Django无法级联删除相关的通用外键对象

时间:2016-05-05 15:01:18

标签: django cascade django-class-based-views generic-foreign-key

我的模型中定义了以下内容:

class TaskLink(models.Model):
    task = model.ForeignKey(Task)
    link_choices = (
        models.Q(app_label="accounts", model="location"),
        # Other models are also linked to here.
    )
    linked_content_type = \
        models.ForeignKey(
            ContentType,
            limit_choices_to=link_choices
        )
    linked_object_id = models.PositiveIntegerField()
    linked_object = \
        generic.GenericForeignKey(
            'linked_object_content_type',
            'linked_object_id'
        )

此模型将Task个对象与link_choices元组中的任何模型相关联。在这种情况下,accounts.Location模型位于此列表中。

当删除Location对象导致相关TaskLink对象的级联删除时,我的问题就出现了。删除失败,并显示以下错误消息:

django.core.exceptions.FieldError: Cannot resolve keyword 'object_id' into field. Choices are: id, linked_object, linked_object_content_type, linked_object_content_type_id, linked_object_id, task, task_id

视图是django.views.generic.DeleteView的一个实例,只有pk_url_kwarg参数和模型集(以及权限装饰器添加到调度方法中);在我将TaskLink模型添加到混合之前,它工作了linked_object_fine。

我错过了什么?

编辑:看起来这可能是Django中的一个错误;当通过通用外键级联删除对象时,Django会忽略您传递给GenericForeignKey字段的构造函数的字段名称字符串,而是查找content_typeobject_id字段,我的情况,不存在。这有效地限制了模型可能具有的通用外键的数量1,除非您不会遇到级联删除。

我已经通过Django邮件列表发送了这个问题,因为这种行为可能是故意的。

1 个答案:

答案 0 :(得分:3)

重命名TaskLink的字段名称

linked_content_type >>> content_type
linked_object_id >>> object_id

或在删除"位置"时写入前置信号删除链接对象的对象" TaskLink"

from django.db.models.signals import pre_delete
from django.dispatch import receiver

@receiver(pre_delete, sender=Location, dispatch_uid='location_delete_signal')
def deleted_gfk_TaskLink(sender, instance, using, **kwargs):
    ctype = ContentType.objects.get_for_model(sender)
    obj = TaskLink.objects.get(linked_content_type=ctype, linked_object_id=instance.id)
    obj.delete()

自定义信号的参考:
https://micropyramid.com/blog/using-djangos-built-in-signals-and-writing-custom-signals/