spring-cloud-gateway之handler

引言

本篇文字主要讲解spring-cloud-gateway的handler部分,参考官方的gateway处理流程图可以知道handler是gateway的核心控制部分,其控制着request在gateway的整个生命周期。我们可以看到,一个请求由Client发出,被spring-web-server监听,经过gateway-handler-mapping之后请求便进入了gateway部分的代码实现部分,也就表明gateway是通过Gateway Web Hnadler来与web框架交接工作的。那么本节的重点则用于讲解handler的工作交接部分以及如何疏通整个工作流的,扮演这个交接工作角色的是FilteringWebHandler,实现web server的WebHandler接口。

FilteringWebHandler | loadFilters(List<GlobalFilter> filters)

关注final List<GatewayFilter> globalFilters的初始化,会将Configuration注入的GlobalFilter bean初始化至globalFilters;这里有两个目标转换类,在loadFilters方法中map函数对每一个GlobalFilter做包装转换,对于实现了Ordered接口最终转换出OrderedGatewayFilter,没有排序的则转换出GatewayFilterAdapter;两个目标类均是GatewayFilter接口的实现类,GatewayFilterAdapter是FilteringWebHandler的私有嵌套类,理想情况不应该存在不实现Ordered接口的全局过滤器;而OrderedGatewayFilter位于filter包下,与GlobalFilter在同一包下声明,仅仅是对GatewayFilterAdapter的包装,将其GatewayFilterAdapter和order映射到字段。

FilteringWebHandler

这里的loadFilters加载的是全局过滤器,与RouteDefinitionRouteLocator中的loadFilters函数不同,RouteDefinitionRouteLocator中是加载的配置的gatewayFilter,即每次请求都需要根据Route和过滤器工厂集合来加载一次,而全局过滤器的加载只发生在进程启动时的一次性加载行为,但是最终加载结果类型一样都是OrderedGatewayFilter类型。

GatewayFilterAdapter
OrderedGatewayFilter

FilteringWebHandler | handle(ServerWebExchange exchange)

handle方法处理一个来自web server(WebFlux)的http(web server将一个请求的request和response抽象成一个ServerWebExchange)请求,首先在exchange的attributes中获取当前请求配置或者代码预先设置的gatewayFilters,然后与当前全局过滤器转换出来的gatewayFilter进行组合,最后new一个GatewayFilterChain来对exchange过滤。

GatewayFilterChain处理过程

RoutePredicateHandlerMapping

RoutePredicateHandlerMapping两个重要属性,一个是FilteringWebHandler;一个是RouteLocator接口的实现,默认是CachingRouteLocator。

webHandler就是代理了FilteringWebHandler的处理能力,调用其handle函数处理请求的ServerWebExchange。

handler模块处理request的入口是getHandlerInternal函数,主要逻辑在lookupRoute(ServerWebExchange exchange)方法中,为每一个请求分配路由,这里并没有将路由作为返回结果,而是将其set进attributes属性集合中。

路由匹配逻辑触发函数

从方法逻辑处分析,首先调用了CachingRouteLocator的getRoutes()获取当前内存中所有已存在的Route集合,然后有序遍历concatMap()中通过filterWhen来使用每一个Route的predicate(AsyncPredicate)来匹配当前的request,这个predicate是一个函数接口实现,其中and操作符将多个配置断言进行聚合操作的,这部分可以参考对Route部分的讲解。

路由断言工厂

gateway路由断言

在spring-cloud-gateway的架构设计中,将路由设计为spring-webflux的一部分,对于webflux mapping的请求进行匹配;在此基础上gateway内置了许多的路由断言,均位于handler包的predicate下;主要用于断言匹配Http请求的不同属性,通过断言组合来实现多属性匹配,断言作用发生于filter之前,在Exchange进入Filter之前对请求属性做处理匹配,所以断言仅仅对于request有效。

从断言的命名可以明白大部分的断言作用,其实在使用过程中,大部分断言应用并不广泛,正常情况下都会配置PathRoutePredicateFactory断言来断言区分path,其他的用于与其组合作用于某个下游服务或者绝对路由设置。下面图片列举出接口-抽象类-实现类的关系图,与GatewayFilter过滤器工厂类继承依赖关系极度相似。

接口-类

对于每一个Route的定义中,均有相应的predicate声明,可以在配置文件或者代码中进行配置。

AfterRoutePredicateFactory

AfterRoutePredicateFactory断言用于断言请求发生时间,配置参数只有一个datetime的String类型参数,在GatewayAutoConfiguration中注入,匹配配置时间之后的所有请求,在一些预热活动等场景可以配合其他断言来使用。

BeforeRoutePredicateFactory

BeforeRoutePredicateFactory与AfterRoutePredicateFactory对应,匹配配置时间之前的所有请求,能够与其他比如Path断言组合使用。

配置示例

BetweenRoutePredicateFactory

BetweenRoutePredicateFactory是BeforeRoutePredicateFactory和AfterRoutePredicateFactory的组合体。

Between配置示例

CloudFoundryRouteServiceRoutePredicateFactory

CloudFoundryRouteServiceRoutePredicateFactory官方并没有作使用说明,从代码提交上看来是配合HeaderRoutePredicateFactory使用,用于对request进行forward的,目前看来应用场景并不明确。

CookieRoutePredicateFactory

CookieRoutePredicateFactory断言,用于匹配指定Cookie的请求,结合官方说明可知其意图在过滤指定的请求;比如配置过滤不同设备端的web请求等场景。

image.png
image.png

HeaderRoutePredicateFactory

HeaderRoutePredicateFactory是Http Header层的断言,匹配指定header参数,同时也可以指定参数所匹配的表达式,没有进行实际应用过。

HostRoutePredicateFactory

HostRoutePredicateFactory配置指定Host模式的请求(比如二级域名的正则匹配),并将匹配结果以键值对形式放入attributes属性集合中。

MethodRoutePredicateFactory

MethodRoutePredicateFactory很好理解,即匹配指定Method的请求,否则被忽略,如果一个路由只允许某些Http方法的请求,可以使用此配置,来限制不合法的请求。

PathRoutePredicateFactory

PathRoutePredicateFactory是断言中应用最广泛的,也是路由匹配的核心组件,根据请求的Uri进行模式匹配区分路由的。

QueryRoutePredicateFactory

QueryRoutePredicateFactory作用于请求的query参数,可以用于限制某些请求必要的query参数。

ReadBodyPredicateFactory

ReadBodyPredicateFactory个人觉得不是一个实用的断言,可以参考Filter中ModifyBody部分的实现,涉及异步流的修改。

RemoteAddrRoutePredicateFactory

RemoteAddrRoutePredicateFactory断言可以过滤掉非指定ip地址段的请求;个人觉得可以写一个ExcludeRemoteAddrRoutePredicateFactory断言,用于拒绝指定ip的请求,或者从黑名单中匹配拒绝,粗略的防止hack攻击。

WeightRoutePredicateFactory

WeightRoutePredicateFactory权重路由,可以用于恢复发布或者AB test场景,互联网应用中常见的需求,这个断言需要集成WeightCalculatorWebFilter过滤器来起作用。

具体配置可以参考:Canary-test using Spring Cloud Gateway

Sumarry

spring-cloud-gateway中predicate是过滤匹配请求的第一步,主要作用是为每个请求精准匹配出其路由,其中关键精彩实现部分在于作者合理利用函数式接口实现类AsyncPredicate类将多个predicate进行了聚合操作,这是值得Java编程学习的。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,558评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,002评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,036评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,024评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,144评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,255评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,295评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,068评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,478评论 1 305
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,789评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,965评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,649评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,267评论 3 318
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,982评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,223评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,800评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,847评论 2 351

推荐阅读更多精彩内容