org.apache.cxf.frontend.WSDLGetInterceptor Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of cxf-rt-frontend-simple Show documentation
Show all versions of cxf-rt-frontend-simple Show documentation
Apache CXF Runtime Simple Frontend
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.cxf.frontend;
import java.util.Iterator;
import java.util.Map;
import org.w3c.dom.Document;
import org.apache.cxf.binding.soap.interceptor.EndpointSelectionInterceptor;
import org.apache.cxf.common.util.StringUtils;
import org.apache.cxf.common.util.UrlUtils;
import org.apache.cxf.endpoint.Endpoint;
import org.apache.cxf.interceptor.Fault;
import org.apache.cxf.interceptor.Interceptor;
import org.apache.cxf.interceptor.MessageSenderInterceptor;
import org.apache.cxf.interceptor.OutgoingChainInterceptor;
import org.apache.cxf.interceptor.StaxOutInterceptor;
import org.apache.cxf.message.Message;
import org.apache.cxf.message.MessageImpl;
import org.apache.cxf.phase.AbstractPhaseInterceptor;
import org.apache.cxf.phase.Phase;
import org.apache.cxf.transport.common.gzip.GZIPOutInterceptor;
public class WSDLGetInterceptor extends AbstractPhaseInterceptor {
public static final WSDLGetInterceptor INSTANCE = new WSDLGetInterceptor();
public static final String DOCUMENT_HOLDER = WSDLGetInterceptor.class.getName() + ".documentHolder";
private static final String TRANSFORM_SKIP = "transform.skip";
private Interceptor wsdlGetOutInterceptor = WSDLGetOutInterceptor.INSTANCE;
public WSDLGetInterceptor() {
super(Phase.READ);
getAfter().add(EndpointSelectionInterceptor.class.getName());
}
public WSDLGetInterceptor(Interceptor outInterceptor) {
this();
// Let people override the wsdlGetOutInterceptor
wsdlGetOutInterceptor = outInterceptor;
}
public void handleMessage(Message message) throws Fault {
String method = (String)message.get(Message.HTTP_REQUEST_METHOD);
String query = (String)message.get(Message.QUERY_STRING);
if (!"GET".equals(method) || StringUtils.isEmpty(query)) {
return;
}
String baseUri = (String)message.get(Message.REQUEST_URL);
String ctx = (String)message.get(Message.PATH_INFO);
WSDLGetUtils utils = (WSDLGetUtils)message.getContextualProperty(WSDLGetUtils.class.getName());
if (utils == null) {
utils = new WSDLGetUtils();
message.put(WSDLGetUtils.class, utils);
}
Map map = UrlUtils.parseQueryString(query);
if (isRecognizedQuery(map)) {
Document doc = getDocument(utils, message, baseUri, map, ctx);
Endpoint e = message.getExchange().getEndpoint();
Message mout = new MessageImpl();
mout.setExchange(message.getExchange());
mout = e.getBinding().createMessage(mout);
mout.setInterceptorChain(OutgoingChainInterceptor.getOutInterceptorChain(message.getExchange()));
message.getExchange().setOutMessage(mout);
mout.put(DOCUMENT_HOLDER, doc);
mout.put(Message.CONTENT_TYPE, "text/xml");
// just remove the interceptor which should not be used
cleanUpOutInterceptors(mout);
// notice this is being added after the purge above, don't swap the order!
mout.getInterceptorChain().add(wsdlGetOutInterceptor);
message.getExchange().put(TRANSFORM_SKIP, Boolean.TRUE);
// skip the service executor and goto the end of the chain.
message.getInterceptorChain().doInterceptStartingAt(
message,
OutgoingChainInterceptor.class.getName());
}
}
protected void cleanUpOutInterceptors(Message outMessage) {
// TODO - how can I improve this to provide a specific interceptor chain that just has the
// stax, gzip and message sender components, while also ensuring that GZIP is only provided
// if its already configured for the endpoint.
Iterator> iterator = outMessage.getInterceptorChain().iterator();
while (iterator.hasNext()) {
Interceptor extends Message> inInterceptor = iterator.next();
if (!inInterceptor.getClass().equals(StaxOutInterceptor.class)
&& !inInterceptor.getClass().equals(GZIPOutInterceptor.class)
&& !inInterceptor.getClass().equals(MessageSenderInterceptor.class)) {
outMessage.getInterceptorChain().remove(inInterceptor);
}
}
}
private Document getDocument(WSDLGetUtils utils,
Message message, String base,
Map params, String ctxUri) {
// cannot have two wsdl's being generated for the same endpoint at the same
// time as the addresses may get mixed up
// For WSDL's the WSDLWriter does not share any state between documents.
// For XSD's, the WSDLGetUtils makes a copy of any XSD schema documents before updating
// any addresses and returning them, so for both WSDL and XSD this is the only part that needs
// to be synchronized.
synchronized (message.getExchange().getEndpoint()) {
return utils.getDocument(message, base, params, ctxUri,
message.getExchange().getEndpoint().getEndpointInfo());
}
}
private boolean isRecognizedQuery(Map map) {
return map.containsKey("wsdl") || map.containsKey("xsd");
}
}