Django AJAX状态更新:解决CSRF令牌缺失问题

花韻仙語
发布: 2025-07-30 14:58:16
原创
796人浏览过

django ajax状态更新:解决csrf令牌缺失问题

本教程详细阐述了在Django项目中,使用AJAX进行数据更新时,因缺少CSRF(跨站请求伪造)令牌而导致状态无法持久化的问题。文章将深入解析Django的CSRF保护机制,提供前端(JavaScript)获取并传递CSRF令牌的完整代码示例,并结合后端视图处理,确保AJAX POST请求能够安全、成功地修改数据,从而实现动态页面内容的稳定更新。

1. 理解Django CSRF保护机制

在Django中,为了防止跨站请求伪造(CSRF)攻击,框架对所有非幂等HTTP方法(如POST、PUT、DELETE)强制执行CSRF验证。这意味着,当你从前端通过AJAX发送POST请求到Django后端时,请求中必须包含一个有效的CSRF令牌。如果缺少这个令牌,或者令牌不匹配,Django会拒绝该请求,导致数据无法更新。

尽管在某些情况下,开发者可能会在视图函数上使用@csrf_exempt装饰器来暂时禁用CSRF保护,但这并非最佳实践,因为它会降低应用的安全性。理想情况下,即使是AJAX请求,也应该正确地包含并验证CSRF令牌。本教程将指导你如何在前端正确地获取并发送CSRF令牌,从而在不牺牲安全性的前提下实现数据更新。

2. 前端集成CSRF令牌

为了在AJAX请求中包含CSRF令牌,我们需要执行以下步骤:

  1. 获取CSRF令牌:CSRF令牌通常作为隐藏字段或cookie在Django渲染的页面中提供。我们可以通过JavaScript从DOM中提取它,或者从名为csrftoken的cookie中获取。从cookie获取是更通用的方法,尤其适用于AJAX请求。
  2. 在请求头中发送令牌:获取到令牌后,将其作为X-CSRFToken头部字段添加到你的AJAX请求中。

以下是获取CSRF令牌的JavaScript函数以及如何在fetch API中使用的示例:

{% extends "base/room_home.html" %}

{% block content %}

<div class="container d-flex align-items-center justify-content-center" style="min-height: 100vh;">
    <div class="text-center">
        <h1>{{ room_bills.title }}</h1>
        <p>Due: {{ room_bills.due }}</p>

        <div class="col-3">
            <h3>Paid Members: </h3>
            <ul>
                {% for submission in submissions %}
                <li>
                    <a href="proof/{{ submission.user.id }}" target="_blank">{{ submission.user.username }} {{ submission.text }}</a>
                    <form>
                        <label for="status">Status:</label>
                        <select name="status" id="status-{{ submission.id }}" onchange="postStatus('{{ submission.id }}')">
                            <option value="P" {% if submission.status == 'P' %}selected{% endif %}>Pending</option>
                            <option value="A" {% if submission.status == 'A' %}selected{% endif %}>Accepted</option>
                            <option value="R" {% if submission.status == 'R' %}selected{% endif %}>Rejected</option>
                        </select>
                    </form>
                </li>
                {% endfor %}
            </ul>
        </div>

        <div class="col-3">
            <h3>Did not pay members: </h3>
            <ul>
                {% for user in did_not_submit %}
                    <li>{{ user.username }}</li>
                {% endfor %}
            </ul>
        </div>
    </div>
</div>

<script>
    // 辅助函数:从cookie中获取CSRF令牌
    function getCookie(name) {
        let cookieValue = null;
        if (document.cookie && document.cookie !== '') {
            const cookies = document.cookie.split(';');
            for (let i = 0; i < cookies.length; i++) {
                const cookie = cookies[i].trim();
                // 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;
    }

    // 更新状态的AJAX请求函数
    function postStatus(submissionId) {
        const selectElement = document.getElementById(`status-${submissionId}`); // 使用更具体的ID
        const status = selectElement.value;
        const csrftoken = getCookie('csrftoken'); // 获取CSRF令牌

        fetch(`/room/{{ room.join_code }}/bills/{{ bills.slug }}/status/`, {
            method: 'POST',
            headers: {
                'Content-Type': 'application/json',
                'X-CSRFToken': csrftoken // 在请求头中添加CSRF令牌
            },
            body: JSON.stringify({
                "submissionId": submissionId, // 保持与后端期望的键名一致
                "status": status
            })
        })
        .then(response => {
            if (!response.ok) {
                // 处理HTTP错误状态,例如403 Forbidden (CSRF token missing/invalid)
                throw new Error(`HTTP error! status: ${response.status}`);
            }
            return response.json();
        })
        .then(data => {
            if (data.success) {
                console.log('Status updated successfully:', data);
                // 可以在此处添加用户反馈,例如显示一个成功消息
            } else {
                console.error('Status update failed:', data);
                // 可以在此处添加用户反馈,例如显示一个错误消息
            }
        })
        .catch(error => {
            console.error('There was a problem with the fetch operation:', error);
            // 可以在此处添加网络错误或解析错误的用户反馈
        });
    }

    // 初始设置select元素的选中状态
    document.addEventListener('DOMContentLoaded', function() {
        {% for submission in submissions %}
            const selectElement = document.getElementById(`status-{{ submission.id }}`);
            if (selectElement) {
                selectElement.value = "{{ submission.status }}";
            }
        {% endfor %}
    });
</script>

{% endblock %}
登录后复制

代码改进点说明:

  • CSRF令牌获取:引入了getCookie函数来安全地从浏览器cookie中获取csrftoken。
  • 请求头:在fetch请求的headers中加入了'X-CSRFToken': csrftoken。
  • 元素ID:将select元素的id改为id="status-{{ submission.id }}"以确保唯一性,并避免潜在的ID冲突。
  • 事件监听:将onblur改为onchange,因为对于select元素,onchange通常是更合适的事件,它在用户选择一个新选项时触发。
  • 初始选中状态:在页面加载时,根据后端数据设置select元素的初始选中状态,确保刷新后能显示当前状态。这通过{% if submission.status == 'P' %}selected{% endif %}在模板中完成,并在DOMContentLoaded中再次确保。
  • 错误处理:为fetch请求添加了.then().catch()链,以便更好地处理网络请求成功与否的响应,并捕获潜在的错误。

3. 后端视图处理

在后端,一旦前端正确地发送了CSRF令牌,Django的CSRF中间件会自动验证它。因此,你的视图函数可以保持相对简洁。最重要的是,如果你之前为了调试或其他原因使用了@csrf_exempt,现在应该考虑移除它,让Django的CSRF保护机制发挥作用。

AI建筑知识问答
AI建筑知识问答

用人工智能ChatGPT帮你解答所有建筑问题

AI建筑知识问答 22
查看详情 AI建筑知识问答
import json
from django.http import JsonResponse
from django.views.decorators.http import require_POST # 推荐使用
# from django.views.decorators.csrf import csrf_exempt # 如果前端已发送CSRF,建议移除此行

# 推荐使用 @require_POST 确保只响应POST请求
# 如果前端已发送CSRF令牌,请移除 @csrf_exempt
# @csrf_exempt 
@require_POST 
def remark_proof_api(request, room_id, bills_slug):
    try:
        data = json.loads(request.body.decode("utf-8"))
        submission_id = data.get("submissionId") # 使用.get()方法更安全
        status = data.get("status")

        if not submission_id or not status:
            return JsonResponse({"success": False, "message": "Missing submissionId or status"}, status=400)

        # 确保 submission_id 是整数
        try:
            sub = Submission.objects.get(id=int(submission_id))
        except Submission.DoesNotExist:
            return JsonResponse({"success": False, "message": "Submission not found"}, status=404)
        except ValueError:
            return JsonResponse({"success": False, "message": "Invalid submissionId format"}, status=400)

        # 验证状态值是否有效(根据你的模型定义)
        valid_statuses = ['P', 'A', 'R'] # 假设你的Submission模型status字段有这些选项
        if status not in valid_statuses:
            return JsonResponse({"success": False, "message": "Invalid status value"}, status=400)

        sub.status = status
        sub.save()

        return JsonResponse({"success": True, "message": "Status updated successfully"})
    except json.JSONDecodeError:
        return JsonResponse({"success": False, "message": "Invalid JSON format"}, status=400)
    except Exception as e:
        # 捕获其他未知错误
        return JsonResponse({"success": False, "message": f"An unexpected error occurred: {str(e)}"}, status=500)
登录后复制

后端视图改进点说明:

  • 移除@csrf_exempt:一旦前端正确发送CSRF令牌,@csrf_exempt就不再需要,甚至应该移除,以恢复Django的内置安全防护。
  • @require_POST:这是一个推荐的装饰器,用于明确指定视图只接受POST请求,提高代码可读性和安全性。
  • 健壮性:增加了对json.loads的错误处理、data.get()的安全访问、对submission_id类型转换的错误处理,以及对Submission对象是否存在和状态值是否有效的检查,返回更具体的错误信息和HTTP状态码。

4. URL配置

URL配置保持不变,它正确地将前端请求路由到后端视图:

from django.urls import path
from . import views # 假设views.py在当前应用目录下

urlpatterns = [
    # ... 其他URL模式
    path('room/<str:room_id>/bills/<str:bills_slug>/status/', views.remark_proof_api, name='remark-proof'),
    # ...
]
登录后复制

5. 总结与注意事项

通过以上步骤,你的Django预算网站的账单状态更新功能将能够正常工作并持久化。关键在于理解并正确实现Django的CSRF保护机制。

  • CSRF令牌的重要性:永远不要低估CSRF保护的重要性。它能有效防止恶意网站利用用户已登录的会话执行未经授权的操作。
  • @csrf_exempt的慎用:虽然它能快速解决问题,但应仅在特定且理解其风险的情况下使用,并在生产环境中尽量避免。
  • 前端反馈:在实际应用中,AJAX请求成功或失败后,应向用户提供适当的视觉反馈(例如,一个短暂的“更新成功”消息或错误提示),以提升用户体验。
  • 错误处理:无论是前端JavaScript还是后端Python,都应包含健壮的错误处理机制,以便在出现问题时能够捕获、记录并向用户提供有用的信息。
  • 初始状态设置:确保页面加载时,select元素的默认选中项与数据库中的当前状态一致,避免用户混淆。

遵循这些最佳实践,你的Django应用将更加安全、稳定和用户友好。

以上就是Django AJAX状态更新:解决CSRF令牌缺失问题的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号