一、Route Predicate Factories(路由谓词工厂)
还是先来查看官网的说明:点击此处
路由谓词工厂是微服务网关(如Spring Cloud Gateway)中用于定义路由规则的一种机制。它们用来决定哪些请求应该被路由到特定的服务。
简单来说就是:
- Route Predicate Factories 提供了一种方式来指定什么样的请求应该被路由到某个服务。这通常是基于请求的属性,比如路径、查询参数、头部信息等。【你可以创建自己的Route Predicate Factories来满足特定的需求,这样就可以定制化地处理请求。】
- 当一个请求到达网关时,网关会根据配置的规则来检查该请求是否符合某个Route Predicate Factory的条件。如果请求匹配了某个Route Predicate Factory的条件,则该请求会被路由到相应的服务。
二、Route Predicate 的配置
针对上图所示的几种内置的Route Predicate 我们来在配置文件中进行配置。
1、配置方法有两种:
- shortcuts
- fully expanded arguments.
配置Route Predicate 的股份描述如下图所示:
2、具体配置如下所示:
server:port: 9527spring:application:name: cloud-gateway #以微服务注册进consul或nacos服务列表内cloud:consul: #配置consul地址host: localhostport: 8500discovery:prefer-ip-address: trueservice-name: ${spring.application.name}gateway:routes:- id: pay_routh1 #pay_routh1 #路由的ID(类似mysql主键ID),没有固定规则但要求唯一,建议配合服务名#uri: http://localhost:8001 #匹配后提供服务的路由地址uri: lb://cloud-payment-service #匹配后提供服务的路由地址predicates:- Path=/pay/gateway/get/** # 断言,路径相匹配的进行路由- After=2023-12-30T23:02:39.079979400+08:00[Asia/Shanghai]#- Cookie=username,zzyy# - Header=X-Request-Id, \d+ # 请求头要有X-Request-Id属性并且值为整数的正则表达式#- Host=**.atguigu.com#- Query=username, \d+ # 要有参数名username并且值还要是整数才能路由#- RemoteAddr=192.168.124.1/24 # 外部访问我的IP限制,最大跨度不超过32,目前是1~24它们是 CIDR 表示法。- id: pay_routh2 #pay_routh2 #路由的ID(类似mysql主键ID),没有固定规则但要求唯一,建议配合服务名#uri: http://localhost:8001 #匹配后提供服务的路由地址uri: lb://cloud-payment-servicepredicates:- Path=/pay/gateway/info/** # 断言,路径相匹配的进行路由
小贴士:
如何获得ZonedDateTime??
很简单,创建一个main方法,代码如下:
import java.time.ZoneId;
import java.time.ZonedDateTime;public class ZonedDateTimeDemo
{public static void main(String[] args){ZonedDateTime zbj = ZonedDateTime.now(); // 默认时区System.out.println(zbj);}
}
三、自定义断言规则
除了上面的内置的几种规则之外,我们还可以自定义断言规则。注意格式是XXRoutePredicateFactory
.
为什么要自定义断言规则呢??
那肯定就是原有的断言配置不满足业务需求喽
话不多说,直接开始自定义路由断言规则步骤:
1.新建类名XXX需要以RoutePredicateFactory
结尾并继承AbstractRoutePredicateFactory
类自
2新建apply方法所需要的静态内部类MyRoutePredicateFactory.Config
这个Config类就是我们的路由断言规则
3.空参构造方法,内部调用super
4.重写apply方法
完整代码如下所示:
import jakarta.validation.constraints.NotEmpty;
import lombok.Getter;
import lombok.Setter;
import org.springframework.cloud.gateway.handler.predicate.AbstractRoutePredicateFactory;
import org.springframework.stereotype.Component;
import org.springframework.validation.annotation.Validated;
import org.springframework.web.server.ServerWebExchange;import java.util.ArrayList;
import java.util.List;
import java.util.function.Predicate;@Component
public class MyRoutePredicateFactory extends AbstractRoutePredicateFactory<MyRoutePredicateFactory.Config>
{public MyRoutePredicateFactory(){super(MyRoutePredicateFactory.Config.class);}@Validatedpublic static class Config{@Setter@Getter@NotEmptyprivate String userType; //钻、金、银等用户等级}@Overridepublic Predicate<ServerWebExchange> apply(MyRoutePredicateFactory.Config config){return new Predicate<ServerWebExchange>(){@Overridepublic boolean test(ServerWebExchange serverWebExchange){//检查request的参数里面,userType是否为指定的值,符合配置就通过String userType = serverWebExchange.getRequest().getQueryParams().getFirst("userType");if (userType == null) return false;//如果说参数存在,就和config的数据进行比较if(userType.equals(config.getUserType())) {return true;}return false;}};}// 如果缺少shortcutFieldOrder方法的实现,就不支持短格式,所以要重写@Overridepublic List<String> shortcutFieldOrder() {return Collections.singletonList("userType");}
}
yml配置如下:和内置的配置使用都是一样的
spring:application:name: cloud-gateway #以微服务注册进consul或nacos服务列表内cloud:consul: #配置consul地址host: localhostport: 8500discovery:prefer-ip-address: trueservice-name: ${spring.application.name}gateway:routes:- id: pay_routh1 uri: lb://cloud-payment-service #匹配后提供服务的路由地址predicates:- Path=/pay/gateway/get/** - After=2023-12-30T23:02:39.079979400+08:00[Asia/Shanghai]- My=diamond # 自定义的配置
我的浏览器地址栏访问如下地址(顺利通过测试):