2

我有一个通过 supervisord 运行 django celery worker 的 docker,程序设置非常简单

[program:celery_priority]
command=python manage.py celery worker -E -Q priority --concurrency=2 --loglevel=ERROR
directory=/var/lib/app
stdout_events_enabled = true
stderr_events_enabled = true
stopwaitsecs = 600

[program:celery_medium]
command=python manage.py celery worker -E -Q medium --concurrency=2 --loglevel=ERROR
directory=/var/lib/app
stdout_events_enabled = true
stderr_events_enabled = true
stopwaitsecs = 600

[program:celerycam]
command=python manage.py celerycam
directory=/var/lib/app
stdout_events_enabled = true
stderr_events_enabled = true
stopwaitsecs = 600

我们的部署周期使用 fig 来管理 docker,这里是我们的 fig.yml 文件对于 worker 的样子

worker:
  build: .docker/worker
  command: normal
  volumes_from:
    - appdata
  hostname: workerprod
  domainname: project.internal
  links:
    - redis
    - rabbit
    - appdata
    - mail

我们面临的问题是,当我们尝试使用fig restart workersupervisord 程序时失败,因为它发现 pid 与以下错误发生冲突

[130.211.XX.XX] out: worker_1     | celery_medium stderr | [2015-02-13 13:40:54,271: WARNING/MainProcess] ERROR: Pidfile (/tmp/med_celery.pid) already exists.
[130.211.XX.XX] out: worker_1     | Seems we're already running? (pid: 17)
[130.211.XX.XX] out: worker_1     | celery_priority stderr | [2015-02-13 13:40:54,272: WARNING/MainProcess] ERROR: Pidfile (/tmp/priority_celery.pid) already exists.
[130.211.XX.XX] out: worker_1     | Seems we're already running? (pid: 16)
[130.211.XX.XX] out: worker_1     | 2015-02-13 18:40:54,359 INFO exited: celery_medium (exit status 0; expected)
[130.211.XX.XX] out: worker_1     | 2015-02-13 18:40:54,359 INFO exited: celery_priority (exit status 0; expected)
[130.211.XX.XX] out: worker_1     | 2015-02-13 18:40:55,360 INFO success: celerycam entered RUNNING state, process has stayed up for > than 1 seconds (startsecs)

然而,当我们使用fig -d up worker它时,它是有效的,因为显然upfig 试图重新创建容器而不使用现有的容器。但这会导致所有链接服务也重新创建,从而丢失 RabbitMQ 数据和 Redis 缓存。

有没有办法使用 simple 重新启动 dockerfig restart worker并确保 pid 在重新启动时清除?请指教

4

1 回答 1

5

创建一个ENTRYPOINT脚本,在运行您的CMD. 例如

FROM someotherimage
COPY entrypoint.sh /entrypoint.sh
ENTRYPOINT ["/entrypoint.sh"]

并在entrypoint.sh

#!/bin/sh
rm -f /tmp/*.pid
exec "$@"

ENTRYPOINT脚本将在每次容器启动时运行,并确保/tmp在运行容器命令之前清除所有 pid 文件。

于 2015-02-13T20:22:32.083 回答