微信支付PHP SDK之微信公眾號支付代碼詳解_php執行個體

來源:互聯網
上載者:User

這裡假設你已經申請完微信支付

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 style="background-color:#FF7F24"><a href="<?php echo 'http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/jsapi.php';?>">JSAPI支付</a></li>  <li style="background-color:#698B22"><a href="<?php echo 'http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/micropay.php';?>">刷卡支付</a></li>  <li style="background-color:#8B6914"><a href="<?php echo 'http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/native.php';?>">掃碼支付</a></li>  <li style="background-color:#CDCD00"><a href="<?php echo 'http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/orderquery.php';?>">訂單查詢</a></li>  <li style="background-color:#CD3278"><a href="<?php echo 'http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/refund.php';?>">訂單退款</a></li>  <li style="background-color:#848484"><a href="<?php echo 'http://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'example/refundquery.php';?>">退款查詢</a></li>  <li style="background-color:#8EE5EE"><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支付就大致分析完成了。

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.