本项目的笔记和资料的Download,请点击这一句话自行获取。html
day01-springboot(理论篇) ;day01-springboot(实践篇)前端
day02-springcloud(理论篇一) ;day02-springcloud(理论篇二) ;day02-springcloud(理论篇三) ;day02-springcloud(理论篇四) ;git
day03-springcloud(Hystix,Feign) ;day03-springcloud(Zuul网关); day04-ES6语法入门(视频第三天讲的ES6,笔记在day04位置)github
经过前面的学习,使用Spring Cloud实现微服务的架构基本成型,大体是这样的:spring
咱们使用Spring Cloud Netflix中的Eureka实现了服务注册中心以及服务注册与发现;而服务间经过Ribbon或Feign实现服务的消费以及均衡负载;经过Spring Cloud Config实现了应用多环境的外部化配置以及版本管理。api
为了使得服务集群更为健壮,使用Hystrix的融断机制来避免在微服务架构中个别服务出现异常时引发的故障蔓延。安全
在该架构中,咱们的服务集群包含:内部服务Service A和Service B,他们都会注册与订阅服务至Eureka Server,而Open Service是一个对外的服务,经过均衡负载公开至服务调用方。咱们把焦点汇集在对外服务这块,直接暴露咱们的服务地址,这样的实现是否合理,或者是否有更好的实现方式呢?springboot
先来讲说这样架构须要作的一些事儿以及存在的不足:架构
面对相似上面的问题,咱们要如何解决呢?答案是:服务网关!app
为了解决上面这些问题,咱们须要将权限控制这样的东西从咱们的服务单元中抽离出去,而最适合这些逻辑的地方就是处于对外访问最前端的地方,咱们须要一个更强大一些的均衡负载器的 服务网关。
服务网关是微服务架构中一个不可或缺的部分。经过服务网关统一贯外系统提供REST API的过程当中,除了具有服务路由、均衡负载功能以外,它还具有了权限控制
等功能。Spring Cloud Netflix中的Zuul就担任了这样的一个角色,为微服务架构提供了前门保护的做用,同时将权限控制这些较重的非业务逻辑内容迁移到服务路由层面,使得服务集群主体可以具有更高的可复用性和可测试性。
官网:https://github.com/Netflix/zuul
Zuul:维基百科:
电影《捉鬼敢死队》中的怪兽,Zuul,在纽约引起了巨大骚乱。
事实上,在微服务架构中,Zuul就是守门的大Boss!一夫当关,万夫莫开!
不论是来自于客户端(PC或移动端)的请求,仍是服务内部调用。一切对服务的请求都会通过Zuul这个网关,而后再由网关来实现 鉴权、动态路由等等操做。
Zuul就是咱们服务的统一入口。
填写基本信息:
添加Zuul依赖:
经过@EnableZuulProxy
注解开启Zuul的功能:
@SpringBootApplication @EnableZuulProxy // 开启Zuul的网关功能 public class ZuulDemoApplication { public static void main(String[] args) { SpringApplication.run(ZuulDemoApplication.class, args); } }
server: port: 10010 #服务端口 spring: application: name: api-gateway #指定服务名
咱们须要用Zuul来代理user-service服务,先看一下控制面板中的服务状态:
映射规则:
zuul: routes: user-service: # 这里是路由id,随意写 path: /user-service/** # 这里是映射路径 url: http://127.0.0.1:8081 # 映射路径对应的实际url地址
咱们将符合path
规则的一切请求,都代理到 url
参数指定的地址
本例中,咱们将 /user-service/**
开头的请求,代理到http://127.0.0.1:8081
访问的路径中须要加上配置规则的映射路径,咱们访问:http://127.0.0.1:8081/user-service/user/10
在刚才的路由规则中,咱们把路径对应的服务地址写死了!若是同一服务有多个实例的话,这样作显然就不合理了。
咱们应该根据服务的名称,去Eureka注册中心查找 服务对应的全部实例列表,而后进行动态路由才对!
<dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId> </dependency>
@SpringBootApplication @EnableZuulProxy // 开启Zuul的网关功能 @EnableDiscoveryClient public class ZuulDemoApplication { public static void main(String[] args) { SpringApplication.run(ZuulDemoApplication.class, args); } }
eureka: client: registry-fetch-interval-seconds: 5 # 获取服务列表的周期:5s service-url: defaultZone: http://127.0.0.1:10086/eureka instance: prefer-ip-address: true ip-address: 127.0.0.1
由于已经有了Eureka客户端,咱们能够从Eureka获取服务的地址信息,所以映射时无需指定IP地址,而是经过服务名称来访问,并且Zuul已经集成了Ribbon的负载均衡功能。
zuul: routes: user-service: # 这里是路由id,随意写 path: /user-service/** # 这里是映射路径 serviceId: user-service # 指定服务名称
再次启动,此次Zuul进行代理时,会利用Ribbon进行负载均衡访问:
日志中能够看到使用了负载均衡器:
在刚才的配置中,咱们的规则是这样的:
zuul.routes.<route>.path=/xxx/**
: 来指定映射路径。<route>
是自定义的路由名zuul.routes.<route>.serviceId=/user-service
:来指定服务名。而大多数状况下,咱们的<route>
路由名称每每和 服务名会写成同样的。所以Zuul就提供了一种简化的配置语法:zuul.routes.<serviceId>=<path>
比方说上面咱们关于user-service的配置能够简化为一条:
zuul: routes: user-service: /user-service/** # 这里是映射路径
省去了对服务名称的配置。
在使用Zuul的过程当中,上面讲述的规则已经大大的简化了配置项。可是当服务较多时,配置也是比较繁琐的。所以Zuul就指定了默认的路由规则:
user-service
,则默认的映射路径就是:/user-service/**
也就是说,刚才的映射规则咱们彻底不配置也是OK的,不信就试试看。
配置示例:
zuul: prefix: /api # 添加路由前缀 routes: user-service: # 这里是路由id,随意写 path: /user-service/** # 这里是映射路径 service-id: user-service # 指定服务名称
咱们经过zuul.prefix=/api
来指定了路由的前缀,这样在发起请求时,路径就要以/api开头。
路径/api/user-service/user/1
将会被代理到/user-service/user/1
Zuul做为网关的其中一个重要功能,就是实现请求的鉴权。而这个动做咱们每每是经过Zuul提供的过滤器来实现的。
ZuulFilter是过滤器的顶级父类。在这里咱们看一下其中定义的4个最重要的方法:
public abstract ZuulFilter implements IZuulFilter{ abstract public String filterType(); abstract public int filterOrder(); boolean shouldFilter();// 来自IZuulFilter Object run() throws ZuulException;// IZuulFilter }
shouldFilter
:返回一个Boolean
值,判断该过滤器是否须要执行。返回true执行,返回false不执行。run
:过滤器的具体业务逻辑。filterType
:返回字符串,表明过滤器的类型。包含如下4种:
pre
:请求在被路由以前执行routing
:在路由请求时调用post
:在routing和errror过滤器以后调用error
:处理请求时发生错误调用filterOrder
:经过返回的int值来定义过滤器的执行顺序,数字越小优先级越高。这张是Zuul官网提供的请求生命周期图,清晰的表现了一个请求在各个过滤器的执行顺序。
全部内置过滤器列表:
场景很是多:
接下来咱们来自定义一个过滤器,模拟一个登陆的校验。基本逻辑:若是请求中有access-token参数,则认为请求有效,放行。
@Component public class LoginFilter extends ZuulFilter{ @Override public String filterType() { // 登陆校验,确定是在前置拦截 return "pre"; } @Override public int filterOrder() { // 顺序设置为1 return 1; } @Override public boolean shouldFilter() { // 返回true,表明过滤器生效。 return true; } @Override public Object run() throws ZuulException { // 登陆校验逻辑。 // 1)获取Zuul提供的请求上下文对象 RequestContext ctx = RequestContext.getCurrentContext(); // 2) 从上下文中获取request对象 HttpServletRequest req = ctx.getRequest(); // 3) 从请求中获取token String token = req.getParameter("access-token"); // 4) 判断 if(token == null || "".equals(token.trim())){ // 没有token,登陆校验失败,拦截 ctx.setSendZuulResponse(false); // 返回401状态码。也能够考虑重定向到登陆页。 ctx.setResponseStatusCode(HttpStatus.UNAUTHORIZED.value()); } // 校验经过,能够考虑把用户信息放入上下文,继续向后执行 return null; } }
没有token参数时,访问失败:
添加token参数后:
Zuul中默认就已经集成了Ribbon负载均衡和Hystix熔断机制。可是全部的超时策略都是走的默认值,好比熔断超时时间只有1S,很容易就触发了。所以建议咱们手动进行配置:
zuul: retryable: true ribbon: ConnectTimeout: 250 # 链接超时时间(ms) ReadTimeout: 2000 # 通讯超时时间(ms) OkToRetryOnAllOperations: true # 是否对全部操做重试 MaxAutoRetriesNextServer: 2 # 同一服务不一样实例的重试次数 MaxAutoRetries: 1 # 同一实例的重试次数 hystrix: command: default: execution: isolation: thread: timeoutInMillisecond: 6000 # 熔断超时时长:6000ms
=============================================
参考资料:
SpringCloud系列八:Zuul 路由访问(Zuul 的基本使用、Zuul 路由功能、zuul 过滤访问、Zuul 服务降级)
Spring Cloud升级最新Finchley版本的全部坑
end