微信公众号jsapi支付php源码分析
微信公众号支付,首先需要通过授权跳转地址里获取code,并进一步向微信获取openid,然后拉起统一支付获取prepay_id,然后再等待用户按下支付,调起支付。支付部分在前端,很多初次使用微信公众号支付的人人对获取code和opendi部分不懂,微信的php,java等demo源代码地址如下
https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=11_1。
微信支付的演示地址(在微信内复制黏贴后点击打开)http://paysdk.weixin.qq.com/
选择jsapi后,会进入https://open.weixin.qq.com/connect/oauth2/authorize?appid=wx426b3015555a46be&redirect_uri=http%3A%2F%2Fpaysdk.weixin.qq.com%2Fexample%2Fjsapi.php&response_type=code&scope=snsapi_base&state=STATE#wechat_redirect
我们发现微信自身提供的跳转地址是http://paysdk.weixin.qq.com/example/jsapi.php这个地址,这样你可以在上面微信的源代码里的example下找到jsapi.php这个页面,下面就是我对这个Php页面和相关调用的一个梳理和分析,值分析获取opendi到得到prepay_id的过程,支付在最下面一段
<?php
ini_set(‘date.timezone‘,‘Asia/Shanghai‘);
//error_reporting(E_ERROR);
require_once "../lib/WxPay.Api.php";
require_once "WxPay.JsApiPay.php";
require_once ‘log.php‘;
//初始化日志
$logHandler= new CLogFileHandler("../logs/".date(‘Y-m-d‘).‘.log‘);
$log = Log::Init($logHandler, 15);
//打印输出数组信息
function printf_info($data)
{
foreach($data as $key=>$value){
echo "<font color=‘#00ff55;‘>$key</font> : $value <br/>";
}
}
//①、获取用户openid
//example/JsApiPay.php里有这个JsApiPay这个类,里面含有获取opendi的函数GetOpenid,
//这个函数首先获取微信传入的code,进一步利用getOpenidFromMp($code)函数向微信服务器获取openid
$tools = new JsApiPay();
$openId = $tools->GetOpenid();
//②、统一下单
$input = new WxPayUnifiedOrder(); //定义在lib Wepay.Data.php里
$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("http://paysdk.weixin.qq.com/example/notify.php");
$input->SetTrade_type("JSAPI");
$input->SetOpenid($openId);
//上面将需要发送到微信服务器进行统一支付的body, attach, trade_no,fee,通知地址等填写在定义在lib下面的Wepay.Data.php里的WxPayUnifiedOrder类的实例$input里,
//然后定义在lib WxPay.Api.php里的WxPayApi类里有个一个unifiedOrder函数,首先判断是否填//写以上各项数据,没有爆出异常,然后在下面的
$order = WxPayApi::unifiedOrder($input);里进一步填写关键的公众号APPID,商户号MCHID,KEY ,APPSECRET(这些作为静态数据放在lib WxPay.Config.php里) ,用户的ip要前台传递给后台,随机字符串和签名在该函数里调用了相应函数进行了获取。
//然后通过该函数里又调用了两个函数将数据封装成xml发送到微信服务器,并将微信响应的含有prepay_id的结果返回,在这里就赋值给了$order,这时的$order是统一下单后的微信返回结果 下面是unifiedOrder函数最后几行
/*
参见 lib WxPay.Api.php 函数 unifiedOrder函数
$xml = $inputObj->ToXml();//将上面$input的统一支付各个数据封装成xml,发送给微信,获取Prepaid_id等
$response = self::postXmlCurl($xml, $url, false, $timeOut);
//发起统一下单,向微信提供的统一下单地址发送填写了各种参数的xml,并获得响应结果
$result = WxPayResults::Init($response); return result;
*/
$order = WxPayApi::unifiedOrder($input);
//这时候$order含有了prepaid_id等微信返回数据
echo ‘<font color="#f00"><b>统一下单支付单信息</b></font><br/>‘;
printf_info($order);
//GetJsApiParameters抽取各项返回数据,封装成json,用于在填充下面的发起对微信支付invoke里所需要传送的数据
$jsApiParameters = $tools->GetJsApiParameters($order);
//获取共享收货地址js函数参数
$editAddress = $tools->GetEditAddressParameters();
//③、在支持成功回调通知中处理成功之后的事宜,见 notify.php
/**
* 注意:
* 1、当你的回调地址不可访问的时候,回调通知会失败,可以通过查询订单来确认支付是否成功
* 2、jsapi支付时需要填入用户openid,WxPay.JsApiPay.php中有获取openid流程 (文档可以参考微信公众平台“网页授权接口”,
* 参考http://mp.weixin.qq.com/wiki/17/c0f37d5704f0b64713d5d2c37b468d75.html)
*/
?>
<html>
<head>
<meta http-equiv="content-type" content="text/html;charset=utf-8"/>
<meta name="viewport" content="width=device-width, initial-scale=1"/>
<title>微信支付样例-支付</title>
<script type="text/javascript">
//调用微信JS api 支付
function jsApiCall()
{
WeixinJSBridge.invoke(
‘getBrandWCPayRequest‘,
//这里将从统一支付获取的参数填充到要发送到微信的部分
<?php echo $jsApiParameters; ?>,
function(res){
WeixinJSBridge.log(res.err_msg);
alert(res.err_code+res.err_desc+res.err_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();
}
}
//获取共享地址
function editAddress()
{
WeixinJSBridge.invoke(
‘editAddress‘,
<?php echo $editAddress; ?>,
function(res){
var value1 = res.proviceFirstStageName;
var value2 = res.addressCitySecondStageName;
var value3 = res.addressCountiesThirdStageName;
var value4 = res.addressDetailInfo;
var tel = res.telNumber;
alert(value1 + value2 + value3 + value4 + ":" + tel);
}
);
}
window.onload = function(){
if (typeof WeixinJSBridge == "undefined"){
if( document.addEventListener ){
document.addEventListener(‘WeixinJSBridgeReady‘, editAddress, false);
}else if (document.attachEvent){
document.attachEvent(‘WeixinJSBridgeReady‘, editAddress);
document.attachEvent(‘onWeixinJSBridgeReady‘, editAddress);
}
}else{
editAddress();
}
};
</script>
</head>
<body>
<br/>
<font color="#9ACD32"><b>该笔订单支付金额为<span style="color:#f00;font-size:50px">1分</span>钱</b></font><br/><br/>
<div align="center">
<button style="width:210px; height:50px; border-radius: 15px; border:0px #FE6714 solid; cursor: pointer; color:white; font-size:16px;" type="button" onclick="callpay()" >立即支付</button>
</div>
</body>
</html>