wmic service 未被引用服务路径_SpringCloudZuul服务网关

我们使用Eureka实现了服务注册中心来服务注册与发现。为了方便统一配置文件的集中化管理我们使用ConfigServer。而服务间通过Ribbon或Feign实现服务的消费以及均衡负载。为了使服务集群更为健壮,使用Hystrix的熔断机制来避免在微服务架构中个别服务出现异常时引起的故障蔓延。

在以上架构设计中,我们的所有服务都注册到注册中心直接对外暴露使用,这样设计是否合理?(能不能加上一个权限和路由的管理功能?)或者说有没有更好的实现方式呢?

一、什么是Zuul?

为了解决上面这些问题,我们需要将权限控制等这样的东西从我们的服务单元中抽离出去,而最适合这些逻辑的地方就是处于对外访问最前端的地方,我们需要一个更强大一些的均衡负载器的服务网关

服务网关是微服务架构中一个不可或缺的部分。通过服务网关统一向外系统提供REST API的过程中,除了具备服务路由、均衡负载功能之外,它还具备了权限控制等功能。Spring Cloud Netflix中的Zuul就担任了这样的一个角色,为微服务架构提供了前门保护的作用,同时将权限控制这些较重的非业务逻辑内容迁移到服务路由层面,使得服务集群主体能够具备更高的可复用性和可测试性。

我们来看一下官方介绍:

662699f77874c6c1b0edc5cb1ee14cc1.png

下面简单画了一个zuul加入后的架构图

546533f3b28d61407afa8703a97cc2ff.png

这样

不管是来自于客户端(PC或移动端)的请求,还是服务内部调用。一切对服务的请求都会经过Zuul这个网关,然后再由网关来实现 鉴权、动态路由等等操作。Zuul就是我们服务的统一入口。

二、快速入门

1.导入zuul依赖
  <dependency>      <groupId>org.springframework.cloudgroupId>      <artifactId>spring-cloud-starter-netflix-zuulartifactId>  dependency>
2.pom文件编写路由配置
server:  port: 10010 # 端口spring:  application:    name: api-gateway # 应用名称,会在Eureka中显示eureka:  client:    service-url:      defaultZone: http://127.0.0.1:8081/eurekazuul:  routes:    service-provider: # 这里是路由id,随意写      path: /service-provider/** # 这里是映射路径#      url: http://127.0.0.1:8001 # 映射路径对应的实际url地址      serviceId: service-provider # 指定服务名称
  • 如果不使用注册中心,我们可以直接通过url访问具体服务路径

  • 因为已经有了Eureka客户端,我们可以从Eureka获取服务的地址信息,因此映射时无需指定IP地址,而是通过服务名称来访问,而且Zuul已经集成了Ribbon的负载均衡功能。

3.启动类

@SpringBootApplication@EnableZuulProxy    //开启Zuul网关功能public class GatewayApp {    public static void main(String[] args) {        SpringApplication.run(GatewayApp.class,args);    }}

4.启动测试,会利用Ribbon进行负载均衡访问

d81120907eba4c97601099f28b08585b.png

36cd4946914bf2fcbf4ba7dcbbfe2e5e.png

日志中可以看到使用了负载均衡器

704440c90aa592f6799b0ebf5f7c2e5d.png

5.简化路由配置

在刚才的配置中,我们的规则是这样的:
  • zuul.routes..path=/xxx/**:来指定映射路径。是自定义的路由名。

  • zuul.routes..serviceId=service-provider:来指定服务名。

而大多数情况下,我们的路由名称往往和服务名会写成一样的。因此Zuul就提供了一种简化的配置语法:zuul.routes.=

比方说上面我们关于service-provider的配置可以简化为一条:

zuul:  routes:    service-provider: /service-provider/** # 这里是映射路径

6.默认路由配置

在使用Zuul的过程中,上面讲述的规则已经大大的简化了配置项。但是当服务较多时,配置也是比较繁琐的。因此Zuul就指定了默认的路由规则:

  • 默认情况下,一切服务的映射路径就是服务名本身。例如服务名为:service-provider,则默认的映射路径就 是:/service-provider/**

也就是说,刚才的映射规则我们完全不配置也是OK的。

7.路由前缀

zuul:  routes:    service-provider: /service-provider/**  prefix: /gateway
  • 我们通过zuul.prefix=/gateway来指定了路由的前缀,这样在发起请求时,路径就要以/gateway开头。

a598df64f81c33a8e189ef57d08956cd.png

三、过滤器使用

Zuul作为网关的其中一个重要功能,就是实现请求的鉴权。而这个动作我们往往是通过Zuul提供的过滤器来实现的。

1.ZuulFilter

ZuulFilter是过滤器的顶级父类。看一下其中定义的4个最重要的方法:

public abstract ZuulFilter implements IZuulFilter{    abstract public String filterType();    abstract public int filterOrder();    boolean shouldFilter();// 来自IZuulFilter    Object run() throws ZuulException;// 来自IZuulFilter}

c17c51dc4088f5b22bcf263bd76b12ba.png

  • shouldFilter:返回一个Boolean值,判断该过滤器是否需要执行。返回true执行,返回false不执行。

  • run:过滤器的具体业务逻辑。

  • filterType:返回字符串,代表过滤器的类型。包含以下4种:

    • pre:请求在被路由之前执行

    • route:在路由请求时调用

    • post:在route和errror过滤器之后调用

    • error:处理请求时发生错误调用

  • filterOrder:通过返回的int值来定义过滤器的执行顺序,数字越小优先级越高。

2.过滤器执行生命周期

Zuul官网提供的请求生命周期图,清晰的表现了一个请求在各个过滤器的执行顺序。

7b1f8d44c58f037ef047dc3ad22bb248.png

正常流程:

  • 请求到达首先会经过pre类型过滤器,而后到达route类型,进行路由,请求就到达真正的服务提供者,执行请求,返回结果后,会到达post过滤器。而后返回响应。

异常流程:

  • 整个过程中,pre或者route过滤器出现异常,都会直接进入error过滤器,在error处理完毕后,会将请求交给post过滤器,最后返回给用户。

  • 如果是error过滤器自己出现异常,最终也会进入post过滤器,将最终结果返回给请求客户端。

  • 如果是post过滤器出现异常,会跳转到error过滤器,但是与pre和route不同的是,请求不会再到达post过滤器了。

使用场景

  • 请求鉴权:一般放在pre类型,如果发现没有访问权限,直接就拦截了

  • 异常处理:一般会在error类型和post类型过滤器中结合来处理。

  • 服务调用时长统计:pre和post结合使用。

3.自定义过滤器

接下来我们来自定义一个过滤器,模拟一个登录的校验。基本逻辑:如果请求中有access-token参数,则认为请求有效,放行。
@Componentpublic class LoginFilter extends ZuulFilter {    /**     * 过滤器类型,前置过滤器     *     * @return     */    public String filterType() {        return "pre";    }    /**     * 过滤器的执行顺序     *     * @return     */    public int filterOrder() {        return 1;    }    /**     * 该过滤器是否生效     *     * @return     */    public boolean shouldFilter() {        return true;    }    /**     * 登陆校验逻辑     *     * @return     * @throws ZuulException     */    public Object run() throws ZuulException {        // 获取zuul提供的上下文对象        RequestContext context = RequestContext.getCurrentContext();        // 从上下文对象中获取请求对象        HttpServletRequest request = context.getRequest();        // 获取token信息        String token = request.getParameter("access-token");        // 判断        if (StringUtils.isBlank(token)) {            // 过滤该请求,不对其进行路由            context.setSendZuulResponse(false);            // 设置响应状态码,401            context.setResponseStatusCode(HttpStatus.SC_UNAUTHORIZED);            // 设置响应信息            context.setResponseBody("{\"status\":\"401\", \"text\":\"request error!\"}");        }        // 校验通过,把登陆信息放入上下文信息,继续向后执行        context.set("token", token);        return null;    }}
测试,没有token参数,请求失败!

8165028c84fee3ce9ad7a4cfffcdaa56.png

测试,有token参数,请求成功!

29e89b6bcafad6f8cdc8764dcc6bc38e.png

4.负载均衡和熔断

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:            timeoutInMilliseconds: 6000 #设置hystrix的超时时间为6000ms

如果您觉得本文对你有帮助,欢迎老铁们帮忙点赞、关注、留言、分享你们的支持是我原创最大的动力

3d2589c1c6c7f92acd8c1c7b6e91c930.png

4d3549848dc592df681974c73888859c.gif

往期精选

SpringCloud配置中心的使用

SpringCloud-Eureka高可用搭建

SpringCloud-Ribbon负载均衡

SpringCloud-Feign远程调用

SpringCloud-Hystrix解决雪崩

Nacos入门案例

Mybatis动态SQL 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值