【问题标题】:Wrong redirects when using defaultRouter() for ModelViewSet in Django-REST-framework在 Django-REST-framework 中为 ModelViewSet 使用 defaultRouter() 时重定向错误
【发布时间】:2019-09-28 07:11:30
【问题描述】:

我有一个 Django 项目,我在其中制作了 3 个不同的应用程序:“博客”、“用户”、“api”。 这是一个可以使用模型 Post 发布消息的网站。我想使用 Django Rest API 来访问模型。它可以工作,但它与“博客”的 UpdateView 和 DeleteView 的一些重定向相混淆。 我认为使用 DefaultRouter() 可能有问题?

当我尝试使用我的 blog/PostupdateView blog/PostDeleteView(继承自 UpdateView 和 DeleteView)视图时,我不断被重定向到 /api/blog/postid/ 而不仅仅是访问我的 detailView路径应该只是 /blog/postid/ 我不知道为什么。

我的帖子模型:

class Post(models.Model):
    ...
    def __str__(self):
        return self.title

    def get_absolute_url(self):
        return reverse('post-detail', kwargs={'pk': self.pk})

我的序列化器:

class PostSerializer(serializers.ModelSerializer):
   class Meta:
        model = Post
        fields = ('id', 'title', 'content', 'date_posted', 'author', 'rooms')

我的帖子 api 视图:

  class PostView(viewsets.ModelViewSet):
        queryset = Post.objects.all()
        serializer_class = PostSerializer

我的网址文件:

主 urls.py:

urlpatterns = [
    ...
    path('', include('blog.urls')),
    path('api/',include('api.urls')),
]

博客/urls.py:

urlpatterns = [
   ...
    path('post/<int:pk>/', PostDetailView.as_view(),name='post-detail'),
    path('post/new/', PostCreateView.as_view(),name='post-create'),
   ...
]

api/urls.py:

router = routers.DefaultRouter()
router.register('post', views.PostView)

urlpatterns  = [
    path('',include(router.urls))
]

我在 blog/views.py 中的 PostCreateView

class PostCreateView( LoginRequiredMixin, UserPassesTestMixin, CreateView):
    model = Post
    fields = ['title', 'content', 'rooms']

    def test_func(self):
        ...

    def get_form(self, form_class=None):
        ...

    def form_valid(self, form):
        ...

当使用 PostCreateView 时,我应该被重定向到创建的 Post 的详细视图,正如 Post 模型中定义的那样。相反,我被重定向到由路由器在 api/urls.py 中生成的 api url

【问题讨论】:

    标签: python django django-rest-framework url-redirection django-rest-viewsets


    【解决方案1】:

    解决问题的简单方法之一是将 get_absolute_url() 方法更改为

    class Post(models.Model):
        ...
    
        def __str__(self):
            return self.title
    
        def get_absolute_url(self):
            return "/blog/{}/".format(self.pk)

    更新

    出了什么问题?

    为 API 和常规视图定义了具有相同名称的 URL,post-detail。您应该使用唯一的名称命名 URL。

    # blog/urls.py
    urlpatterns = [
        ...
        path('post/<int:pk>/', PostDetailView.as_view(), name='blog-post-detail'),
        path('post/new/', PostCreateView.as_view(), name='blog-post-create'),
        ...
    ]
    #models.py
    class Post(models.Model):
        ...
    
        def __str__(self):
            return self.title
    
        def get_absolute_url(self):
            return reverse('blog-post-detail', kwargs={'pk': self.pk})
    

    【讨论】:

    • 这很好用!非常感谢 !有人可以解释使用 reverse('post-detail', kwargs={'pk': self.pk}) 有什么问题吗?
    【解决方案2】:

    我是新手,无法发表评论,所以我会在已接受答案的评论中回答 OP 的问题,以供其他任何发现此问题的人使用。

    当您使用默认路由器注册路由时,Django 会自动为它们分配一组 url,格式为{basename}-{something else},其中{something else} 取决于您调用的操作和 HTTP 方法。基本名称由视图集的queryset 属性确定(因此您的基本名称将是post)。

    对于“GET”+“retrieve”操作,url 将是 {basename}-detail,对于您的查询集,它会转换为 post-detail,它会覆盖您为非 api 视图定义的 post-detail

    你可以通过注册一个basename的路由器来绕过它,所以非api的post-detail URL名称将对应post/&lt;pk&gt; URL,{other basename}-detail可以对应API视图:

    router.register('post', views.PostView, basename='api-post')
    

    https://www.django-rest-framework.org/api-guide/routers/

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 2014-08-24
      • 1970-01-01
      • 2021-08-10
      • 2021-12-02
      • 1970-01-01
      • 2019-01-13
      • 1970-01-01
      相关资源
      最近更新 更多