开放 奔腾 server 手机号 产品 etop 名称 ota lds
这两天完成了公司网站手机和PC端的支付对接,就是支付宝和微信.
对接完后有所感触,我们来聊一聊,微信支付的坑,为什么这么说呢,因为我在对接完支付宝后是很愉快的,基本上在demo上稍加修改就ok了,
对于支付宝我觉得基本上都没什么可以说的,而微信支付...
网上搜了一下,基本上没有看到有针对h5支付的成功教程,全都是复制粘贴官方文档,有意思吗.
恰好今天空了,来聊一聊微信支付的坑,当然本文也不是光吐槽了事,我会把完整的代码放到文末尾处,并加以适当的注释.
------------------------------
首先第一步去微信商户平台开通h5支付的.
因为公司的app中的微信支付是我对接的,所以在开始做手机网站的微信支付前,我首先想到去微信开放平台,新建了一个网站应用,
最后审核下来居然没有支付接口,只有获取用户信息的接口,我有点蒙笔,然后我打了一下客服电话,那边客服说h5支付在公众号平台中完成(注意:这里客服mm向我扔了一个巨坑,接着往下看)
既然这样,我就去了微信公众号平台,我们有服务号,拿着appid和mch_id就开始搞,一切都很好,但是提示[商户号该产品权限预开通中,请等待产品开通后重试],
我赶紧登录商户平台看一下,确定h5是开通的没问题,这个时候我注意到一个问题,就是我的公众号平台中的商户号和商户平台中的商户号不一致,我检查了公司信息,邮箱,手机号,两者都是完全一样的.
这个时候我把代码中的商户号(微信公众号平台的)改成了商户平台的商户号,请求接口返回[商户号和appid不匹配],我再次坚定,应该用微信公众号平台的商户号,因为appid是用的微信公众号平台的.
这个时候我陷入了焦灼,网上找了很久,没有搜到任何跟[商户号该产品权限预开通中,请等待产品开通后重试]有关的结果,我给微信支付技术发了一封邮件,阐述了我遇到的问题,
第二天技术回复了我,说"你开通h5支付的商户号是这个(商户平台的),而不是你现在用这个(微信公众号平台的)",这就是前面那个客服mm给我扔的大坑!!
邮件中还告诉我,我开通h5支付的商户号对应的appid是这个[appid**],而这个appid是我在开放平台中以前建的移动应用,也就是我们之前的app中用的...此时一万只草泥马在开始奔腾.
前后跨度太大,让我一时无法接受这个事实,
你先告诉我,h5支付为什么跟移动应用扯上关系了,其次你文档中提到了一星半点吗,没有!,客服mm也是调皮,动不动就h5支付跟开放平台没关系,只需要公众号平台.
还有你微信产品真多微信公众号平台(配置),微信开放平台(文档),微信商户平台(配置),我对接个微信支付要在三大平台之间来回切换,你真的666!!!
---------------------------------
平复一下心情,来看一下h5支付的流程,整个逻辑是这样的,首先用户打开手机网站某个服务页面
点击开通某个产品时,此时通过ajax请求带上产品价格,名称等参数来请求一个php脚本(wxpay_wap.php),
这个脚本根据官方文档要求的参数,组合,生成签名,请求统一下单接口,
返回一个支付url,然后用js生成一个a标签并模拟点击或者直接location.href应该都是可以的,
就会调起微信支付,支付完成后微信会发送一个请求到你设定的回调脚本(wxpay_wap_notify.php),在回调脚本里写业务逻辑
大体就是这样的,以下代码稍加修改就可以直接用了
---------------------------------
wxpay_wap.php
- // 文档中说的,价格单位是分
- $money = $_POST[‘money‘]*100; //商品价格
- // 前台请求的参数
- $title = $_POST[‘title‘];//商品名称
- $userid = $_POST[‘userid‘];//用户id
- $time = time();
- $nonce_str = "hcuasduvihasdiovjerjgvujsaru"; //随机字符串
- $appid = "wx**************33"; //在微信开放平台中的appid(先要创建一个移动应用)
- $mch_id = "147****642"; //商户号,在商户平台中查看
- $key = "51b*************************f15e"; //在微信开放平台中的
- $notify_url = "http://www.xxx.com/xxx/wxnotify.php"; //用户支付完后微信会来触发这个脚本,是处理业务逻辑的地方
- //订单号可以灵活使用,比如我这个地方把userid加进去,在异步回调的时候方便直接操作用户
- $out_trade_no = $time."__".$userid;
- // 下面的参数含义直接看文档
- $tmpArr = array(
- ‘appid‘=>$appid, //不要填成了 公众号原始id
- ‘attach‘=>$title,
- ‘body‘=>$title,
- ‘mch_id‘=>$mch_id,
- ‘nonce_str‘=>$nonce_str,
- ‘notify_url‘=>$notify_url,
- ‘out_trade_no‘=>$out_trade_no,
- ‘spbill_create_ip‘=>$_SERVER[‘REMOTE_ADDR‘],
- ‘total_fee‘=>$money,
- ‘trade_type‘=>‘Mweb‘
- );
- // 签名逻辑官网有说明,签名步骤就不解释了
- ksort($tmpArr);
- $buff = "";
- foreach ($tmpArr as $k => $v)
- {
- $buff .= $k . "=" . $v . "&";
- }
- $buff = trim($buff, "&");
- $stringSignTemp=$buff."&key=51b3363e91fe317fc346526f5933f15e";
- $sign= strtoupper(md5($stringSignTemp)); //签名
- $xml = "<xml>
- <appid>".$appid."</appid>
- <attach>".$title."</attach>
- <body>".$title."</body>
- <mch_id>".$mch_id."</mch_id>
- <nonce_str>".$nonce_str."</nonce_str>
- <notify_url>".$notify_url."</notify_url>
- <out_trade_no>".$out_trade_no."</out_trade_no>
- <spbill_create_ip>".$_SERVER[‘REMOTE_ADDR‘]."</spbill_create_ip>
- <total_fee>".$money."</total_fee>
- <trade_type>MWEB</trade_type>
- <sign>".$sign."</sign>
- </xml> ";
- $posturl = "https://api.mch.weixin.qq.com/pay/unifiedorder";
- $ch = curl_init($posturl);
- curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
- curl_setopt($ch, CURLOPT_POST, 1);
- curl_setopt($ch, CURLOPT_POSTFIELDS, $xml);
- $response = curl_exec($ch);
- curl_close($ch);
- $xmlobj = json_decode(json_encode(simplexml_load_string($response, ‘SimpleXMLElement‘, LIBXML_NOCDATA )));
- exit($xmlobj->mweb_url);
------------------------------------------
wxpay_wap_notify.php
- // 本脚本的业务逻辑只是个例子,仅供参考
- // 拿到原始数据并通过xml类解析为对象
- $postObj = simplexml_load_string(file_get_contents("php://input"), ‘SimpleXMLElement‘, LIBXML_NOCDATA);
- // 你可以通过下面这种方式来看一下微信究竟返回了那些参数,请保证log.php存在并且有写权限
- // file_put_contents(dirname(__file__)."/log.php",file_get_contents("php://input"));
- $arr = array();
- foreach($postObj as $key = >$value) {
- $arr[$key] = $value;
- }
- // 订单状态
- $status = $arr[‘result_code‘];
- if (trim($status) == "SUCCESS") {
- // 微信订单
- $out_trade_no = $arr[‘transaction_id‘];
- // 价格
- $money = $arr[‘total_fee‘] / 100;
- // 在商户订单号中提取用户id,上一个脚本中我说了这个商户订单号可以灵活使用
- $uid = explode("__", $arr[‘out_trade_no‘])[1];
- // 在数据库中检查这个订单号是否已经处理过了以免重复处理,因为很多原因微信可能多次触发本脚本
- // checkrepeat(orderid);
- /////////////////////////////////////////////////////////////
- //// /////
- //// 这里处理业务逻辑.... /////
- //// /////
- /////////////////////////////////////////////////////////////
- // 处理完逻辑返回这个xml数据,告诉微信服务器,这个订单号已经处理完了不要在来骚扰我了
- $xml = "
- <xml>
- <return_code><![CDATA[".$status."]]></return_code>
- <return_msg><![CDATA[OK]]></return_msg>
- </xml>";
- echo $xml;
- }
来源: http://www.bubuko.com/infodetail-2333850.html