容器如何识别它在一组缩放的 docker-compose 服务中属于哪个容器?
How can a container identify which container it is in a set of a scaled docker-compose service?
我有一个名为 node
的 docker 容器,我想将其扩展到 n 个容器。
给定的 node
容器需要知道它是 n 个缩放的 node
容器集合中的哪个容器。
total可以放在环境变量里,但是current比较麻烦
$ docker-compose scale node=100
Starting projectdir_node_1 ... done
Creating and starting projectdir_node_2 ... done
Creating and starting projectdir_node_3 ... done
Creating and starting projectdir_node_4 ... done
Creating and starting projectdir_node_5 ... done
Creating and starting projectdir_node_6 ... done
Creating and starting projectdir_node_7 ... done
Creating and starting projectdir_node_8 ... done
Creating and starting projectdir_node_9 ... done
...
Creating and starting projectdir_node_99 ... done
Creating and starting projectdir_node_100 ... done
projectdir_node_100
如何知道它是节点100
?
我看到 $HOSTNAME
是容器 ID(例如 2c73136347cd
),但没有找到主机名的 ENV 变量和我需要的编号。
作为参考,我的 docker-compose.yml
:
version: '2'
services:
node:
build: ./node/
volumes:
- ./node/code/:/code:ro
entrypoint: ["/bin/bash"]
我找到了未解How to reach additional containers by the hostname after docker-compose scale?,但我仍然不知道我是哪个容器
这可以通过使用 Docker api.
来解决
虽然此答案中的代码是 Python,而我的链接示例是 Java,但它完全独立于您的语言。只需将 Docker API 与您选择的库一起使用,或者直接向 Docker API 套接字发出请求。
我使用 docker-py 包来访问它。
api 为每个容器公开了一个 labels
字典,键 com.docker.compose.container-number
、com.docker.compose.project
和 com.docker.compose.service
完成了构建主机名.
下面的代码是我现在使用的代码的简化版。
您可以在 Github 和 luckydonald/pbft/dockerus.ServiceInfos (python) as well as a similar java version at luckydonald/pbft-java/de.luckydonald.utils.dockerus.Dockerus (java) 找到我的高级代码,其中包含缓存和花哨的东西,它试图在 java 中做同样的事情,但可能更难阅读,并且没有缓存。
某事
让我们分步解决这个问题:
0。使 API 对容器可用。
我们需要让套接字文件对卷可用,因此在 docker-compose.yml
文件的卷部分添加 /var/run/docker.sock:/var/run/docker.sock
:
version: '2'
services:
node:
build: .
volumes:
- /var/run/docker.sock:/var/run/docker.sock
这会将套接字文件映射到 docker 容器中。
因为我们不暴露任何套接字端口,所以我们不必担心外界的防火墙。
1.连接到 API
现在我们可以连接到它了。因为我使用 python,所以我使用 docker-py。
from docker import Client # pip install docker-py
cli = Client(base_url='unix://var/run/docker.sock')
2。通过我们自己的项目名称和服务名称过滤容器,获取同一规模组中的所有容器。
找到我们自己
要知道我们是哪个容器,我们将$HOSTNAME
环境变量与容器Id
进行比较。
import os
HOSTNAME = os.environ.get("HOSTNAME")
all_containers = cli.containers()
# filter out ourself by HOSTNAME
our_container = [c for c in all_containers if c['Id'][:12] == HOSTNAME[:12]][0]
hostname 应该是 Id
的 12 个字符,所以我们在比较的时候去掉 id 以确保它们相等。
our_container
现在是我们自己的 api 代表。耶。
接下来是获取其他容器。
我们将搜索具有相同项目和服务名称的容器。
这样我们就知道它们是我们自己的实例。
service_name = our_container.Labels['com.docker.compose.service']
project_name = our_container.Labels['com.docker.compose.project']
filters = [
'com.docker.compose.project={}'.format(project_name),
'com.docker.compose.service={}'.format(service_name)
]
# The python wrapper has a filter function to do that work for us.
containers = cli.containers(filters={'label': filters})
我们只想要 com.docker.compose.project
和 com.docker.compose.service
标签与我们自己的容器相同的每个容器。
最后建立一个主机名列表
hostname_list = list()
for container in containers:
project = container.Labels["com.docker.compose.project"]
service = container.Labels["com.docker.compose.service"]
number = container.Labels["com.docker.compose.container-number"]
hostname = "{project}_{service}_{i}".format(project=project, service=service, i=number)
hostname_list.append(hostname)
# end for
所以,我们得到了 hostname_list
。
我也将其用作 class,并将值缓存一分钟:
dockerus.ServiceInfos (backup at gist.github.com)
docker 的最新版本应该能够自动发现缩放服务。例如:
services:
app: {}
worker: {}
运行 启动 docker-compose 后缩放:
docker-compose scale worker=10
您的应用容器可以通过以下方式调用服务:
http://worker{:port}
Docker 将随机选择一个 运行 worker 容器将请求转发到。
我有一个名为 node
的 docker 容器,我想将其扩展到 n 个容器。
给定的 node
容器需要知道它是 n 个缩放的 node
容器集合中的哪个容器。
total可以放在环境变量里,但是current比较麻烦
$ docker-compose scale node=100
Starting projectdir_node_1 ... done
Creating and starting projectdir_node_2 ... done
Creating and starting projectdir_node_3 ... done
Creating and starting projectdir_node_4 ... done
Creating and starting projectdir_node_5 ... done
Creating and starting projectdir_node_6 ... done
Creating and starting projectdir_node_7 ... done
Creating and starting projectdir_node_8 ... done
Creating and starting projectdir_node_9 ... done
...
Creating and starting projectdir_node_99 ... done
Creating and starting projectdir_node_100 ... done
projectdir_node_100
如何知道它是节点100
?
我看到 $HOSTNAME
是容器 ID(例如 2c73136347cd
),但没有找到主机名的 ENV 变量和我需要的编号。
作为参考,我的 docker-compose.yml
:
version: '2'
services:
node:
build: ./node/
volumes:
- ./node/code/:/code:ro
entrypoint: ["/bin/bash"]
我找到了未解How to reach additional containers by the hostname after docker-compose scale?,但我仍然不知道我是哪个容器
这可以通过使用 Docker api.
来解决虽然此答案中的代码是 Python,而我的链接示例是 Java,但它完全独立于您的语言。只需将 Docker API 与您选择的库一起使用,或者直接向 Docker API 套接字发出请求。
我使用 docker-py 包来访问它。
api 为每个容器公开了一个 labels
字典,键 com.docker.compose.container-number
、com.docker.compose.project
和 com.docker.compose.service
完成了构建主机名.
下面的代码是我现在使用的代码的简化版。 您可以在 Github 和 luckydonald/pbft/dockerus.ServiceInfos (python) as well as a similar java version at luckydonald/pbft-java/de.luckydonald.utils.dockerus.Dockerus (java) 找到我的高级代码,其中包含缓存和花哨的东西,它试图在 java 中做同样的事情,但可能更难阅读,并且没有缓存。
某事
让我们分步解决这个问题:
0。使 API 对容器可用。
我们需要让套接字文件对卷可用,因此在 docker-compose.yml
文件的卷部分添加 /var/run/docker.sock:/var/run/docker.sock
:
version: '2'
services:
node:
build: .
volumes:
- /var/run/docker.sock:/var/run/docker.sock
这会将套接字文件映射到 docker 容器中。 因为我们不暴露任何套接字端口,所以我们不必担心外界的防火墙。
1.连接到 API
现在我们可以连接到它了。因为我使用 python,所以我使用 docker-py。
from docker import Client # pip install docker-py
cli = Client(base_url='unix://var/run/docker.sock')
2。通过我们自己的项目名称和服务名称过滤容器,获取同一规模组中的所有容器。
找到我们自己
要知道我们是哪个容器,我们将$HOSTNAME
环境变量与容器Id
进行比较。
import os
HOSTNAME = os.environ.get("HOSTNAME")
all_containers = cli.containers()
# filter out ourself by HOSTNAME
our_container = [c for c in all_containers if c['Id'][:12] == HOSTNAME[:12]][0]
hostname 应该是 Id
的 12 个字符,所以我们在比较的时候去掉 id 以确保它们相等。
our_container
现在是我们自己的 api 代表。耶。
接下来是获取其他容器。
我们将搜索具有相同项目和服务名称的容器。 这样我们就知道它们是我们自己的实例。
service_name = our_container.Labels['com.docker.compose.service']
project_name = our_container.Labels['com.docker.compose.project']
filters = [
'com.docker.compose.project={}'.format(project_name),
'com.docker.compose.service={}'.format(service_name)
]
# The python wrapper has a filter function to do that work for us.
containers = cli.containers(filters={'label': filters})
我们只想要 com.docker.compose.project
和 com.docker.compose.service
标签与我们自己的容器相同的每个容器。
最后建立一个主机名列表
hostname_list = list()
for container in containers:
project = container.Labels["com.docker.compose.project"]
service = container.Labels["com.docker.compose.service"]
number = container.Labels["com.docker.compose.container-number"]
hostname = "{project}_{service}_{i}".format(project=project, service=service, i=number)
hostname_list.append(hostname)
# end for
所以,我们得到了 hostname_list
。
我也将其用作 class,并将值缓存一分钟: dockerus.ServiceInfos (backup at gist.github.com)
docker 的最新版本应该能够自动发现缩放服务。例如:
services:
app: {}
worker: {}
运行 启动 docker-compose 后缩放:
docker-compose scale worker=10
您的应用容器可以通过以下方式调用服务:
http://worker{:port}
Docker 将随机选择一个 运行 worker 容器将请求转发到。