【问题标题】:Why does Django post_save signal give me pre_save data?为什么 Django post_save 信号会给我 pre_save 数据?
【发布时间】:2010-11-16 08:26:26
【问题描述】:

我正在尝试将“信息”对象连接到许多“客户”(参见下面的代码)

当一个信息对象更新时,我想向每个连接到该信息的客户发送电子邮件。

但是,当我记录信号接收到的 sold_to 字段时,我总是会得到保存前的数据。

我猜这是因为它的 ManyToManyField 和数据存储在单独的表中,但不应该在所有关系都更新后调用 post_save 信号吗?

有人有解决方案的建议吗?

class Customer
    name = models.CharField(max_length=200)
    category = models.ManyToManyField('Category',symmetrical=False)
    contact = models.EmailField()

class Information
    name = models.CharField(max_length=200)
    email = models.EmailField(max_length=200)
    mod_date = models.DateTimeField(auto_now=True)
    sold_to = models.ManyToManyField(Customer, null=True, blank=True)


def send_admin_email(sender, instance, signal, *args, **kwargs):
    from myapp import settings
    for cust in instance.sold_to.all():
        settings.debug(cust.name)

post_save.connect(send_admin_email, sender=Information)

编辑:#django 中的 apollo13 提醒我注意这一点: “相关项(保存到多对多关系中的事物) 正如您所发现的那样,不会作为模型保存方法的一部分保存。” - http://groups.google.com/group/django-users/msg/2b734c153537f970

但是从 2006 年 7 月 9 日开始,我真的很希望有一个解决方案。

【问题讨论】:

    标签: python django django-signals


    【解决方案1】:

    对于您面临的问题here,有一张未解决的票。您可以密切关注它何时发布,或者您可以尝试应用它提供的补丁,看看是否有帮助。

    【讨论】:

    • 感谢您的快速回复。我尝试了这个补丁,经过一点黑客攻击后,我让它工作了。将在下面发布解决方案。
    【解决方案2】:

    这是我的解决方案,在应用上述 code.djangoproject.com 的补丁后。

    在models.py中添加了这个:

    from django.db.models.signals import m2m_changed
    m2m_changed.connect(send_admin_email, sender=Information)
    

    还有 send_admin_email 功能:

    def send_customer_email(sender, instance, action, model, field_name, reverse, objects, **kwargs):
        if ("add" == action):
            # do stuff
    

    【讨论】:

      【解决方案3】:

      我遇到了同样的问题,因为我的模型中有 M2M 字段,我也得到了类似 pre_save 的数据。

      在这种情况下,问题是在 M2M 字段中,两个相关模型都应该保存,以便获取自动生成的 ID。

      在我的解决方案中,我既没有使用 post_save 信号,也没有使用 m2m_changed 信号,而是在 ModelAdmin 类定义中使用了 log_addition 和 log_change 方法。

      在您的自定义 ModelAdmin 类中:

          class CustomModelAdmin(admin.ModelAdmin):
               def log_addition(self, request, object):
               """
               Log that an object has been successfully added.
               """
                   super(CustomModelAdmin, self).log_addition(request, object)
                   #call post_save callback here object created
      
               def log_change(self, request, object):
               """
               Log that an object has been successfully changed.
               """
                   super(CustomModelAdmin, self).log_change(request, object)
                   #call post_save callback here object changed
      

      如果你愿意,你也可以重写 log_deletion() 方法。

      愉快的覆盖...

      【讨论】:

      • 我喜欢这个解决方案。在 Django 3 中,log_addition、log_change 采用 3 个位置参数,例如 log_addition(request, object, message) 或 log_change(request, object, message)。省略第三个参数会引发错误
      猜你喜欢
      • 1970-01-01
      • 2018-08-09
      • 2017-04-26
      • 2021-04-09
      • 2014-03-16
      • 2011-05-30
      • 2013-07-14
      • 1970-01-01
      • 2011-12-22
      相关资源
      最近更新 更多