1. ホーム
  2. docker

[解決済み] ホストからコンテナ名でDockerコンテナにアクセスする

2022-10-08 16:08:02

質問

私はサービスを開発しており、postgres、redis、elasticsearchのようなサービスを回すために、そこでdocker composeを使用しています。私は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アプリケーションを実行したいのですが、次のようなURLでpostgresインスタンスにアクセスできるようにしたいです。 postgresql://username:password@postgres/database 現在、それは不可能です。

$ 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 ディレクティブを使用しましたが、何の効果もありませんでした。

私は、この設定を正しく構成する方法の提案の助けを感謝します。

アップデイト

インターネットのリソースを見て回った結果、ボックスに固定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

どのように解決するのですか?

私はbashスクリプトで /etc/hosts . なぜこのような解決策をとるのでしょうか?

  • 短いスクリプト、レビューが容易 (多くの依存関係を持つ未レビューのアプリケーションに Docker ソケットへのアクセス (ルートアクセスを意味します) を与えたくなかった)
  • このスクリプトでは docker events を使用して、コンテナが起動または停止するたびに実行します (ここに投稿された他のソリューションはループで毎秒実行しますが、これは非常に効率的ではありません)。
  • 更新 /etc/hosts を行うので、別途 DNS サーバーは必要ありません。
  • 依存するのは 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

注意: このスクリプトは _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