在分布式系统系统中,有商品、订单、用户、广告、支付等等一大批的服务,前端怎么调用呢?和每个服务一个个打交道?这显然是不可能的,这就需要有一个角色充当所有请求的入口,这个角色就是服务网关(API gateway)。
为了解决上面这些问题,我们需要将权限控制这样的东西从我们的服务单元中抽离出去,而最适合这些逻辑的地方就是处于对外访问最前端的地方,我们需要一个更强大一些的均衡负载器,它就是本文将来介绍的:服务网关。
服务网关是微服务架构中一个不可或缺的部分。通过服务网关统一向外系统提供REST API的过程中,除了具备服务路 由、均衡负载功能之外,它还具备了权限控制等功能。Spring Cloud Netflflix 中的 Zuul 就担任了这样的一个角色,为 微服务架构提供了前门保护的作用,同时将权限控制这些较重的非业务逻辑内容迁移到服务路由层面,使得服务集群主体能够具备更高的可复用性和可测试性。
5.1 新建zuul的module工程
5.2 新建pom文件
org.springframework.cloud spring-cloud-starter-netflix-eureka-client org.springframework.cloud spring-cloud-starter-netflix-zuul
eureka:client:serviceUrl:defaultZone: http://localhost:8888/eureka/
server:port: 9000
spring:application:name: study-zuul
zuul:routes:#路由名称,随意命名api-order:path: /api-order/**serviceId: study-userapi-user:path: /api-user/**serviceId: study-user
5.4 启动类
package com.study;import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;
import org.springframework.cloud.netflix.zuul.EnableZuulProxy;@SpringBootApplication
// 开启zuul功能
@EnableZuulProxy
@EnableEurekaClient
public class ZuulApp {public static void main(String[] args) {SpringApplication.run(ZuulApp.class, args);}
}
5.5测试路由访问
直接访问该端口结果如下:
通过路由网关访问,和上面结果一样
5.6 配置统一前缀访问
zuul:routes:#路由名称,随意命名api-order:path: /api-order/**serviceId: study-userapi-user:path: /api-user/**serviceId: study-user#前缀访问prefix: /study
添加统一前缀以后之前访问地址:http://localhost:9000/api-user/user/3
需要增加前缀才能访问,如下:
http://localhost:9000/study/api-user/user/3
5.7 忽略服务名serviceId访问
zuul:routes:#路由名称,随意命名api-order:path: /api-order/**serviceId: study-userapi-user:path: /api-user/**serviceId: study-user#忽略服务名serviceId访问ignored-services: "*"
添加ignored-services: "*"以后,之前直接通过服务名访问的端口将无法直接访问,必须通过路由访问。
5.8 配url绑定映射
zuul:routes:testurl:url: http://www.baidu.compath: /testurl/**
添加以后只要访问http://地址/testurl/**的将直接跳转到http://www.baidu.com。
testurl可以自己设置名称。
5.9 配置URL映射负载
ribbon:eureka:enabled: false
#Ribbon请求的微服务serviceId
study-user:ribbon:listOfServers: http://www.huya.com,http://www.douyu.com
zuul:routes:testurl:serviceId: study-userpath: /testurl/**
访问/testurl/**,会自动跳转http://www.huya.com或者http://www.douyu.com根据负载均衡来进行。
如http://localhost:9000/testurl/
Zuul本身是一系列过滤器的集成,那么他当然也就提供了自定义过滤器的功能,zuul提供了四种过滤器:前置过滤器,路由过滤器,错误过滤器,简单过滤器,实现起来也十分简单,只需要编写一个类去实现zuul提供的接口。
使用zuul过滤器
package com.study.Filter;import com.netflix.zuul.ZuulFilter;
import org.springframework.cloud.netflix.zuul.filters.support.FilterConstants;
import org.springframework.stereotype.Component;@Component
public class Filter1 extends ZuulFilter {/*** 类型包含 pre post route error* pre 代表在路由代理之前执行* route 代表代理的时候执行* error 代表出现错的时候执行* post 代表在route 或者是 error 执行完成后执行*/@Overridepublic String filterType() {// 路由之前(前置过滤器)return FilterConstants.PRE_TYPE;}@Overridepublic int filterOrder() {// 优先级,数字越大,优先级越低return 2;}@Overridepublic boolean shouldFilter() {// 是否执行该过滤器,true代表需要过滤return true;}@Overridepublic Object run() {System.out.println("Filter1 run");return null;}
}
设置两个过滤器,一个前置,一个后置,查看运行结果。