Как расширить менеджерский метод в Django? - PullRequest
0 голосов
/ 07 апреля 2019

Я хотел бы расширить метод _filter_or_exclude на моем менеджере, но по какой-то причине метод никогда не вызывается.

Чего я хочу добиться, так это неявно фильтровать модели, у которых нет удаленного поля =0 (со всеми get() и filter() и all() и exclude() и, возможно, другими методами)

Мои модели

from django.db.models import Model, Manager, Q, QuerySet


class BaseManager(Manager):

    def _filter_or_exclude(self, *args, get_deleted=False,  **kwargs):
        clone = super()._filter_or_exclude(*args, **kwargs)
        if not get_deleted:
            clone.query.add(Q(deleted=0))
        return clone

# Just for sure, I extended the method on a querySet as well
class BaseQuerySet(QuerySet):

    def _filter_or_exclude(self, *args, get_deleted=False, **kwargs):
        clone = super()._filter_or_exclude(*args, **kwargs)
        if not get_deleted:
            clone.query.add(Q(deleted=0))
        return clone


class BaseModel(Model):
    id = models.UUIDField(primary_key=True, default=uuid.uuid4, editable=False)
    deleted = models.IntegerField(default=0)

    ...

    objects = BaseManager.from_queryset(BaseQuerySet)

    class Meta:
        abstract = True

    def delete(self, from_db=False, using=None, keep_parents=False):    
        if from_db:
            super().delete(using, keep_parents)
            return
        self.deleted = 1
        self.save()

class Operation(BaseModel):

    ...

Теперь, когда я запускаю

Operation.objects.get(id="...")

Расширенный метод _filter_or_exclude вызывается не в пользовательском менеджере, а в стандартном.

Что я делаю не так?

1 Ответ

1 голос
/ 07 апреля 2019

Вы можете переопределить get_queryset метод для менеджера.И чтобы добавить все экземпляры, включая удаленные, вы можете добавить другой метод.вот так.

class BaseManager(Manager):
        def get_queryset(self):
            qs = super().get_queryset()
            qs = qs.filter(deleted=1) # or whatever your condition is
            return qs

    def all_objects(self):
        return super().get_queryset()

И используйте это вот так.

Model.objects.get() # filtered results
Model.objects.all_objects.get() # include deleted ones.
...