我在模型中定义了以下内容:
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中的一个bug;当通过泛型外键级联删除对象时,Django会忽略传递给GenericForeignKey
字段构造函数的字段名字符串,而是查找content_type
和object_id
字段,在我的情况下,这两个字段并不存在。这有效地将模型可能具有的通用外键的数量限制为1,除非您不会遇到级联删除。
我已经通过Django邮件列表发送了这个问题,因为这种行为可能是故意的。
重命名TaskLink 的字段名
linked_content_type >>> content_type
linked_object_id >>> object_id
或在删除"Location"对象时写入预信号以删除链接对象"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/