在macOS 10.12上两个Docker容器之间的通信

11

我正在macOS 10.12上使用Docker 1.12.5,搭建一个开发环境。 我有一个应用程序镜像和一个共享的redis镜像,其中包含一些预配置变量。

即使按照一些教程(并阅读了关于Mac上docker0不可用的内容),我仍然无法连接这两个容器。

我使用以下命令启动我的redis镜像:

docker run -d -p 6379:6379 (IMAGE ID)

在我的redis镜像中,我拥有:

CONTAINER ID        IMAGE               COMMAND                  CREATED             STATUS              PORTS                    NAMES
dffb89854618        d59                 "docker-entrypoint.sh"   20 seconds ago      Up 19 seconds       0.0.0.0:6379->6379/tcp   drunk_williams

我可以从我的 Mac 上使用 redis-cli 命令成功连接,并且没有任何问题。

但是,当我启动一个简单的 ubuntu 镜像时,似乎无法连接到这个单独的 redis 镜像:

root@2d4eda315f4f:/# ifconfig
eth0      Link encap:Ethernet  HWaddr 02:42:ac:11:00:03
          inet addr:172.17.0.3  Bcast:0.0.0.0  Mask:255.255.0.0
          inet6 addr: fe80::42:acff:fe11:3/64 Scope:Link
          UP BROADCAST RUNNING MULTICAST  MTU:1500  Metric:1
          RX packets:20707 errors:0 dropped:0 overruns:0 frame:0
          TX packets:11515 errors:0 dropped:0 overruns:0 carrier:0
          collisions:0 txqueuelen:0
          RX bytes:28252929 (28.2 MB)  TX bytes:635848 (635.8 KB)

lo        Link encap:Local Loopback
          inet addr:127.0.0.1  Mask:255.0.0.0
          inet6 addr: ::1/128 Scope:Host
          UP LOOPBACK RUNNING  MTU:65536  Metric:1
          RX packets:12 errors:0 dropped:0 overruns:0 frame:0
          TX packets:12 errors:0 dropped:0 overruns:0 carrier:0
          collisions:0 txqueuelen:1
          RX bytes:680 (680.0 B)  TX bytes:680 (680.0 B)
root@2d4eda315f4f:/# telnet localhost 6379
Trying ::1...
Trying 127.0.0.1...
telnet: Unable to connect to remote host: Connection refused
root@2d4eda315f4f:/# telnet 172.17.0.3 6379
Trying 172.17.0.3...
telnet: Unable to connect to remote host: Connection refused

这是因为主机没有可用的docker0接口吗?在开发环境中,是否有一些简单的解决方法允许这些容器进行通信(在同一主机上运行)?

更新: 尝试使用命名容器,仍然无法连接。

docker run -d --name redis_server redis

结果为:

CONTAINER ID        IMAGE               COMMAND                  CREATED             STATUS              PORTS                    NAMES
5d05820aa985        redis               "docker-entrypoint.sh"   43 hours ago        Up 1 seconds        6379/tcp                 redis_server

但是如果我启动一个新的Ubuntu容器:

root@e92b47419bc4:/# redis-cli -h redis_server
Could not connect to Redis at redis_server:6379: Name or service not known

我不确定如何找到/连接第一个redis_server容器。

1个回答

19

每个容器都有自己的本地主机

每个服务都在自己的容器中运行。从Ubuntu容器的视角来看,Redis并没有在本地主机上进行监听。

使用Docker网络

为了让你的容器相互通信,它们应该在同一个Docker网络中。这包括以下三个步骤:

  1. 创建Docker网络
  2. 为你的容器命名
  3. 将你的容器连接到你创建的网络中

完成这些步骤后,容器可以使用它们的名称相互通信,就像它们是主机名一样。

有不止一种方法可以实现这个目标……我会在这个回答中介绍两种方法,但可能还有其他几种方法,我不熟悉(例如使用Kubernetes或Swarm)。

手动操作

你可以使用docker network命令为这个应用程序创建一个网络。

# Show the current list of networks
docker network ls

# Create a network for your app
docker network create my_redis_app

运行 Redis 容器时,请确保它有一个名称,并且在此网络上。如果您想将端口暴露到 macOS 上(使用 -p),则可以进行外部暴露,但这对于其他容器与 Redis 通信并不必要。

docker run -d -p 6379:6379 --name redis_server --network my_redis_app <IMAGE ID>

现在运行您的Ubuntu容器。如果您愿意,也可以为其命名,但在这个示例中我不会费心去命名,因为这个示例并没有运行任何服务。

docker run -it --network my_redis_app ubuntu bash

现在从Ubuntu容器内部,您应该能够通过名称redis_server访问Redis,就像它是一个DNS名称一样。

使用Compose完成

我倾向于使用Compose构建这样的设置,因为将其编写为YAML文件更容易(在我看来)。下面是上述内容以docker-compose.yml形式重写的示例:

version: '2'
services:
  redis:
    image: <IMAGE ID>
    networks:
      - my_redis_app
    ports: 6379:6379
  ubuntu:
    image: ubuntu:latest
    networks:
      - my_redis_app
networks:
  my_redis_app:
    driver: bridge

有了这个设置,您可以运行docker-compose up -d redis,并使用特定的Docker网络使您的redis服务在线。如果不存在,Compose将为您创建网络。

按照这种方式运行Ubuntu容器没有太多意义......它当然是交互式的。但我假设一旦您运行了Redis,您会添加某种应用程序容器,以及像nginx这样的Web代理... 将其他容器也放到services下面,这样就可以一起管理它们了。

由于ubuntu是交互式的,您可以交互式地运行它:

# without -d, container is run interactively
docker-compose run ubuntu bash

现在在Ubuntu中,您应该能够使用其名称连接到Redis,在此示例中名称为redis


1
这在 Mac 上也是这种情况吗?我尝试了上面的方法,但没有成功。(请参见问题的更新。) - Craig Otis
@CraigOtis 是的,没错。请看我的更新答案,我忘了提醒你还应该使用 --link。我倾向于使用 docker-compose 来处理这些事情,你可以将所有内容写入一个 YAML 文件中,而不是在命令行标志上摆弄...之前忘记了。 - Dan Lowe
好的,我一直避免使用 --link,因为它已经被废弃了:https://docs.docker.com/engine/userguide/networking/default_network/dockerlinks/鉴于这个配置是为一个新项目准备的,我想避免使用已经过时的工具。 :-) - Craig Otis
@CraigOtis 如果它们都在“default”网络上,那么它没有工作是很奇怪的。我本来以为应该可以工作...我对此没有一个很好的解释。 - Dan Lowe
这个方案在第三个容器上不能工作,有什么原因吗?我已经成功让两个容器进行通信(一种是主机和代理的情况),但是我无法从第三台机器上使用curl命令连接到“代理”容器。 - Russell
显示剩余4条评论

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接