zoukankan      html  css  js  c++  java
  • 微信支付服务商开发相关的那点事

    项目背景

    不是什么大项目,使用微信服务商来管理多个子商户,并使用服务商的接口替子商户下单,服务商后台才能接收到回调

    使用场景是web扫码支付

    准备

    • 域名应该要在服务商所归属的公众号内“网页授权域名”设置好(不知此操作是否需要?)
    •  在支付服务商后台设置好回调地址(子商户应该是不用设置)
    • 项目使用apache+php为后台服务,下载官方支付php demo(native)
    • 我们直接先按照demo的目录结构来玩,直接把解压的example和lib,2个目录都到服务器根目录
    • 在example目录下,创建cert目录,进服务商后台-账户中心-api安全,下载证书,放到这个目录内
    • 在example目录下(或根目录,老衲改了改代码中log的路径 $logHandler= new CLogFileHandler("logs/".date('Y-m-d').'.log');),创建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="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);
    
    
    ?>

    JSAPI方式配置

    • 首先在微信支付后台开发配置,设置JSAPI支付授权目录
    • 在公从号功能 设置里面,设置好一样的JS接口安全域名,但不知此步是否必须
    • 打开WxPay.JsApiPay.php文件,在方法GetOpenid里面,跳转地址改为:$baseUrl = urlencode('https://'.$_SERVER['HTTP_HOST'].$_SERVER['REQUEST_URI'].'?'.$_SERVER['QUERY_STRING']);其实就是改为https,和加了个问号,否则就会跳转404的错误
  • 相关阅读:
    DNS 壓力測試
    2008IT技术精英年会数据库分论坛热点扫描
    DOS command
    说说大型高并发高负载网站的系统架构
    DNS Server &Bind的配置与使用
    IoC 容器和Dependency Injection 模式
    Inversion of Control Containers and the Dependency Injection pattern
    Windows 2003网络负载均衡的实现
    UVA 10369 Arctic Network
    UVA 10397 Connect the Campus
  • 原文地址:https://www.cnblogs.com/vbyzc/p/11113364.html
Copyright © 2011-2022 走看看