FundLogCollector.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513
  1. <?php
  2. namespace App\Module\Game\Logics\UserLogCollectors;
  3. use App\Module\Fund\Enums\FUND_TYPE;
  4. use App\Module\Fund\Models\FundLogModel;
  5. use App\Module\Fund\Services\AccountService;
  6. use App\Module\Game\Enums\REWARD_SOURCE_TYPE;
  7. /**
  8. * 资金日志收集器
  9. *
  10. * 收集fund_logs表的新增记录,转换为用户友好的日志消息
  11. */
  12. class FundLogCollector extends BaseLogCollector
  13. {
  14. /**
  15. * 源表名
  16. *
  17. * @var string
  18. */
  19. protected string $sourceTable = 'fund_logs';
  20. /**
  21. * 默认源类型 - 使用系统奖励枚举
  22. *
  23. * @var string
  24. */
  25. protected string $sourceType = 'system';
  26. /**
  27. * 商店商品名称缓存
  28. *
  29. * @var array
  30. */
  31. private array $shopItemNameCache = [];
  32. /**
  33. * 预加载商店商品信息
  34. *
  35. * @param \Illuminate\Database\Eloquent\Collection $records 记录集合
  36. * @return void
  37. */
  38. private function preloadShopItems($records): void
  39. {
  40. $shopItemIds = [];
  41. foreach ($records as $record) {
  42. if (strpos($record->remark, 'shop_buy') !== false) {
  43. if (preg_match('/ID:(\d+)/', $record->remark, $matches)) {
  44. $shopItemIds[] = (int)$matches[1];
  45. }
  46. }
  47. }
  48. if (!empty($shopItemIds)) {
  49. $shopItemIds = array_unique($shopItemIds);
  50. $shopItems = \App\Module\Shop\Models\ShopItem::whereIn('id', $shopItemIds)->get();
  51. foreach ($shopItems as $shopItem) {
  52. $this->shopItemNameCache[$shopItem->id] = $shopItem->name;
  53. }
  54. // 为未找到的商品设置默认名称
  55. foreach ($shopItemIds as $itemId) {
  56. if (!isset($this->shopItemNameCache[$itemId])) {
  57. $this->shopItemNameCache[$itemId] = "商品{$itemId}";
  58. }
  59. }
  60. }
  61. }
  62. /**
  63. * 获取新的记录
  64. *
  65. * @param int $lastProcessedId 上次处理的最大ID
  66. * @return \Illuminate\Database\Eloquent\Collection
  67. */
  68. protected function getNewRecords(int $lastProcessedId)
  69. {
  70. // 使用原始查询避免枚举转换问题
  71. $records = FundLogModel::selectRaw('*')
  72. ->where('id', '>', $lastProcessedId)
  73. ->orderBy('id')
  74. ->limit($this->maxRecords)
  75. ->get()
  76. ->map(function ($record) {
  77. // 手动设置原始属性,避免枚举转换
  78. $record->setRawAttributes($record->getAttributes(), true);
  79. return $record;
  80. });
  81. // 预加载商店商品信息
  82. $this->preloadShopItems($records);
  83. return $records;
  84. }
  85. /**
  86. * 获取源表的最大ID
  87. *
  88. * 重写父类方法,使用模型查询以获得更好的性能
  89. *
  90. * @return int
  91. */
  92. public function getSourceTableMaxId(): int
  93. {
  94. return FundLogModel::max('id') ?: 0;
  95. }
  96. /**
  97. * 转换记录为用户日志数据
  98. *
  99. * @param FundLogModel $record 资金日志记录
  100. * @return array|null 用户日志数据,null表示跳过
  101. */
  102. protected function convertToUserLog($record): ?array
  103. {
  104. try {
  105. // 获取资金名称
  106. $fundName = $this->getFundName($record->fund_id);
  107. // 判断是获得还是消耗
  108. $amount = abs($record->amount);
  109. $action = $record->amount > 0 ? '增加' : '减少';
  110. // 解析备注信息,生成用户友好的消息
  111. $message = $this->buildUserFriendlyMessage($record, $fundName, $action, $amount);
  112. // 使用原始记录的时间
  113. $createdAt = date('Y-m-d H:i:s', $record->create_time);
  114. // 根据操作类型选择合适的source_type
  115. $sourceType = $this->getSourceTypeByRemark($record->remark);
  116. return $this->createUserLogDataWithSourceType(
  117. $record->user_id,
  118. $message,
  119. $record->id,
  120. $createdAt,
  121. $sourceType
  122. );
  123. } catch (\Exception $e) {
  124. \Illuminate\Support\Facades\Log::error("转换资金日志失败", [
  125. 'record_id' => $record->id,
  126. 'error' => $e->getMessage()
  127. ]);
  128. return null;
  129. }
  130. }
  131. /**
  132. * 构建用户友好的消息
  133. *
  134. * @param FundLogModel $record 资金日志记录
  135. * @param string $fundName 资金名称
  136. * @param string $action 操作类型(获得/消耗)
  137. * @param float $amount 金额
  138. * @return string
  139. */
  140. private function buildUserFriendlyMessage(FundLogModel $record, string $fundName, string $action, float $amount): string
  141. {
  142. // 解析备注信息
  143. $remarkInfo = $this->parseRemark($record->remark);
  144. // 根据来源类型生成不同的消息格式
  145. if (isset($remarkInfo['source']) && isset($remarkInfo['id'])) {
  146. $sourceMessage = $this->getSourceMessage($remarkInfo['source'], $remarkInfo['id'], $action);
  147. if ($sourceMessage) {
  148. return "{$sourceMessage}{$action}{$fundName} {$amount}";
  149. }
  150. }
  151. // 特殊处理operate_type=3(管理员操作)的情况
  152. try {
  153. // 直接从属性数组获取值,避免枚举转换问题
  154. $operateTypeValue = $record->getAttributes()['operate_type'] ?? null;
  155. if ($operateTypeValue == 3 || $operateTypeValue === '3') {
  156. $sourceMessage = $this->getAdminOperationMessage($record, $action);
  157. if ($sourceMessage) {
  158. return "{$sourceMessage}{$action}{$fundName} {$amount}";
  159. }
  160. }
  161. } catch (\Exception $e) {
  162. // 如果类型转换失败,记录错误但继续处理
  163. \Illuminate\Support\Facades\Log::warning("处理operate_type失败", [
  164. 'record_id' => $record->id,
  165. 'operate_type_raw' => $record->getAttributes()['operate_type'] ?? 'null',
  166. 'error' => $e->getMessage(),
  167. 'collector' => $this->collectorName
  168. ]);
  169. }
  170. // 如果无法解析来源信息,使用默认格式
  171. return "{$action}{$fundName} {$amount}";
  172. }
  173. /**
  174. * 获取资金名称
  175. *
  176. * @param mixed $fundId 资金ID
  177. * @return string
  178. */
  179. private function getFundName(FUND_TYPE $fundId): string
  180. {
  181. return FUND_TYPE::getName($fundId);
  182. }
  183. /**
  184. * 解析备注信息
  185. *
  186. * @param string $remark 备注内容
  187. * @return array 解析后的信息数组
  188. */
  189. private function parseRemark(string $remark): array
  190. {
  191. $info = [];
  192. // 解析格式:币种消耗:2,消耗组:16,来源:shop_buy,ID:7
  193. // 使用更宽松的正则表达式来匹配中文和英文字符
  194. if (preg_match_all('/([^:,]+):([^,]+)/', $remark, $matches)) {
  195. for ($i = 0; $i < count($matches[1]); $i++) {
  196. $key = trim($matches[1][$i]);
  197. $value = trim($matches[2][$i]);
  198. // 转换中文键名为英文
  199. switch ($key) {
  200. case '来源':
  201. $info['source'] = $value;
  202. break;
  203. case 'ID':
  204. $info['id'] = (int)$value;
  205. break;
  206. case '消耗组':
  207. $info['consume_group'] = (int)$value;
  208. break;
  209. case '币种消耗':
  210. $info['fund_type'] = (int)$value;
  211. break;
  212. default:
  213. $info[$key] = $value;
  214. }
  215. }
  216. }
  217. return $info;
  218. }
  219. /**
  220. * 根据来源信息获取操作描述
  221. *
  222. * @param string $source 来源类型
  223. * @param int $id 来源ID
  224. * @param string $action 操作类型
  225. * @return string|null 操作描述,null表示使用默认格式
  226. */
  227. private function getSourceMessage(string $source, int $id, string $action): ?string
  228. {
  229. switch ($source) {
  230. case 'shop_buy':
  231. $itemName = $this->getShopItemName($id);
  232. return "购买{$itemName}";
  233. case '开启宝箱':
  234. case 'chest_open':
  235. return "开启宝箱";
  236. case 'house_upgrade':
  237. return "房屋升级";
  238. case 'land_upgrade':
  239. return "土地升级";
  240. case 'task_reward':
  241. return "任务奖励";
  242. case 'system_gift':
  243. return "系统赠送";
  244. case 'admin_operation':
  245. return "管理员操作";
  246. case 'test_command':
  247. return "测试操作";
  248. default:
  249. return null;
  250. }
  251. }
  252. /**
  253. * 获取管理员操作消息
  254. *
  255. * @param FundLogModel $record 资金日志记录
  256. * @param string $action 操作类型
  257. * @return string|null 操作描述,null表示使用默认格式
  258. */
  259. private function getAdminOperationMessage(FundLogModel $record, string $action): ?string
  260. {
  261. try {
  262. // 尝试从fund_admin表获取详细信息
  263. // 用户不需要知道这么多,隐藏详情
  264. // $adminOperation = \App\Module\Fund\Models\FundAdminModel::find($record->operate_id);
  265. //
  266. // if ($adminOperation) {
  267. // // 如果找到管理员操作记录,使用其备注信息
  268. // if (!empty($adminOperation->remark) && $adminOperation->remark !== 'Admin Approved') {
  269. // return "管理员操作({$adminOperation->remark})";
  270. // }
  271. //
  272. // // 获取管理员信息
  273. // $adminInfo = $this->getAdminInfo($adminOperation->admin_id);
  274. // return "管理员操作({$adminInfo})";
  275. // }
  276. // 如果没有找到对应的管理员操作记录,尝试使用fund_logs的备注
  277. if (!empty($record->remark) && $record->remark !== 'Admin Approved') {
  278. return "管理员操作({$record->remark})";
  279. }
  280. // 如果备注也没有有用信息,显示操作ID
  281. return "管理员操作";
  282. } catch (\Exception $e) {
  283. \Illuminate\Support\Facades\Log::warning("获取管理员操作信息失败", [
  284. 'fund_log_id' => $record->id,
  285. 'operate_id' => $record->operate_id,
  286. 'error' => $e->getMessage(),
  287. 'collector' => $this->collectorName
  288. ]);
  289. return "管理员操作";
  290. }
  291. }
  292. /**
  293. * 获取管理员信息
  294. *
  295. * @param int $adminId 管理员ID
  296. * @return string
  297. */
  298. private function getAdminInfo(int $adminId): string
  299. {
  300. try {
  301. // 尝试获取管理员用户信息
  302. $adminUser = \Dcat\Admin\Models\Administrator::find($adminId);
  303. if ($adminUser && !empty($adminUser->name)) {
  304. return $adminUser->name;
  305. }
  306. return "管理员{$adminId}";
  307. } catch (\Exception $e) {
  308. return "管理员{$adminId}";
  309. }
  310. }
  311. /**
  312. * 获取商店商品名称
  313. *
  314. * @param int $itemId 商品ID
  315. * @return string
  316. */
  317. private function getShopItemName(int $itemId): string
  318. {
  319. // 检查缓存
  320. if (isset($this->shopItemNameCache[$itemId])) {
  321. return $this->shopItemNameCache[$itemId];
  322. }
  323. try {
  324. $shopItem = \App\Module\Shop\Models\ShopItem::find($itemId);
  325. if ($shopItem && $shopItem->name) {
  326. $this->shopItemNameCache[$itemId] = $shopItem->name;
  327. return $shopItem->name;
  328. }
  329. $defaultName = "商品{$itemId}";
  330. $this->shopItemNameCache[$itemId] = $defaultName;
  331. return $defaultName;
  332. } catch (\Exception $e) {
  333. \Illuminate\Support\Facades\Log::error("获取商店商品名称失败", [
  334. 'item_id' => $itemId,
  335. 'error' => $e->getMessage(),
  336. 'collector' => $this->collectorName
  337. ]);
  338. $defaultName = "商品{$itemId}";
  339. $this->shopItemNameCache[$itemId] = $defaultName;
  340. return $defaultName;
  341. }
  342. }
  343. /**
  344. * 是否应该记录此日志
  345. *
  346. * @param FundLogModel $record
  347. * @return bool
  348. */
  349. private function shouldLogRecord(FundLogModel $record): bool
  350. {
  351. // 可以在这里添加过滤规则
  352. // 例如:只记录金额大于某个值的变更
  353. // 跳过金额为0的记录
  354. if ($record->amount == 0) {
  355. return false;
  356. }
  357. // 可以添加更多过滤条件
  358. // 例如:跳过某些操作类型
  359. // if (in_array($record->operate_type, [...])) {
  360. // return false;
  361. // }
  362. return true;
  363. }
  364. /**
  365. * 重写转换方法,添加过滤逻辑
  366. *
  367. * @param FundLogModel $record 资金日志记录
  368. * @return array|null 用户日志数据,null表示跳过
  369. */
  370. protected function convertToUserLogWithFilter($record): ?array
  371. {
  372. // 检查是否应该记录此日志
  373. if (!$this->shouldLogRecord($record)) {
  374. return null;
  375. }
  376. return $this->convertToUserLog($record);
  377. }
  378. /**
  379. * 根据备注信息获取合适的source_type
  380. *
  381. * @param string $remark 备注内容
  382. * @return string
  383. */
  384. private function getSourceTypeByRemark(string $remark): string
  385. {
  386. $remarkInfo = $this->parseRemark($remark);
  387. if (isset($remarkInfo['source'])) {
  388. switch ($remarkInfo['source']) {
  389. case 'shop_buy':
  390. return REWARD_SOURCE_TYPE::SHOP_PURCHASE->value;
  391. case 'chest_open':
  392. case '开启宝箱':
  393. return REWARD_SOURCE_TYPE::CHEST->value;
  394. case 'house_upgrade':
  395. case 'land_upgrade':
  396. return REWARD_SOURCE_TYPE::FARM_UPGRADE->value;
  397. case 'task_reward':
  398. return REWARD_SOURCE_TYPE::TASK->value;
  399. case 'system_gift':
  400. return REWARD_SOURCE_TYPE::SYSTEM->value;
  401. case 'admin_operation':
  402. return REWARD_SOURCE_TYPE::ADMIN_GRANT->value;
  403. case 'test_command':
  404. return REWARD_SOURCE_TYPE::TEST->value;
  405. default:
  406. return REWARD_SOURCE_TYPE::SYSTEM->value;
  407. }
  408. }
  409. return REWARD_SOURCE_TYPE::SYSTEM->value;
  410. }
  411. /**
  412. * 创建用户日志数据数组(带自定义source_type)
  413. *
  414. * @param int $userId 用户ID
  415. * @param string $message 日志消息
  416. * @param int $sourceId 来源记录ID
  417. * @param string|null $originalTime 原始时间(业务发生时间),null则使用当前时间
  418. * @param string $sourceType 来源类型
  419. * @return array
  420. */
  421. protected function createUserLogDataWithSourceType(
  422. int $userId,
  423. string $message,
  424. int $sourceId,
  425. ?string $originalTime = null,
  426. string $sourceType = null
  427. ): array
  428. {
  429. $now = now()->toDateTimeString();
  430. $originalTime = $originalTime ?? $now;
  431. $sourceType = $sourceType ?? $this->sourceType;
  432. return [
  433. 'user_id' => $userId,
  434. 'message' => $message,
  435. 'source_type' => $sourceType,
  436. 'source_id' => $sourceId,
  437. 'source_table' => $this->sourceTable,
  438. 'original_time' => $originalTime, // 原始业务时间
  439. 'collected_at' => $now, // 收集时间
  440. 'created_at' => $now, // 兼容字段
  441. ];
  442. }
  443. }