首页 > 编程 > C# > 正文

C#创建自签名认证文件的方法

2019-10-29 21:41:01
字体:
来源:转载
供稿:网友

这篇文章主要介绍了C#创建自签名认证文件的方法,实例分析了C#自签名认证文件的实现技巧,具有一定参考借鉴价值,需要的朋友可以参考下

本文实例讲述了C#创建自签名认证文件的方法。分享给大家供大家参考。具体如下:

 

 
  1. using System; 
  2. using System.Runtime.InteropServices; 
  3. using System.Security.Cryptography.X509Certificates; 
  4. using SecureString = System.Security.SecureString; 
  5. using RuntimeHelpers = System.Runtime.CompilerServices.RuntimeHelpers; 
  6. internal class Certificate 
  7. public static byte[] CreateSelfSignCertificatePfx( 
  8. string x500, 
  9. DateTime startTime, 
  10. DateTime endTime) 
  11. byte[] pfxData = CreateSelfSignCertificatePfx( 
  12. x500, 
  13. startTime, 
  14. endTime, 
  15. (SecureString)null); 
  16. return pfxData; 
  17. public static byte[] CreateSelfSignCertificatePfx( 
  18. string x500, 
  19. DateTime startTime, 
  20. DateTime endTime, 
  21. string insecurePassword) 
  22. byte[] pfxData; 
  23. SecureString password = null
  24. try 
  25. if (!string.IsNullOrEmpty(insecurePassword)) 
  26. password = new SecureString(); 
  27. foreach (char ch in insecurePassword) 
  28. password.AppendChar(ch); 
  29. password.MakeReadOnly(); 
  30. pfxData = CreateSelfSignCertificatePfx( 
  31. x500, 
  32. startTime, 
  33. endTime, 
  34. password); 
  35. finally 
  36. if (password != null
  37. password.Dispose(); 
  38. return pfxData; 
  39. public static byte[] CreateSelfSignCertificatePfx( 
  40. string x500, 
  41. DateTime startTime, 
  42. DateTime endTime, 
  43. SecureString password) 
  44. byte[] pfxData; 
  45. if (x500 == null
  46. x500 = ""
  47. SystemTime startSystemTime = ToSystemTime(startTime); 
  48. SystemTime endSystemTime = ToSystemTime(endTime); 
  49. string containerName = Guid.NewGuid().ToString(); 
  50. GCHandle dataHandle = new GCHandle(); 
  51. IntPtr providerContext = IntPtr.Zero; 
  52. IntPtr cryptKey = IntPtr.Zero; 
  53. IntPtr certContext = IntPtr.Zero; 
  54. IntPtr certStore = IntPtr.Zero; 
  55. IntPtr storeCertContext = IntPtr.Zero; 
  56. IntPtr passwordPtr = IntPtr.Zero; 
  57. RuntimeHelpers.PrepareConstrainedRegions(); 
  58. try 
  59. Check(NativeMethods.CryptAcquireContextW( 
  60. out providerContext, 
  61. containerName, 
  62. null
  63. 1, // PROV_RSA_FULL 
  64. 8)); // CRYPT_NEWKEYSET 
  65. Check(NativeMethods.CryptGenKey( 
  66. providerContext, 
  67. 1, // AT_KEYEXCHANGE 
  68. 1, // CRYPT_EXPORTABLE 
  69. out cryptKey)); 
  70. IntPtr errorStringPtr; 
  71. int nameDataLength = 0; 
  72. byte[] nameData; 
  73. // errorStringPtr gets a pointer into the middle of the x500 string, 
  74. // so x500 needs to be pinned until after we've copied the value 
  75. // of errorStringPtr. 
  76. dataHandle = GCHandle.Alloc(x500, GCHandleType.Pinned); 
  77. if (!NativeMethods.CertStrToNameW( 
  78. 0x00010001, // X509_ASN_ENCODING | PKCS_7_ASN_ENCODING 
  79. dataHandle.AddrOfPinnedObject(), 
  80. 3, // CERT_X500_NAME_STR = 3 
  81. IntPtr.Zero, 
  82. null
  83. ref nameDataLength, 
  84. out errorStringPtr)) 
  85. string error = Marshal.PtrToStringUni(errorStringPtr); 
  86. throw new ArgumentException(error); 
  87. nameData = new byte[nameDataLength]; 
  88. if (!NativeMethods.CertStrToNameW( 
  89. 0x00010001, // X509_ASN_ENCODING | PKCS_7_ASN_ENCODING 
  90. dataHandle.AddrOfPinnedObject(), 
  91. 3, // CERT_X500_NAME_STR = 3 
  92. IntPtr.Zero, 
  93. nameData, 
  94. ref nameDataLength, 
  95. out errorStringPtr)) 
  96. string error = Marshal.PtrToStringUni(errorStringPtr); 
  97. throw new ArgumentException(error); 
  98. dataHandle.Free(); 
  99. dataHandle = GCHandle.Alloc(nameData, GCHandleType.Pinned); 
  100. CryptoApiBlob nameBlob = new CryptoApiBlob( 
  101. nameData.Length, 
  102. dataHandle.AddrOfPinnedObject()); 
  103. CryptKeyProviderInformation kpi = new CryptKeyProviderInformation(); 
  104. kpi.ContainerName = containerName; 
  105. kpi.ProviderType = 1; // PROV_RSA_FULL 
  106. kpi.KeySpec = 1; // AT_KEYEXCHANGE 
  107. certContext = NativeMethods.CertCreateSelfSignCertificate( 
  108. providerContext, 
  109. ref nameBlob, 
  110. 0, 
  111. ref kpi, 
  112. IntPtr.Zero, // default = SHA1RSA 
  113. ref startSystemTime, 
  114. ref endSystemTime, 
  115. IntPtr.Zero); 
  116. Check(certContext != IntPtr.Zero); 
  117. dataHandle.Free(); 
  118. certStore = NativeMethods.CertOpenStore( 
  119. "Memory"// sz_CERT_STORE_PROV_MEMORY 
  120. 0, 
  121. IntPtr.Zero, 
  122. 0x2000, // CERT_STORE_CREATE_NEW_FLAG 
  123. IntPtr.Zero); 
  124. Check(certStore != IntPtr.Zero); 
  125. Check(NativeMethods.CertAddCertificateContextToStore( 
  126. certStore, 
  127. certContext, 
  128. 1, // CERT_STORE_ADD_NEW 
  129. out storeCertContext)); 
  130. NativeMethods.CertSetCertificateContextProperty( 
  131. storeCertContext, 
  132. 2, // CERT_KEY_PROV_INFO_PROP_ID 
  133. 0, 
  134. ref kpi); 
  135. if (password != null
  136. passwordPtr = Marshal.SecureStringToCoTaskMemUnicode(password); 
  137. CryptoApiBlob pfxBlob = new CryptoApiBlob(); 
  138. Check(NativeMethods.PFXExportCertStoreEx( 
  139. certStore, 
  140. ref pfxBlob, 
  141. passwordPtr, 
  142. IntPtr.Zero, 
  143. 7)); // EXPORT_PRIVATE_KEYS | REPORT_NO_PRIVATE_KEY | REPORT_NOT_ABLE_TO_EXPORT_PRIVATE_KEY 
  144. pfxData = new byte[pfxBlob.DataLength]; 
  145. dataHandle = GCHandle.Alloc(pfxData, GCHandleType.Pinned); 
  146. pfxBlob.Data = dataHandle.AddrOfPinnedObject(); 
  147. Check(NativeMethods.PFXExportCertStoreEx( 
  148. certStore, 
  149. ref pfxBlob, 
  150. passwordPtr, 
  151. IntPtr.Zero, 
  152. 7)); // EXPORT_PRIVATE_KEYS | REPORT_NO_PRIVATE_KEY | REPORT_NOT_ABLE_TO_EXPORT_PRIVATE_KEY 
  153. dataHandle.Free(); 
  154. finally 
  155. if (passwordPtr != IntPtr.Zero) 
  156. Marshal.ZeroFreeCoTaskMemUnicode(passwordPtr); 
  157. if (dataHandle.IsAllocated) 
  158. dataHandle.Free(); 
  159. if (certContext != IntPtr.Zero) 
  160. NativeMethods.CertFreeCertificateContext(certContext); 
  161. if (storeCertContext != IntPtr.Zero) 
  162. NativeMethods.CertFreeCertificateContext(storeCertContext); 
  163. if (certStore != IntPtr.Zero) 
  164. NativeMethods.CertCloseStore(certStore, 0); 
  165. if (cryptKey != IntPtr.Zero) 
  166. NativeMethods.CryptDestroyKey(cryptKey); 
  167. if (providerContext != IntPtr.Zero) 
  168. NativeMethods.CryptReleaseContext(providerContext, 0); 
  169. NativeMethods.CryptAcquireContextW( 
  170. out providerContext, 
  171. containerName, 
  172. null
  173. 1, // PROV_RSA_FULL 
  174. 0x10); // CRYPT_DELETEKEYSET 
  175. return pfxData; 
  176. private static SystemTime ToSystemTime(DateTime dateTime) 
  177. long fileTime = dateTime.ToFileTime(); 
  178. SystemTime systemTime; 
  179. Check(NativeMethods.FileTimeToSystemTime(ref fileTime, out systemTime)); 
  180. return systemTime; 
  181. private static void Check(bool nativeCallSucceeded) 
  182. if (!nativeCallSucceeded) 
  183. int error = Marshal.GetHRForLastWin32Error(); 
  184. Marshal.ThrowExceptionForHR(error); 
  185. [StructLayout(LayoutKind.Sequential)] 
  186. private struct SystemTime 
  187. public short Year; 
  188. public short Month; 
  189. public short DayOfWeek; 
  190. public short Day; 
  191. public short Hour; 
  192. public short Minute; 
  193. public short Second; 
  194. public short Milliseconds; 
  195. [StructLayout(LayoutKind.Sequential)] 
  196. private struct CryptoApiBlob 
  197. public int DataLength; 
  198. public IntPtr Data; 
  199. public CryptoApiBlob(int dataLength, IntPtr data) 
  200. this.DataLength = dataLength; 
  201. this.Data = data; 
  202. [StructLayout(LayoutKind.Sequential)] 
  203. private struct CryptKeyProviderInformation 
  204. [MarshalAs(UnmanagedType.LPWStr)] public string ContainerName; 
  205. [MarshalAs(UnmanagedType.LPWStr)] public string ProviderName; 
  206. public int ProviderType; 
  207. public int Flags; 
  208. public int ProviderParameterCount; 
  209. public IntPtr ProviderParameters; // PCRYPT_KEY_PROV_PARAM 
  210. public int KeySpec; 
  211. private static class NativeMethods 
  212. [DllImport("kernel32.dll", SetLastError = true, ExactSpelling = true)] 
  213. [return: MarshalAs(UnmanagedType.Bool)] 
  214. public static extern bool FileTimeToSystemTime( 
  215. [In] ref long fileTime, 
  216. out SystemTime systemTime); 
  217. [DllImport("AdvApi32.dll", SetLastError = true, ExactSpelling = true)] 
  218. [return: MarshalAs(UnmanagedType.Bool)] 
  219. public static extern bool CryptAcquireContextW( 
  220. out IntPtr providerContext, 
  221. [MarshalAs(UnmanagedType.LPWStr)] string container, 
  222. [MarshalAs(UnmanagedType.LPWStr)] string provider, 
  223. int providerType, 
  224. int flags); 
  225. [DllImport("AdvApi32.dll", SetLastError = true, ExactSpelling = true)] 
  226. [return: MarshalAs(UnmanagedType.Bool)] 
  227. public static extern bool CryptReleaseContext( 
  228. IntPtr providerContext, 
  229. int flags); 
  230. [DllImport("AdvApi32.dll", SetLastError = true, ExactSpelling = true)] 
  231. [return: MarshalAs(UnmanagedType.Bool)] 
  232. public static extern bool CryptGenKey( 
  233. IntPtr providerContext, 
  234. int algorithmId, 
  235. int flags, 
  236. out IntPtr cryptKeyHandle); 
  237. [DllImport("AdvApi32.dll", SetLastError = true, ExactSpelling = true)] 
  238. [return: MarshalAs(UnmanagedType.Bool)] 
  239. public static extern bool CryptDestroyKey( 
  240. IntPtr cryptKeyHandle); 
  241. [DllImport("Crypt32.dll", SetLastError = true, ExactSpelling = true)] 
  242. [return: MarshalAs(UnmanagedType.Bool)] 
  243. public static extern bool CertStrToNameW( 
  244. int certificateEncodingType, 
  245. IntPtr x500, 
  246. int strType, 
  247. IntPtr reserved, 
  248. [MarshalAs(UnmanagedType.LPArray)] [Out] byte[] encoded, 
  249. ref int encodedLength, 
  250. out IntPtr errorString); 
  251. [DllImport("Crypt32.dll", SetLastError = true, ExactSpelling = true)] 
  252. public static extern IntPtr CertCreateSelfSignCertificate( 
  253. IntPtr providerHandle, 
  254. [In] ref CryptoApiBlob subjectIssuerBlob, 
  255. int flags, 
  256. [In] ref CryptKeyProviderInformation keyProviderInformation, 
  257. IntPtr signatureAlgorithm, 
  258. [In] ref SystemTime startTime, 
  259. [In] ref SystemTime endTime, 
  260. IntPtr extensions); 
  261. [DllImport("Crypt32.dll", SetLastError = true, ExactSpelling = true)] 
  262. [return: MarshalAs(UnmanagedType.Bool)] 
  263. public static extern bool CertFreeCertificateContext( 
  264. IntPtr certificateContext); 
  265. [DllImport("Crypt32.dll", SetLastError = true, ExactSpelling = true)] 
  266. public static extern IntPtr CertOpenStore( 
  267. [MarshalAs(UnmanagedType.LPStr)] string storeProvider, 
  268. int messageAndCertificateEncodingType, 
  269. IntPtr cryptProvHandle, 
  270. int flags, 
  271. IntPtr parameters); 
  272. [DllImport("Crypt32.dll", SetLastError = true, ExactSpelling = true)] 
  273. [return: MarshalAs(UnmanagedType.Bool)] 
  274. public static extern bool CertCloseStore( 
  275. IntPtr certificateStoreHandle, 
  276. int flags); 
  277. [DllImport("Crypt32.dll", SetLastError = true, ExactSpelling = true)] 
  278. [return: MarshalAs(UnmanagedType.Bool)] 
  279. public static extern bool CertAddCertificateContextToStore( 
  280. IntPtr certificateStoreHandle, 
  281. IntPtr certificateContext, 
  282. int addDisposition, 
  283. out IntPtr storeContextPtr); 
  284. [DllImport("Crypt32.dll", SetLastError = true, ExactSpelling = true)] 
  285. [return: MarshalAs(UnmanagedType.Bool)] 
  286. public static extern bool CertSetCertificateContextProperty( 
  287. IntPtr certificateContext, 
  288. int propertyId, 
  289. int flags, 
  290. [In] ref CryptKeyProviderInformation data); 
  291. [DllImport("Crypt32.dll", SetLastError = true, ExactSpelling = true)] 
  292. [return: MarshalAs(UnmanagedType.Bool)] 
  293. public static extern bool PFXExportCertStoreEx( 
  294. IntPtr certificateStoreHandle, 
  295. ref CryptoApiBlob pfxBlob, 
  296. IntPtr password, 
  297. IntPtr reserved, 
  298. int flags); 

希望本文所述对大家的C#程序设计有所帮助。

发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表