WeChatPayBackService.cs 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106
  1. using System;
  2. using System.Collections.Generic;
  3. using Library;
  4. using LitJson;
  5. using System.Linq;
  6. using MySystem.PxcModels;
  7. using Org.BouncyCastle.Crypto.Modes;
  8. using Org.BouncyCastle.Crypto.Engines;
  9. using Org.BouncyCastle.Crypto.Parameters;
  10. using System.Text;
  11. using System.Threading;
  12. namespace MySystem
  13. {
  14. public class WeChatPayBackService
  15. {
  16. public readonly static WeChatPayBackService Instance = new WeChatPayBackService();
  17. private WeChatPayBackService()
  18. { }
  19. public void Start()
  20. {
  21. Thread th = new Thread(dosomething);
  22. th.IsBackground = true;
  23. th.Start();
  24. }
  25. public void dosomething()
  26. {
  27. bool op = true;
  28. while (op)
  29. {
  30. string content = RedisDbconn.Instance.RPop<string>("WeChatPayBack");
  31. if (!string.IsNullOrEmpty(content))
  32. {
  33. try
  34. {
  35. JsonData jsonObj = JsonMapper.ToObject(content);
  36. if (jsonObj.Count > 0)
  37. {
  38. if (jsonObj["event_type"].ToString() == "TRANSACTION.SUCCESS")
  39. {
  40. JsonData resource = jsonObj["resource"];
  41. //开始解密
  42. string WxPayResourceDecryptModel = AesGcmDecrypt(resource["associated_data"].ToString(), resource["nonce"].ToString(), resource["ciphertext"].ToString());
  43. // {\"sp_mchid\":\"1611167423\",\"sub_mchid\":\"1622024882\",\"sp_appid\":\"wxe2c051b3e46c0f6f\",\"out_trade_no\":\"2022022621562926396898863\",\"transaction_id\":\"4200001412202202267619496496\",\"trade_type\":\"JSAPI\",\"trade_state\":\"SUCCESS\",\"trade_state_desc\":\"支付成功\",\"bank_type\":\"OTHERS\",\"attach\":\"\",\"success_time\":\"2022-02-26T21:56:42+08:00\",\"payer\":{\"sp_openid\":\"omawy5W6jb0pgPfuKUVs6K3bEhzk\",\"sub_openid\":\"\"},\"amount\":{\"total\":1,\"payer_total\":1,\"currency\":\"CNY\",\"payer_currency\":\"CNY\"}}
  44. JsonData orderObj = JsonMapper.ToObject(WxPayResourceDecryptModel);
  45. string OrderNo = orderObj["out_trade_no"].ToString();
  46. WebCMSEntities db = new WebCMSEntities();
  47. ConsumerOrderForNo forNo = db.ConsumerOrderForNo.FirstOrDefault(m => m.OrderNo == OrderNo);
  48. if (forNo != null)
  49. {
  50. ConsumerOrders order = db.ConsumerOrders.FirstOrDefault(m => m.Id == forNo.OrderIds && m.Status == 0);
  51. if (order != null)
  52. {
  53. order.Status = 1;
  54. order.UpdateDate = DateTime.Now;
  55. order.PayMoney = order.PayMoney * 10000;
  56. order.MaxDivi = order.MaxDivi * 10000;
  57. MerchantInfo merchant = db.MerchantInfo.FirstOrDefault(m => m.Id == order.MerchantId) ?? new MerchantInfo();
  58. order.UserId = merchant.UserId;
  59. db.SaveChanges();
  60. RedisDbconn.Instance.AddList("ConsumerOrdersStat", order.Id);
  61. RedisDbconn.Instance.AddList("ConsumerOrders:Divi:List", order.Id.ToString());
  62. RedisDbconn.Instance.AddRightList("ConsumerOrders:Divi:" + order.MerchantId, order);
  63. // ConsumerOrdersStatService.Instance.Stat(order);
  64. }
  65. }
  66. db.Dispose();
  67. }
  68. }
  69. }
  70. catch (Exception ex)
  71. {
  72. function.WriteLog(DateTime.Now.ToString() + ":" + ex.ToString(), "public_service");
  73. }
  74. }
  75. else
  76. {
  77. Thread.Sleep(2000);
  78. }
  79. }
  80. }
  81. private string ALGORITHM = "AES/GCM/NoPadding";
  82. private int TAG_LENGTH_BIT = 128;
  83. private int NONCE_LENGTH_BYTE = 12;
  84. public string AesGcmDecrypt(string associatedData, string nonce, string ciphertext)
  85. {
  86. GcmBlockCipher gcmBlockCipher = new GcmBlockCipher(new AesEngine());
  87. AeadParameters aeadParameters = new AeadParameters(
  88. new KeyParameter(Encoding.UTF8.GetBytes(AppConfig.WeChatParam.AesGemKey)),
  89. 128,
  90. Encoding.UTF8.GetBytes(nonce),
  91. Encoding.UTF8.GetBytes(associatedData));
  92. gcmBlockCipher.Init(false, aeadParameters);
  93. byte[] data = Convert.FromBase64String(ciphertext);
  94. byte[] plaintext = new byte[gcmBlockCipher.GetOutputSize(data.Length)];
  95. int length = gcmBlockCipher.ProcessBytes(data, 0, data.Length, plaintext, 0);
  96. gcmBlockCipher.DoFinal(plaintext, length);
  97. return Encoding.UTF8.GetString(plaintext);
  98. }
  99. }
  100. }