WeChatPayBackService.cs 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. using System;
  2. using System.Collections.Generic;
  3. using Library;
  4. using LitJson;
  5. using System.Linq;
  6. using MySystem.Models;
  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(Listen);
  22. th.IsBackground = true;
  23. th.Start();
  24. }
  25. public void Listen()
  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. dosomething(content);
  36. }
  37. catch (Exception ex)
  38. {
  39. LogHelper.Instance.WriteLog(DateTime.Now.ToString() + ":" + ex.ToString(), "微信支付回调异常");
  40. }
  41. }
  42. else
  43. {
  44. Thread.Sleep(2000);
  45. }
  46. }
  47. }
  48. public void dosomething(string content)
  49. {
  50. JsonData jsonObj = JsonMapper.ToObject(content);
  51. if (jsonObj.Count > 0)
  52. {
  53. if (jsonObj["event_type"].ToString() == "TRANSACTION.SUCCESS")
  54. {
  55. JsonData resource = jsonObj["resource"];
  56. //开始解密
  57. string WxPayResourceDecryptModel = AesGcmDecrypt(resource["associated_data"].ToString(), resource["nonce"].ToString(), resource["ciphertext"].ToString());
  58. // {\"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\"}}
  59. JsonData orderObj = JsonMapper.ToObject(WxPayResourceDecryptModel);
  60. string OrderNo = orderObj["out_trade_no"].ToString();
  61. string TradeNo = orderObj["transaction_id"].ToString();
  62. WebCMSEntities db = new WebCMSEntities();
  63. ConsumerOrderForNo forNo = db.ConsumerOrderForNo.FirstOrDefault(m => m.OrderNo == OrderNo);
  64. if (forNo != null)
  65. {
  66. ConsumerOrders order = db.ConsumerOrders.FirstOrDefault(m => m.Id == forNo.OrderIds && m.Status == 0);
  67. if (order != null)
  68. {
  69. order.Status = 1;
  70. order.UpdateDate = DateTime.Now;
  71. order.PayMoney = order.PayMoney;
  72. order.MaxDivi = order.MaxDivi;
  73. order.SeoTitle = TradeNo;
  74. MerchantInfo merchant = db.MerchantInfo.FirstOrDefault(m => m.Id == order.MerchantId) ?? new MerchantInfo();
  75. MerchantAddInfo addMer = db.MerchantAddInfo.FirstOrDefault(m => m.Id == order.MerchantId) ?? new MerchantAddInfo();
  76. order.UserId = merchant.UserId;
  77. ConsumerOrders check = db.ConsumerOrders.FirstOrDefault(m => m.Id < order.Id && m.Status > 0 && m.PayMode == order.PayMode && m.MerchantId == order.MerchantId);
  78. if (check != null)
  79. {
  80. order.QueryCount = check.QueryCount + 1;
  81. }
  82. else
  83. {
  84. order.QueryCount = 1;
  85. }
  86. db.SaveChanges();
  87. //语音播报
  88. var machines = db.PosMachines.Select(m => new { m.DeviceKind, m.BrandId, m.Detail, m.BindMerchantId }).Where(m => m.DeviceKind == "1" && m.BindMerchantId == order.MerchantId);
  89. foreach (var machine in machines)
  90. {
  91. if (machine.BrandId == 2)
  92. {
  93. TianYuVoiceHelper.Instance.doSomething(DateTime.Now.ToString("yyyyMMddHHmmssfff"), machine.Detail, order.PayMoney.ToString("f2"));
  94. }
  95. else if (machine.BrandId == 3)
  96. {
  97. MqLinksHelper.Instance.doSomething(machine.Detail, order.PayMoney.ToString("f2"));
  98. }
  99. }
  100. //发送订单到客小爽MQ
  101. OrderMessageHelper.SendOrderMsg(order);
  102. //推送交易给java
  103. PosPushDataNewHelper.Trade(order, addMer);
  104. }
  105. }
  106. db.Dispose();
  107. }
  108. }
  109. }
  110. public string AesGcmDecrypt(string associatedData, string nonce, string ciphertext)
  111. {
  112. GcmBlockCipher gcmBlockCipher = new GcmBlockCipher(new AesEngine());
  113. AeadParameters aeadParameters = new AeadParameters(
  114. new KeyParameter(Encoding.UTF8.GetBytes(AppConfig.WeChatParam.AesGemKey)),
  115. 128,
  116. Encoding.UTF8.GetBytes(nonce),
  117. Encoding.UTF8.GetBytes(associatedData));
  118. gcmBlockCipher.Init(false, aeadParameters);
  119. byte[] data = Convert.FromBase64String(ciphertext);
  120. byte[] plaintext = new byte[gcmBlockCipher.GetOutputSize(data.Length)];
  121. int length = gcmBlockCipher.ProcessBytes(data, 0, data.Length, plaintext, 0);
  122. gcmBlockCipher.DoFinal(plaintext, length);
  123. return Encoding.UTF8.GetString(plaintext);
  124. }
  125. //分账队列
  126. public void StartProfitShare()
  127. {
  128. Thread th = new Thread(StartProfitShareListen);
  129. th.IsBackground = true;
  130. th.Start();
  131. }
  132. public void StartProfitShareListen()
  133. {
  134. bool op = true;
  135. while (op)
  136. {
  137. try
  138. {
  139. WebCMSEntities db = new WebCMSEntities();
  140. DateTime checkDate = DateTime.Now.AddMinutes(-1);
  141. var list = db.ConsumerOrders.Select(m => new { m.Id, m.CreateDate, m.Status, m.IsAct, m.PayMode }).Where(m => m.Status == 1 && m.PayMode == 2 && m.IsAct == 1 && m.CreateDate < checkDate).OrderBy(m => m.Id).Take(10).ToList();
  142. foreach (var sub in list)
  143. {
  144. LogHelper.Instance.WriteLog(DateTime.Now.ToString(), "微信分账队列异常");
  145. LogHelper.Instance.WriteLog("Id:" + sub.Id, "微信分账队列异常");
  146. ConsumerOrders order = db.ConsumerOrders.FirstOrDefault(m => m.Id == sub.Id);
  147. if (order != null)
  148. {
  149. order.Status = 2;
  150. MerchantInfo merchant = db.MerchantInfo.FirstOrDefault(m => m.Id == order.MerchantId) ?? new MerchantInfo();
  151. MerchantParamSet set = db.MerchantParamSet.FirstOrDefault(m => m.Id == order.MerchantId) ?? new MerchantParamSet();
  152. LogHelper.Instance.WriteLog("IsAct:" + order.IsAct, "微信分账队列监听");
  153. LogHelper.Instance.WriteLog("PayMoney:" + order.PayMoney, "微信分账队列监听");
  154. LogHelper.Instance.WriteLog("MinPayMoney:" + set.MinPayMoney, "微信分账队列监听");
  155. if (order.IsAct == 1 && order.PayMoney >= set.MinPayMoney)
  156. {
  157. MerchantAddInfo merchantAdd = db.MerchantAddInfo.FirstOrDefault(m => m.Id == order.MerchantId) ?? new MerchantAddInfo();
  158. //添加分账接收方
  159. if (merchant.Version == 0)
  160. {
  161. WeChatFunction.Instance.AddReceive(merchantAdd.SubMchid, "MERCHANT_ID", WeChatFunction.Instance.MchId, WeChatFunction.Instance.MchName, "PARTNER");
  162. merchant.Version = 1;
  163. db.SaveChanges();
  164. }
  165. LogHelper.Instance.WriteLog("准备分账", "微信分账队列监听");
  166. //发起分账
  167. decimal fee = order.PayMoney; //单位:分
  168. if (fee >= 1)
  169. {
  170. string TradeNo = order.SeoTitle;
  171. string OrderNo = order.OrderNo;
  172. // decimal hdfee = 0.0038M * order.PayMoney * (1 - set.GetPercent - 0.01M);
  173. // hdfee = decimal.Parse(hdfee.ToString("f2"));
  174. // decimal servicefee = 0.01M * order.PayMoney - hdfee;
  175. // servicefee = servicefee * 100;
  176. // int servicefeeNum = int.Parse(servicefee.ToString("f0"));
  177. List<ReceiverList> Receivers = new List<ReceiverList>();
  178. Receivers.Add(new ReceiverList()
  179. {
  180. type = "MERCHANT_ID", //分账接收方类型
  181. account = WeChatFunction.Instance.MchId, //分账接收方账号
  182. amount = int.Parse(fee.ToString("f0")), //分账金额
  183. description = "服务费", //分账描述
  184. });
  185. LogHelper.Instance.WriteLog("TradeNo:" + TradeNo, "微信分账队列监听");
  186. LogHelper.Instance.WriteLog("OrderNo:" + OrderNo, "微信分账队列监听");
  187. LogHelper.Instance.WriteLog(Newtonsoft.Json.JsonConvert.SerializeObject(Receivers), "微信分账队列监听");
  188. string ProfitShareResult = WeChatFunction.Instance.ProfitShare(merchantAdd.SubMchid, TradeNo, OrderNo, Receivers);
  189. LogHelper.Instance.WriteLog("分账结果:" + ProfitShareResult, "微信分账队列监听");
  190. order.DivideLog = "请求分账日志:" + ProfitShareResult;
  191. order.DivideFlag = 1;
  192. db.SaveChanges();
  193. //开始监听分账状态
  194. Dictionary<string, object> req = new Dictionary<string, object>();
  195. req.Add("SubMchid", merchantAdd.SubMchid); //子商户号
  196. req.Add("TradeNo", TradeNo); //微信订单号
  197. req.Add("OrderNo", OrderNo); //商户订单号
  198. RedisDbconn.Instance.AddList("ProfitShareQueue", Newtonsoft.Json.JsonConvert.SerializeObject(req));
  199. }
  200. else
  201. {
  202. RedisDbconn.Instance.AddList("ConsumerOrders:Divi:2:List", order.Id.ToString());
  203. // RedisDbconn.Instance.AddRightList("ConsumerOrders:Divi:2:" + order.MerchantId, order);
  204. }
  205. }
  206. }
  207. LogHelper.Instance.WriteLog("\n\n", "微信分账队列监听");
  208. }
  209. db.SaveChanges();
  210. db.Dispose();
  211. }
  212. catch (Exception ex)
  213. {
  214. LogHelper.Instance.WriteLog(DateTime.Now.ToString() + ":" + ex.ToString(), "微信分账队列异常");
  215. }
  216. Thread.Sleep(2000);
  217. }
  218. }
  219. //触发分账接口
  220. public void StartDivi()
  221. {
  222. Thread th = new Thread(ListenDivi);
  223. th.IsBackground = true;
  224. th.Start();
  225. }
  226. public void ListenDivi()
  227. {
  228. bool op = true;
  229. while (op)
  230. {
  231. string content = RedisDbconn.Instance.RPop<string>("AddWeChatDiviQueue");
  232. if (!string.IsNullOrEmpty(content))
  233. {
  234. try
  235. {
  236. int id = int.Parse(function.CheckInt(content));
  237. WebCMSEntities db = new WebCMSEntities();
  238. ConsumerOrders order = db.ConsumerOrders.FirstOrDefault(m => m.Id == id);
  239. if (order != null)
  240. {
  241. MerchantAddInfo merchantAdd = db.MerchantAddInfo.FirstOrDefault(m => m.Id == order.MerchantId) ?? new MerchantAddInfo();
  242. MerchantParamSet set = db.MerchantParamSet.FirstOrDefault(m => m.Id == order.MerchantId) ?? new MerchantParamSet();
  243. if (order.IsAct == 1 && order.PayMoney >= set.MinPayMoney)
  244. {
  245. //发起分账
  246. decimal fee = order.PayMoney; //单位:分
  247. if (fee >= 1)
  248. {
  249. string TradeNo = order.SeoTitle;
  250. string OrderNo = order.OrderNo;
  251. List<ReceiverList> Receivers = new List<ReceiverList>();
  252. Receivers.Add(new ReceiverList()
  253. {
  254. type = "MERCHANT_ID", //分账接收方类型
  255. account = WeChatFunction.Instance.MchId, //分账接收方账号
  256. amount = int.Parse(fee.ToString("f0")), //分账金额
  257. description = "服务费", //分账描述
  258. });
  259. LogHelper.Instance.WriteLog("TradeNo:" + TradeNo, "微信分账队列监听");
  260. LogHelper.Instance.WriteLog("OrderNo:" + OrderNo, "微信分账队列监听");
  261. LogHelper.Instance.WriteLog(Newtonsoft.Json.JsonConvert.SerializeObject(Receivers), "微信分账队列监听");
  262. string ProfitShareResult = WeChatFunction.Instance.ProfitShare(merchantAdd.SubMchid, TradeNo, OrderNo, Receivers);
  263. LogHelper.Instance.WriteLog("分账结果:" + ProfitShareResult, "微信分账队列监听");
  264. order.DivideLog = "请求分账日志:" + ProfitShareResult;
  265. order.DivideFlag = 1;
  266. db.SaveChanges();
  267. //开始监听分账状态
  268. Dictionary<string, object> req = new Dictionary<string, object>();
  269. req.Add("SubMchid", merchantAdd.SubMchid); //子商户号
  270. req.Add("TradeNo", TradeNo); //微信订单号
  271. req.Add("OrderNo", OrderNo); //商户订单号
  272. RedisDbconn.Instance.AddList("ProfitShareQueue", Newtonsoft.Json.JsonConvert.SerializeObject(req));
  273. }
  274. }
  275. }
  276. db.Dispose();
  277. }
  278. catch (Exception ex)
  279. {
  280. LogHelper.Instance.WriteLog(DateTime.Now.ToString() + ":" + ex.ToString(), "补微信分账异常");
  281. }
  282. }
  283. else
  284. {
  285. Thread.Sleep(2000);
  286. }
  287. }
  288. }
  289. }
  290. }