拦截应用 error日志并发送到钉钉群|Java 开发实战
愿天堂没有BUG
共 4360字,需浏览 9分钟
· 2021-06-12
开篇
现在应用都需要对日志进行监控或者报警,现在普遍的做法是采用EKL收集日志,然后再由Grafana进行内容展示和及告警策略等,那如果项目架构比较简单(单体应用),又不想搞那么多中间件依赖怎么办,这里有一种简单的方式可以实现~
继承UnsynchronizedAppenderBase
Springboot默认集成的是logback,所以自定义Appender
非常简单,继承一下AppenderBase
类即可。
再来看看AppenderBase
之上的Appender
下都有哪些实现
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>
需要引用该name
, class
属性的值就是刚刚上面定义实现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>
复制代码
效果如下:
作者:蒋老湿
链接:https://juejin.cn/post/6970996702320787470
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
评论
【送书福利】《Java面试八股文:高频面试题与求职攻略一本通》
先来唠唠最近粉丝面试回来跟我聊天,基本上都提到一个点,在面试过程中八股文占比很高(八股文70%、项目20%、10%算法)除了一些搞算法突出的厂除外。其实现在很多厂八股都是逐渐深入的方式来问,所以大家在学习的过程中,针对一些重点的内容,最好深入去学习,不然还是比较难应对这种追问式的问题。最近刚好从一位
Java后端技术
0
【每周一课#06】MidJourney应用实战
#AI绘画# #MJ# #文生图#时间:4月24日周三 21:00课程大纲:1、关于AIGC:概念、发展历程、就业前景2、MJ基础认识:如何使用、底层逻辑、MJ与SD优缺点比较3、MJ基础功能介绍:任务指令、后缀参数、图生图、图生文、垫图、局部修改等4、MJ应用场景与变现方向
Python涨薪研究所
0
高并发实战案例 100 讲,已更新24节,即将涨价,抓紧了
大家好,我是路人。本人亲自录制的《Java 高并发 & 微服务 & 性能调优实战案例 100 讲》已发布了 24 个课时,干货满满。59元,一杯咖啡的价格,100个实战案例,即将涨价到 99,需要的小伙伴速度啦,文末附下单方式。已发布 24 个课时1、SpringBoot实
路人甲Java
0
【每周一课#06】MidJourney 应用实战
#AI绘画# #MJ# #文生图#时间:4月24日周三 21:00课程大纲:1、关于AIGC:概念、发展历程、就业前景2、MJ基础认识:如何使用、底层逻辑、MJ与SD优缺点比较3、MJ基础功能介绍:任务指令、后缀参数、图生图、图生文、垫图、局部修改等4、MJ应用场景与变现方向
Python涨薪研究所
0
Java项目实战——打造一款股票区间交易盯盘系统
点击上方“Java进阶学习交流”,进行关注后台回复“Java”即可获赠Java学习资料今日鸡汤身无彩凤双飞翼,心有灵犀一点通。一、简介大家好,我是Snowball。今天给大家分享的内容是基于Java编程,实现股票交易相关功能开发,如果读者对股票或金融衍生物交易不太了解,又比较感兴趣的话可自行查询相关
Java进阶学习交流
0
AI智能视觉检测技术在工业级测量领域的创新应用--AMB Tube-Q导管数字化测量系统
技术背景在深度学习算法出来之前,对于视觉算法来说,大致可以分为以下5个步骤:特征感知,图像预处理,特征提取,特征筛选,推理预测与识别。早期的机器学习中,占优势的统计机器学习群体中,对特征是不大关心的。深度学习是机器学习技术的一个方面,由人工神经网络提供支持。深度学习技术的工作原理是教机器通过实例学习
机器视觉
0
豆瓣9.7,这部Java神作第3版重磅上市!
Java 程序员们开年就有重磅好消息,《Effective Java 中文版(原书第 3 版)》要上市啦!该书的第1版出版于 2001 年,当时就在业界流传开来,受到广泛赞誉。时至今日,已热销近20年,本书第 3 版已是 Java 程序员的必读神书,被誉为“Java 四大名著之一”,甚至连 Java
菜鸟学Python
0
想赚点小钱,来淘金副业群!
前两天参加一个圈内非常有名的社群聚会,我自己只能算是小卡拉米,一直很想破圈,看看外边有什么机会,或者有什么好的赛道和风口,有的时候比你道行高的人一句话就可能点醒你。因为向上链接非常重要,我也是在朋友的推荐下进入了这个圈子,结果让我大开眼界,获益良多,受益匪浅。当天到场的嘉宾老师都很牛逼,我简单列一下
菜鸟学Python
0