csrf(20)

cross site request forgery:跨站请求伪造

  1. 用户C打开浏览器,访问受信任网站A,输入用户名和密码请求登录网站A;

  2. 在用户信息通过验证后,网站A产生Cookie信息并返回给浏览器,此时用户登录网站A成功,可以正常发送请求到网站A;

  3. 用户未退出网站A之前,在同一浏览器中,打开一个TAB页访问网站B;

  4. 网站B接收到用户请求后,返回一些攻击性代码,并发出一个请求要求访问第三方站点A;

  5. 浏览器在接收到这些攻击性代码后,根据网站B的请求,在用户不知情的情况下携带Cookie信息,向网站A发出请求。网站A并不知道该请求其实是由B发起的,所以会根据用户C的Cookie信息以C的权限处理该请求,导致来自网站B的恶意代码被执行。

跨站请求伪造可以发送邮件、发消息,盗取你的账号,添加系统管理员,甚至于购买商品、虚拟货币转账、删除数据等

csrf 攻击

get 请求发送数据

正常网站删除 topic 的连接如右下角所示

图片.png

在同一个浏览器下访问其他网页,如果点击如下诱惑性超链接,将会发起 csrf 攻击

图片.png
图片.png

如果将诱导性超链接换成图片元素,那么它会自动加载,不需要额外点击就能发起攻击;

post 请求发送数据

图片.png

分析被攻击网站的信息,可以看到,发送到服务器的字段有三个:topic 的 title 字段,所属的 board_id 字段和 content 字段,那么攻击也需要发送这三个字段。

图片.png

还可以通过 js 实现自动提交:

<script>
    var e = function(sel) {
        return document.querySelector(sel)
    }
    e('#add').click()
</script>








csrf 防御

csrf_token

  1. 先生成 token(随机字符串),建立与当前用户的对应关系;
  2. 将该 token 通过视图函数传入到需要 "增删查改" 的页面中;
  3. 在 "增删查改" 的 html 块中加入 token,以 get 或 post 方式传递;
  4. 编写权限函数,有这个 token,然后这个 token 对应 当前用户 id 即有权限;

CSRF 攻击之所以能够成功,是因为黑客可以完全伪造用户的请求,该请求中所有的用户验证信息都是存在于 cookie(通过相同的浏览器就能拿到) 中,因此黑客可以在不知道这些验证信息的情况下直接利用用户自己的 cookie 来通过安全验证。要抵御 CSRF,关键在于在请求中放入黑客所不能伪造的信息,并且该信息不存在于 cookie 之中。可以在 HTTP 请求中以参数的形式加入一个随机产生的 token,并在服务器端建立一个拦截器来验证这个 token,如果请求中没有 token 或者 token 内容不正确,则认为可能是 CSRF 攻击而拒绝该请求。

浏览器发送登录请求的时候,服务器产生 csrf_token 随机字符串并保存,返回给浏览器,就像 session 和 cookie。当浏览器请求时,会检查请求携带的随机数是否在是不是在服务器保存的随机数里面,如果存在就是一个合法的请求。
一般来说,每次刷新页面,产生的随机字符串都会发生变化。但是对于同一台电脑同一台浏览器同一个用户,就配一个相同的 csrf_token。

生成 csrf_token
图片.png
  1. 生成 随机字符串 token
import uuid
csrf_token = {}
def new_csrf_token():
    u = current_user()
    token = str(uuid.uuid4())
    csrf_tokens[token] = u.id
    return token
  • 生成一个随机度比较高的随机数:uuid.uuid4()
  • 将随机数与当前用户的对应关系保存,就像session,只是key不一样,值都是用户 id

验证函数

from functools import wraps
csrf_tokens = dict()
def csrf_required(f):
    @wraps(f)
    def wrapper(*args, **kwargs):
        token = request.args.get('token')
        u = current_user()
        if token in csrf_tokens and csrf_tokens[token] == u.id:
            csrf_tokens.pop(token)
            return f(*args, **kwargs)
        else:
            # abort(404)
            return redirect(url_for('topic.index'))
    return wrapper
  • token in csrf_tokens 可以判断字典中是否存在 token 键

在需要保护的函数前面加权限验证

@main.route("/add", methods=["POST"])
@csrf_required
def add():
    form = request.form
    u = current_user()
    Topic.new(form, user_id=u.id)
    return redirect(url_for('.index'))
  1. 传入需要保护的页面
  • get
@main.route("/")
def index():
    board_id = int(request.args.get('board_id', -1))
    if board_id == -1:
        ms = Topic.all()
    else:
        ms = Topic.find_all(board_id=board_id)
    token = new_csrf_token()
    bs = Board.all()
    return render_template("topic/index.html", ms=ms, token=token, bs=bs, bid=board_id)

删除的 get 请求里面带上数据

这里需要强调一下利用 url 通过 get 方法传递数据到路由函数:
<a href="{{ url_for('topic.index') }}?board_id={{ b.id }}">{{ b.title }}</a>
会自动转换为字典形式:
board_id = int(request.args.get('board_id', -1))

<!--<a class="topic_title" href="{{ url_for('topic.delete', id=t.id) }}">-->
<a class="topic_title" href="{{ url_for('topic.delete', id=t.id, token=token) }}">
    删除
</a>
  • post
<form id="create_topic_form" method="post" action="{{ url_for('.add', token=token) }}">

<input type='text' name='token' value={{token}}>

上面一种方式是 get 请求,下面的才是 post 请求,故需要更改权限函数;

注意:

@main.route("/add", methods=["POST"])
@csrf_required
def add():
    form = request.form
    u = current_user()
    Topic.new(form, user_id=u.id)
    return redirect(url_for('.index'))

@csrf_required 对应的是 get 方法,获取 url 里面的 token;
request.form 对应的是表单里面上传的字段;
如果将 token 也以 post(在 form 里面加 input)传输,则要更改权限函数和 add() 函数;

验证码

服务器生成验证码的文字和 id
用户在浏览器输入相应的 文字,服务器会进行对比验证;

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,744评论 6 502
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,505评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,105评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,242评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,269评论 6 389
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,215评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,096评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,939评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,354评论 1 311
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,573评论 2 333
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,745评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,448评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,048评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,683评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,838评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,776评论 2 369
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,652评论 2 354