ThirdPartyLogController.php 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. <?php
  2. namespace App\Module\ThirdParty\AdminControllers;
  3. use UCore\DcatAdmin\AdminController;
  4. use App\Module\ThirdParty\Models\ThirdPartyLog;
  5. use App\Module\ThirdParty\Models\ThirdPartyService;
  6. use App\Module\ThirdParty\Repositorys\ThirdPartyLogRepository;
  7. use App\Module\ThirdParty\Enums\LOG_LEVEL;
  8. use Dcat\Admin\Grid;
  9. use Dcat\Admin\Show;
  10. /**
  11. * 第三方服务调用日志管理控制器
  12. *
  13. * 路由: /admin/thirdparty/logs
  14. */
  15. class ThirdPartyLogController extends AdminController
  16. {
  17. /**
  18. * 页面标题
  19. *
  20. * @var string
  21. */
  22. protected $title = '调用日志管理';
  23. /**
  24. * 数据仓库
  25. *
  26. * @return string
  27. */
  28. protected function repository()
  29. {
  30. return ThirdPartyLogRepository::class;
  31. }
  32. /**
  33. * 列表页面
  34. *
  35. * @return Grid
  36. */
  37. protected function grid(): Grid
  38. {
  39. return Grid::make(new ThirdPartyLogRepository(), function (Grid $grid) {
  40. // 基础设置
  41. $grid->column('id', 'ID')->sortable();
  42. // 关联服务
  43. $grid->column('service.name', '服务名称')->sortable();
  44. $grid->column('service.code', '服务代码');
  45. $grid->column('request_id', '请求ID')->copyable();
  46. // HTTP方法
  47. $grid->column('method', '方法')->display(function ($method) {
  48. $colors = [
  49. 'GET' => 'success',
  50. 'POST' => 'primary',
  51. 'PUT' => 'warning',
  52. 'DELETE' => 'danger',
  53. 'PATCH' => 'info',
  54. ];
  55. $color = $colors[$method] ?? 'secondary';
  56. return "<span class='badge badge-{$color}'>{$method}</span>";
  57. });
  58. // URL(截断显示)
  59. $grid->column('url', 'URL')->display(function ($url) {
  60. $shortUrl = strlen($url) > 50 ? substr($url, 0, 50) . '...' : $url;
  61. return "<span title='{$url}'>{$shortUrl}</span>";
  62. });
  63. // 响应状态码
  64. $grid->column('response_status', '状态码')->display(function ($status) {
  65. if (!$status) {
  66. return '<span class="badge badge-secondary">-</span>';
  67. }
  68. $color = 'secondary';
  69. if ($status >= 200 && $status < 300) {
  70. $color = 'success';
  71. } elseif ($status >= 300 && $status < 400) {
  72. $color = 'info';
  73. } elseif ($status >= 400 && $status < 500) {
  74. $color = 'warning';
  75. } elseif ($status >= 500) {
  76. $color = 'danger';
  77. }
  78. return "<span class='badge badge-{$color}'>{$status}</span>";
  79. });
  80. // 响应时间
  81. $grid->column('response_time', '响应时间')->display(function ($time) {
  82. if (!$time) {
  83. return '-';
  84. }
  85. $color = 'success';
  86. if ($time > 5000) {
  87. $color = 'danger';
  88. } elseif ($time > 2000) {
  89. $color = 'warning';
  90. } elseif ($time > 1000) {
  91. $color = 'info';
  92. }
  93. return "<span class='badge badge-{$color}'>{$time}ms</span>";
  94. })->sortable();
  95. // 日志级别
  96. $grid->column('level', '级别')->display(function ($level) {
  97. $logLevel = LOG_LEVEL::tryFrom($level);
  98. if ($logLevel) {
  99. $color = $logLevel->getColor();
  100. $icon = $logLevel->getIcon();
  101. $label = $logLevel->getLabel();
  102. return "<span class='badge badge-{$color}'><i class='{$icon}'></i> {$label}</span>";
  103. }
  104. return $level;
  105. });
  106. // 错误信息(截断显示)
  107. $grid->column('error_message', '错误信息')->display(function ($error) {
  108. if (!$error) {
  109. return '-';
  110. }
  111. $shortError = strlen($error) > 30 ? substr($error, 0, 30) . '...' : $error;
  112. return "<span class='text-danger' title='{$error}'>{$shortError}</span>";
  113. });
  114. // 用户信息
  115. $grid->column('user_id', '用户ID');
  116. $grid->column('ip_address', 'IP地址');
  117. // 创建时间
  118. $grid->column('created_at', '调用时间')->sortable();
  119. // 过滤器
  120. $grid->filter(function (Grid\Filter $filter) {
  121. $filter->equal('service_id', '服务')->select(
  122. ThirdPartyService::pluck('name', 'id')->toArray()
  123. );
  124. $filter->equal('method', 'HTTP方法')->select([
  125. 'GET' => 'GET',
  126. 'POST' => 'POST',
  127. 'PUT' => 'PUT',
  128. 'DELETE' => 'DELETE',
  129. 'PATCH' => 'PATCH',
  130. ]);
  131. $filter->equal('level', '日志级别')->select(LOG_LEVEL::getOptions());
  132. $filter->between('response_status', '状态码');
  133. $filter->between('response_time', '响应时间(ms)');
  134. $filter->like('request_id', '请求ID');
  135. $filter->like('url', 'URL');
  136. $filter->like('error_message', '错误信息');
  137. $filter->equal('user_id', '用户ID');
  138. $filter->like('ip_address', 'IP地址');
  139. $filter->between('created_at', '调用时间')->datetime();
  140. });
  141. // 批量操作
  142. $grid->batchActions(function (Grid\Tools\BatchActions $batch) {
  143. // TODO: 创建批量操作类
  144. // $batch->add('批量删除', new \App\Module\ThirdParty\AdminActions\BatchDeleteLogAction());
  145. });
  146. // 工具栏
  147. $grid->tools(function (Grid\Tools $tools) {
  148. $tools->append('<a href="/admin/thirdparty/logs/stats" class="btn btn-sm btn-info"><i class="fa fa-chart-bar"></i> 统计分析</a>');
  149. $tools->append('<a href="/admin/thirdparty/logs/export" class="btn btn-sm btn-success"><i class="fa fa-download"></i> 导出日志</a>');
  150. $tools->append('<a href="/admin/thirdparty/logs/cleanup" class="btn btn-sm btn-warning"><i class="fa fa-trash"></i> 清理日志</a>');
  151. });
  152. // 行操作
  153. $grid->actions(function (Grid\Displayers\Actions $actions) {
  154. // 移除编辑按钮,日志只读
  155. $actions->disableEdit();
  156. if ($actions->row->error_message) {
  157. $actions->append('<a href="/admin/thirdparty/logs/' . $actions->getKey() . '/retry" class="btn btn-xs btn-warning"><i class="fa fa-redo"></i> 重试</a>');
  158. }
  159. });
  160. // 默认排序(最新的在前)
  161. $grid->model()->orderBy('created_at', 'desc');
  162. // 禁用创建按钮
  163. $grid->disableCreateButton();
  164. });
  165. }
  166. /**
  167. * 详情页面
  168. *
  169. * @return Show
  170. */
  171. protected function detail($id): Show
  172. {
  173. return Show::make($id, new ThirdPartyLogRepository(), function (Show $show) {
  174. $show->field('id', 'ID');
  175. $show->field('service.name', '服务名称');
  176. $show->field('service.code', '服务代码');
  177. $show->field('credential.name', '使用凭证');
  178. $show->field('request_id', '请求ID');
  179. $show->field('method', 'HTTP方法');
  180. $show->field('url', '请求URL');
  181. $show->field('headers', '请求头')->json();
  182. $show->field('params', '请求参数')->json();
  183. $show->field('body', '请求体')->code();
  184. $show->field('response_status', '响应状态码');
  185. $show->field('response_headers', '响应头')->json();
  186. $show->field('response_body', '响应体')->code();
  187. $show->field('response_time', '响应时间')->as(function ($time) {
  188. return $time ? $time . ' ms' : '-';
  189. });
  190. $show->field('level', '日志级别')->as(function ($level) {
  191. $logLevel = LOG_LEVEL::tryFrom($level);
  192. return $logLevel ? $logLevel->getLabel() : $level;
  193. });
  194. $show->field('error_message', '错误信息');
  195. $show->field('user_id', '用户ID');
  196. $show->field('ip_address', 'IP地址');
  197. $show->field('user_agent', 'User Agent');
  198. $show->field('created_at', '调用时间');
  199. // 相关日志
  200. $show->relation('relatedLogs', '相关日志', function ($model) {
  201. $grid = new Grid(new ThirdPartyLog());
  202. $grid->model()->where('service_id', $model->service_id)
  203. ->where('id', '!=', $model->id)
  204. ->orderBy('created_at', 'desc')
  205. ->limit(10);
  206. $grid->column('request_id', '请求ID');
  207. $grid->column('method', '方法');
  208. $grid->column('response_status', '状态码');
  209. $grid->column('response_time', '响应时间(ms)');
  210. $grid->column('level', '级别');
  211. $grid->column('created_at', '时间');
  212. $grid->disableCreateButton();
  213. $grid->disableActions();
  214. $grid->disableFilter();
  215. $grid->disableBatchActions();
  216. return $grid;
  217. });
  218. });
  219. }
  220. }