Данные поста Fetch API не принимаются в представлении Django

Что я пытаюсь сделать?

Я пытаюсь отправить пост-запрос с данными в представление Django, используя fetch API, следующим образом:

javascript:

const data = {
  search_text: "",
  months: 6,
  property_type: "all"
};
const headers = {
   'Accept': 'application/json',
   'Content-Type':'application/json',
   'X-Requested-With':'XMLHttpRequest'
}

fetch("some-url", { method: "POST", headers: headers, body: JSON.stringify(data) })
  .then((response) => response.json())
  .then((data) => console.log(data));

view:

class MyView(View):

  def post(self, request, *args, **kwargs):
     print("request: ", request.POST)
     # do stuff with post data

урлы:

re_path(r"^my_view/$", login_required(csrf_exempt(MyView.as_view())), name="my_view"),

Проблема

Когда я пытаюсь получить доступ к данным поста в моем представлении Django, я получаю пустой QueryDict, а вывод терминала выглядит так:

request:  <QueryDict: {}>
[06/Jan/2022 06:48:19] "POST /my_app/my_view/ HTTP/1.1" 200 114
Forbidden (CSRF token missing or incorrect.): /inbox/notifications/api/unread_list/
[06/Jan/2022 06:48:22] "{"search_text":"","months":"6","property_type":"all"}GET /inbox/notifications/api/unread_list/?max=5 HTTP/1.1" 403 12291

Я пробовал искать это, но ничего не работает. Мы используем react и я знаю, что axios также может быть использован, но он должен работать с fetch api, почему он не работает, пожалуйста, помогите.

Вы, вероятно, используете SessionAuthentication. Он в любом случае обеспечивает проверку csrf. Поэтому вам все равно придется добавить X-CSRFToken, чтобы продолжать использовать SessionAuthentication. Некоторые подробности об этом. Итак, добавьте к вашему headers:

const headers = {
    'Accept': 'application/json',
    'Content-Type': 'application/json',
    'X-Requested-With': 'XMLHttpRequest',
    'X-CSRFToken': getCookie('csrftoken')  // <-- this
}

Где getCookie реализовано следующим образом:

function getCookie(c_name)
{
    if (document.cookie.length > 0)
    {
        c_start = document.cookie.indexOf(c_name + "=");
        if (c_start != -1)
        {
            c_start = c_start + c_name.length + 1;
            c_end = document.cookie.indexOf(";", c_start);
            if (c_end == -1) c_end = document.cookie.length;
            return unescape(document.cookie.substring(c_start,c_end));
        }
    }
    return "";
}

Это происходит потому, что ваше представление защищено CSRF-protection-middleware.

У вас есть две возможности:

  • Сделайте представление освобожденным от этой защиты, см. csrf_exempt
  • .
  • Добавьте CSRF-токен к вашему запросу (см. ответ Евгения Космака)

Хорошо, исправил проблему, похоже, что данные, которые я искал, были не в request.POST, а вместо этого в request.body Сделал следующие изменения:

import json

class MyView(View):

  def post(self, request, *args, **kwargs):
     print("request: ", json.loads(request.body))
     # do stuff with post data

request.body возвращает байтовую строку, поэтому необходимо преобразовать ее в json с помощью json.loads для получения дополнительной информации читайте docs

Вернуться на верх