我正在尝试将CryptoJS的这段代码转换为Kotlin:
const hash = CryptoJS.HmacSHA256(message, key);
const signature = CryptoJS.enc.Hex.stringify(hash);
这是与上述代码段等效的kotlin代码:
private fun generateSignature(key: String, payload: String): String {
val algorithm = "HmacSHA256"
return Mac.getInstance(algorithm)
.apply { init(SecretKeySpec(key.toByteArray(), algorithm)) }
.run { doFinal(payload.toByteArray()) }
.let { HexUtils.toHexString(it) }
}
但是它根本不起作用。它们产生不同的结果。 CryptoJS生成具有8个位置的字节数组,Java代码生成具有32个位置的字节数组。
我不知道我在做什么错。我需要使我的Kotlin代码与javascript代码完全一样。
更新:我无法更改Javascript方式。我必须在科特林做完全相同的事情
Update2:这是一个测试,其中JS代码和Kotlin代码生成不同的结果。
输入:
key = 's21fk4vb-5415-46c7-aade-303dcf432bb4'
message = 'POST,/wallets/3323461f96-bdf3-4e03-bc93-7da1fb27aee7/withdraw/,1573148023809,{"amount":"1.0","bank":{"bank":"789","agency":"456","account":"12378","accountDigit":"6","name":"joao","taxId":"33206913098","holderType":"personal"}}'
带有JS代码的结果:
Result of encrypt in bytes:
{sigBytes: 32, words: [8]}
sigBytes: 32
words: [8]
0: 2102759135
1: -196086391
2: -2099697915
3: -1620551271
4: 2463524
5: 1757965357
6: -1039993965
7: -1798822705
Bytes to Hex:
7d558edff44ff58982d927059f6859990025972468c86c2dc202f39394c824cf
使用Kotlin代码的结果:
Result of encrypt in bytes:
{byte[32]@1126}
0 = 82
1 = -110
2 = -100
3 = -128
4 = -63
5 = 22
6 = -103
7 = -31
8 = 83
9 = -125
10 = -72
11 = 109
12 = -91
13 = -69
14 = 54
15 = -41
16 = 27
17 = -107
18 = -60
19 = -110
20 = -57
21 = -29
22 = -20
23 = -32
24 = -66
25 = 88
26 = 87
27 = -50
28 = -47
29 = -18
30 = -96
31 = 25
Bytes to Hex:
52929c80c11699e15383b86da5bb36d71b95c492c7e3ece0be5857ced1eea019
没有SHA-256哈希只能有8个字节的位置。顾名思义,输出应为256位或32字节。我怀疑发生的事是stringify
的输入已经假定为字节,而CryptoJS函数返回的是32位字的WordArray
。由于8 * 32 = 256,这似乎是合理的。
因此,我想您可以简单地通过在WordArray
上使用一个函数来解决此问题,例如hash.toString('hex')
。