Django queryset monkeypatching для добавления нового атрибута к возвращаемым моделям в объекте QuerySet

Я использую Django 3.2

У меня есть такая модель:

class MyModel(models.Model): last_valid_field_values = models.TextField(help_text='JSON-поле с именами полей и значениями') # ...

Я хочу иметь возможность monkeypatch моих запросов, чтобы при получении экземпляров MyModel, я мог добавить атрибут last_values_dict - который является просто загруженным JSON из last_valid_field_values

Так что у меня будет что-то вроде этого (псевдокод):

def callback_func(instance):
    instance.last_values_dict = json.loads(instance.last_valid_field_values)
    
    
MyModel.objects.all().apply_some_function_to_monkey_patch(callback_func)

Как я могу это сделать? Я также думаю, что возможно это можно сделать через генератор, итерирующий QuerySet?

Вы можете подкласс QuerySet для реализации apply_some_function_to_monkey_patch и переопределить _fetch_all для запуска функции обратного вызова, аналогично тому, как QuerySet.prefetch_related реализовано . Вам также необходимо переопределить __init__ и _clone для работы с фильтрами.

class MyQuerySet(models.QuerySet):
    def __init__(self, model=None, query=None, using=None, hints=None):
        super().__init__(model=model, query=query, using=using, hints=hints)
        self._callback_funcs = ()
        self._callback_done = False

    def apply_some_function_to_monkey_patch(self, *callback_funcs):
        qs = self._chain()
        qs._callback_funcs = qs._callback_funcs + callback_funcs
        return qs

    def _callback(self):
        for callback_func in self._callback_funcs:
            for item in self._result_cache:
                callback_func(item)
        self._callback_done = True

    def _clone(self):
        clone = super()._clone()
        clone._callback_funcs = self._callback_funcs[:]
        return clone

    def _fetch_all(self):
        super()._fetch_all()
        if self._callback_funcs and not self._callback_done:
            self._callback()

Использование:

class MyManager(BaseManager.from_queryset(MyQuerySet)):
    pass


class MyModel(models.Model):
    objects = MyManager()
def callback_func(instance):
    instance.last_values_dict = json.loads(instance.last_valid_field_values)
    
    
MyModel.objects.all().apply_some_function_to_monkey_patch(callback_func)
Вернуться на верх