Django 按最新相关对象过滤

Nat*_*ler 3 django foreign-keys filter django-admin django-queryset

我已经设置了一个系统来跟踪数据库中各种对象的审核任务。它们通过通用外键关系链接。给定一个 ObjectModerationState 对象,我必须根据最新的 StateTransisition 对象确定其状态。

该解决方案必须能够在 SimpleListFilter 中使用,以便使用 list_filter 对 list_display 进行排序。

例如给定状态 - 排队 - 已完成 - 验证

我应该能够根据 StateTransisition 对象中的最新状态值过滤 ObjectModerationState 对象。

我一直倾向于在 ObjectModerationState 上进行某种注释。

这是我所拥有的不起作用的东西。

模型.py:

class ObjectModerationState(models.Model):
    job = models.ForeignKey(JobDescription)
    object_id = models.TextField(help_text="Primary key of the model under moderation.")
    content_type = models.ForeignKey(ContentType, help_text="Content type of the model under moderation.")
    object = generic.GenericForeignKey()

class StateTransisition(models.Model):
    state = models.ForeignKey(State)
    moderation_details = models.ForeignKey("ObjectModerationState", related_name='states')
    changed_by = models.ForeignKey("auth.User", related_name='+', null=False,
                                   help_text="If you create the task, then usually this will be you.")
    date_updated = models.DateTimeField(_("Date Set"))
    last_modified = models.DateTimeField(auto_now=True)


    def __unicode__(self):
        return "%s in state %s by %s" % (self.moderation_details.object, self.state, self.changed_by)

    class Meta:
        ordering = ['last_modified', ]
        get_latest_by = 'last_modified'

class State(Orderable):
    label = models.CharField(_("State Label"), max_length=100, unique=True)
    description = models.TextField(_("Description"), max_length=200, null=True, blank=True,
                                   help_text=_("Explination of this state in context"))
    valid_transitions = models.ManyToManyField("State", null=True, blank=True,
                                               help_text=_("You must add these mappings."))

    def __unicode__(self):
        return "%s" % (self.label)

    class Meta:
        abstract = False
Run Code Online (Sandbox Code Playgroud)

管理员.py

class CurrentStateFilter(SimpleListFilter):
    title = 'current state'
    parameter_name = 'current state'

    def lookups(self, request, model_admin):
        states = State.objects.all()
        return  [(c.id, c) for c in states]

    def queryset(self, request, queryset):
        if self.value():
            queryset.filter(states__state__id=self.value()) <<< ????
        else:
            return queryset
Run Code Online (Sandbox Code Playgroud)

编辑

TGO*_*TGO 5

对于这种情况,缓存绝对是一个好方法。但是,为了完整起见,我想表明我已经能够使用以下 Django ORM 查询做同样的事情:

EventGroup.objects.annotate(
    latest_event_date=Max('events__timestamp')
).filter(
    events__timestamp=F('latest_event_date'),
    events__state=EventState.STATE_PROCESSED,
)
Run Code Online (Sandbox Code Playgroud)

因此,对于每个EventGrouplatest_event_date被注释timestamp为该组事件的最大值。此外,我进行过滤以仅包括EventGroup其最新事件状态为的s EventState.STATE_PROCESSED

我相当肯定这样的事情在你的情况下会起作用。


Nat*_*ler 1

经过大量研究后,我不确定是否可以在不非常hacky的情况下完成它。我最终将值直接缓存在列表显示中显示的模型上。我使用保存后信号来做到这一点。它运作得很好。