Spring Cloud Gateway 内置了许多路由断言工厂,可以通过配置的方式直接使用,也可以组合使用多个路由断言工厂。接下来为大家介绍几个常用的路由断言工厂类。

1)Path 路由断言工厂

Path 路由断言工厂接收一个参数,根据 Path 定义好的规则来判断访问的 URI 是否匹配。

  1. spring:
  2. cloud:
  3. gateway:
  4. routes:
  5. - id: host_route
  6. uri: http://c.biancheng.net
  7. predicates:
  8. - Path=/blog/detail/{segment}

如果请求路径为 /blog/detail/xxx,则此路由将匹配。也可以使用正则,例如 /blog/detail/** 来匹配 /blog/detail/ 开头的多级 URI。

我们访问本地的网关:http://localhost:2001/blog/detail/36185 ,可以看到显示的是 http://c.biancheng.net/blog/detail/36185 对应的内容。

2)Query 路由断言工厂

Query 路由断言工厂接收两个参数,一个必需的参数和一个可选的正则表达式。

  1. spring:
  2. cloud:
  3. gateway:
  4. routes:
  5. - id: query_route
  6. uri: http://c.biancheng.net
  7. predicates:
  8. - Query=foo, ba.

如果请求包含一个值与 ba 匹配的 foo 查询参数,则此路由将匹配。bar 和 baz 也会匹配,因为第二个参数是正则表达式。

测试链接:http://localhost:2001/?foo=baz。

3)Method 路由断言工厂

Method 路由断言工厂接收一个参数,即要匹配的 HTTP 方法。

  1. spring:
  2. cloud:
  3. gateway:
  4. routes:
  5. - id: method_route
  6. uri: http://baidu.com
  7. predicates:
  8. - Method=GET

4)Header 路由断言工厂

Header 路由断言工厂接收两个参数,分别是请求头名称和正则表达式。

  1. spring:
  2. cloud:
  3. gateway:
  4. routes:
  5. - id: header_route
  6. uri: http://example.org
  7. predicates:
  8. - Header=X-Request-Id, \d+

如果请求中带有请求头名为 x-request-id,其值与 \d+ 正则表达式匹配(值为一个或多个数字),则此路由匹配。

如果你想学习更多路由断言工厂的用法,可以参考官方文档进行学习。

自定义路由断言工厂

自定义路由断言工厂需要继承 AbstractRoutePredicateFactory 类,重写 apply 方法的逻辑。

在 apply 方法中可以通过 exchange.getRequest() 拿到 ServerHttpRequest 对象,从而可以获取到请求的参数、请求方式、请求头等信息。

apply 方法的参数是自定义的配置类,在使用的时候配置参数,在 apply 方法中直接获取使用。

命名需要以 RoutePredicateFactory 结尾,比如 CheckAuthRoutePredicateFactory,那么在使用的时候 CheckAuth 就是这个路由断言工厂的名称。代码如下所示。

  1. @Component
  2. public class CheckAuthRoutePredicateFactory
  3. extends AbstractRoutePredicateFactory<CheckAuthRoutePredicateFactory.Config> {
  4. public CheckAuthRoutePredicateFactory() {
  5. super(Config.class);
  6. }
  7. @Override
  8. public Predicate<ServerWebExchange> apply(Config config) {
  9. return exchange -> {
  10. System.err.println("进入了CheckAuthRoutePredicateFactory\t" + config.getName());
  11. if (config.getName().equals("zhangsan")) {
  12. return true;
  13. }
  14. return false;
  15. };
  16. }
  17. public static class Config {
  18. private String name;
  19. public void setName(String name) {
  20. this.name = name;
  21. }
  22. public String getName() {
  23. return name;
  24. }
  25. }
  26. }

使用示例如下所示:

  1. spring:
  2. cloud:
  3. gateway:
  4. routes:
  5. - id: customer_route
  6. uri: http://c.biancheng.net
  7. predicates:
  8. - name: CheckAuth
  9. args:
  10. name: zhangsan