2
0

CertService.cs 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188
  1. using FastGithub.Configuration;
  2. using Microsoft.Extensions.Caching.Memory;
  3. using Microsoft.Extensions.Logging;
  4. using System;
  5. using System.Collections.Generic;
  6. using System.Diagnostics;
  7. using System.IO;
  8. using System.Linq;
  9. using System.Net;
  10. using System.Security.Cryptography.X509Certificates;
  11. namespace FastGithub.HttpServer
  12. {
  13. /// <summary>
  14. /// 证书服务
  15. /// </summary>
  16. sealed class CertService
  17. {
  18. private const string CACERT_PATH = "cacert";
  19. private const int KEY_SIZE_BITS = 2048;
  20. private readonly IMemoryCache serverCertCache;
  21. private readonly ILogger<CertService> logger;
  22. /// <summary>
  23. /// 获取证书文件路径
  24. /// </summary>
  25. public string CaCerFilePath { get; } = $"{CACERT_PATH}/fastgithub.cer";
  26. /// <summary>
  27. /// 获取私钥文件路径
  28. /// </summary>
  29. public string CaKeyFilePath { get; } = $"{CACERT_PATH}/fastgithub.key";
  30. /// <summary>
  31. /// 证书服务
  32. /// </summary>
  33. /// <param name="logger"></param>
  34. public CertService(
  35. IMemoryCache serverCertCache,
  36. ILogger<CertService> logger)
  37. {
  38. this.serverCertCache = serverCertCache;
  39. this.logger = logger;
  40. Directory.CreateDirectory(CACERT_PATH);
  41. }
  42. /// <summary>
  43. /// 生成CA证书
  44. /// </summary>
  45. public bool CreateCaCertIfNotExists()
  46. {
  47. if (File.Exists(this.CaCerFilePath) && File.Exists(this.CaKeyFilePath))
  48. {
  49. return false;
  50. }
  51. File.Delete(this.CaCerFilePath);
  52. File.Delete(this.CaKeyFilePath);
  53. var validFrom = DateTime.Today.AddDays(-1);
  54. var validTo = DateTime.Today.AddYears(10);
  55. CertGenerator.GenerateBySelf(new[] { nameof(FastGithub) }, KEY_SIZE_BITS, validFrom, validTo, this.CaCerFilePath, this.CaKeyFilePath);
  56. return true;
  57. }
  58. /// <summary>
  59. /// 安装和信任CA证书
  60. /// </summary>
  61. public void InstallAndTrustCaCert()
  62. {
  63. if (OperatingSystem.IsWindows())
  64. {
  65. this.InstallAndTrustCaCertAtWindows();
  66. }
  67. else if (OperatingSystem.IsLinux())
  68. {
  69. this.logger.LogWarning($"请根据具体linux发行版手工安装CA证书{this.CaCerFilePath}");
  70. }
  71. else if (OperatingSystem.IsMacOS())
  72. {
  73. this.logger.LogWarning($"请手工安装CA证书然后设置信任CA证书{this.CaCerFilePath}");
  74. }
  75. else
  76. {
  77. this.logger.LogWarning($"请根据你的系统平台手工安装和信任CA证书{this.CaCerFilePath}");
  78. }
  79. GitConfigSslverify(false);
  80. }
  81. /// <summary>
  82. /// 设置ssl验证
  83. /// </summary>
  84. /// <param name="value">是否验证</param>
  85. /// <returns></returns>
  86. public static bool GitConfigSslverify(bool value)
  87. {
  88. try
  89. {
  90. Process.Start(new ProcessStartInfo
  91. {
  92. FileName = "git",
  93. Arguments = $"config --global http.sslverify {value.ToString().ToLower()}",
  94. UseShellExecute = true,
  95. CreateNoWindow = true,
  96. WindowStyle = ProcessWindowStyle.Hidden
  97. });
  98. return true;
  99. }
  100. catch (Exception)
  101. {
  102. return false;
  103. }
  104. }
  105. /// <summary>
  106. /// 安装CA证书
  107. /// </summary>
  108. private void InstallAndTrustCaCertAtWindows()
  109. {
  110. try
  111. {
  112. using var store = new X509Store(StoreName.Root, StoreLocation.LocalMachine);
  113. store.Open(OpenFlags.ReadWrite);
  114. var caCert = new X509Certificate2(this.CaCerFilePath);
  115. var subjectName = caCert.Subject[3..];
  116. foreach (var item in store.Certificates.Find(X509FindType.FindBySubjectName, subjectName, false))
  117. {
  118. if (item.Thumbprint != caCert.Thumbprint)
  119. {
  120. store.Remove(item);
  121. }
  122. }
  123. if (store.Certificates.Find(X509FindType.FindByThumbprint, caCert.Thumbprint, true).Count == 0)
  124. {
  125. store.Add(caCert);
  126. }
  127. store.Close();
  128. }
  129. catch (Exception)
  130. {
  131. this.logger.LogWarning($"请手工安装CA证书{this.CaCerFilePath}到“将所有的证书都放入下载存储”\\“受信任的根证书颁发机构”");
  132. }
  133. }
  134. /// <summary>
  135. /// 获取颁发给指定域名的证书
  136. /// </summary>
  137. /// <param name="domain"></param>
  138. /// <returns></returns>
  139. public X509Certificate2 GetOrCreateServerCert(string? domain)
  140. {
  141. var key = $"{nameof(CertService)}:{domain}";
  142. return this.serverCertCache.GetOrCreate(key, GetOrCreateCert);
  143. // 生成域名的1年证书
  144. X509Certificate2 GetOrCreateCert(ICacheEntry entry)
  145. {
  146. var domains = GetDomains(domain).Distinct();
  147. var validFrom = DateTime.Today.AddDays(-1);
  148. var validTo = DateTime.Today.AddYears(1);
  149. entry.SetAbsoluteExpiration(validTo);
  150. return CertGenerator.GenerateByCa(domains, KEY_SIZE_BITS, validFrom, validTo, this.CaCerFilePath, this.CaKeyFilePath);
  151. }
  152. }
  153. /// <summary>
  154. /// 获取域名
  155. /// </summary>
  156. /// <param name="domain"></param>
  157. /// <returns></returns>
  158. private static IEnumerable<string> GetDomains(string? domain)
  159. {
  160. if (string.IsNullOrEmpty(domain) == false)
  161. {
  162. yield return domain;
  163. yield break;
  164. }
  165. yield return LocalMachine.Name;
  166. yield return IPAddress.Loopback.ToString();
  167. }
  168. }
  169. }