初识 diamond client

上一篇文章,我们简单的看了一下diamond server的功能及所做的事情,本篇我们看一下diamond client 所做的事情。

首先我们说明的是,应用系统想要获得diamond中的动态配置,需要设置监听器,当监听到diamond的数据发生改变的时候,就会通知监听器接收改变后的数据。所以应用程序需要做的就是设置diamond的监听器。

Diamond.java

static public void addListener(String dataId, String group, ManagerListener listener) 
{    
           defaultEnv.addListeners(dataId, group, Arrays.asList(listener));
}

在使用这个方法时,会首先执行静态代码块的内容

static {   
         try {          
                 initLog();       
                 checkSnapshotValidity();    
              } 
         catch (Exception e)
              {        
                e.printStackTrace();        
                throw new RuntimeException(e);    
              }
          }
static private void checkSnapshotValidity()
    {    
             List<String> localServerlist = LocalConfigInfoProcessor.readServerlist(defaultEnv);    
             List<String> apacheServerlist = defaultEnv.getServerUrls(); 
             log.info("[apache-urls] " + apacheServerlist);    
             log.info("[cache-urls] " + localServerlist);    
             boolean isNotChange = apacheServerlist.equals(localServerlist);                if (isNotChange) {       
                log.info(LogConstants.PREFFIX + "environment ok.");   
            } else {        
                log.warn(LogConstants.PREFFIX + "environment changed. clear cache.");        
                LocalConfigInfoProcessor.cleanAllSnapshot();
                LocalConfigInfoProcessor.saveServerlist(defaultEnv, apacheServerlist);   
             }
     }

在这里又静态导入了defaultEnv,所以导致了defaultEnv的初始化。

static public final DiamondEnv defaultEnv = new DiamondEnv(new ServerListManager());

DiamondEnv 的初始化需要ServerListManager的实例。

protected DiamondEnv(ServerListManager serverListMgr) {
{
       initServerManager(serverListMgr);   //初始化ServerListManager 
       cacheMap = new AtomicReference<Map<String, CacheData>>(new HashMap<String, CacheData>());          //初始化放CacheData的容器
       worker = new ClientWorker(this);    //初始化ClientWorker
 }

看一下如何初始化的ServerListManager

public void initServerManager(ServerListManager _serverMgr) 
{          
    _serverMgr.setEnv(this);     //为ServerListManager 设置env
    serverMgr = _serverMgr;      //为DiamondEnv设置ServerListManager 
    serverMgr.start();           //ServerListManager 启动
    agent = new ServerHttpAgent(serverMgr); //创建一个http的Client,用于同diamond server进行http的通信。
}

关键是serverMgr.start();

public synchronized void start() 
{   
     if (isStarted || isFixed)   {  return; }               
     GetServerListTask getServersTask = new GetServerListTask(DiamondConfigure.singleton.getDiamondHttpUrl());
     while (serverUrls.isEmpty()) {       
        getServersTask.run();       
        try {            
           Thread.sleep(1000L); } 
        catch (Exception e) {}  
     }    
     TimerService.scheduleWithFixedDelay(getServersTask, 0L, 30L,TimeUnit.SECONDS);    
     isStarted = true;
}

这里有一个定时的线程池每30s执行这个任务,这个任务就是去diamond服务器取服务器的ip列表。我们发现有一个DiamondConfigure.singleton.getDiamondHttpUrl(),所以我们需要在设置监听前要设置这个diamond的http url。这个很重要,要不然client起不来。
根据服务器上获取的ip列表与本地的进行比较,不一致的话

LocalConfigInfoProcessor.saveServerlist(name, serverUrls);

更新本地的ip列表,否则直接return。这是initServerManager所做的事情。我们再来看一下

worker = new ClientWorker(this);

这里面做的事情就比较多了。

    ClientWorker(final DiamondEnv env) {
        this.env = env;   //ClientWorker中的env进行赋值
        executor = Executors.newScheduledThreadPool(1, new ThreadFactory() {
            @Override
            public Thread newThread(Runnable r) {
                Thread t = new Thread(r);
                t.setName("com.taobao.diamond.client.Worker."+ env.serverMgr.name);
                t.setDaemon(true);
                return t;
            }
        });

        executor.scheduleWithFixedDelay(new Runnable() {
            public void run() {
                try {
                    checkLocalConfigInfo();
                    checkServerConfigInfo();
                } catch (Throwable e) {
                    log.error("[sub-error-rotate] rotate check error", e);
                }
            }
        }, 1L, 1L, TimeUnit.MILLISECONDS);
    }

创建了一个定时的线程池,每隔1毫秒,进行checkLocalConfigInfo和checkServerConfigInfo,checkLocalConfigInfo是本地容灾文件的处理,
我们着重看一下checkServerConfigInfo。

static public void checkServerConfigInfo(DiamondEnv env) {
        for (String groupKey : checkUpdateDataIds(env)) {
            String dataId = GroupKey.parseKey(groupKey)[0];
            String group = GroupKey.parseKey(groupKey)[1];
            try {
                String content = getServerConfig(env, dataId, group, 3000L);
                CacheData cache = env.getCache(dataId, group);
                cache.setContent(content);

                log.info("[data-received] dataId=" + dataId + ", group=" + group + ", md5="
                        + cache.getMd5() + ", content=" + ContentUtils.truncateContent(content));
            } catch (IOException ioe) {
                log.warn(ioe.toString(), ioe);
            }
        }

        checkListenerMd5(env);
    }

checkUpdateDataIds 是将本地缓存的CacheData中的group,dataId和md5作为字符串拼接起来,作为参数,post到服务器上进行处理,根据group和dataId,查询到md5与传递过来的MD5进行比对,如果不一致,将不一致的group和dataId回传回来。

/**
     * 从DiamondServer获取值变化了的DataID列表。返回的对象里只有dataId和group是有效的。 保证不返回NULL。
     */
    static List<String> checkUpdateDataIds(DiamondEnv env) {
        /*
        if (MockServer.isTestMode()) {
            List<String> updateList = new ArrayList<String>();
            //与DiamondEnv的模拟数据源比较,得出数据变化列表
            for(CacheData cacheData : env.getAllCacheDataSnapshot()){
                CacheData mockServerData = env.getMockCache(cacheData.dataId, cacheData.group);
                if(mockServerData == null || !mockServerData.getMd5().equals(cacheData.getMd5()))
                    updateList.add(GroupKey.getKey(cacheData.dataId, cacheData.group));
            }
            return updateList;
        } */
        if (MockServer.isTestMode()) {
            // 避免 test mode cpu% 过高
            try {
                Thread.sleep(3000l);
            } catch (InterruptedException e) {}
            List<String> updateList = new ArrayList<String>();
            for(CacheData cacheData : env.getAllCacheDataSnapshot()){
                if(!CacheData.getMd5String(MockServer.getConfigInfo(cacheData.dataId, cacheData.group, env))
                        .equals(cacheData.getMd5())) {
                    updateList.add(GroupKey.getKey(cacheData.dataId, cacheData.group));
                }
            }
            return updateList;
        }


        String probeUpdateString = getProbeUpdateString(env);
        List<String> params = Arrays.asList(Constants.PROBE_MODIFY_REQUEST, probeUpdateString);
        long timeout = TimeUnit.SECONDS.toMillis(30L);

        List<String> headers = Arrays.asList("longPullingTimeout", "" + timeout);

        if (StringUtils.isBlank(probeUpdateString)) {
            return Collections.emptyList();
        }

        try {
            HttpResult result = env.agent.httpPost("/config.co", headers, params, Constants.ENCODE,
                    timeout);

            if (HttpURLConnection.HTTP_OK == result.code) {
                return parseUpdateDataIdResponse(result.content);
            } else {
                log.warn("[check-update] get changed dataId error, HTTP State: " + result.code);
            }
        } catch (IOException e) {
            log.warn("[check-update] get changed dataId exception, " + e.toString());
        }
        return Collections.emptyList();
    }

那么回传回来的group和dataId说明是有变化的,随后再请求服务器中的数据(最新的数据),放入本地的snapshot文件再放入放入CacheData

    String content = getServerConfig(env, dataId, group, 3000L);
    CacheData cache = env.getCache(dataId, group);
    cache.setContent(content);
/**
     * 对于404响应码,返回NULL.
     * 
     * @throws IOException  
     */
    static String getServerConfig(DiamondEnv env, String dataId, String group, long readTimeout)
            throws IOException {
        if (StringUtils.isBlank(group)) {
            group = Constants.DEFAULT_GROUP;
        }


        if (MockServer.isTestMode()) {
            return MockServer.getConfigInfo(dataId, group, env);
        }

        HttpResult result = null;
        try {
            List<String> params = Arrays.asList("dataId", dataId, "group", group);
            result = env.agent.httpGet("/config.co", null, params, Constants.ENCODE, readTimeout);
        } catch (IOException e) {
            log.warn("[sub-server] get server config exception, dataId=" + dataId + ", group="
                    + group + ", " + e.toString());
            throw e;
        }

        switch (result.code) {
        case HttpURLConnection.HTTP_OK:
            // if (env == defaultEnv) {
            LocalConfigInfoProcessor.saveSnapshot(env, dataId, group, result.content);
            // }
            return result.content;
        case HttpURLConnection.HTTP_NOT_FOUND:
            // if (env == defaultEnv) {
            LocalConfigInfoProcessor.saveSnapshot(env, dataId, group, null);
            // }
            return null;
        case HttpURLConnection.HTTP_CONFLICT: {
            log.warn("[sub-server-error] data being modified");
            throw new IOException("data being modified");
        }
        default: {
            log.warn("[sub-server-error] error code " + result.code);
            throw new IOException("http code: " + result.code);
        }
        }
    }

CacheData处理完了,在处理监听器。

    static void checkListenerMd5(DiamondEnv env) {
        for (CacheData cacheData : env.getAllCacheDataSnapshot()) {
            cacheData.checkListenerMd5();
        }
    }
    void checkListenerMd5() {
        for (ManagerListenerWrap wrap : listeners) {
            if (!md5.equals(wrap.lastCallMd5)) {
                safeNotifyListener(dataId, group, content, md5, wrap);
            }
    }
    static void safeNotifyListener(final String dataId, final String group, final String content,
            final String md5, ManagerListenerWrap listenerWrap) {
        final ManagerListener listener = listenerWrap.listener;
        listenerWrap.lastCallMd5 = md5;

        Runnable job = new Runnable() {
            public void run() {
                try {
                    listener.receiveConfigInfo(content);
                    log.info("[notify-ok] " + dataId + ", " + group + ", md5=" + md5
                            + ", listener=" + listener);
                } catch (Throwable t) {
                    log.error("[notify-error] " + dataId + ", " + group + ", md5=" + md5
                            + ", listener=" + listener.toString(), t);
                }
            }
        };

        try {
            if (null != listener.getExecutor()) {
                listener.getExecutor().execute(job);
            } else {
                job.run();
            }
        } catch (Throwable t) {
            log.error("[notify-error] " + dataId + ", " + group + ", md5=" + md5 + ", listener="
                    + listener.toString(), t);
        }
    }

每次调用safeNotifyListener时都会开启一个线程(有线程池用线程池),将改变后的内容传递给监听器进行处理。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 200,045评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,114评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 147,120评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,902评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,828评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,132评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,590评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,258评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,408评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,335评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,385评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,068评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,660评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,747评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,967评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,406评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,970评论 2 341

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,566评论 18 139
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,160评论 11 349
  • 1、OC中创建线程的方法是什么?如果指定在主线程中执行代码?如何延时执行代码。【难度系数★★】 1)创建线程的方法...
    木旁_G阅读 1,934评论 2 16
  • 若每天的生活夹带着欢声笑语,这样的日子真的令人期待,难道不是吗? 一处温暖的小窝,我们聊生活,聊综艺,聊...
    小混混儿阅读 281评论 0 0
  • 2007-08-31 15:28 今天早上6点不到就醒了,于是起床烧了开水,等LG把早饭放入电饭锅之后,我们就去方...
    AnnaFan阅读 576评论 0 0