PHP:微信支付服务商开发相关的那点事
项目背景
不是什么大项目,使用微信服务商来管理多个子商户,并使用服务商的接口替子商户下单,服务商后台才能接收到回调
使用场景是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教程栏目进行学习!
Atas ialah kandungan terperinci PHP:微信支付服务商开发相关的那点事. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Alat AI Hot

Undresser.AI Undress
Apl berkuasa AI untuk mencipta foto bogel yang realistik

AI Clothes Remover
Alat AI dalam talian untuk mengeluarkan pakaian daripada foto.

Undress AI Tool
Gambar buka pakaian secara percuma

Clothoff.io
Penyingkiran pakaian AI

AI Hentai Generator
Menjana ai hentai secara percuma.

Artikel Panas

Alat panas

Notepad++7.3.1
Editor kod yang mudah digunakan dan percuma

SublimeText3 versi Cina
Versi Cina, sangat mudah digunakan

Hantar Studio 13.0.1
Persekitaran pembangunan bersepadu PHP yang berkuasa

Dreamweaver CS6
Alat pembangunan web visual

SublimeText3 versi Mac
Perisian penyuntingan kod peringkat Tuhan (SublimeText3)

Topik panas



PHP 8.4 membawa beberapa ciri baharu, peningkatan keselamatan dan peningkatan prestasi dengan jumlah penamatan dan penyingkiran ciri yang sihat. Panduan ini menerangkan cara memasang PHP 8.4 atau naik taraf kepada PHP 8.4 pada Ubuntu, Debian, atau terbitan mereka

Kod Visual Studio, juga dikenali sebagai Kod VS, ialah editor kod sumber percuma — atau persekitaran pembangunan bersepadu (IDE) — tersedia untuk semua sistem pengendalian utama. Dengan koleksi sambungan yang besar untuk banyak bahasa pengaturcaraan, Kod VS boleh menjadi c

Jika anda seorang pembangun PHP yang berpengalaman, anda mungkin merasakan bahawa anda telah berada di sana dan telah melakukannya. Anda telah membangunkan sejumlah besar aplikasi, menyahpenyahpepijat berjuta-juta baris kod dan mengubah suai sekumpulan skrip untuk mencapai op

Tutorial ini menunjukkan cara memproses dokumen XML dengan cekap menggunakan PHP. XML (bahasa markup extensible) adalah bahasa markup berasaskan teks yang serba boleh yang direka untuk pembacaan manusia dan parsing mesin. Ia biasanya digunakan untuk penyimpanan data

JWT adalah standard terbuka berdasarkan JSON, yang digunakan untuk menghantar maklumat secara selamat antara pihak, terutamanya untuk pengesahan identiti dan pertukaran maklumat. 1. JWT terdiri daripada tiga bahagian: header, muatan dan tandatangan. 2. Prinsip kerja JWT termasuk tiga langkah: menjana JWT, mengesahkan JWT dan muatan parsing. 3. Apabila menggunakan JWT untuk pengesahan di PHP, JWT boleh dijana dan disahkan, dan peranan pengguna dan maklumat kebenaran boleh dimasukkan dalam penggunaan lanjutan. 4. Kesilapan umum termasuk kegagalan pengesahan tandatangan, tamat tempoh, dan muatan besar. Kemahiran penyahpepijatan termasuk menggunakan alat debugging dan pembalakan. 5. Pengoptimuman prestasi dan amalan terbaik termasuk menggunakan algoritma tandatangan yang sesuai, menetapkan tempoh kesahihan dengan munasabah,

Rentetan adalah urutan aksara, termasuk huruf, nombor, dan simbol. Tutorial ini akan mempelajari cara mengira bilangan vokal dalam rentetan yang diberikan dalam PHP menggunakan kaedah yang berbeza. Vokal dalam bahasa Inggeris adalah a, e, i, o, u, dan mereka boleh menjadi huruf besar atau huruf kecil. Apa itu vokal? Vokal adalah watak abjad yang mewakili sebutan tertentu. Terdapat lima vokal dalam bahasa Inggeris, termasuk huruf besar dan huruf kecil: a, e, i, o, u Contoh 1 Input: String = "TutorialSpoint" Output: 6 menjelaskan Vokal dalam rentetan "TutorialSpoint" adalah u, o, i, a, o, i. Terdapat 6 yuan sebanyak 6

Mengikat statik (statik: :) Melaksanakan pengikatan statik lewat (LSB) dalam PHP, yang membolehkan kelas panggilan dirujuk dalam konteks statik dan bukannya menentukan kelas. 1) Proses parsing dilakukan pada masa runtime, 2) Cari kelas panggilan dalam hubungan warisan, 3) ia boleh membawa overhead prestasi.

Apakah kaedah sihir PHP? Kaedah sihir PHP termasuk: 1. \ _ \ _ Membina, digunakan untuk memulakan objek; 2. \ _ \ _ Destruct, digunakan untuk membersihkan sumber; 3. \ _ \ _ Call, mengendalikan panggilan kaedah yang tidak wujud; 4. \ _ \ _ Mendapatkan, melaksanakan akses atribut dinamik; 5. \ _ \ _ Set, melaksanakan tetapan atribut dinamik. Kaedah ini secara automatik dipanggil dalam situasi tertentu, meningkatkan fleksibiliti dan kecekapan kod.
