UrsRelationCacheLogic.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343
  1. <?php
  2. namespace App\Module\UrsPromotion\Logics;
  3. use App\Module\UrsPromotion\Enums\UrsPromotionRelationLevel;
  4. use App\Module\UrsPromotion\Models\UrsUserReferral;
  5. use App\Module\UrsPromotion\Models\UrsUserRelationCache;
  6. use App\Module\UrsPromotion\Services\UrsUserMappingService;
  7. use Illuminate\Support\Facades\DB;
  8. use Illuminate\Support\Facades\Log;
  9. use Illuminate\Support\Facades\Redis;
  10. /**
  11. * URS关系缓存逻辑类
  12. *
  13. * 处理URS用户关系缓存的核心业务逻辑,包括生成关系缓存、清理关系缓存、
  14. * 重建关系缓存等功能。该类仅供内部使用,不对外提供服务。
  15. */
  16. class UrsRelationCacheLogic
  17. {
  18. /**
  19. * 生成用户的关系缓存
  20. *
  21. * @param int $ursUserId URS用户ID
  22. * @return bool
  23. */
  24. public function generateUserRelationCache(int $ursUserId): bool
  25. {
  26. try {
  27. // 清除用户的现有关系缓存
  28. $this->clearUserRelationCache($ursUserId);
  29. // 获取用户的直接推荐人
  30. $referral = UrsUserReferral::where('urs_user_id', $ursUserId)
  31. ->where('status', UrsUserReferral::STATUS_VALID)
  32. ->first();
  33. if (!$referral) {
  34. Log::info("URS用户 {$ursUserId} 没有推荐人,无需生成关系缓存");
  35. return true;
  36. }
  37. $ursReferrerId = $referral->urs_referrer_id;
  38. // 获取农场用户ID
  39. $farmUserId = UrsUserMappingService::getFarmUserId($ursUserId);
  40. $farmReferrerId = UrsUserMappingService::getFarmUserId($ursReferrerId);
  41. // 如果农场用户ID为0,跳过缓存生成
  42. if ($farmUserId <= 0 || $farmReferrerId <= 0) {
  43. Log::info("URS用户 {$ursUserId} 或推荐人 {$ursReferrerId} 未进入农场,跳过缓存生成");
  44. return true;
  45. }
  46. // 创建直接关系缓存
  47. $directRelation = new UrsUserRelationCache();
  48. $directRelation->user_id = $farmUserId;
  49. $directRelation->related_user_id = $farmReferrerId;
  50. $directRelation->urs_user_id = $ursUserId;
  51. $directRelation->urs_related_user_id = $ursReferrerId;
  52. $directRelation->level = UrsPromotionRelationLevel::DIRECT;
  53. $directRelation->path = (string)$farmReferrerId;
  54. $directRelation->urs_path = (string)$ursReferrerId;
  55. $directRelation->depth = 1;
  56. $directRelation->save();
  57. // 获取推荐人的所有上级(限制在20代以内)
  58. $upperRelations = UrsUserRelationCache::where('urs_user_id', $ursReferrerId)
  59. ->where('depth', '<', UrsPromotionRelationLevel::getMaxLevel())
  60. ->get();
  61. // 创建间接关系缓存(最多20代)
  62. foreach ($upperRelations as $upperRelation) {
  63. $newDepth = $upperRelation->depth + 1;
  64. // 限制最大深度为20代
  65. if ($newDepth > UrsPromotionRelationLevel::getMaxLevel()) {
  66. continue;
  67. }
  68. $indirectRelation = new UrsUserRelationCache();
  69. $indirectRelation->user_id = $farmUserId;
  70. $indirectRelation->related_user_id = $upperRelation->related_user_id;
  71. $indirectRelation->urs_user_id = $ursUserId;
  72. $indirectRelation->urs_related_user_id = $upperRelation->urs_related_user_id;
  73. $indirectRelation->level = UrsPromotionRelationLevel::getLevelByDepth($newDepth);
  74. $indirectRelation->path = $farmReferrerId . ',' . $upperRelation->path;
  75. $indirectRelation->urs_path = $ursReferrerId . ',' . $upperRelation->urs_path;
  76. $indirectRelation->depth = $newDepth;
  77. $indirectRelation->save();
  78. }
  79. // 清除Redis缓存
  80. Redis::del("urs_promotion:user:{$ursUserId}:all_referrers");
  81. Redis::del("urs_promotion:farm_user:{$farmUserId}:all_referrers");
  82. Log::info("URS用户 {$ursUserId} 关系缓存生成成功");
  83. return true;
  84. } catch (\Exception $e) {
  85. Log::error("生成URS用户关系缓存失败", [
  86. 'urs_user_id' => $ursUserId,
  87. 'error' => $e->getMessage()
  88. ]);
  89. return false;
  90. }
  91. }
  92. /**
  93. * 清除用户的关系缓存
  94. *
  95. * @param int $ursUserId URS用户ID
  96. * @return bool
  97. */
  98. public function clearUserRelationCache(int $ursUserId): bool
  99. {
  100. try {
  101. // 获取农场用户ID
  102. $farmUserId = UrsUserMappingService::getFarmUserId($ursUserId);
  103. // 删除数据库中的关系缓存
  104. UrsUserRelationCache::where('urs_user_id', $ursUserId)->delete();
  105. // 清除Redis缓存
  106. Redis::del("urs_promotion:user:{$ursUserId}:all_referrers");
  107. if ($farmUserId > 0) {
  108. Redis::del("urs_promotion:farm_user:{$farmUserId}:all_referrers");
  109. }
  110. return true;
  111. } catch (\Exception $e) {
  112. Log::error("清除URS用户关系缓存失败", [
  113. 'urs_user_id' => $ursUserId,
  114. 'error' => $e->getMessage()
  115. ]);
  116. return false;
  117. }
  118. }
  119. /**
  120. * 重建所有用户的关系缓存
  121. *
  122. * @param int $batchSize 批处理大小
  123. * @return array 包含成功和失败的数量
  124. */
  125. public function rebuildAllRelationCache(int $batchSize = 100): array
  126. {
  127. try {
  128. // 清空关系缓存表
  129. UrsUserRelationCache::truncate();
  130. // 清除所有Redis缓存
  131. $keys = Redis::keys("urs_promotion:user:*:all_referrers");
  132. if (!empty($keys)) {
  133. Redis::del($keys);
  134. }
  135. $keys = Redis::keys("urs_promotion:farm_user:*:all_referrers");
  136. if (!empty($keys)) {
  137. Redis::del($keys);
  138. }
  139. // 获取所有URS用户ID
  140. $ursUserIds = UrsUserReferral::where('status', UrsUserReferral::STATUS_VALID)
  141. ->pluck('urs_user_id')
  142. ->toArray();
  143. $successCount = 0;
  144. $failCount = 0;
  145. // 分批处理
  146. $chunks = array_chunk($ursUserIds, $batchSize);
  147. foreach ($chunks as $chunk) {
  148. foreach ($chunk as $ursUserId) {
  149. if ($this->generateUserRelationCache($ursUserId)) {
  150. $successCount++;
  151. } else {
  152. $failCount++;
  153. }
  154. }
  155. }
  156. Log::info("URS关系缓存重建完成", [
  157. 'success' => $successCount,
  158. 'fail' => $failCount,
  159. 'total' => count($ursUserIds)
  160. ]);
  161. return [
  162. 'success' => $successCount,
  163. 'fail' => $failCount,
  164. 'total' => count($ursUserIds)
  165. ];
  166. } catch (\Exception $e) {
  167. Log::error("重建所有URS关系缓存失败: " . $e->getMessage());
  168. return [
  169. 'success' => 0,
  170. 'fail' => 0,
  171. 'total' => 0,
  172. 'error' => $e->getMessage()
  173. ];
  174. }
  175. }
  176. /**
  177. * 检查关系缓存的完整性
  178. *
  179. * @return array 包含检查结果
  180. */
  181. public function checkRelationCacheIntegrity(): array
  182. {
  183. try {
  184. $totalUsers = UrsUserReferral::where('status', UrsUserReferral::STATUS_VALID)->count();
  185. $usersWithCache = UrsUserRelationCache::distinct('urs_user_id')->count('urs_user_id');
  186. $missingUsers = $totalUsers - $usersWithCache;
  187. // 检查循环推荐关系
  188. $circularRelations = DB::select("
  189. SELECT r1.urs_user_id, r1.urs_referrer_id
  190. FROM kku_urs_promotion_user_referrals r1
  191. JOIN kku_urs_promotion_user_referrals r2 ON r1.urs_referrer_id = r2.urs_user_id
  192. WHERE r2.urs_referrer_id = r1.urs_user_id
  193. AND r1.status = 1 AND r2.status = 1
  194. ");
  195. // 检查孤立的缓存
  196. $orphanedCaches = DB::select("
  197. SELECT c.urs_user_id
  198. FROM kku_urs_promotion_user_relation_cache c
  199. LEFT JOIN kku_urs_promotion_user_referrals r ON c.urs_user_id = r.urs_user_id AND r.status = 1
  200. WHERE r.urs_user_id IS NULL
  201. ");
  202. return [
  203. 'total_users' => $totalUsers,
  204. 'users_with_cache' => $usersWithCache,
  205. 'missing_users' => $missingUsers,
  206. 'circular_relations' => count($circularRelations),
  207. 'orphaned_caches' => count($orphanedCaches)
  208. ];
  209. } catch (\Exception $e) {
  210. Log::error("检查URS关系缓存完整性失败: " . $e->getMessage());
  211. return [
  212. 'error' => $e->getMessage()
  213. ];
  214. }
  215. }
  216. /**
  217. * 修复关系缓存问题
  218. *
  219. * @return array 包含修复结果
  220. */
  221. public function fixRelationCacheIssues(): array
  222. {
  223. try {
  224. // 检查完整性
  225. $integrity = $this->checkRelationCacheIntegrity();
  226. $fixed = [
  227. 'missing_users' => 0,
  228. 'orphaned_caches' => 0
  229. ];
  230. // 修复缺失的用户缓存
  231. if ($integrity['missing_users'] > 0) {
  232. $usersWithCache = UrsUserRelationCache::distinct('urs_user_id')->pluck('urs_user_id')->toArray();
  233. $allUsers = UrsUserReferral::where('status', UrsUserReferral::STATUS_VALID)
  234. ->pluck('urs_user_id')
  235. ->toArray();
  236. $missingUsers = array_diff($allUsers, $usersWithCache);
  237. foreach ($missingUsers as $ursUserId) {
  238. if ($this->generateUserRelationCache($ursUserId)) {
  239. $fixed['missing_users']++;
  240. }
  241. }
  242. }
  243. // 清理孤立的缓存
  244. if ($integrity['orphaned_caches'] > 0) {
  245. $orphanedCaches = DB::select("
  246. SELECT c.urs_user_id
  247. FROM kku_urs_promotion_user_relation_cache c
  248. LEFT JOIN kku_urs_promotion_user_referrals r ON c.urs_user_id = r.urs_user_id AND r.status = 1
  249. WHERE r.urs_user_id IS NULL
  250. ");
  251. foreach ($orphanedCaches as $cache) {
  252. if ($this->clearUserRelationCache($cache->urs_user_id)) {
  253. $fixed['orphaned_caches']++;
  254. }
  255. }
  256. }
  257. return [
  258. 'integrity' => $integrity,
  259. 'fixed' => $fixed
  260. ];
  261. } catch (\Exception $e) {
  262. Log::error("修复URS关系缓存问题失败: " . $e->getMessage());
  263. return [
  264. 'error' => $e->getMessage()
  265. ];
  266. }
  267. }
  268. /**
  269. * 为特定用户更新农场用户ID相关的缓存
  270. *
  271. * @param int $ursUserId URS用户ID
  272. * @param int $farmUserId 农场用户ID
  273. * @return bool
  274. */
  275. public function updateFarmUserIdInCache(int $ursUserId, int $farmUserId): bool
  276. {
  277. try {
  278. // 更新该用户作为被推荐人的缓存记录
  279. UrsUserRelationCache::where('urs_user_id', $ursUserId)
  280. ->update(['user_id' => $farmUserId]);
  281. // 更新该用户作为推荐人的缓存记录
  282. UrsUserRelationCache::where('urs_related_user_id', $ursUserId)
  283. ->update(['related_user_id' => $farmUserId]);
  284. // 重新生成该用户的关系缓存(确保路径正确)
  285. $this->generateUserRelationCache($ursUserId);
  286. Log::info("URS用户关系缓存中的农场用户ID更新成功", [
  287. 'urs_user_id' => $ursUserId,
  288. 'farm_user_id' => $farmUserId
  289. ]);
  290. return true;
  291. } catch (\Exception $e) {
  292. Log::error("更新URS用户关系缓存中的农场用户ID失败", [
  293. 'urs_user_id' => $ursUserId,
  294. 'farm_user_id' => $farmUserId,
  295. 'error' => $e->getMessage()
  296. ]);
  297. return false;
  298. }
  299. }
  300. }