WxPayData.cs 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. using System.Collections.Generic;
  2. using System.Linq;
  3. using System.Security.Cryptography;
  4. using System.Text;
  5. using System.Web;
  6. using System.Xml;
  7. using LitJson;
  8. namespace GxPress.Common.WechatPay
  9. {
  10. public class WxPayData
  11. {
  12. public const string SIGN_TYPE_MD5 = "MD5";
  13. public const string SIGN_TYPE_HMAC_SHA256 = "HMAC-SHA256";
  14. public WxPayData()
  15. {
  16. }
  17. //采用排序的Dictionary的好处是方便对数据包进行签名,不用再签名之前再做一次排序
  18. private SortedDictionary<string, object> m_values = new SortedDictionary<string, object>();
  19. /**
  20. * 设置某个字段的值
  21. * @param key 字段名
  22. * @param value 字段值
  23. */
  24. public void SetValue(string key, object value)
  25. {
  26. m_values[key] = value;
  27. }
  28. /**
  29. * 根据字段名获取某个字段的值
  30. * @param key 字段名
  31. * @return key对应的字段值
  32. */
  33. public object GetValue(string key)
  34. {
  35. object o = null;
  36. m_values.TryGetValue(key, out o);
  37. return o;
  38. }
  39. /**
  40. * 判断某个字段是否已设置
  41. * @param key 字段名
  42. * @return 若字段key已被设置,则返回true,否则返回false
  43. */
  44. public bool IsSet(string key)
  45. {
  46. object o = null;
  47. m_values.TryGetValue(key, out o);
  48. if (null != o)
  49. return true;
  50. else
  51. return false;
  52. }
  53. /**
  54. * @将Dictionary转成xml
  55. * @return 经转换得到的xml串
  56. * @throws WxPayException
  57. **/
  58. public string ToXml()
  59. {
  60. //数据为空时不能转化为xml格式
  61. if (0 == m_values.Count)
  62. {
  63. //Log.Error(this.GetType().ToString(), "WxPayData数据为空!");
  64. throw new WxPayException("WxPayData数据为空!");
  65. }
  66. string xml = "<xml>";
  67. foreach (KeyValuePair<string, object> pair in m_values)
  68. {
  69. //字段值不能为null,会影响后续流程
  70. if (pair.Value == null)
  71. {
  72. //Log.Error(this.GetType().ToString(), "WxPayData内部含有值为null的字段!");
  73. throw new WxPayException("WxPayData内部含有值为null的字段!");
  74. }
  75. if (pair.Value.GetType() == typeof(int))
  76. {
  77. xml += "<" + pair.Key + ">" + pair.Value + "</" + pair.Key + ">";
  78. }
  79. else if (pair.Value.GetType() == typeof(string))
  80. {
  81. xml += "<" + pair.Key + ">" + "<![CDATA[" + pair.Value + "]]></" + pair.Key + ">";
  82. }
  83. else//除了string和int类型不能含有其他数据类型
  84. {
  85. //Log.Error(this.GetType().ToString(), "WxPayData字段数据类型错误!");
  86. throw new WxPayException("WxPayData字段数据类型错误!");
  87. }
  88. }
  89. xml += "</xml>";
  90. return xml;
  91. }
  92. /**
  93. * @将xml转为WxPayData对象并返回对象内部的数据
  94. * @param string 待转换的xml串
  95. * @return 经转换得到的Dictionary
  96. * @throws WxPayException
  97. */
  98. public SortedDictionary<string, object> FromXml(string xml)
  99. {
  100. if (string.IsNullOrEmpty(xml))
  101. {
  102. //Log.Error(this.GetType().ToString(), "将空的xml串转换为WxPayData不合法!");
  103. throw new WxPayException("将空的xml串转换为WxPayData不合法!");
  104. }
  105. SafeXmlDocument xmlDoc = new SafeXmlDocument();
  106. xmlDoc.LoadXml(xml);
  107. XmlNode xmlNode = xmlDoc.FirstChild;//获取到根节点<xml>
  108. XmlNodeList nodes = xmlNode.ChildNodes;
  109. foreach (XmlNode xn in nodes)
  110. {
  111. XmlElement xe = (XmlElement)xn;
  112. m_values[xe.Name] = xe.InnerText;//获取xml的键值对到WxPayData内部的数据中
  113. }
  114. try
  115. {
  116. //2015-06-29 错误是没有签名
  117. if(m_values["return_code"] != "SUCCESS")
  118. {
  119. return m_values;
  120. }
  121. CheckSign();//验证签名,不通过会抛异常
  122. }
  123. catch(WxPayException ex)
  124. {
  125. throw new WxPayException(ex.Message);
  126. }
  127. return m_values;
  128. }
  129. /**
  130. * @Dictionary格式转化成url参数格式
  131. * @ return url格式串, 该串不包含sign字段值
  132. */
  133. public string ToUrl()
  134. {
  135. string buff = "";
  136. foreach (KeyValuePair<string, object> pair in m_values)
  137. {
  138. if (pair.Value == null)
  139. {
  140. //Log.Error(this.GetType().ToString(), "WxPayData内部含有值为null的字段!");
  141. throw new WxPayException("WxPayData内部含有值为null的字段!");
  142. }
  143. if (pair.Key != "sign" && pair.Value.ToString() != "")
  144. {
  145. buff += pair.Key + "=" + pair.Value + "&";
  146. }
  147. }
  148. buff = buff.Trim('&');
  149. return buff;
  150. }
  151. /**
  152. * @Dictionary格式化成Json
  153. * @return json串数据
  154. */
  155. public string ToJson()
  156. {
  157. string jsonStr = JsonMapper.ToJson(m_values);
  158. return jsonStr;
  159. }
  160. /**
  161. * @values格式化成能在Web页面上显示的结果(因为web页面上不能直接输出xml格式的字符串)
  162. */
  163. public string ToPrintStr()
  164. {
  165. string str = "";
  166. foreach (KeyValuePair<string, object> pair in m_values)
  167. {
  168. if (pair.Value == null)
  169. {
  170. //Log.Error(this.GetType().ToString(), "WxPayData内部含有值为null的字段!");
  171. throw new WxPayException("WxPayData内部含有值为null的字段!");
  172. }
  173. str += string.Format("{0}={1}\n", pair.Key, pair.Value.ToString());
  174. }
  175. str = HttpUtility.HtmlEncode(str);
  176. //Log.Debug(this.GetType().ToString(), "Print in Web Page : " + str);
  177. return str;
  178. }
  179. /**
  180. * @生成签名,详见签名生成算法
  181. * @return 签名, sign字段不参加签名
  182. */
  183. public string MakeSign(string signType){
  184. //转url格式
  185. string str = ToUrl();
  186. //在string后加入API KEY
  187. str += "&key=" + WxPayConfig.GetConfig().GetKey();
  188. if (signType == SIGN_TYPE_MD5)
  189. {
  190. var md5 = MD5.Create();
  191. var bs = md5.ComputeHash(Encoding.UTF8.GetBytes(str));
  192. var sb = new StringBuilder();
  193. foreach (byte b in bs)
  194. {
  195. sb.Append(b.ToString("x2"));
  196. }
  197. //所有字符转为大写
  198. return sb.ToString().ToUpper();
  199. }
  200. else if(signType==SIGN_TYPE_HMAC_SHA256)
  201. {
  202. return CalcHMACSHA256Hash(str, WxPayConfig.GetConfig().GetKey());
  203. }else{
  204. throw new WxPayException("sign_type 不合法");
  205. }
  206. }
  207. /**
  208. * @生成签名,详见签名生成算法
  209. * @return 签名, sign字段不参加签名 SHA256
  210. */
  211. public string MakeSign()
  212. {
  213. return MakeSign(SIGN_TYPE_HMAC_SHA256);
  214. }
  215. /**
  216. *
  217. * 检测签名是否正确
  218. * 正确返回true,错误抛异常
  219. */
  220. public bool CheckSign(string signType)
  221. {
  222. //如果没有设置签名,则跳过检测
  223. if (!IsSet("sign"))
  224. {
  225. //Log.Error(this.GetType().ToString(), "WxPayData签名存在但不合法!");
  226. throw new WxPayException("WxPayData签名存在但不合法!");
  227. }
  228. //如果设置了签名但是签名为空,则抛异常
  229. else if (GetValue("sign") == null || GetValue("sign").ToString() == "")
  230. {
  231. //Log.Error(this.GetType().ToString(), "WxPayData签名存在但不合法!");
  232. throw new WxPayException("WxPayData签名存在但不合法!");
  233. }
  234. //获取接收到的签名
  235. string return_sign = GetValue("sign").ToString();
  236. //在本地计算新的签名
  237. string cal_sign = MakeSign(signType);
  238. if (cal_sign == return_sign)
  239. {
  240. return true;
  241. }
  242. //Log.Error(this.GetType().ToString(), "WxPayData签名验证错误!");
  243. throw new WxPayException("WxPayData签名验证错误!");
  244. }
  245. /**
  246. *
  247. * 检测签名是否正确
  248. * 正确返回true,错误抛异常
  249. */
  250. public bool CheckSign()
  251. {
  252. return CheckSign(SIGN_TYPE_HMAC_SHA256);
  253. }
  254. /**
  255. * @获取Dictionary
  256. */
  257. public SortedDictionary<string, object> GetValues()
  258. {
  259. return m_values;
  260. }
  261. private string CalcHMACSHA256Hash(string plaintext, string salt)
  262. {
  263. string result = "";
  264. var enc = Encoding.Default;
  265. byte[]
  266. baText2BeHashed = enc.GetBytes(plaintext),
  267. baSalt = enc.GetBytes(salt);
  268. System.Security.Cryptography.HMACSHA256 hasher = new HMACSHA256(baSalt);
  269. byte[] baHashedText = hasher.ComputeHash(baText2BeHashed);
  270. result = string.Join("", baHashedText.ToList().Select(b => b.ToString("x2")).ToArray());
  271. return result;
  272. }
  273. }
  274. }