8fc30896db
Former-commit-id: c477e03582759447177c6d4bf412cd2355aad476
224 lines
6.4 KiB
C#
224 lines
6.4 KiB
C#
//
|
|
// X509Helper2.cs
|
|
//
|
|
// Authors:
|
|
// Martin Baulig <martin.baulig@xamarin.com>
|
|
//
|
|
// Copyright (C) 2016 Xamarin, Inc. (http://www.xamarin.com)
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
// a copy of this software and associated documentation files (the
|
|
// "Software"), to deal in the Software without restriction, including
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
// the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be
|
|
// included in all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
//
|
|
|
|
#if SECURITY_DEP
|
|
#if MONO_SECURITY_ALIAS
|
|
extern alias MonoSecurity;
|
|
#endif
|
|
|
|
#if MONO_SECURITY_ALIAS
|
|
using MonoSecurity::Mono.Security.Interface;
|
|
using MX = MonoSecurity::Mono.Security.X509;
|
|
#else
|
|
#if MONO_FEATURE_BTLS
|
|
using Mono.Security.Interface;
|
|
#endif
|
|
using MX = Mono.Security.X509;
|
|
#endif
|
|
|
|
#if MONO_FEATURE_BTLS
|
|
using Mono.Btls;
|
|
#endif
|
|
#endif
|
|
|
|
using System.IO;
|
|
using System.Text;
|
|
|
|
namespace System.Security.Cryptography.X509Certificates
|
|
{
|
|
internal static class X509Helper2
|
|
{
|
|
internal static long GetSubjectNameHash (X509Certificate certificate)
|
|
{
|
|
return GetSubjectNameHash (certificate.Impl);
|
|
}
|
|
|
|
internal static long GetSubjectNameHash (X509CertificateImpl impl)
|
|
{
|
|
#if SECURITY_DEP
|
|
using (var x509 = GetNativeInstance (impl))
|
|
return GetSubjectNameHash (x509);
|
|
#else
|
|
throw new NotSupportedException ();
|
|
#endif
|
|
}
|
|
|
|
internal static void ExportAsPEM (X509Certificate certificate, Stream stream, bool includeHumanReadableForm)
|
|
{
|
|
ExportAsPEM (certificate.Impl, stream, includeHumanReadableForm);
|
|
}
|
|
|
|
internal static void ExportAsPEM (X509CertificateImpl impl, Stream stream, bool includeHumanReadableForm)
|
|
{
|
|
#if SECURITY_DEP
|
|
using (var x509 = GetNativeInstance (impl))
|
|
ExportAsPEM (x509, stream, includeHumanReadableForm);
|
|
#else
|
|
throw new NotSupportedException ();
|
|
#endif
|
|
}
|
|
|
|
#if SECURITY_DEP
|
|
internal static void Initialize ()
|
|
{
|
|
X509Helper.InstallNativeHelper (new MyNativeHelper ());
|
|
}
|
|
|
|
internal static void ThrowIfContextInvalid (X509CertificateImpl impl)
|
|
{
|
|
X509Helper.ThrowIfContextInvalid (impl);
|
|
}
|
|
|
|
#if !MONO_FEATURE_BTLS
|
|
static X509Certificate GetNativeInstance (X509CertificateImpl impl)
|
|
{
|
|
throw new PlatformNotSupportedException ();
|
|
}
|
|
#else
|
|
static MonoBtlsX509 GetNativeInstance (X509CertificateImpl impl)
|
|
{
|
|
ThrowIfContextInvalid (impl);
|
|
var btlsImpl = impl as X509CertificateImplBtls;
|
|
if (btlsImpl != null)
|
|
return btlsImpl.X509.Copy ();
|
|
else
|
|
return MonoBtlsX509.LoadFromData (impl.GetRawCertData (), MonoBtlsX509Format.DER);
|
|
}
|
|
|
|
internal static long GetSubjectNameHash (MonoBtlsX509 x509)
|
|
{
|
|
using (var subject = x509.GetSubjectName ())
|
|
return subject.GetHash ();
|
|
}
|
|
|
|
internal static void ExportAsPEM (MonoBtlsX509 x509, Stream stream, bool includeHumanReadableForm)
|
|
{
|
|
using (var bio = MonoBtlsBio.CreateMonoStream (stream)) {
|
|
x509.ExportAsPEM (bio, includeHumanReadableForm);
|
|
}
|
|
}
|
|
#endif // !MONO_FEATURE_BTLS
|
|
|
|
internal static X509Certificate2Impl Import (byte[] rawData, string password, X509KeyStorageFlags keyStorageFlags, bool disableProvider = false)
|
|
{
|
|
if (rawData == null || rawData.Length == 0)
|
|
return null;
|
|
|
|
#if MONO_FEATURE_BTLS
|
|
if (!disableProvider) {
|
|
var provider = MonoTlsProviderFactory.GetProvider ();
|
|
if (provider.HasNativeCertificates) {
|
|
var impl = provider.GetNativeCertificate (rawData, password, keyStorageFlags);
|
|
return impl;
|
|
}
|
|
}
|
|
#endif // MONO_FEATURE_BTLS
|
|
var impl2 = new X509Certificate2ImplMono ();
|
|
impl2.Import (rawData, password, keyStorageFlags);
|
|
return impl2;
|
|
}
|
|
|
|
internal static X509Certificate2Impl Import (X509Certificate cert, bool disableProvider = false)
|
|
{
|
|
if (cert.Impl == null)
|
|
return null;
|
|
|
|
#if MONO_FEATURE_BTLS
|
|
if (!disableProvider) {
|
|
var provider = MonoTlsProviderFactory.GetProvider ();
|
|
if (provider.HasNativeCertificates) {
|
|
var impl = provider.GetNativeCertificate (cert);
|
|
return impl;
|
|
}
|
|
}
|
|
#endif // MONO_FEATURE_BTLS
|
|
var impl2 = cert.Impl as X509Certificate2Impl;
|
|
if (impl2 != null)
|
|
return (X509Certificate2Impl)impl2.Clone ();
|
|
return Import (cert.GetRawCertData (), null, X509KeyStorageFlags.DefaultKeySet);
|
|
}
|
|
|
|
/*
|
|
* This is used by X509ChainImplMono
|
|
*
|
|
* Some of the missing APIs such as X509v3 extensions can be added to the native
|
|
* BTLS implementation.
|
|
*
|
|
* We should also consider replacing X509ChainImplMono with a new X509ChainImplBtls
|
|
* at some point.
|
|
*/
|
|
[MonoTODO ("Investigate replacement; see comments in source.")]
|
|
internal static MX.X509Certificate GetMonoCertificate (X509Certificate2 certificate)
|
|
{
|
|
var impl2 = certificate.Impl as X509Certificate2Impl;
|
|
if (impl2 == null)
|
|
impl2 = Import (certificate, true);
|
|
var fallbackImpl = impl2.FallbackImpl as X509Certificate2ImplMono;
|
|
if (fallbackImpl == null)
|
|
throw new NotSupportedException ();
|
|
return fallbackImpl.MonoCertificate;
|
|
}
|
|
|
|
internal static X509ChainImpl CreateChainImpl (bool useMachineContext)
|
|
{
|
|
return new X509ChainImplMono (useMachineContext);
|
|
}
|
|
|
|
public static bool IsValid (X509ChainImpl impl)
|
|
{
|
|
return impl != null && impl.IsValid;
|
|
}
|
|
|
|
internal static void ThrowIfContextInvalid (X509ChainImpl impl)
|
|
{
|
|
if (!IsValid (impl))
|
|
throw GetInvalidChainContextException ();
|
|
}
|
|
|
|
internal static Exception GetInvalidChainContextException ()
|
|
{
|
|
return new CryptographicException (Locale.GetText ("Chain instance is empty."));
|
|
}
|
|
|
|
class MyNativeHelper : INativeCertificateHelper
|
|
{
|
|
public X509CertificateImpl Import (
|
|
byte[] data, string password, X509KeyStorageFlags flags)
|
|
{
|
|
return X509Helper2.Import (data, password, flags);
|
|
}
|
|
|
|
public X509CertificateImpl Import (X509Certificate cert)
|
|
{
|
|
return X509Helper2.Import (cert);
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
}
|