この記事では主に ThinkPHP で WeChat 決済 (jsapi 決済) を実装する方法についての詳細なチュートリアルを紹介します。必要な方は参考にしてください。
WeChat 決済 (jsapi 決済) を実装するプロセスについては以前に記事を書きました。 PHP 詳細については、「PHP による WeChat 支払い (jsapi 支払い) プロセスの実装」を参照してください。
当時の環境はフレームワークを使用せず、ドメイン名で指定されたディレクトリの下に直接新規ディレクトリを作成し、そのディレクトリにアクセスするという方法で実装していましたが、フレームワークに適用するとまだいくつか問題がありました。 ThinkPHPでは、ルーティングルールと支払い承認ディレクトリにより齟齬が生じるため、エラーが報告されます。この記事では、WeChat 支払いを TP に統合するプロセスについて説明します。
Goose Factory によって作成された SDK とドキュメントは理解するのが難しく、使用するとどれだけ新鮮であるかがわかります。ドキュメントと SDK はできるだけシンプルで理解しやすいものであるべきではないでしょうか?精力的な再構築だけがGoose Factoryプログラマーの優れた技術を示すことができるのでしょうか?うーん…新人属性をさらけてしまったのか…実際、SDKは非常に使いやすいのですが、前回の記事で見たように、支払い完了のコールバック関数が非常にわかりにくいのです。
公式にバイパスされたくない、TP で WeChat 決済を使用したい方は、公式ドキュメントに基づいてマスターによって再構築され、効率化された TP に適した決済 SDK をご覧ください。ソースコードをダウンロードして見てみました。コードが書かれています。導入は非常にエレガントで、プロセスはシンプルで理解しやすいです。詳細については、ブログ投稿を参照してください: http://baijunyao.com/article/78
私はまだ眉をひそめながらも公式 SDK を使用し、支払いを正常に実装しました:
1.SDK のダウンロード変更
これについてはあまり詳しく説明しませんので、ご存じない方は、私の前回の記事をご覧ください: PHP は WeChat 支払い (jsapi 支払い) プロセスを実装しており、ダウンロードしたファイルを変更する必要があることが詳しく説明されています。
2. 公式アカウントの設定
A. Web ページの認証ドメイン名を設定する必要がありますが、これは特別なことではありません
B. ここでの支払い認証ディレクトリに注意してください。モード (REWRITE モード) を使用するか、REWRITE モードの使用中に擬似静的モードを使用します。この時点で生成されるリンクは次のとおりです: http://serverName/Home/Blog/read/id/1; PATHINFO モードを使用している場合は、リンクは http://serverName/index.php/Home/Blog/read/id/1 です。たとえば、Home モジュールの Blog コントローラーのメソッドを通じて支払いを行う場合、支払いの認証ディレクトリは http である必要があります。 : //serverName/Home/Blog/ または http://serverName/index.php/Home/Blog/。これは、独自の TP によって設定された URL パターンによって異なります。
3. 支払いプロセス(1) 統合された注文
注文の支払いパラメータの設定は、基本的に前のものと同じです。注意すべき重要なことは、支払いコールバック検証リンクである必要があるためです。複数回呼び出される場合は、パラメータ設定を Application/Common/Common/function.php に直接カプセル化しました。SDK はプロジェクトのルート ディレクトリの下の Api ディレクトリに配置されているため、SDK の導入時に Vendor 関数は使用しません。
/** * 微信支付 * @param string $openId openid * @param string $goods 商品名称 * @param string $attach 附加参数,我们可以选择传递一个参数,比如订单ID * @param string $order_sn 订单号 * @param string $total_fee 金额 */ function wxpay($openId,$goods,$order_sn,$total_fee,$attach){ require_once APP_ROOT."/Api/wxpay/lib/WxPay.Api.php"; require_once APP_ROOT."/Api/wxpay/payment/WxPay.JsApiPay.php"; require_once APP_ROOT.'/Api/wxpay/payment/log.php'; //初始化日志 $logHandler= new CLogFileHandler(APP_ROOT."/Api/wxpay/logs/".date('Y-m-d').'.log'); $log = Log::Init($logHandler, 15); $tools = new JsApiPay(); if(empty($openId)) $openId = $tools->GetOpenid(); $input = new WxPayUnifiedOrder(); $input->SetBody($goods); //商品名称 $input->SetAttach($attach); //附加参数,可填可不填,填写的话,里边字符串不能出现空格 $input->SetOut_trade_no($order_sn); //订单号 $input->SetTotal_fee($total_fee); //支付金额,单位:分 $input->SetTime_start(date("YmdHis")); //支付发起时间 $input->SetTime_expire(date("YmdHis", time() + 600));//支付超时 $input->SetGoods_tag("test3"); //$input->SetNotify_url("http://".$_SERVER['HTTP_HOST']."/payment.php"); //支付回调验证地址 $input->SetNotify_url("http://".$_SERVER['HTTP_HOST']."/payment.php/WexinApi/WeixinPay/notify"); $input->SetTrade_type("JSAPI"); //支付类型 $input->SetOpenid($openId); //用户openID $order = WxPayApi::unifiedOrder($input); //统一下单 $jsApiParameters = $tools->GetJsApiParameters($order); return $jsApiParameters; }
注意、注意、ここが重要なポイントです:
そのリンクに自分でアクセスし、認証のためにログインして登録する必要がある場合は、許可なしに検証する必要があります。アクセス可能なリンクは可能ですが、一連のパラメータを渡さないでください。最もシンプルで粗雑なhttp://serverName/xxx.php。それに対応するモジュール(WexinApi. Application/ ディレクトリ内のコントローラー (WeixinPay)、およびメソッド (通知):
// 检测PHP环境 if(version_compare(PHP_VERSION,'5.3.0','<')) die('require PHP > 5.3.0 !'); // $_GET['m']='Admin'; // 开启调试模式 建议开发阶段开启 部署阶段注释或者设为false define('APP_DEBUG',True); //指定模块控制器和方法 $_GET['m']='WexinApi'; $_GET['c']='WeixinPay'; $_GET['a']='notify'; // 定义应用目录 define('APP_PATH','./Application/'); define("APP_ROOT",dirname(__FILE__)); // 引入ThinkPHP入口文件 require './ThinkCore/ThinkCore.php'; // 亲^_^ 后面不需要任何代码了 就是如此简单
次に、http://serverName/payment.php にアクセスし、http://serverName/payment/weixinPay/ と直接入力します。通知するため、コールバック検証リンクは http://serverName/payment.php または http://serverName/payment.php/WexinApi/WeixinPay/notify のように記述できます。
(2) 支払いの開始
はまだ非常に簡単です:
/** * 支付测试 * 微信访问:http://daoshi.sdxiaochengxu.com/payment.php/WexinApi/WeixinPay/pay */ public function pay(){ $order_sn = getrand_num(true); $openId = ''; $jsApiParameters = wxpay($openId,'江南极客',$order_sn,1); $this->assign(array( 'data' => $jsApiParameters )); $this->display(); } <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 type="text/javascript"> //调用微信JS api 支付 function jsApiCall() { var data={$data}; WeixinJSBridge.invoke( 'getBrandWCPayRequest', data, function(res){ WeixinJSBridge.log(res.err_msg); //alert('err_code:'+res.err_code+'err_desc:'+res.err_desc+'err_msg:'+res.err_msg); //alert(res.err_code+res.err_desc+res.err_msg); //alert(res); if(res.err_msg == "get_brand_wcpay_request:ok"){ alert("支付成功!"); window.location.href="http://m.blog.csdn.net/article/details?id=72765676" rel="external nofollow" ; }else if(res.err_msg == "get_brand_wcpay_request:cancel"){ alert("用户取消支付!"); }else{ alert("支付失败!"); } } ); } 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(); } } </script> </head> <body> <br/> <font color="#9ACD32"><b>该笔订单支付金额为<span style="color:#f00;font-size:50px">1分</span>钱</b></font><br/><br/> <font color="#9ACD32"><b><span style="color:#f00;font-size:50px;margin-left:40%;">1分</span>钱也是爱</b></font><br/><br/> <p align="center"> <button style="width:210px; height:50px; border-radius: 15px;background-color:#FE6714; border:0px #FE6714 solid; cursor: pointer; color:white; font-size:16px;" type="button" onclick="callpay()" >果断买买买^_^</button> </p> </body> </html>
しかし、支払いページの URL には多くのパラメータが必要であるため、支払いページの URL には注意する必要があります。先ほど、で使用される REWRITE モードについて説明しました。 TP、あなたのリンク [ http://serverName/Home/Blog/read/id/1 ] に似ていますが、より多くのパラメーターがある場合があります。この時点で、WeChat Pay はあなたの支払い承認ディレクトリが [ http:/ ] であると認識します。 /serverName/Home/ Blog/read/id/ ] ですが、実際に許可されているディレクトリは [ http://serverName/Home/Blog/ ] であるため、エラーが報告されます。解決策は、支払いページに入るときに URL を再構築し、通常モード ([http://serverName/Home/Blog/read?id=1]) で記述することです。それだけです。
(3) 成功コールバックをサポートします
支払いが完了したので、以前に書き込んだリンクに対応するメソッドを入力します。これは、[ http://serverName/payment.php/WexinApi/WeixinPay/notify ] です。 ]:
//微信支付回调验证 public function notify(){ $xml = $GLOBALS['HTTP_RAW_POST_DATA']; // 这句file_put_contents是用来查看服务器返回的XML数据 测试完可以删除了 file_put_contents('./Api/wxpay/logs/log.txt',$xml,FILE_APPEND); //将服务器返回的XML数据转化为数组 //$data = json_decode(json_encode(simplexml_load_string($xml,'SimpleXMLElement',LIBXML_NOCDATA)),true); $data = xmlToArray($xml); // 保存微信服务器返回的签名sign $data_sign = $data['sign']; // sign不参与签名算法 unset($data['sign']); $sign = $this->makeSign($data); // 判断签名是否正确 判断支付状态 if ( ($sign===$data_sign) && ($data['return_code']=='SUCCESS') && ($data['result_code']=='SUCCESS') ) { $result = $data; // 这句file_put_contents是用来查看服务器返回的XML数据 测试完可以删除了 file_put_contents('./Api/wxpay/logs/log1.txt',$xml,FILE_APPEND); //获取服务器返回的数据 $order_sn = $data['out_trade_no']; //订单单号 $order_id = $data['attach']; //附加参数,选择传递订单ID $openid = $data['openid']; //付款人openID $total_fee = $data['total_fee']; //付款金额 //更新数据库 $this->updateDB($order_id,$order_sn,$openid,$total_fee); }else{ $result = false; } // 返回状态给微信服务器 if ($result) { $str='<xml><return_code><![CDATA[SUCCESS]]></return_code><return_msg><![CDATA[OK]]></return_msg></xml>'; }else{ $str='<xml><return_code><![CDATA[FAIL]]></return_code><return_msg><![CDATA[签名失败]]></return_msg></xml>'; } echo $str; return $result; }
セキュリティ上の理由から、返された署名は再検証される必要があります:
/** * 生成签名 * @return 签名,本函数不覆盖sign成员变量 */ protected function makeSign($data){ //获取微信支付秘钥 require_once APP_ROOT."/Api/wxpay/lib/WxPay.Api.php"; $key = \WxPayConfig::KEY; // 去空 $data=array_filter($data); //签名步骤一:按字典序排序参数 ksort($data); $string_a=http_build_query($data); $string_a=urldecode($string_a); //签名步骤二:在string后加入KEY //$config=$this->config; $string_sign_temp=$string_a."&key=".$key; //签名步骤三:MD5加密 $sign = md5($string_sign_temp); // 签名步骤四:所有字符转为大写 $result=strtoupper($sign); return $result; }
至此,TP中微信支付也就搞定了。这是集成了官方的SDK实现的,如果不使用SDK,可以使用更简单的方法,见:PHP实现微信支付(jsapi支付)和退款(无需集成支付SDK)
以上就是本文的全部内容,希望对大家的学习有所帮助,更多相关内容请关注PHP中文网!
相关推荐:
以上がThinkPHP は WeChat 支払い (jsapi 支払い) を実装しますの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

ホットAIツール

Undresser.AI Undress
リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover
写真から衣服を削除するオンライン AI ツール。

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

Video Face Swap
完全無料の AI 顔交換ツールを使用して、あらゆるビデオの顔を簡単に交換できます。

人気の記事

ホットツール

SublimeText3 Mac版
神レベルのコード編集ソフト(SublimeText3)

SublimeText3 Linux 新バージョン
SublimeText3 Linux 最新バージョン

ZendStudio 13.5.1 Mac
強力な PHP 統合開発環境

SublimeText3 英語版
推奨: Win バージョン、コードプロンプトをサポート!

Safe Exam Browser
Safe Exam Browser は、オンライン試験を安全に受験するための安全なブラウザ環境です。このソフトウェアは、あらゆるコンピュータを安全なワークステーションに変えます。あらゆるユーティリティへのアクセスを制御し、学生が無許可のリソースを使用するのを防ぎます。
