2026-03-01 03:04:24 +08:00
|
|
|
|
import { z } from 'zod';
|
2026-03-06 15:11:47 +08:00
|
|
|
|
import fs from 'fs';
|
2026-03-01 03:04:24 +08:00
|
|
|
|
|
|
|
|
|
|
const optionalTrimmedString = z.preprocess((value) => {
|
|
|
|
|
|
if (typeof value !== 'string') return value;
|
|
|
|
|
|
const trimmed = value.trim();
|
|
|
|
|
|
return trimmed === '' ? undefined : trimmed;
|
|
|
|
|
|
}, z.string().optional());
|
|
|
|
|
|
|
refactor: extract pay page components and migrate zpay → easypay
Components:
- Add PayPageLayout, OrderFilterBar, MobileOrderList, OrderTable, OrderSummaryCards
- Extract shared pay-utils (types, constants, helper functions)
- Simplify pay/page.tsx and orders/page.tsx
EasyPay migration:
- Remove src/lib/zpay/, api/zpay/notify, zpay test, zpay.md
- Simplify config.ts: single envSchema, no ZPAY_* fallback
- Rename DB field zpay_trade_no → payment_trade_no (migration added)
- Update OrderDetail label: ZPAY订单号 → 支付单号
- Update CLAUDE.md project structure
2026-03-01 15:55:43 +08:00
|
|
|
|
const envSchema = z.object({
|
2026-03-01 03:04:24 +08:00
|
|
|
|
DATABASE_URL: z.string().min(1),
|
|
|
|
|
|
|
|
|
|
|
|
SUB2API_BASE_URL: z.string().url(),
|
|
|
|
|
|
SUB2API_ADMIN_API_KEY: z.string().min(1),
|
|
|
|
|
|
|
2026-03-06 13:57:52 +08:00
|
|
|
|
// ── 支付服务商(显式声明启用哪些服务商,逗号分隔:easypay, alipay, wxpay, stripe) ──
|
2026-03-02 02:04:53 +08:00
|
|
|
|
PAYMENT_PROVIDERS: z
|
|
|
|
|
|
.string()
|
|
|
|
|
|
.default('')
|
2026-03-05 23:10:44 +08:00
|
|
|
|
.transform((v) =>
|
|
|
|
|
|
v
|
|
|
|
|
|
.split(',')
|
|
|
|
|
|
.map((s) => s.trim().toLowerCase())
|
|
|
|
|
|
.filter(Boolean),
|
|
|
|
|
|
),
|
2026-03-02 02:04:53 +08:00
|
|
|
|
|
|
|
|
|
|
// ── Easy-Pay(PAYMENT_PROVIDERS 含 easypay 时必填) ──
|
2026-03-01 17:58:08 +08:00
|
|
|
|
EASY_PAY_PID: optionalTrimmedString,
|
|
|
|
|
|
EASY_PAY_PKEY: optionalTrimmedString,
|
|
|
|
|
|
EASY_PAY_API_BASE: optionalTrimmedString,
|
|
|
|
|
|
EASY_PAY_NOTIFY_URL: optionalTrimmedString,
|
|
|
|
|
|
EASY_PAY_RETURN_URL: optionalTrimmedString,
|
2026-03-01 03:04:24 +08:00
|
|
|
|
EASY_PAY_CID: optionalTrimmedString,
|
|
|
|
|
|
EASY_PAY_CID_ALIPAY: optionalTrimmedString,
|
|
|
|
|
|
EASY_PAY_CID_WXPAY: optionalTrimmedString,
|
|
|
|
|
|
|
2026-03-05 01:52:59 +08:00
|
|
|
|
// ── 支付宝直连(PAYMENT_PROVIDERS 含 alipay 时必填) ──
|
2026-03-06 15:11:47 +08:00
|
|
|
|
// 支持直接传密钥内容,也支持传文件路径(自动读取)
|
2026-03-05 01:48:10 +08:00
|
|
|
|
ALIPAY_APP_ID: optionalTrimmedString,
|
|
|
|
|
|
ALIPAY_PRIVATE_KEY: optionalTrimmedString,
|
|
|
|
|
|
ALIPAY_PUBLIC_KEY: optionalTrimmedString,
|
|
|
|
|
|
ALIPAY_NOTIFY_URL: optionalTrimmedString,
|
|
|
|
|
|
ALIPAY_RETURN_URL: optionalTrimmedString,
|
|
|
|
|
|
|
2026-03-06 13:57:52 +08:00
|
|
|
|
// ── 微信支付直连(PAYMENT_PROVIDERS 含 wxpay 时必填) ──
|
|
|
|
|
|
WXPAY_APP_ID: optionalTrimmedString,
|
|
|
|
|
|
WXPAY_MCH_ID: optionalTrimmedString,
|
|
|
|
|
|
WXPAY_PRIVATE_KEY: optionalTrimmedString,
|
|
|
|
|
|
WXPAY_CERT_SERIAL: optionalTrimmedString,
|
|
|
|
|
|
WXPAY_API_V3_KEY: optionalTrimmedString,
|
|
|
|
|
|
WXPAY_NOTIFY_URL: optionalTrimmedString,
|
|
|
|
|
|
WXPAY_PUBLIC_KEY: optionalTrimmedString,
|
|
|
|
|
|
WXPAY_PUBLIC_KEY_ID: optionalTrimmedString,
|
|
|
|
|
|
|
2026-03-02 02:04:53 +08:00
|
|
|
|
// ── Stripe(PAYMENT_PROVIDERS 含 stripe 时必填) ──
|
2026-03-01 17:58:08 +08:00
|
|
|
|
STRIPE_SECRET_KEY: optionalTrimmedString,
|
|
|
|
|
|
STRIPE_PUBLISHABLE_KEY: optionalTrimmedString,
|
|
|
|
|
|
STRIPE_WEBHOOK_SECRET: optionalTrimmedString,
|
|
|
|
|
|
|
refactor: extract pay page components and migrate zpay → easypay
Components:
- Add PayPageLayout, OrderFilterBar, MobileOrderList, OrderTable, OrderSummaryCards
- Extract shared pay-utils (types, constants, helper functions)
- Simplify pay/page.tsx and orders/page.tsx
EasyPay migration:
- Remove src/lib/zpay/, api/zpay/notify, zpay test, zpay.md
- Simplify config.ts: single envSchema, no ZPAY_* fallback
- Rename DB field zpay_trade_no → payment_trade_no (migration added)
- Update OrderDetail label: ZPAY订单号 → 支付单号
- Update CLAUDE.md project structure
2026-03-01 15:55:43 +08:00
|
|
|
|
ORDER_TIMEOUT_MINUTES: z.string().default('5').transform(Number).pipe(z.number().int().positive()),
|
|
|
|
|
|
MIN_RECHARGE_AMOUNT: z.string().default('1').transform(Number).pipe(z.number().positive()),
|
2026-03-01 19:41:44 +08:00
|
|
|
|
MAX_RECHARGE_AMOUNT: z.string().default('1000').transform(Number).pipe(z.number().positive()),
|
|
|
|
|
|
// 每日每用户最大累计充值额,0 = 不限制
|
2026-03-01 19:49:42 +08:00
|
|
|
|
MAX_DAILY_RECHARGE_AMOUNT: z.string().default('10000').transform(Number).pipe(z.number().min(0)),
|
2026-03-01 21:53:09 +08:00
|
|
|
|
|
2026-03-01 22:51:09 +08:00
|
|
|
|
// 每日各渠道全平台总限额,可选覆盖(0 = 不限制)。
|
|
|
|
|
|
// 未设置时由各 PaymentProvider.defaultLimits 提供默认值。
|
2026-03-05 23:10:44 +08:00
|
|
|
|
MAX_DAILY_AMOUNT_ALIPAY: z
|
|
|
|
|
|
.string()
|
|
|
|
|
|
.optional()
|
|
|
|
|
|
.transform((v) => (v !== undefined ? Number(v) : undefined))
|
|
|
|
|
|
.pipe(z.number().min(0).optional()),
|
2026-03-06 15:33:22 +08:00
|
|
|
|
MAX_DAILY_AMOUNT_ALIPAY_DIRECT: z
|
|
|
|
|
|
.string()
|
|
|
|
|
|
.optional()
|
|
|
|
|
|
.transform((v) => (v !== undefined ? Number(v) : undefined))
|
|
|
|
|
|
.pipe(z.number().min(0).optional()),
|
2026-03-05 23:10:44 +08:00
|
|
|
|
MAX_DAILY_AMOUNT_WXPAY: z
|
|
|
|
|
|
.string()
|
|
|
|
|
|
.optional()
|
|
|
|
|
|
.transform((v) => (v !== undefined ? Number(v) : undefined))
|
|
|
|
|
|
.pipe(z.number().min(0).optional()),
|
|
|
|
|
|
MAX_DAILY_AMOUNT_STRIPE: z
|
|
|
|
|
|
.string()
|
|
|
|
|
|
.optional()
|
|
|
|
|
|
.transform((v) => (v !== undefined ? Number(v) : undefined))
|
|
|
|
|
|
.pipe(z.number().min(0).optional()),
|
refactor: extract pay page components and migrate zpay → easypay
Components:
- Add PayPageLayout, OrderFilterBar, MobileOrderList, OrderTable, OrderSummaryCards
- Extract shared pay-utils (types, constants, helper functions)
- Simplify pay/page.tsx and orders/page.tsx
EasyPay migration:
- Remove src/lib/zpay/, api/zpay/notify, zpay test, zpay.md
- Simplify config.ts: single envSchema, no ZPAY_* fallback
- Rename DB field zpay_trade_no → payment_trade_no (migration added)
- Update OrderDetail label: ZPAY订单号 → 支付单号
- Update CLAUDE.md project structure
2026-03-01 15:55:43 +08:00
|
|
|
|
PRODUCT_NAME: z.string().default('Sub2API Balance Recharge'),
|
2026-03-01 03:04:24 +08:00
|
|
|
|
|
2026-03-13 23:03:01 +08:00
|
|
|
|
ADMIN_TOKEN: z.string().min(16),
|
2026-03-01 03:04:24 +08:00
|
|
|
|
|
|
|
|
|
|
NEXT_PUBLIC_APP_URL: z.string().url(),
|
2026-03-02 02:46:51 +08:00
|
|
|
|
PAY_HELP_IMAGE_URL: optionalTrimmedString,
|
|
|
|
|
|
PAY_HELP_TEXT: optionalTrimmedString,
|
2026-03-06 17:34:42 +08:00
|
|
|
|
|
|
|
|
|
|
// ── 支付方式前端描述(sublabel)覆盖,不设置则使用默认值 ──
|
|
|
|
|
|
PAYMENT_SUBLABEL_ALIPAY: optionalTrimmedString,
|
|
|
|
|
|
PAYMENT_SUBLABEL_ALIPAY_DIRECT: optionalTrimmedString,
|
|
|
|
|
|
PAYMENT_SUBLABEL_WXPAY: optionalTrimmedString,
|
|
|
|
|
|
PAYMENT_SUBLABEL_WXPAY_DIRECT: optionalTrimmedString,
|
|
|
|
|
|
PAYMENT_SUBLABEL_STRIPE: optionalTrimmedString,
|
2026-03-01 03:04:24 +08:00
|
|
|
|
});
|
|
|
|
|
|
|
refactor: extract pay page components and migrate zpay → easypay
Components:
- Add PayPageLayout, OrderFilterBar, MobileOrderList, OrderTable, OrderSummaryCards
- Extract shared pay-utils (types, constants, helper functions)
- Simplify pay/page.tsx and orders/page.tsx
EasyPay migration:
- Remove src/lib/zpay/, api/zpay/notify, zpay test, zpay.md
- Simplify config.ts: single envSchema, no ZPAY_* fallback
- Rename DB field zpay_trade_no → payment_trade_no (migration added)
- Update OrderDetail label: ZPAY订单号 → 支付单号
- Update CLAUDE.md project structure
2026-03-01 15:55:43 +08:00
|
|
|
|
export type Env = z.infer<typeof envSchema>;
|
2026-03-01 03:04:24 +08:00
|
|
|
|
|
|
|
|
|
|
let cachedEnv: Env | null = null;
|
|
|
|
|
|
|
2026-03-06 15:11:47 +08:00
|
|
|
|
/**
|
|
|
|
|
|
* 如果值看起来是文件路径且文件存在,则读取文件内容作为实际值;
|
|
|
|
|
|
* 否则直接返回原值。
|
|
|
|
|
|
*/
|
|
|
|
|
|
function resolveKeyValue(value: string | undefined): string | undefined {
|
|
|
|
|
|
if (!value) return undefined;
|
|
|
|
|
|
// 密钥内容不会以 / 或盘符开头,文件路径才会
|
|
|
|
|
|
if ((value.startsWith('/') || /^[A-Za-z]:[/\\]/.test(value)) && fs.existsSync(value)) {
|
|
|
|
|
|
try {
|
|
|
|
|
|
return fs.readFileSync(value, 'utf-8').trim();
|
|
|
|
|
|
} catch (err) {
|
|
|
|
|
|
throw new Error(`Failed to read key file ${value}: ${(err as Error).message}`);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
return value;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-01 03:04:24 +08:00
|
|
|
|
export function getEnv(): Env {
|
|
|
|
|
|
if (cachedEnv) return cachedEnv;
|
|
|
|
|
|
|
refactor: extract pay page components and migrate zpay → easypay
Components:
- Add PayPageLayout, OrderFilterBar, MobileOrderList, OrderTable, OrderSummaryCards
- Extract shared pay-utils (types, constants, helper functions)
- Simplify pay/page.tsx and orders/page.tsx
EasyPay migration:
- Remove src/lib/zpay/, api/zpay/notify, zpay test, zpay.md
- Simplify config.ts: single envSchema, no ZPAY_* fallback
- Rename DB field zpay_trade_no → payment_trade_no (migration added)
- Update OrderDetail label: ZPAY订单号 → 支付单号
- Update CLAUDE.md project structure
2026-03-01 15:55:43 +08:00
|
|
|
|
const parsed = envSchema.safeParse(process.env);
|
2026-03-01 03:04:24 +08:00
|
|
|
|
if (!parsed.success) {
|
|
|
|
|
|
console.error('Invalid environment variables:', parsed.error.flatten().fieldErrors);
|
|
|
|
|
|
throw new Error('Invalid environment variables');
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-06 15:11:47 +08:00
|
|
|
|
const env = parsed.data;
|
|
|
|
|
|
|
|
|
|
|
|
// 支付宝密钥:支持直接传内容或传文件路径
|
|
|
|
|
|
env.ALIPAY_PRIVATE_KEY = resolveKeyValue(env.ALIPAY_PRIVATE_KEY);
|
|
|
|
|
|
env.ALIPAY_PUBLIC_KEY = resolveKeyValue(env.ALIPAY_PUBLIC_KEY);
|
|
|
|
|
|
|
2026-03-06 22:10:50 +08:00
|
|
|
|
// 微信支付密钥:支持直接传内容或传文件路径
|
|
|
|
|
|
env.WXPAY_PRIVATE_KEY = resolveKeyValue(env.WXPAY_PRIVATE_KEY);
|
|
|
|
|
|
env.WXPAY_PUBLIC_KEY = resolveKeyValue(env.WXPAY_PUBLIC_KEY);
|
|
|
|
|
|
|
2026-03-06 15:11:47 +08:00
|
|
|
|
cachedEnv = env;
|
2026-03-01 03:04:24 +08:00
|
|
|
|
return cachedEnv;
|
|
|
|
|
|
}
|