feat(payment): add TokenPay payment plugin
Register a new TokenPay payment plugin with configurable API credentials, payment URL generation, and signed callback verification. Also improve admin config fetching to support single-group lookups and add backwards-compatible subscribe template loading from legacy settings and bundled files when the database table is unavailable.
This commit is contained in:
@@ -72,12 +72,14 @@ class ConfigController extends Controller
|
|||||||
public function fetch(Request $request)
|
public function fetch(Request $request)
|
||||||
{
|
{
|
||||||
$key = $request->input('key');
|
$key = $request->input('key');
|
||||||
$configMappings = $this->getConfigMappings();
|
if ($key) {
|
||||||
if ($key && isset($configMappings[$key])) {
|
$configMapping = $this->getConfigMapping($key);
|
||||||
return $this->success([$key => $configMappings[$key]]);
|
if ($configMapping !== null) {
|
||||||
|
return $this->success([$key => $configMapping]);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->success($configMappings);
|
return $this->success($this->getConfigMappings());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -87,7 +89,36 @@ class ConfigController extends Controller
|
|||||||
*/
|
*/
|
||||||
private function getConfigMappings(): array
|
private function getConfigMappings(): array
|
||||||
{
|
{
|
||||||
return [
|
$keys = [
|
||||||
|
'invite',
|
||||||
|
'site',
|
||||||
|
'subscribe',
|
||||||
|
'frontend',
|
||||||
|
'server',
|
||||||
|
'email',
|
||||||
|
'telegram',
|
||||||
|
'app',
|
||||||
|
'safe',
|
||||||
|
'subscribe_template',
|
||||||
|
];
|
||||||
|
|
||||||
|
$configMappings = [];
|
||||||
|
foreach ($keys as $key) {
|
||||||
|
$configMappings[$key] = $this->getConfigMapping($key);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $configMappings;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取单个配置分组
|
||||||
|
*
|
||||||
|
* @param string $key 配置分组键名
|
||||||
|
* @return array|null 配置分组内容
|
||||||
|
*/
|
||||||
|
private function getConfigMapping(string $key): ?array
|
||||||
|
{
|
||||||
|
return match ($key) {
|
||||||
'invite' => [
|
'invite' => [
|
||||||
'invite_force' => (bool) admin_setting('invite_force', 0),
|
'invite_force' => (bool) admin_setting('invite_force', 0),
|
||||||
'invite_commission' => admin_setting('invite_commission', 10),
|
'invite_commission' => admin_setting('invite_commission', 10),
|
||||||
@@ -205,8 +236,9 @@ class ConfigController extends Controller
|
|||||||
'subscribe_template_stash' => subscribe_template('stash') ?? '',
|
'subscribe_template_stash' => subscribe_template('stash') ?? '',
|
||||||
'subscribe_template_surge' => subscribe_template('surge') ?? '',
|
'subscribe_template_surge' => subscribe_template('surge') ?? '',
|
||||||
'subscribe_template_surfboard' => subscribe_template('surfboard') ?? ''
|
'subscribe_template_surfboard' => subscribe_template('surfboard') ?? ''
|
||||||
]
|
],
|
||||||
];
|
default => null,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
public function save(ConfigSave $request)
|
public function save(ConfigSave $request)
|
||||||
|
|||||||
@@ -2,8 +2,16 @@
|
|||||||
|
|
||||||
namespace App\Models;
|
namespace App\Models;
|
||||||
|
|
||||||
|
use App\Protocols\Clash;
|
||||||
|
use App\Protocols\ClashMeta;
|
||||||
|
use App\Protocols\SingBox;
|
||||||
|
use App\Protocols\Stash;
|
||||||
|
use App\Protocols\Surfboard;
|
||||||
|
use App\Protocols\Surge;
|
||||||
use Illuminate\Database\Eloquent\Model;
|
use Illuminate\Database\Eloquent\Model;
|
||||||
|
use Illuminate\Database\QueryException;
|
||||||
use Illuminate\Support\Facades\Cache;
|
use Illuminate\Support\Facades\Cache;
|
||||||
|
use Illuminate\Support\Facades\File;
|
||||||
|
|
||||||
class SubscribeTemplate extends Model
|
class SubscribeTemplate extends Model
|
||||||
{
|
{
|
||||||
@@ -15,32 +23,131 @@ class SubscribeTemplate extends Model
|
|||||||
];
|
];
|
||||||
|
|
||||||
private static string $cachePrefix = 'subscribe_template:';
|
private static string $cachePrefix = 'subscribe_template:';
|
||||||
|
private static array $legacyTemplateFiles = [
|
||||||
|
'singbox' => [SingBox::CUSTOM_TEMPLATE_FILE, SingBox::DEFAULT_TEMPLATE_FILE],
|
||||||
|
'clash' => [Clash::CUSTOM_TEMPLATE_FILE, Clash::DEFAULT_TEMPLATE_FILE],
|
||||||
|
'clashmeta' => [
|
||||||
|
ClashMeta::CUSTOM_TEMPLATE_FILE,
|
||||||
|
ClashMeta::CUSTOM_CLASH_TEMPLATE_FILE,
|
||||||
|
ClashMeta::DEFAULT_TEMPLATE_FILE,
|
||||||
|
],
|
||||||
|
'stash' => [
|
||||||
|
Stash::CUSTOM_TEMPLATE_FILE,
|
||||||
|
Stash::CUSTOM_CLASH_TEMPLATE_FILE,
|
||||||
|
Stash::DEFAULT_TEMPLATE_FILE,
|
||||||
|
],
|
||||||
|
'surge' => [Surge::CUSTOM_TEMPLATE_FILE, Surge::DEFAULT_TEMPLATE_FILE],
|
||||||
|
'surfboard' => [Surfboard::CUSTOM_TEMPLATE_FILE, Surfboard::DEFAULT_TEMPLATE_FILE],
|
||||||
|
];
|
||||||
|
|
||||||
public static function getContent(string $name): ?string
|
public static function getContent(string $name): ?string
|
||||||
{
|
{
|
||||||
$cacheKey = self::$cachePrefix . $name;
|
$cacheKey = self::$cachePrefix . $name;
|
||||||
|
|
||||||
return Cache::store('redis')->remember($cacheKey, 3600, function () use ($name) {
|
return Cache::store('redis')->remember($cacheKey, 3600, function () use ($name) {
|
||||||
return self::where('name', $name)->value('content');
|
return self::resolveContent($name);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function setContent(string $name, ?string $content): void
|
public static function setContent(string $name, ?string $content): void
|
||||||
{
|
{
|
||||||
|
try {
|
||||||
self::updateOrCreate(
|
self::updateOrCreate(
|
||||||
['name' => $name],
|
['name' => $name],
|
||||||
['content' => $content]
|
['content' => $content]
|
||||||
);
|
);
|
||||||
|
} catch (QueryException) {
|
||||||
|
admin_setting([self::legacySettingKey($name) => $content]);
|
||||||
|
}
|
||||||
Cache::store('redis')->forget(self::$cachePrefix . $name);
|
Cache::store('redis')->forget(self::$cachePrefix . $name);
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function getAllContents(): array
|
public static function getAllContents(): array
|
||||||
{
|
{
|
||||||
return self::pluck('content', 'name')->toArray();
|
try {
|
||||||
|
$contents = self::pluck('content', 'name')->toArray();
|
||||||
|
return $contents ?: self::getLegacyContents();
|
||||||
|
} catch (QueryException) {
|
||||||
|
return self::getLegacyContents();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public static function flushCache(string $name): void
|
public static function flushCache(string $name): void
|
||||||
{
|
{
|
||||||
Cache::store('redis')->forget(self::$cachePrefix . $name);
|
Cache::store('redis')->forget(self::$cachePrefix . $name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Resolve template content with backwards-compatible fallbacks.
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @return string|null
|
||||||
|
*/
|
||||||
|
private static function resolveContent(string $name): ?string
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
$template = self::query()
|
||||||
|
->select('content')
|
||||||
|
->where('name', $name)
|
||||||
|
->first();
|
||||||
|
|
||||||
|
if ($template !== null) {
|
||||||
|
return $template->content;
|
||||||
|
}
|
||||||
|
} catch (QueryException) {
|
||||||
|
// Fall back to legacy sources when the new table is unavailable.
|
||||||
|
}
|
||||||
|
|
||||||
|
return self::getLegacyContent($name);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Collect all legacy template contents by protocol name.
|
||||||
|
*
|
||||||
|
* @return array<string, string>
|
||||||
|
*/
|
||||||
|
private static function getLegacyContents(): array
|
||||||
|
{
|
||||||
|
$contents = [];
|
||||||
|
|
||||||
|
foreach (array_keys(self::$legacyTemplateFiles) as $name) {
|
||||||
|
$contents[$name] = self::getLegacyContent($name) ?? '';
|
||||||
|
}
|
||||||
|
|
||||||
|
return $contents;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Resolve template content from legacy settings and bundled files.
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @return string|null
|
||||||
|
*/
|
||||||
|
private static function getLegacyContent(string $name): ?string
|
||||||
|
{
|
||||||
|
$settingValue = admin_setting(self::legacySettingKey($name));
|
||||||
|
if ($settingValue !== null && $settingValue !== '') {
|
||||||
|
return $settingValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (self::$legacyTemplateFiles[$name] ?? [] as $file) {
|
||||||
|
$path = base_path($file);
|
||||||
|
if (File::exists($path)) {
|
||||||
|
return File::get($path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build the legacy settings key for a template name.
|
||||||
|
*
|
||||||
|
* @param string $name
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
private static function legacySettingKey(string $name): string
|
||||||
|
{
|
||||||
|
return "subscribe_template_{$name}";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,3 +6,5 @@
|
|||||||
!Epay
|
!Epay
|
||||||
!Mgate
|
!Mgate
|
||||||
!Telegram
|
!Telegram
|
||||||
|
!TokenPay/
|
||||||
|
!TokenPay/**
|
||||||
|
|||||||
@@ -0,0 +1,123 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Plugin\TokenPay;
|
||||||
|
|
||||||
|
use App\Contracts\PaymentInterface;
|
||||||
|
use App\Exceptions\ApiException;
|
||||||
|
use App\Services\Plugin\AbstractPlugin;
|
||||||
|
use Curl\Curl;
|
||||||
|
|
||||||
|
class Plugin extends AbstractPlugin implements PaymentInterface
|
||||||
|
{
|
||||||
|
private const FIXED_RETURN_URL = 'https://www.spkun.com/dashboard/finance/orders';
|
||||||
|
|
||||||
|
public function boot(): void
|
||||||
|
{
|
||||||
|
$this->filter('available_payment_methods', function ($methods) {
|
||||||
|
if ($this->getConfig('enabled', true)) {
|
||||||
|
$methods['TokenPay'] = [
|
||||||
|
'name' => $this->getConfig('display_name', 'TokenPay'),
|
||||||
|
'icon' => $this->getConfig('icon', '🪙'),
|
||||||
|
'plugin_code' => $this->getPluginCode(),
|
||||||
|
'type' => 'plugin'
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $methods;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
public function form(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'token_pay_url' => [
|
||||||
|
'label' => 'API 地址',
|
||||||
|
'type' => 'string',
|
||||||
|
'required' => true,
|
||||||
|
'description' => '您的 TokenPay API 接口地址,例如:https://token-pay.xxx.com'
|
||||||
|
],
|
||||||
|
'token_pay_apitoken' => [
|
||||||
|
'label' => 'API Token',
|
||||||
|
'type' => 'string',
|
||||||
|
'required' => true,
|
||||||
|
'description' => '您的 TokenPay API Token'
|
||||||
|
],
|
||||||
|
'token_pay_currency' => [
|
||||||
|
'label' => '币种',
|
||||||
|
'type' => 'string',
|
||||||
|
'required' => true,
|
||||||
|
'description' => '您的 TokenPay 币种,如 USDT_TRC20、TRX'
|
||||||
|
]
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function pay($order): array
|
||||||
|
{
|
||||||
|
$params = [
|
||||||
|
'ActualAmount' => $order['total_amount'] / 100,
|
||||||
|
'OutOrderId' => $order['trade_no'],
|
||||||
|
'OrderUserKey' => (string) $order['user_id'],
|
||||||
|
'Currency' => $this->getConfig('token_pay_currency'),
|
||||||
|
'RedirectUrl' => self::FIXED_RETURN_URL,
|
||||||
|
'NotifyUrl' => $order['notify_url'],
|
||||||
|
];
|
||||||
|
|
||||||
|
ksort($params);
|
||||||
|
$str = stripslashes(urldecode(http_build_query($params))) . $this->getConfig('token_pay_apitoken');
|
||||||
|
$params['Signature'] = md5($str);
|
||||||
|
|
||||||
|
$curl = new Curl();
|
||||||
|
$curl->setUserAgent('TokenPay');
|
||||||
|
$curl->setOpt(CURLOPT_SSL_VERIFYPEER, 0);
|
||||||
|
$curl->setOpt(CURLOPT_HTTPHEADER, ['Content-Type: application/json']);
|
||||||
|
$curl->post(rtrim((string) $this->getConfig('token_pay_url'), '/') . '/CreateOrder', json_encode($params));
|
||||||
|
|
||||||
|
$result = $curl->response;
|
||||||
|
$error = $curl->error ? ($curl->errorMessage ?: 'TokenPay request failed') : null;
|
||||||
|
$curl->close();
|
||||||
|
|
||||||
|
if ($error) {
|
||||||
|
throw new ApiException($error);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isset($result->success) || !$result->success) {
|
||||||
|
$message = isset($result->message) ? (string) $result->message : 'Failed to create TokenPay order';
|
||||||
|
throw new ApiException($message);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isset($result->data) || !is_string($result->data) || $result->data === '') {
|
||||||
|
throw new ApiException('TokenPay payment url is invalid');
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
'type' => 1,
|
||||||
|
'data' => $result->data
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
public function notify($params): array|bool
|
||||||
|
{
|
||||||
|
if (!isset($params['Signature'])) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$sign = $params['Signature'];
|
||||||
|
unset($params['Signature']);
|
||||||
|
ksort($params);
|
||||||
|
$str = stripslashes(urldecode(http_build_query($params))) . $this->getConfig('token_pay_apitoken');
|
||||||
|
|
||||||
|
if ($sign !== md5($str)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!isset($params['Status']) || (int) $params['Status'] !== 1) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
'trade_no' => $params['OutOrderId'] ?? '',
|
||||||
|
'callback_no' => $params['Id'] ?? '',
|
||||||
|
'custom_result' => 'ok'
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
"name": "TokenPay",
|
||||||
|
"code": "token_pay",
|
||||||
|
"type": "payment",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"description": "TokenPay payment plugin",
|
||||||
|
"author": "Micah123321"
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user