zoukankan      html  css  js  c++  java
  • Node.js 使用 RSA 做加密

    RSA

    RSA加密算法是一种非对称加密算法。

    假设 A 与 B 通信。A 和 B 都提供一个公开的公钥。A 把需要传递的信息,先用自己的私钥签名,再用 B 的公钥加密。B 接收到这串密文后,用自己的私钥解密,用 A 提供的公钥验签。

    为什么要先签名后加密?如果你先加密后签名,非法用户通过获取的公钥就可以破解签名,破解之后就可以替换签名。

    详细的原理可以参考以下文档:
    RSA算法原理(一)
    RSA算法原理(二)

    node-rsa

    在 node.js 中使用 rsa 算法,我们使用的是 node-rsa 这个包。

    const NodeRSA = require('node-rsa');
    
    const a_public_key_data = '-----BEGIN PUBLIC KEY----- ... -----END PUBLIC KEY-----';
    const a_private_key_data = '-----BEGIN PRIVATE KEY----- ... -----END PRIVATE KEY-----';
    
    const b_public_key_data = '-----BEGIN PUBLIC KEY----- ... -----END PUBLIC KEY-----';
    const b_private_key_data = '-----BEGIN PRIVATE KEY----- ... -----END PRIVATE KEY-----';
    
    // 生成 A 的公私钥对象
    const a_public_key = new NodeRSA(a_public_key_data);
    const a_private_key = new NodeRSA(a_private_key_data);
    
    // 生成 B 的公私钥对象
    const a_public_key = new NodeRSA(a_public_key_data);
    const a_private_key = new NodeRSA(a_private_key_data);
    
    const text = 'Hello RSA!';
    
    // 加签并加密
    const sign = a_private_key.sign(text, 'base64', 'utf8');
    console.log('A 私钥加签:', sign);
    
    const encrypted = a_public_key.encrypt(sign, 'base64');
    console.log('B 公钥加密:', encrypted);
    
    // 解密并验签
    const decrypted = a_public_key.decrypt(encrypted, 'utf8');
    console.log('B 私钥解密:', decrypted);
    
    const verify = a_public_key.verify(text, decrypted, 'utf8', 'base64');
    console.log('A 公钥验签:', verify);
    

    serialize

    接口传递的一般是复杂的对象,所以我们需要把对象按一定的顺序排列并序列化成字符串再进行签名加密的操作

    const serialize = (obj) => {
      const str = [];
      Object.keys(obj).sort().forEach((key) => {
        if (obj.hasOwnProperty(key)) {
          str.push(encodeURIComponent(key) + '=' + encodeURIComponent(obj[key]));
        }
      });
      return str.join('&');
    };
    

    注意

    RSA 算法有一定的计算量,加上 Node 不适合做计算密集型的操作。当接口被频繁调用可能会占用主线程,阻塞其他接口,使用了 RSA 的接口并发量会下降十倍左右。如非必要,谨慎在 Node 里使用 RSA。

    博客首发地址:https://www.jianshu.com/u/13cd86311525

  • 相关阅读:
    Java I/O基础
    Python基础篇(九)
    Python基础篇(八)
    thinkphp5中Indirect modification of overloaded element of XXX has no effect的解决办法
    关于json_encode转义斜杠的问题
    [thinkexceptionErrorException] glob() has been disabled for security reasons
    CentOS7 安装svn
    -bash: /usr/bin/yum: /usr/bin/python: bad interpreter: No such file or directory
    php在线编辑本地文件方法共享
    python一些不错的东西
  • 原文地址:https://www.cnblogs.com/chaohangz/p/9824811.html
Copyright © 2011-2022 走看看