luc*_*ald 7 docker docker-compose
我有一个名为"node"的docker容器,我想扩展.该节点需要知道它实际上有多少节点的节点.总数可能是一个环境变量,但目前更令人不安.
$ 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
Run Code Online (Sandbox Code Playgroud)
怎么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"]
Run Code Online (Sandbox Code Playgroud)
我找到了未解决的如何在docker-compose scale之后通过主机名访问其他容器?,但我还是不知道我是哪个容器.
我可以解决此问题的方法是使用docker api。我使用了docker-py软件包来访问它。
api公开了labels
每个容器的字典和键com.docker.compose.container-number
,com.docker.compose.project
并com.docker.compose.service
完成了构建主机名所需的操作。
下面的代码是我现在正在使用的代码的简化。您可以在Github上的luckydonald / pbft / dockerus.ServiceInfos(位于gist.github.com上的备份)中找到有关缓存和奇特东西的高级代码。
让我们分步解决:
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
Run Code Online (Sandbox Code Playgroud)
这会将套接字文件映射到docker容器中。因为我们不公开任何套接字端口,所以我们不必担心外部防火墙。
1.连接到API现在我们可以连接到它了。当我使用python时,我为此使用了docker-py。
from docker import Client # pip install docker-py
cli = Client(base_url='unix://var/run/docker.sock')
Run Code Online (Sandbox Code Playgroud)
2.通过按我们自己的项目名称和服务名称过滤容器,将所有容器归入同一比例组。找到我们自己
要知道我们是哪个容器,我们将$HOSTNAME
环境变量与container 进行比较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]
Run Code Online (Sandbox Code Playgroud)
主机名应为的12个字符Id
,因此在比较时我们将其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})
Run Code Online (Sandbox Code Playgroud)
我们只希望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
Run Code Online (Sandbox Code Playgroud)
因此,我们得到了hostname_list
。
我将其用作一个类,将值缓存了一分钟: dockerus.ServiceInfos(在gist.github.com上的备份)
归档时间: |
|
查看次数: |
1007 次 |
最近记录: |