Gateway 网关学习
1.概述
Gateway是在Spring生态系统之上构建的API网关服务,基于Spring5,Spring Boot2和Project Reactor等技术。
Gateway旨在替工一种简单而有效的方式来对API进行路由,以及提供一些强大的过滤器功能,例如:熔断,限流,重试等。
总的来说,SpringCloud Gateway 使用的Webflux中的reactor-netty响应式编程组件,底层使用了Netty通讯框架。
2.为什么要使用Gateway网关
3.三大核心:
Route(路由):是构建网关的基本模块,它由ID,目标URL,一系列的断言和过滤器组成,如果断言为true则匹配该路由。
Predicate(断言):开发人员可以匹配HTTP请求重的所有内容(例如请求头或请求参数),如果请求与断言相匹配则进行路由·
Filter(过滤):指的是Spring框架中GatewayFilter的实例,使用过滤器,可以在请求被路由前或者之后对请求进行修改。
4.添加新的模块(9527)进行测试
添加pom 修改yml 写主启动类
另外pom中要记得取消web模块,否则会报错
server:
port: 9527
spring:
application:
name: cloud-gateway
cloud:
gateway:
discovery:
locator:
enabled: true #开启从注册中心动态创建路由的功能,利用微服务名进行路由
routes:
- id: payment_routh #payment_route #路由的ID,没有固定规则但要求唯一,建议配合服务名
uri: http://localhost:8001 #匹配后提供服务的路由地址
#uri: lb://cloud-payment-service #匹配后提供服务的路由地址
predicates:
- Path=/payment/get/** # 断言,路径相匹配的进行路由
- id: payment_routh2 #payment_route #路由的ID,没有固定规则但要求唯一,建议配合服务名
uri: http://localhost:8001 #匹配后提供服务的路由地址
#uri: lb://cloud-payment-service #匹配后提供服务的路由地址
predicates:
- Path=/payment/lb/** # 断言,路径相匹配的进行路由
#- After=2020-02-21T15:51:37.485+08:00[Asia/Shanghai]
#- Cookie=username,zzyy
#- Header=X-Request-Id, \d+ # 请求头要有X-Request-Id属性并且值为整数的正则表达式
eureka:
instance:
hostname: cloud-gateway-service
client: #服务提供者provider注册进eureka服务列表内
service-url:
register-with-eureka: true
fetch-registry: true
defaultZone: http://eureka7001.com:7001/eureka
启动7001Eureka注册中心 8001Payment支付模块 最后启动网关9527
测试发现注册中心中有9527和8001两个模块
然后启动8001查询,得到结果;用端口号9527查询也是相同结果。由此发现,网关9527隐藏了真实端口8001。
5.Gateway的网关配置:
Gateway的网关配置有两种方法:
(1):在配置文件yml中配置,见上方案例。就是yml文件会越来越庞大。
(2):代码中注入RouteLocator的Bean
业务案例:通过9527网关访问到外网的新闻百度地址。
新建一个配置类(Config)
@Configuration
public class GateWayConfig {
@Bean
public RouteLocator customRouteLocator(RouteLocatorBuilder routeLocatorBuilder){
RouteLocatorBuilder.Builder routes = routeLocatorBuilder.routes();
routes.route("path_route_atguigu" ,
r -> r.path("/guonei")
.uri("http://news.baidu.com/guonei")).build();
return routes.build();
}
}
由此成功访问到国内新闻地址
6.通过微服务名实现动态路由
启动注册中心7001,以及支付模块8001,8002
之后在9527中的yml配置文件中:开启从注册中心动态创建路由的功能,利用微服务名进行路由;并对地址进行修改换成uri:lb
discovery:
locator:
enabled: true #开启从注册中心动态创建路由的功能,利用微服务名进行路由
#uri: http://localhost:8001 #匹配后提供服务的路由地址
uri: lb://cloud-payment-service #匹配后提供服务的路由地址
#uri: http://localhost:8001 #匹配后提供服务的路由地址
uri: lb://cloud-payment-service #匹配后提供服务的路由地址
由测试结果得,通过这样的方式实现了动态路由。
7.Predicate的使用
启动网关9527发现:
所以我们上面使用的path(路径相匹配的进行路由)只是其中一种,还可以使用after,before等
After后跟时间,限制访问时间。between和before使用基本相同。
Cookie:
predicates: - Cookie=chocolate , ch.p
- Cookie=username,zzyy
Header:
- Header=X-Request-Id, \d+ # 请求头要有X-Request-Id属性并且值为整数的正则表达式
所以,说白了,Predicate就是为了实现一组匹配规则,让请求过来找到对应的Route进行处理.这么多使断言获得了十分强大得功能支持。
8.Filter(过滤器)的使用
Filter过滤指的是Spring框架中GatewayFilter的实例,使用过滤器,可以在请求被路由前或者之后对请求进行修改。
自定义一个过滤器:
@Component
@Slf4j
public class MyLogGateWayFilter implements GlobalFilter , Ordered {
@Override
public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
log.info("**************come in MyLogGateWayFilter" + new Date());
String uname = exchange.getRequest().getQueryParams().getFirst("uname");
if (uname == null) { //判断用户是否合法,非法用户离开,合法用户放行
log.info("**********用户名为null,非法用户,/(ㄒoㄒ)/~~");
exchange.getResponse().setStatusCode(HttpStatus.NOT_ACCEPTABLE);
return exchange.getResponse().setComplete();
}
return chain.filter(exchange);
}
@Override
public int getOrder() {
return 0;
}
}
测试发现网关生效。