博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Docker Compose 容器连接
阅读量:6120 次
发布时间:2019-06-21

本文共 3507 字,大约阅读时间需要 11 分钟。

hot3.png

在Docker中,容器之间的链接是一种很常见的操作:它提供了访问其中的某个容器的网络服务而不需要将所需的端口暴露给Docker Host主机的功能。Docker Compose中对该特性的支持同样是很方便的。然而,如果需要链接的容器没有定义在同一个docker-compose.yml中的时候,这个时候就稍微麻烦复杂了点。

在不使用Docker Compose的时候,将两个容器链接起来使用—link参数,相对来说比较简单,以nginx镜像为例子:

12
docker run --rm --name test1 -d nginx  #开启一个实例test1docker run --rm --name test2 --link test1 -d nginx #开启一个实例test2并与test1建立链接

这样,test2test1便建立了链接,就可以在test2中使用访问test1中的服务了。

如果使用Docker Compose,那么这个事情就更简单了,还是以上面的nginx镜像为例子,编辑docker-compose.yml文件为:

12345678910
version: "3"services:  test2:    image: nginx    depends_on:      - test1    links:      - test1  test1:    image: nginx

最终效果与使用普通的Docker命令docker run xxxx建立的链接并无区别。这只是一种最为理想的情况。


  1. 如果容器没有定义在同一个docker-compose.yml文件中,应该如何链接它们呢?
  2. 又如果定义在docker-compose.yml文件中的容器需要与docker run xxx启动的容器链接,需要如何处理?

针对这两种典型的情况,下面给出我个人测试可行的办法:


方式一:让需要链接的容器同属一个外部网络

我们还是使用nginx镜像来模拟这样的一个情景:假设我们需要将两个使用Docker Compose管理的nignx容器(test1test2)链接起来,使得test2能够访问test1中提供的服务,这里我们以能ping通为准。

首先,我们定义容器test1docker-compose.yml文件内容为:

1234567891011
version: "3"services:  test2:    image: nginx    container_name: test1    networks:      - default      - app_netnetworks:  app_net:    external: true

容器test2内容与test1基本一样,只是多了一个external_links,需要特别说明的是:最近发布的Docker版本已经不需要使用external_links来链接容器,容器的DNS服务可以正确的作出判断,因此如果你你需要兼容较老版本的Docker的话,那么容器test2docker-compose.yml文件内容为:

12345678910111213
version: "3"services:  test2:    image: nginx    networks:      - default      - app_net    external_links:      - test1    container_name: test2networks:  app_net:    external: true

否则的话,test2docker-compose.ymltest1的定义完全一致,不需要额外多指定一个external_links。相关的问题请参见stackoverflow上的相关问题:

正如你看到的那样,这里两个容器的定义里都使用了同一个外部网络app_net,因此,我们需要在启动这两个容器之前通过以下命令再创建外部网络:

1
docker network create app_net

之后,通过docker-compose up -d命令启动这两个容器,然后执行docker exec -it test2 ping test1,你将会看到如下的输出:

12345678
docker exec -it test2 ping test1PING test1 (172.18.0.2): 56 data bytes64 bytes from 172.18.0.2: icmp_seq=0 ttl=64 time=0.091 ms64 bytes from 172.18.0.2: icmp_seq=1 ttl=64 time=0.146 ms64 bytes from 172.18.0.2: icmp_seq=2 ttl=64 time=0.150 ms64 bytes from 172.18.0.2: icmp_seq=3 ttl=64 time=0.145 ms64 bytes from 172.18.0.2: icmp_seq=4 ttl=64 time=0.126 ms64 bytes from 172.18.0.2: icmp_seq=5 ttl=64 time=0.147 ms

证明这两个容器是成功链接了,反过来在test1中pingtest2也是能够正常ping通的。

如果我们通过docker run --rm --name test3 -d nginx这种方式来先启动了一个容器(test3)并且没有指定它所属的外部网络,而需要将其与test1或者test2链接的话,这个时候手动链接外部网络即可:

1
docker network connect app_net test3

这样,三个容器都可以相互访问了。


方式二:更改需要链接的容器的网络模式

通过更改你想要相互链接的容器的网络模式为bridge,并指定需要链接的外部容器(external_links)即可。与同属外部网络的容器可以相互访问的链接方式一不同,这种方式的访问是单向的。

还是以nginx容器镜像为例子,如果容器实例nginx1需要访问容器实例nginx2,那么nginx2doker-compose.yml定义为:

123456
version: "3"services:  nginx2:    image: nginx    container_name: nginx2    network_mode: bridge

与其对应的,nginx1docker-compose.yml定义为:

12345678
version: "3"services:  nginx1:    image: nginx    external_links:      - nginx2    container_name: nginx1    network_mode: bridge

需要特别说明的是,这里的external_links是不能省略的,而且nginx1的启动必须要在nginx2之后,否则可能会报找不到容器nginx2的错误。

接着我们使用ping来测试下连通性:

12345678
$ docker exec -it nginx1 ping nginx2  # nginx1 to nginx2PING nginx2 (172.17.0.4): 56 data bytes64 bytes from 172.17.0.4: icmp_seq=0 ttl=64 time=0.141 ms64 bytes from 172.17.0.4: icmp_seq=1 ttl=64 time=0.139 ms64 bytes from 172.17.0.4: icmp_seq=2 ttl=64 time=0.145 ms$ docker exec -it nginx2 ping nginx1 #nginx2 to nginx1ping: unknown host

以上也能充分证明这种方式是属于单向联通的。

在实际应用中根据自己的需要灵活的选择这两种链接方式,如果想偷懒的话,大可选择第二种。不过我更推荐第一种,不难看出无论是联通性还是灵活性,较为更改网络模式的第二种都更为友好。

转载于:https://my.oschina.net/10000000000/blog/3059609

你可能感兴趣的文章
Web语音处理 - Web Audio API & WebRTC
查看>>
org.tinygroup.validatecomponent-流程校验组件
查看>>
Linux磁盘管理和文件系统管理
查看>>
我的友情链接
查看>>
Jumpserver堡垒机安装配置全过程(二)-配置
查看>>
iOS layer
查看>>
我的友情链接
查看>>
第一次自己写存储过程去进行设备录入——存做纪念
查看>>
我的友情链接
查看>>
PTR的故事
查看>>
Exchange2013灾难恢复演练--Exchange管理员必须掌握的技能
查看>>
本地ASP.NET开发页面使用AzureAD(AAD)验证登录
查看>>
5年了..
查看>>
使用域组策略/脚本统一配置防火墙
查看>>
我的友情链接
查看>>
开源网络备份软件bacula安装配置(一)
查看>>
二进制相关
查看>>
FreeBSD防火墙浅解
查看>>
OpenStack网络模块核心OpenvSwitch的全面解读
查看>>
命令screen
查看>>