浅析node应用的timing-attack安全漏洞


Posted in Javascript onFebruary 28, 2018

前言

假如你在项目中遇到过 eslint 报错 Potential timing attack ,不可忽视!这是一个涉及到安全的问题:时序攻击。
eslint 报错原因

首先eslint引入了一个叫做eslint-plugin-security的插件,这个插件有助于识别出潜在的安全问题,但同时也会产生误报的问题,附上插件 源码地址。

var keywords = '((' + [
  'password',
  'secret',
  'api',
  'apiKey',
  'token',
  'auth',
  'pass',
  'hash'
 ].join(')|(') + '))';

 var re = new RegExp('^' + keywords + '$', 'im');

 function containsKeyword (node) {
  if (node.type === 'Identifier') {
   if (re.test(node.name)) return true;
  }
  return
 }
 if (node.test.operator === '==' || node.test.operator === '===' || node.test.operator === '!=' || node.test.operator === '!==') {
  // 在这里 console 出错误
 }

首先这个插件会判断本次的运算符是否为 ==、===、!=、!==其中一种,其次检查标识符(字段名)是否包含特殊字符串password、secret、api、apiKey、token、auth、pass、hash,如果同时满足二者情况,eslint 就会编译报错 Potential timing attack。

攻击定义

timing attack:时序攻击,属于侧信道攻击 / 旁路攻击,侧信道攻击指的是利用信道外的信息,比如加解密的数据、数据比较时间、密文传输的流量和途径进行攻击的方式,相当于是“旁敲侧击”。

攻击点

首先讲讲js比较两个字符串大小的原理:

  • 判断字符串长度是否为0,如果为0,就可以直接比较出结果;反之,进入到第二步。
  • 字符串是由一个个字符组成,通过每个字符的charCode进行比较。
  • 在第二步中,只要出现一个字符不同,就 return false,剩余的字符不再做比较。

单个字符的比较是很快的,攻击者可以细化测量时间精度到微秒,通过响应时间的差异推算出是从哪一个字符开始不用的,这样一次次实验或者用 Python 写个脚本去跑,就可以试出正确的密码,密码破解的难度也降低了不少。

容易受攻击的写法

if (user.password === password) {
  return { state: true }; // 登录成功
 }

防御措施

每次不同的输入会造成处理时间的不同。为了防止它,我们需要使字符串比较花费相同的时间量,无论输入的密码是什么。
不容易受攻击的写法

系统中每一个密码的长度是固定的,每次比较密码是否相同时,使用正确密码的长度作为比较次数,使用异或比较每一个字符的 Unicode 编码是否相等,并且把每一次的比较结果存放到一个数组中,最后再判断数组的每一个元素是否为0(为 0 表示两个字符相同)。

// psdReceived 为用户输入密码;
 // psdDb 为系统中存储的正确用户密码
 const correctUser = (psdDb, psdReceived) => {
  const state = [];
  for (let i = 0; i < psdDb.length; ++i) {
   if (!psdReceived[i]) {
    state.push(false);
   } else {
    state.push(psdReceived.charCodeAt(i) ^ psdDb.charCodeAt(i));
   }
  }
  return state.length !== 0 && state.every(item => !item);
 }

三方包推荐

也可以使用 cryptiles 这个 npm 模块来解决这个问题

import cryptiles from 'cryptiles';

......
return cryptiles.fixedTimeCimparison(passwordFromDb, passwordReceived);
Javascript 相关文章推荐
JavaScript入门教程(2) JS基础知识
Jan 31 Javascript
Javascript中的变量使用说明
May 18 Javascript
jquery 触发a链接点击事件解决方案
May 02 Javascript
如何使用Jquery获取Form表单中被选中的radio值
Aug 09 Javascript
underscore之Collections_动力节点Java学院整理
Jul 10 Javascript
js实现canvas图片与img图片的相互转换的示例
Aug 31 Javascript
javascript高级模块化require.js的具体使用方法
Oct 31 Javascript
20行JS代码实现粘贴板复制功能
Feb 06 Javascript
vue中axios实现数据交互与跨域问题
May 12 Javascript
微信小程序后端无法保持session的原因及解决办法问题
Mar 20 Javascript
JS实现随机点名器
Apr 12 Javascript
uni-app 微信小程序授权登录的实现步骤
Feb 18 Javascript
vue组件传递对象中实现单向绑定的示例
Feb 28 #Javascript
在Vue组件中使用 TypeScript的方法
Feb 28 #Javascript
React组件中的this的具体使用
Feb 28 #Javascript
浅谈Vue网络请求之interceptors实际应用
Feb 28 #Javascript
Node.js中DNS模块学习总结
Feb 28 #Javascript
Vue自定义指令实现checkbox全选功能的方法
Feb 28 #Javascript
如何在vue中使用ts的示例代码
Feb 28 #Javascript
You might like
md5 16位二进制与32位字符串相互转换示例
2013/12/30 PHP
thinkphp文件引用与分支结构用法实例
2014/11/26 PHP
如何批量清理系统临时文件(语言:C#、 C/C++、 php 、python 、java )
2016/02/01 PHP
laravel如何开启跨域功能示例详解
2017/08/31 PHP
Jquery:ajax实现翻页无刷新功能代码
2013/08/05 Javascript
JavaScript如何禁止Backspace键
2015/12/02 Javascript
动态创建按钮的JavaScript代码
2016/01/29 Javascript
javascript和jquery实现用户登录验证
2016/05/04 Javascript
jQuery中each循环的跳出和结束实例
2017/08/16 jQuery
jQuery简单实现对数组去重及排序操作实例
2017/10/31 jQuery
JS实现的ajax和同源策略(实例讲解)
2017/12/01 Javascript
vue2 mint-ui loadmore实现下拉刷新,上拉更多功能
2018/03/21 Javascript
vue input实现点击按钮文字增删功能示例
2019/01/29 Javascript
ES6基础之默认参数值
2019/02/21 Javascript
ES6 Iterator接口和for...of循环用法分析
2019/07/31 Javascript
layui switch 开关监听 弹出确定状态转换的例子
2019/09/21 Javascript
Python strip lstrip rstrip使用方法
2008/09/06 Python
Python读取ini文件、操作mysql、发送邮件实例
2015/01/01 Python
python字符串的常用操作方法小结
2016/05/21 Python
Python中的FTP通信模块ftplib的用法整理
2016/07/08 Python
Django实现网页分页功能
2019/10/31 Python
Pytorch 实现计算分类器准确率(总分类及子分类)
2020/01/18 Python
python实现全排列代码(回溯、深度优先搜索)
2020/02/26 Python
python实现PDF中表格转化为Excel的方法
2020/06/16 Python
Python爬虫之Selenium下拉框处理的实现
2020/12/04 Python
美国开幕式潮店:Opening Ceremony
2018/02/10 全球购物
JD Sports比利时官网:英国领先的运动鞋和运动服饰零售商
2018/10/10 全球购物
《手指教学》反思
2014/02/14 职场文书
住宅使用说明书
2014/05/09 职场文书
优秀实习生主要事迹
2014/05/29 职场文书
计算机专业求职信
2014/06/02 职场文书
校园文化标语
2014/06/18 职场文书
运动会100米加油稿
2015/07/21 职场文书
入党宣誓大会后的感想
2015/08/10 职场文书
六一儿童节致辞稿(3篇)
2019/07/11 职场文书
MySQL数字类型自增的坑
2021/05/07 MySQL