-
Notifications
You must be signed in to change notification settings - Fork 3
Expand file tree
/
Copy pathGlobalExceptionHandler.java
More file actions
118 lines (107 loc) · 5.09 KB
/
GlobalExceptionHandler.java
File metadata and controls
118 lines (107 loc) · 5.09 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
package com.involutionhell.backend.common.error;
import cn.dev33.satoken.exception.NotLoginException;
import cn.dev33.satoken.exception.NotPermissionException;
import cn.dev33.satoken.exception.NotRoleException;
import com.involutionhell.backend.common.api.ApiResponse;
import jakarta.validation.ConstraintViolationException;
import java.util.Optional;
import org.springframework.http.HttpStatus;
import org.springframework.http.ResponseEntity;
import org.springframework.validation.BindException;
import org.springframework.web.bind.MethodArgumentNotValidException;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.RestControllerAdvice;
@RestControllerAdvice
public class GlobalExceptionHandler {
// ==========================================
// Sa-Token 异常拦截
// ==========================================
/**
* Sa-Token: 拦截未登录异常
*/
@ExceptionHandler(NotLoginException.class)
public ResponseEntity<ApiResponse<Void>> handleNotLoginException(NotLoginException e) {
// 判断场景值,定制化异常信息
String message = switch (e.getType()) {
case NotLoginException.NOT_TOKEN -> "未提供 Token";
case NotLoginException.INVALID_TOKEN -> "Token 无效";
case NotLoginException.TOKEN_TIMEOUT -> "Token 已过期";
case NotLoginException.BE_REPLACED -> "Token 已被顶下线";
case NotLoginException.KICK_OUT -> "Token 已被踢下线";
case NotLoginException.TOKEN_FREEZE -> "Token 已被冻结";
case NotLoginException.NO_PREFIX -> "未按照指定前缀提交 Token";
default -> "当前会话未登录";
};
return ResponseEntity.status(HttpStatus.UNAUTHORIZED)
.body(ApiResponse.fail(message));
}
/**
* Sa-Token: 拦截权限不足异常
*/
@ExceptionHandler(NotPermissionException.class)
public ResponseEntity<ApiResponse<Void>> handleNotPermissionException(NotPermissionException e) {
return ResponseEntity.status(HttpStatus.FORBIDDEN)
.body(ApiResponse.fail("拒绝访问: 缺少权限 [" + e.getCode() + "]"));
}
/**
* Sa-Token: 拦截角色不足异常
*/
@ExceptionHandler(NotRoleException.class)
public ResponseEntity<ApiResponse<Void>> handleNotRoleException(NotRoleException e) {
return ResponseEntity.status(HttpStatus.FORBIDDEN)
.body(ApiResponse.fail("拒绝访问: 缺少角色 [" + e.getRole() + "]"));
}
// ==========================================
// 业务与通用异常拦截
// ==========================================
/**
* 将参数校验异常转换为 400 响应
*/
@ExceptionHandler({
MethodArgumentNotValidException.class,
BindException.class,
ConstraintViolationException.class
})
public ResponseEntity<ApiResponse<Void>> handleValidation(Exception exception) {
return ResponseEntity.badRequest()
.body(ApiResponse.fail(resolveValidationMessage(exception)));
}
/**
* 将业务校验异常转换为 400 响应
*/
@ExceptionHandler({IllegalArgumentException.class, IllegalStateException.class})
public ResponseEntity<ApiResponse<Void>> handleBusiness(Exception exception) {
return ResponseEntity.badRequest().body(ApiResponse.fail(exception.getMessage()));
}
/**
* 兜底处理未预期异常并返回 500 响应
*/
@ExceptionHandler(Exception.class)
public ResponseEntity<ApiResponse<Void>> handleUnexpected(Exception exception) {
exception.printStackTrace(); // 建议在开发阶段打印堆栈,生产环境应使用日志框架
return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR)
.body(ApiResponse.fail("服务器内部错误"));
}
/**
* 提取不同校验异常中的首个可读错误信息
*/
private String resolveValidationMessage(Exception exception) {
if (exception instanceof MethodArgumentNotValidException methodArgumentNotValidException) {
return Optional.ofNullable(methodArgumentNotValidException.getBindingResult().getFieldError())
.map(fieldError -> fieldError.getField() + ": " + fieldError.getDefaultMessage())
.orElse("请求参数不合法");
}
if (exception instanceof BindException bindException) {
return Optional.ofNullable(bindException.getBindingResult().getFieldError())
.map(fieldError -> fieldError.getField() + ": " + fieldError.getDefaultMessage())
.orElse("请求参数不合法");
}
if (exception instanceof ConstraintViolationException constraintViolationException) {
return constraintViolationException.getConstraintViolations().stream()
.findFirst()
.map(violation -> violation.getPropertyPath() + ": " + violation.getMessage())
.orElse("请求参数不合法");
}
return "请求参数不合法";
}
}