#python #django #django-models #django-queryset
Вопрос:
У меня есть три модели: Бренд, журнал и абстрактная модель MWModel.
class Log(models.Model):
app = models.CharField(max_length=20, choices=get_modules())
content_type = models.ForeignKey(ContentType, on_delete=models.CASCADE)
object_id = models.PositiveIntegerField()
content_object = GenericForeignKey('content_type', 'object_id')
method = models.CharField(max_length=8, choices=HTTP_METHODS)
url = models.TextField()
body = models.TextField(null=True)
code = models.IntegerField()
message = models.TextField(null=True)
created_at = models.DateTimeField(auto_now_add=True)
def __str__(self):
return self.created_at.isoformat()
class MWModel(models.Model):
log = GenericRelation(Log)
created_at = models.DateTimeField(auto_now_add=True)
modified_at = models.DateTimeField(auto_now=True)
class Meta:
abstract = True
class Brand(MWModel):
name = models.CharField(max_length=50)
status = models.BooleanField(default=True)
def __str__(self):
return self.name
Как получить поле кода последней записи журнала, созданной для каждого бренда?
У меня есть это
Brand.objects.filter(
merchant=account.merchant,
identity__app=F('merchant__erp')
).annotate(
log__last__created_at=Max('log__created_at'),
log__last__code=Log.objects.filter(
content_type__model=Brand._meta.model_name,
object_id=OuterRef('pk')
).order_by('-created_at').values('code')[:1]
).values('pk', 'name', 'log__last__created_at', 'log__last__code')
И мне нужно что-то вроде этого
Brand.objects.filter(
merchant=account.merchant,
identity__app=F('merchant__erp')
).annotate(
log__last__created_at=Max('log__created_at'),
log__last__code=F('log__code', filter=Q('log__created_at'=F('log__last__created_at')))
).values('pk', 'name', 'log__last__created_at', 'log__last__code')
Но я не нашел, как закодировать что-то подобное.
Я был бы очень признателен вам за помощь. Спасибо.
Ответ №1:
Это может быть достигнуто с помощью подзапроса. Вы были действительно близки, просто не хватало выражения подзапроса.
Brand.objects.annotate(
log__last__code=Subquery(
Log.objects.filter(
content_type__model=Brand._meta.model_name,
object_id=OuterRef('pk')
).order_by('-created_at').values('code')[:1]
)
)
Комментарии:
1. Здравствуйте, Александр, без подзапроса аннотация также работает для меня. Мне действительно нужно заменить подзапрос выражениями в аннотированном коде журнала__last__ . Итак, я получаю
log__last__created_at=Max('log__created_at')
и мне нужно что-то подобное, чтобы получить лог__последний__код2. Это никоим образом не работает без выражения подзапроса… Вы хотите
get code field of last Log record created for each Brand?
или хотите заменить подзапрос чем-то другим? Я давал решение для первого вопроса.3. Подожди, что? Он работает без подзапроса, но это странно. Тогда в чем проблема?
4. Я думаю, что мой вопрос неясен, мне нужно выражение, которое позволяет мне получить код последнего журнала без использования подзапроса. Что-то похожее на то, как я получаю журнал__последний__созданный_по адресу .
5. Я не думаю, что это возможно, так как вы не можете
Max
code
пройти мимоcreated_at
. Так что этот трюк работает только тогда, когда вам нужно максимальное/минимальное значение одного и того же столбца. Однако, в чем ваша проблема с подзапросом? Для меня это звучит как проблема X/Y