【问题标题】:Using custom methods in filter with django-rest-framework在带有 django-rest-framework 的过滤器中使用自定义方法
【发布时间】:2014-08-16 09:15:30
【问题描述】:

我想在我的 REST API - see django docs on this 中过滤查询参数。 但是,我希望过滤的一个参数只能通过模型​​@property 获得

示例models.py:

class Listing(models.Model):
    product = models.OneToOneField(Product, related_name='listing')
    ...
    @property
    def category(self):
        return self.product.assets[0].category.name

这是根据django-filter docs 为我的Listing API 设置的

    class ListingFilter(django_filters.FilterSet):
        product = django_filters.CharFilter(name='product__name')
        category = django_filters.CharFilter(name='category') #DOES NOT WORK!!

        class Meta:
            model = Listing
            fields = ['product','category']

    class ListingList(generics.ListCreateAPIView):
        queryset = Listing.objects.all()
        serializer_class = ListingSerializer
        filter_class = ListingFilter

如何按listing.category 适当过滤?它不能直接在列表模型上使用。

【问题讨论】:

标签: python django django-rest-framework django-filter


【解决方案1】:

使用 'action' 参数指定自定义方法 - see django-filter docs

首先定义一个使用类别参数值过滤查询集的方法:

    def filter_category(queryset, value):
        if not value:
            return queryset

        queryset = ...custom filtering on queryset using 'value'...
        return queryset

列表过滤器应如下所示:

    class ListingFilter(django_filters.FilterSet):
        ...
        category = django_filters.CharFilter(action=filter_category)
        ...

【讨论】:

【解决方案2】:

为了数据库速度,您应该只将类别添加到您的列表模型中

class Listing(models.Model):
    product = models.OneToOneField(Product, related_name='listing')
    category = models.ForeignKey(Category)

然后使用post_save signal 保持字段更新

from django.dispatch import receiver
from django.db.models.signals import post_save

@receiver(post_save, sender=Product)
def updateCategory(sender, instance, created, update_fields, **kwargs):
    product = instance
    product.listing.category = product.assets[0].category.name
    product.listing.save()

然后像过滤任何其他字段一样按其名称过滤:

class ListingFilter(django_filters.FilterSet):
    ...
    category = django_filters.CharFilter(name='category__name')
    ...

【讨论】:

    猜你喜欢
    • 2018-01-20
    • 2017-06-13
    • 2022-11-18
    • 2020-07-08
    • 2021-06-21
    • 2021-08-10
    • 2018-02-23
    • 2019-05-15
    • 2021-07-16
    相关资源
    最近更新 更多