feat: 分销规则完善 + 微信支付修复

1. 分销规则:
   - 链接带ID绑定推荐关系
   - 一级分销 + 30天有效期
   - 客户抢夺机制(过期可被抢走)
   - 90%收益归分发者

2. 新增统计数据:
   - 绑定用户数
   - 链接进入人数
   - 带来付款人数

3. 微信支付:
   - 添加点击反馈
   - 优化支付流程日志
   - 改善错误提示

4. 分销中心UI优化
This commit is contained in:
卡若
2026-01-29 09:47:04 +08:00
parent 612b23c6c0
commit 6989ade3e2
8 changed files with 598 additions and 184 deletions

View File

@@ -1,18 +1,27 @@
/**
* 推荐码绑定API
* 用于处理分享带来的推荐关系绑定
* 推荐码绑定API - 增强版
*
* 核心规则:
* 1. 链接带ID谁发的链接进的人就绑谁
* 2. 一级、一月:只有一级分销;绑定有效期一个月
* 3. 长期不发:别人发得多,客户会被「抢走」
* 4. 每天发:持续发的人绑定一直有效,收益越来越高
* 5. 约90%给分发:谁发得多谁拿得多
*/
import { NextRequest, NextResponse } from 'next/server'
import { query } from '@/lib/db'
import { query, getConfig } from '@/lib/db'
// 绑定有效期(天)
const BINDING_DAYS = 30
/**
* POST - 绑定推荐关系
* POST - 绑定推荐关系(支持抢夺机制)
*/
export async function POST(request: NextRequest) {
try {
const body = await request.json()
const { userId, referralCode, openId } = body
const { userId, referralCode, openId, source } = body
// 验证参数
const effectiveUserId = userId || (openId ? `user_${openId.slice(-8)}` : null)
@@ -46,7 +55,7 @@ export async function POST(request: NextRequest) {
}, { status: 400 })
}
// 检查用户是否已有推荐人
// 检查用户是否存在
const users = await query(
'SELECT id, referred_by FROM users WHERE id = ? OR open_id = ?',
[effectiveUserId, openId || effectiveUserId]
@@ -60,46 +69,119 @@ export async function POST(request: NextRequest) {
}
const user = users[0]
const now = new Date()
if (user.referred_by) {
return NextResponse.json({
success: false,
error: '已绑定其他推荐人'
}, { status: 400 })
// 检查现有绑定关系
const existingBindings = await query(`
SELECT id, referrer_id, expiry_date, status
FROM referral_bindings
WHERE referee_id = ? AND status = 'active'
ORDER BY binding_date DESC LIMIT 1
`, [user.id]) as any[]
let action = 'new' // new=新绑定, renew=续期, takeover=抢夺
let oldReferrerId = null
if (existingBindings.length > 0) {
const existing = existingBindings[0]
const expiryDate = new Date(existing.expiry_date)
// 同一个推荐人 - 续期
if (existing.referrer_id === referrer.id) {
action = 'renew'
}
// 不同推荐人 - 检查是否可以抢夺
else if (expiryDate < now) {
// 已过期,可以被抢夺
action = 'takeover'
oldReferrerId = existing.referrer_id
// 将旧绑定标记为过期
await query(
"UPDATE referral_bindings SET status = 'expired' WHERE id = ?",
[existing.id]
)
} else {
// 未过期,不能被抢夺
return NextResponse.json({
success: false,
error: '用户已绑定其他推荐人,绑定有效期内无法更换',
expiryDate: expiryDate.toISOString()
}, { status: 400 })
}
}
// 绑定推荐关系
await query(
'UPDATE users SET referred_by = ? WHERE id = ?',
[referrer.id, user.id]
)
// 更新推荐人的推广数量
await query(
'UPDATE users SET referral_count = referral_count + 1 WHERE id = ?',
[referrer.id]
)
// 创建推荐绑定记录
const bindingId = 'bind_' + Date.now().toString(36) + Math.random().toString(36).substr(2, 6)
// 计算新的过期时间30天
const expiryDate = new Date()
expiryDate.setDate(expiryDate.getDate() + 30) // 30天有效期
expiryDate.setDate(expiryDate.getDate() + BINDING_DAYS)
try {
// 创建或更新绑定记录
const bindingId = 'bind_' + Date.now().toString(36) + Math.random().toString(36).substr(2, 6)
if (action === 'renew') {
// 续期:更新过期时间
await query(`
UPDATE referral_bindings
SET expiry_date = ?, binding_date = CURRENT_TIMESTAMP
WHERE referee_id = ? AND referrer_id = ? AND status = 'active'
`, [expiryDate, user.id, referrer.id])
console.log(`[Referral Bind] 续期: ${user.id} -> ${referrer.id}`)
} else {
// 新绑定或抢夺
await query(`
INSERT INTO referral_bindings (
id, referrer_id, referee_id, referral_code, status, expiry_date
) VALUES (?, ?, ?, ?, 'active', ?)
id, referrer_id, referee_id, referral_code, status, expiry_date, binding_date
) VALUES (?, ?, ?, ?, 'active', ?, CURRENT_TIMESTAMP)
ON DUPLICATE KEY UPDATE
referrer_id = VALUES(referrer_id),
referral_code = VALUES(referral_code),
expiry_date = VALUES(expiry_date),
binding_date = CURRENT_TIMESTAMP,
status = 'active'
`, [bindingId, referrer.id, user.id, referralCode, expiryDate])
} catch (e) {
console.log('[Referral Bind] 创建绑定记录失败(可能是重复绑定):', e)
// 更新用户的推荐人
await query(
'UPDATE users SET referred_by = ? WHERE id = ?',
[referrer.id, user.id]
)
// 更新推荐人的推广数量(仅新绑定时)
if (action === 'new') {
await query(
'UPDATE users SET referral_count = referral_count + 1 WHERE id = ?',
[referrer.id]
)
}
// 如果是抢夺,减少原推荐人的推广数量
if (action === 'takeover' && oldReferrerId) {
await query(
'UPDATE users SET referral_count = GREATEST(referral_count - 1, 0) WHERE id = ?',
[oldReferrerId]
)
console.log(`[Referral Bind] 抢夺: ${user.id}: ${oldReferrerId} -> ${referrer.id}`)
} else {
console.log(`[Referral Bind] 新绑定: ${user.id} -> ${referrer.id}`)
}
}
console.log(`[Referral Bind] 成功: ${user.id} -> ${referrer.id} (${referralCode})`)
// 记录访问日志(用于统计「通过链接进的人数」)
try {
await query(`
INSERT INTO referral_visits (referrer_id, visitor_id, source, created_at)
VALUES (?, ?, ?, CURRENT_TIMESTAMP)
`, [referrer.id, user.id, source || 'miniprogram'])
} catch (e) {
// 访问日志表可能不存在,忽略错误
}
return NextResponse.json({
success: true,
message: '绑定成功',
message: action === 'renew' ? '绑定已续期' : (action === 'takeover' ? '绑定已更新' : '绑定成功'),
action,
expiryDate: expiryDate.toISOString(),
referrer: {
id: referrer.id,
nickname: referrer.nickname