通过扩展logback日志来发送异常信息邮件

运维 系统运维
这篇是通过扩展 logback 的日志插件来处理 err 级别的日志异常信息来发送邮件的,通过这篇的学习,可以掌握如何扩展 logback 的日志类,来实现自己不可告人的目的。

系统异常了,​​上篇​​是通过在全局异常中通过调用发送邮件的处理器代码进行邮件的发送,总是觉得还不那么优雅。

这篇是通过扩展 logback 的日志插件来处理 err 级别的日志异常信息来发送邮件的。

通过这篇的学习,可以掌握如何扩展 logback 的日志类,来实现自己不可告人的目的。

下面直接上代码。

首先自定义一个日志处理处理类  wlcLogLogbackAppender。

import ch.qos.logback.classic.spi.LoggingEvent;
import ch.qos.logback.classic.spi.ThrowableProxy;
import ch.qos.logback.core.AppenderBase;
import com.handler.WlcAsyncHandler;
import com.utils.SpringContextUtil;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import javax.servlet.http.HttpServletRequest;
import java.util.Objects;
/**
* 描述: 自定义日志处理类 <br>
* 时间: 2021-07-01 11:11 <br>
* 作者:IT学习道场
*/
public class wlcLogLogbackAppender extends AppenderBase<LoggingEvent> {
@Override
protected void append(LoggingEvent eventObject) {
if(eventObject instanceof LoggingEvent){
LoggingEvent loggingEvent = (LoggingEvent)eventObject;
//拿到ThrowableProxy
ThrowableProxy throwableProxy = (ThrowableProxy) loggingEvent.getThrowableProxy();
if (Objects.nonNull(throwableProxy)) {
//获取 throwable 顶级异常
Throwable throwable = throwableProxy.getThrowable();
//获取log的msg
String formattedMessage = loggingEvent.getFormattedMessage();
//获取请求request
HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder
.getRequestAttributes()).getRequest();
//获取spring的spring.mail.enable属性值,下面的逻辑你也可以自己根据想法自己玩也行
String mailEnableProperty = SpringContextUtil.getEnvironmentProperty("spring.mail.enable");
mailEnableProperty = (mailEnableProperty == null) ? "false" : mailEnableProperty;
boolean mailEnable = Boolean.valueOf(mailEnableProperty);
if (mailEnable){
//获取邮件的处理类
WlcAsyncHandler emailHandler = SpringContextUtil.getBean(WlcAsyncHandler.class);
emailHandler.handle(formattedMessage, throwable, request);
}
}
}
super.doAppend(eventObject);
}
}

自定义的日志处理类搞完了,剩下的就是在logback-spring.xml中配置下就行,配置级别是 error 级别。

<?xml version="1.0" encoding="UTF-8"?>
<configuration>
<!-- 修改一下路径-->
<property name="PATH" value="./log"></property>
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
<encoder>
<!-- <Pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} %highlight(%-5level) %blue(%-50logger{50}:%-4line) %thread %green(%-18X{LOG_ID}) %msg%n</Pattern>-->
<Pattern>%d{ss.SSS} %highlight(%-5level) %blue(%-30logger{30}:%-4line) %thread %green(%-18X{LOG_ID}) %msg%n</Pattern>
</encoder>
</appender>
<appender name="TRACE_FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
<file>${PATH}/trace/trace.log</file>
<rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
<FileNamePattern>${PATH}/trace/trace.%d{yyyy-MM-dd}.%i.log</FileNamePattern>
<timeBasedFileNamingAndTriggeringPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP">
<maxFileSize>10MB</maxFileSize>
</timeBasedFileNamingAndTriggeringPolicy>
<!--因为 FileNamePattern 配置规则是yyyy-MM-dd,就是天,所以maxHistory代表多少天-->
<cleanHistoryOnStart>true</cleanHistoryOnStart>
<!--maxHistory生效必须加 <cleanHistoryOnStart>true</cleanHistoryOnStart>-->
<maxHistory>15</maxHistory>
</rollingPolicy>
<layout>
<pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} %-5level %-50logger{50}:%-4line %green(%-18X{LOG_ID}) %msg%n</pattern>
</layout>
</appender>
<appender name="ERROR_FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
<file>${PATH}/error/error.log</file>
<rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
<FileNamePattern>${PATH}/error/error.%d{yyyy-MM-dd}.%i.log</FileNamePattern>
<timeBasedFileNamingAndTriggeringPolicy class="ch.qos.logback.core.rolling.SizeAndTimeBasedFNATP">
<maxFileSize>10MB</maxFileSize>
</timeBasedFileNamingAndTriggeringPolicy>
<cleanHistoryOnStart>true</cleanHistoryOnStart>
<maxHistory>15</maxHistory>
</rollingPolicy>
<layout>
<pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} %-5level %-50logger{50}:%-4line %green(%-18X{LOG_ID}) %msg%n</pattern>
</layout>
<filter class="ch.qos.logback.classic.filter.LevelFilter">
<level>ERROR</level>
<onMatch>ACCEPT</onMatch>
<onMismatch>DENY</onMismatch>
</filter>
</appender>
<!--配置自定义的日志处理了-->
<appender name="WlcLog" class="com.logback.wlcLogLogbackAppender">
<filter class="ch.qos.logback.classic.filter.LevelFilter">
<level>ERROR</level>
<onMatch>ACCEPT</onMatch>
<onMismatch>DENY</onMismatch>
</filter>
</appender>
<!--ERROR级别调用-->
<root level="ERROR">
<appender-ref ref="ERROR_FILE" />
<appender-ref ref="WlcLog" />
</root>
<root level="TRACE">
<appender-ref ref="TRACE_FILE" />
</root>
<root level="INFO">
<appender-ref ref="STDOUT" />
</root>
</configuration>

当然由于:

图片

判断了异常不为空时,想要进去判断,就需要在log输出日志时,要注入 Throwable。

如下:

图片

必须用  log.error("这是一个日志msg", e)。

这样 :

ThrowableProxy throwableProxy = (ThrowableProxy) loggingEvent.getThrowableProxy();

throwableProxy 就不等于 null。

log.error("这是一个日志msg")。 

throwableProxy 就等于 null。

这样自定义一个日志处理处理类  wlcLogLogbackAppender 就生效了。

你可以  wlcLogLogbackAppender  在里面你可以拿到日志信息,做一些自己想做的事情。比如我这边就是在里面发送邮件,当然你可以做任何事。

发邮件的代码就不分享了,有兴趣的可以去看​​上一篇文章​​。

责任编辑:姜华 来源: IT学习道场
相关推荐

2022-01-28 15:04:57

Python日志邮件

2009-01-03 15:05:31

ibmdwEclipseSymphony

2021-02-18 07:46:07

日志框架

2011-03-16 16:06:46

iptables日志MySQL

2010-11-26 11:25:17

2011-03-28 13:08:13

Nagios

2024-10-29 14:34:46

2020-09-08 08:44:36

日志记录基础设施安全漏洞

2013-01-04 10:48:12

IBMdW

2017-03-02 09:05:56

CentOS 7Sendmail邮件

2017-04-26 09:00:23

Python发送邮件脚本

2010-03-10 14:34:52

Python异常处理

2020-07-27 16:52:59

Mailmerge定制邮件系统应用

2022-09-02 15:08:02

Python邮件发送

2012-03-12 12:45:27

JavaLogback

2011-04-07 15:56:24

2011-04-07 16:08:16

日志信息路由器Cisco IOS

2023-12-06 11:10:08

2009-12-14 17:23:31

2023-12-05 07:21:17

IstioEnvoy
点赞
收藏

51CTO技术栈公众号