阿里Sentinel整合Zuul網關詳解

前面咱們講解了Sentinel整合Spring Cloud Gateway,詳細請查看文章:阿里Sentinel支持Spring Cloud Gateway啦git

目前來講,大部分公司線上的網關應該是Zuul,因此今天咱們就來看看如何在Zuul中整合Sentinel。原本想基於Spring Cloud Alibaba來進行整合講解,整合的時候發現目前還沒更新版本,依賴仍是以前的版本,我們就以最原生的方式進行整合吧,等Spring Cloud Alibaba更新以後,Sentinel的整合只會變得更簡單。github

加入zuul-adapter依賴:json

<dependency>
    <groupId>com.alibaba.csp</groupId>
    <artifactId>sentinel-zuul-adapter</artifactId>
    <version>1.6.0</version>
</dependency>

配置Sentinel提供的限流過濾器和限流規則:api

@Configuration
public class ZuulConfig {

    @Bean
    public ZuulFilter sentinelZuulPreFilter() {
        return new SentinelZuulPreFilter();
    }

    @Bean
    public ZuulFilter sentinelZuulPostFilter() {
        return new SentinelZuulPostFilter();
    }

    @Bean
    public ZuulFilter sentinelZuulErrorFilter() {
        return new SentinelZuulErrorFilter();
    }

    @PostConstruct
    public void doInit() {
        // 註冊 FallbackProvider
        ZuulBlockFallbackManager.registerProvider(new MyBlockFallbackProvider());
        initGatewayRules();
    }

    /**
     * 配置限流規則
     */
    private void initGatewayRules() {
        Set<GatewayFlowRule> rules = new HashSet<>();
        rules.add(new GatewayFlowRule("yinjihuan").setCount(1) // 限流閾值
                .setIntervalSec(1) // 統計時間窗口,單位是秒,默認是 1 秒
        );

        GatewayRuleManager.loadRules(rules);
    }
}
  • SentinelZuulPreFilter

pre過濾器,在請求路由以前匹配routeId和api,進行限流操做微信

  • SentinelZuulPostFilter

post過濾器,路由以後恢復資源app

  • SentinelZuulErrorFilter

error過濾器,異常後的處理框架

最後再配置一個簡單的路由,路由名稱yinjihuan,跟上面規則中的名稱一致:ide

zuul.routes.yinjihuan.path=/cxytiandi/**
zuul.routes.yinjihuan.url=http://cxytiandi.com

觸發限流後會返回固定的提示:post

{
    "code":429,
    "message":"Sentinel block exception",
    "route":"yinjihuan"
}

若是想修改提示內容能夠本身實現ZuulBlockFallbackProvider接口,框架默認提供的實現是DefaultBlockFallbackProvider,源碼以下:學習

public class DefaultBlockFallbackProvider implements ZuulBlockFallbackProvider {

    @Override
    public String getRoute() {
        return "*";
    }

    @Override
    public BlockResponse fallbackResponse(String route, Throwable cause) {
        if (cause instanceof BlockException) {
            return new BlockResponse(429, "Sentinel block exception", route);
        } else {
            return new BlockResponse(500, "System Error", route);
        }
    }
}

用法其實跟Zuul中的FallbackProvider一致,可是FallbackProvider比較好的是返回的ClientHttpResponse,咱們能夠自定義響應內容。

Sentinel提供的ZuulBlockFallbackProvider接口中定義的返回對象是BlockResponse ,也就意味着限制了響應的字段,BlockResponse中有code,message,route三個字段,若是我想返回status, msg這兩個字段目前我沒找到其它的方式,不知道後續會不會支持,其實最好的是也返回ClientHttpResponse,這樣就能夠自定義響應內容了。

這邊有個小插曲,就是咱們自定義fallbackResponse的時候若是用中文message的話,響應內容是亂碼,以下:

{
code: 429,
message: "??????",
route: "yinjihuan"
}

我看了下SentinelZuulPreFilter中的代碼,以下:

這邊是構造了BlockResponse,而後設置到ResponseBody中,可是沒有進行編碼設置,我本身改了下源碼,加了一行代碼:

ctx.getResponse().setContentType("application/json; charset=utf-8");

加了上面的代碼後,中文就不會亂碼了,效果以下:

{
code: 429,
message: "訪問太頻繁啦",
route: "yinjihuan"
}

不說了,我仍是去提個issues吧: https://github.com/alibaba/Sentinel/issues/733

歡迎加入個人知識星球,一塊兒交流技術,免費學習猿天地的課程(http://cxytiandi.com/course

PS:目前星球中正在星主的帶領下組隊學習Spring Cloud,等你哦!

微信掃碼加入猿天地知識星球

猿天地

相關文章
相關標籤/搜索