/* Copyright 2009, 2012 predic8 GmbH, www.predic8.com Licensed 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 com.predic8.membrane.core.interceptor; import static com.predic8.membrane.core.Constants.WSDL_HTTP_NS; import static com.predic8.membrane.core.Constants.WSDL_SOAP11_NS; import static com.predic8.membrane.core.Constants.WSDL_SOAP12_NS; import static com.predic8.membrane.core.Constants.XSD_NS; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.InputStreamReader; import java.io.OutputStreamWriter; import java.io.StringWriter; import java.io.UnsupportedEncodingException; import java.net.MalformedURLException; import java.net.URL; import java.net.URLDecoder; import javax.xml.namespace.QName; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import com.googlecode.jatl.Html; import com.predic8.membrane.annot.MCAttribute; import com.predic8.membrane.annot.MCElement; import com.predic8.membrane.core.Router; import com.predic8.membrane.core.exchange.Exchange; import com.predic8.membrane.core.http.Request; import com.predic8.membrane.core.http.Response; import com.predic8.membrane.core.transport.http.HttpClient; import com.predic8.membrane.core.util.MessageUtil; import com.predic8.membrane.core.ws.relocator.Relocator; /** * @description * <p>The <i>wsdlRewriter</i> rewrites endpoint addresses of services and XML Schema locations in WSDL documents.</p> * @topic 8. SOAP based Web Services */ @MCElement(name="wsdlRewriter") public class WSDLInterceptor extends RelocatingInterceptor { private static Logger log = LoggerFactory.getLogger(WSDLInterceptor.class.getName()); private String registryWSDLRegisterURL; private boolean rewriteEndpoint = true; private HttpClient hc; public WSDLInterceptor() { name = "WSDL Rewriting Interceptor"; setFlow(Flow.Set.RESPONSE); } @Override public void init(Router router) throws Exception { super.init(router); hc = router.getResolverMap().getHTTPSchemaResolver().getHttpClient(); } @Override protected void rewrite(Exchange exc) throws Exception, IOException { log.debug("Changing endpoint address in WSDL"); ByteArrayOutputStream stream = new ByteArrayOutputStream(); Relocator relocator = new Relocator(new OutputStreamWriter(stream, exc.getResponse().getCharset()), getLocationProtocol(), getLocationHost(exc), getLocationPort(exc), pathRewriter); if (rewriteEndpoint) { relocator.getRelocatingAttributes().put( new QName(WSDL_SOAP11_NS, "address"), "location"); relocator.getRelocatingAttributes().put( new QName(WSDL_SOAP12_NS, "address"), "location"); relocator.getRelocatingAttributes().put( new QName(WSDL_HTTP_NS, "address"), "location"); } relocator.getRelocatingAttributes().put(new QName(XSD_NS, "import"), "schemaLocation"); relocator.getRelocatingAttributes().put(new QName(XSD_NS, "include"), "schemaLocation"); relocator.relocate(new InputStreamReader(exc.getResponse().getBodyAsStreamDecoded(), exc.getResponse().getCharset())); if (relocator.isWsdlFound()) { registerWSDL(exc); } exc.getResponse().setBodyContent(stream.toByteArray()); } private void registerWSDL(Exchange exc) { if (registryWSDLRegisterURL == null) return; StringBuilder buf = new StringBuilder(); buf.append(registryWSDLRegisterURL); buf.append("?wsdl="); try { buf.append(URLDecoder.decode(getWSDLURL(exc), "US-ASCII")); } catch (UnsupportedEncodingException e) { // ignored } callRegistry(buf.toString()); log.debug(buf.toString()); } private void callRegistry(String uri) { try { Response res = hc.call(createExchange(uri)).getResponse(); if (res.getStatusCode() != 200) log.warn("{}",res); } catch (Exception e) { log.error("", e); } } private Exchange createExchange(String uri) throws MalformedURLException { URL url = new URL(uri); Request req = MessageUtil.getGetRequest(getCompletePath(url)); req.getHeader().setHost(url.getHost()); Exchange exc = new Exchange(null); exc.setRequest(req); exc.getDestinations().add(uri); return exc; } private String getCompletePath(URL url) { if (url.getQuery() == null) return url.getPath(); return url.getPath() + "?" + url.getQuery(); } private String getWSDLURL(Exchange exc) { StringBuilder buf = new StringBuilder(); buf.append(getLocationProtocol()); buf.append("://"); buf.append(getLocationHost(exc)); if (getLocationPort(exc) != 80) { buf.append(":"); buf.append(getLocationPort(exc)); } buf.append("/"); buf.append(exc.getRequest().getUri()); return buf.toString(); } @MCAttribute public void setRegistryWSDLRegisterURL(String registryWSDLRegisterURL) { this.registryWSDLRegisterURL = registryWSDLRegisterURL; } public String getRegistryWSDLRegisterURL() { return registryWSDLRegisterURL; } @Override public String getShortDescription() { return "Rewrites SOAP endpoint addresses and XML Schema locations in WSDL and XSD documents."; } @Override public String getLongDescription() { StringBuilder sb = new StringBuilder(); sb.append(getShortDescription()); sb.append("<br/>"); sb.append("The protocol, host and port of the incoming request will be used for the substitution"); if (protocol != null || port != null || host != null) { sb.append(" except the following fixed values:"); StringWriter sw = new StringWriter(); new Html(sw){{ table(); thead(); tr(); th().text("Part").end(); th().text("Value").end(); end(); end(); tbody(); if (protocol != null) { tr(); td().text("Protocol").end(); td().text(protocol).end(); end(); } if (host != null) { tr(); td().text("Host").end(); td().text(host).end(); end(); } if (port != null) { tr(); td().text("Port").end(); td().text(port).end(); end(); } end(); end(); }}; sb.append(sw.toString()); } else { sb.append("."); } return sb.toString(); } public void setRewriteEndpoint(boolean rewriteEndpoint) { this.rewriteEndpoint = rewriteEndpoint; } /** * @description The protocol the endpoint should be changed to. * @default Don't change the endpoint's protocol. * @example http */ @MCAttribute @Override public void setProtocol(String protocol) { super.setProtocol(protocol); } /** * @description The host the endpoint should be changed to. * @default Don't change the endpoint's host. * @example localhost */ @MCAttribute @Override public void setHost(String host) { super.setHost(host); } /** * @description The port the endpoint should be changed to. * @default Don't change the endpoint's port. * @example 4000 */ @MCAttribute @Override public void setPort(String port) { super.setPort(port); } }