拦截应用 error日志并发送到钉钉群|Java 开发实战

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 现在应用都需要对日志进行监控或者报警,现在普遍的做法是采用EKL收集日志,然后再由Grafana进行内容展示和及告警策略等,那如果项目架构比较简单(单体应用),又不想搞那么多中间件依赖怎么办,这里有一种简单的方式可以实现~

开篇

现在应用都需要对日志进行监控或者报警,现在普遍的做法是采用EKL收集日志,然后再由Grafana进行内容展示和及告警策略等,那如果项目架构比较简单(单体应用),又不想搞那么多中间件依赖怎么办,这里有一种简单的方式可以实现~

继承UnsynchronizedAppenderBase

Springboot默认集成的是logback,所以自定义Appender非常简单,继承一下AppenderBase类即可。

3e4a8fed7d47221d62f4668cf608b03.png

再来看看AppenderBase之上的Appender下都有哪些实现

UnsynchronizedAppenderBase  ch.qos.logback.core)  
cAsyncAppenderBase (ch.qos.ogback.core) cOutputStreamAppender (ch.qos.logback.core9:DBAppenderBase (ch.qos.logback.core.db)
SendErrorMsgAppender (com.mty.jls.config)(cmAppenderBase (ch.qos.loqback.core)
AbstractServerSocketAppender(ch.qos.logback.core.net.se C:NOPAppender (ch.qos.logback.core.helpers)@=SyslogAppenderBase (ch.gos.logback.core.net)(CSMTPAppenderBase (ch.qos.logback.core.net)@SiftingAppenderBase (ch.qos.logback.core.sift)
CCycicBufferAppender (ch.qos.logback.core.read)@AbstractSocketAppender (ch.qos.logback.core.net)
ListAppender (ch.qos.logback.core.read) 

UnsynchronizedAppenderBase。从名字就能看出来是异步的、普通的、不加锁的。 它类似于AppenderBase,只是派生的Appender实现类需要自己处理线程同步。

演示

定义一个SendErrorMsgAppender

这个类要继承UnsynchronizedAppenderBase,并实现其中的抽象方法,不需要手动让它线程同步,异步就好,这样可以避免出错了导致业务操作回滚。

/**
 * UnsynchronizedAppenderBase 用于异步处理,不阻塞主线程, 拦截error日志,并发送到钉钉群
 */
@Getter
@Setter
public class SendErrorMsgAppender extends UnsynchronizedAppenderBase<ILoggingEvent> {
    // ILoggingEvent里面是日志的内容
    Layout<ILoggingEvent> layout;
    //自定义配置
    String printString;
    @Override
    public void start() {
        //这里可以做些初始化判断 比如layout不能为null ,
        if (layout == null) {
            addWarn("Layout was not defined");
        }
        //或者写入数据库、redis时的初始化连接等等
        super.start();
    }
    @Override
    public void stop() {
        //释放相关资源,如数据库连接,redis线程池等等
        if (!isStarted()) {
            return;
        }
        super.stop();
    }
    @Override
    public void append(ILoggingEvent event) {
        if (event.getLevel() == Level.ERROR) {
            try {
                var isEnableSendLog = "true".equals(SpringUtil.environment(PropertiesConstant.IS_ENABLE_SEND_LOG));
                if (isEnableSendLog) {
                    //获取服务器Ip,告知哪台服务器抛异常
                    var ip = InetAddress.getLocalHost().getHostAddress();
                    var message = ip + "  " + layout.doLayout(event);
                    sendMsgToDingDing(message);
                }
            } catch (UnknownHostException ignored) {
                addWarn("获取服务器ip失败");
            }
        }
    }
    /**
    * 发送日志消息到钉钉的逻辑
    **/
    private void sendMsgToDingDing(String msg) {
        Text text = new Text();
        text.setContent(msg);
        DdMsgBody msgBody = DdMsgBody.builder().msgtype("text").text(text).build();
        Long timestamp = System.currentTimeMillis();
        String secret = SpringUtil.environment(PropertiesConstant.OAPI_DINGTALK_SECRET);
        var charsetName = "UTF-8";
        // 把timestamp+"\n"+密钥当做签名字符串
        String stringToSign = timestamp + "\n" + secret;
        try {
            // 使用HmacSHA256算法计算签名
            Mac mac = Mac.getInstance("HmacSHA256");
            mac.init(new SecretKeySpec(secret.getBytes(charsetName), "HmacSHA256"));
            byte[] signData = mac.doFinal(stringToSign.getBytes());
            // 最后再把签名参数再进行urlEncode,得到最终的签名(需要使用UTF-8字符集)
            String sign = URLEncoder.encode(Base64.getEncoder().encodeToString(signData), charsetName);
            var paramMap = Map.of("timestamp", timestamp, "sign", sign);
            var sendUrl = RestTemplateUtils.buildGetUrlByMap(SpringUtil.environment(PropertiesConstant.OAPI_DINGTALK_URL), paramMap);
            RestTemplateUtils.executeHttpPost(sendUrl, msgBody);
        } catch (Exception e) {
        }
    }
    @Builder
    @Data
    private static class DdMsgBody {
        private String msgtype;
        private Text text;
        private Markdown markdown;
    }
    @Accessors(chain = true)
    @Data
    private class Markdown {
        private String title;
    }
    @Accessors(chain = true)
    @Data
    private class Text {
        private String content;
    }
}
复制代码

追加配置

<appender> 标签的name随意设置,但<appender-ref>需要引用该nameclass属性的值就是刚刚上面定义实现UnsynchronizedAppenderBase的类

<?xml version="1.0" encoding="UTF-8"?>
<configuration>
    <appender name="sendErrorMsg" class="com.mty.jls.config.SendErrorMsgAppender">
        <filter class="ch.qos.logback.classic.filter.ThresholdFilter">
            <level>INFO</level>
        </filter>
        <layout class="ch.qos.logback.classic.PatternLayout">
            <!--格式化输出:%d表示日期,%thread表示线程名,%-5level:级别从左显示5个字符宽度%msg:日志消息,%n是换行符 -->
            <pattern>%d{yyyy-MM-dd HH:mm:ss} [ %t:%r ] - [ %p ] %m%n</pattern>
        </layout>
    </appender>
    <root level="info">
        <appender-ref ref="stdout"/>
        <appender-ref ref="sendErrorMsg"/>
    </root>
</configuration>
复制代码

效果如下:

192.168.561 2020-10-3117:15:06「http-nio-80-exec-2;12234031-[ERROR I
mybatisPlusInterceptor 没有获取到登录人的tenantId,可能是登录成功前调用了sql操作
com.mty.jls.contract.exception.BaseException:登录状态过期
atcom.mtyjls.utils.RbacUtil.getSecurityUser(RbacUtil.java:47)
at
com.mtyjls.config.mybatisplus.MybatisPlusConfig$1.getTenantId(MybatisPlusConfig java:53
at
com.baomidou.mybatisplus.extension.plugins.inner.TenantLineInnerInterceptor.builderExpr ession(TenantLineInnerInterceptor.java:295)
at
com.baomidou.mybatisplus.extension.plugins.inner.TenantLineInnerInterceptor.processJoin( TenantLineInnerInterceptor.java:285)
com.baomidou.mybatisplus.extension.plugins.inner.TenantLineInnerInterceptor.lambda$pro cessPlainSelect$2(TenantLineInnerInterceptor.iava:239)
at java.base/iava.util.ArrayList.forEach(ArrayListjava:1511)
at
com.baomidou.mybatisplus.extension.plugins.inner.TenantLineInnerInterceptor.processPlai nSelect(TenantLineInnerInterceptor.java:238)


相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
6天前
|
人工智能 数据可视化 API
10 分钟构建 AI 客服并应用到网站、钉钉或微信中测试评
10 分钟构建 AI 客服并应用到网站、钉钉或微信中测试评
32 2
|
1天前
|
人工智能 运维 负载均衡
10 分钟构建 AI 客服并应用到网站、钉钉或微信中
《10分钟构建AI客服并应用到网站、钉钉或微信中》的解决方案通过详尽的文档和示例代码,使具有一定编程基础的用户能够快速上手,顺利完成AI客服集成。方案涵盖高可用性、负载均衡及定制化选项,满足生产环境需求。然而,若文档不清晰或存在信息缺失,则可能导致部署障碍。实际部署中可能遇到网络、权限等问题,需逐一排查。云产品的功能、性能及操作配置便捷性直接影响解决方案效果,详尽的产品手册有助于快速解决问题。总体而言,该方案在各方面表现出色,值得推荐。
|
9天前
|
缓存 监控 Java
Java中的并发编程:理解并应用线程池
在Java的并发编程中,线程池是提高应用程序性能的关键工具。本文将深入探讨如何有效利用线程池来管理资源、提升效率和简化代码结构。我们将从基础概念出发,逐步介绍线程池的配置、使用场景以及最佳实践,帮助开发者更好地掌握并发编程的核心技巧。
|
6天前
|
SQL JavaScript 前端开发
用Java来开发Hive应用
用Java来开发Hive应用
20 7
|
6天前
|
SQL JavaScript 前端开发
用Java、Python来开发Hive应用
用Java、Python来开发Hive应用
18 6
|
5天前
|
Java 数据库连接 开发者
Java中的异常处理机制:理解与应用
在Java编程中,异常处理是一个核心概念,它允许程序在遇到错误时优雅地恢复或终止。本文将深入探讨Java的异常处理机制,包括异常的分类、如何正确使用try-catch-finally块以及throw关键字。我们将通过实例来说明如何在Java应用程序中有效地捕获和处理异常,以确保程序的健壮性和稳定性。
|
5天前
|
Java 调度 开发者
Java中的多线程基础及其应用
【9月更文挑战第13天】本文将深入探讨Java中的多线程概念,从基本理论到实际应用,带你一步步了解如何有效使用多线程来提升程序的性能。我们将通过实际代码示例,展示如何在Java中创建和管理线程,以及如何利用线程池优化资源管理。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的见解和技巧,帮助你更好地理解和应用多线程编程。
|
6天前
|
存储 负载均衡 Java
Jetty技术深度解析及其在Java中的实战应用
【9月更文挑战第3天】Jetty,作为一款开源的、轻量级、高性能的Java Web服务器和Servlet容器,自1995年问世以来,凭借其卓越的性能、灵活的配置和丰富的扩展功能,在Java Web应用开发中占据了举足轻重的地位。本文将详细介绍Jetty的背景、核心功能点以及在Java中的实战应用,帮助开发者更好地理解和利用Jetty构建高效、可靠的Web服务。
21 2
|
10天前
|
Java 数据处理
技术分享:高效与灵活并存——Java版通用树形结构转换工具的实现与应用
在软件开发中,树形结构的数据表现形式无处不在,从文件系统的目录树到组织架构的部门树,再到各类产品的分类结构。处理这些具有层级关系的数据时,将其转换为树形结构以便展示和操作显得尤为重要。Java作为一门成熟的编程语言,虽然提供了强大的集合框架,但并未直接提供树形结构转换的内置工具。因此,开发一个高效且灵活的通用树形结构转换工具成为许多项目中的必备需求。
20 2
|
JavaScript Dubbo Java
这份日志格式规范,拿走不谢(Java版)
这份日志格式规范,拿走不谢(Java版)