golang 程序在 docker 無法找到其他容器

閱讀對象

假設閱讀者了解 docker,docker-compose以及 go 的語法

問題描述

我有三個應用分別叫做mysql逢并,goApp,javaApp。 他們的依賴關系如下圖所示:

image.png
  • goApp 通過調用 javaApp 的服務完成邏輯立肘。

  • javaApp 直接和 mysql 數(shù)據(jù)庫打交道。

為了讓他們三個很容易的在 docker 容器里跑起來我使用了 docker-compose名扛。具體的配置文件如下:


version: '2'
services:
  mysql:
    container_name: mysql
    image: mysql:5.7
    restart: always
    hostname: mysql
    environment:
       MYSQL_DATABASE: ${MYSQL_DATABASE}
       MYSQL_USER: ${MYSQL_USER}
       MYSQL_PASSWORD: ${MYSQL_PASSWORD}
       MYSQL_ROOT_PASSWORD: ${MYSQL_ROOT_PASSWORD}
  goApp:
    container_name: goApp
    image: goApp:0.1
    hostname: goApp
    ports:
      - "8080:8080"    
javaApp:
    container_name: javaApp
    image: javaApp:0.1
    hostname: javaApp
    ports:
      - "6031:6031"  

這個結構很順暢的跑起來了谅年,而且一直很穩(wěn)定。突然有一天肮韧,我給我的服務器修改了一個 hostname 修改為 app.crop.cn融蹂,然后就跑不起來了旺订。具體的現(xiàn)象如下:

  • javaApp 的程序能夠正常訪問。能夠訪問到mysql 數(shù)據(jù)庫超燃。

  • goApp 無法通過 javaApp 訪問到 javaApp 這個應用区拳。

  • 進入 javaApp和 goApp 這個容器,能夠互相 ping 通意乓。

問題分析

現(xiàn)象中有兩個關鍵點:

  • 在容器里樱调,所有 ping 都是能互通的。說明 docker 內部的 dns 是能工作的届良。

  • 只有 go 的程序不能根據(jù) hostname 找到對應的容器笆凌。

根據(jù)第二疑點猜測:go 語言的 dns 解析機制和java 的不一樣。沿著這條路我我找到了士葫,最后在官方文檔中找到如下內容乞而。

On Unix systems, the resolver has two options for resolving names. 
It can use a pure Go resolver that sends DNS requests directly to
 the servers listed in /etc/resolv.conf, or it can use a cgo-based 
resolver that calls C library routines such as getaddrinfo and 
getnameinfo.

也就是說 go 自己實現(xiàn)了一套請求 dns 解析的方法。其他程序應該使用基礎的c 標準庫getaddrinfo. 所以他們的表現(xiàn)不一樣慢显。還說啥啊爪模,去看代碼吧。在dnsconfig_unix.go 文件中找到了實現(xiàn)荚藻。


func dnsDefaultSearch() []string {
    hn, err := getHostname()
    if err != nil {
        // best effort
        return nil
    }
    if i := byteIndex(hn, '.'); i >= 0 && i < len(hn)-1 {
        return []string{ensureRooted(hn[i+1:])}
    }
    return nil
}

func ensureRooted(s string) string {
    if len(s) > 0 && s[len(s)-1] == '.' {
        return s
    }
    return s + "."
}


通過代碼可以看出屋灌,默認的 Search 是獲取的當前主機的 hostname,第一個點(“.”) 后面部分的內容应狱。例如 我上面把主機名修改成了 app.crop.cn, 那么他獲取到的默認的 search 就是 crop.cn. 了声滥。所以當我請求 javaApp 的時候,他會像 docker 內建的 dns 請求 javaApp.crop.cn 這個域名侦香。當然就請求不到了落塑。

問題解決方案

找到原因以后下面就比較簡單了。只需要在啟動 容器的是時候設定上 把dns-search 這個參數(shù)設定成 “.”就可以了罐韩。

最后修改 docker-compose.yml 為:


version: '2'
services:
  mysql:
    container_name: mysql
    image: mysql:5.7
    restart: always
    hostname: mysql
    environment:
       MYSQL_DATABASE: ${MYSQL_DATABASE}
       MYSQL_USER: ${MYSQL_USER}
       MYSQL_PASSWORD: ${MYSQL_PASSWORD}
       MYSQL_ROOT_PASSWORD: ${MYSQL_ROOT_PASSWORD}
  goApp:
    container_name: goApp
    image: goApp:0.1
    dns-search: .
    hostname: goApp
    ports:
      - "8080:8080"    
    javaApp:
    container_name: javaApp
    image: javaApp:0.1
    hostname: javaApp
    ports:
      - "6031:6031" 

就這樣憾赁,這個看上去奇葩的問題被搞定了。

參考資料

  1. NameResolver - glibc wiki

  2. https://golang.org/pkg/net/#hdr-Name_Resolution

  3. https://docs.docker.com/engine/userguide/networking/default_network/configure-dns/

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末散吵,一起剝皮案震驚了整個濱河市龙考,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌矾睦,老刑警劉巖晦款,帶你破解...
    沈念sama閱讀 218,204評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異枚冗,居然都是意外死亡缓溅,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,091評論 3 395
  • 文/潘曉璐 我一進店門赁温,熙熙樓的掌柜王于貴愁眉苦臉地迎上來坛怪,“玉大人淤齐,你說我怎么就攤上這事⊥嗄洌” “怎么了更啄?”我有些...
    開封第一講書人閱讀 164,548評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長居灯。 經常有香客問我祭务,道長,這世上最難降的妖魔是什么怪嫌? 我笑而不...
    開封第一講書人閱讀 58,657評論 1 293
  • 正文 為了忘掉前任待牵,我火速辦了婚禮,結果婚禮上喇勋,老公的妹妹穿的比我還像新娘。我一直安慰自己偎行,他們只是感情好川背,可當我...
    茶點故事閱讀 67,689評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著蛤袒,像睡著了一般熄云。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上妙真,一...
    開封第一講書人閱讀 51,554評論 1 305
  • 那天缴允,我揣著相機與錄音,去河邊找鬼珍德。 笑死练般,一個胖子當著我的面吹牛,可吹牛的內容都是我干的锈候。 我是一名探鬼主播薄料,決...
    沈念sama閱讀 40,302評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼泵琳!你這毒婦竟也來了摄职?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 39,216評論 0 276
  • 序言:老撾萬榮一對情侶失蹤获列,失蹤者是張志新(化名)和其女友劉穎谷市,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體击孩,經...
    沈念sama閱讀 45,661評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡迫悠,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,851評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了巩梢。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片及皂。...
    茶點故事閱讀 39,977評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡甫男,死狀恐怖,靈堂內的尸體忽然破棺而出验烧,到底是詐尸還是另有隱情板驳,我是刑警寧澤,帶...
    沈念sama閱讀 35,697評論 5 347
  • 正文 年R本政府宣布碍拆,位于F島的核電站若治,受9級特大地震影響,放射性物質發(fā)生泄漏感混。R本人自食惡果不足惜端幼,卻給世界環(huán)境...
    茶點故事閱讀 41,306評論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望弧满。 院中可真熱鬧婆跑,春花似錦、人聲如沸庭呜。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,898評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽募谎。三九已至扶关,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間数冬,已是汗流浹背节槐。 一陣腳步聲響...
    開封第一講書人閱讀 33,019評論 1 270
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留拐纱,地道東北人铜异。 一個月前我還...
    沈念sama閱讀 48,138評論 3 370
  • 正文 我出身青樓,卻偏偏與公主長得像秸架,于是被迫代替她去往敵國和親熙掺。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,927評論 2 355

推薦閱讀更多精彩內容

  • Docker — 云時代的程序分發(fā)方式 要說最近一年云計算業(yè)界有什么大事件咕宿?Google Compute Engi...
    ahohoho閱讀 15,532評論 15 147
  • Spring Cloud為開發(fā)人員提供了快速構建分布式系統(tǒng)中一些常見模式的工具(例如配置管理币绩,服務發(fā)現(xiàn),斷路器府阀,智...
    卡卡羅2017閱讀 134,657評論 18 139
  • Docker從2013年發(fā)布第一個版本以來缆镣,已經火遍全球,技術迭代也比較頻繁试浙,其周邊產品和技術也越來越豐富董瞻。Doc...
    歸海聽雪閱讀 12,284評論 7 44
  • 一、Docker 簡介 Docker 兩個主要部件:Docker: 開源的容器虛擬化平臺Docker Hub: 用...
    R_X閱讀 4,388評論 0 27
  • 螢火蟲找朋友,它帶著小燈籠去找朋友钠糊。螢火蟲飛啊飛挟秤,看見了一只小螞蟻。螢火蟲要跟它交朋友抄伍,小螞蟻說:“我迷路了艘刚,你...
    啟迪源教育閱讀 803評論 1 0