gpt4 book ai didi

java - Logback SMTPAppender 在特定时间只发送一封电子邮件,但有所有异常(exception)情况

转载 作者:塔克拉玛干 更新时间:2023-11-03 03:16:05 25 4
gpt4 key购买 nike

有没有办法在 LogBack 中配置 SMTPAppender 以满足以下条件?

  1. 将所有异常分组到一条消息中
  2. 仅在发生异常时发送每日日志报告
  3. 仅在一天中的特定时间发送一次报告,将其分组在一封电子邮件中。

我当前的实现远未实现上述功能,但目前它会在发生异常时发送 3 封电子邮件 - 异常消息、堆栈跟踪和缓冲区刷新。

<!-- Filter duplicate Log Messages - Very important for Email Reports -->
<turboFilter class="ch.qos.logback.classic.turbo.DuplicateMessageFilter">
<AllowedRepetitions>1</AllowedRepetitions>
<CacheSize>1000</CacheSize>
</turboFilter>

<!--
############################################################
BASIC APPENDER
############################################################
-->

<appender name="Console" class="ch.qos.logback.core.ConsoleAppender">
<encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder">
<pattern>%d{HH:mm:ss.SSS} %-55(%X{user} %level [%thread] %logger{20}) - %msg%n</pattern>
</encoder>
</appender>


<!--
############################################################
EMAIL APPENDER
############################################################
-->
<statusListener class="ch.qos.logback.core.status.OnConsoleStatusListener" />

<appender name="Email" class="ch.qos.logback.classic.net.SMTPAppender">
<smtpHost>SERVER</smtpHost>
<smtpPort>PORT</smtpPort>
<asynchronousSending>false</asynchronousSending>
<from>SENDER</from>
<to>RECIPIENT</to>
<subject>SUBJECT</subject>

<layout class="ch.qos.logback.classic.PatternLayout">
<pattern>%d{HH:mm:ss.SSS} %-55(%X{user} %level [%thread] %logger{20}) - %msg%n</pattern>
</layout>

</appender>

<!--
############################################################
OTHER
############################################################
-->
<root level="INFO">
<appender-ref ref="Console"/>
<appender-ref ref="RollingFile"/>
<appender-ref ref="Email"/>
</root>

最佳答案

一个简单的解决方案是将这些错误记录到一个文件中,并在您的服务器/机器上安装一个脚本,每天读取一次该文件并发送一封电子邮件。

如果你想使用附加程序,在我看来你需要自己动手,因为我认为标准的 SMTPAppender 不能让你每天发送一次电子邮件:

  • 扩展 SMTPAppender
  • 覆盖 SMTPAppenderBase 中的 sendBuffer 方法,以便它简单地将日志消息添加到集合中
  • ScheduledExecutorService 添加到每天运行一次 sendEmail 方法的 appender
  • sendEmail 方法将在 this 上同步以确保线程安全,检查集合是否为空,发送包含所有错误的电子邮件并清除集合

基本实现可能类似于下面的类(我没有测试过它——我使用的是 Java 8 语法,但如果需要,您可以将其替换为匿名类)。请注意,我只保留导致异常的事件,您可能还想在 sendBuffer 方法中保留 CyclicBuffer 的内容和/或在 sendEmail 方法中的错误之间添加一些错误分隔符。这可能会变得非常复杂,需要根据您的要求进行微调。

public class ScheduledSMTPAppender extends SMTPAppender {
private final ThreadFactory tf = r -> {
Thread t = new Thread(r, "ScheduledSMTPAppender Thread");
t.setDaemon(true); //make daemon or it will prevent your program to exit
return t;
};
private final ScheduledExecutorService scheduler = Executors.newScheduledThreadPool(1, tf);
private final List<ILoggingEvent> events = new ArrayList<> ();

private int maxMessages = 10;

public ScheduledSMTPAppender() { super(); }
public ScheduledSMTPAppender(EventEvaluator<ILoggingEvent> eventEvaluator) { super(eventEvaluator); }

@Override public void start() {
super.start();
scheduler.scheduleAtFixedRate(this::sendEmail, 1, 1, TimeUnit.DAYS);
}

@Override protected void sendBuffer(CyclicBuffer<ILoggingEvent> cb, ILoggingEvent lastEventObject) {
events.add(lastEventObject);
if (events.size() > maxMessages) sendEmail();
}

//needs to be synchronized for thread safety
private synchronized void sendEmail() {
try {
if (events.isEmpty()) return;
ILoggingEvent lastEvent = events.get(events.size() - 1);
events.remove(events.size() - 1);
CyclicBuffer<ILoggingEvent> cb;
if (events.isEmpty()) {
cb = new CyclicBuffer<>(1);
} else {
cb = new CyclicBuffer<>(events.size());
for (ILoggingEvent e : events) cb.add(e);
}
super.sendBuffer(cb, lastEvent);
events.clear();
} catch (Exception e) {
//Important to have a catch all here or the scheduled task will die
addError("Error occurred while sending e-mail notification.", e);
}
}

//this allows to make "maxMessages" a parameter of your appender
public int getMaxMessages() { return maxMessages; }
public void setMaxMessages(int maxMessages) { this.maxMessages = maxMessages; }
}

你的 logback 配置文件看起来像:

<appender name="Email" class="your.package.ScheduledSMTPAppender">
<smtpHost>SERVER</smtpHost>
<smtpPort>PORT</smtpPort>
<asynchronousSending>false</asynchronousSending>
<from>SENDER</from>
<to>RECIPIENT</to>
<subject>SUBJECT</subject>
<maxMessages>10</maxMessages>

<layout class="ch.qos.logback.classic.PatternLayout">
<pattern>%d{HH:mm:ss.SSS} %-55(%X{user} %level [%thread] %logger{20}) - %msg%n</pattern>
</layout>
</appender>

更进一步,您可以添加参数,例如发送邮件的时间、每天的电子邮件数量等。

关于java - Logback SMTPAppender 在特定时间只发送一封电子邮件,但有所有异常(exception)情况,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31024910/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com