Django: как получить доступ к оригинальному (неизмененному) экземпляру в сигнале post_save - PullRequest
32 голосов
/ 07 апреля 2011

Я хочу провести денормализацию данных для повышения производительности и поместить сумму голосов, которую мой пост в блоге получает внутри модели поста:

class Post(models.Model):
    """ Blog entry """
    author          = models.ForeignKey(User)
    title           = models.CharField(max_length=255)
    text            = models.TextField()
    rating          = models.IntegerField(default=0) # here is the sum of votes!

class Vote(models.Model):
    """ Vote for blog entry """
    post            = models.ForeignKey(Post)
    voter           = models.ForeignKey(User)
    value           = models.IntegerField()

Конечно, мне нужно сохранить значение Post.rating актуальным. Обычно я бы использовал для этого триггеры базы данных, но теперь я решил подать сигнал post_save (чтобы сократить время обработки базы данных):

# vote was saved
@receiver(post_save, sender=Vote)
def update_post_votes(sender, instance, created, **kwargs):
    """ Update post rating """
    if created:
        instance.post.rating += instance.value
        instance.post.save()
    else:
        # if vote was updated, we need to remove the old vote value and add the new one
        # but how...?

Как получить доступ к значению экземпляра до его сохранения? В триггерах базы данных я бы имел предопределения OLD и NEW для этого, но есть ли что-то подобное в сигналах post_save?

UPDATE

Решение, основанное на ответе Марка:

# vote was saved
@receiver(pre_save, sender=Vote)
def update_post_votes_on_save(sender, instance, **kwargs):
    """ Update post rating """
    # if vote is being updated, then we must remove previous value first
    if instance.id:
        old_vote = Vote.objects.get(pk=instance.id)
        instance.post.rating -= old_vote.value
    # now adding the new vote
    instance.post.rating += instance.value
    instance.post.save()

1 Ответ

48 голосов
/ 07 апреля 2011

Я считаю, post_save слишком поздно, чтобы получить неизмененную версию.Как следует из названия, данные уже были записаны в БД на тот момент.Вы должны использовать pre_save вместо этого.В этом случае вы можете извлечь модель из БД через pk: old = Vote.objects.get(pk=instance.pk) и проверить различия в текущем и предыдущем экземплярах.

...