您好,登錄后才能下訂單哦!
本篇文章為大家展示了nacos中DistroFilter的作用是什么,內容簡明扼要并且容易理解,絕對能使你眼前一亮,通過這篇文章的詳細介紹希望你能有所收獲。
nacos-1.1.3/naming/src/main/java/com/alibaba/nacos/naming/web/CanDistro.java
@Retention(RetentionPolicy.RUNTIME) public @interface CanDistro { }
CanDistro用于標識一個方法需要判斷是否應該根據distro被重定向
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 doFilter(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
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, Map<String, 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
上述內容就是nacos中DistroFilter的作用是什么,你們學到知識或技能了嗎?如果還想學到更多技能或者豐富自己的知識儲備,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。