微信異步通知:
[AcceptVerbs("POST")] public void Notify() { //編碼(101-登錄無效,102-賬號無效,200-成功,201-失敗,202~299-其他原因1-99,300-無效提交方式,400-無效參數) MessagesDataCodeModel json = new MessagesDataCodeModel(false, "無效參數", 401); int notify_id = 0; string result = "failed"; try { //得到微信推送來的xml數據 Stream s = System.Web.HttpContext.Current.Request.InputStream; byte[] b = new byte[s.Length]; s.Read(b, 0, (int)s.Length); string postStr = Encoding.UTF8.GetString(b); SortedDictionary<string, string> requestXML = Common.TenpayUtil.GetInfoFromXml(postStr); if (requestXML != null && requestXML.Count > 0) { Models.WeiXinNotifyRecords model = GetNotifyModel(requestXML);//將xml數據轉換為Model model.remark = "";//postStr; model.CreateDate = DateTime.Now; notify_id = WeiXinNotifyRecordsBLL.Append(model);//記錄微信通知 model.ID = notify_id; #region 驗簽 //微信返回的簽名字符串 string sign = requestXML["sign"]; requestXML.Remove("sign"); //待簽名字符串 string signStr = AlipaySignature.GetSignContent(requestXML) + "&key=" + Common.ConfigApi.WeiXinPay_API_Key;//借用阿里的方法 string newsign = Utils.GetMD5(signStr).ToUpper();//MD5加密,轉大寫 bool ValidateSign = sign == newsign;//驗證簽名是否一致 #endregion #region 處理訂單 if (ValidateSign) { Models.TradeInfo tradeInfo = TradeInfoBLL.GetEntityByTradeNo(model.out_trade_no); if (tradeInfo != null && model.appid == ConfigApi.WeiXinPay_App_app_id && model.mch_id == ConfigApi.WeiXinPay_App_mch_id) { result = "success";//TODO:處理訂單邏輯,完成后 result="success" } else { result = "TradeError"; logger.Error("WeiXinPayController.Notify【訂單數據與通知數據不符】"); } } else { result = "CheckSignError"; logger.Error("WeiXinPayController.Notify【驗簽失敗】"); } #endregion } } catch (Exception ex) { logger.Error("WeiXinPayController.Notify【程序異常】", ex); result = "exception"; } string xmlstr = @"<xml><return_code><![CDATA[SUCCESS]]></return_code><return_msg><![CDATA[OK]]></return_msg></xml>"; if (result != "success") { xmlstr = @"<xml><return_code><![CDATA[FAIL]]></return_code><return_msg><![CDATA[FAIL]]></return_msg></xml>"; } HttpContext.Current.Response.Write(xmlstr); HttpContext.Current.Response.End(); }
把XML數據轉換為SortedDictionary<string, string>集合:
/// <summary> /// 把XML數據轉換為SortedDictionary<string, string>集合 /// </summary> /// <param name="strxml"></param> /// <returns></returns> public static SortedDictionary<string, string> GetInfoFromXml(string xmlstring) { SortedDictionary<string, string> sParams = new SortedDictionary<string, string>(); try { XmlDocument doc = new XmlDocument(); doc.LoadXml(xmlstring); XmlElement root = doc.DocumentElement; int len = root.ChildNodes.Count; for (int i = 0; i < len; i++) { string name = root.ChildNodes[i].Name; if (!sParams.ContainsKey(name)) { sParams.Add(name.Trim(), root.ChildNodes[i].InnerText.Trim()); } } } catch { } return sParams; }
把參數排序后拼接,得到簽名字符串:
public static string GetSignContent(IDictionary<string, string> parameters) { // 第一步:把字典按Key的字母順序排序 IDictionary<string, string> sortedParams = new SortedDictionary<string, string>(parameters); IEnumerator<KeyValuePair<string, string>> dem = sortedParams.GetEnumerator(); // 第二步:把所有參數名和參數值串在一起 StringBuilder query = new StringBuilder(""); while (dem.MoveNext()) { string key = dem.Current.Key; string value = dem.Current.Value; if (!string.IsNullOrEmpty(key) && !string.IsNullOrEmpty(value)) { query.Append(key).Append("=").Append(value).Append("&"); } } string content = query.ToString().Substring(0, query.Length - 1); return content; }
簽名算法文檔:https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=4_3