具有ENV变量和可选参数的Docker ENTRYPOINT

时间:2018-03-06 14:32:00

标签: python docker dockerfile

我有一个带有ENTRYPOINT的Dockerfile,它使用ENV变量。我无法获得ENTRYPOINT结构,因此容器也可以接受其他命令行参数。这是Dockerfile的相关部分:

ARG MODULE_NAME
ENV MODULE_NAME=$MODULE_NAME
ENTRYPOINT /usr/bin/python3 -m ${MODULE_NAME}

如果我只想在没有其他参数的情况下启动容器,那就可以了。

docker run my-image

但我需要能够将其他命令行参数(例如,“ - debug”标志)传递给python进程,如下所示:

docker run my-image --debug

使用上面的ENTRYPOINT形式,“ - debug”arg不会传递给python进程。我已经尝试了ENTRYPOINT的exec形式和shell形式,但无法使用ENV变量命令行参数。我试过的其他几种形式:

此操作但不接受其他参数:

ENTRYPOINT ["/bin/bash", "-c", "/usr/bin/python3 -m ${MODULE_NAME}"]

这给出了“/ usr / bin / python3:没有名为$ {MODULE_NAME}的模块”:

ENTRYPOINT ["/usr/bin/python3", "-m ${MODULE_NAME}"]

这给出了“/ usr / bin / python3:没有名为$ {MODULE_NAME}的模块”:

ENTRYPOINT ["/usr/bin/python3", "-m", "${MODULE_NAME}"]

1 个答案:

答案 0 :(得分:7)

似乎无法创建直接支持变量扩展其他命令行参数的ENTRYPOINT。虽然ENTRYPOINT的shell形式将在运行时扩展ENV变量,但它不接受来自docker run命令的附加(附加)参数。虽然ENTRYPOINT的exec形式确实支持其他命令行参数,但默认情况下它不会创建shell环境,因此不会展开ENV变量。

为了解决这个问题,可以在exec表单中显式调用bash来执行一个脚本,然后扩展ENV变量并将命令行参数传递给python进程。下面是一个示例Dockerfile:

FROM ubuntu:16.04
ARG MODULE_NAME=foo
ENV MODULE_NAME=${MODULE_NAME}

RUN apt-get update -y && apt-get install -y python3.5

# Create the module to be run
RUN echo "import sys; print('Args are', sys.argv)" > /foo.py

# Create a script to pass command line args to python
RUN echo "/usr/bin/python3.5 -m $MODULE_NAME \$@" > /run_module.sh

ENTRYPOINT ["/bin/bash", "/run_module.sh"]

码头图片的输出:

$ docker run my-image
Args are ['/foo.py']

$ docker run my-image a b c
Args are ['/foo.py', 'a', 'b', 'c']

请注意,在RUN命令期间发生变量扩展(因为它们使用shell形式),因此图像中run_script.py的内容为:

/usr/bin/python3.5 -m foo $@

如果最终的RUN命令被替换为:

    RUN echo "/usr/bin/python3.5 -m \$MODULE_NAME \$@" > /run_module.sh

然后run_script.sh将包含

/usr/bin/python3.5 -m $MODULE_NAME $@

但是运行容器的输出将是相同的,因为变量扩展将在运行时发生。第二个版本的潜在好处是可以在运行时覆盖模块而不替换ENTRYPOINT。