PHP实现⽀付(jsapi⽀付)流程步骤详解
最近接触到⼀个项⽬,涉及到⽀付,搞开发这么久以来,还没搞过⽀付,之前也就搞过发红包,感谢前辈们的探索,我看了他们的博⽂,让我少⾛了很多弯路。
前期准备:
1.认证服务号,并且开通了⽀付
⽅法步骤:
1.demo⽂件处理
(1)将官⽅的demo下载下来,⽂件名为WxpayAPI_php_v3,把这⽂件重命名为wxpay,为了后边书写⽬录⽅便;
(2)打开lib⽂件夹下的WxPay.Api.php⽂件,在537⾏有⼀段curl⽹络请求配置代码:
curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,TRUE);
curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,2);//严格校验
替换成:
curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,FALSE);
curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,2);//严格校验
为了禁⽌ cURL 验证对等证书(peer's certificate)。
(3)打开lib⽂件夹下的WxPay.Config.php⽂件,第25⾏开始,根据⾃⼰的账号完成基本信息设置;
const APPID = '公众账号APPID';
const MCHID = '商户号';
const KEY = '商户⽀付密钥';
const APPSECRET = '公众帐号secert';
(4)打开lib⽂件夹下的WxPay.Notify.php⽂件,第79⾏的代码:
if($needSign == true &&
$this->GetReturn_code($return_code) == "SUCCESS")
{
$this->SetSign();
}
改成:
if($needSign == true &&
$this->GetReturn_code() == "SUCCESS")
{
$this->SetSign();
}
(5)打开cert证书⽬录,将⾥边的两个证书换成⾃⼰的⽀付证书。
2.后台设置
(1)配置⽹页授权域名,我的域名是();
(1)配置⽀付授权⽬录,域名是(),我将demo放到此域名指向的⽬录的weixinopen/⽂件夹下,demo中jsapi.php⽂件位于example/⽬录下,所以⽀付授权⽬录为:/weixinopen/wxpay/example/
3.⽀付流程
打开example⽬录下的jsapi.php⽂件,⽀付发起和处理,都是在这⾥完成。
(1)获取⽤户openid
之前配置好了⾃⼰的APPID和APPSecert,所以这⾥不⽤处理。
//①、获取⽤户openid
$tools = new JsApiPay();
$openId = $tools->GetOpenid();
这⾥⾸先初始化的⼀个JsApiPay()类得到⼀个对象,⽂件对应example/⽬录下的WxPay.JsApiPay.php,调⽤GetOpenid()⽅法,会⾃动获取⾃⼰的openID。
(2)统⼀下单
//②、统⼀下单
$input = new WxPayUnifiedOrder();
$input->SetBody("test");
$input->SetAttach("test");
$input->SetOut_trade_no(WxPayConfig::MCHID.date("YmdHis"));
$input->SetTotal_fee("1");
$input->SetTime_start(date("YmdHis"));
$input->SetTime_expire(date("YmdHis", time() + 600));
$input->SetGoods_tag("test");
$input->SetNotify_url("paysdk.weixin.qq/example/notify.php");
$input->SetTrade_type("JSAPI");
$input->SetOpenid($openId);
$order = WxPayApi::unifiedOrder($input);
echo '<font color="#f00"><b>统⼀下单⽀付单信息</b></font><br/>';
printf_info($order);
$jsApiParameters = $tools->GetJsApiParameters($order);
对应WxPay.Api.php的第24⾏的unifiedOrder()⽅法,配置订单信息和⽀付回调函数,这⾥需要修改⼏个参数:
A. 商品名称:
$input->SetBody("test");
B. 订单号
$input->SetOut_trade_no(WxPayConfig::MCHID.date("YmdHis"));
C. ⽀付⾦额
$input->SetTotal_fee("1");
D. ⽀付验证链接
也可以写其他地址,当然要在⽀付授权域名之下,⽀付成功之后就会⾃动回调到该链接指定的⽅法⾥边,可以在⾥边进⾏判断和数据库操作.
$input->SetNotify_url(paysdk.weixin.qq/example/notify.php);
E. 附加参数
$input->SetAttach("test");
附加参数,可填可不填,填写的话,⾥边字符串最好不要出现空格。
这时候,点击⽀付应该就可以成功⽀付了。
(3)发起⽀付
<script type="text/javascript">
//调⽤JS api ⽀付
function jsApiCall()
{
WeixinJSBridge.invoke(
'getBrandWCPayRequest',
<?php echo $jsApiParameters; ?>,
function(res){
WeixinJSBridge._msg);
___msg);
}
);
}
function callpay()
{
if (typeof WeixinJSBridge == "undefined"){
if( document.addEventListener ){
document.addEventListener('WeixinJSBridgeReady', jsApiCall, false);
}else if (document.attachEvent){
document.attachEvent('WeixinJSBridgeReady', jsApiCall);
document.attachEvent('onWeixinJSBridgeReady', jsApiCall);
}
}else{
支付分怎么开通jsApiCall();
}
}
</script>
点击⽴即⽀付按钮调⽤的就是 callpay() 函数,他有会调⽤jsApiCall() 函数打开⽀付程序。
jsApiCall() 函数会监听每⼀步动作:
<_msg 为get_brand_wcpay_request:cancel 表明前端判断的取消⽀付,es.err_msg 为get_brand_wcpay_request:ok 表明前端判断的⽀付成功,我们可以根据这个将⽀付跳转到成功页⾯。
(4)⽀持成功回调
通过前端jsApiCall()函数可以监听⽀付结果,但是这个并不可信。确认是否⽀付成功还是应当通过notify.php 处理业务逻辑。前边配置好了⽀付验证链接SetNotify_url(),⽀付完成后,服务器会根据链接⾃动请求你的notify.php⽂件,打开这个⽂件,其实这个⽂件最主要的代码就两⾏:
$notify = new PayNotifyCallBack();
$notify->Handle(false);
由此跟踪到WxPay.Notify.php类⽂件的Handle()函数:
/**
*
* 回调⼊⼝
* @param bool $needSign 是否需要签名输出
*/
final public function Handle($needSign = true)
{
$msg = "OK";
//当返回false的时候,表⽰notify中调⽤NotifyCallBack回调失败获取签名校验失败,此时直接回复失败
$result = WxpayApi::notify(array($this, 'NotifyCallBack'), $msg);
if($result == false){
$this->SetReturn_code("FAIL");
$this->SetReturn_msg($msg);
$this->ReplyNotify(false);
return;
} else {
//该分⽀在成功回调到NotifyCallBack⽅法,处理完成之后流程
$this->SetReturn_code("SUCCESS");
$this->SetReturn_msg("OK");
}
$this->ReplyNotify($needSign);
}
主要代码:
$result = WxpayApi::notify(array($this, 'NotifyCallBack'), $msg);
然后来到WxPay.Api.php⽂件的第411⾏,notify()函数:
/**
*