org.apache.camel.component.spring.ws.SpringWebserviceProducer Maven / Gradle / Ivy
/*
* 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.camel.component.spring.ws;
import java.io.IOException;
import java.net.HttpURLConnection;
import java.net.URI;
import java.security.GeneralSecurityException;
import java.util.Iterator;
import javax.net.ssl.HttpsURLConnection;
import javax.net.ssl.SSLContext;
import javax.xml.namespace.QName;
import javax.xml.transform.Source;
import javax.xml.transform.TransformerException;
import org.apache.camel.CamelContext;
import org.apache.camel.Endpoint;
import org.apache.camel.Exchange;
import org.apache.camel.Message;
import org.apache.camel.RuntimeCamelException;
import org.apache.camel.attachment.AttachmentMessage;
import org.apache.camel.support.DefaultProducer;
import org.apache.camel.support.ExchangeHelper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.ws.WebServiceMessage;
import org.springframework.ws.client.core.WebServiceMessageCallback;
import org.springframework.ws.client.core.WebServiceTemplate;
import org.springframework.ws.mime.Attachment;
import org.springframework.ws.soap.SoapHeader;
import org.springframework.ws.soap.SoapHeaderElement;
import org.springframework.ws.soap.SoapMessage;
import org.springframework.ws.soap.addressing.client.ActionCallback;
import org.springframework.ws.soap.addressing.core.EndpointReference;
import org.springframework.ws.soap.client.core.SoapActionCallback;
import org.springframework.ws.transport.WebServiceConnection;
import org.springframework.ws.transport.WebServiceMessageSender;
import org.springframework.ws.transport.http.AbstractHttpWebServiceMessageSender;
import org.springframework.ws.transport.http.HttpComponentsMessageSender;
import org.springframework.ws.transport.http.HttpUrlConnection;
import org.springframework.ws.transport.http.HttpUrlConnectionMessageSender;
import static org.apache.camel.component.spring.ws.SpringWebserviceHelper.toResult;
public class SpringWebserviceProducer extends DefaultProducer {
private static final Logger LOG = LoggerFactory.getLogger(SpringWebserviceProducer.class);
public SpringWebserviceProducer(Endpoint endpoint) {
super(endpoint);
prepareMessageSenders(getEndpoint().getConfiguration());
}
@Override
public SpringWebserviceEndpoint getEndpoint() {
return (SpringWebserviceEndpoint) super.getEndpoint();
}
@Override
public void process(Exchange exchange) throws Exception {
// Let Camel TypeConverter hierarchy handle the conversion of XML messages to Source objects
Source sourcePayload = exchange.getIn().getMandatoryBody(Source.class);
// Extract optional headers
String endpointUriHeader = exchange.getIn().getHeader(SpringWebserviceConstants.SPRING_WS_ENDPOINT_URI, String.class);
String soapActionHeader = exchange.getIn().getHeader(SpringWebserviceConstants.SPRING_WS_SOAP_ACTION, String.class);
URI wsAddressingActionHeader
= exchange.getIn().getHeader(SpringWebserviceConstants.SPRING_WS_ADDRESSING_ACTION, URI.class);
URI wsReplyToHeader
= exchange.getIn().getHeader(SpringWebserviceConstants.SPRING_WS_ADDRESSING_PRODUCER_REPLY_TO, URI.class);
URI wsFaultToHeader
= exchange.getIn().getHeader(SpringWebserviceConstants.SPRING_WS_ADDRESSING_PRODUCER_FAULT_TO, URI.class);
Source soapHeaderSource = exchange.getIn().getHeader(SpringWebserviceConstants.SPRING_WS_SOAP_HEADER, Source.class);
WebServiceMessageCallback callback = new DefaultWebserviceMessageCallback(
soapActionHeader, wsAddressingActionHeader,
wsReplyToHeader, wsFaultToHeader, soapHeaderSource, getEndpoint().getConfiguration(), exchange);
if (endpointUriHeader == null) {
endpointUriHeader = getEndpoint().getConfiguration().getWebServiceTemplate().getDefaultUri();
}
getEndpoint().getConfiguration().getWebServiceTemplate().sendAndReceive(endpointUriHeader,
new WebServiceMessageCallback() {
@Override
public void doWithMessage(WebServiceMessage requestMessage) throws IOException, TransformerException {
toResult(sourcePayload, requestMessage.getPayloadResult());
callback.doWithMessage(requestMessage);
}
}, new WebServiceMessageCallback() {
@Override
public void doWithMessage(WebServiceMessage responseMessage) throws IOException, TransformerException {
SoapMessage soapMessage = (SoapMessage) responseMessage;
if (ExchangeHelper.isOutCapable(exchange)) {
exchange.getOut().copyFromWithNewBody(exchange.getIn(), soapMessage.getPayloadSource());
populateHeaderAndAttachmentsFromResponse(exchange.getOut(AttachmentMessage.class), soapMessage);
} else {
exchange.getIn().setBody(soapMessage.getPayloadSource());
populateHeaderAndAttachmentsFromResponse(exchange.getIn(AttachmentMessage.class), soapMessage);
}
}
});
}
/**
* Populates soap message headers and attachments from soap response
*/
private void populateHeaderAndAttachmentsFromResponse(AttachmentMessage inOrOut, SoapMessage soapMessage) {
if (soapMessage.getSoapHeader() != null && getEndpoint().getConfiguration().isAllowResponseHeaderOverride()) {
populateMessageHeaderFromResponse(inOrOut, soapMessage.getSoapHeader());
}
if (soapMessage.getAttachments() != null && getEndpoint().getConfiguration().isAllowResponseAttachmentOverride()) {
populateMessageAttachmentsFromResponse(inOrOut, soapMessage.getAttachments());
}
}
/**
* Populates message headers from soapHeader response
*/
private void populateMessageHeaderFromResponse(Message message, SoapHeader soapHeader) {
message.setHeader(SpringWebserviceConstants.SPRING_WS_SOAP_HEADER, soapHeader.getSource());
// Set header values for the soap header attributes
Iterator attIter = soapHeader.getAllAttributes();
while (attIter.hasNext()) {
QName name = attIter.next();
message.getHeaders().put(name.getLocalPart(), soapHeader.getAttributeValue(name));
}
// Set header values for the soap header elements
Iterator elementIter = soapHeader.examineAllHeaderElements();
while (elementIter.hasNext()) {
SoapHeaderElement element = elementIter.next();
QName name = element.getName();
message.getHeaders().put(name.getLocalPart(), element);
}
}
/**
* Populates message attachments from soap response attachments
*/
private void populateMessageAttachmentsFromResponse(AttachmentMessage inOrOut, Iterator attachments) {
while (attachments.hasNext()) {
Attachment attachment = attachments.next();
inOrOut.addAttachment(attachment.getContentId(), attachment.getDataHandler());
}
}
private void prepareMessageSenders(SpringWebserviceConfiguration configuration) {
// Skip this whole thing if none of the relevant config options are set.
if (!(configuration.getTimeout() > -1) && configuration.getSslContextParameters() == null) {
return;
}
WebServiceTemplate webServiceTemplate = configuration.getWebServiceTemplate();
WebServiceMessageSender[] messageSenders = webServiceTemplate.getMessageSenders();
for (int i = 0; i < messageSenders.length; i++) {
WebServiceMessageSender messageSender = messageSenders[i];
if (messageSender instanceof HttpComponentsMessageSender) {
if (configuration.getSslContextParameters() != null) {
LOG.warn("Not applying SSLContextParameters based configuration to HttpComponentsMessageSender. "
+ "If you are using this MessageSender, which you are not by default, you will need "
+ "to configure SSL using the Commons HTTP 3.x Protocol registry.");
}
if (configuration.getTimeout() > -1) {
if (messageSender.getClass().equals(HttpComponentsMessageSender.class)) {
((HttpComponentsMessageSender) messageSender).setReadTimeout(configuration.getTimeout());
} else {
LOG.warn("Not applying timeout configuration to HttpComponentsMessageSender based implementation. "
+ "You are using what appears to be a custom MessageSender, which you are not doing by default. "
+ "You will need configure timeout on your own.");
}
}
} else if (messageSender.getClass().equals(HttpUrlConnectionMessageSender.class)) {
// Only if exact match denoting likely use of default configuration. We don't want to get
// sub-classes that might have been otherwise injected.
messageSenders[i] = new AbstractHttpWebServiceMessageSenderDecorator(
(HttpUrlConnectionMessageSender) messageSender, configuration, getEndpoint().getCamelContext());
} else {
// For example this will be the case during unit-testing with the net.javacrumbs.spring-ws-test API
LOG.warn("Ignoring the timeout and SSLContextParameters options for {}. You will need to configure "
+ "these options directly on your custom configured WebServiceMessageSender",
messageSender);
}
}
}
/**
* A decorator of {@link HttpUrlConnectionMessageSender} instances that can apply configuration options from the
* Camel component/endpoint configuration without replacing the actual implementation which may actually be an
* end-user implementation and not one of the built-in implementations.
*/
protected static final class AbstractHttpWebServiceMessageSenderDecorator extends AbstractHttpWebServiceMessageSender {
private final AbstractHttpWebServiceMessageSender delegate;
private final SpringWebserviceConfiguration configuration;
private final CamelContext camelContext;
private SSLContext sslContext;
public AbstractHttpWebServiceMessageSenderDecorator(AbstractHttpWebServiceMessageSender delegate,
SpringWebserviceConfiguration configuration,
CamelContext camelContext) {
this.delegate = delegate;
this.configuration = configuration;
this.camelContext = camelContext;
}
@Override
public WebServiceConnection createConnection(URI uri) throws IOException {
WebServiceConnection wsc = delegate.createConnection(uri);
if (wsc instanceof HttpUrlConnection) {
HttpURLConnection connection = ((HttpUrlConnection) wsc).getConnection();
if (configuration.getTimeout() > -1) {
connection.setReadTimeout(configuration.getTimeout());
}
if (configuration.getSslContextParameters() != null && connection instanceof HttpsURLConnection) {
try {
synchronized (this) {
if (sslContext == null) {
sslContext = configuration.getSslContextParameters().createSSLContext(camelContext);
}
}
} catch (GeneralSecurityException e) {
throw new RuntimeCamelException("Error creating SSLContext based on SSLContextParameters.", e);
}
((HttpsURLConnection) connection).setSSLSocketFactory(sslContext.getSocketFactory());
}
} else {
throw new RuntimeCamelException(
"Unsupported delegate. Delegate must return a org.springframework.ws.transport.http.HttpUrlConnection. Found "
+ wsc.getClass());
}
return wsc;
}
@Override
public boolean isAcceptGzipEncoding() {
return delegate.isAcceptGzipEncoding();
}
@Override
public void setAcceptGzipEncoding(boolean acceptGzipEncoding) {
delegate.setAcceptGzipEncoding(acceptGzipEncoding);
}
@Override
public boolean supports(URI uri) {
return delegate.supports(uri);
}
}
protected static class DefaultWebserviceMessageCallback implements WebServiceMessageCallback {
private final String soapActionHeader;
private final URI wsAddressingActionHeader;
private final URI wsReplyToHeader;
private final URI wsFaultToHeader;
private final Source soapHeaderSource;
private final SpringWebserviceConfiguration configuration;
private final Exchange exchange;
public DefaultWebserviceMessageCallback(String soapAction, URI wsAddressingAction, URI wsReplyTo, URI wsFaultTo,
Source soapHeaderSource,
SpringWebserviceConfiguration configuration, Exchange exchange) {
this.soapActionHeader = soapAction;
this.wsAddressingActionHeader = wsAddressingAction;
this.wsReplyToHeader = wsReplyTo;
this.wsFaultToHeader = wsFaultTo;
this.soapHeaderSource = soapHeaderSource;
this.configuration = configuration;
this.exchange = exchange;
}
@Override
public void doWithMessage(WebServiceMessage message) throws IOException, TransformerException {
// Add SoapAction to webservice request. Note that exchange header
// takes precedence over endpoint option
String soapAction = soapActionHeader != null ? soapActionHeader : configuration.getSoapAction();
if (soapAction != null) {
new SoapActionCallback(soapAction).doWithMessage(message);
}
// Add WS-Addressing Action to webservice request (the WS-Addressing
// 'to' header will default to the URL of the connection).
// Note that exchange header takes precedence over endpoint option
URI wsAddressingAction
= wsAddressingActionHeader != null ? wsAddressingActionHeader : configuration.getWsAddressingAction();
URI wsReplyTo = wsReplyToHeader != null ? wsReplyToHeader : configuration.getReplyTo();
URI wsFaultTo = wsFaultToHeader != null ? wsFaultToHeader : configuration.getFaultTo();
// Create the SOAP header
if (soapHeaderSource != null) {
SoapHeader header = ((SoapMessage) message).getSoapHeader();
toResult(soapHeaderSource, header.getResult());
}
if (wsAddressingAction != null) {
ActionCallback actionCallback = new ActionCallback(wsAddressingAction);
if (configuration.getMessageIdStrategy() != null) {
actionCallback.setMessageIdStrategy(configuration.getMessageIdStrategy());
}
if (wsReplyTo != null) {
actionCallback.setReplyTo(new EndpointReference(wsReplyTo));
}
if (wsFaultTo != null) {
actionCallback.setFaultTo(new EndpointReference(wsFaultTo));
}
actionCallback.doWithMessage(message);
}
configuration.getMessageFilter().filterProducer(exchange, message);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy