使用 GenericForeignKey 级联删除模型而不使用 GenericRelation

问题描述

我正在创建一个可重复使用的 django 应用程序,其中包含一个带有 GenericForeignKey 的模型,我需要将其级联删除

此模型可以连接到任何其他模型。我无法控制目标模型类,因为它在应用程序之外。这意味着我无法向其中添加 GenericRelation 字段,也无法强制用户添加它,因为目标可能位于其他第三方应用中。

假设我们有这样的模型(无法控制 PostPostGroup):

class Tag(models.Model):
    content_type = models.ForeignKey(ContentType,on_delete=models.CASCADE)
    object_id = models.PositiveIntegerField()
    object = GenericForeignKey()

class PostGroup(models.Model):
    title = models.CharField(max_length=255)

class Post(models.Model):
    title = models.CharField(max_length=255)
    group = models.ForeignKey(PostGroup,on_delete=models.CASCADE)

如果 Tag 查询集被删除,有没有办法删除 PostGroup

例如不仅post_group.delete(),还有PostGroup.objects.delete()

解决方法

您可以使用 pre_delete 信号来实现此目的:

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


@receiver(pre_delete) # add relevant sender to signal (not mandatory)
def post_group_deleted(sender,instance,using,**kwargs):
    # Query tags with the instance of PostGroup and delete them
    if isinstance(instance,Tag):
        return

    Tag.objects.filter(
        content_type=ContentType.objects.get_for_model(instance),object_id=instance.pk
    ).delete()

查看文档 here

与 ForeignKey 不同,GenericForeignKey 不接受 on_delete 参数来自定义此行为;如果需要,您可以通过不使用 GenericRelation 来避免级联删除,并且可以通过 pre_delete 信号提供替代行为。