从 .wsdl 为 java 生成 Web 服务客户端安全策略

Generate web service client secure policy from .wsdl for java

我一直在研究如何从 .wsdl 文件实施 Web 服务客户端策略。

Web 服务的策略涉及使用带有必要密钥的 .jks 文件进行签名和加密(用于签名的非对称私钥和用于加密的对称私钥)。政策是:用户名:oracle/wss10_username_token_with_message_protection_service_policy.

我可以使用 java 的 wsimport 工具(或使用 cxf 或 axis2)制作 .xsd 文件(请求、响应和服务对象)。我无法解决的是如何制定正确的政策。

有什么方法可以从 .wsdl 自动生成策略,还是我必须自己制作它们

用户名:oracle/wss10_username_token_with_message_protection_service_policy 用spring ws这样解决:

<!-- == Ougoing interceptor == -->
<bean id="loginOutgoingWss4jSecurityInterceptor" class="org.springframework.ws.soap.security.wss4j2.Wss4jSecurityInterceptor">
    <property name="securementActions" value="Timestamp Signature Encrypt" />

    <!--  == Set Outgoing Signature properties == -->
    <property name="securementUsername" value="alias"/>
    <property name="securementPassword" value="aliasPass"/>
    <property name="securementSignatureKeyIdentifier" value="DirectReference"/>
    <property name="securementSignatureCrypto" ref="cryptoFactoryBean" />
    <property name="securementSignatureParts" value="{Element}{}Body;{Element}{http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-utility-1.0.xsd}Timestamp;" />

    <!--  == Set Outgoing Encryption properties == -->
    <property name="securementEncryptionUser" value="alias"/> 
    <property name="securementEncryptionCrypto" ref="cryptoFactoryBean" />
    <property name="securementEncryptionKeyIdentifier" value="DirectReference"/>
    <property name="securementEncryptionParts" value="{Content}{}Body;" />
</bean>

<!-- == Incoming interceptor == -->
 <bean id="loginIncomingWss4jSecurityInterceptor" class="org.springframework.ws.soap.security.wss4j2.Wss4jSecurityInterceptor">
    <property name="validationActions" value="Timestamp Signature Encrypt" />

    <!--  == Set Validations Response, This validate signature and decrypts response == -->
    <property name="validateResponse" value="true" />

    <!-- The lower operation validation. Less time consume-->
    <property name="validateRequest" value="false" />
    <property name="enableSignatureConfirmation" value="false"/>

    <!--  == Set Incoming Signature/Decryption keystore == -->
    <property name="validationDecryptionCrypto" ref="cryptoFactoryBean" />
    <property name="validationSignatureCrypto" ref="cryptoFactoryBean" />

    <!-- Sets the {@link org.apache.ws.security.WSPasswordCallback} handler to use when validating messages -->
    <property name="validationCallbackHandler">
        <bean class="org.springframework.ws.soap.security.wss4j2.callback.KeyStoreCallbackHandler">
            <property name="privateKeyPassword" value="aliasPass"/>
        </bean>
    </property> 
 </bean>

如果您在 wsdl 中使用 WS-SecurityPolicy(1.1 或更高版本)中的策略,则无需生成策略,也无需使用 Apache CXF 在客户端创建它们。使用 WS-SecurityPolicy,CXF 的安全运行时是策略驱动的。

1) 您遵循 CXF 的 WSDL 优先方法生成客户端代码,使用 wsdl2java 命令行工具或 Maven cxf-codegen-plugin(wsdl2java 目标)。这在 CXF 文档的 How to develop a client.

中有所描述

2) 在 WS-SecurityPolicy usage, you configure the client security properties for the wsdl port you want to use, either using JAX-WS API (on the client's RequestContext) or Spring XML configuration. For the list of possible properties, there are the generic XML security ones and WS-Security-specific ones. Example with Spring XML for UsernameToken policy (from Glen Mazza's blog samples 上关注 CXF 的文档):

<beans xmlns="http://www.springframework.org/schema/beans"
   xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
   xmlns:jaxws="http://cxf.apache.org/jaxws"
   xsi:schemaLocation="http://www.springframework.org/schema/beans 
   http://www.springframework.org/schema/beans/spring-beans.xsd
   http://cxf.apache.org/jaxws 
   http://cxf.apache.org/schemas/jaxws.xsd">
<jaxws:client name="{http://www.example.org/contract/DoubleIt}DoubleItPort" createdFromAPI="true">
        <!-- Use this for the UsernameToken Symmetric Binding w/X.509 for secret key derivation -->
        <jaxws:properties>
            <entry key="ws-security.username" value="alice"/>        
            <entry key="ws-security.callback-handler" value="client.ClientPasswordCallback"/>        
            <entry key="ws-security.encryption.properties" value="clientKeystore.properties"/>
            <entry key="ws-security.encryption.username" value="myservicekey"/>
        </jaxws:properties>

        <!-- Use this for the UsernameToken Symmetric Binding w/UT password for secret key derivation -->
        <!--jaxws:properties>
            <entry key="ws-security.username" value="alice"/>        
            <entry key="ws-security.callback-handler" value="client.ClientPasswordCallback"/>        
        </jaxws:properties-->
</jaxws:client>
</beans>

将其放在 class 路径的 /cxf.xml 中。警告:该示例使用 CallbackHandler subclass(本例中为 client.ClientPasswordCallback)来提供密码。所以你需要提供你自己的实现。

3) 回到 CXF 文档的 How to develop a client - 最后一部分 - 在应用程序代码中,使用带有参数的 JAX-WS API 初始化客户端:a) WSDL 的位置(URL) 具有 WS-SecurityPolicy 策略(据我所知,您已经拥有); b) 客户端使用的服务和端口的 QName,如 WSDL 中所定义:

final Service service = Service.create(wsdlLocation, SERVICE_QNAME);
final DoubleItPortType transportPort = service.getPort(PORT_QNAME, DoubleItPortType.class);

4) 确保在运行时 class 路径上有 cxf-rt-ws-policycxf-rt-ws-security 模块以启用 WS-SecurityPolicy 支持。