关于Django Ajax CSRF 认证

CSRF(Cross-site request forgery跨站请求伪造,也被称为“one click attack”或者session riding,一般缩写为CSRF或者XSRF,是一种对网站的恶意利用。尽管听起来像跨站脚本(XSS),但它与XSS很是不一样,而且攻击方式几乎相左。XSS利用站点内的信任用户,而CSRF则经过假装来自受信任用户的请求来利用受信任的网站。与XSS攻击相比,CSRF攻击每每不大流行(所以对其进行防范的资源也至关稀少)和难以防范,因此被认为比XSS更具危险性。html

Django 中自带了 防止CSRF攻击的功能,可是一些新手不知道如何使用,给本身编程带来了麻烦。经常会出现下面django csrf token missing or incorrect的错误。ajax

GET 请求不须要 CSRF 认证,POST 请求须要正确认证才能获得正确的返回结果。通常在POST表单中加入 {% csrf_token %}django

1
2
3
4
5
< form  method = "POST"  action = "/post-url/" >
     {% csrf_token %}
     
     < input  name = 'zqxt'  value = "自强学堂学习Django技术" >
</ form >

 

若是使用Ajax调用的时候,就要麻烦一些。须要注意如下几点:编程

  1. 在视图中使用 render (而不要使用 render_to_response)cookie

  2. 使用 jQuery 的 ajax 或者 post 以前 加入这个 js 代码:http://www.ziqiangxuetang.com/media/django/csrf.jssession

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
jQuery(document).ajaxSend( function (event, xhr, settings) {
     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 = jQuery.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  sameOrigin(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));
     }
     function  safeMethod(method) {
         return  (/^(GET|HEAD|OPTIONS|TRACE)$/.test(method));
     }
 
     if  (!safeMethod(settings.type) && sameOrigin(settings.url)) {
         xhr.setRequestHeader( "X-CSRFToken" , getCookie('csrftoken'));
     }
});

或者 更为优雅简洁的代码(不能写在 .js 中,要直接写在模板文件中):post

1
2
3
$.ajaxSetup({
     data: {csrfmiddlewaretoken:  '{{ csrf_token }}'  },
});

这样以后,就能够像原来同样的使用 jQuery.ajax() 和 jQuery.post()了学习

最后,附上一个 Django Ajax CSRF 实例网站

相关文章
相关标签/搜索