微服务分布式架构实现日志链路跟踪的方法

作者:码农架构 时间:2023-11-29 15:43:16 

Logback 背景

Logback是由log4j创始人设计的另一个开源日志组件,官方网站:http://logback.qos.ch。它当前分为下面下个模块:

  • logback-core:其它两个模块的基础模块

  • logback-classic:它是log4j的一个改良版本,同时它完整实现了slf4j API使你可以很方便地更换成其它日志系统如log4j或JDK14 Logging

  • logback-access:访问模块与Servlet容器集成提供通过Http来访问日志的功能

普通debug日志

微服务分布式架构实现日志链路跟踪的方法

SQL执行日志

微服务分布式架构实现日志链路跟踪的方法

Logback 配置案例

微服务分布式架构实现日志链路跟踪的方法

日志级别排序为:TRACE < DEBUG < INFO < WARN < ERROR

  • %d:表示日期

  • %n:换行

  • %thread:表示线程名

  • %level:日志级别

  • %msg:日志消息

  • %file:表示文件名

  • %class:表示文件名

  • %logger:Java类名(含包名,这里设定了36位,若超过36位,包名会精简为类似a.b.c.JavaBean)

  • %line:Java类的行号

微服务分布式架构实现日志链路跟踪的方法 

注意:

%-4relative %d{yyyy-MM-dd HH:mm:ss.SSS} [%thread][%X{TRACE_ID}] %-5level %logger{100}.%M\(%line\) - %msg%n

在logback中,%relative表示自应用程序启动以来打印相对时间戳(以毫秒为单位). %-4只是元素的对齐方式.

案例

3452487 2021-08-03 15:19:36.940 [thread-monitor-daemon][] WARN  com.xxxx.common.util.MonitorLogger.warn(27) - 发现超时线程notify-replay-consumer...

微服务分布式架构实现日志链路跟踪的方法

由于案例中是守护线程thread-monitor-daemon,所以不记录链路ID。

对在系统设计的时候对于线程的命名规范也是有约束的

微服务分布式架构实现日志链路跟踪的方法


这里就不做详细展开后续有机会会分享。
回归正题比如下面的例子中记录了请求的链路ID

19006989 2021-08-04 22:35:25.776 [http-nio-0.0.0.0-8010-exec-10][1fc8pebmgwukw863w2p342rp2936a3r157w0:0:] INFO  com.xxx.framework.eureka.core.listener.EurekaStateChangeListener.listen(58) - 服务实例[XX-PAAS]注册成功,当前服务器已注册服务实例数量[3]

微服务分布式架构实现日志链路跟踪的方法

对于上图中显示的系统启动时间、当前时间、当前线程、对应路径按照logback官方配置就可以逐步完善对于的日志信息,但是对于链路ID的生成写入就需要特殊处理。

链路ID设计

对于链路追踪设计我个人比较喜欢两种方案

第一种

微服务分布式架构实现日志链路跟踪的方法

在每一次请求中链路编号(traceId)、单元编号(spanId)都是通过HttpHeader的方式进行传递,日志的起始位置会主动生成traceId、spanId,而起始位置的Parent SpanId则是不存在的,值为null。

这样每次通过restTemplate、Openfeign的形式访问其他服务的接口时,就会携带起始位置生成的traceId、spanId到下一个服务单元。

 第二种

微服务分布式架构实现日志链路跟踪的方法 

在每一次请求中链路编号(traceId),没经过一次微服务对于深度(Deep)加1


public static class ThreadTraceListener implements ThreadListener {
   @Override
   public void onThreadBegin(HttpServletRequest request) {
     String traceToken = ThreadLocalUtil.getTranVar(TRACE_ID);
     String fromServer = ThreadLocalUtil.getTranVar(FROM_SERVER);
     int deep;
     String traceId;
     if (StringUtils.isBlank(traceToken)) {
       traceId = IDGenerator.generateID();
       deep = 0;
       traceToken = StringHelper.join(traceId, ":0");
     } else {
       int index = traceToken.lastIndexOf(':');
       traceId = traceToken.substring(0, index);
       deep = Integer.valueOf(traceToken.substring(index + 1));
     }
     ThreadLocalUtil.setLocalVar(TRACE_ID, traceId);
     ThreadLocalUtil.setLocalVar(TRACE_DEEP, deep);
     ThreadLocalUtil.setTranVar(TRACE_ID, StringHelper.join(traceId, ":", deep + 1));
     ThreadLocalUtil.setLocalVar(FROM_SERVER, fromServer);
     ThreadLocalUtil.setTranVar(FROM_SERVER, getCurrentServer());
     MDC.put(TRACE_ID, StringHelper.join(traceToken, ":", fromServer));
   }

@Override
   public void onThreadEnd(HttpServletRequest request) {
     MDC.remove(TRACE_ID);
   }
 }

针对请求拦截


protected void doFilterInternal(HttpServletRequest request,
     HttpServletResponse response,
     FilterChain chain) throws ServletException, IOException {
   long startTime = System.currentTimeMillis();
   // 从Header中装载传递过来的变量
   Map<String, Object> tranVar = new HashMap<String, Object>();
   Enumeration<String> headers = request.getHeaderNames();
   while (headers.hasMoreElements()) {
     String key = headers.nextElement();
     if (!StringUtils.isEmpty(key)
         && key.startsWith(ThreadLocalUtil.TRAN_PREFIX)) {
       tranVar.put(key.substring(ThreadLocalUtil.TRAN_PREFIX.length()),
           request.getHeader(key));
     }
   }
   ThreadLocalHolder.begin(tranVar, request);
   try {
     if (isGateway) {
       response.addHeader("X-TRACE-ID", TraceUtil.getTraceId());
     }
     // 检查RPC调用深度
     checkRpcDeep(request, response);
     // 业务处理
     chain.doFilter(request, response);
     // 记录RPC调用次数
     logRpcCount(request, response);
   } catch (Throwable ex) {
     // 错误处理
     Response<?> result = ExceptionUtil.toResponse(ex);
     Determine determine = ExceptionUtil.determineType(ex);
     ExceptionUtil.doLog(result, determine.getStatus(), ex);
     response.setStatus(determine.getStatus().value());
     response.setCharacterEncoding("UTF-8");
     response.setContentType(MediaType.APPLICATION_JSON_UTF8_VALUE);
     response.getWriter().write(JsonUtil.toJsonString(result));
   } finally {
     try {
       doMonitor(request, response, startTime);
       if (TraceUtil.isTraceLoggerOn()) {
         log.warn(StringHelper.join(
             "TRACE-HTTP-", request.getMethod(),
             " URI:", request.getRequestURI(),
             ", dt:", System.currentTimeMillis() - startTime,
             ", rpc:", TraceUtil.getRpcCount(),
             ", status:", response.getStatus()));
       } else if (log.isTraceEnabled()) {
         log.trace(StringHelper.join(request.getMethod(),
             " URI:", request.getRequestURI(),
             ", dt:", System.currentTimeMillis() - startTime,
             ", rpc:", TraceUtil.getRpcCount(),
             ", status:", response.getStatus()));
       }
     } finally {
       ThreadLocalHolder.end(request);
     }
   }
}

来源:https://mp.weixin.qq.com/s/s_dVunSq9bkJGeRVN9qRaA

标签:微服务,分布式架构,日志链路,跟踪
0
投稿

猜你喜欢

  • java使用IO流对数组排序实例讲解

    2023-09-04 02:24:19
  • SpringBoot中@Import注解如何正确使用

    2023-07-28 12:36:16
  • Spring Boot缓存实战 EhCache示例

    2023-08-30 12:23:35
  • 使用mybatis-plus的insert方法遇到的问题及解决方法(添加时id值不存在异常)

    2023-11-12 19:25:55
  • java编程下字符串的16位,32位md5加密实现方法

    2023-07-29 21:34:28
  • SpringBoot自动配置原理详解

    2023-08-19 09:25:55
  • Java SpringMVC框架开发之数据导出Excel文件格式实例详解

    2023-11-10 21:11:18
  • SpringBoot微信消息接口配置详解

    2023-08-23 09:51:21
  • Android 实现永久保存数据的方法详解

    2023-06-23 22:13:23
  • SpringBoot如何在运行时动态添加数据源

    2023-11-13 21:36:40
  • 用代码更新你的jar包

    2023-09-19 23:01:04
  • Java多线程回调方法实例解析

    2023-11-04 01:40:01
  • mybatis多个区间处理方式(双foreach循环)

    2023-11-26 09:01:42
  • SpringBoot和Swagger结合提高API开发效率

    2023-11-25 01:23:16
  • Jersey Restful接口如何获取参数的问题

    2023-10-29 14:44:16
  • 简单了解Java编程中线程的创建与守护线程

    2023-11-25 06:59:52
  • Java代码实现简单酒店管理系统

    2023-11-21 03:38:57
  • Java新手学习之IO流的简单使用

    2023-08-09 15:57:33
  • Spring spel表达式使用方法示例

    2023-08-25 00:43:32
  • 解决java.lang.ClassCastException的java类型转换异常的问题

    2023-11-29 04:34:10
  • asp之家 软件编程 m.aspxhome.com