Spring Boot将错误日志发送到企微微信或钉钉群

1.创建 WeComAppender ,重写 AppenderBase 的 append 方法

复制代码
import ch.qos.logback.classic.spi.ILoggingEvent;
import ch.qos.logback.core.AppenderBase;
import cn.hutool.core.thread.ThreadUtil;
import com.alibaba.fastjson2.JSON;
import com.alibaba.fastjson2.JSONObject;

import java.io.OutputStream;
import java.net.HttpURLConnection;
import java.net.URL;
import java.nio.charset.StandardCharsets;
import java.time.ZoneOffset;

/**
 * @author: joe
 */
public class WeComAppender extends AppenderBase<ILoggingEvent> {

    @Override
    protected void append(ILoggingEvent event) {
        try {
            // 按项目需求自定义拼装消息串,此处仅供参考
            StringBuilder sb = new StringBuilder();
            sb.append(event.getInstant().atOffset(ZoneOffset.ofHours(8)));
            sb.append(event);
            if (event.getThrowableProxy() != null) {
                sb.append(" ");
                sb.append(event.getThrowableProxy().getMessage());
            }
            // 异步请求,避免影响主线程
            ThreadUtil.execAsync(() -> sendHttpRequest(sb.toString()));
        } catch (Exception e) {
        }
    }

    /**
     * 发送请求
     * @param message
     */
    public void sendHttpRequest(String message) {
        try {
        	// 机器人 webhook 地址
            String webhookUrl = "https://qyapi.weixin.qq.com/cgi-bin/webhook/send?key=xxxxxxxxxxxxxxxxxxxxxxxxxx";
            URL url = new URL(webhookUrl);
            HttpURLConnection connection = (HttpURLConnection) url.openConnection();
            connection.setRequestMethod("POST");
            connection.setRequestProperty("Content-Type", "application/json; utf-8");
            connection.setRequestProperty("Accept", "application/json");
            connection.setDoOutput(true);

            // 企微机器人固定的请求结构,详情请参考企微文档(钉钉的请参考钉钉文档)
            String jsonInputString = "{\"msgtype\": \"text\", \"text\": {\"content\": \"\"}}";
            JSONObject jsonObject = JSON.parseObject(jsonInputString);
            jsonObject.getJSONObject("text").put("content", message);

            try (OutputStream os = connection.getOutputStream()) {
                byte[] input = jsonObject.toString().getBytes(StandardCharsets.UTF_8);
                os.write(input, 0, input.length);
            }
            int responseCode = connection.getResponseCode();
        } catch (Exception e) {
        }
    }
}

2.配置 logback.xml

复制代码
<?xml version="1.0" encoding="UTF-8"?>
<configuration>
    <!-- WECOM:企微告警 -->
    <!-- class 设置为 WeComAppender -->
    <appender name="WECOM" class="com.kjwl.framework.logback.WeComLogbackAppender">
        <filter class="ch.qos.logback.classic.filter.ThresholdFilter">
        	<!-- 日志等级为 ERROR -->
            <level>ERROR</level>
        </filter>
        <encoder>
            <pattern>%d{yyyy-MM-dd HH:mm:ss} %level [%thread] %logger{36} %msg%n</pattern>
        </encoder>
    </appender>

    <root level="INFO">
        <appender-ref ref="WECOM" />
    </root>

</configuration>
相关推荐
葫芦和十三8 小时前
图解 MongoDB 19|Oplog:复制的真正载体,不是文档是操作
后端·mongodb·agent
葫芦和十三8 小时前
图解 MongoDB 20|复制延迟与 catch up:Secondary 为什么跟不上
后端·mongodb·agent
IT_陈寒13 小时前
SpringBoot自动配置的坑,我的API突然就404了
前端·人工智能·后端
ServBay14 小时前
为什么说 MCP 是 2026 年开发者必须掌握的黄金协议?
后端·mcp
程序员夏洛14 小时前
Spring Boot 多模块项目中 IDEA 提示 Cannot resolve symbol 的一次排查记录
后端
子兮曰14 小时前
OpenMontage 深度解剖:你的 AI 编程助手,其实是个视频工作室
前端·后端·ai编程
子兮曰14 小时前
前端工具链的「Rust 化」:一场没有赢家的军备竞赛?
前端·后端·rust
爱勇宝15 小时前
从 Ctrl+CV 到 Enter:程序员正在失去什么
前端·后端·程序员
码事漫谈15 小时前
EdgeOne Makers + WorkBuddy:零基础也能快速搭建可上线的 AI 智能体(附图文教程)
后端