soul網(wǎng)關學習5-Hystrix插件流程解析

Hystrix插件圖

Hystrix類繼承圖

Bootstrap依賴模塊

bootstrap依賴模塊

網(wǎng)關插件啟動關鍵段

  • 將網(wǎng)關插件列表注入到webHandler
    // @see org.dromara.soul.web.configuration.SoulConfiguration
    @Bean("webHandler")
    public SoulWebHandler soulWebHandler(final ObjectProvider<List<SoulPlugin>> plugins) {
        // 將所有的網(wǎng)關插件躯肌,掛載到webHandler中蛉拙;且按照順序掛載
List<SoulPlugin> pluginList = plugins.getIfAvailable(Collections::emptyList);
        final List<SoulPlugin> soulPlugins = pluginList.stream()
                .sorted(Comparator.comparingInt(SoulPlugin::getOrder)).collect(Collectors.toList());
        soulPlugins.forEach(soulPlugin -> log.info("load plugin:[{}] [{}]", soulPlugin.named(), soulPlugin.getClass().getName()));
        return new SoulWebHandler(soulPlugins);
    }
  • 推測webHandler會將傳入的插件列表,生成插件鏈奴烙,在處理流程中依次處理
   // @see org.dromara.soul.web.handler.SoulWebHandler
   public Mono<Void> handle(@NonNull final ServerWebExchange exchange) {
        MetricsTrackerFacade.getInstance().counterInc(MetricsLabelEnum.REQUEST_TOTAL.getName());
        Optional<HistogramMetricsTrackerDelegate> startTimer = MetricsTrackerFacade.getInstance().histogramStartTimer(MetricsLabelEnum.REQUEST_LATENCY.getName());
        // 直接將插件列表生成一個插件鏈DefaultSoulPluginChain叮叹,責任鏈的execute方法會遍歷所有插件列表,依次處理
       // Mono的subscribeOn沒看明白乏苦???【就是講線程池綁定到Mono請求流程中】汇荐;難道是經(jīng)過Mono的請求洞就,都會提交給這個線程池執(zhí)行?Mono本身是有默認線程池吧掀淘,這里為什么要自用線程池旬蟋?應該是自行創(chuàng)建的Mono對象
        return new DefaultSoulPluginChain(plugins).execute(exchange).subscribeOn(scheduler)
                .doOnSuccess(t -> startTimer.ifPresent(time -> MetricsTrackerFacade.getInstance().histogramObserveDuration(time)));
    }
  • 我們看下網(wǎng)關插件鏈DefaultSoulPluginChain.execute的邏輯實現(xiàn)
      // 1. 執(zhí)行方法,遍歷插件列表革娄,并調(diào)用插件的執(zhí)行邏輯
      // 2. 返回Mono對象倾贰,能方便的使用Reactive的事件驅(qū)動編程的機制
       public Mono<Void> execute(final ServerWebExchange exchange) {
            return Mono.defer(() -> {
                if (this.index < plugins.size()) {
                    SoulPlugin plugin = plugins.get(this.index++);
                    // 先判斷當前請求是否滿足插件處理的邏輯,如果不滿足拦惋,則跳過不處理匆浙;否則進行處理
                    Boolean skip = plugin.skip(exchange);
                    if (skip) {
                       // 跳過,直接執(zhí)行下一個插件
                        return this.execute(exchange);
                    }
                   // 執(zhí)行當前插件
                    return plugin.execute(exchange, this);
                }
                // 所有插件執(zhí)行完畢
                return Mono.empty();
            });
        }
  • 我們看下插件HystrixPlugin的執(zhí)行邏輯架忌,從上面的類圖可以得知是執(zhí)行的AbstractSoulPluginexecute方法
    public Mono<Void> execute(final ServerWebExchange exchange, final SoulPluginChain chain) {
        String pluginName = named();
        // 獲取插件的配置數(shù)據(jù)
        final PluginData pluginData = BaseDataCache.getInstance().obtainPluginData(pluginName);
        if (pluginData != null && pluginData.getEnabled()) {
            // 獲取插件選擇器
            final Collection<SelectorData> selectors = BaseDataCache.getInstance().obtainSelectorData(pluginName);
            // 插件選擇器為空的處理
            if (CollectionUtils.isEmpty(selectors)) {
                return handleSelectorIsNull(pluginName, exchange, chain);
            }
            // 匹配選擇器
            final SelectorData selectorData = matchSelector(exchange, selectors);
            if (Objects.isNull(selectorData)) {
                return handleSelectorIsNull(pluginName, exchange, chain);
            }
            selectorLog(selectorData, pluginName);
            // 獲取選擇器的規(guī)則
            final List<RuleData> rules = BaseDataCache.getInstance().obtainRuleData(selectorData.getId());
            // 規(guī)則為空處理
            if (CollectionUtils.isEmpty(rules)) {
                return handleRuleIsNull(pluginName, exchange, chain);
            }
            // 匹配規(guī)則
            RuleData rule;
            if (selectorData.getType() == SelectorTypeEnum.FULL_FLOW.getCode()) {
                //get last
                rule = rules.get(rules.size() - 1);
            } else {
                rule = matchRule(exchange, rules);
            }
            // 未匹配到規(guī)則的處理
            if (Objects.isNull(rule)) {
                return handleRuleIsNull(pluginName, exchange, chain);
            }
            ruleLog(rule, pluginName);
            // 將匹配到插件配置數(shù)據(jù)吞彤,傳遞給插件本身處理自身的業(yè)務邏輯
            return doExecute(exchange, chain, selectorData, rule);
        }
        // 插件數(shù)據(jù)沒有我衬,則直接進入到下一個插件
        return chain.execute(exchange);
    }
  • 再看HystrixPlugindoExecute方法
    protected Mono<Void> doExecute(final ServerWebExchange exchange, final SoulPluginChain chain, final SelectorData selector, final RuleData rule) {
        final SoulContext soulContext = exchange.getAttribute(Constants.CONTEXT);
        assert soulContext != null;
        // 構(gòu)造hystrix處理
        final HystrixHandle hystrixHandle = GsonUtils.getInstance().fromJson(rule.getHandle(), HystrixHandle.class);
        if (StringUtils.isBlank(hystrixHandle.getGroupKey())) {
            hystrixHandle.setGroupKey(Objects.requireNonNull(soulContext).getModule());
        }
        if (StringUtils.isBlank(hystrixHandle.getCommandKey())) {
            hystrixHandle.setCommandKey(Objects.requireNonNull(soulContext).getMethod());
        }
        // 生成hystrix command
        Command command = fetchCommand(hystrixHandle, exchange, chain);
        // 這里就木有看懂了叹放,需額外學習RxJava
        return Mono.create(s -> {
            Subscription sub = command.fetchObservable().subscribe(s::success,
                    s::error, s::success);
            s.onCancel(sub::unsubscribe);
            if (command.isCircuitBreakerOpen()) {
                log.error("hystrix execute have circuitBreaker is Open! groupKey:{},commandKey:{}", hystrixHandle.getGroupKey(), hystrixHandle.getCommandKey());
            }
        }).doOnError(throwable -> {
            log.error("hystrix execute exception:", throwable);
            exchange.getAttributes().put(Constants.CLIENT_RESPONSE_RESULT_TYPE, ResultEnum.ERROR.getName());
            chain.execute(exchange);
        }).then();
    }

TODO

  1. soul網(wǎng)關如何調(diào)用hystrix進行控制的?
  2. Hystrix原理學習
  3. RxJava學習
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末挠羔,一起剝皮案震驚了整個濱河市井仰,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌破加,老刑警劉巖俱恶,帶你破解...
    沈念sama閱讀 221,430評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異范舀,居然都是意外死亡合是,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,406評論 3 398
  • 文/潘曉璐 我一進店門锭环,熙熙樓的掌柜王于貴愁眉苦臉地迎上來聪全,“玉大人,你說我怎么就攤上這事辅辩∧牙瘢” “怎么了?”我有些...
    開封第一講書人閱讀 167,834評論 0 360
  • 文/不壞的土叔 我叫張陵玫锋,是天一觀的道長蛾茉。 經(jīng)常有香客問我,道長撩鹿,這世上最難降的妖魔是什么谦炬? 我笑而不...
    開封第一講書人閱讀 59,543評論 1 296
  • 正文 為了忘掉前任,我火速辦了婚禮节沦,結(jié)果婚禮上键思,老公的妹妹穿的比我還像新娘窜管。我一直安慰自己,他們只是感情好稚机,可當我...
    茶點故事閱讀 68,547評論 6 397
  • 文/花漫 我一把揭開白布幕帆。 她就那樣靜靜地躺著,像睡著了一般赖条。 火紅的嫁衣襯著肌膚如雪失乾。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,196評論 1 308
  • 那天纬乍,我揣著相機與錄音碱茁,去河邊找鬼。 笑死仿贬,一個胖子當著我的面吹牛纽竣,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播茧泪,決...
    沈念sama閱讀 40,776評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼蜓氨,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了队伟?” 一聲冷哼從身側(cè)響起穴吹,我...
    開封第一講書人閱讀 39,671評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎嗜侮,沒想到半個月后港令,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,221評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡锈颗,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,303評論 3 340
  • 正文 我和宋清朗相戀三年顷霹,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片击吱。...
    茶點故事閱讀 40,444評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡淋淀,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出姨拥,到底是詐尸還是另有隱情绅喉,我是刑警寧澤,帶...
    沈念sama閱讀 36,134評論 5 350
  • 正文 年R本政府宣布叫乌,位于F島的核電站柴罐,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏憨奸。R本人自食惡果不足惜革屠,卻給世界環(huán)境...
    茶點故事閱讀 41,810評論 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧似芝,春花似錦那婉、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,285評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至寞奸,卻和暖如春呛谜,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背枪萄。 一陣腳步聲響...
    開封第一講書人閱讀 33,399評論 1 272
  • 我被黑心中介騙來泰國打工隐岛, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人瓷翻。 一個月前我還...
    沈念sama閱讀 48,837評論 3 376
  • 正文 我出身青樓聚凹,卻偏偏與公主長得像,于是被迫代替她去往敵國和親齐帚。 傳聞我的和親對象是個殘疾皇子妒牙,可洞房花燭夜當晚...
    茶點故事閱讀 45,455評論 2 359

推薦閱讀更多精彩內(nèi)容