如何解决带有一个文件的 docker-compose 命名卷:错误:无法为服务创建容器,源不是目录
我正在尝试使容器 /bin/wkhtmltopdf
中的二进制文件 wkhtmltopdf
在 web
容器中可用。我尝试使用命名卷来实现这一点。
我的 docker-compose.yml
中有以下 docker 容器设置:
services:
web:
image: php:7.4-apache
command: sh -c "mkdir -p /usr/local/bin && touch /usr/local/bin/wkhtmltopdf"
entrypoint: sh -c "exec 'apache2-foreground'"
volumes:
- wkhtmltopdfvol:/usr/local/bin/wkhtmltopdf
wkhtmltopdf:
image: madnight/docker-alpine-wkhtmltopdf
command: sh -c "touch /bin/wkhtmltopdf"
entrypoint: sh -c "tail -f /dev/null" # workaround to keep container running
volumes:
- wkhtmltopdfvol:/bin/wkhtmltopdf
volumes:
wkhtmltopdfvol:
但是,我在运行 docker-compose up
时收到以下错误:
ERROR: for wkhtmltopdf Cannot create container for service wkhtmltopdf:
source /var/lib/docker/overlay2/42e7082b8024ae4ebb13a4f0003a9e17bc18b33ef0677431dd002da3c21dde88/merged/bin/wkhtmltopdf is not directory
.../bin/wkhtmltopdf 不是目录
这是否意味着我不能在容器之间共享一个文件,而只能通过命名卷共享目录?我如何实现这一目标?
编辑:我还注意到 /usr/local/bin/wkhtmltopdf
容器内的 web
是一个目录,而不是我预期的文件。
解决方法
像这样在容器之间共享二进制文件可能很棘手。卷可能不是您要寻找的机制。
如果您查看 Docker Hub page for the php
image,您可以看到 php:7.4-apache
是(当前)php:7.4.15-apache-buster
的别名,其中“Buster”是 Debian 版本的名称。然后您可以搜索 https://packages.debian.org/ 以发现 Debian 有一个预先打包的 wkhtmltopdf 软件包。您可以使用自定义 Dockerfile 安装它:
FROM php:7.4-apache
RUN apt-get update \
&& DEBIAN_FRONTEND=noninteractive \
apt-get install --assume-yes --no-install-recommends \
wkhtmltopdf
# COPY ...
# Base image provides EXPOSE,CMD
然后您的 docker-compose.yml
文件需要构建此映像:
version: '3.8'
services:
web:
build: .
# no image:,volumes:,or command: override
就这样共享二进制文件的机制而言,当二进制文件需要目标容器中不存在的共享库时,您可能会遇到麻烦。 apt-get install
机制会为您处理此问题。如果容器具有不同的共享库生态系统(尤其是基于 Alpine 的容器),或者使用来自不同操作系统的主机二进制文件,也会存在潜在问题。
您展示的 Compose 文件以一种实际上不起作用的方式混合了几个概念。命名卷始终是一个目录,因此尝试将其挂载到第二个容器中的 /bin/wkhtmltopdf
文件会导致您看到的错误。容器首先启动并创建卷存在依赖性问题。一个容器只运行一个命令,如果您同时拥有 entrypoint:
和 command:
,那么该命令将作为额外的参数传递给入口点(如果入口点是一个 sh -c ...
调用,则有效地忽略)。
如果你真的想尝试这种方法,你应该让 web: {depends_on: [wkhtmltopdf]}
强制依赖顺序。第二个容器应该在其他地方安装卷,它可能不应该有 entrypoint:
,它应该做类似 command: cp -a /bin/wkhtmltopdf /export
的事情。 (一旦此 cp
完成,它将立即退出,但这应该无关紧要。)然后第一个容器可以将卷安装到,例如,/usr/local/bin
,而不是专门设置 command:
或entrypoint:
。仍然会出现轻微的竞争条件(您不能保证 cp
命令会在 Apache 启动之前完成),但它可能不会成为实际问题。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。