Как сделать обратный фильтр на основе последней записи в Django

У меня есть таблица клиентов и таблица заказов клиентов, как указано ниже

class Customer(models.Model):
  name = models.CharField()

class CustomerOder(models.Model):
  customer = models.ForiengKey(Customer)
  item = models.CharField()
  datetime_of_purchase = models.DateField()

Образец записи в базе данных

Таблица: Customer

id    name
1     A
2     B
3     C

Таблица: CustomerOrder

id   customer   item     datetime_of_purchase
1    A          item_3   2021-11-10 10:00:00
2    A          item_2   2021-11-11 10:00:00
3    B          item_1   2021-11-11 10:00:00
4    A          item_1   2021-11-12 10:00:00
5    B          item_2   2021-11-12 10:00:00
6    C          item_1   2021-11-13 10:00:00

Предположим, мне нужно отфильтровать клиентов, чья последняя покупка - 'items_1', поэтому выборка после фильтрации должна получить только клиентов 'A' и 'C', чья последняя покупка - 'item_1'

Sample:
<QuerySet [<Customer: A>, <Customer: C>]>

Вы можете получить последний вычисленный элемент с помощью выражения Subquery [Django-doc]:

from django.db.models import OuterRef, Subquery

Customer.objects.alias(
    latest_purchase=Subquery(
        CustomerOrder.objects.filter(
            customer_id=OuterRef('pk')
        ).order_by('-datetime_of_purchase').values('item')[:1]
    )
).filter(latest_purchase='item_1')

Перед , используется .annotate(…) [Django-doc] вместо .alias(…) [Django-doc]:

from django.db.models import OuterRef, Subquery

Customer.objects.annotate(
    latest_purchase=Subquery(
        CustomerOrder.objects.filter(
            customer_id=OuterRef('pk')
        ).order_by('-datetime_of_purchase').values('item')[:1]
    )
).filter(latest_purchase='item_1')
Вернуться на верх