1👍
✅
the error is raised by router class.
rest_framework.routers.SimpleRouter:
class SimpleRouter(BaseRouter):
routes = [
# List route.
...
]
...
def get_routes(self, viewset):
known_actions = list(flatten([route.mapping.values() for route in self.routes if isinstance(route, Route)]))
extra_actions = viewset.get_extra_actions()
# checking action names against the known actions list
not_allowed = [
action.__name__ for action in extra_actions
if action.__name__ in known_actions
]
if not_allowed:
msg = ('Cannot use the @action decorator on the following '
'methods, as they are existing routes: %s')
raise ImproperlyConfigured(msg % ', '.join(not_allowed))
...
so we can simply create a custom router and change the "routes" class attribute.
first way
(the updated parts are marked by ######### changed)
class CustomRouter(DefaultRouter):
"""
sets action as detail = False for all default urls
"""
routes = [
# List route.
Route(
url=r'^{prefix}{trailing_slash}$',
mapping={
'get': 'list',
'post': 'create'
},
name='{basename}-list',
detail=False,
initkwargs={'suffix': 'List'}
),
# Dynamically generated list routes. Generated using
# @action(detail=False) decorator on methods of the viewset.
DynamicRoute(
url=r'^{prefix}/{url_path}{trailing_slash}$',
name='{basename}-{url_name}',
detail=False,
initkwargs={}
),
# Detail route.
Route(
url=r'^{prefix}{trailing_slash}$', ######### changed
mapping={
'get': 'retrieve',
'put': 'update',
'patch': 'partial_update',
'delete': 'destroy'
},
name='{basename}-detail',
detail=False, ######### changed
initkwargs={'suffix': 'Instance'}
),
# Dynamically generated detail routes. Generated using
# @action(detail=True) decorator on methods of the viewset.
DynamicRoute(
url=r'^{prefix}/{url_path}{trailing_slash}$', ######### changed
name='{basename}-{url_name}',
detail=False, ######### changed
initkwargs={}
),
]
views.py:
class ProfileViewSet(mixins.RetrieveModelMixin,
mixins.UpdateModelMixin,
GenericViewSet, ):
serializer_class = ProfileSerializer
def get_object(self):
return self.request.user
urls.py:
profile_router = CustomRouter()
profile_router.register('profile', ProfileViewSet, basename='profile')
second way
class CustomRouter(DefaultRouter):
routes = []
views.py:
class ProfileViewSet(mixins.RetrieveModelMixin,
mixins.UpdateModelMixin,
GenericViewSet, ):
serializer_class = ProfileSerializer
def get_object(self):
return self.request.user
@action(detail=False)
def retrieve(self, request, *args, **kwargs):
pass
urls.py: same as the first way
Source:stackexchange.com