⽀付PHPSDK之⽀付代码详解这⾥假设你已经申请完⽀付
1. 后台配置如图
我们先进⾏测试,所以先把测试授权⽬录和测试⽩名单添加上。测试授权⽬录是你要发起请求的哪个⽂件所在的⽬录。
例如jsapi 发起请求⼀般是jsapi.php所在⽬录为测试⽬录,测试⽩名单即开发⼈员的号。
正式的⽀付授权⽬录不能和测试的⼀样否则会报错。不填写或者填错授权⽬录以及测试⽩名单都会报错。
报错样例:
NaNsystem:access_denied
不在测试⽩名单
2. 配置 lib/WxPay.Config.php⽂件
最主要配置⼀下四项:
const APPID = '';
const MCHID = '';
const KEY = '';
const APPSECRET = '';
APPID 和 APPSECRET都可以在后台中到。
MCHID 在申请⽀付后发来的邮件中可以到,KEY 则根据邮件提⽰
去商户平台配置即可。
3. 访问起始 index.php
⾸先访问 index.php 你可以看到界⾯
我们⾸先需要的是 JSAPI⽀付。但是看代码 index.php 最下⾯的链接。他默认是个demo的链接,改为我们⾃定义的即可
<ul>
<li ><a href="<?php echo ''.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/jsapi.php';?>">JSAPI⽀付</a></li>
<li ><a href="<?php echo ''.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/micropay.php';?>">刷卡⽀付</a></li> <li ><a href="<?php echo ''.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/native.php';?>">扫码⽀付</a></li>
<li ><a href="<?php echo ''.$_SERVER['HTTP_HOST'].$_
SERVER['REQUEST_URI'].'example/orderquery.php';?>">订单查询</a></li> <li ><a href="<?php echo ''.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/refund.php';?>">订单退款</a></li>
<li ><a href="<?php echo ''.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/refundquery.php';?>">退款查询</a></li> <li ><a href="<?php echo ''.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/download.php';?>">下载订单</a></li> </ul>
当然你也可以直接写死为⾃⼰的访问链接。
4. JSAPI ⽀付
必要代码解析:
$logHandler= new CLogFileHandler("../logs/".date('Y-m-d').'.log');
$log = Log::Init($logHandler, 15);
调⽤⽇志类可以通过 $log->DEBUG(‘test‘); 打印调试信息。其实也可以直接使⽤ $Log::DEBUG(‘test‘); 来调试
$tools = new JsApiPay();
$openId = $tools->GetOpenid();
主要是为了获取 openid 其中GetOpenid() 函数定义在⽂件 WxPay.JsApiPay.php ⽂件中
public function GetOpenid()
{
//通过code获得openid
if (!isset($_GET['code'])){
//触发返回code码
$baseUrl = urlencode(''.$_SERVER['HTTP_HOST'].$_SERVER['PHP_SELF'].$_SERVER['QUERY_STRING']);
$url = $this->__CreateOauthUrlForCode($baseUrl);
Header("Location: $url");
exit();
} else {
//获取code码,以获取openid
$code = $_GET['code'];
$openid = $this->getOpenidFromMp($code);
return $openid;
}
}
$baseUrl 其实就是为了在跳转回来这个页⾯。可以继续跟踪函数__CreateOauthUrlForCode() 其实就是通过的Auth2.0 来获取Openid
这就需要你把的⽹页授权接⼝也设置好。
获取到 Openid 就可以调⽤⽀付的统⼀下单接⼝了。回到⽂件 jsapi.php 如下代码
$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);
这⾥⾯的代码:
$input->SetAttach("test");
如果把值改为 $input->SetAttach("test this is attach");就会存在bug 后⾯再说,其实这个参数不是必须的⼲脆可以去掉。
代码:
$input->SetNotify_url(paysdk.weixin.qq/example/notify.php);
是设置接收⽀付结果通知的Url 这⾥是默认的demo 链接我们可以设置成我们的:
$input->SetNotify_url(dirname(''.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI']).'/notify.php');
当然你也可以选择直接写死。
其中的函数 unifiedOrder($input) 可以到WxPay.Api.php 中⽂件跟踪,其实就是调⽤统⼀下单接⼝。
在 WxPay.Api.php 中需要更改的⼀处代码是:
//异步通知url未设置,则使⽤配置⽂件中的url
if(!$inputObj->IsNotify_urlSet()){
$inputObj->SetNotify_url(WxPayConfig::NOTIFY_URL);//异步通知url
}
就是当没设置 notifyUrl 的时候回去配置⽂件中,但是配置⽂件中根本没有设置。
所以你可以选择在配置⽂件WxPay.Config.php 中加上这个配置,也可以直接写⼀个默认的notify链接。
函数 GetJsApiParameters() 是获取jsApi⽀付的参数给变量 $jsApiParameters ⽅便在下⾯的Js中调⽤
jsapi.php 中js的代码:
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();
}
}
其中点击⽴即⽀付按钮调⽤的就是 callpay() 函数,他有会调⽤jsApiCall() 函数打开⽀付程序。
此后输⼊密码完成⽀付。
在完成⽀付页⾯点击完成会回到这个⽀付页⾯,并弹出⽀付成功的提⽰框
这个其实就是 js函数 jsApiCall ⾥⾯的alter 弹出的对话框
其中 _msg 为get_brand_wcpay_request:ok 表明前端判断的⽀付成功,我们可以根据这个将⽀付跳转到成功页⾯。
但是这个并不可信。确认是否⽀付成功还是应当通过notify.php 处理业务逻辑。
5. ⽀付结果通知 notify.php
其实这个页⾯最主要的代码就两⾏
$notify = new PayNotifyCallBack();
$notify->Handle(false);
其中⼤部分逻辑在 Handle 函数中处理⽂件 WxPay.Notify.php
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);
跟踪函数 notify ⽂件WxPay.Api.php
public static function notify($callback, &$msg)
{
//获取通知的数据
$xml = $GLOBALS['HTTP_RAW_POST_DATA'];
//如果返回成功则验证签名
try {
$result = WxPayResults::Init($xml);
} catch (WxPayException $e){
$msg = $e->errorMessage();
return false;
}
return call_user_func($callback, $result);
}
通过 $GLOBALS[‘HTTP_RAW_POST_DATA‘]; 获取同志数据然后 Init 函数验证签名等。验签成功运⾏代码
return call_user_func($callback, $result);
即调⽤了⼀个回调函数,NotifyCallBack() 函数并传递参数 $result 在NotifyCallBack函数中会调⽤我们重写的NotifyProcess()函数(此函数在notify.php 中被重写)NotifyProcess() 判断也没有问题就会设置返回 success的xml信息
$this->SetReturn_code("SUCCESS");
$this->SetReturn_msg("OK");
并最终调⽤函数 $this->ReplyNotify($needSign); echo success的结果
函数ReplyNotify 需要修改⼀处代码:
final private function ReplyNotify($needSign = true)
{
//如果需要签名
if($needSign == true &&
$this->GetReturn_code($return_code) == "SUCCESS")
{
$this->SetSign();
}
WxpayApi::replyNotify($this->ToXml());
}
$this->GetReturn_code($return_code) == "SUCCESS")
改为
$this->GetReturn_code() == "SUCCESS")
即可。
这样整个流程就结束了。上⾯提到了传递订单参数
$input->SetAttach("test");
如果我设置值为 test this is attach (其实只要有空格就会存在bug)
如图传递的订单信息
可以看到 attach 信息正常,当然⽀付也是正常的没有任何问题。
但是发现总是会收到notify 通知,即意味着没有返回给服务器正确的结果通知。打印服务器发来的通知数据
可以看到 attach 是 test+this+is+attach 即空格被转化为加号
打印接收到的签名和程序算出来的签名发现签名不同,即认为接收结果异常。
所以我们要是想使⽤attach 这个值就不能有空格,要么⼲脆不使⽤这个参数
(等待修复这个bug, 也可能是我这边有哪个地⽅不会? - -#)
这样⽀付的 JsApi⽀付就⼤致分析完成了。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论