cookie_parser 深入学习

发布于 2024-11-01 06:23:38 字数 5487 浏览 13 评论 0

cookie-parser 是 Express 的中间件,用来实现 cookie 的解析,是官方脚手架内置的中间件之一。

它的使用非常简单,但在使用过程中偶尔也会遇到问题。一般都是因为对 Express + cookie-parser 的签名、验证机制不了解导致的。

本文深入讲解 Express + cookie-parser 的签名和验证的实现机制,以及 cookie 签名是如何增强网站的安全性的。

入门例子:cookie 设置与解析

先从最简单的例子来看下 cookie-parser 的使用,下面采用默认配置。

  • cookie 设置:使用 Express 的内置方法 res.cookie
  • cookie 解析:使用 cookie-parser 中间件。
var express = require('express');
var cookieParser = require('cookie-parser');
var app = express();

app.use(cookieParser());

app.use(function (req, res, next) {
  console.log(req.cookies.nick); // 第二次访问,输出 chyingp
  next();
});

app.use(function (req, res, next) {  
  res.cookie('nick', 'chyingp');
  res.end('ok');
});

app.listen(3000);

在当前场景下, cookie-parser 中间件大致实现如下:

app.use(function (req, res, next) {
  req.cookies = cookie.parse(req.headers.cookie);
  next();
});

进阶例子:cookie 签名与解析

出于安全的考虑,我们通常需要对 cookie 进行签名。

例子改写如下,有两个注意点:

  1. cookieParser 初始化时,传入 secret 作为签名的秘钥。
  2. 设置 cookie 时,将 signed 设置为 true ,表示对 cookie 进行签名。
  3. 获取 cookie 时,可以同时通过 req.cookies ,也可以通过 req.signedCookies 获取。
var express = require('express');
var cookieParser = require('cookie-parser');
var app = express();

// 初始化中间件,传入的第一个参数为 singed secret
app.use(cookieParser('secret'));

app.use(function (req, res, next) {
  console.log(req.cookies.nick); // chyingp
  console.log(req.signedCookies.nick); // chyingp
  next();
});

app.use(function (req, res, next) {  
  // 传入第三个参数 {signed: true},表示要对 cookie 进行摘要计算
  res.cookie('nick', 'chyingp', {signed: true});
  res.end('ok');
});

app.listen(3000);

签名前的 cookie 值为 chyingp ,签名后的 cookie 值为 s%3Achyingp.uVofnk6k%2B9mHQpdPlQeOfjM8B5oa6mppny9d%2BmG9rD0

下面就来分析下,cookie 的签名、解析是如何实现的。

cookie 签名、解析实现剖析

Express 完成 cookie 值的签名, cookie-parser 实现签名 cookie 的解析。两者公用同一个秘钥。

cookie 签名

Express 对 cookie 的设置(包括签名),都是通过 res.cookie 这个方法实现的。

精简后的代码如下:

res.cookie = function (name, value, options) {  
  var secret = this.req.secret;
  var signed = opts.signed;

  // 如果 options.signed 为 true,则对 cookie 进行签名
  if (signed) {
    val = 's:' + sign(val, secret);
  }

  this.append('Set-Cookie', cookie.serialize(name, String(val), opts));

  return this;
};

sign 为签名函数。伪代码如下,其实就是把 cookie 的原始值,跟 hmac 后的值拼接起来。

敲黑板划重点:签名后的 cookie 值,包含了原始值。

function sign (val, secret) {
  return val + '.' + hmac(val, secret);
}

这里的 secret 哪来的呢?是 cookie-parser 初始化的时候传入的。如下伪代码所示:

var cookieParser = function (secret) {
  return function (req, res, next) {
    req.secret = secret;
    // ...
    next();
  };
};

app.use(cookieParser('secret'));

签名 cookie 解析

知道了 cookie 签名的机制后,如何"解析"签名 cookie 就很清楚了。这个阶段,中间件主要做了两件事:

  1. 将签名 cookie 对应的原始值提取出来
  2. 验证签名 cookie 是否合法

实现代码如下:

// str:签名后的 cookie,比如 "s:chyingp.uVofnk6k+9mHQpdPlQeOfjM8B5oa6mppny9d+mG9rD0"
// secret:秘钥,比如 "secret"
function signedCookie(str, secret) {

  // 检查是否 s: 开头,确保只对签过名的 cookie 进行解析
  if (str.substr(0, 2) !== 's:') {
    return str;
  }

  // 校验签名的值是否合法,如合法,返回 true,否则,返回 false
  var val = unsign(str.slice(2), secret);

  if (val !== false) {
    return val;
  }

  return false;
}

判断、提取 cookie 原始值比较简单。只是是 unsign 方法名比较有迷惑性。

一般只会对签名进行合法校验,并没有所谓的反签名。

unsign 方法的代码如下。首先,从传入的 cookie 值中,分别提取出原始值 A1、签名值 B1。用同样的秘钥对 A1 进行签名,得到 A2。根据 A2、B1 是否相等,判断签名是否合法。

exports.unsign = function(val, secret){
  var str = val.slice(0, val.lastIndexOf('.'))
    , mac = exports.sign(str, secret);

  return sha1(mac) == sha1(val) ? str : false;
};

cookie 签名的作用

主要是出于安全考虑,防止 cookie 被篡改,增强安全性。

举个小例子来看下 cookie 签名是如何实现防篡改的。

基于前面的例子展开。假设网站通过 nick 这个 cookie 来区分当前登录的用户是谁。在前面例子中,登录用户的 cookie 中,nick 对应的值如下:(decode 后的)

s:chyingp.uVofnk6k+9mHQpdPlQeOfjM8B5oa6mppny9d+mG9rD0

此时,有人试图修改这个 cookie 值,来达到伪造身份的目的。比如修改成 xiaoming

s:xiaoming.uVofnk6k+9mHQpdPlQeOfjM8B5oa6mppny9d+mG9rD0

当网站收到请求,对签名 cookie 进行解析,发现签名验证不通过。由此可判断,cookie 是伪造的。

hmac("xiaoming", "secret") !== "uVofnk6k+9mHQpdPlQeOfjM8B5oa6mppny9d+mG9rD0"

签名就能够确保安全吗

当然不是。

上个小节的例子,仅通过 nick 这个 cookie 的值来判断登录的是哪个用户,这是一个非常糟糕的设计。虽然在秘钥未知的情况下,很难伪造签名 cookie 的,但原始值相同的情况下,签名也是相同的。这种情况下,其实是很容易伪造的。

另外,开源组件的算法是公开的,因此秘钥的安全性就成了关键,要确保秘钥不泄露。

还有很多,这里不展开。

小结

本文主要对 Express + cookie-parser 的签名和解析机制进行相对深入的介绍。不少类似的总结文章中,把 cookie 的签名说成了加密,这是一个常见的错误,读者朋友可以注意一下。

签名部分的介绍,稍微涉及一些简单的安全知识,对这块不熟悉的同学可以留言交流。为讲解方便,部分段落、用词可能不够严谨。如有错漏,敬请指出。

相关链接

https://github.com/expressjs/cookie-parser

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据

关于作者

风轻花落早

暂无简介

文章
评论
25 人气
更多

推荐作者

七七

文章 0 评论 0

囍笑

文章 0 评论 0

盛夏尉蓝

文章 0 评论 0

ゞ花落谁相伴

文章 0 评论 0

Sherlocked

文章 0 评论 0

    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文