javascript - 使用 CryptoJS 的 Java SHA-1 到 javascript
问题描述
我有这样的代码来生成用 Java 编写的密码
MessageDigest messageDigestPassword = MessageDigest.getInstance("SHA1");
messageDigestPassword .reset();
byte[] password = "password".getBytes();
messageDigestPassword .update(password);
byte[] encryptedPassword = messageDigestPassword .digest();
String date = "2019-10-22T11:33:13.393Z";
byte[] dateBytes = date.getBytes(StandardCharsets.UTF_8);
int offset = 0;
byte[] outputBytes = new byte[dateBytes.length + encryptedPassword .length];
System.arraycopy(dateBytes, 0, outputBytes, offset, dateBytes.length);
offset += dateBytes.length;
System.arraycopy(encryptedPassword , 0, outputBytes, offset, encryptedPassword .length);
MessageDigest finalMessageDigeset = MessageDigest.getInstance("SHA-1");
finalMessageDigeset.reset();
finalMessageDigeset.update(outputBytes);
byte[] finalPasswordBytes= finalMessageDigeset .digest();
String finalBase64Password = new String(Base64.encode(finalPasswordBytes));
并且我试图将它重写为 JavaScript 以在邮递员中使用它 - CryptoJS 到目前为止,我有:
function wordArrayToByteArray(wordArray, length) {
if (wordArray.hasOwnProperty("sigBytes") &&
wordArray.hasOwnProperty("words")) {
length = wordArray.sigBytes;
wordArray = wordArray.words;
}
var result = [],
bytes,
i = 0;
while (length > 0) {
bytes = wordToByteArray(wordArray[i], Math.min(4, length));
length -= bytes.length;
result.push(bytes);
i++;
}
return [].concat.apply([], result);
}
function stringToBytes ( str ) {
var ch, st, re = [];
for (var i = 0; i < str.length; i++ ) {
ch = str.charCodeAt(i); // get char
st = []; // set up "stack"
do {
st.push( ch & 0xFF ); // push byte to stack
ch = ch >> 8; // shift value down by 1 byte
}
while ( ch );
// add stack contents to result
// done because chars have "wrong" endianness
re = re.concat( st.reverse() );
}
// return an array of bytes
return re;
}
var dateFixed = "2019-10-22T11:33:13.393Z";
var fixedDateBytes = stringToBytes(dateFixed);
var sha1Password= CryptoJS.SHA1("password");
console.log("sha1Password",sha1Password.toString(CryptoJS.enc.Hex));
var sha1PasswordBytes= wordArrayToByteArray(sha1Password, 20);
var concatedBytes= fixedDateBytes.concat(sha1PasswordBytes);
var finalShaPassWords= CryptoJS.SHA1(concatedBytes);
console.log("finalShaPassWords",finalShaPassWords.toString(CryptoJS.enc.Hex));
console.log("finalShaPassWords",finalShaPassWords.toString(CryptoJS.enc.Base64));
然而不幸的是,用这两种语言编写的 Base64 表示不匹配。
我已经检查过,并且从日期开始的字节数是相等的。来自散列密码的字节不是。因此,在 JavaScript 中 concat 后散列失败。
我检查了第一个密码散列和生成的字节,它们都是相同的。所以我的猜测线var sha1PasswordBytes= wordArrayToByteArray(sha1Password, 20);
导致该线 var finalShaPassWords= CryptoJS.SHA1(concatedBytes);
返回错误的值。
有人可以告诉我什么是错的吗?也许它应该写成不同的?
解决方案
由于无论如何您都在使用 CryptoJS,因此您还可以使用CryptoJS 编码器和WordArray#concat
-method,这大大简化了代码:
var CryptoJS = require("crypto-js");
// Input
var inPwd = "password";
var inDate = "2019-10-22T11:33:13.393Z";
// Processing
var pwdHash = CryptoJS.SHA1(inPwd); // hash and convert to WordArray
var date = CryptoJS.enc.Utf8.parse(inDate); // convert to WordArray
var joinedData = date.clone().concat(pwdHash); // join date and hashed password
var joinedDataHash = CryptoJS.SHA1(joinedData); // hash joined data
var joinedDataHashB64 = CryptoJS.enc.Base64.stringify(joinedDataHash); // convert to Base64 string
// Output
console.log("Result: " + joinedDataHashB64 ); // Output: D235TBTZMfpSyB/CDl5MHAjH5fI=
此代码的输出与 Java 代码的输出相同:D235TBTZMfpSyB/CDl5MHAjH5fI=
推荐阅读
- jestjs - 告诉 jest 处理 Vite 的 "?raw" 语法
- python - 如何合并具有不同标题的多个csv文件?
- javascript - 根据先前状态的数组项的值设置状态
- javascript - 如何使用 IF 切换运行代码?
- javascript - 如何制作一个触发下载用户输入的特定键的特定文件的功能?
- reactjs - 撤消转换回零时使用Spring的奇怪行为
- python - 如何只在满足特定条件时运行的while循环
- rust - 排序然后在一行中删除一个整数的锈向量
- flutter - 对零安全性感到困惑
- c++ - 未定义符号:GOMP_parallel 使用 openmp 运行我的 c++ 应用程序时发生?(g++ 版本 7.5.0,Ubuntu~18.04)