(easywechat + Laravel 5.8) PC 上で WeChat スキャンコード支払いのプロセス全体を整理します
ビジネス シナリオの説明:
ユーザーがサイト ページで [購入] をクリック --> QR コードがポップアップ --> ユーザーが WeChat を使用して QR コードをスキャン--> ; WeChat の指示に従って支払いを完了します --> 支払いが成功すると、支払いが成功したことを示すメッセージがページに表示され、ジャンプします
WeChat とのやり取りは 3 つのステップで行われます。
1. パラメーターを渡し、WeChat 統合注文インターフェイスをリクエストし、支払い QR コードを取得します
2. WeChat から通知を受信します (WeChat はコールバックを通じて支払い結果をサーバーに送信します)前のステップのパラメータのアドレス)
3. WeChat に注文インターフェイスの表示をリクエストします。支払いが成功すると、ページがジャンプします。
次のレコードは基本的に上記のプロセスに従います.
準備作業:
インストールovertrue/laravel-wechat
composer require "overtrue/laravel-wechat:~5.0"
設定ファイルの作成:
php artisan vendor:publish --provider="Overtrue\LaravelWeChat\ServiceProvider"
アプリケーションのルート ディレクトリにある対応する config/wechat.php パラメータを変更します (この部分をコピー/ペーストするだけです):
'payment' => [ 'default' => [ 'sandbox' => env('WECHAT_PAYMENT_SANDBOX', false), 'app_id' => env('WECHAT_PAYMENT_APPID', ''), 'mch_id' => env('WECHAT_PAYMENT_MCH_ID', 'your-mch-id'), 'key' => env('WECHAT_PAYMENT_KEY', 'key-for-signature'), 'cert_path' => env('WECHAT_PAYMENT_CERT_PATH', 'path/to/cert/apiclient_cert.pem'), // XXX: 绝对路径!!!! 'key_path' => env('WECHAT_PAYMENT_KEY_PATH', 'path/to/cert/apiclient_key.pem'), // XXX: 绝对路径!!!! 'notify_url' => env('WECHAT_PAYMENT_NOTIFY_URL',''), // 默认支付结果通知地址 ], // ... ],
設定する必要があるのは上記の配列の内容ですが、実際にはすべて.env ファイルで設定する必要があります:
# wechat_payment WECHAT_PAYMENT_SANDBOX=false # 真正需要配置的就下面这四行 WECHAT_PAYMENT_APPID=xxxxxxxxxxxxxxx // 自己的 WECHAT_PAYMENT_MCH_ID=xxxxxxx // 自己的 WECHAT_PAYMENT_KEY=xxxxxxxxxxxxxxxxxxxx // 自己的 WECHAT_PAYMENT_NOTIFY_URL='test.abc.com/payment/notify' // 这个地址只要是外网能够访问到项目的任何地址都可以, 不是需要在微信那里配置的那种, 现在还不知道怎么定义没关系, 后面用到的时候自然就有了 SWAGGER_VERSION=3.0
インストール シンプルな QrCode QR コード パッケージの生成
composer.json に次のコードを追加しますfile:
"require": { "simplesoftwareio/simple-qrcode": "~2" }
ターミナルで実行:composer update 、後で使用します。
---------------------- ------------------------ ----以上が準備作業です。手順を追っていきましょう ------------------------ ----------------------------------- --
ユーザーが「購入」をクリック 注文する --> ポップアップ QR コード
WeChat のリクエストです統合注文インターフェイス。
私が処理するロジックは次のとおりです。 :
ユーザーが購入リクエストを開始すると、まず支払いログにレコードを作成し、ユーザーが支払いを完了するまで待ってから注文レコードを作成します。
具体的には新しい PaymentController を作成しますWeChat 支払いのロジックを処理します (OrderController とは異なります)。「購入」をクリックするというユーザーのリクエストに対して、「place_order」メソッドを使用して処理します。つまり、ここで WeChat の [統合注文] インターフェイスをリクエストします。
##注文リクエストを開始するページ上の部分Html 部分:#(QR コードのモーダル ボックスはブートストラップに従って記述されています) document)
<button type="button" id="order" class="btn btn-secondary btn-block"> 扫码支付 </button> <!-- 二维码, 随便放在当前页面的那里都可以, 因为是通过 axios 控制, 请求成功后才会弹出的 --> <div class="modal fade" id="qrcode" tabindex="-1" role="dialog" aria-hidden="true"> <div class="modal-dialog modal-sm" role="document"> <div class="modal-content bg-transparent" style="border:none"> <div class="modal-body align-items-center text-center"> <p class="modal-title" id="exampleModalLabel" style="color:white">微信扫码支付</p> <br> {{--生成的二维码会放在这里--}} <div id="qrcode2"></div> </div> </div> </div> </div>JS 部分:
$('#order').click(function () {
/** 请求下单接口 **/
axios.get("/payment/place_order", {
params: {
id: "{{ $post->id }}"
}
}).then(function (response) {
if (response.data.code == 200) {
/** 把生成的二维码放到页面上 */
$('#qrcode2').html(response.data.html);
/** 弹出二维码 **/
$('#qrcode').modal('show');
/** 设置定时器, 即一弹出二维码就开始不断请求查看支付状态, 直到收到支付成功的返回, 再终止定时器 **/
var timer = setInterval(function () {
/** 在这里请求微信支付状态的接口 **/
axios.get('/payment/paid', {
params: {
'out_trade_no':response.data.order_sn,
}
}).then(function (response) {
if (response.data.code == 200) {
/** 如果支付成功, 就取消定时器, 并重新加载页面 */
window.clearInterval(timer);
window.location.reload();
}
}).catch(function (error) {
console.log(error);
});
}, 3000);
}
}).catch(function (error) {
console.log(error);
});
});
ここでは、まず要求された 2 つのルートを JS に記述します上の部分 これは書かれています。まず最初の部分を説明しましょう:
// 请求微信统一下单接口 Route::get('/payment/place_order', 'PaymentController@place_order')->name('web.payment.place_order'); // 请求微信接口, 查看订单支付状态 Route::get('/payment/paid', 'PaymentController@paid')->name('web.payment.paid'); PaymentController 里的支付逻辑 下面是具体的逻辑,用户点击支付后,先创建一条记录在 PayLog (用来记录支付的详细信息,所以这里还需要建 Paylog 的 model 和 migration, migration 的内容我附在最后了,都是微信返回的字段,基本可以直接 copy 来用的) class PaymentController extends Controller { // 请求微信接口的公用配置, 所以单独提出来 private function payment() { $config = [ // 必要配置, 这些都是之前在 .env 里配置好的 'app_id' => config('wechat.payment.default.app_id'), 'mch_id' => config('wechat.payment.default.mch_id'), 'key' => config('wechat.payment.default.key'), // API 密钥 'notify_url' => config('wechat.payment.default.notify_url'), // 通知地址 ]; // 这个就是 easywechat 封装的了, 一行代码搞定, 照着写就行了 $app = Factory::payment($config); return $app; } // 向微信请求统一下单接口, 创建预支付订单 public function place_order($id) { // 因为没有先创建订单, 所以这里先生成一个随机的订单号, 存在 pay_log 里, 用来标识订单, 支付成功后再把这个订单号存到 order 表里 $order_sn = date('ymd').substr(time(),-5).substr(microtime(),2,5); // 根据文章 id 查出文章价格 $post_price = optional(Post::where('id', $id)->first())->pirce; // 创建 Paylog 记录 PayLog::create([ 'appid' => config('wechat.payment.default.app_id'), 'mch_id' => config('wechat.payment.default.mch_id'), 'out_trade_no' => $order_sn, 'post_id' => $id ]); $app = $this->payment(); $total_fee = env('APP_DEBUG') ? 1 : $post_price; // 用 easywechat 封装的方法请求微信的统一下单接口 $result = $app->order->unify([ 'trade_type' => 'NATIVE', // 原生支付即扫码支付,商户根据微信支付协议格式生成的二维码,用户通过微信“扫一扫”扫描二维码后即进入付款确认界面,输入密码即完成支付。 'body' => '投资平台-订单支付', // 这个就是会展示在用户手机上巨款界面的一句话, 随便写的 'out_trade_no' => $order_sn, 'total_fee' => $total_fee, 'spbill_create_ip' => request()->ip(), // 可选,如不传该参数,SDK 将会自动获取相应 IP 地址 ]); if ($result['result_code'] == 'SUCCESS') { // 如果请求成功, 微信会返回一个 'code_url' 用于生成二维码 $code_url = $result['code_url']; return [ 'code' => 200, // 订单编号, 用于在当前页面向微信服务器发起订单状态查询请求 'order_sn' => $order_sn, // 生成二维码 'html' => QrCode::size(200)->generate($code_url), ]; } } }
----------- WeChat と対話する最初のステップ (統合された注文インターフェイスの要求) 完了----- - -----
WeChat から WeChat リクエストをルーティングするアップロードパラメータのnotify_urlは、サーバーに支払い結果を送信するようリクエストするため、ポストリクエストである必要があります:
Route::post('/payment/notify', 'paymentController@notify');Cancel csrf validation
ただし、WeChat サーバーが開始されました 投稿リクエストは csrf トークン検証に合格できないため、WeChat に使用されるルートの検証をキャンセルする必要があります。app/Http/Middleware/VerifyCsrfToken ファイル内:
protected $except = [ // 'payment/notify' ]; 在 PaymentController.php 文件中处理接收微信信息的逻辑 // 接收微信支付状态的通知 public function notify() { $app = $this->payment(); // 用 easywechat 封装的方法接收微信的信息, 根据 $message 的内容进行处理, 之后要告知微信服务器处理好了, 否则微信会一直请求这个 url, 发送信息 $response = $app->handlePaidNotify(function($message, $fail){ // 首先查看 order 表, 如果 order 表有记录, 表示已经支付过了 $order = Order::where('order_sn', $message['out_trade_no'])->first(); if ($order) { return true; // 如果已经生成订单, 表示已经处理完了, 告诉微信不用再通知了 } // 查看支付日志 $payLog = PayLog::where('out_trade_no', $message['out_trade_no'])->first(); if (!$payLog || $payLog->paid_at) { // 如果订单不存在 或者 订单已经支付过了 return true; // 告诉微信,我已经处理完了,订单没找到,别再通知我了 } // return_code 表示通信状态,不代表支付状态 if ($message['return_code'] === 'SUCCESS') { // 用户是否支付成功 if ($message['result_code'] === 'SUCCESS') { // 更新支付时间为当前时间 $payLog->paid_at = now(); $post_id = $payLog->post_id; // 联表查询 post 的相关信息 $post_title = $payLog->post->title; $post_price = $payLog->post->price; $post_original_price = $payLog->post->original_price; $post_cover = $payLog->post->post_cover; $post_description = $payLog->post->description; $user_id = $payLog->post->user_id; // 创建订单记录 Order::create([ 'order_sn' => $message['out_trade_no'], 'total_fee' => $message['total_fee'], 'pay_log_id' => $payLog->id, 'status' => 1, 'user_id' => $user_id, 'paid_at' => $payLog->paid_at, 'post_id' => $post_id, 'post_title' => $post_title, 'post_price' => $post_price, 'post_original_price' => $post_original_price, 'post_cover' => $post_cover, 'post_description' => $post_description, ]); // 更新 PayLog, 这里的字段都是根据微信支付结果通知的字段设置的(https://pay.weixin.qq.com/wiki/doc/api/native.php?chapter=9_7&index=8) PayLog::where('out_trade_no', $message['out_trade_no'])->update([ 'appid' => $message['appid'], 'bank_type' => $message['bank_type'], 'total_fee' => $message['total_fee'], 'trade_type' => $message['trade_type'], 'is_subscribe' => $message['is_subscribe'], 'mch_id' => $message['mch_id'], 'nonce_str' => $message['nonce_str'], 'openid' => $message['openid'], 'sign' => $message['sign'], 'cash_fee' => $message['cash_fee'], 'fee_type' => $message['fee_type'], 'transaction_id' => $message['transaction_id'], 'time_end' => $payLog->paid_at, 'result_code' => $message['result_code'], 'return_code' => $message['return_code'], ]); } } else { // 如果支付失败, 也更新 PayLog, 跟上面一样, 就是多了 error 信息 PayLog::where('out_trade_no', $message['out_trade_no'])->update([ 'appid' => $message['appid'], 'bank_type' => $message['bank_type'], 'total_fee' => $message['total_fee'], 'trade_type' => $message['trade_type'], 'is_subscribe' => $message['is_subscribe'], 'mch_id' => $message['mch_id'], 'nonce_str' => $message['nonce_str'], 'openid' => $message['openid'], 'sign' => $message['sign'], 'cash_fee' => $message['cash_fee'], 'fee_type' => $message['fee_type'], 'transaction_id' => $message['transaction_id'], 'time_end' => $payLog->paid_at, 'result_code' => $message['result_code'], 'return_code' => $message['return_code'], 'err_code' => $message['err_code'], 'err_code_des' => $message['err_code_des'], ]); return $fail('通信失败,请稍后再通知我'); } return true; // 返回处理完成 }); // 这里是必须这样返回的, 会发送给微信服务器处理结果 return $response; }
上記はpay_logs テーブルと Posts テーブルを使用したジョイント テーブル クエリ。投稿には複数の pay_log を含めることができるため、1 対多の関係になります。PayLog.php で設定します:
public function post() { return $this->belongsTo(Post::class); }
------ --------WeChat との対話 2 番目のステップ (情報の受信)、完了 --------------
WeChat のリクエストView orderInterface WeChat に注文ステータス インターフェイスの表示をリクエストします。ルーティングは対話の最初のステップで記述されています
public function paid(Request $request) { $out_trade_no = $request->get('out_trade_no'); $app = $this->payment(); // 用 easywechat 封装的方法请求微信 $result = $app->order->queryByOutTradeNumber($out_trade_no); if ($result['trade_state'] === 'SUCCESS') return [ 'code' => 200, 'msg' => 'paid' ]; }else{ return [ 'code' => 202, 'msg' => 'not paid' ]; } }
--------- -------WeChat との対話の 3 番目のステップ (注文ステータスの確認)、完了しました-----
添付ファイル: pay_logs テーブルの移行
この表のフィールドは基本的に WeChat 決済なので、次回使いやすいように結果通知のフィールドを以下に添付します。
public function up() { Schema::create('pay_logs', function (Blueprint $table) { $table->bigIncrements('id'); // 根据自身业务设计的字段 $table->integer('post_id')->default(0)->comment('文章id'); // 以下均是微信支付结果通知接口返回的字段 $table->string('appid', 255)->default('')->comment('微信分配的公众账号ID'); $table->string('mch_id', 255)->default('')->comment('微信支付分配的商户号'); $table->string('bank_type', 16)->default('')->comment('付款银行'); $table->integer('cash_fee')->default(0)->comment('现金支付金额'); $table->string('fee_type', 8)->default('')->comment('货币种类'); $table->string('is_subscribe', 1)->default('')->comment('是否关注公众账号'); $table->string('nonce_str', 32)->default('')->comment('随机字符串'); $table->string('openid', 128)->default('')->comment('用户标识'); $table->string('out_trade_no', 32)->default('')->comment('商户系统内部订单号'); $table->string('result_code', 16)->default('')->comment('业务结果'); $table->string('return_code', 16)->default('')->comment('通信标识'); $table->string('sign', 32)->default('')->comment('签名'); $table->string('prepay_id', 64)->default('')->comment('微信生成的预支付回话标识,用于后续接口调用中使用,该值有效期为2小时'); $table->dateTime('time_end')->nullable()->comment('支付完成时间'); $table->integer('total_fee')->default(0)->comment('订单金额'); $table->string('trade_type', 16)->default('')->comment('交易类型'); $table->string('transaction_id', 32)->default('')->comment('微信支付订单号'); $table->string('err_code', 32)->default('')->comment('错误代码'); $table->string('err_code_des', 128)->default('')->comment('错误代码描述'); $table->string('device_info', 32)->default('')->comment('设备号'); $table->text('attach')->nullable()->comment('商家数据包'); $table->nullableTimestamps(); }); }
上記はページから注文までの全過程の記録ですお支払いへページジャンプへ。ずっと前に一度 Laravel-china のチュートリアルに従ったことを除けば、実際に自分で調べて自分のニーズに合わせて作成したのは今回が初めてです。インターネット上には多くの記事やチュートリアルが共有されていますが、どれもマスターレベルであり、多くの場所が一度に言及されているため、私のような初心者にとって、ある場所ではハンマー、別の場所では槌のような感じです。分かりにくいので、頑張ってメモを詳しくまとめてみましたが、私のような初心者の方の参考になれば幸いです。非常に長く見えますが、少なくとも私の現在のレベルでは、非常に多くの内容が必要です。
以上が(easywechat + Laravel 5.8) PC 上で WeChat スキャンコード支払いのプロセス全体を整理しますの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

Laravelは、バックエンド開発で強く機能し、Eloquentorm、コントローラー、サービスクラスを介してデータベース操作を簡素化し、ビジネスロジックを処理し、キュー、イベント、その他の機能を提供します。 1)Eloquentormは、モデルを介してデータベーステーブルをマップしてクエリを簡素化します。 2)ビジネスロジックは、モジュール性と保守性を向上させるために、コントローラーとサービスクラスで処理されます。 3)キューシステムなどのその他の機能は、複雑なニーズの処理に役立ちます。

Laravel Developmentプロジェクトは、さまざまなサイズと複雑さのニーズに合う柔軟性とパワーのために選択されました。 Laravelは、ルーティングシステム、Eloquentorm、Artisan Command Lineおよびその他の機能を提供し、簡単なブログから複雑なエンタープライズレベルのシステムへの開発をサポートしています。

開発環境とエコシステムにおけるLaravelとPythonの比較は次のとおりです。1。Laravelの開発環境は簡単で、PHPと作曲家のみが必要です。 Laravelforgeなどの豊富な範囲の拡張パッケージを提供しますが、拡張パッケージのメンテナンスはタイムリーではない場合があります。 2。Pythonの開発環境もシンプルで、PythonとPIPのみが必要です。エコシステムは巨大で複数のフィールドをカバーしていますが、バージョンと依存関係の管理は複雑な場合があります。

Laravelはバックエンドロジックでどのように役割を果たしますか?ルーティングシステム、Eloquentorm、認証と承認、イベントとリスナー、パフォーマンスの最適化を通じてバックエンド開発を簡素化および強化します。 1.ルーティングシステムにより、URL構造の定義とリクエスト処理ロジックが可能になります。 2.Eloquentormは、データベースの相互作用を簡素化します。 3.認証および承認システムは、ユーザー管理に便利です。 4.イベントとリスナーは、ゆるく結合したコード構造を実装します。 5.パフォーマンスの最適化により、キャッシュとキューイングを通じてアプリケーションの効率が向上します。

Laravelの人気には、単純化された開発プロセスが含まれ、快適な開発環境を提供し、豊富な機能が提供されます。 1)Rubyonrailsの設計哲学を吸収し、PHPの柔軟性を組み合わせています。 2)Eloquentorm、Bladeテンプレートエンジンなどのツールを提供して、開発効率を向上させます。 3)そのMVCアーキテクチャと依存関係噴射メカニズムにより、コードがよりモジュール化され、テスト可能になります。 4)キャッシュシステムやベストプラクティスなどの強力なデバッグツールとパフォーマンス最適化方法を提供します。

DjangoとLaravelはどちらもフルスタックのフレームワークです。 DjangoはPython開発者や複雑なビジネスロジックに適していますが、LaravelはPHP開発者とエレガントな構文に適しています。 1.DjangoはPythonに基づいており、迅速な発展と高い並行性に適した「バッテリーコンプリート」哲学に従います。 2. LaravelはPHPに基づいており、開発者エクスペリエンスを強調しており、小規模から中規模のプロジェクトに適しています。

LaravelはPHPベースのフレームワークであるため、PHPとLaravelは直接匹敵するものではありません。 1.PHPは、シンプルで直接的であるため、小規模プロジェクトや迅速なプロトタイピングに適しています。 2。LARAVELは、豊富な機能とツールを提供するため、大規模なプロジェクトや効率的な開発に適していますが、急な学習曲線があり、純粋なPHPほど良くない場合があります。

laravelisabackendframeworkbuiltonphp、designforwebapplicationdevelopment.itfocusonserver-sidelogic、databasemanagement、およびapplicationStructure、およびbueithedendtechnologiesvue.jsorreactforfull-stackdevelymentと統合されていること。


ホットAIツール

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

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

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

AI Hentai Generator
AIヘンタイを無料で生成します。

人気の記事

ホットツール

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

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

DVWA
Damn Vulnerable Web App (DVWA) は、非常に脆弱な PHP/MySQL Web アプリケーションです。その主な目的は、セキュリティ専門家が法的環境でスキルとツールをテストするのに役立ち、Web 開発者が Web アプリケーションを保護するプロセスをより深く理解できるようにし、教師/生徒が教室環境で Web アプリケーションを教え/学習できるようにすることです。安全。 DVWA の目標は、シンプルでわかりやすいインターフェイスを通じて、さまざまな難易度で最も一般的な Web 脆弱性のいくつかを実践することです。このソフトウェアは、

SublimeText3 中国語版
中国語版、とても使いやすい

EditPlus 中国語クラック版
サイズが小さく、構文の強調表示、コード プロンプト機能はサポートされていません
