kjt*_*n99 3 python django cron docker
为了定期在我的 Django 应用程序中执行任务,我安装了django_crontab扩展。我的应用程序由 dockerized 数据库和 Django 应用程序组成。
https://pypi.org/project/django-crontab/
我已按照设置段落中的描述完成了每一步。
settings.py
INSTALLED_APPS = [
...
'django_crontab',
]
...
CRONJOBS = [
('*/1 * * * *', 'config.cron.fun')
]
Run Code Online (Sandbox Code Playgroud)
cron.py
def fun():
print("hello cron")
with open("./test.txt", "a") as f:
f.write("Hello")
Run Code Online (Sandbox Code Playgroud)
我还在以下位置添加了 cron 作业docker-entrypoint.yml:
python manage.py crontab add
python manage.py crontab show
Run Code Online (Sandbox Code Playgroud)
输出:
webapp | no crontab for root
webapp | adding cronjob: (ebcca28ea3199afe6d09a445db5d5fd8) -> ('*/1 * * * *', 'config.cron.fun')
webapp | Currently active jobs in crontab:
webapp | ebcca28ea3199afe6d09a445db5d5fd8 -> ('*/1 * * * *', 'config.cron.fun')
Run Code Online (Sandbox Code Playgroud)
在Dockerfile我使用python:3.8图像并安装cron:
RUN apt-get install -y cron && touch /var/log/cron.log
Run Code Online (Sandbox Code Playgroud)
并且运行容器后没有任何反应。
当我进入容器时,我可以看到 cron 看到了作业,但仍然没有执行它。
root@bar:/back# crontab -l
*/1 * * * * /usr/local/bin/python /back/manage.py crontab run ebcca28ea3199afe6d09a445db5d5fd8 # django-cronjobs for config
Run Code Online (Sandbox Code Playgroud)
我该如何修复它?
编辑:整个 Dockerfile:
FROM python:3.8
ENV PYTHONUNBUFFERED 1
RUN mkdir /back
WORKDIR /back
COPY . /back/
RUN apt-get update
RUN apt-get install -y cron && touch /var/log/cron.log
RUN pip install --upgrade pip
RUN pip install -r requirements.txt
ENTRYPOINT ["python3", "manage.py"]
CMD ["runserver", "0.0.0.0:8080"]
Run Code Online (Sandbox Code Playgroud)
docker-compose.yml:
version: "3"
services:
db:
image: postgres:latest
restart: unless-stopped
container_name: database
env_file:
- .env
webapp:
build: ./back
container_name: webapp
restart: unless-stopped
ports:
- "8000:8000"
env_file:
- .env
depends_on:
- db
links:
- db:db
volumes:
- ./back:/back
entrypoint: "bash /back/docker-entrypoint.sh"
command: "runserver 0.0.0.0:8000"
Run Code Online (Sandbox Code Playgroud)
docker-entrypoint.yml- 这里是进行迁移、等待db和提到的crontab命令
一个 Docker 容器只运行一个进程。您需要运行两个独立的东西:主应用程序服务器和(经过一些初始设置后)cron 守护程序。通过正确的设置,您可以直接运行两个单独的容器,从同一映像运行两个单独的命令。
我在这里要做的第一个更改是将您当前的拆分合并ENTRYPOINT为CMD一个CMD:
# no ENTRYPOINT, but
CMD ["python3", "manage.py", "runserver", "0.0.0.0:8080"]
Run Code Online (Sandbox Code Playgroud)
通过此设置,您可以使用此默认命令启动一个容器,并从同一映像启动第二个命令,但command:在 Compose 设置中覆盖。
version: '3.8'
services:
db: { ... }
webapp:
build: ./back
restart: unless-stopped
ports:
- "8000:8000"
env_file:
- .env
depends_on:
- db
# run the default command: from the image; no override
# skip unnecessary links:, container_name:, volumes: options
cron:
build: ./back # same as main application
restart: unless-stopped
env_file:
- .env
depends_on:
- db
command: cron -f # as a long-running foreground process
Run Code Online (Sandbox Code Playgroud)
我们在这里没有做的事情是在容器启动时填充 crontab。这就是ENTRYPOINT可以发挥作用的地方:如果你有一个ENTRYPOINT,它会代替 运行CMD,但它会作为参数传递CMD。这里一个非常典型的模式是使用ENTRYPOINTshell 脚本进行一些首次设置,然后以exec "$@"运行所提供的任何内容CMD结束command:。
#!/bin/sh
# docker-entrypoint.sh
# If this is going to be a cron container, set up the crontab.
if [ "$1" = cron ]; then
./manage.py crontab add
fi
# Launch the main container command passed as arguments.
exec "$@"
Run Code Online (Sandbox Code Playgroud)
然后在 Dockerfile 中,将此脚本指定为ENTRYPOINT.
...
WORKDIR /back
COPY . ./ # including docker-entrypoint.sh
...
ENTRYPOINT ["./docker-entrypoint.sh"] # must be JSON-array syntax
CMD ["./manage.py", "runserver", "0.0.0.0:8080"] # as before
Run Code Online (Sandbox Code Playgroud)
您不需要entrypoint:在这里覆盖。docker-compose run web app bash像获取交互式调试 shell这样的命令可以正常工作;他们将通过该入口点脚本,并运行 shell 作为最后exec "$@"一行。
| 归档时间: |
|
| 查看次数: |
4630 次 |
| 最近记录: |