无法在 Django 中运行 Celery 任务 - 我要么得到“AppRegistryNotReady:应用程序尚未加载”或“RuntimeError:populate() 不可重入”

问题描述

我正在尝试使用 Django 中的 Celery 设置一个任务,使其每天在 23:00 运行。

app = Celery('App.tasks',broker='redis://localhost')
os.environ.setdefault("DJANGO_SETTINGS_MODULE","App.settings")

django.setup() <== PROBLEM

@app.on_after_configure.connect
def setup_periodic_tasks(sender,**kwargs):
    sender.add_periodic_task(
        crontab(hour=23),calc_average_rating.s(),)


@app.task
def calc_average_rating(final_content_id):

问题是在这函数中,我有rating = apps.get_model(app_label='App',model_name='rating'),如果我不调用django.setup(),那么我得到django.core.exceptions.AppRegistryNotReady: Apps aren't loaded yet.

但是,如果我调用 django.setup(),任务运行良好,但我无法执行 manage.py runserver,因为我得到 RuntimeError: populate() isn't reentrant

有什么解决办法吗?

解决方法

我不确定如何准确地重现您所处的环境,因此这里是我的环境中的一些观察结果,希望对您有所帮助

我唯一拥有 Celery() 对象的地方是一个独立文件,保存在“manage.py startproject”生成的包中,

我认为与大多数 django 用户相比,我布置 django 应用程序的方式不同寻常,因此描述一下:

# .git/  # top folder is my vcs
# setup.py  # packaging for exampleapp
# env/  # python venv created to this service
# exampleapp/  # package generated from startapp
# exampleapp/tasks.py  # package generated from startapp
# exampleproject/  # folder generated from startproject
# exampleproject/exampleproject/  # package generated by startproject
# exampleproject/exampleproject/settings.py  # generated
# exampleproject/exampleproject/celery.py  # created based on celery docs
# exampleproject/exampleproject/celery.py
import os

from celery import Celery

os.environ.setdefault('DJANGO_SETTINGS_MODULE','exampleproject.settings')

app = Celery('exampleproject')

app.config_from_object('django.conf:settings',namespace='CELERY')
app.autodiscover_tasks()

@app.task(bind=True)
def debug_task(self):
    print('Request: {self.request!r}'.format(self=self))

if __name__ == '__main__':
    app.start()

我像下面这样开始芹菜作业,其中我的 python 虚拟环境文件夹 'env' 是生成的 exampleproject 包的兄弟

(
        cd exampleproject
        ../env/bin/python3 -m celery -A exampleproject  worker -l INFO
        # or 
        ../env/bin/python3 -m celery -A exampleproject  beat -l INFO --scheduler django_celery_beat.schedulers:DatabaseScheduler

)
# and for django
./env/bin/python3 exampleproject/manage.py runserver

可能也有兴趣

# exampleapp/tasks.py
from celery import shared_task

@shared_task
def add(x,y):
    return x+y
# exampleproject/exampleproject/settings.py
# suffixed to end of generated file

INSTALLED_APPS.extend([
    'django_celery_results','django_celery_beat',])

CELERY_TASK_TRACK_STARTED = True
CELERY_TASK_TIME_LIMIT = 30 * 60
CELERY_RESULT_BACKEND = 'django-db'
#CELERY_RESULT_BACKEND = 'django-cache'


使用这些部分,我没有注意到加载入口点有任何问题

相关问答

Selenium Web驱动程序和Java。元素在(x,y)点处不可单击。其...
Python-如何使用点“。” 访问字典成员?
Java 字符串是不可变的。到底是什么意思?
Java中的“ final”关键字如何工作?(我仍然可以修改对象。...
“loop:”在Java代码中。这是什么,为什么要编译?
java.lang.ClassNotFoundException:sun.jdbc.odbc.JdbcOdbc...