0

0

告别SOAP安全认证的噩梦:php-soap/psr18-wsse-middleware助你轻松搞定WSSE/WSA

聖光之護

聖光之護

发布时间:2025-11-07 12:57:06

|

468人浏览过

|

来源于php中文网

原创

告别soap安全认证的噩梦:php-soap/psr18-wsse-middleware助你轻松搞定wsse/wsa

最近在负责一个项目,需要与一个老旧但核心的SOAP服务进行数据交互。这个服务对安全性要求极高,不仅需要基本的用户名密码认证,还强制要求对请求进行数字签名,并且需要特定的WS-Addressing头。

你是否也曾遇到过这样的场景?当SOAP服务的文档摆在你面前,里面充斥着WSSE、WSA、X.509证书、XML数字签名、时间戳等各种术语,你是不是瞬间感到一阵眩晕?我当时就是这样。

尝试过最初的方案,是手动拼接XML,或者使用一些老旧的SOAP客户端,但很快就遇到了瓶颈:

  1. 复杂且易错的XML操作: WSSE头部的结构非常复杂,包含各种命名空间、元素顺序、时间戳、二进制安全令牌、签名信息等。手动构建或修改这些XML字符串,就像在走钢丝,一个字符的错误都可能导致服务拒绝。
  2. 证书和密钥管理: 数字签名需要用到私钥和公钥证书。如何安全地加载、使用这些密钥,并正确地将它们嵌入到XML中,成了一个大难题。
  3. 调试困难: 当请求被拒绝时,SOAP服务通常只会返回一个模糊的错误信息,很难定位是WSSE头部的结构问题、签名问题,还是证书配置问题。
  4. 缺乏灵活性: 不同的SOAP服务可能有不同的WSSE配置要求(例如,签名算法、摘要算法、KeyIdentifier类型)。手动方案很难快速适应这些变化。

眼看项目进度受阻,我开始寻找一个更现代、更健壮的解决方案。这时,Composer的强大生态再次展现了它的魅力。

Composer在线学习地址:学习地址

立即学习PHP免费学习笔记(深入)”;

我发现了 php-soap/psr18-wsse-middleware 这个库,它简直是我的救星!这个包是为基于PSR-18的HTTP SOAP传输而设计的中间件,它能将复杂的WSSE和WSA安全逻辑封装起来,让我们以更简洁、更PHP友好的方式进行配置。

核心思想:中间件与可配置条目

php-soap/psr18-wsse-middleware 的核心在于它作为一个PSR-18 HTTP客户端的中间件,可以轻松地插入到你的HTTP请求流程中。它将WSSE的各种安全特性抽象为一系列可配置的“条目(Entry)”,例如:UsernameTimestampBinarySecurityTokenSignatureEncryption 等。你只需要根据SOAP服务的具体要求,组合这些条目即可。

如何使用它解决问题?

首先,通过Composer安装它:

composer require php-soap/psr18-wsse-middleware

这个包通常与 php-soap/psr18-transport 配合使用,后者提供了PSR-18标准的SOAP传输能力。

1. 添加WS-Addressing (WSA) 头

如果你的SOAP服务需要WSA头(例如,ActionTo 等),你可以使用 WsaMiddlewareWsaMiddleware2005

use Http\Client\Common\PluginClient;
use Soap\Psr18Transport\Psr18Transport;
use Soap\Psr18WsseMiddleware\WsaMiddleware;
use Soap\Psr18WsseMiddleware\WsaMiddleware2005;
use GuzzleHttp\Client as GuzzleClient; // 假设你使用Guzzle作为PSR-18客户端

$psr18Client = new GuzzleClient(); // 你的PSR-18兼容HTTP客户端

$transport = Psr18Transport::createForClient(
    new PluginClient($psr18Client, [
        new WsaMiddleware(), // 或 new WsaMiddleware2005(),取决于WSA版本
    ])
);

// 接下来,你可以使用这个 $transport 对象来发送SOAP请求
// ...

通过简单的实例化和添加到插件客户端,WSA头就自动注入到你的SOAP请求中了,省去了手动构建XML的麻烦。

QIMI奇觅
QIMI奇觅

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

下载

2. 实现WS-Security (WSSE)

这才是真正的“大头”。WsseMiddleware 允许你配置各种WSSE条目。它的设计理念非常棒,如果你在SoapUI中配置过WS-Security,你会发现这里的PHP代码结构与SoapUI的UI配置非常相似,这极大地降低了学习成本。

示例一:添加用户名和密码认证

这是最常见的WSSE需求之一。

use Soap\Psr18WsseMiddleware\WsseMiddleware;
use Soap\Psr18WsseMiddleware\WSSecurity\Entry;
use Http\Client\Common\PluginClient;
use Soap\Psr18Transport\Psr18Transport;
use GuzzleHttp\Client as GuzzleClient;

$user = 'myUsername';
$password = 'myPassword';

$wsseMiddleware = new WsseMiddleware(
    outgoing: [
        (new Entry\Username($user))
            ->withPassword($password)
            ->withDigest(false), // 如果密码不需要摘要,设置为false
        new Entry\Timestamp(60), // 添加一个有效期60秒的时间戳
    ]
);

$transport = Psr18Transport::createForClient(
    new PluginClient(new GuzzleClient(), [
        $wsseMiddleware,
    ])
);

// 使用 $transport 发送SOAP请求,WSSE头会自动添加

几行代码,就完成了复杂的用户名密码和时间戳注入,比手动拼接XML要清晰和安全得多。

示例二:使用X.509证书对请求进行数字签名

这是我遇到的最棘手的问题之一。服务要求使用PKCS12证书进行签名。这个库提供了 KeyStore 类来管理密钥和证书。

use Soap\Psr18WsseMiddleware\WsseMiddleware;
use Soap\Psr18WsseMiddleware\WSSecurity\Entry;
use Soap\Psr18WsseMiddleware\WSSecurity\KeyStore\Certificate;
use Soap\Psr18WsseMiddleware\WSSecurity\KeyStore\Key;
use Soap\Psr18WsseMiddleware\WSSecurity\KeyIdentifier;
use Soap\Psr18WsseMiddleware\WSSecurity\SignatureMethod;
use Soap\Psr18WsseMiddleware\WSSecurity\DigestMethod;
use Http\Client\Common\PluginClient;
use Soap\Psr18Transport\Psr18Transport;
use GuzzleHttp\Client as GuzzleClient;

// 假设你的私钥和公钥证书文件
$privateKeyFile = 'path/to/security_token.priv'; // 从PKCS12转换出的私钥
$publicKeyFile = 'path/to/security_token.pub';   // 从PKCS12转换出的公钥证书
$passphrase = 'your_key_passphrase';

$privKey = Key::fromFile($privateKeyFile)->withPassphrase($passphrase);
$pubKey = Certificate::fromFile($publicKeyFile);

$wsseMiddleware = new WsseMiddleware(
    outgoing: [
        new Entry\Timestamp(60),
        new Entry\BinarySecurityToken($pubKey), // 将公钥证书作为二进制安全令牌
        (new Entry\Signature(
            $privKey,
            new KeyIdentifier\BinarySecurityTokenIdentifier() // 指定签名时使用二进制安全令牌作为标识
        ))
            ->withSignatureMethod(SignatureMethod::RSA_SHA256) // 签名算法
            ->withDigestMethod(DigestMethod::SHA256)       // 摘要算法
            ->withSignAllHeaders(true)                      // 签名所有SOAP头
            ->withSignBody(true)                            // 签名SOAP Body
    ]
);

$transport = Psr18Transport::createForClient(
    new PluginClient(new GuzzleClient(), [
        $wsseMiddleware,
    ])
);

// 现在,你的SOAP请求会自动包含签名和证书信息

这段代码清晰地展示了如何加载密钥、配置签名算法、摘要算法以及签名范围。它将所有复杂性封装在 Entry\Signature 对象中,我们只需提供正确的参数,大大降低了出错的概率。

示例三:加密和解密敏感数据

某些服务甚至要求对SOAP请求或响应中的敏感部分进行加密。php-soap/psr18-wsse-middleware 也提供了 Entry\EncryptionEntry\Decryption

// ... (密钥和证书加载与签名类似)

$wsseMiddleware = new WsseMiddleware(
    outgoing: [
        // ... 其他如Timestamp, BinarySecurityToken, Signature 条目
        (new Entry\Encryption(
            $signKey, // 用于加密的公钥证书
            new KeyIdentifier\X509SubjectKeyIdentifier($signKey)
        ))
            ->withKeyEncryptionMethod(KeyEncryptionMethod::RSA_OAEP_MGF1P)
            ->withDataEncryptionMethod(DataEncryptionMethod::AES256_CBC)
    ],
    incoming: [
        new Entry\Decryption($privKey) // 用于解密响应的私钥
    ]
);
// ...

注意:官方文档提到,加密功能在底层库中存在一个已知bug,并提供了Composer补丁的解决方案。这体现了在实际使用中需要关注细节和社区支持。

总结其优势和实际应用效果

使用 php-soap/psr18-wsse-middleware 之后,我深刻体会到了它带来的巨大优势:

  1. 告别XML噩梦: 最直接的感受就是再也不用手动去拼接和解析复杂的WSSE/WSA XML头了。库内部处理了所有细节,大大减少了出错的可能性。
  2. 开发效率大幅提升: 以前需要花费数天甚至一周来调试和解决的SOAP安全认证问题,现在通过阅读文档和配置几个 Entry 对象,往往只需几个小时就能搞定。
  3. 代码可读性和可维护性增强: 清晰的面向对象API,让WSSE配置逻辑一目了然。当服务方调整安全要求时,我们只需修改或添加相应的 Entry,而不是修改大段的XML字符串。
  4. 高度灵活性: 它支持多种WSSE/WSA标准和算法,无论是用户名密码、数字签名、时间戳、SAML断言,还是加密解密,都能轻松应对。这使得它能够适应各种复杂的SOAP服务集成场景。
  5. 现代化集成: 基于PSR-18 HTTP客户端,使得它可以与现代PHP生态中的HTTP客户端(如Guzzle)无缝集成,符合现代PHP开发的最佳实践。

在我的项目中,通过引入 php-soap/psr18-wsse-middleware,我们成功地与遗留的SOAP服务建立了安全、稳定的连接,并按时完成了项目交付。它不仅解决了技术难题,更重要的是,将开发者从繁琐且易错的SOAP安全配置中解放出来,让我们能够专注于业务逻辑的实现。

如果你也在为SOAP服务的WSSE/WSA安全认证而烦恼,强烈推荐你尝试一下 php-soap/psr18-wsse-middleware。它将是你在SOAP集成之路上的得力助手!

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
composer是什么插件
composer是什么插件

Composer是一个PHP的依赖管理工具,它可以帮助开发者在PHP项目中管理和安装依赖的库文件。Composer通过一个中央化的存储库来管理所有的依赖库文件,这个存储库包含了各种可用的依赖库的信息和版本信息。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

154

2023.12.25

什么是中间件
什么是中间件

中间件是一种软件组件,充当不兼容组件之间的桥梁,提供额外服务,例如集成异构系统、提供常用服务、提高应用程序性能,以及简化应用程序开发。想了解更多中间件的相关内容,可以阅读本专题下面的文章。

178

2024.05.11

Golang 中间件开发与微服务架构
Golang 中间件开发与微服务架构

本专题系统讲解 Golang 在微服务架构中的中间件开发,包括日志处理、限流与熔断、认证与授权、服务监控、API 网关设计等常见中间件功能的实现。通过实战项目,帮助开发者理解如何使用 Go 编写高效、可扩展的中间件组件,并在微服务环境中进行灵活部署与管理。

215

2025.12.18

go语言 面向对象
go语言 面向对象

本专题整合了go语言面向对象相关内容,阅读专题下面的文章了解更多详细内容。

56

2025.09.05

java面向对象
java面向对象

本专题整合了java面向对象相关内容,阅读专题下面的文章了解更多详细内容。

52

2025.11.27

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指的是可扩展标记语言,标准通用标记语言的子集,是一种用于标记电子文件使其具有结构性的标记语言。想了解更多相关的内容,可阅读本专题下面的相关文章。

1069

2024.11.28

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

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

11

2026.01.29

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
第二十四期_PHP8编程
第二十四期_PHP8编程

共86课时 | 3.4万人学习

成为PHP架构师-自制PHP框架
成为PHP架构师-自制PHP框架

共28课时 | 2.5万人学习

第二十三期_PHP编程
第二十三期_PHP编程

共93课时 | 6.9万人学习

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

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