springmvc请求参数异常统一处理,结合钉钉报告信息定位bug位置
参考之前一篇博客:springmvc请求参数异常统一处理
1、ExceptionHandlerController
package com.oy.controller; import java.text.MessageFormat;
import java.util.ResourceBundle; import javax.servlet.http.HttpServletRequest; import org.springframework.beans.TypeMismatchException;
import org.springframework.http.HttpStatus;
import org.springframework.http.converter.HttpMessageNotReadableException;
import org.springframework.web.HttpRequestMethodNotSupportedException;
import org.springframework.web.bind.MissingServletRequestParameterException;
import org.springframework.web.bind.annotation.ControllerAdvice;
import org.springframework.web.bind.annotation.ExceptionHandler;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.bind.annotation.ResponseStatus;
import org.springframework.web.servlet.mvc.multiaction.NoSuchRequestHandlingMethodException; import com.alibaba.fastjson.JSONObject; import com.oy.exception.ForbiddenException;
import com.oy.exception.JwebException;
import com.oy.utils.ErrCode;
import com.oy.utils.Response;
import com.oy.utils.UtilFunctions; @ControllerAdvice
public class ExceptionHandlerController { @ExceptionHandler(RuntimeException.class)
@ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
@ResponseBody
public JSONObject runtimeExceptionHandler(RuntimeException ex) {
UtilFunctions.log.error("runtimeExceptionHandler, msg: {}, exception: {}", ex.toString(), ex);
UtilFunctions.reportError("runtimeExceptionHandler: " + ex.toString(), ex);
JSONObject response = new JSONObject();
response.put("message", "Internal Server Error");
return response;
} @ExceptionHandler(NullPointerException.class)
@ResponseStatus(HttpStatus.INTERNAL_SERVER_ERROR)
@ResponseBody
public JSONObject nullPointerExceptionHandler(NullPointerException ex) {
UtilFunctions.log.error("nullPointerExceptionHandler, msg: {}, exception: {}", ex.toString(), ex);
UtilFunctions.reportError("runtimeExceptionHandler: " + ex.toString(), ex);
JSONObject response = new JSONObject();
response.put("message", "Internal Server Error");
return response;
} /*----- REQUEST ERROR -----*/
@ExceptionHandler({ ForbiddenException.class })
@ResponseStatus(HttpStatus.FORBIDDEN)
@ResponseBody
public JSONObject requestForbidden(ForbiddenException ex) {
UtilFunctions.log.error("ForbiddenExceptionHandler, msg: {}, exception: {}", ex.toString(), ex);
JSONObject response = new JSONObject();
response.put("message", ex.getMessage());
return response;
} @ExceptionHandler({ TypeMismatchException.class })
@ResponseStatus(HttpStatus.BAD_REQUEST)
@ResponseBody
public JSONObject requestTypeMismatch(TypeMismatchException ex) {
UtilFunctions.log.error("TypeMismatchExceptionHandler, msg: {}, exception: {}", ex.toString(), ex);
JSONObject response = new JSONObject();
// response.put("message", "Bad Request");
// response.put("message", "Bad Request, parameter type of " + ex.getPropertyName() + " need be " + ex.getRequiredType()); if (Double.class.equals(ex.getRequiredType()) || Integer.class.equals(ex.getRequiredType())) {
response.put("message", "Bad Request, " + ex.getValue() + " not a number");
} else {
String strTemplate = "Bad Request, {0} is invalid, a type of {1} is needed";
response.put("message", MessageFormat.format(strTemplate, ex.getValue(), ex.getRequiredType().getName()));
}
return response;
} @ExceptionHandler({ MissingServletRequestParameterException.class })
@ResponseStatus(HttpStatus.BAD_REQUEST)
@ResponseBody
public JSONObject requestMissingServletRequest(MissingServletRequestParameterException ex) {
UtilFunctions.log.error("MissingServletRequestParameterExceptionHandler, msg: {}, exception: {}", ex.toString(), ex);
JSONObject response = new JSONObject();
// response.put("message", "Bad Request");
String strTemplate = "Bad Request, param:{0} is required, type:{1}";
response.put("message", MessageFormat.format(strTemplate, ex.getParameterName(), ex.getParameterType()));
return response;
} @ExceptionHandler({ NoSuchRequestHandlingMethodException.class })
@ResponseStatus(HttpStatus.BAD_REQUEST)
@ResponseBody
public JSONObject NoSuchRequestHandlingMethodExceptionHandler(NoSuchRequestHandlingMethodException ex) {
UtilFunctions.log.error("NoSuchRequestHandlingMethodExceptionHandler, msg: {}, exception: {}", ex.toString(), ex);
JSONObject response = new JSONObject();
response.put("message", "Not Found");
return response;
} /*----- REQUEST ERROR -----*/
@ExceptionHandler({ HttpMessageNotReadableException.class })
@ResponseStatus(HttpStatus.BAD_REQUEST)
@ResponseBody
public JSONObject requestNotReadable(HttpMessageNotReadableException ex) {
UtilFunctions.log.error("HttpMessageNotReadableExceptionHandler, msg: {}, exception: {}", ex.toString(), ex);
JSONObject response = new JSONObject();
response.put("message", "Bad Request");
return response;
} @ExceptionHandler({ HttpRequestMethodNotSupportedException.class })
@ResponseStatus(HttpStatus.METHOD_NOT_ALLOWED)
@ResponseBody
public JSONObject request405(HttpRequestMethodNotSupportedException ex) {
UtilFunctions.log.error("HttpRequestMethodNotSupportedExceptionHandler, msg: {}, exception: {}", ex.toString(), ex);
JSONObject response = new JSONObject();
// response.put("message", "Method Not Allowed");
response.put("message", ex.getMessage());
return response;
} @ExceptionHandler({ JwebException.class })
@ResponseStatus(HttpStatus.OK)
@ResponseBody
public JSONObject jwebExceptionHandler(JwebException ex, HttpServletRequest request) {
UtilFunctions.log.error("jwebExceptionHandler, info: {}, exception: {}", ex.getMessage(), ex);
UtilFunctions.reportError("jwebExceptionHandler, " + ex.toString(), null);
//JSONObject response = new JSONObject();
//response.put("code", ErrCode.SER_INTERNAL_ERR);
//response.put("message", ex.getMessage());
//response.put("message", "The system is busy. Please try again later.");
ResourceBundle resourceBundle = (ResourceBundle) request.getAttribute("resourceBundle");
String message = UtilFunctions.getMessage(resourceBundle, "SYSTEM_BUSY");
return new Response(ErrCode.SER_INTERNAL_ERR, message).toJson();
}
}
2、springmvc全局异常捕获,整合钉钉打印异常相关信息
public class UtilFunctions {
    public static Logger log = LoggerFactory.getLogger("jweb");
    public static void reportError(String msg, Exception err) {
        if (err != null) {
            log.error("msg:{}, err:{}", msg, err);
        }
        JSONObject content = new JSONObject();
        //content.put("content", "hostname:" + Config.HOSTNAME + "\n" + "errMsg:" + msg);
        content.put("content", "hostname:" + Config.HOSTNAME + "\n" + "errMsg:" + msg + "\n" + getStackTraceInfo(err));
        JSONObject obj = new JSONObject();
        obj.put("msgtype", "text");
        obj.put("text", content.toString());
        String query = obj.toString();
        try {
            URL url = new URL("https://oapi.dingtalk.com/robot/send?access_token=" + Config.DINGTOKEN);
            HttpURLConnection connection = (HttpURLConnection) url.openConnection();
            connection.setDoInput(true);
            connection.setDoOutput(true);
            connection.setRequestMethod("POST");
            connection.setUseCaches(false);
            connection.setInstanceFollowRedirects(true);
            connection.setRequestProperty("Content-Type", "application/json");
            connection.connect();
            try (OutputStream os = connection.getOutputStream()) {
                os.write(query.getBytes("UTF-8"));
            }
            try (BufferedReader reader = new BufferedReader(new InputStreamReader(connection.getInputStream()))) {
            }
            connection.disconnect();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    public static String printStackTraceToString(Throwable t) {
     if (t == null) {
            return "";
        }
        StringWriter sw = null;
        PrintWriter pw = null;
        try {
            sw = new StringWriter();
            pw =  new PrintWriter(sw);
            t.printStackTrace(pw);
            pw.flush();
            sw.flush();
            return sw.getBuffer().toString();
        } finally {
            if (sw != null) {
                try {
                    sw.close();
                } catch (IOException e1) {
                    e1.printStackTrace();
                }
            }
            if (pw != null) {
                pw.close();
            }
        }
    }
    public static String getStackTraceInfo(Throwable t) {
        String stackTraceInfo = "";
        String stackTrace = printStackTraceToString(t);
        String regex = "at com.oy(\\S*)";
        Pattern pattern = Pattern.compile(regex);
        Matcher matcher = pattern.matcher(stackTrace);
        while (matcher.find()) {
            stackTraceInfo += matcher.group() + "\n";
            // UtilFunctions.log.info("===== matcher.group:{}" + matcher.group());
        }
        return stackTraceInfo;
    }
}
3、测试
3.1、代码出现RuntimeException,比如 int a = 1/0; 钉钉报告信息:
hostname:APC058
errMsg:runtimeExceptionHandler:java.lang.ArithmeticException:/by zero
at com.oy.controller.TradeController.getOrder(TradeController.java:220)
通过钉钉报告信息可以定位bug位置是TradeController#getOrder()方法220行。
3.2、手动抛出JwebException
注意:不要轻易抛出JwebException,即使时try...catch...捕获异常时。因为抛出JwebException,则程序交给ExceptionHandlerController来统一处理,并且程序中断。
需要根据业务来判断,比如需求:登陆成功后记录登陆信息,要记录ip及国家城市信息,如果查询不到城市信息,如果抛出JwebException,则导致登陆失败,这是不可以的。此种情况可以记录log,发送钉钉。
if (true) {
    String errMsg = MessageFormat.format("TradeController#getOrder error, tradeOrder:{0}, uid:{1}, hashId:{2}", tradeOrder, uid, hashId);
    throw new JwebException(errMsg);
}
钉钉报告信息:
hostname:APC058
errMsg:jwebExceptionHandler, com.oy.exception.JwebException:TradeController#getOrder error, tradeOrder:null, uid:222, hashId:xxx
通过钉钉报告信息可以定位bug位置是TradeController#getOrder()方法,tradeOrder为null。
4、其他
4.1、上面ExceptionHandlerController中捕获JwebException后,通过注解@ResponseStatus(HttpStatus.OK)给前端返回http状态码200,并返回json数据:
{
    "code": 50005,
    "message": "The system is busy. Please try again later!"
}
4.2、前面代码中ResourceBundle是用来支持国际化的。
4.3、将异常栈信息保存为一个字符串
public static String printStackTraceToString(Throwable t) {
     if (t == null) {
        return "";
    }
    StringWriter sw = null;
    PrintWriter pw = null;
    try {
        sw = new StringWriter();
        pw =  new PrintWriter(sw);
        t.printStackTrace(pw);
        pw.flush();
        sw.flush();
        return sw.getBuffer().toString();
    } finally {
        if (sw != null) {
            try {
                sw.close();
            } catch (IOException e1) {
                e1.printStackTrace();
            }
        }
        if (pw != null) {
            pw.close();
        }
    }
}
4.4、从异常栈信息中截取需要的数据
@Test
public void test3() {
try{
int a = 1/0;
} catch(Exception e) {
System.out.println(printStackTraceToString(e));
getStackTraceInfo(e);
}
} public static String getStackTraceInfo(Throwable t) {
String stackTraceInfo = ""; String stackTrace = printStackTraceToString(t);
String regex = "com.oy.Hello(\\S*)";
// String regex = "at (\\S*)$"; // 截取异常栈最后一句。$表示字符串结尾
Pattern pattern = Pattern.compile(regex); Matcher matcher = pattern.matcher(stackTrace);
while (matcher.find()) {
stackTraceInfo += matcher.group();
System.out.println("=====matcher.group:{}" + matcher.group());
}
return stackTraceInfo;
}
springmvc请求参数异常统一处理,结合钉钉报告信息定位bug位置的更多相关文章
- springmvc请求参数异常统一处理
		1.ExceptionHandlerController package com.oy.controller; import java.text.MessageFormat; import org.s ... 
- SpringMVC请求参数接收总结
		前提 在日常使用SpringMVC进行开发的时候,有可能遇到前端各种类型的请求参数,这里做一次相对全面的总结.SpringMVC中处理控制器参数的接口是HandlerMethodArgumentRes ... 
- SpringMVC请求参数接收总结(一)
		前提 在日常使用SpringMVC进行开发的时候,有可能遇到前端各种类型的请求参数,这里做一次相对全面的总结.SpringMVC中处理控制器参数的接口是HandlerMethodArgumentRes ... 
- SpringMVC请求参数总结
		前提 在日常使用SpringMVC进行开发的时候,有可能遇到前端各种类型的请求参数,这里做一次相对全面的总结.SpringMVC中处理控制器参数的接口是HandlerMethodArgumentRes ... 
- 2.5万字长文简单总结SpringMVC请求参数接收
		这是公众号<Throwable文摘>发布的第22篇原创文章,暂时收录于专辑<架构与实战>.暂定下一篇发布的长文是<图文分析JUC同步器框架>,下一篇发布的短文是&l ... 
- springmvc请求参数异常处理
		接着上一篇<springmvc 通过异常增强返回给客户端统一格式>讲通过spring ControllerAdvice对各种异常进行拦截处理,统一格式返回给客户端. 接下来我们更精细的讲, ... 
- springmvc 请求参数解析细节
		springmvc 的请求流程,相信大家已经很熟悉了,不熟悉的同学可以参考下资料! 有了整体流程的概念,是否对其中的实现细节就很清楚呢?我觉得不一定,比如:单是参数解析这块,就是个大学问呢? 首先,我 ... 
- SpringMVC请求参数注解两个小问题
		今天遇到使用SpringMVC请求注解遇到的两个小问题: 如果用@requestBody注解,则请求体内容类型一般要为application/json,如果其类型为multipart/form-dat ... 
- SpringMVC请求参数和响应结果全局加密和解密
		前提 前段时间在做一个对外的网关项目,涉及到加密和解密模块,这里详细分析解决方案和适用的场景.为了模拟真实的交互场景,先定制一下整个交互流程.第三方传输(包括请求和响应)数据报文包括三个部分: 1.t ... 
随机推荐
- 【神经网络与深度学习】深度学习实战——caffe windows 下训练自己的网络模型
			1.相关准备 1.1 手写数字数据集 这篇博客上有.jpg格式的图片下载,附带标签信息,有需要的自行下载,博客附带百度云盘下载地址(手写数字.jpg 格式):http://blog.csdn.net/ ... 
- 深度解析Maven
			此文来源于: https://www.cnblogs.com/hafiz/p/8119964.html 带你深度解析Maven 一.What`s Maven? Maven是基于项目对象模型(POM ... 
- java.math包简介
			java.math包提供了java中的数学类 包括基本的浮点库.复杂运算以及任意精度的数据运算 '可以看得到,主要包括三个类一个枚举 BigDecimal和BigInteger接下来会详细介绍 先 ... 
- DVWA、 DSVM 环境搭建简述
			DVWA(http://www.dvwa.co.uk/)是一个本地的漏洞演示环境基于PHP和Mysql . 另外还有基于Python的DSVM部署起来也十分简单.(https://github.com ... 
- [转帖]SSH命令总结
			SSH命令总结 ssh-keygen ssh-copy-id 等命令自己用过 但是知道的不系统 也知道 转发命令 但是也只是知道一点点... ttps://www.cnblogs.com/chenfa ... 
- css精灵图使用
			1. 精灵技术的使用 CSS 精灵其实是将网页中的一些背景图像整合到一张大图中(精灵图),然而,各个网页元素通常只需要精灵图中不同位置的某个小图,要想精确定位到精灵图中的某个小图,就需要使用CSS的b ... 
- Jquery复习(七)之尺寸
			jQuery 尺寸 方法 jQuery 提供多个处理尺寸的重要方法: width() height() innerWidth() innerHeight() outerWidth() outerHei ... 
- mysql优化--explain关键字
			MySQL性能优化---EXPLAIN 参见:https://blog.csdn.net/jiadajing267/article/details/81269067 参见:https://www.cn ... 
- 十二、LaTex中数学公式多行排版
- redis数据库如何用Django框架缓存数据
			---恢复内容开始--- 一.python 使用redis 1.1 安装 pip install redis 测试有一些基本的数据类型 import redis # redis 是一个缓存数据库 # ... 
