問題描述
我在用 C# 編寫的服務器應用程序中使用 AES 加密了一些數據.例如,我使用預定義的密鑰(32 字節)和 IV(16 字節)...
I encrypt some data using AES in a server application, which is written in C#. I use a predefined key (32 bytes) and IV (16 bytes), for instance...
Key: 81fe1681..6a451c1c
IV: e83c..ae76
這是我用來加密數據的 C# 代碼:
This is my C# code I use to encrypt the data:
async Task<byte[]> Encrypt(string privateKey, string pin, byte[] data)
{
using (var sha = SHA256.Create())
{
byte[] keyHash = sha.ComputeHash(Encoding.UTF8.GetBytes($"{privateKey}"));
byte[] pinHash = sha.ComputeHash(Encoding.UTF8.GetBytes($"{pin}"));
using (Aes aes = Aes.Create())
{
byte[] key = keyHash.Slice(0, aes.Key.Length);
byte[] iv = pinHash.Slice(0, aes.IV.Length);
using (ICryptoTransform transform = aes.CreateEncryptor(key, iv))
using (var stream = new MemoryStream())
using (var cryptStream = new CryptoStream(stream, transform, CryptoStreamMode.Write))
{
await cryptStream.WriteAsync(data, 0, data.Length);
await cryptStream.FlushAsync();
return stream.ToArray();
}
}
}
}
加密的結果數據看起來像...
The encrypted result data looks like...
534c..28f5
現在,我想使用 CryptoJS 解密客戶端應用程序中的數據.我使用完全相同的密鑰和IV信息,但解密似乎失敗了……至少解密的結果總是空的.
Now, I want to decrypt the data in a client application using CryptoJS. I use the exact same key and IV information, but decryption seems to fail... at least the decrypted result is always empty.
所以,我在客戶端加密了數據(當然密鑰和 IV 相同),結果密文不同;更準確地說,它是相同的,但最后有更多數據......
So, I encrypted the data on the client (of course same key and IV) and in result the ciphered text is different; more precisely it is identical but has more data at the end...
534c..28f5bbd5..ac0e
如果我對服務器上的數據進行加密,我最后得不到的額外數據是什么?
What is this additional data at the end that I don′t get if I encrypt the data on the server?
如果我解密已在客戶端加密的密文,則解密有效.順便提一下,模式和填充在服務器和客戶端都是默認的,即CBC
和Pkcs7
;密鑰大小應為 256
.這是我用來解密服務器加密的數據的代碼:
If I decrypt the ciphered text that has been encrypted on the client, the decryption works. Just to mention it, mode and padding are default on both server and client, which is CBC
and Pkcs7
; keysize should be 256
. This is the code I use to decrypt the data that has been ciphered by the server:
let keyHash: WordArray = CryptoJS.SHA256(CryptoJS.enc.Utf8.parse(privateKey));
let key: WordArray = CryptoJS.lib.WordArray.create(keyHash.words.slice(0, 8), 32);
let pinHash: WordArray = CryptoJS.SHA256(CryptoJS.enc.Utf8.parse(pin));
let iv: WordArray = CryptoJS.lib.WordArray.create(pinHash.words.slice(0, 4), 16);
let cfg: CryptoJS.lib.IBlockCipherCfg = { iv: iv };
let paramsData: CryptoJS.lib.CipherParamsData = {
ciphertext: cipherBuffer
};
let decrypted: WordArray = CryptoJS.AES.decrypt(paramsData, key, cfg);
推薦答案
對于寫入,塊的刷新出現問題.FlushFinalBlock()
不同于 Flush()
(或 FlushAsync()
).您必須同時執行它們,或者簡單地處理 CryptoStream
.這將解決代碼沒有寫入最后一個數據塊的事實.
For the write there was a problem with the flushing of the blocks. The FlushFinalBlock()
is distinct from the Flush()
(or from the FlushAsync()
). You have to do them both, or simply dispose the CryptoStream
. This will solve the fact that the code wasn't writing the last block of data.
async static Task<byte[]> Encrypt(string privateKey, string pin, byte[] data)
{
using (var sha = SHA256.Create())
{
byte[] keyHash = sha.ComputeHash(Encoding.UTF8.GetBytes($"{privateKey}"));
byte[] pinHash = sha.ComputeHash(Encoding.UTF8.GetBytes($"{pin}"));
using (Aes aes = Aes.Create())
{
byte[] key = keyHash.Slice(0, aes.Key.Length);
byte[] iv = pinHash.Slice(0, aes.IV.Length);
Trace.WriteLine($"Key length: { key.Length }, iv length: { iv.Length }, block mode: { aes.Mode }, padding: { aes.Padding }");
using (var stream = new MemoryStream())
using (ICryptoTransform transform = aes.CreateEncryptor(key, iv))
{
using (var cryptStream = new CryptoStream(stream, transform, CryptoStreamMode.Write))
{
await cryptStream.WriteAsync(data, 0, data.Length);
}
return stream.ToArray();
}
}
}
}
打字稿代碼似乎可以解密.
The typescript code seems to be able to decrypt it.
工作小提琴:https://jsfiddle.net/uj58twrr/3/
這篇關于使用 C# 和 CryptoJS 的不同加密結果的文章就介紹到這了,希望我們推薦的答案對大家有所幫助,也希望大家多多支持html5模板網!