|
@@ -4,6 +4,7 @@ import cn.hutool.core.exceptions.ExceptionUtil;
|
|
|
import cn.hutool.core.map.MapUtil;
|
|
|
import cn.hutool.core.util.ObjUtil;
|
|
|
import cn.hutool.core.util.StrUtil;
|
|
|
+import cn.hutool.extra.servlet.JakartaServletUtil;
|
|
|
import cn.iocoder.yudao.framework.apilog.core.service.ApiErrorLogFrameworkService;
|
|
|
import cn.iocoder.yudao.framework.common.exception.ServiceException;
|
|
|
import cn.iocoder.yudao.framework.common.exception.util.ServiceExceptionUtil;
|
|
@@ -14,9 +15,14 @@ import cn.iocoder.yudao.framework.common.util.monitor.TracerUtils;
|
|
|
import cn.iocoder.yudao.framework.common.util.servlet.ServletUtils;
|
|
|
import cn.iocoder.yudao.framework.web.core.util.WebFrameworkUtils;
|
|
|
import cn.iocoder.yudao.module.infra.api.logger.dto.ApiErrorLogCreateReqDTO;
|
|
|
+import com.fasterxml.jackson.databind.exc.InvalidFormatException;
|
|
|
+import jakarta.servlet.http.HttpServletRequest;
|
|
|
+import jakarta.validation.ConstraintViolation;
|
|
|
+import jakarta.validation.ConstraintViolationException;
|
|
|
+import jakarta.validation.ValidationException;
|
|
|
import lombok.AllArgsConstructor;
|
|
|
import lombok.extern.slf4j.Slf4j;
|
|
|
-import org.apache.commons.lang3.exception.ExceptionUtils;
|
|
|
+import org.springframework.http.converter.HttpMessageNotReadableException;
|
|
|
import org.springframework.security.access.AccessDeniedException;
|
|
|
import org.springframework.util.Assert;
|
|
|
import org.springframework.validation.BindException;
|
|
@@ -28,16 +34,18 @@ import org.springframework.web.bind.annotation.ExceptionHandler;
|
|
|
import org.springframework.web.bind.annotation.RestControllerAdvice;
|
|
|
import org.springframework.web.method.annotation.MethodArgumentTypeMismatchException;
|
|
|
import org.springframework.web.servlet.NoHandlerFoundException;
|
|
|
+import org.springframework.web.servlet.resource.NoResourceFoundException;
|
|
|
|
|
|
-import javax.servlet.http.HttpServletRequest;
|
|
|
-import javax.validation.ConstraintViolation;
|
|
|
-import javax.validation.ConstraintViolationException;
|
|
|
-import javax.validation.ValidationException;
|
|
|
import java.time.LocalDateTime;
|
|
|
import java.util.Map;
|
|
|
import java.util.Set;
|
|
|
|
|
|
-import static cn.iocoder.yudao.framework.common.exception.enums.GlobalErrorCodeConstants.*;
|
|
|
+import static cn.iocoder.yudao.framework.common.exception.enums.GlobalErrorCodeConstants.BAD_REQUEST;
|
|
|
+import static cn.iocoder.yudao.framework.common.exception.enums.GlobalErrorCodeConstants.FORBIDDEN;
|
|
|
+import static cn.iocoder.yudao.framework.common.exception.enums.GlobalErrorCodeConstants.INTERNAL_SERVER_ERROR;
|
|
|
+import static cn.iocoder.yudao.framework.common.exception.enums.GlobalErrorCodeConstants.METHOD_NOT_ALLOWED;
|
|
|
+import static cn.iocoder.yudao.framework.common.exception.enums.GlobalErrorCodeConstants.NOT_FOUND;
|
|
|
+import static cn.iocoder.yudao.framework.common.exception.enums.GlobalErrorCodeConstants.NOT_IMPLEMENTED;
|
|
|
|
|
|
/**
|
|
|
* 全局异常处理器,将 Exception 翻译成 CommonResult + 对应的异常编号
|
|
@@ -87,7 +95,10 @@ public class GlobalExceptionHandler {
|
|
|
return validationException((ValidationException) ex);
|
|
|
}
|
|
|
if (ex instanceof NoHandlerFoundException) {
|
|
|
- return noHandlerFoundExceptionHandler(request, (NoHandlerFoundException) ex);
|
|
|
+ return noHandlerFoundExceptionHandler((NoHandlerFoundException) ex);
|
|
|
+ }
|
|
|
+ if (ex instanceof NoResourceFoundException) {
|
|
|
+ return noResourceFoundExceptionHandler(request, (NoResourceFoundException) ex);
|
|
|
}
|
|
|
if (ex instanceof HttpRequestMethodNotSupportedException) {
|
|
|
return httpRequestMethodNotSupportedExceptionHandler((HttpRequestMethodNotSupportedException) ex);
|
|
@@ -119,7 +130,7 @@ public class GlobalExceptionHandler {
|
|
|
*/
|
|
|
@ExceptionHandler(MethodArgumentTypeMismatchException.class)
|
|
|
public CommonResult<?> methodArgumentTypeMismatchExceptionHandler(MethodArgumentTypeMismatchException ex) {
|
|
|
- log.warn("[missingServletRequestParameterExceptionHandler]", ex);
|
|
|
+ log.warn("[methodArgumentTypeMismatchExceptionHandler]", ex);
|
|
|
return CommonResult.error(BAD_REQUEST.getCode(), String.format("请求参数类型错误:%s", ex.getMessage()));
|
|
|
}
|
|
|
|
|
@@ -145,6 +156,22 @@ public class GlobalExceptionHandler {
|
|
|
return CommonResult.error(BAD_REQUEST.getCode(), String.format("请求参数不正确:%s", fieldError.getDefaultMessage()));
|
|
|
}
|
|
|
|
|
|
+ /**
|
|
|
+ * 处理 SpringMVC 请求参数类型错误
|
|
|
+ *
|
|
|
+ * 例如说,接口上设置了 @RequestBody实体中 xx 属性类型为 Integer,结果传递 xx 参数类型为 String
|
|
|
+ */
|
|
|
+ @ExceptionHandler(HttpMessageNotReadableException.class)
|
|
|
+ public CommonResult<?> methodArgumentTypeInvalidFormatExceptionHandler(HttpMessageNotReadableException ex) {
|
|
|
+ log.warn("[methodArgumentTypeInvalidFormatExceptionHandler]", ex);
|
|
|
+ if(ex.getCause() instanceof InvalidFormatException) {
|
|
|
+ InvalidFormatException invalidFormatException = (InvalidFormatException) ex.getCause();
|
|
|
+ return CommonResult.error(BAD_REQUEST.getCode(), String.format("请求参数类型错误:%s", invalidFormatException.getValue()));
|
|
|
+ }else {
|
|
|
+ return defaultExceptionHandler(ServletUtils.getRequest(), ex);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
/**
|
|
|
* 处理 Validator 校验不通过产生的异常
|
|
|
*/
|
|
@@ -173,11 +200,20 @@ public class GlobalExceptionHandler {
|
|
|
* 2. spring.mvc.static-path-pattern 为 /statics/**
|
|
|
*/
|
|
|
@ExceptionHandler(NoHandlerFoundException.class)
|
|
|
- public CommonResult<?> noHandlerFoundExceptionHandler(HttpServletRequest req, NoHandlerFoundException ex) {
|
|
|
+ public CommonResult<?> noHandlerFoundExceptionHandler(NoHandlerFoundException ex) {
|
|
|
log.warn("[noHandlerFoundExceptionHandler]", ex);
|
|
|
return CommonResult.error(NOT_FOUND.getCode(), String.format("请求地址不存在:%s", ex.getRequestURL()));
|
|
|
}
|
|
|
|
|
|
+ /**
|
|
|
+ * 处理 SpringMVC 请求地址不存在
|
|
|
+ */
|
|
|
+ @ExceptionHandler(NoResourceFoundException.class)
|
|
|
+ private CommonResult<?> noResourceFoundExceptionHandler(HttpServletRequest req, NoResourceFoundException ex) {
|
|
|
+ log.warn("[noResourceFoundExceptionHandler]", ex);
|
|
|
+ return CommonResult.error(NOT_FOUND.getCode(), String.format("请求地址不存在:%s", ex.getResourcePath()));
|
|
|
+ }
|
|
|
+
|
|
|
/**
|
|
|
* 处理 SpringMVC 请求方法不正确
|
|
|
*
|
|
@@ -240,7 +276,7 @@ public class GlobalExceptionHandler {
|
|
|
// 情况二:处理异常
|
|
|
log.error("[defaultExceptionHandler]", ex);
|
|
|
// 插入异常日志
|
|
|
- this.createExceptionLog(req, ex);
|
|
|
+ createExceptionLog(req, ex);
|
|
|
// 返回 ERROR CommonResult
|
|
|
return CommonResult.error(INTERNAL_SERVER_ERROR.getCode(), INTERNAL_SERVER_ERROR.getMsg());
|
|
|
}
|
|
@@ -266,7 +302,7 @@ public class GlobalExceptionHandler {
|
|
|
errorLog.setExceptionName(e.getClass().getName());
|
|
|
errorLog.setExceptionMessage(ExceptionUtil.getMessage(e));
|
|
|
errorLog.setExceptionRootCauseMessage(ExceptionUtil.getRootCauseMessage(e));
|
|
|
- errorLog.setExceptionStackTrace(ExceptionUtils.getStackTrace(e));
|
|
|
+ errorLog.setExceptionStackTrace(ExceptionUtil.stacktraceToString(e));
|
|
|
StackTraceElement[] stackTraceElements = e.getStackTrace();
|
|
|
Assert.notEmpty(stackTraceElements, "异常 stackTraceElements 不能为空");
|
|
|
StackTraceElement stackTraceElement = stackTraceElements[0];
|
|
@@ -279,12 +315,12 @@ public class GlobalExceptionHandler {
|
|
|
errorLog.setApplicationName(applicationName);
|
|
|
errorLog.setRequestUrl(request.getRequestURI());
|
|
|
Map<String, Object> requestParams = MapUtil.<String, Object>builder()
|
|
|
- .put("query", ServletUtils.getParamMap(request))
|
|
|
- .put("body", ServletUtils.getBody(request)).build();
|
|
|
+ .put("query", JakartaServletUtil.getParamMap(request))
|
|
|
+ .put("body", JakartaServletUtil.getBody(request)).build();
|
|
|
errorLog.setRequestParams(JsonUtils.toJsonString(requestParams));
|
|
|
errorLog.setRequestMethod(request.getMethod());
|
|
|
errorLog.setUserAgent(ServletUtils.getUserAgent(request));
|
|
|
- errorLog.setUserIp(ServletUtils.getClientIP(request));
|
|
|
+ errorLog.setUserIp(JakartaServletUtil.getClientIP(request));
|
|
|
errorLog.setExceptionTime(LocalDateTime.now());
|
|
|
}
|
|
|
|
|
@@ -301,51 +337,51 @@ public class GlobalExceptionHandler {
|
|
|
}
|
|
|
// 1. 数据报表
|
|
|
if (message.contains("report_")) {
|
|
|
- log.error("[报表模块 yudao-module-report - 表结构未导入][参考 https://doc.iocoder.cn/report/ 开启]");
|
|
|
+ log.error("[报表模块 yudao-module-report - 表结构未导入][参考 https://cloud.iocoder.cn/report/ 开启]");
|
|
|
return CommonResult.error(NOT_IMPLEMENTED.getCode(),
|
|
|
- "[报表模块 yudao-module-report - 表结构未导入][参考 https://doc.iocoder.cn/report/ 开启]");
|
|
|
+ "[报表模块 yudao-module-report - 表结构未导入][参考 https://cloud.iocoder.cn/report/ 开启]");
|
|
|
}
|
|
|
// 2. 工作流
|
|
|
if (message.contains("bpm_")) {
|
|
|
- log.error("[工作流模块 yudao-module-bpm - 表结构未导入][参考 https://doc.iocoder.cn/bpm/ 开启]");
|
|
|
+ log.error("[工作流模块 yudao-module-bpm - 表结构未导入][参考 https://cloud.iocoder.cn/bpm/ 开启]");
|
|
|
return CommonResult.error(NOT_IMPLEMENTED.getCode(),
|
|
|
- "[工作流模块 yudao-module-bpm - 表结构未导入][参考 https://doc.iocoder.cn/bpm/ 开启]");
|
|
|
+ "[工作流模块 yudao-module-bpm - 表结构未导入][参考 https://cloud.iocoder.cn/bpm/ 开启]");
|
|
|
}
|
|
|
// 3. 微信公众号
|
|
|
if (message.contains("mp_")) {
|
|
|
- log.error("[微信公众号 yudao-module-mp - 表结构未导入][参考 https://doc.iocoder.cn/mp/build/ 开启]");
|
|
|
+ log.error("[微信公众号 yudao-module-mp - 表结构未导入][参考 https://cloud.iocoder.cn/mp/build/ 开启]");
|
|
|
return CommonResult.error(NOT_IMPLEMENTED.getCode(),
|
|
|
- "[微信公众号 yudao-module-mp - 表结构未导入][参考 https://doc.iocoder.cn/mp/build/ 开启]");
|
|
|
+ "[微信公众号 yudao-module-mp - 表结构未导入][参考 https://cloud.iocoder.cn/mp/build/ 开启]");
|
|
|
}
|
|
|
// 4. 商城系统
|
|
|
if (StrUtil.containsAny(message, "product_", "promotion_", "trade_")) {
|
|
|
- log.error("[商城系统 yudao-module-mall - 已禁用][参考 https://doc.iocoder.cn/mall/build/ 开启]");
|
|
|
+ log.error("[商城系统 yudao-module-mall - 已禁用][参考 https://cloud.iocoder.cn/mall/build/ 开启]");
|
|
|
return CommonResult.error(NOT_IMPLEMENTED.getCode(),
|
|
|
- "[商城系统 yudao-module-mall - 已禁用][参考 https://doc.iocoder.cn/mall/build/ 开启]");
|
|
|
+ "[商城系统 yudao-module-mall - 已禁用][参考 https://cloud.iocoder.cn/mall/build/ 开启]");
|
|
|
}
|
|
|
// 5. ERP 系统
|
|
|
if (message.contains("erp_")) {
|
|
|
- log.error("[ERP 系统 yudao-module-erp - 表结构未导入][参考 https://doc.iocoder.cn/erp/build/ 开启]");
|
|
|
+ log.error("[ERP 系统 yudao-module-erp - 表结构未导入][参考 https://cloud.iocoder.cn/erp/build/ 开启]");
|
|
|
return CommonResult.error(NOT_IMPLEMENTED.getCode(),
|
|
|
- "[ERP 系统 yudao-module-erp - 表结构未导入][参考 https://doc.iocoder.cn/erp/build/ 开启]");
|
|
|
+ "[ERP 系统 yudao-module-erp - 表结构未导入][参考 https://cloud.iocoder.cn/erp/build/ 开启]");
|
|
|
}
|
|
|
// 6. CRM 系统
|
|
|
if (message.contains("crm_")) {
|
|
|
- log.error("[CRM 系统 yudao-module-crm - 表结构未导入][参考 https://doc.iocoder.cn/crm/build/ 开启]");
|
|
|
+ log.error("[CRM 系统 yudao-module-crm - 表结构未导入][参考 https://cloud.iocoder.cn/crm/build/ 开启]");
|
|
|
return CommonResult.error(NOT_IMPLEMENTED.getCode(),
|
|
|
- "[CRM 系统 yudao-module-crm - 表结构未导入][参考 https://doc.iocoder.cn/crm/build/ 开启]");
|
|
|
+ "[CRM 系统 yudao-module-crm - 表结构未导入][参考 https://cloud.iocoder.cn/crm/build/ 开启]");
|
|
|
}
|
|
|
// 7. 支付平台
|
|
|
if (message.contains("pay_")) {
|
|
|
- log.error("[支付模块 yudao-module-pay - 表结构未导入][参考 https://doc.iocoder.cn/pay/build/ 开启]");
|
|
|
+ log.error("[支付模块 yudao-module-pay - 表结构未导入][参考 https://cloud.iocoder.cn/pay/build/ 开启]");
|
|
|
return CommonResult.error(NOT_IMPLEMENTED.getCode(),
|
|
|
- "[支付模块 yudao-module-pay - 表结构未导入][参考 https://doc.iocoder.cn/pay/build/ 开启]");
|
|
|
+ "[支付模块 yudao-module-pay - 表结构未导入][参考 https://cloud.iocoder.cn/pay/build/ 开启]");
|
|
|
}
|
|
|
// 8. AI 大模型
|
|
|
if (message.contains("ai_")) {
|
|
|
- log.error("[AI 大模型 yudao-module-ai - 表结构未导入][参考 https://doc.iocoder.cn/ai/build/ 开启]");
|
|
|
+ log.error("[AI 大模型 yudao-module-ai - 表结构未导入][参考 https://cloud.iocoder.cn/ai/build/ 开启]");
|
|
|
return CommonResult.error(NOT_IMPLEMENTED.getCode(),
|
|
|
- "[AI 大模型 yudao-module-ai - 表结构未导入][参考 https://doc.iocoder.cn/ai/build/ 开启]");
|
|
|
+ "[AI 大模型 yudao-module-ai - 表结构未导入][参考 https://cloud.iocoder.cn/ai/build/ 开启]");
|
|
|
}
|
|
|
return null;
|
|
|
}
|