123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281 |
- #if !BESTHTTP_DISABLE_ALTERNATE_SSL && (!UNITY_WEBGL || UNITY_EDITOR)
- #pragma warning disable
- using System;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Crypto.Parameters;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.CryptoPro;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.EdEC;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.Oiw;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.Pkcs;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.Rosstandart;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.X509;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.X9;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Crypto;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Math;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Math.EC;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities;
- using BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities.Encoders;
- namespace BestHTTP.SecureProtocol.Org.BouncyCastle.X509
- {
- /// <summary>
- /// A factory to produce Public Key Info Objects.
- /// </summary>
- public sealed class SubjectPublicKeyInfoFactory
- {
- private SubjectPublicKeyInfoFactory()
- {
- }
- /// <summary>
- /// Create a Subject Public Key Info object for a given public key.
- /// </summary>
- /// <param name="publicKey">One of ElGammalPublicKeyParameters, DSAPublicKeyParameter, DHPublicKeyParameters, RsaKeyParameters or ECPublicKeyParameters</param>
- /// <returns>A subject public key info object.</returns>
- /// <exception cref="Exception">Throw exception if object provided is not one of the above.</exception>
- public static SubjectPublicKeyInfo CreateSubjectPublicKeyInfo(
- AsymmetricKeyParameter publicKey)
- {
- if (publicKey == null)
- throw new ArgumentNullException("publicKey");
- if (publicKey.IsPrivate)
- throw new ArgumentException("Private key passed - public key expected.", "publicKey");
- if (publicKey is ElGamalPublicKeyParameters)
- {
- ElGamalPublicKeyParameters _key = (ElGamalPublicKeyParameters)publicKey;
- ElGamalParameters kp = _key.Parameters;
- SubjectPublicKeyInfo info = new SubjectPublicKeyInfo(
- new AlgorithmIdentifier(
- OiwObjectIdentifiers.ElGamalAlgorithm,
- new ElGamalParameter(kp.P, kp.G).ToAsn1Object()),
- new DerInteger(_key.Y));
- return info;
- }
- if (publicKey is DsaPublicKeyParameters)
- {
- DsaPublicKeyParameters _key = (DsaPublicKeyParameters) publicKey;
- DsaParameters kp = _key.Parameters;
- Asn1Encodable ae = kp == null
- ? null
- : new DsaParameter(kp.P, kp.Q, kp.G).ToAsn1Object();
- return new SubjectPublicKeyInfo(
- new AlgorithmIdentifier(X9ObjectIdentifiers.IdDsa, ae),
- new DerInteger(_key.Y));
- }
- if (publicKey is DHPublicKeyParameters)
- {
- DHPublicKeyParameters _key = (DHPublicKeyParameters) publicKey;
- DHParameters kp = _key.Parameters;
- SubjectPublicKeyInfo info = new SubjectPublicKeyInfo(
- new AlgorithmIdentifier(
- _key.AlgorithmOid,
- new DHParameter(kp.P, kp.G, kp.L).ToAsn1Object()),
- new DerInteger(_key.Y));
- return info;
- } // End of DH
- if (publicKey is RsaKeyParameters)
- {
- RsaKeyParameters _key = (RsaKeyParameters) publicKey;
- SubjectPublicKeyInfo info = new SubjectPublicKeyInfo(
- new AlgorithmIdentifier(PkcsObjectIdentifiers.RsaEncryption, DerNull.Instance),
- new RsaPublicKeyStructure(_key.Modulus, _key.Exponent).ToAsn1Object());
- return info;
- } // End of RSA.
- if (publicKey is ECPublicKeyParameters)
- {
-
- ECPublicKeyParameters _key = (ECPublicKeyParameters) publicKey;
- if (_key.Parameters is ECGost3410Parameters)
- {
- ECGost3410Parameters gostParams = (ECGost3410Parameters)_key.Parameters;
- BigInteger bX = _key.Q.AffineXCoord.ToBigInteger();
- BigInteger bY = _key.Q.AffineYCoord.ToBigInteger();
- bool is512 = (bX.BitLength > 256);
- Gost3410PublicKeyAlgParameters parameters = new Gost3410PublicKeyAlgParameters(
- gostParams.PublicKeyParamSet,
- gostParams.DigestParamSet,
- gostParams.EncryptionParamSet);
- int encKeySize;
- int offset;
- DerObjectIdentifier algIdentifier;
- if (is512)
- {
- encKeySize = 128;
- offset = 64;
- algIdentifier = RosstandartObjectIdentifiers.id_tc26_gost_3410_12_512;
- }
- else
- {
- encKeySize = 64;
- offset = 32;
- algIdentifier = RosstandartObjectIdentifiers.id_tc26_gost_3410_12_256;
- }
- byte[] encKey = new byte[encKeySize];
-
- ExtractBytes(encKey, encKeySize / 2, 0, bX);
- ExtractBytes(encKey, encKeySize / 2, offset, bY);
-
- return new SubjectPublicKeyInfo(new AlgorithmIdentifier(algIdentifier, parameters), new DerOctetString(encKey));
-
- } // End of ECGOST3410_2012
- if (_key.AlgorithmName == "ECGOST3410")
- {
- if (_key.PublicKeyParamSet == null)
- throw BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities.Platform.CreateNotImplementedException("Not a CryptoPro parameter set");
- ECPoint q = _key.Q.Normalize();
- BigInteger bX = q.AffineXCoord.ToBigInteger();
- BigInteger bY = q.AffineYCoord.ToBigInteger();
- byte[] encKey = new byte[64];
- ExtractBytes(encKey, 0, bX);
- ExtractBytes(encKey, 32, bY);
- Gost3410PublicKeyAlgParameters gostParams = new Gost3410PublicKeyAlgParameters(
- _key.PublicKeyParamSet, CryptoProObjectIdentifiers.GostR3411x94CryptoProParamSet);
- AlgorithmIdentifier algID = new AlgorithmIdentifier(
- CryptoProObjectIdentifiers.GostR3410x2001,
- gostParams.ToAsn1Object());
- return new SubjectPublicKeyInfo(algID, new DerOctetString(encKey));
- }
- else
- {
- X962Parameters x962;
- if (_key.PublicKeyParamSet == null)
- {
- ECDomainParameters kp = _key.Parameters;
- X9ECParameters ecP = new X9ECParameters(kp.Curve, kp.G, kp.N, kp.H, kp.GetSeed());
- x962 = new X962Parameters(ecP);
- }
- else
- {
- x962 = new X962Parameters(_key.PublicKeyParamSet);
- }
- byte[] pubKey = _key.Q.GetEncoded(false);
- AlgorithmIdentifier algID = new AlgorithmIdentifier(
- X9ObjectIdentifiers.IdECPublicKey, x962.ToAsn1Object());
- return new SubjectPublicKeyInfo(algID, pubKey);
- }
- } // End of EC
- if (publicKey is Gost3410PublicKeyParameters)
- {
- Gost3410PublicKeyParameters _key = (Gost3410PublicKeyParameters) publicKey;
- if (_key.PublicKeyParamSet == null)
- throw BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities.Platform.CreateNotImplementedException("Not a CryptoPro parameter set");
- byte[] keyEnc = _key.Y.ToByteArrayUnsigned();
- byte[] keyBytes = new byte[keyEnc.Length];
- for (int i = 0; i != keyBytes.Length; i++)
- {
- keyBytes[i] = keyEnc[keyEnc.Length - 1 - i]; // must be little endian
- }
- Gost3410PublicKeyAlgParameters algParams = new Gost3410PublicKeyAlgParameters(
- _key.PublicKeyParamSet, CryptoProObjectIdentifiers.GostR3411x94CryptoProParamSet);
- AlgorithmIdentifier algID = new AlgorithmIdentifier(
- CryptoProObjectIdentifiers.GostR3410x94,
- algParams.ToAsn1Object());
- return new SubjectPublicKeyInfo(algID, new DerOctetString(keyBytes));
- }
- if (publicKey is X448PublicKeyParameters)
- {
- X448PublicKeyParameters key = (X448PublicKeyParameters)publicKey;
- return new SubjectPublicKeyInfo(new AlgorithmIdentifier(EdECObjectIdentifiers.id_X448), key.GetEncoded());
- }
- if (publicKey is X25519PublicKeyParameters)
- {
- X25519PublicKeyParameters key = (X25519PublicKeyParameters)publicKey;
- return new SubjectPublicKeyInfo(new AlgorithmIdentifier(EdECObjectIdentifiers.id_X25519), key.GetEncoded());
- }
- if (publicKey is Ed448PublicKeyParameters)
- {
- Ed448PublicKeyParameters key = (Ed448PublicKeyParameters)publicKey;
- return new SubjectPublicKeyInfo(new AlgorithmIdentifier(EdECObjectIdentifiers.id_Ed448), key.GetEncoded());
- }
- if (publicKey is Ed25519PublicKeyParameters)
- {
- Ed25519PublicKeyParameters key = (Ed25519PublicKeyParameters)publicKey;
- return new SubjectPublicKeyInfo(new AlgorithmIdentifier(EdECObjectIdentifiers.id_Ed25519), key.GetEncoded());
- }
- throw new ArgumentException("Class provided no convertible: " + BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities.Platform.GetTypeName(publicKey));
- }
- private static void ExtractBytes(
- byte[] encKey,
- int offset,
- BigInteger bI)
- {
- byte[] val = bI.ToByteArray();
- int n = (bI.BitLength + 7) / 8;
- for (int i = 0; i < n; ++i)
- {
- encKey[offset + i] = val[val.Length - 1 - i];
- }
- }
- private static void ExtractBytes(byte[] encKey, int size, int offSet, BigInteger bI)
- {
- byte[] val = bI.ToByteArray();
- if (val.Length < size)
- {
- byte[] tmp = new byte[size];
- Array.Copy(val, 0, tmp, tmp.Length - val.Length, val.Length);
- val = tmp;
- }
- for (int i = 0; i != size; i++)
- {
- encKey[offSet + i] = val[val.Length - 1 - i];
- }
- }
- }
- }
- #pragma warning restore
- #endif
|