首页
/ Pistache HTTPS 服务器在客户端主机名验证失败时的异常处理问题分析

Pistache HTTPS 服务器在客户端主机名验证失败时的异常处理问题分析

2025-06-24 09:14:56作者:尤辰城Agatha

在 Pistache 开源 HTTP 服务器项目中,近期发现了一个与 HTTPS 连接处理相关的严重问题。当客户端(如 cURL 8.9.0 及以上版本)在进行 TLS 主机名验证失败时,服务器会变得无响应,只能通过重启服务来恢复。

问题现象

当 Pistache HTTPS 服务器配置了不符合客户端验证要求的证书(如缺少 SAN 扩展或 CN 不匹配),且客户端启用了主机名验证(cURL 默认行为)时,服务器会进入异常状态。具体表现为:

  1. 客户端发送请求后,服务器不再处理后续任何请求
  2. 虽然应用层代码仍能执行请求处理逻辑,但 SSL_write 调用不再被执行
  3. 客户端只能收到空响应

根本原因分析

经过深入调查,发现问题源于 OpenSSL 错误队列的处理机制。当客户端(如新版本 cURL)在验证失败时不发送 TLS close_notify 消息而直接关闭连接时:

  1. SSL_read 返回 <=0(表示读取失败)
  2. SSL_get_error 返回 SSL_ERROR_SSL(致命错误)
  3. OpenSSL 错误队列中会记录 SSL routines::unexpected eof while reading 错误
  4. 由于错误队列未被清空,导致后续所有 SSL I/O 操作不可靠

这与 OpenSSL 文档中明确指出的要求相违背:"当前线程的错误队列在尝试 TLS/SSL I/O 操作前必须为空,否则 SSL_get_error 将无法可靠工作"。

解决方案

修复方案相对简单但有效:在 SSL_read 失败后,主动清空 OpenSSL 错误队列。具体实现是在 Transport::handleIncoming 方法中添加错误队列清理逻辑:

bytes = SSL_read(reinterpret_cast<SSL*>(peer->ssl()),
               buffer + totalBytes,
               static_cast<int>(Const::MaxBuffer - totalBytes));
if (bytes <= 0) {
    int ssl_get_error_res = SSL_get_error(
        reinterpret_cast<SSL*>(peer->ssl()),
        static_cast<int>(bytes));
    while (ERR_get_error() != 0); // 清空错误队列
    retry = (ssl_get_error_res == SSL_ERROR_WANT_READ);
}

验证测试

为了验证修复效果,开发了一个专门的测试用例,通过 SSL_CTX_set_quiet_shutdown 显式禁用客户端的关闭通知发送:

TEST(https_server_test, basic_tls_request_with_no_shutdown_from_peer) {
    // 配置服务器和客户端
    // 启用quiet shutdown模拟客户端不发送关闭通知
    SSL_CTX_set_quiet_shutdown(reinterpret_cast<SSL_CTX*>(sslctx), 1);
    
    // 执行多次请求验证服务器稳定性
    for (const auto& req_i : { 0, 1, 2, 3 }) {
        res = curl_easy_perform(curl);
        EXPECT_EQ(res, CURLE_OK);
        EXPECT_EQ(buffer, "Hello, World!");
    }
}

测试结果表明,修复后服务器能够正确处理客户端异常关闭连接的情况,保持稳定运行。

安全启示

这一问题的发现和修复提醒我们:

  1. TLS 连接终止处理是安全通信中容易被忽视但至关重要的环节
  2. OpenSSL 错误队列管理是实现稳定 SSL/TLS 通信的基础要求
  3. 客户端行为变化(如 cURL 8.9.0 的修改)可能暴露出服务器实现的潜在问题
  4. 全面的异常情况测试是保证服务可靠性的必要手段

该修复已合并到 Pistache 主分支,显著提升了 HTTPS 服务在异常情况下的稳定性。

登录后查看全文
热门项目推荐

热门内容推荐

最新内容推荐

项目优选

收起
ohos_react_nativeohos_react_native
React Native鸿蒙化仓库
C++
176
262
RuoYi-Vue3RuoYi-Vue3
🎉 (RuoYi)官方仓库 基于SpringBoot,Spring Security,JWT,Vue3 & Vite、Element Plus 的前后端分离权限管理系统
Vue
863
511
ShopXO开源商城ShopXO开源商城
🔥🔥🔥ShopXO企业级免费开源商城系统,可视化DIY拖拽装修、包含PC、H5、多端小程序(微信+支付宝+百度+头条&抖音+QQ+快手)、APP、多仓库、多商户、多门店、IM客服、进销存,遵循MIT开源协议发布、基于ThinkPHP8框架研发
JavaScript
93
15
openGauss-serveropenGauss-server
openGauss kernel ~ openGauss is an open source relational database management system
C++
129
182
openHiTLSopenHiTLS
旨在打造算法先进、性能卓越、高效敏捷、安全可靠的密码套件,通过轻量级、可剪裁的软件技术架构满足各行业不同场景的多样化要求,让密码技术应用更简单,同时探索后量子等先进算法创新实践,构建密码前沿技术底座!
C
259
300
kernelkernel
deepin linux kernel
C
22
5
cherry-studiocherry-studio
🍒 Cherry Studio 是一款支持多个 LLM 提供商的桌面客户端
TypeScript
596
57
CangjieCommunityCangjieCommunity
为仓颉编程语言开发者打造活跃、开放、高质量的社区环境
Markdown
1.07 K
0
HarmonyOS-ExamplesHarmonyOS-Examples
本仓将收集和展示仓颉鸿蒙应用示例代码,欢迎大家投稿,在仓颉鸿蒙社区展现你的妙趣设计!
Cangjie
398
371
Cangjie-ExamplesCangjie-Examples
本仓将收集和展示高质量的仓颉示例代码,欢迎大家投稿,让全世界看到您的妙趣设计,也让更多人通过您的编码理解和喜爱仓颉语言。
Cangjie
332
1.08 K