Django 项目中基于 CSRF 令牌 AJAX 的帖子

Posted

技术标签:

【中文标题】Django 项目中基于 CSRF 令牌 AJAX 的帖子【英文标题】:CSRF token AJAX based post in a Django Project 【发布时间】:2017-07-06 14:13:15 【问题描述】:

所以我发现了错误,它在我的 html 中。我刚刚添加了% csrf_token %,它成功了 :) 感谢大家的帮助!

(我使用了第一个答案中给我的 JS sn-p 但我仍然 收到 403 Forbidden 错误!)我可能做错了什么?

我最近学习了 JS,并为 AJAX POST 尝试了以下 JS 代码,但我收到 403 错误。做了一些进一步的研究,发现我需要传递一个 CSRF 令牌。我在网上浏览了很多教程,但我能找到的唯一解决方案是 JQuery,我不知道该语法是如何工作的。我需要知道如何通过基于 javascript AJAX 的帖子为 django 项目传递 CSRF 令牌。我的代码是;

var upvoteBtn = document.querySelector('#upvote');
var downvoteBtn = document.querySelector('#downvote');

upvoteBtn.addEventListener('click', jL);
downvoteBtn.addEventListener('click', cL);

function jL(event) 
    document.getElementById("upvote").style.display='none';
    document.getElementById("downvote").style.display='none';
    var http = new XMLHttpRequest ();
    var url = 'entered my url here';
    var data = 'title=Post%20Title&body=Body';
    var method = 'POST';

    http.open(method, url, true);
    http.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded')
    http.setRequestHeader("X-CSRFToken", csrftoken);
    http.onreadystatechange = function() 

    if (http.readyState === XMLHttpRequest.DONE && http.status === 200)
        document.getElementById("first").innerHTML = "this post has been voted";
        console.log("upvote given");
    
    else if (http.readyState === XMLHttpRequest.DONE && http.status !== 200)
        console.log("error!", http.responseText);
    
;

http.send(data);


function cL(event)
    document.getElementById("upvote").style.display='none';
    document.getElementById("downvote").style.display='none';
    var http = new XMLHttpRequest ();
    var url = 'entered my url here';
    var data = 'title=Post%20Title&body=Body';
    var method = 'POST';

    http.open(method, url, true);
    http.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded')
    http.setRequestHeader("X-CSRFToken", csrftoken);
    http.onreadystatechange = function() 

    if (http.readyState === XMLHttpRequest.DONE && http.status === 200)
        document.getElementById("first").innerHTML = "got downvoted";
        console.log("downvoted!");
    
    else if (http.readyState === XMLHttpRequest.DONE && http.status !== 200)
        console.log("error!", http.responseText);
    
; 
http.send(data);


//function for CSRF token
function getCookie(cname) 
    var name = cname + "=";
    var decodedCookie = decodeURIComponent(document.cookie);
    var ca = decodedCookie.split(';');
    for(var i = 0; i <ca.length; i++) 
        var c = ca[i];
        while (c.charAt(0) == ' ') 
            c = c.substring(1);
        
        if (c.indexOf(name) == 0) 
            return c.substring(name.length, c.length);
        
    
    return "";
    
var csrftoken = getCookie('csrftoken'); 

【问题讨论】:

查看浏览器的网络监视器,并将 Django 在 Cookie 标头中发送给您的值与您的 JavaScript 代码在 X-CSRFToken 标头中发送回 Django 的值进行比较。 【参考方案1】:

您需要致电:

xhr.setRequestHeader("X-CSRFToken", csrftoken);

当您准备 xhr 请求时。 (在您的示例中,xhr 被命名为 http

您可以从 cookie 中获取 csrftoken,但为此您需要实现 getCookie 函数。

这样的事情应该可以解决问题:

function getCookie(cname) 
    var name = cname + "=";
    var decodedCookie = decodeURIComponent(document.cookie);
    var ca = decodedCookie.split(';');
    for(var i = 0; i <ca.length; i++) 
        var c = ca[i];
        while (c.charAt(0) == ' ') 
            c = c.substring(1);
        
        if (c.indexOf(name) == 0) 
            return c.substring(name.length, c.length);
        
    
    return "";


var csrftoken = getCookie('csrftoken');

更新

在你的代码中应该是这样的:

upvoteBtn.addEventListener('click', jL);
downvoteBtn.addEventListener('click', cL);

//first define your getCookie function
function getCookie(cname) 
    var name = cname + "=";
    var decodedCookie = decodeURIComponent(document.cookie);
    var ca = decodedCookie.split(';');
    for(var i = 0; i <ca.length; i++) 
        var c = ca[i];
        while (c.charAt(0) == ' ') 
            c = c.substring(1);
        
        if (c.indexOf(name) == 0) 
            return c.substring(name.length, c.length);
        
    
    return "";


function jL(event) 
    //...
    //Then when you prepare you data fetch the token
    var csrftoken = getCookie('csrftoken');

    http.open(method, url, true);
    http.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded')
    //Now set it
    http.setRequestHeader("X-CSRFToken", csrftoken);
    //... the rest of your code


function cL(event)
    //do the same here

【讨论】:

所以在按照您的指示添加代码后,我仍然收到 Forbidden 403 错误。虽然我在 http.open 之后添加了 http 请求。我做错了什么?我尝试在这里添加代码,但它不允许我:( 这很奇怪,我可以打开它没有问题。试试这个:pastebin.com/raw/Zks2hkJ7 :( 与以前相同的错误;未找到 在此服务器上未找到请求的 URL /raw/Zks2hkJ7。Apache/2.2.15 (CentOS) 服务器在 pastebin.com 端口 80 你可以添加如果代码很短,可以在此处添加代码,或者只是在最后编辑我的原始帖子并说这是一个可能的解决方案? @shahz 是的,刚刚更新了我的答案,希望对您有所帮助。【参考方案2】:

将这段代码添加到你的 JS 中:

function getCookie(name) 
    var cookieValue = null;
    if (document.cookie && document.cookie != '') 
        var cookies = document.cookie.split(';');
        for (var i = 0; i < cookies.length; i++) 
            var cookie = $.trim(cookies[i]);
            // Does this cookie string begin with the name we want?
            if (cookie.substring(0, name.length + 1) == (name + '=')) 
                cookieValue = decodeURIComponent(cookie.substring(name.length + 1));
                break;
            
        
    
    return cookieValue;


function csrfSafeMethod(method) 
    // these HTTP methods do not require CSRF protection
    return (/^(GET|HEAD|OPTIONS|TRACE)$/.test(method));


function sameOrigin(url) 
    // test that a given url is a same-origin URL
    // url could be relative or scheme relative or absolute
    var host = document.location.host; // host + port
    var protocol = document.location.protocol;
    var sr_origin = '//' + host;
    var origin = protocol + sr_origin;
    // Allow absolute or scheme relative URLs to same origin
    return (url == origin || url.slice(0, origin.length + 1) == origin + '/') ||
        (url == sr_origin || url.slice(0, sr_origin.length + 1) == sr_origin + '/') ||
        // or any other URL that isn't scheme relative or absolute i.e relative.
        !(/^(\/\/|http:|https:).*/.test(url));


if (!csrfSafeMethod(http.responseType) && sameOrigin(http.responseUrl)) 
    http.setRequestHeader("X-CSRFToken", getCookie('csrftoken'));

基本上,您需要将 CSRF 令牌添加到标头中。我找不到找到这段代码的链接,如果有人知道会很棒。

【讨论】:

嘿,我尝试使用您的评论,但起初我收到一个错误,即 http 未定义,后来我收到 Uncaught ReferenceError: csrftoken is not defined at HTMLButtonElement.jL (button.js:49)当我将 http 定义为一个变量时,如上所述,我已经在我的代码中进行了操作。 我刚刚在我为这个问题粘贴的代码末尾添加了你给我的代码 我无法复制您的代码,因此您必须提供详细的错误信息。给出错误发生的行不会有帮助。无论如何,您必须在 Content-Type 标头之后添加我的代码。嗯,功能可以出去,但是if必须去那里。 我已经更新了原始问题中的代码,还添加了错误消息的屏幕截图。虽然新更新的代码没有你给我的代码,但我也尝试了你的方法,将 3 个函数排除在外,但 if 语句就在“Content-Type”下方,我仍然遇到与上述相同的错误。

以上是关于Django 项目中基于 CSRF 令牌 AJAX 的帖子的主要内容,如果未能解决你的问题,请参考以下文章

Django Rest Framework,ajax POST 工作,但 PATCH 抛出 CSRF 失败:CSRF 令牌丢失或不正确

无模板 Django + AJAX:Django 的 CSRF 令牌是不是在浏览会话过程中更新?

Django - 在同一个模板(ajax 和表单)中处理多个 CSRF 令牌禁止(CSRF 令牌丢失或不正确。)

在 django 中使用 ajax 表单发布 csrf 令牌的这种方法有啥问题?

用于 Ajax 的 Django csrf 令牌

Django 1.9 AJAX 表单 CSRF 令牌 403 错误 - “未设置 CSRF cookie”