PHP实现微信支付(jsapi支付)流程

最近接触到一个项目,涉及到微信支付,搞微信开发这么久以来,还没搞过支付,之前也就搞过公众号发红包,感谢前辈们的探索,我看了他们的博文,让我少走了很多弯路。

前期准备:
1.微信认证服务号,并且开通了微信支付
2.微信支付SDK,下载地址:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=11_1
3.登录微信支付平台https://pay.weixin.qq.com/index.php/account/api_cert下载支付证书

方法步骤:
1.demo文件处理
(1)将官方的demo下载下来,文件名为WxpayAPI_php_v3,把这文件重命名为wxpay,为了后边书写目录方便;

(2)打开lib文件夹下的WxPay.Api.PHP文件,在537行有一段curl网络请求配置代码:

  1. curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,TRUE);  

  2. curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,2);//严格校验  

替换成:

  1. curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,FALSE);  

  2. curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,2);//严格校验  

为了禁止 cURL 验证对等证书(peer's certificate)。


(3)打开lib文件夹下的WxPay.Config.php文件,第25行开始,根据自己的账号完成基本信息设置;

  1. const APPID = '公众账号APPID';  

  2. const MCHID = '商户号';  

  3. const KEY = '商户支付密钥';  

  4. const APPSECRET = '公众帐号secert';  

(4)打开lib文件夹下的WxPay.Notify.php文件,第79行的代码:

  1. if($needSign == true &&   

  2.     $this->GetReturn_code($return_code) == "SUCCESS")  

  3. {  

  4.     $this->SetSign();  

  5. }  

改成:


  1. if($needSign == true &&   

  2.     $this->GetReturn_code() == "SUCCESS")  

  3. {  

  4.     $this->SetSign();  

  5. }  

(5)打开cert证书目录,将里边的两个证书换成自己的支付证书。


2.公众号后台设置

(1)配置网页授权域名,我的域名是(xy.chuyin.ren);

1.png

(1)配置支付授权目录,域名是(xy.chuyin.ren),我将demo放到此域名指向的目录的weixinopen/文件夹下,demo中jsapi.php文件位于example/目录下,所以支付授权目录为:xy.chuyin.ren/weixinopen/wxpay/example/

2.png

3.支付流程
打开example目录下的jsapi.php文件,支付发起和处理,都是在这里完成。
(1)获取用户openid

之前配置好了自己的APPID和APPSecert,所以这里不用处理。

  1. //①、获取用户openid  

  2. $tools = new JsApiPay();  

  3. $openId = $tools->GetOpenid();  

这里首先初始化的一个JsApiPay()类得到一个对象,文件对应example/目录下的WxPay.JsApiPay.php,调用GetOpenid()方法,会自动获取自己的openID。


(2)统一下单

  1. //②、统一下单  

  2. $input = new WxPayUnifiedOrder();  

  3. $input->SetBody("test");  

  4. $input->SetAttach("test");  

  5. $input->SetOut_trade_no(WxPayConfig::MCHID.date("YmdHis"));  

  6. $input->SetTotal_fee("1");  

  7. $input->SetTime_start(date("YmdHis"));  

  8. $input->SetTime_expire(date("YmdHis", time() + 600));  

  9. $input->SetGoods_tag("test");  

  10. $input->SetNotify_url("http://paysdk.weixin.qq.com/example/notify.php");  

  11. $input->SetTrade_type("JSAPI");  

  12. $input->SetOpenid($openId);  

  13. $order = WxPayApi::unifiedOrder($input);  

  14. echo '<font color="#f00"><b>统一下单支付单信息</b></font><br/>';  

  15. printf_info($order);  

  16. $jsApiParameters = $tools->GetJsApiParameters($order);  

对应WxPay.Api.php的第24行的unifiedOrder()方法,配置订单信息和支付回调函数,这里需要修改几个参数:


A. 商品名称:

  1. $input->SetBody("test");  

B. 订单号

  1. $input->SetOut_trade_no(WxPayConfig::MCHID.date("YmdHis"));  

C. 支付金额

$input->SetTotal_fee("1");

D. 支付验证链接

设置为你的notify.php文件所在的位置,所以我这里设置为:http://xy.chuyin.ren/weixinopen/wxpay/example/notify.php

也可以写其他地址,当然要在支付授权域名之下,支付成功之后就会自动回调到该链接指定的方法里边,可以在里边进行判断和数据库操作.

$input->SetNotify_url("http://paysdk.weixin.qq.com/example/notify.php");

E. 附加参数

  1. $input->SetAttach("test");  


附加参数,可填可不填,填写的话,里边字符串最好不要出现空格。
这时候,点击支付应该就可以成功支付了。

(3)发起支付

[javascript] view plain copy
  1. <script type="text/javascript">  

  2. //调用微信JS api 支付  

  3. function jsApiCall()  

  4. {  

  5.     WeixinJSBridge.invoke(  

  6.         'getBrandWCPayRequest',  

  7.         <?php echo $jsApiParameters; ?>,  

  8.         function(res){  

  9.             WeixinJSBridge.log(res.err_msg);  

  10.             alert(res.err_code+res.err_desc+res.err_msg);  

  11.         }  

  12.     );  

  13. }  

  14.   

  15. function callpay()  

  16. {  

  17.     if (typeof WeixinJSBridge == "undefined"){  

  18.         if( document.addEventListener ){  

  19.             document.addEventListener('WeixinJSBridgeReady', jsApiCall, false);  

  20.         }else if (document.attachEvent){  

  21.             document.attachEvent('WeixinJSBridgeReady', jsApiCall);   

  22.             document.attachEvent('onWeixinJSBridgeReady', jsApiCall);  

  23.         }  

  24.     }else{  

  25.         jsApiCall();  

  26.     }  

  27. }  

  28. </script>  

点击立即支付按钮调用的就是 callpay() 函数,他有会调用jsApiCall() 函数打开支付程序。


3.png

jsApiCall() 函数会监听每一步动作:

4.png

 res.err_msg 为get_brand_wcpay_request:cancel 表明前端判断的取消支付,es.err_msg 为get_brand_wcpay_request:ok 表明前端判断的支付成功,我们可以根据这个将支付跳转到成功页面。

(4)支持成功回调

通过前端jsApiCall()函数可以监听支付结果,但是这个并不可信。确认是否支付成功还是应当通过notify.php 处理业务逻辑。前边配置好了支付验证链接SetNotify_url(),支付完成后,微信服务器会根据链接自动请求你的notify.php文件,打开这个文件,其实这个文件最主要的代码就两行:

  1. $notify = new PayNotifyCallBack();  

  2. $notify->Handle(false);  

由此跟踪到WxPay.Notify.php类文件的Handle()函数:

  1. /** 

  2.  *  

  3.  * 回调入口 

  4.  * @param bool $needSign  是否需要签名输出 

  5.  */  

  6. final public function Handle($needSign = true)  

  7. {  

  8.     $msg = "OK";  

  9.     //当返回false的时候,表示notify中调用NotifyCallBack回调失败获取签名校验失败,此时直接回复失败  

  10.     $result = WxpayApi::notify(array($this'NotifyCallBack'), $msg);  

  11.     if($result == false){  

  12.         $this->SetReturn_code("FAIL");  

  13.         $this->SetReturn_msg($msg);  

  14.         $this->ReplyNotify(false);  

  15.         return;  

  16.     } else {  

  17.         //该分支在成功回调到NotifyCallBack方法,处理完成之后流程  

  18.         $this->SetReturn_code("SUCCESS");  

  19.         $this->SetReturn_msg("OK");  

  20.     }  

  21.     $this->ReplyNotify($needSign);  

  22. }  

主要代码:

  1. $result = WxpayApi::notify(array($this'NotifyCallBack'), $msg);  

然后来到WxPay.Api.php文件的第411行,notify()函数:

  1. /** 

  2.  *  

  3.  * 支付结果通用通知 

  4.  * @param function $callback 

  5.  * 直接回调函数使用方法: notify(you_function); 

  6.  * 回调类成员函数方法:notify(array($this, you_function)); 

  7.  * $callback  原型为:function function_name($data){} 

  8.  */  

  9. public static function notify($callback, &$msg)  

  10. {  

  11.     //获取通知的数据  

  12.     $xml = $GLOBALS['HTTP_RAW_POST_DATA'];  

  13.     //file_put_contents('log.txt',$xml,FILE_APPEND);  

  14.     //如果返回成功则验证签名  

  15.     try {  

  16.         $result = WxPayResults::Init($xml);  

  17.     } catch (WxPayException $e){  

  18.         $msg = $e->errorMessage();  

  19.         return false;  

  20.     }  

  21.       

  22.     return call_user_func($callback$result);  

  23. }  

这里面的$xml=$GLOBALS['HTTP_RAW_POST_DATA'],就是支付成功后用户返回给你的一个结果,他是一个xml格式的字符串。


4-1.png

我们可以将这里返回的xml数据记录下来,然后打开看看$out_trade_no就是在支付之前我自己设置的订单号码,$attach就是设置的附加参数。

得到了这个订单号,然后我就直接在下面写支付成功后的逻辑了,比如改变数据库中的数据等等。

这样 微信支付的 JsApi支付就大致分析完成了。

5.png