网关(GateWay)
第一代网关zuul 1.X
Netflix开源的网关,使用Java开发,基于Servlet架构构建,便于二次开发。因为基于Servlet内部延迟严重,并发场景不友好,一个线程只能处理一次连接请求。
但由于 zuul 采取的是 servlet 2.5 阻塞IO,性能较低且 zuul2 迟迟未发布,目前不再推荐使用。
spring 推出的 “GateWay” 网关组件
Spring Cloud Gateway 使用的Webflux中的reactor-netty响应式编程组件,底层使用了Netty通讯框架。
Yaml 配置:
路由(Route)是GateWay中最基本的组件之一,表示一个具体的路由信息载体,主要由下面几个部分组成:
- id:路由唯一标识,区别于其他的route
- url: 路由指向的目的地URL,客户端请求最终被转发到的微服务
- order: 用于多个Route之间的排序,数值越小越靠前,匹配优先级越高
- predicate:断言的作用是进行条件判断,只有断言为true,才执行路由
- filter: 过滤器用于修改请求和响应信息
server:
port: 9527
spring:
application:
name: 微服务名
cloud:
gateway:
routes:
- id: 类似数据库主键字段,全局唯一即可
# uri: http://localhost:8001 # 匹配后提空服务的路由地址
uri: lb://cloud-payment-service # 注册中心中的服务名
predicates:
- Path=/payment/get/** # 断言,路径相匹配的进行路由跳转,controller请求地址
注册中心配置...
作用:
- 反向代理
- 鉴权
- 流量控制
- 熔断
- 日志监控
三大核心概念
一. Route(路由)
路由是构建网关的基本模块,它由 ID ,目标 URI ,一系列的断言和过滤器组成,如果断言为 true 则匹配该路由。
二. Predicate(断言)
参考的是 java 8的 java.util.function.Predicate 开发人员可以匹配 HTTP 请求中的所有内容(例如请求头或请求参数),如果请求与断言相匹配则进行路由。在 Yaml 文件中配置在 Predicates 结点下
1) After:在某时间点之后才可以进行访问
// 获取格式化时区时间的 Java 方法
public static void main(String[] args) {
ZonedDateTime zbj = ZonedDateTime.now(); // 根据系统的默认时区获取
System.out.print(zbj); // 2022-10-04T16:49:23.334+08:00[Asia/Shanghai]
}
# Yaml 中的配置
cloud:
gateway:
routes:
predicates: - After=2022-10-04T16:49:23.334+08:00[Asia/Shanghai]
2)Between、Before 时间同理
3)Cookie:携带 Cookie 才可以访问,以携带用户名和zzyy字段为例
# Yaml 中的配置
cloud:
gateway:
routes:
predicates:
- Cookie=username,zzyy
4)Header:请求头中携带指定字段
5)Host:请求需为指定主机(ip)
6)Method:请求需为指定请求类型(GET)
三. Filter(过滤)
指的是 Spring 框架中 GatewayFilter 的实例,使用过滤器,可以在请求被路由前或者之后对请求进行修改。
1) 生命周期 Only Two
pre:请求之前
post:请求之后
2) 种类 Only Two
GatewayFilter:单一网关过滤器
GlobalFilter:全局网关过滤器
3)自定义过滤器:
/**
* 全局网关过滤器配置,以“请求头”必须携带用户名为例
* Mono 相当于 MVC 中的 ModelAndView
*
* @Author: 魏一yi
* @Date: 2022/10/4 14:58
*/
@Component
@Slf4j
public class MyLogGateWayFilter implements GlobalFilter, Ordered {
@Override
public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
log.info("******come in MyLogGateWayFilter" + new Date());
// exchange.getRequest() 获取的对象类似于HttpServletRequest
// 请求必须带uname
String uname = exchange.getRequest().getQueryParams().getFirst("uname");
if(uname == null) {
log.info("******用户名为空,非法用户");
exchange.getResponse().setStatusCode(HttpStatus.NOT_ACCEPTABLE);
return exchange.getResponse().setComplete();
}
return chain.filter(exchange);
}
@Override
public int getOrder() {
// 数字越小优先级越高
return 0;
}
}