我想要实现的是使用这个堆栈进行负载平衡:Docker、Docker Compose、Registrator、Consul、Consul Template、NGINX,最后是一个在浏览器中打印出“Hello world”的小型服务。所以,此时我有一个 docker-compose.yml 文件。它看起来像这样:
version: '2'
services:
accent:
build:
context: ./accent
image: accent
container_name: accent
restart: always
ports:
- 80
consul:
image: gliderlabs/consul-server:latest
container_name: consul
hostname: ${MYHOST}
restart: always
ports:
- 8300:8300
- 8400:8400
- 8500:8500
- 8600:53/udp
command: -advertise ${MYHOST} -data-dir /tmp/consul -bootstrap -client 0.0.0.0
registrator:
image: gliderlabs/registrator:latest
container_name: registrator
hostname: ${MYHOST}
network_mode: host
restart: always
volumes:
- /var/run/docker.sock:/tmp/docker.sock
command: -ip ${MYHOST} consul://${MYHOST}:8500
nginx:
container_name: nginx
image: nginx:latest
restart: always
volumes:
- /etc/nginx
ports:
- 8181:80
consul-template:
container_name: consul-template
build:
context: ./consul-template
network_mode: host
restart: always
volumes_from:
- nginx
volumes:
- /var/run/docker.sock:/tmp/docker.sock
command: -consul=${MYHOST}:8500 -wait=5s -template="/etc/ctmpl/nginx.ctmpl:/etc/nginx/nginx.conf:docker kill -s HUP nginx"
第一项服务 - 重音 - 是我需要负载平衡的 Web 服务。当我运行这个命令时:
$ docker-compose up
我看到所有服务都开始运行,并且没有看到任何错误消息。看起来好像一切都很完美。当我跑
$ docker ps
我在控制台中看到了这个:
... NAMES STATUS PORTS
consul-template Up 45 seconds
consul Up 56 seconds 0.0.0.0:8300->8300/tcp, 0.0.0.0:8400->8400/tcp, 8301-8302/tcp, 8301-8302/udp, 0.0.0.0:8500->8500/tcp, 8600/tcp, 8600/udp, 0.0.0.0:8600->53/udp
nginx Up 41 seconds 0.0.0.0:8181->80/tcp
registrator Up 56 seconds
accent Up 56 seconds 0.0.0.0:32792->80/tcp
请注意最后一行,尤其是 PORTS 列。如您所见,该服务发布了 32792 端口。为了检查我的 Web 服务是否可以实现,我转到127.0.0.1:32972
我的主机(我运行 docker compose up 的机器)并在浏览器中查看:
Hello World
这正是我想看到的。然而,这并不是我最终想要的。请看一下 docker ps 命令的输出,你会看到,我的 nginx 服务发布了 8181 端口。所以,我的期望是,当我转到这个地址时127.0.0.1:8181
——我会看到完全相同的“Hello world”页面。然而,事实并非如此。在浏览器中,我看到 Bad Gateway 错误消息,在 nginx 日志中,我看到此错误消息
nginx | 2017/01/18 06:16:45 [错误] 5#5: *5 connect() 在连接到上游时失败(111:连接被拒绝),客户端:172.18.0.1,服务器:,请求:“GET /favicon. ico HTTP/1.1”,上游:“ http://127.0.0.1:32792/index.php ”,主机:“127.0.0.1:8181”
这真的很有趣,因为 nginx 做了我期望它做的事情——上游到“ http://127.0.0.1:32792/index.php ”。但我不确定它为什么会失败。顺便说一句,这就是 nginx.conf(使用 Consul 模板自动创建)的样子:
worker_processes 1;
events {
worker_connections 1024;
}
http {
sendfile on;
upstream app_servers {
server 127.0.0.1:32792;
}
server {
listen 80;
root /code;
index index.php index.html;
location / {
try_files $uri/ $uri/ /index.php;
}
location ~ \.php$ {
proxy_pass http://app_servers;
proxy_redirect off;
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_set_header X-Forwarded-Host $server_name;
}
location ~ /\.ht {
deny all;
}
}
}
我不会改变任何东西,因为这个 nginx.conf 对我来说看起来不错。为了理解它为什么不起作用,我打开了 nginx 容器并发出了几个命令:
$ curl accent
Hello World
$ curl 127.0.0.1:32972
curl: (7) Failed to connect to 127.0.0.1 port 32972: Connection refused
$ curl accent:32972
curl: (7) Failed to connect to accent port 32972: Connection refused
同样,这很有趣,因为 nginx 容器在端口下80
而不是在其发布的32972
端口下看到我的 Web 服务。无论如何,在这个阶段我不知道为什么它不起作用以及如何解决它。我只是猜测,它以某种方式连接到方式,网络是如何在 docker-compose.yml 中配置的。我尝试了 on 重音和 nginx 服务的各种组合network_mode: host
,但无济于事 - 口音停止工作或 nginx 或两者兼而有之。所以,我需要一些帮助。