[Django]-Django >= 3.1 and is_ajax

30๐Ÿ‘

โœ…

Check HTTP_X_REQUESTED_WITH header

def sample_view(request):
    is_ajax = request.META.get('HTTP_X_REQUESTED_WITH') == 'XMLHttpRequest'

From the Release Notes of 3.1

The HttpRequest.is_ajax() method is deprecated as it relied on a jQuery-specific way of signifying AJAX calls, while current usage tends to use the JavaScript Fetch API. Depending on your use case, you can either write your own AJAX detection method, or use the new HttpRequest.accepts() method if your code depends on the client Accept HTTP header.

๐Ÿ‘คJPG

8๐Ÿ‘

Funny thing โ€” the quoted deprecation blurb only gets you halfway there. Thereโ€™s no indication of how you "use the new HttpRequest.accepts method" to replace HttpRequest.is_ajax โ€” not in the deprecation text, nor the documentation, nor the release notes.

So, here it is: if request.accepts("application/json")

(At least thatโ€™s what worked for me.)

๐Ÿ‘คPaul Bissex

6๐Ÿ‘

Instead of:

if request.is_ajax():

Helped me:

if request.headers.get('x-requested-with') == 'XMLHttpRequest':
๐Ÿ‘คPatrick Bond

3๐Ÿ‘

Combining the suggestions works for our use casesโ€ฆ

def is_ajax(request: django.http.request.HttpRequest) -> bool:
    """
    https://stackoverflow.com/questions/63629935
    """
    return (
        request.headers.get('x-requested-with') == 'XMLHttpRequest'
        or request.accepts("application/json")
    )

And then replace all instances of request.is_ajax() with is_ajax(request).

๐Ÿ‘คcclauss

0๐Ÿ‘

have you tried to check HttpRequest.headers?
HttpRequest.is_ajax() depends on HTTP_X_REQUESTED_WITH header.
so you can check this header, if it is true it would be AJAX other wise it would be a request from a browser.

HttpRequest.headers['HTTP_X_REQUESTED_WITH']
๐Ÿ‘คAhmed Munir

0๐Ÿ‘

I did what arakkal-abu said but I also added
'X-Requested-With' header with the same value
ie. 'XMLHttpRequest'

to my request and it worked

๐Ÿ‘คMrBunny

0๐Ÿ‘

Make sure to import this at the top

import re
from django.http import JsonResponse
from django.utils.translation import gettext_lazy as _
from django.conf.urls import handler404

You can have this inside your function/method to determine if from browser or ajax call

    requested_html = re.search(r'^text/html', request.META.get('HTTP_ACCEPT'))
    if requested_html:
        # requested from browser, do as per your wish
    # ajax call. Returning as per wish 
    return JsonResponse({
        'detail': _('Requested API URL not found')
    }, status=404, safe=False)

Explanation

If you request to load a page from a browser, you would see in the network tab under the requested headers of that request, text/html is at the beginning of requested headers.
enter image description here

However, if you are making an ajax call from the browser, the requested headers has */* in the beginning. If you attach

Accept: application/json

in the header, then requested headers become this
enter image description here

From this, you can understand how the accept header is different in these cases.

๐Ÿ‘คKoushik Das

Leave a comment