我正在使用 (EC)DHE 加密类型 x25519,我在计算共享密钥时遇到了一个大问题。
Microsoft 没有椭圆曲线 x25519 的默认实现。然而,他们对加密 Diffie Hellman 对象的实现允许我们定义自己的曲线。
一旦我们定义了要使用的曲线 (x25519),我们就可以使用 Microsoft 的ECDiffieHellmanCng
实现来导入曲线、生成密钥并创建共享密钥。
感谢Yasar_yy关于 x25519 上一个不相关主题的问题,他为我们实现了曲线。
ECCurve
我们使用类实现曲线
public static ECCurve Curve25519 {get; init;} = new ECCurve()
{
CurveType = ECCurve.ECCurveType.PrimeMontgomery,
B = new byte[] { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1 },
A = new byte[] { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0x07, 0x6d, 0x06 }, // 486662
G = new ECPoint()
{
X = new byte[] { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 9 },
Y = new byte[] { 0x20, 0xae, 0x19, 0xa1, 0xb8, 0xa0, 0x86, 0xb4, 0xe0, 0x1e, 0xdd, 0x2c, 0x77, 0x48, 0xd1, 0x4c,
0x92, 0x3d, 0x4d, 0x7e, 0x6d, 0x7c, 0x61, 0xb2, 0x29, 0xe9, 0xc5, 0xa2, 0x7e, 0xce, 0xd3, 0xd9 }
},
Prime = new byte[] { 0x7f, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xed },
//Prime = new byte[] { 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1 },
Order = new byte[] { 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
0x14, 0xde, 0xf9, 0xde, 0xa2, 0xf7, 0x9c, 0xd6, 0x58, 0x12, 0x63, 0x1a, 0x5c, 0xf5, 0xd3, 0xed },
Cofactor = new byte[] { 8 }
};
在我们定义了我们想要使用的曲线之后,我们只需要为曲线生成键,其余的都是使用ECDiffieHellmanCng
类的标准。
public class Person
{
public string Name {get; set;}
public byte[] PublicKey {get; private set;}
public byte[] PrivateKey {get; private set;}
private ECParameters EncryptionParameters;
public void GenerateInitialKeys()
{
using (ECDiffieHellmanCng bob = new ECDiffieHellmanCng())
{
// we have to generate the key explicitly using the curve we defined, the auto-generated keys can not be used
bob.GenerateKey(Curve25519);
// assign what algorithms for derivation and hashing should be used
bob.KeyDerivationFunction = ECDiffieHellmanKeyDerivationFunction.Hash;
bob.HashAlgorithm = CngAlgorithm.Sha256;
// save the keys
PublicKey = bob.PublicKey.ToByteArray();
PrivateKey = bob.ExportECPrivateKey();
// export the curve information so we can create a shared secret later
EncryptionParameters = bob.ExportParameters(true);
}
}
public void CreateSharedSecret(byte[] OtherPublicKey)
{
if(EncryptionParameters is null)
{
throw new NullReferenceException($"{nameof(EncryptionParameters)} must not be null, invoke {nameof(GenerateInitialKeys)} to generate new keys and {nameof(EncryptionParameters)}");
}
using (ECDiffieHellmanCng bob = new ECDiffieHellmanCng())
{
// import the curve information from when generated our initial keys
bob.ImportParameters(EncryptionParameters);
// assign what algorithms for derivation and hashing should be used
bob.KeyDerivationFunction = ECDiffieHellmanKeyDerivationFunction.Hash;
bob.HashAlgorithm = CngAlgorithm.Sha256;
// import the byte[] as a key, note EccFullPublicBlob is required, otherwise a generic runtime error will throw and will contain absolutely no useful information
CngKey otherKey = CngKey.Import(OtherPublicKey, CngKeyBlobFormat.EccFullPublicBlob)
// Save the shared secret
PrivateKey = bob.DeriveKeyMaterial(otherKey);
}
}
// This is just here to visually verify the private keys for equality because we don't speak or read byte[]
public string ExportPrivateKey()
{
return Convert.ToBase64String(PrivateKey ?? Array.Empty<byte>());
}
}
要使用这个非常基本的类,我们只需调用GenerateKeys
和 随后CreateSharedSecret
。
Person alice = new();
Person bob = new();
alice.GenerateInitialKeys();
bob.GenerateInitialKeys();
alice.CreateSharedSecret(bob.PublicKey);
bob.CreateSharedSecret(alice.PublicKey);
Console.WriteLine(alice.ExportPrivateKey() == bob.ExportPrivateKey());
// ideally should output: true