设为首页收藏本站
天天打卡

 找回密码
 立即注册
搜索
查看: 62|回复: 13

详解python flask是如何预防CSRF攻击

[复制链接]

2

主题

54

回帖

154

积分

注册会员

积分
154
发表于 2024-4-20 09:45:41 | 显示全部楼层 |阅读模式
目录


CSRF攻击防范

详述CSRF(Cross-site request forgery),中文名称:跨站请求伪造,也被称为:one click attack/session riding,缩写为:CSRF/XSRF。
攻击者通过HTTP请求将数据传送到服务器,从而盗取回话的cookie。盗取会话cookie之后,攻击者不仅可以获取用户的信息,还可以修改该cookie关联的账户信息。
跨站请求攻击,简单地说,是攻击者通过一些技术手段欺骗用户的浏览器去访问一个自己曾经认证过的网站并运行一些操作(如发邮件,发消息,甚至财产操作如转账和购买商品)。由于浏览器曾经认证过,所以被访问的网站会认为是真正的用户操作而去运行。这利用了web中用户身份验证的一个漏洞:简单的身份验证只能保证请求发自某个用户的浏览器,却不能保证请求本身是用户自愿发出的。
CSRF防范方式一:
同源检测
Cookie的同源和浏览器的同源策略有所区别:
浏览器同源策略:协议、域名和端口都相同即同源;

  • Cookie同源策略:域名相同即同源;
    在HTTP协议中,每个异步请求都会携带两个header,用来标记来源域名:
  • Origin Header
  • Referer Header
    这两个Header在浏览器发起请求时,大多数情况会自动带上,并且不能由前端修改,服务器接收到后可以根据这两个Header确定来源的域名;
特殊情况: 如果Origin和Referer都不存在,建议直接进行阻止,特别是如果您没有使用随机CSRF Token(参考下方)作为第二次检查。
另外,CSRF大多数情况下来自第三方域名,但并不能排除本域发起。如果攻击者有权限在本域发布评论(含链接、图片等),那么它可以直接在本域发起攻击,这种情况下同源策略无法达到防护的作用。
CSRF防范方式二:
添加校验token
由于CSRF的本质在于攻击者欺骗用户去访问自己设置的地址,所以如果要求在访问敏感数据请求时,要求用户浏览器提供不保存在cookie中,并且攻击者无法伪造的数据作为校验,那么攻击者就无法再运行CSRF攻击。这种数据通常是窗体中的一个数据项。服务器将其生成并附加在窗体中,其内容是一个伪随机数。当客户端通过窗体提交请求时,这个伪随机数也一并提交上去以供校验。正常的访问时,客户端浏览器能够正确得到并传回这个伪随机数,而通过CSRF传来的欺骗性攻击中,攻击者无从事先得知这个伪随机数的值,服务端就会因为校验token的值为空或者错误,拒绝这个可疑请求
CSRF: 跨域请求伪造攻击。flask预防CSRF攻击需要下载包,flask_wtf。这个就是基于token校验来防范CSRF攻击的。

插件下载
  1. pip install flask_wtf
复制代码
flask_wtf本身提供了生成表单HTML页面的功能(基于wtforms提供),常用于开发前后端不分离的表单页面,同时Flask-wtf 扩展模块还提供了一套完善的 csrf 防护体系,对于我们开发者来说,使用flask_wtf模块就可以非常简单解决CSRF攻击问题。

工作流程

1 在表单响应的页面中加入{% csrf_token %}标签,那么在进行模板渲染是会生成如下标签
  1. <input type="hidden" name="csrfmiddlewaretoken" value="ppwN8yg1wVEyXDxtMpVIrc4zV3gHiDKKb9rwGPLaSGRc0HKhXAwpNrKjGDUHIxjj">
复制代码
2 并且在响应还有这个{% csrf_token %}标签的页面时,会添加cookie键值对,如下
csrftoken:lsMQeJgVbIKKxlfz6umgYM8WOWx1Njr77cHzM0L4xtXoApsnhFXXk1OGzwb1dd0G
3 当用户从该页面提交数据时,会携带csrfmiddlewaretoken:ppwN8yg1wVEyXDxtMpVIrc4zV3gHiDKKb9rwGPLaSGRc0HKhXAwpNrKjGDUHIxjj和cookie键值对
4 取出cookie中的csrftoken值和请求数据部分的csrfmiddlewaretoken的值,两者进行比较,这个随机字符串叫做token字符串.flask如果在请求数据部分找tokne值,如果这个键对应的值和cookie中csrftoken对应的值相同,也能通过认证.

具体配置

设置应用程序的 secret_key,用于加密生成的 csrf_token 的值
  1. #1. session加密的时候已经配置过了.如果没有在配置项中设置,则如下:
  2. app.secret_key = "#此处可以写随机字符串#"

  3. #2. 也可以写在配置类中。
  4. class Config(object):
  5.     DEBUG = True
  6.     SECRET_KEY = "dsad32DASSLD*13%^32"
  7.    
  8. """加载配置"""
  9. app.config.from_object(Config)
复制代码
导入 flask_wtf 中的 CSRFProtect类,进行初始化,并在初始化的时候关联 app
  1. # 方式1:
  2. from flask_wtf import CSRFProtect
  3. csrf = CSRFProtect() # 这块代码可能在文件中。
  4. app = Flask(import_name=__name__, template_folder="templates")
  5. # 项目配置代码之后
  6. csrf.init_app(app) # 避免出现引用导包,所以提供了init_app的用法

  7. # 方式2:
  8. # from flask_wtf import CSRFProtect
  9. # app = Flask(import_name=__name__, template_folder="templates")
  10. # 项目配置代码之后
  11. # CSRFProtect(app)
复制代码
完整视图代码
  1. from flask import Flask, render_template

  2. from flask_wtf import CSRFProtect
  3. csrf = CSRFProtect() # 这块代码可能在文件中。
  4. app = Flask(import_name=__name__, template_folder="templates")

  5. # 项目配置代码之后
  6. csrf.init_app(app) # 将插件注册到app中去


  7. #设置应用程序的 secret_key,用于加密生成的 csrf_token 的值
  8. app.config['SECRET_KEY'] = 'dafssg231bfvxvdsfwrqdqfafaffsgsbfsfsgs'


  9. @app.route("/user")
  10. def user():
  11.     title = "用户中心"
  12.     html = render_template("index12.html", **locals())
  13.     return html

  14. @app.route("/transfer", methods=["post"])
  15. def transfer():
  16.     return "转账成功!"

  17. if __name__ == '__main__':
  18.     app.run(debug=True)
复制代码
在表单中使用 CSRF 令牌
  1. <!doctype html>
  2. <html lang="en">
  3. <head>
  4.   <meta charset="UTF-8">
  5.   <title>Document</title>
  6. </head>
  7. <body>
  8.   <form action="http://127.0.0.1:5000/transfer" method="post">
  9.     <input type="hidden" name="csrf_token" value="{{ csrf_token() }}">
  10.     账号:<input type="text" name="username"><br><br>
  11.     密码:<input type="text" name="password"><br><br>
  12.     <input type="submit" value="转账">
  13.   </form>
  14. </body>
  15. </html>
复制代码
浏览器访问/user页面,表单里面会多一个隐藏的input标签,

当我们点击转账时,form表单中只要添加了{%csrf_token%},页面就会自动添加个隐藏的input标签,每次请求得到的的value值都不相同

只要手动改下这个隐藏的input里面的value值,再提交数据就会forbidden。这个值只有是后台给的才能通过鉴权

点击转账,校验失败,数据提交不成功

这样,第三方请求在提交表单时,就算猜到csrf_token的变量名,也无法猜测到服务器传来的值,所以,就可以防范csrf攻击

总结

简单总结一下本文flask中CSRF攻击的防护策略:
自动防护策略:同源检测(Origin和Referer验证);
主动防护策略:token验证以及配合SameSite设置;
为了更好的防御CSRF,最佳实践应该是结合上面总结的防御措施方式中的优缺点来综合考虑,结合当前Web应用程序自身的情况做合适的选择,才能更好的预防CSRF的发生。
以上就是详解python flask是如何预防CSRF攻击的详细内容,更多关于python flask预防CSRF攻击的资料请关注脚本之家其它相关文章!

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

×

0

主题

47

回帖

95

积分

注册会员

积分
95
发表于 2024-5-11 17:27:05 | 显示全部楼层
已测试,非常不错

2

主题

46

回帖

138

积分

注册会员

积分
138
发表于 2024-5-17 11:38:22 | 显示全部楼层
说得太好了,完全同意!

0

主题

25

回帖

67

积分

注册会员

积分
67

热心会员付费会员

发表于 2024-5-19 08:15:15 | 显示全部楼层
666666666666

1

主题

45

回帖

112

积分

注册会员

积分
112
发表于 2024-6-5 10:59:44 | 显示全部楼层
同意!

0

主题

64

回帖

128

积分

注册会员

积分
128
发表于 2024-6-22 23:52:16 | 显示全部楼层
已测试,非常不错

0

主题

55

回帖

111

积分

注册会员

积分
111
发表于 2024-6-28 13:09:44 | 显示全部楼层
你的信息来源是?我想了解更多。

2

主题

53

回帖

151

积分

注册会员

积分
151
发表于 2024-7-18 23:35:51 | 显示全部楼层
友善的讨论氛围是非常重要的。

0

主题

42

回帖

83

积分

注册会员

积分
83
发表于 2024-8-12 00:18:00 | 显示全部楼层
同意你的观点,我们有共鸣。

1

主题

49

回帖

117

积分

注册会员

积分
117
发表于 2024-8-21 01:32:03 | 显示全部楼层
同意你的观点,我们有共鸣。
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|爱云论坛 - d.taiji888.cn - 技术学习 免费资源分享 ( 蜀ICP备2022010826号 )|天天打卡

GMT+8, 2024-11-15 12:12 , Processed in 0.096387 second(s), 28 queries .

Powered by i云网络 Licensed

© 2023-2028 正版授权

快速回复 返回顶部 返回列表