当前位置: 首页 > news >正文

每个 node 应用可能存在的 timing-attack 安全漏洞

前言

假如你在项目中遇到过 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);
复制代码

关注微信公众号:创宇前端(KnownsecFED),码上获取更多优质干货!

相关文章:

  • 自己做的js甘特图插件
  • 流式计算与计算抽象化------《Designing Data-Intensive Applications》读书笔记15
  • codis proxy处理流程
  • 【异周话题 第 20 期】三年后,人工智能将彻底改变前端开发?
  • IOS评论框不贴底(ios12新bug)
  • iOS点击获取短信验证码按钮
  • 快速开始Python/WSGI应用程序
  • JavaScript原生实现《贪吃蛇》
  • JPA关系映射系列二:one-to-one主键关联
  • CSS选择器——伪元素选择器之处理父元素高度及外边距溢出
  • 有了这四个“最”,AI或许可以成功预测地震
  • JS设计模式之工厂模式
  • Web前端开发必备手册(Cheat sheet)
  • 如何在招聘中考核.NET架构师
  • 《ActiveMQ 系列》- HelloWorld
  • [译] 理解数组在 PHP 内部的实现(给PHP开发者的PHP源码-第四部分)
  • 《Javascript数据结构和算法》笔记-「字典和散列表」
  • CSS 三角实现
  • css属性的继承、初识值、计算值、当前值、应用值
  • JavaScript 基本功--面试宝典
  • JS字符串转数字方法总结
  • Sass 快速入门教程
  • 案例分享〡三拾众筹持续交付开发流程支撑创新业务
  • 百度地图API标注+时间轴组件
  • 纯 javascript 半自动式下滑一定高度,导航栏固定
  • 短视频宝贝=慢?阿里巴巴工程师这样秒开短视频
  • 番外篇1:在Windows环境下安装JDK
  • 普通函数和构造函数的区别
  • 悄悄地说一个bug
  • 微信端页面使用-webkit-box和绝对定位时,元素上移的问题
  • 要让cordova项目适配iphoneX + ios11.4,总共要几步?三步
  • 移动端解决方案学习记录
  • 用element的upload组件实现多图片上传和压缩
  • 在 Chrome DevTools 中调试 JavaScript 入门
  • AI又要和人类“对打”,Deepmind宣布《星战Ⅱ》即将开始 ...
  • 测评:对于写作的人来说,Markdown是你最好的朋友 ...
  • ​ ​Redis(五)主从复制:主从模式介绍、配置、拓扑(一主一从结构、一主多从结构、树形主从结构)、原理(复制过程、​​​​​​​数据同步psync)、总结
  • ​​​​​​​​​​​​​​Γ函数
  • #includecmath
  • #NOIP 2014# day.1 T2 联合权值
  • (4.10~4.16)
  • (Arcgis)Python编程批量将HDF5文件转换为TIFF格式并应用地理转换和投影信息
  • (cos^2 X)的定积分,求积分 ∫sin^2(x) dx
  • (Matlab)基于蝙蝠算法实现电力系统经济调度
  • (八)Spring源码解析:Spring MVC
  • (附源码)计算机毕业设计ssm-Java网名推荐系统
  • (机器学习-深度学习快速入门)第三章机器学习-第二节:机器学习模型之线性回归
  • (原創) 如何使用ISO C++讀寫BMP圖檔? (C/C++) (Image Processing)
  • (转)MVC3 类型“System.Web.Mvc.ModelClientValidationRule”同时存在
  • (转)清华学霸演讲稿:永远不要说你已经尽力了
  • (转)重识new
  • ./configure、make、make install 命令
  • ./和../以及/和~之间的区别
  • .NET Core MongoDB数据仓储和工作单元模式封装
  • .net core使用ef 6