文章目录

  • 1. GateWay的自动配置
  • 2. GateWay的源码执行流程
    • ①:进入路由断言HandlerMapping,扫描yml文件,匹配路由信息
    • ②:找到对应的适配器HandlerAdaptor,执行过滤器链
  • 3:Gateway的负载均衡是如何实现的?

1. GateWay的自动配置

springboot 在引入一个新的组件时,一般都会有对应的XxxAutoConfiguration类来对该组件进行配置,GateWay也不例外,在引入了以下配置后,就会生成对应的GatewayAutoConfiguration自动配置类

    <!-- gateway网关 --><dependency><groupId>org.springframework.cloud</groupId><artifactId>spring-cloud-starter-gateway</artifactId></dependency>

GatewayAutoConfiguration自动配置类信息如下:

@Configuration(proxyBeanMethods = false)
//spring.cloud.gateway.enabled配置项必须为true,自动配置才生效,默认为true
@ConditionalOnProperty(name = "spring.cloud.gateway.enabled", matchIfMissing = true)
@EnableConfigurationProperties
//自动配置前置条件:引入了WebFlux 和 HttpHandler 组件
@AutoConfigureBefore({ HttpHandlerAutoConfiguration.class,WebFluxAutoConfiguration.class })
//自动配置后置组件:负载均衡组件
@AutoConfigureAfter({ GatewayLoadBalancerClientAutoConfiguration.class,GatewayClassPathWarningAutoConfiguration.class })
@ConditionalOnClass(DispatcherHandler.class)
public class GatewayAutoConfiguration {..... // 初始化各种bean
}

从配置类上的注解,可以了解到

  • spring.cloud.gateway.enabled配置项必须为true,自动配置才生效,默认为true
  • 在使用Gataway之前,必须存在WebFluxHttpHandler 组件
  • 注入Gataway之后,需要对请求负载到某一台服务器上,所以后置组件为负载均衡组件

自动配置类GatewayAutoConfiguration在内部初始化了很多bean,列举几个重要的如下:

  • PropertiesRouteDefinitionLocator:用于从配置文件(yml/properties)中读取路由配置信息!
  • RouteDefinitionLocator:把 RouteDefinition 转化为 Route
  • RoutePredicateHandlerMapping:类似于 mvc 的HandlerMapping,不过这里是 Gateway实现的。用于匹配对应的请求route
  • GatewayProperties:yml配置信息封装在 GatewayProperties 对象中
  • AfterRoutePredicateFactory:各种路由断言工厂,正是这些断言工厂在启动时已经生成对应的bean,我们才可以在 yml 中配置一下,即可生效
  • RetryGatewayFilterFactory:各种 Gateway 过滤器,正是这些过滤器在启动时已经生成对应的bean,我们才可以在 yml 中配置一下,即可生效
  • GlobalFilter实现类:全局过滤器

2. GateWay的源码执行流程

GateWay采用的是webFlux的响应式编程,其整个流程与spring mvc 类似

框架 Gateway spring mvc
请求分发 DispatcherHandler DispatcherServlet
请求映射 HandlerMapping HandlerMapping
请求适配 HanderAdaper HanderAdaper
请求处理 WebHander Hander

所有请求都会经过 gateway 的DispatcherHandler中的handle方法!可以看到该方法使用的就是webFlux的响应式编程

 @Overridepublic Mono<Void> handle(ServerWebExchange exchange) {if (this.handlerMappings == null) {return createNotFoundError();}if (CorsUtils.isPreFlightRequest(exchange.getRequest())) {return handlePreFlight(exchange);}return Flux// 1.遍历所有的 handlerMapping.fromIterable(this.handlerMappings) // 2.获取对应的handlerMapping ,比如常用的 RequestMappingHandlerMapping、RoutePredicateHandlerMapping.concatMap(mapping -> mapping.getHandler(exchange)).next().switchIfEmpty(createNotFoundError())// 3.获取对应的适配器,调用对应的处理器.flatMap(handler -> invokeHandler(exchange, handler))// 4.返回处理结果.flatMap(result -> handleResult(exchange, result));}

其实这就是 gateway的核心逻辑

①:进入路由断言HandlerMapping,扫描yml文件,匹配路由信息

上文核心逻辑代码中getHandler(exchange)方法是获取对应的HandlerMapping。由于是网关组件,当请求进入时,会先判断路由,所以会进入实现类RoutePredicateHandlerMapping

org.springframework.cloud.gateway.handler.RoutePredicateHandlerMapping # getHandlerInternal 方法如下:

 @Overrideprotected Mono<?> getHandlerInternal(ServerWebExchange exchange) {if (this.managementPortType == DIFFERENT && this.managementPort != null&& exchange.getRequest().getURI().getPort() == this.managementPort) {return Mono.empty();}exchange.getAttributes().put(GATEWAY_HANDLER_MAPPER_ATTR, getSimpleName());//寻找并匹配路由return lookupRoute(exchange).flatMap((Function<Route, Mono<?>>) r -> {//移除上下文中旧的属性exchange.getAttributes().remove(GATEWAY_PREDICATE_ROUTE_ATTR);if (logger.isDebugEnabled()) {logger.debug("Mapping [" + getExchangeDesc(exchange) + "] to " + r);}//把该路由与上下文绑定,后续负载均衡会用exchange.getAttributes().put(GATEWAY_ROUTE_ATTR, r);//返回 webHandlerreturn Mono.just(webHandler);}).switchIfEmpty(Mono.empty().then(Mono.fromRunnable(() -> {exchange.getAttributes().remove(GATEWAY_PREDICATE_ROUTE_ATTR);if (logger.isTraceEnabled()) {logger.trace("No RouteDefinition found for ["+ getExchangeDesc(exchange) + "]");}})));}

其中lookupRoute方法会找到yml中配置的所有的路由断言工厂(Before、After、Path等等),并执行apply方法,进行路由匹配,判断是否允许请求通过!执行顺序由springboot自动配置时自己制定

 protected Mono<Route> lookupRoute(ServerWebExchange exchange) {// getRoutes 获取所有的断言工厂return this.routeLocator.getRoutes().concatMap(route -> Mono.just(route).filterWhen(r -> {exchange.getAttributes().put(GATEWAY_PREDICATE_ROUTE_ATTR, r.getId());// 先获取Route内部的predicate属性//然后调用apply方法 执行断言!判断请求是否通过return r.getPredicate().apply(exchange);})

其中getRoutes()方法就是通过RouteDefinitionRouteLocator从配置文件中获取所有路由的,然后把找到的路由转换成Route

 public Flux<Route> getRoutes() {// getRouteDefinitions() 从配置文件中获取所有路由Flux<Route> routes = this.routeDefinitionLocator.getRouteDefinitions()// convertToRoute():把找到的路由转换成Route.map(this::convertToRoute);

Route内部结构如下

public class Route implements Ordered {//路由idprivate final String id;//请求URIprivate final URI uri;   //排序private final int order;    //断言private final AsyncPredicate<ServerWebExchange> predicate;    //过滤器private final List<GatewayFilter> gatewayFilters;    //元数据private final Map<String, Object> metadata;

②:找到对应的适配器HandlerAdaptor,执行过滤器链

Gateway由于在第①步匹配路由后返回的是webHandler类型的,所以也需要找到对应的HandlerAdaptor,进入获取对应的适配器方法 invokeHandler(exchange, handler)


SimpleHandlerAdapter 中的handle方法如下

public class SimpleHandlerAdapter implements HandlerAdapter {@Overridepublic Mono<HandlerResult> handle(ServerWebExchange exchange, Object handler) {//处理WebHandler 类型WebHandler webHandler = (WebHandler) handler;Mono<Void> mono = webHandler.handle(exchange);return mono.then(Mono.empty());}
}

其中webHandler.handle方法就是处理所有过滤器链的方法,该过滤器链包括globalFiltersgatewayFilters

 @Overridepublic Mono<Void> handle(ServerWebExchange exchange) {// 1. 根据路由与上下文绑定关系,获取对应的路由RouteRoute route = exchange.getRequiredAttribute(GATEWAY_ROUTE_ATTR);List<GatewayFilter> gatewayFilters = route.getFilters();// 2. 收集所有的 globalFilters 并放入List<GatewayFilter>//注意这里使用了适配器模式List<GatewayFilter> combined = new ArrayList<>(this.globalFilters);// 3. 把 gatewayFilters 也放入List<GatewayFilter>,形成一条过滤器立案combined.addAll(gatewayFilters);// 4. 根据order排序AnnotationAwareOrderComparator.sort(combined);if (logger.isDebugEnabled()) {logger.debug("Sorted gatewayFilterFactories: " + combined);}// 5. 执行过滤器链中的每一个过滤器方法!return new DefaultGatewayFilterChain(combined).filter(exchange);}

注意:在组装过滤器链的时候,是把globalFiltersgatewayFilters两种过滤器都放进了List<GatewayFilter>中,这是怎么做的呢?

这其实用到了一种 适配器 的设计模式!

  • 如果放入的是globalFilters,会先把globalFilters转化成GatewayFilterAdapterGatewayFilterAdapter在内部集成了GlobalFilter,同时也实现了GatewayFilter,使 globalFiltersgatewayFilters适配器GatewayFilterAdapter中共存!
  • 如果放入的是gatewayFilters,直接放入即可!
 //使用适配器类GatewayFilterAdapter 解决了 globalFilters想要放入List<GatewayFilter>中的类型不一致问题private static class GatewayFilterAdapter implements GatewayFilter {//集成了`GlobalFilter`private final GlobalFilter delegate;GatewayFilterAdapter(GlobalFilter delegate) {this.delegate = delegate;}//调用filter时,调的是globalFilters的filter方法!@Overridepublic Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {return this.delegate.filter(exchange, chain);}@Overridepublic String toString() {final StringBuilder sb = new StringBuilder("GatewayFilterAdapter{");sb.append("delegate=").append(delegate);sb.append('}');return sb.toString();}}

然后在执行过滤器链中的globalFiltersgatewayFiltersfilter方法时,就会为请求加上请求头、请求参数等扩展点!

3:Gateway的负载均衡是如何实现的?

Gateway的负载均衡只需要在yml中配置 uri: lb://mall-order即可实现负载均衡,底层是由全局过滤器LoadBalancerClientFilterfilter方法去做的!

以订单服务的http://localhost:8888/order/findOrderById/1为例!8888为网关Gateway的端口

 public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {// 1. 根据路由与上下文绑定关系// 获取原始的url:http://localhost:8888/order/findOrderById/1URI url = exchange.getAttribute(GATEWAY_REQUEST_URL_ATTR);String schemePrefix = exchange.getAttribute(GATEWAY_SCHEME_PREFIX_ATTR);if (url == null|| (!"lb".equals(url.getScheme()) && !"lb".equals(schemePrefix))) {return chain.filter(exchange);}addOriginalRequestUrl(exchange, url);if (log.isTraceEnabled()) {log.trace("LoadBalancerClientFilter url before: " + url);}// 2. 通过ribbon的负载均衡算法,根据服务名 去nacos选择一个实例!// 该实例就有order服务真正的 url 地址:http://localhost:9001/order/findOrderById/1final ServiceInstance instance = choose(exchange);if (instance == null) {throw NotFoundException.create(properties.isUse404(),"Unable to find instance for " + url.getHost());}// 3. 拿到原生的 uri :http://localhost:8888/order/findOrderById/1URI uri = exchange.getRequest().getURI();String overrideScheme = instance.isSecure() ? "https" : "http";if (schemePrefix != null) {overrideScheme = url.getScheme();}// 4. 拿服务实例instance的uri替换原生的uri地址 得到 新的url// 新的url: http://localhost:8888/order/findOrderById/1URI requestUrl = loadBalancer.reconstructURI(new DelegatingServiceInstance(instance, overrideScheme), uri);if (log.isTraceEnabled()) {log.trace("LoadBalancerClientFilter url chosen: " + requestUrl);}// 5. 再次记录上下文关系exchange.getAttributes().put(GATEWAY_REQUEST_URL_ATTR, requestUrl);// 6. 执行过滤器链中的其他过滤请求return chain.filter(exchange);}

Gateway源码分析相关推荐

  1. Docker源码分析(八):Docker Container网络(下)

    http://www.infoq.com/cn/articles/docker-source-code-analysis-part8 1.Docker Client配置容器网络模式 Docker目前支 ...

  2. Istio Pilot 源码分析(一)

    张海东, ‍多点生活(成都)云原生开发工程师. Istio 作为目前 Servic Mesh 方案中的翘楚,吸引着越来越多的企业及开发者.越来越多的团队想将其应用于微服务的治理,但在实际落地时却因为不 ...

  3. springcloud gateway 源码解析、请求响应流程、第三方响应结果在 gateway 的经过

    大家好,我是烤鸭: 1.  官方介绍 官方文档: 看的是 2.2.5.RELEASE 版本的 https://docs.spring.io/spring-cloud-gateway/docs/2.2. ...

  4. LWIP源码分析——ip4.c

    LWIP源码分析--ip4.c ipv4是IP栈中重要的一部分,实现功能使用了上千行代码,分析起来可能会稍显复杂,这部分采用的分析的思路是,重点思想总结部分放在前面,剩下的结合代码穿插分析 1.ipv ...

  5. Spring Cloud Alibaba - 27 Gateway源码解析

    文章目录 How it works 入口 自动配置类源码分析 How it works https://docs.spring.io/spring-cloud-gateway/docs/current ...

  6. Kafka源码分析10:副本状态机ReplicaStateMachine详解 (图解+秒懂+史上最全)

    文章很长,建议收藏起来,慢慢读! Java 高并发 发烧友社群:疯狂创客圈 奉上以下珍贵的学习资源: 免费赠送 经典图书:<Java高并发核心编程(卷1)> 面试必备 + 大厂必备 +涨薪 ...

  7. Flask源码分析(一)

    知识背景 Flask是python web框架,主要包含werkzeug和jinja2,前者是一个WSGI工具集,后者用来实现模板处理. WSGI,Werkzeug WSGI WSGI(Web Ser ...

  8. RuoYi-Vue-Plus 与 RuoYi-Cloud-Plus 高端进阶 源码分析 系列教程

    专栏地址 由项目成员 MichelleChung 书写 vue版本专栏 https://blog.csdn.net/michelle_zhong/category_11109741.html Clou ...

  9. 【Golang源码分析】Go Web常用程序包gorilla/mux的使用与源码简析

    目录[阅读时间:约10分钟] 一.概述 二.对比: gorilla/mux与net/http DefaultServeMux 三.简单使用 四.源码简析 1.NewRouter函数 2.HandleF ...

最新文章

  1. 开源网站Open-Open
  2. MongoDB命令及SQL语法对比
  3. 数据结构Queue:poll、offer、element、peek
  4. 视觉SLAM找工作面试问题集锦(转自网络)
  5. 多线程笔试题(linux)
  6. 【ElasticSearch】Es 源码之 LicenseService 源码解读
  7. mysql 断号查询_怎么查询SQL中连续编号中间的断号
  8. jquery jqplot pierenderer 饼图百分比小于3的无法显示DataLabels
  9. markdown中打勾,对号和打叉,表格内换行
  10. Android半透明对话框实现
  11. vue插槽面试题_vue面试题(一)
  12. vue中.lazy 相当于change事件
  13. ES6 import命令和import()函数区别
  14. dhcp计算机毕业论文,计算机网络毕业设计(论文)dhcp在校园网中的应用.pdf
  15. 【基础算法】Gossip协议
  16. lo流知识(字节流 字符流)
  17. R语言:根据经纬度在世界地图上画出各个点
  18. 如何一眼辨别谁有男朋友/女朋友?哈哈哈哈哈哈哈
  19. 误删了efi分区,怎么样恢复,使电脑开机回到windows
  20. Office 连供打印机无法进纸怎么办 卡纸,塞纸怎么办

热门文章

  1. JavaScript(四)——具体对象(Math、字符串对象、Date对象、Number对象及Boolean对象)
  2. 计算机网络基本操作命令的使用,计算机网络-路由器基本命令操作实验指导书--华为...
  3. 单片机c语言编程RGB,C语言将raw data(rgb/rgba)写成bmp文件(bmp24或32)
  4. C盘文件内容及清理思路
  5. 利用计算机语言实现ID3算法,机器学习之决策树学习-id3算法-原理分析及c语言代码实现.pdf...
  6. python defaultdict 类属性_Python collections.defaultdict模块用法详解
  7. 卸载nginx php mysql_ubuntu16.04彻底删除nginx+php
  8. Linux设备树语法详解【转】
  9. Centos R安装
  10. [Azure][PowerShell][ASM][12]ACL