docker - 无法在 Google Cloud Run 上部署 Ubuntu 20.04 Docker 容器
问题描述
我正在尝试通过 Google Cloud Run 部署一个基于 Ubuntu 20.04 的简单的基于 Python 的 Docker 容器。我已经成功构建了镜像,但是当我尝试部署 Cloud Run 服务时,我收到以下错误(省略了项目详细信息):
Cloud Run error: Invalid argument error. Invalid ENTRYPOINT. [name: "gcr.io/{PROJECT_ID}/{SERVICE_NAME}@sha256:{HASH}"
error: "Invalid command \"/bin/sh\": fil
e not found"
e not found"
]....failed
Deployment failed
不过,奇怪的是,如果我在本地拉取并运行图像,它就可以正常工作。
docker run --rm --publish 5000:5000 -e PORT=5000 -it gcr.io/{PROJECT_ID}/{SERVICE_NAME}@sha256:{HASH}
我的 Dockerfile 几乎是最基本的:
FROM ubuntu:20.04
COPY . /app
WORKDIR /app
RUN apt-get update \
&& DEBIAN_FRONTEND=noninteractive apt-get install -y python3 python3-pip \
&& pip3 install gunicorn Flask flask-cors
CMD exec gunicorn --bind :$PORT --worker-tmp-dir /dev/shm --timeout 900 wsgi:app
更奇怪的是,如果我用 替换基本图像debian:buster-slim
,它就可以正常工作。
有谁知道会发生什么?
附加信息:
status:
conditions:
- type: Ready
status: 'False'
message: |-
Cloud Run error: Invalid argument error. Invalid ENTRYPOINT. [name: "gcr.io/{PROJECT_ID}/{SERVICE_NAME}@sha256:{HASH}"
error: "Invalid command \"/bin/sh\": file not found"
].
lastTransitionTime: '2020-05-12T07:40:12.804Z'
- type: ConfigurationsReady
status: 'False'
message: |-
Cloud Run error: Invalid argument error. Invalid ENTRYPOINT. [name: "gcr.io/{PROJECT_ID}/{SERVICE_NAME}@sha256:{HASH}"
error: "Invalid command \"/bin/sh\": file not found"
].
lastTransitionTime: '2020-05-12T07:40:12.804Z'
- type: RoutesReady
status: 'True'
lastTransitionTime: '2020-05-12T06:19:12.224Z'
解决方案
我遇到了同样的问题。似乎是断断续续的:白天我无法部署到 Cloud Run,但在深夜,各种变通办法大约有一半的时间起作用。
我发现最可靠的解决方法是在 CMD 或 ENTRYPOINT 中不依赖 /bin/sh 或 /bin/bash。/bin/sh 似乎在运行时存在,但有时在 Cloud Run 在部署前测试容器时不存在。
而不是在 Dockerfile 中:
CMD exec gunicorn --bind :$PORT --workers 1 --threads 4 main:app
我改用这个:
CMD ["/usr/bin/python3", "/app/run_gunicorn.py", "--workers", "1", "--threads", "4", "main:app"]
然后我添加了 run_gunicorn.py 脚本:
import os
import sys
from gunicorn.app.wsgiapp import run
port = os.environ['PORT']
sys.argv[-1:-1] = ['--bind', f':{port}']
print("sys.argv:", sys.argv)
run()
这是保持端口号动态的一种方法。
推荐阅读
- python - 如果字典为空,则字典上的 for 循环将失败
- c++ - 在 Eclipse CDT 中折叠 C++ 类的私有部分?
- python - scrapy - 多次解析
- docker - traefik 的 ACME 证书超时
- python - 计算键并将它们存储到相应的值
- python - 为什么 libvirt python 模块会停止我的脚本?
- java - 如何在 Android 中正确使用快餐栏
- sql - INSERT INTO 后 Redshift 更新中缺少数据
- java - 如何使用 H2 嵌入式数据库导出和部署 JavaFX 应用程序?
- ios - 作为 MTKViews 子视图的 UIViews 是否在 GPU 上处理/呈现?