com.predic8.membrane.core.interceptor.WSDLInterceptor.java Source code

Java tutorial

Introduction

Here is the source code for com.predic8.membrane.core.interceptor.WSDLInterceptor.java

Source

/* 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.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

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 Log log = LogFactory.getLog(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);
    }
}