Handler.php 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. <?php
  2. namespace App\Exceptions;
  3. use App\Utils\IP;
  4. use ErrorException;
  5. use Illuminate\Auth\AuthenticationException;
  6. use Illuminate\Foundation\Exceptions\Handler as ExceptionHandler;
  7. use Illuminate\Http\Client\ConnectionException;
  8. use Illuminate\Http\Request;
  9. use Illuminate\Session\TokenMismatchException;
  10. use Illuminate\Validation\ValidationException;
  11. use Log;
  12. use ReflectionException;
  13. use Response;
  14. use Symfony\Component\HttpKernel\Exception\AccessDeniedHttpException;
  15. use Symfony\Component\HttpKernel\Exception\MethodNotAllowedHttpException;
  16. use Symfony\Component\HttpKernel\Exception\NotFoundHttpException;
  17. use Throwable;
  18. class Handler extends ExceptionHandler
  19. {
  20. protected $dontReport = [
  21. ConnectionException::class,
  22. ValidationException::class,
  23. ];
  24. /**
  25. * The list of the inputs that are never flashed to the session on validation exceptions.
  26. *
  27. * @var array<int, string>
  28. */
  29. protected $dontFlash = [
  30. 'current_password',
  31. 'password',
  32. 'password_confirmation',
  33. ];
  34. /**
  35. * Report or log an exception.
  36. *
  37. * @return void
  38. *
  39. * @throws Throwable
  40. */
  41. public function report(Throwable $exception)
  42. {
  43. if (config('app.debug')) { // 调试模式下记录错误详情
  44. Log::debug('来源:'.url()->full().PHP_EOL.'访问者IP:'.IP::getClientIP().PHP_EOL.$exception);
  45. }
  46. parent::report($exception);
  47. }
  48. /**
  49. * Render an exception into an HTTP response.
  50. *
  51. * @param Request $request
  52. * @return \Symfony\Component\HttpFoundation\Response
  53. *
  54. * @throws Throwable
  55. */
  56. public function render($request, Throwable $exception)
  57. {
  58. // 调试模式下直接返回错误信息,非调试模式下渲染在返回
  59. if (! config('app.debug')) {
  60. switch ($exception) {
  61. case $exception instanceof NotFoundHttpException: // 捕获访问异常
  62. Log::warning('异常请求:'.$request->fullUrl().',IP:'.IP::getClientIp());
  63. if ($request->ajax() || $request->wantsJson()) {
  64. return Response::json(['status' => 'fail', 'message' => trans('http-statuses.404')], 404);
  65. }
  66. return Response::view('auth.error', ['message' => trans('http-statuses.404')], 404);
  67. case $exception instanceof AuthenticationException: // 捕获身份校验异常
  68. if ($request->ajax() || $request->wantsJson()) {
  69. return Response::json(['status' => 'fail', 'message' => trans('http-statuses.401')], 401);
  70. }
  71. return Response::view('auth.error', ['message' => trans('http-statuses.401')], 401);
  72. case $exception instanceof AccessDeniedHttpException: // 捕获权限拒绝异常
  73. if ($request->ajax() || $request->wantsJson()) {
  74. return Response::json(['status' => 'fail', 'message' => trans('http-statuses.401')], 401);
  75. }
  76. return Response::view('auth.error', ['message' => trans('http-statuses.401')], 401);
  77. case $exception instanceof TokenMismatchException: // 捕获CSRF异常
  78. if ($request->ajax() || $request->wantsJson()) {
  79. return Response::json(['status' => 'fail', 'message' => trans('http-statuses.419')], 419);
  80. }
  81. return Response::view('auth.error', ['message' => trans('errors.refresh_page').'<a href="'.route('login').'" target="_blank">'.trans('errors.refresh').'</a>'],
  82. 419);
  83. case $exception instanceof ReflectionException:
  84. if ($request->ajax() || $request->wantsJson()) {
  85. return Response::json(['status' => 'fail', 'message' => trans('http-statuses.500')], 500);
  86. }
  87. return Response::view('auth.error', ['message' => trans('http-statuses.500')], 500);
  88. case $exception instanceof MethodNotAllowedHttpException:
  89. if ($request->ajax() || $request->wantsJson()) {
  90. return Response::json(['status' => 'fail', 'message' => trans('http-statuses.405')], 405);
  91. }
  92. return Response::view('auth.error', ['message' => trans('http-statuses.405')], 405);
  93. case $exception instanceof ErrorException: // 捕获系统错误异常
  94. if ($request->ajax() || $request->wantsJson()) {
  95. return Response::json([
  96. 'status' => 'fail',
  97. 'message' => trans('http-statuses.500').', '.trans('errors.visit').'<a href="'.route('log-viewer::dashboard').'" target="_blank">'.trans('errors.log').'</a>',
  98. ], 500);
  99. }
  100. return Response::view('auth.error',
  101. ['message' => trans('http-statuses.500').', '.trans('errors.visit').'<a href="'.route('log-viewer::dashboard').'" target="_blank">'.trans('errors.log').'</a>'],
  102. 500);
  103. case $exception instanceof ConnectionException:
  104. if ($request->ajax() || $request->wantsJson()) {
  105. return Response::json(['status' => 'fail', 'message' => $exception->getMessage()], 408);
  106. }
  107. return Response::view('auth.error', ['message' => $exception->getMessage()], 408);
  108. default:
  109. if ($request->ajax() || $request->wantsJson()) {
  110. return Response::json(['status' => 'fail', 'message' => $exception->getMessage()], 400);
  111. }
  112. return Response::view('auth.error', ['message' => $exception->getMessage()], 400);
  113. }
  114. }
  115. return parent::render($request, $exception);
  116. }
  117. }