spring-cloud-sleuth+zipkin源码探究

2024-05-02 10:32

本文主要是介绍spring-cloud-sleuth+zipkin源码探究,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. spring-cloud-sleuth+zipkin源码探究

1.1. 前言

  粗略看了下spring cloud sleuth core源码,发现内容真的有点多,它支持了很多类型的链路追踪,我就找其中一个比较有代表性的深入剖析下源码结构和内容

1.2. spring-cloud-sleuth-core源码解析

1.2.1. 结构

751560-20190409153305960-776037222.png

  1. 可以看到源码中支持的追踪类型有很多,支持async,hystrix,websocket,rxjava,Spring mvc,servlet,spring restTemplate,feign,zuul等等,这里我着重探讨spring web mvc的链路追踪
  2. 打开web包,找到TraceWebAutoConfiguration,这里配置了主要的初始化类
    751560-20190409154107309-390771196.png

1.2.2. 过滤器注册

  1. 当启动初始化程序时,跟踪代码如下
    @Beanpublic FilterRegistrationBean traceWebFilter(TraceFilter traceFilter) {FilterRegistrationBean filterRegistrationBean = new FilterRegistrationBean(traceFilter);filterRegistrationBean.setDispatcherTypes(ASYNC, ERROR, FORWARD, INCLUDE,REQUEST);filterRegistrationBean.setOrder(TraceFilter.ORDER);return filterRegistrationBean;}@Bean@ConditionalOnMissingBeanpublic TraceFilter traceFilter(BeanFactory beanFactory,SkipPatternProvider skipPatternProvider) {return new TraceFilter(beanFactory, skipPatternProvider.skipPattern());}
  1. 初始化traceFilter,进行过滤器注册

1.2.3. 拦截器注册

  1. 然后看TraceWebMvcConfigurer类,它会进行拦截器的注册
@Configuration
class TraceWebMvcConfigurer extends WebMvcConfigurerAdapter {@Autowired BeanFactory beanFactory;@Beanpublic TraceHandlerInterceptor traceHandlerInterceptor(BeanFactory beanFactory) {return new TraceHandlerInterceptor(beanFactory);}@Overridepublic void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(this.beanFactory.getBean(TraceHandlerInterceptor.class));}
}
  1. TraceHandlerInterceptor类中,preHandle,afterCompletion方法可以看出,这是对请求进行拦截进行span的包装
    @Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response,Object handler) throws Exception {String spanName = spanName(handler);boolean continueSpan = getRootSpanFromAttribute(request) != null;Span span = continueSpan ? getRootSpanFromAttribute(request) : getTracer().createSpan(spanName);if (log.isDebugEnabled()) {log.debug("Handling span " + span);}addClassMethodTag(handler, span);addClassNameTag(handler, span);setSpanInAttribute(request, span);if (!continueSpan) {setNewSpanCreatedAttribute(request, span);}return true;}
    @Overridepublic void afterCompletion(HttpServletRequest request, HttpServletResponse response,Object handler, Exception ex) throws Exception {if (isErrorControllerRelated(request)) {if (log.isDebugEnabled()) {log.debug("Skipping closing of a span for error controller processing");}return;}Span span = getRootSpanFromAttribute(request);if (ex != null) {getErrorParser().parseErrorTags(span, ex);}if (getNewSpanFromAttribute(request) != null) {if (log.isDebugEnabled()) {log.debug("Closing span " + span);}Span newSpan = getNewSpanFromAttribute(request);getTracer().continueSpan(newSpan);getTracer().close(newSpan);clearNewSpanCreatedAttribute(request);}}

1.2.4. zipkin端点提交

  1. 这里首先会初始化HttpZipkinSpanReporter类,,用来进行span端点提交,然后初始化ZipkinSpanListenerspan的监听器,用来监听并调用端点提交,以上配置再下图位置
    751560-20190409161404026-1390041560.png

1.2.5. 调用http接口时,进入过滤器

  1. 首先进入TraceFilter中的过滤方法doFilter,这里会做span的创建
private Span createSpan(HttpServletRequest request,boolean skip, Span spanFromRequest, String name) {if (spanFromRequest != null) {if (log.isDebugEnabled()) {log.debug("Span has already been created - continuing with the previous one");}return spanFromRequest;}//加入调用链路ZipkinHttpSpanExtractor,此链路在TraceHttpAutoConfiguration中配置实例化,调用链还没有时,返回为空,作为头节点Span parent = spanExtractor().joinTrace(new HttpServletRequestTextMap(request));if (parent != null) {if (log.isDebugEnabled()) {log.debug("Found a parent span " + parent + " in the request");}addRequestTagsForParentSpan(request, parent);spanFromRequest = parent;tracer().continueSpan(spanFromRequest);if (parent.isRemote()) {parent.logEvent(Span.SERVER_RECV);}request.setAttribute(TRACE_REQUEST_ATTR, spanFromRequest);if (log.isDebugEnabled()) {log.debug("Parent span is " + parent + "");}} else {if (skip) {spanFromRequest = tracer().createSpan(name, NeverSampler.INSTANCE);}else {String header = request.getHeader(Span.SPAN_FLAGS);if (Span.SPAN_SAMPLED.equals(header)) {spanFromRequest = tracer().createSpan(name, new AlwaysSampler());} else {//创建span节点spanFromRequest = tracer().createSpan(name);}}spanFromRequest.logEvent(Span.SERVER_RECV);request.setAttribute(TRACE_REQUEST_ATTR, spanFromRequest);if (log.isDebugEnabled()) {log.debug("No parent span present - creating a new span");}}return spanFromRequest;}

1.2.6. 进入拦截器

  1. preHandle方法中,对span进行包装,然后把span放入请求头header中
  2. 最后再DefaultTracer中进行span的关闭和spanReporter的提交

参考:https://blog.csdn.net/zhllansezhilian/article/details/83001870

这篇关于spring-cloud-sleuth+zipkin源码探究的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/954045

相关文章

SpringBoot中SM2公钥加密、私钥解密的实现示例详解

《SpringBoot中SM2公钥加密、私钥解密的实现示例详解》本文介绍了如何在SpringBoot项目中实现SM2公钥加密和私钥解密的功能,通过使用Hutool库和BouncyCastle依赖,简化... 目录一、前言1、加密信息(示例)2、加密结果(示例)二、实现代码1、yml文件配置2、创建SM2工具

Spring WebFlux 与 WebClient 使用指南及最佳实践

《SpringWebFlux与WebClient使用指南及最佳实践》WebClient是SpringWebFlux模块提供的非阻塞、响应式HTTP客户端,基于ProjectReactor实现,... 目录Spring WebFlux 与 WebClient 使用指南1. WebClient 概述2. 核心依

Spring Boot @RestControllerAdvice全局异常处理最佳实践

《SpringBoot@RestControllerAdvice全局异常处理最佳实践》本文详解SpringBoot中通过@RestControllerAdvice实现全局异常处理,强调代码复用、统... 目录前言一、为什么要使用全局异常处理?二、核心注解解析1. @RestControllerAdvice2

Spring IoC 容器的使用详解(最新整理)

《SpringIoC容器的使用详解(最新整理)》文章介绍了Spring框架中的应用分层思想与IoC容器原理,通过分层解耦业务逻辑、数据访问等模块,IoC容器利用@Component注解管理Bean... 目录1. 应用分层2. IoC 的介绍3. IoC 容器的使用3.1. bean 的存储3.2. 方法注

Spring事务传播机制最佳实践

《Spring事务传播机制最佳实践》Spring的事务传播机制为我们提供了优雅的解决方案,本文将带您深入理解这一机制,掌握不同场景下的最佳实践,感兴趣的朋友一起看看吧... 目录1. 什么是事务传播行为2. Spring支持的七种事务传播行为2.1 REQUIRED(默认)2.2 SUPPORTS2

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

java中新生代和老生代的关系说明

《java中新生代和老生代的关系说明》:本文主要介绍java中新生代和老生代的关系说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、内存区域划分新生代老年代二、对象生命周期与晋升流程三、新生代与老年代的协作机制1. 跨代引用处理2. 动态年龄判定3. 空间分

Java设计模式---迭代器模式(Iterator)解读

《Java设计模式---迭代器模式(Iterator)解读》:本文主要介绍Java设计模式---迭代器模式(Iterator),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录1、迭代器(Iterator)1.1、结构1.2、常用方法1.3、本质1、解耦集合与遍历逻辑2、统一

Java内存分配与JVM参数详解(推荐)

《Java内存分配与JVM参数详解(推荐)》本文详解JVM内存结构与参数调整,涵盖堆分代、元空间、GC选择及优化策略,帮助开发者提升性能、避免内存泄漏,本文给大家介绍Java内存分配与JVM参数详解,... 目录引言JVM内存结构JVM参数概述堆内存分配年轻代与老年代调整堆内存大小调整年轻代与老年代比例元空