All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.cxf.transport.http.HTTPConduit Maven / Gradle / Ivy

There is a newer version: 2.7.18
Show newest version
/**
 * 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.transport.http;


import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PushbackInputStream;
import java.net.HttpURLConnection;
import java.net.InetSocketAddress;
import java.net.MalformedURLException;
import java.net.Proxy;
import java.net.URL;
import java.net.URLConnection;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.logging.Level;
import java.util.logging.Logger;

import javax.xml.namespace.QName;

import org.apache.cxf.Bus;
import org.apache.cxf.common.logging.LogUtils;
import org.apache.cxf.common.util.Base64Utility;
import org.apache.cxf.common.util.StringUtils;
import org.apache.cxf.configuration.Configurable;
import org.apache.cxf.configuration.jsse.TLSClientParameters;
import org.apache.cxf.configuration.security.AuthorizationPolicy;
import org.apache.cxf.configuration.security.ProxyAuthorizationPolicy;
import org.apache.cxf.helpers.CastUtils;
import org.apache.cxf.helpers.HttpHeaderHelper;
import org.apache.cxf.helpers.IOUtils;
import org.apache.cxf.helpers.LoadingByteArrayOutputStream;
import org.apache.cxf.io.AbstractThresholdOutputStream;
import org.apache.cxf.io.CacheAndWriteOutputStream;
import org.apache.cxf.io.CachedOutputStream;
import org.apache.cxf.message.Exchange;
import org.apache.cxf.message.ExchangeImpl;
import org.apache.cxf.message.Message;
import org.apache.cxf.message.MessageImpl;
import org.apache.cxf.service.model.EndpointInfo;
import org.apache.cxf.transport.AbstractConduit;
import org.apache.cxf.transport.Destination;
import org.apache.cxf.transport.DestinationFactory;
import org.apache.cxf.transport.DestinationFactoryManager;
import org.apache.cxf.transport.MessageObserver;
import org.apache.cxf.transport.http.policy.PolicyUtils;
import org.apache.cxf.transports.http.configuration.HTTPClientPolicy;
import org.apache.cxf.version.Version;
import org.apache.cxf.workqueue.AutomaticWorkQueue;
import org.apache.cxf.workqueue.WorkQueueManager;
import org.apache.cxf.ws.addressing.EndpointReferenceType;
import org.apache.cxf.ws.policy.Assertor;
import org.apache.cxf.ws.policy.PolicyEngine;
import org.apache.cxf.wsdl.EndpointReferenceUtils;

import static org.apache.cxf.message.Message.DECOUPLED_CHANNEL_MESSAGE;


/*
 * HTTP Conduit implementation.
 * 

* This implementation is a based on the java.net.URLConnection interface and * dependent upon installed implementations of that URLConnection, * HttpURLConnection, and HttpsURLConnection. Currently, this implementation * has been known to work with the Sun JDK 1.5 default implementations. The * HttpsURLConnection is part of Sun's implementation of the JSSE. * Presently, the source code for the Sun JSSE implementation is unavailable * and therefore we may only lay a guess of whether its HttpsURLConnection * implementation correctly works as far as security is concerned. *

* The Trust Decision. If a MessageTrustDecider is configured/set for the * Conduit, it is called upon the first flush of the headers in the * WrappedOutputStream. This reason for this approach is two-fold. * Theoretically, in order to get connection information out of the * URLConnection, it must be "connected". We assume that its implementation will * only follow through up to the point at which it will be ready to send * one byte of data down to the endpoint, but through proxies, and the * commpletion of a TLS handshake in the case of HttpsURLConnection. * However, if we force the connect() call right away, the default * implementations will not allow any calls to add/setRequestProperty, * throwing an exception that the URLConnection is already connected. *

* We need to keep the semantic that later CXF interceptors may add to the * PROTOCOL_HEADERS in the Message. This architectual decision forces us to * delay the connection until after that point, then pulling the trust decision. *

* The security caveat is that we don't really know when the connection is * really established. The call to "connect" is stated to force the * "connection," but it is a no-op if the connection was already established. * It is entirely possible that an implementation of an URLConnection may * indeed connect at will and start sending the headers down the connection * during calls to add/setRequestProperty! *

* We know that the JDK 1.5 sun.com.net.www.HttpURLConnection does not send * this information before the "connect" call, because we can look at the * source code. However, we can only assume, not verify, that the JSSE 1.5 * HttpsURLConnection does the same, in that it is probable that the * HttpsURLConnection shares the HttpURLConnection implementation. *

* Due to these implementations following redirects without trust checks, we * force the URLConnection implementations not to follow redirects. If * client side policy dictates that we follow redirects, trust decisions are * placed before each retransmit. On a redirect, any authorization information * dynamically acquired by a BasicAuth UserPass supplier is removed before * being retransmitted, as it may no longer be applicable to the new url to * which the connection is redirected. */ /** * This Conduit handles the "http" and "https" transport protocols. An * instance is governed by policies either explicitly set or by * configuration. */ public class HTTPConduit extends AbstractConduit implements Configurable, Assertor { /** * This constant is the Message(Map) key for the HttpURLConnection that * is used to get the response. */ public static final String KEY_HTTP_CONNECTION = "http.connection"; /** * This constant is the Message(Map) key for a list of visited URLs that * is used in redirect loop protection. */ private static final String KEY_VISITED_URLS = "VisitedURLs"; /** * This constant is the Message(Map) key for a list of URLs that * is used in authorization loop protection. */ private static final String KEY_AUTH_URLS = "AuthURLs"; /** * The Logger for this class. */ private static final Logger LOG = LogUtils.getL7dLogger(HTTPConduit.class); /** * This constant holds the suffix ".http-conduit" that is appended to the * Endpoint Qname to give the configuration name of this conduit. */ private static final String SC_HTTP_CONDUIT_SUFFIX = ".http-conduit"; /** * This field holds the connection factory, which primarily is used to * factor out SSL specific code from this implementation. *

* This field is "protected" to facilitate some contrived UnitTesting so * that an extended class may alter its value with an EasyMock URLConnection * Factory. */ protected HttpURLConnectionFactory connectionFactory; /** * This field holds a reference to the CXF bus associated this conduit. */ private final Bus bus; /** * This field is used for two reasons. First it provides the base name for * the conduit for Spring configuration. The other is to hold default * address information, should it not be supplied in the Message Map, by the * Message.ENDPOINT_ADDRESS property. */ private final EndpointInfo endpointInfo; /** * This field holds the "default" URL for this particular conduit, which * is created on demand. */ private URL defaultEndpointURL; private boolean fromEndpointReferenceType; private Destination decoupledDestination; private MessageObserver decoupledObserver; private int decoupledDestinationRefCount; // Configurable values /** * This field holds the QoS configuration settings for this conduit. * This field is injected via spring configuration based on the conduit * name. */ private HTTPClientPolicy clientSidePolicy; /** * This field holds the password authorization configuration. * This field is injected via spring configuration based on the conduit * name. */ private AuthorizationPolicy authorizationPolicy; /** * This field holds the password authorization configuration for the * configured proxy. This field is injected via spring configuration based * on the conduit name. */ private ProxyAuthorizationPolicy proxyAuthorizationPolicy; /** * This field holds the configuration TLS configuration which * is programmatically configured. */ private TLSClientParameters tlsClientParameters; /** * This field contains the MessageTrustDecider. */ private MessageTrustDecider trustDecider; /** * This field contains the HttpAuthSupplier. */ private HttpAuthSupplier authSupplier; /** * This boolean signfies that that finalizeConfig is called, which is * after the HTTPTransportFactory configures this object via spring. * At this point, any change by a "setter" is dynamic, and any change * should be handled as such. */ private boolean configFinalized; /** * Variables for holding session state if sessions are supposed to be maintained */ private Map sessionCookies = new ConcurrentHashMap(); private boolean maintainSession; /** * Constructor * * @param b the associated Bus * @param ei the endpoint info of the initiator * @throws IOException */ public HTTPConduit(Bus b, EndpointInfo ei) throws IOException { this(b, ei, null); } /** * Constructor * * @param b the associated Bus. * @param endpoint the endpoint info of the initiator. * @param t the endpoint reference of the target. * @throws IOException */ public HTTPConduit(Bus b, EndpointInfo ei, EndpointReferenceType t) throws IOException { super(getTargetReference(ei, t, b)); bus = b; endpointInfo = ei; if (t != null) { fromEndpointReferenceType = true; } initializeConfig(); } /** * This method returns the registered Logger for this conduit. */ protected Logger getLogger() { return LOG; } /** * This method returns the name of the conduit, which is based on the * endpoint name plus the SC_HTTP_CONDUIT_SUFFIX. * @return */ public final String getConduitName() { return endpointInfo.getName() + SC_HTTP_CONDUIT_SUFFIX; } /** * This method is called from the constructor which initializes * the configuration. The TransportFactory will call configureBean * on this object after construction. */ private void initializeConfig() { // wsdl extensors are superseded by policies which in // turn are superseded by injection PolicyEngine pe = bus.getExtension(PolicyEngine.class); if (null != pe && pe.isEnabled() && endpointInfo.getService() != null) { clientSidePolicy = PolicyUtils.getClient(pe, endpointInfo, this); } } /** * This call gets called by the HTTPTransportFactory after it * causes an injection of the Spring configuration properties * of this Conduit. */ protected void finalizeConfig() { // See if not set by configuration, if there are defaults // in order from the Endpoint, Service, or Bus. if (this.clientSidePolicy == null) { clientSidePolicy = endpointInfo.getTraversedExtensor( new HTTPClientPolicy(), HTTPClientPolicy.class); } if (this.authorizationPolicy == null) { authorizationPolicy = endpointInfo.getTraversedExtensor( new AuthorizationPolicy(), AuthorizationPolicy.class); } if (this.proxyAuthorizationPolicy == null) { proxyAuthorizationPolicy = endpointInfo.getTraversedExtensor( new ProxyAuthorizationPolicy(), ProxyAuthorizationPolicy.class); } if (this.tlsClientParameters == null) { tlsClientParameters = endpointInfo.getTraversedExtensor( null, TLSClientParameters.class); } if (this.trustDecider == null) { trustDecider = endpointInfo.getTraversedExtensor( null, MessageTrustDecider.class); } if (this.authSupplier == null) { authSupplier = endpointInfo.getTraversedExtensor( null, HttpAuthSupplier.class); } if (trustDecider == null) { if (LOG.isLoggable(Level.FINE)) { LOG.log(Level.FINE, "No Trust Decider configured for Conduit '" + getConduitName() + "'"); } } else { if (LOG.isLoggable(Level.FINE)) { LOG.log(Level.FINE, "Message Trust Decider of class '" + trustDecider.getClass().getName() + "' with logical name of '" + trustDecider.getLogicalName() + "' has been configured for Conduit '" + getConduitName() + "'"); } } if (authSupplier == null) { if (LOG.isLoggable(Level.FINE)) { LOG.log(Level.FINE, "No Auth Supplier configured for Conduit '" + getConduitName() + "'"); } } else { if (LOG.isLoggable(Level.FINE)) { LOG.log(Level.FINE, "HttpAuthSupplier of class '" + authSupplier.getClass().getName() + "' with logical name of '" + authSupplier.getLogicalName() + "' has been configured for Conduit '" + getConduitName() + "'"); } } if (this.tlsClientParameters != null) { if (LOG.isLoggable(Level.FINE)) { LOG.log(Level.FINE, "Conduit '" + getConduitName() + "' has been configured for TLS " + "keyManagers " + tlsClientParameters.getKeyManagers() + "trustManagers " + tlsClientParameters.getTrustManagers() + "secureRandom " + tlsClientParameters.getSecureRandom() + "Disable Common Name (CN) Check: " + tlsClientParameters.isDisableCNCheck()); } } else { if (LOG.isLoggable(Level.FINE)) { LOG.log(Level.FINE, "Conduit '" + getConduitName() + "' has been configured for plain http."); } } // Get the correct URLConnection factory based on the // configuration. retrieveConnectionFactory(); // We have finalized the configuration. Any configurable entity // set now, must make changes dynamically. configFinalized = true; } /** * Allow access to the cookies that the conduit is maintaining * @return the sessionCookies map */ public Map getCookies() { return sessionCookies; } /** * This method sets the connectionFactory field for this object. It is called * after an SSL Client Policy is set or an HttpsHostnameVerifier * because we need to reinitialize the connection factory. *

* This method is "protected" so that this class may be extended and override * this method to put an EasyMock URL Connection factory for some contrived * UnitTest that will of course break, should the calls to the URL Connection * Factory get altered. */ protected synchronized void retrieveConnectionFactory() { connectionFactory = AbstractHTTPTransportFactory.getConnectionFactory(this); } protected synchronized void retrieveConnectionFactory(String url) { connectionFactory = AbstractHTTPTransportFactory.getConnectionFactory(this, url); } protected synchronized HttpURLConnectionFactory getConnectionFactory(URL url) { if (connectionFactory == null || !url.getProtocol().equals(connectionFactory.getProtocol())) { retrieveConnectionFactory(url.toString()); } return connectionFactory; } /** * Prepare to send an outbound HTTP message over this http conduit to a * particular endpoint. *

* If the Message.PATH_INFO property is set it gets appended * to the Conduit's endpoint URL. If the Message.QUERY_STRING * property is set, it gets appended to the resultant URL following * a "?". *

* If the Message.HTTP_REQUEST_METHOD property is NOT set, the * Http request method defaults to "POST". *

* If the Message.PROTOCOL_HEADERS is not set on the message, it is * initialized to an empty map. *

* This call creates the OutputStream for the content of the message. * It also assigns the created Http(s)URLConnection to the Message * Map. * * @param message The message to be sent. */ public void prepare(Message message) throws IOException { Map> headers = getSetProtocolHeaders(message); // This call can possibly change the conduit endpoint address and // protocol from the default set in EndpointInfo that is associated // with the Conduit. URL currentURL = setupURL(message); // The need to cache the request is off by default boolean needToCacheRequest = false; HTTPClientPolicy csPolicy = getClient(message); HttpURLConnection connection = getConnectionFactory(currentURL) .createConnection(getProxy(csPolicy), currentURL); connection.setDoOutput(true); //TODO using Message context to decided HTTP send properties long timeout = csPolicy.getConnectionTimeout(); if (timeout > Integer.MAX_VALUE) { timeout = Integer.MAX_VALUE; } connection.setConnectTimeout((int)timeout); timeout = csPolicy.getReceiveTimeout(); if (timeout > Integer.MAX_VALUE) { timeout = Integer.MAX_VALUE; } connection.setReadTimeout((int)timeout); connection.setUseCaches(false); // We implement redirects in this conduit. We do not // rely on the underlying URLConnection implementation // because of trust issues. connection.setInstanceFollowRedirects(false); // If the HTTP_REQUEST_METHOD is not set, the default is "POST". String httpRequestMethod = (String)message.get(Message.HTTP_REQUEST_METHOD); if (null != httpRequestMethod) { connection.setRequestMethod(httpRequestMethod); } else { connection.setRequestMethod("POST"); } boolean isChunking = false; int chunkThreshold = 0; // We must cache the request if we have basic auth supplier // without preemptive basic auth. if (authSupplier != null) { String auth = authSupplier.getPreemptiveAuthorization( this, currentURL, message); if (auth == null || authSupplier.requiresRequestCaching()) { needToCacheRequest = true; isChunking = false; LOG.log(Level.FINE, "Auth Supplier, but no Premeptive User Pass or Digest auth (nonce may be stale)" + " We must cache request."); } message.put("AUTH_VALUE", auth); } if (csPolicy.isAutoRedirect()) { needToCacheRequest = true; LOG.log(Level.FINE, "AutoRedirect is turned on."); } if (csPolicy.getMaxRetransmits() > 0) { needToCacheRequest = true; LOG.log(Level.FINE, "MaxRetransmits is set > 0."); } // DELETE does not work and empty PUTs cause misleading exceptions // if chunking is enabled // TODO : ensure chunking can be enabled for non-empty PUTs - if requested if (connection.getRequestMethod().equals("POST") && csPolicy.isAllowChunking()) { //TODO: The chunking mode be configured or at least some // documented client constant. //use -1 and allow the URL connection to pick a default value isChunking = true; chunkThreshold = csPolicy.getChunkingThreshold(); if (chunkThreshold <= 0) { chunkThreshold = 0; connection.setChunkedStreamingMode(-1); } } //Do we need to maintain a session? maintainSession = Boolean.TRUE.equals((Boolean)message.get(Message.MAINTAIN_SESSION)); //If we have any cookies and we are maintaining sessions, then use them if (maintainSession && sessionCookies.size() > 0) { List cookies = null; for (String s : headers.keySet()) { if (HttpHeaderHelper.COOKIE.equalsIgnoreCase(s)) { cookies = headers.remove(s); break; } } if (cookies == null) { cookies = new ArrayList(); } else { cookies = new ArrayList(cookies); } headers.put(HttpHeaderHelper.COOKIE, cookies); for (Cookie c : sessionCookies.values()) { cookies.add(c.requestCookieHeader()); } } // The trust decision is relegated to after the "flushing" of the // request headers. // We place the connection on the message to pick it up // in the WrappedOutputStream. message.put(KEY_HTTP_CONNECTION, connection); // Set the headers on the message according to configured // client side policy. setHeadersByPolicy(message, currentURL, headers); message.setContent(OutputStream.class, new WrappedOutputStream( message, connection, needToCacheRequest, isChunking, chunkThreshold)); // We are now "ready" to "send" the message. } public void close(Message msg) throws IOException { InputStream in = msg.getContent(InputStream.class); try { if (in != null) { int count = 0; byte buffer[] = new byte[1024]; while (in.read(buffer) != -1 && count < 25) { //don't do anything, we just need to pull off the unread data (like //closing tags that we didn't need to read //however, limit it so we don't read off gigabytes of data we won't use. ++count; } } } finally { super.close(msg); } } /** * This call must take place before anything is written to the * URLConnection. The URLConnection.connect() will be called in order * to get the connection information. * * This method is invoked just after setURLRequestHeaders() from the * WrappedOutputStream before it writes data to the URLConnection. * * If trust cannot be established the Trust Decider implemenation * throws an IOException. * * @param message The message being sent. * @throws IOException This exception is thrown if trust cannot be * established by the configured MessageTrustDecider. * @see MessageTrustDecider */ private void makeTrustDecision(Message message) throws IOException { HttpURLConnection connection = (HttpURLConnection) message.get(KEY_HTTP_CONNECTION); MessageTrustDecider decider2 = message.get(MessageTrustDecider.class); if (trustDecider != null || decider2 != null) { try { // We must connect or we will not get the credentials. // The call is (said to be) ingored internally if // already connected. connection.connect(); URLConnectionInfo info = getConnectionFactory(connection.getURL()) .getConnectionInfo(connection); if (trustDecider != null) { trustDecider.establishTrust( getConduitName(), info, message); if (LOG.isLoggable(Level.FINE)) { LOG.log(Level.FINE, "Trust Decider " + trustDecider.getLogicalName() + " considers Conduit " + getConduitName() + " trusted."); } } if (decider2 != null) { decider2.establishTrust(getConduitName(), info, message); if (LOG.isLoggable(Level.FINE)) { LOG.log(Level.FINE, "Trust Decider " + decider2.getLogicalName() + " considers Conduit " + getConduitName() + " trusted."); } } } catch (UntrustedURLConnectionIOException untrustedEx) { // This cast covers HttpsURLConnection as well. ((HttpURLConnection)connection).disconnect(); if (LOG.isLoggable(Level.FINE)) { LOG.log(Level.FINE, "Trust Decider " + trustDecider.getLogicalName() + " considers Conduit " + getConduitName() + " untrusted.", untrustedEx); } throw untrustedEx; } } else { // This case, when there is no trust decider, a trust // decision should be a matter of policy. if (LOG.isLoggable(Level.FINE)) { LOG.log(Level.FINE, "No Trust Decider for Conduit '" + getConduitName() + "'. An afirmative Trust Decision is assumed."); } } } /** * This function sets up a URL based on ENDPOINT_ADDRESS, PATH_INFO, * and QUERY_STRING properties in the Message. The QUERY_STRING gets * added with a "?" after the PATH_INFO. If the ENDPOINT_ADDRESS is not * set on the Message, the endpoint address is taken from the * "defaultEndpointURL". *

* The PATH_INFO is only added to the endpoint address string should * the PATH_INFO not equal the end of the endpoint address string. * * @param message The message holds the addressing information. * * @return The full URL specifying the HTTP request to the endpoint. * * @throws MalformedURLException */ private URL setupURL(Message message) throws MalformedURLException { String result = (String)message.get(Message.ENDPOINT_ADDRESS); String pathInfo = (String)message.get(Message.PATH_INFO); String queryString = (String)message.get(Message.QUERY_STRING); if (result == null) { if (pathInfo == null && queryString == null) { URL url = getURL(); message.put(Message.ENDPOINT_ADDRESS, url.toString()); return url; } result = getURL().toString(); message.put(Message.ENDPOINT_ADDRESS, result); } // REVISIT: is this really correct? if (null != pathInfo && !result.endsWith(pathInfo)) { result = result + pathInfo; } if (queryString != null) { result = result + "?" + queryString; } return new URL(result); } /** * Retreive the back-channel Destination. * * @return the backchannel Destination (or null if the backchannel is * built-in) */ public synchronized Destination getBackChannel() { if (decoupledDestination == null && getClient().getDecoupledEndpoint() != null) { setUpDecoupledDestination(); } return decoupledDestination; } /** * Close the conduit */ public void close() { if (defaultEndpointURL != null) { try { URLConnection connect = defaultEndpointURL.openConnection(); if (connect instanceof HttpURLConnection) { ((HttpURLConnection)connect).disconnect(); } } catch (IOException ex) { //ignore } //defaultEndpointURL = null; } // in decoupled case, close response Destination if reference count // hits zero // if (decoupledDestination != null) { releaseDecoupledDestination(); } } /** * @return the default target address */ protected String getAddress() throws MalformedURLException { if (defaultEndpointURL != null) { return defaultEndpointURL.toExternalForm(); } else if (fromEndpointReferenceType) { return getTarget().getAddress().getValue(); } return endpointInfo.getAddress(); } /** * @return the default target URL */ protected synchronized URL getURL() throws MalformedURLException { return getURL(true); } /** * @param createOnDemand create URL on-demand if null * @return the default target URL */ protected synchronized URL getURL(boolean createOnDemand) throws MalformedURLException { if (defaultEndpointURL == null && createOnDemand) { if (fromEndpointReferenceType && getTarget().getAddress().getValue() != null) { defaultEndpointURL = new URL(this.getTarget().getAddress().getValue()); return defaultEndpointURL; } if (endpointInfo.getAddress() == null) { throw new MalformedURLException("Invalid address. Endpoint address cannot be null."); } defaultEndpointURL = new URL(endpointInfo.getAddress()); } return defaultEndpointURL; } /** * While extracting the Message.PROTOCOL_HEADERS property from the Message, * this call ensures that the Message.PROTOCOL_HEADERS property is * set on the Message. If it is not set, an empty map is placed there, and * then returned. * * @param message The outbound message * @return The PROTOCOL_HEADERS map */ private Map> getSetProtocolHeaders(Message message) { Map> headers = CastUtils.cast((Map)message.get(Message.PROTOCOL_HEADERS)); if (null == headers) { headers = new LinkedHashMap>(); } else if (headers instanceof HashMap) { headers = new LinkedHashMap>(headers); } message.put(Message.PROTOCOL_HEADERS, headers); return headers; } /** * This procedure sets the URLConnection request properties * from the PROTOCOL_HEADERS in the message. */ private void transferProtocolHeadersToURLConnection( Message message, URLConnection connection ) { Map> headers = getSetProtocolHeaders(message); for (String header : headers.keySet()) { List headerList = headers.get(header); if (HttpHeaderHelper.CONTENT_TYPE.equalsIgnoreCase(header)) { continue; } if (HttpHeaderHelper.COOKIE.equalsIgnoreCase(header)) { for (String s : headerList) { connection.addRequestProperty(HttpHeaderHelper.COOKIE, s); } } else { StringBuilder b = new StringBuilder(); for (int i = 0; i < headerList.size(); i++) { b.append(headerList.get(i)); if (i + 1 < headerList.size()) { b.append(','); } } connection.setRequestProperty(header, b.toString()); } } if (!connection.getRequestProperties().containsKey("User-Agent")) { connection.addRequestProperty("User-Agent", Version.getCompleteVersionString()); } } /** * This procedure logs the PROTOCOL_HEADERS from the * Message at the specified logging level. * * @param level The Logging Level. * @param headers The Message protocol headers. */ private void logProtocolHeaders( Level level, Message message ) { Map> headers = getSetProtocolHeaders(message); for (String header : headers.keySet()) { List headerList = headers.get(header); for (String value : headerList) { LOG.log(level, header + ": " + value); } } } /** * Put the headers from Message.PROTOCOL_HEADERS headers into the URL * connection. * Note, this does not mean they immediately get written to the output * stream or the wire. They just just get set on the HTTP request. * * @param message The outbound message. * @throws IOException */ private void setURLRequestHeaders(Message message) throws IOException { HttpURLConnection connection = (HttpURLConnection)message.get(KEY_HTTP_CONNECTION); String ct = (String) message.get(Message.CONTENT_TYPE); String enc = (String) message.get(Message.ENCODING); if (null != ct) { if (enc != null && ct.indexOf("charset=") == -1 && !ct.toLowerCase().contains("multipart/related")) { ct = ct + "; charset=" + enc; } } else if (enc != null) { ct = "text/xml; charset=" + enc; } else { ct = "text/xml"; } connection.setRequestProperty(HttpHeaderHelper.CONTENT_TYPE, ct); if (LOG.isLoggable(Level.FINE)) { LOG.fine("Sending " + connection.getRequestMethod() + " Message with Headers to " + connection.getURL() + " Conduit :" + getConduitName() + "\nContent-Type: " + ct + "\n"); logProtocolHeaders(Level.FINE, message); } transferProtocolHeadersToURLConnection(message, connection); } /** * Set up the decoupled Destination if necessary. */ private void setUpDecoupledDestination() { EndpointReferenceType reference = EndpointReferenceUtils.getEndpointReference( getClient().getDecoupledEndpoint()); if (reference != null) { String decoupledAddress = reference.getAddress().getValue(); LOG.info("creating decoupled endpoint: " + decoupledAddress); try { decoupledDestination = getDestination(decoupledAddress); duplicateDecoupledDestination(); } catch (Exception e) { // REVISIT move message to localizable Messages.properties LOG.log(Level.WARNING, "decoupled endpoint creation failed: ", e); } } } /** * @param address the address * @return a Destination for the address */ private Destination getDestination(String address) throws IOException { Destination destination = null; DestinationFactoryManager factoryManager = bus.getExtension(DestinationFactoryManager.class); DestinationFactory factory = factoryManager.getDestinationFactoryForUri(address); if (factory != null) { EndpointInfo ei = new EndpointInfo(); ei.setAddress(address); destination = factory.getDestination(ei); decoupledObserver = new InterposedMessageObserver(); destination.setMessageObserver(decoupledObserver); } return destination; } /** * @return the decoupled observer */ protected MessageObserver getDecoupledObserver() { return decoupledObserver; } private synchronized void duplicateDecoupledDestination() { decoupledDestinationRefCount++; } private synchronized void releaseDecoupledDestination() { if (--decoupledDestinationRefCount == 0) { LOG.log(Level.FINE, "shutting down decoupled destination"); decoupledDestination.shutdown(); //this way we can release the port of decoupled destination decoupledDestination.setMessageObserver(null); } } /** * This predicate returns true iff the exchange indicates * a oneway MEP. * * @param exchange The exchange in question */ private boolean isOneway(Exchange exchange) { return exchange != null && exchange.isOneWay(); } /** * @return true if expecting a decoupled response */ private boolean isDecoupled() { return decoupledDestination != null; } /** * Get an input stream containing the partial response if one is present. * * @param connection the connection in question * @param responseCode the response code * @return an input stream if a partial response is pending on the connection */ protected static InputStream getPartialResponse( HttpURLConnection connection, int responseCode ) throws IOException { InputStream in = null; if (responseCode == HttpURLConnection.HTTP_ACCEPTED || responseCode == HttpURLConnection.HTTP_OK) { if (connection.getContentLength() > 0) { in = connection.getInputStream(); } else if (hasChunkedResponse(connection) || hasEofTerminatedResponse(connection)) { // ensure chunked or EOF-terminated response is non-empty in = getNonEmptyContent(connection); } } return in; } /** * @param connection the given HttpURLConnection * @return true iff the connection has a chunked response pending */ private static boolean hasChunkedResponse(HttpURLConnection connection) { return HttpHeaderHelper.CHUNKED.equalsIgnoreCase( connection.getHeaderField(HttpHeaderHelper.TRANSFER_ENCODING)); } /** * @param connection the given HttpURLConnection * @return true iff the connection has a chunked response pending */ private static boolean hasEofTerminatedResponse( HttpURLConnection connection ) { return HttpHeaderHelper.CLOSE.equalsIgnoreCase( connection.getHeaderField(HttpHeaderHelper.CONNECTION)); } /** * @param connection the given HttpURLConnection * @return an input stream containing the response content if non-empty */ private static InputStream getNonEmptyContent( HttpURLConnection connection ) { InputStream in = null; try { PushbackInputStream pin = new PushbackInputStream(connection.getInputStream()); int c = pin.read(); if (c != -1) { pin.unread((byte)c); in = pin; } } catch (IOException ioe) { // ignore } return in; } /** * This method returns the Proxy server should it be set on the * Client Side Policy. * * @return The proxy server or null, if not set. */ private Proxy getProxy(HTTPClientPolicy policy) { Proxy proxy = null; if (policy != null && policy.isSetProxyServer() && !StringUtils.isEmpty(policy.getProxyServer())) { proxy = new Proxy( Proxy.Type.valueOf(policy.getProxyServerType().toString()), new InetSocketAddress(policy.getProxyServer(), policy.getProxyServerPort())); } return proxy; } /** * This call places HTTP Header strings into the headers that are relevant * to the Authorization policies that are set on this conduit by * configuration. *

* An AuthorizationPolicy may also be set on the message. If so, those * policies are merged. A user name or password set on the messsage * overrides settings in the AuthorizationPolicy is retrieved from the * configuration. *

* The precedence is as follows: * 1. AuthorizationPolicy that is set on the Message, if exists. * 2. Authorization from AuthSupplier, if exists. * 3. AuthorizationPolicy set/configured for conduit. * * REVISIT: Since the AuthorizationPolicy is set on the message by class, then * how does one override the ProxyAuthorizationPolicy which is the same * type? * * @param message * @param headers */ private void setHeadersByAuthorizationPolicy( Message message, URL url, Map> headers ) { AuthorizationPolicy authPolicy = getAuthorization(); AuthorizationPolicy newPolicy = message.get(AuthorizationPolicy.class); String authString = null; if (authSupplier != null && (newPolicy == null || (!"Basic".equals(newPolicy.getAuthorizationType()) && newPolicy.getAuthorization() == null))) { authString = (String)message.get("AUTH_VALUE"); if (authString == null) { authString = authSupplier.getPreemptiveAuthorization( this, url, message); } else { message.remove("AUTH_VALUE"); } if (authString != null) { headers.put("Authorization", createMutableList(authString)); } return; } String userName = null; String passwd = null; if (null != newPolicy) { userName = newPolicy.getUserName(); passwd = newPolicy.getPassword(); } if (userName == null && authPolicy != null && authPolicy.isSetUserName()) { userName = authPolicy.getUserName(); } if (userName != null) { if (passwd == null && authPolicy != null && authPolicy.isSetPassword()) { passwd = authPolicy.getPassword(); } setBasicAuthHeader(userName, passwd, headers); } else if (authPolicy != null && authPolicy.isSetAuthorizationType() && authPolicy.isSetAuthorization()) { String type = authPolicy.getAuthorizationType(); type += " "; type += authPolicy.getAuthorization(); headers.put("Authorization", createMutableList(type)); } AuthorizationPolicy proxyAuthPolicy = getProxyAuthorization(); if (proxyAuthPolicy != null && proxyAuthPolicy.isSetUserName()) { userName = proxyAuthPolicy.getUserName(); if (userName != null) { passwd = ""; if (proxyAuthPolicy.isSetPassword()) { passwd = proxyAuthPolicy.getPassword(); } setProxyBasicAuthHeader(userName, passwd, headers); } else if (proxyAuthPolicy.isSetAuthorizationType() && proxyAuthPolicy.isSetAuthorization()) { String type = proxyAuthPolicy.getAuthorizationType(); type += " "; type += proxyAuthPolicy.getAuthorization(); headers.put("Proxy-Authorization", createMutableList(type)); } } } private static List createMutableList(String val) { return new ArrayList(Arrays.asList(new String[] {val})); } /** * This call places HTTP Header strings into the headers that are relevant * to the ClientPolicy that is set on this conduit by configuration. * * REVISIT: A cookie is set statically from configuration? */ private void setHeadersByClientPolicy( Message message, Map> headers ) { HTTPClientPolicy policy = getClient(message); if (policy == null) { return; } if (policy.isSetCacheControl()) { headers.put("Cache-Control", createMutableList(policy.getCacheControl().value())); } if (policy.isSetHost()) { headers.put("Host", createMutableList(policy.getHost())); } if (policy.isSetConnection()) { headers.put("Connection", createMutableList(policy.getConnection().value())); } if (policy.isSetAccept()) { headers.put("Accept", createMutableList(policy.getAccept())); } else if (!headers.containsKey("Accept")) { headers.put("Accept", createMutableList("*/*")); } if (policy.isSetAcceptEncoding()) { headers.put("Accept-Encoding", createMutableList(policy.getAcceptEncoding())); } if (policy.isSetAcceptLanguage()) { headers.put("Accept-Language", createMutableList(policy.getAcceptLanguage())); } if (policy.isSetContentType()) { message.put(Message.CONTENT_TYPE, policy.getContentType()); } if (policy.isSetCookie()) { headers.put("Cookie", createMutableList(policy.getCookie())); } if (policy.isSetBrowserType()) { headers.put("BrowserType", createMutableList(policy.getBrowserType())); } if (policy.isSetReferer()) { headers.put("Referer", createMutableList(policy.getReferer())); } } /** * This call places HTTP Header strings into the headers that are relevant * to the polices that are set on this conduit by configuration for the * ClientPolicy and AuthorizationPolicy. * * * @param message The outgoing message. * @param url The URL the message is going to. * @param headers The headers in the outgoing message. */ private void setHeadersByPolicy( Message message, URL url, Map> headers ) { setHeadersByAuthorizationPolicy(message, url, headers); setHeadersByClientPolicy(message, headers); } /** * This is part of the Configurable interface which retrieves the * configuration from spring injection. */ // REVISIT:What happens when the endpoint/bean name is null? public String getBeanName() { if (endpointInfo.getName() != null) { return endpointInfo.getName().toString() + ".http-conduit"; } return null; } /** * This method gets the Authorization Policy that was configured or * explicitly set for this HTTPConduit. */ public AuthorizationPolicy getAuthorization() { return authorizationPolicy; } /** * This method is used to set the Authorization Policy for this conduit. * Using this method will override any Authorization Policy set in * configuration. */ public void setAuthorization(AuthorizationPolicy authorization) { this.authorizationPolicy = authorization; } public HTTPClientPolicy getClient(Message message) { return PolicyUtils.getClient(message, clientSidePolicy); } /** * This method retrieves the Client Side Policy set/configured for this * HTTPConduit. */ public HTTPClientPolicy getClient() { return clientSidePolicy; } /** * This method sets the Client Side Policy for this HTTPConduit. Using this * method will override any HTTPClientPolicy set in configuration. */ public void setClient(HTTPClientPolicy client) { this.clientSidePolicy = client; } /** * This method retrieves the Proxy Authorization Policy for a proxy that is * set/configured for this HTTPConduit. */ public ProxyAuthorizationPolicy getProxyAuthorization() { return proxyAuthorizationPolicy; } /** * This method sets the Proxy Authorization Policy for a specified proxy. * Using this method overrides any Authorization Policy for the proxy * that is set in the configuration. */ public void setProxyAuthorization( ProxyAuthorizationPolicy proxyAuthorization ) { this.proxyAuthorizationPolicy = proxyAuthorization; } /** * This method returns the TLS Client Parameters that is set/configured * for this HTTPConduit. */ public TLSClientParameters getTlsClientParameters() { return tlsClientParameters; } /** * This method sets the TLS Client Parameters for this HTTPConduit. * Using this method overrides any TLS Client Parameters that is configured * for this HTTPConduit. */ public void setTlsClientParameters(TLSClientParameters params) { this.tlsClientParameters = params; if (this.tlsClientParameters != null) { if (LOG.isLoggable(Level.FINE)) { LOG.log(Level.FINE, "Conduit '" + getConduitName() + "' has been (re) configured for TLS " + "keyManagers " + tlsClientParameters.getKeyManagers() + "trustManagers " + tlsClientParameters.getTrustManagers() + "secureRandom " + tlsClientParameters.getSecureRandom()); } } else { if (LOG.isLoggable(Level.FINE)) { LOG.log(Level.FINE, "Conduit '" + getConduitName() + "' has been (re)configured for plain http."); } } // If this is called after the HTTPTransportFactory called // finalizeConfig, we need to update the connection factory. if (configFinalized) { retrieveConnectionFactory(); } } /** * This method gets the Trust Decider that was set/configured for this * HTTPConduit. * @return The Message Trust Decider or null. */ public MessageTrustDecider getTrustDecider() { return this.trustDecider; } /** * This method sets the Trust Decider for this HTTP Conduit. * Using this method overrides any trust decider configured for this * HTTPConduit. */ public void setTrustDecider(MessageTrustDecider decider) { this.trustDecider = decider; } /** * This method gets the Auth Supplier that was set/configured for this * HTTPConduit. * @return The Auth Supplier or null. */ public HttpAuthSupplier getAuthSupplier() { return this.authSupplier; } public void setAuthSupplier(HttpAuthSupplier supplier) { this.authSupplier = supplier; } /** * This function processes any retransmits at the direction of redirections * or "unauthorized" responses. *

* If the request was not retransmitted, it returns the given connection. * If the request was retransmitted, it returns the new connection on * which the request was sent. * * @param connection The active URL connection. * @param message The outgoing message. * @param cachedStream The cached request. * @return * @throws IOException */ private HttpURLConnection processRetransmit( HttpURLConnection connection, Message message, CacheAndWriteOutputStream cachedStream ) throws IOException { int responseCode = connection.getResponseCode(); if ((message != null) && (message.getExchange() != null)) { message.getExchange().put(Message.RESPONSE_CODE, responseCode); } // Process Redirects first. switch(responseCode) { case HttpURLConnection.HTTP_MOVED_PERM: case HttpURLConnection.HTTP_MOVED_TEMP: connection = redirectRetransmit(connection, message, cachedStream); break; case HttpURLConnection.HTTP_UNAUTHORIZED: connection = authorizationRetransmit(connection, message, cachedStream); break; default: break; } return connection; } /** * This method performs a redirection retransmit in response to * a 302 or 305 response code. * * @param connection The active URL connection * @param message The outbound message. * @param cachedStream The cached request. * @return This method returns the new HttpURLConnection if * redirected. If it cannot be redirected for some reason * the same connection is returned. * * @throws IOException */ private HttpURLConnection redirectRetransmit( HttpURLConnection connection, Message message, CacheAndWriteOutputStream cachedStream ) throws IOException { // If we are not redirecting by policy, then we don't. if (!getClient(message).isAutoRedirect()) { return connection; } // We keep track of the redirections for redirect loop protection. Set visitedURLs = getSetVisitedURLs(message); String lastURL = connection.getURL().toString(); visitedURLs.add(lastURL); String newURL = extractLocation(connection.getHeaderFields()); if (newURL != null) { // See if we are being redirected in a loop as best we can, // using string equality on URL. if (visitedURLs.contains(newURL)) { // We are in a redirect loop; -- bail if (LOG.isLoggable(Level.INFO)) { LOG.log(Level.INFO, "Redirect loop detected on Conduit \"" + getConduitName() + "\" on '" + newURL + "'"); } throw new IOException("Redirect loop detected on Conduit \"" + getConduitName() + "\" on '" + newURL + "'"); } // We are going to redirect. // Remove any Server Authentication Information for the previous // URL. Map> headers = getSetProtocolHeaders(message); headers.remove("Authorization"); headers.remove("Proxy-Authorization"); URL url = new URL(newURL); // If user configured this Conduit with preemptive authorization // it is meant to make it to the end. (Too bad that information // went to every URL along the way, but that's what the user // wants! // TODO: Make this issue a security release note. setHeadersByAuthorizationPolicy(message, url, headers); connection = retransmit( connection, url, message, cachedStream); } return connection; } /** * This function gets the Set of URLs on the message that is used to * keep track of the URLs that were used in getting authorization * information. * * @param message The message where the Set of URLs is stored. * @return The modifiable set of URLs that were visited. */ private Set getSetAuthoriationURLs(Message message) { @SuppressWarnings("unchecked") Set authURLs = (Set) message.get(KEY_AUTH_URLS); if (authURLs == null) { authURLs = new HashSet(); message.put(KEY_AUTH_URLS, authURLs); } return authURLs; } /** * This function get the set of URLs on the message that is used to keep * track of the URLs that were visited in redirects. * * If it is not set on the message, an new empty set is stored. * @param message The message where the Set is stored. * @return The modifiable set of URLs that were visited. */ private Set getSetVisitedURLs(Message message) { @SuppressWarnings("unchecked") Set visitedURLs = (Set) message.get(KEY_VISITED_URLS); if (visitedURLs == null) { visitedURLs = new HashSet(); message.put(KEY_VISITED_URLS, visitedURLs); } return visitedURLs; } /** * This method performs a retransmit for authorization information. * * @param connection The currently active connection. * @param message The outbound message. * @param cachedStream The cached request. * @return A new connection if retransmitted. If not retransmitted * then this method returns the same connection. * @throws IOException */ private HttpURLConnection authorizationRetransmit( HttpURLConnection connection, Message message, CacheAndWriteOutputStream cachedStream ) throws IOException { // If we don't have a dynamic supply of user pass, then // we don't retransmit. We just die with a Http 401 response. if (authSupplier == null) { String auth = connection.getHeaderField("WWW-Authenticate"); if (auth.startsWith("Digest ")) { authSupplier = new DigestAuthSupplier(); } else { return connection; } } URL currentURL = connection.getURL(); String realm = extractAuthorizationRealm(connection.getHeaderFields()); Set authURLs = getSetAuthoriationURLs(message); // If we have been here (URL & Realm) before for this particular message // retransmit, it means we have already supplied information // which must have been wrong, or we wouldn't be here again. // Otherwise, the server may be 401 looping us around the realms. if (authURLs.contains(currentURL.toString() + realm)) { if (LOG.isLoggable(Level.INFO)) { LOG.log(Level.INFO, "Authorization loop detected on Conduit \"" + getConduitName() + "\" on URL \"" + "\" with realm \"" + realm + "\""); } throw new IOException("Authorization loop detected on Conduit \"" + getConduitName() + "\" on URL \"" + "\" with realm \"" + realm + "\""); } String up = authSupplier.getAuthorizationForRealm( this, currentURL, message, realm, connection.getHeaderField("WWW-Authenticate")); // No user pass combination. We give up. if (up == null) { return connection; } // Register that we have been here before we go. authURLs.add(currentURL.toString() + realm); Map> headers = getSetProtocolHeaders(message); headers.put("Authorization", createMutableList(up)); return retransmit( connection, currentURL, message, cachedStream); } /** * This method retransmits the request. * * @param connection The currently active connection. * @param newURL The newURL to connection to. * @param message The outbound message. * @param stream The cached request. * @return This function returns a new connection if * retransmitted, otherwise it returns the given * connection. * * @throws IOException */ private HttpURLConnection retransmit( HttpURLConnection connection, URL newURL, Message message, CacheAndWriteOutputStream stream ) throws IOException { // Disconnect the old, and in with the new. connection.disconnect(); HTTPClientPolicy cp = getClient(message); connection = getConnectionFactory(newURL).createConnection(getProxy(cp), newURL); connection.setDoOutput(true); // TODO: using Message context to deceided HTTP send properties connection.setConnectTimeout((int)cp.getConnectionTimeout()); connection.setReadTimeout((int)cp.getReceiveTimeout()); connection.setUseCaches(false); connection.setInstanceFollowRedirects(false); // If the HTTP_REQUEST_METHOD is not set, the default is "POST". String httpRequestMethod = (String)message.get(Message.HTTP_REQUEST_METHOD); if (null != httpRequestMethod) { connection.setRequestMethod(httpRequestMethod); } else { connection.setRequestMethod("POST"); } message.put(KEY_HTTP_CONNECTION, connection); connection.setFixedLengthStreamingMode(stream.size()); // Need to set the headers before the trust decision // because they are set before the connect(). setURLRequestHeaders(message); // // This point is where the trust decision is made because the // Sun implementation of URLConnection will not let us // set/addRequestProperty after a connect() call, and // makeTrustDecision needs to make a connect() call to // make sure the proper information is available. // makeTrustDecision(message); // If this is a GET method we must not touch the output // stream as this automagically turns the request into a POST. if (connection.getRequestMethod().equals("GET")) { return connection; } // Trust is okay, write the cached request OutputStream out = connection.getOutputStream(); stream.writeCacheTo(out); if (LOG.isLoggable(Level.FINE)) { LOG.fine("Conduit \"" + getConduitName() + "\" Retransmit message to: " + connection.getURL() + ": " + new String(stream.getBytes())); } return connection; } /** * This function extracts the authorization realm from the * "WWW-Authenticate" Http response header. * * @param headers The Http Response Headers * @return The realm, or null if it is non-existent. */ private String extractAuthorizationRealm( Map> headers ) { List auth = headers.get("WWW-Authenticate"); if (auth != null) { for (String a : auth) { int idx = a.indexOf("realm="); if (idx != -1) { a = a.substring(idx + 6); if (a.charAt(0) == '"') { a = a.substring(1, a.indexOf('"', 1)); } else if (a.contains(",")) { a = a.substring(0, a.indexOf(',')); } return a; } } } return null; } /** * This method extracts the value of the "Location" Http * Response header. * * @param headers The Http response headers. * @return The value of the "Location" header, null if non-existent. */ private String extractLocation( Map> headers ) { for (Map.Entry> head : headers.entrySet()) { if ("Location".equalsIgnoreCase(head.getKey())) { List locs = head.getValue(); if (locs != null && locs.size() > 0) { return locs.get(0); } } } return null; } /** * This procedure sets the "Authorization" header with the * BasicAuth token, which is Base64 encoded. * * @param userid The user's id, which cannot be null. * @param password The password, it may be null. * * @param headers The headers map that gets the "Authorization" header set. */ private void setBasicAuthHeader( String userid, String password, Map> headers ) { String userpass = userid; userpass += ":"; if (password != null) { userpass += password; } String token = Base64Utility.encode(userpass.getBytes()); headers.put("Authorization", createMutableList("Basic " + token)); } /** * This procedure sets the "ProxyAuthorization" header with the * BasicAuth token, which is Base64 encoded. * * @param userid The user's id, which cannot be null. * @param password The password, it may be null. * * @param headers The headers map that gets the "Proxy-Authorization" * header set. */ private void setProxyBasicAuthHeader( String userid, String password, Map> headers ) { String userpass = userid; userpass += ":"; if (password != null) { userpass += password; } String token = Base64Utility.encode(userpass.getBytes()); headers.put("Proxy-Authorization", createMutableList("Basic " + token)); } /** * Wrapper output stream responsible for flushing headers and handling * the incoming HTTP-level response (not necessarily the MEP response). */ protected class WrappedOutputStream extends AbstractThresholdOutputStream { /** * This field contains the currently active connection. */ protected HttpURLConnection connection; /** * This boolean is true if the request must be cached. */ protected boolean cachingForRetransmission; /** * If we are going to be chunking, we won't flush till close which causes * new chunks, small network packets, etc.. */ protected final boolean chunking; /** * This field contains the output stream with which we cache * the request. It maybe null if we are not caching. */ protected CacheAndWriteOutputStream cachedStream; protected Message outMessage; protected WrappedOutputStream( Message m, HttpURLConnection c, boolean possibleRetransmit, boolean isChunking, int chunkThreshold ) { super(chunkThreshold); this.outMessage = m; connection = c; cachingForRetransmission = possibleRetransmit; chunking = isChunking; } @Override public void thresholdNotReached() { if (chunking) { connection.setFixedLengthStreamingMode(buffer.size()); } } @Override public void thresholdReached() { if (chunking) { connection.setChunkedStreamingMode(-1); } } /** * Perform any actions required on stream flush (freeze headers, * reset output stream ... etc.) */ @Override protected void onFirstWrite() throws IOException { try { handleHeadersTrustCaching(); } catch (IOException e) { if (e.getMessage() != null && e.getMessage().contains("HTTPS hostname wrong:")) { throw new IOException("The https URL hostname does not match the " + "Common Name (CN) on the server certificate. To disable this check " + "(NOT recommended for production) set the CXF client TLS configuration " + "property \"disableCNCheck\" to true."); } else { throw e; } } } protected void handleHeadersTrustCaching() throws IOException { // Need to set the headers before the trust decision // because they are set before the connect(). setURLRequestHeaders(outMessage); // // This point is where the trust decision is made because the // Sun implementation of URLConnection will not let us // set/addRequestProperty after a connect() call, and // makeTrustDecision needs to make a connect() call to // make sure the proper information is available. // makeTrustDecision(outMessage); // Trust is okay, set up for writing the request. // If this is a GET method we must not touch the output // stream as this automatically turns the request into a POST. // Nor it should be done in case of DELETE/HEAD/OPTIONS // - strangely, empty PUTs work ok if (!"POST".equals(connection.getRequestMethod()) && !"PUT".equals(connection.getRequestMethod())) { return; } // If we need to cache for retransmission, store data in a // CacheAndWriteOutputStream. Otherwise write directly to the output stream. if (cachingForRetransmission) { cachedStream = new CacheAndWriteOutputStream(connection.getOutputStream()); wrappedStream = cachedStream; } else { wrappedStream = connection.getOutputStream(); } } public void flush() throws IOException { if (!chunking) { super.flush(); } } /** * Perform any actions required on stream closure (handle response etc.) */ public void close() throws IOException { if (buffer != null && buffer.size() > 0) { thresholdNotReached(); LoadingByteArrayOutputStream tmp = buffer; buffer = null; super.write(tmp.getRawBytes(), 0, tmp.size()); } if (!written) { handleHeadersTrustCaching(); } super.flush(); if (!cachingForRetransmission) { super.close(); } else { cachedStream.getOut().close(); cachedStream.closeFlowthroughStream(); } try { handleResponse(); } finally { if (cachingForRetransmission && cachedStream != null) { cachedStream.close(); } } } /** * This procedure handles all retransmits, if any. * * @throws IOException */ protected void handleRetransmits() throws IOException { // If we have a cachedStream, we are caching the request. if (cachedStream != null) { if (LOG.isLoggable(Level.FINE)) { LOG.fine("Conduit \"" + getConduitName() + "\" Transmit cached message to: " + connection.getURL() + ": " + new String(cachedStream.getBytes())); } HttpURLConnection oldcon = connection; HTTPClientPolicy policy = getClient(outMessage); // Default MaxRetransmits is -1 which means unlimited. int maxRetransmits = (policy == null) ? -1 : policy.getMaxRetransmits(); // MaxRetransmits of zero means zero. if (maxRetransmits == 0) { return; } int nretransmits = 0; connection = processRetransmit(connection, outMessage, cachedStream); while (connection != oldcon) { nretransmits++; oldcon = connection; // A negative max means unlimited. if (maxRetransmits < 0 || nretransmits < maxRetransmits) { connection = processRetransmit( connection, outMessage, cachedStream); } } } } /** * This procedure is called on the close of the output stream so * we are ready to handle the response from the connection. * We may retransmit until we finally get a response. * * @throws IOException */ protected void handleResponse() throws IOException { // Process retransmits until we fall out. handleRetransmits(); if (outMessage == null || outMessage.getExchange() == null || outMessage.getExchange().isSynchronous()) { handleResponseInternal(); } else { Runnable runnable = new Runnable() { public void run() { try { handleResponseInternal(); } catch (Exception e) { Message inMessage = new MessageImpl(); inMessage.setExchange(outMessage.getExchange()); inMessage.setContent(Exception.class, e); incomingObserver.onMessage(inMessage); } } }; WorkQueueManager mgr = outMessage.getExchange().get(Bus.class) .getExtension(WorkQueueManager.class); AutomaticWorkQueue queue = mgr.getNamedWorkQueue("http-conduit"); if (queue == null) { queue = mgr.getAutomaticWorkQueue(); } queue.execute(runnable); } } protected void handleResponseInternal() throws IOException { int responseCode = connection.getResponseCode(); if ((outMessage != null) && (outMessage.getExchange() != null)) { outMessage.getExchange().put(Message.RESPONSE_CODE, responseCode); } if (LOG.isLoggable(Level.FINE)) { LOG.fine("Response Code: " + responseCode + " Conduit: " + getConduitName()); LOG.fine("Content length: " + connection.getContentLength()); Map> headerFields = connection.getHeaderFields(); if (null != headerFields) { StringBuffer buf = new StringBuffer(); buf.append("Header fields: "); buf.append(System.getProperty("line.separator")); for (String h : headerFields.keySet()) { buf.append(" "); buf.append(h); buf.append(": "); buf.append(headerFields.get(h)); buf.append(System.getProperty("line.separator")); } LOG.fine(buf.toString()); } } if (responseCode == HttpURLConnection.HTTP_NOT_FOUND) { throw new IOException(connection.getResponseMessage()); } Exchange exchange = outMessage.getExchange(); InputStream in = null; if (isOneway(exchange) || isDecoupled()) { in = getPartialResponse(connection, responseCode); if (in == null) { // oneway operation or decoupled MEP without // partial response connection.getInputStream().close(); return; } } else { //not going to be resending or anything, clear out the stuff in the out message //to free memory outMessage.removeContent(OutputStream.class); if (cachingForRetransmission) { cachedStream.close(); } cachedStream = null; } Message inMessage = new MessageImpl(); inMessage.setExchange(exchange); Map> headers = new HashMap>(); for (String key : connection.getHeaderFields().keySet()) { if (key != null) { headers.put(HttpHeaderHelper.getHeaderKey(key), connection.getHeaderFields().get(key)); } } inMessage.put(Message.PROTOCOL_HEADERS, headers); inMessage.put(Message.RESPONSE_CODE, responseCode); String ct = connection.getContentType(); inMessage.put(Message.CONTENT_TYPE, ct); String charset = HttpHeaderHelper.findCharset(ct); String normalizedEncoding = HttpHeaderHelper.mapCharset(charset); if (normalizedEncoding == null) { String m = new org.apache.cxf.common.i18n.Message("INVALID_ENCODING_MSG", LOG, charset).toString(); LOG.log(Level.WARNING, m); throw new IOException(m); } inMessage.put(Message.ENCODING, normalizedEncoding); if (maintainSession) { List cookies = connection.getHeaderFields().get("Set-Cookie"); Cookie.handleSetCookie(sessionCookies, cookies); } in = in == null ? connection.getErrorStream() == null ? connection.getInputStream() : connection.getErrorStream() : in; // if (in == null) : it's perfectly ok for non-soap http services // have no response body : those interceptors which do need it will check anyway inMessage.setContent(InputStream.class, in); incomingObserver.onMessage(inMessage); } } /** * Used to set appropriate message properties, exchange etc. * as required for an incoming decoupled response (as opposed * what's normally set by the Destination for an incoming * request). */ protected class InterposedMessageObserver implements MessageObserver { /** * Called for an incoming message. * * @param inMessage */ public void onMessage(Message inMessage) { // disposable exchange, swapped with real Exchange on correlation inMessage.setExchange(new ExchangeImpl()); inMessage.getExchange().put(Bus.class, bus); inMessage.put(DECOUPLED_CHANNEL_MESSAGE, Boolean.TRUE); // REVISIT: how to get response headers? //inMessage.put(Message.PROTOCOL_HEADERS, req.getXXX()); getSetProtocolHeaders(inMessage); inMessage.put(Message.RESPONSE_CODE, HttpURLConnection.HTTP_OK); // remove server-specific properties inMessage.remove(AbstractHTTPDestination.HTTP_REQUEST); inMessage.remove(AbstractHTTPDestination.HTTP_RESPONSE); inMessage.remove(Message.ASYNC_POST_RESPONSE_DISPATCH); //cache this inputstream since it's defer to use in case of async try { InputStream in = inMessage.getContent(InputStream.class); if (in != null) { CachedOutputStream cos = new CachedOutputStream(); IOUtils.copy(in, cos); inMessage.setContent(InputStream.class, cos.getInputStream()); } incomingObserver.onMessage(inMessage); } catch (IOException e) { e.printStackTrace(); } } } public void assertMessage(Message message) { PolicyUtils.assertClientPolicy(message, clientSidePolicy); } public boolean canAssert(QName type) { return PolicyUtils.HTTPCLIENTPOLICY_ASSERTION_QNAME.equals(type); } @Deprecated public void setBasicAuthSupplier(HttpBasicAuthSupplier basicAuthSupplier) { setAuthSupplier(basicAuthSupplier); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy