UrsTalentLogic.php 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  1. <?php
  2. namespace App\Module\UrsPromotion\Logics;
  3. use App\Module\UrsPromotion\Models\UrsUserReferral;
  4. use App\Module\UrsPromotion\Models\UrsUserTalent;
  5. use App\Module\UrsPromotion\Models\UrsTalentConfig;
  6. use App\Module\UrsPromotion\Enums\UrsPromotionRelationLevel;
  7. use Illuminate\Support\Facades\Log;
  8. /**
  9. * URS达人等级逻辑类
  10. *
  11. * 处理URS推广系统的达人等级计算和升级逻辑
  12. */
  13. class UrsTalentLogic
  14. {
  15. /**
  16. * 更新用户的团队统计和达人等级
  17. *
  18. * @param int $userId 用户ID
  19. * @return UrsUserTalent|null
  20. */
  21. public function updateUserTalent(int $userId): ?UrsUserTalent
  22. {
  23. try {
  24. // 计算团队统计数据
  25. $teamStats = $this->calculateTeamStats($userId);
  26. // 获取或创建用户达人记录
  27. $talent = UrsUserTalent::firstOrCreate(
  28. ['user_id' => $userId],
  29. [
  30. 'talent_level' => 0,
  31. 'direct_count' => 0,
  32. 'indirect_count' => 0,
  33. 'third_count' => 0,
  34. 'promotion_count' => 0,
  35. ]
  36. );
  37. // 更新团队统计(包含20代总人数)
  38. $talent->updateTeamStats(
  39. $teamStats['direct_count'],
  40. $teamStats['indirect_count'],
  41. $teamStats['third_count'],
  42. $teamStats['promotion_count'] // 传递20代总人数
  43. );
  44. // 计算新的达人等级
  45. $newLevel = $this->calculateTalentLevel(
  46. $teamStats['direct_count'],
  47. $teamStats['promotion_count']
  48. );
  49. // 检查是否需要升级
  50. if ($newLevel > $talent->talent_level) {
  51. $oldLevel = $talent->talent_level;
  52. $talent->upgradeTalentLevel($newLevel);
  53. Log::info("用户达人等级升级", [
  54. 'user_id' => $userId,
  55. 'old_level' => $oldLevel,
  56. 'new_level' => $newLevel,
  57. 'direct_count' => $teamStats['direct_count'],
  58. 'promotion_count' => $teamStats['promotion_count']
  59. ]);
  60. }
  61. $talent->save();
  62. return $talent;
  63. } catch (\Exception $e) {
  64. Log::error("更新用户达人等级失败", [
  65. 'user_id' => $userId,
  66. 'error' => $e->getMessage()
  67. ]);
  68. return null;
  69. }
  70. }
  71. /**
  72. * 计算用户的团队统计数据(支持20代统计)
  73. *
  74. * @param int $userId 用户ID
  75. * @return array
  76. */
  77. private function calculateTeamStats(int $userId): array
  78. {
  79. $stats = [
  80. 'direct_count' => 0,
  81. 'indirect_count' => 0,
  82. 'third_count' => 0,
  83. 'promotion_count' => 0,
  84. ];
  85. // 获取直推用户
  86. $directUsers = UrsUserReferral::where('referrer_id', $userId)
  87. ->where('status', UrsUserReferral::STATUS_VALID)
  88. ->pluck('user_id')
  89. ->toArray();
  90. $stats['direct_count'] = count($directUsers);
  91. if (empty($directUsers)) {
  92. return $stats;
  93. }
  94. // 获取间推用户(直推用户的直推)
  95. $indirectUsers = UrsUserReferral::whereIn('referrer_id', $directUsers)
  96. ->where('status', UrsUserReferral::STATUS_VALID)
  97. ->pluck('user_id')
  98. ->toArray();
  99. $stats['indirect_count'] = count($indirectUsers);
  100. if (empty($indirectUsers)) {
  101. $stats['promotion_count'] = $stats['direct_count'];
  102. return $stats;
  103. }
  104. // 获取三推用户(间推用户的直推)
  105. $thirdUsers = UrsUserReferral::whereIn('referrer_id', $indirectUsers)
  106. ->where('status', UrsUserReferral::STATUS_VALID)
  107. ->pluck('user_id')
  108. ->toArray();
  109. $stats['third_count'] = count($thirdUsers);
  110. // 计算20代总人数(使用递归方式统计所有层级)
  111. $allTeamUsers = $this->getAllTeamMembers($userId, UrsPromotionRelationLevel::getTeamStatsDepth());
  112. $stats['promotion_count'] = count($allTeamUsers);
  113. return $stats;
  114. }
  115. /**
  116. * 递归获取所有团队成员(支持多层级)
  117. *
  118. * @param int $userId 用户ID
  119. * @param int $maxLevels 最大层级数
  120. * @return array 所有团队成员的用户ID数组
  121. */
  122. private function getAllTeamMembers(int $userId, ?int $maxLevels = null): array
  123. {
  124. // 如果没有指定层级数,使用配置的团队统计深度
  125. if ($maxLevels === null) {
  126. $maxLevels = UrsPromotionRelationLevel::getTeamStatsDepth();
  127. }
  128. $allMembers = [];
  129. $currentLevelUsers = [$userId];
  130. for ($level = 1; $level <= $maxLevels; $level++) {
  131. $nextLevelUsers = [];
  132. foreach ($currentLevelUsers as $currentUserId) {
  133. $directReferrals = UrsUserReferral::where('referrer_id', $currentUserId)
  134. ->where('status', UrsUserReferral::STATUS_VALID)
  135. ->pluck('user_id')
  136. ->toArray();
  137. $nextLevelUsers = array_merge($nextLevelUsers, $directReferrals);
  138. }
  139. if (empty($nextLevelUsers)) {
  140. break; // 没有下级了
  141. }
  142. $allMembers = array_merge($allMembers, $nextLevelUsers);
  143. $currentLevelUsers = $nextLevelUsers;
  144. }
  145. return array_unique($allMembers);
  146. }
  147. /**
  148. * 根据团队数据计算达人等级
  149. *
  150. * @param int $directCount 直推人数
  151. * @param int $promotionCount 团队总人数
  152. * @return int
  153. */
  154. private function calculateTalentLevel(int $directCount, int $promotionCount): int
  155. {
  156. // 获取所有启用的达人等级配置,按等级倒序
  157. $configs = UrsTalentConfig::where('status', UrsTalentConfig::STATUS_ENABLED)
  158. ->orderBy('level', 'desc')
  159. ->get();
  160. foreach ($configs as $config) {
  161. if ($config->meetsRequirements($directCount, $promotionCount)) {
  162. return $config->level;
  163. }
  164. }
  165. return 0; // 默认为非达人
  166. }
  167. /**
  168. * 批量更新推荐人的达人等级
  169. *
  170. * @param array $userIds 用户ID数组
  171. * @return array 更新结果
  172. */
  173. public function batchUpdateTalents(array $userIds): array
  174. {
  175. $results = [
  176. 'success' => 0,
  177. 'failed' => 0,
  178. 'upgraded' => [],
  179. ];
  180. foreach ($userIds as $userId) {
  181. $talent = $this->updateUserTalent($userId);
  182. if ($talent) {
  183. $results['success']++;
  184. // 记录升级的用户
  185. if ($talent->wasChanged('talent_level')) {
  186. $results['upgraded'][] = [
  187. 'user_id' => $userId,
  188. 'old_level' => $talent->getOriginal('talent_level'),
  189. 'new_level' => $talent->talent_level,
  190. ];
  191. }
  192. } else {
  193. $results['failed']++;
  194. }
  195. }
  196. return $results;
  197. }
  198. /**
  199. * 获取用户的推荐关系树(三代)
  200. *
  201. * @param int $userId 用户ID
  202. * @return array
  203. */
  204. public function getUserReferralTree(int $userId): array
  205. {
  206. $tree = [
  207. 'user_id' => $userId,
  208. 'direct' => [],
  209. 'indirect' => [],
  210. 'third' => [],
  211. ];
  212. // 获取直推用户
  213. $directUsers = UrsUserReferral::where('referrer_id', $userId)
  214. ->where('status', UrsUserReferral::STATUS_VALID)
  215. ->with('user')
  216. ->get();
  217. foreach ($directUsers as $referral) {
  218. $tree['direct'][] = [
  219. 'user_id' => $referral->user_id,
  220. 'referral_time' => $referral->referral_time,
  221. 'user' => $referral->user,
  222. ];
  223. }
  224. // 获取间推用户
  225. if (!empty($tree['direct'])) {
  226. $directUserIds = array_column($tree['direct'], 'user_id');
  227. $indirectUsers = UrsUserReferral::whereIn('referrer_id', $directUserIds)
  228. ->where('status', UrsUserReferral::STATUS_VALID)
  229. ->with('user')
  230. ->get();
  231. foreach ($indirectUsers as $referral) {
  232. $tree['indirect'][] = [
  233. 'user_id' => $referral->user_id,
  234. 'referrer_id' => $referral->referrer_id,
  235. 'referral_time' => $referral->referral_time,
  236. 'user' => $referral->user,
  237. ];
  238. }
  239. }
  240. // 获取三推用户
  241. if (!empty($tree['indirect'])) {
  242. $indirectUserIds = array_column($tree['indirect'], 'user_id');
  243. $thirdUsers = UrsUserReferral::whereIn('referrer_id', $indirectUserIds)
  244. ->where('status', UrsUserReferral::STATUS_VALID)
  245. ->with('user')
  246. ->get();
  247. foreach ($thirdUsers as $referral) {
  248. $tree['third'][] = [
  249. 'user_id' => $referral->user_id,
  250. 'referrer_id' => $referral->referrer_id,
  251. 'referral_time' => $referral->referral_time,
  252. 'user' => $referral->user,
  253. ];
  254. }
  255. }
  256. return $tree;
  257. }
  258. /**
  259. * 获取达人等级配置列表
  260. *
  261. * @return array
  262. */
  263. public function getTalentConfigs(): array
  264. {
  265. return UrsTalentConfig::where('status', UrsTalentConfig::STATUS_ENABLED)
  266. ->orderBy('sort_order')
  267. ->get()
  268. ->toArray();
  269. }
  270. /**
  271. * 检查用户是否可以升级到指定等级
  272. *
  273. * @param int $userId 用户ID
  274. * @param int $targetLevel 目标等级
  275. * @return bool
  276. */
  277. public function canUpgradeToLevel(int $userId, int $targetLevel): bool
  278. {
  279. $talent = UrsUserTalent::where('user_id', $userId)->first();
  280. if (!$talent) {
  281. return false;
  282. }
  283. $config = UrsTalentConfig::where('level', $targetLevel)
  284. ->where('status', UrsTalentConfig::STATUS_ENABLED)
  285. ->first();
  286. if (!$config) {
  287. return false;
  288. }
  289. return $config->meetsRequirements($talent->direct_count, $talent->promotion_count);
  290. }
  291. }