soul网关学习11-配置数据同步1-HttpLongPolling_2

在上篇中我们分析了配置数据同步中HttpLongPollingsoul-bootstrap端的源码分析。在这一篇中,我们会分析soul-admin端的源码。
进入正题。。。

找切入点

  • soul-bootstrap端在长轮询中调用了soul-admin的两个接口:
# 拉取特定类型的配置
/configs/fetch
# 配置变更的监听
/configs/listener
  • 全局搜/configs是怎么提供的服务
    search-cibfugs
  • 我们定位到org.dromara.soul.admin.controller.ConfigController
    ConfigController

拉取配置fetchConfigs

分析

  • org.dromara.soul.admin.listener.AbstractDataChangedListener
    public ConfigData<?> fetchConfig(final ConfigGroupEnum groupKey) {
        // 配置数据的缓存
        ConfigDataCache config = CACHE.get(groupKey.name());
        // 不同类型则传入对应类型,返回configData
        switch (groupKey) {
            case APP_AUTH:
                List<AppAuthData> appAuthList = GsonUtils.getGson().fromJson(config.getJson(), new TypeToken<List<AppAuthData>>() {
                }.getType());
                // 对于每次的数据更新都有记录cache的md5值,最后更新时间
                return new ConfigData<>(config.getMd5(), config.getLastModifyTime(), appAuthList);
            case PLUGIN:
                List<PluginData> pluginList = GsonUtils.getGson().fromJson(config.getJson(), new TypeToken<List<PluginData>>() {
                }.getType());
                return new ConfigData<>(config.getMd5(), config.getLastModifyTime(), pluginList);
            case RULE:
                List<RuleData> ruleList = GsonUtils.getGson().fromJson(config.getJson(), new TypeToken<List<RuleData>>() {
                }.getType());
                return new ConfigData<>(config.getMd5(), config.getLastModifyTime(), ruleList);
            case SELECTOR:
                List<SelectorData> selectorList = GsonUtils.getGson().fromJson(config.getJson(), new TypeToken<List<SelectorData>>() {
                }.getType());
                return new ConfigData<>(config.getMd5(), config.getLastModifyTime(), selectorList);
            case META_DATA:
                List<MetaData> metaList = GsonUtils.getGson().fromJson(config.getJson(), new TypeToken<List<MetaData>>() {
                }.getType());
                return new ConfigData<>(config.getMd5(), config.getLastModifyTime(), metaList);
            default:
                throw new IllegalStateException("Unexpected groupKey: " + groupKey);
        }
    }
  • 这里注意到,对应配置数据是直接从内存cache中拿的,那什么时候将配置数据放到内存cache的?
  • 先来寻找cache的使用情况
    AbstractDataChangedListener.cache
  • 找到updateCache
    protected <T> void updateCache(final ConfigGroupEnum group, final List<T> data) {
        String json = GsonUtils.getInstance().toJson(data);
        ConfigDataCache newVal = new ConfigDataCache(group.name(), json, Md5Utils.md5(json), System.currentTimeMillis());
        ConfigDataCache oldVal = CACHE.put(newVal.getGroup(), newVal);
        log.info("update config cache[{}], old: {}, updated: {}", group, oldVal, newVal);
    }
  • 看起来这里没啥东西,没有找到出处;继续找updateCache的使用之处
    updateCache.usage
  • 点进去看一个,到updateSelectorCache,再继续往上找onSelectorChanged,再到org.dromara.soul.admin.listener.DataChangedEventDispatcher
    DataChangedEventDispatcher
  • DataChangedEventDispatcher使用了spring的内存应用事件机制,为事件消费端,再找下事件发布端
    DataChangedEventDispatcher.event
  • 查找关键字DataChangedEvent,看下事件发布的地方
  • 差不多可以了,找到了源头的地方,下面总结一下

总结

  1. ConfigController提供接口配置获取/configs/fetch,供soul-bootstrap调用
  2. http长轮询数据变更监听器HttpLongPollingDataChangedListener,提供fetchConfig方法,其中,所有配置数据是存放在其成员变量cache中的;拉取特定类型的配置,只需要从cache中取出来就行了
  3. 关于配置数据的存放,则是用户在soul-admin的web界面,对配置数据更新时,会通过spring的应用事件机制,将变更的数据发布出来,事件为DataChangedEvent;而监听器端则监听DataChangedEvent事件,实现对应数据变更的存放
  4. 上述是增量数据的处理;
  5. 全量数据是如何加载到cache中的?
  6. 仔细看HttpLongPollingDataChangedListener,发现在实例化的过程中,会创建一个定时任务线程池,其提供一个后台守护线程,默认情况下会每隔5min钟会从数据库中拉取配置数据加载到内存。
   /**
    * Instantiates a new Http long polling data changed listener.
    * @param httpSyncProperties the HttpSyncProperties
    */
   public HttpLongPollingDataChangedListener(final HttpSyncProperties httpSyncProperties) {
       this.clients = new ArrayBlockingQueue<>(1024);
       // 后台定期reload数据库配置数据的线程池
       this.scheduler = new ScheduledThreadPoolExecutor(1,
               SoulThreadFactory.create("long-polling", true));
       this.httpSyncProperties = httpSyncProperties;
   }

   @Override
   protected void afterInitialize() {
       long syncInterval = httpSyncProperties.getRefreshInterval().toMillis();
       // Periodically check the data for changes and update the cache
       // 启动这个定时任务线程池,用于reload数据库配置到本地缓存
       scheduler.scheduleWithFixedDelay(() -> {
           log.info("http sync strategy refresh config start.");
           try {
               this.refreshLocalCache();
               log.info("http sync strategy refresh config success.");
           } catch (Exception e) {
               log.error("http sync strategy refresh config error!", e);
           }
       }, syncInterval, syncInterval, TimeUnit.MILLISECONDS);
       log.info("http sync strategy refresh interval: {}ms", syncInterval);
   }

   private void refreshLocalCache() {
       this.updateAppAuthCache();
       this.updatePluginCache();
       this.updateRuleCache();
       this.updateSelectorCache();
       this.updateMetaDataCache();
   }
  • 该操作只会reload,并不会生成update的事件,通知给soul-bootstrap
    现在就只剩下一个问题了,当本地缓存数据有更新时,是如何通知到soul-bootstrap的呢?下面我们来分析这个问题。

配置变更的监听与响应

分析

  • 我们知道soul-bootstrap是通过回调长轮询的方式完成配置的监听,那实际上我们只要跟踪监听的接口逻辑就行
  • 监听接口/config/listener中调用HttpLongPollingDataChangedListener.doLongPolling
public void doLongPolling(final HttpServletRequest request, final HttpServletResponse response) {

        // compare group md5
        // 根据监听传入的md5与更新时间戳找到变化的配置数据
        List<ConfigGroupEnum> changedGroup = compareChangedGroup(request);
        String clientIp = getRemoteIp(request);

        // response immediately.
        // 如果此次存在变化的配置数据,则直接响应请求,将变化的配置类型返回给soul-bootstrap
        if (CollectionUtils.isNotEmpty(changedGroup)) {
            this.generateResponse(response, changedGroup);
            log.info("send response with the changed group, ip={}, group={}", clientIp, changedGroup);
            return;
        }

        // listen for configuration changed.
        // 否则将当前请求异步化
        final AsyncContext asyncContext = request.startAsync();

        // AsyncContext.settimeout() does not timeout properly, so you have to control it yourself
        // 不设置超时
        asyncContext.setTimeout(0L);

        // block client's thread.
        // 通过调度线程池去执行监听长轮询任务,这里的execute是立即执行的
        scheduler.execute(new LongPollingClient(asyncContext, clientIp, HttpConstants.SERVER_MAX_HOLD_TIMEOUT));
    }
  • 接入的请求会开启异步(servelet3.0支持),并将其封装成长轮询客户端LongPollingClient后丢给调度线程池,并立即执行
  • LongPollingClient中的run方法有点精巧,里边的执行逻辑并没有立即执行,而是先丢给调度线程池,并延迟60s执行;同时LongPollingClient会添加到长轮询队列clients
       public void run() {
            // 这里并没有立即执行,会将其丢到调度线程池,延迟60s执行
            this.asyncTimeoutFuture = scheduler.schedule(() -> {
                // 执行时,先将当前长轮询的client从长轮询队列队列中移除
                clients.remove(LongPollingClient.this);
                // 检查是否存在变更的配置
                List<ConfigGroupEnum> changedGroups = compareChangedGroup((HttpServletRequest) asyncContext.getRequest());
                // 返回结果
                sendResponse(changedGroups);
            }, timeoutTime, TimeUnit.MILLISECONDS);
            // 将当前长轮询的client放入长轮询队列中
            clients.add(this);
        }
  • 上述做法的目的是,在这延迟的60s中,如果有配置变更产生,则会由配置变更的任务DataChangeTask,遍历现有的长轮询队列clients,依次移除,并完成LongPollingClient的返回结果设置,将异步化的请求操作完结掉;
        public void run() {
            // 遍历当前所有正在长轮询的client,将变更的数据作为此次轮询响应的结果返回给长轮询的client
            //TODO question 这里是否会存在配置丢失的情况?
            // 如果两次间隔很近的配置变更过来,第一次配置变更还在返回给client,此时的client并没有重新轮询进来,
            // 则会导致第二次配置变更没有通知到第一次已通知的client,从而使得某些client节点丢失配置
            // 在admin是集群的情况下,该数据同步机制可能更不可靠
            for (Iterator<LongPollingClient> iter = clients.iterator(); iter.hasNext();) {
                // 从长轮询队列中移除client
                LongPollingClient client = iter.next();
                iter.remove();
                // 并将变更的数据返回给长轮询client
                client.sendResponse(Collections.singletonList(groupKey));
                log.info("send response with the changed group,ip={}, group={}, changeTime={}", client.ip, groupKey, changeTime);
            }
        }
  • LongPollingClient 的返回结果设置
       void sendResponse(final List<ConfigGroupEnum> changedGroups) {
            // cancel scheduler
            // 如果在延迟60s的窗口中,存在配置变更的数据,则会提前结束,把变更的数据给到长轮询client;
            // 这里的asyncTimeoutFuture便会为空,从而可以取消当前延迟执行的任务
            if (null != asyncTimeoutFuture) {
                asyncTimeoutFuture.cancel(false);
            }
            generateResponse((HttpServletResponse) asyncContext.getResponse(), changedGroups);
            asyncContext.complete();
        }
  • 分析结束,做下总结

总结

  1. soul-bootstrapHttpLongPollingTask中采用请求回调轮询的方式,去轮询soul-admin中的配置监听接口/configs/listener,其中每次请求的超时时间为90s
  2. soul-admin中通过HttpLongPollingDataChangedListener.doLongPolling方法开启请求的异步支持request.startAsync(),避免阻塞住soul-admin端的请求Acceptor线程;
  3. 将异步化请求AsyncContext封装成长轮询客户端任务LongPollingClient,通过调度线程池scheduler执行。
  4. 长轮询客户端任务LongPollingClientrun方法,将自身逻辑丢给调度线程池scheduler延迟60s执行,这样就实现请求/configs/listener至少会保持60s
  5. 长轮询客户端任务LongPollingClient还会将自身加入到长轮询客户端队列clients
  6. 如果在请求保持的60s中,存在有配置变更产生(产生来源是用户在配置web界面操作,包括对插件、选择器、规则的增删改,此类操作会自动触发配置变更事件;还有web端提供的一些强制同步功能,如各个插件中的同步、插件管理的同步,也会产生配置变更事件),则会由数据变更任务DataChangeTask,遍历现有的长轮询队列clients,依次移除,并完成长轮询客户端任务LongPollingClient的返回结果设置,将异步化的请求操作完结掉
  7. 上述就是soul-boostrapsoul-admin之间,http长轮询HttpLongPolling同步方式的配置监听与响应的大致流程。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,491评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,856评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,745评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,196评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,073评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,112评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,531评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,215评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,485评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,578评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,356评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,215评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,583评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,898评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,497评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,697评论 2 335

推荐阅读更多精彩内容