org.apache.camel.impl.engine.DefaultFluentProducerTemplate Maven / Gradle / Ivy
Show all versions of camel-base-engine Show documentation
/*
* 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.impl.engine;
import java.util.HashMap;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.concurrent.Future;
import java.util.function.Consumer;
import java.util.function.Supplier;
import org.apache.camel.CamelContext;
import org.apache.camel.CamelExecutionException;
import org.apache.camel.Endpoint;
import org.apache.camel.EndpointProducerResolver;
import org.apache.camel.Exchange;
import org.apache.camel.ExchangePattern;
import org.apache.camel.FluentProducerTemplate;
import org.apache.camel.Message;
import org.apache.camel.Processor;
import org.apache.camel.ProducerTemplate;
import org.apache.camel.support.ExchangeHelper;
import org.apache.camel.support.processor.ConvertBodyProcessor;
import org.apache.camel.support.service.ServiceHelper;
import org.apache.camel.support.service.ServiceSupport;
import org.apache.camel.util.ObjectHelper;
/**
* This implementation is based on the usage pattern, that a top level DefaultFluentProducerTemplate instance is created
* as singleton and provided to the Camel end user (such as injected into a POJO).
*
* The top level instance is then cloned once per message that is being built using the fluent method calls and then
* reset when the message has been sent.
*
* Each cloned instance is not thread-safe as its assumed that its a single thread that calls the fluent method to build
* up the message to be sent.
*/
public class DefaultFluentProducerTemplate extends ServiceSupport implements FluentProducerTemplate {
// transient state of endpoint, headers, exchange properties, variables, and body which needs to be thread local scoped to be thread-safe
private Map headers;
private Map exchangeProperties;
private Map variables;
private Object body;
private Supplier exchangeSupplier;
private Supplier processorSupplier;
private Consumer templateCustomizer;
private final CamelContext context;
private final ClassValue resultProcessors;
private Endpoint defaultEndpoint;
private int maximumCacheSize;
private boolean eventNotifierEnabled;
private volatile Endpoint endpoint;
private volatile String endpointUri;
private volatile ProducerTemplate template;
private volatile boolean cloned;
private volatile boolean useDefaultEndpoint = true;
public DefaultFluentProducerTemplate(CamelContext context) {
this.context = context;
this.eventNotifierEnabled = true;
this.resultProcessors = new ClassValue<>() {
@Override
protected Processor computeValue(Class> type) {
return new ConvertBodyProcessor(type);
}
};
}
private DefaultFluentProducerTemplate(CamelContext context,
ClassValue resultProcessors,
Endpoint defaultEndpoint, int maximumCacheSize, boolean eventNotifierEnabled,
ProducerTemplate template, Endpoint endpoint, String endpointUri) {
this.context = context;
this.resultProcessors = resultProcessors;
this.defaultEndpoint = defaultEndpoint;
this.maximumCacheSize = maximumCacheSize;
this.eventNotifierEnabled = eventNotifierEnabled;
this.template = template;
this.endpoint = endpoint;
this.endpointUri = endpointUri;
this.cloned = true;
}
private DefaultFluentProducerTemplate newClone() {
return new DefaultFluentProducerTemplate(
context, resultProcessors, defaultEndpoint, maximumCacheSize, eventNotifierEnabled, template, endpoint,
endpointUri);
}
private DefaultFluentProducerTemplate checkCloned() {
if (!cloned) {
return newClone();
} else {
return this;
}
}
@Override
public CamelContext getCamelContext() {
return context;
}
@Override
public int getCurrentCacheSize() {
if (template == null) {
return 0;
}
return template.getCurrentCacheSize();
}
@Override
public void cleanUp() {
if (template != null) {
template.cleanUp();
}
}
@Override
public void setDefaultEndpointUri(String endpointUri) {
setDefaultEndpoint(getCamelContext().getEndpoint(endpointUri));
}
@Override
public Endpoint getDefaultEndpoint() {
return defaultEndpoint;
}
@Override
public void setDefaultEndpoint(Endpoint defaultEndpoint) {
if (this.defaultEndpoint != null && isStarted()) {
throw new IllegalArgumentException("Not allowed after template has been started");
}
this.defaultEndpoint = defaultEndpoint;
}
@Override
public int getMaximumCacheSize() {
return maximumCacheSize;
}
@Override
public void setMaximumCacheSize(int maximumCacheSize) {
if (this.maximumCacheSize != 0 && isStarted()) {
throw new IllegalArgumentException("Not allowed after template has been started");
}
this.maximumCacheSize = maximumCacheSize;
}
@Override
public boolean isEventNotifierEnabled() {
return eventNotifierEnabled;
}
@Override
public void setEventNotifierEnabled(boolean eventNotifierEnabled) {
if (isStarted()) {
throw new IllegalArgumentException("Not allowed after template has been started");
}
this.eventNotifierEnabled = eventNotifierEnabled;
}
@Override
public FluentProducerTemplate withHeaders(Map headers) {
DefaultFluentProducerTemplate clone = checkCloned();
if (clone.processorSupplier != null) {
throw new IllegalArgumentException("Cannot use both withHeaders and withProcessor with FluentProducerTemplate");
}
Map map = clone.headers;
if (map == null) {
map = new LinkedHashMap<>();
clone.headers = map;
}
map.putAll(headers);
return clone;
}
@Override
public FluentProducerTemplate withHeader(String key, Object value) {
DefaultFluentProducerTemplate clone = checkCloned();
if (clone.processorSupplier != null) {
throw new IllegalArgumentException("Cannot use both withHeader and withProcessor with FluentProducerTemplate");
}
Map map = clone.headers;
if (map == null) {
map = new LinkedHashMap<>();
clone.headers = map;
}
map.put(key, value);
return clone;
}
@Override
public FluentProducerTemplate withExchangeProperties(Map properties) {
DefaultFluentProducerTemplate clone = checkCloned();
if (clone.processorSupplier != null) {
throw new IllegalArgumentException(
"Cannot use both withExchangeProperties and withProcessor with FluentProducerTemplate");
}
Map map = clone.exchangeProperties;
if (map == null) {
map = new LinkedHashMap<>();
clone.exchangeProperties = map;
}
map.putAll(properties);
return clone;
}
@Override
public FluentProducerTemplate withExchangeProperty(String key, Object value) {
DefaultFluentProducerTemplate clone = checkCloned();
if (clone.processorSupplier != null) {
throw new IllegalArgumentException(
"Cannot use both withExchangeProperty and withProcessor with FluentProducerTemplate");
}
Map map = clone.exchangeProperties;
if (map == null) {
map = new LinkedHashMap<>();
clone.exchangeProperties = map;
}
map.put(key, value);
return clone;
}
@Override
public FluentProducerTemplate withVariables(Map variables) {
DefaultFluentProducerTemplate clone = checkCloned();
if (clone.processorSupplier != null) {
throw new IllegalArgumentException("Cannot use both withVariables and withProcessor with FluentProducerTemplate");
}
Map map = clone.variables;
if (map == null) {
map = new LinkedHashMap<>();
clone.variables = map;
}
map.putAll(variables);
return clone;
}
@Override
public FluentProducerTemplate withVariable(String key, Object value) {
DefaultFluentProducerTemplate clone = checkCloned();
if (clone.processorSupplier != null) {
throw new IllegalArgumentException("Cannot use both withVariable and withProcessor with FluentProducerTemplate");
}
Map map = clone.variables;
if (map == null) {
map = new LinkedHashMap<>();
clone.variables = map;
}
map.put(key, value);
return clone;
}
@Override
public FluentProducerTemplate withBody(Object body) {
DefaultFluentProducerTemplate clone = checkCloned();
if (clone.processorSupplier != null) {
throw new IllegalArgumentException("Cannot use both withBody and withProcessor with FluentProducerTemplate");
}
clone.body = body;
return clone;
}
@Override
public FluentProducerTemplate withBodyAs(Object body, Class> type) {
DefaultFluentProducerTemplate clone = checkCloned();
if (clone.processorSupplier != null) {
throw new IllegalArgumentException("Cannot use both withBodyAs and withProcessor with FluentProducerTemplate");
}
clone.body = type != null
? clone.context.getTypeConverter().convertTo(type, body)
: body;
return clone;
}
@Override
public FluentProducerTemplate withDefaultEndpoint(String endpointUri) {
if (cloned) {
throw new IllegalArgumentException("Default endpoint must be set before template has been used");
}
this.defaultEndpoint = getCamelContext().getEndpoint(endpointUri);
return this;
}
@Override
public FluentProducerTemplate withDefaultEndpoint(EndpointProducerResolver resolver) {
if (cloned) {
throw new IllegalArgumentException("Default endpoint must be set before template has been used");
}
this.defaultEndpoint = resolver.resolve(getCamelContext());
return this;
}
@Override
public FluentProducerTemplate withDefaultEndpoint(Endpoint endpoint) {
if (cloned) {
throw new IllegalArgumentException("Default endpoint must be set before template has been used");
}
this.defaultEndpoint = endpoint;
return this;
}
@Override
public FluentProducerTemplate withTemplateCustomizer(final Consumer templateCustomizer) {
if (this.templateCustomizer != null && isStarted()) {
throw new IllegalArgumentException("Not allowed after template has been started");
}
this.templateCustomizer = templateCustomizer;
if (template != null) {
// need to re-initialize template since we have a customizer
ServiceHelper.stopService(template);
templateCustomizer.accept(template);
ServiceHelper.startService(template);
}
return this;
}
@Override
public FluentProducerTemplate withExchange(final Exchange exchange) {
return withExchange(() -> exchange);
}
@Override
public FluentProducerTemplate withExchange(final Supplier exchangeSupplier) {
DefaultFluentProducerTemplate clone = checkCloned();
clone.exchangeSupplier = exchangeSupplier;
return clone;
}
@Override
public FluentProducerTemplate withProcessor(final Processor processor) {
return withProcessor(() -> processor);
}
@Override
public FluentProducerTemplate withProcessor(final Supplier processorSupplier) {
DefaultFluentProducerTemplate clone = checkCloned();
if (clone.body != null) {
throw new IllegalArgumentException("Cannot use both withBody and withProcessor with FluentProducerTemplate");
}
clone.processorSupplier = processorSupplier;
return clone;
}
@Override
public FluentProducerTemplate to(String endpointUri) {
DefaultFluentProducerTemplate clone = checkCloned();
clone.useDefaultEndpoint = false;
clone.endpointUri = endpointUri;
clone.endpoint = context.getEndpoint(endpointUri);
return clone;
}
@Override
public FluentProducerTemplate to(Endpoint endpoint) {
DefaultFluentProducerTemplate clone = checkCloned();
clone.useDefaultEndpoint = false;
clone.endpoint = endpoint;
return clone;
}
// ************************
// REQUEST
// ************************
@Override
public Object request() throws CamelExecutionException {
return request(Object.class);
}
@Override
@SuppressWarnings("unchecked")
public T request(Class type) throws CamelExecutionException {
if (exchangeSupplier != null && exchangeSupplier.get() != null) {
throw new IllegalArgumentException(
"withExchange not supported on FluentProducerTemplate.request method. Use send method instead.");
}
DefaultFluentProducerTemplate clone = checkCloned();
// Determine the target endpoint
final Endpoint target = clone.target();
// Create the default processor if not provided.
Processor processor = clone.processorSupplier != null ? clone.processorSupplier.get() : null;
final Processor processorSupplier = processor != null ? processor : clone.defaultProcessor();
T result;
if (type == Exchange.class) {
result = (T) clone.template().request(target, processorSupplier);
} else if (type == Message.class) {
Exchange exchange = clone.template().request(target, processorSupplier);
result = (T) exchange.getMessage();
} else {
Exchange exchange = clone.template().send(
target,
ExchangePattern.InOut,
processorSupplier,
clone.resultProcessors.get(type));
result = clone.context.getTypeConverter().convertTo(
type,
ExchangeHelper.extractResultBody(exchange, exchange.getPattern()));
}
// reset cloned flag so when we use it again it has to set values again
cloned = false;
useDefaultEndpoint = true;
return result;
}
@Override
public Future