首页 > 其他分享 >为全局请求添加 TraceId ,看日志再也不懵逼

为全局请求添加 TraceId ,看日志再也不懵逼

时间:2022-12-17 10:00:51浏览次数:43  
标签:traceId MDC public TraceId 线程 executor 日志 全局

不知道大家有没有一堆日志就是定位不到那块是异常部分,接口错误无法复现,也找不到报错信息等比较棘手的问题。

其实解决上面的问题很简单,只要我们为每一个请求都分配一个唯一的 RequestId 或者叫 TraceId ,一旦出了问题,只需要拿着 Id 去日志里一搜,妖魔鬼怪立马原形毕露。

对于分布式链路追踪,有很多开源中间件,本文主要通过 logback 的 MDC 实现。

请求拦截器

@Component
public class TraceIdInterceptor extends HandlerInterceptorAdapter {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
String traceId = String.format("%s - %s",request.getRequestURI(), UUID.fastUUID().toString(true));
MDC.put("traceId", traceId);
return true;
}

注册拦截器

@Component
@RequiredArgsConstructor
public class GlobalWebMvcConfigurer implements WebMvcConfigurer {

private final TraceIdInterceptor traceIdInterceptor;
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(traceIdInterceptor);
}
}

统一返回值

@Data
@NoArgsConstructor
@AllArgsConstructor
public class CommonResponse<T> implements Serializable {

/** 错误码 */
private Integer code;

/** 错误消息 */
private String message;

/** 泛型响应数据 */
private T Data;

private String traceId;

public CommonResponse(Integer code, String message) {

this.code = code;
this.message = message;
this.traceId = MDC.get("traceId");
}

}

日志配置

logback.xml

<?xml version="1.0" encoding="UTF-8"?>
<configuration>
<property name="LOG_PATTERN"
value="%d{yyyy-MM-dd} %d{HH:mm:ss.SSS} [%highlight(%-5level)] [%boldYellow(%X{traceId})] [%boldYellow(%thread)] %boldGreen(%logger{36} %F.%L) %msg%n">
</property>

<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
<encoder>
<pattern>${LOG_PATTERN}</pattern>
</encoder>
<!-- 控制台打印INFO及以上级别的日志 -->
<filter class="ch.qos.logback.classic.filter.ThresholdFilter">
<level>INFO</level>
</filter>
</appender>

<root>
<appender-ref ref="STDOUT"/>
</root>
</configuration>

测试

接口返回值

{
"code": 0,
"message": "",
"traceId": "/commerce-user/user/findById - 73e470120ef24d57a111de6b671d030b",
"data": {
"id": 1,
"username": "test1",
"password": "test",
"extraInfo": "{}",
"createTime": "2022-06-20T09:23:18.000+00:00",
"updateTime": "2022-06-20T09:23:18.000+00:00",
"balance": 0
}
}

日志

为全局请求添加 TraceId ,看日志再也不懵逼_ide

如此,如果线上有接口出现问题,拿着 traceId 到日志文件搜索,就能检索到该请求的日志调用链路,处理问题就变得简单了。

异步调用配置

因为是 ThreadLocal ,异步任务必然是获取不到 traceId 的,需要再线程池配置中手动添加。


为全局请求添加 TraceId ,看日志再也不懵逼_xml_02

线程池配置

@Configuration
public class AsyncConfig implements AsyncConfigurer {
@Bean
@Override
public Executor getAsyncExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();

executor.setCorePoolSize(10);
executor.setMaxPoolSize(20);
executor.setQueueCapacity(20);
executor.setKeepAliveSeconds(60);
executor.setThreadNamePrefix("user-async-");

// 等待所有任务结果候再关闭线程池
executor.setWaitForTasksToCompleteOnShutdown(true);
executor.setAwaitTerminationSeconds(60);
// 定义拒绝策略
executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
//设置线程装饰器
executor.setTaskDecorator(runnable -> ThreadMdcUtils.wrapAsync(runnable, MDC.getCopyOfContextMap()));
// 初始化线程池, 初始化 core 线程
executor.initialize();

return executor;
}
}
public class ThreadMdcUtils {
public static Runnable wrapAsync(Runnable task, Map<String,String> context){
return () -> {
if(context==null){
MDC.clear();
}else {
MDC.setContextMap(context);
}
if(MDC.get("traceId")==null){
MDC.put("traceId", UUID.fastUUID().toString(true));
}
try {
task.run();
}finally {
MDC.clear();
}
};
}
}

再次测试

为全局请求添加 TraceId ,看日志再也不懵逼_线程池_03

标签:traceId,MDC,public,TraceId,线程,executor,日志,全局
From: https://blog.51cto.com/u_15733182/5949481

相关文章

  • EasyPytest测试平台开发日志之系统设计
    EasyPytest测试平台,帮你高效管理Pytest测试用例。EP平台的定位是Pytest测试用例管理平台,80%的自动化用例开发在PyCharm等专业的IDE中进行,20%的用例调度和结果查看就交给E......
  • ELK日志分析系统
    一.ELK概述ELK平台是一套完整的日志集中处理解决方案,将ElasticSearch、Logstash和Kiabana三个开源工具配合使用,完成更强大的用户对日志的查询、排序、统计需求。1......
  • ②. 日志切割logrotate
    1、关于日志切割日志文件包含了关于系统中发生的事件的有用信息,在排障过程中或者系统性能分析时经常被用到。对于忙碌的服务器,日志文件大小会增长极快,服务器会很快消耗磁......
  • ELK日志分析系统
    一,ELK日志分析系统概述1.ELK简介ElasticSearch:是基于Lucene(-一个全文检索引擎的架构)开发的分布式存储检索引擎,用来存储各类日志Elasticsearch是用Java开发的,可通......
  • ELK+FileBeat日志分析系统
    一、ELK简介ELK平台是一套完整的日志集中处理解决方案,将ElasticSearch、Logstash和Kiabana三个开源工具配合使用,完成更强大的用户对日志的查询、排序、统计需求。......
  • ELK+Filebeat日志分析系统
    一、ELK日志分析系统简介ELK平台是一套完整的日志集中处理解决方案,将ElasticSearch、Logstash和Kiabana三个开源工具配合使用,完成更强大的用户对日志的查询、排序、......
  • 基于基于全局差错能量函数的双目图像立体匹配算法matlab仿真,并提取图像的深度信息
    1.算法概述全局的能量函数公式如下:E(f)=Edata(f)+Esmooth(f)其中,Edata表示能量函数的数据项,意为该像素只考虑自身的视差值的倾向,不考虑邻域内其他像素的影响;N表......
  • 基于基于全局差错能量函数的双目图像立体匹配算法matlab仿真,并提取图像的深度信息
    1.算法概述        全局的能量函数公式如下:E(f)=Edata(f)+Esmooth(f)其中,Edata表示能量函数的数据项,意为该像素只考虑自身的视差值的倾向,不考虑邻域内其......
  • python 日志 logging模块详解
    1、基本使用配置logging基本的设置,然后在控制台输出日志,importlogginglogging.basicConfig(level=logging.INFO,format='%(asctime)s-%(name)s-%(levelname)s-......
  • Linux:通过命令查找日志文件中的某字段
    工作中有用到,做个记录。1.查询某字段,显示行号:cat-nfile_name|grep'查找字段'[root@ZWZF-CWY-LZY-12CWY]#cat-nnohup.out|grep'JVM'104452022-12-151......