using System; using System.Collections.Generic; using System.IO; using Org.BouncyCastle.Asn1; using Org.BouncyCastle.Asn1.Cms; using Org.BouncyCastle.Asn1.Ocsp; using Org.BouncyCastle.Asn1.Sec; using Org.BouncyCastle.Asn1.X509; using Org.BouncyCastle.Asn1.X9; using Org.BouncyCastle.Utilities.Collections; using Org.BouncyCastle.Utilities.IO; using Org.BouncyCastle.X509; namespace Org.BouncyCastle.Cms { internal static class CmsUtilities { // TODO Is there a .NET equivalent to this? // private static readonly Runtime RUNTIME = Runtime.getRuntime(); internal static int MaximumMemory { get { // TODO Is there a .NET equivalent to this? long maxMem = int.MaxValue;//RUNTIME.maxMemory(); if (maxMem > int.MaxValue) { return int.MaxValue; } return (int)maxMem; } } internal static ContentInfo ReadContentInfo(byte[] input) { using (var asn1In = new Asn1InputStream(input)) { return ReadContentInfo(asn1In); } } internal static ContentInfo ReadContentInfo(Stream input) { using (var asn1In = new Asn1InputStream(input, MaximumMemory, leaveOpen: true)) { return ReadContentInfo(asn1In); } } private static ContentInfo ReadContentInfo(Asn1InputStream asn1In) { try { return ContentInfo.GetInstance(asn1In.ReadObject()); } catch (IOException e) { throw new CmsException("IOException reading content.", e); } catch (InvalidCastException e) { throw new CmsException("Malformed content.", e); } catch (ArgumentException e) { throw new CmsException("Malformed content.", e); } } internal static byte[] StreamToByteArray(Stream inStream) { return Streams.ReadAll(inStream); } internal static byte[] StreamToByteArray(Stream inStream, int limit) { return Streams.ReadAllLimited(inStream, limit); } internal static List GetAttributeCertificatesFromStore( IStore attrCertStore) { var result = new List(); if (attrCertStore != null) { foreach (var attrCert in attrCertStore.EnumerateMatches(null)) { result.Add(new DerTaggedObject(false, 2, attrCert.AttributeCertificate)); } } return result; } internal static List GetCertificatesFromStore(IStore certStore) { var result = new List(); if (certStore != null) { foreach (var cert in certStore.EnumerateMatches(null)) { result.Add(cert.CertificateStructure); } } return result; } internal static List GetCrlsFromStore(IStore crlStore) { var result = new List(); if (crlStore != null) { foreach (var crl in crlStore.EnumerateMatches(null)) { result.Add(crl.CertificateList); } } return result; } internal static List GetOtherRevocationInfosFromStore( IStore otherRevocationInfoStore) { var result = new List(); if (otherRevocationInfoStore != null) { foreach (var otherRevocationInfo in otherRevocationInfoStore.EnumerateMatches(null)) { ValidateOtherRevocationInfo(otherRevocationInfo); result.Add(new DerTaggedObject(false, 1, otherRevocationInfo)); } } return result; } internal static List GetOtherRevocationInfosFromStore(IStore otherRevInfoStore, DerObjectIdentifier otherRevInfoFormat) { var result = new List(); if (otherRevInfoStore != null && otherRevInfoFormat != null) { foreach (var otherRevInfo in otherRevInfoStore.EnumerateMatches(null)) { var otherRevocationInfo = new OtherRevocationInfoFormat(otherRevInfoFormat, otherRevInfo); ValidateOtherRevocationInfo(otherRevocationInfo); result.Add(new DerTaggedObject(false, 1, otherRevocationInfo)); } } return result; } internal static Asn1Set CreateBerSetFromList(IEnumerable elements) { Asn1EncodableVector v = new Asn1EncodableVector(); foreach (Asn1Encodable element in elements) { v.Add(element); } return BerSet.FromVector(v); } internal static Asn1Set CreateDerSetFromList(IEnumerable elements) { Asn1EncodableVector v = new Asn1EncodableVector(); foreach (Asn1Encodable element in elements) { v.Add(element); } return DerSet.FromVector(v); } internal static TbsCertificateStructure GetTbsCertificateStructure(X509Certificate cert) { return cert.CertificateStructure.TbsCertificate; } internal static IssuerAndSerialNumber GetIssuerAndSerialNumber(X509Certificate cert) { TbsCertificateStructure tbsCert = GetTbsCertificateStructure(cert); return new IssuerAndSerialNumber(tbsCert.Issuer, tbsCert.SerialNumber); } internal static Asn1.Cms.AttributeTable ParseAttributeTable(Asn1SetParser parser) { Asn1EncodableVector v = new Asn1EncodableVector(); IAsn1Convertible o; while ((o = parser.ReadObject()) != null) { Asn1SequenceParser seq = (Asn1SequenceParser)o; v.Add(seq.ToAsn1Object()); } return new Asn1.Cms.AttributeTable(new DerSet(v)); } internal static void ValidateOtherRevocationInfo(OtherRevocationInfoFormat otherRevocationInfo) { if (CmsObjectIdentifiers.id_ri_ocsp_response.Equals(otherRevocationInfo.InfoFormat)) { OcspResponse ocspResponse = OcspResponse.GetInstance(otherRevocationInfo.Info); if (OcspResponseStatus.Successful != ocspResponse.ResponseStatus.IntValueExact) throw new ArgumentException("cannot add unsuccessful OCSP response to CMS SignedData"); } } internal static bool IsMqv(DerObjectIdentifier oid) => X9ObjectIdentifiers.MqvSinglePassSha1KdfScheme.Equals(oid) || SecObjectIdentifiers.mqvSinglePass_sha224kdf_scheme.Equals(oid) || SecObjectIdentifiers.mqvSinglePass_sha256kdf_scheme.Equals(oid) || SecObjectIdentifiers.mqvSinglePass_sha384kdf_scheme.Equals(oid) || SecObjectIdentifiers.mqvSinglePass_sha512kdf_scheme.Equals(oid); } }