最近公司上的app需要增加一个微信支付的功能。初看微信支付API开发文档还是很简单的,但是在简单的背后却隐藏这几个不小的坑。
公司要接入的是APP支付(微信支付有如下几种[图1]),所以这里只谈APP支付遇到的坑,其它支付估计也有类似的坑吧。
微信给出的业务流程
商户系统和微信支付系统主要交互说明:
步骤1:用户在商户APP中选择商品,提交订单,选择微信支付。
步骤2:商户后台收到用户支付单,调用微信支付统一下单接口。参见【统一下单API】。
步骤3:统一下单接口返回正常的prepay_id,再按签名规范重新生成签名后,将数据传输给APP。参与签名的字段名为appId,partnerId,prepayId,nonceStr,timeStamp,package。注意:package的值格式为Sign=WXPay
步骤4:商户APP调起微信支付。api参见本章节【app端开发步骤说明】
步骤5:商户后台接收支付通知。api参见【支付结果通知API】
步骤6:商户后台查询支付结果。,api参见【查询订单API】
1、统一下单API遇到的坑
商户后台系统需要把一些关于支付的信息post到微信的统一下单接口,post信息是xml格式的,如下图2;一开始我直接通过拼接字符串的方式写的,结果就各种不成功,由于提交的xml里面有中文,所以需要在xml里指定utf-8的编码,如此就可以提交成功得到prepay_id了。
2、APP支付业务流程步骤3:统一下单接口返回正常的prepay_id,再按签名规范重新生成签名后,将数据传输给APP。参与签名的字段名为appId,partnerId,prepayId,nonceStr,timeStamp,package。注意:package的值格式为Sign=WXPay遇到的坑。
由于需要签名,然后我就根据微信支付签名算法把步骤3中提到的字段加上key值进行签名了,结果android和ios端用接口返回的prepayId和签名掉起微信时总是报参数错误,以为是签名签的不对,最后发现签名的这些字段名必须全部小些才可以。
附C#部分关键代码
1、签名算法
string strA="appid=wx111111111111&body=xxx付费&mch_id=141111111&nonce_str=37A749D808E46495A8DA1E5352D03CAE¬ify_url=https://weixinpay.com/notify.ashx&out_trade_no=20170315113530201659&spbill_create_ip=127.0.0.1&total_fee=1&trade_type=APP&key=imkey121";
string sign=System.Web.Security.FormsAuthentication.HashPasswordForStoringInConfigFile(strA, "MD5").ToUpper();
2、制作需要post到微信接口的xml数据
XmlDataDocument doc = new XmlDataDocument();
XmlNode node = doc.CreateXmlDeclaration("1.0", "utf-8", "");
doc.AppendChild(node);
XmlNode root = doc.CreateElement("xml");
doc.AppendChild(root);
CreateNode(doc, root, "appid", appid);
CreateNode(doc, root, "mch_id", mch_id);
CreateNode(doc, root, "nonce_str", nonce_str);
CreateNode(doc, root, "sign", sign);
CreateNode(doc, root, "body", body);
CreateNode(doc, root, "out_trade_no", out_trade_no);
CreateNode(doc, root, "total_fee", total_fee);
CreateNode(doc, root, "spbill_create_ip", spbill_create_ip);
CreateNode(doc, root, "notify_url", notify_url);
CreateNode(doc, root, "trade_type", trade_type);
//CreateNode方法
private void CreateNode(XmlDocument xmlDoc, XmlNode parentNode, string name, string value)
{
XmlNode node = xmlDoc.CreateNode(XmlNodeType.Element, name, null);
node.InnerText = value;
parentNode.AppendChild(node);
}
3、post方法调用微信接口
//menuInfo=需要post的xml参数
//postUrl 微信接口地址
//returnValue 通过微信接口获取的返回值xml格式
byte[] byteData = Encoding.UTF8.GetBytes(menuInfo);
Uri uri = new Uri(postUrl);
HttpWebRequest webReq = (HttpWebRequest)WebRequest.Create(uri);
webReq.Method = "POST";
webReq.ContentType = "application/x-www-form-urlencoded";
webReq.ContentLength = byteData.Length;
//定义Stream信息
Stream stream = webReq.GetRequestStream();
stream.Write(byteData, 0, byteData.Length);
stream.Close();
//获取返回信息
HttpWebResponse response = (HttpWebResponse)webReq.GetResponse();
StreamReader streamReader = new StreamReader(response.GetResponseStream(), Encoding.UTF8);
returnValue = streamReader.ReadToEnd();
//关闭信息
streamReader.Close();
response.Close();
stream.Close();