You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
349 lines
10 KiB
349 lines
10 KiB
8 months ago
|
#if !BESTHTTP_DISABLE_ALTERNATE_SSL && (!UNITY_WEBGL || UNITY_EDITOR)
|
||
|
#pragma warning disable
|
||
|
using System;
|
||
|
using System.Collections;
|
||
|
|
||
|
using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1;
|
||
|
using BestHTTP.SecureProtocol.Org.BouncyCastle.Asn1.X509;
|
||
|
using BestHTTP.SecureProtocol.Org.BouncyCastle.Crypto;
|
||
|
using BestHTTP.SecureProtocol.Org.BouncyCastle.Crypto.Operators;
|
||
|
using BestHTTP.SecureProtocol.Org.BouncyCastle.Math;
|
||
|
using BestHTTP.SecureProtocol.Org.BouncyCastle.Security;
|
||
|
using BestHTTP.SecureProtocol.Org.BouncyCastle.Security.Certificates;
|
||
|
using BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities;
|
||
|
using BestHTTP.SecureProtocol.Org.BouncyCastle.X509.Extension;
|
||
|
|
||
|
namespace BestHTTP.SecureProtocol.Org.BouncyCastle.X509
|
||
|
{
|
||
|
/// <summary>
|
||
|
/// A class to Generate Version 3 X509Certificates.
|
||
|
/// </summary>
|
||
|
public class X509V3CertificateGenerator
|
||
|
{
|
||
|
private readonly X509ExtensionsGenerator extGenerator = new X509ExtensionsGenerator();
|
||
|
|
||
|
private V3TbsCertificateGenerator tbsGen;
|
||
|
private DerObjectIdentifier sigOid;
|
||
|
private AlgorithmIdentifier sigAlgId;
|
||
|
private string signatureAlgorithm;
|
||
|
|
||
|
public X509V3CertificateGenerator()
|
||
|
{
|
||
|
tbsGen = new V3TbsCertificateGenerator();
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Reset the Generator.
|
||
|
/// </summary>
|
||
|
public void Reset()
|
||
|
{
|
||
|
tbsGen = new V3TbsCertificateGenerator();
|
||
|
extGenerator.Reset();
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Set the certificate's serial number.
|
||
|
/// </summary>
|
||
|
/// <remarks>Make serial numbers long, if you have no serial number policy make sure the number is at least 16 bytes of secure random data.
|
||
|
/// You will be surprised how ugly a serial number collision can Get.</remarks>
|
||
|
/// <param name="serialNumber">The serial number.</param>
|
||
|
public void SetSerialNumber(
|
||
|
BigInteger serialNumber)
|
||
|
{
|
||
|
if (serialNumber.SignValue <= 0)
|
||
|
{
|
||
|
throw new ArgumentException("serial number must be a positive integer", "serialNumber");
|
||
|
}
|
||
|
|
||
|
tbsGen.SetSerialNumber(new DerInteger(serialNumber));
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Set the distinguished name of the issuer.
|
||
|
/// The issuer is the entity which is signing the certificate.
|
||
|
/// </summary>
|
||
|
/// <param name="issuer">The issuer's DN.</param>
|
||
|
public void SetIssuerDN(
|
||
|
X509Name issuer)
|
||
|
{
|
||
|
tbsGen.SetIssuer(issuer);
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Set the date that this certificate is to be valid from.
|
||
|
/// </summary>
|
||
|
/// <param name="date"/>
|
||
|
public void SetNotBefore(
|
||
|
DateTime date)
|
||
|
{
|
||
|
tbsGen.SetStartDate(new Time(date));
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Set the date after which this certificate will no longer be valid.
|
||
|
/// </summary>
|
||
|
/// <param name="date"/>
|
||
|
public void SetNotAfter(
|
||
|
DateTime date)
|
||
|
{
|
||
|
tbsGen.SetEndDate(new Time(date));
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Set the DN of the entity that this certificate is about.
|
||
|
/// </summary>
|
||
|
/// <param name="subject"/>
|
||
|
public void SetSubjectDN(
|
||
|
X509Name subject)
|
||
|
{
|
||
|
tbsGen.SetSubject(subject);
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Set the public key that this certificate identifies.
|
||
|
/// </summary>
|
||
|
/// <param name="publicKey"/>
|
||
|
public void SetPublicKey(
|
||
|
AsymmetricKeyParameter publicKey)
|
||
|
{
|
||
|
tbsGen.SetSubjectPublicKeyInfo(SubjectPublicKeyInfoFactory.CreateSubjectPublicKeyInfo(publicKey));
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Set the signature algorithm that will be used to sign this certificate.
|
||
|
/// </summary>
|
||
|
/// <param name="signatureAlgorithm"/>
|
||
|
|
||
|
public void SetSignatureAlgorithm(
|
||
|
string signatureAlgorithm)
|
||
|
{
|
||
|
this.signatureAlgorithm = signatureAlgorithm;
|
||
|
|
||
|
try
|
||
|
{
|
||
|
sigOid = X509Utilities.GetAlgorithmOid(signatureAlgorithm);
|
||
|
}
|
||
|
catch (Exception)
|
||
|
{
|
||
|
throw new ArgumentException("Unknown signature type requested: " + signatureAlgorithm);
|
||
|
}
|
||
|
|
||
|
sigAlgId = X509Utilities.GetSigAlgID(sigOid, signatureAlgorithm);
|
||
|
|
||
|
tbsGen.SetSignature(sigAlgId);
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Set the subject unique ID - note: it is very rare that it is correct to do this.
|
||
|
/// </summary>
|
||
|
/// <param name="uniqueID"/>
|
||
|
public void SetSubjectUniqueID(
|
||
|
bool[] uniqueID)
|
||
|
{
|
||
|
tbsGen.SetSubjectUniqueID(booleanToBitString(uniqueID));
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Set the issuer unique ID - note: it is very rare that it is correct to do this.
|
||
|
/// </summary>
|
||
|
/// <param name="uniqueID"/>
|
||
|
public void SetIssuerUniqueID(
|
||
|
bool[] uniqueID)
|
||
|
{
|
||
|
tbsGen.SetIssuerUniqueID(booleanToBitString(uniqueID));
|
||
|
}
|
||
|
|
||
|
private DerBitString booleanToBitString(
|
||
|
bool[] id)
|
||
|
{
|
||
|
byte[] bytes = new byte[(id.Length + 7) / 8];
|
||
|
|
||
|
for (int i = 0; i != id.Length; i++)
|
||
|
{
|
||
|
if (id[i])
|
||
|
{
|
||
|
bytes[i / 8] |= (byte)(1 << ((7 - (i % 8))));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int pad = id.Length % 8;
|
||
|
|
||
|
if (pad == 0)
|
||
|
{
|
||
|
return new DerBitString(bytes);
|
||
|
}
|
||
|
|
||
|
return new DerBitString(bytes, 8 - pad);
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Add a given extension field for the standard extensions tag (tag 3).
|
||
|
/// </summary>
|
||
|
/// <param name="oid">string containing a dotted decimal Object Identifier.</param>
|
||
|
/// <param name="critical">Is it critical.</param>
|
||
|
/// <param name="extensionValue">The value.</param>
|
||
|
public void AddExtension(
|
||
|
string oid,
|
||
|
bool critical,
|
||
|
Asn1Encodable extensionValue)
|
||
|
{
|
||
|
extGenerator.AddExtension(new DerObjectIdentifier(oid), critical, extensionValue);
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Add an extension to this certificate.
|
||
|
/// </summary>
|
||
|
/// <param name="oid">Its Object Identifier.</param>
|
||
|
/// <param name="critical">Is it critical.</param>
|
||
|
/// <param name="extensionValue">The value.</param>
|
||
|
public void AddExtension(
|
||
|
DerObjectIdentifier oid,
|
||
|
bool critical,
|
||
|
Asn1Encodable extensionValue)
|
||
|
{
|
||
|
extGenerator.AddExtension(oid, critical, extensionValue);
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Add an extension using a string with a dotted decimal OID.
|
||
|
/// </summary>
|
||
|
/// <param name="oid">string containing a dotted decimal Object Identifier.</param>
|
||
|
/// <param name="critical">Is it critical.</param>
|
||
|
/// <param name="extensionValue">byte[] containing the value of this extension.</param>
|
||
|
public void AddExtension(
|
||
|
string oid,
|
||
|
bool critical,
|
||
|
byte[] extensionValue)
|
||
|
{
|
||
|
extGenerator.AddExtension(new DerObjectIdentifier(oid), critical, new DerOctetString(extensionValue));
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Add an extension to this certificate.
|
||
|
/// </summary>
|
||
|
/// <param name="oid">Its Object Identifier.</param>
|
||
|
/// <param name="critical">Is it critical.</param>
|
||
|
/// <param name="extensionValue">byte[] containing the value of this extension.</param>
|
||
|
public void AddExtension(
|
||
|
DerObjectIdentifier oid,
|
||
|
bool critical,
|
||
|
byte[] extensionValue)
|
||
|
{
|
||
|
extGenerator.AddExtension(oid, critical, new DerOctetString(extensionValue));
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Add a given extension field for the standard extensions tag (tag 3),
|
||
|
/// copying the extension value from another certificate.
|
||
|
/// </summary>
|
||
|
public void CopyAndAddExtension(
|
||
|
string oid,
|
||
|
bool critical,
|
||
|
X509Certificate cert)
|
||
|
{
|
||
|
CopyAndAddExtension(new DerObjectIdentifier(oid), critical, cert);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* add a given extension field for the standard extensions tag (tag 3)
|
||
|
* copying the extension value from another certificate.
|
||
|
* @throws CertificateParsingException if the extension cannot be extracted.
|
||
|
*/
|
||
|
public void CopyAndAddExtension(
|
||
|
DerObjectIdentifier oid,
|
||
|
bool critical,
|
||
|
X509Certificate cert)
|
||
|
{
|
||
|
Asn1OctetString extValue = cert.GetExtensionValue(oid);
|
||
|
|
||
|
if (extValue == null)
|
||
|
{
|
||
|
throw new CertificateParsingException("extension " + oid + " not present");
|
||
|
}
|
||
|
|
||
|
try
|
||
|
{
|
||
|
Asn1Encodable value = X509ExtensionUtilities.FromExtensionValue(extValue);
|
||
|
|
||
|
this.AddExtension(oid, critical, value);
|
||
|
}
|
||
|
catch (Exception e)
|
||
|
{
|
||
|
throw new CertificateParsingException(e.Message, e);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Generate an X509Certificate.
|
||
|
/// </summary>
|
||
|
/// <param name="privateKey">The private key of the issuer that is signing this certificate.</param>
|
||
|
/// <returns>An X509Certificate.</returns>
|
||
|
|
||
|
public X509Certificate Generate(
|
||
|
AsymmetricKeyParameter privateKey)
|
||
|
{
|
||
|
return Generate(privateKey, null);
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Generate an X509Certificate using your own SecureRandom.
|
||
|
/// </summary>
|
||
|
/// <param name="privateKey">The private key of the issuer that is signing this certificate.</param>
|
||
|
/// <param name="random">You Secure Random instance.</param>
|
||
|
/// <returns>An X509Certificate.</returns>
|
||
|
|
||
|
public X509Certificate Generate(
|
||
|
AsymmetricKeyParameter privateKey,
|
||
|
SecureRandom random)
|
||
|
{
|
||
|
return Generate(new Asn1SignatureFactory(signatureAlgorithm, privateKey, random));
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Generate a new X509Certificate using the passed in SignatureCalculator.
|
||
|
/// </summary>
|
||
|
/// <param name="signatureCalculatorFactory">A signature calculator factory with the necessary algorithm details.</param>
|
||
|
/// <returns>An X509Certificate.</returns>
|
||
|
public X509Certificate Generate(ISignatureFactory signatureCalculatorFactory)
|
||
|
{
|
||
|
tbsGen.SetSignature ((AlgorithmIdentifier)signatureCalculatorFactory.AlgorithmDetails);
|
||
|
|
||
|
if (!extGenerator.IsEmpty)
|
||
|
{
|
||
|
tbsGen.SetExtensions(extGenerator.Generate());
|
||
|
}
|
||
|
|
||
|
TbsCertificateStructure tbsCert = tbsGen.GenerateTbsCertificate();
|
||
|
|
||
|
IStreamCalculator streamCalculator = signatureCalculatorFactory.CreateCalculator();
|
||
|
|
||
|
byte[] encoded = tbsCert.GetDerEncoded();
|
||
|
|
||
|
streamCalculator.Stream.Write(encoded, 0, encoded.Length);
|
||
|
|
||
|
BestHTTP.SecureProtocol.Org.BouncyCastle.Utilities.Platform.Dispose(streamCalculator.Stream);
|
||
|
|
||
|
return GenerateJcaObject(tbsCert, (AlgorithmIdentifier)signatureCalculatorFactory.AlgorithmDetails, ((IBlockResult)streamCalculator.GetResult()).Collect());
|
||
|
}
|
||
|
|
||
|
private X509Certificate GenerateJcaObject(
|
||
|
TbsCertificateStructure tbsCert,
|
||
|
AlgorithmIdentifier sigAlg,
|
||
|
byte[] signature)
|
||
|
{
|
||
|
return new X509Certificate(
|
||
|
new X509CertificateStructure(tbsCert, sigAlg, new DerBitString(signature)));
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Allows enumeration of the signature names supported by the generator.
|
||
|
/// </summary>
|
||
|
public IEnumerable SignatureAlgNames
|
||
|
{
|
||
|
get { return X509Utilities.GetAlgNames(); }
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
#pragma warning restore
|
||
|
#endif
|