如果我有一个包含以下代码的视图集:
class ExtraRouteViewset(viewsets.GenericViewSet):
@list_route(methods=['get'])
def somefunction(self, request):
return Response({
'key': 'value',
'reverse': reverse('extraroute-somefunction'),
})
@list_route(methods=['get'], url_path='arguments/(?P<thing>[^/]+)')
def arguments(self, request, thing):
return Response({
'key': thing,
'reverse': reverse('extraroute-arguments', kwargs={'thing': 'something'}),
})
我希望这两种方法都有效。但是,第二个reverse
会引发NoReverseMatch
。检查网址格式(通过导航到不存在的网址)显示以下网址格式:
^demo/ ^ ^extraroute/arguments/(?P<thing>[^/]+)/$ [name='extraroute-arguments/(?P<thing>[^/]+)']
^demo/ ^ ^extraroute/arguments/(?P<thing>[^/]+)/\.(?P<format>[a-z0-9]+)$ [name='extraroute-arguments/(?P<thing>[^/]+)']
^demo/ ^ ^extraroute/somefunction/$ [name='extraroute-somefunction']
^demo/ ^ ^extraroute/somefunction/\.(?P<format>[a-z0-9]+)$ [name='extraroute-somefunction']
视图名称似乎是extraroute-arguments/(?P<thing>[^/]+)
而不是extraroute-arguments
?事实上,如果我使用reverse('extraroute-arguments/(?P<thing>[^/]+)', kwargs={'thing': 'something'})
它可行。我在这里遗漏了一些非常明显的东西,或者这是django-rest-framework
中的错误吗?
这是使用Django 1.8a和django-rest-framework 3.0.5。
答案 0 :(得分:8)
嗯,在第二个示例中,您发送了url_path='arguments/(?P<thing>[^/]+)'
。 Django REST框架use it用于创建URL pattern
和URL Name
。但the implementation太纯粹无法剥离正则表达式。
#inside urls.py
router = SimpleRouter()
router.routes.append(
Route(
url=r'^{prefix}/arguments/(?P<thing>[^/]+)$',
name='{basename}-arguments',
mapping={
'get': 'arguments',
},
initkwargs={}
),
)
router.register('extraroute', ExtraRouteViewset, base_name='extraroute')
urlpatterns = router.urls
然后在views.py中删除@list_route
装饰器,因为它不再需要(并且会导致路由冲突)
#inside views.py
class ExtraRouteViewset(viewsets.GenericViewSet):
#...
def arguments(self, request, thing):
return Response({
'key': thing,
'reverse': reverse('extraroute-arguments', kwargs={'thing': 'something'}),
})
我必须提一下,这实际上会在默认的Route
中添加一个硬编码的SimpleRouter
模式(其中包含列表,创建,检索,更新,部分更新,销毁模式)。这意味着通过此路由器实例注册的每个视图集将能够实现arguments
方法,并且当正则表达式匹配时,将调用此方法。