【问题标题】:django-rest-framework- Filtering using 'or' on multiple values from one url parameterdjango-rest-framework- 使用“或”过滤来自一个 url 参数的多个值
【发布时间】:2020-01-10 19:14:24
【问题描述】:

我为我的 API 公开的模型设置了一个标记系统。模型看起来像这样:

class TaggableModel(models.Model):
    name = models.CharField(max_length=255)
    tags = models.ManyToManyField(Tag, related_name="taggable_models")

class Tag(models.Model):
    tag = models.CharField(max_length=32)

然后我设置了一个序列化程序和视图,如下所示:

class TaggableModelSerializer(serializers.ModelSerializer):
    class Meta:
        model = TaggableModel
        fields = ('id', 'name', 'tags',)
        read_only_fields = ('id',)

class TaggableModelViewSet(viewsets.ModelViewSet):
    queryset = TaggableModel.objects.all()
    serializer_class = TaggableModelSerializer
    permission_classes = (AllowAny,)
    filter_backend = [DjangoFilterBackend]
    filterset_fields = ['tags']

如果我想获取标签 ID 为 1、2 或 3 的所有 TaggableModels,我可以通过以下方式进行:

https://my-api-domain/api/taggable-models?tags=1&tags=2&tags=3

有没有办法在分隔符上拆分,所以我可以将它全部作为一个参数传递?例如:

https://my-api-domain/api/taggable-models?tags=1,2,3

看起来我可以编写自己的自定义 DjangoFilterBackend 过滤器,但我有点不确定从哪里开始。或者也许有更简单的方法可以做到这一点?

【问题讨论】:

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


    【解决方案1】:

    当然,您可以通过具有特定字段“小部件”的自定义过滤器集类来做到这一点(这就是它在 django-filters 中的调用方式)

    您可以尝试以下示例:

    # filters.py
    
    from django_filters.rest_framework import FilterSet, filters
    from django_filters.widgets import CSVWidget
    
    from .your_models import Tag, TaggableModel
    
    class TaggableModelFilterSet(FilterSet):
        tags = filters.ModelMultipleChoiceFilter(
            queryset=Tag.objects.all(), widget=CSVWidget,
            help_text=_("A list of ids, comma separated, identifying tags"),
            method='filter_tags'
        )
    
        class Meta:
            model = TaggableModel
            fields = ['tags']
    
        def filter_tags(self, queryset, name, value):
            if value:
                queryset = queryset.filter(tags__in=value)
            return queryset
    
    # views.py
    
    class TaggableModelViewSet(viewsets.ModelViewSet):
        queryset = TaggableModel.objects.all()
        serializer_class = TaggableModelSerializer
        permission_classes = (AllowAny,)
        filter_backends = [DjangoFilterBackend]
        filter_class = TaggableModelFilterSet
    

    【讨论】:

    • 谢谢你,它没有很好的记录。对于只想转换现有字段的人:使这项工作有效的关键设置是:指定 ModelMultipleChoiceFilter,包括对象的查询集和设置小部件。
    【解决方案2】:

    使用django-filter package 有一种更简单的方法来实现此目的。在django-filter documentation 的深处,它提到您可以使用“映射到查找列表的字段名称字典”。

    您的代码将像这样更新:

    # views.py
    
    from django_filters.rest_framework import DjangoFilterBackend
    
    class TaggableModelViewSet(viewsets.ModelViewSet):
        queryset = TaggableModel.objects.all()
        serializer_class = TaggableModelSerializer
        permission_classes = (AllowAny,)
        filter_backend = [DjangoFilterBackend]
        filterset_fields = {
            'tags': ["in", "exact"] # note the 'in' field
        }
    

    现在在 URL 中,您可以在提供参数列表之前将 __in 添加到过滤器中,它会按您的预期工作:

    https://my-api-domain/api/taggable-models?tags__in=1,2,3
    

    关于可用查找过滤器的django-filter 文档非常差,但Django documentation 本身提到了in 查找过滤器。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2017-01-15
      • 2021-03-09
      • 1970-01-01
      • 1970-01-01
      • 2018-10-03
      • 1970-01-01
      • 2017-02-27
      • 2014-10-01
      相关资源
      最近更新 更多