`
大涛学长
  • 浏览: 105543 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

如何基于 Nacos 和 Sentinel ,实现灰度路由和流量防护一体化

阅读更多
Nepxion Discovery框架在实现灰度发布和路由功能前提下,结合Nacos和Sentinel,对流量再实施一层防护措施,更能达到企业级的流量安全控制的目的。它的功能包括:

*   封装远程配置中心和本地规则文件的读取逻辑,即优先读取远程配置,如果不存在或者规则错误,则读取本地规则文件。动态实现远程配置中心对于规则的热刷新
*   封装NacosDataSource和ApolloDataSource,支持Nacos和Apollo两个远程配置中心,零代码实现Sentinel功能。更多的远程配置中心,请参照Sentinel官方的DataSource并自行集成
*   支持原生的流控规则、降级规则、授权规则、系统规则、热点参数流控规则
*   支持扩展LimitApp的机制,通过动态的Http Header方式实现组合式防护机制,包括基于服务名、基于灰度组、基于灰度版本、基于灰度区域、基于机器地址和端口等防护机制,支持自定义任意的业务参数组合实现该功能,例如,根据传入的微服务灰度版本号+用户名,组合在一起进行熔断
*   支持微服务侧Actuator、Swagger和Rest三种方式的规则写入
*   支持控制台侧基于微服务名的Actuator、Swagger和Rest三种方式的批量规则写入
*   支持开关关闭上述功能spring.application.strategy.sentinel.enabled=true,默认是关闭的

\[**Nacos**\] 阿里巴巴中间件部门开发的新一代集服务注册发现中心和配置中心为一体的中间件。它是构建以“服务”为中心的现代应用架构 (例如微服务范式、云原生范式) 的服务基础设施,支持几乎所有主流类型的“服务”的发现、配置和管理,更敏捷和容易地构建、交付和管理微服务平台

\[**Sentinel**\] 阿里巴巴中间件部门开发的新一代以流量为切入点,从流量控制、熔断降级、系统负载保护等多个维度保护服务的稳定性的分布式系统的流量防卫兵。它承接了阿里巴巴近10年的双十一大促流量的核心场景,例如秒杀(即突发流量控制在系统容量可以承受的范围)、消息削峰填谷、集群流量控制、实时熔断下游不可用应用等

环境搭建及依赖引入
---------

服务端在Discovery框架原有依赖的基础上,再引入如下依赖

```
<dependency>
    <groupId>com.nepxion</groupId>
    <artifactId>discovery-plugin-strategy-starter-service-sentinel</artifactId>
    <version>${discovery.version}</version>
</dependency>

<dependency>
    <groupId>com.nepxion</groupId>
    <artifactId>discovery-plugin-strategy-sentinel-starter-nacos</artifactId>
    <!-- <artifactId>discovery-plugin-strategy-sentinel-starter-apollo</artifactId> -->
    <version>${discovery.version}</version>
</dependency>
```

![]()![]( "点击并拖拽以移动")

原生Sentinel注解
------------

参照下面代码,为接口方法增加@SentinelResource注解,value为sentinel-resource,blockHandler和fallback是防护其作用后需要执行的方法

```
@RestController
@ConditionalOnProperty(name = DiscoveryConstant.SPRING_APPLICATION_NAME, havingValue = "discovery-guide-service-b")
public class BFeignImpl extends AbstractFeignImpl implements BFeign {
    private static final Logger LOG = LoggerFactory.getLogger(BFeignImpl.class);

    @Override
    @SentinelResource(value = "sentinel-resource", blockHandler = "handleBlock", fallback = "handleFallback")
    public String invoke(@PathVariable(value = "value") String value) {
        value = doInvoke(value);

        LOG.info("调用路径:{}", value);

        return value;
    }

    public String handleBlock(String value, BlockException e) {
        return value + "-> B server sentinel block, cause=" + e.getClass().getName() + ", rule=" + e.getRule() + ", limitApp=" + e.getRuleLimitApp();
    }

    public String handleFallback(String value) {
        return value + "-> B server sentinel fallback";
    }
}
```

![]()![]( "点击并拖拽以移动")

原生Sentinel规则
------------

Sentinel在配置中心订阅的Key格式,如下:

```
1. Nacos的Key格式:Group为元数据中配置的[组名],Data Id为[服务名]-[规则类型]
2. Apollo的Key格式:[组名]-[服务名]-[规则类型]
```

![]()![]( "点击并拖拽以移动")

Sentinel规则的用法,请参照Sentinel官方文档

### 流控规则

增加服务discovery-guide-service-b的规则,Group为discovery-guide-group,Data Id为discovery-guide-service-b-sentinel-flow,规则内容如下:

```
[
    {
        "resource": "sentinel-resource",
        "limitApp": "default",
        "grade": 1,
        "count": 1,
        "strategy": 0,
        "refResource": null,
        "controlBehavior": 0,
        "warmUpPeriodSec": 10,
        "maxQueueingTimeMs": 500,
        "clusterMode": false,
        "clusterConfig": null
    }
]
```

![]()![]( "点击并拖拽以移动")

如图所示 
![1](https://yqfile.alicdn.com/e133cf771a41e5c2e8795501aee92603a2ed77c3.jpeg)

### 降级规则

增加服务discovery-guide-service-b的规则,Group为discovery-guide-group,Data Id为discovery-guide-service-b-sentinel-degrade,规则内容如下:

```
[
    {
        "resource": "sentinel-resource",
        "limitApp": "default",
        "count": 2,
        "timeWindow": 10,
        "grade": 0,
        "passCount": 0
    }
]
```

![]()![]( "点击并拖拽以移动")

如图所示 
![2](https://yqfile.alicdn.com/a995a3d1091cdcedaffc14f5f009727ebb1faa21.jpeg)

### 授权规则

增加服务discovery-guide-service-b的规则,Group为discovery-guide-group,Data Id为discovery-guide-service-b-sentinel-authority,规则内容如下:

```
[
    {
        "resource": "sentinel-resource",
        "limitApp": "discovery-guide-service-a",
        "strategy": 0
    }
]
```

![]()![]( "点击并拖拽以移动")

如图所示 
![3](https://yqfile.alicdn.com/004196045e42b362f4d5f719963dadd9934b0e8b.jpeg)

### 系统规则

增加服务discovery-guide-service-b的规则,Group为discovery-guide-group,Data Id为discovery-guide-service-b-sentinel-system,规则内容如下:

```
[
    {
        "resource": null,
        "limitApp": null,
        "highestSystemLoad": -1.0,
        "highestCpuUsage": -1.0,
        "qps": 200.0,
        "avgRt": -1,
        "maxThread": -1
    }
]
```

![]()![]( "点击并拖拽以移动")

如图所示 
![4](https://yqfile.alicdn.com/06ea05edf66db24b918ba3b01ae51278f958cd50.jpeg)

### 热点参数流控规则

增加服务discovery-guide-service-b的规则,Group为discovery-guide-group,Data Id为discovery-guide-service-b-sentinel-param-flow,规则内容如下:

```
[
    {
        "resource": "sentinel-resource",
        "limitApp": "default",
        "grade": 1,
        "paramIdx": 0,
        "count": 1,
        "controlBehavior": 0,
        "maxQueueingTimeMs": 0,
        "burstCount": 0,
        "durationInSec": 1,
        "paramFlowItemList": [],
        "clusterMode": false
    }
]
```

![]()![]( "点击并拖拽以移动")

如图所示 
![5](https://yqfile.alicdn.com/c200583fc33e1ca010b5f970632234e27ee68a37.jpeg)

基于灰度路由和Sentinel-LimitApp扩展的防护机制
-------------------------------

该方式对于上面5种规则都有效,这里以授权规则展开阐述

授权规则中,limitApp,如果有多个,可以通过“,”分隔。"strategy": 0 表示白名单,"strategy": 1 表示黑名单

### 基于服务名的防护机制

修改配置项Sentinel Request Origin Key为服务名的Header名称,修改授权规则中limitApp为对应的服务名,可实现基于服务名的防护机制

配置项,该配置项默认为n-d-service-id,可以不配置

```
spring.application.strategy.service.sentinel.request.origin.key=n-d-service-id
```

![]()![]( "点击并拖拽以移动")

增加服务discovery-guide-service-b的规则,Group为discovery-guide-group,Data Id为discovery-guide-service-b-sentinel-authority,规则内容如下,表示所有discovery-guide-service-a服务允许访问discovery-guide-service-b服务

```
[
    {
        "resource": "sentinel-resource",
        "limitApp": "discovery-guide-service-a",
        "strategy": 0
    }
]
```

![]()![]( "点击并拖拽以移动")

### 基于灰度组的防护机制

修改配置项Sentinel Request Origin Key为灰度组的Header名称,修改授权规则中limitApp为对应的组名,可实现基于组名的防护机制

配置项

```
spring.application.strategy.service.sentinel.request.origin.key=n-d-service-group
```

![]()![]( "点击并拖拽以移动")

增加服务discovery-guide-service-b的规则,Group为discovery-guide-group,Data Id为discovery-guide-service-b-sentinel-authority,规则内容如下,表示隶属my-group组的所有服务都允许访问服务discovery-guide-service-b

```
[
    {
        "resource": "sentinel-resource",
        "limitApp": "my-group",
        "strategy": 0
    }
]
```

![]()![]( "点击并拖拽以移动")

### 基于灰度版本的防护机制

修改配置项Sentinel Request Origin Key为灰度版本的Header名称,修改授权规则中limitApp为对应的版本,可实现基于版本的防护机制

配置项

```
spring.application.strategy.service.sentinel.request.origin.key=n-d-service-version
```

![]()![]( "点击并拖拽以移动")

增加服务discovery-guide-service-b的规则,Group为discovery-guide-group,Data Id为discovery-guide-service-b-sentinel-authority,规则内容如下,表示版本为1.0的所有服务都允许访问服务discovery-guide-service-b

```
[
    {
        "resource": "sentinel-resource",
        "limitApp": "1.0",
        "strategy": 0
    }
]
```

![]()![]( "点击并拖拽以移动")

### 基于灰度区域的防护机制

修改配置项Sentinel Request Origin Key为灰度区域的Header名称,修改授权规则中limitApp为对应的区域,可实现基于区域的防护机制

配置项

```
spring.application.strategy.service.sentinel.request.origin.key=n-d-service-region
```

![]()![]( "点击并拖拽以移动")

增加服务discovery-guide-service-b的规则,Group为discovery-guide-group,Data Id为discovery-guide-service-b-sentinel-authority,规则内容如下,表示区域为dev的所有服务都允许访问服务discovery-guide-service-b

```
[
    {
        "resource": "sentinel-resource",
        "limitApp": "dev",
        "strategy": 0
    }
]
```

![]()![]( "点击并拖拽以移动")

### 基于机器地址和端口的防护机制

修改配置项Sentinel Request Origin Key为灰度区域的Header名称,修改授权规则中limitApp为对应的区域值,可实现基于机器地址和端口的防护机制

配置项

```
spring.application.strategy.service.sentinel.request.origin.key=n-d-service-address
```

![]()![]( "点击并拖拽以移动")

增加服务discovery-guide-service-b的规则,Group为discovery-guide-group,Data Id为discovery-guide-service-b-sentinel-authority,规则内容如下,表示地址和端口为192.168.0.88:8081和192.168.0.88:8082的服务都允许访问服务discovery-guide-service-b

```
[
    {
        "resource": "sentinel-resource",
        "limitApp": "192.168.0.88:8081,192.168.0.88:8082",
        "strategy": 0
    }
]
```

![]()![]( "点击并拖拽以移动")

### 自定义业务参数的组合式防护机制

通过适配类实现自定义业务参数的组合式防护机制

```
// 版本号+用户名,实现组合式熔断
public class MyServiceSentinelRequestOriginAdapter extends DefaultServiceSentinelRequestOriginAdapter {
    @Override
    public String parseOrigin(HttpServletRequest request) {
        String version = request.getHeader(DiscoveryConstant.N_D_SERVICE_VERSION);
        String user = request.getHeader("user");

        return version + "&" + user;
    }
}
```

![]()![]( "点击并拖拽以移动")

在配置类里@Bean方式进行适配类创建

```
@Bean
public ServiceSentinelRequestOriginAdapter ServiceSentinelRequestOriginAdapter() {
    return new MyServiceSentinelRequestOriginAdapter();
}
```

![]()![]( "点击并拖拽以移动")

增加服务discovery-guide-service-b的规则,Group为discovery-guide-group,Data Id为discovery-guide-service-b-sentinel-authority,规则内容如下,表示版本为1.0且传入的Http Header的user=zhangsan,同时满足这两个条件下的所有服务都允许访问服务discovery-guide-service-b

```
[
    {
        "resource": "sentinel-resource",
        "limitApp": "1.0&zhangsan",
        "strategy": 0
    }
]
```

![]()![]( "点击并拖拽以移动")

运行效果

*   当传递的Http Header中user=zhangsan,当全链路调用中,API网关负载均衡discovery-guide-service-a服务到1.0版本后再去调用discovery-guide-service-b服务,最终调用成功

如图所示 
![6](https://yqfile.alicdn.com/199db580ac33bd2c9bd01f1cd55a4c39b8d03d23.jpeg)

*   当传递的Http Header中user=lisi,不满足条件,最终调用在discovery-guide-service-b服务端被拒绝掉

如图所示 
![7](https://yqfile.alicdn.com/03418cbe8f759ad5f576b23dfcb2dd2ffbe11cc1.jpeg)

*   当传递的Http Header中user=zhangsan,满足条件之一,当全链路调用中,API网关负载均衡discovery-guide-service-a服务到1.1版本后再去调用discovery-guide-service-b服务,不满足version=1.0的条件,最终调用在discovery-guide-service-b服务端被拒绝掉

如图所示 
![8](https://yqfile.alicdn.com/a53719d86ce29f92d8a3dd1b2e59fd548cdd9811.jpeg)

基于Swagger的Sentinel规则推送
----------------------

分为基于单个服务实例和基于服务名对应的多个服务实例的Sentinel规则推送

### 基于单个服务实例的Sentinel规则推送

直接访问该服务实例的Swagger主页即可

如图所示 
![9](https://yqfile.alicdn.com/1f73f71a029bbaf3ad07101e9fa1c321d60f28eb.jpeg)

### 基于服务名对应的多个服务实例的Sentinel规则推送

需要开启discovery-console服务,并访问其Swagger主页即可

如图所示 
![10](https://yqfile.alicdn.com/04e4029daab9f1f8d72d95281c653a58570cee74.jpeg)

**本文作者:**任浩军, 10 多年开源经历,Github ID:@HaojunRen,Nepxion 开源社区创始人,Nacos Group Member,Spring Cloud Alibaba & Nacos & Sentinel Committer ,曾就职于平安银行平台架构部,负责银行 PaaS 系统基础服务框架研发。 
王伟华, 10 余年 Java 开发,Github ID:@vipweihua,对微服务架构研究多年,当前更多关注于微服务中的网关、限流熔断、灰度路由等,现就职于平安银行平台架构部,从事银行 PaaS 系统基础服务框架研发。

 

 

[原文链接](https://link.zhihu.com/?target=https%3A//yq.aliyun.com/articles/721532%3Futm_content%3Dg_1000084426)

本文为云栖社区原创内容,未经允许不得转载。
分享到:
评论

相关推荐

    Nacos技术分享.pptx

    Spring Cloud Alibaba 是一套基于Spring Cloud的微服务解决方案,其中Nacos 提供了服务注册与发现和分布式配置中心的功能,同时还包含了如Sentinel(流量控制和服务降级)、RocketMQ(消息驱动)、Seata(分布式事务...

    Nepxion Discovery【探索】框架指南 V5.4.0.pdf

    全链路服务限流熔断降级权限,集成阿里巴巴Sentinel,有机整合灰度路由,扩展LimitApp的 机制,通过动态的Http Header方式实现组合式防护机制,包括基于服务名、基于灰度组、基于 灰度版本、基于灰度区域、基于机器...

    NepxionDiscovery-微服务企业级解决方案.docx

    10. **技术栈组件**:Nepxion Discovery依赖于阿里巴巴的中间件产品,如Nacos(服务注册发现和配置中心)、Sentinel(流量防护)和Spring Cloud Alibaba(微服务开发套件),这些组件协同工作,构建出强大的微服务...

    教育云原生技术架构实践.pdf

    在故障转移和恢复过程中,Nacos Eureka Sync提供了基于一致性Hash的分片路由策略,保证数据的一致性和服务的可用性。通过自动化运维工具如Ansible进行演练,验证了即使在极端情况下,如大部分节点故障,系统也能快速...

    DiscoveryGuide::sun: Nepxion Discovery is a solution for Spring Cloud with blue green, gray, route, limitation, circuit breaker, degrade, isolation, tracing, dye, failover 蓝绿、灰度、路由、限流、熔断、降级、隔离、追踪、流量染色、故障转移的指南

    Discovery【探索】云原生微服务解决方案   如果您觉得本框架具有一定的...Spring Cloud Alibaba、Nacos、Sentinel、OpenTracing Committer & Contributor 商业化合作 ① Discovery系列 框架名称 框架版本 支持Spr

    springcloudtext_springcloud学习笔记_

    SpringCloud是基于Spring Boot实现的一套微服务解决方案,它为开发者提供了在分布式系统(如配置管理、服务发现、断路器、智能路由、微代理、控制总线、一次性令牌、全局锁、领导选举、分布式会话、集群状态等)中...

    2020年springcloud alibaba课程源码.zip

    Spring Cloud Alibaba 是一套基于 Spring Boot 实现的微服务框架,它提供了全面的微服务解决方案,包含服务注册与发现、配置中心、全链路监控、分布式事务处理等多个关键功能。本课程源码是针对 2020 年的小滴课堂 ...

    Nepxion Discovery【探索】微服务企业级解决方案 V5.4.0.pdf

    • 支持Spring Cloud Gateway、Zuul网关和微服务三大模块的灰度发布和路由等一系列功能 兼容的Spring相关技术栈 • 兼容Spring Cloud Edgware版,兼容Spring Cloud Alibaba 1.5.x.RELEASE,兼容Spring Boot 1.5.x....

    sctest.rar

    在"**sctest.rar**"这个压缩包中,我们可以看到几个关键的服务组件的使用示例,这些组件包括Eureka、Zuul、Zookeeper、Sentinel、Nacos、Sleuth、Security和Gateway。下面将对这些组件进行详细介绍。 1. **Eureka**...

    微服务架构实战指南: 构建与治理高可用微服务系统

    - 实现了流量分担,可以针对不同模块进行优化和水平扩展。 - 提高了容错率,因为一个系统的问题不会影响其他系统。 - **缺点**: - 系统之间相互独立,无法直接相互调用。 - 存在重复开发的问题。 - **1.1.3 ...

    疯狂Spring Cloud微服务架构实战视频教程

    - **定义与作用**:Spring Cloud是一套用于构建微服务架构的工具集,基于Spring Boot实现,旨在提供一种简单的方式来进行配置管理、服务发现、断路器模式、智能路由、微代理、控制总线、一次性令牌协调器服务等操作...

    SpringCloud实战

    - **Alibaba Sentinel**:阿里巴巴开源的一款流量控制组件,适用于微服务、Docker 和 Kubernetes 架构。 - **服务网关**: - **Zuul**:早期版本的服务网关组件,现在已经被 Gateway 替代。 - **Gateway**:...

    test-spring-cloud-alibaba

    4. 测试与部署:自动化测试,灰度发布,滚动升级,保证服务的稳定性和可回滚性。 通过以上对Spring Cloud Alibaba的详细介绍,相信读者已经对这个框架有了深入的理解。在实际开发中,灵活运用这些组件,可以极大地...

Global site tag (gtag.js) - Google Analytics