Django進階(CSRF、Ajax)


1、跨站請求偽造:CSRF

django為用戶實現:防止跨站請求偽造的功能,通過中間件 django.middleware.csrf.CsrfViewMiddleware 來完成。而對於django中設置防跨站請求偽造功能有分為全局和局部。
全局:
  中間件 django.middleware.csrf.CsrfViewMiddleware
局部:
    @csrf_protect,為當前函數強制設置防跨站請求偽造功能,即便settings中沒有設置全局中間件。
    @csrf_exempt,取消當前函數防跨站請求偽造功能,即便settings中設置了全局中間件

注:from django.views.decorators.csrf import csrf_exempt,csrf_protect

html中設置Token:
  {% csrf_token %}

view視圖返回方法:

 

from django.template.context import RequestContext

return render_to_response('Account/Login.html',data,context_instance=RequestContext(request) 
或者
return render(request, 'xxx.html', data)
#render()方法是render_to_response的一個嶄新的快捷方式,render()方法會自動使用RequestContext=RequestContext(request)

 

2、Ajax

AJAX = 異步JavaScript 和 XML(Asynchronous JavaScript and XML)
簡短地說,在不重載整個網頁的情況下,AJAX 通過后台加載數據,並在網頁上進行顯示。

 示例:

view.py

from django.views.decorators.csrf import csrf_protect
import json

@csrf_protect
def ajax_data(request):
    if request.method == 'POST':
        ret = {'status':True,'msg':"",'data':[11,22,33,44]}
        try:
            print request.POST
        except Exception,e:
            ret['status'] = False
            ret['msg'] = str(e)
        return HttpResponse(json.dumps(ret))  #默認只能發送字符串,用json把字典轉換成字符穿
    else:
        return render(request, 'ajax_app/aaa.html')
        #render()方法是render_to_response的一個嶄新的快捷方式,前者會自動使用RequestContext=RequestContext(request)

 aaa.html

    {% csrf_token %}
    <input id="ajax_id" type="text"/>
    <input type="button" value="點擊執行ajax請求" onclick="myajax();"/>  {#點擊時就調用myajax()函數 #}

    <script src="/static/jquery-3.3.1.min.js"></script>  {# #配置js文件位置 #}
    <script src="/static/jquery.cookie.js"></script>
    <script type="text/javascript">
        var csrftokens = $.cookie('csrftoken'); //獲取當前csrftoken值
        function csrfSafeMethod(method) {
            // 根據請求方法判斷加不加 CSRF protection
            return (/^(GET|HEAD|OPTIONS|TRACE)$/.test(method));
        }
        $.ajaxSetup({
            // 發送ajax之前,執行以下這個函數
            beforeSend: function(xhr, settings) { // xhr=XmlHttpRequest 對象,所有ajax請求底層操作都是使用的它來做的
                if (!csrfSafeMethod(settings.type) && !this.crossDomain) {
                    xhr.setRequestHeader("X-CSRFToken", csrftokens);
                }
            }
        });

        function myajax(){  // 創建點擊事件函數
            var temp = $('#ajax_id').val(); //獲取input標簽id=ajax_id的值,用戶輸入
            var user_list = [
            {'username':'yangmv','age':18},
            {'username':'wj','age':28}];

            $.ajax({
                url:'/ajax_app/test',
                type:'POST',
                tradition:true,  {#對數據原生處理,不做加工#}
                data:{data0:temp,data1:JSON.stringify(user_list)}, {#類似以json.dumps轉換為字符串的方式#}
                success:function (arg) {   //arg為返回的數據
                    var obj = jQuery.parseJSON(arg); {#把返回的字符串數據用json.load成字典#}
                    if(obj.status){   //status字段為view中定義好的字段,為True
                        alert('okkkkk');
                    }else{
                        alert('obj.msg');
                }
                }
            });

        }
    </script>

 在setting.py文件中設置static文件夾路徑:

STATIC_URL = '/static/'
STATICFILES_DIRS = (os.path.join(BASE_DIR, 'static'),)

CSRF  with Ajax

 

{{ csrf_token }}  # 在html這樣寫,前端就會顯示它
{% csrf_token %}  # 在html這樣寫,不會顯示,但是會生成一個隱藏的input框,type=hidden

 

 如果不是按照如上面使用jquery.cookie.js那樣的方法,沒有使用csrfSafeMethod、$.ajaxSetup;那就要在data中加入:csrfmiddlewaretoken:'{{ csrf_token }}'

function myajax(){
	var temp = $('#ajax_id').val();
	$.ajax({
		url:'/ajax_app/test',
		type:'POST',
		data:{data1:temp,csrfmiddlewaretoken:'{{ csrf_token  }}'},
		success:function (arg) {
			var obj = jQuery.parseJSON(arg);
			$('#ajax_id').val(obj.msg);
		},
		error:function () {
			alert("failed");
		}
	});

}

 否則ajax的post請求會報錯,forbidden
Forbidden (CSRF token missing or incorrect.): /ajax_app/test

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM