`

Java 调用Web service 添加认证头(soapenv:Header)

 
阅读更多
 
 

前言

有时候调用web service 会出现

Message does not conform to configured policy [ AuthenticationTokenPolicy(S) ]:  No Security Header found 

这样的错误。

以在 soapui 调用的结果来看, 会出现如下的返回

出现这种错误的原因 是webservice 的服务端需要提供 soap 认证的表头。

举例来说, 可能需要加上如下的认证头:

 

[html] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. <soapenv:Header>  
  2.   <wsse:Security xmlns:wsse="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd" soapenv:mustUnderstand="1">  
  3.     <wsse:UsernameToken>  
  4.       <wsse:Username>UserName</wsse:Username>  
  5.       <wsse:Password Type="http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-username-token-profile-1.0#PasswordText">Password</wsse:Password>  
  6.     </wsse:UsernameToken>  
  7.   </wsse:Security>  
  8. </soapenv:Header>  
(这个细部的格式和服务端的要求有关, 具体的username和pass也是服务端提供的)

在sopaui 调用的时候, 加上类似, 就能呼叫成功了。

 

 

在soapui 调用, 可以用以上方式来做。在把wsdl 转为java 后, 又该如何加上认证的头信息呢?

 

cxf 加上认证头

(以上的认证头, 比较接近cxf 的调用方式。)

如果使用的是cxf生产的客户端的代码。

(如何生成,参考 CXF 生成Web Service Client(将WSDl 转化成 Java代码)

在 _Client 调用的时候加上 如下代码:(在方法调用的代码之前)

 

[java] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. Map<String, Object> props = new HashMap<String, Object>();  
  2. props.put(WSHandlerConstants.ACTION, WSHandlerConstants.USERNAME_TOKEN);        
  3. props.put(WSHandlerConstants.PASSWORD_TYPE,WSConstants.PW_TEXT);  
  4. props.put(WSHandlerConstants.USER, "UserName");  
  5. props.put(WSHandlerConstants.PW_CALLBACK_CLASS, PasswordHandler.class.getName());  
  6. WSS4JOutInterceptor wssOut = new WSS4JOutInterceptor(props);  
  7. Client client = ClientProxy.getClient(port);  
  8. client.getOutInterceptors().add(wssOut);  

在client 的java 文件中, 新增以下内部类

 

 

[java] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. public static class PasswordHandler implements CallbackHandler  
  2. {  
  3.     public void handle(javax.security.auth.callback.Callback[] callbacks) {  
  4.         for (int i = 0; i < callbacks.length; i++) {  
  5.             WSPasswordCallback pc = (WSPasswordCallback)callbacks[i];  
  6.             pc.setPassword("password");  
  7.         }  
  8.     }  
  9.   
  10. }  


 

Axis2 加上认证头

针对以上的认证头在axis2 产生的java 文件中如何添加呢(Axis2自动产生的java 文件并不会自动产生main的测试文件, 需要自己写。  XXXXProxy.java 这是供调用的类文件。 不过这些和添加认证头关系不大)

axis2会产生一个  XXXXPortBindingStub.java 的文件。 这里面的内容就是实际的方法体。

找到我们需要调用的那个方法体:

在方法调用之前,加入以下代码:

 

[java] view plaincopy在CODE上查看代码片派生到我的代码片
 
  1. //Begin add for Header  
  2.         String AUTH_PREFIX = "wsse";  
  3.         String AUTH_NS = "http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-wssecurity-secext-1.0.xsd";  
  4.          try{  
  5.             SOAPElement wsSecHeaderElm = soapFactory.createElement("Security", AUTH_PREFIX, AUTH_NS);          
  6.             SOAPElement userNameTokenElm = soapFactory.createElement("UsernameToken",AUTH_PREFIX, AUTH_NS);            
  7.             SOAPElement userNameElm = soapFactory.createElement("Username",AUTH_PREFIX, AUTH_NS);           
  8.             SOAPElement passwdElm = soapFactory.createElement("Password",AUTH_PREFIX, AUTH_NS);                   
  9.             passwdElm.setAttribute("Type""http://docs.oasis-open.org/wss/2004/01/oasis-200401-wss-username-token-profile-1.0#PasswordText");  
  10.             
  11.        userNameElm.addTextNode("vend_bmc01");  
  12.        passwdElm.addTextNode("mediatek");  
  13.               
  14.       userNameTokenElm.addChildElement(userNameElm);  
  15.       userNameTokenElm.addChildElement(passwdElm);  
  16.       wsSecHeaderElm.addChildElement(userNameTokenElm);           
  17.       SOAPHeaderElement soapHeaderElement =  new SOAPHeaderElement(wsSecHeaderElm);  
  18.       soapHeaderElement.setMustUnderstand(true);  
  19.       _call.addHeader(soapHeaderElement);    
  20.         }catch(Exception e)  
  21.         {  
  22.             e.printStackTrace();  
  23.         }          
  24.       //End add for Header  
  25.    
  26. ===》call method  
  27.        java.lang.Object _resp = _call.invoke(XXXX);  

原理很简单。 加上类似xml 的头, _call_addHeader

 

加上之后,再触发方法 _call.invoke

分享到:
评论

相关推荐

    Java Tutorial: Creating Web Services

    ### Java 教程:创建 Web 服务 #### 1. 引言 在互联网的早期阶段,Web 应用程序通过 HTML 提供静态网页。这种网站开发方式虽然简单,但静态内容很容易过时,因此网站的内容管理变得尤为重要。为了向 Web 用户提供...

    java调用soap接口案例

    Java调用SOAP(Simple Object Access Protocol)接口是Web服务交互中的常见操作,SOAP是一种基于XML的协议,用于在Web上交换结构化的和类型化的信息。本案例主要关注如何使用Java来实现SOAP请求并获取响应,具体我们...

    金蝶调用Webservice接口-SOAP

    在金蝶中调用Web服务的第一步是创建服务对象。这通过`Service.create()`方法完成,传入WSDL(Web Services Description Language)URL和QName( Qualified Name)来定义服务的命名空间和本地名称。例如: ```java ...

    Web 服务编程技巧和窍门: 手工创建的 SOAP 消息中命名空间的处理

    4. 创建SOAP Body或Header:在`SOAPEnvelope`上,使用`addBody()`或`addHeader()`方法创建相应的部分。 5. 添加元素:在SOAP Body或Header中,使用`createElement()`方法创建元素,并指定命名空间前缀和本地名称。...

    模拟soapui调用webservice

    此外,如果Web服务需要身份验证,我们还需要在请求头中添加认证信息,如Basic Auth或Token。 博客文章可能还涵盖了如何解析返回的SOAP响应,这通常涉及XML解析库,如Java的JAXB或DOM,或者Python的ElementTree。...

    webservice接口的wsdl文件已经生成java代码.docx

    4. 使用Java调用Web服务: 生成的Java代码通常包含一个服务代理类(Stub),开发者可以通过这个类来调用Web服务。例如,使用CXF,可以设置服务端点(Endpoint): ```java MyWebService service = new MyWebService...

Global site tag (gtag.js) - Google Analytics