今天项目上面需要做一个CXF+ws-security和HttpURLConnection调用第三方的webService进行访问,这里我主要做客户端访问,服务器端和客户端完整请看原创。
客户端需要jar(纯java调用):
asm-3.3.jar
commons-logging.jar
cxf-2.7.18.jar
cxf-api-2.7.18.jar
cxf-rt-frontend-jaxws-2.7.8.jar
cxf-rt-ws-security-2.7.8.jar
javapns-jdk16-2.2.1.jar
neethi-3.0.3.jar
stax2-api-3.1.4.jar
woodstox-core-asl-4.4.1.jar
wsdl4j-1.6.3.jar
wss4j-1.6.19.jar
xmlschema-core-2.1.0.jar
xmlsec-1.5.8.jar
---------------------
maven依赖:
asm
asm
3.3
org.apache.cxf
cxf-api
2.7.18
org.apache.cxf
cxf
2.7.18
pom
org.apache.cxf
cxf-rt-frontend-jaxws
2.7.8
org.apache.cxf
cxf-rt-ws-security
2.7.8
com.github.fernandospr
javapns-jdk16
2.2.1
org.apache.neethi
neethi
3.0.3
org.codehaus.woodstox
stax2-api
3.1.4
org.codehaus.woodstox
woodstox-core-asl
4.4.1
wsdl4j
wsdl4j
1.6.3
org.apache.ws.security
wss4j
1.6.19
org.apache.ws.xmlschema
xmlschema-core
2.1.0
org.apache.santuario
xmlsec
1.5.8
jaxen
jaxen
1.1-beta-6
纯java调用webService包含两步:
一、创建验证辅助类,如本实列的PasswordHandler,用于向客户端传递密码。
二、创建调用类,如本实列的WsClientUtil,进行调用。
说明:
同服务端一样,PasswordHandler也是实现了CallbackHandler接口,与服务器端不同的是客户端该类的用途只是为了设置密码,不许要做任何验证。详情见代码。
一、方法updateMobileInfo和getMobileInfo是使用cxf+ws-security做的webService安全请求验证。
二、方法getDirectoryEntryByAccount和resetAccountPassword是使用HttpURLConnection做的webService请求。
package com.hxzq.s0026.my168.util.webservice;
import java.io.BufferedReader;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.net.HttpURLConnection;
import java.net.URL;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import org.apache.cxf.endpoint.Client;
import org.apache.cxf.jaxws.endpoint.dynamic.JaxWsDynamicClientFactory;
import org.apache.cxf.ws.security.wss4j.WSS4JOutInterceptor;
import org.apache.ws.security.WSConstants;
import org.apache.ws.security.handler.WSHandlerConstants;
import org.dom4j.Attribute;
import org.dom4j.DocumentException;
import org.dom4j.DocumentHelper;
import org.dom4j.Element;
import com.hxzq.s0026.my168.util.LogUtil;
import com.thinkive.base.config.Configuration;
import com.thinkive.base.exception.CommonException;
import com.thinkive.base.util.StringHelper;
/**
*
* @说明:调用第三方接口,采用CXF +ws-security 实现webservice客户端调用
* @类型名称:WsClientUtil
* @创建者: 敬进
* @创建时间: 2018年10月30日 上午10:11:26
* @修改者: 敬进
* @修改时间: 2018年10月30日 上午10:11:26
*/
public class WsClientUtil {
private static JaxWsDynamicClientFactory dcf = null;
private static WSS4JOutInterceptor wssOut = null;
private static String url_90 = null;
private static String url_91 = null;
private static String namespace = null;
static {
// 获取配置文件信息
url_90 = Configuration.getString("ws0002.url_90");
url_91 = Configuration.getString("ws0001.url_91");
namespace = Configuration.getString("ws0002.namespace");
String account = Configuration.getString("ws0001.account");
// 这个是用cxf 客户端访问cxf部署的webservice服务
// 千万记住,访问cxf的webservice必须加上namespace ,否则通不过
dcf = JaxWsDynamicClientFactory.newInstance();
Map props = new HashMap();
props.put(WSHandlerConstants.ACTION, WSHandlerConstants.USERNAME_TOKEN);
// 密码类型 明文:PasswordText密文:PasswordDigest
props.put(WSHandlerConstants.PASSWORD_TYPE, WSConstants.PW_TEXT);
// 用户名
props.put(WSHandlerConstants.USER, account);
// 将PasswordHandler 的类名传递给服务器,相当于传递了密码给服务器
props.put(WSHandlerConstants.PW_CALLBACK_CLASS, PasswordHandler.class.getName());
props.put(WSHandlerConstants.MUST_UNDERSTAND, "0");
wssOut = new WSS4JOutInterceptor(props);
}
/**
*
* @说明: updateMobileInfo修改用户手机号方法
* @方法名称: updateMobileInfo
* @参数 @param loginName 用户名
* @参数 @param phone 电话号码
* @参数 @return
* @返回类型 String
* @创建者: 敬进
* @创建时间: 2018年10月30日 下午2:07:44
* @修改者: 敬进
* @修改时间: 2018年10月30日 下午2:07:44
*/
public static String updateMobileInfo(String loginName, String phone) {
try {
// 创建一个客户端
LogUtil.info(url_91);
Client client = dcf.createClient(url_91);
// 添加访问参数
LogUtil.info("------开始------");
client.getOutInterceptors().add(wssOut);
LogUtil.info("------添加访问参数------");
// 执行方法
Object[] objects = client.invoke("updateMobileInfo", loginName, phone);
LogUtil.info("------执行方法------");
return (String) objects[0];
} catch (Exception e) {
// TODO: handle exception
LogUtil.error(e.getMessage());
throw new CommonException(1, "修改用户手机号失败!");
}
}
/**
*
* @说明:getMobileInfo获取用户联系信息方法
* @方法名称: getMobileInfo
* @参数 @param loginName 用户名
* @参数 @return
* @返回类型 String
* @创建者: 敬进
* @创建时间: 2018年10月30日 下午2:10:34
* @修改者: 敬进
* @修改时间: 2018年10月30日 下午2:10:34
*/
public static String getMobileInfo(String loginName) {
try {
Client client = dcf.createClient(url_91);
client.getOutInterceptors().add(wssOut);
Object[] objects = client.invoke("getMobileInfo", loginName);
return (String) objects[0];
} catch (Exception e) {
// TODO: handle exception
LogUtil.error(e.getMessage());
throw new CommonException(1, "获取用户联系信息失败!");
}
}
/**
* 、
* @说明: 获取密码过期时间方法
* @方法名称: getDirectoryEntryByAccount
* @参数 @param loginName 登录名
* @参数 @param password 密码
* @参数 @return
* @返回类型 Map key:Result、OperationMessage、BizData
* @创建者: 敬进
* @创建时间: 2018年11月1日 上午10:29:09
* @修改者: 敬进
* @修改时间: 2018年11月1日 上午10:29:09
*/
public static Map getDirectoryEntryByAccount(String loginName, String password) {
StringBuilder sb = new StringBuilder("");
sb.append(
"")
.append(" ").append("")
.append("")
.append("").append(loginName).append(" ")
.append("").append(password).append(" ")
.append(" ")
.append(" ").append(" ");
String dataXml = sb.toString();
String soapAction = namespace + "GetDirectoryEntryByAccount";
String resultXml = httpConnUtil(dataXml, soapAction);
// xml解析
Map resultMap=readStringXml(resultXml);
return resultMap;
}
/**
*
* @说明: 修改用户密码方法
* @方法名称: resetAccountPassword
* @参数 @param loginName 登录名
* @参数 @param oldPassword 旧密码
* @参数 @param newPassword 新密码
* @参数 @return
* @返回类型 Map key:Result、OperationMessage、BizData
* @创建者: 敬进
* @创建时间: 2018年10月30日 下午2:14:38
* @修改者: 敬进
* @修改时间: 2018年10月30日 下午2:14:38
*/
public static Map resetAccountPassword(String loginName, String oldPassword, String newPassword) {
StringBuilder sb = new StringBuilder("");
sb.append(
"")
.append(" ").append("")
.append("")
.append("").append(loginName).append(" ")
.append("").append(oldPassword).append(" ")
.append("").append(newPassword).append(" ")
.append(" ")
.append(" ").append(" ");
String dataXml = sb.toString();
String soapAction = namespace + "ResetAccountPassword";
String resultXml = httpConnUtil(dataXml, soapAction);
// xml解析
Map resultMap=readStringXml(resultXml);
return resultMap;
}
/**
*
* @说明: http请求处理webservice
* @方法名称: httpConnUtil
* @参数 @param dataXml xml入参
* @参数 @param soapAction 命名空间加方法名
* @参数 @return
* @返回类型 map
* @创建者: 敬进
* @创建时间: 2018年10月30日 下午2:12:24
* @修改者: 敬进
* @修改时间: 2018年10月30日 下午2:12:24
*/
private static String httpConnUtil(String dataXml, String soapAction) {
String contentType = "text/xml; charset=utf-8";
HttpURLConnection httpConn = null;
OutputStream out = null;
try {
httpConn = (HttpURLConnection) new URL(url_90).openConnection();
httpConn.setRequestProperty("Content-Type", contentType);
if (null != soapAction) {
httpConn.setRequestProperty("SOAPAction", soapAction);
}
httpConn.setRequestMethod("POST");
httpConn.setDoOutput(true);
httpConn.setDoInput(true);
httpConn.connect();
out = httpConn.getOutputStream();// 获取输出流对象
httpConn.getOutputStream().write(dataXml.getBytes("UTF-8"));// 将要提交服务器的SOAP请求字符流写入输出流
out.flush();
out.close();
int code = httpConn.getResponseCode();// 用来获取服务器相应状态
String tempString = null;
StringBuffer sb = new StringBuffer();
if (code == HttpURLConnection.HTTP_OK) {
BufferedReader reader = new BufferedReader(new InputStreamReader(httpConn.getInputStream(), "UTF-8"));
while ((tempString = reader.readLine()) != null) {
sb.append(tempString);
}
if (null != reader) {
reader.close();
}
} else {
BufferedReader reader = new BufferedReader(new InputStreamReader(httpConn.getErrorStream(), "UTF-8"));
// 一次读入一行,直到读入null为文件结束
while ((tempString = reader.readLine()) != null) {
sb.append(tempString);
}
if (null != reader) {
reader.close();
}
}
return sb.toString();
} catch (Exception e) {
// TODO: handle exception
LogUtil.error(e.getMessage());
throw new CommonException(1, "员工密码修改失败!");
}
}
/**
*
* @说明: dom4j解析xml
* @方法名称: readStringXml
* @参数 @param xml
* @参数 @return
* @返回类型 Map
* @创建者: 敬进
* @创建时间: 2018年11月1日 上午8:43:25
* @修改者: 敬进
* @修改时间: 2018年11月1日 上午8:43:25
*/
@SuppressWarnings("unchecked")
private static Map readStringXml(String xml) {
org.dom4j.Document doc = null;
Map resultMap=null;
try {
doc = DocumentHelper.parseText(xml);
Element rootElt = doc.getRootElement();
//获取根节点下body子节点
List elements = rootElt.selectNodes("soap:Body");
if (elements.size()!=0) {
for (Element element : elements) {
Element parent=element.getParent();
resultMap=new HashMap();
arrayNodes(resultMap, element, rootElt);
//如果存在多个item数据才调用这个
List attributes = parent.attributes();
if (attributes.size()>0) {
for (Attribute attribute : attributes) {
//将属性名(key)和属性值(value)添加到map对象中去
resultMap.put(attribute.getName(), attribute.getValue());
}
}else{
LogUtil.info("该节点没有任何属性节点!");
}
}
}else{
LogUtil.info("xPath(DOM树路径)出现错误!");
}
} catch (DocumentException e) {
// TODO Auto-generated catch block
LogUtil.info(e.getMessage());
throw new CommonException(1,"DOM解析失败!");
}
return resultMap;
}
//一般类型的解析
private static void arrayNodes(Map nodeMap, Element node, Element root) {
LogUtil.info("----------------------------");
// 当前节点的名称、文本内容和属性
LogUtil.info("当前节点名称:" + node.getName());// 当前节点名称
if (!(node.getTextTrim().equals(""))) {
LogUtil.info("当前节点的内容:" + node.getTextTrim());// 当前节点名称
}
String nodeName = node.getName();
String nodeValue = node.getTextTrim();
if (StringHelper.isNotEmpty(nodeValue)) {
nodeMap.put(nodeName, nodeValue);
}
// 当前节点下面子节点迭代器
@SuppressWarnings("unchecked")
Iterator it = node.elementIterator();
// 遍历
while (it.hasNext()) {
// 获取某个子节点对象
Element e = it.next();
// 对子节点进行遍历
arrayNodes(nodeMap, e, root);
}
}
}
在做cxf+ws-security时需要传输一个password
package com.hxzq.s0026.my168.util.webservice;
import java.io.IOException;
import javax.security.auth.callback.Callback;
import javax.security.auth.callback.CallbackHandler;
import javax.security.auth.callback.UnsupportedCallbackException;
import org.apache.ws.security.WSPasswordCallback;
import com.thinkive.base.config.Configuration;
/**
*
* @说明:webService验证辅助类,用于向ws-security发送密码
* @类型名称:PasswordHandler
* @创建者: 敬进
* @创建时间: 2018年10月30日 下午1:11:21
* @修改者: 敬进
* @修改时间: 2018年10月30日 下午1:11:21
*/
public class PasswordHandler implements CallbackHandler {
@Override
public void handle(Callback[] callbacks) throws IOException, UnsupportedCallbackException {
// TODO Auto-generated method stub
String password=Configuration.getString("ws0001.password");
for (int i = 0; i < callbacks.length; i++) {
WSPasswordCallback pc=(WSPasswordCallback) callbacks[i];
pc.setPassword(password);
}
}
}
其中namespace是wsdl中的命名空间,url_90和url_91是webService接口。
使用HttpURLConnection做webService客户端时,需要拼接发送请求的参数,然后使用dom4j解析返回的参数,如
调用getDirectoryEntryByAccount方法,拼接参数类似。
使用方法readStringXml解析返回的参数。
cxf+ws-secur原创ity
webService调用方法还有axis和axis2
一般遇到的错误