how to discriminate based on HTTP method in django urlpatterns

Question:

I’m having some difficulties finding information about this, probably it’s not the right approach. I’d like to route a request to two different view functions based on the http method (GET or POST or DELETE or PUT).

As it is usually done in REST apis, this would mean that the same url has different meaning based on the HTTP method.

I don’t see a way to do this in the urls.py file of django, I’d like something like:

url(r'^tasks$', 'app.views.get_tasks', method='get'),
url(r'^tasks$', 'app.views.create_task',  method='post'),

(note: I’m working with django 1.4)

Asked By: luca

||

Answers:

I don’t think you can do this with different functions without adding a bunch of logic to the URL (which is never a good idea), but you can check inside the function for the request method:

def myview(request):
    if request.method == 'GET':
        # Code for GET requests
    elif request.method == 'POST':
        # Code for POST requests

You could also switch to class-based views. You would then only need to define a method for each of the HTTP methods:

class CreateMyModelView(CreateView):
    def get(self, request, *args, **kwargs):
        # Code for GET requests

    def post(self, request, *args, **kwargs):
        # Code for POST requests

If you decide to go the class-based route, another good resource is http://ccbv.co.uk/.

Answered By: hellsgate

Because Django allows you to use callables in url config, you can do it with a helper function.

def method_dispatch(**table):
    def invalid_method(request, *args, **kwargs):
        logger.warning('Method Not Allowed (%s): %s', request.method, request.path,
            extra={
                'status_code': 405,
                'request': request
            }
        )
        return HttpResponseNotAllowed(table.keys())

    def d(request, *args, **kwargs):
        handler = table.get(request.method, invalid_method)
        return handler(request, *args, **kwargs)
    return d

To use it:

url(r'^foo',
    method_dispatch(POST = post_handler,
                    GET = get_handler)),
Answered By: Juri Pakaste

For Django 3.2 you can use require_http_methods

from django.views.decorators.http import require_http_methods

@require_http_methods(["GET", "POST"])
def my_view(request):
    # I can assume now that only GET or POST requests make it this far
    # ...
    pass

See https://docs.djangoproject.com/en/3.2/topics/http/decorators/#django.views.decorators.http.require_http_methods for details

Answered By: Kim Stacks
Categories: questions Tags: , ,
Answers are sorted by their score. The answer accepted by the question owner as the best is marked with
at the top-right corner.