Browse Source

feat: 增加支付宝和微信转账支持

在WalletWithdrawRecordService中,新增了对支付宝和微信转账的处理逻辑。根据提现方式调用相应的转账接口,并处理转账结果。此更改旨在提升提现功能的灵活性,支持更多支付方式,优化用户体验。
刘学玺 2 months ago
parent
commit
677ad67ae8
4 changed files with 401 additions and 8 deletions
  1. 104 7
      app/Services/Admin/WalletWithdrawRecordService.php
  2. 1 0
      composer.json
  3. 242 1
      composer.lock
  4. 54 0
      config/alipay.php

+ 104 - 7
app/Services/Admin/WalletWithdrawRecordService.php

@@ -10,6 +10,7 @@ use App\Enums\TransactionType;
 use App\Enums\TransactionStatus;
 use Illuminate\Support\Facades\DB;
 use Illuminate\Support\Facades\Log;
+use App\Enums\WithdrawMethod;
 
 class WalletWithdrawRecordService
 {
@@ -215,15 +216,111 @@ class WalletWithdrawRecordService
 
     /**
      * 处理实际打款
-     * TODO: 对接实际支付服务
+     *
+     * @param WalletWithdrawRecord $record 提现记录
+     * @return array [
+     *     'success' => bool 是否成功
+     *     'trade_no' => string 交易流水号
+     *     'message' => string 结果信息
+     * ]
+     * @throws \Exception
      */
     private function processPayment(WalletWithdrawRecord $record): array
     {
-        // 模拟打款成功
-        return [
-            'success' => true,
-            'trade_no' => 'PAY' . time() . random_int(1000, 9999),
-            'message' => '打款成功'
-        ];
+        try {
+            // 根据提现方式调用不同的转账接口
+            if ($record->withdraw_type === WithdrawMethod::WECHAT->value) {
+                // 使用 EasyWeChat 6.x 调用微信转账到零钱接口
+                $config = [
+                    'app_id' => config('wechat.payment.default.app_id'),
+                    'mch_id' => config('wechat.payment.default.mch_id'),
+                    'key' => config('wechat.payment.default.key'),
+                    'cert_path' => config('wechat.payment.default.cert_path'),
+                    'key_path' => config('wechat.payment.default.key_path'),
+                ];
+
+                $app = \EasyWeChat\Pay\Application::fromConfig($config);
+
+                $result = $app->transfer->toBalance([
+                    'out_batch_no' => $record->external_no,
+                    'batch_name' => '用户提现',
+                    'batch_remark' => '用户提现到微信零钱',
+                    'total_amount' => (int)bcmul($record->amount, '100'), // 转为分
+                    'total_num' => 1,
+                    'transfer_detail_list' => [[
+                        'out_detail_no' => $record->external_no . 'D1',
+                        'transfer_amount' => (int)bcmul($record->amount, '100'),
+                        'transfer_remark' => '提现到微信零钱',
+                        'openid' => $record->withdraw_account, // withdraw_account 存储的是 openid
+                    ]],
+                ]);
+
+                if ($result['return_code'] === 'SUCCESS' && $result['result_code'] === 'SUCCESS') {
+                    return [
+                        'success' => true,
+                        'trade_no' => $result['payment_no'] ?? $result['partner_trade_no'],
+                        'message' => '微信转账成功'
+                    ];
+                }
+
+                throw new \Exception($result['err_code_des'] ?? '微信转账失败');
+            } elseif ($record->withdraw_type === WithdrawMethod::ALIPAY->value) {
+                // 使用支付宝 SDK 调用转账接口
+                $config = [
+                    'app_id' => config('alipay.app_id'),
+                    'private_key' => config('alipay.private_key'),
+                    'alipay_public_key' => config('alipay.alipay_public_key'),
+                ];
+
+                // 初始化支付宝 SDK
+                \Alipay\EasySDK\Kernel\Factory::setOptions($config);
+
+                // 调用转账接口
+                $result = \Alipay\EasySDK\Kernel\Factory::payment()->common()
+                    ->transfer([
+                        'out_biz_no' => $record->external_no,
+                        'trans_amount' => $record->amount,
+                        'product_code' => 'TRANS_ACCOUNT_NO_PWD',
+                        'biz_scene' => 'DIRECT_TRANSFER',
+                        'payee_info' => [
+                            'identity' => $record->withdraw_account,
+                            'identity_type' => 'ALIPAY_LOGON_ID',
+                            'name' => $record->withdraw_account_name,
+                        ],
+                        'remark' => '用户提现',
+                    ]);
+
+                if ($result->code === '10000') {
+                    return [
+                        'success' => true,
+                        'trade_no' => $result->order_id,
+                        'message' => '支付宝转账成功'
+                    ];
+                }
+
+                throw new \Exception($result->sub_msg ?? '支付宝转账失败');
+            } elseif ($record->withdraw_type === WithdrawMethod::BANK->value) {
+                // 银行卡转账暂时使用模拟逻辑
+                return [
+                    'success' => true,
+                    'trade_no' => 'BANK' . time() . random_int(1000, 9999),
+                    'message' => '模拟银行卡打款成功'
+                ];
+            } else {
+                throw new \Exception('不支持的提现方式');
+            }
+        } catch (\Exception $e) {
+            Log::error('转账处理失败', [
+                'withdraw_id' => $record->id,
+                'withdraw_type' => $record->withdraw_type,
+                'error' => $e->getMessage()
+            ]);
+
+            return [
+                'success' => false,
+                'trade_no' => '',
+                'message' => $e->getMessage()
+            ];
+        }
     }
 }

+ 1 - 0
composer.json

@@ -6,6 +6,7 @@
     "license": "MIT",
     "require": {
         "php": "^8.2",
+        "alipaysdk/easysdk": "^2.2",
         "huifurepo/dg-php-sdk": "^2.0",
         "iwzh/owl-scheduling": "^1.0",
         "knuckleswtf/scribe": "^4.0",

+ 242 - 1
composer.lock

@@ -4,8 +4,249 @@
         "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
         "This file is @generated automatically"
     ],
-    "content-hash": "44486a9a1b206d281de829f0a81c5ef3",
+    "content-hash": "0b94b47b777b531a2c91338a4b4b7ff1",
     "packages": [
+        {
+            "name": "adbario/php-dot-notation",
+            "version": "2.5.0",
+            "source": {
+                "type": "git",
+                "url": "https://github.com/adbario/php-dot-notation.git",
+                "reference": "081e2cca50c84bfeeea2e3ef9b2c8d206d80ccae"
+            },
+            "dist": {
+                "type": "zip",
+                "url": "https://api.github.com/repos/adbario/php-dot-notation/zipball/081e2cca50c84bfeeea2e3ef9b2c8d206d80ccae",
+                "reference": "081e2cca50c84bfeeea2e3ef9b2c8d206d80ccae",
+                "shasum": "",
+                "mirrors": [
+                    {
+                        "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%",
+                        "preferred": true
+                    }
+                ]
+            },
+            "require": {
+                "ext-json": "*",
+                "php": "^5.5 || ^7.0 || ^8.0"
+            },
+            "require-dev": {
+                "phpunit/phpunit": "^4.8|^5.7|^6.6|^7.5|^8.5|^9.5",
+                "squizlabs/php_codesniffer": "^3.6"
+            },
+            "type": "library",
+            "autoload": {
+                "files": [
+                    "src/helpers.php"
+                ],
+                "psr-4": {
+                    "Adbar\\": "src"
+                }
+            },
+            "notification-url": "https://packagist.org/downloads/",
+            "license": [
+                "MIT"
+            ],
+            "authors": [
+                {
+                    "name": "Riku Särkinen",
+                    "email": "riku@adbar.io"
+                }
+            ],
+            "description": "PHP dot notation access to arrays",
+            "homepage": "https://github.com/adbario/php-dot-notation",
+            "keywords": [
+                "ArrayAccess",
+                "dotnotation"
+            ],
+            "support": {
+                "issues": "https://github.com/adbario/php-dot-notation/issues",
+                "source": "https://github.com/adbario/php-dot-notation/tree/2.5.0"
+            },
+            "time": "2022-10-14T20:31:46+00:00"
+        },
+        {
+            "name": "alibabacloud/tea",
+            "version": "3.2.1",
+            "source": {
+                "type": "git",
+                "url": "https://github.com/aliyun/tea-php.git",
+                "reference": "1619cb96c158384f72b873e1f85de8b299c9c367"
+            },
+            "dist": {
+                "type": "zip",
+                "url": "https://api.github.com/repos/aliyun/tea-php/zipball/1619cb96c158384f72b873e1f85de8b299c9c367",
+                "reference": "1619cb96c158384f72b873e1f85de8b299c9c367",
+                "shasum": "",
+                "mirrors": [
+                    {
+                        "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%",
+                        "preferred": true
+                    }
+                ]
+            },
+            "require": {
+                "adbario/php-dot-notation": "^2.4",
+                "ext-curl": "*",
+                "ext-json": "*",
+                "ext-libxml": "*",
+                "ext-mbstring": "*",
+                "ext-openssl": "*",
+                "ext-simplexml": "*",
+                "ext-xmlwriter": "*",
+                "guzzlehttp/guzzle": "^6.3|^7.0",
+                "php": ">=5.5"
+            },
+            "require-dev": {
+                "phpunit/phpunit": "*",
+                "symfony/dotenv": "^3.4",
+                "symfony/var-dumper": "^3.4"
+            },
+            "suggest": {
+                "ext-sockets": "To use client-side monitoring"
+            },
+            "type": "library",
+            "autoload": {
+                "psr-4": {
+                    "AlibabaCloud\\Tea\\": "src"
+                }
+            },
+            "notification-url": "https://packagist.org/downloads/",
+            "license": [
+                "Apache-2.0"
+            ],
+            "authors": [
+                {
+                    "name": "Alibaba Cloud SDK",
+                    "email": "sdk-team@alibabacloud.com",
+                    "homepage": "http://www.alibabacloud.com"
+                }
+            ],
+            "description": "Client of Tea for PHP",
+            "homepage": "https://www.alibabacloud.com/",
+            "keywords": [
+                "alibabacloud",
+                "client",
+                "cloud",
+                "tea"
+            ],
+            "support": {
+                "issues": "https://github.com/aliyun/tea-php/issues",
+                "source": "https://github.com/aliyun/tea-php"
+            },
+            "time": "2023-05-16T06:43:41+00:00"
+        },
+        {
+            "name": "alibabacloud/tea-fileform",
+            "version": "0.3.4",
+            "source": {
+                "type": "git",
+                "url": "https://github.com/alibabacloud-sdk-php/tea-fileform.git",
+                "reference": "4bf0c75a045c8115aa8cb1a394bd08d8bb833181"
+            },
+            "dist": {
+                "type": "zip",
+                "url": "https://api.github.com/repos/alibabacloud-sdk-php/tea-fileform/zipball/4bf0c75a045c8115aa8cb1a394bd08d8bb833181",
+                "reference": "4bf0c75a045c8115aa8cb1a394bd08d8bb833181",
+                "shasum": "",
+                "mirrors": [
+                    {
+                        "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%",
+                        "preferred": true
+                    }
+                ]
+            },
+            "require": {
+                "alibabacloud/tea": "^3.0",
+                "php": ">5.5"
+            },
+            "require-dev": {
+                "phpunit/phpunit": "^4.8.35|^5.4.3"
+            },
+            "type": "library",
+            "autoload": {
+                "psr-4": {
+                    "AlibabaCloud\\Tea\\FileForm\\": "src"
+                }
+            },
+            "notification-url": "https://packagist.org/downloads/",
+            "license": [
+                "Apache-2.0"
+            ],
+            "authors": [
+                {
+                    "name": "Alibaba Cloud SDK",
+                    "email": "sdk-team@alibabacloud.com"
+                }
+            ],
+            "description": "Alibaba Cloud Tea File Library for PHP",
+            "support": {
+                "issues": "https://github.com/alibabacloud-sdk-php/tea-fileform/issues",
+                "source": "https://github.com/alibabacloud-sdk-php/tea-fileform/tree/0.3.4"
+            },
+            "time": "2020-12-01T07:24:35+00:00"
+        },
+        {
+            "name": "alipaysdk/easysdk",
+            "version": "2.2.3",
+            "source": {
+                "type": "git",
+                "url": "https://github.com/alipay/alipay-easysdk.git",
+                "reference": "c6008839a22a5fca08e9f8536730f7abfed522d5"
+            },
+            "dist": {
+                "type": "zip",
+                "url": "https://api.github.com/repos/alipay/alipay-easysdk/zipball/c6008839a22a5fca08e9f8536730f7abfed522d5",
+                "reference": "c6008839a22a5fca08e9f8536730f7abfed522d5",
+                "shasum": "",
+                "mirrors": [
+                    {
+                        "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%",
+                        "preferred": true
+                    }
+                ]
+            },
+            "require": {
+                "alibabacloud/tea": "^3.1",
+                "alibabacloud/tea-fileform": "^0.3.2",
+                "ext-ctype": "*",
+                "ext-curl": "*",
+                "ext-dom": "*",
+                "ext-fileinfo": "*",
+                "ext-json": "*",
+                "ext-libxml": "*",
+                "ext-mbstring": "*",
+                "ext-openssl": "*",
+                "ext-simplexml": "*",
+                "ext-xmlwriter": "*",
+                "guzzlehttp/guzzle": ">=6.3",
+                "php": ">=7.0"
+            },
+            "require-dev": {
+                "phpunit/phpunit": "^7.5"
+            },
+            "type": "library",
+            "autoload": {
+                "psr-4": {
+                    "Alipay\\EasySDK\\": "php/src/"
+                }
+            },
+            "notification-url": "https://packagist.org/downloads/",
+            "license": [
+                "Apache-2.0"
+            ],
+            "authors": [
+                {
+                    "name": "junying.wjy",
+                    "email": "junying.wjy@antfin.com"
+                }
+            ],
+            "description": "支付宝官方 Alipay Easy SDK",
+            "support": {
+                "source": "https://github.com/alipay/alipay-easysdk/tree/v2.2.3"
+            },
+            "time": "2022-11-28T14:04:57+00:00"
+        },
         {
             "name": "aws/aws-crt-php",
             "version": "v1.2.7",

+ 54 - 0
config/alipay.php

@@ -0,0 +1,54 @@
+<?php
+
+return [
+    /*
+    |--------------------------------------------------------------------------
+    | 支付宝基础配置
+    |--------------------------------------------------------------------------
+    */
+
+    // 应用ID,您的APPID。
+    'app_id' => env('ALIPAY_APP_ID', ''),
+
+    // 商户私钥,您的原始格式RSA私钥
+    'private_key' => env('ALIPAY_PRIVATE_KEY', ''),
+
+    // 支付宝公钥,查看地址:https://openhome.alipay.com/platform/keyManage.htm 对应APPID下的支付宝公钥。
+    'alipay_public_key' => env('ALIPAY_PUBLIC_KEY', ''),
+
+    // 网关地址
+    'gateway_url' => env('ALIPAY_GATEWAY_URL', 'https://openapi.alipay.com/gateway.do'),
+
+    // 异步通知地址
+    'notify_url' => env('ALIPAY_NOTIFY_URL', ''),
+
+    // 同步跳转
+    'return_url' => env('ALIPAY_RETURN_URL', ''),
+
+    // 编码格式
+    'charset' => env('ALIPAY_CHARSET', 'UTF-8'),
+
+    // 签名方式
+    'sign_type' => env('ALIPAY_SIGN_TYPE', 'RSA2'),
+
+    // 支付宝网关
+    'gatewayUrl' => env('ALIPAY_GATEWAY', 'https://openapi.alipay.com/gateway.do'),
+
+    // 支付宝沙箱网关
+    'gatewayUrlSandbox' => 'https://openapi.alipaydev.com/gateway.do',
+
+    // 是否使用沙箱环境
+    'sandbox' => env('ALIPAY_SANDBOX', false),
+
+    /*
+    |--------------------------------------------------------------------------
+    | 日志配置
+    |--------------------------------------------------------------------------
+    */
+    'log' => [
+        'file' => storage_path('logs/alipay.log'),
+        'level' => env('ALIPAY_LOG_LEVEL', 'debug'),
+        'type' => 'daily',
+        'max_file' => 30,
+    ],
+];