如何在 django rest 框架中的 posgres JSONField 上应用过滤器?

udo*_*udo 5 python django django-filter django-rest-framework

我有一个类似的模型:

# segment/models.py
from django.db import models
from django.contrib.postgres.fields import JSONField

class Segment(models.Model):
    name = models.CharField(max_length=100)
    # docs: https://docs.djangoproject.com/en/2.2/ref/contrib/postgres/fields/#jsonfield
    data = JSONField('data', 'data')

Run Code Online (Sandbox Code Playgroud)

在字段数据中,我存储与此类似的 JSON 数据

{
    "length": 123.45
    "difficulty": {
        "avg": "easy"
        "max": "advanced"
    }
}
Run Code Online (Sandbox Code Playgroud)

我希望能够查询数据如下:

  • api/segments/?data__difficulty__avg=easy

过滤所有记录,其中

  • data.difficulty.avg="easy"

为此,我设置了序列化程序:

# api/serializers.py
from rest_framework import serializers

class SegmentSerializer(serializers.ModelSerializer):
    class Meta:
        model = Segment
        fields = ['id', 'name', 'data']
Run Code Online (Sandbox Code Playgroud)

视图如下所示:

# api/views.py
from django_filters.rest_framework import DjangoFilterBackend
from .serializers import SegmentSerializer
from segment.models import Segment

class SegmentViewSet(viewsets.ReadOnlyModelViewSet):
    queryset = Segment.objects.all()
    serializer_class = SegmentSerializer
    filter_backends = [DjangoFilterBackend]
    filterset_fields = ['id', 'name', 'data']
Run Code Online (Sandbox Code Playgroud)

当我打电话

  • api/段

我收到以下错误

AssertionError at /api/segments/
AutoFilterSet resolved field 'data' with 'exact' lookup to an unrecognized field type JSONField. Try adding an override to 'Meta.filter_overrides'. See: https://django-filter.readthedocs.io/en/master/ref/filterset.html#customise-filter-generation-with-filter-overrides

Request Method: GET
Request URL: http://localhost:8000/api/segments/
Django Version: 2.2.7
Python Executable: /Users/udos/.virtualenvs/ts/bin/python
Python Version: 3.6.4
.
.
.
Run Code Online (Sandbox Code Playgroud)

我尝试将data = serializers.JSONField()添加到序列化程序:

# api/serializers.py
from rest_framework import serializers

class SegmentSerializer(serializers.ModelSerializer):
    data = serializers.JSONField()
    class Meta:
        model = Segment
        fields = ['id', 'name', 'data']
Run Code Online (Sandbox Code Playgroud)

但错误仍然存​​在。所以我没有正确应用它:|

这是如何正确完成的?

小智 0

我正在尝试同样的事情并发现了SearchFilter.

所以对你来说,它会是这样的:

# api/views.py
from rest_framework import filters
from .serializers import SegmentSerializer
from segment.models import Segment

class SegmentViewSet(viewsets.ReadOnlyModelViewSet):
    queryset = Segment.objects.all()
    serializer_class = SegmentSerializer
    filterset_fields = ['id', 'name']
    filter_backends = [filters.SearchFilter]
    search_fields = ['data__difficulty__avg']
Run Code Online (Sandbox Code Playgroud)