Django order by aggregate value from non-related table

У меня есть три модели, Accrual и Member, и общим полем этих двух моделей является register_no, но это поле не является внешним ключом

class Accrual(models.Model):
  register_no = models.PositiveIntegerField(verbose_name=_('Register No'))
  amount=models.DecimalField(decimal_places=2, max_digits=17, verbose_name=_('Total Amount'))

class Member(models.Model):
    register_no = models.PositiveIntegerField(unique=True, verbose_name=_('Register No'))

class Driver(models.Model):
    register_no = models.PositiveIntegerField(unique=True, verbose_name=_('Register No'))

Я хочу перечислить долг, который есть у каждого члена/водителя. Это можно сделать с помощью @property;

class Member(models.Model):
    register_no = models.PositiveIntegerField(unique=True, verbose_name=_('Register No'))

    @property
    def debt(self):
        ret_val = Accrual.objects.filter(register_no=self.register_no).aggregate(
            debt=Sum('amount'))

        debt = ret_val.get('debt', 0)
        return debt if debt else 0

Но я не могу использовать order_by таким образом. Я хочу отсортировать каждого члена по долгу. Как я могу решить эту проблему?

Вы можете использовать подзапрос для добавления всех связанных начислений на основе register_no:

from django.db.models import OuterRef, Subquery, Sum


accruals = Accrual.objects.filter(
  register_no=OuterRef('register_no')
).values('register_no').annotate(debt=Sum('amount'))

Member.objects.annotate(
  debt=Subquery(accruals.values('debt'))
).order_by('debt')

что приводит к такому sql:

SELECT "id", "register_no", (
    SELECT SUM(U0."amount") AS "debt" 
    FROM "accrual" U0 
    WHERE U0."register_no" = ("register_no") 
    GROUP BY U0."register_no"
) AS "debt" 
FROM "member" 
ORDER BY "debt" ASC
Вернуться на верх