最近开发微信小程序中用到了微信支付功能, 接口开发用的 ASP.NET web API;
在支付成功后, 接口接受到微信服务器的支付通知结果, 处理完数据, 接口返回给微信服务数据时出现了问题.
微信服务器识别不到返回的数据, 导致重复通知.
最终解决代码如下:
- protected virtual HttpResponseMessage RetMessage(object msg)
- {
- return new HttpResponseMessage
- {
- Content = new StringContent(msg.ToString(),new UTF8Encoding(false)
- , "text/plain")
- };
- }
- [HttpPost]
- public HttpResponseMessage Wx_Notify()
- {
- // 业务处理代码, 以下为返回
- WxPayData res = new WxPayData();
- res.SetValue("return_code", "SUCCESS");
- res.SetValue("return_msg", "OK");
- return RetMessage(res.ToXml());
- }
WxPayData 为腾讯官方提供源码中的类
- namespace WxPayAPI
- {
- /// <summary>
- /// 微信支付协议接口数据类, 所有的 API 接口通信都依赖这个数据结构,
- /// 在调用接口之前先填充各个字段的值, 然后进行接口通信,
- /// 这样设计的好处是可扩展性强, 用户可随意对协议进行更改而不用重新设计数据结构,
- /// 还可以随意组合出不同的协议数据包, 不用为每个协议设计一个数据包结构
- /// </summary>
- public class WxPayData
- {
- public WxPayData()
- {
- }
- // 采用排序的 Dictionary 的好处是方便对数据包进行签名, 不用再签名之前再做一次排序
- private SortedDictionary<string, object> m_values = new SortedDictionary<string, object>();
- /**
- * 设置某个字段的值
- * @param key 字段名
- * @param value 字段值
- */
- public void SetValue(string key, object value)
- {
- m_values[key] = value;
- }
- /**
- * 根据字段名获取某个字段的值
- * @param key 字段名
- * @return key 对应的字段值
- */
- public object GetValue(string key)
- {
- object o = null;
- m_values.TryGetValue(key, out o);
- return o;
- }
- /**
- * 判断某个字段是否已设置
- * @param key 字段名
- * @return 若字段 key 已被设置, 则返回 true, 否则返回 false
- */
- public bool IsSet(string key)
- {
- object o = null;
- m_values.TryGetValue(key, out o);
- if (null != o)
- return true;
- else
- return false;
- }
- /**
- * @将 Dictionary 转成 xml
- * @return 经转换得到的 xml 串
- * @throws WxPayException
- **/
- public string ToXml()
- {
- // 数据为空时不能转化为 xml 格式
- if (0 == m_values.Count)
- {
- Log.Error(this.GetType().ToString(), "WxPayData 数据为空!");
- throw new WxPayException("WxPayData 数据为空!");
- }
- string xml = "<xml>";
- foreach (KeyValuePair<string, object> pair in m_values)
- {
- // 字段值不能为 null, 会影响后续流程
- if (pair.Value == null)
- {
- Log.Error(this.GetType().ToString(), "WxPayData 内部含有值为 null 的字段!");
- throw new WxPayException("WxPayData 内部含有值为 null 的字段!");
- }
- if (pair.Value.GetType() == typeof(int))
- {
- xml += "<" + pair.Key + ">" + pair.Value + "</" + pair.Key + ">";
- }
- else if (pair.Value.GetType() == typeof(string))
- {
- xml += "<" + pair.Key + ">" + "<![CDATA[" + pair.Value + "]]></" + pair.Key + ">";
- }
- else// 除了 string 和 int 类型不能含有其他数据类型
- {
- Log.Error(this.GetType().ToString(), "WxPayData 字段数据类型错误!");
- throw new WxPayException("WxPayData 字段数据类型错误!");
- }
- }
- xml += "</xml>";
- return xml;
- }
- /**
- * @将 xml 转为 WxPayData 对象并返回对象内部的数据
- * @param string 待转换的 xml 串
- * @return 经转换得到的 Dictionary
- * @throws WxPayException
- */
- public SortedDictionary<string, object> FromXml(string xml)
- {
- if (string.IsNullOrEmpty(xml))
- {
- Log.Error(this.GetType().ToString(), "将空的 xml 串转换为 WxPayData 不合法!");
- throw new WxPayException("将空的 xml 串转换为 WxPayData 不合法!");
- }
- XmlDocument xmlDoc = new XmlDocument();
- xmlDoc.LoadXml(xml);
- XmlNode xmlNode = xmlDoc.FirstChild;// 获取到根节点 < xml>
- XmlNodeList nodes = xmlNode.ChildNodes;
- foreach (XmlNode xn in nodes)
- {
- XmlElement xe = (XmlElement)xn;
- m_values[xe.Name] = xe.InnerText;// 获取 xml 的键值对到 WxPayData 内部的数据中
- }
- try
- {
- //2015-06-29 错误是没有签名
- if(m_values["return_code"] != "SUCCESS")
- {
- return m_values;
- }
- CheckSign();// 验证签名, 不通过会抛异常
- }
- catch(WxPayException ex)
- {
- throw new WxPayException(ex.Message);
- }
- return m_values;
- }
- /**
- * @Dictionary 格式转化成 url 参数格式
- * @ return url 格式串, 该串不包含 sign 字段值
- */
- public string ToUrl()
- {
- string buff = "";
- foreach (KeyValuePair<string, object> pair in m_values)
- {
- if (pair.Value == null)
- {
- Log.Error(this.GetType().ToString(), "WxPayData 内部含有值为 null 的字段!");
- throw new WxPayException("WxPayData 内部含有值为 null 的字段!");
- }
- if (pair.Key != "sign" && pair.Value.ToString() != "")
- {
- buff += pair.Key + "=" + pair.Value + "&";
- }
- }
- buff = buff.Trim('&');
- return buff;
- }
- /**
- * @Dictionary 格式化成 JSON
- * @return JSON 串数据
- */
- public string ToJson()
- {
- string jsonStr = JSON.ToJson(m_values);
- return jsonStr;
- }
- /**
- * @values 格式化成能在 Web 页面上显示的结果 (因为 Web 页面上不能直接输出 xml 格式的字符串)
- */
- public string ToPrintStr()
- {
- string str = "";
- foreach (KeyValuePair<string, object> pair in m_values)
- {
- if (pair.Value == null)
- {
- Log.Error(this.GetType().ToString(), "WxPayData 内部含有值为 null 的字段!");
- throw new WxPayException("WxPayData 内部含有值为 null 的字段!");
- }
- str += string.Format("{0}={1}<br>", pair.Key, pair.Value.ToString());
- }
- Log.Debug(this.GetType().ToString(), "Print in Web Page :" + str);
- return str;
- }
- /**
- * @生成签名, 详见签名生成算法
- * @return 签名, sign 字段不参加签名
- */
- public string MakeSign()
- {
- // 转 url 格式
- string str = ToUrl();
- // 在 string 后加入 API KEY
- str += "&key=" + WxPayConfig.KEY;
- //MD5 加密
- var md5 = MD5.Create();
- var bs = md5.ComputeHash(Encoding.UTF8.GetBytes(str));
- var sb = new StringBuilder();
- foreach (byte b in bs)
- {
- sb.Append(b.ToString("x2"));
- }
- // 所有字符转为大写
- return sb.ToString().ToUpper();
- }
- /**
- *
- * 检测签名是否正确
- * 正确返回 true, 错误抛异常
- */
- public bool CheckSign()
- {
- // 如果没有设置签名, 则跳过检测
- if (!IsSet("sign"))
- {
- Log.Error(this.GetType().ToString(), "WxPayData 签名存在但不合法!");
- throw new WxPayException("WxPayData 签名存在但不合法!");
- }
- // 如果设置了签名但是签名为空, 则抛异常
- else if(GetValue("sign") == null || GetValue("sign").ToString() == "")
- {
- Log.Error(this.GetType().ToString(), "WxPayData 签名存在但不合法!");
- throw new WxPayException("WxPayData 签名存在但不合法!");
- }
- // 获取接收到的签名
- string return_sign = GetValue("sign").ToString();
- // 在本地计算新的签名
- string cal_sign = MakeSign();
- if (cal_sign == return_sign)
- {
- return true;
- }
- Log.Error(this.GetType().ToString(), "WxPayData 签名验证错误!");
- throw new WxPayException("WxPayData 签名验证错误!");
- }
- /**
- * @获取 Dictionary
- */
- public SortedDictionary<string, object> GetValues()
- {
- return m_values;
- }
- }
- }
来源: http://www.bubuko.com/infodetail-2990173.html