Encrypt.cs 42 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109
  1. using System;
  2. using System.IO;
  3. using System.Linq;
  4. using System.Security.Cryptography;
  5. using System.Text;
  6. using System.Text.RegularExpressions;
  7. namespace Masuit.Tools.Security
  8. {
  9. /// <summary>
  10. /// 常用的加密解密算法
  11. /// </summary>
  12. public static class Encrypt
  13. {
  14. #region DES对称加密解密
  15. /// <summary>
  16. /// 加密密钥,默认取“masuit”的MD5值
  17. /// </summary>
  18. public static string DefaultEncryptKey = "masuit".MDString2();
  19. /// <summary>
  20. /// 使用默认加密
  21. /// </summary>
  22. /// <param name="strText">被加密的字符串</param>
  23. /// <returns>加密后的数据</returns>
  24. public static string DesEncrypt(this string strText)
  25. {
  26. try
  27. {
  28. return DesEncrypt(strText, DefaultEncryptKey);
  29. }
  30. catch
  31. {
  32. return "";
  33. }
  34. }
  35. /// <summary>
  36. /// 使用默认解密
  37. /// </summary>
  38. /// <param name="strText">需要解密的 字符串</param>
  39. /// <returns>解密后的数据</returns>
  40. public static string DesDecrypt(this string strText)
  41. {
  42. try
  43. {
  44. return DesDecrypt(strText, DefaultEncryptKey);
  45. }
  46. catch
  47. {
  48. return "";
  49. }
  50. }
  51. /// <summary>
  52. /// 加密字符串
  53. /// 加密密钥必须为8位
  54. /// </summary>
  55. /// <param name="strText">被加密的字符串</param>
  56. /// <param name="strEncrKey">8位长度密钥</param>
  57. /// <returns>加密后的数据</returns>
  58. public static string DesEncrypt(this string strText, string strEncrKey)
  59. {
  60. if (strEncrKey.Length < 8)
  61. {
  62. throw new Exception("密钥长度无效,密钥必须是8位!");
  63. }
  64. StringBuilder ret = new StringBuilder();
  65. using var des = new DESCryptoServiceProvider();
  66. byte[] inputByteArray = Encoding.Default.GetBytes(strText);
  67. des.Key = Encoding.ASCII.GetBytes(strEncrKey.Substring(0, 8));
  68. des.IV = Encoding.ASCII.GetBytes(strEncrKey.Substring(0, 8));
  69. MemoryStream ms = new MemoryStream();
  70. using var cs = new CryptoStream(ms, des.CreateEncryptor(), CryptoStreamMode.Write);
  71. cs.Write(inputByteArray, 0, inputByteArray.Length);
  72. cs.FlushFinalBlock();
  73. foreach (byte b in ms.ToArray())
  74. {
  75. ret.AppendFormat($"{b:X2}");
  76. }
  77. return ret.ToString();
  78. }
  79. /// <summary>
  80. /// DES加密文件
  81. /// </summary>
  82. /// <param name="fin">文件输入流</param>
  83. /// <param name="outFilePath">文件输出路径</param>
  84. /// <param name="strEncrKey">加密密钥</param>
  85. public static void DesEncrypt(this FileStream fin, string outFilePath, string strEncrKey)
  86. {
  87. byte[] iv =
  88. {
  89. 0x12,
  90. 0x34,
  91. 0x56,
  92. 0x78,
  93. 0x90,
  94. 0xAB,
  95. 0xCD,
  96. 0xEF
  97. };
  98. var byKey = Encoding.UTF8.GetBytes(strEncrKey.Substring(0, 8));
  99. using var fout = new FileStream(outFilePath, FileMode.OpenOrCreate, FileAccess.Write);
  100. fout.SetLength(0);
  101. byte[] bin = new byte[100];
  102. long rdlen = 0;
  103. long totlen = fin.Length;
  104. DES des = new DESCryptoServiceProvider();
  105. var encStream = new CryptoStream(fout, des.CreateEncryptor(byKey, iv), CryptoStreamMode.Write);
  106. while (rdlen < totlen)
  107. {
  108. var len = fin.Read(bin, 0, 100);
  109. encStream.Write(bin, 0, len);
  110. rdlen += len;
  111. }
  112. }
  113. /// <summary>
  114. /// DES解密文件
  115. /// </summary>
  116. /// <param name="fin">输入文件流</param>
  117. /// <param name="outFilePath">文件输出路径</param>
  118. /// <param name="sDecrKey">解密密钥</param>
  119. public static void DesDecrypt(this FileStream fin, string outFilePath, string sDecrKey)
  120. {
  121. byte[] iv =
  122. {
  123. 0x12,
  124. 0x34,
  125. 0x56,
  126. 0x78,
  127. 0x90,
  128. 0xAB,
  129. 0xCD,
  130. 0xEF
  131. };
  132. var byKey = Encoding.UTF8.GetBytes(sDecrKey.Substring(0, 8));
  133. using var fout = new FileStream(outFilePath, FileMode.OpenOrCreate, FileAccess.Write);
  134. fout.SetLength(0);
  135. byte[] bin = new byte[100];
  136. long rdlen = 0;
  137. long totlen = fin.Length;
  138. using DES des = new DESCryptoServiceProvider();
  139. var encStream = new CryptoStream(fout, des.CreateDecryptor(byKey, iv), CryptoStreamMode.Write);
  140. while (rdlen < totlen)
  141. {
  142. var len = fin.Read(bin, 0, 100);
  143. encStream.Write(bin, 0, len);
  144. rdlen += len;
  145. }
  146. }
  147. /// <summary>
  148. /// DES解密算法
  149. /// 密钥为8位
  150. /// </summary>
  151. /// <param name="pToDecrypt">需要解密的字符串</param>
  152. /// <param name="sKey">密钥</param>
  153. /// <returns>解密后的数据</returns>
  154. public static string DesDecrypt(this string pToDecrypt, string sKey)
  155. {
  156. if (sKey.Length < 8)
  157. {
  158. throw new Exception("密钥长度无效,密钥必须是8位!");
  159. }
  160. var ms = new MemoryStream();
  161. using var des = new DESCryptoServiceProvider();
  162. var inputByteArray = new byte[pToDecrypt.Length / 2];
  163. for (int x = 0; x < pToDecrypt.Length / 2; x++)
  164. {
  165. int i = Convert.ToInt32(pToDecrypt.Substring(x * 2, 2), 16);
  166. inputByteArray[x] = (byte)i;
  167. }
  168. des.Key = Encoding.ASCII.GetBytes(sKey.Substring(0, 8));
  169. des.IV = Encoding.ASCII.GetBytes(sKey.Substring(0, 8));
  170. using var cs = new CryptoStream(ms, des.CreateDecryptor(), CryptoStreamMode.Write);
  171. cs.Write(inputByteArray, 0, inputByteArray.Length);
  172. cs.FlushFinalBlock();
  173. return Encoding.Default.GetString(ms.ToArray());
  174. }
  175. #endregion
  176. #region 对称加密算法AES RijndaelManaged加密解密
  177. private static readonly string Default_AES_Key = "@#kim123";
  178. private static byte[] Keys =
  179. {
  180. 0x41,
  181. 0x72,
  182. 0x65,
  183. 0x79,
  184. 0x6F,
  185. 0x75,
  186. 0x6D,
  187. 0x79,
  188. 0x53,
  189. 0x6E,
  190. 0x6F,
  191. 0x77,
  192. 0x6D,
  193. 0x61,
  194. 0x6E,
  195. 0x3F
  196. };
  197. /// <summary>
  198. /// 对称加密算法AES RijndaelManaged加密(RijndaelManaged(AES)算法是块式加密算法)
  199. /// </summary>
  200. /// <param name="encryptString">待加密字符串</param>
  201. /// <returns>加密结果字符串</returns>
  202. public static string AESEncrypt(this string encryptString)
  203. {
  204. return AESEncrypt(encryptString, Default_AES_Key);
  205. }
  206. /// <summary>
  207. /// 对称加密算法AES RijndaelManaged加密(RijndaelManaged(AES)算法是块式加密算法)
  208. /// </summary>
  209. /// <param name="encryptString">待加密字符串</param>
  210. /// <param name="encryptKey">加密密钥,须半角字符</param>
  211. /// <returns>加密结果字符串</returns>
  212. public static string AESEncrypt(this string encryptString, string encryptKey)
  213. {
  214. encryptKey = GetSubString(encryptKey, 32, "");
  215. encryptKey = encryptKey.PadRight(32, ' ');
  216. using var rijndaelProvider = new RijndaelManaged
  217. {
  218. Key = Encoding.UTF8.GetBytes(encryptKey.Substring(0, 32)),
  219. IV = Keys
  220. };
  221. using ICryptoTransform rijndaelEncrypt = rijndaelProvider.CreateEncryptor();
  222. byte[] inputData = Encoding.UTF8.GetBytes(encryptString);
  223. byte[] encryptedData = rijndaelEncrypt.TransformFinalBlock(inputData, 0, inputData.Length);
  224. return Convert.ToBase64String(encryptedData);
  225. }
  226. /// <summary>
  227. /// 对称加密算法AES RijndaelManaged解密字符串
  228. /// </summary>
  229. /// <param name="decryptString">待解密的字符串</param>
  230. /// <returns>解密成功返回解密后的字符串,失败返源串</returns>
  231. public static string AESDecrypt(this string decryptString)
  232. {
  233. return AESDecrypt(decryptString, Default_AES_Key);
  234. }
  235. /// <summary>
  236. /// 对称加密算法AES RijndaelManaged解密字符串
  237. /// </summary>
  238. /// <param name="decryptString">待解密的字符串</param>
  239. /// <param name="decryptKey">解密密钥,和加密密钥相同</param>
  240. /// <returns>解密成功返回解密后的字符串,失败返回空</returns>
  241. public static string AESDecrypt(this string decryptString, string decryptKey)
  242. {
  243. try
  244. {
  245. decryptKey = GetSubString(decryptKey, 32, "");
  246. decryptKey = decryptKey.PadRight(32, ' ');
  247. using var rijndaelProvider = new RijndaelManaged()
  248. {
  249. Key = Encoding.UTF8.GetBytes(decryptKey),
  250. IV = Keys
  251. };
  252. using ICryptoTransform rijndaelDecrypt = rijndaelProvider.CreateDecryptor();
  253. byte[] inputData = Convert.FromBase64String(decryptString);
  254. byte[] decryptedData = rijndaelDecrypt.TransformFinalBlock(inputData, 0, inputData.Length);
  255. return Encoding.UTF8.GetString(decryptedData);
  256. }
  257. catch
  258. {
  259. return string.Empty;
  260. }
  261. }
  262. /// <summary>
  263. /// 按字节长度(按字节,一个汉字为2个字节)取得某字符串的一部分
  264. /// </summary>
  265. /// <param name="sourceString">源字符串</param>
  266. /// <param name="length">所取字符串字节长度</param>
  267. /// <param name="tailString">附加字符串(当字符串不够长时,尾部所添加的字符串,一般为"...")</param>
  268. /// <returns>某字符串的一部分</returns>
  269. private static string GetSubString(this string sourceString, int length, string tailString)
  270. {
  271. return GetSubString(sourceString, 0, length, tailString);
  272. }
  273. /// <summary>
  274. /// 按字节长度(按字节,一个汉字为2个字节)取得某字符串的一部分
  275. /// </summary>
  276. /// <param name="sourceString">源字符串</param>
  277. /// <param name="startIndex">索引位置,以0开始</param>
  278. /// <param name="length">所取字符串字节长度</param>
  279. /// <param name="tailString">附加字符串(当字符串不够长时,尾部所添加的字符串,一般为"...")</param>
  280. /// <returns>某字符串的一部分</returns>
  281. private static string GetSubString(this string sourceString, int startIndex, int length, string tailString)
  282. {
  283. //当是日文或韩文时(注:中文的范围:\u4e00 - \u9fa5, 日文在\u0800 - \u4e00, 韩文为\xAC00-\xD7A3)
  284. if (Regex.IsMatch(sourceString, "[\u0800-\u4e00]+") || Regex.IsMatch(sourceString, "[\xAC00-\xD7A3]+"))
  285. {
  286. //当截取的起始位置超出字段串长度时
  287. if (startIndex >= sourceString.Length)
  288. {
  289. return string.Empty;
  290. }
  291. return sourceString.Substring(startIndex, length + startIndex > sourceString.Length ? (sourceString.Length - startIndex) : length);
  292. }
  293. //中文字符,如"中国人民abcd123"
  294. if (length <= 0)
  295. {
  296. return string.Empty;
  297. }
  298. byte[] bytesSource = Encoding.Default.GetBytes(sourceString);
  299. //当字符串长度大于起始位置
  300. if (bytesSource.Length > startIndex)
  301. {
  302. int endIndex = bytesSource.Length;
  303. //当要截取的长度在字符串的有效长度范围内
  304. if (bytesSource.Length > (startIndex + length))
  305. {
  306. endIndex = length + startIndex;
  307. }
  308. else
  309. {
  310. //当不在有效范围内时,只取到字符串的结尾
  311. length = bytesSource.Length - startIndex;
  312. tailString = "";
  313. }
  314. var anResultFlag = new int[length];
  315. int nFlag = 0;
  316. //字节大于127为双字节字符
  317. for (int i = startIndex; i < endIndex; i++)
  318. {
  319. if (bytesSource[i] > 127)
  320. {
  321. nFlag++;
  322. if (nFlag == 3)
  323. {
  324. nFlag = 1;
  325. }
  326. }
  327. else
  328. {
  329. nFlag = 0;
  330. }
  331. anResultFlag[i] = nFlag;
  332. }
  333. //最后一个字节为双字节字符的一半
  334. if ((bytesSource[endIndex - 1] > 127) && (anResultFlag[length - 1] == 1))
  335. {
  336. length++;
  337. }
  338. byte[] bsResult = new byte[length];
  339. Array.Copy(bytesSource, startIndex, bsResult, 0, length);
  340. var myResult = Encoding.Default.GetString(bsResult);
  341. myResult += tailString;
  342. return myResult;
  343. }
  344. return string.Empty;
  345. }
  346. /// <summary>
  347. /// 加密文件流
  348. /// </summary>
  349. /// <param name="fs">需要加密的文件流</param>
  350. /// <param name="decryptKey">加密密钥</param>
  351. /// <returns>加密流</returns>
  352. public static CryptoStream AESEncryptStrream(this FileStream fs, string decryptKey)
  353. {
  354. decryptKey = GetSubString(decryptKey, 32, "");
  355. decryptKey = decryptKey.PadRight(32, ' ');
  356. using var rijndaelProvider = new RijndaelManaged()
  357. {
  358. Key = Encoding.UTF8.GetBytes(decryptKey),
  359. IV = Keys
  360. };
  361. using var encrypto = rijndaelProvider.CreateEncryptor();
  362. return new CryptoStream(fs, encrypto, CryptoStreamMode.Write);
  363. }
  364. /// <summary>
  365. /// 解密文件流
  366. /// </summary>
  367. /// <param name="fs">需要解密的文件流</param>
  368. /// <param name="decryptKey">解密密钥</param>
  369. /// <returns>加密流</returns>
  370. public static CryptoStream AESDecryptStream(this FileStream fs, string decryptKey)
  371. {
  372. decryptKey = GetSubString(decryptKey, 32, "");
  373. decryptKey = decryptKey.PadRight(32, ' ');
  374. using var rijndaelProvider = new RijndaelManaged()
  375. {
  376. Key = Encoding.UTF8.GetBytes(decryptKey),
  377. IV = Keys
  378. };
  379. using var decrypto = rijndaelProvider.CreateDecryptor();
  380. return new CryptoStream(fs, decrypto, CryptoStreamMode.Read);
  381. }
  382. /// <summary>
  383. /// 对指定文件AES加密
  384. /// </summary>
  385. /// <param name="input">源文件流</param>
  386. /// <param name="outputPath">输出文件路径</param>
  387. public static void AESEncryptFile(this FileStream input, string outputPath)
  388. {
  389. using var fren = new FileStream(outputPath, FileMode.Create);
  390. using var enfr = AESEncryptStrream(fren, Default_AES_Key);
  391. byte[] bytearrayinput = new byte[input.Length];
  392. input.Read(bytearrayinput, 0, bytearrayinput.Length);
  393. enfr.Write(bytearrayinput, 0, bytearrayinput.Length);
  394. }
  395. /// <summary>
  396. /// 对指定的文件AES解密
  397. /// </summary>
  398. /// <param name="input">源文件流</param>
  399. /// <param name="outputPath">输出文件路径</param>
  400. public static void AESDecryptFile(this FileStream input, string outputPath)
  401. {
  402. using FileStream frde = new FileStream(outputPath, FileMode.Create);
  403. using CryptoStream defr = AESDecryptStream(input, Default_AES_Key);
  404. byte[] bytearrayoutput = new byte[1024];
  405. while (true)
  406. {
  407. var count = defr.Read(bytearrayoutput, 0, bytearrayoutput.Length);
  408. frde.Write(bytearrayoutput, 0, count);
  409. if (count < bytearrayoutput.Length)
  410. {
  411. break;
  412. }
  413. }
  414. }
  415. #endregion
  416. #region Base64加密解密
  417. /// <summary>
  418. /// Base64加密
  419. /// </summary>
  420. /// <param name="str">需要加密的字符串</param>
  421. /// <returns>加密后的数据</returns>
  422. public static string Base64Encrypt(this string str)
  423. {
  424. byte[] encbuff = Encoding.UTF8.GetBytes(str);
  425. return Convert.ToBase64String(encbuff);
  426. }
  427. /// <summary>
  428. /// Base64解密
  429. /// </summary>
  430. /// <param name="str">需要解密的字符串</param>
  431. /// <returns>解密后的数据</returns>
  432. public static string Base64Decrypt(this string str)
  433. {
  434. byte[] decbuff = Convert.FromBase64String(str);
  435. return Encoding.UTF8.GetString(decbuff);
  436. }
  437. #endregion
  438. /// <summary>
  439. /// SHA256函数
  440. /// </summary>
  441. /// <param name="str">原始字符串</param>
  442. /// <returns>SHA256结果(返回长度为44字节的字符串)</returns>
  443. public static string SHA256(this string str)
  444. {
  445. byte[] sha256Data = Encoding.UTF8.GetBytes(str);
  446. using var sha256 = new SHA256Managed();
  447. byte[] result = sha256.ComputeHash(sha256Data);
  448. return Convert.ToBase64String(result); //返回长度为44字节的字符串
  449. }
  450. #region MD5加密算法
  451. #region 获取数组的Hex值
  452. /// <summary>
  453. /// 获取数组的Hex值
  454. /// </summary>
  455. /// <param name="array">需要求Hex值的数组</param>
  456. /// <param name="uppercase">是否转大写</param>
  457. /// <returns>字节数组的16进制表示</returns>
  458. public static string ArrayToHexString(this byte[] array, bool uppercase)
  459. {
  460. var format = "x2";
  461. if (uppercase)
  462. {
  463. format = "X2";
  464. }
  465. return array.Aggregate("", (current, b) => current + b.ToString(format));
  466. }
  467. #endregion
  468. #region 对字符串进行MD5加密
  469. /// <summary>
  470. /// 对字符串进行MD5加密
  471. /// </summary>
  472. /// <param name="message">需要加密的字符串</param>
  473. /// <returns>加密后的结果</returns>
  474. public static string MDString(this string message)
  475. {
  476. MD5 md5 = MD5.Create();
  477. byte[] buffer = Encoding.Default.GetBytes(message);
  478. byte[] bytes = md5.ComputeHash(buffer);
  479. return ArrayToHexString(bytes, false);
  480. }
  481. /// <summary>
  482. /// 对字符串进行MD5二次加密
  483. /// </summary>
  484. /// <param name="message">需要加密的字符串</param>
  485. /// <returns>加密后的结果</returns>
  486. public static string MDString2(this string message) => MDString(MDString(message));
  487. /// <summary>
  488. /// MD5 三次加密算法
  489. /// </summary>
  490. /// <param name="s">需要加密的字符串</param>
  491. /// <returns>MD5字符串</returns>
  492. public static string MDString3(this string s)
  493. {
  494. using MD5 md5 = MD5.Create();
  495. byte[] bytes = Encoding.ASCII.GetBytes(s);
  496. byte[] bytes1 = md5.ComputeHash(bytes);
  497. byte[] bytes2 = md5.ComputeHash(bytes1);
  498. byte[] bytes3 = md5.ComputeHash(bytes2);
  499. return ArrayToHexString(bytes3, false);
  500. }
  501. /// <summary>
  502. /// 对字符串进行MD5加盐加密
  503. /// </summary>
  504. /// <param name="message">需要加密的字符串</param>
  505. /// <param name="salt">盐</param>
  506. /// <returns>加密后的结果</returns>
  507. public static string MDString(this string message, string salt) => MDString(message + salt);
  508. /// <summary>
  509. /// 对字符串进行MD5二次加盐加密
  510. /// </summary>
  511. /// <param name="message">需要加密的字符串</param>
  512. /// <param name="salt">盐</param>
  513. /// <returns>加密后的结果</returns>
  514. public static string MDString2(this string message, string salt) => MDString(MDString(message + salt), salt);
  515. /// <summary>
  516. /// MD5 三次加密算法
  517. /// </summary>
  518. /// <param name="s">需要加密的字符串</param>
  519. /// <param name="salt">盐</param>
  520. /// <returns>MD5字符串</returns>
  521. public static string MDString3(this string s, string salt)
  522. {
  523. using MD5 md5 = MD5.Create();
  524. byte[] bytes = Encoding.ASCII.GetBytes(s + salt);
  525. byte[] bytes1 = md5.ComputeHash(bytes);
  526. byte[] bytes2 = md5.ComputeHash(bytes1);
  527. byte[] bytes3 = md5.ComputeHash(bytes2);
  528. return ArrayToHexString(bytes3, false);
  529. }
  530. #endregion
  531. #region 获取文件的MD5值
  532. /// <summary>
  533. /// 获取文件的MD5值
  534. /// </summary>
  535. /// <param name="fileName">需要求MD5值的文件的文件名及路径</param>
  536. /// <returns>MD5字符串</returns>
  537. public static string MDFile(this string fileName)
  538. {
  539. using var fs = File.Open(fileName, FileMode.Open, FileAccess.Read);
  540. var array = new byte[fs.Length];
  541. fs.Read(array, 0, (int)fs.Length);
  542. using MD5 md5 = MD5.Create();
  543. byte[] bytes = md5.ComputeHash(array);
  544. return ArrayToHexString(bytes, false);
  545. }
  546. /// <summary>
  547. /// 获取数据流的MD5值
  548. /// </summary>
  549. /// <param name="stream"></param>
  550. /// <returns>MD5字符串</returns>
  551. public static string MDString(this Stream stream)
  552. {
  553. var bytes = stream.ToArray();
  554. using var md5 = MD5.Create();
  555. var mdstr = ArrayToHexString(md5.ComputeHash(bytes), false);
  556. stream.Position = 0;
  557. return mdstr;
  558. }
  559. #endregion
  560. #endregion MD5加密算法
  561. }
  562. /// <summary>
  563. /// RC2加密解密算法
  564. /// </summary>
  565. public static class RC2
  566. {
  567. private static ASCIIEncoding _asciiEncoding;
  568. private static byte[] _iv;
  569. private static byte[] _key;
  570. private static RC2CryptoServiceProvider _rc2Csp;
  571. private static UnicodeEncoding _textConverter;
  572. static RC2()
  573. {
  574. InitializeComponent();
  575. }
  576. private static void InitializeComponent()
  577. {
  578. _key = new byte[]
  579. {
  580. 106,
  581. 51,
  582. 25,
  583. 141,
  584. 157,
  585. 142,
  586. 23,
  587. 111,
  588. 234,
  589. 159,
  590. 187,
  591. 154,
  592. 215,
  593. 34,
  594. 37,
  595. 204
  596. };
  597. _iv = new byte[]
  598. {
  599. 135,
  600. 186,
  601. 133,
  602. 136,
  603. 184,
  604. 149,
  605. 153,
  606. 144
  607. };
  608. _asciiEncoding = new ASCIIEncoding();
  609. _textConverter = new UnicodeEncoding();
  610. _rc2Csp = new RC2CryptoServiceProvider();
  611. }
  612. #region 将文本数据加密后写入一个文件
  613. /// <summary>
  614. /// 将文本数据加密后写入一个文件,其中,这个文件是用InitBinFile建立的,这个文件将被分成十块,
  615. /// 用来分别保存10组不同的数据,第一个byte位保留,第2位到第21位分别用来存放每块数据的长度,但
  616. /// 一个byte的取值为0-127,所以,用两个byte来存放一个长度。
  617. /// </summary>
  618. /// <param name="toEncryptText">要加密的文本数据</param>
  619. /// <param name="filePath">要写入的文件</param>
  620. /// <param name="dataIndex">写入第几块,取值为1--10</param>
  621. /// <returns>是否操作成功</returns>
  622. public static bool EncryptToFile(this string toEncryptText, string filePath, int dataIndex)
  623. {
  624. var r = false;
  625. if ((dataIndex > 10) && (dataIndex < 1))
  626. {
  627. return r;
  628. }
  629. //打开要写入的文件,主要是为了保持原文件的内容不丢失
  630. var tmpFileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.None, 1024, true);
  631. var index = new byte[10261];
  632. //将读取的内容写到byte数组
  633. tmpFileStream.Read(index, 0, 10261);
  634. tmpFileStream.Close();
  635. //定义基本的加密转换运算
  636. using var Encryptor = _rc2Csp.CreateEncryptor(_key, _iv);
  637. var msEncrypt = new MemoryStream();
  638. //在此加密转换流中,加密将从csEncrypt,加密后,结果在msEncrypt流中。
  639. using var csEncrypt = new CryptoStream(msEncrypt, Encryptor, CryptoStreamMode.Write);
  640. //将要加密的文本转换成UTF-16 编码,保存在tmp数组。
  641. var tmp = _textConverter.GetBytes(toEncryptText);
  642. //将tmp输入csEncrypt,将通过Encryptor来加密。
  643. csEncrypt.Write(tmp, 0, tmp.Length);
  644. //输出到msEnctypt
  645. csEncrypt.FlushFinalBlock();
  646. //将流转成byte[]
  647. var encrypted = msEncrypt.ToArray();
  648. if (encrypted.Length > 1024)
  649. return false;
  650. //得到加密后数据的大小,将结果存在指定的位置。
  651. index[dataIndex * 2 - 1] = Convert.ToByte(Convert.ToString(encrypted.Length / 128));
  652. index[dataIndex * 2] = Convert.ToByte(Convert.ToString(encrypted.Length % 128));
  653. //将加密后的结果写入index(覆盖)
  654. for (var i = 0; i < encrypted.Length; i++)
  655. index[1024 * (dataIndex - 1) + 21 + i] = encrypted[i];
  656. //建立文件流
  657. tmpFileStream = new FileStream(filePath, FileMode.Truncate, FileAccess.Write, FileShare.None, 1024, true);
  658. //写文件
  659. tmpFileStream.Write(index, 0, 10261);
  660. tmpFileStream.Flush();
  661. r = true;
  662. tmpFileStream.Close();
  663. return r;
  664. }
  665. #endregion
  666. #region 从一个文件中解密出一段文本,其中,这个文件是由InitBinFile建立的,并且由 EncryptToFile加密的
  667. /// <summary>
  668. /// 从一个文件中解密出一段文本,其中,这个文件是由InitBinFile建立的,并且由 EncryptToFile加密的
  669. /// </summary>
  670. /// <param name="filePath">要解密的文件</param>
  671. /// <param name="dataIndex">要从哪一个块中解密</param>
  672. /// <returns>解密后的文本</returns>
  673. public static string DecryptFromFile(this string filePath, int dataIndex)
  674. {
  675. if (dataIndex > 10 && dataIndex < 1)
  676. {
  677. return "";
  678. }
  679. using var tmpFileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.None, 1024, true);
  680. using var decryptor = _rc2Csp.CreateDecryptor(_key, _iv);
  681. var msDecrypt = new MemoryStream();
  682. using var csDecrypt = new CryptoStream(msDecrypt, decryptor, CryptoStreamMode.Write);
  683. var index = new byte[10261];
  684. tmpFileStream.Read(index, 0, 10261);
  685. var count = index[dataIndex * 2 - 1] * 128 + index[dataIndex * 2];
  686. var tmp = new byte[count];
  687. Array.Copy(index, 1024 * (dataIndex - 1) + 21, tmp, 0, count);
  688. csDecrypt.Write(tmp, 0, count);
  689. csDecrypt.FlushFinalBlock();
  690. var decrypted = msDecrypt.ToArray();
  691. return _textConverter.GetString(decrypted, 0, decrypted.Length);
  692. }
  693. #endregion
  694. #region 将一段文本加密后保存到一个文件
  695. /// <summary>
  696. /// 将一段文本加密后保存到一个文件
  697. /// </summary>
  698. /// <param name="toEncryptText">要加密的文本数据</param>
  699. /// <param name="filePath">要保存的文件</param>
  700. /// <returns>是否加密成功</returns>
  701. public static void EncryptToFile(this string toEncryptText, string filePath)
  702. {
  703. using var tmpFileStream = new FileStream(filePath, FileMode.OpenOrCreate, FileAccess.Write, FileShare.None, 1024, true);
  704. using var encryptor = _rc2Csp.CreateEncryptor(_key, _iv);
  705. var msEncrypt = new MemoryStream();
  706. using var csEncrypt = new CryptoStream(msEncrypt, encryptor, CryptoStreamMode.Write);
  707. var tmp = _textConverter.GetBytes(toEncryptText);
  708. csEncrypt.Write(tmp, 0, tmp.Length);
  709. csEncrypt.FlushFinalBlock();
  710. var encrypted = msEncrypt.ToArray();
  711. tmpFileStream.Write(encrypted, 0, encrypted.Length);
  712. }
  713. #endregion
  714. #region 将一个被加密的文件解密
  715. /// <summary>
  716. /// 将一个被加密的文件解密
  717. /// </summary>
  718. /// <param name="filePath">要解密的文件</param>
  719. /// <returns>解密后的文本</returns>
  720. public static string DecryptFromFile(this string filePath)
  721. {
  722. using var tmpFileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.None, 1024, true);
  723. using var decryptor = _rc2Csp.CreateDecryptor(_key, _iv);
  724. var msDecrypt = new MemoryStream();
  725. using var csDecrypt = new CryptoStream(msDecrypt, decryptor, CryptoStreamMode.Write);
  726. var tmp = new byte[tmpFileStream.Length];
  727. tmpFileStream.Read(tmp, 0, tmp.Length);
  728. csDecrypt.Write(tmp, 0, tmp.Length);
  729. csDecrypt.FlushFinalBlock();
  730. var decrypted = msDecrypt.ToArray();
  731. return _textConverter.GetString(decrypted, 0, decrypted.Length);
  732. }
  733. #endregion
  734. #region 将文本数据加密后写入一个文件
  735. /// <summary>
  736. /// 将文本数据加密后写入一个文件,其中,这个文件是用InitBinFile建立的,这个文件将被分成十块,
  737. /// 用来分别保存10组不同的数据,第一个byte位保留,第2位到第21位分别用来存放每块数据的长度,但
  738. /// 一个byte的取值为0-127,所以,用两个byte来存放一个长度。
  739. /// </summary>
  740. /// <param name="toEncryptText">要加密的文本数据</param>
  741. /// <param name="filePath">要写入的文件</param>
  742. /// <param name="dataIndex">写入第几块,取值为1--10</param>
  743. /// <param name="IV">初始化向量</param>
  744. /// <param name="Key">加密密匙</param>
  745. /// <returns>是否操作成功</returns>
  746. public static void EncryptToFile(this string toEncryptText, string filePath, int dataIndex, byte[] IV, byte[] Key)
  747. {
  748. if ((dataIndex > 10) && (dataIndex < 1))
  749. {
  750. return;
  751. }
  752. //打开要写入的文件,主要是为了保持原文件的内容不丢失
  753. using var fs = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.None, 1024, true);
  754. var index = new byte[10261];
  755. //将读取的内容写到byte数组
  756. fs.Read(index, 0, 10261);
  757. //定义基本的加密转换运算
  758. using var encryptor = _rc2Csp.CreateEncryptor(Key, IV);
  759. var msEncrypt = new MemoryStream();
  760. //在此加密转换流中,加密将从csEncrypt,加密后,结果在msEncrypt流中。
  761. using var csEncrypt = new CryptoStream(msEncrypt, encryptor, CryptoStreamMode.Write);
  762. var tmp = _textConverter.GetBytes(toEncryptText);
  763. //将tmp输入csEncrypt,将通过Encryptor来加密。
  764. csEncrypt.Write(tmp, 0, tmp.Length);
  765. //输出到msEnctypt
  766. csEncrypt.FlushFinalBlock();
  767. //将流转成byte[]
  768. var encrypted = msEncrypt.ToArray();
  769. if (encrypted.Length > 1024)
  770. {
  771. return;
  772. }
  773. //得到加密后数据的大小,将结果存在指定的位置。
  774. index[dataIndex * 2 - 1] = Convert.ToByte(Convert.ToString(encrypted.Length / 128));
  775. index[dataIndex * 2] = Convert.ToByte(Convert.ToString(encrypted.Length % 128));
  776. //将加密后的结果写入index(覆盖)
  777. for (int i = 0; i < encrypted.Length; i++)
  778. {
  779. index[1024 * (dataIndex - 1) + 21 + i] = encrypted[i];
  780. }
  781. //建立文件流
  782. using var newStream = new FileStream(filePath, FileMode.Truncate, FileAccess.Write, FileShare.None, 1024, true);
  783. newStream.Write(index, 0, 10261);
  784. newStream.Flush();
  785. }
  786. #endregion
  787. #region 从一个文件中解密出一段文本
  788. /// <summary>
  789. /// 从一个文件中解密出一段文本,其中,这个文件是由InitBinFile建立的,并且由 EncryptToFile加密的
  790. /// </summary>
  791. /// <param name="filePath">要解密的文件</param>
  792. /// <param name="dataIndex">要从哪一个块中解密</param>
  793. /// <param name="iv">初始化向量</param>
  794. /// <param name="key">解密密匙</param>
  795. /// <returns>解密后的文本</returns>
  796. public static string DecryptFromFile(this string filePath, int dataIndex, byte[] iv, byte[] key)
  797. {
  798. if ((dataIndex > 10) && (dataIndex < 1))
  799. {
  800. return "";
  801. }
  802. using var tmpFileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.None, 1024, true);
  803. using var decryptor = _rc2Csp.CreateDecryptor(key, iv);
  804. var msDecrypt = new MemoryStream();
  805. using var csDecrypt = new CryptoStream(msDecrypt, decryptor, CryptoStreamMode.Write);
  806. var index = new byte[10261];
  807. tmpFileStream.Read(index, 0, 10261);
  808. var count = index[dataIndex * 2 - 1] * 128 + index[dataIndex * 2];
  809. var tmp = new byte[count];
  810. Array.Copy(index, 1024 * (dataIndex - 1) + 21, tmp, 0, count);
  811. csDecrypt.Write(tmp, 0, count);
  812. csDecrypt.FlushFinalBlock();
  813. var decrypted = msDecrypt.ToArray();
  814. return _textConverter.GetString(decrypted, 0, decrypted.Length);
  815. }
  816. #endregion
  817. #region 将一段文本加密后保存到一个文件
  818. /// <summary>
  819. /// 将一段文本加密后保存到一个文件
  820. /// </summary>
  821. /// <param name="toEncryptText">要加密的文本数据</param>
  822. /// <param name="filePath">要保存的文件</param>
  823. /// <param name="iv">初始化向量</param>
  824. /// <param name="key">加密密匙</param>
  825. /// <returns>是否加密成功</returns>
  826. public static void EncryptToFile(this string toEncryptText, string filePath, byte[] iv, byte[] key)
  827. {
  828. using var tmpFileStream = new FileStream(filePath, FileMode.OpenOrCreate, FileAccess.Write, FileShare.None, 1024, true);
  829. using var encryptor = _rc2Csp.CreateEncryptor(key, iv);
  830. var msEncrypt = new MemoryStream();
  831. using var csEncrypt = new CryptoStream(msEncrypt, encryptor, CryptoStreamMode.Write);
  832. var tmp = _textConverter.GetBytes(toEncryptText);
  833. csEncrypt.Write(tmp, 0, tmp.Length);
  834. csEncrypt.FlushFinalBlock();
  835. var encrypted = msEncrypt.ToArray();
  836. tmpFileStream.Write(encrypted, 0, encrypted.Length);
  837. tmpFileStream.Flush();
  838. }
  839. #endregion
  840. #region 将一个被加密的文件解密
  841. /// <summary>
  842. /// 将一个被加密的文件解密
  843. /// </summary>
  844. /// <param name="filePath">要解密的文件</param>
  845. /// <param name="iv">初始化向量</param>
  846. /// <param name="key">解密密匙</param>
  847. /// <returns>解密后的文本</returns>
  848. public static string DecryptFromFile(this string filePath, byte[] iv, byte[] key)
  849. {
  850. using var tmpFileStream = new FileStream(filePath, FileMode.Open, FileAccess.Read, FileShare.None, 1024, true);
  851. using var decryptor = _rc2Csp.CreateDecryptor(key, iv);
  852. var msDecrypt = new MemoryStream();
  853. using var csDecrypt = new CryptoStream(msDecrypt, decryptor, CryptoStreamMode.Write);
  854. var tmp = new byte[tmpFileStream.Length];
  855. tmpFileStream.Read(tmp, 0, tmp.Length);
  856. csDecrypt.Write(tmp, 0, tmp.Length);
  857. csDecrypt.FlushFinalBlock();
  858. var decrypted = msDecrypt.ToArray();
  859. return _textConverter.GetString(decrypted, 0, decrypted.Length);
  860. }
  861. #endregion
  862. #region 设置加密或解密的初始化向量
  863. /// <summary>
  864. /// 设置加密或解密的初始化向量
  865. /// </summary>
  866. /// <param name="s">长度等于8的ASCII字符集的字符串</param>
  867. public static void SetIV(this string s)
  868. {
  869. if (s.Length != 8)
  870. {
  871. _iv = null;
  872. return;
  873. }
  874. try
  875. {
  876. _iv = _asciiEncoding.GetBytes(s);
  877. }
  878. catch (Exception)
  879. {
  880. _iv = null;
  881. }
  882. }
  883. #endregion
  884. #region 设置加密或解密的密匙
  885. /// <summary>
  886. /// 设置加密或解密的密匙
  887. /// </summary>
  888. /// <param name="s">长度等于16的ASCII字符集的字符串</param>
  889. public static void SetKey(this string s)
  890. {
  891. if (s.Length != 16)
  892. {
  893. _key = null;
  894. return;
  895. }
  896. try
  897. {
  898. _key = _asciiEncoding.GetBytes(s);
  899. }
  900. catch (Exception)
  901. {
  902. _key = null;
  903. }
  904. }
  905. #endregion
  906. }
  907. /// <summary>
  908. /// 对称加密解密算法类
  909. /// </summary>
  910. public static class Rijndael
  911. {
  912. private static string _key;
  913. private static SymmetricAlgorithm _mobjCryptoService;
  914. /// <summary>
  915. /// 对称加密类的构造函数
  916. /// </summary>
  917. public static void SymmetricMethod()
  918. {
  919. _mobjCryptoService = new RijndaelManaged();
  920. _key = "Guz(%&hj7x89H$yuBI0456FtmaT5&fvHUFCy76*h%(HilJ$lhj!y6&(*jkP87jH7";
  921. }
  922. /// <summary>
  923. /// 获得密钥
  924. /// </summary>
  925. /// <returns>密钥</returns>
  926. private static byte[] GetLegalKey()
  927. {
  928. var sTemp = _key;
  929. _mobjCryptoService.GenerateKey();
  930. var bytTemp = _mobjCryptoService.Key;
  931. var keyLength = bytTemp.Length;
  932. if (sTemp.Length > keyLength)
  933. {
  934. sTemp = sTemp.Substring(0, keyLength);
  935. }
  936. else if (sTemp.Length < keyLength)
  937. {
  938. sTemp = sTemp.PadRight(keyLength, ' ');
  939. }
  940. return Encoding.ASCII.GetBytes(sTemp);
  941. }
  942. /// <summary>
  943. /// 获得初始向量IV
  944. /// </summary>
  945. /// <returns>初试向量IV</returns>
  946. private static byte[] GetLegalIV()
  947. {
  948. var sTemp = "E4ghj*Ghg7!rNIfb&95GUY86GfghUber57HBh(u%g6HJ($jhWk7&!hg4ui%$hjk";
  949. _mobjCryptoService.GenerateIV();
  950. var bytTemp = _mobjCryptoService.IV;
  951. var ivLength = bytTemp.Length;
  952. if (sTemp.Length > ivLength)
  953. {
  954. sTemp = sTemp.Substring(0, ivLength);
  955. }
  956. else if (sTemp.Length < ivLength)
  957. {
  958. sTemp = sTemp.PadRight(ivLength, ' ');
  959. }
  960. return Encoding.ASCII.GetBytes(sTemp);
  961. }
  962. /// <summary>
  963. /// 加密方法
  964. /// </summary>
  965. /// <param name="source">待加密的串</param>
  966. /// <returns>经过加密的串</returns>
  967. public static string Encrypto(this string source)
  968. {
  969. var bytIn = Encoding.UTF8.GetBytes(source);
  970. var ms = new MemoryStream();
  971. _mobjCryptoService.Key = GetLegalKey();
  972. _mobjCryptoService.IV = GetLegalIV();
  973. using var encrypto = _mobjCryptoService.CreateEncryptor();
  974. using var cs = new CryptoStream(ms, encrypto, CryptoStreamMode.Write);
  975. cs.Write(bytIn, 0, bytIn.Length);
  976. cs.FlushFinalBlock();
  977. var bytOut = ms.ToArray();
  978. return Convert.ToBase64String(bytOut);
  979. }
  980. /// <summary>
  981. /// 解密方法
  982. /// </summary>
  983. /// <param name="source">待解密的串</param>
  984. /// <returns>经过解密的串</returns>
  985. public static string Decrypto(this string source)
  986. {
  987. var bytIn = Convert.FromBase64String(source);
  988. var ms = new MemoryStream(bytIn, 0, bytIn.Length);
  989. _mobjCryptoService.Key = GetLegalKey();
  990. _mobjCryptoService.IV = GetLegalIV();
  991. using var encrypto = _mobjCryptoService.CreateDecryptor();
  992. using var cs = new CryptoStream(ms, encrypto, CryptoStreamMode.Read);
  993. using var sr = new StreamReader(cs);
  994. return sr.ReadToEnd();
  995. }
  996. }
  997. }