PHP:微信支付服務商開發相關的那點事

2020-07-16 10:05:40

專案背景

不是什麼大專案,使用微信服務商來管理多個子商戶,並使用服務商的介面替子商戶下單,服務商後台才能接收到回撥

使用場景是web掃碼支付

準備

域名應該要在服務商所歸屬的公眾號內「網頁授權域名」設定好(不知此操作是否需要?)

在支付服務商後台設定好回撥地址(子商戶應該是不用設定)

專案使用apache+php為後台服務,下載官方支付php demo(native)

我們直接先按照demo的目錄結構來玩,直接把解壓的example和lib,2個目錄都到伺服器根目錄

在example目錄下,建立cert目錄,進服務商後台-賬戶中心-api安全,下載證書,放到這個目錄內

在example目錄下,建立logs目錄,用於微信支付log類寫紀錄檔檔案

由於微信支付相關都要使用https,所以檢視存取紀錄檔在apache目錄下的logs目錄,ssl_request.txt檔案,最下面,可以看到回撥地址是否被請求

注意

官方demo有2個方式的掃碼支付,第一種方式已經不提供,都使用第二種

官方的demo,會有不能顯示二維條碼的bug,例子頁面是native.php

列印print_r($result); 這個,會顯示錯誤,主要是關於於curl的錯誤,自行百度解決

設定

在WxPay.Config.Interface.php中的介面物件中增加一個公共方法public abstract function GetSubMchId(); //獲取子商戶id在WxPay.Config.php內中,設定需要的引數,自行百度,並且增加一個方法public function GetSubMchId(){ return '8888888888'; //返回子商戶號 by vbyzc }在lib/WxPay.Api.php 內,在統一下單方法unifiedOrder中,下面的引數那段位置,增加$inputObj->SetSub_mch_id($config->GetSubMchId());//子商戶號 by vbyzc在各個需要查詢訂單的的地方回撥,付款頁面實時檢測訂單支付狀態的請求頁面,都要使用此方法來設定子商戶id:
$input->SetSub_mch_id($config->GetSubMchId());注意,有的地方可能沒有$config物件,請引入WxPay.Config.php ,並初始化:$config = new WxPayConfig();

部分程式碼

掃碼頁面:native.php

<?php
/**
*
* example目錄下為簡單的支付樣例,僅能用於搭建快速體驗微信支付使用
* 樣例的作用僅限於指導如何使用sdk,在安全上面僅做了簡單處理, 複製使用樣例程式碼時請慎重
* 請勿直接直接使用樣例對外提供服務
* 
**/

require_once "../lib/WxPay.Api.php";
require_once "WxPay.NativePay.php";
require_once 'log.php';

//初始化紀錄檔
$logHandler= new CLogFileHandler("logs/".date('Y-m-d').'.log');
$log = Log::Init($logHandler, 15);

//模式一
//官方不再提供模式一支付方式

$notify = new NativePay();

//模式二
/**
 * 流程:
 * 1、呼叫統一下單,取得code_url,生成二維條碼
 * 2、使用者掃描二維條碼,進行支付
 * 3、支付完成之後,微信伺服器會通知支付成功
 * 4、在支付成功通知中需要查單確認是否真正支付成功(見:notify.php)
 */

$out_trade_no = "vbyzc_for_jstx".date("YmdHis"); 

$input = new WxPayUnifiedOrder();
$input->SetBody("test_body");
$input->SetAttach("test_Attach");//成功支付的回撥裡會返回這個
$input->SetOut_trade_no($out_trade_no);//自定義訂單號
$input->SetTotal_fee("1"); // 金額
$input->SetTime_start(date("YmdHis"));
// $input->SetTime_expire(date("YmdHis", time() + 500));
$input->SetGoods_tag("test_goodsTag");
$input->SetNotify_url("https://service.ktfqs.com/example/wx_pay_callback.php");
$input->SetTrade_type("NATIVE");
$input->SetProduct_id("123456789"); //此id為二維條碼中包含的商品ID,商戶自行定義。

$result = $notify->GetPayUrl($input);
$url2 = $result["code_url"];

echo "<div>這是返回:$url2</div>";
print_r($result);
?>

<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 src="https://cdn.bootcss.com/jquery/1.12.4/jquery.min.js"></script>
</head>
<body>

    <div style="margin-left: 10px;color:#556B2F;font-size:30px;font-weight: bolder;">掃描支付模式二</div><br/>
    <div> 訂單編號<input id="out_trade_no" type="hidden"  value="<?php echo $out_trade_no;?>"> </div>
    <img alt="模式二掃碼支付" src="qrcode.php?data=<?php echo urlencode($url2);?>" style="width:150px;height:150px;"/>
    <div>支付提示:<span id="query_result" style="color: red">WAITING...</span></div>
    <script>
        var t1;
        var sum=0;
        $(document).ready(function () {
            t1=setInterval("ajaxstatus()", 4000);
        });
        function ajaxstatus() {
            sum++;
            if(sum>100){ window.clearInterval(t1);return false;}
            if ($("#out_trade_no").val() != 0) {
                $.post("orderqueryajax.php", { out_trade_no:$("#out_trade_no").val() }, function (data) {
                    data = $.trim(data);
                    $("#query_result").html(data);
                    if (data=="SUCCESS") {
                        $("#query_result").html("哈哈哈!!支付成功,即將跳轉...");
                        window.clearInterval(t1)
                        <?php
                            // 插入php程式碼
                            /*
                            if (isset($_POST['history_go']) && $_POST['history_go'] == 3){
                                echo 'window.setTimeout("history.go(-3);",2000);';
                            }else{
                                echo 'window.setTimeout("history.go(-2);",2000);';
                            }
                            */
                        ?>
                    }
                });
            }
        }
    </script>
</body>
</html>

查詢並返回訂單狀態頁面:orderqueryajax.php

<?php
/**
*
* ajax非同步查詢訂單是否完成
* 
**/
require_once "../lib/WxPay.Api.php";
require_once 'log.php';
require_once "WxPay.Config.php";

//初始化紀錄檔
$logHandler= new CLogFileHandler("../logs/".date('Y-m-d').'.log');
$log = Log::Init($logHandler, 15);

$v = $_POST["out_trade_no"];
if(isset($v) && $v != ""){
    $out_trade_no = $v;
    $config = new WxPayConfig();
    $input = new WxPayOrderQuery();
    $input->SetOut_trade_no($out_trade_no);
    $input->SetSub_mch_id($config->GetSubMchId());//子商戶號 by vbyzc
    $result = WxPayApi::orderQuery($config, $input);
    if ($result['return_code'] == 'SUCCESS' && $result['result_code'] == 'SUCCESS'){//返回查詢結果
        echo $result['trade_state'];
    }else{
        echo "FAIL";
    }
}
?>

回撥頁:notify.php

<?php
date_default_timezone_set('PRC');
/**
*
* example目錄下為簡單的支付樣例,僅能用於搭建快速體驗微信支付使用
* 樣例的作用僅限於指導如何使用sdk,在安全上面僅做了簡單處理, 複製使用樣例程式碼時請慎重
* 請勿直接直接使用樣例對外提供服務
* 
**/
// 連結資料庫
include_once('../include/conn_db.php');
include_once('../include/db_class.php');
mysql_connect(HOST,NAME,PASS) or die(mysql_error());
mysql_select_db(DBNAME);
mysql_query('SET NAMES '.CODEPAGE);

require_once "../lib/WxPay.Api.php";
require_once '../lib/WxPay.Notify.php';
require_once "WxPay.Config.php";
require_once 'log.php';

//初始化紀錄檔
$logHandler= new CLogFileHandler("logs/".date('Y-m-d').'.log');
$log = Log::Init($logHandler, 15);

class PayNotifyCallBack extends WxPayNotify
{
    //查詢訂單
    public function Queryorder($transaction_id)
    {
        $input = new WxPayOrderQuery();
        $config = new WxPayConfig();
        $input->SetTransaction_id($transaction_id);
        $input->SetSub_mch_id($config->GetSubMchId()); //設定子商戶號  by vbyzc
        $result = WxPayApi::orderQuery($config, $input);
        Log::DEBUG("query:" . json_encode($result));
        if(array_key_exists("return_code", $result)
            && array_key_exists("result_code", $result)
            && $result["return_code"] == "SUCCESS"
            && $result["result_code"] == "SUCCESS")
        {
            return true;
        }
        return false;
    }

    /**
    *
    * 回包前的回撥方法
    * 業務可以繼承該方法,列印紀錄檔方便定位
    * @param string $xmlData 返回的xml引數
    *
    **/
    public function LogAfterProcess($xmlData)
    {
        Log::DEBUG("call back, return xml:" . $xmlData);
        return;
    }
    
    //重寫回撥處理常式
    /**
     * @param WxPayNotifyResults $data 回撥解釋出的引數
     * @param WxPayConfigInterface $config
     * @param string $msg 如果回撥處理失敗,可以將錯誤資訊輸出到該方法
     * @return true回撥出來完成不需要繼續回撥,false回撥處理未完成需要繼續回撥
     */
    public function NotifyProcess($objData, $config, &$msg)
    {
        $data = $objData->GetValues();
        //TODO 1、進行引數校驗
        if(!array_key_exists("return_code", $data) 
            ||(array_key_exists("return_code", $data) && $data['return_code'] != "SUCCESS")) {
            //TODO失敗,不是支付成功的通知
            //如果有需要可以做失敗時候的一些清理處理,並且做一些監控
            $msg = "異常異常";
            return false;
        }
        if(!array_key_exists("transaction_id", $data)){
            $msg = "輸入引數不正確";
            return false;
        }

        //TODO 2、進行簽名驗證
        try {
            $checkResult = $objData->CheckSign($config);
            if($checkResult == false){
                //簽名錯誤
                Log::ERROR("簽名錯誤...");
                return false;
            }
        } catch(Exception $e) {
            Log::ERROR(json_encode($e));
        }

        //TODO 3、處理業務邏輯
        Log::DEBUG("call back JSON:" . json_encode($data));
        $notfiyOutput = array();
        /* 返回的格式 
        {
            "appid": "wxa664cef2fee1b641", //呼叫介面提交的公眾賬號ID
            "attach": "test",//附加資料,在查詢API和支付通知中原樣返回,該欄位主要用於商戶攜帶訂單的自定義資料 (使用SetAttach設定的)
            "bank_type": "LQT",//不知什麼鬼東西
            "cash_fee": "1",// 金額
            "fee_type": "CNY",//貨幣型別
            "is_subscribe": "N",//不知什麼鬼東西
            "mch_id": "154133502151",// 商戶號(服務商)
            "nonce_str": "jw0bvddz275qyvxnpdfoaam55h3dw6uk",//微信返回的隨機字串
            "openid": "opnVE5pDPx2hWAoLLxyQW5KQt8GA",// 使用者openid(應該是對於系結的公從號)
            "out_trade_no": "vbyzc_for_jstx20190701010509",// 發起訂單時自定義訂單號
            "result_code": "SUCCESS",// 業務結果
            "return_code": "SUCCESS",// 此欄位是通訊標識,非交易標識,交易是否成功需要檢視result_code來判斷
            "sign": "80E46C6CC50C25E6B5099AE4E03DA3C6FEFD5B172A99B03A56FAC4A9E11EC8F3",//
            "sub_mch_id": "154172463171",// 子商戶id
            "time_end": "20190701090530",// 交易結束時間??
            "total_fee": "1",// 總金額
            "trade_type": "NATIVE",// 支付方式
            "transaction_id": "4200000301201907011310094985" // 微信支付單號
        }
        */
        //查詢訂單,判斷訂單真實性
        if(!$this->Queryorder($data["transaction_id"])){
            $msg = "訂單查詢失敗";
            Log::DEBUG("vbyzc run to here : order querySelect faild!!!!!" );
            return false;
        }
        // 根據微信官方原始碼的業務流程,應該是如下:
        // 支會成功後微信會不斷請求回撥,在上面的程式碼 應該是包函了回撥回應的程式碼,
        // 如果成功回應,微信支付應該就停止請求回撥,才能執行下面的程式碼 
        Log::DEBUG("vbyzc run to here :<<<<<<<<<<<<<<start to mysql record" );

        $openid = $data['openid'];// 微信使用者
        $trade_no = $data['transaction_id'];// 微信支付單號
        $mch_id = $data['mch_id'];// 商戶號
        $sub_mch_id = $data['sub_mch_id'];// 子商戶id
        $trade_status = $data['result_code'];// 業務結果
        $total_amount = $data['total_fee'];// 總金額
        $out_trade_no = $data['out_trade_no'];// 商戶自定義訂單號

        $cmd = "insert into myorder(openid,trade_no,mch_id,sub_mch_id,trade_status,total_amount,out_trade_no,datetime) 
        values ('$openid','$trade_no','$mch_id','$sub_mch_id','$trade_status',$total_amount,'$out_trade_no',NOW())";
        mysql_query($cmd);
        Log::DEBUG("vbyzc run to here :end to mysql record>>>>>>>>>>" );
        return true;
    }
}

$config = new WxPayConfig();
Log::DEBUG("begin notify");
$notify = new PayNotifyCallBack();
$notify->Handle($config, false);


?>

更多PHP相關技術文章,請存取PHP教學欄目進行學習!

以上就是PHP:微信支付服務商開發相關的那點事的詳細內容,更多請關注TW511.COM其它相關文章!