一尘不染

将我的signal.py文件保留在Django项目中的正确位置

django

根据我正在阅读的Django文档,似乎signals.py在app文件夹中是一个不错的起点,但是我面临的问题是,当我为创建信号pre_save并尝试从模型中导入类时,它与import在我的模型中。

# models.py

from django.contrib.auth.models import User
from django.db import models
from django.utils.translation import gettext as _
from signals import *

class Comm_Queue(CommunicatorAbstract):
    queue_statuses = (
        ('P', _('Pending')),
        ('S', _('Sent')),
        ('E', _('Error')),
        ('R', _('Rejected')),
    )
    status          = models.CharField(max_length=10, db_index=True, default='P')
    is_html         = models.BooleanField(default=False)
    language        = models.CharField(max_length=6, choices=settings.LANGUAGES)
    sender_email    = models.EmailField()
    recipient_email = models.EmailField()
    subject         = models.CharField(max_length=100)
    content         = models.TextField()
# signals.py

from django.conf import settings
from django.db.models.signals import pre_save
from django.dispatch import receiver
from models import Comm_Queue

@receiver(pre_save, sender=Comm_Queue)
def get_sender_email_from_settings(sender, **kwargs):
    obj=kwargs['instance']
    if not obj.sender_email:
        obj.sender_email='%s' % settings.ADMINS[0][1]

此代码将无法运行,因为我Comm_Queue在内部signals.py导入了信号,也在内部也导入了信号models.py。

谁能建议我如何解决这个问题?


阅读 360

收藏
2020-03-27

共2个答案

一尘不染

对于Django <1.7的原始答案:
你可以通过导入signals.py应用程序的__init__.py文件来注册信号:

# __init__.py
import signals

这将允许从中进行导入models.py,signals.py而不会出现循环导入错误。

这种方法的一个问题是,如果你使用coverage.py,它会弄乱覆盖率结果。

2020-03-27
一尘不染

如果您使用的是Django <= 1.6,我建议使用Kamagatos解决方案:只需将信号导入模型模块的末尾即可。

对于将来的Django版本(> = 1.7),建议的方法是将信号模块导入应用程序的config ready()函数中:

my_app/apps.py

from django.apps import AppConfig

class MyAppConfig(AppConfig):
name = ‘my_app’

def ready(self):
    import my_app.signals

my_app/init.py

default_app_config = ‘my_app.apps.MyAppConfig’

2020-03-27