UrsUserMappingService.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537
  1. <?php
  2. namespace App\Module\UrsPromotion\Services;
  3. use App\Module\UrsPromotion\Dtos\UrsUserMappingDto;
  4. use App\Module\UrsPromotion\Models\UrsUserMapping;
  5. use App\Module\User\Services\UserService;
  6. use Illuminate\Support\Facades\DB;
  7. use Illuminate\Support\Facades\Log;
  8. use Illuminate\Support\Str;
  9. /**
  10. * URS用户映射服务
  11. *
  12. * 管理URS用户ID与农场用户ID的映射关系
  13. */
  14. class UrsUserMappingService
  15. {
  16. /**
  17. * 创建用户映射关系(用户进入农场时调用)
  18. *
  19. * @param int $ursUserId URS用户ID
  20. * @param int $farmUserId 农场用户ID
  21. * @return UrsUserMappingDto
  22. * @throws \Exception
  23. */
  24. public static function createMapping(int $ursUserId, int $farmUserId): UrsUserMappingDto
  25. {
  26. try {
  27. DB::beginTransaction();
  28. // 检查是否已存在映射
  29. $existingMapping = UrsUserMapping::where('urs_user_id', $ursUserId)->first();
  30. if ($existingMapping) {
  31. if ($existingMapping->user_id === $farmUserId) {
  32. // 相同映射,直接返回
  33. DB::rollBack();
  34. return UrsUserMappingDto::fromModel($existingMapping);
  35. } else {
  36. // 不同映射,抛出异常
  37. throw new \Exception("URS用户ID {$ursUserId} 已映射到农场用户ID {$existingMapping->user_id}");
  38. }
  39. }
  40. // 检查农场用户ID是否已被映射
  41. $existingFarmMapping = UrsUserMapping::where('user_id', $farmUserId)->first();
  42. if ($existingFarmMapping) {
  43. throw new \Exception("农场用户ID {$farmUserId} 已映射到URS用户ID {$existingFarmMapping->urs_user_id}");
  44. }
  45. // 创建新映射
  46. $mapping = UrsUserMapping::create([
  47. 'urs_user_id' => $ursUserId,
  48. 'user_id' => $farmUserId,
  49. 'mapping_time' => now(),
  50. 'status' => UrsUserMapping::STATUS_VALID,
  51. ]);
  52. DB::commit();
  53. Log::info('URS用户映射创建成功', [
  54. 'urs_user_id' => $ursUserId,
  55. 'farm_user_id' => $farmUserId,
  56. 'mapping_id' => $mapping->id
  57. ]);
  58. return UrsUserMappingDto::fromModel($mapping);
  59. } catch (\Exception $e) {
  60. DB::rollBack();
  61. Log::error('URS用户映射创建失败', [
  62. 'urs_user_id' => $ursUserId,
  63. 'farm_user_id' => $farmUserId,
  64. 'error' => $e->getMessage()
  65. ]);
  66. throw $e;
  67. }
  68. }
  69. /**
  70. * 根据URS用户ID获取农场用户ID,不存在时自动创建新用户并绑定关系
  71. *
  72. * @param int $ursUserId URS用户ID
  73. * @return int|null 农场用户ID,创建失败时返回null
  74. */
  75. public static function getFarmUserId(int $ursUserId): ?int
  76. {
  77. // 首先尝试获取现有映射关系
  78. $existingFarmUserId = UrsUserMapping::getFarmUserIdByUrsUserId($ursUserId);
  79. if ($existingFarmUserId !== null) {
  80. return $existingFarmUserId;
  81. }
  82. // 不存在映射关系,自动创建新用户并建立映射
  83. try {
  84. DB::beginTransaction();
  85. // 生成用户名和随机密码
  86. $username = 'urs-' . $ursUserId;
  87. $password = Str::random(12);
  88. // 创建新用户
  89. $userDto = UserService::create($username, $password);
  90. if (is_string($userDto)) {
  91. // 创建失败
  92. DB::rollBack();
  93. Log::error('自动创建URS用户失败', [
  94. 'urs_user_id' => $ursUserId,
  95. 'username' => $username,
  96. 'error' => $userDto
  97. ]);
  98. return null;
  99. }
  100. // 创建用户映射关系
  101. $mapping = UrsUserMapping::create([
  102. 'urs_user_id' => $ursUserId,
  103. 'user_id' => $userDto->id,
  104. 'mapping_time' => now(),
  105. 'status' => UrsUserMapping::STATUS_VALID,
  106. ]);
  107. DB::commit();
  108. Log::info('自动创建URS用户映射成功', [
  109. 'urs_user_id' => $ursUserId,
  110. 'farm_user_id' => $userDto->id,
  111. 'username' => $username,
  112. 'mapping_id' => $mapping->id
  113. ]);
  114. return $userDto->id;
  115. } catch (\Exception $e) {
  116. DB::rollBack();
  117. Log::error('自动创建URS用户映射失败', [
  118. 'urs_user_id' => $ursUserId,
  119. 'error' => $e->getMessage()
  120. ]);
  121. return null;
  122. }
  123. }
  124. /**
  125. * 根据农场用户ID获取URS用户ID
  126. *
  127. * @param int $farmUserId 农场用户ID
  128. * @return int|null URS用户ID,如果未找到返回null
  129. */
  130. public static function getUrsUserId(int $farmUserId): ?int
  131. {
  132. return UrsUserMapping::getUrsUserIdByFarmUserId($farmUserId);
  133. }
  134. /**
  135. * 批量获取URS用户ID对应的农场用户ID
  136. *
  137. * @param array $ursUserIds URS用户ID数组
  138. * @return array 映射关系数组 [urs_user_id => farm_user_id]
  139. */
  140. public static function batchGetFarmUserIds(array $ursUserIds): array
  141. {
  142. return UrsUserMapping::getFarmUserIdsByUrsUserIds($ursUserIds);
  143. }
  144. /**
  145. * 批量获取农场用户ID对应的URS用户ID
  146. *
  147. * @param array $farmUserIds 农场用户ID数组
  148. * @return array 映射关系数组 [farm_user_id => urs_user_id]
  149. */
  150. public static function batchGetUrsUserIds(array $farmUserIds): array
  151. {
  152. return UrsUserMapping::getUrsUserIdsByFarmUserIds($farmUserIds);
  153. }
  154. /**
  155. * 检查URS用户是否已进入农场
  156. *
  157. * @param int $ursUserId URS用户ID
  158. * @return bool
  159. */
  160. public static function hasEnteredFarm(int $ursUserId): bool
  161. {
  162. return UrsUserMapping::hasEnteredFarm($ursUserId);
  163. }
  164. /**
  165. * 获取已进入农场的URS用户ID列表
  166. *
  167. * @param array $ursUserIds URS用户ID数组
  168. * @return array 已进入农场的URS用户ID数组
  169. */
  170. public static function getEnteredFarmUsers(array $ursUserIds): array
  171. {
  172. return UrsUserMapping::getEnteredFarmUrsUserIds($ursUserIds);
  173. }
  174. /**
  175. * 禁用用户映射关系
  176. *
  177. * @param int $ursUserId URS用户ID
  178. * @return bool
  179. */
  180. public static function disableMapping(int $ursUserId): bool
  181. {
  182. try {
  183. $mapping = UrsUserMapping::where('urs_user_id', $ursUserId)->first();
  184. if (!$mapping) {
  185. return false;
  186. }
  187. $mapping->status = UrsUserMapping::STATUS_INVALID;
  188. $mapping->save();
  189. Log::info('URS用户映射已禁用', [
  190. 'urs_user_id' => $ursUserId,
  191. 'mapping_id' => $mapping->id
  192. ]);
  193. return true;
  194. } catch (\Exception $e) {
  195. Log::error('URS用户映射禁用失败', [
  196. 'urs_user_id' => $ursUserId,
  197. 'error' => $e->getMessage()
  198. ]);
  199. return false;
  200. }
  201. }
  202. /**
  203. * 启用用户映射关系
  204. *
  205. * @param int $ursUserId URS用户ID
  206. * @return bool
  207. */
  208. public static function enableMapping(int $ursUserId): bool
  209. {
  210. try {
  211. $mapping = UrsUserMapping::where('urs_user_id', $ursUserId)->first();
  212. if (!$mapping) {
  213. return false;
  214. }
  215. $mapping->status = UrsUserMapping::STATUS_VALID;
  216. $mapping->save();
  217. Log::info('URS用户映射已启用', [
  218. 'urs_user_id' => $ursUserId,
  219. 'mapping_id' => $mapping->id
  220. ]);
  221. return true;
  222. } catch (\Exception $e) {
  223. Log::error('URS用户映射启用失败', [
  224. 'urs_user_id' => $ursUserId,
  225. 'error' => $e->getMessage()
  226. ]);
  227. return false;
  228. }
  229. }
  230. /**
  231. * 获取映射统计信息
  232. *
  233. * @return array
  234. */
  235. public static function getMappingStats(): array
  236. {
  237. return [
  238. 'total_mappings' => UrsUserMapping::count(),
  239. 'valid_mappings' => UrsUserMapping::where('status', UrsUserMapping::STATUS_VALID)->count(),
  240. 'invalid_mappings' => UrsUserMapping::where('status', UrsUserMapping::STATUS_INVALID)->count(),
  241. 'today_mappings' => UrsUserMapping::whereDate('created_at', today())->count(),
  242. ];
  243. }
  244. /**
  245. * 获取用户映射详情
  246. *
  247. * @param int $ursUserId URS用户ID
  248. * @return UrsUserMappingDto|null
  249. */
  250. public static function getMappingDetail(int $ursUserId): ?UrsUserMappingDto
  251. {
  252. $mapping = UrsUserMapping::where('urs_user_id', $ursUserId)->first();
  253. if (!$mapping) {
  254. return null;
  255. }
  256. return UrsUserMappingDto::fromModel($mapping);
  257. }
  258. /**
  259. * 验证映射关系有效性
  260. *
  261. * @param UrsUserMapping $mapping 映射关系对象
  262. * @return array 验证结果 ['valid' => bool, 'updated' => bool, 'details' => array, 'reasons' => array]
  263. */
  264. public static function validateMapping(UrsUserMapping $mapping): array
  265. {
  266. $result = [
  267. 'valid' => false,
  268. 'updated' => false,
  269. 'details' => [],
  270. 'reasons' => []
  271. ];
  272. try {
  273. $isValid = true;
  274. $details = [];
  275. $reasons = [];
  276. // 检查URS用户ID是否有效
  277. if ($mapping->urs_user_id <= 0) {
  278. $isValid = false;
  279. $reasons[] = 'URS用户ID无效';
  280. } else {
  281. $details[] = "URS用户ID: {$mapping->urs_user_id}";
  282. }
  283. // 检查农场用户ID是否有效
  284. if ($mapping->user_id <= 0) {
  285. $isValid = false;
  286. $reasons[] = '农场用户ID无效';
  287. } else {
  288. // 检查农场用户是否存在
  289. $farmUser = \App\Module\User\Models\User::find($mapping->user_id);
  290. if (!$farmUser) {
  291. $isValid = false;
  292. $reasons[] = '农场用户不存在';
  293. } else {
  294. $details[] = "农场用户: {$farmUser->username} (ID: {$mapping->user_id})";
  295. }
  296. }
  297. // 检查是否存在重复映射
  298. $duplicateUrsMapping = UrsUserMapping::where('urs_user_id', $mapping->urs_user_id)
  299. ->where('id', '!=', $mapping->id)
  300. ->first();
  301. if ($duplicateUrsMapping) {
  302. $isValid = false;
  303. $reasons[] = "URS用户ID {$mapping->urs_user_id} 存在重复映射";
  304. }
  305. $duplicateFarmMapping = UrsUserMapping::where('user_id', $mapping->user_id)
  306. ->where('id', '!=', $mapping->id)
  307. ->first();
  308. if ($duplicateFarmMapping) {
  309. $isValid = false;
  310. $reasons[] = "农场用户ID {$mapping->user_id} 存在重复映射";
  311. }
  312. // 检查映射时间是否合理
  313. if ($mapping->mapping_time && $mapping->mapping_time->isFuture()) {
  314. $isValid = false;
  315. $reasons[] = '映射时间不能是未来时间';
  316. } else if ($mapping->mapping_time) {
  317. $details[] = "映射时间: {$mapping->mapping_time->format('Y-m-d H:i:s')}";
  318. }
  319. // 更新映射状态
  320. $originalStatus = $mapping->status;
  321. $newStatus = $isValid ? UrsUserMapping::STATUS_VALID : UrsUserMapping::STATUS_INVALID;
  322. if ($originalStatus !== $newStatus) {
  323. $mapping->status = $newStatus;
  324. $mapping->save();
  325. $result['updated'] = true;
  326. Log::info('映射关系状态已更新', [
  327. 'mapping_id' => $mapping->id,
  328. 'urs_user_id' => $mapping->urs_user_id,
  329. 'farm_user_id' => $mapping->user_id,
  330. 'old_status' => $originalStatus,
  331. 'new_status' => $newStatus
  332. ]);
  333. }
  334. $result['valid'] = $isValid;
  335. $result['details'] = $details;
  336. $result['reasons'] = $reasons;
  337. } catch (\Exception $e) {
  338. Log::error('验证映射关系失败', [
  339. 'mapping_id' => $mapping->id,
  340. 'error' => $e->getMessage()
  341. ]);
  342. $result['reasons'][] = '验证过程发生异常: ' . $e->getMessage();
  343. }
  344. return $result;
  345. }
  346. /**
  347. * 同步用户信息
  348. *
  349. * @param UrsUserMapping $mapping 映射关系对象
  350. * @return array 同步结果 ['success' => bool, 'updated_fields' => array, 'sync_time' => string, 'error' => string]
  351. */
  352. public static function syncUserInfo(UrsUserMapping $mapping): array
  353. {
  354. $result = [
  355. 'success' => false,
  356. 'updated_fields' => [],
  357. 'sync_time' => '',
  358. 'error' => ''
  359. ];
  360. try {
  361. // 检查映射状态
  362. if ($mapping->status !== UrsUserMapping::STATUS_VALID) {
  363. $result['error'] = '映射关系状态无效,无法同步';
  364. return $result;
  365. }
  366. // 获取农场用户信息
  367. $farmUser = \App\Module\User\Models\User::find($mapping->user_id);
  368. if (!$farmUser) {
  369. $result['error'] = '农场用户不存在';
  370. return $result;
  371. }
  372. $updatedFields = [];
  373. $syncTime = now()->format('Y-m-d H:i:s');
  374. // 这里可以根据实际需求添加从URS系统同步用户信息的逻辑
  375. // 目前只是模拟同步过程,更新最后同步时间
  376. // 模拟同步用户昵称(如果需要的话)
  377. $expectedUsername = 'urs-' . $mapping->urs_user_id;
  378. if ($farmUser->username !== $expectedUsername) {
  379. // 检查新用户名是否已存在
  380. $existingUser = \App\Module\User\Models\User::where('username', $expectedUsername)
  381. ->where('id', '!=', $farmUser->id)
  382. ->first();
  383. if (!$existingUser) {
  384. $oldUsername = $farmUser->username;
  385. $farmUser->username = $expectedUsername;
  386. $farmUser->save();
  387. $updatedFields[] = "用户名: {$oldUsername} -> {$expectedUsername}";
  388. }
  389. }
  390. // 更新映射记录的最后同步时间(如果表中有这个字段的话)
  391. // $mapping->last_sync_time = now();
  392. // $mapping->save();
  393. $result['success'] = true;
  394. $result['updated_fields'] = $updatedFields;
  395. $result['sync_time'] = $syncTime;
  396. Log::info('用户信息同步成功', [
  397. 'mapping_id' => $mapping->id,
  398. 'urs_user_id' => $mapping->urs_user_id,
  399. 'farm_user_id' => $mapping->user_id,
  400. 'updated_fields' => $updatedFields,
  401. 'sync_time' => $syncTime
  402. ]);
  403. } catch (\Exception $e) {
  404. Log::error('用户信息同步失败', [
  405. 'mapping_id' => $mapping->id,
  406. 'error' => $e->getMessage()
  407. ]);
  408. $result['error'] = '同步过程发生异常: ' . $e->getMessage();
  409. }
  410. return $result;
  411. }
  412. /**
  413. * 获取活跃用户统计信息
  414. *
  415. * @return array
  416. */
  417. public static function getActiveUserStats(): array
  418. {
  419. return UrsUserMapping::getActiveUserStats();
  420. }
  421. /**
  422. * 批量更新用户活跃状态
  423. *
  424. * @param array $activeData 活跃数据 [urs_user_id => ['is_active' => 1, 'active_days_count' => 5]]
  425. * @return int 更新的记录数
  426. */
  427. public static function batchUpdateActiveStatus(array $activeData): int
  428. {
  429. return UrsUserMapping::batchUpdateActiveStatus($activeData);
  430. }
  431. /**
  432. * 获取活跃的URS用户ID列表
  433. *
  434. * @param array $ursUserIds URS用户ID数组
  435. * @return array 活跃的URS用户ID数组
  436. */
  437. public static function getActiveUrsUserIds(array $ursUserIds): array
  438. {
  439. return UrsUserMapping::getActiveUrsUserIds($ursUserIds);
  440. }
  441. /**
  442. * 检查用户是否活跃
  443. *
  444. * @param int $ursUserId URS用户ID
  445. * @return bool
  446. */
  447. public static function isUserActive(int $ursUserId): bool
  448. {
  449. $mapping = UrsUserMapping::where('urs_user_id', $ursUserId)
  450. ->where('status', UrsUserMapping::STATUS_VALID)
  451. ->first();
  452. return $mapping ? $mapping->isActive() : false;
  453. }
  454. /**
  455. * 更新单个用户的活跃状态
  456. *
  457. * @param int $ursUserId URS用户ID
  458. * @return bool
  459. */
  460. public static function updateUserActiveStatus(int $ursUserId): bool
  461. {
  462. return UrsActiveUserService::updateUserActiveStatus($ursUserId);
  463. }
  464. }