SubjectPublicKeyInfoFactory.cs 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217
  1. #if !BESTHTTP_DISABLE_ALTERNATE_SSL && (!UNITY_WEBGL || UNITY_EDITOR)
  2. #pragma warning disable
  3. using System;
  4. using BestHTTP.SecureProtocol.Org.BouncyCastle.Crypto.Parameters;
  5. using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1;
  6. using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.CryptoPro;
  7. using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.EdEC;
  8. using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.Oiw;
  9. using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.Pkcs;
  10. using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.X509;
  11. using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.X9;
  12. using BestHTTP.SecureProtocol.Org.BouncyCastle.Crypto;
  13. using BestHTTP.SecureProtocol.Org.BouncyCastle.Math;
  14. using BestHTTP.SecureProtocol.Org.BouncyCastle.Math.EC;
  15. using BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities;
  16. namespace BestHTTP.SecureProtocol.Org.BouncyCastle.X509
  17. {
  18. /// <summary>
  19. /// A factory to produce Public Key Info Objects.
  20. /// </summary>
  21. public sealed class SubjectPublicKeyInfoFactory
  22. {
  23. private SubjectPublicKeyInfoFactory()
  24. {
  25. }
  26. /// <summary>
  27. /// Create a Subject Public Key Info object for a given public key.
  28. /// </summary>
  29. /// <param name="publicKey">One of ElGammalPublicKeyParameters, DSAPublicKeyParameter, DHPublicKeyParameters, RsaKeyParameters or ECPublicKeyParameters</param>
  30. /// <returns>A subject public key info object.</returns>
  31. /// <exception cref="Exception">Throw exception if object provided is not one of the above.</exception>
  32. public static SubjectPublicKeyInfo CreateSubjectPublicKeyInfo(
  33. AsymmetricKeyParameter publicKey)
  34. {
  35. if (publicKey == null)
  36. throw new ArgumentNullException("publicKey");
  37. if (publicKey.IsPrivate)
  38. throw new ArgumentException("Private key passed - public key expected.", "publicKey");
  39. if (publicKey is ElGamalPublicKeyParameters)
  40. {
  41. ElGamalPublicKeyParameters _key = (ElGamalPublicKeyParameters)publicKey;
  42. ElGamalParameters kp = _key.Parameters;
  43. SubjectPublicKeyInfo info = new SubjectPublicKeyInfo(
  44. new AlgorithmIdentifier(
  45. OiwObjectIdentifiers.ElGamalAlgorithm,
  46. new ElGamalParameter(kp.P, kp.G).ToAsn1Object()),
  47. new DerInteger(_key.Y));
  48. return info;
  49. }
  50. if (publicKey is DsaPublicKeyParameters)
  51. {
  52. DsaPublicKeyParameters _key = (DsaPublicKeyParameters) publicKey;
  53. DsaParameters kp = _key.Parameters;
  54. Asn1Encodable ae = kp == null
  55. ? null
  56. : new DsaParameter(kp.P, kp.Q, kp.G).ToAsn1Object();
  57. return new SubjectPublicKeyInfo(
  58. new AlgorithmIdentifier(X9ObjectIdentifiers.IdDsa, ae),
  59. new DerInteger(_key.Y));
  60. }
  61. if (publicKey is DHPublicKeyParameters)
  62. {
  63. DHPublicKeyParameters _key = (DHPublicKeyParameters) publicKey;
  64. DHParameters kp = _key.Parameters;
  65. SubjectPublicKeyInfo info = new SubjectPublicKeyInfo(
  66. new AlgorithmIdentifier(
  67. _key.AlgorithmOid,
  68. new DHParameter(kp.P, kp.G, kp.L).ToAsn1Object()),
  69. new DerInteger(_key.Y));
  70. return info;
  71. } // End of DH
  72. if (publicKey is RsaKeyParameters)
  73. {
  74. RsaKeyParameters _key = (RsaKeyParameters) publicKey;
  75. SubjectPublicKeyInfo info = new SubjectPublicKeyInfo(
  76. new AlgorithmIdentifier(PkcsObjectIdentifiers.RsaEncryption, DerNull.Instance),
  77. new RsaPublicKeyStructure(_key.Modulus, _key.Exponent).ToAsn1Object());
  78. return info;
  79. } // End of RSA.
  80. if (publicKey is ECPublicKeyParameters)
  81. {
  82. ECPublicKeyParameters _key = (ECPublicKeyParameters) publicKey;
  83. if (_key.AlgorithmName == "ECGOST3410")
  84. {
  85. if (_key.PublicKeyParamSet == null)
  86. throw BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities.Platform.CreateNotImplementedException("Not a CryptoPro parameter set");
  87. ECPoint q = _key.Q.Normalize();
  88. BigInteger bX = q.AffineXCoord.ToBigInteger();
  89. BigInteger bY = q.AffineYCoord.ToBigInteger();
  90. byte[] encKey = new byte[64];
  91. ExtractBytes(encKey, 0, bX);
  92. ExtractBytes(encKey, 32, bY);
  93. Gost3410PublicKeyAlgParameters gostParams = new Gost3410PublicKeyAlgParameters(
  94. _key.PublicKeyParamSet, CryptoProObjectIdentifiers.GostR3411x94CryptoProParamSet);
  95. AlgorithmIdentifier algID = new AlgorithmIdentifier(
  96. CryptoProObjectIdentifiers.GostR3410x2001,
  97. gostParams.ToAsn1Object());
  98. return new SubjectPublicKeyInfo(algID, new DerOctetString(encKey));
  99. }
  100. else
  101. {
  102. X962Parameters x962;
  103. if (_key.PublicKeyParamSet == null)
  104. {
  105. ECDomainParameters kp = _key.Parameters;
  106. X9ECParameters ecP = new X9ECParameters(kp.Curve, kp.G, kp.N, kp.H, kp.GetSeed());
  107. x962 = new X962Parameters(ecP);
  108. }
  109. else
  110. {
  111. x962 = new X962Parameters(_key.PublicKeyParamSet);
  112. }
  113. byte[] pubKey = _key.Q.GetEncoded(false);
  114. AlgorithmIdentifier algID = new AlgorithmIdentifier(
  115. X9ObjectIdentifiers.IdECPublicKey, x962.ToAsn1Object());
  116. return new SubjectPublicKeyInfo(algID, pubKey);
  117. }
  118. } // End of EC
  119. if (publicKey is Gost3410PublicKeyParameters)
  120. {
  121. Gost3410PublicKeyParameters _key = (Gost3410PublicKeyParameters) publicKey;
  122. if (_key.PublicKeyParamSet == null)
  123. throw BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities.Platform.CreateNotImplementedException("Not a CryptoPro parameter set");
  124. byte[] keyEnc = _key.Y.ToByteArrayUnsigned();
  125. byte[] keyBytes = new byte[keyEnc.Length];
  126. for (int i = 0; i != keyBytes.Length; i++)
  127. {
  128. keyBytes[i] = keyEnc[keyEnc.Length - 1 - i]; // must be little endian
  129. }
  130. Gost3410PublicKeyAlgParameters algParams = new Gost3410PublicKeyAlgParameters(
  131. _key.PublicKeyParamSet, CryptoProObjectIdentifiers.GostR3411x94CryptoProParamSet);
  132. AlgorithmIdentifier algID = new AlgorithmIdentifier(
  133. CryptoProObjectIdentifiers.GostR3410x94,
  134. algParams.ToAsn1Object());
  135. return new SubjectPublicKeyInfo(algID, new DerOctetString(keyBytes));
  136. }
  137. if (publicKey is X448PublicKeyParameters)
  138. {
  139. X448PublicKeyParameters key = (X448PublicKeyParameters)publicKey;
  140. return new SubjectPublicKeyInfo(new AlgorithmIdentifier(EdECObjectIdentifiers.id_X448), key.GetEncoded());
  141. }
  142. if (publicKey is X25519PublicKeyParameters)
  143. {
  144. X25519PublicKeyParameters key = (X25519PublicKeyParameters)publicKey;
  145. return new SubjectPublicKeyInfo(new AlgorithmIdentifier(EdECObjectIdentifiers.id_X25519), key.GetEncoded());
  146. }
  147. if (publicKey is Ed448PublicKeyParameters)
  148. {
  149. Ed448PublicKeyParameters key = (Ed448PublicKeyParameters)publicKey;
  150. return new SubjectPublicKeyInfo(new AlgorithmIdentifier(EdECObjectIdentifiers.id_Ed448), key.GetEncoded());
  151. }
  152. if (publicKey is Ed25519PublicKeyParameters)
  153. {
  154. Ed25519PublicKeyParameters key = (Ed25519PublicKeyParameters)publicKey;
  155. return new SubjectPublicKeyInfo(new AlgorithmIdentifier(EdECObjectIdentifiers.id_Ed25519), key.GetEncoded());
  156. }
  157. throw new ArgumentException("Class provided no convertible: " + BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities.Platform.GetTypeName(publicKey));
  158. }
  159. private static void ExtractBytes(
  160. byte[] encKey,
  161. int offset,
  162. BigInteger bI)
  163. {
  164. byte[] val = bI.ToByteArray();
  165. int n = (bI.BitLength + 7) / 8;
  166. for (int i = 0; i < n; ++i)
  167. {
  168. encKey[offset + i] = val[val.Length - 1 - i];
  169. }
  170. }
  171. }
  172. }
  173. #pragma warning restore
  174. #endif