天天看点

一文详解springcloud中的重要组件 “网关” Gateway的前世今生

网关(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中最基本的组件之一,表示一个具体的路由信息载体,主要由下面几个部分组成:
  1. id:路由唯一标识,区别于其他的route
  2. url: 路由指向的目的地URL,客户端请求最终被转发到的微服务
  3. order: 用于多个Route之间的排序,数值越小越靠前,匹配优先级越高
  4. predicate:断言的作用是进行条件判断,只有断言为true,才执行路由
  5. 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;
    }
}