2017-04-18 115 views
0

有人可以帮我解决这个问题吗?CSRF令牌丢失或不正确AJAX/DJANGO

我有我的网页上的任务列表。每个任务都有自己的表单,用户可以发送评论。当我添加新的任务AJAX更新评论列表,然后我尝试发送评论的形式,并提出错误:“CSRF令牌丢失或不正确”。在其他任何情况下,表格都很好。我的表格中有{% csrf_token %}。好像我需要在AJAX中发送CSRF。我哪里错了?为什么它没有工作?

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 = 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; 
} 
var csrftoken = getCookie('csrftoken'); 

// TASK 

$(function() { 
    var loadForm = function() { 
     var btn = $(this); 
     $.ajax({ 
      url: btn.attr("data-url"), 
      type: 'get', 
      dataType: 'json', 
      beforeSend: function() { 
       $("#modal").modal("show"); 
      }, 
      success: function (data) { 
       $("#modal .modal-content").html(data.html_task_form); 
      } 
     }); 
    }; 

    var saveForm = function() { 
     var form = $(this); 
     $.ajax({ 
      url: form.attr("action"), 
      data: { 
       data: form.serialize(), 
       csrfmiddlewaretoken: getCookie('csrftoken') 
      }, 
      type: form.attr("method"), 
      dataType: 'json', 
      success: function (data) { 
       if (data.form_is_valid) { 
        $("#task-list").html(data.html_task); 
        $("#modal").modal("hide"); 
       } 
       else { 
        $("#modal .modal-content").html(data.html_task_form); 
       } 
      } 
     }); 
     $.ajaxSetup({ 
      beforeSend: function(xhr, settings) { 
       if (!csrfSafeMethod(settings.type) && sameOrigin(settings.url)) { 
        xhr.setRequestHeader("X-CSRFToken", csrftoken); 
       } 
      } 
     }); 
     return false; 
    }; 

    // Create TASK 
    $("#task-add-button").click(loadForm); 
    $("#modal").on("submit", ".js-task-add-form", saveForm); 
    // Update TASK 
    $("#task-list").on("click", "#js-edit-task-button", loadForm); 
    $("#modal").on("submit", ".js-task-edit-form", saveForm); 
}); 

CODE征求意见地址:

views.py:

def task_comment_add(request, project_code, task_code): 
    data = dict() 
    project = get_object_or_404(Project, pk=project_code) 
    task = get_object_or_404(Task, pk=task_code) 
    if request.method == 'POST': 
     form = CommentForm(request.POST) 
     if form.is_valid(): 
      comment = form.save(commit=False) 
      comment.author = request.user 
      comment.save() 
      task.comments.add(comment) 
      data['form_is_valid'] = True 
      data['html_task_comment'] = render_to_string('project/task_comment_list.html' {'task': group_task}) 
     else: 
      data['form_is_valid'] = False 
    else: 
     form = CommentForm() 
    context = {'project': project, 'task': task, 'form': form} 
    data['html_task_comment_form'] = render_to_string('project/task_comment_form.html', context, request=request) 
    return JsonResponse(data) 

JS:

// TASK COMMENT ADD 
$(".task-comment-form").submit(function(event) { 
    event.preventDefault(); 
    console.log(event.preventDefault()); 
    var form = $(this); 
    $.ajax({ 
     url: form.attr("action"), 
     data: form.serialize(), 
     type: form.attr("method"), 
     dataType: 'json', 
     success: function (data) { 
      var current_group = form.closest('.custom-list-group'); 
      if (data.form_is_valid) { 
       current_group.find(".task-comments").html(data.html_task_comment); 
      } 
      else { 
       current_group.find(".task-comment-form").html(data.html_task_comment_form); 
      } 
     } 
    }); 
    form[0].reset(); 
    return false; 
}); 

enter image description here

回答

0

尝试添加这一项

$.ajaxSetup(
{ 
    headers: 
    { 
     'X-CSRF-Token': $('input[name="_token"]').val() 
    } 
}); 
+0

我试过这个,但不幸的是它没有为我工作。你有什么想法? –

+0

尝试提醒您的csrf标记..如果你会得到任何价值? –

+0

如何检查该值?在控制台?顺便说一句,在添加新任务后,我也在PyCharm的控制台中添加了消息:UserWarning:{%csrf_token%}被用在模板中,但上下文没有提供值。这通常是由于没有使用RequestContext引起的。 “在模板中使用了{%csrf_token%},但上下文' –

0

您没有设置令牌上的Ajax请求:

https://docs.djangoproject.com/en/1.11/ref/csrf/#setting-the-token-on-the-ajax-request

此的getCookie后添加到您的代码( )电话:

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

$.ajaxSetup({ 
    beforeSend: function(xhr, settings) { 
     if (!csrfSafeMethod(settings.type) && !this.crossDomain) { 
      xhr.setRequestHeader("X-CSRFToken", csrftoken); 
     } 
    } 
}); 
+0

你可以说我在我的JS代码的哪一部分我需要把'csrfSafeMethod'函数和'$ .ajaxSetup '?我有点舒服。另外我认为'saveForm'函数的'data argument'不是正确的。 –

+0

粘贴该行下面的所有代码:var csrftoken = getCookie('csrftoken');它需要在ajax调用之前运行。 –

+0

我把这段代码放在下面的'var csrftoken = getCookie('csrftoken');''无论如何我仍然有错误。在我看来,我需要把'$ .ajaxSetup'放入我的'saveForm函数'中。你对此有何看法? –

相关问题