CXF:为CXF SOAP/REST Web服务的WSDL和WADL响应添加隐式标头

rap*_*apt 11 java rest soap web-services cxf

我正在尝试为CXF SOAP/REST Web服务(由Camel管理)的WSDL和WADL响应添加隐式标头.

(这些不一定是安全标题....)

通过"隐式标头",我的意思是,点击服务的WSDL/WADL URL将显示客户端期望在请求中提供标头.

但我不想在Web服务的签名中明确指定标头.

我有一个CXF拦截器,它为每个SOAP/REST响应添加一个隐式头.

因此,由于WSDL/WADL文档是作为对某些GET请求的响应而发送的,所以我想以某种方式使用类似的拦截器将头数据添加到WSDL/WADL响应中.我怎么能进行这样一个奇妙的壮举?

这是CXF拦截器,它为每个SOAP/REST响应添加一个隐式头:

public class MyInterceptor extends AbstractPhaseInterceptor<Message> {

    public MyInterceptor()
    {
        super(Phase.RECEIVE);
    }

    @Override
    public void handleMessage(Message message)
    {   
        try
        {
            //soap
            if (message instanceof SoapMessage)
            {               
                List<Header> headers = ((SoapMessage)message).getHeaders();

                Header dummyHeader = new Header(new QName("uri:org.apache.cxf", "dummy"), "decapitated", new JAXBDataBinding(String.class));

                headers.add(dummyHeader);
            }
            //rest
            else
            {
                Map<String, List> headers = (Map<String, List>) message.get(Message.PROTOCOL_HEADERS);

                String dummyHeader = "decapitated";

                headers.put("dummy", Collections.singletonList(dummyHeader));
            }

        }
        catch (JAXBException e)
        {
            throw new Fault(e);
        }
    }

    @Override
    public void handleFault(Message messageParam)
    {
    }
}
Run Code Online (Sandbox Code Playgroud)

GPI*_*GPI 7

CXF 2.7.4

在CXF中,WSDL是通过一个名为the的链中的Interceptor生成的WSDLGetInterceptor,它位于READ链中.

它的基本设计是

  1. 检查呼叫是否是HTTP GET
  2. 准备输出消息以返回
  3. 访问wsdl(来自Java或来自静态资源)
  4. 将wsdl写入输出消息
  5. 中断拦截器链以提供输出消息

采取行动的最简单方法是抢占这个拦截器,使其无法通过之前注册您自己的实现来完成其工作.

删除标准CXF拦截器在默认总线上是一件"难以做到的事"(最简单的方法是注册你自己的拦截器,把它放在链中,并让它删除其他类似的拦截器 message.getInterceptorChain().remove(removeInterceptor);

但是在标准WSDL拦截器之前添加自己的东西很简单:

public MyWSDLGetInterceptor() {
    super(Phase.READ);
    addBefore(WSDLGetInterceptor.class.getName());
}
Run Code Online (Sandbox Code Playgroud)

MyWSDLGetInterceptor会扩展标准WSDLGetInterceptor和你只覆盖:

public Document getDocument(Message message,
                            String base,
                            Map<String, String> params,
                            String ctxUri,
                            EndpointInfo endpointInfo) {
    Document domDocument = super.getDocument(message, base, params, ctxUri, endpointInfo);
    domDocument.getChildNodes(); // Whatever you need to add remove
    return domDocument; // Once modified
}
Run Code Online (Sandbox Code Playgroud)

您可以动态修改生成的DOM Document(添加/创建DOM节点)或通过XSLT,无论您最喜欢什么,您都可以通过标准API处理标准XML.

CXF 2.7.x(其中x大于4且<10)

校长是相同的,但拦截器的工作方式不同.

  1. 它(在私有方法中)将WSDL作为DOM Document作为out消息属性
  2. 它清除所有拦截器的输出拦截器链(绝对必要除外)
  3. WSDLGetOutInterceptor在输出链中 注册一个
  4. 它会停止IN链并进入停止链
  5. WSDLGetOutInterceptor做它的序列化工作

所以它有点难/干净.但是使用相同的预占基础拦截器的原则(在之前注册自己),您可以cleanUpOutInteceptors通过访问WSDL来覆盖操纵消息,就像在2.7.4情况下一样.outMessage.get(DOCUMENT_HOLDER)

WADL

对不起,我没有专业知识,但我猜CXF对两者都有相同的架构......