Eureka源碼剖析之五:服務(wù)下線

Eureka源碼剖析之一:初始化-啟動(dòng)

Eureka源碼剖析之二:服務(wù)注冊(cè)

Eureka源碼剖析之三:服務(wù)拉取

Eureka源碼剖析之四:服務(wù)續(xù)約

現(xiàn)在研究下Eureka服務(wù)下線的源碼虾啦。由服務(wù)續(xù)約的源碼我們知道,如果客戶端在90秒內(nèi)沒(méi)有繼續(xù)跟服務(wù)端進(jìn)行心跳的話滤淳,服務(wù)端會(huì)進(jìn)行下線客戶端并且更改狀態(tài)將其剔除勺鸦,并且也會(huì)在集群中告知(同步)其它節(jié)點(diǎn)亲配。

〓Eureka Client

    /**
     * 注銷服務(wù),調(diào)用client的cancel服務(wù)善榛,往里面看也就是調(diào)用了服務(wù)端的http delete 請(qǐng)求進(jìn)行服務(wù)下線
     */     void unregister() {
        // It can be null if shouldRegisterWithEureka == false         if(eurekaTransport != null && eurekaTransport.registrationClient != null) {
            try {
                logger.info("Unregistering ...");
                EurekaHttpResponse<Void> httpResponse = eurekaTransport.registrationClient.cancel(instanceInfo.getAppName(), instanceInfo.getId());
                logger.info(PREFIX + appPathIdentifier + " - deregister  status: " + httpResponse.getStatusCode());
            } catch (Exception e) {
                logger.error(PREFIX + appPathIdentifier + " - de-registration failed" + e.getMessage(), e);
            }
        }
    }

    @Override     public EurekaHttpResponse<Void> cancel(String appName, String id) {
        String urlPath = "apps/" + appName + '/' + id;
        ClientResponse response = null;
        try {
            Builder resourceBuilder = jerseyClient.resource(serviceUrl).path(urlPath).getRequestBuilder();
            addExtraHeaders(resourceBuilder);
            response = resourceBuilder.delete(ClientResponse.class);
            return anEurekaHttpResponse(response.getStatus()).headers(headersOf(response)).build();
        } finally {
            if (logger.isDebugEnabled()) {
                logger.debug("Jersey HTTP DELETE {}/{}; statusCode={}", serviceUrl, urlPath, response == null ? "N/A" : response.getStatus());
            }
            if (response != null) {
                response.close();
            }
        }
    } 

接下來(lái)再看是哪里調(diào)用注銷服務(wù):

    @PreDestroy     @Override     public synchronized void shutdown() {
        if (isShutdown.compareAndSet(false, true)) {
            logger.info("Shutting down DiscoveryClient ...");
            // 注銷服務(wù)狀態(tài)的監(jiān)聽(tīng)器             if (statusChangeListener != null && applicationInfoManager != null) {
                applicationInfoManager.unregisterStatusChangeListener(statusChangeListener.getId());
            }
            // 取消定時(shí)器任務(wù)砾肺,關(guān)閉線程池等             cancelScheduledTasks();

            // 服務(wù)實(shí)例以及被注冊(cè),那么設(shè)置實(shí)例狀態(tài)為DOWN翩蘸,并且進(jìn)行注銷操作             if (applicationInfoManager != null && clientConfig.shouldRegisterWithEureka()) {
                applicationInfoManager.setInstanceStatus(InstanceStatus.DOWN);
                unregister();
            }
            // 關(guān)閉eurekaTransport             if (eurekaTransport != null) {
                eurekaTransport.shutdown();
            }
            // 關(guān)閉監(jiān)控             heartbeatStalenessMonitor.shutdown();
            registryStalenessMonitor.shutdown();

            logger.info("Completed shut down of DiscoveryClient");
        }
    }

    // DiscoveryManager調(diào)用了DiscveryClient的shutdown方法所意,這里就是服務(wù)下線的入口     public void shutdownComponent() {
        if (discoveryClient != null) {
            try {
                discoveryClient.shutdown();
                discoveryClient = null;
            } catch (Throwable th) {
                logger.error("Error in shutting down client", th);
            }
        }
    } 

〓Eureka Server

    // 服務(wù)端提供的對(duì)外服務(wù)下線接口     @DELETE     public Response cancelLease(
            @HeaderParam(PeerEurekaNode.HEADER_REPLICATION) String isReplication) {
        boolean isSuccess = registry.cancel(app.getName(), id,
                "true".equals(isReplication));

        if (isSuccess) {
            logger.debug("Found (Cancel): " + app.getName() + " - " + id);
            return Response.ok().build();
        } else {
            logger.info("Not Found (Cancel): " + app.getName() + " - " + id);
            return Response.status(Status.NOT_FOUND).build();
        }
    }

    // 服務(wù)端下線入口     @Override     public boolean cancel(final String appName, final String id,
                          final boolean isReplication) {
        // 調(diào)用父類cancel方法,將實(shí)例信息添加到最近下線隊(duì)列催首、最近變更隊(duì)列扶踊,并且使本地緩存失效操作         if (super.cancel(appName, id, isReplication)) {
            // 服務(wù)端集群之間進(jìn)行Cancel同步操作             replicateToPeers(Action.Cancel, appName, id, null, null, isReplication);
            synchronized (lock) {
                if (this.expectedNumberOfRenewsPerMin > 0) {
                    // Since the client wants to cancel it, reduce the threshold (1 for 30 seconds, 2 for a minute)                     this.expectedNumberOfRenewsPerMin = this.expectedNumberOfRenewsPerMin - 2;
                    this.numberOfRenewsPerMinThreshold =
                            (int) (this.expectedNumberOfRenewsPerMin * serverConfig.getRenewalPercentThreshold());
                }
            }
            return true;
        }
        return false;
    }

    // 最終調(diào)用這個(gè)方法進(jìn)行實(shí)例信息移除     protected boolean internalCancel(String appName, String id, boolean isReplication) {
        try {
            // 讀鎖             read.lock();
            // 取消(下線)計(jì)數(shù)             CANCEL.increment(isReplication);
            // 獲取續(xù)約實(shí)例             Map<String, Lease<InstanceInfo>> gMap = registry.get(appName);
            Lease<InstanceInfo> leaseToCancel = null;
            if (gMap != null) {
                leaseToCancel = gMap.remove(id);
            }
            // 添加到近期取消隊(duì)列             recentCanceledQueue.add(new Pair<Long, String>(System.currentTimeMillis(), appName + "(" + id + ")"));
            // 移除實(shí)例狀態(tài)             InstanceStatus instanceStatus = overriddenInstanceStatusMap.remove(id);
            if (instanceStatus != null) {
                logger.debug("Removed instance id {} from the overridden map which has value {}", id, instanceStatus.name());
            }
            if (leaseToCancel == null) {
                // 續(xù)約實(shí)例不存在,返回false                 CANCEL_NOT_FOUND.increment(isReplication);
                logger.warn("DS: Registry: cancel failed because Lease is not registered for: {}/{}", appName, id);
                return false;
            } else {
                // 續(xù)約實(shí)例執(zhí)行取消操作:設(shè)置剔除時(shí)間                 leaseToCancel.cancel();
                InstanceInfo instanceInfo = leaseToCancel.getHolder();
                String vip = null;
                String svip = null;
                if (instanceInfo != null) {
                    // 設(shè)置實(shí)例行為為delete                     instanceInfo.setActionType(ActionType.DELETED);
                    // 添加到近期變化隊(duì)列                     recentlyChangedQueue.add(new RecentlyChangedItem(leaseToCancel));
                    instanceInfo.setLastUpdatedTimestamp();
                    vip = instanceInfo.getVIPAddress();
                    svip = instanceInfo.getSecureVipAddress();
                }
                // 使響應(yīng)緩存失效                 invalidateCache(appName, vip, svip);
                logger.info("Cancelled instance {}/{} (replication={})", appName, id, isReplication);
                return true;
            }
        } finally {
            // 釋放讀鎖             read.unlock();
        }
    } 

總結(jié)

Eureka client執(zhí)行DiscoveryManager調(diào)用了DiscveryClient的shutdown方法進(jìn)行服務(wù)的下線操作郎任,然后服務(wù)端接收到http delete請(qǐng)求之后進(jìn)行服務(wù)的相關(guān)下線操作姻檀,并且同步到集群中的其它節(jié)點(diǎn)。Eureka server則會(huì)將實(shí)例信息進(jìn)行剔除處理涝滴,并且添加到近期變化隊(duì)列和近期取消隊(duì)列里绣版。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市歼疮,隨后出現(xiàn)的幾起案子杂抽,更是在濱河造成了極大的恐慌,老刑警劉巖韩脏,帶你破解...
    沈念sama閱讀 211,194評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件缩麸,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡赡矢,警方通過(guò)查閱死者的電腦和手機(jī)杭朱,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,058評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)吹散,“玉大人弧械,你說(shuō)我怎么就攤上這事】彰瘢” “怎么了刃唐?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,780評(píng)論 0 346
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)界轩。 經(jīng)常有香客問(wèn)我画饥,道長(zhǎng),這世上最難降的妖魔是什么浊猾? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,388評(píng)論 1 283
  • 正文 為了忘掉前任抖甘,我火速辦了婚禮,結(jié)果婚禮上葫慎,老公的妹妹穿的比我還像新娘衔彻。我一直安慰自己薇宠,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,430評(píng)論 5 384
  • 文/花漫 我一把揭開(kāi)白布米奸。 她就那樣靜靜地躺著昼接,像睡著了一般爽篷。 火紅的嫁衣襯著肌膚如雪悴晰。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,764評(píng)論 1 290
  • 那天逐工,我揣著相機(jī)與錄音铡溪,去河邊找鬼。 笑死泪喊,一個(gè)胖子當(dāng)著我的面吹牛棕硫,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播袒啼,決...
    沈念sama閱讀 38,907評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼哈扮,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了蚓再?” 一聲冷哼從身側(cè)響起滑肉,我...
    開(kāi)封第一講書(shū)人閱讀 37,679評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎摘仅,沒(méi)想到半個(gè)月后靶庙,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,122評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡娃属,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,459評(píng)論 2 325
  • 正文 我和宋清朗相戀三年六荒,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片矾端。...
    茶點(diǎn)故事閱讀 38,605評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡掏击,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出秩铆,到底是詐尸還是另有隱情铐料,我是刑警寧澤,帶...
    沈念sama閱讀 34,270評(píng)論 4 329
  • 正文 年R本政府宣布豺旬,位于F島的核電站钠惩,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏族阅。R本人自食惡果不足惜篓跛,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,867評(píng)論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望坦刀。 院中可真熱鬧愧沟,春花似錦蔬咬、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,734評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至混坞,卻和暖如春狐援,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背究孕。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,961評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工啥酱, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人厨诸。 一個(gè)月前我還...
    沈念sama閱讀 46,297評(píng)論 2 360
  • 正文 我出身青樓镶殷,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親微酬。 傳聞我的和親對(duì)象是個(gè)殘疾皇子绘趋,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,472評(píng)論 2 348

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