UserLogLogic.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506
  1. <?php
  2. // 玩家日志逻
  3. namespace app\v1\logic\dataReport;
  4. use plugin\saiadmin\basic\BaseLogic;
  5. use plugin\saiadmin\service\OpenSpoutWriter;
  6. use support\think\Db;
  7. class UserLogLogic extends BaseLogic
  8. {
  9. // 注册日志
  10. public function getRegLogList($where): mixed
  11. {
  12. $params = $this->searchByAuth($where);
  13. $regTime = $params['reg_time'];
  14. $tableName = 'sdk_reg_log_' . date('Ym', strtotime($regTime));
  15. $params['reg_time'] = strtotime($regTime . " 00:00:00") . "<=" . strtotime($regTime . " 23:59:59");
  16. $orderBy = request()->input('orderBy', 'reg_time');
  17. $orderType = request()->input('orderType', 'desc');
  18. $page = request()->input('page', 1);
  19. $limit = request()->input('limit', 10);
  20. // 获取这一天的注册日志
  21. $regLogQuery = Db::connect('db_game_log')
  22. ->table($tableName)
  23. ->where('reg_time', '>=', strtotime($regTime . ' 00:00:00'))
  24. ->where('reg_time', '<=', strtotime($regTime . ' 23:59:59'));
  25. // 添加其他查询条件
  26. if (!empty($params['game_id'])) {
  27. $regLogQuery = $regLogQuery->where('game_id', 'in', is_array($params['game_id']) ? implode(',', $params['game_id']) : $params['game_id']);
  28. }
  29. if (!empty($params['user_name'])) {
  30. $regLogQuery = $regLogQuery->where('user_name', $params['user_name']);
  31. }
  32. if (!empty($params['media_id'])) {
  33. $regLogQuery = $regLogQuery->where('media_id', $params['media_id']);
  34. }
  35. if (!empty($params['site_id'])) {
  36. $regLogQuery = $regLogQuery->where('site_id', $params['site_id']);
  37. }
  38. if (!empty($params['agent_id'])) {
  39. $regLogQuery = $regLogQuery->where('agent_id', $params['agent_id']);
  40. }
  41. if (!empty($params['auth_id'])){
  42. $regLogQuery = $regLogQuery->where('auth_id', 'in', is_array($params['auth_id']) ? implode(',', $params['auth_id']) : $params['auth_id']);
  43. }
  44. if (!empty($params['vt'])) {
  45. $regLogQuery = $regLogQuery->where('vt', $params['vt']);
  46. }
  47. // 自然量ID, auth_id=0为自然量
  48. $whereRaw = [];
  49. if(!empty($params['nomal_game_id'])){
  50. for($i=0;$i<count($params['nomal_game_id']);$i++){
  51. $whereRaw[] = "(game_id = {$params['nomal_game_id'][$i]} AND auth_id=0)";
  52. }
  53. }
  54. $whereRaw = implode(' OR ', $whereRaw);
  55. $regLogQuery = $regLogQuery->whereRaw($whereRaw);
  56. echo $regLogQuery->buildSql();
  57. // 分页
  58. $data = $regLogQuery->paginate($limit)->toArray();
  59. return $data;
  60. }
  61. // 登录日志
  62. public function getLoginLogList($where): mixed
  63. {
  64. $orderBy = request()->input('orderBy', 'login_time');
  65. $orderType = request()->input('orderType', 'desc');
  66. $page = request()->input('page', 1);
  67. $limit = request()->input('limit', 10);
  68. $params = $this->searchByAuth($where);
  69. $loginTime = $params['login_time'];
  70. $tableName = 'sdk_login_log_' . date('Ym', strtotime($loginTime));
  71. $loginLogQuery = Db::connect('db_game_log')
  72. ->table($tableName)
  73. ->where('login_time', '>=', strtotime($params['login_time'] . ' 00:00:00'))
  74. ->where('login_time', '<=', strtotime($params['login_time'] . ' 23:59:59'));
  75. if (!empty($params['game_id'])) {
  76. $loginLogQuery = $loginLogQuery->where('game_id', 'in', is_array($params['game_id']) ? implode(',', $params['game_id']) : $params['game_id']);
  77. }
  78. if (!empty($params['user_name'])) {
  79. $loginLogQuery = $loginLogQuery->where('user_name', $params['user_name']);
  80. }
  81. if (!empty($params['media_id'])) {
  82. $loginLogQuery = $loginLogQuery->where('media_id', $params['media_id']);
  83. }
  84. if (!empty($params['site_id'])) {
  85. $loginLogQuery = $loginLogQuery->where('site_id', $params['site_id']);
  86. }
  87. if (!empty($params['auth_id'])) {
  88. $loginLogQuery = $loginLogQuery->where('auth_id', 'in', is_array($params['auth_id']) ? implode(',', $params['auth_id']) : $params['auth_id']);
  89. }
  90. if (!empty($params['agent_id'])) {
  91. $loginLogQuery = $loginLogQuery->where('agent_id', $params['agent_id']);
  92. }
  93. if (!empty($params['reg_time'])) {
  94. $loginLogQuery = $loginLogQuery
  95. ->where('reg_time', '>=', strtotime($params['reg_time'] . ' 00:00:00'))
  96. ->where('reg_time', '<=', strtotime($params['reg_time'] . ' 23:59:59'));
  97. }
  98. if (!empty($params['login_time'])) {
  99. $loginLogQuery = $loginLogQuery
  100. ->where('login_time', '>=', strtotime($params['login_time'] . ' 00:00:00'))
  101. ->where('login_time', '<=', strtotime($params['login_time'] . ' 23:59:59'));
  102. }
  103. // 自然量ID, auth_id=0为自然量
  104. $whereRaw = [];
  105. if(!empty($params['nomal_game_id'])){
  106. for($i=0;$i<count($params['nomal_game_id']);$i++){
  107. $whereRaw[] = "(game_id = {$params['nomal_game_id'][$i]} AND auth_id=0)";
  108. }
  109. }
  110. $whereRaw = implode(' OR ', $whereRaw);
  111. $loginLogQuery = $loginLogQuery->whereRaw($whereRaw);
  112. // 执行查询并排序
  113. $loginLogQuery->order($orderBy, $orderType);
  114. $data = $loginLogQuery->paginate($limit)->toArray();
  115. return $data;
  116. }
  117. // 充值明细
  118. public function exportRechargeDetailList($where): mixed
  119. {
  120. $data = $this->getRechargeDetailList($where, 'all');
  121. $data = $this->trandformListColumn($data, ['game', 'ip', 'agent', 'auth', 'pay_channel']);
  122. $data = array_map(function ($item) {
  123. return [
  124. 'order_id' => $item['order_id'],
  125. 'user_name' => $item['user_name'],
  126. 'agent_id' => $item['agent_id'],
  127. 'site_id' => $item['site_id'],
  128. 'game_name' => $item['game_name'],
  129. 'server_id' => $item['server_id'],
  130. 'server_name' => $item['server_name'],
  131. 'pay_channel_name' => $item['pay_channel_name'],
  132. 'money' => $item['money'],
  133. 'pay_date' => $item['pay_date'],
  134. 'reg_date' => $item['reg_date'],
  135. 'first_payment' => $item['first_payment'],
  136. 'agent_name' => $item['agent_name'],
  137. 'auth_name' => $item['auth_name'],
  138. ];
  139. }, $data);
  140. // 是否首次付费
  141. $filter = [
  142. 'first_payment' => [
  143. ['value' => 1, 'label' => '是'],
  144. ['value' => 0, 'label' => '否']
  145. ]
  146. ];
  147. $file_name = '充值明细_' . date('YmdHis') . '.xlsx';
  148. $header = ['订单号', '用户名', '渠道ID', '广告位ID', '游戏名', '服务器ID', '服务器名', '支付方式', '充值金额', '充值时间', '注册时间', '是否首次付费', '渠道名', '负责人'];
  149. $writer = new OpenSpoutWriter($file_name);
  150. $writer->setWidth([15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15]);
  151. $writer->setHeader($header);
  152. // $writer->setData($data, null);
  153. $writer->setData($data, null, $filter);
  154. $file_path = $writer->returnFile();
  155. return response()->download($file_path, urlencode($file_name));
  156. }
  157. // 充值明细导出
  158. public function getRechargeDetailList($where, $type = "list"): mixed
  159. {
  160. $orderBy = request()->input('orderBy', 'pay_date');
  161. $orderType = request()->input('orderType', 'desc');
  162. $page = request()->input('page', 1);
  163. $limit = request()->input('limit', 10);
  164. $params = $this->searchByAuth($where);
  165. $tableName = 'sdk_order_success';
  166. $rechargeDetailQuery = Db::connect('db_game_log')
  167. ->table($tableName)
  168. ->where('pay_date', '>=', $params['pay_date'][0] . ' 00:00:00')
  169. ->where('pay_date', '<=', $params['pay_date'][1] . ' 23:59:59');
  170. if (!empty($params['game_id'])) {
  171. $rechargeDetailQuery = $rechargeDetailQuery->where('game_id', 'in', is_array($params['game_id']) ? implode(',', $params['game_id']) : $params['game_id']);
  172. }
  173. if (!empty($params['user_name'])) {
  174. $rechargeDetailQuery = $rechargeDetailQuery->where('user_name', $params['user_name']);
  175. }
  176. if (!empty($params['media_id'])) {
  177. $rechargeDetailQuery = $rechargeDetailQuery->where('media_id', $params['media_id']);
  178. }
  179. if (!empty($params['auth_id'])) {
  180. $rechargeDetailQuery = $rechargeDetailQuery->where('auth_id', 'in', is_array($params['auth_id']) ? implode(',', $params['auth_id']) : $params['auth_id']);
  181. }
  182. if (!empty($params['agent_id'])) {
  183. $rechargeDetailQuery = $rechargeDetailQuery->where('agent_id', $params['agent_id']);
  184. }
  185. if (!empty($params['site_id'])) {
  186. $rechargeDetailQuery = $rechargeDetailQuery->where('site_id', $params['site_id']);
  187. }
  188. if (!empty($params['server_id'])) {
  189. $rechargeDetailQuery = $rechargeDetailQuery->where('server_id', $params['server_id']);
  190. }
  191. if (!empty($params['server_name'])) {
  192. $rechargeDetailQuery = $rechargeDetailQuery->where('server_name', $params['server_name']);
  193. }
  194. if (!empty($params['reg_date'])) {
  195. $rechargeDetailQuery = $rechargeDetailQuery
  196. ->where('reg_date', '>=', $params['reg_date'][0] . ' 00:00:00')
  197. ->where('reg_date', '<=', $params['reg_date'][1] . ' 23:59:59');
  198. }
  199. // 自然量ID, auth_id=0为自然量
  200. $whereRaw = [];
  201. if(!empty($params['nomal_game_id'])){
  202. for($i=0;$i<count($params['nomal_game_id']);$i++){
  203. $whereRaw[] = "(game_id = {$params['nomal_game_id'][$i]} AND auth_id=0)";
  204. }
  205. }
  206. $whereRaw = implode(' OR ', $whereRaw);
  207. $rechargeDetailQuery = $rechargeDetailQuery->whereRaw($whereRaw);
  208. $rechargeDetailQuery->order($orderBy, $orderType);
  209. if ($type == 'all') {
  210. $data = $rechargeDetailQuery->select()->toArray();
  211. } else {
  212. $data = $rechargeDetailQuery->paginate($limit)->toArray();
  213. }
  214. return $data;
  215. }
  216. // 充值排行
  217. public function getRechargeRankList($where): mixed
  218. {
  219. $orderBy = request()->input('orderBy', 'searchTotalMoney');
  220. $orderType = request()->input('orderType', 'desc');
  221. $page = request()->input('page', 1);
  222. $limit = request()->input('limit', 10);
  223. $params = $this->searchByAuth($where);
  224. $tableName = 'sdk_order_success';
  225. $rechargeRankQuery = Db::connect('db_game_log')
  226. ->table($tableName)
  227. ->where('pay_date', '>=', $params['pay_date'][0] . ' 00:00:00')
  228. ->where('pay_date', '<=', $params['pay_date'][1] . ' 23:59:59');
  229. if (!empty($params['game_id'])) {
  230. $rechargeRankQuery = $rechargeRankQuery->where('game_id', 'in', is_array($params['game_id']) ? implode(',', $params['game_id']) : $params['game_id']);
  231. }
  232. if (!empty($params['user_name'])) {
  233. $rechargeRankQuery = $rechargeRankQuery->where('user_name', $params['user_name']);
  234. }
  235. if (!empty($params['media_id'])) {
  236. $rechargeRankQuery = $rechargeRankQuery->where('media_id', $params['media_id']);
  237. }
  238. if (!empty($params['auth_id'])) {
  239. $rechargeRankQuery = $rechargeRankQuery->where('auth_id', 'in', is_array($params['auth_id']) ? implode(',', $params['auth_id']) : $params['auth_id']);
  240. }
  241. if (!empty($params['agent_id'])) {
  242. $rechargeRankQuery = $rechargeRankQuery->where('agent_id', $params['agent_id']);
  243. }
  244. if (!empty($params['site_id'])) {
  245. $rechargeRankQuery = $rechargeRankQuery->where('site_id', $params['site_id']);
  246. }
  247. if (!empty($params['server_id'])) {
  248. $rechargeRankQuery = $rechargeRankQuery->where('server_id', $params['server_id']);
  249. }
  250. if (!empty($params['server_name'])) {
  251. $rechargeRankQuery = $rechargeRankQuery->where('server_name', $params['server_name']);
  252. }
  253. if (!empty($params['reg_date'])) {
  254. $rechargeRankQuery = $rechargeRankQuery
  255. ->where('reg_date', '>=', $params['reg_date'][0] . ' 00:00:00')
  256. ->where('reg_date', '<=', $params['reg_date'][1] . ' 23:59:59');
  257. }
  258. // 自然量ID, auth_id=0为自然量
  259. $whereRaw = [];
  260. if(!empty($params['nomal_game_id'])){
  261. for($i=0;$i<count($params['nomal_game_id']);$i++){
  262. $whereRaw[] = "(game_id = {$params['nomal_game_id'][$i]} AND auth_id=0)";
  263. }
  264. }
  265. $whereRaw = implode(' OR ', $whereRaw);
  266. $rechargeRankQuery = $rechargeRankQuery->whereRaw($whereRaw);
  267. // 先选择字段,包括计算字段
  268. $rechargeRankQuery->field('user_name,sum(money) as searchTotalMoney,
  269. order_id,
  270. game_id,
  271. media_id,
  272. agent_id,
  273. site_id,
  274. uid,
  275. pay_channel_id,
  276. auth_id,
  277. server_id,
  278. server_name,
  279. pay_date,
  280. reg_date,
  281. role_name,
  282. role_id');
  283. $rechargeRankQuery->group('uid, game_id,site_id,server_id,media_id,auth_id,agent_id,server_id,auth_id,role_id');
  284. $rechargeRankQuery->order($orderBy, $orderType);
  285. $data = $rechargeRankQuery->paginate($limit)->toArray();
  286. $data['data'] = $this->trandformListColumn($data['data'], ['game', 'auth', 'agent', 'pay_channel']);
  287. // // 告警提示,查询最近登录时间,查询最近充值时间,和现在时间对比,如果超过3天,则告警,并提示
  288. foreach ($data['data'] as &$item) {
  289. // 最后登录时间
  290. $item['login_time'] = Db::connect('db_origin')->table('user_' . $item['uid'] % 10)->where('uid', $item['uid'])->value('login_time');
  291. // 最近注册时间
  292. $item['pay_time'] = Db::connect('db_origin')->table('user_' . $item['uid'] % 10)->where('uid', $item['uid'])->value('pay_time');
  293. // 现在时间
  294. $item['now_time'] = time();
  295. // 最近登录时间与现在时间对比,如果超过3天,则告警,并提示
  296. if ($item['login_time'] < $item['now_time'] - 3 * 24 * 60 * 60) {
  297. $item['login_alert'] = 1;
  298. } else {
  299. $item['login_alert'] = 0;
  300. }
  301. // 最近充值时间与现在时间对比,如果超过3天,则告警,并提示
  302. if ($item['pay_time'] < $item['now_time'] - 3 * 24 * 60 * 60) {
  303. $item['pay_alert'] = 1;
  304. } else {
  305. $item['pay_alert'] = 0;
  306. }
  307. }
  308. $totalWhere = [];
  309. if (!empty($params['game_id'])) {
  310. $totalWhere[] = ['game_id', 'in', is_array($params['game_id']) ? implode(',', $params['game_id']) : $params['game_id']];
  311. }
  312. if (!empty($where['auth_id'])) {
  313. $totalWhere[] = ['auth_id', 'in', is_array($params['auth_id']) ? implode(',', $params['auth_id']) : $params['auth_id']];
  314. }
  315. if (!empty($where['agent_id'])) {
  316. $totalWhere['agent_id'] = $params['agent_id'];
  317. }
  318. if (!empty($where['site_id'])) {
  319. $totalWhere['site_id'] = $params['site_id'];
  320. }
  321. if (!empty($where['server_id'])) {
  322. $totalWhere['server_id'] = $params['server_id'];
  323. }
  324. if (!empty($where['user_name'])) {
  325. $totalWhere['user_name'] = $params['user_name'];
  326. }
  327. if (!empty($where['server_name'])) {
  328. $totalWhere['server_name'] = $params['server_name'];
  329. }
  330. if (!empty($where['reg_date'])) {
  331. $totalWhere['reg_date'] = $params['reg_date'];
  332. }
  333. // 自然量ID, auth_id=0为自然量
  334. $whereRaw = [];
  335. if(!empty($params['nomal_game_id'])){
  336. for($i=0;$i<count($params['nomal_game_id']);$i++){
  337. $whereRaw[] = "(game_id = {$params['nomal_game_id'][$i]} AND auth_id=0)";
  338. }
  339. }
  340. $whereRaw = implode(' OR ', $whereRaw);
  341. $totalList = Db::connect('db_game_log')->table('sdk_order_success')
  342. ->where($totalWhere)
  343. ->whereRaw($whereRaw)
  344. ->field('game_id,uid,sum(money) as totalMoney')
  345. ->group('uid,game_id,site_id,server_id,media_id,auth_id,agent_id,server_id,auth_id,role_id')
  346. ->select()->toArray();
  347. // 查询累计充值金额(对每一行数据单独查询历史充值金额)
  348. $totalMoneyMap = [];
  349. if (!empty($data['data']) && !empty($totalList)) {
  350. foreach ($totalList as $totalItem) {
  351. $key = $totalItem['uid'] . '_' . $totalItem['game_id'];
  352. $totalMoneyMap[$key] = $totalItem['totalMoney'] ?? 0;
  353. }
  354. }
  355. // 为搜索结果添加累计充值金额
  356. foreach ($data['data'] as &$item) {
  357. $key = $item['uid'] . '_' . $item['game_id'];
  358. $item['totalMoney'] = $totalMoneyMap[$key] ?? 0;
  359. }
  360. return $data;
  361. }
  362. // 角色数据
  363. public function getRoleDataList($where): mixed
  364. {
  365. $orderBy = request()->input('orderBy', 'reg_time');
  366. $orderType = request()->input('orderType', 'desc');
  367. $page = request()->input('page', 1);
  368. $limit = request()->input('limit', 10);
  369. $params = $this->searchByAuth($where);
  370. $regTime = $params['reg_time'];
  371. $egTableName = 'sdk_reg_log_' . date('Ym', strtotime($regTime));
  372. $userData = Db::connect('db_game_log')->table($egTableName)->where('reg_time', '>=', strtotime($regTime . ' 00:00:00'))->where('reg_time', '<=', strtotime($regTime . ' 23:59:59'))->select()->toArray();
  373. $where_sql = $this->generateWhereSql($params);
  374. $uids = array_column($userData, 'uid');
  375. if (empty($userData)) {
  376. return [];
  377. }
  378. $where_sql = " AND uid in (" . implode(',', $uids) . ") {$where_sql}";
  379. $sql_parts = [];
  380. $sql_parts[] = "SELECT * FROM role_data_and WHERE 1=1 {$where_sql}";
  381. $sql_parts[] = "SELECT * FROM role_data_ios WHERE 1=1 {$where_sql}";
  382. $unionSql = implode(" UNION ALL ", $sql_parts);
  383. // 分页
  384. $offset = ($page - 1) * $limit;
  385. $unionSql .= " LIMIT {$offset}, {$limit}";
  386. $roleData = Db::connect('db_game_log')->query($unionSql);
  387. // 合并相同uid的数据,将角色信息放到数组中
  388. $mergedData = [];
  389. foreach ($roleData as $item) {
  390. $uid = $item['uid'];
  391. if (!isset($mergedData[$uid])) {
  392. // 创建用户基础信息,排除角色相关字段
  393. $baseInfo = array_diff_key($item, array_flip(['role_id', 'role_name', 'role_level']));
  394. $mergedData[$uid] = $baseInfo + ['roles' => []];
  395. }
  396. // 添加角色信息到数组
  397. $mergedData[$uid]['roles'][] = [
  398. 'server_name' => $item['server_name'],
  399. 'role_name' => $item['role_name'],
  400. 'role_level' => $item['role_level'],
  401. ];
  402. }
  403. // 转换回数组格式
  404. $roleData = array_values($mergedData);
  405. $roleData = $this->trandformListColumn($roleData, ['game', 'ip', 'auth', 'agent']);
  406. $count = count($roleData);
  407. return [
  408. 'data' => $roleData,
  409. 'current_page' => $page,
  410. 'per_page' => $limit,
  411. 'last_page' => ceil($count / $limit),
  412. 'has_more' => $page < ceil($count / $limit),
  413. 'total' => $count
  414. ];
  415. }
  416. }