最近公司上的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();