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

相关文章

Spring Boot集成Druid实现数据源管理与监控的详细步骤

《SpringBoot集成Druid实现数据源管理与监控的详细步骤》本文介绍如何在SpringBoot项目中集成Druid数据库连接池,包括环境搭建、Maven依赖配置、SpringBoot配置文件... 目录1. 引言1.1 环境准备1.2 Druid介绍2. 配置Druid连接池3. 查看Druid监控

Java中读取YAML文件配置信息常见问题及解决方法

《Java中读取YAML文件配置信息常见问题及解决方法》:本文主要介绍Java中读取YAML文件配置信息常见问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录1 使用Spring Boot的@ConfigurationProperties2. 使用@Valu

创建Java keystore文件的完整指南及详细步骤

《创建Javakeystore文件的完整指南及详细步骤》本文详解Java中keystore的创建与配置,涵盖私钥管理、自签名与CA证书生成、SSL/TLS应用,强调安全存储及验证机制,确保通信加密和... 目录1. 秘密键(私钥)的理解与管理私钥的定义与重要性私钥的管理策略私钥的生成与存储2. 证书的创建与

浅析Spring如何控制Bean的加载顺序

《浅析Spring如何控制Bean的加载顺序》在大多数情况下,我们不需要手动控制Bean的加载顺序,因为Spring的IoC容器足够智能,但在某些特殊场景下,这种隐式的依赖关系可能不存在,下面我们就来... 目录核心原则:依赖驱动加载手动控制 Bean 加载顺序的方法方法 1:使用@DependsOn(最直

SpringBoot中如何使用Assert进行断言校验

《SpringBoot中如何使用Assert进行断言校验》Java提供了内置的assert机制,而Spring框架也提供了更强大的Assert工具类来帮助开发者进行参数校验和状态检查,下... 目录前言一、Java 原生assert简介1.1 使用方式1.2 示例代码1.3 优缺点分析二、Spring Fr

java使用protobuf-maven-plugin的插件编译proto文件详解

《java使用protobuf-maven-plugin的插件编译proto文件详解》:本文主要介绍java使用protobuf-maven-plugin的插件编译proto文件,具有很好的参考价... 目录protobuf文件作为数据传输和存储的协议主要介绍在Java使用maven编译proto文件的插件

Java中的数组与集合基本用法详解

《Java中的数组与集合基本用法详解》本文介绍了Java数组和集合框架的基础知识,数组部分涵盖了一维、二维及多维数组的声明、初始化、访问与遍历方法,以及Arrays类的常用操作,对Java数组与集合相... 目录一、Java数组基础1.1 数组结构概述1.2 一维数组1.2.1 声明与初始化1.2.2 访问

Javaee多线程之进程和线程之间的区别和联系(最新整理)

《Javaee多线程之进程和线程之间的区别和联系(最新整理)》进程是资源分配单位,线程是调度执行单位,共享资源更高效,创建线程五种方式:继承Thread、Runnable接口、匿名类、lambda,r... 目录进程和线程进程线程进程和线程的区别创建线程的五种写法继承Thread,重写run实现Runnab

Java 方法重载Overload常见误区及注意事项

《Java方法重载Overload常见误区及注意事项》Java方法重载允许同一类中同名方法通过参数类型、数量、顺序差异实现功能扩展,提升代码灵活性,核心条件为参数列表不同,不涉及返回类型、访问修饰符... 目录Java 方法重载(Overload)详解一、方法重载的核心条件二、构成方法重载的具体情况三、不构

Java通过驱动包(jar包)连接MySQL数据库的步骤总结及验证方式

《Java通过驱动包(jar包)连接MySQL数据库的步骤总结及验证方式》本文详细介绍如何使用Java通过JDBC连接MySQL数据库,包括下载驱动、配置Eclipse环境、检测数据库连接等关键步骤,... 目录一、下载驱动包二、放jar包三、检测数据库连接JavaJava 如何使用 JDBC 连接 mys