我很难复制oracle如何使用AES 256 + CBC + PKCS5/7将它复制到C#中进行加密。感谢您的帮助。将Oracle AES 256加密复制到C#
我在Oracle包下面的函数(简体):
--Character set used by GEN_ENCRYPT_PASSWORD and GEN_DECRYPT_PASSWORD
G_CHARACTER_SET VARCHAR2(10) := 'AL32UTF8';
FUNCTION GEN_ENCRYPT_PASSWORD(p_in_val IN VARCHAR2
,p_key IN VARCHAR2
,p_iv IN VARCHAR2 := NULL)
RETURN RAW
IS
l_enc_val RAW(4000);
l_enc_algo PLS_INTEGER;
l_in RAW(4000);
l_iv RAW(4000);
l_key RAW(4000);
--l_ret VARCHAR2(4000 CHAR);
v_mod NUMBER;
BEGIN
l_in := UTL_I18N.STRING_TO_RAW(data => p_in_val, dst_charset => G_CHARACTER_SET);
l_iv := UTL_I18N.STRING_TO_RAW(p_iv, G_CHARACTER_SET);
l_key := UTL_I18N.STRING_TO_RAW(data => p_key, dst_charset => G_CHARACTER_SET);
l_enc_algo := DBMS_CRYPTO.encrypt_aes256;
--chain_cbc: Cipher Block Chaining. Plaintext is XORed with the previous ciphertext block before it is encrypted.
--pad_pkcs5: Provides padding which complies with the PKCS #5: Password-Based Cryptography Standard.
v_mod := (l_enc_algo + DBMS_CRYPTO.chain_cbc + DBMS_CRYPTO.pad_pkcs5);
l_enc_val := DBMS_CRYPTO.encrypt(src => l_in, KEY => l_key, typ => v_mod);
--l_ret := RAWTOHEX(l_enc_val);
RETURN l_enc_val;
END GEN_ENCRYPT_PASSWORD;
FUNCTION GEN_DECRYPT_PASSWORD(p_in_val IN RAW
,p_key IN VARCHAR2
,p_iv IN VARCHAR2 := NULL)
RETURN VARCHAR2
IS
l_enc_val RAW(4000);
l_enc_algo PLS_INTEGER;
l_in RAW(4000);
l_iv RAW(4000);
l_key RAW(4000);
l_ret VARCHAR2(4000 CHAR);
v_mod NUMBER;
BEGIN
l_in := p_in_val; --UTL_I18N.STRING_TO_RAW(data => p_in_val, dst_charset => G_CHARACTER_SET);
l_iv := UTL_I18N.STRING_TO_RAW(p_iv, G_CHARACTER_SET);
l_key := UTL_I18N.STRING_TO_RAW(data => p_key, dst_charset => G_CHARACTER_SET);
l_enc_algo := DBMS_CRYPTO.encrypt_aes256;
--chain_cbc: Cipher Block Chaining. Plaintext is XORed with the previous ciphertext block before it is encrypted.
--pad_pkcs5: Provides padding which complies with the PKCS #5: Password-Based Cryptography Standard.
v_mod := (l_enc_algo + DBMS_CRYPTO.chain_cbc + DBMS_CRYPTO.pad_pkcs5);
l_enc_val := DBMS_CRYPTO.decrypt(src => l_in, KEY => l_key, typ => v_mod);
l_ret := UTL_I18N.raw_to_char(data => l_enc_val, src_charset => G_CHARACTER_SET);
RETURN l_ret;
END GEN_DECRYPT_PASSWORD;
然后我跑的查询:
SELECT pkg_Encyption_Test.GEN_ENCRYPT_PASSWORD('Test', '123456789', '26744a68b53dd87a') Encrypted FROM DUAL;
--Result: 7D2894678D46C769B3001BD75F603E3C
和解密结果:
SELECT pkg_Encyption_Test.GEN_DECRYPT_PASSWORD('7D2894678D46C769B3001BD75F603E3C', '123456789', '26744a68b53dd87a') Decrypt from dual;
--Result: Test
所以以上所有的作品,问题是我怎样才能转换为由C#使用来解密结果? 示例控制台应用程序:
using System;
using System.IO;
using System.Security.Cryptography;
using System.Text;
namespace SSO_EncryptionTest
{
class Program
{
/* ORACLE
* TEXT ENC: 7D2894678D46C769B3001BD75F603E3C
* TEXT: Test
* KEY: 123456789
* .NET
* ENC: EAAAADI2NzQ0YTY4YjUzZGQ4N2GfHsbuE8t1/hhwz3v9isJ1
*/
static void Main(string[] args)
{
//Use the values from Oracle
string inputText = "Test";
string encryptedTextValue = "7D2894678D46C769B3001BD75F603E3C";
string encryptPrivateKey = "123456789";
string encryptSharedIV = "26744a68b53dd87a";
Console.WriteLine("******************** Initial values from Oracle ******************");
Console.WriteLine("inputText: '{0}'", inputText);
Console.WriteLine("encryptedTextValue: '{0}'", encryptedTextValue);
Console.WriteLine("encryptPrivateKey: '{0}'", encryptPrivateKey);
Console.WriteLine("encryptSharedIV: '{0}'", encryptSharedIV);
Console.WriteLine();
//This is just here to convert the Encrypted byte array to a string for viewing purposes.
UTF8Encoding UTF = new UTF8Encoding();
byte[] inputTextByte = UTF.GetBytes(inputText);
byte[] encryptedTextValueByte = UTF.GetBytes(encryptedTextValue);
byte[] encryptPrivateKeyByte = UTF.GetBytes(encryptPrivateKey);
byte[] encryptSharedIvByte = UTF.GetBytes(encryptSharedIV);
string Encrypted_Text;
//string Decrypted;
try
{
Console.WriteLine("********************Encryption Example******************");
Console.WriteLine("Plain text is: '{0}'", inputText);
Encrypted_Text = EncryptOracleAES(inputTextByte, encryptPrivateKeyByte, encryptSharedIvByte);
Console.WriteLine("Encrypted text is: '{0}'", Encrypted_Text);
Console.WriteLine();
/*Console.WriteLine("********************Decryption Example******************");
Console.WriteLine("Input Encrypted text is '{0}'", XXXXXXXXXXXX);
Decrypted = "";
Console.WriteLine("Decrypted text is: '{0}'", Decrypted);*/
}
catch (Exception e)
{
Console.WriteLine("Exception: {0}", e.Message);
}
Console.WriteLine("Press enter to exit");
Console.ReadLine();
}
public static string EncryptOracleAES(byte[] plainText, byte[] privateKey, byte[] sharedIVKey)
{
//select UTL_I18N.STRING_TO_RAW('Test', 'AL32UTF8') from dual;
//54657374
string outStr = null; // Encrypted string to return
AesManaged aesAlg = null;
try
{
aesAlg = new AesManaged();
aesAlg.Key = privateKey;
aesAlg.KeySize = 256;
aesAlg.BlockSize = 128;
aesAlg.Padding = PaddingMode.PKCS7; //Same as PKCS5/7
aesAlg.Mode = CipherMode.CBC;
aesAlg.IV = sharedIVKey;
ICryptoTransform encryptor = aesAlg.CreateEncryptor(aesAlg.Key, sharedIVKey);
using (MemoryStream msEncrypt = new MemoryStream())
{
// prepend the IV
msEncrypt.Write(BitConverter.GetBytes(aesAlg.IV.Length), 0, sizeof(int));
msEncrypt.Write(aesAlg.IV, 0, aesAlg.IV.Length);
using (CryptoStream csEncrypt = new CryptoStream(msEncrypt, encryptor, CryptoStreamMode.Write))
{
using (StreamWriter swEncrypt = new StreamWriter(csEncrypt))
{
//Write all data to the stream.
swEncrypt.Write(plainText);
}
}
outStr = Convert.ToBase64String(msEncrypt.ToArray());
}
}
finally
{
if (aesAlg != null)
aesAlg.Clear();
}
// Return the encrypted bytes from the memory stream.
return outStr;
}
}
}
但我的结果并不一致:
******************** Initial values from Oracle ******************
inputText: 'Test'
encryptedTextValue: '7D2894678D46C769B3001BD75F603E3C'
encryptPrivateKey: '123456789'
encryptSharedIV: '26744a68b53dd87a'
********************Encryption Example******************
Plain text is: 'Test'
Encrypted text is: 'EAAAADI2NzQ0YTY4YjUzZGQ4N2FCXNXYzo2xWZym3dNFwCSJ'
我缺少什么?
'UTF.GetBytes( “AB”)'将返回字符代码字符A和B'{65,66}'不是您需要的'0xAB'(171)的字节值。您需要[将十六进制字符串转换为字节数组](http://stackoverflow.com/questions/321370/how-can-i-convert-a-hex-string-to-a-byte-array) –
对不起我得到了逻辑,但没有看到何时需要进行转换。我使用了LINQ调用:public static byte [] StringToByteArray(string hex) { return Enumerable.Range(0,hex.Length) .Where(x => x%2 == 0) .Select(x = > Convert.ToByte(hex.Substring(x,2),16)) .ToArray(); } – AhsenB
您可以用链接答案 –