我是靠谱客的博主 包容天空,最近开发中收集的这篇文章主要介绍(十三)soul的springcloud插件底层原理((十三)soul的springcloud插件底层原理,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

((十三)soul的springcloud插件底层原理

目标

  • springcloud插件底层原理

springcloud插件底层原理

springcloud插件的具体使用使用在小节三里有分析了,这里不解释
启动一个adminadmin和一个网关,两个order-service,一个eureka

  • 我们在SpringCloudPlugin的doExecute打断点

在这里插入图片描述

从SoulWebHandler#execute, SoulWebHandler是插件链的处理器,依次按照order大小依次遍历插件列表执行

        @Override
        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) {
                        return this.execute(exchange);
                    }
                    return plugin.execute(exchange, this);
                }
                return Mono.empty();
            });
        }

plugins对象如下图
在这里插入图片描述

plugin.execute(exchange, this)执行来到,AbstractSoulPlugin的doExecute
if (pluginData != null && pluginData.getEnabled()) :如果对应插件没打开,则直接回到上面SoulWebHandler#execute,接着下一个插件处理器处理
如果对应插件打开,则走doExecute(exchange, chain, selectorData, rule),此处用到模板设计模式,具体实现在继承类

  public Mono<Void> execute(final ServerWebExchange exchange, final SoulPluginChain chain) {
       String pluginName = named();
       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);
           final List<RuleData> rules = BaseDataCache.getInstance().obtainRuleData(selectorData.getId());
           if (CollectionUtils.isEmpty(rules)) {
               return handleRuleIsNull(pluginName, exchange, chain);
           }
           RuleData rule;
           if (selectorData.getType() == SelectorTypeEnum.FULL_FLOW.getCode()) {
               //get last
               rule = rules.get(rules.size() - 1);
           } else {
               rule = matchRule(exchange, rules);
           }
           if (Objects.isNull(rule)) {
               return handleRuleIsNull(pluginName, exchange, chain);
           }
           ruleLog(rule, pluginName);
           return doExecute(exchange, chain, selectorData, rule);
       }
       return chain.execute(exchange);
   }

doExecute(exchange, chain, selectorData, rule)到springcloud插件
1:exchange.getAttribute(Constants.CONTEXT):拿到我们的admin配置信息
2:realURL真实请求业务实例的ip
loadBalancer.choose(selectorHandle.getServiceId())通过ribbon的负载均衡,来获取具体的哪个调用的实例ip信息
负载均衡的实例是因为依赖了包spring-cloud-starter-netflix-ribbon,netflix-ribbon包具体处理负载

   @Override
   protected Mono<Void> doExecute(final ServerWebExchange exchange, final SoulPluginChain chain, final SelectorData selector, final RuleData rule) {
       if (Objects.isNull(rule)) {
           return Mono.empty();
       }
       final SoulContext soulContext = exchange.getAttribute(Constants.CONTEXT);
       assert soulContext != null;
       final SpringCloudRuleHandle ruleHandle = GsonUtils.getInstance().fromJson(rule.getHandle(), SpringCloudRuleHandle.class);
       final SpringCloudSelectorHandle selectorHandle = GsonUtils.getInstance().fromJson(selector.getHandle(), SpringCloudSelectorHandle.class);
       if (StringUtils.isBlank(selectorHandle.getServiceId()) || StringUtils.isBlank(ruleHandle.getPath())) {
           Object error = SoulResultWrap.error(SoulResultEnum.CANNOT_CONFIG_SPRINGCLOUD_SERVICEID.getCode(), SoulResultEnum.CANNOT_CONFIG_SPRINGCLOUD_SERVICEID.getMsg(), null);
           return WebFluxResultUtils.result(exchange, error);
       }

       final ServiceInstance serviceInstance = loadBalancer.choose(selectorHandle.getServiceId());
       if (Objects.isNull(serviceInstance)) {
           Object error = SoulResultWrap.error(SoulResultEnum.SPRINGCLOUD_SERVICEID_IS_ERROR.getCode(), SoulResultEnum.SPRINGCLOUD_SERVICEID_IS_ERROR.getMsg(), null);
           return WebFluxResultUtils.result(exchange, error);
       }
       final URI uri = loadBalancer.reconstructURI(serviceInstance, URI.create(soulContext.getRealUrl()));

       String realURL = buildRealURL(uri.toASCIIString(), soulContext.getHttpMethod(), exchange.getRequest().getURI().getQuery());

       exchange.getAttributes().put(Constants.HTTP_URL, realURL);
       //set time out.
       exchange.getAttributes().put(Constants.HTTP_TIME_OUT, ruleHandle.getTimeout());
       return chain.execute(exchange);
   }

springcloud的插件探活

  • 我们依赖了包:spring-cloud-starter-netflix-eureka-client,服务的注册和发现,它探活完成使用eureka这套,交由eureke判断业务服务实例是否可用

总结

  • 今天下班太晚了,明天继续
    ring-cloud-starter-netflix-eureka-client,服务的注册和发现,它探活完成使用eureka这套,交由eureke判断业务服务实例是否可用

总结

  • 今天下班太晚了,明天继续

最后

以上就是包容天空为你收集整理的(十三)soul的springcloud插件底层原理((十三)soul的springcloud插件底层原理的全部内容,希望文章能够帮你解决(十三)soul的springcloud插件底层原理((十三)soul的springcloud插件底层原理所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(67)

评论列表共有 0 条评论

立即
投稿
返回
顶部