gpt4 book ai didi

sql-server - C# 从 SQL Server EncryptByPassPhrase 解密字节?

转载 作者:行者123 更新时间:2023-12-02 22:39:18 28 4
gpt4 key购买 nike

已关注 Replicate T-SQL DecryptByPassPhrase in C# ,我无法使用 MSSQL 进行简单加密以在 C# 中解密。某些列中的加密值是必要的,因为表格会定期导出到 Excel 和 Access 中,因此简单的加密足以“阻止”值,而无需让开发人员(重新)执行 View 等操作。

在 SQL Server 2012 中:

    select EncryptByPassPhrase( N'hello' , N'world'  ) 
-- returns 0x01000000AA959FFB3A8E4B06B734051437E198C8B72000A058ACE91D617123DA102287EB

在 C# 中:

byte[] buf = System.Text.Encoding.UTF8.GetBytes( "0x010000003A95FA870ED699A5F90D33C2BF01491D9132F61BA162998E96F37117AF5DA0905D51EB6FB298EC88" );
// bytes emitted from the database
var cp = new TripleDESCryptoServiceProvider();
var m = new MemoryStream(buf);
cp.Key = System.Text.Encoding.UTF8.GetBytes( "hello" ); // throws
cp.IV = System.Text.Encoding.UTF8.GetBytes( "hello" ); // throws
CryptoStream cs = new CryptoStream( m , cp.CreateDecryptor( cp.Key , cp.IV ) , CryptoStreamMode.Read );
StreamReader reader = new StreamReader( cs );
string plainText = reader.ReadToEnd();

有效的 C# 代码应该是什么样子?

谢谢。

最佳答案

SQL Server 2017 使用 SHA256 密码哈希 + AES-256 加密

旧版本使用 SHA1 密码哈希 + 3DES-128 加密

IV 大小与 block 大小相同:AES = 128 位,3DES = 64 位

填充模式:PKCS #7加密方式:CBC

2017年服务器加密的数据以“0x02”开头,旧版本以“0x01”开头。

// Example decrypt:
// UInt32 - "magic" (0xbaadf00d): 0d f0 ad ba
// UInt16 - unknown (always zero): 00 00
// UInt16 - decrypted data length (16): 10 00
// byte[] - decrypted data: 4c 65 74 54 68 65 53 75 6e 53 68 69 6e 69 6e 67

DecryptCombined("0x02000000266AD4F387FA9474E825B013B0232E73A398A5F72B79BC90D63BD1E45AE3AA5518828D187125BECC285D55FA7CAFED61", "Radames");
DecryptCombined("0x010000007854E155CEE338D5E34808BA95367D506B97C63FB5114DD4CE687FE457C1B5D5", "banana");


void DecryptCombined(string FromSql, string Password)
{
// Encode password as UTF16-LE
byte[] passwordBytes = Encoding.Unicode.GetBytes(Password);

// Remove leading "0x"
FromSql = FromSql.Substring(2);

int version = BitConverter.ToInt32(StringToByteArray(FromSql.Substring(0, 8)), 0);
byte[] encrypted = null;

HashAlgorithm hashAlgo = null;
SymmetricAlgorithm cryptoAlgo = null;
int keySize = (version == 1 ? 16 : 32);

if (version == 1)
{
hashAlgo = SHA1.Create();
cryptoAlgo = TripleDES.Create();
cryptoAlgo.IV = StringToByteArray(FromSql.Substring(8, 16));
encrypted = StringToByteArray(FromSql.Substring(24));
}
else if (version == 2)
{
hashAlgo = SHA256.Create();
cryptoAlgo = Aes.Create();
cryptoAlgo.IV = StringToByteArray(FromSql.Substring(8, 32));
encrypted = StringToByteArray(FromSql.Substring(40));
}
else
{
throw new Exception("Unsupported encryption");
}

cryptoAlgo.Padding = PaddingMode.PKCS7;
cryptoAlgo.Mode = CipherMode.CBC;

hashAlgo.TransformFinalBlock(passwordBytes, 0, passwordBytes.Length);
cryptoAlgo.Key = hashAlgo.Hash.Take(keySize).ToArray();

byte[] decrypted = cryptoAlgo.CreateDecryptor().TransformFinalBlock(encrypted, 0, encrypted.Length);
int decryptLength = BitConverter.ToInt16(decrypted, 6);
UInt32 magic = BitConverter.ToUInt32(decrypted, 0);
if (magic != 0xbaadf00d)
{
throw new Exception("Decrypt failed");
}

byte[] decryptedData = decrypted.Skip(8).ToArray();
bool isUtf16 = (Array.IndexOf(decryptedData, (byte)0) != -1);
string decryptText = (isUtf16 ? Encoding.Unicode.GetString(decryptedData) : Encoding.UTF8.GetString(decryptedData));

Console.WriteLine("Result: {0}", decryptText);
}

// Method taken from https://stackoverflow.com/questions/321370/how-can-i-convert-a-hex-string-to-a-byte-array?answertab=votes#tab-top
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();
}

关于sql-server - C# 从 SQL Server EncryptByPassPhrase 解密字节?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/21684733/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com