Dubbo 2.7.1 踩坑記

Dubbo 2.7.1 踩坑記

Dubbo 2.7 版本增長新特性,新系統開始使用 Dubbo 2.7.1 嚐鮮新功能。使用過程當中不慎踩到這個版本的 Bug。git

系統架構

Spring Boot 2.14-Release + Dubbo 2.7.1github

現象

Dubbo 服務者啓動成功,正常提供服務,消費者調用偶現失敗的狀況。錯誤以下圖:面試

15583219929142ef5dbd0806c47d988aa1ad24920b1b6.png

能夠看出,主要緣由爲 cause: message can not send, because channel is closed。 可是檢查提供者,卻發現服務進程正常。apache

登錄 Dubbo admin 查看提供者服務,發現這個服務存在兩個節點。服務器

1558321992969f7ec73433dfc4c35a79484edb908d748.png

192.168.164.77 爲測試服務器的 ip,提供者位於這臺機器,而另外一個 10.20.80.67 倒是本地電腦的 IP,可是此時本地並未運行這個服務。架構

再次查看服務報錯的緣由,能夠看到提供者調用l本地提供 RPC 的服務。因爲本地服務已中止,致使調用失敗。app

這個問題在以前版本從未碰到,剛開始隱約記得 Dubbo 服務提供者註冊使用 ZooKeeper 臨時節點,服務斷開,會刪除該節點。ide

問題緣由

在 Dubbo 主頁搜索相關 issue,看到一樣的問題 Dubbo-2.7.1 providers 重複註冊.源碼分析

查看相關回復,能夠看到問題主要因爲 dynamic 默認值變成 false ,而 2.7.1 以前版本默認不賦值,初始值爲 null。測試

15583219929464f88d2277240422a93e80bf95ff69ec1.png

後續 PR 中已修復該問題 Fix issue 3785,修復代碼將 dynamic 默認設置成 true。可是截止 20190515 該版本暫未發佈。

源碼分析

知道問題緣由,這裏咱們從源碼分析一下,爲何 dynamic 設置成 false 會致使該問題。

注:下面分析的是 Dubbo 2.7.1 的源碼

下面咱們使用 Dubbo xml 配置相關。

在 xml 配置中,能夠在如下兩個地方設置 dynamic 屬性。

15583219930733ceca2a340fd49ce96bd3c9574ae065b.png

服務啓動時將會使用 DubboNamespaceHandler 解析,注入 Spring 容器。

15583219929464afc4fc277e84dcfbf4c75adb2b97df3.png

其中會將 provider 標籤解析成 ProviderConfig 對象,service 標籤解析成 ServiceBean 對象。

155832199300620acb303fee84d2ab16131920cacc700.png

查看繼承關係,能夠看到以上兩個類都繼承 AbstractServiceConfig , dynamic 位於這個父對象中。

15583219930294e661481d55741d6a31110f373002ef1.png

能夠看到該字段默認值爲 false

接着查看 Dubbo 服務導出過程,位於 ServiceBean#export,略過其餘代碼,咱們直接跳到關鍵 ServiceConfig#doExportUrlsFor1Protocol

15583219930461ed4b203ed234f9092fe71e784a4ea85.png

能夠看到這裏調用了屢次 appendParameters 方法。 這個方法將利用反射,獲取對象的中全部字段信息,而後添加到 map 中。其中字段名字爲鍵值,字段實際值爲內容。此時 map 鍵值內容爲:

1558321993050358b0a09c89549f194de00ec97c25787.png

能夠看到 map 中還有一個 default.dynamic,你們翻看代碼本身思考一下,爲何會出現這個?

接着咱們跳到後面:

1558321993054a18c4fbfc9984685a8daa45c011a5413.png

在這裏會將上面獲得 map 組裝到 URL 對象中,而後再註冊到註冊中心。。

因爲註冊中心使用的是 ZooKeeper,因此這裏將會使用 ZookeeperRegistry 實現類。

1558321993020fa3f1b4cd5cc4566acb835e8c2bd14f4.png

首先查看 url##getParameter 方法,這裏 Constants.DYNAMIC_KEY 值爲 dynamic。

15583219930714b971def40924ad2891aa5eb33d04f02.png

該方法會先從 parameters 中根據鍵值取值。若不存在,會再根據 default 做爲前綴拼接再次取值。若還不存在則使用傳入的默認值。

查看此時的 parameters 對象。

155832199307775de17130ae44fe5bcf445ae93d57de9.png

url.getParameter(Constants.DYNAMIC_KEY, true) 返回爲 false。

而後分析 zkClient#create 方法,

15583219930223ed0dc3cf43b4191a26877d11056e84d.png

因爲 ephemeral 爲 false,因此這個服務註冊到 ZooKeeper 的節點爲持久節點。

臨時節點,客戶端斷開,會話超時後,ZooKeeper 將會自動刪除這個節點。zookeeper-faq

面試題:服務提供者能實現失效踢出是什麼原理(高頻題)

服務宕機的時候,該節點因爲是持久節點會永遠存在,並且當服務再次重啓的時候會將從新註冊一個新節點。這樣就致使 ZooKeeper 中存在額外失效的節點,且該節點還沒法天然消除(除非手動調用 ZooKeeper 刪除節點方法)。

總結

因爲 Dubbo 2.7.2 暫未發佈,因此建議若想使用 Dubbo 2.7 新功能的同窗,使用 2.7.0 版本。若如今正在使用 2.7.2 版本,也不要慌張。只要服務不是異常宕機或者使用 kill -9 強制殺死進程,以上的現象將不會碰到。正常服務關閉的時候,Dubbo 服務會主動去 ZooKeeper 註銷該服務,並刪除這個節點。

還未使用該版本的同窗們,建議使用 2.7.0 或者等 2.7.2 發佈之後,再使用。

相關文章
相關標籤/搜索