Files
MikuSB/SdkServer/Utils/CertHelper.cs
2026-05-13 06:36:26 +09:00

98 lines
3.8 KiB
C#

using System.Collections.Concurrent;
using System.Security.Cryptography;
using System.Security.Cryptography.X509Certificates;
namespace MikuSB.SdkServer.Utils;
public static class CertHelper
{
private const string Password = "MikuSB.SdkServer.LocalTLS";
private static readonly ConcurrentDictionary<string, X509Certificate2> Certificates =
new(StringComparer.OrdinalIgnoreCase);
private static readonly Lazy<X509Certificate2> WildcardCert =
new(() => LoadOrCreatePersisted("wildcard.xoyo.games", "*.xoyo.games"));
private static string CertificateDirectory => Path.Combine(AppContext.BaseDirectory, "sdk-certs");
public static X509Certificate2 GetOrCreate(string? serverName)
{
if (string.IsNullOrEmpty(serverName) ||
serverName.EndsWith(".xoyo.games", StringComparison.OrdinalIgnoreCase))
return WildcardCert.Value;
var normalized = serverName.Trim().TrimEnd('.').ToLowerInvariant();
return Certificates.GetOrAdd(normalized, host => LoadOrCreatePersisted(host, host));
}
private static X509Certificate2 LoadOrCreatePersisted(string fileHost, string subjectHost)
{
Directory.CreateDirectory(CertificateDirectory);
var pfxPath = Path.Combine(CertificateDirectory, $"{SanitizeFileName(fileHost)}.pfx");
if (File.Exists(pfxPath))
return LoadPkcs12(File.ReadAllBytes(pfxPath));
var certificate = CreateSelfSigned(subjectHost);
File.WriteAllBytes(pfxPath, certificate.Export(X509ContentType.Pfx, Password));
return LoadPkcs12(File.ReadAllBytes(pfxPath));
}
private static X509Certificate2 LoadPkcs12(byte[] pfx)
{
return X509CertificateLoader.LoadPkcs12(
pfx,
Password,
X509KeyStorageFlags.UserKeySet |
X509KeyStorageFlags.PersistKeySet |
X509KeyStorageFlags.Exportable);
}
private static X509Certificate2 CreateSelfSigned(string host)
{
// CNG key must have AllowPlainTextExport so the private key is included in PFX export.
// Without this, Export(Pfx) produces a cert-only PFX, and EphemeralKeySet loads a
// keyless cert that Kestrel cannot use for TLS.
var cngParams = new CngKeyCreationParameters
{
ExportPolicy = CngExportPolicies.AllowPlaintextExport,
KeyUsage = CngKeyUsages.AllUsages
};
cngParams.Parameters.Add(new CngProperty("Length",
BitConverter.GetBytes(2048), CngPropertyOptions.None));
using var cngKey = CngKey.Create(CngAlgorithm.Rsa, null, cngParams);
using var rsa = new RSACng(cngKey);
var req = new CertificateRequest(
new X500DistinguishedName($"CN={host}"),
rsa,
HashAlgorithmName.SHA256,
RSASignaturePadding.Pkcs1);
var san = new SubjectAlternativeNameBuilder();
san.AddDnsName(host);
req.CertificateExtensions.Add(san.Build());
req.CertificateExtensions.Add(new X509KeyUsageExtension(
X509KeyUsageFlags.KeyEncipherment | X509KeyUsageFlags.DigitalSignature,
critical: false));
req.CertificateExtensions.Add(new X509EnhancedKeyUsageExtension(
new OidCollection { new Oid("1.3.6.1.5.5.7.3.1") },
critical: false));
var cert = req.CreateSelfSigned(
DateTimeOffset.UtcNow.AddHours(-1),
DateTimeOffset.UtcNow.AddYears(10));
// Private key is now exportable — PFX includes key material
var pfx = cert.Export(X509ContentType.Pfx, Password);
return LoadPkcs12(pfx);
}
private static string SanitizeFileName(string host)
{
var invalidChars = Path.GetInvalidFileNameChars();
return string.Concat(host.Select(ch => invalidChars.Contains(ch) ? '_' : ch));
}
}