JSEncrypt长文本分段加解密 解决长字符串加解密失败
JSEncrypt长文本分段加解密 解决长字符串加解密失败
github: https://github.com/zhangwen9229/jsencrypt
修改源码解决方案:
//文件路径 src/JSEncrypt.ts,增加以下方法
// 分段加密长字符串
public encryptLong(str:string) {
try {
return hex2b64(this.getKey().encryptLong(str));
} catch (ex) {
return false;
}
}
// 分段解密长字符串
public decryptLong(str:string) {
try {
return this.getKey().decryptLong(b64tohex(str));
} catch (ex) {
return false;
}
}
//文件路径 lib/jsbn/rsa.ts,增加以下方法
// 分段加密长字符串
public encryptLong(text:string) {
let ct = "";
// RSA每次加密117bytes,需要辅助方法判断字符串截取位置
// 1.获取字符串截取点
const bytes = new Array();
bytes.push(0);
let byteNo = 0;
const len = text.length;
let c;
let temp = 0;
for (let i = 0; i < len; i++) {
c = text.charCodeAt(i);
if (c >= 0x010000 && c <= 0x10FFFF) { // 特殊字符,如Ř,Ţ
byteNo += 4;
} else if (c >= 0x000800 && c <= 0x00FFFF) { // 中文以及标点符号
byteNo += 3;
} else if (c >= 0x000080 && c <= 0x0007FF) { // 特殊字符,如È,Ò
byteNo += 2;
} else { // 英文以及标点符号
byteNo += 1;
}
if ((byteNo % 117) >= 114 || (byteNo % 117) == 0) {
if (byteNo - temp >= 114) {
bytes.push(i);
temp = byteNo;
}
}
}
// 2.截取字符串并分段加密
if (bytes.length > 1) {
for (let i = 0; i < bytes.length - 1; i++) {
let str;
if (i == 0) {
str = text.substring(0, bytes[i + 1] + 1);
} else {
str = text.substring(bytes[i] + 1, bytes[i + 1] + 1);
}
const t1 = this.encrypt(str);
ct += t1;
}
if (bytes[bytes.length - 1] != text.length - 1) {
const lastStr = text.substring(bytes[bytes.length - 1] + 1);
ct += this.encrypt(lastStr);
}
return (ct);
}
const t = this.encrypt(text);
return t;
}
// 分段解密长字符串
public decryptLong(text:string) {
const maxLength = ((this.n.bitLength() + 7) >> 3);
try {
if (text.length > maxLength) {
let ct = "";
const lt = text.match(/.{1,256}/g);
lt.forEach((entry) => {
const t1 = this.decrypt(entry);
ct += t1;
});
return ct;
}
const y = this.decrypt(text);
return y;
} catch (ex) {
return false;
}
}
- 修改完后,执行gulp打包
- 注意:该方案长文本解密,中文直接加密会出现乱码(解决方案:可以让后台加密之前,先将字符串encodeURI后再加密,前端解密后,再decodeURIComponent,则可避免该问题)
参考:https://jackiedark.github.io/2018/02/05/JSEncrypt长文本分段加解密/