我将 Celery 与 Django 和 Redis 一起用于日常任务。它实际上有效,但有时芹菜会因
redis.exceptions.ResponseError
崩溃,这是我不能允许的。我看到了配置 redis 身份验证的解决方案,但它对我不起作用(或者我做错了什么)。
[2024-01-24 08:49:17,548: INFO/MainProcess] Connected to redis://default:**@redis:6379/0
[2024-01-24 08:51:10,365: CRITICAL/MainProcess] Unrecoverable error: ResponseError('UNBLOCKED force unblock from blocking operation, instance state changed (master -> replica?)')
Traceback (most recent call last):
File "/usr/local/lib/python3.12/site-packages/celery/worker/worker.py", line 202, in start
self.blueprint.start(self)
File "/usr/local/lib/python3.12/site-packages/celery/bootsteps.py", line 116, in start
step.start(parent)
File "/usr/local/lib/python3.12/site-packages/celery/bootsteps.py", line 365, in start
return self.obj.start()
^^^^^^^^^^^^^^^^
File "/usr/local/lib/python3.12/site-packages/celery/worker/consumer/consumer.py", line 340, in start
blueprint.start(self)
File "/usr/local/lib/python3.12/site-packages/celery/bootsteps.py", line 116, in start
step.start(parent)
File "/usr/local/lib/python3.12/site-packages/celery/worker/consumer/consumer.py", line 742, in start
c.loop(*c.loop_args())
File "/usr/local/lib/python3.12/site-packages/celery/worker/loops.py", line 97, in asynloop
next(loop)
File "/usr/local/lib/python3.12/site-packages/kombu/asynchronous/hub.py", line 373, in create_loop
cb(*cbargs)
File "/usr/local/lib/python3.12/site-packages/kombu/transport/redis.py", line 1344, in on_readable
self.cycle.on_readable(fileno)
File "/usr/local/lib/python3.12/site-packages/kombu/transport/redis.py", line 569, in on_readable
chan.handlers[type]()
File "/usr/local/lib/python3.12/site-packages/kombu/transport/redis.py", line 962, in _brpop_read
dest__item = self.client.parse_response(self.client.connection,
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/usr/local/lib/python3.12/site-packages/redis/client.py", line 553, in parse_response
response = connection.read_response()
^^^^^^^^^^^^^^^^^^^^^^^^^^
File "/usr/local/lib/python3.12/site-packages/redis/connection.py", line 524, in read_response
raise response
redis.exceptions.ResponseError: UNBLOCKED force unblock from blocking operation, instance state changed (master -> replica?)
services:
redis:
image: redis:latest
container_name: redis
restart: on-failure
volumes:
- ./redis.conf:/usr/local/etc/redis/redis.conf
ports:
- "6379:6379"
django:
image: django-service
container_name: django
ports:
- '8000:8000'
env_file:
- .django
depends_on:
- redis
command: /django-start.sh
celery:
image: django-service
container_name: celery
restart: on-failure
env_file:
- .django
depends_on:
- django
command: /celery-start.sh
celery-beat:
image: django-service
container_name: celery-beat
restart: on-failure
env_file:
- .django
depends_on:
- django
command: /celery-beat-start.sh
requirepass someInsanePass
#!/bin/bash
celery -A celery_worker.app worker --loglevel=debug -E --uid=nobody --gid=nogroup
#!/bin/bash
celery -A celery_worker.app beat --loglevel=debug
...
# CELERY
# --------------------------------------------------
CELERY_BROKER_URL = os.environ.get("CELERY_BROKER_URL", "redis://default:someInsanePass@localhost:6379/0")
CELERY_RESULT_BACKEND = os.environ.get("CELERY_RESULT_BACKEND", "redis://default:someInsanePass@localhost:6379/0")
CELERY_TIMEZONE = TIME_ZONE
CELERY_EAGER_PROPAGATES_EXCEPTIONS = True
CELERY_BEAT_SCHEDULE = {
"some_everyday_task_1": {
"task": "apps.app1.tasks.task1",
"schedule": crontab(hour=0, minute=0),
},
"some_everyday_task_2": {
"task": "apps.app2.tasks.task2",
"schedule": crontab(hour=4, minute=0),
},
}
...
# ...
DJANGO_SETTINGS_MODULE=config.settings
CELERY_BROKER_URL=redis://default:someInsanePass@redis:6379/0
CELERY_RESULT_BACKEND=redis://default:someInsanePass@redis:6379/0
# ...
from celery import Celery, Task
app = Celery("tasks")
app.config_from_object('django.conf:settings', namespace='CELERY')
class BaseTask(Task):
def on_failure(self, exc, task_id, args, kwargs, einfo):
raise exc
我在 github 上发现了这个问题 itamarhaber 说:
我在这里进行大胆的猜测 - 您的 Redis 服务器没有密码保护并且向公众开放。如果是这种情况,则表示恶意实体正在尝试通过
命令劫持服务器。SLAVEOF
就我而言,为 redis 设置密码并不是解决方案。但是关闭 Redis 的端口是。
services:
redis:
image: redis:latest
container_name: redis
restart: on-failure
volumes:
- ./redis.conf:/usr/local/etc/redis/redis.conf