目录
通过 CSRF 保护保护 Django AJAX 请求
应用程序设置
启用 CORS 和 CSRF 保护
获取并使用 CSRF 令牌
修改 POST 请求
重要注意事项
资源
首页 后端开发 Python教程 将 CSRF 保护与 Django 和 AJAX 请求结合使用

将 CSRF 保护与 Django 和 AJAX 请求结合使用

Jan 07, 2025 pm 06:15 PM

Using CSRF Protection with Django and AJAX Requests

通过 CSRF 保护保护 Django AJAX 请求

Django 的内置 CSRF(跨站请求伪造)保护在使用 django-admin startproject 创建项目时默认启用,利用 CSRF 令牌来防范恶意请求。 此中间件已添加到您的 settings.py.

对 Django 应用程序的每个 POST 请求都需要有效的 CSRF 令牌。在 Django 模板中,这是通过使用 POST 方法在任何表单中包含 {% csrf_token %} 来实现的。 然而,使用单独的前端 AJAX 请求处理 CSRF 保护需要不同的方法。

本教程演示使用来自单独前端的 AJAX 请求来保护简单的 Django 应用程序。

应用程序设置

我们的示例应用程序具有两个端点:

  • GET /get-picture:检索存储在服务器上的图像的 URL。
  • POST /set-picture:更新存储在服务器上的图像的 URL。

为了简单起见,省略了错误处理。 初始后端代码(在urls.py中)如下:

from django.urls import path
from django.http import JsonResponse
import json

picture_url = "https://picsum.photos/id/247/720/405"

def get_picture(request):
    return JsonResponse({"picture_url": picture_url})

def set_picture(request):
    if request.method == "POST":
        global picture_url
        picture_url = json.loads(request.body)["picture_url"]
        return JsonResponse({"picture_url": picture_url})

urlpatterns = [
    path("get-picture", get_picture),
    path("set-picture", set_picture)
]
登录后复制

对应的前端功能(简体):

// GET request to retrieve the image URL
async function get_picture() {
    const res = await fetch("http://localhost:8000/get-picture");
    const data = await res.json();
    return data.picture_url;
}

// POST request to update the image URL
async function set_picture(picture_url) {
    const res = await fetch("http://localhost:8000/set-picture", {
        method: "POST",
        body: JSON.stringify({ "picture_url": picture_url })
    });
}
登录后复制

为了处理跨源资源共享(CORS),我们将使用 django-cors-headers 包。

启用 CORS 和 CSRF 保护

安装django-cors-headers

pip install django-cors-headers
登录后复制

配置settings.py

INSTALLED_APPS = [
    "corsheaders",
    # ... other apps
]

MIDDLEWARE = [
    "corsheaders.middleware.CorsMiddleware",
    # ... other middleware
]

CORS_ALLOWED_ORIGINS = ["http://localhost:4040"] # Adjust port as needed
CSRF_TRUSTED_ORIGINS = ["http://localhost:4040"] # Add your frontend origin
登录后复制

虽然 GET 请求现在可以正常工作,但 POST 请求将由于 CSRF 保护而失败。 为了解决这个问题,我们需要手动管理 CSRF 令牌。

获取并使用 CSRF 令牌

创建一个新视图来提供 CSRF 令牌:

from django.views.decorators.csrf import ensure_csrf_cookie
from django.http import JsonResponse

@ensure_csrf_cookie
def get_csrf_token(request):
    return JsonResponse({"success": True})

urlpatterns = [
    # ... other paths
    path("get-csrf-token", get_csrf_token),
]
登录后复制

更新前端以获取令牌(使用js-cookie):

fetch("http://localhost:8000/get-csrf-token", { credentials: "include" });
登录后复制

credentials: "include" 选项确保浏览器处理任何 Set-Cookie 标头,存储 csrftoken cookie。 检查浏览器开发者工具中的网络选项卡以验证 cookie 是否已设置。

修改 POST 请求

最后,修改 set_picture 函数以在标头中包含 CSRF 令牌:

async function set_picture(picture_url) {
    const res = await fetch("http://localhost:8000/set-picture", {
        method: "POST",
        credentials: "include",
        headers: {
            'X-CSRFToken': Cookies.get("csrftoken")
        },
        body: JSON.stringify({ "picture_url": picture_url })
    });
}
登录后复制

这会添加 X-CSRFToken 标头以及 csrftoken cookie 中的值,从而启用成功的 POST 请求。

重要注意事项

这种方法有局限性,尤其是在不同域上部署前端和后端时。 浏览器安全策略可能会阻止设置或访问第三方 cookie,从而影响 CSRF 令牌管理。

资源

以上是将 CSRF 保护与 Django 和 AJAX 请求结合使用的详细内容。更多信息请关注PHP中文网其他相关文章!

本站声明
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover

AI Clothes Remover

用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool

Undress AI Tool

免费脱衣服图片

Clothoff.io

Clothoff.io

AI脱衣机

Video Face Swap

Video Face Swap

使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热工具

记事本++7.3.1

记事本++7.3.1

好用且免费的代码编辑器

SublimeText3汉化版

SublimeText3汉化版

中文版,非常好用

禅工作室 13.0.1

禅工作室 13.0.1

功能强大的PHP集成开发环境

Dreamweaver CS6

Dreamweaver CS6

视觉化网页开发工具

SublimeText3 Mac版

SublimeText3 Mac版

神级代码编辑软件(SublimeText3)

如何在使用 Fiddler Everywhere 进行中间人读取时避免被浏览器检测到? 如何在使用 Fiddler Everywhere 进行中间人读取时避免被浏览器检测到? Apr 02, 2025 am 07:15 AM

使用FiddlerEverywhere进行中间人读取时如何避免被检测到当你使用FiddlerEverywhere...

在Linux终端中使用python --version命令时如何解决权限问题? 在Linux终端中使用python --version命令时如何解决权限问题? Apr 02, 2025 am 06:36 AM

Linux终端中使用python...

如何在10小时内通过项目和问题驱动的方式教计算机小白编程基础? 如何在10小时内通过项目和问题驱动的方式教计算机小白编程基础? Apr 02, 2025 am 07:18 AM

如何在10小时内教计算机小白编程基础?如果你只有10个小时来教计算机小白一些编程知识,你会选择教些什么�...

如何绕过Investing.com的反爬虫机制获取新闻数据? 如何绕过Investing.com的反爬虫机制获取新闻数据? Apr 02, 2025 am 07:03 AM

攻克Investing.com的反爬虫策略许多人尝试爬取Investing.com(https://cn.investing.com/news/latest-news)的新闻数据时,常常�...

Python 3.6加载pickle文件报错ModuleNotFoundError: No module named '__builtin__'怎么办? Python 3.6加载pickle文件报错ModuleNotFoundError: No module named '__builtin__'怎么办? Apr 02, 2025 am 06:27 AM

Python3.6环境下加载pickle文件报错:ModuleNotFoundError:Nomodulenamed...

使用Scapy爬虫时,管道文件无法写入的原因是什么? 使用Scapy爬虫时,管道文件无法写入的原因是什么? Apr 02, 2025 am 06:45 AM

使用Scapy爬虫时管道文件无法写入的原因探讨在学习和使用Scapy爬虫进行数据持久化存储时,可能会遇到管道文�...

See all articles