UrsReferralService.php 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328
  1. <?php
  2. namespace App\Module\UrsPromotion\Services;
  3. use App\Module\UrsPromotion\Enums\UrsPromotionRelationLevel;
  4. use App\Module\UrsPromotion\Models\UrsUserMapping;
  5. use App\Module\UrsPromotion\Models\UrsUserReferral;
  6. use App\Module\UrsPromotion\Models\UrsUserTalent;
  7. use Illuminate\Support\Facades\DB;
  8. use Illuminate\Support\Facades\Log;
  9. /**
  10. * URS推荐关系服务
  11. *
  12. * 管理URS用户之间的推荐关系
  13. */
  14. class UrsReferralService
  15. {
  16. /**
  17. * 获取用户的推荐人
  18. *
  19. * @param int $ursUserId URS用户ID
  20. * @return int|null 推荐人URS用户ID,如果没有推荐人返回null
  21. */
  22. public static function getReferrer(int $ursUserId): ?int
  23. {
  24. $referral = UrsUserReferral::where('urs_user_id', $ursUserId)
  25. ->where('status', UrsUserReferral::STATUS_VALID)
  26. ->first();
  27. return $referral ? $referral->urs_referrer_id : null;
  28. }
  29. /**
  30. * 获取用户的直推下级列表
  31. *
  32. * @param int $ursUserId URS用户ID
  33. * @return array 直推下级URS用户ID数组
  34. */
  35. public static function getDirectReferrals(int $ursUserId): array
  36. {
  37. return UrsUserReferral::where('urs_referrer_id', $ursUserId)
  38. ->where('status', UrsUserReferral::STATUS_VALID)
  39. ->pluck('urs_user_id')
  40. ->toArray();
  41. }
  42. /**
  43. * 获取用户的推荐关系链(向上查找)
  44. *
  45. * @param int $ursUserId URS用户ID
  46. * @param int $maxLevels 最大层级数
  47. * @return array 推荐关系链,按层级排序 [level => urs_user_id]
  48. */
  49. public static function getReferralChain(int $ursUserId, int $maxLevels = 10): array
  50. {
  51. $chain = [];
  52. $currentUserId = $ursUserId;
  53. for ($level = 1; $level <= $maxLevels; $level++) {
  54. $referrerId = self::getReferrer($currentUserId);
  55. if (!$referrerId) {
  56. break; // 没有上级了
  57. }
  58. $chain[$level] = $referrerId;
  59. $currentUserId = $referrerId;
  60. }
  61. return $chain;
  62. }
  63. /**
  64. * 获取用户的团队成员(向下查找)
  65. *
  66. * @param int $ursUserId URS用户ID
  67. * @param int $maxLevels 最大层级数(默认20代,支持达人等级统计)
  68. * @return array 团队成员,按层级分组 [level => [urs_user_ids]]
  69. */
  70. public static function getTeamMembers(int $ursUserId, ?int $maxLevels = null): array
  71. {
  72. // 如果没有指定层级数,使用配置的团队统计深度
  73. if ($maxLevels === null) {
  74. $maxLevels = UrsPromotionRelationLevel::getTeamStatsDepth();
  75. }
  76. $team = [];
  77. $currentLevelUsers = [$ursUserId];
  78. for ($level = 1; $level <= $maxLevels; $level++) {
  79. $nextLevelUsers = [];
  80. foreach ($currentLevelUsers as $userId) {
  81. $directReferrals = self::getDirectReferrals($userId);
  82. $nextLevelUsers = array_merge($nextLevelUsers, $directReferrals);
  83. }
  84. if (empty($nextLevelUsers)) {
  85. break; // 没有下级了
  86. }
  87. $team[$level] = $nextLevelUsers;
  88. $currentLevelUsers = $nextLevelUsers;
  89. }
  90. return $team;
  91. }
  92. /**
  93. * 更新推荐人的团队统计
  94. *
  95. * @param int $ursReferrerId URS推荐人ID
  96. */
  97. private static function updateReferrerStats(int $ursReferrerId): void
  98. {
  99. $user_id = UrsUserMapping::getFarmUserIdByUrsUserId($ursReferrerId);
  100. // 获取团队成员统计(使用配置的团队统计深度)
  101. $teamMembers = self::getTeamMembers($ursReferrerId);
  102. $directCount = count($teamMembers[1] ?? []);
  103. $indirectCount = count($teamMembers[2] ?? []);
  104. $thirdCount = count($teamMembers[3] ?? []);
  105. // 计算20代总人数
  106. $totalCount = 0;
  107. foreach ($teamMembers as $members) {
  108. $totalCount += count($members);
  109. }
  110. // 获取或创建用户达人记录
  111. $talent = UrsUserTalent::firstOrCreate(
  112. ['user_id' => $user_id],
  113. [
  114. 'talent_level' => 0,
  115. 'direct_count' => 0,
  116. 'indirect_count' => 0,
  117. 'third_count' => 0,
  118. 'promotion_count' => 0,
  119. ]
  120. );
  121. // 使用模型方法更新团队统计,确保逻辑一致性
  122. $talent->updateTeamStats($directCount, $indirectCount, $thirdCount, $totalCount);
  123. $talent->save();
  124. }
  125. /**
  126. * 批量更新团队统计
  127. *
  128. * @param array $ursUserIds URS用户ID数组
  129. */
  130. public static function batchUpdateTeamStats(array $ursUserIds): void
  131. {
  132. foreach ($ursUserIds as $ursUserId) {
  133. self::updateReferrerStats($ursUserId);
  134. }
  135. }
  136. /**
  137. * 获取推荐关系统计
  138. *
  139. * @param int $ursUserId URS用户ID
  140. * @return array
  141. */
  142. public static function getReferralStats(int $ursUserId): array
  143. {
  144. $teamMembers = self::getTeamMembers($ursUserId);
  145. $referralChain = self::getReferralChain($ursUserId);
  146. return [
  147. 'urs_user_id' => $ursUserId,
  148. 'referrer_id' => self::getReferrer($ursUserId),
  149. 'referrer_level' => count($referralChain),
  150. 'direct_count' => count($teamMembers[1] ?? []),
  151. 'indirect_count' => count($teamMembers[2] ?? []),
  152. 'third_count' => count($teamMembers[3] ?? []),
  153. 'total_team_count' => array_sum(array_map('count', $teamMembers)),
  154. 'team_members' => $teamMembers,
  155. 'referral_chain' => $referralChain,
  156. ];
  157. }
  158. /**
  159. * 验证推荐关系的有效性
  160. *
  161. * @param int $ursUserId URS用户ID
  162. * @return bool
  163. */
  164. public static function validateReferral(int $ursUserId): bool
  165. {
  166. $referral = UrsUserReferral::where('urs_user_id', $ursUserId)->first();
  167. return $referral && $referral->isValid();
  168. }
  169. /**
  170. * 禁用推荐关系
  171. *
  172. * @param int $ursUserId URS用户ID
  173. * @return bool
  174. */
  175. public static function disableReferral(int $ursUserId): bool
  176. {
  177. try {
  178. $referral = UrsUserReferral::where('urs_user_id', $ursUserId)->first();
  179. if (!$referral) {
  180. return false;
  181. }
  182. $referral->setInvalid();
  183. $referral->save();
  184. // 更新推荐人的团队统计
  185. self::updateReferrerStats($referral->urs_referrer_id);
  186. Log::info('URS推荐关系已禁用', [
  187. 'urs_user_id' => $ursUserId,
  188. 'referral_id' => $referral->id
  189. ]);
  190. return true;
  191. } catch (\Exception $e) {
  192. Log::error('URS推荐关系禁用失败', [
  193. 'urs_user_id' => $ursUserId,
  194. 'error' => $e->getMessage()
  195. ]);
  196. return false;
  197. }
  198. }
  199. /**
  200. * 批量更新推荐关系表中的农场用户ID
  201. *
  202. * 当用户进入农场后,批量更新推荐关系表中的农场用户ID字段
  203. *
  204. * @return array 更新结果
  205. */
  206. public static function batchUpdateFarmUserIds(): array
  207. {
  208. try {
  209. Log::info('开始批量更新推荐关系表中的农场用户ID');
  210. // 使用SQL批量更新,提高性能
  211. $updateSql = "
  212. UPDATE `kku_urs_promotion_user_referrals` r
  213. LEFT JOIN `kku_urs_promotion_user_mappings` um ON r.urs_user_id = um.urs_user_id AND um.status = 1
  214. LEFT JOIN `kku_urs_promotion_user_mappings` rm ON r.urs_referrer_id = rm.urs_user_id AND rm.status = 1
  215. SET
  216. r.user_id = COALESCE(um.user_id, 0),
  217. r.referrer_id = COALESCE(rm.user_id, 0)
  218. WHERE r.status = 1
  219. ";
  220. $affectedRows = DB::update($updateSql);
  221. Log::info('批量更新推荐关系表中的农场用户ID完成', [
  222. 'affected_rows' => $affectedRows
  223. ]);
  224. return [
  225. 'success' => true,
  226. 'affected_rows' => $affectedRows,
  227. 'message' => "成功更新 {$affectedRows} 条推荐关系记录"
  228. ];
  229. } catch (\Exception $e) {
  230. Log::error('批量更新推荐关系表中的农场用户ID失败', [
  231. 'error' => $e->getMessage()
  232. ]);
  233. return [
  234. 'success' => false,
  235. 'affected_rows' => 0,
  236. 'message' => '批量更新失败: ' . $e->getMessage()
  237. ];
  238. }
  239. }
  240. /**
  241. * 更新单个推荐关系的农场用户ID
  242. *
  243. * 当特定用户进入农场时,更新其相关的推荐关系记录
  244. *
  245. * @param int $ursUserId URS用户ID
  246. * @param int $farmUserId 农场用户ID
  247. * @return bool
  248. */
  249. public static function updateFarmUserIdForUser(int $ursUserId, int $farmUserId): bool
  250. {
  251. try {
  252. DB::beginTransaction();
  253. // 更新该用户作为被推荐人的记录
  254. UrsUserReferral::where('urs_user_id', $ursUserId)
  255. ->where('status', UrsUserReferral::STATUS_VALID)
  256. ->update(['user_id' => $farmUserId]);
  257. // 更新该用户作为推荐人的记录
  258. UrsUserReferral::where('urs_referrer_id', $ursUserId)
  259. ->where('status', UrsUserReferral::STATUS_VALID)
  260. ->update(['referrer_id' => $farmUserId]);
  261. DB::commit();
  262. Log::info('更新用户推荐关系农场用户ID成功', [
  263. 'urs_user_id' => $ursUserId,
  264. 'farm_user_id' => $farmUserId
  265. ]);
  266. return true;
  267. } catch (\Exception $e) {
  268. DB::rollBack();
  269. Log::error('更新用户推荐关系农场用户ID失败', [
  270. 'urs_user_id' => $ursUserId,
  271. 'farm_user_id' => $farmUserId,
  272. 'error' => $e->getMessage()
  273. ]);
  274. return false;
  275. }
  276. }
  277. }