惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

Exploit-DB.com RSS Feed
Exploit-DB.com RSS Feed
Cisco Talos Blog
Cisco Talos Blog
T
Threat Research - Cisco Blogs
P
Privacy International News Feed
S
Schneier on Security
P
Privacy & Cybersecurity Law Blog
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
云风的 BLOG
云风的 BLOG
P
Proofpoint News Feed
Scott Helme
Scott Helme
人人都是产品经理
人人都是产品经理
G
GRAHAM CLULEY
O
OpenAI News
CTFtime.org: upcoming CTF events
CTFtime.org: upcoming CTF events
PCI Perspectives
PCI Perspectives
GbyAI
GbyAI
宝玉的分享
宝玉的分享
Y
Y Combinator Blog
T
Troy Hunt's Blog
让小产品的独立变现更简单 - ezindie.com
让小产品的独立变现更简单 - ezindie.com
C
CXSECURITY Database RSS Feed - CXSecurity.com
腾讯CDC
C
Check Point Blog
Spread Privacy
Spread Privacy
L
LINUX DO - 最新话题
Recent Announcements
Recent Announcements
大猫的无限游戏
大猫的无限游戏
P
Palo Alto Networks Blog
Hacker News: Ask HN
Hacker News: Ask HN
M
MIT News - Artificial intelligence
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
The Hacker News
The Hacker News
H
Hacker News: Front Page
Microsoft Azure Blog
Microsoft Azure Blog
I
InfoQ
T
Tor Project blog
Martin Fowler
Martin Fowler
博客园 - 叶小钗
罗磊的独立博客
C
Cyber Attacks, Cyber Crime and Cyber Security
H
Heimdal Security Blog
V
Vulnerabilities – Threatpost
Simon Willison's Weblog
Simon Willison's Weblog
Latest news
Latest news
WordPress大学
WordPress大学
G
Google Developers Blog
N
Netflix TechBlog - Medium
S
Security Affairs
S
Secure Thoughts
Know Your Adversary
Know Your Adversary

博客园_首页

Plist 二进制格式 Milvus 和 PGVector,哪个更好? OpenClaw 已过时?在 VS Code 中运行 Hermes Agent! 第30篇文章:一个大三计科生的自白 Manim如何在数学公式中完美显示中文? Docker 部署 RocketMQ 5 并发编程核心概念辨析 C#事务处理最佳实践:别再让“主表存了、明细丢了”的破事发生 CLI 是什么?为什么大厂突然集体卷命令行? 【从0到1构建一个ClaudeAgent】协作-自主Agent UIImageView 设置图片不生效的原因排查 最小二乘问题详解20:无先验约束下的增量式SFM自由网平差 痞子衡嵌入式:大话双核i.MXRT1180之XIP应用里借助MU实现可靠Flash IAP的方法 AI Chat 封装, SemanticKerne.AiProvider.Unified 已发布 Windows下右键编辑js文件无法打开记事本——在注册表中使用环境变量 在后台服务中使用 Scoped 服务,为什么总是报错? H200 安装驱动并使用sglang启动模型 wireshark 抓包Trap上报告警内容 我用 AI 辅助开发了一系列小工具(2):图片压缩工具 [A Primer On MC and CC] 2.1 Memory Consistency 1 - 指令重排序和 SC 模型 Oracle数据库SCN推进技术详解与实践指南 玩转控件:封装个带图片的Label控件 Claude Code 4.7 真正该升级的不是模型,而是你的工作流 前端小白一句话,AI 帮我做了个颜值拉满的桌面媒体播放器。当代码不再是门槛,一句话编程就是现实。 5. WorkBuddy: 小龙虾的灵魂三件套,让你的小龙虾不只是工具 SQLite 分片方案实战:三种分片策略的深度对比 告别简陋 UI!一款基于 Fluent Design 和基于 WinUI 的开源免费、现代化的 Avalonia UI 控件库 关于二进制排列组合枚举的总结 AI开发-python-LangGraph框架(3-27-LangGraph从零实现大模型智能决策工作流) ElasticSearch主分片和副本分片概念详解 【002】HTTPS 粗解:证书、TLS 握手与对后端配置的影响 Hermes Agent 一周暴涨五万 Star,但我劝你别急着追 明明连接的是Redis的DB0,为什么能查到DB3的数据? 【从0到1构建一个ClaudeAgent】协作-Agent团队 熟悉电子元器件之后,电子小白下一步该怎么走? MAF快速入门(23)通过C#类定义Skills .NET 高级开发 | 手写一个对象映射框架 FastAPI数据库ORM怎么选?我肝了三个Demo后,终于不再纠结了 mysqldump 参数拾遗:在遗忘与铭记之间 C# .NET 周刊|2026年3月5期 Claude code入门 - 陈彦斌 一文学习入门 ThingsBoard 开源物联网平台 GitHub 热门项目 | 2026年04月16日 如何为GIT设置全局勾子,为每次提交追加信息 Number.isFinite和isFinite与isNaN()和Number.isNaN的区别 PortSwigger SQL注入LAB2 推荐一个测试人必备的Skills,从功能到性能全搞定(附详细实操和安装下载方式) 筑基期:掌握Odoo基础核心知识点02(Odoo XML 开发方式详解) GLM模型这么火,咱们用vllm也咧一个呗! 深入理解 AbortController:从底层原理到跨语言设计哲学 字符串学习笔记 多租户系统框架的基础模块设计和分析设计 Apache SeaTunnel Zeta 为什么能做到“又快又稳”? AI开发-python-LangGraph框架(3-26-LangGraph基本概念及第一个简单样例) Vue 3 组件通信,别只会用 Props 和 Emits 了,这几个狠活儿你得看看 ElasticSearch7.X版本配置密码 用Manim实现动态交点计算--从一个动点问题说起 团结引擎+Addressable+Instant Game打包抖音小游戏 function call 实战:让 LLM 自动判断 pod 异常、调用日志工具并完成故障分析 bubseek —— 让 Agent 的足迹,变成团队的洞察 通过 C# 读取并导出 PDF 书签 如何用 GitHub Actions 实现 Steam 自动化发布 【从0到1构建一个ClaudeAgent】并发-后台任务 .NET 高级开发 | 定制 ASP.NET Core 框架 电子小白:什么是运算放大器(运放) zero2Agent:面向大厂面试的 Agent 工程教程,从概念到生产的完整学习路线 堆上的ORW HC32F460 USB CDC通信异常:非对齐访问异常排查 20260413-Hyperbridge 攻击事件:发生在默克尔山上的验证绕过 那些喊着AI 要淘汰你的人,正在靠你的焦虑赚大钱! 深度学习进阶(八)Swin Transformer 最小二乘问题详解19:带先验约束的增量式SFM优化与实现 SnapTranslate 3.0 正式发布:全局划词翻译 + 完整英语学习闭环,一站式搞定查词、记词、复习 工作的意义、工作的困难认知再思考 .NET + AI 进阶实战:基于类的技能开发 - 打造可治理的 Agent 能力模块 【从0到1构建一个ClaudeAgent】规划与协调-技能 上周热点回顾(4.6-4.12) 电子小白的工具三件套:面包板、杜邦线、万能板 单表五亿数据的查询优化 | Mysql、StarRocks 2. WorkBuddy:从“我是谁”到“帮我干活” C# 如何减少代码运行时间:7 个实战技巧 基于HelixToolkit.SharpDX 渲染3D模型 - 笺上知微 从零开始的双臂具身VLA起源及现阶段发展综述 - SkyXZ 记对 xonsh shell 的使用, 脚本编写, 迁移及调优 - pluvium27 受够了Vibe Coding的失控?换个起点,让AI事半功倍 从开始配置漏洞环境到漏洞复现流程 - 難しい 关于10年工作经验的程序员对OpenClaw的实战经验分享以及看法 - 虚无境 Any metadata 的内存布局 C# .NET 周刊|2026年3月2期 - InCerry 我帮你测过了,测试圈排名第二的 Skill 依然很牛逼 Skill Discovery | 无监督技能发现的经典工作总结 - MoonOut 上下文工程是什么?过时了么?一文讲明白! - 一枫说码 开了 TUN 模式还是直连?90% 的人都踩过这个坑 AScript扩展多种脚本语言 - rockey627 AI 学习笔记:Agent 的记忆机制 你能被装进一个文件里吗?——7 万人把同事"蒸馏"成了 AI - 我没有三颗心脏 Claude Code 通关手册(七):给 AI 装上技能包——Skills 完全指南 - 暮色之狐 在浏览器中快速编辑代码:VSCode Web 集成实践 - Newbe36524 蒸馏自己 skill?基于 Deepseek 的蒸馏器,丐版蒸馏方式,简单便捷 - To_Carpe_Diem Spring AI Aliababa和AgentScope,哪个更好? - 苏三说技术
记一次微服务架构下的HTTP请求头“大小写”丢失排查之旅
杜劲松 · 2026-06-18 · via 博客园_首页

在最近的微服务排障过程中,业务方反馈了一个诡异的问题:客户端发起请求时,明确携带了驼峰写法的请求头(如 appKey: asd),但请求经过反向代理和网关,到达后端具体的 Spring Boot 业务服务时,业务代码里取出来的请求头全变成了小写(appkey: asd)。

面对这种链路较长的问题,最忌讳的就是靠猜。是 Nginx 做了转换?是 Gateway 的某个 Filter 偷偷改了头?还是 Spring Boot 本身的问题?为了用事实说话,我直接上 tcpdumpWireshark,在链路的各个节点进行了分段抓包。

测试环境拓扑与机器信息

在开始抓包前,我们需要明确当前测试环境的具体流量拓扑和节点的 IP、端口信息。根据排查梳理,当前链路如下:

  • 客户端 (Postman):IP 10.20.4.84
  • Nginx (反向代理):IP 10.100.38.11,对外暴露端口 9009
  • Spring Cloud Gateway (网关):IP 10.100.22.48,服务端口 8081
  • Spring Boot 业务服务:IP 10.100.22.48,服务端口 8071注意:业务服务与网关部署在同一台物理机上

完整的流量流向: 客户端(10.20.4.84) -> Nginx(10.100.38.11:9009) -> Gateway(10.100.22.48:8081) -> Spring Boot(10.100.22.48:8071)

第一阶段:抓取 客户端 -> Nginx(入向请求)

首先确认客户端发出的报文到底对不对。在 Nginx 所在机器(10.100.38.11)上抓取目标端口为 9009 的入向流量:

Bash

sudo tcpdump -i ens192 -s 0 -nn 'tcp dst port 9009' -w /tmp/nginx_before.pcap

将文件导出到本地后,通过 Wireshark 打开。为了快速定位,可以使用包含接口特征的过滤条件,例如: tcp contains "responseSpecialBufferSizeOfPost"

定位到目标数据包后,右键点击该数据包 -> 追踪流 (Follow) -> TCP 流 (TCP Stream),即可看到直观的 HTTP 原始报文。

还原出的原始 HTTP 请求如下:

HTTP

POST /api-gateway-dev/demo-business-service/sms/responseSpecialBufferSizeOfPost HTTP/1.1
appid: demo-business-service
appKey: asd
Content-Type: application/json
User-Agent: PostmanRuntime/7.54.0
Host: 10.100.38.11:9009
Content-Length: 1868

{"xn":"0462add21538f...[报文过长,此处省略]...1b4f7b1911"}

结论: 客户端确实发送了驼峰格式的 appKey: asd,源头没问题。

第二阶段:抓取 Nginx -> Gateway(出向请求)

接着排查是不是 Nginx 在转发时对 Header 做了手脚。依然在 Nginx 机器上,抓取 Nginx 发往 Gateway(10.100.22.48:8081)的流量:

Bash

sudo tcpdump -i ens192 -s 0 -nn 'tcp and dst host 10.100.22.48 and dst port 8081' -w /tmp/nginx_after.pcap

查看报文内容:

HTTP

POST /demo-business-service/sms/responseSpecialBufferSizeOfPost HTTP/1.1
Host: 10.100.38.11
X-Real-IP: 10.20.4.84
X-Real-Port: 50649
X-Forwarded-For: 10.20.4.84
Content-Length: 1868
appid: demo-business-service
appKey: asd

结论: Nginx 增加了几个 X- 开头的代理头,但原封不动地保留了 appKey 的驼峰格式。Nginx 洗清嫌疑。

第三阶段:抓取 Nginx -> Gateway(入向请求确认)

为了严谨,我们前往 Gateway 所在机器(10.100.22.48),确认网关网卡实际收到的报文内容。

💡 技术要点:为什么这里不筛选 Nginx 的源端口(9009)? > 因为 Nginx 在作为反向代理将请求转发给上游服务器时,它自己扮演了“客户端”的角色。系统会为这个新的 TCP 连接随机分配一个临时的动态端口(Ephemeral Port,例如 53971),而不是复用外部客户端访问 Nginx 时的 9009 端口。如果强行加上 src port 9009,将抓不到任何转发包。

因此,过滤条件只需指定源 IP 和目标端口:

Bash

sudo tcpdump -i ens33 -s 0 -nn 'src host 10.100.38.11 and dst port 8081' -w /tmp/gateway_before.pcap

报文显示 appKey: asd 依然坚挺。网关入向流量正常。

第四阶段:抓取 Gateway -> Spring Boot(出向请求)

这是最关键的一环。Spring Cloud Gateway 底层基于 Netty 构建,中间经过了一系列的 Routing Filter 配置,它会修改 Header 吗?

由于 Gateway 和最终的下游 Spring Boot 业务服务部署在同一台机器10.100.22.48)上,我们需要在网关机器上抓取发往下游业务服务(8071端口)的流量。

💡 技术要点:为什么网卡参数使用的是 -i any > 当服务部署在同一台机器时,它们之间的网络通信通常不会经过外部的物理网卡(如 ens33),而是直接走系统的本地回环网卡(Loopback interface,即 lo,IP 为 127.0.0.1 或是通过内网 IP 内部路由)。使用 -i any 可以监听本机上所有网络接口的流量,无论它们是走物理网卡出网,还是在本地回环网络内通信,都能做到“一网打尽”,避免因选错网卡而抓不到包的情况。

Bash

sudo tcpdump -i any -s 0 -nn 'src host 10.100.22.48 and dst port 8071 and tcp' -w /tmp/gateway_after.pcap

通过 Wireshark 追踪这部分报文,我们看到了网关发出的最终内容:

HTTP

POST /sms/responseSpecialBufferSizeOfPost HTTP/1.1
X-Real-IP: 10.20.4.84
X-Real-Port: 61676
X-Forwarded-For: 10.20.4.84,10.100.38.11
Content-Length: 720
appid: demo-business-service
appKey: asd
Content-Type: application/json
Forwarded: proto=http;host=10.100.38.11;for="10.100.38.11:58748"
host: 10.100.22.48:8071
sw8: 1-YWM4Nj...[链路追踪头信息省略]...MTAuMTAwLjIyLjQ4OjgwNzE=

结论: 网关追加了 SkyWalking 链路追踪相关的头(sw8 等)以及 Forwarded 路由信息,但依然完美透传appKey: asd。网关也洗清了嫌疑!

峰回路转:真相在最后一公里

既然整个网络链路(Nginx -> Gateway -> 本地网卡)都没有修改请求头的大小写,那问题必定出在 Spring Boot 服务内部。

为了验证,我直接使用 Postman 直连 Spring Boot 服务(10.100.22.48:8071)发起请求,并在代码中断点调试以下获取请求头的方法:

Java

Enumeration<String> headerNames = request.getHeaderNames();
while (headerNames.hasMoreElements()) {
    String headerName = headerNames.nextElement();
    System.out.println(headerName + " : " + request.getHeader(headerName));
}

运行结果让人大跌眼镜:遍历打印出来的 headerName 全是小写的 appkey

根因剖析:RFC 规范与 Tomcat 源码实现

为什么 Spring Boot 会把 Header 的名字变成小写?

实际上,HTTP/1.1 规范(RFC 2616 章节 4.2)明确规定:HTTP Header 的字段名称是大小写不敏感的(case-insensitive)。 到了 HTTP/2,规范更是直接强制要求所有的 Header 名称在传输层必须被转换为全小写。

为了遵循这一规范并提高匹配效率,Spring Boot 内嵌的 Tomcat 容器在解析 HTTP 协议的极底层代码中,直接完成了大写到小写的转换。以当前项目的 Spring Boot 2.7.18(默认内嵌 Tomcat 9.0.83)为例,核心转换逻辑发生在 Coyote HTTP/1.1 处理器的 Http11InputBuffer 类中。

Tomcat 会逐字节读取 HTTP 报文。当它在 parseHeader() 方法中解析到 Header Name 时,会直接在底层的 ByteBuffer 里进行原地替换(In-place conversion)。具体源码信息如下:

关键代码片段截取:

Java

// 截取自 Http11InputBuffer#parseHeader() 读取 Header Name 字节流的逻辑
if (chr >= Constants.A && chr <= Constants.Z) {
    byteBuffer.put(byteBuffer.position() - 1, (byte) (chr - Constants.LC_OFFSET));
}

原理解释:

  • chr 是当前读取到的单个字节。
  • 当判断 chr 为大写字母(在 A (65) 和 Z (90) 的 ASCII 码之间)时,执行 chr - Constants.LC_OFFSET
  • Constants.LC_OFFSET 的定义是 A - a(即 65 - 97 = -32)。所以 chr - (-32) 实质上就是 chr + 32,这正是 ASCII 码表中大写字母转换为小写字母的数学偏移量。

经过 Http11InputBuffer 这样极其底层的按字节剥离处理后,被强制转为小写的 Header Name 最终才会被封装成 MessageBytes 对象,并存入 org.apache.tomcat.util.http.MimeHeaders 结构中供后续路由和业务使用。因此,当你通过 request.getHeaderNames() 获取枚举迭代器时,暴露出来的迭代值自然就是小写的形式了。

但需要特别注意的是,虽然遍历出来是小写,由于规范要求“大小写不敏感”,且 Tomcat 底层查找逻辑做了忽略大小写的兼容,所以当你使用指定 key 去获取时: request.getHeader("appKey")request.getHeader("APPKEY") 以及 request.getHeader("appkey") 都能正确获取到对应的值 "asd"

总结与避坑指南

  1. 排查思路: 遇到跨多节点的网络问题,善用 tcpdump 结合 Wireshark 追踪 TCP 流是最直接高效的手段。不要过度依赖主观猜测。清楚流量拓扑、正确选择网卡(同机用 anylo)以及理解代理转发时的端口分配机制,能避免在抓包时走弯路。
  2. 开发规范: 永远不要依赖 HTTP 请求头的大小写来做业务逻辑判断(比如 if(name.equals("appKey")))。如果必须遍历 Headers 并做精确匹配,请使用 equalsIgnoreCase 进行比较。
  3. 获取方式: 推荐直接通过 @RequestHeader("appKey") 注解或 request.getHeader("appKey") 明确获取,容器底层会帮我们处理好大小写兼容的问题。尽量避免通过 request.getHeaderNames() 遍历取 key 后再做强校验。