UserLogLogic.php 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338
  1. <?php
  2. // 玩家日志逻
  3. namespace app\v1\logic\dataReport;
  4. use plugin\saiadmin\exception\ApiException;
  5. use plugin\saiadmin\basic\BaseLogic;
  6. use plugin\saiadmin\service\OpenSpoutWriter;
  7. use support\think\Db;
  8. class UserLogLogic extends BaseLogic
  9. {
  10. // 注册日志
  11. public function getRegLogList($where): mixed
  12. {
  13. $orderBy = request()->input('orderBy', 'reg_time');
  14. $orderType = request()->input('orderType', 'desc');
  15. $params = $this->searchByAuth($where);
  16. $limit = request()->input('limit', 10);
  17. // 检查注册时间只能查询3个月内
  18. if (empty($params['reg_time']) || count($params['reg_time']) != 2) {
  19. throw new ApiException('请选择注册时间范围');
  20. }
  21. $startTime = strtotime($params['reg_time'][0]);
  22. $endTime = strtotime($params['reg_time'][1]);
  23. if ($endTime < $startTime) {
  24. throw new ApiException('注册时间范围不正确');
  25. }
  26. // 计算时间差,限制为3个月(90天)内
  27. $maxDays = 90;
  28. if (($endTime - $startTime) > ($maxDays * 86400)) {
  29. throw new ApiException('注册时间只能查询3个月内');
  30. }
  31. // 公共处理完的where, 自然量ID, auth_id=0为自然量
  32. [$where, $whereRaw] = $this->getUlogCommonWhere($params);
  33. $monthRange = getMonthsBetweenDates($params['reg_time'][0], $params['reg_time'][1]);
  34. $db = Db::connect('db_game_log');
  35. $unionQuery = [];
  36. foreach ($monthRange as $month){
  37. $tableName = 'sdk_reg_log_' . $month;
  38. $unionQuery[] = $db->table($tableName)->where($where)->whereRaw(where: $whereRaw)->buildSql();
  39. }
  40. $fullSql = "(" . implode(' UNION ALL ', $unionQuery) . ") as unTable";
  41. return $db->table($fullSql)->order($orderBy, $orderType)->paginate($limit)->toArray();
  42. }
  43. // 登录日志
  44. public function getLoginLogList($where): mixed
  45. {
  46. $orderBy = request()->input('orderBy', 'login_time');
  47. $orderType = request()->input('orderType', 'desc');
  48. $page = request()->input('page', 1);
  49. $limit = request()->input('limit', 10);
  50. $params = $this->searchByAuth($where);
  51. // 检查注册时间只能查询3个月内
  52. if (empty($params['login_time']) || count($params['login_time']) != 2) {
  53. throw new ApiException('请选择登录时间范围');
  54. }
  55. $startTime = strtotime($params['login_time'][0]);
  56. $endTime = strtotime($params['login_time'][1]);
  57. if ($endTime < $startTime) {
  58. throw new ApiException('登录时间范围不正确');
  59. }
  60. // 计算时间差,限制为3个月(90天)内
  61. $maxDays = 90;
  62. if (($endTime - $startTime) > ($maxDays * 86400)) {
  63. throw new ApiException('登录时间只能查询3个月内');
  64. }
  65. // 检查登录时间只能查询3个月内
  66. if(!empty($params['reg_time'])){
  67. $startTime = strtotime($params['reg_time'][0]);
  68. $endTime = strtotime($params['reg_time'][1]);
  69. if ($endTime < $startTime) {
  70. throw new ApiException('注册时间范围不正确');
  71. }
  72. // 计算时间差,限制为3个月(90天)内
  73. $maxDays = 90;
  74. if (($endTime - $startTime) > ($maxDays * 86400)) {
  75. throw new ApiException('注册时间只能查询3个月内');
  76. }
  77. }
  78. // 公共处理完的where, 自然量ID, auth_id=0为自然量
  79. [$where, $whereRaw] = $this->getUlogCommonWhere($params);
  80. $monthRange = getMonthsBetweenDates($params['login_time'][0], $params['login_time'][1]);
  81. $db = Db::connect('db_game_log');
  82. $unionQuery = [];
  83. foreach ($monthRange as $month){
  84. $tableName = 'sdk_login_log_' . $month;
  85. $unionQuery[] = $db->table($tableName)->where($where)->whereRaw(where: $whereRaw)->buildSql();
  86. }
  87. $fullSql = "(" . implode(' UNION ALL ', $unionQuery) . ") as unTable";
  88. return $db->table($fullSql)->order($orderBy, $orderType)->paginate($limit)->toArray();
  89. }
  90. // 充值明细
  91. public function getRechargeDetailList($where, $type = "list"): mixed
  92. {
  93. $orderBy = request()->input('orderBy', 'pay_date');
  94. $orderType = request()->input('orderType', 'desc');
  95. $page = request()->input('page', 1);
  96. $limit = request()->input('limit', 10);
  97. $params = $this->searchByAuth($where);
  98. $tableName = 'sdk_order_success';
  99. $rechargeDetailQuery = Db::connect('db_game_log')->table($tableName);
  100. // 公共处理完的where, 自然量ID, auth_id=0为自然量
  101. [$where, $whereRaw] = $this->getUlogCommonWhere($params);
  102. if($where){
  103. $rechargeDetailQuery = $rechargeDetailQuery->where($where);
  104. }
  105. if($whereRaw){
  106. $rechargeDetailQuery = $rechargeDetailQuery->whereRaw($whereRaw);
  107. }
  108. $rechargeDetailQuery->order($orderBy, $orderType);
  109. if ($type == 'all') {
  110. $data = $rechargeDetailQuery->select()->toArray();
  111. } else {
  112. $data = $rechargeDetailQuery->paginate($limit)->toArray();
  113. }
  114. return $data;
  115. }
  116. // 充值明细导出
  117. public function exportRechargeDetailList($where): mixed
  118. {
  119. $data = $this->getRechargeDetailList($where, 'all');
  120. $data = $this->trandformListColumn($data, ['game', 'ip', 'agent', 'auth', 'pay_channel']);
  121. $data = array_map(function ($item) {
  122. return [
  123. 'order_id' => $item['order_id'],
  124. 'user_name' => $item['user_name'],
  125. 'agent_id' => $item['agent_id'],
  126. 'site_id' => $item['site_id'],
  127. 'game_name' => $item['game_name'],
  128. 'server_id' => $item['server_id'],
  129. 'server_name' => $item['server_name'],
  130. 'pay_channel_name' => $item['pay_channel_name'],
  131. 'money' => $item['money'],
  132. 'pay_date' => $item['pay_date'],
  133. 'reg_date' => $item['reg_date'],
  134. 'first_payment' => $item['first_payment'],
  135. 'agent_name' => $item['agent_name'],
  136. 'auth_name' => $item['auth_name'],
  137. ];
  138. }, $data);
  139. // 是否首次付费
  140. $filter = [
  141. 'first_payment' => [
  142. ['value' => 1, 'label' => '是'],
  143. ['value' => 0, 'label' => '否']
  144. ]
  145. ];
  146. $file_name = '充值明细_' . date('YmdHis') . '.xlsx';
  147. $header = ['订单号', '用户名', '渠道ID', '广告位ID', '游戏名', '服务器ID', '服务器名', '支付方式', '充值金额', '充值时间', '注册时间', '是否首次付费', '渠道名', '负责人'];
  148. $writer = new OpenSpoutWriter($file_name);
  149. $writer->setWidth([15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15, 15]);
  150. $writer->setHeader($header);
  151. $writer->setData($data, null, $filter);
  152. $file_path = $writer->returnFile();
  153. return response()->download($file_path, urlencode($file_name));
  154. }
  155. // 充值排行
  156. public function getRechargeRankList($where): mixed
  157. {
  158. $orderBy = request()->input('orderBy', 'searchTotalMoney');
  159. $orderType = request()->input('orderType', 'desc');
  160. $page = request()->input('page', 1);
  161. $limit = request()->input('limit', 10);
  162. $params = $this->searchByAuth($where);
  163. $tableName = 'sdk_order_success';
  164. $rechargeRankQuery = Db::connect('db_game_log')->table($tableName);
  165. // 公共处理完的where, 自然量ID, auth_id=0为自然量
  166. [$where, $whereRaw] = $this->getUlogCommonWhere($params);
  167. if($where){
  168. $rechargeRankQuery = $rechargeRankQuery->where($where);
  169. }
  170. if($whereRaw){
  171. $rechargeRankQuery = $rechargeRankQuery->whereRaw($whereRaw);
  172. }
  173. // 先选择字段,包括计算字段
  174. $rechargeRankQuery->field('user_name,sum(money) as searchTotalMoney,
  175. order_id,
  176. game_id,
  177. media_id,
  178. agent_id,
  179. site_id,
  180. uid,
  181. pay_channel_id,
  182. auth_id,
  183. server_id,
  184. server_name,
  185. pay_date,
  186. reg_date,
  187. role_name,
  188. role_id');
  189. $rechargeRankQuery->group('uid, game_id,site_id,server_id,media_id,auth_id,agent_id,server_id,auth_id,role_id');
  190. $rechargeRankQuery->order($orderBy, $orderType);
  191. $data = $rechargeRankQuery->paginate($limit)->toArray();
  192. $data['data'] = $this->trandformListColumn($data['data'], ['game', 'auth', 'agent', 'pay_channel']);
  193. // 告警提示,查询最近登录时间,查询最近充值时间,和现在时间对比,如果超过3天,则告警,并提示
  194. foreach ($data['data'] as &$item) {
  195. // 最后登录时间
  196. $item['login_time'] = Db::connect('db_origin')->table('user_' . $item['uid'] % 10)->where('uid', $item['uid'])->value('login_time');
  197. // 最近注册时间
  198. $item['pay_time'] = Db::connect('db_origin')->table('user_' . $item['uid'] % 10)->where('uid', $item['uid'])->value('pay_time');
  199. // 现在时间
  200. $item['now_time'] = time();
  201. // 最近登录时间与现在时间对比,如果超过3天,则告警,并提示
  202. if ($item['login_time'] < $item['now_time'] - 3 * 86400) {
  203. $item['login_alert'] = 1;
  204. } else {
  205. $item['login_alert'] = 0;
  206. }
  207. // 最近充值时间与现在时间对比,如果超过3天,则告警,并提示
  208. if ($item['pay_time'] < $item['now_time'] - 3 * 86400) {
  209. $item['pay_alert'] = 1;
  210. } else {
  211. $item['pay_alert'] = 0;
  212. }
  213. }
  214. $totalList = Db::connect('db_game_log')->table('sdk_order_success')
  215. ->where($where)
  216. ->whereRaw($whereRaw)
  217. ->field('game_id,uid,sum(money) as totalMoney')
  218. ->group('uid,game_id,site_id,server_id,media_id,auth_id,agent_id,server_id,auth_id,role_id')
  219. ->select()->toArray();
  220. // 查询累计充值金额(对每一行数据单独查询历史充值金额)
  221. $totalMoneyMap = [];
  222. if (!empty($data['data']) && !empty($totalList)) {
  223. foreach ($totalList as $totalItem) {
  224. $key = $totalItem['uid'] . '_' . $totalItem['game_id'];
  225. $totalMoneyMap[$key] = $totalItem['totalMoney'] ?? 0;
  226. }
  227. }
  228. // 为搜索结果添加累计充值金额
  229. foreach ($data['data'] as &$item) {
  230. $key = $item['uid'] . '_' . $item['game_id'];
  231. $item['totalMoney'] = $totalMoneyMap[$key] ?? 0;
  232. }
  233. return $data;
  234. }
  235. // 角色数据
  236. public function getRoleDataList($where): mixed
  237. {
  238. $page = request()->input('page', 1);
  239. $limit = request()->input('limit', 10);
  240. $params = $this->searchByAuth($where);
  241. $where_sql = $this->generateWhereSql($params);
  242. $where_sql = str_replace('AND reg_time', 'AND create_time', $where_sql);
  243. $sql_parts = [];
  244. $sql_parts[] = "SELECT * FROM role_data_and WHERE 1=1 {$where_sql}";
  245. $sql_parts[] = "SELECT * FROM role_data_ios WHERE 1=1 {$where_sql}";
  246. $unionSql = implode(" UNION ALL ", $sql_parts);
  247. // 分页
  248. $offset = ($page - 1) * $limit;
  249. $unionSql .= " LIMIT {$offset}, {$limit}";
  250. $roleData = Db::connect('db_game_log')->query($unionSql);
  251. // 合并相同uid的数据,将角色信息放到数组中
  252. $mergedData = [];
  253. foreach ($roleData as $item) {
  254. $uid = $item['uid'];
  255. if (!isset($mergedData[$uid])) {
  256. // 创建用户基础信息,排除角色相关字段
  257. $baseInfo = array_diff_key($item, array_flip(['role_id', 'role_name', 'role_level']));
  258. $mergedData[$uid] = $baseInfo + ['roles' => []];
  259. }
  260. // 添加角色信息到数组
  261. $mergedData[$uid]['roles'][] = [
  262. 'server_name' => $item['server_name'],
  263. 'role_name' => $item['role_name'],
  264. 'role_level' => $item['role_level'],
  265. ];
  266. }
  267. // 转换回数组格式
  268. $roleData = array_values($mergedData);
  269. $roleData = $this->trandformListColumn($roleData, ['game', 'ip', 'auth', 'agent']);
  270. $count = count($roleData);
  271. return [
  272. 'data' => $roleData,
  273. 'current_page' => $page,
  274. 'per_page' => $limit,
  275. 'last_page' => ceil($count / $limit),
  276. 'has_more' => $page < ceil($count / $limit),
  277. 'total' => $count
  278. ];
  279. }
  280. }