这可以通过使用 Docker api 来解决。
虽然此答案中的代码是 Python,而我的链接示例是 Java,但这完全独立于您的语言。只需将 Docker API 与您选择的库一起使用,或直接向 Docker API 套接字发出请求。
我使用docker-py包来访问它。
apilabels
为每个容器公开了一个字典,以及 keys com.docker.compose.container-number
,com.docker.compose.project
并com.docker.compose.service
完成了构建hostname所需的操作。
下面的代码是我现在使用的代码的简化。你可以在 Github 的luckydonald/pbft/dockerus.ServiceInfos (python)上找到我的高级代码,其中包含缓存和花哨的东西,以及在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]
主机名应该是 12 个字符Id
,所以我们在比较时剪掉了 id 以确保它是相等的。
our_container
now 是我们自己的 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
.
我也将它用作一个类,将值缓存一分钟:
dockerus.ServiceInfos(gist.github.com 上的备份)