LogsController.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419
  1. <?php
  2. namespace App\Http\Controllers\Admin;
  3. use App\Components\IP;
  4. use App\Http\Controllers\Controller;
  5. use App\Models\Node;
  6. use App\Models\NodeOnlineIp;
  7. use App\Models\NotificationLog;
  8. use App\Models\Order;
  9. use App\Models\PaymentCallback;
  10. use App\Models\User;
  11. use App\Models\UserBanedLog;
  12. use App\Models\UserCreditLog;
  13. use App\Models\UserDataFlowLog;
  14. use App\Models\UserDataModifyLog;
  15. use Illuminate\Http\Request;
  16. use phpseclib3\Crypt\EC\Curves\brainpoolP160r1;
  17. use function Matrix\diagonal;
  18. class LogsController extends Controller
  19. {
  20. // 订单列表
  21. public function orderList(Request $request)
  22. {
  23. $email = $request->input('email');
  24. $sn = $request->input('sn');
  25. $is_coupon = $request->input('is_coupon');
  26. $is_expire = $request->input('is_expire');
  27. $pay_way = $request->input('pay_way');
  28. $status = $request->input('status');
  29. $range_time = $request->input('range_time');
  30. $sort = $request->input('sort'); // 0-按创建时间降序、1-按创建时间升序
  31. $order_id = $request->input('id');
  32. $trade_no = $request->input('trade_no');
  33. $query = Order::with(['user:id,email', 'goods:id,name', 'coupon:id,name,sn']);
  34. if (isset($email)) {
  35. $query->whereHas('user', static function ($q) use ($email) {
  36. $q->where('email', 'like', '%'.$email.'%');
  37. });
  38. }
  39. if (isset($sn)) {
  40. $query->where('sn', 'like', '%'.$sn.'%');
  41. }
  42. if (isset($is_coupon)) {
  43. if ($is_coupon) {
  44. $query->where('coupon_id', '<>', null);
  45. } else {
  46. $query->whereCouponId(null);
  47. }
  48. }
  49. if (isset($is_expire)) {
  50. $query->whereIsExpire($is_expire);
  51. }
  52. if (isset($pay_way)) {
  53. $query->wherePayWay($pay_way);
  54. }
  55. if (isset($status)) {
  56. $query->whereStatus($status);
  57. }
  58. if (isset($range_time) && $range_time !== ',') {
  59. $range_time = explode(',', $range_time);
  60. $query->where('created_at', '>=', $range_time[0])->where('created_at', '<=', $range_time[1]);
  61. }
  62. if (isset($order_id)) {
  63. $query->whereId($order_id);
  64. }
  65. if ($sort) {
  66. $query->orderBy('id');
  67. } else {
  68. $query->orderByDesc('id');
  69. }
  70. if (isset($trade_no)){
  71. $query->whereHas('payment' , function ($q) use ($trade_no) {
  72. $q->where('trade_no', 'like', '%'.$trade_no.'%');
  73. });
  74. }else {
  75. // $query->payment();
  76. }
  77. // //dump( ['orders' => $query->latest()->paginate(15)->appends($request->except('page'))]); die();
  78. // $data = $query->latest()->paginate(15)->appends($request->except('page'));
  79. // foreach ($data as $key => $v){
  80. // echo $v;
  81. // }
  82. // die();
  83. // var_dump($query->latest()->paginate(15)->appends($request->except('page'))); die();
  84. return view('admin.logs.order', ['orders' => $query->latest()->paginate(15)->appends($request->except('page'))]);
  85. }
  86. // 流量日志
  87. public function trafficLog(Request $request)
  88. {
  89. $port = $request->input('port');
  90. $user_id = $request->input('user_id');
  91. $email = $request->input('email');
  92. $nodeId = $request->input('nodeId');
  93. $startTime = $request->input('startTime');
  94. $endTime = $request->input('endTime');
  95. $lastHours = $request->input('last_hours');
  96. $isAbnormal = $request->input('is_abnormal'); // 新增异常用户筛选
  97. $query = UserDataFlowLog::with(['user', 'node']);
  98. if (isset($port)) {
  99. $query->whereHas('user', static function ($q) use ($port) {
  100. $q->wherePort($port);
  101. });
  102. }
  103. if (isset($user_id)) {
  104. $query->whereUserId($user_id);
  105. }
  106. if (isset($email)) {
  107. $query->whereHas('user', static function ($q) use ($email) {
  108. $q->where('email', 'like', '%'.$email.'%');
  109. });
  110. }
  111. if (isset($nodeId)) {
  112. $query->whereNodeId($nodeId);
  113. }
  114. // 处理时间范围
  115. if ($lastHours) {
  116. $query->where('log_time', '>=', time() - 3600 * intval($lastHours));
  117. } else {
  118. if (isset($startTime)) {
  119. $query->where('log_time', '>=', strtotime($startTime));
  120. }
  121. if (isset($endTime)) {
  122. $query->where('log_time', '<=', strtotime($endTime));
  123. }
  124. }
  125. $baseTrafficBanValue = sysConfig('traffic_ban_value') * GB;
  126. if ($isAbnormal) {
  127. // 只查最近N小时,避免全表扫描
  128. $hours = $lastHours ?: 1;
  129. $since = time() - 3600 * intval($hours);
  130. // 先查出最近N小时内所有流量日志
  131. $logs = UserDataFlowLog::with('node')
  132. ->where('log_time', '>=', $since)
  133. ->get();
  134. // 统计每个用户的"加权流量"
  135. $userWeighted = [];
  136. foreach ($logs as $log) {
  137. $rate = ($log->node && $log->node->traffic_rate) ? $log->node->traffic_rate : 1;
  138. if ($rate <= 0) $rate = 1;
  139. $weighted = ($log->u + $log->d) / $rate;
  140. if (!isset($userWeighted[$log->user_id])) {
  141. $userWeighted[$log->user_id] = 0;
  142. }
  143. $userWeighted[$log->user_id] += $weighted;
  144. }
  145. // 找出超限的用户ID
  146. $abnormalUserIds = [];
  147. foreach ($userWeighted as $userId => $total) {
  148. if ($total > $baseTrafficBanValue) {
  149. $abnormalUserIds[] = $userId;
  150. }
  151. }
  152. // 只查异常用户
  153. $query->whereIn('user_id', $abnormalUserIds);
  154. }
  155. $dataFlowLogs = $query->latest('log_time')->paginate(20)->appends($request->except('page'));
  156. foreach ($dataFlowLogs as $log) {
  157. $rate = ($log->node && $log->node->traffic_rate) ? $log->node->traffic_rate : 1;
  158. if ($rate <= 0) $rate = 1;
  159. $weighted = ($log->u + $log->d) / $rate;
  160. $log->weighted_traffic = $weighted;
  161. $log->is_abnormal = $weighted > $baseTrafficBanValue;
  162. // 先保存原始流量值
  163. $original_u = $log->u;
  164. $original_d = $log->d;
  165. // 转换为可读格式
  166. $log->u = flowAutoShow($original_u);
  167. $log->d = flowAutoShow($original_d);
  168. // 增加转换后的流量显示(转换为1倍流量)
  169. if ($rate != 1) {
  170. $converted_u = $original_u / $rate;
  171. $converted_d = $original_d / $rate;
  172. $log->converted_u = flowAutoShow($converted_u);
  173. $log->converted_d = flowAutoShow($converted_d);
  174. }
  175. $log->log_time = date('Y-m-d H:i:s', $log->log_time);
  176. }
  177. return view('admin.logs.traffic', [
  178. 'totalTraffic' => flowAutoShow($query->sum('u') + $query->sum('d')),
  179. 'dataFlowLogs' => $dataFlowLogs,
  180. 'nodes' => Node::whereStatus(1)->orderByDesc('sort')->latest()->get(),
  181. 'isAbnormal' => $isAbnormal // 传递筛选状态到视图
  182. ]);
  183. }
  184. // 邮件发送日志列表
  185. public function notificationLog(Request $request)
  186. {
  187. $email = $request->input('email');
  188. $type = $request->input('type');
  189. $query = NotificationLog::query();
  190. if (isset($email)) {
  191. $query->where('address', 'like', '%'.$email.'%');
  192. }
  193. if (isset($type)) {
  194. $query->whereType($type);
  195. }
  196. return view('admin.logs.notification', ['notificationLogs' => $query->latest()->paginate(15)->appends($request->except('page'))]);
  197. }
  198. // 在线IP监控(实时)
  199. public function onlineIPMonitor(Request $request, $id = null)
  200. {
  201. $ip = $request->input('ip');
  202. $email = $request->input('email');
  203. $port = $request->input('port');
  204. $nodeId = $request->input('nodeId');
  205. $query = NodeOnlineIp::with(['node:id,name', 'user:id,email'])->where('created_at', '>=', strtotime('-2 minutes'));
  206. if (isset($ip)) {
  207. $query->whereIp($ip);
  208. }
  209. if (isset($email)) {
  210. $query->whereHas('user', static function ($q) use ($email) {
  211. $q->where('email', 'like', '%'.$email.'%');
  212. });
  213. }
  214. if (isset($port)) {
  215. $query->whereHas('user', static function ($q) use ($port) {
  216. $q->wherePort($port);
  217. });
  218. }
  219. if (isset($nodeId)) {
  220. $query->whereHas('node', static function ($q) use ($nodeId) {
  221. $q->whereId($nodeId);
  222. });
  223. }
  224. if (isset($id)) {
  225. $query->whereHas('user', static function ($q) use ($id) {
  226. $q->whereId($id);
  227. });
  228. }
  229. $onlineIPLogs = $query->groupBy('user_id', 'node_id')->latest()->paginate(20)->appends($request->except('page'));
  230. // foreach ($onlineIPLogs as $log) {
  231. // // 跳过上报多IP的
  232. // if ($log->ip === null || strpos($log->ip, ',') !== false) {
  233. // continue;
  234. // }
  235. // //$ipInfo = IP::getIPInfo($log->ip);
  236. //
  237. // //$log->ipInfo = implode(' ', $ipInfo);
  238. // }
  239. return view('admin.logs.onlineIPMonitor', [
  240. 'onlineIPLogs' => $onlineIPLogs,
  241. 'nodes' => Node::whereStatus(1)->orderByDesc('sort')->latest()->get(),
  242. ]);
  243. }
  244. // 用户余额变动记录
  245. public function userCreditLogList(Request $request)
  246. {
  247. $email = $request->input('email');
  248. $query = UserCreditLog::with('user:id,email')->latest();
  249. if (isset($email)) {
  250. $query->whereHas('user', static function ($q) use ($email) {
  251. $q->where('email', 'like', '%'.$email.'%');
  252. });
  253. }
  254. return view('admin.logs.userCreditHistory', ['userCreditLogs' => $query->paginate(15)->appends($request->except('page'))]);
  255. }
  256. // 用户封禁记录
  257. public function userBanLogList(Request $request)
  258. {
  259. $email = $request->input('email');
  260. $isBanned = $request->input('is_banned'); // 新增封禁状态筛选
  261. $query = UserBanedLog::with('user:id,email,t')->latest();
  262. if (isset($email)) {
  263. $query->whereHas('user', static function ($q) use ($email) {
  264. $q->where('email', 'like', '%'.$email.'%');
  265. });
  266. }
  267. // 添加封禁状态筛选
  268. if (isset($isBanned)) {
  269. if ($isBanned == 1) {
  270. $query->where('time', '>', 0); // 有封禁时长的记录
  271. } else {
  272. $query->where('time', 0); // 无封禁时长的记录
  273. }
  274. }
  275. return view('admin.logs.userBanHistory', [
  276. 'userBanLogs' => $query->paginate(15)->appends($request->except('page')),
  277. 'isBanned' => $isBanned // 传递当前筛选状态到视图
  278. ]);
  279. }
  280. // 用户流量变动记录
  281. public function userTrafficLogList(Request $request)
  282. {
  283. $email = $request->input('email');
  284. $query = UserDataModifyLog::with(['user:id,email', 'order.goods:id,name']);
  285. if (isset($email)) {
  286. $query->whereHas('user', static function ($q) use ($email) {
  287. $q->where('email', 'like', '%'.$email.'%');
  288. });
  289. }
  290. return view('admin.logs.userTraffic', ['userTrafficLogs' => $query->latest()->paginate(15)->appends($request->except('page'))]);
  291. }
  292. // 用户在线IP记录
  293. public function userOnlineIPList(Request $request)
  294. {
  295. $email = $request->input('email');
  296. $port = $request->input('port');
  297. $wechat = $request->input('wechat');
  298. $qq = $request->input('qq');
  299. $query = User::activeUser();
  300. if (isset($email)) {
  301. $query->where('email', 'like', '%'.$email.'%');
  302. }
  303. if (isset($wechat)) {
  304. $query->where('wechat', 'like', '%'.$wechat.'%');
  305. }
  306. if (isset($qq)) {
  307. $query->where('qq', 'like', '%'.$qq.'%');
  308. }
  309. if (isset($port)) {
  310. $query->wherePort($port);
  311. }
  312. $userList = $query->paginate(15)->appends($request->except('page'));
  313. $nodeOnlineIPs = NodeOnlineIp::with('node:id,name')->where('created_at', '>=', strtotime('-10 minutes'))->latest()->distinct()->get();
  314. foreach ($userList as $user) {
  315. //Todo node_online_ip表 api可以用user_id
  316. // 最近5条在线IP记录,如果后端设置为60秒上报一次,则为10分钟内的在线IP
  317. $user->onlineIPList = $nodeOnlineIPs->where('port', $user->port)->take(5);
  318. }
  319. return view('admin.logs.userOnlineIP', ['userList' => $userList]);
  320. }
  321. // 用户流量监控
  322. public function userTrafficMonitor(User $user)
  323. {
  324. return view('admin.logs.userMonitor', array_merge(['email' => $user->email], $this->dataFlowChart($user->id)));
  325. }
  326. // 回调日志
  327. public function callbackList(Request $request)
  328. {
  329. $status = $request->input('status', 0);
  330. $out_trade_no = $request->input('out_trade_no', 0);
  331. $query = PaymentCallback::query();
  332. if (isset($status)) {
  333. $query->whereStatus($status);
  334. }
  335. if (isset($out_trade_no)){
  336. $query->where('out_trade_no',"=",$out_trade_no);
  337. }
  338. $data = $query->latest()->paginate(10)->appends($request->except('page'));
  339. // die();
  340. return view('admin.logs.callback', ['callbackLogs' => $data]);
  341. }
  342. }