Django-фильтры не работают с Viewset - PullRequest
0 голосов
/ 17 сентября 2018

Я пытался использовать django-filters , но объекты не фильтруются.Кроме того, разрешение не работает для partal_update представлений

У меня есть Viewset, который имеет основные действия, такие как - list (), retrieve (), destroy () ,partal_update () инесколько других действий, и пытается применить фильтр для того же.

После некоторых исследований я обнаружил, что, поскольку я создаю набор запросов с помощью фильтров, мне придется переопределить метод get_queryset () .Однако, это также, кажется, не работает.Работает ли фильтр только с ModelViewSet или ListApiView ?

ViewSet -

class PostViewSet(viewsets.ViewSet):
"""
The Endpoint to list, retrieve, create and delete Posts.

"""
filter_backends = (DjangoFilterBackend, )
# filterset_class = PostFilter
filter_fields = ('pet_age', 'pet_gender', 'breed')

def get_permissions(self):

    if self.action == 'partial_update' or self.action == 'update':
        permission_classes = [IsPostAuthor, ]

    elif self.action == 'create' or self.action == 'destroy':
        permission_classes = [IsAuthenticated, ]

    else:
        permission_classes = [AllowAny, ]

    return[permission() for permission in permission_classes]

def get_queryset(self):
    return Post.objects.active() # This is implemented via custom Manager

def list(self, request, *args, **kwargs):
    """
    Method for Post listing. It can be accessed by anyone.
    """

    serializer = PostListSerializer(self.get_queryset(), many=True, context={"request": request})

    return Response(serializer.data)
# REST CODE TRUNCATED

Разрешение -

class IsPostAuthor(permissions.BasePermission):
"""
Object-level permission to only allow owners of an object to edit it.
"""

def has_object_permission(self, request, view, obj):
    if request.user.is_authenticated:
        if view.action in ['partial_update', 'update']:
            return obj.user.id == request.user.id
        return False

    return False

PostFilter -

class PostFilter(filters.FilterSet):

class Meta:
    model = Post
    fields = ('pet_age', 'pet_gender', 'breed', )

Менеджер -

class PostManager(models.Manager):

def active(self):
    return self.filter(post_status='Active')

Любая помощь будетвысоко ценится.

Ответы [ 2 ]

0 голосов
/ 18 сентября 2018

Ладно, вот наконец-то нашли решение из DRF Docs . Проблема заключалась в том, что в случае обычного ViewSet вы должны переопределить метод filter_queryset () и вернуть соответствующий queryset соответственно. Затем используйте queryset в filter_queryset , как указано Aman -

serializer = PostListSerializer(self.filter_queryset(self.get_queryset()), many=True, context={"request": request})

Ниже приведен код для справки для тех, кто все еще сталкивается с проблемами -

filter_queryset -

def filter_queryset(self, queryset):
    filter_backends = (DjangoFilterBackend, )

    # Other condition for different filter backend goes here

    for backend in list(filter_backends):
        queryset = backend().filter_queryset(self.request, queryset, view=self)
    return queryset
0 голосов
/ 18 сентября 2018

Вы перезаписали метод list, поэтому он не работает, вызовите метод filter_queryet.

def list(self, request, *args, **kwargs):
    """
     Method for Post listing. It can be accessed by anyone.
     """

    serializer = PostListSerializer(self.filter_queryset(self.get_queryset()), many=True, context= . 
     {"request": request})

    return Response(serializer.data)
...