使用Django 1.8.5和Celery进行ImportError

时间:2015-10-17 00:41:11

标签: python django python-3.x celery

我正在尝试让Celery使用Django,遵循官方文档,以及位于此处的视频:https://godjango.com/63-deferred-tasks-and-scheduled-jobs-with-celery-31-django-17-and-redis/

我不知道我做错了什么,但这总是导致一个ImportError。

该项目名为“clubmgmt”

“clubmgmt / celery.py”的内容:

from __future__ import absolute_import
import os
import django
from celery import Celery
from django.conf import settings

# set the default Django settings module for the 'celery' program.
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'clubmgmt.settings')
django.setup()

app = Celery('clubmgmt')

# Using a string here means the worker will not have to
# pickle the object when using Windows.
app.config_from_object('django.conf:settings')
app.autodiscover_tasks(lambda: settings.INSTALLED_APPS)

“clubmgmt / __ init __。py”

的内容
from __future__ import absolute_import

# This will make sure the app is always imported when
# Django starts so that shared_task will use this app.
from clubmgmt.celery import app as celery_app

任务在名为“激活”的应用中定义

“activation / tasks.py”的内容:

from activation.models import Activation
from django.conf import settings
from django.template.loader import render_to_string
import smtplib
from email.mime.text import MIMEText
from email.mime.multipart import MIMEMultipart
from clubmgmt.celery import app

__author__ = 'jeroenjacobs'

@app.task
def send_activation_mail(activation_pk):
    activation = Activation.objects.get(pk=activation_pk)
    mail = activation.user.email

    msg = MIMEMultipart('alternative')
    msg['Subject'] = "Please activate your account"
    msg['From'] = settings.SMTP_SENDER_ADDRESS
    msg['To'] = mail
    ...

我总是收到以下错误:

(clubmgmt) > $ python manage.py runserver                                                                                                                      [±master ●●●]
Traceback (most recent call last):
  File "manage.py", line 10, in <module>
    execute_from_command_line(sys.argv)
  File "/Users/jeroenjacobs/.pyenv/versions/clubmgmt/lib/python3.4/site-packages/django/core/management/__init__.py", line 351, in execute_from_command_line
    utility.execute()
  File "/Users/jeroenjacobs/.pyenv/versions/clubmgmt/lib/python3.4/site-packages/django/core/management/__init__.py", line 303, in execute
    settings.INSTALLED_APPS
  File "/Users/jeroenjacobs/.pyenv/versions/clubmgmt/lib/python3.4/site-packages/django/conf/__init__.py", line 48, in __getattr__
    self._setup(name)
  File "/Users/jeroenjacobs/.pyenv/versions/clubmgmt/lib/python3.4/site-packages/django/conf/__init__.py", line 44, in _setup
    self._wrapped = Settings(settings_module)
  File "/Users/jeroenjacobs/.pyenv/versions/clubmgmt/lib/python3.4/site-packages/django/conf/__init__.py", line 92, in __init__
    mod = importlib.import_module(self.SETTINGS_MODULE)
  File "/Users/jeroenjacobs/.pyenv/versions/3.4.3/Python.framework/Versions/3.4/lib/python3.4/importlib/__init__.py", line 109, in import_module
    return _bootstrap._gcd_import(name[level:], package, level)
  File "<frozen importlib._bootstrap>", line 2254, in _gcd_import
  File "<frozen importlib._bootstrap>", line 2237, in _find_and_load
  File "<frozen importlib._bootstrap>", line 2212, in _find_and_load_unlocked
  File "<frozen importlib._bootstrap>", line 321, in _call_with_frames_removed
  File "<frozen importlib._bootstrap>", line 2254, in _gcd_import
  File "<frozen importlib._bootstrap>", line 2237, in _find_and_load
  File "<frozen importlib._bootstrap>", line 2226, in _find_and_load_unlocked
  File "<frozen importlib._bootstrap>", line 1200, in _load_unlocked
  File "<frozen importlib._bootstrap>", line 1129, in _exec
  File "<frozen importlib._bootstrap>", line 1471, in exec_module
  File "<frozen importlib._bootstrap>", line 321, in _call_with_frames_removed
  File "/Users/jeroenjacobs/PycharmProjects/clubmgmt/clubmgmt/__init__.py", line 5, in <module>
    from clubmgmt.celery import app as celery_app
  File "/Users/jeroenjacobs/PycharmProjects/clubmgmt/clubmgmt/celery.py", line 9, in <module>
    django.setup()
  File "/Users/jeroenjacobs/.pyenv/versions/clubmgmt/lib/python3.4/site-packages/django/__init__.py", line 18, in setup
    apps.populate(settings.INSTALLED_APPS)
  File "/Users/jeroenjacobs/.pyenv/versions/clubmgmt/lib/python3.4/site-packages/django/apps/registry.py", line 115, in populate
    app_config.ready()
  File "/Users/jeroenjacobs/PycharmProjects/clubmgmt/activation/apps.py", line 10, in ready
    import activation.handlers
  File "/Users/jeroenjacobs/PycharmProjects/clubmgmt/activation/handlers.py", line 6, in <module>
    from activation.utils import create_or_update_activation
  File "/Users/jeroenjacobs/PycharmProjects/clubmgmt/activation/utils.py", line 2, in <module>
    from activation.tasks import send_activation_mail
  File "/Users/jeroenjacobs/PycharmProjects/clubmgmt/activation/tasks.py", line 7, in <module>
    from clubmgmt.celery import app
ImportError: cannot import name 'app'

有人可以解释一下为什么“app”无法导入?

BTW:这是在python 3.4下测试的。回到Python 2不是一个选项,所以请不要建议这个解决方案。如果Celery不兼容Python3,还有其他选择吗?

1 个答案:

答案 0 :(得分:1)

问题是clubmgmt.celery.app尝试导入clubmgm.celery.app导致循环导入。

如果您查看堆栈跟踪clubmgmt.__init__导入clubmgmt.celery.app,它会使django运行其设置例程,最终触发clubmgmt.activation.apps的导入再次点击clubmgmt.activation.tasks导入clubmgmt.celery.app,同时导入ImportError

__init__.py中删除该导入肯定会解决问题,但如果您仍然需要它,则可以使用activation/apps.py代替。

到目前为止,我学到的一件事是90%的时间ImportError由于循环导入而发生,另外10%是由于拼写错误,或类似的东西。