3

我必须模型:一个图书馆可以有很多书。

现在我有一个在特定图书馆中对书籍执行 CRUD 的 URL:

router.register(r'books/(?P<library_id>[0-9]+)', BookViewSet, base_name='books')

和相应的观点:

class BookViewSet(viewsets.ModelViewSet):

    serializer_class = BookSerializer
    def get_queryset(self):

        genre_query = self.request.query_params.get('genre', None)
        status_query = self.request.query_params.get('status', None)
        author_query = self.request.query_params.get('author', None)

        books = Book.objects.filter(which_library=self.kwargs.get('library_id'))
        if genre_query:
            books = books.filter(genre=genre_query)
        if status_query:
            books = books.filter(status=status_query)
        if author_query:
            books = books.filter(author=author_query)
        return books 

我最初没有使用 ModelViewSet,而是使用了带有 @api_view 装饰器的函数,其中一个如下(返回最近两周添加的书籍,我为此函数有一个单独的 URL,即 api/books//new_arrivals):

@api_view(['GET'])
def new_arrivals(request, library_id):
    """
    List all new arrival books in a specific library
    """
    d=timezone.now()-timedelta(days=14)
    if request.method == 'GET':
        books = Book.objects.filter(which_library=library_id)
        books = books.filter(when_added__gte=d)
        serializer = BookSerializer(books, many=True)
        return Response(serializer.data)

在使用 ModelViewSets 时,我该怎么做?我必须添加另一个 URL,然后为 new_arrivals 编写另一个类或在现有 BookViewSet 中编写一个函数吗?在这种情况下如何实现处理这两个 GET 方法?

4

2 回答 2

2

您可以通过添加特殊方法来扩展传统的 viewSetlist_routedetail_route. 使用这个装饰器,您可以添加 ViewSet 生成的新 url。在这种情况下list_route更适合:

from rest_framework.decorators import list_route

class BookViewSet(viewsets.ModelViewSet):
    ...
    @list_route()
    def new_arrivals(self, request):
        books = self.get_queryset()
        d=timezone.now()-timedelta(days=14)
        books = books.filter(when_added__gte=d 
        serializer = self.get_serializer(books, many=True)
        return Response(serializer.data)

这将向 viewSet: 添加额外的 url books/{library_id}/new_arrivals

于 2018-01-04T16:08:50.497 回答
1

list_route并被detail_route折旧并与装饰器操作合并。

https://www.django-rest-framework.org/community/3.8-announcement/#deprecations

于 2019-04-01T07:11:42.490 回答