where('order_id', $orderId) ->whereIn('type', ['reward_direct', 'reward_umbrella']) ->count(); if ($exists > 0) { Log::info("[PointsReward] 订单 {$orderId} 已有积分奖励记录,跳过重复发放"); return; } $buyer = $this->userDao->get($orderUid); if (!$buyer || !$buyer['spread_uid']) { return; } $this->propagateReward($buyer['spread_uid'], $orderUid, $orderId, 0, 0, $orderDbId); } catch (\Throwable $e) { Log::error("[PointsReward] 积分奖励失败 orderUid={$orderUid} orderId={$orderId}: " . $e->getMessage()); } } /** * 向上递归发放级差积分 * * @param int $uid 当前被奖励用户 * @param int $fromUid 触发方(下级)用户 ID * @param string $orderId 来源订单号 * @param int $lowerReward 下级已获得的直推/伞下奖励积分(用于级差扣减) * @param int $depth 递归深度 */ private function propagateReward( int $uid, int $fromUid, string $orderId, int $lowerReward, int $depth = 0, int $orderDbId = 0 ): void { if ($depth >= 10 || $uid <= 0) { return; } $user = $this->userDao->get($uid); if (!$user) { return; } $agentLevelId = (int)($user['agent_level'] ?? 0); $grade = $this->agentLevelServices->getGradeByLevelId($agentLevelId); if ($grade === 0) { if ($user['spread_uid']) { $this->propagateReward((int)$user['spread_uid'], $uid, $orderId, 0, $depth + 1, $orderDbId); } return; } $isDirect = ($depth === 0); $reward = $isDirect ? $this->agentLevelServices->getDirectRewardPoints($agentLevelId) : $this->agentLevelServices->getUmbrellaRewardPoints($agentLevelId); $actual = max(0, $reward - $lowerReward); if ($actual > 0) { $this->grantFrozenPoints( $uid, $actual, $orderId, $isDirect ? 'reward_direct' : 'reward_umbrella', ($isDirect ? '直推奖励' : '伞下奖励(级差)') . " - 来源订单 {$orderId}", $orderDbId ); } if ($user['spread_uid']) { $this->propagateReward( (int)$user['spread_uid'], $uid, $orderId, $reward, $depth + 1, $orderDbId ); } } /** * 写入待释放积分(frozen_points)并记录明细 */ private function grantFrozenPoints( int $uid, int $points, string $orderId, string $type, string $mark, int $orderDbId = 0 ): void { Db::transaction(function () use ($uid, $points, $orderId, $type, $mark, $orderDbId) { $this->userDao->bcInc($uid, 'frozen_points', $points, 'uid'); $this->logDao->save([ 'uid' => $uid, 'points' => $points, 'pm' => 1, 'type' => $type, 'title' => ($type === 'reward_direct') ? '直推奖励' : '伞下奖励', 'mark' => $mark, 'status' => 'frozen', 'order_id' => $orderId, ]); // PRD §3.3:营销后台积分日志读 eb_user_bill,双写待释放积分明细(不增加可消费 integral) $integralBalance = (int)($this->userDao->value(['uid' => $uid], 'integral') ?: 0); $billType = ($type === 'reward_direct') ? 'hjf_reward_direct_integral' : 'hjf_reward_umbrella_integral'; $this->userBillServices->income($billType, $uid, $points, $integralBalance, $orderDbId, 0, $mark); }); } }