SpringCloud Gateway 路由配置定位原理分析

作者:mrr 时间:2022-06-10 19:57:47 

环境:springcloud Hoxton.SR11

本节主要了解系统中的谓词与配置的路由信息是如何进行初始化关联生成路由对象的。每个谓词工厂中的Config对象又是如何被解析配置的。

所有的谓词工厂中的Config中属性值是如何被配置的。

在SpringCloud Gateway中的所有谓词工厂如下:

SpringCloud Gateway 路由配置定位原理分析

命名规则:XxxRoutePredicateFactory。所有的这些谓词工厂都是如下的继承关系


public class MethodRoutePredicateFactory extends   AbstractRoutePredicateFactory<MethodRoutePredicateFactory.Config>
//
public class PathRoutePredicateFactory extends AbstractRoutePredicateFactory<PathRoutePredicateFactory.Config>
// ...

所有的谓词工厂继承的
AbstractRoutePredicateFactory中的泛型都是内部类的Config。这个是如何被配置上值的呢?

6.1 gateway自动配置

在下面这个类中配置了所有的Predicate和Filter。


public class GatewayAutoConfiguration {
 @Bean
 @ConditionalOnEnabledPredicate
 public PathRoutePredicateFactory pathRoutePredicateFactory() {
   return new PathRoutePredicateFactory();
 }
 @Bean
 @ConditionalOnEnabledPredicate
 public QueryRoutePredicateFactory queryRoutePredicateFactory() {
   return new QueryRoutePredicateFactory();
 }
 @Bean
 public RouteLocator routeDefinitionRouteLocator(GatewayProperties properties, List<GatewayFilterFactory> gatewayFilters, List<RoutePredicateFactory> predicates, RouteDefinitionLocator routeDefinitionLocator, ConfigurationService configurationService) {
   return new RouteDefinitionRouteLocator(routeDefinitionLocator, predicates,
           gatewayFilters, properties, configurationService);
 }
 @Bean
 @Primary
 @ConditionalOnMissingBean(name = "cachedCompositeRouteLocator")
 public RouteLocator cachedCompositeRouteLocator(List<RouteLocator> routeLocators) {
   return new CachingRouteLocator(new CompositeRouteLocator(Flux.fromIterable(routeLocators)));
 }
}

这里会层层委托最终查找查找路由定位会交给
RouteDefinitionRouteLocator。CachingRouteLocator起到缓存的作用,将配置的所有路由信息保存。

注意:这里的路由信息是在容器启动后就会被初始化的。


public class CachingRouteLocator {
 private final RouteLocator delegate;

private final Flux<Route> routes;

private final Map<String, List> cache = new ConcurrentHashMap<>();

private ApplicationEventPublisher applicationEventPublisher;

public CachingRouteLocator(RouteLocator delegate) {
   this.delegate = delegate;
   routes = CacheFlux.lookup(cache, CACHE_KEY, Route.class) .onCacheMissResume(this::fetch);
 }

private Flux<Route> fetch() {
   return this.delegate.getRoutes().sort(AnnotationAwareOrderComparator.INSTANCE);
 }
}

实例化CachingRouteLocator就开始查找所有配置的Route信息。最终的会委托给
RouteDefinitionRouteLocator

RouteDefinitionRouteLocator构造函数中的initFactories方法用来映射路由工厂的XxxRoutePredicateFactory。


private void initFactories(List<RoutePredicateFactory> predicates) {
 predicates.forEach(factory -> {
   String key = factory.name();
   if (this.predicates.containsKey(key)) {
     this.logger.warn("A RoutePredicateFactory named " + key + " already exists, class: " + this.predicates.get(key) + ". It will be overwritten.");
   }
   this.predicates.put(key, factory);
 });
}

方法中解析每一个谓词工厂对应的名称然后缓存到predicates 集合中。

factory.name()方法解析谓词名称。


default String name() {
 return NameUtils.normalizeRoutePredicateName(getClass());
}

CachingRouteLocator是个缓存路由 * ,是个首选的RouteLocator(@Primary),这里将
RouteDefinitionRouteLocator进行了合并。

6.2 生成路由对象Route及Config配置

getRoutes---》convertToRoute---》combinePredicates---》lookup。

根据上面的自动配置也知道了在服务启动时就进行初始化所有路由信息了。

获取路由信息


public Flux<Route> getRoutes() {
 Flux<Route> routes = this.routeDefinitionLocator.getRouteDefinitions() .map(this::convertToRoute);
 routes = routes.onErrorContinue((error, obj) -> {
   return routes.map(route -> {
           return route;
 });
}

合并谓词


private AsyncPredicate<ServerWebExchange> combinePredicates(
           RouteDefinition routeDefinition) {
 // other code
 for (PredicateDefinition andPredicate : predicates.subList(1, predicates.size())) {
   AsyncPredicate<ServerWebExchange> found = lookup(routeDefinition, andPredicate);
   predicate = predicate.and(found);
 }
 return predicate;
}

进入lookup中


private AsyncPredicate<ServerWebExchange> lookup(RouteDefinition route, PredicateDefinition predicate) {
 RoutePredicateFactory<Object> factory = this.predicates.get(predicate.getName());
 if (factory == null) {
   throw new IllegalArgumentException("Unable to find RoutePredicateFactory with name " + predicate.getName());
 }
 // 这里将配置中(yml文件)配置的name,args和谓词工厂中的Config进行关联设置值
 Object config = this.configurationService.with(factory)
   .name(predicate.getName())
   .properties(predicate.getArgs())
   .eventFunction((bound, properties) -> new PredicateArgsEvent(
       RouteDefinitionRouteLocator.this, route.getId(), properties))
   .bind();
 // 最终调用谓词工厂(XxxRoutePredicateFactory的apply方法返回RoutePredicate该对象继承Predicate)
 return factory.applyAsync(config);
}

lookup方法中查找,也就是在这里将对应的谓词Config与RouteDefinition(Predicate)中定义的相对应的属性关联。

进入factory.applyAsync方法


@FunctionalInterface
public interface RoutePredicateFactory<C> extends ShortcutConfigurable, Configurable<C> {
 default AsyncPredicate<ServerWebExchange> applyAsync(C config) {
   return toAsyncPredicate(apply(config)); // 查看下面的6.2-1图当前apply所有的实现就是系统内部定义的XxxRoutePredicateFactory
 }
}
// apply(config),如这里配置了Path谓词,那么就会进入PathRoutePredicateFactory中的apply方法
public Predicate<ServerWebExchange> apply(Config config) {
 // other code    
 return new GatewayPredicate() {
   public boolean test() {
     // todo    
   }
 }
}
// 最后返回一个异步的谓词
public static AsyncPredicate<ServerWebExchange> toAsyncPredicate(Predicate<? super ServerWebExchange> predicate) {
 Assert.notNull(predicate, "predicate must not be null");
 // 这里from就是返回一个DefaultAsyncPredicate默认的异步谓词
 return AsyncPredicate.from(predicate);
}
static AsyncPredicate<ServerWebExchange> from( Predicate<? super ServerWebExchange> predicate) {
 return new DefaultAsyncPredicate<>(GatewayPredicate.wrapIfNeeded(predicate));
}

图6.2-1

SpringCloud Gateway 路由配置定位原理分析

最后在combinePredicates方法中将当前路由中配置的所有谓词进行了and操作返回。最终回到convertToRoute方法中将当前路由中配置的谓词,过滤器进行了整合包装返回Route(一个路由对象)


public class Route implements Ordered {
 private final String id;

private final URI uri;

private final int order;

private final AsyncPredicate<ServerWebExchange> predicate;

private final List<GatewayFilter> gatewayFilters;

private final Map<String, Object> metadata;
}

这些Route对象会被保存在上面说的
CachingRouteLocator.routes中。

6.3 定位路由

根据上面的配置RouteLocator 该类用来定位路由(查找具体的使用哪个路由);当一个请求过来会查找是哪个路由。

RouteLocator中定义了一个方法


public interface RouteLocator {

Flux<Route> getRoutes();

}

查看这个getRoutes方法是谁调用的

SpringCloud Gateway 路由配置定位原理分析

看到这个
RoutePredicateHandlerMapping是不是想起了Spring MVC中的HandlerMapping(我们所有的Controller都会被RequestMappingHanlderMapping 匹配)。通过名称也就知道了该HandlerMapping用来匹配我们的路由谓词的谁来处理路由。

接下来回到前面说的
RequestMappingHanlderMapping 对象,当我们请求一个路由地址时会执行该类中的lookup方法查找路由


protected Mono<Route> lookupRoute(ServerWebExchange exchange) {
 // 这里的this.routeLocator就是 CachingRouteLocator对象
 return this.routeLocator.getRoutes()
     .concatMap(route -> Mono.just(route).filterWhen(r -> {
       exchange.getAttributes().put(GATEWAY_PREDICATE_ROUTE_ATTR, r.getId());
       // 过滤查找符合的路由  
       return r.getPredicate().apply(exchange);
    }).doOnError(e -> logger.error(
         "Error applying predicate for route: " + route.getId(),
    e)).onErrorResume(e -> Mono.empty()))
       .next()
       .map(route -> {
         if (logger.isDebugEnabled()) {
           logger.debug("Route matched: " + route.getId());
         }
         validateRoute(route, exchange);
         return route;
    });
}

进入r.getPredicate().apply(exchange)


public interface AsyncPredicate<T> extends Function<T, Publisher<Boolean>> {
 static AsyncPredicate<ServerWebExchange> from(Predicate<? super ServerWebExchange> predicate) {
 return new DefaultAsyncPredicate<>(GatewayPredicate.wrapIfNeeded(predicate));
 }

class DefaultAsyncPredicate<T> implements AsyncPredicate<T> {

private final Predicate<T> delegate;

public DefaultAsyncPredicate(Predicate<T> delegate) {
     this.delegate = delegate;
   }

@Override
   public Publisher<Boolean> apply(T t) {
     return Mono.just(delegate.test(t));
   }

@Override
   public String toString() {
     return this.delegate.toString();
   }

}

}

这里会调用Predicate.test方法(XxxRoutePredicateFactory中的apply方法返回的GatewayPredicate)。

调用GatewayPredicate.test返回判断当前请求的路由是否匹配。

整体的一个流程:

1、系统先初始化所有的Predicate(谓词)和Filter(过滤器)

2、根据配置的路由信息(过滤器,谓词)包装返回Route对象

3、根据请求路由路径查找匹配的路由

完毕!!!

标签:SpringCloud,Gateway,路由,原理
0
投稿

猜你喜欢

  • Android仿微信图片上传带加号且超过最大数隐藏功能

    2022-12-27 13:20:03
  • android实现简单拼图游戏

    2021-11-17 09:28:23
  • 总结Java的Struts框架的异常处理方法

    2022-04-12 01:29:44
  • c# 实现语音合成

    2021-06-16 00:17:47
  • spring aop action中验证用户登录状态的实例代码

    2021-12-04 19:47:01
  • clion最新激活码+汉化的步骤详解(亲测可用激活到2089)

    2023-07-17 08:03:53
  • Java中实现可拖放图片剪裁入门教程

    2022-04-23 12:11:03
  • SpringBoot整合Dozer映射框架流程详解

    2023-03-08 02:23:48
  • Java中Date和Calendar常用方法

    2023-11-12 07:35:32
  • java开发分布式服务框架Dubbo调用过程

    2022-04-13 14:29:55
  • phonegap教程使用jspdf库在应用中生成pdf文件(pdf生成方法)

    2022-10-29 05:50:11
  • Java毕业设计实战项目之仓库管理系统的实现流程

    2021-11-11 09:42:42
  • Mybatis Plugin拦截器开发过程详解

    2021-07-15 22:30:18
  • JAVA8 Stream流中的reduce()方法详解

    2023-09-01 21:10:33
  • java 可变参数详解及实例

    2021-06-01 10:02:42
  • Java集合继承体系详解

    2023-12-03 00:22:13
  • Android 读取文件内容实现方法总结

    2022-09-15 12:57:56
  • Java Iterator迭代器_动力节点Java学院整理

    2022-11-08 13:50:39
  • Android 验证码功能实现代码

    2023-06-12 23:19:06
  • SpringBoot+Eureka实现微服务负载均衡的示例代码

    2021-09-19 07:10:13
  • asp之家 软件编程 m.aspxhome.com