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

简介: 现在应用都需要对日志进行监控或者报警,现在普遍的做法是采用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)


相关实践学习
通过日志服务实现云资源OSS的安全审计
本实验介绍如何通过日志服务实现云资源OSS的安全审计。
目录
相关文章
|
6月前
|
安全 前端开发 Java
《深入理解Spring》:现代Java开发的核心框架
Spring自2003年诞生以来,已成为Java企业级开发的基石,凭借IoC、AOP、声明式编程等核心特性,极大简化了开发复杂度。本系列将深入解析Spring框架核心原理及Spring Boot、Cloud、Security等生态组件,助力开发者构建高效、可扩展的应用体系。(238字)
|
7月前
|
消息中间件 人工智能 Java
抖音微信爆款小游戏大全:免费休闲/竞技/益智/PHP+Java全筏开源开发
本文基于2025年最新行业数据,深入解析抖音/微信爆款小游戏的开发逻辑,重点讲解PHP+Java双引擎架构实战,涵盖技术选型、架构设计、性能优化与开源生态,提供完整开源工具链,助力开发者从理论到落地打造高留存、高并发的小游戏产品。
|
8月前
|
JavaScript 安全 前端开发
Java开发:最新技术驱动的病人挂号系统实操指南与全流程操作技巧汇总
本文介绍基于Spring Boot 3.x、Vue 3等最新技术构建现代化病人挂号系统,涵盖技术选型、核心功能实现与部署方案,助力开发者快速搭建高效、安全的医疗挂号平台。
390 3
|
8月前
|
安全 Java 数据库
Java 项目实战病人挂号系统网站设计开发步骤及核心功能实现指南
本文介绍了基于Java的病人挂号系统网站的技术方案与应用实例,涵盖SSM与Spring Boot框架选型、数据库设计、功能模块划分及安全机制实现。系统支持患者在线注册、登录、挂号与预约,管理员可进行医院信息与排班管理。通过实际案例展示系统开发流程与核心代码实现,为Java Web医疗项目开发提供参考。
382 2
|
7月前
|
存储 Java 关系型数据库
Java 项目实战基于面向对象思想的汽车租赁系统开发实例 汽车租赁系统 Java 面向对象项目实战
本文介绍基于Java面向对象编程的汽车租赁系统技术方案与应用实例,涵盖系统功能需求分析、类设计、数据库设计及具体代码实现,帮助开发者掌握Java在实际项目中的应用。
271 0
|
8月前
|
移动开发 Cloud Native 安全
Java:跨平台之魂,企业级开发的磐石
Java:跨平台之魂,企业级开发的磐石
|
11月前
|
机器学习/深度学习 消息中间件 存储
【高薪程序员必看】万字长文拆解Java并发编程!(9-2):并发工具-线程池
🌟 ​大家好,我是摘星!​ 🌟今天为大家带来的是并发编程中的强力并发工具-线程池,废话不多说让我们直接开始。
396 0
|
安全 Java 调度
解锁Java并发编程高阶技能:深入剖析无锁CAS机制、揭秘魔法类Unsafe、精通原子包Atomic,打造高效并发应用
【8月更文挑战第4天】在Java并发编程中,无锁编程以高性能和低延迟应对高并发挑战。核心在于无锁CAS(Compare-And-Swap)机制,它基于硬件支持,确保原子性更新;Unsafe类提供底层内存操作,实现CAS;原子包java.util.concurrent.atomic封装了CAS操作,简化并发编程。通过`AtomicInteger`示例,展现了线程安全的自增操作,突显了这些技术在构建高效并发程序中的关键作用。
258 1
|
11月前
|
缓存 安全 Java
【高薪程序员必看】万字长文拆解Java并发编程!(3-1):并发共享问题的解决与分析
活锁:多个线程相互影响对方退出同步代码块的条件而导致线程一直运行的情况。例如,线程1的退出条件是count=5,而线程2和线程3在其代码块中不断地是count进行自增自减的操作,导致线程1永远运行。内存一致性问题:由于JIT即时编译器对缓存的优化和指令重排等造成的内存可见性和有序性问题,可以通过synchronized,volatile,并发集合类等机制来解决。这里的线程安全是指,多个线程调用它们同一个实例的方法时,是线程安全的,但仅仅能保证当前调用的方法是线程安全的,不同方法之间是线程不安全的。
188 0