我有一个使用anApp
后端的模型。现在只允许按字段OrderingFilter
排序。
我想给出其他字段的排序选项,但不必在我的数据库中公开字段名称。有没有办法做到这一点?
答案 0 :(得分:6)
pip install django-filter
在您看来:
from rest_framework import viewsets, filters
from django_filters.filters import OrderingFilter
from .models import MyList
from .serializers import MyListSerializer
class MyFilter(django_filters.FilterSet):
surname = django_filters.CharFilter(name="model_field_name_2")
order_by_field = 'ordering'
ordering = OrderingFilter(
# fields(('model field name', 'parameter name'),)
fields=(
('model_field_name_1', 'name'),
('model_field_name_2', 'surname'),
('model_field_name_3', 'email'),
)
)
class Meta:
model = MyList
fields = ['model_field_name_2',]
class MyListViewSet(viewsets.ModelViewSet):
serializer_class = MyListSerializer
filter_backends = (filters.DjangoFilterBackend,)
filter_class = MyFilter
def get_queryset(self):
return MyList.objects.all()
你可以这样做:
/my-list?ordering=name,surname,email
/my-list?ordering=-email&surname=taylor
django-filter docs: http://django-filter.readthedocs.io/en/develop/ref/filters.html#orderingfilter
drf docs: http://www.django-rest-framework.org/api-guide/filtering/#djangofilterbackend
答案 1 :(得分:4)
我创建了一个AliasedOrderingFilter,它可以很好地满足这个需求。它扩展了ordering_fields属性,允许字段和字符串的元组。例如,您可以将视图ordering_fields设置为:
ordering_fields = (('alias1', 'field1'),('alias2', 'field2'), 'field3')
在ordering=alias1,-alias2,field3
的请求中使用此类将导致:
qs.order_by('field1', '-field2', 'field3)
班级:
class AliasedOrderingFilter(OrderingFilter):
''' this allows us to "alias" fields on our model to ensure consistency at the API level
We do so by allowing the ordering_fields attribute to accept a list of tuples.
You can mix and match, i.e.:
ordering_fields = (('alias1', 'field1'), 'field2', ('alias2', 'field2')) '''
def remove_invalid_fields(self, queryset, fields, view):
valid_fields = getattr(view, 'ordering_fields', self.ordering_fields)
if valid_fields is None or valid_fields == '__all__':
return super(AliasedOrderingFilter, self).remove_invalid_fields(queryset, fields, view)
aliased_fields = {}
for field in valid_fields:
if isinstance(field, basestring):
aliased_fields[field] = field
else:
aliased_fields[field[0]] = field[1]
ordering = []
for raw_field in fields:
invert = raw_field[0] == '-'
field = raw_field.lstrip('-')
if field in aliased_fields:
if invert:
ordering.append('-{}'.format(aliased_fields[field]))
else:
ordering.append(aliased_fields[field])
return ordering
答案 2 :(得分:0)
是的,您可以使用视图中的ordering_fields
属性执行此操作,如:
class YourViewSet(viewsets.ModelViewSet):
serializer_class = yourSerializer
filter_backends = (filters.OrderingFilter,)
ordering_fields = ('field1', 'field2')
在ordering_fields
中,您提到了您要允许订购的所有字段。
此处field1
和field2
不需要出现在序列化程序fields
属性中,因此它们不会出现在您的API响应中。
更新: 您的请求网址应该是:
http://example.com/api/users?ordering=field1
ordering
是查询参数。您需要将ordering
设置为您要为其配置查询集的字段名称。
您可以了解更多here