7.GateWay 服务网关

7.GateWay 服务网关,第1张

7.GateWay 服务网关 1.GateWay概述 1.1 GateWay的出现

上一代zuul 1.x官网

https://github.com/Netflix/zuul/wiki

新一代Gateway官网

https://docs.spring.io/spring-cloud-gateway/docs/current/reference/html/

SpringCloud自研了一个网关SpringCloud Gateway替代Zuul,即Gateway是原zuul1.x版的替代。

1.2 GateWay是什么?

​ SpringCloud Gateway是Spring Cloud的一个全新项目,基于Spring 5.0+Spring Boot 2.0和Project Reactor等技术,在Spring生态系统之上构建的API网关服务。它旨在为微服务架构提供—种简单有效的统一的API路由管理方式,以及提供一些强大的过滤器功能,例如:熔断、限流、重试等。

​ SpringCloud Gateway作为Spring Cloud 生态系统中的网关,目标是替代Zuul,在Spring Cloud 2.0以上版本中,没有对新版本的Zuul 2.0 以上最新高性能版本进行集成,仍然还是使用的Zuul 1.x 非Reactor模式的老版本。而为了提升网关的性能,SpringCloud自研的Gateway是基于WebFlux框架实现的,而WebFlux框架底层则使用了高性能的Reactor模式通信框架Netty。

1.3 GateWay的作用

​ Spring Cloud Gateway的目标提供统一的路由方式且基于 Filter链的方式提供了网关基本的功能,例如:安全,监控/指标,和限流。其作用是:方向代理、鉴权、流量控制、熔断、日志监控等。

网关的作用如下:

**反向路由:**很多时候,公司不想让外部人员看到我们公司的内部,就需要网关来进行反向路由。即将外部请求转换成内部具体服务调用。**安全认证:**网络中会有很多恶意访问,譬如爬虫,譬如黑客攻击,网关维护安全功能。**限流熔断:**当请求很多服务不堪重负,会让我们的服务自动关闭,导致不能用服务。限流熔断可以有效的避免这类问题。**日志监控:**所有的外面的请求都会经过网关,这样我们就可以使用网关来记录日志信息。**灰度发布,蓝绿部署。**是指能够平滑过渡的一种发布方式。在其上可以进行 A/B testing。即让一部分用户继续用产品特性 A,一部分用户开始用产品特性 B,如果用户对 B 没有什么反对意见,那么逐步扩大范围,把所有用户都迁移到 B 上面来。

微服务架构图

1.4 GateWay非阻塞异步模型

Gateway是基于异步非阻塞模型上进行开发的

SpringCloud Gateway具有如下特性

基于Spring framework 5,Project Reactor和Spring Boot 2.0进行构建;

动态路由:能够匹配任何请求属性;

可以对路由指定Predicate (断言)和Filter(过滤器);

集成Spring Cloud 服务发现功能;

易于编写的Predicate (断言)和Filter (过滤器);

请求限流功能;

支持路径重写;

1.5 SpringCloud Gateway与Zuul的区别

在SpringCloud Finchley正式版之前,Spring Cloud推荐的网关是Netflix提供的Zuul。

Zuul 1.x,是一个基于阻塞I/O的API Gateway。Zuul 1.x基于Servlet 2.5使用阻塞架构它不支持任何长连接( 如WebSocket ),Zuul的设计模式和Nginx较像,每次I/О *** 作都是从工作线程中选择一个执行,请求线程被阻塞到工作线程完成,但是差别是Nginx用C++实现,Zuul用Java实现,而JVM本身会有第一次加载较慢的情况,使得Zuul的性能相对较差。

Zuul 2.x,理念更先进,想基于Netty非阻塞和支持长连接,但SpringCloud目前还没有整合。Zuul 2.x的性能较Zuul 1.x有较大提升。

在性能方面,根据官方提供的基准测试 Spring Cloud Gateway的RPS(每秒请求数)是Zuul的1.6倍。
Spring Cloud Gateway 建立在Spring framework 5、Project Reactor和Spring Boot2之上,使用非阻塞API。
Spring Cloud Gateway 还支持WebSocket,并且与Spring紧密集成拥有更好的开发体验。

Zuul 1.x 模型

Springcloud中所集成的Zuul版本,采用的是Tomcat容器,使用的是传统的Serviet IO处理模型。

Servlet的生命周期?

servlet由servlet container进行生命周期管理。container启动时构造servlet对象并调用servlet init()进行初始化;
container运行时接受请求,并为每个请求分配一个线程(一般从线程池中获取空闲线程)然后调用service);
container关闭时调用servlet destory()销毁servlet。

Zuul 1.x 模式的缺点:

Servlet是一个简单的网络IO模型,当请求进入Servlet container时,Servlet container就会为其绑定一个线程,在并发不高的场景下这种模型是适用的。但是一旦高并发,线程数量就会上涨,而线程资源代价是昂贵的(上线文切换,内存消耗大)严重影响请求的处理时间。在一些简单业务场景下,不希望为每个request分配一个线程,只需要1个或几个线程就能应对极大并发的请求,这种业务场景下servlet模型没有优势。

所以Zuul 1.x 是基于servlet之上的一个阻塞式处理模型,即Spring实现了处理所有request请求的一个servlet (DispatcherServlet) 并由该servlet阻塞式处理处理。所以SpringCloud Zuul无法摆脱servlet模型的弊端。

Gateway模型

传统的Web框架,比如说: Struts2,SpringMVC等都是基于Servlet APl与Servlet容器基础之上运行的。

但是在Servlet3.1之后有了异步非阻塞的支持。而WebFlux是一个典型非阻塞异步的框架,它的核心是基于Reactor的相关API实现的。相对于传统的web框架来说,它可以运行在诸如Netty,Undertow及支持Servlet3.1的容器上。正因为非阻塞式+函数式编程,所以Spring 5必须让你使用JDK8。

Spring WebFlux是Spring 5.0 引入的新的响应式框架,区别于Spring MVC,它不需要依赖Servlet APl,它是完全异步非阻塞的,并且基于Reactor来实现响应式流规范。

Spring Cloud Gateway requires the Netty runtime provided by Spring Boot and Spring Webflux. It does not work in a traditional Servlet Container or when built as a WAR.

https://docs.spring.io/spring-cloud-gateway/docs/current/reference/html/

2.Gateway工作流程 2.1 三大核心概念

Route(路由) - 路由是构建网关的基本模块,它由ID,目标URI,一系列的断言和过滤器组成,如断言为true则匹配该路由;Predicate(断言) - 开发人员可以匹配HTTP请求中的所有内容(例如请求头或请求参数),如果请求与断言相匹配则进行路由;(参考的是Java8 的java.util.function.Predicate)Filter(过滤) - 指的是Spring框架中GateWayFilter的实例,使用过滤器,可以在请求被路由前或者之后对请求进行修改。

Web请求,通过一些匹配条件,定位到真正的服务节点。并在这个转发过程的前后,进行一些精细化控制。Predicate就是我们的匹配条件;而Fliter,就可以理解为一个无所不能的拦截器。有了这两个元素,再加上目标URI,就可以实现一个具体的路由了。

2.2 Gateway工作流程

官网 https://docs.spring.io/spring-cloud-gateway/docs/current/reference/html/#gateway-how-it-works

1.客户端向Spring Cloud Gateway发出请求。然后在Gateway Handler Mapping 中找到与请求相匹配的路由,将其发送到GatewayWeb Handler。

2.Handler再通过指定的过滤器链来将请求发送到我们实际的服务执行业务逻辑,然后返回。

3.过滤器之间用虚线分开是因为过滤器可能会在发送代理请求之前(“pre”)或之后(“post")执行业务逻辑。

4.Filter在“pre”类型的过滤器可以做参数校验、权限校验、流量监控、日志输出、协议转换等,在“post”类型的过滤器中可以做响应内容、响应头的修改,日志的输出,流量监控等有着非常重要的作用。

核心逻辑:路由转发 + 执行过滤器链。

3.Gateway8899搭建 3.1 新建Module模块

cloudalibaba-gateway-8899

3.2 修改POM文件


    
        clouddemo
        com.igeek.cloud
        1.0-SNAPSHOT
    
    4.0.0

    cloudalibaba-gateway-8899

    
        
        
            org.springframework.cloud
            spring-cloud-starter-gateway
        

        
        
            com.alibaba.cloud
            spring-cloud-starter-alibaba-nacos-config
        

        
        
            com.alibaba.cloud
            spring-cloud-starter-alibaba-nacos-discovery
            
                
                    org.springframework.cloud
                    spring-cloud-starter-netflix-ribbon
                
            
        
        
        
            org.springframework.cloud
            spring-cloud-starter-loadbalancer
            2.2.1.RELEASE
        

        
        
            com.igeek.cloud
            cloud-api-commons
            ${project.version}
        

        
        
            org.springframework.boot
            spring-boot-devtools
            runtime
            true
        
        
            org.projectlombok
            lombok
            true
        
        
            org.springframework.boot
            spring-boot-starter-test
            test
        
    


3.3 修改YML文件
server:
  port: 8899

spring:
  application:
    name: cloud-gateway
  cloud:
    nacos:
      discovery:
        #server-addr: localhost:8848  #配置Nacos地址,注册在此处
        server-addr: 106.15.66.190:80 #配置虚拟机上通过Nginx访问Nacos

#消费者将要去访问的微服务名称(注册成功进nacos的微服务提供者)
service-url:
  nacos-user-service: http://nacos-payment-provider
3.4 新建主启动类

GateWayMain8899类

package com.igeek.springcloud;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.discovery.EnableDiscoveryClient;

@EnableDiscoveryClient
@SpringBootApplication
public class GateWayMain8899 {

    public static void main(String[] args) {
        SpringApplication.run(GateWayMain8899.class , args);
    }

}
3.5 YML新增网关配置

8899网关如何做路由映射 ?

cloudalibaba-provider-payment-9002 模块,controller的访问地址如下:

/payment/nacos/{id}

/payment/get/{id}

若不想暴露9002端口,则在9002外面套一层8899

application.yml中新增网关配置

server:
  port: 8899

spring:
  application:
    name: cloud-gateway
  cloud:
    #############################网关配置 gateway ###########################
    gateway:
      routes:
        - id: payment_route1 #payment_route    #路由的ID,没有固定规则但要求唯一,建议配合服务名
          uri: http://localhost:9002           #匹配后提供服务的路由地址
          predicates:
            - Path=/payment/get/**             # 断言,路径相匹配的进行路由

        - id: payment_route2 #payment_route    #路由的ID,没有固定规则但要求唯一,建议配合服务名
          uri: http://localhost:9002           #匹配后提供服务的路由地址
          predicates:
            - Path=/payment/nacos/**           # 断言,路径相匹配的进行路由

    #############################服务注册中心 nacos #######################################
    nacos:
      discovery:
        #server-addr: localhost:8848  #配置Nacos地址,注册在此处
        server-addr: 106.15.66.190:80 #配置虚拟机上通过Nginx访问Nacos

#消费者将要去访问的微服务名称(注册成功进nacos的微服务提供者)
service-url:
  nacos-user-service: http://nacos-payment-provider
3.6 测试

启动9002 cloudalibaba-provider-payment-9002微服务

启动8899 cloudalibaba-gateway-8899网关

访问地址

添加网关前 - http://localhost:9002/payment/get/1

添加网关后 - http://localhost:8899/payment/get/1

两者访问成功,返回相同结果 4.Gateway配置路由的两种方式

方式一:在配置文件yml中配置

方式二:代码中注入RouteLocator的Bean

4.1 官网参考

https://docs.spring.io/spring-cloud-gateway/docs/current/reference/html/#modifying-the-way-remote-addresses-are-resolved

GatewayConfig.java

RemoteAddressResolver resolver = XForwardedRemoteAddressResolver
    .maxTrustedIndex(1);

...

.route("direct-route",
    r -> r.remoteAddr("10.1.1.1", "10.10.1.1/24")
        .uri("https://downstream1")
.route("proxied-route",
    r -> r.remoteAddr(resolver, "10.10.1.1", "10.10.1.1/24")
        .uri("https://downstream2")
)

https://docs.spring.io/spring-cloud-gateway/docs/current/reference/html/#spring-cloud-circuitbreaker-filter-factory

@Bean
public RouteLocator routes(RouteLocatorBuilder builder) {
    return builder.routes()
        .route("circuitbreaker_route", r -> r.path("/consumingServiceEndpoint")
            .filters(f -> f.circuitBreaker(c -> c.name("myCircuitBreaker").fallbackUri("forward:/inCaseOfFailureUseThis"))
                .rewritePath("/consumingServiceEndpoint", "/backingServiceEndpoint")).uri("lb://backing-service:8088")
        .build();
}
4.2 新建GateWayConfig配置类

在cloudalibaba-gateway-8899模块中,新建GateWayConfig类

package com.igeek.springcloud.config;

import org.springframework.cloud.gateway.route.RouteLocator;
import org.springframework.cloud.gateway.route.builder.RouteLocatorBuilder;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;


@Configuration
public class GateWayConfig {

    @Bean
    public RouteLocator customRouteLocator(RouteLocatorBuilder routeLocatorBuilder)
    {
        RouteLocatorBuilder.Builder routes = routeLocatorBuilder.routes();

        routes.route("payment_route1",
                r -> r.path("/payment/get/**")
                        .uri("http://localhost:9002")).build();

        routes.route("payment_route2",
                r -> r.path("/payment/nacos/**")
                        .uri("http://localhost:9002")).build();

        return routes.build();
    }

}
4.3 测试

重启8899 cloudalibaba-gateway-8899网关

访问地址

网关1 - http://localhost:8899/payment/nacos/1

网关2 - http://localhost:8899/payment/get/1

5.GateWay配置动态路由

默认情况下Gateway会根据注册中心注册的服务列表,以注册中心上微服务名为路径创建动态路由进行转发,从而实现动态路由的功能。

5.1 修改YML文件

DiscoveryLocatorProperties 类

第一步:开启从注册中心动态创建路由的功能,利用微服务名进行路由

spring.cloud.gateway.discovery.locator.enabled=true

第二步:需要注意的是uri的协议为lb,表示启用Gateway的负载均衡功能。

lb://serviceName是Spring Cloud GateWay在微服务中自动为我们创建的负载均衡uri。

application.yml

server:
  port: 8899

spring:
  application:
    name: cloud-gateway
  cloud:
    #############################网关配置 gateway ###########################
    gateway:
      routes:
        - id: payment_route1 #payment_route    #路由的ID,没有固定规则但要求唯一,建议配合服务名
          #uri: http://localhost:9002          #匹配后提供服务的路由地址
          uri: lb://nacos-payment-provider   #匹配后提供服务的路由地址
          predicates:
            - Path=/payment/get/**            # 断言,路径相匹配的进行路由

        - id: payment_route2 #payment_route   #路由的ID,没有固定规则但要求唯一,建议配合服务名
          #uri: http://localhost:9002          #匹配后提供服务的路由地址
          uri: lb://nacos-payment-provider   #匹配后提供服务的路由地址
          predicates:
            - Path=/payment/nacos/**          # 断言,路径相匹配的进行路由
      discovery:
        locator:
          enabled: true  #开启从注册中心动态创建路由的功能,利用微服务名进行路由

    #############################服务注册中心 nacos #######################################
    nacos:
      discovery:
        #server-addr: localhost:8848  #配置Nacos地址,注册在此处
        server-addr: 106.15.66.190:80 #配置虚拟机上通过Nginx访问Nacos

#消费者将要去访问的微服务名称(注册成功进nacos的微服务提供者)
service-url:
  nacos-user-service: http://nacos-payment-provider
5.2 启动及测试

启动

cloudalibaba-provider-payment-9001/cloudalibaba-provider-payment-9002

cloudalibaba-gateway-8899

测试

浏览器输入 - http://localhost:8899/payment/nacos/1

结果

不停刷新页面,9001/9002两个端口切换

6.GateWay常用的Predicate

官方文档

https://docs.spring.io/spring-cloud-gateway/docs/current/reference/html/#gateway-request-predicates-factories

6.1 Route Predicate Factories是什么?

Spring Cloud Gateway matches routes as part of the Spring WebFlux HandlerMapping infrastructure. Spring Cloud Gateway includes many built-in route predicate factories. All of these predicates match on different attributes of the HTTP request. You can combine multiple route predicate factories with logical and statements.

Spring Cloud Gateway将路由匹配作为Spring WebFlux HandlerMapping基础架构的一部分。Spring Cloud Gateway包括许多内置的Route Predicate工厂。所有这些Predicate都与HTTP请求的不同属性匹配。你可以将多个Route Predicate工厂可以进行组合。

Spring Cloud Gateway创建Route 对象时,使用RoutePredicateFactory 创建 Predicate对象,Predicate 对象可以赋值给Route。Spring Cloud Gateway包含许多内置的Route Predicate Factories。
所有这些谓词都匹配HTTP请求的不同属性。多种谓词工厂可以组合,并通过逻辑and。

启动cloudalibaba-gateway-8899项目时,控制台上输出信息:

6.2 常用的Route Predicate Factory

The After Route Predicate Factory
The Before Route Predicate Factory
The Between Route Predicate Factory
The cookie Route Predicate Factory
The Header Route Predicate Factory
The Host Route Predicate Factory
The Method Route Predicate Factory
The Path Route Predicate Factory
The Query Route Predicate Factory
The RemoteAddr Route Predicate Factory
The weight Route Predicate Factory

1)、The After Route Predicate Factory
spring:
  cloud:
    gateway:
      routes:
      - id: after_route
        uri: https://example.org
        predicates:
        # 这个时间后才能起效
        - After=2017-01-20T17:42:47.789-07:00[America/Denver]

可以通过下述方法获得上述格式的时间戳字符串

import java.time.ZonedDateTime;


public class TimeTest{
    public static void main(String[] args){
        ZonedDateTime zbj = ZonedDateTime.now(); // 默认时区
        System.out.println(zbj);

       //2021-01-16T11:51:37.485+08:00[Asia/Shanghai]
    }
}
2)、The Before Route Predicate Factory
spring:
  cloud:
    gateway:
      routes:
      - id: before_route
        uri: https://example.org
        predicates:
         # 这个时间前才能起效
        - Before=2017-01-20T17:42:47.789-07:00[America/Denver]
3)、The Between Route Predicate Factory
spring:
  cloud:
    gateway:
      routes:
      - id: between_route
        uri: https://example.org
        # 两个时间点之间
        predicates:
        - Between=2017-01-20T17:42:47.789-07:00[America/Denver], 2017-01-21T17:42:47.789-07:00[America/Denver]
4)、The cookie Route Predicate Factory
spring:
  cloud:
    gateway:
      routes:
      - id: cookie_route
        uri: https://example.org
        predicates:
        - cookie=chocolate, ch.p

The cookie route predicate factory takes two parameters, the cookie name and a regular expression.

cookie Route Predicate需要两个参数,一个是cookie name ,一个是正则表达式。

This predicate matches cookies that have the given name and whose values match the regular expression.

路由规则会通过获取对应的cookie name值和正则表达式去匹配,如果匹配上就会执行路由,如果没有匹配上则不执行。

yml文件

predicates:
	- Path=/payment/nacos/**          # 断言,路径相匹配的进行路由
	- Between=2021-01-16T11:30:11.626+08:00[Asia/Shanghai], 2021-01-16T11:50:11.626+08:00[Asia/Shanghai]
	- cookie=username,cm

测试

# 该命令相当于发get请求,且没带cookie
curl http://localhost:8899/payment/nacos/1

结果:404

# 带cookie的
curl http://localhost:8899/payment/nacos/1 --cookie "username=cm"

5)、The Header Route Predicate Factory
spring:
  cloud:
    gateway:
      routes:
      - id: header_route
        uri: https://example.org
        predicates:
        	#请求头要有X-Request-Id属性,并且值为整数的正则表达式
        	- Header=X-Request-Id, d+   

The header route predicate factory takes two parameters, the header name and a regular expression.

This predicate matches with a header that has the given name whose value matches the regular expression.

两个参数:一个是属性名称和一个正则表达式,这个属性值和正则表达式匹配则执行。

测试

# 带指定请求头的参数的CURL命令
curl http://localhost:8899/payment/nacos/1 -H "X-Request-Id:123"

6)、The Host Route Predicate Factory
spring:
  cloud:
    gateway:
      routes:
      - id: host_route
        uri: https://example.org
        predicates:
        - Host=**.somehost.org,**.anotherhost.org

Host Route Predicate 接收一组参数,一组匹配的域名列表,这个模板是一个ant分隔的模板,用.号作为分隔符。它通过参数中的主机地址作为匹配规则。

yml文件

predicates:
	- Path=/payment/nacos/**          # 断言,路径相匹配的进行路由
	- Host=**.igeekhome.com           #一组匹配的域名列表

测试

curl http://localhost:8899/payment/nacos/1 -H "Host:www.igeekhome.com"

7)、The Method Route Predicate Factory
spring:
  cloud:
    gateway:
      routes:
      - id: method_route
        uri: https://example.org
        predicates:
        	- Method=GET,POST

This route matches if the request method was a GET or a POST.

匹配规则为请求方式。

yml文件

predicates:
	- Path=/payment/nacos/**    # 断言,路径相匹配的进行路由
	- Method=GET                # 请求方式为GET

测试

curl http://localhost:8899/payment/nacos/1

8)、The Path Route Predicate Factory
spring:
  cloud:
    gateway:
      routes:
      - id: path_route
        uri: https://example.org
        predicates:
        	- Path=/red/{segment},/blue/{segment}

Path 路径相匹配的进行路由

9)、The Query Route Predicate Factory
spring:
  cloud:
    gateway:
      routes:
      - id: query_route
        uri: https://example.org
        predicates:
        	- Query=green

The preceding route matches if the request contained a green query parameter.

必须要有参数名为green的值才能路由。

支持传入两个参数,一个是属性名,一个为属性值,属性值可以是正则表达式。

yml文件

Query=username,d+  #要有参数名username并且值还是整数才能路由

测试

curl http://localhost:8899/payment/nacos/1?username=123

10)、The RemoteAddr Route Predicate Factory
spring:
  cloud:
    gateway:
      routes:
      - id: remoteaddr_route
        uri: https://example.org
        predicates:
        	- RemoteAddr=192.168.1.1/24

This route matches if the remote address of the request was, for example, 192.168.1.10.

根据请求的远程IP进行匹配,如果能匹配到,则将请求路由到指定URI上;

配置规则如下:

指定一个固定的IP指定多个IP,用逗号隔开

yml文件

predicates:
	- Path=/payment/nacos/**   # 断言,路径相匹配的进行路由
	- RemoteAddr=127.0.0.1     # 根据请求的远程IP进行匹配

测试

curl http://localhost:8899/payment/nacos/1

11)、The Weight Route Predicate Factory
spring:
  cloud:
    gateway:
      routes:
      - id: weight_high
        uri: https://weighthigh.org
        predicates:
        	- Weight=group1, 8
      - id: weight_low
        uri: https://weightlow.org
        predicates:
        	- Weight=group1, 2

This route would forward ~80% of traffic to weighthigh.org and ~20% of traffic to weighlow.org

根据权重来分发请求,权重是根据group来计算的。

存在参数:

group 组,权重根据组来计算
weight 权重值,是一个 Int 的值

12)、总结

Predicate就是为了实现一组匹配规则,让请求过来找到对应的Route进行处理。

7.GateWay的Filter

官方文档

https://docs.spring.io/spring-cloud-gateway/docs/current/reference/html/#gatewayfilter-factories

Route filters allow the modification of the incoming HTTP request or outgoing HTTP response in some manner. Route filters are scoped to a particular route. Spring Cloud Gateway includes many built-in GatewayFilter Factories.

路由过滤器可用于修改进入的HTTP请求和返回的HTTP响应,路由过滤器只能指定路由进行使用。Spring Cloud Gateway内置了多种路由过滤器,他们都由GatewayFilter的工厂类来产生。

生命周期:pre 、post

7.1 Filter的种类

GatewayFilter - 有31种
GlobalFilter - 有10种

常用的GatewayFilter:AddRequestParameter GatewayFilter…

例如:The AddRequestParameter GatewayFilter Factory

spring:
  cloud:
    gateway:
      routes:
      - id: add_request_parameter_route
        uri: https://example.org
        filters:
        	- AddRequestParameter=red, blue

yml文件

- id: payment_route2 #payment_route      #路由的ID,没有固定规则但要求唯一,建议配合服务名
	uri: lb://nacos-payment-provider     #匹配后提供服务的路由地址
	predicates:
		- Path=/payment/nacos/**         # 断言,路径相匹配的进行路由
	filters:
		- AddRequestParameter=name,zs    #  
@GetMapping(value = "/payment/nacos/{id}")
public String getPayment(@PathVariable("id") Integer id , @RequestParam String name) {
  return "nacos registry, serverPort: "+ serverPort+"t id = "+id +" ,name = "+name;
}

测试

curl http://localhost:8899/payment/nacos/1?name=zs
#nacos registry, serverPort: 9002 id = 1 ,name = zs,zs

curl http://localhost:8899/payment/nacos/1?name=ls
#nacos registry, serverPort: 9001         id = 1
7.2 自定义全局GlobalFilter

主要可以做 全局日志记录、统一网关鉴权等功能。

cloudalibaba-gateway-8899项目

1)、添加MyLogGateWayFilter类
package com.igeek.springcloud.filter;

import lombok.extern.slf4j.Slf4j;
import org.springframework.cloud.gateway.filter.GatewayFilterChain;
import org.springframework.cloud.gateway.filter.GlobalFilter;
import org.springframework.core.annotation.Order;
import org.springframework.http.HttpStatus;
import org.springframework.http.server.reactive.ServerHttpRequest;
import org.springframework.http.server.reactive.ServerHttpResponse;
import org.springframework.stereotype.Component;
import org.springframework.util.StringUtils;
import org.springframework.web.server.ServerWebExchange;
import reactor.core.publisher.Mono;

import java.util.Date;

@Component
@Slf4j
@Order(value = 0)
public class MyLogGateWayFilter implements GlobalFilter {

    @Override
    public Mono filter(ServerWebExchange exchange, GatewayFilterChain chain) {
        log.info("===============MyLogGateWayFilter "+new Date());

        ServerHttpRequest request = exchange.getRequest();
        ServerHttpResponse response = exchange.getResponse();
        //获取请求参数
        String username = request.getQueryParams().getFirst("username");
        if(StringUtils.hasLength(username)){
            return chain.filter(exchange);
        }

        //
        log.info("=================username为null");
        response.setStatusCode(HttpStatus.BAD_REQUEST);
        return response.setComplete();
    }

}
2)、测试

1.启动项目

cloudalibaba-provider-payment-9001

cloudalibaba-provider-payment-9002

cloudalibaba-gateway-8899

2.浏览器输入:

http://localhost:8899/payment/nacos/1 - 访问异常
http://localhost:8899/payment/nacos/1?username=ls - 携带参数可正常访问

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zaji/5709981.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-12-17
下一篇 2022-12-17

发表评论

登录后才能评论

评论列表(0条)

保存