解决Django提交表单报错:CSRF token missing or incorrect的问题


Posted in Python onMarch 13, 2020

1、在Django提交表单时报错:Django提交表单报错:

CSRF token missing or incorrect

具体报错页面如下:

解决Django提交表单报错:CSRF token missing or incorrect的问题

2、有道词典翻译后如下:

通常,当存在真正的跨站点请求伪造时,或者Django的CSRF机制没有被正确使用时,就会出现这种情况。至于邮递表格,你须确保:

您的浏览器正在接受cookie。

视图函数将一个请求传递给模板的呈现方法。

在模板中,每个POST表单中都有一个{% csrf_token %}模板标记,目标是一个内部URL。

如果您没有使用CsrfViewMiddleware,那么您必须在任何使用csrf_token模板标签的视图以及那些接受POST数据的视图上使用csrf_protect。

该表单有一个有效的CSRF令牌。在登录另一个浏览器选项卡或登录后单击back按钮之后,您可能需要使用表单重新加载页面,因为登录后令牌会旋转。

您将看到这个页面的帮助部分,因为在Django设置文件中有DEBUG = True。将其更改为False,将只显示初始错误消息。

您可以使用CSRF_FAILURE_VIEW设置自定义这个页面。

3、解决办法:

(1)、确定项目底下的settings.py文件,是否存在django.middleware.csrf.CsrfViewMiddleware:

MIDDLEWARE = [
  'django.middleware.security.SecurityMiddleware',
  'django.contrib.sessions.middleware.SessionMiddleware',
  'django.middleware.common.CommonMiddleware',
  'django.middleware.csrf.CsrfViewMiddleware',
  'django.contrib.auth.middleware.AuthenticationMiddleware',
  'django.contrib.messages.middleware.MessageMiddleware',
  'django.middleware.clickjacking.XFrameOptionsMiddleware',
]

(2)、如果确定settings.py文件有配置了,还是报错,就要在form表单底下加一行代码:

{% csrf_token %}

这样应该就不会报上面错误了,以上内容仅供学习参考,谢谢!

补充知识:Django中csrf token验证原理

我多年没维护的博客园,有一篇初学Django时的笔记,记录了关于django-csrftoekn使用笔记,当时几乎是照抄官网的使用示例,后来工作全是用的flask。博客园也没有维护。直到我的博客收到了如下评论,确实把我给问倒了,而且我也仔细研究了这个问题。

1. Django是怎么验证csrfmiddlewaretoken合法性的?

2. 每次刷新页面的时候<input>中的csrf的value都会更新,每次重复登录的时候cookie的csrf令牌都会刷新,那么这两个csrf-token有什么区别?

解决Django提交表单报错:CSRF token missing or incorrect的问题

image.png
CSRF(Cross Site Request Forgery protection),中文简称跨站请求伪造。

django 第一次响应来自某个客户端的请求时,会在服务器端随机生成一个 token,把这个 token 放在 cookie 里。然后每次 POST 请求都会带上这个 token,
这样就能避免被 CSRF 攻击。

这样子看起来似乎没毛病,但是评论中的第三个问题,每次刷新页面,form表单中的token都会刷新,而cookie中的token却只在每次登录时刷新。我对csrftoken的验证方式起了疑问,后来看了一段官方文档的解释。

When validating the ‘csrfmiddlewaretoken' field value, only the secret, not the full token, is compared with the secret in the cookie value. This allows the use of ever-changing tokens. While each request may use its own token, the secret remains common to all.
This check is done by CsrfViewMiddleware.

官方文档中说到,检验token时,只比较secret是否和cookie中的secret值一样,而不是比较整个token。

我又有疑问了,同一次登录,form表单中的token每次都会变,而cookie中的token不便,django把那个salt存储在哪里才能保证验证通过呢。

直到看到源码。

def _compare_salted_tokens(request_csrf_token, csrf_token):
  # Assume both arguments are sanitized -- that is, strings of
  # length CSRF_TOKEN_LENGTH, all CSRF_ALLOWED_CHARS.
  return constant_time_compare(
    _unsalt_cipher_token(request_csrf_token),
    _unsalt_cipher_token(csrf_token),
  )

def _unsalt_cipher_token(token):
  """
  Given a token (assumed to be a string of CSRF_ALLOWED_CHARS, of length
  CSRF_TOKEN_LENGTH, and that its first half is a salt), use it to decrypt
  the second half to produce the original secret.
  """
  salt = token[:CSRF_SECRET_LENGTH]
  token = token[CSRF_SECRET_LENGTH:]
  chars = CSRF_ALLOWED_CHARS
  pairs = zip((chars.index(x) for x in token), (chars.index(x) for x in salt))
  secret = ''.join(chars[x - y] for x, y in pairs) # Note negative values are ok
  return secret

token字符串的前32位是salt, 后面是加密后的token, 通过salt能解密出唯一的secret。

django会验证表单中的token和cookie中token是否能解出同样的secret,secret一样则本次请求合法。

同样也不难解释,为什么ajax请求时,需要从cookie中拿取token添加到请求头中。

网上有不少关于django csrf token验证原理的文章都是错的,是因为他们根本不知道csrf-token的结构组成,我也是卡在第三条评论那.然后看了官方文档,和CsrfViewMiddleware中间件源码然后才恍然大悟。

以上这篇解决Django提交表单报错:CSRF token missing or incorrect的问题就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持三水点靠木。

Python 相关文章推荐
python实现图片变亮或者变暗的方法
Jun 01 Python
深入理解Python中的内置常量
May 20 Python
python实现飞机大战
Sep 11 Python
PyQt弹出式对话框的常用方法及标准按钮类型
Feb 27 Python
python画图的函数用法以及技巧
Jun 28 Python
Python在Matplotlib图中显示中文字体的操作方法
Jul 29 Python
Pycharm+django2.2+python3.6+MySQL实现简单的考试报名系统
Sep 05 Python
Python 跨.py文件调用自定义函数说明
Jun 01 Python
解决pyinstaller打包运行程序时出现缺少plotly库问题
Jun 02 Python
python中rc1什么意思
Jun 19 Python
PyQt5结合matplotlib绘图的实现示例
Sep 15 Python
python四种出行路线规划的实现
Jun 23 Python
python爬虫实现获取下一页代码
Mar 13 #Python
Python3 利用face_recognition实现人脸识别的方法
Mar 13 #Python
在django中使用post方法时,需要增加csrftoken的例子
Mar 13 #Python
python 安装教程之Pycharm安装及配置字体主题,换行,自动更新
Mar 13 #Python
详解用Python进行时间序列预测的7种方法
Mar 13 #Python
django-xadmin根据当前登录用户动态设置表单字段默认值方式
Mar 13 #Python
在django项目中导出数据到excel文件并实现下载的功能
Mar 13 #Python
You might like
php 不同编码下的字符串长度区分
2009/09/26 PHP
phpmyadmin3 安装配置图解教程
2012/03/29 PHP
破解.net程序(dll文件)编译和反编译方法
2013/01/31 PHP
php calender(日历)二个版本代码示例(解决2038问题)
2013/12/24 PHP
php实现的日历程序
2015/06/18 PHP
PHP设计模式之注册树模式分析
2018/01/26 PHP
js 刷新页面的代码小结 推荐
2010/04/02 Javascript
javascript延时加载之defer测试
2012/12/28 Javascript
Javascript 中 null、NaN和undefined的区别总结
2013/04/10 Javascript
使用js+jquery实现无限极联动
2013/05/23 Javascript
借助javascript代码判断网页是静态还是伪静态
2014/05/05 Javascript
javascript实现动态模态绑定grid过程代码
2014/09/22 Javascript
基于BootStrap Metronic开发框架经验小结【七】数据的导入、导出及附件的查看处理
2016/05/12 Javascript
使用JQuery中的trim()方法去掉前后空格
2016/09/16 Javascript
使用gulp搭建本地服务器并实现模拟ajax
2017/04/05 Javascript
vue中axios处理http发送请求的示例(Post和get)
2017/10/13 Javascript
JavaScript实现旋转木马轮播图
2020/03/16 Javascript
从0搭建vue-cli4脚手架
2020/06/17 Javascript
Python实现好友全头像的拼接实例(推荐)
2017/06/24 Python
利用信号如何监控Django模型对象字段值的变化详解
2017/11/27 Python
Python中判断输入是否为数字的实现代码
2018/05/26 Python
pandas 条件搜索返回列表的方法
2018/10/30 Python
python正则表达式匹配[]中间为任意字符的实例
2018/12/25 Python
django认证系统实现自定义权限管理的方法
2019/08/28 Python
Python GUI库Tkiner使用方法代码示例
2020/11/27 Python
python爬虫中抓取指数的实例讲解
2020/12/01 Python
新西兰领先的鞋类和靴子网上商城:Merchant 1948
2017/09/08 全球购物
奥地利票务门户网站:oeticket.com
2019/12/31 全球购物
法国春天百货官网:Printemps.com
2020/06/29 全球购物
会计系毕业个人自荐信格式
2013/09/23 职场文书
通信专业个人自我鉴定
2013/10/21 职场文书
一位农村小子的自荐信
2014/04/07 职场文书
好学生评语大全
2014/05/05 职场文书
给医院的感谢信
2015/01/21 职场文书
2015年教研组工作总结
2015/05/04 职场文书
在HTML5 localStorage中存储对象的示例代码
2021/04/21 Javascript