初識 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閱讀 229,117評論 6 537
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,860評論 3 423
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 177,128評論 0 381
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,291評論 1 315
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,025評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,421評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,477評論 3 444
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,642評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,177評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,970評論 3 356
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,157評論 1 371
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,717評論 5 362
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,410評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,821評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,053評論 1 289
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,896評論 3 395
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,157評論 2 375

推薦閱讀更多精彩內容

  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,806評論 18 139
  • 從三月份找實習到現在,面了一些公司,掛了不少,但最終還是拿到小米、百度、阿里、京東、新浪、CVTE、樂視家的研發崗...
    時芥藍閱讀 42,329評論 11 349
  • 1、OC中創建線程的方法是什么?如果指定在主線程中執行代碼?如何延時執行代碼。【難度系數★★】 1)創建線程的方法...
    木旁_G閱讀 1,981評論 2 16
  • 若每天的生活夾帶著歡聲笑語,這樣的日子真的令人期待,難道不是嗎? 一處溫暖的小窩,我們聊生活,聊綜藝,聊...
    小混混兒閱讀 292評論 0 0
  • 2007-08-31 15:28 今天早上6點不到就醒了,于是起床燒了開水,等LG把早飯放入電飯鍋之后,我們就去方...
    AnnaFan閱讀 599評論 0 0