gateway网关
一、gateWay网关
1.1 gateway简介
1.2 微服务架构
1.3 spring cloud gateway的特性
1.4 与zuul的区别
1.5 zuul 1.x的模型
1. 6 webflux
1.7 流程
二、简单案例
2.1 案例pro
2.2.1 pom
<dependencies>>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-gateway</artifactId>
<version>2.2.3.RELEASE</version>
</dependency>
<!--eureka-client-->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
<!-- 引入自己定义的api通用包,可以使用Payment支付Entity -->
<dependency>
<groupId>com.atguigu.springcloud</groupId>
<artifactId>cloud-api-commons</artifactId>
<version>${project.version}</version>
</dependency>
<!-- <dependency>-->
<!-- <groupId>org.springframework.boot</groupId>-->
<!-- <artifactId>spring-boot-devtools</artifactId>-->
<!-- <scope>runtime</scope>-->
<!-- <optional>true</optional>-->
<!-- </dependency>-->
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<optional>true</optional>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
2.2.2 yml
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 #匹配后提供服务的路由地址
predicates:
- Path=/payment/get/** # 断言,路径相匹配的进行路由
eureka:
instance:
hostname: cloud-gateway-service
client: #服务提供者provider注册进eureka服务列表内
service-url:
register-with-eureka: true
fetch-registry: true
defaultZone: http://eureka7001.com:7001/eureka
2.2.3 config
@Configuration
public class configGateway {
@Bean
public RouteLocator customRouteLocator(configGateway routeLocatorBuilder) {
RouteLocatorBuilder.Builder routes = routeLocatorBuilder.routes();
routes.route("path_route_atguigu", r -> r.path("/guoji").uri("http://news.baidu.com/guonei"))
.build();
return routes.build();
}
}
2.2.5 主启动类
@SpringBootApplication
@EnableEurekaClient
public class gatewayApp {
public static void main(String[] args) {
SpringApplication.run(gatewayApp.class, args);
System.out.println("启动成功");
}
}
2.3 gateWay效果
- 原8001端口的接口,现在9527也可以访问的到
2.4 路由配置方式
- yaml配置
- 代码中注入RouterLocator的bean
package com.pack.gateway.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 configGateway {
@Bean
public RouteLocator customRouteLocator(RouteLocatorBuilder routeLocatorBuilder) {
RouteLocatorBuilder.Builder routes = routeLocatorBuilder.routes();
routes.route("path_route_atguigu", r -> r.path("/guoji").uri("http://news.baidu.com/guonei"))
.build();
routes.route("path_route_atguigu", r -> r.path("/guonei").uri("http://news.baidu.com/guoji"))
.build();
return routes.build();
}
}
- 作用
访问http://localhost:9527/guoj会转到http://news.baidu.com/guonei,但是路由还是http://localhost:9527/guoj
2.5 动态路由
2.5.1 动态路由概念
2.5.2 YML配置
- 开启动态路由,去掉具体的uri配置
- uri: lb://CONSUMERORDER-80:配置uri,uri对应eureka中的服务名称,lb是负载均衡的意思,动态路由就是由eureka动态调用多个同名的服务提供者,(轮询…等其他方式)
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://CONSUMERORDER-80 #匹配后提供服务的路由地址
predicates:
- Path=/payment/get/** # 断言,路径相匹配的进行路由
- id: payment_routh #payment_route #路由的ID,没有固定规则但要求唯一,建议配合服务名
# uri: http://localhost:8001 #匹配后提供服务的路由地址
#lb是负载均衡的意思
uri: lb://CONSUMERORDER-80 #匹配后提供服务的路由地址
predicates:
- Path=/getServiceInfo # 断言,路径相匹配的进行路由
- id: payment_routh2 #payment_route #路由的ID,没有固定规则但要求唯一,建议配合服务名
# uri: http://localhost:8001 #匹配后提供服务的路由地址
#uri: lb://cloud-payment-service #匹配后提供服务的路由地址
predicates:
- Path=/payment/lb/** # 断言,路径相匹配的进行路由
- After=2020-03-08T10:59:34.102+08:00[Asia/Shanghai]
# - Before=2020-03-08T10:59:34.102+08:00[Asia/Shanghai]
# - Between=2020-03-08T10:59:34.102+08:00[Asia/Shanghai] , 2020-03-08T10:59:34.102+08:00[Asia/Shanghai]
# curl http://localhost:9527/payment/lb --cookie "username=zzyy"
# - Cookie=username,zzyy #Cookie=cookieName,正则表达式
# 请求头要有X-Request-Id属性并且值为整数的正则表达式 curl http://localhost:9527/payment/lb --cookie "username=zzyy" -H "X-Request-Id:11"
# - Header=X-Request-Id, \d+
# - Host=**.atguigu.com # curl http://localhost:9527/payment/lb -H "Host:afae.atguigu.com"
eureka:
instance:
hostname: cloud-gateway-service
client: #服务提供者provider注册进eureka服务列表内
service-url:
register-with-eureka: true
fetch-registry: true
defaultZone: http://eureka7001.com:7001/eureka
2.6 Predicate
2.6.1 作用
- 可以对路由的判定加一些条件,比如路由什么时间生效之类的
- 简单的说就是对请求添加一些先决的条件,只有通过条件才去进行转发。
cloud:
gateway:
discovery:
locator:
enabled: true #开启从注册中心动态创建路由的功能,利用微服务名进行路由
routes:
- id: payment_routh #payment_route #路由的ID,没有固定规则但要求唯一,建议配合服务名
# uri: http://localhost:8001 #匹配后提供服务的路由地址
uri: lb://CONSUMERORDER-80 #匹配后提供服务的路由地址
predicates:
- Path=/payment/get/** # 断言,路径相匹配的进行路由
- After=2022-06-21T20:27:25.987+08:00[Asia/Shanghai]
- 路由的时间格式可以通过ZonedDateTime获得
public class tets {
public static void main(String[] args) {
ZonedDateTime now = ZonedDateTime.now();
System.out.println(now);
}
}
2.6.2 cookie相关
- predicates中设置请求Cookie的username是zzyy
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://CONSUMERORDER-80 #匹配后提供服务的路由地址
predicates:
- Path=/payment/get/** # 断言,路径相匹配的进行路由
- After=2022-06-21T20:27:25.987+08:00[Asia/Shanghai]
- Cookie=username,zzyy
- 通过cmd模拟请求
2.6.3 对请求头等其他断言
predicates:
- Path=/payment/get/** # 断言,路径相匹配的进行路由
# - After=2022-06-21T20:27:25.987+08:00[Asia/Shanghai]
# - Cookie=username,zzyy
- Header=X-Request-Id, \d+ #请求头断言
- Host=**.atguigu.com #Host断言
C:\Users\Administrator>curl http://localhost:9527/payment/get/Cookie -H "X-Request-Id;12"
2.7 filter
2.7.1 官网提供的过滤器
- 在请求前后对请求进行修改
2.7.2 自定义过滤器
package com.pack.gateway.filter;
import lombok.extern.slf4j.Slf4j;
import org.springframework.cloud.gateway.filter.GatewayFilterChain;
import org.springframework.cloud.gateway.filter.GlobalFilter;
import org.springframework.core.Ordered;
import org.springframework.http.HttpStatus;
import org.springframework.stereotype.Component;
import org.springframework.web.server.ServerWebExchange;
import reactor.core.publisher.Mono;
import java.util.Date;
@Component
@Slf4j
class MyLogGateWayFilter implements GlobalFilter, Ordered {
@Override
public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
log.info("***********come in MyLogGateWayFilter: " + new Date());
// 获取请求参数中的 uname
String uname = exchange.getRequest().getQueryParams().getFirst("uname");
if (uname == null) {
log.info("*******用户名为null,非法用户,o(╥﹏╥)o");
exchange.getResponse().setStatusCode(HttpStatus.NOT_ACCEPTABLE);
return exchange.getResponse().setComplete();
}
//合法放行,到新的过滤器,传到新的过滤链
return chain.filter(exchange);
}
/**
* 加载过滤器顺序,数字越小优先级越高
*
* @return
*/
@Override
public int getOrder() {
return 0;
}
}
- 上述自定义全局过滤器的作用就是没有加uname时中断请求
三、原理
3.1原理图
3.2 先启动网关?
先启动微服务,等微服务注册到nacos上,网关再从nacos上获取微服务信息。