0

0

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

花韻仙語

花韻仙語

发布时间:2025-07-30 14:58:16

|

800人浏览过

|

来源于php中文网

原创

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 %}

{{ room_bills.title }}

Due: {{ room_bills.due }}

Paid Members:

Did not pay members:

    {% for user in did_not_submit %}
  • {{ user.username }}
  • {% endfor %}
{% endblock %}

代码改进点说明:

星绘
星绘

豆包旗下 AI 写真、P 图、换装和视频生成

下载
  • 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保护机制发挥作用。

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//bills//status/', views.remark_proof_api, name='remark-proof'),
    # ...
]

5. 总结与注意事项

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

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

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

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
什么是中间件
什么是中间件

中间件是一种软件组件,充当不兼容组件之间的桥梁,提供额外服务,例如集成异构系统、提供常用服务、提高应用程序性能,以及简化应用程序开发。想了解更多中间件的相关内容,可以阅读本专题下面的文章。

178

2024.05.11

Golang 中间件开发与微服务架构
Golang 中间件开发与微服务架构

本专题系统讲解 Golang 在微服务架构中的中间件开发,包括日志处理、限流与熔断、认证与授权、服务监控、API 网关设计等常见中间件功能的实现。通过实战项目,帮助开发者理解如何使用 Go 编写高效、可扩展的中间件组件,并在微服务环境中进行灵活部署与管理。

214

2025.12.18

json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

418

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

535

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

311

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

77

2025.09.10

ajax教程
ajax教程

php中文网为大家带来ajax教程合集,Ajax是一种用于创建快速动态网页的技术。通过在后台与服务器进行少量数据交换,Ajax可以使网页实现异步更新。这意味着可以在不重新加载整个网页的情况下,对网页的某部分进行更新。php中文网还为大家带来ajax的相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

159

2023.06.14

ajax中文乱码解决方法
ajax中文乱码解决方法

ajax中文乱码解决方法有设置请求头部的字符编码、在服务器端设置响应头部的字符编码和使用encodeURIComponent对中文进行编码。本专题为大家提供ajax中文乱码相关的文章、下载、课程内容,供大家免费下载体验。

160

2023.08.31

俄罗斯Yandex引擎入口
俄罗斯Yandex引擎入口

2026年俄罗斯Yandex搜索引擎最新入口汇总,涵盖免登录、多语言支持、无广告视频播放及本地化服务等核心功能。阅读专题下面的文章了解更多详细内容。

24

2026.01.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.3万人学习

Django 教程
Django 教程

共28课时 | 3.6万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

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