SpringCloud Zuul(五)之编程指导

一、Zuul Servlet

Zuul被实现为Servlet。对于一般情况,Zuul已嵌入到Spring Dispatch机制中。这使Spring MVC可以控制路由。在这种情况下,Zuul缓冲请求。如果需要在不缓冲请求的情况下进行Zuul操作(例如,对于大文件上传),则Servlet也会安装在Spring Dispatcher的外部。默认情况下,该servlet的地址为/zuul。可以使用zuul.servlet-path属性更改此路径。

(1)Zuul RequestContext

要在过滤器之间传递信息,Zuul使用RequestContext。它的数据保存在ThreadLocal每个请求的特定内容中。有关在何处路由请求,错误以及实际HttpServletRequestHttpServletResponse的信息存储在此处。在RequestContext扩展ConcurrentHashMap,所以什么都可以存储在上下文。FilterConstants包含由Spring Cloud Netflix安装的过滤器使用的密钥(稍后会详细介绍)。

(2)@EnableZuulProxy与@EnableZuulServer

Spring Cloud Netflix安装了许多过滤器,具体取决于用于启用Zuul的注释。@EnableZuulProxy是的超集@EnableZuulServer。换句话说,@EnableZuulProxy包含所安装的所有过滤器@EnableZuulServer。“代理”中的其他过滤器启用路由功能。如果您想要“空白” Zuul,则应使用@EnableZuulServer

(3)@EnableZuulServer筛选器

@EnableZuulServer创建一个SimpleRouteLocator可从Spring Boot配置文件加载路由定义的。

安装了以下过滤器(作为普通的Spring Bean):

  • 前置过滤器:

    • ServletDetectionFilter:检测请求是否通过Spring Dispatcher。设置键为的布尔值FilterConstants.IS_DISPATCHER_SERVLET_REQUEST_KEY

    • FormBodyWrapperFilter:解析表单数据并为下游请求重新编码。

    • DebugFilter:如果debug设置了请求参数,则将RequestContext.setDebugRouting()和设置RequestContext.setDebugRequest()true

  • 路由过滤器:

    • SendForwardFilter:通过使用Servlet转发请求RequestDispatcher。转发位置存储在RequestContext属性中FilterConstants.FORWARD_TO_KEY。这对于转发到当前应用程序中的端点很有用。

  • 后置过滤器:

    • SendResponseFilter:将代理请求的响应写入当前响应。

  • 错误过滤器:

    • SendErrorFilter:如果RequestContext.getThrowable()不为null,则转发到/error(默认情况下)。您可以通过设置error.path属性来更改默认转发路径(/error)。

(4)@EnableZuulProxy筛选器

创建一个DiscoveryClientRouteLocator可从DiscoveryClient(例如Eureka)以及属性加载路线定义的。路线为每个创建serviceId从所述DiscoveryClient。添加新服务后,将刷新路由。

除了前面描述的过滤器之外,还安装了以下过滤器(作为普通的Spring Bean):

  • 前置过滤器:

    • PreDecorationFilter:根据提供的确定位置和路线RouteLocator。它还为下游请求设置了各种与代理相关的标头。

  • 路由过滤器:

    • RibbonRoutingFilter:使用Ribbon,Hystrix和可插拔HTTP客户端发送请求。服务ID在RequestContext属性中找到FilterConstants.SERVICE_ID_KEY。此过滤器可以使用不同的HTTP客户端:

      • Apache HttpClient:默认客户端。

      • Squareup OkHttpClientv3:通过将com.squareup.okhttp3:okhttp库放在类路径上并设置来启用ribbon.okhttp.enabled=true

      • Netflix Ribbon HTTP客户端:通过设置启用ribbon.restclient.enabled=true。该客户端具有局限性,包括不支持PATCH方法,但还具有内置的重试功能。

    • SimpleHostRoutingFilter:通过Apache HttpClient将请求发送到预定的URL。网址位于中RequestContext.getRouteHost()

 

二、自定义Zuul过滤器示例

下面的大多数“如何编写”示例都包含在示例Zuul过滤器项目中。在该存储库中也有一些处理请求或响应正文的示例。

(1)如何编写预过滤器

前置过滤器会在中设置数据,以RequestContext供下游过滤器使用。主要用例是设置路由过滤器所需的信息。以下示例显示了Zuul预过滤器:

public class QueryParamPreFilter extends ZuulFilter {@Overridepublic int filterOrder() {return PRE_DECORATION_FILTER_ORDER - 1; // run before PreDecoration}@Overridepublic String filterType() {return PRE_TYPE;}@Overridepublic boolean shouldFilter() {RequestContext ctx = RequestContext.getCurrentContext();return !ctx.containsKey(FORWARD_TO_KEY) // a filter has already forwarded&& !ctx.containsKey(SERVICE_ID_KEY); // a filter has already determined serviceId}@Overridepublic Object run() {RequestContext ctx = RequestContext.getCurrentContext();HttpServletRequest request = ctx.getRequest();if (request.getParameter("sample") != null) {// put the serviceId in `RequestContext`ctx.put(SERVICE_ID_KEY, request.getParameter("foo"));}return null;}
}

前面的过滤器SERVICE_ID_KEYsamplerequest参数填充。实际上,您不应该执行这种直接映射。相反,应该从sample相反的值中查找服务ID 。

现在SERVICE_ID_KEY已填充,PreDecorationFilter无法运行RibbonRoutingFilter

注:如果要路由到完整URL,请致电ctx.setRouteHost(url)

要修改路由过滤器转发到的路径,请设置REQUEST_URI_KEY

(2)如何编写路由过滤器

路由过滤器在预过滤器之后运行,并向其他服务发出请求。此处的许多工作是在请求和响应数据与客户端所需的模型之间进行转换。以下示例显示了Zuul路由过滤器:

public class OkHttpRoutingFilter extends ZuulFilter {@Autowiredprivate ProxyRequestHelper helper;@Overridepublic String filterType() {return ROUTE_TYPE;}@Overridepublic int filterOrder() {return SIMPLE_HOST_ROUTING_FILTER_ORDER - 1;}@Overridepublic boolean shouldFilter() {return RequestContext.getCurrentContext().getRouteHost() != null&& RequestContext.getCurrentContext().sendZuulResponse();}@Overridepublic Object run() {OkHttpClient httpClient = new OkHttpClient.Builder()// customize.build();RequestContext context = RequestContext.getCurrentContext();HttpServletRequest request = context.getRequest();String method = request.getMethod();String uri = this.helper.buildZuulRequestURI(request);Headers.Builder headers = new Headers.Builder();Enumeration<String> headerNames = request.getHeaderNames();while (headerNames.hasMoreElements()) {String name = headerNames.nextElement();Enumeration<String> values = request.getHeaders(name);while (values.hasMoreElements()) {String value = values.nextElement();headers.add(name, value);}}InputStream inputStream = request.getInputStream();RequestBody requestBody = null;if (inputStream != null && HttpMethod.permitsRequestBody(method)) {MediaType mediaType = null;if (headers.get("Content-Type") != null) {mediaType = MediaType.parse(headers.get("Content-Type"));}requestBody = RequestBody.create(mediaType, StreamUtils.copyToByteArray(inputStream));}Request.Builder builder = new Request.Builder().headers(headers.build()).url(uri).method(method, requestBody);Response response = httpClient.newCall(builder.build()).execute();LinkedMultiValueMap<String, String> responseHeaders = new LinkedMultiValueMap<>();for (Map.Entry<String, List<String>> entry : response.headers().toMultimap().entrySet()) {responseHeaders.put(entry.getKey(), entry.getValue());}this.helper.setResponse(response.code(), response.body().byteStream(),responseHeaders);context.setRouteHost(null); // prevent SimpleHostRoutingFilter from runningreturn null;}
}

前面的过滤器将Servlet请求信息转换为OkHttp3请求信息,执行HTTP请求,并将OkHttp3响应信息转换为Servlet响应。

(3)如何编写后置过滤器

后置过滤器通常操纵响应。以下过滤器将随机数添加UUIDX-Sample标题:

public class AddResponseHeaderFilter extends ZuulFilter {@Overridepublic String filterType() {return POST_TYPE;}@Overridepublic int filterOrder() {return SEND_RESPONSE_FILTER_ORDER - 1;}@Overridepublic boolean shouldFilter() {return true;}@Overridepublic Object run() {RequestContext context = RequestContext.getCurrentContext();HttpServletResponse servletResponse = context.getResponse();servletResponse.addHeader("X-Sample", UUID.randomUUID().toString());return null;}
}
注:其他操作,例如转换响应主体,则更加复杂且计算量大。

三、Zuul错误如何工作

如果在Zuul过滤器生命周期的任何部分抛出异常,则将执行错误过滤器。该SendErrorFilter如果只运行RequestContext.getThrowable()null。然后,它javax.servlet.error.*在请求中设置特定的属性,并将请求转发到Spring Boot错误页面。

 

四、Zuul饥饿加载应用程序上下文

Zuul在内部使用Ribbon来调用远程URL。默认情况下,ribbon客户端在第一次调用时由Spring Cloud延迟加载。可以使用以下配置为Zuul更改此行为,这将导致在应用程序启动时即时加载与子Ribbon相关的应用程序上下文。以下示例显示了如何启用饥饿加载:

application.yml
zuul:ribbon:eager-load:enabled: true

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/322092.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Comet OJ(Contest #8)-D菜菜种菜【树状数组,指针】

前言 话说昨晚写题的时候贼NMNMNM惊险&#xff0c;最后22秒把程序交了上去竟然过了 正题 题目链接:https://cometoj.com/contest/58/problem/D?problem_id2758 题目大意 nnn个点mmm条单向边&#xff0c;然后每次询问一个区间[L,R][L,R][L,R]求若只选择这个区间的点&#xf…

微软正式开源Blazor ,将.NET带回到浏览器

微软 ASP.NET 团队近日正式开源了 Blazor &#xff0c;这是一个 Web UI 框架&#xff0c;可通过 WebAssembly 在任意浏览器中运行 .Net 。Blazor 旨在简化快速的单页面 .Net 浏览器应用的构建过程&#xff0c;它虽然使用了诸如 CSS 和 HTML 之类的 Web 技术&#xff0c;但它使…

SpringCloud Zuul(六)之PRE Filter

一、PRE Filter 前置过滤器一般用来区分请求来源、转换数据格式、debug日志、校验权限&#xff0c;增加请求装饰标识等待操作。 有一下几个重要方法&#xff1a; &#xff08;1&#xff09;filterType 确定过滤器类型 &#xff08;2&#xff09;filterOrder 过滤器执行顺序…

在.NetCore中使用Myrmec检测文件真实格式

Myrmec 是什么&#xff1f;Myrmec 是一个用于检测文件格式的库&#xff0c;Myrmec不同于其它库或者手写检测代码&#xff0c;Myrmec不依赖文件扩展名&#xff08;在实际使用中&#xff0c;你的用户很可能使用虚假的扩展名欺骗你的应用程序&#xff09;&#xff0c;Myrmec会检测…

jzoj3736-[NOI2014模拟7.11]数学题(math)【计算几何】

正题 题目大意 给定两个向量a(x1,y1),b(x2,y2)a(x_1,y_1),b(x_2,y_2)a(x1​,y1​),b(x2​,y2​)&#xff0c;然后求∣λ1aλ2b∣|\lambda _1a\lambda _2b|∣λ1​aλ2​b∣的最小值&#xff0c;要求λ1,λ2\lambda_1,\lambda _2λ1​,λ2​不同时为0。 解题思路 我们先考虑若…

SpringCloud Zuul(七)之POST Filter

一、POST Filter 后置过滤器一般使用来转换响应数据的格式&#xff0c;截取请求响应数据进行流量录制等操作。 Zuul已定义的后置过滤器SendResponseFilter&#xff0c;将代理请求的响应写入当前响应。 二、自定义后置过滤器 本文自定义后置过滤器用来截取响应体的快照文本发…

Orleans之EventSourcing

引入:如果没有意外,我再这篇文章中用ES代替EventSourcing,如果碰到"事件回溯","事件溯源","事溯"等词语,都一般代表Eventsourcing.如果引入Orleans而不用es的话,那就只用了Orleans一半的优点,多线程编程的逻辑\排错的简化以及可分布式.下面我聊聊…

jzoj3737-[NOI2014模拟7.11]挖宝藏(treasure)【斯坦纳树,SPFA,状压】

正题 题目大意 hhh层&#xff0c;每层n∗mn*mn∗m个石头&#xff0c;挖开不同位置的石头有不同的消耗&#xff0c;只能从高层下到低层。有一些宝藏&#xff0c;求拿到所有宝藏的最小代价。 解题思路 先考虑只有111层的情况&#xff0c;因为挖开的不用再挖 &#xff0c;我们可以…

SpringCloud Zuul(八)之ERROR Filter

一、ERROR Filter 错误过滤器用来处理zuul异常&#xff0c;一般使作为打印异常堆栈、跳转异常页面、转换异常信息格式返回等操作。 Zuul已定义的错误过滤器SendErrorFilter&#xff0c;如果RequestContext.getThrowable()不为null&#xff0c;则转发到/error&#xff08;默认…

Alex: 2018年对混合现实MR的展望

原文作者&#xff1a;Alex Kipman&#xff0c; 微软操作系统工程院技术院士 Hello 大家好&#xff01;难以置信我们已经走过了2018年的头两个月了。每年一月份我都会去巴西省亲&#xff0c;和我的家人欢聚一堂&#xff0c;度过一个美好的假日。在我省亲的同时&#xff0c;我想了…

jzoj3738-[NOI2014模拟7.11]理想城市(city)【树,模型转换】

正题 题目大意 一个理想城市有nnn个块构成&#xff0c;有以下性质 任意两个块之间可以通过其他块到达任意两个块之间可以不通过其他块(通过空位)到达 然后求每个块之间的距离之和。 解题思路 我们将横竖的距离分开计算。 假设现在我们考虑计算竖向的边的距离&#xff0c;我…

SpringCloud Ribbon(二)之自定义负载均衡策略IRule

一、Ribbon负载均衡策略 一个服务对应一个LoadBalancer&#xff0c;一个LoadBalancer只有一个Rule&#xff0c;LoadBalancer记录服务的注册地址&#xff0c;Rule提供从服务的注册地址中找出一个地址的规则。 Ribbon提供七种负载均衡策略&#xff0c;默认的负载均衡策略是轮训策…

欢乐纪中A组赛【2019.8.10】

前言 昨天&#xff1a; MdMdMd今天还真爆零了 顺便%%%ZZYRank1\%\%\%ZZY\ Rank1%%%ZZY Rank1 成绩 JJJ表示初中&#xff0c;HHH表示高中后面加的是几年级 RankRankRankPersonPersonPersonScoreScoreScoreAAABBBCCC111(H−1)ZZY(H-1)ZZY(H−1)ZZY1501501502020203030301001001…

Blazor正式成为Microsoft官方.NET 和WebAssembly项目

Microsoft从Blazor的开发者Steve Sanderson手中接手了这款应用程序&#xff0c;自此&#xff0c;将.NET在浏览器运行的计划又更进了一步。由此&#xff0c;Microsoft又进一步扩充了自己的WebAssembly/.NET栈&#xff0c;更进一步帮助.NET开发人员搭建基于浏览器的应用程序。在一…

SpringCloud Ribbon(一)之自定义负载均衡器ILoadBalancer

一、Ribbon负载均衡 一个服务对应一个LoadBalancer&#xff0c;一个LoadBalancer只有一个Rule&#xff0c;LoadBalancer记录服务的注册地址&#xff0c;提供更新服务的注册地址&#xff0c;Rule提供从服务的注册地址中找出一个地址的规则。 二、 自定义负载均衡 本文自定义负…

bzoj4403-序列统计【Lucas,组合数学】

正题 题目链接:https://www.lydsy.com/JudgeOnline/problem.php?id4403 题目大意 求有多少个长度为nnn的单调不升序列&#xff0c;且对于每个元素都∈[L,R]\in[L,R]∈[L,R] 解题思路 我们让mR−L1mR-L1mR−L1&#xff0c;因为序列的要求起始起始不会影响结果 然后我们开始考…

.NET Core 2.1路线图

Microsoft的Scott Hunter发布了Microsoft .NET Core 2.1版本的路线图。Hunter宣布Microsoft .NET Core每天约有五十万开发人员的使用量。根据Microsoft所收集的数据&#xff0c;在2017年9月.NET Core 2的使用量已经超过了.NET Core 1.X。有了之前成功的发布经验&#xff0c;Mic…

SpringCloud Ribbon(三)之IPing机制

一、IPing机制 IPing是一个主动探测服务节点存活的机制&#xff0c;通过判断服务节点的当前状态&#xff0c;设置节点的可用状态。只有当节点为可用时候才会作为负载均衡器的选取节点。 IPing有以下几种模式: DummyPing&#xff1a;默认返回true&#xff0c;即认为所有节点都…

.NET Core使用swagger进行API接口文档管理

一、问题背景随着技术的发展&#xff0c;现在的开发模式已经更多的转向了前后端分离的模式&#xff0c;在前后端开发的过程中&#xff0c;联系的方式也变成了API接口&#xff0c;但是目前项目中对于API的管理很多时候还是通过手工编写文档&#xff0c;每次的需求变更只要涉及到…

CF451E-Devu and Flowers【组合计数,容斥】

正题 题目链接:https://www.luogu.org/problem/CF451E 题目大意 长度为nnn的序列aia_iai​&#xff0c;有如下要求 ai∈[0...xi]a_i\in[0...x_i]ai​∈[0...xi​]∑i1nais\sum_{i1}^na_is∑i1n​ai​s 求序列个数。 解题思路 考虑容斥&#xff0c;这样我们就可以将条件转换…