我正在尝试为 Django 模型实现通用的软删除模式。
模型被赋予一个 is_deleted 字段,该字段将已删除的对象保留在数据库中,但出于所有实际目的隐藏它们:应遵循级联等的所有正常规则,除了实际删除之外。然而,管理应用程序仍然应该能够处理已删除的对象,以便擦除(肯定将它们扔掉)或恢复它们。 (见下面的代码)
问题:这会破坏级联。我期望发生的事情是通过我在模型和自定义查询集上重写的方法来级联发生。实际发生的情况是,它们被默认的查询集/管理器绕过,而默认的查询集/管理器也恰好使用快速 _raw_delete 内部 API。因此,要么级联删除不会发生,要么如果我在模型上调用 super().delete() 方法(然后调用 save() ),则会对相关对象执行标准删除。
我已经尝试过中建议的内容使用自定义模型删除方法进行级联删除 https://stackoverflow.com/questions/29856011/cascading-delete-w-custom-model-delete-method,但这会严重破坏事情 - 除此之外它提倡使用已弃用的 use_for_lated_fields 管理器属性。
我开始认为,如果不影响 Django 私有的主要部分,我想要实现的目标是不可能的 - 很奇怪,因为这种软删除行为是许多 DBMS 情况下的标准模式。
这就是我现在所处的位置:
-
我为带有 is_deleted 字段的对象创建了一个自定义管理器和查询集:
from django.db import models
from django.db.models.query import QuerySet
class SoftDeleteQuerySet(QuerySet):
#https://stackoverflow.com/questions/28896237/override-djangos-model-delete-method-for-bulk-deletion
def __init__(self,*args,**kwargs):
return super(self.__class__,self).__init__(*args,**kwargs)
def delete(self,*args,**kwargs):
for obj in self: obj.delete()
#http://codespatter.com/2009/07/01/django-model-manager-soft-delete-how-to-customize-admin/
# but use get_queryset, not get_query_set !!!
class SoftDeleteManager(models.Manager):
""" Use this manager to get objects that have a is_deleted field """
def get_queryset(self,*args,**kwargs):
return SoftDeleteQuerySet(model=self.model, using=self._db, hints=self._hints).filter(is_deleted=False)
def all_with_deleted(self,*args,**kwargs):
return SoftDeleteQuerySet(model=self.model, using=self._db, hints=self._hints).filter()
def deleted_set(self,*args,**kwargs):
return SoftDeleteQuerySet(model=self.model, using=self._db, hints=self._hints).filter(is_deleted=True)
def get(self, *args, **kwargs):
""" if a specific record was requested, return it even if it's deleted """
return self.all_with_deleted().get(*args, **kwargs)
def filter(self, *args, **kwargs):
""" if pk was specified as a kwarg, return even if it's deleted """
if 'pk' in kwargs:
return self.all_with_deleted().filter(*args, **kwargs)
return self.get_queryset().filter(*args, **kwargs)
-
添加了一个基本模型来使用它:
class SoftDeleteModel(models.Model):
objects=SoftDeleteManager()
is_deleted = models.BooleanField(default=False, verbose_name="Is Deleted")
def delete(self,*args,**kwargs):
if self.is_deleted : return
self.is_deleted=True
self.save()
def erase(self,*args,**kwargs):
"""
Actually delete from database.
"""
super(SoftDeleteModel,self).delete(*args,**kwargs)
def restore(self,*args,**kwargs):
if not self.deleted: return
self.is_deleted=False
self.save()
def __unicode__(self): return "%r %s of %s"%(self.__class__,str(self.id))
class Meta:
abstract = True
-
以及管理类来处理擦除、恢复等:
# for definitive deletion of models in admin
def erase_model(modeladmin,request,queryset):
"""
Completely remove models from db
"""
for obj in queryset:
obj.erase(user=request.user)
def restore_model(modeladmin,request,queryset):
"""
Restore a softdeletd model set
"""
for obj in queryset:
obj.restore(user=request.user)
#http://codespatter.com/2009/07/01/django-model-manager-soft-delete-how-to-customize-admin/
# but the method is now get_queryset.
class SoftDeleteAdmin(admin.ModelAdmin):
list_display = ('pk', '__unicode__', 'is_deleted',)
list_filter = ('is_deleted',)
actions=[erase_model, restore_model]
def get_queryset(self, request):
""" Returns a QuerySet of all model instances that can be edited by the
admin site. This is used by changelist_view. """
# Default: qs = self.model._default_manager.get_query_set()
qs = self.model._default_manager.all_with_deleted()
#TR()
# TODO: this should be handled by some parameter to the ChangeList.
ordering = self.ordering or () # otherwise we might try to *None, which is bad ;)
if ordering:
qs = qs.order_by(*ordering)
return qs
queryset=get_queryset
Ideas?
EDIT:所有这一切的要点(除了更彻底地搜索打包解决方案:-))是可以覆盖删除并使其正确,但这对于胆小的人来说并不容易。我将使用的包 - django-softdelete,我的起点的演变,从http://codespatter.com/2009/07/01/django-model-manager-soft-delete-how-to-customize-admin/ http://codespatter.com/2009/07/01/django-model-manager-soft-delete-how-to-customize-admin/- 使用通过 Contenttype API 计算的 ChangeSet。
除此之外,有几种情况根本不会调用重写的delete()(基本上,每次发生组删除时,django都会采用跳过model.delete()头部的快捷方式)。
在我看来,这是一个设计错误。如果覆盖它需要如此大量的大脑爆炸,则 model.delete() 实际上应该是 model._delete()。
Django 软删除 https://github.com/scoursen/django-softdelete是一个为 Django 实现软删除的库,具有级联功能。它还存储变更集,并允许恢复删除(例如恢复整个级联)。
我不确定它的维护状态和质量如何,但它至少可以作为灵感,即使它本身不是解决方案。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)