捕获全局异常推送至钉钉群

场景描述

在现有异常捕获的基础上加入异常捕获并推送钉钉群机器人,因为我们项目中使用的全局异常捕获是组件提供的包,所以我们不能自定义全局异常处理类,因为会覆盖掉组件包中的异常处理逻辑(这部分视公司使用特性而定,可能包含AJAX请求、Zipkin链路日志处理)。所以我们要使用java语言类的继承特性,继承组件包,重写日志捕获方法,调用父类包中的日志处理逻辑,加上我们自己的扩展,以此完成该功能。

本项目中异常处理器

/**
 * @Author: zhangdc
 * @Date: 2021/5/27 18:30
 */
@Component
public class GlobalExceptionHandler extends 组件全局异常处理器 {

    @Autowired
    private DingTalkTool dingTalkTool;

    //是否启用环境配置
    @Value("${xxx.xxx.dingTalkEnable}")
    private Boolean dingTalkIsEnable;

    @Override
    protected ModelAndView doResolveHandlerMethodException(HttpServletRequest request, HttpServletResponse response, HandlerMethod handlerMethod, Exception e) {
        //调用父类的默认处理逻辑
        ModelAndView modelAndView = super.doResolveHandlerMethodException(request, response, handlerMethod, e);
        //加上我们自己的扩展
        //是否启用钉钉群机器人异常推送
        if (Boolean.TRUE.equals(dingTalkIsEnable)) {
            try {
                //推送钉钉机器人通知
                dingTalkTool.sendDingTalkMsg();
            } catch (ApiException apiException) {
                apiException.printStackTrace();
            }
        }
        return modelAndView;
    }

}

dingTalk工具

/**
 * @Author: zhangdc
 * @Date: 2021/5/26 15:30
 */
@Component
@Slf4j
public class DingTalkTool {

    //zipkinURL
    @Value("${xxx.xxx.zipkinURL}")
    private String zipkinURL;

    //Webhook
    @Value("${xxx.xxx.dingTalkWebhook}")
    private String dingTalkWebhook;

    //阿波罗配置信息获取
    private static final ConfigUtil CONFIG = ApolloInjector.getInstance(ConfigUtil.class);

    private static final String DATEFORMAT = "yyyy-MM-dd HH:mm:ss";

    private static final String MSGTYPE = "markdown";

    private static final String TITLE = "监控报警通知";

    public void sendDingTalkMsg() throws ApiException {

        //url为钉钉机器人webHook地址
        DefaultDingTalkClient client = new DefaultDingTalkClient(dingTalkWebhook);
        //构建钉钉消息
        OapiRobotSendRequest request = buildMeaages();
        //发送请求到钉钉
        OapiRobotSendResponse response = client.execute(request);
        log.info("=====================钉钉机器人发送消息{}", response.getErrcode());

    }

    private OapiRobotSendRequest buildMeaages() {
        OapiRobotSendRequest request = new OapiRobotSendRequest();
        //通过阿波罗获取环境信息
        Env env = CONFIG.getApolloEnv();
        //获取链路
        String traceId = ZipkinUtils.getTraceId(null);
        //获取当前线程用户信息
        UserInfoEntity userInfo = LoginDataHelper.getUserInfo();
        //获取当前请求对象
        ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
        HttpServletRequest req = attributes.getRequest();

        Date date = new Date();
        SimpleDateFormat sdf = new SimpleDateFormat(DATEFORMAT);

        request.setMsgtype(MSGTYPE);
        //初始化发送的markdown信息
        OapiRobotSendRequest.Markdown markdown = new OapiRobotSendRequest.Markdown();
        markdown.setTitle(TITLE);
        markdown.setText("## 监控报警通知\n" +
                "\n" +
                "> 环境:\n" + env + "\n" +
                "\n" +
                "> 请求方式:\n" + req.getMethod() + "\n" +
                "\n" +
                "> 请求地址:\n" + req.getRequestURI() + "\n" +
                "\n" +
                "> 操作人:" + userInfo.getPersonName() + "\n" +
                "\n" +
                "> email:\n" + userInfo.getEmail() + "\n" +
                "\n" +
                "> 异常时间:\n" + sdf.format(date) + "\n" +
                "\n" +
                "> zipkinURL:\n" + zipkinURL + traceId);
        request.setMarkdown(markdown);
        return request;
    }
}

至此我们已经实现了将异常信息推送至钉钉群的功能,并且包含了开关、多环境的切换,可配置化增强。

以下是钉钉机器人截图:
在这里插入图片描述

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java 中,我们可以通过捕获全局异常来处理程序中未被处理的异常全局异常处理器可以处理任何未被捕获异常,无论是在应用程序的主线程还是在其他线程中发生。 下面是一个简单的示例代码,演示如何使用全局异常处理器: ```java public class GlobalExceptionHandler implements Thread.UncaughtExceptionHandler { @Override public void uncaughtException(Thread t, Throwable e) { // 处理异常的代码 System.out.println("An exception occurred: " + e.getMessage()); } public static void main(String[] args) { // 设置全局异常处理器 Thread.setDefaultUncaughtExceptionHandler(new GlobalExceptionHandler()); // 抛出一个异常 throw new RuntimeException("Oops, something went wrong!"); } } ``` 在上面的代码中,我们定义了一个名为 `GlobalExceptionHandler` 的类,它实现了 `Thread.UncaughtExceptionHandler` 接口,并重写了其中的 `uncaughtException` 方法。在 `main` 方法中,我们通过调用 `Thread.setDefaultUncaughtExceptionHandler` 方法来设置全局异常处理器为 `GlobalExceptionHandler` 类的实例。接下来,我们抛出一个运行时异常,这会导致全局异常处理器被调用。在这个处理器中,我们可以编写代码来处理异常,例如记录日志、发送错误报告等等。 需要注意的是,全局异常处理器只能处理未被捕获异常。如果异常已经被 `try-catch` 块捕获,那么全局异常处理器将不会被调用。因此,在编写代码时,我们应该尽可能地使用 `try-catch` 块来捕获异常,并在必要时使用全局异常处理器来处理未被捕获异常
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值