Encrypt.cs 42 KB

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