CXF 中自定义SOAPHeader

Interceptor是CXF架构中一个很有特色的模式。你可以在不对核心模块进行修改的情况下,动态添加很多功能。这对于CXF这个以处理消息为中心的服务框架来说是非常有用的,CXF通过在Interceptor中对消息进行特殊处理,实现了很多重要功能模块,例如:日志记录,Soap消息处理,消息的压缩处。简单的说,可以在收到请求后,还未进行业务处理前,进行处理。或者在请求包发送前,进行报文的处理。

Interceptor

定义两个方法,一个处理消息 handleMessage, 一个是处理错误 handleFault。

InterceptorChain

单个的Interceptor功能有限,CXF要实现一个SOAP消息处理,需要将许许多多的Interceptor组合在一起使用。因此设计了 InterceptorChain,在我看了InterceptorChain就像是一个Interceptor的小队长。 小队长有调配安置Interceptor的权力(add,remove),也有控制消息处理的权力(doInterceptor,pause,resume,reset,abort),同时也有交付错误处理的权力( {get|set}FaultObserver)。更有意思的是为灵活控制Interceptor的处理消息顺序(doInterceptStartingAt,doInterceptorStartingAfter),这也是InterceptorChain比较难理解的地方。

Fault

定义了CXF中的错误消息。

InterceptorProvider

  这里定义了Interceptor的后备保障部队。我们可以在InterceptorProvider中设置In,Out,InFault,OutFault 后备小分队,添加我们所希望添加的Interceptor。而InterceptorChain会根据这些后备小分队,组建自己的小分队实例,完成具体的作战功能任务。

AbstractAttributedInterceptorProvider

InterceptorProvider实现的抽象类,由于这个类来继承了HashMap,我们可以像这个类中存储一些属性信息。

AbrstractBasicInterceptorProvider

与AbstractAttributedInterceptorProvider不同,这个Interceptor只是简单实现了InterceptorProvider的功能,并不提供对其属性存储的扩展。

Message

由于Interceptor是针对Message来进行处理的,当你打开Message这个类文件时,你会发现在Message中定义了很多常量,同时你还可以从Message中获取到很多与Message操作相关的信息。可以获取设置的对象有InterceptorChain Exchange Destination,还有获取设置Content的泛型接口,是不是感觉Message和Bus差不多,都成了大杂货铺,一切与消息处理相关的信息都可以放在Message中。

服务端拦截器

package hs.cxf.soapHeader;

import javax.xml.soap.SOAPException;

import javax.xml.soap.SOAPHeader;

import javax.xml.soap.SOAPMessage;

import org.apache.cxf.binding.soap.SoapMessage;

import org.apache.cxf.binding.soap.saaj.SAAJInInterceptor;

import org.apache.cxf.interceptor.Fault;

import org.apache.cxf.phase.AbstractPhaseInterceptor;

import org.apache.cxf.phase.Phase;

import org.w3c.dom.NodeList;

/**

 * 

 * @Title:获取soap头信息

 * 

 * @Description:

 * 

 * @Copyright:

 * 

 * @author zz

 * @version 1.00.000

 * 

 */

public class ReadSoapHeader extends AbstractPhaseInterceptor<SoapMessage> {

    private SAAJInInterceptor saa = new SAAJInInterceptor();

    public ReadSoapHeader() {

        super(Phase.PRE_PROTOCOL);

        getAfter().add(SAAJInInterceptor.class.getName());

    }

    public void handleMessage(SoapMessage message) throws Fault {

        SOAPMessage mess = message.getContent(SOAPMessage.class);

        if (mess == null) {

            saa.handleMessage(message);

            mess = message.getContent(SOAPMessage.class);

        }

        SOAPHeader head = null;

        try {

            head = mess.getSOAPHeader();

        } catch (SOAPException e) {

            e.printStackTrace();

        }

        if (head == null) {

            return;

        }

        try {

            //读取自定义的节点

            NodeList nodes = head.getElementsByTagName("tns:spId");

            NodeList nodepass = head.getElementsByTagName("tns:spPassword");

            //获取节点值,简单认证

            if (nodes.item(0).getTextContent().equals("wdw")) {

                if (nodepass.item(0).getTextContent().equals("wdwsb")) {

                    System.out.println("认证成功");

                }

            } else {

                SOAPException soapExc = new SOAPException("认证错误");

                throw new Fault(soapExc);

            }

        } catch (Exception e) {

            SOAPException soapExc = new SOAPException("认证错误");

            throw new Fault(soapExc);

        }

    }

}

配置文件中新增拦截器配置

<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">   

    <import resource="classpath:META-INF/cxf/cxf.xml" />   

    <import resource="classpath:META-INF/cxf/cxf-extension-soap.xml" />   

    <import resource="classpath:META-INF/cxf/cxf-servlet.xml" />   

    <bean id="jaxWsServiceFactoryBean"  

        class="org.apache.cxf.jaxws.support.JaxWsServiceFactoryBean">   

        <property name="wrapped" value="true" />   

        <property name="dataBinding" ref="aegisBean" />   

    </bean>   

    <bean id="aegisBean"  

        class="org.apache.cxf.aegis.databinding.AegisDatabinding" />   

    <jaxws:endpoint id="CollectiveServices"  

        implementor="hs.cxf.server.WebServiceSampleImpl" address="/HelloWorld">   

        <jaxws:inInterceptors>   

          <!-- 日志拦截器 -->      

          <bean class="org.apache.cxf.interceptor.LoggingOutInterceptor"/>   

          <!-- 自定义拦截器 --> 

          <bean class="hs.cxf.soapHeader.ReadSoapHeader"/>   

          </jaxws:inInterceptors>    

        <jaxws:serviceFactory>   

            <ref bean="jaxWsServiceFactoryBean"/>   

        </jaxws:serviceFactory>   

    </jaxws:endpoint>   

</beans>

客户端拦截器:

package hs.cxf.client.SoapHeader;

import java.util.List;

import javax.xml.namespace.QName;

import org.apache.cxf.binding.soap.SoapHeader;

import org.apache.cxf.binding.soap.SoapMessage;

import org.apache.cxf.binding.soap.interceptor.AbstractSoapInterceptor;

import org.apache.cxf.headers.Header;

import org.apache.cxf.helpers.DOMUtils;

import org.apache.cxf.interceptor.Fault;

import org.apache.cxf.phase.Phase;

import org.w3c.dom.Document;

import org.w3c.dom.Element;

/**

 * 

 * @Title:在发送消息前,封装Soap Header 信息

 * 

 * @Description:

 * 

 * @Copyright: 

 *

 * @author zz

 * @version 1.00.000

 *

 */

public class AddSoapHeader extends AbstractSoapInterceptor {

      private static String nameURI="http://127.0.0.1:8080/cxfTest/ws/HelloWorld";   

        public AddSoapHeader(){   

            super(Phase.WRITE);   

        }   

        public void handleMessage(SoapMessage message) throws Fault {   

            String spPassword="wdwsb";   

            String spName="wdw";   

            QName qname=new QName("RequestSOAPHeader");   

            Document doc=DOMUtils.createDocument();   

            //自定义节点

            Element spId=doc.createElement("tns:spId");   

            spId.setTextContent(spName);   

            //自定义节点

            Element spPass=doc.createElement("tns:spPassword");   

            spPass.setTextContent(spPassword);   

            Element root=doc.createElementNS(nameURI, "tns:RequestSOAPHeader");   

            root.appendChild(spId);   

            root.appendChild(spPass);   

            SoapHeader head=new SoapHeader(qname,root);   

            List<Header> headers=message.getHeaders();   

            headers.add(head);   

            System.out.println(">>>>>添加header<<<<<<<");

        }   

}

客户端调用时:

package hs.cxf.client;

import hs.cxf.client.SoapHeader.AddSoapHeader;

import java.util.ArrayList;

import javax.xml.bind.JAXBElement;

import javax.xml.namespace.QName;

import org.apache.cxf.endpoint.Client;

import org.apache.cxf.frontend.ClientProxy;

import org.apache.cxf.interceptor.Interceptor;

import org.apache.cxf.jaxws.JaxWsProxyFactoryBean;

import org.apache.cxf.transport.http.HTTPConduit;

import org.apache.cxf.transports.http.configuration.HTTPClientPolicy;

/**

 * @Title:

 * 

 * @Description:

 * 

 * @Copyright: 

 * 

 * @author zz

 * @version 1.00.000

 * 

 */

public class TestClient {

    /**

     * 测试1

     */

    @SuppressWarnings("unchecked")

    public void testSend1() {

        try {

            JaxWsProxyFactoryBean factory = new JaxWsProxyFactoryBean();

            ArrayList<Interceptor> list = new ArrayList<Interceptor>();

            // 添加soap header 

            list.add(new AddSoapHeader());

            // 添加soap消息日志打印

            list.add(new org.apache.cxf.interceptor.LoggingOutInterceptor());

            factory.setOutInterceptors(list);

            factory.setServiceClass(WebServiceSample.class);

            factory.setAddress("http://127.0.0.1:8080/cxfTest/ws/HelloWorld");

            Object obj = factory.create();

            System.out.println(obj == null ? "NULL" : obj.getClass().getName());

            if (obj != null) {

                WebServiceSample ws = (WebServiceSample) obj;

                String str = ws.say("test");

                System.out.println(str);

                str = ws.say("1111");

                System.out.println(str);

                User u = new User();

                JAXBElement<String> je = new JAXBElement<String>(new QName(

                        "http://bean.cxf.hs", "name"), String.class, "张三");

                u.setName(je);

                str = ws.sayUserName(u);

                System.out.println(str);

                // 通过对象来交互

                ReqBean req = new ReqBean();

                req.setExp(new JAXBElement<String>(new QName(

                        "http://bean.cxf.hs", "exp"), String.class,

                        "<exp>111<exp>"));

                req.setSeqId(new JAXBElement<String>(new QName(

                        "http://bean.cxf.hs", "seqId"), String.class,

                        "12345678"));

                RespBean resp = ws.action(req);

                System.out.println("resp_id:" + resp.getRespId().getValue());

                System.out.println("resp_exp:" + resp.getExp().getValue());

            }

        } catch (Exception ex) {

            ex.printStackTrace();

        }

    }

    /**

     * 测试2

     */

    @SuppressWarnings("unchecked")

    public void testSend2() {

        String webServiceUrl = "http://127.0.0.1:8080/cxfTest/ws/HelloWorld";

        String webServiceConTimeout = "60000";

        String webServiceRevTimeout = "60000";

        JaxWsProxyFactoryBean factory = new JaxWsProxyFactoryBean();

        ArrayList<Interceptor> list = new ArrayList<Interceptor>();

        // 添加soap header 信息

        list.add(new AddSoapHeader());

        // 添加soap消息日志打印

        list.add(new org.apache.cxf.interceptor.LoggingOutInterceptor());

        factory.setOutInterceptors(list);

        factory.setServiceClass(WebServiceSample.class);

        factory.setAddress(webServiceUrl);

        WebServiceSample service = (WebServiceSample) factory.create();

        //超时时间设置

        Client clientP = ClientProxy.getClient(service);

        HTTPConduit http = (HTTPConduit) clientP.getConduit();

        HTTPClientPolicy httpClientPolicy = new HTTPClientPolicy();

        httpClientPolicy.setConnectionTimeout(Integer

                .valueOf(webServiceConTimeout));

        httpClientPolicy.setReceiveTimeout(Integer

                .valueOf(webServiceRevTimeout));

        httpClientPolicy.setAllowChunking(false);

        http.setClient(httpClientPolicy);

        // 通过对象来交互

        ReqBean req = new ReqBean();

        req.setExp(new JAXBElement<String>(new QName(

                "http://bean.cxf.hs", "exp"), String.class,

                "<exp>111<exp>"));

        req.setSeqId(new JAXBElement<String>(new QName(

                "http://bean.cxf.hs", "seqId"), String.class,

                "12345678"));

        System.out.println(">>>>>>发送消息<<<<<<<<<");

        RespBean resp = service.action(req);

        System.out.println("resp_id:" + resp.getRespId().getValue());

        System.out.println("resp_exp:" + resp.getExp().getValue());

    }

    /**

     * @param args

     */

    public static void main(String[] args) {

        TestClient tc = new TestClient();

        tc.testSend1();

        System.out.println(">>>>>>>>>>>>2<<<<<<<<<<<<<");

        tc.testSend2();

        System.out.println(">>>>>>>>>>>>END<<<<<<<<<<<<<");

    }

}
时间: 2024-11-05 12:26:07

CXF 中自定义SOAPHeader的相关文章

vue中自定义指令

在vue中自定义标签,首先要调用vue中一个directive的方法,具体方法:Vue.direction('指令名称',function(){ }); 例如我们要写一个关于颜色的指令,叫v-colorred: 1 Vue.directive('colorred',function(){ 2 3 this.el.style.color='red'; 4 }); 在html中,我直接用v-colorred指令就可以了,例如: 1 <p v-colorred>1234567890</p>

在Tableau中自定义版块地图 (Polygon)

在Tableau的地图报表中有一个‘Filed Map’的类型,可以根据版块来显示数据. 但实际应用中Tableau固有的版块划分可能不是我们想要的,下面介绍如何自定义版块并且用作数据分析. 自定义版块 在Tableau中自定义版块是非常容易的.如下图我们把每个点链接起来就是一个多边形的版块 上面的经纬度就不用说了.State 是我们版块的名字.其中Point Order告诉Tableau链接的顺序. Polygon ID 用来指定各个闭合的区域.如下图,State 都叫Michigan,由两块

linux中自定义回收站

myrm(){ D=/tmp/$(date +%Y%m%d%H%M%S); mkdir -p $D; mv "[email protected]" $D && echo "moved to $D ok"; } [[email protected] test]# myrm(){ D=/tmp/$(date +%Y%m%d%H%M%S); mkdir -p $D;  mv "[email protected]" $D &&am

PB中自定义事件ID含义

PB中自定义事件ID含义 单选或多选按钮消息(前缀:pbm_bm) pbm_bmgetcheck 单选按钮或多选按钮是否被选. pbm_bmgetstate 按钮是否加亮. pbm_bmsetcheck 将无线按钮或确认框的选中状态改为未选中状态,反之亦然. pbm_bmsetstate 加亮或不加亮按钮. pbm_bmchange 改变按钮的风格,例如,改为单选按钮或组合框. 单选或多选按钮通知消息(前缀:pbm_bn) pbm_bnclicked 按钮控件被点中. pbm_bndisable

对NSArray中自定义的对象进行排序

本文译自How to sort NSArray with custom objects. 我们开发的每个程序都会使用到一些数据,而这些数据一般被封装在一个自定义的类中.例如一个音乐程序可能会有一个Song类,聊天程序则又一个Friend类,点菜程序会有一个Recipe类等.有时候我们希望在程序中显示的列表数据是按照一定顺序进行排列的,本文我们就来看看在iOS中有哪些方法可以对NSArray中的对象进行排序.下面是目录: 小引 使用NSComparator进行排序 使用NSDescriptor进行

iOS开发中自定义字体的方法

http://www.cnblogs.com/iyou/archive/2014/05/25/3751669.html 1. 首先下载你想要设置的字体库,例如设置方正启体简体 2. 添加到工程,一定要注意勾选红色框框处,默认是不勾选的  添加以后 3.在plist文件中添加 4.现在已经添加成功了,但是要使用就必须知道FontName,用以下代码可查到 NSArray *familyNames = [[NSArray alloc] initWithArray:[UIFont familyName

在.net桌面程序中自定义鼠标光标

有的时候,一个自定义的鼠标光标能给你的程序增色不少.本文这里介绍一下如何在.net桌面程序中自定义鼠标光标.由于.net的桌面程序分为WinForm和WPF两种,这里分别介绍一下. WinForm程序 对于WinForm程序,可以通过修改Control.Cursor属性来实现光标的修改,如果我们有光标文件的话,可以直接通过如下代码实现自定义光标: this.Cursor = new Cursor("myCursor.cur"); 但这种方式不是本文介绍的重点,本文主要介绍如何自己绘制光

Android中自定义下拉样式Spinner

Android中自定义下拉样式Spinner 本文继续介绍android自定义控件系列,自定义Spinner控件的使用. 实现思路 1.定义下拉控件布局(ListView及子控件布局) 2.自定义SpinerPopWindow类 3.定义填充数据的Adapter 效果图 一.定义控件布局 <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http:/

Android中自定义View的MeasureSpec使用

有时,Android系统控件无法满足我们的需求,因此有必要自定义View.具体方法参见官方开发文档:http://developer.android.com/guide/topics/ui/custom-components.html 一般来说,自定义控件都会去重写View的onMeasure方法,因为该方法指定该控件在屏幕上的大小. protected void onMeasure (int widthMeasureSpec, int heightMeasureSpec) onMeasure传