0

0

Spring Sleuth与JAX-WS SOAP调用的Baggage传播指南

聖光之護

聖光之護

发布时间:2025-09-23 10:09:03

|

510人浏览过

|

来源于php中文网

原创

Spring Sleuth与JAX-WS SOAP调用的Baggage传播指南

本文探讨了Spring Sleuth在不同HTTP客户端中传播自定义Baggage字段(如Caller-Id)的机制。Sleuth默认支持RestTemplate、WebClient和Feign等客户端的自动传播。然而,对于JAX-WS SOAP客户端,由于缺乏开箱即用的集成,需要通过自定义SOAP处理器手动从Sleuth上下文中获取并注入Baggage字段到SOAP消息头中,以确保分布式追踪上下文的完整传递。

Spring Sleuth的自动传播机制

spring sleuth作为spring cloud生态系统中的分布式追踪解决方案,能够自动为http请求添加追踪id(x-b3-traceid、x-b3-spanid等)以及自定义的baggage字段,并在服务间进行传播。这种自动化的能力极大地简化了分布式系统的可观测性配置。

对于以下几种主流的HTTP客户端,Sleuth提供了开箱即用的集成:

  • RestTemplate: Spring框架中用于同步HTTP请求的客户端。
  • WebClient: Spring WebFlux提供的非阻塞、响应式HTTP客户端。
  • Feign: 一种声明式Web服务客户端,常用于微服务间的REST调用。

当应用程序使用这些客户端发起对外调用时,Sleuth会自动拦截请求,将当前的追踪上下文(包括Baggage字段)注入到请求头中。例如,在application.yaml中配置remote-fields:

sleuth:
  async:
    enabled: true
  baggage:
    remote-fields:
      - Caller-Id

配置后,通过Feign发起的REST调用,其请求头将包含X-B3-*追踪信息以及自定义的caller-id:

Request headers: {Accept=[application/json; distances], Authorization=[Bearer ...], X-B3-TraceId=[...], X-B3-SpanId=[...], X-B3-Sampled=[1], caller-id=[value]}

JAX-WS SOAP调用的挑战

然而,对于使用JAX-WS(Java API for XML Web Services)框架发起的SOAP调用,即使项目中引入了jaxws-spring等辅助库,Spring Sleuth的默认自动传播机制通常不覆盖这类客户端。这意味着,虽然X-B3-*追踪头可能在某些情况下通过底层HTTP传输层被传递(取决于JAX-WS客户端的具体实现和配置),但自定义的Baggage字段(如Caller-Id)则不会自动注入到SOAP消息头中。

观察到的现象是,SOAP请求头可能仅包含如Authorization等标准HTTP头,而缺少Sleuth期望传播的caller-id字段:

SOAP Headers - {Authorization=[Bearer...]}

这种差异源于Sleuth的自动插桩是针对特定的HTTP客户端库进行的,JAX-WS客户端(特别是其SOAP消息处理管道)需要不同的集成方式。

解决方案:手动注入Baggage到SOAP消息头

要解决JAX-WS SOAP调用中Baggage字段无法传播的问题,我们需要利用JAX-WS提供的扩展点——Handler(处理器),手动将Sleuth上下文中的Baggage字段提取出来,并注入到SOAP消息头中。

QIMI奇觅
QIMI奇觅

美图推出的游戏行业广告AI制作与投放一体化平台

下载

1. 创建自定义SOAP处理器

创建一个实现javax.xml.ws.handler.soap.SOAPHandler接口的类。这个处理器将在SOAP消息发送前被调用,允许我们修改消息内容。

import brave.baggage.BaggageField;
import javax.xml.namespace.QName;
import javax.xml.soap.SOAPEnvelope;
import javax.xml.soap.SOAPHeader;
import javax.xml.soap.SOAPHeaderElement;
import javax.xml.soap.SOAPMessage;
import javax.xml.ws.handler.MessageContext;
import javax.xml.ws.handler.soap.SOAPHandler;
import javax.xml.ws.handler.soap.SOAPMessageContext;
import java.util.Collections;
import java.util.Set;

/**
 * 自定义SOAP处理器,用于将Spring Sleuth的Baggage字段注入到SOAP消息头中。
 */
public class SleuthBaggageSOAPClientHandler implements SOAPHandler {

    // 定义要传播的Baggage字段名称
    private static final String CALLER_ID_FIELD_NAME = "Caller-Id";
    // 定义SOAP头元素的命名空间,根据实际需求调整
    private static final String SOAP_HEADER_NAMESPACE_URI = "http://your.service.namespace/headers";
    private static final String SOAP_HEADER_PREFIX = "hdr"; // 可选前缀

    @Override
    public Set getHeaders() {
        // 声明此Handler可能处理或添加的SOAP头QName
        return Collections.singleton(new QName(SOAP_HEADER_NAMESPACE_URI, CALLER_ID_FIELD_NAME));
    }

    @Override
    public boolean handleMessage(SOAPMessageContext context) {
        Boolean outboundProperty = (Boolean) context.get(MessageContext.MESSAGE_OUTBOUND_PROPERTY);

        if (outboundProperty) { // 仅处理出站消息(客户端发出的请求)
            try {
                SOAPMessage soapMessage = context.getMessage();
                SOAPEnvelope soapEnvelope = soapMessage.getSOAPPart().getEnvelope();
                SOAPHeader soapHeader = soapEnvelope.getHeader();

                // 如果SOAP头不存在,则创建
                if (soapHeader == null) {
                    soapHeader = soapEnvelope.addHeader();
                }

                // 从Sleuth的当前追踪上下文中获取Caller-Id Baggage字段的值
                // BaggageField.getByName()会从当前TraceContext中查找字段
                String callerIdValue = BaggageField.getByName(CALLER_ID_FIELD_NAME).value();

                if (callerIdValue != null && !callerIdValue.isEmpty()) {
                    // 创建SOAP头元素并设置值
                    QName callerIdQName = new QName(SOAP_HEADER_NAMESPACE_URI, CALLER_ID_FIELD_NAME, SOAP_HEADER_PREFIX);
                    SOAPHeaderElement headerElement = soapHeader.addHeaderElement(callerIdQName);
                    headerElement.setValue(callerIdValue);
                    System.out.println("SleuthBaggageSOAPClientHandler: Added Caller-Id to SOAP header: " + callerIdValue);
                } else {
                    System.out.println("SleuthBaggageSOAPClientHandler: Caller-Id baggage field is null or empty, not added to SOAP header.");
                }

                // 保存对SOAP消息的更改
                soapMessage.saveChanges();

            } catch (Exception e) {
                System.err.println("Error in SleuthBaggageSOAPClientHandler while adding baggage: " + e.getMessage());
                // 根据需要决定是否抛出异常或继续处理
                return false; // 阻止消息继续处理
            }
        }
        return true; // 允许消息继续处理
    }

    @Override
    public boolean handleFault(SOAPMessageContext context) {
        // 错误处理逻辑(可选)
        return true;
    }

    @Override
    public void close(MessageContext context) {
        // 清理资源(可选)
    }
}

代码说明:

  • BaggageField.getByName(CALLER_ID_FIELD_NAME).value():这是获取当前Sleuth追踪上下文中指定Baggage字段值的关键。它依赖于Spring Sleuth已正确初始化,并且当前线程的追踪上下文是激活的。
  • SOAP_HEADER_NAMESPACE_URI和SOAP_HEADER_PREFIX:这些需要根据SOAP服务提供方期望的SOAP头命名空间和前缀进行调整。

2. 配置JAX-WS客户端以使用Handler

在创建JAX-WS客户端代理时,需要将自定义的SleuthBaggageSOAPClientHandler添加到其处理器链中。

import javax.xml.ws.BindingProvider;
import javax.xml.ws.handler.Handler;
import java.util.ArrayList;
import java.util.List;

// 假设这是你的JAX-WS服务接口和实现
// import com.example.soap.YourSoapService;
// import com.example.soap.YourSoapPort;

public class SoapClientConfig {

    public YourSoapPort createSoapClient() {
        // 1. 创建JAX-WS服务和端口实例
        YourSoapService service = new YourSoapService(); // 替换为你的服务类
        YourSoapPort port = service.getYourSoapPort();   // 替换为你的端口类

        // 2. 获取客户端代理的BindingProvider
        BindingProvider bindingProvider = (BindingProvider) port;

        // 3. 获取当前的Handler链
        List handlerChain = bindingProvider.getBinding().getHandlerChain();
        if (handlerChain == null) {
            handlerChain = new ArrayList<>();
        }

        // 4. 将自定义的SleuthBaggageSOAPClientHandler添加到Handler链中
        handlerChain.add(new SleuthBaggageSOAPClientHandler());

        // 5. 设置更新后的Handler链
        bindingProvider.getBinding().setHandlerChain(handlerChain);

        // 可选:设置SOAP服务的endpoint地址
        // bindingProvider.getRequestContext().put(BindingProvider.ENDPOINT_ADDRESS_PROPERTY, "http://localhost:8080/your/soap/service");

        return port;
    }

    // 示例使用
    public static void main(String[] args) {
        SoapClientConfig config = new SoapClientConfig();
        YourSoapPort client = config.createSoapClient();
        // 现在,当通过client调用SOAP方法时,Sleuth Baggage将自动注入
        // client.someSoapOperation("data");
    }
}

代码说明:

  • BindingProvider:JAX-WS客户端代理实现了此接口,允许访问和配置底层绑定和请求上下文。
  • getBinding().getHandlerChain():获取当前客户端的处理器链。
  • setHandlerChain():设置更新后的处理器链。

注意事项

  1. Baggage字段配置: 确保application.yaml中sleuth.baggage.remote-fields下配置的字段名称与SleuthBaggageSOAPClientHandler中尝试获取的字段名称完全一致(例如Caller-Id)。
  2. SOAP Header命名空间: SleuthBaggageSOAPClientHandler中定义的SOAP_HEADER_NAMESPACE_URI和SOAP_HEADER_PREFIX必须与SOAP服务提供方期望接收的SOAP头命名空间和前缀保持一致。不匹配会导致接收方无法正确解析。
  3. 安全性: 如果Baggage字段包含敏感信息,应考虑加密或使用其他安全传输机制,因为SOAP头是明文传输的。
  4. 错误处理: 在handleMessage方法中,应添加健壮的错误处理和日志记录,以便在注入Baggage失败时能够及时发现问题。
  5. jaxws-spring集成: 虽然jaxws-spring简化了JAX-WS客户端的Spring集成,但它主要关注客户端的创建和配置,不一定提供Sleuth Baggage的自动传播。上述手动Handler的方法是通用的JAX-WS解决方案。如果jaxws-spring提供了更高级的拦截器或扩展点,可以考虑结合使用。
  6. TraceId/SpanId传播: 对于X-B3-TraceId和X-B3-SpanId等核心追踪信息,某些JAX-WS客户端可能通过底层HTTP传输层自动处理。但为了确保完整性和一致性,也可以考虑在SleuthBaggageSOAPClientHandler中一并手动注入这些B3头,使用brave.Span.current().context().traceIdString()等方法获取。

总结

Spring Sleuth在分布式追踪领域提供了强大的自动上下文传播能力,但其开箱即用的支持主要集中在主流的HTTP客户端(RestTemplate、WebClient、Feign)。对于JAX-WS SOAP客户端,由于其消息处理机制的特殊性,需要开发者通过实现自定义的SOAPHandler来手动获取Sleuth上下文中的Baggage字段,并将其注入到SOAP消息头中。这种手动集成方式确保了分布式追踪上下文在异构客户端技术中的完整传递,从而维护了整个调用链的可观测性。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
spring框架介绍
spring框架介绍

本专题整合了spring框架相关内容,想了解更多详细内容,请阅读专题下面的文章。

115

2025.08.06

Java Spring Security 与认证授权
Java Spring Security 与认证授权

本专题系统讲解 Java Spring Security 框架在认证与授权中的应用,涵盖用户身份验证、权限控制、JWT与OAuth2实现、跨站请求伪造(CSRF)防护、会话管理与安全漏洞防范。通过实际项目案例,帮助学习者掌握如何 使用 Spring Security 实现高安全性认证与授权机制,提升 Web 应用的安全性与用户数据保护。

30

2026.01.26

什么是分布式
什么是分布式

分布式是一种计算和数据处理的方式,将计算任务或数据分散到多个计算机或节点中进行处理。本专题为大家提供分布式相关的文章、下载、课程内容,供大家免费下载体验。

329

2023.08.11

分布式和微服务的区别
分布式和微服务的区别

分布式和微服务的区别在定义和概念、设计思想、粒度和复杂性、服务边界和自治性、技术栈和部署方式等。本专题为大家提供分布式和微服务相关的文章、下载、课程内容,供大家免费下载体验。

235

2023.10.07

pdf怎么转换成xml格式
pdf怎么转换成xml格式

将 pdf 转换为 xml 的方法:1. 使用在线转换器;2. 使用桌面软件(如 adobe acrobat、itext);3. 使用命令行工具(如 pdftoxml)。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

1900

2024.04.01

xml怎么变成word
xml怎么变成word

步骤:1. 导入 xml 文件;2. 选择 xml 结构;3. 映射 xml 元素到 word 元素;4. 生成 word 文档。提示:确保 xml 文件结构良好,并预览 word 文档以验证转换是否成功。想了解更多xml的相关内容,可以阅读本专题下面的文章。

2091

2024.08.01

xml是什么格式的文件
xml是什么格式的文件

xml是一种纯文本格式的文件。xml指的是可扩展标记语言,标准通用标记语言的子集,是一种用于标记电子文件使其具有结构性的标记语言。想了解更多相关的内容,可阅读本专题下面的相关文章。

1070

2024.11.28

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1130

2023.10.19

clawdbot ai使用教程 保姆级clawdbot部署安装手册
clawdbot ai使用教程 保姆级clawdbot部署安装手册

Clawdbot是一个“有灵魂”的AI助手,可以帮用户清空收件箱、发送电子邮件、管理日历、办理航班值机等等,并且可以接入用户常用的任何聊天APP,所有的操作均可通过WhatsApp、Telegram等平台完成,用户只需通过对话,就能操控设备自动执行各类任务。

13

2026.01.29

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 3万人学习

C# 教程
C# 教程

共94课时 | 7.9万人学习

Java 教程
Java 教程

共578课时 | 52.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号