CAUtil.cs 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  1. using Org.BouncyCastle.Crypto;
  2. using Org.BouncyCastle.X509;
  3. using System;
  4. using System.Collections.Generic;
  5. using System.Linq;
  6. using System.Text;
  7. using System.Threading.Tasks;
  8. namespace CAUtilLib
  9. {
  10. public class CaUtil
  11. {
  12. public string RootPath = "/home/cert";
  13. public const string RcaDirectoryName = "rca";
  14. public const string TrustedDirectoryName = "tcrt";
  15. public const string RcaKeyName = "rca_key.pem";
  16. public const string RcaCertName = "rca.pem";
  17. public string RcaKeyPath => Path.Combine(RootPath, RcaDirectoryName, RcaKeyName);
  18. public string RcaCertPath => Path.Combine(RootPath, RcaDirectoryName, RcaCertName);
  19. public string TrustedCertsPath => Path.Combine(RootPath, TrustedDirectoryName);
  20. public async Task<bool> CreateRootCA()
  21. {
  22. AsymmetricCipherKeyPair kp = BouncyCastleWrapper.GenerateRsaKeyPair(4096);
  23. var saveKeyResult = await BouncyCastleWrapper.TrySaveAsPemAsync(RcaKeyPath, new object[] { kp.Private });
  24. if (!saveKeyResult)
  25. {
  26. return false;
  27. }
  28. var subject = BouncyCastleWrapper.CreateX509Name(commonName: "Zerova");
  29. X509Certificate cert = BouncyCastleWrapper.GenerateSelfSignedRootCertificate(subject, kp);
  30. var saveCertResult = await BouncyCastleWrapper.TrySaveAsPemAsync(RcaCertPath, new object[] { cert, kp.Private });
  31. if (!saveCertResult)
  32. {
  33. return false;
  34. }
  35. return true;
  36. }
  37. public async Task<string?> SignCsr(string csr)
  38. {
  39. var parsedCsr = BouncyCastleWrapper.LoadPemCsrFromString(csr);
  40. if (parsedCsr is null)
  41. {
  42. return null;
  43. }
  44. var rca = await BouncyCastleWrapper.LoadPemCertFromFile(RcaCertPath);
  45. if (rca is null)
  46. {
  47. return null;
  48. }
  49. var rcaKey = await BouncyCastleWrapper.LoadPemKeyFromFile(RcaKeyPath);
  50. if (rcaKey is null)
  51. {
  52. return null;
  53. }
  54. var generatedCrt = BouncyCastleWrapper.SignCertificate(parsedCsr, rca, rcaKey);
  55. return await BouncyCastleWrapper.ToStringAsPem(generatedCrt);
  56. }
  57. public async Task<int> VerifyCrt(string crt)
  58. {
  59. var parsedCrt = BouncyCastleWrapper.LoadPemCertFromString(crt);
  60. if (parsedCrt is null)
  61. {
  62. return -1;
  63. }
  64. var isSignedByRca = await VerifyCrtSignedByRCA(parsedCrt!);
  65. if (isSignedByRca)
  66. {
  67. return 0;
  68. }
  69. var isSignedByTrusted = await VerifyCrtSignedByTrused(parsedCrt!);
  70. if (isSignedByTrusted)
  71. {
  72. return 1;
  73. }
  74. return -1;
  75. }
  76. private async Task<bool> VerifyCrtSignedByRCA(X509Certificate crt)
  77. {
  78. var rcaKey = await BouncyCastleWrapper.LoadPemKeyFromFile(RcaKeyPath);
  79. if (rcaKey is null)
  80. {
  81. return false;
  82. }
  83. return BouncyCastleWrapper.ValidateCert(crt, rcaKey.Public);
  84. }
  85. private async Task<bool> VerifyCrtSignedByTrused(X509Certificate crt)
  86. {
  87. var trustedCerts = Directory.GetFiles(TrustedCertsPath);
  88. foreach (var trustedCert in trustedCerts)
  89. {
  90. var cert = await BouncyCastleWrapper.LoadPemCertFromFile(trustedCert);
  91. var checkResult = BouncyCastleWrapper.ValidateCert(crt, cert.GetPublicKey());
  92. if (checkResult)
  93. {
  94. return true;
  95. }
  96. }
  97. return false;
  98. }
  99. }
  100. }