在微信支付 開發者文檔頁面 下載最新的 php SDKjavascript
http://mch.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=11_1php
這裏假設你已經申請完微信支付html
1. 微信後臺配置 如圖
前端
咱們先進行測試,因此先把測試受權目錄和 測試白名單添加上。測試受權目錄是你要發起微信請求的哪一個文件所在的目錄。java
例如jsapi 發起請求通常是jsapi.php所在目錄 爲測試目錄,測試白名單即開發人員的微信號。api
正式的支付受權目錄不能和測試的同樣不然會報錯。不填寫或者填錯受權目錄以及測試白名單都會報錯。服務器
報錯樣例:微信
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 style="<a href="<?php echo 'http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/jsapi.php';?>">JSAPI支付</a></li> <li style="<a href="<?php echo 'http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/micropay.php';?>">刷卡支付</a></li> <li style="<a href="<?php echo 'http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/native.php';?>">掃碼支付</a></li> <li style="<a href="<?php echo 'http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/orderquery.php';?>">訂單查詢</a></li> <li style="<a href="<?php echo 'http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/refund.php';?>">訂單退款</a></li> <li style="<a href="<?php echo 'http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/refundquery.php';?>">退款查詢</a></li> <li style="<a href="<?php echo 'http://'.$_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('http://'.$_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
參考連接:http://mp.weixin.qq.com/wiki/17/c0f37d5704f0b64713d5d2c37b468d75.html
這就須要你把微信的 網頁受權接口也設置好。
獲取到 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("http://paysdk.weixin.qq.com/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("http://paysdk.weixin.qq.com/example/notify.php");
是設置接收支付結果通知的Url 這裏是默認的demo 連接咱們能夠設置成咱們的:
$input->SetNotify_url(dirname('http://'.$_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.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(); } }
其中點擊當即支付按鈕調用的就是 callpay() 函數,他有會調用jsApiCall() 函數打開支付程序。
此後輸入密碼完成支付。
在完成支付頁面點擊完成會回到這個支付頁面,並彈出 支付成功的提示框
這個其實就是 js函數 jsApiCall 裏面的alter 彈出的對話框
其中 res.err_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支付就大體分析完成了。
http://www.mamicode.com/info-detail-587828.html