目录

Gateway实战入门(四)、断言-请求头以及请求权重分流等

spring cloud-Gateway:断言-请求头以及请求权重分流等

一、断言Header信息要求

需要参数header 和regexp(正则表达式),即key和value,匹配请求携带的信息。实际就是对应请求头携带的信息,官网的案例是X-Request-Id,所以我们第一个案例就以它为例 ;)

项目前置环境要求

  • 启动nacos服务,对应版本: 2.2.3 ,jdk1.8或以上。
  • 至少1个服务提供者,端口9001,提供一个web接口,【我的是/hello,下文也是以此接口为例】;
  • 一个网关服务端口 9966 ;

案例一、断言-请求头信息-匹配X-Request-Id

1、配置文件及代码

网关gateway的yaml配置文件:

cpp 复制代码
server:
  port: 9966
spring:
  application:
    name: gateway-service
  cloud:
    nacos:
      discovery:
        server-addr: 192.168.0.103:80
    gateway:
      discovery:
        locator:
          enabled: false # 关闭注册中心路由功能
      routes:
        - id: nacos-provider
          uri: lb://nacos-provider #路由到注册中心,服务为nacos-provider的服务
          predicates:
            - Path=/nancy/** # 路径匹配,则进行路由
            - Header=X-Request-Id,\d+ #表示数字
          filters:
            - StripPrefix=1 # 截取掉断言路径的第一部分
management:
  endpoint:
    web:
      exposure:
        include: '*'

网关gateway主启动类:

cpp 复制代码
package com.nancy.gateway9966;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.discovery.EnableDiscoveryClient;

@EnableDiscoveryClient
@SpringBootApplication
public class Gateway9966Application {
    public static void main(String[] args) {
        SpringApplication.run(Gateway9966Application.class, args);
    }
}

服务提供者9001,接口

cpp 复制代码
/**
 * @Auther: gina
 * @Date: 2025-03-10
 * @Description:
 */
@RestController
@Slf4j
public class UserController {

    @GetMapping("/hello")
    public String hello(HttpServletRequest request) {
        log.info("........Nacos Provider run.......... ");
        return "9001-----hello Nancy...";
    }
}

服务提供者yaml

cpp 复制代码
server:
  port: 9001

spring:
  application:
    name: nacos-provider
  cloud:
    discovery:
      server-addr: 192.168.0.103:80
management:
  endpoint:
    web:
      exposure:
        include: '*'

服务提供者主启动类

cpp 复制代码
@EnableDiscoveryClient
@SpringBootApplication
public class NacosProvider9001Application {
    public static void main(String[] args) {
        SpringApplication.run(NacosProvider9001Application.class, args);
    }

}

服务提供者,引入maven依赖

cpp 复制代码
 		<dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
        </dependency>
        <dependency>
            <groupId>com.alibaba.cloud</groupId>
            <artifactId>spring-cloud-starter-alibaba-nacos-discovery</artifactId>
        </dependency>
        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>
2、测试

启动9001服务,gateway9966网关侧服务。访问链接:http://localhost:9966/nancy/hello

案例二、断言-请求头信息-匹配API版本

场景

案例一的粒度要粗一些,属于服务之间的隔离;但实际工作中会有粒度更细的场景,比如API 可以同时存在多个版本的场景,举个实际场景的例子:

手机APP,新业务迭代,对于老版本app也要做兼容,但老app界面框架无法更改(涉及到合规和用户体验的考量,一般不会选择强制更新),这时就会有相同业务不同api版本的诉求。

而落地实现的方式之一,可以基于spring cloud 的gateway 的断言来实现,通过断言做API版本隔离,不同版本路由到不同的服务;话不多说直接上实例。

主要配置信息

网关gateway9966的yaml文件关于路由的内容设置:

cpp 复制代码
 spring:
  cloud:
    gateway:
      routes:
        - id: route_api_v1
          uri: http://nacos-consumer/testV1
          predicates:
            - Header=X-API-Version, 1.1.0  # 如果 API 版本为 1.1.0,则路由到 v1 服务
        - id: route_api_v2
          uri: http://nacos-consumer/testV2
          predicates:
            - Header=X-API-Version, 1.2.0  # 如果 API 版本为 1.2.0,则路由到 v2 服务

测试同案例一,不再赘述。

案例三、断言-请求头信息:匹配请求来源

场景

在处理跨域请求时,会在请求Header中添加Origin字段。使用Header断言可以根据Origin字段的值判断请求的来源,从而设定只限某些来源的请求来访问资源。

主要配置信息

gateway网关侧yaml文件主要配置信息如下:

dart 复制代码
spring:
  cloud:
    gateway:
      routes:
        - id: allowed_origin_route
          uri: http://backend-service
          predicates:
            - Header=Origin, https://nancy.com  # 只允许来自 https://nancy.com 的请求

案例四、断言-请求头信息:匹配用户权限/角色

场景

微服务架构中,在请求Header里可能会增加用户的角色信息,比如X-User-Role。基于 此断言的Header信息里用户的角色,将请求路由到不同的资源或服务。

主要配置信息

gateway网关侧yaml文件主要配置信息如下:

dart 复制代码
spring:
  cloud:
    gateway:
      routes:
        - id: admin_route
          uri: http://nacos-consumer/admin
          predicates:
            - Header=X-User-Role, admin  # 用户角色是管理员,则路由到admin服务
        - id: user_route
          uri: http://nacos-consumer/user
          predicates:
            - Header=X-User-Role, user  # 用户角色是普通用户,则路由到user服务

二、断言-请求方式

Method:可以设置一个或多个参数,匹配HTTP请求,比如GET、POST

案例、请求方式Demo

主要配置信息
css 复制代码
server:
  port: 9966
spring:
  application:
    name: gateway-service
  cloud:
    nacos:
      discovery:
        server-addr: 192.168.0.103:80
    gateway:
      discovery:
        locator:
          enabled: false # 开启注册中心路由功能
      routes:
        - id: nacos-provider
          uri: lb://nacos-provider #路由到注册中心,服务为nacos-provider的服务
          predicates:
            - Path=/nancy/** # 路径匹配,则进行路由
            - Method=POST,GET # 匹配GET请求或者POST请求

          filters:
            - StripPrefix=1 # 截取掉断言路径的第一部分
management:
  endpoint:
    web:
      exposure:
        include: '*'
测试

访问路径:http://localhost:9966/nancy/hello

1、代码中设置的请求方式只有post和get两种,如果访问put方式,会报404,如图:

2、如果请求方式设置为get,则接口是通的,由网关路由到9002端口的hello接口,如下图:

三、断言-查询条件

案例、断言-查询条件QueryDemo

Query:需要指定一个或者多个参数,一个必须参数和一个可选的正则表达式,匹配请求中是否包含第一个参数,

如果有两个参数,则匹配请求中第一个参数的值是否符合正则表达式。

主要配置信息

gateway的配置信息

css 复制代码
server:
  port: 9966
spring:
  application:
    name: gateway-service
  cloud:
    nacos:
      discovery:
        server-addr: 192.168.0.103:80
    gateway:
      discovery:
        locator:
          enabled: false # 开启注册中心路由功能
      routes:
        - id: nacos-provider
          uri: lb://nacos-provider #路由到注册中心,服务为nacos-provider的服务
          predicates:
            - Path=/nancy/** # 路径匹配,则进行路由
            - Query=id,.+ # 匹配请求参数,这里如果需要匹配多个参数,可以写多个Query

          filters:
            - StripPrefix=1 # 截取掉断言路径的第一部分
management:
  endpoint:
    web:
      exposure:
        include: '*'
测试

测试链接:http://localhost:9966/nancy/hello

当不传id时,测试不通过:

传id以及value值时,测试通过:

四、断言-将请求按设定的权重进行分流

实现路由权重,需要参数 group 和weight(int);按照路由权重选择同一个分组中的路由。

涉及到要用权重分流的场景

案例、断言-WeightDemo

gateway的配置信息

该路由配置 约 80%的流量转发到服务 nacos-consumer ,约20%的流量转发给服务nacos-consumer-openFeign

路由属性Weight 接收两个参数:groupweight(int),分流权重按组计算,在断言里配置权重路由配置如下:

主要配置信息

gateway的yaml文件配置如下:

css 复制代码
 server:
  port: 9999
spring:
  application:
    name: gateway-service
  cloud:
    nacos:
      discovery:
        server-addr: 172.18.101.197:80
    gateway:
      discovery:
        locator:
          enabled: false # 开启注册中心路由功能
      routes:
        # 配置流量分流权重
        - id: nacos-consumer-router
          uri: lb://nacos-consumer #路由到注册中心,服务为nacos-consumer的服务
          predicates:
            - Path=/testC/** # 路径匹配,则进行路由
            - Weight=group1, 2
          filters:
            - StripPrefix=1 # 截取掉断言路径的第一部分

        - id: nacos-consumer-openfeign
          uri: lb://nacos-consumer-openfeign #路由到注册中心,服务为nacos-consumer的服务
          predicates:
            - Path=/testC/** # 路径匹配,则进行路由
            - Weight=group1, 8
          filters:
            - StripPrefix=1 # 截取掉断言路径的第一部分
 
management:
  endpoint:
    web:
      exposure:
        include: '*'
 

9101和9102接口,分别为:

接口映射明不变,返回值为对应端口

css 复制代码
	//9102
	static  AtomicInteger a1=new AtomicInteger(0);

    @GetMapping("/abTest")
    public String versionB() throws InterruptedException {
        log.info("端口9102 调用次数 " + a1.incrementAndGet());
        return "version_B--9102";
    }
   //9101
    private static AtomicInteger a = new AtomicInteger(0);

    @GetMapping("/abTest")
    @SentinelResource(value = "version_A")
    public String versionB() throws InterruptedException {
        log.info("端口9101 调用次数 " + a.incrementAndGet());
        return "version_A--9101";
    } 
测试

启动三个服务,注册中心可以看到服务均已注册成功:

测试链接:http://localhost:9999/testC/abTest

验证分流情况,预期结果:9101 约20%的流量,9102约80%的流量。

实际结果:

通过jmeter测试,总请求数量设定1000,从控制台可观测到结果:

可看出请求分流权重,实际结果跟预期相符。

本文是转载文章,点击查看原文
如有侵权,请联系 xyy@jishuzhan.net 删除
相关推荐
艾妮艾妮9 分钟前
C语言常见3种排序
java·c语言·开发语言·c++·算法·c#·排序算法
java技术小馆10 分钟前
Zookeeper中的Zxid是如何设计的
java·分布式·zookeeper·云原生
葵野寺20 分钟前
【多线程】synchronized锁升级和优化
java·开发语言·java-ee
SimonKing29 分钟前
因为不知道条件注解@Conditional,错失15K的Offer!
java·后端·架构
橘猫云计算机设计30 分钟前
基于springboot微信小程序的旅游攻略系统(源码+lw+部署文档+讲解),源码可白嫖!
java·spring boot·后端·微信小程序·毕业设计·旅游
落榜程序员31 分钟前
Java 基础-30-单例设计模式:懒汉式与饿汉式
java·开发语言
顾林海31 分钟前
深度解析ArrayList工作原理
android·java·面试
雷渊33 分钟前
spring-IoC容器启动流程源码分析
java·后端·面试
用户33154891110738 分钟前
一招搞定Java线程池炸弹,系统吞吐量暴增10倍!
java·后端
努力的搬砖人.42 分钟前
maven如何使用
java·后端·面试·maven