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

阅读对象

假设阅读者了解 docker,docker-compose以及 go 的语法

问题描述

我有三个应用分别叫做mysql,goApp,javaApp。 他们的依赖关系如下图所示:

image.png
  • goApp 通过调用 javaApp 的服务完成逻辑。

  • javaApp 直接和 mysql 数据库打交道。

为了让他们三个很容易的在 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"  

这个结构很顺畅的跑起来了,而且一直很稳定。突然有一天,我给我的服务器修改了一个 hostname 修改为 app.crop.cn,然后就跑不起来了。具体的现象如下:

  • javaApp 的程序能够正常访问。能够访问到mysql 数据库。

  • goApp 无法通过 javaApp 访问到 javaApp 这个应用。

  • 进入 javaApp和 goApp 这个容器,能够互相 ping 通。

问题分析

现象中有两个关键点:

  • 在容器里,所有 ping 都是能互通的。说明 docker 内部的 dns 是能工作的。

  • 只有 go 的程序不能根据 hostname 找到对应的容器。

根据第二疑点猜测: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 自己实现了一套请求 dns 解析的方法。其他程序应该使用基础的c 标准库getaddrinfo. 所以他们的表现不一样。还说啥啊,去看代码吧。在dnsconfig_unix.go 文件中找到了实现。


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 这个参数设定成 “.”就可以了。

最后修改 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/

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

推荐阅读更多精彩内容

  • Docker — 云时代的程序分发方式 要说最近一年云计算业界有什么大事件?Google Compute Engi...
    ahohoho阅读 15,508评论 15 147
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,596评论 18 139
  • Docker从2013年发布第一个版本以来,已经火遍全球,技术迭代也比较频繁,其周边产品和技术也越来越丰富。Doc...
    归海听雪阅读 12,258评论 7 44
  • 一、Docker 简介 Docker 两个主要部件:Docker: 开源的容器虚拟化平台Docker Hub: 用...
    R_X阅读 4,377评论 0 27
  • 萤火虫找朋友,它带着小灯笼去找朋友。萤火虫飞啊飞,看见了一只小蚂蚁。萤火虫要跟它交朋友,小蚂蚁说:“我迷路了,你...
    启迪源教育阅读 792评论 1 0