7

在我的 Tastypie 资源中,我正在注释我的查询集,但我没有看到该注释流向 JSON Tastypie 生成并传回。代码很简单:

class CompetitionResource(ModelResource):
    total_tickets = fields.IntegerField(readonly=True)

    class Meta:
        queryset = Competition.objects.all().annotate(total_tickets=Count('ticket__ticketownership__user__id', distinct=True))

我在查询集中生成和注释的那个计数根本不会出现在最终的 JSON 中。最终的 JSON 有一个 total_users 字段(因为我在 ModelResource 中声明了一个),但它为空。我是否遗漏了任何明显的东西来确保通过这样的注释?如果没有,有什么方法可以解决这个问题?

一种方法是在我的模型中创建一个属性,然后将我的 ModelResource 中的 total_users 字段绑定到该属性。但这可能会导致我从数据库中提取的每个竞赛的计数查询,这并不好。我想在一个注释类型的查询中做到这一点。

4

3 回答 3

5

好,我知道了。您可以简单地使用可以添加到 ModelResource 的自定义 dehydrate_[field name] 方法。对于每个 ModelResource 字段,Tastypie 会检查您是否指定了 dehydrate_[field name] 方法,如果您指定了,它会在将对象处理成包时调用该方法(然后以 JSON 或 XML 或其他形式输出)。这个 dehydrate_[field name] 方法获取 Tastypie 在此之前为该特定对象创建的包。好消息是这个bundle中有原始对象,在bundle.obj下。并且该对象仍将具有您在 get_object_list 中提供的原始注释(如上面的答案所示)。所以你可以使用下面的代码。

class CompetitionResource(ModelResource):
    total_tickets = fields.IntegerField(readonly=True)

    class Meta:
        queryset = Competition.objects.all()

    def get_object_list(self, request):
        return super(CompetitionResource, self).get_object_list(request).annotate(total_tickets=Count('ticket__ticketownership__user__id', distinct=True))

    def dehydrate_total_tickets(self, bundle):
        return bundle.obj.total_tickets

无论您从自定义 dehydrate_[field name] 方法返回什么,都将作为该字段的最终值正确存储在该对象的包中,然后正确处理为输出。

于 2012-09-18T14:02:25.617 回答
0

它没有出现在文档中,但是查看源代码有一个属性参数可以传递给字段声明,该字段声明可用于将其绑定到模型实例的属性。

可选地接受一个attribute,它应该是一个实例属性的字符串,或者 dehydratehydrate. 默认为None,表示将手动访问数据。

因此,对于您的示例,以下内容应该可以解决问题。

class CompetitionResource(ModelResource):
    total_tickets = fields.IntegerField(readonly=True, attribute='total_tickets')

    class Meta:
        queryset = Competition.objects.all().annotate(total_tickets=Count('ticket__ticketownership__user__id', distinct=True))

脱水解决方案适用于使用值填充发送的对象,但不允许您轻松利用 Tastypie 的一些其他功能,例如内置过滤(我相信排序)。使用带有属性参数的字段定义可以。

于 2018-07-12T16:58:16.097 回答
-1

我认为您的问题可能与 Tastypie 文档中给出的警告有关queryset

如果您在其中放置任何可调用对象,它们只会被评估一次(当 Meta 类被实例化时)。这尤其会影响与日期/时间相关的事物。请参阅 :ref:cookbook 以了解解决此问题的方法。

查看食谱中的相关部分,我认为您应该尝试以下操作:

class CompetitionResource(ModelResource):
    total_users = fields.IntegerField(readonly=True)

    class Meta:
        queryset = Competition.objects.all()

    def get_object_list(self, request):
        return super(CompetitionResource, self).get_object_list(request).annotate(total_tickets=Count('ticket__ticketownership__user__id', distinct=True))
于 2012-09-17T18:22:39.647 回答