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

RSAUtil 前端 JavaScript JSEncrypt 实现 RSA (长文本)加密解密

文章归档:https://www.yuque.com/u27599042/coding_star/cl4dl599pdmtllw1

依赖

  1. import JSEncrypt from ‘jsencrypt’
pnpm i jsencrypt
  1. import {stringIsNull} from “@/utils/string_utils.js”:https://www.yuque.com/u27599042/coding_star/slncupw7un3ce7cb
  2. import {isNumber} from “@/utils/number_utils.js”:https://www.yuque.com/u27599042/coding_star/tuwmm3ghf5lgo4bw

导入依赖

import JSEncrypt from 'jsencrypt'
import {stringIsNull} from "@/utils/string_utils.js"
import {isNumber} from "@/utils/number_utils.js"

内部变量

/*** RSA 加密算法获取密钥对中的公钥使用的 key** @type {string}*/
export const PUBLIC_KEY = 'RSAPublicKey'/*** RSA 加密算法获取密钥对中的密钥使用的 key** @type {string}*/
export const PRIVATE_KEY = 'RSAPrivateKey'/*** RSA 密钥对的 bit 数(密钥对的长度)。* 常用 1024、2048,密钥对的 bit 数,越大越安全,但是越大对服务器的消耗越大** @type {number}*/
let keySize = 1024/*** keySize bit 数下的 RSA 密钥对所能够加密的最大明文大小。* RSA 算法一次能加密的明文长度与密钥长度(RSA 密钥对的 bit 数)成正比,* 默认情况下,Padding 方式为 OPENSSL_PKCS1_PADDING,RSA 算法会使* 用 11 字节的长度用于填充,所以默认情况下,RSA 所能够加密的最大明文大* 小为 (keySize / 8 - 11) byte** @type {number}*/
let maxEncryptPlainTextLen = 117/*** keySize bit 数下的 RSA 密钥对所能够解密的最大密文大小。* (keySize / 8) byte** @type {number}*/
let maxDecryptCipherTextLen = 128/*** 密钥长度为 1024 bit 下,通过公钥生成的密文字符串的长度** @type {number}*/
let cipherTextStrLen = 172

设置 RSA 密钥对的 bit 数

/*** 为 RSA 密钥对的 bit 数赋值,同时重新计算 keySize bit 数下的* RSA 密钥对所能够加密的最大明文大小、所能够解密的最大密文大小** @param size RSA 密钥对的 bit 数*/
export function setKeySize(size) {if (!isNumber(size) || size <= 0) {throw new TypeError("参数 {size} 需要是大于 0 的整数")}keySize = sizemaxEncryptPlainTextLen = keySize / 8 - 11maxDecryptCipherTextLen = keySize / 8
}

获取指定字符的 UTF8 字节大小

/*** 获取指定字符的 UTF8 字节大小* 代码来源: https://blog.csdn.net/csdn_yuan_/article/details/107428744** @param charCode 字符编码* @return {number} 指定字符的 UTF8 字节大小*/
export function getCharByteSizeUTF8(charCode) {if (!isNumber(charCode) || charCode < 0) {throw new TypeError("参数 {charCode} 需要是大于 0 的整数")}//字符代码在000000 – 00007F之间的,用一个字节编码if (charCode <= 0x007f) {return 1}//000080 – 0007FF之间的字符用两个字节else if (charCode <= 0x07ff) {return 2}//000800 – 00D7FF 和 00E000 – 00FFFF之间的用三个字节,注: Unicode在范围 D800-DFFF 中不存在任何字符else if (charCode <= 0xffff) {return 3}//010000 – 10FFFF之间的用4个字节else {return 4}
}

获取字符串的 UTF8 字节长度

/*** 获取字符串的 UTF8 字节长度* 代码来源: https://blog.csdn.net/csdn_yuan_/article/details/107428744** @param str 字符串* @returns {number} 字符串的 UTF8 字节长度*/
export function getStrByteLenUTF8(str) {if (stringIsNull(str)) {throw new TypeError("参数 {str} 需要非空字符串")}// 获取字符串的字符长度const strLen = str.length// 保存字符串的字节长度let strByteLen = 0// 遍历判断字符串中的每个字符,统计字符串的 UTF8 字节长度for (let i = 0; i < strLen; i++) {// 获取当前遍历字符的编码let charCode = str.charCodeAt(i);// 获取并记录当前遍历字符的 UTF8 字节大小strByteLen += getCharByteSizeUTF8(charCode)}return strByteLen
}

获取字符串的 UTF8 字节长度, 同时获取按照指定的子字符串字节长度划分的子字符串数组

/*** 获取字符串的 UTF8 字节长度,同时获取按照指定的子字符串字节长度划分的子字符串数组* 代码参考: https://blog.csdn.net/csdn_yuan_/article/details/107428744** @param str 字符串* @param subStrByteLen 子字符串字节长度* @return {[]} 按照指定的子字符串字节长度划分的子字符串数组*/
export function getStrByteLenUTF8AndSubStrs(str, subStrByteLen) {if (stringIsNull(str)) {throw new TypeError("参数 {str} 需要非空字符串")}if (!isNumber(subStrByteLen) || subStrByteLen <= 0) {throw new TypeError("参数 {subStrByteLen} 需要是大于 0 的整数")}// 获取字符串的字符长度const strLen = str.length// 保存字符串的字节长度let strByteLen = 0// 记录上一次分隔的字符串的位置let preIdx = 0;// 记录当前子字符串的字节大小let subStrByteSize = 0;// 记录子字符串const subStrs = []// 遍历判断字符串中的每个字符,统计字符串的 UTF8 字节长度for (let i = 0; i < strLen; i++) {// 获取当前遍历字符的编码let charCode = str.charCodeAt(i);// 获取并记录当前遍历字符的 UTF8 字节大小let charByteSizeUTF8 = getCharByteSizeUTF8(charCode)strByteLen += charByteSizeUTF8// 当前子字符串的字节大小subStrByteSize += charByteSizeUTF8// 子字符串达到切割长度if (subStrByteSize > subStrByteLen) {// 当前子字符串加入返回结果数组中subStrs.push(str.substring(preIdx, i))// 更新数据preIdx = isubStrByteSize = charByteSizeUTF8}}// 如果还有子字符串还为加入返回结果数组中if (subStrByteSize > 0) {subStrs.push(str.substring(preIdx))}return {strByteLen,subStrs}
}

使用公钥对明文进行加密(支持长文本)

/*** 使用公钥对明文进行加密(支持长文本)** @param publicKey 公钥* @param plainText 明文* @returns {string} 明文加密后的密文*/
export function encryptByPublicKey(publicKey, plainText) {if (stringIsNull(publicKey) || stringIsNull(plainText)) {throw new TypeError("参数 {publicKey} {plainText} 需要非空字符串")}// 获取明文字符串的字节大小和根据指定字节大小划分的子字符串数组const { strByteLen: plainTextByteSize, subStrs: plainTextSubStrArr } = getStrByteLenUTF8AndSubStrs(plainText,maxEncryptPlainTextLen)// 明文加密后的完整密文let cipherText = ""// 对明文进行分段加密plainTextSubStrArr.forEach(subStr => {// 获取加密解密器const encryptor = new JSEncrypt()// 设置公钥encryptor.setPublicKey(publicKey)// 加密cipherText += encryptor.encrypt(subStr)})return cipherText
}

使用私钥对密文进行解密(支持长文本)

注意: 此方法只适用于使用和上述加密方法逻辑相同的加密处理得到的密文的解密

/*** 使用私钥对密文进行解密(支持长文本)* 注意: 此方法只适用于使用和上述加密方法逻辑相同的加密处理得到的密文的解密** @param privateKey 密钥* @param cipherText 密文* @return {string} 密文解密后的明文*/
export function decryptByPrivateKey(privateKey, cipherText) {if (stringIsNull(privateKey) || stringIsNull(cipherText)) {throw new TypeError("参数 {privateKey} {cipherText} 需要非空字符串")}// 获取密文的字符长度let cipherTextLen = cipherText.length// 计算分段解密的次数, cipherTextStrLen 每段密文长度let decryptCount = cipherTextLen / cipherTextStrLen// 解密后的完整明文let plainText = ""// 对密文进行分段解密for (let i = 0; i < decryptCount; i++) {// 分段密文距离开始位置的偏移量let offSet = i * cipherTextStrLenlet subCipherText = cipherText.substring(offSet, offSet + cipherTextLen)// 加密解密器const encryptor = new JSEncrypt()// 设置私钥encryptor.setPrivateKey(privateKey)// 解密plainText += encryptor.decrypt(subCipherText)}return plainText
}

与 RSAUtil 搭配的 Java 后端 RSAUtilInteractiveWithFrontEnd

完整源码

import JSEncrypt from 'jsencrypt'
import {stringIsNull} from "@/utils/string_utils.js"
import {isNumber} from "@/utils/number_utils.js"/*** RSA 加密算法获取密钥对中的公钥使用的 key** @type {string}*/
export const PUBLIC_KEY = 'RSAPublicKey'/*** RSA 加密算法获取密钥对中的密钥使用的 key** @type {string}*/
export const PRIVATE_KEY = 'RSAPrivateKey'/*** RSA 密钥对的 bit 数(密钥对的长度)。* 常用 1024、2048,密钥对的 bit 数,越大越安全,但是越大对服务器的消耗越大** @type {number}*/
let keySize = 1024/*** keySize bit 数下的 RSA 密钥对所能够加密的最大明文大小。* RSA 算法一次能加密的明文长度与密钥长度(RSA 密钥对的 bit 数)成正比,* 默认情况下,Padding 方式为 OPENSSL_PKCS1_PADDING,RSA 算法会使* 用 11 字节的长度用于填充,所以默认情况下,RSA 所能够加密的最大明文大* 小为 (keySize / 8 - 11) byte** @type {number}*/
let maxEncryptPlainTextLen = 117/*** keySize bit 数下的 RSA 密钥对所能够解密的最大密文大小。* (keySize / 8) byte** @type {number}*/
let maxDecryptCipherTextLen = 128/*** 密钥长度为 1024 bit 下,通过公钥生成的密文字符串的长度** @type {number}*/
let cipherTextStrLen = 172/*** 为 RSA 密钥对的 bit 数赋值,同时重新计算 keySize bit 数下的* RSA 密钥对所能够加密的最大明文大小、所能够解密的最大密文大小** @param size RSA 密钥对的 bit 数*/
export function setKeySize(size) {if (!isNumber(size) || size <= 0) {throw new TypeError("参数 {size} 需要是大于 0 的整数")}keySize = sizemaxEncryptPlainTextLen = keySize / 8 - 11maxDecryptCipherTextLen = keySize / 8
}/*** 获取指定字符的 UTF8 字节大小* 代码来源: https://blog.csdn.net/csdn_yuan_/article/details/107428744** @param charCode 字符编码* @return {number} 指定字符的 UTF8 字节大小*/
export function getCharByteSizeUTF8(charCode) {if (!isNumber(charCode) || charCode < 0) {throw new TypeError("参数 {charCode} 需要是大于 0 的整数")}//字符代码在000000 – 00007F之间的,用一个字节编码if (charCode <= 0x007f) {return 1}//000080 – 0007FF之间的字符用两个字节else if (charCode <= 0x07ff) {return 2}//000800 – 00D7FF 和 00E000 – 00FFFF之间的用三个字节,注: Unicode在范围 D800-DFFF 中不存在任何字符else if (charCode <= 0xffff) {return 3}//010000 – 10FFFF之间的用4个字节else {return 4}
}/*** 获取字符串的 UTF8 字节长度* 代码来源: https://blog.csdn.net/csdn_yuan_/article/details/107428744** @param str 字符串* @returns {number} 字符串的 UTF8 字节长度*/
export function getStrByteLenUTF8(str) {if (stringIsNull(str)) {throw new TypeError("参数 {str} 需要非空字符串")}// 获取字符串的字符长度const strLen = str.length// 保存字符串的字节长度let strByteLen = 0// 遍历判断字符串中的每个字符,统计字符串的 UTF8 字节长度for (let i = 0; i < strLen; i++) {// 获取当前遍历字符的编码let charCode = str.charCodeAt(i);// 获取并记录当前遍历字符的 UTF8 字节大小strByteLen += getCharByteSizeUTF8(charCode)}return strByteLen
}/*** 获取字符串的 UTF8 字节长度,同时获取按照指定的子字符串字节长度划分的子字符串数组* 代码参考: https://blog.csdn.net/csdn_yuan_/article/details/107428744** @param str 字符串* @param subStrByteLen 子字符串字节长度* @return {[]} 按照指定的子字符串字节长度划分的子字符串数组*/
export function getStrByteLenUTF8AndSubStrs(str, subStrByteLen) {if (stringIsNull(str)) {throw new TypeError("参数 {str} 需要非空字符串")}if (!isNumber(subStrByteLen) || subStrByteLen <= 0) {throw new TypeError("参数 {subStrByteLen} 需要是大于 0 的整数")}// 获取字符串的字符长度const strLen = str.length// 保存字符串的字节长度let strByteLen = 0// 记录上一次分隔的字符串的位置let preIdx = 0;// 记录当前子字符串的字节大小let subStrByteSize = 0;// 记录子字符串const subStrs = []// 遍历判断字符串中的每个字符,统计字符串的 UTF8 字节长度for (let i = 0; i < strLen; i++) {// 获取当前遍历字符的编码let charCode = str.charCodeAt(i);// 获取并记录当前遍历字符的 UTF8 字节大小let charByteSizeUTF8 = getCharByteSizeUTF8(charCode)strByteLen += charByteSizeUTF8// 当前子字符串的字节大小subStrByteSize += charByteSizeUTF8// 子字符串达到切割长度if (subStrByteSize > subStrByteLen) {// 当前子字符串加入返回结果数组中subStrs.push(str.substring(preIdx, i))// 更新数据preIdx = isubStrByteSize = charByteSizeUTF8}}// 如果还有子字符串还为加入返回结果数组中if (subStrByteSize > 0) {subStrs.push(str.substring(preIdx))}return {strByteLen,subStrs}
}/*** 使用公钥对明文进行加密(支持长文本)** @param publicKey 公钥* @param plainText 明文* @returns {string} 明文加密后的密文*/
export function encryptByPublicKey(publicKey, plainText) {if (stringIsNull(publicKey) || stringIsNull(plainText)) {throw new TypeError("参数 {publicKey} {plainText} 需要非空字符串")}// 获取明文字符串的字节大小和根据指定字节大小划分的子字符串数组const { strByteLen: plainTextByteSize, subStrs: plainTextSubStrArr } = getStrByteLenUTF8AndSubStrs(plainText,maxEncryptPlainTextLen)// 明文加密后的完整密文let cipherText = ""// 对明文进行分段加密plainTextSubStrArr.forEach(subStr => {// 获取加密解密器const encryptor = new JSEncrypt()// 设置公钥encryptor.setPublicKey(publicKey)// 加密cipherText += encryptor.encrypt(subStr)})return cipherText
}/*** 使用私钥对密文进行解密(支持长文本)* 注意: 此方法只适用于使用和上述加密方法逻辑相同的加密处理得到的密文的解密** @param privateKey 密钥* @param cipherText 密文* @return {string} 密文解密后的明文*/
export function decryptByPrivateKey(privateKey, cipherText) {if (stringIsNull(privateKey) || stringIsNull(cipherText)) {throw new TypeError("参数 {privateKey} {cipherText} 需要非空字符串")}// 获取密文的字符长度let cipherTextLen = cipherText.length// 计算分段解密的次数, cipherTextStrLen 每段密文长度let decryptCount = cipherTextLen / cipherTextStrLen// 解密后的完整明文let plainText = ""// 对密文进行分段解密for (let i = 0; i < decryptCount; i++) {// 分段密文距离开始位置的偏移量let offSet = i * cipherTextStrLenlet subCipherText = cipherText.substring(offSet, offSet + cipherTextLen)// 加密解密器const encryptor = new JSEncrypt()// 设置私钥encryptor.setPrivateKey(privateKey)// 解密plainText += encryptor.decrypt(subCipherText)}return plainText
}
http://www.lryc.cn/news/207113.html

相关文章:

  • uniapp map polygons 区域填充色(fillColor)在ios显示正常,但在安卓手机显示是黑色的,怎么解决?
  • OSCAR数据库上锁问题如何排查
  • FPGA与人工智能泛谈-01
  • 【VASP】POTCAR文件
  • 棒球俱乐部青少年成长体系·棒球1号位
  • 折叠式菜单怎么做编程,初学编程系统化教程初级1上线
  • 与AI对话,如何写好prompt?
  • 基于YOLOv8模型和UA-DETRAC数据集的车辆目标检测系统(PyTorch+Pyside6+YOLOv8模型)
  • 0037【Edabit ★☆☆☆☆☆】【修改Bug 2】Buggy Code (Part 2)
  • 【算法中的Java】— 判断语句
  • 【单例模式】饿汉式,懒汉式?JAVA如何实现单例?线程安全吗?
  • Spark_SQL-DataFrame数据写出以及读写数据库(以MySQl为例)
  • Linux进程终止
  • 0036【Edabit ★☆☆☆☆☆】【让我加油】Let‘s Fuel Up!
  • React 中常用的几种路由跳转方式
  • C++内存管理:其七、标准库中的allocator
  • 【机器学习合集】人脸表情分类任务Pytorch实现TensorBoardX的使用 ->(个人学习记录笔记)
  • Maven - 国内 Maven 镜像仓库(加速包,冲冲冲~)
  • 【Solidity】智能合约案例——③版权保护合约
  • Cisco IOS XE Web UI 命令执行漏洞
  • qwen大模型,推理速度慢,单卡/双卡速度慢,flash-attention安装,解决方案
  • 3.SpringSecurity基于数据库的认证与授权
  • 【软件测试】自动化测试selenium
  • ​​​​​​​如何解决Google play开发者新注册账号,身份验证的地址证明问题?
  • Gin vs Beego: Golang的Web框架之争
  • javascript IP地址正则表达式
  • 【Bash】记录一个长命令换行的BUG
  • 【.net core】yisha框架imageupload组件多图上传修改
  • vscode markdown 使用技巧 -- 如何快速打出一个Tab 或多个空格
  • I/O 模型学习笔记【全面理解BIO/NIO/AIO】