fix(fsgx): 直推积分奖励链路校验(问题2)

- 买家直推人非分销员时中断直推级差链,避免祖先越级拿 reward_direct
- 同步更新 fsgx-issues-0330 文档问题2用例说明

Made-with: Cursor
This commit is contained in:
apple
2026-03-31 15:25:48 +08:00
parent 583105e23a
commit 1c0cf5204f
2 changed files with 65 additions and 30 deletions

View File

@@ -82,14 +82,20 @@ class PointsRewardServices extends BaseServices
* max(0, 自身 direct_reward_points - 链中已见最高 direct_reward_points) * qty
* 非分销员grade=0跳过但继续向上传递$lowerDirectReward 不变。
*
* @param int $uid 当前被奖励用户
* @param int $fromUid 触发方(下级)用户 ID
* @param string $orderId 来源订单号
* @param int $lowerDirectReward 链中已被下级拿走的最高 direct_reward_points级差下限
* @param int $depth 递归深度(防止无限递归)
* @param int $orderDbId 订单表主键 id
* @param array $preUpgradeLevels 升级前各用户的 agent_level 快照 [uid => level_id]B2
* @param int $qty 报单商品数量积分按数量倍乘B3
* 直推关系约束($directCascadeActive
* 当买家直接推荐人depth=0不是分销员grade=0直推级差链终止
* 其所有上级祖先不得获得 reward_direct不存在直推关系
* 伞下奖励reward_umbrella不受此约束仍按自身开关独立发放。
*
* @param int $uid 当前被奖励用户
* @param int $fromUid 触发方(下级)用户 ID
* @param string $orderId 来源订单号
* @param int $lowerDirectReward 链中已被下级拿走的最高 direct_reward_points级差下限
* @param int $depth 递归深度(防止无限递归)
* @param int $orderDbId 订单表主键 id
* @param array $preUpgradeLevels 升级前各用户的 agent_level 快照 [uid => level_id]B2
* @param int $qty 报单商品数量积分按数量倍乘B3
* @param bool $directCascadeActive 直推级差链是否仍然有效
*/
private function propagateReward(
int $uid,
@@ -99,7 +105,8 @@ class PointsRewardServices extends BaseServices
int $depth = 0,
int $orderDbId = 0,
array $preUpgradeLevels = [],
int $qty = 1
int $qty = 1,
bool $directCascadeActive = true
): void {
if ($depth >= 10 || $uid <= 0) {
return;
@@ -117,33 +124,50 @@ class PointsRewardServices extends BaseServices
$grade = $this->agentLevelServices->getGradeByLevelId($agentLevelId);
if ($grade === 0) {
// 非分销员:跳过奖励,继续向上传递$lowerDirectReward 保持不变
// 非分销员:跳过奖励,继续向上传递
// 若当前节点是买家的直接推荐人depth=0直推级差链从此中断
// 上级祖先与买家之间没有直推关系,不应获得 reward_direct
$nextCascadeActive = ($depth === 0) ? false : $directCascadeActive;
if ($user['spread_uid']) {
$this->propagateReward((int)$user['spread_uid'], $uid, $orderId, $lowerDirectReward, $depth + 1, $orderDbId, $preUpgradeLevels, $qty);
$this->propagateReward((int)$user['spread_uid'], $uid, $orderId, $lowerDirectReward, $depth + 1, $orderDbId, $preUpgradeLevels, $qty, $nextCascadeActive);
}
return;
}
$directReward = $this->agentLevelServices->getDirectRewardPoints($agentLevelId);
// 级差:本节点只拿超出下级已取走部分的差额,再乘以数量
// 间接上级depth>0即非直推父节点受伞下奖励开关控制hjf_umbrella_reward_enable默认关闭
if ($depth > 0) {
$umbrellaRewardEnable = (int)sys_config('hjf_umbrella_reward_enable', 0);
$actual = $umbrellaRewardEnable ? max(0, $directReward - $lowerDirectReward) * $qty : 0;
} else {
// 直推级差:仅在直推链有效时发放(买家直接推荐人必须是分销员)
if ($directCascadeActive) {
$actual = max(0, $directReward - $lowerDirectReward) * $qty;
if ($actual > 0) {
$this->grantFrozenPoints(
$uid,
$actual,
$orderId,
'reward_direct',
'直推奖励(级差)' . " x{$qty} - 来源订单 {$orderId}",
$orderDbId
);
}
}
if ($actual > 0) {
$this->grantFrozenPoints(
$uid,
$actual,
$orderId,
'reward_direct',
'直推奖励(级差)' . " x{$qty} - 来源订单 {$orderId}",
$orderDbId
);
// 伞下积分奖励独立逻辑仅对间接上级depth>0生效受开关控制
// 使用 umbrella_reward_points 平额发放(非级差),各等级独立拿自身额度
if ($depth > 0) {
$umbrellaRewardEnable = (int)sys_config('hjf_umbrella_reward_enable', 0);
if ($umbrellaRewardEnable) {
$umbrellaPoints = $this->agentLevelServices->getUmbrellaRewardPoints($agentLevelId);
if ($umbrellaPoints > 0) {
$this->grantFrozenPoints(
$uid,
$umbrellaPoints * $qty,
$orderId,
'reward_umbrella',
'伞下奖励' . " x{$qty} - 来源订单 {$orderId}",
$orderDbId
);
}
}
}
// 向上传递时,下限取当前节点与已有下限中的较大值,确保上级只拿增量
@@ -157,7 +181,8 @@ class PointsRewardServices extends BaseServices
$depth + 1,
$orderDbId,
$preUpgradeLevels,
$qty
$qty,
$directCascadeActive
);
}
}