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

相关文章

Java实现字节字符转bcd编码

《Java实现字节字符转bcd编码》BCD是一种将十进制数字编码为二进制的表示方式,常用于数字显示和存储,本文将介绍如何在Java中实现字节字符转BCD码的过程,需要的小伙伴可以了解下... 目录前言BCD码是什么Java实现字节转bcd编码方法补充总结前言BCD码(Binary-Coded Decima

SpringBoot全局域名替换的实现

《SpringBoot全局域名替换的实现》本文主要介绍了SpringBoot全局域名替换的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录 项目结构⚙️ 配置文件application.yml️ 配置类AppProperties.Ja

Java使用Javassist动态生成HelloWorld类

《Java使用Javassist动态生成HelloWorld类》Javassist是一个非常强大的字节码操作和定义库,它允许开发者在运行时创建新的类或者修改现有的类,本文将简单介绍如何使用Javass... 目录1. Javassist简介2. 环境准备3. 动态生成HelloWorld类3.1 创建CtC

JavaScript中的高级调试方法全攻略指南

《JavaScript中的高级调试方法全攻略指南》什么是高级JavaScript调试技巧,它比console.log有何优势,如何使用断点调试定位问题,通过本文,我们将深入解答这些问题,带您从理论到实... 目录观点与案例结合观点1观点2观点3观点4观点5高级调试技巧详解实战案例断点调试:定位变量错误性能分

Java实现将HTML文件与字符串转换为图片

《Java实现将HTML文件与字符串转换为图片》在Java开发中,我们经常会遇到将HTML内容转换为图片的需求,本文小编就来和大家详细讲讲如何使用FreeSpire.DocforJava库来实现这一功... 目录前言核心实现:html 转图片完整代码场景 1:转换本地 HTML 文件为图片场景 2:转换 H

Java使用jar命令配置服务器端口的完整指南

《Java使用jar命令配置服务器端口的完整指南》本文将详细介绍如何使用java-jar命令启动应用,并重点讲解如何配置服务器端口,同时提供一个实用的Web工具来简化这一过程,希望对大家有所帮助... 目录1. Java Jar文件简介1.1 什么是Jar文件1.2 创建可执行Jar文件2. 使用java

SpringBoot实现不同接口指定上传文件大小的具体步骤

《SpringBoot实现不同接口指定上传文件大小的具体步骤》:本文主要介绍在SpringBoot中通过自定义注解、AOP拦截和配置文件实现不同接口上传文件大小限制的方法,强调需设置全局阈值远大于... 目录一  springboot实现不同接口指定文件大小1.1 思路说明1.2 工程启动说明二 具体实施2

Java实现在Word文档中添加文本水印和图片水印的操作指南

《Java实现在Word文档中添加文本水印和图片水印的操作指南》在当今数字时代,文档的自动化处理与安全防护变得尤为重要,无论是为了保护版权、推广品牌,还是为了在文档中加入特定的标识,为Word文档添加... 目录引言Spire.Doc for Java:高效Word文档处理的利器代码实战:使用Java为Wo

SpringBoot日志级别与日志分组详解

《SpringBoot日志级别与日志分组详解》文章介绍了日志级别(ALL至OFF)及其作用,说明SpringBoot默认日志级别为INFO,可通过application.properties调整全局或... 目录日志级别1、级别内容2、调整日志级别调整默认日志级别调整指定类的日志级别项目开发过程中,利用日志

Java中的抽象类与abstract 关键字使用详解

《Java中的抽象类与abstract关键字使用详解》:本文主要介绍Java中的抽象类与abstract关键字使用详解,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、抽象类的概念二、使用 abstract2.1 修饰类 => 抽象类2.2 修饰方法 => 抽象方法,没有