[Answered ]-Djagno rest error: Cannot use the @action decorator on the following methods, as they are existing routes

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

Leave a comment