为什么Django模型信号不起作用



我正在尝试根据用户的状态创建用户的活动流。

型号:

class Status(models.Model):
    body = models.TextField(max_length=200)
    image = models.ImageField(blank=True, null=True, upload_to=get_upload_file_name)
    privacy = models.CharField(max_length=1,choices=PRIVACY, default='F')
    pub_date = models.DateTimeField(auto_now_add=True, auto_now=False)
    user = models.ForeignKey(User)
class Activity(models.Model):
    actor = models.ForeignKey(User)
    action = models.CharField(max_length=100)
    content_type = models.ForeignKey(ContentType)
    object_id = models.PositiveIntegerField()
    content_object = generic.GenericForeignKey('content_type', 'object_id')
    pub_date = models.DateTimeField(auto_now_add=True, auto_now=False)

然而,尽管我创建了一个新的状态,但它不会根据post_save信号创建新的活动。

信号:

from django.contrib.contenttypes.models import ContentType
from django.db.models.signals import post_save
from status.models import Status
from models import Activity
def create_activity_item(sender, instance, signal, *args, **kwargs):
    if kwargs.get('created', True):
        ctype = ContentType.objects.get_for_model(instance)
        if ctype.name == 'Status':
            action = ' shared '
            activity = Activity.objects.get_or_create(
                actor = instance.user,
                action = action,
                content_type = ctype,
                object_id = instance.id,
                pub_date = instance.pubdate
            )
post_save.connect(create_activity_item, sender=Status)

我做错了什么?请帮我解决这个问题。我将非常感激。非常感谢。

更新:

然而,这样做会产生活动:

@receiver(post_save, sender=Status)
def create(sender, instance, **kwargs):
    if kwargs.get('created',True):
        ctype = ContentType.objects.get_for_model(instance)
        activity = Activity.objects.get_or_create(
            actor = instance.user,
            action = ' shared ',
            content_type = ctype,
            object_id = instance.id,
            pub_date = instance.pub_date
        )

那么,为什么上述方法不起作用呢?

似乎您的post_save.connect没有执行。您应该将signals导入某个位置。对于django 1.7,建议在应用程序的config ready((函数中执行此操作。阅读文档中的"这个代码应该在哪里?"旁注。

例如,如果您的应用程序名为activity:

活动/__init__.py

default_app_config = 'activity.apps.ActivityAppConfig'

活动/apps.py

from django.apps import AppConfig
class ActivityAppConfig(AppConfig):
    name = 'activity'
    def ready(self):
        import activity.signals

别忘了将dispatch_uid添加到您的connect()呼叫中:

post_save.connect(create_activity_item, sender=Status,
                  dispatch_uid="create_activity_item")

UPDATEContentTypename属性总是小写。因此,您应该将if语句更改为:

if ctype.name == 'status':

如果在signals.py中所有内容都写得正确但不起作用,请检查以下步骤。。。(假设在名为AppName的应用程序中(

  1. __init__.py中,放置行

    default_app_config = 'AppName.apps.AppnameConfig'
    
  2. apps.py文件中,放入块

    from django.apps import AppConfig
    
    class AppnameConfig(AppConfig):
        name = 'AppName'
        def ready(self):
            import AppName.signals
    

假设您的应用程序名称为blog,在项目的settings.py文件中,确保在主项目的settings.py文件的INSTALLED_app变量中将blog应用程序注册为blog.apps.BlogConfig,而不仅仅是blog。这对我很有效。

在不接触应用程序的情况下。py这对我有效。

class MyModel(models.Model):
    """ MyModel fields go """
    body = models.TextField(max_length=200)
    pub_date = models.DateTimeField(auto_now_add=True, auto_now=False)

def post_save_actions(sender, instance, created, **kwargs):
    if created:
        pass
        # post save actions if new instance is created,
        # do something with `instance` or another models
        # be careful about circular imports. m/

和信号挂钩,

post_save.connect(post_save_user_actions, sender=MyModel)

简单、可扩展、可重复使用的答案是…

如果您计划在应用程序(例如:posts(中使用信号(例如:signals.py(,只需养成每次将此方法添加到apps.py AppConfig的习惯。

def ready(self):
    from . import signals

你不需要像别人说的那样触摸__init__.py。然后你的信号就会起作用。

您应该添加

default_app_config='activity.apps.ActivityAppConfig'

__init__.py文件中

最新更新