聊聊nacos的DistroFilter
阿新 • • 發佈:2019-12-31
序
本文主要研究一下nacos的DistroFilter
CanDistro
nacos-1.1.3/naming/src/main/java/com/alibaba/nacos/naming/web/CanDistro.java
@Retention(RetentionPolicy.RUNTIME)
public @interface CanDistro {
}
複製程式碼
- CanDistro用於標識一個方法需要判斷是否應該根據distro被重定向
DistroFilter
nacos-1.1.3/naming/src/main/java/com/alibaba/nacos/naming/web/DistroFilter.java
public class DistroFilter implements Filter {
private static final int PROXY_CONNECT_TIMEOUT = 2000;
private static final int PROXY_READ_TIMEOUT = 2000;
@Autowired
private DistroMapper distroMapper;
@Autowired
private SwitchDomain switchDomain;
@Autowired
private FilterBase filterBase;
@Override
public void init(FilterConfig filterConfig) throws ServletException {
}
@Override
public void do Filter(ServletRequest servletRequest,ServletResponse servletResponse,FilterChain filterChain) throws IOException,ServletException {
HttpServletRequest req = (HttpServletRequest) servletRequest;
HttpServletResponse resp = (HttpServletResponse) servletResponse;
String urlString = req.getRequestURI();
if (StringUtils.isNotBlank(req.getQueryString())) {
urlString += "?" + req.getQueryString();
}
try {
String path = new URI(req.getRequestURI()).getPath();
String serviceName = req.getParameter(CommonParams.SERVICE_NAME);
// For client under 0.8.0:
if (StringUtils.isBlank(serviceName)) {
serviceName = req.getParameter("dom");
}
Method method = filterBase.getMethod(req.getMethod(),path);
if (method == null) {
throw new NoSuchMethodException(req.getMethod() + " " + path);
}
String groupName = req.getParameter(CommonParams.GROUP_NAME);
if (StringUtils.isBlank(groupName)) {
groupName = Constants.DEFAULT_GROUP;
}
// use groupName@@serviceName as new service name:
String groupedServiceName = serviceName;
if (StringUtils.isNotBlank(serviceName) && !serviceName.contains(Constants.SERVICE_INFO_SPLITER)) {
groupedServiceName = groupName + Constants.SERVICE_INFO_SPLITER + serviceName;
}
// proxy request to other server if necessary:
if (method.isAnnotationPresent(CanDistro.class) && !distroMapper.responsible(groupedServiceName)) {
String userAgent = req.getHeader("User-Agent");
if (StringUtils.isNotBlank(userAgent) && userAgent.contains(UtilsAndCommons.NACOS_SERVER_HEADER)) {
// This request is sent from peer server,should not be redirected again:
Loggers.SRV_LOG.error("receive invalid redirect request from peer {}",req.getRemoteAddr());
resp.sendError(HttpServletResponse.SC_BAD_REQUEST,"receive invalid redirect request from peer " + req.getRemoteAddr());
return;
}
List<String> headerList = new ArrayList<>(16);
Enumeration<String> headers = req.getHeaderNames();
while (headers.hasMoreElements()) {
String headerName = headers.nextElement();
headerList.add(headerName);
headerList.add(req.getHeader(headerName));
}
HttpClient.HttpResult result =
HttpClient.request("http://" + distroMapper.mapSrv(groupedServiceName) + urlString,headerList,StringUtils.isBlank(req.getQueryString()) ? HttpClient.translateParameterMap(req.getParameterMap()) : new HashMap<>(2),PROXY_CONNECT_TIMEOUT,PROXY_READ_TIMEOUT,"UTF-8",req.getMethod());
try {
resp.setCharacterEncoding("UTF-8");
resp.getWriter().write(result.content);
resp.setStatus(result.code);
} catch (Exception ignore) {
Loggers.SRV_LOG.warn("[DISTRO-FILTER] request failed: " + distroMapper.mapSrv(groupedServiceName) + urlString);
}
return;
}
OverrideParameterRequestWrapper requestWrapper = OverrideParameterRequestWrapper.buildRequest(req);
requestWrapper.addParameter(CommonParams.SERVICE_NAME,groupedServiceName);
filterChain.doFilter(requestWrapper,resp);
} catch (AccessControlException e) {
resp.sendError(HttpServletResponse.SC_FORBIDDEN,"access denied: " + UtilsAndCommons.getAllExceptionMsg(e));
return;
} catch (NoSuchMethodException e) {
resp.sendError(HttpServletResponse.SC_NOT_IMPLEMENTED,"no such api: " + e.getMessage());
return;
} catch (Exception e) {
resp.sendError(HttpServletResponse.SC_INTERNAL_SERVER_ERROR,"Server failed," + UtilsAndCommons.getAllExceptionMsg(e));
return;
}
}
@Override
public void destroy() {
}
}
複製程式碼
- DistroFilter實現了servlet的Filter介面;其doFilter方法會從servletRequest中讀取serviceName、method、groupName等,然後判斷method是否標註CanDistro,如果是而且distroMapper不負責該service則構建http請求然後將結果寫回Filter;如果不需要重定向則繼續filterChain.doFilter
HttpClient.request
nacos-1.1.3/naming/src/main/java/com/alibaba/nacos/naming/misc/HttpClient.java
public class HttpClient {
private static final int TIME_OUT_MILLIS = 10000;
private static final int CON_TIME_OUT_MILLIS = 5000;
private static AsyncHttpClient asyncHttpClient;
private static CloseableHttpClient postClient;
//......
public static HttpResult request(String url,List<String> headers,Map<String,String> paramValues,int connectTimeout,int readTimeout,String encoding,String method) {
HttpURLConnection conn = null;
try {
String encodedContent = encodingParams(paramValues,encoding);
url += (null == encodedContent) ? "" : ("?" + encodedContent);
conn = (HttpURLConnection) new URL(url).openConnection();
conn.setConnectTimeout(connectTimeout);
conn.setReadTimeout(readTimeout);
conn.setRequestMethod(method);
conn.addRequestProperty("Client-Version",UtilsAndCommons.SERVER_VERSION);
conn.addRequestProperty("User-Agent",UtilsAndCommons.SERVER_VERSION);
setHeaders(conn,headers,encoding);
conn.connect();
return getResult(conn);
} catch (Exception e) {
Loggers.SRV_LOG.warn("Exception while request: {},caused: {}",url,e);
return new HttpResult(500,e.toString(),Collections.<String,String>emptyMap());
} finally {
if (conn != null) {
conn.disconnect();
}
}
}
private static HttpResult getResult(HttpURLConnection conn) throws IOException {
int respCode = conn.getResponseCode();
InputStream inputStream;
if (HttpURLConnection.HTTP_OK == respCode) {
inputStream = conn.getInputStream();
} else {
inputStream = conn.getErrorStream();
}
Map<String,String> respHeaders = new HashMap<String,String>(conn.getHeaderFields().size());
for (Map.Entry<String,List<String>> entry : conn.getHeaderFields().entrySet()) {
respHeaders.put(entry.getKey(),entry.getValue().get(0));
}
String gzipEncoding = "gzip";
if (gzipEncoding.equals(respHeaders.get(HttpHeaders.CONTENT_ENCODING))) {
inputStream = new GZIPInputStream(inputStream);
}
HttpResult result = new HttpResult(respCode,IOUtils.toString(inputStream,getCharset(conn)),respHeaders);
inputStream.close();
return result;
}
public static class HttpResult {
final public int code;
final public String content;
final private Map<String,String> respHeaders;
public HttpResult(int code,String content,String> respHeaders) {
this.code = code;
this.content = content;
this.respHeaders = respHeaders;
}
public String getHeader(String name) {
return respHeaders.get(name);
}
}
//......
}
複製程式碼
- HttpClient的request方法直接使用jdk的HttpURLConnection進行請求,返回結果封裝為HttpResult,其content即為響應的body
小結
DistroFilter實現了servlet的Filter介面;其doFilter方法會從servletRequest中讀取serviceName、method、groupName等,然後判斷method是否標註CanDistro,如果是而且distroMapper不負責該service則構建http請求然後將結果寫回Filter;如果不需要重定向則繼續filterChain.doFilter