Я пытаюсь получить множество моделей для обновления, когда я сохраняю связанную модель. Это должно быть возможно с помощью m2m_changed signal (и это работает! Но не в админ?), Например.Django admin save не отправляет post_remove действие с m2m_changed signal
# i want the references field to update when related model is saved.
# so just call count_references
class Tag(models.Model):
"""Group everything into categories"""
# stuff stuff stuff
references = models.IntegerField(default=0, editable=False)
def count_references(self):
# just add up references each time to save headaches
self.references = 0
# search for reverse managers
sets = re.compile('^\w+_set$')
for rel_set in [method for method in dir(self) if sets.match(method)]:
self.references += getattr(self, rel_set).count()
self.save()
class Entry(models.Model):
"""Blog entry"""
# stuff stuff stuff
tags = models.ManyToManyField('Tag', blank=True)
# this will call count_references when entry adds or removes tags
@receiver(m2m_changed, sender=Entry.tags.through)
def update_tag_ref_count(sender, instance, action, reverse, model, pk_set, **kwargs):
print action
if not reverse and action == 'post_add' or action == 'post_remove':
for tag_pk in pk_set:
print tag_pk
Tag.objects.get(pk=tag_pk).count_references()
print Tag.objects.get(pk=tag_pk).references
Все работает отлично при запуске в оболочке. например с tests.py как так:
t = Tag.objects.all()[0]
s = Snippet.objects.all()[0]
s.tags.remove(t)
s.save()
s.tags.add(t)
s.save()
я получаю следующее (где 'тест' имя тега печатается):
pre_remove
post_remove
test
0
pre_add
post_add
test
1
отлично! И когда я добавляю тег к записи в admin, я получаю следующее (между HTTP-файлом):
pre_clear
post_clear
pre_add
post_add
test
1
все еще хорошо! не знаю, для чего был вызван pre/post_clear ... и когда я удаляю:
pre_clear
post_clear
argh! pre/post_remove не вызывается! pre/post_clear бесполезен, а также не содержит никаких первичных ключей. это похоже на ошибку в реализации администратора. какие-либо предложения?
Обновление: Bug #16073 подано и принято.
Пожалуйста, ответьте на ваш вопрос. – jpic
Ошибка была отмечена как дубликат [# 6707] (https://code.djangoproject.com/ticket/6707), так как это признак исходной проблемы. – rbanffy