使用容器名称从主机访问 docker 容器

新手上路,请多包涵

我正在开发一项服务并使用 docker compose 来旋转诸如 postgres、redis、elasticsearch 之类的服务。我有一个基于 RubyOnRails 的 Web 应用程序,可以读写所有这些服务。

这是我的 docker-compose.yml

 version: '2'

services:
  redis:
    image: redis:2.8
    networks:
      - frontapp

  elasticsearch:
    image: elasticsearch:2.2
    networks:
      - frontapp

  postgres:
    image: postgres:9.5
    environment:
      POSTGRES_USER: elephant
      POSTGRES_PASSWORD: smarty_pants
      POSTGRES_DB: elephant
    volumes:
      - /var/lib/postgresql/data
    networks:
      - frontapp

networks:
  frontapp:
    driver: bridge

我可以在这个网络中 ping 容器

$ docker-compose run redis /bin/bash
root@777501e06c03:/data# ping postgres
PING postgres (172.20.0.2): 56 data bytes
64 bytes from 172.20.0.2: icmp_seq=0 ttl=64 time=0.346 ms
64 bytes from 172.20.0.2: icmp_seq=1 ttl=64 time=0.047 ms
...

到目前为止,一切都很好。现在我想在我的主机上运行 ruby on rails 应用程序,但能够使用 postgresql://username:password@postgres/database 类的 url 访问 postgres 实例,目前这是不可能的

$ ping postgres
ping: unknown host postgres

我可以在 docker 中看到我的网络

$ docker network ls
NETWORK ID          NAME                DRIVER
ac394b85ce09        bridge              bridge
0189d7e86b33        elephant_default    bridge
7e00c70bde3b        elephant_frontapp   bridge
a648554a72fa        host                host
4ad9f0f41b36        none                null

我可以看到它的接口

$ ifconfig
br-0189d7e86b33 Link encap:Ethernet  HWaddr 02:42:76:72:bb:c2
          inet addr:172.18.0.1  Bcast:0.0.0.0  Mask:255.255.0.0
          inet6 addr: fe80::42:76ff:fe72:bbc2/64 Scope:Link
          UP BROADCAST MULTICAST  MTU:1500  Metric:1
          RX packets:36 errors:0 dropped:0 overruns:0 frame:0
          TX packets:60 errors:0 dropped:0 overruns:0 carrier:0
          collisions:0 txqueuelen:0
          RX bytes:2000 (2.0 KB)  TX bytes:8792 (8.7 KB)

br-7e00c70bde3b Link encap:Ethernet  HWaddr 02:42:e7:d1:fe:29
          inet addr:172.20.0.1  Bcast:0.0.0.0  Mask:255.255.0.0
          inet6 addr: fe80::42:e7ff:fed1:fe29/64 Scope:Link
          UP BROADCAST RUNNING MULTICAST  MTU:1500  Metric:1
          RX packets:1584 errors:0 dropped:0 overruns:0 frame:0
          TX packets:1597 errors:0 dropped:0 overruns:0 carrier:0
          collisions:0 txqueuelen:0
          RX bytes:407137 (407.1 KB)  TX bytes:292299 (292.2 KB)
...

但我不确定下一步该怎么做。我试着玩了一下 /etc/resolv.conf ,主要是 nameserver 指令,但这没有效果。

对于如何正确配置此设置的建议,我将不胜感激。

更新

浏览 Internet 资源后,我设法为盒子分配了静态 IP 地址。就目前而言,这足以让我继续开发。这是我目前的 docker-compose.yml

 version: '2'

services:
  redis:
    image: redis:2.8
    networks:
      frontapp:
        ipv4_address: 172.25.0.11

  elasticsearch:
    image: elasticsearch:2.2
    networks:
      frontapp:
        ipv4_address: 172.25.0.12

  postgres:
    image: postgres:9.5
    environment:
      POSTGRES_USER: elephant
      POSTGRES_PASSWORD: smarty_pants
      POSTGRES_DB: elephant
    volumes:
      - /var/lib/postgresql/data
    networks:
      frontapp:
        ipv4_address: 172.25.0.10

networks:
  frontapp:
    driver: bridge
    ipam:
      driver: default
      config:
        - subnet: 172.25.0.0/16
          gateway: 172.25.0.1

原文由 Max 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 481
2 个回答

我正在使用 bash 脚本来更新 /etc/hosts 。为什么这个解决方案?

  • 简短的脚本,易于审查(不想让一些未经审查的具有大量依赖项的应用程序访问 Docker 套接字(这意味着 root 访问))
  • 它使用 docker events 每次启动或停止容器时运行(此处发布的其他解决方案每秒循环运行,效率较低)
  • 更新 /etc/hosts ,不需要单独的 DNS 服务器。
  • Only dependencies are bash , mktemp , grep , xargs , sed , jqdocker ,我已经安装了所有这些。

只需将脚本放在某处,例如 /usr/local/bin/docker-update-hosts

 #!/usr/bin/env bash
set -e -u -o pipefail

hosts_file=/etc/hosts
begin_block="# BEGIN DOCKER CONTAINERS"
end_block="# END DOCKER CONTAINERS"

if ! grep -Fxq "$begin_block" "$hosts_file"; then
    echo -e "\n${begin_block}\n${end_block}\n" >> "$hosts_file"
fi

(echo "| container start |" && docker events) | \
while read event; do
    if [[ "$event" == *" container start "* ]] || [[ "$event" == *" network disconnect "* ]]; then
        hosts_file_tmp="$(mktemp)"
        docker container ls -q | xargs -r docker container inspect | \
        jq -r '.[]|"\(.NetworkSettings.Networks[].IPAddress|select(length > 0) // "# no ip address:") \(.Name|sub("^/"; "")|sub("_1$"; ""))"' | \
        sed -ne "/^${begin_block}$/ {p; r /dev/stdin" -e ":a; n; /^${end_block}$/ {p; b}; ba}; p" "$hosts_file" \
        > "$hosts_file_tmp"
        chmod 644 "$hosts_file_tmp"
        mv "$hosts_file_tmp" "$hosts_file"
    fi
done

注意:该脚本会从容器名称中删除由 docker-compose 添加的 _1 后缀。如果您不想这样做,只需从脚本中删除 |sub("_1$"; "")

您可以使用 systemd 服务与 Docker 同步运行: /etc/systemd/system/docker-update-hosts.service

 [Unit]
Description=Update Docker containers in /etc/hosts
Requires=docker.service
After=docker.service
PartOf=docker.service

[Service]
ExecStart=/usr/local/bin/docker-update-hosts

[Install]
WantedBy=docker.service

要激活,请运行:

 systemctl daemon-reload
systemctl enable docker-update-hosts.service
systemctl start docker-update-hosts.service

原文由 fnkr 发布,翻译遵循 CC BY-SA 4.0 许可协议

有一个开源应用程序可以解决这个问题,它被称为 DNS 代理服务器这里有一些 来自官方存储库的示例

它是一个解析容器主机名的 DNS 服务器,如果找不到匹配的主机名,那么也可以从互联网上解决它

启动 DNS 服务器

$ docker run --hostname dns.mageddo --restart=unless-stopped -p 5380:5380 \
-v /var/run/docker.sock:/var/run/docker.sock \
-v /etc/resolv.conf:/etc/resolv.conf \
defreitas/dns-proxy-server

它将自动设置为您的默认 DNS(并在停止时恢复为原始 DNS)

创建一些容器进行测试

检查 docker-compose 文件

$ cat docker-compose.yml
version: '3'
services:
  nginx-1:
    image: nginx
    hostname: nginx-1.docker
    network_mode: bridge
  linux-1:
    image: alpine
    hostname: linux-1.docker
    command: sh -c 'apk add --update bind-tools && tail -f /dev/null'
    network_mode: bridge # that way he can solve others containers names even inside, solve nginx-2, for example

启动容器

$ docker-compose up

解决容器

来自主机

nslookup nginx-1.docker
Server:     13.0.0.5
Address:    13.0.0.5#53
Non-authoritative answer:
Name:   nginx-1.docker
Address: 13.0.0.6

从另一个容器

$ docker-compose exec linux-1 ping nginx-1.docker
PING nginx-1.docker (13.0.0.6): 56 data bytes
64 bytes from 13.0.0.6: seq=0 ttl=64 time=0.034 ms

它还解决了互联网主机名

$ nslookup google.com
Server:     13.0.0.5
Address:    13.0.0.5#53

Non-authoritative answer:
Name:   google.com
Address: 216.58.202.78

原文由 deFreitas 发布,翻译遵循 CC BY-SA 4.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题