Docker多容器連線-以Nginx+PHP為例

管宜堯發表於2015-05-31

Docker提供了多個容器直接訪問的方法,最簡單的方式是直接使用埠對映-p引數指定對映的埠或者-P對映所有埠,多個容器直接通過網路埠進行訪問。

但網路埠對映方式並不是Docker中連線多個容器的唯一方式,還可以使用Docker的連線系統(--link)連線多個容器,當容器連線到一起時,接受者容器就可以看到源容器的資訊。

建立容器之間的連線 - 以Nginx+PHP為例

在容器直接建立連線要使用--link選項

--link <name or id>:alias

這裡我們通過建立一個 nginx/php-fpm 的服務,示例一下如何在兩個或者多個容器之間建立連線。

要建立容器連線的話,就要依賴容器的名字了,使用--name指定源容器的名字為phpfpm

docker run --name phpfpm -d -v /Users/mylxsw/codes/php:/app php:5.6-fpm

接下來建立nginx容器,並且連線到phpfpm容器上去

docker run --name nginx_server -d -p 80:80 --link phpfpm:phpfpm -v /Users/mylxsw/Dockers/php/nginx.conf:/etc/nginx/nginx.conf --volumes-from phpfpm  nginx

這裡通過--link選項指定了要連線的容器是phpfpm,並且使用--volumes-from phpfpm將phpfpm容器掛載的卷也掛載到了nginx容器上,另外,這裡使用自定義的nginx配置檔案(nginx.conf)覆蓋了原先的配置,新的 nginx.conf 內容如下:

...
root   /app; # 這裡設定了專案掛載的容器的根目錄

location ~ \.php$ {
    fastcgi_pass   phpfpm:9000;# phpfpm訪問地址
...

需要注意的是,在該配置檔案中設定了伺服器的根目錄(root)為/app目錄,也就是我們掛載的目錄,另外是phpfpm的配置,我們將fastcgi_pass的值從127.0.0.1:9000改為了phpfpm:9000,這裡的phpfpm是域名,在nginx容器的/etc/hosts檔案中自動配置為phpfpm容器的訪問IP。

容器互通訊息

建立兩個容器之間的連線之後,在接收容器(Recipient)中必然會需要訪問源容器(Source)的資源,我們在為容器建立連線時,源容器在建立時並沒有使用-p/-P指定要暴露出來的埠,因此如何訪問源容器的資訊呢?

為了可以讓接收容器能夠訪問源容器的資訊,Docker提供了兩種方式:

  • 環境變數
  • /etc/hosts檔案

環境變數

Docker在連線容器的時候,會根據--link提供的引數自動的在接收者容器中建立一些環境變數,包括源容器的Dockerfile中使用ENV命令設定的環境變數和源容器啟動時(docker run),使用-e或者--env--env-file引數指定的環境變數。

主要包含以下環境變數,這裡假設alias=webdb

<alias>_NAME
<name>_PORT_<port>_<protocol>
<prefix>_ADDR
<prefix>_PORT
<prefix>_PROTO

例如:

$ docker run  -i -t --rm --link phpfpm:php php:5.6-fpm env
PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin
HOSTNAME=e5973c0d639f
TERM=xterm
PHP_PORT=tcp://172.17.0.74:9000
PHP_PORT_9000_TCP=tcp://172.17.0.74:9000
PHP_PORT_9000_TCP_ADDR=172.17.0.74
PHP_PORT_9000_TCP_PORT=9000
PHP_PORT_9000_TCP_PROTO=tcp
PHP_NAME=/tender_banach/php
PHP_ENV_PHP_INI_DIR=/usr/local/etc/php
PHP_ENV_GPG_KEYS=6E4F6AB321FDC07F2C332E3AC2BF0BC433CFC8B3 0BD78B5F97500D450838F95DFE857D9A90D90EC1
PHP_ENV_PHP_VERSION=5.6.9
PHP_INI_DIR=/usr/local/etc/php
PHP_EXTRA_CONFIGURE_ARGS=--enable-fpm --with-fpm-user=www-data --with-fpm-group=www-data
GPG_KEYS=6E4F6AB321FDC07F2C332E3AC2BF0BC433CFC8B3 0BD78B5F97500D450838F95DFE857D9A90D90EC1
PHP_VERSION=5.6.9
HOME=/root

上述例子中,指定了容器的別名為php,因此所有環境變數都是以PHP_開頭。

注意的是,如果源容器重啟,接收容器中的環境變數資訊並不會自動更新,因此,如果要使用源容器的IP地址,請使用/etc/hosts中配置的主機資訊。

/etc/hosts檔案

除了環境變數之外,Docker也在接收容器的/etc/hosts檔案中更新了hosts資訊。

$ docker run  -i -t --rm --link phpfpm:php php:5.6-fpm /bin/bash
root@4678acd72dca:/var/www/html#
root@4678acd72dca:/var/www/html# cat /etc/hosts
172.17.0.77    4678acd72dca
...
172.17.0.74    php f81b2615a6a8 phpfpm

從上可以看出,在接收容器的hosts檔案中增加了兩條額外的資訊,本機IP和別名以及源容器的IP和別名(php)。

與環境變數不同的是,如果源容器重啟了,接收容器中/etc/hosts中的資訊會自動更新。


參考:

Linking Containers Together

How can I use environment variables in Nginx.conf

相關文章