容器如何识别它是一组可扩展的 docker-compose 服务中的哪个容器?

问题描述 投票:0回答:3

我有一个名为

node
的 docker 容器,我想将其扩展到 n 个容器。 给定的
node
容器需要知道它是 n 个缩放的
node
容器集合中的哪个容器。 总数可能位于环境变量中,但当前情况更令人不安。

$ 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"]

我发现了未解决的如何在docker-compose scale之后通过主机名到达其他容器?,但我仍然不知道我是哪个容器。

docker docker-compose docker-container
3个回答
10
投票

这可以通过使用 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) 上找到我的带有缓存和奇特内容的高级代码,以及类似的 java 版本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]

主机名应该是

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

我也将其用作类,并将值缓存一分钟: dockerus.ServiceInfos(备份于gist.github.com


0
投票

如果将 docker 套接字映射到容器中,您可以获得此信息,而无需比安装在容器中的

curl
更复杂。

这将为您提供有关容器的原始数据(与您从

docker inspect <id>
获得的信息相同):

curl -s --unix-socket /var/run/docker.sock "http://v1.41/containers/$(hostname)/json"

如果容器中有

jq
,您可以将其通过管道传输到
jq -r .Name
以获取名称,如果没有
jq
,您可以使用它:
tr ',' '\n' | tr -d '"' | grep '^Name:/' | awk -F: '{print $2}'


-3
投票

最新版本的 docker 应该能够自动发现扩展的服务。例如:

services:
  app: {}
  worker: {}
  

启动 docker-compose 后运行scale:

docker-compose scale worker=10

您的应用程序容器可以通过以下方式调用服务:

http://worker{:port}

Docker 将选择一个随机运行的工作容器来将请求转发到。

© www.soinside.com 2019 - 2024. All rights reserved.