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

org.apache.camel.support.ExchangeHelper Maven / Gradle / Ivy

The 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.camel.support;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.nio.channels.ReadableByteChannel;
import java.nio.charset.Charset;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.StandardOpenOption;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.CopyOnWriteArrayList;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;

import org.apache.camel.CamelContext;
import org.apache.camel.CamelExecutionException;
import org.apache.camel.Endpoint;
import org.apache.camel.Exchange;
import org.apache.camel.ExchangeExtension;
import org.apache.camel.ExchangePattern;
import org.apache.camel.ExchangePropertyKey;
import org.apache.camel.Message;
import org.apache.camel.MessageHistory;
import org.apache.camel.NoSuchBeanException;
import org.apache.camel.NoSuchEndpointException;
import org.apache.camel.NoSuchHeaderException;
import org.apache.camel.NoSuchPropertyException;
import org.apache.camel.NoTypeConversionAvailableException;
import org.apache.camel.Route;
import org.apache.camel.RuntimeCamelException;
import org.apache.camel.StreamCache;
import org.apache.camel.TypeConversionException;
import org.apache.camel.VariableAware;
import org.apache.camel.WrappedFile;
import org.apache.camel.spi.NormalizedEndpointUri;
import org.apache.camel.spi.UnitOfWork;
import org.apache.camel.spi.VariableRepository;
import org.apache.camel.spi.VariableRepositoryFactory;
import org.apache.camel.util.IOHelper;
import org.apache.camel.util.ObjectHelper;
import org.apache.camel.util.Scanner;
import org.apache.camel.util.StringHelper;

/**
 * Some helper methods for working with {@link Exchange} objects
 */
public final class ExchangeHelper {

    private static final String DEFAULT_CHARSET_NAME
            = ObjectHelper.getSystemProperty(Exchange.DEFAULT_CHARSET_PROPERTY, "UTF-8");
    private static final Charset DEFAULT_CHARSET = Charset.forName(DEFAULT_CHARSET_NAME);

    private static Exchange DUMMY;

    /**
     * Utility classes should not have a public constructor.
     */
    private ExchangeHelper() {
    }

    /**
     * Gets a singleton dummy exchange used for special purposes only.
     */
    public static Exchange getDummy(CamelContext context) {
        // we dont worry about thread-safety
        if (DUMMY == null) {
            DUMMY = new DefaultExchange(context);
        }
        return DUMMY;
    }

    /**
     * Extracts the Exchange.BINDING of the given type or null if not present
     *
     * @param  exchange the message exchange
     * @param  type     the expected binding type
     * @return          the binding object of the given type or null if it could not be found or converted
     */
    public static  T getBinding(Exchange exchange, Class type) {
        return exchange != null ? exchange.getProperty(Exchange.BINDING, type) : null;
    }

    /**
     * Attempts to resolve the endpoint for the given value
     *
     * @param  exchange                the message exchange being processed
     * @param  value                   the value which can be an {@link Endpoint} or an object which provides a String
     *                                 representation of an endpoint via {@link #toString()}
     * @return                         the endpoint
     * @throws NoSuchEndpointException if the endpoint cannot be resolved
     */
    public static Endpoint resolveEndpoint(Exchange exchange, Object value) throws NoSuchEndpointException {
        return resolveEndpoint(exchange.getContext(), value);
    }

    /**
     * Attempts to resolve the endpoint for the given value
     *
     * @param  context                 the camel context
     * @param  value                   the value which can be an {@link Endpoint} or an object which provides a String
     *                                 representation of an endpoint via {@link #toString()}
     * @return                         the endpoint
     * @throws NoSuchEndpointException if the endpoint cannot be resolved
     */
    public static Endpoint resolveEndpoint(CamelContext context, Object value) throws NoSuchEndpointException {
        if (value == null) {
            throw new NoSuchEndpointException("null");
        }
        Endpoint endpoint;
        if (value instanceof Endpoint ep) {
            endpoint = ep;
        } else if (value instanceof NormalizedEndpointUri nu) {
            endpoint = CamelContextHelper.getMandatoryEndpoint(context, nu);
        } else {
            String uri = value.toString().trim();
            endpoint = CamelContextHelper.getMandatoryEndpoint(context, uri);
        }
        return endpoint;
    }

    /**
     * Attempts to resolve the endpoint (prototype scope) for the given value
     *
     * @param  exchange                the message exchange being processed
     * @param  value                   the value which can be an {@link Endpoint} or an object which provides a String
     *                                 representation of an endpoint via {@link #toString()}
     * @return                         the endpoint
     * @throws NoSuchEndpointException if the endpoint cannot be resolved
     */
    public static Endpoint resolvePrototypeEndpoint(Exchange exchange, Object value) throws NoSuchEndpointException {
        return resolvePrototypeEndpoint(exchange.getContext(), value);
    }

    /**
     * Attempts to resolve the endpoint (prototype scope) for the given value
     *
     * @param  context                 the camel context
     * @param  value                   the value which can be an {@link Endpoint} or an object which provides a String
     *                                 representation of an endpoint via {@link #toString()}
     * @return                         the endpoint
     * @throws NoSuchEndpointException if the endpoint cannot be resolved
     */
    public static Endpoint resolvePrototypeEndpoint(CamelContext context, Object value) throws NoSuchEndpointException {
        if (value == null) {
            throw new NoSuchEndpointException("null");
        }
        Endpoint endpoint;
        if (value instanceof Endpoint ep) {
            endpoint = ep;
        } else if (value instanceof NormalizedEndpointUri nu) {
            endpoint = CamelContextHelper.getMandatoryPrototypeEndpoint(context, nu);
        } else {
            String uri = value.toString().trim();
            endpoint = CamelContextHelper.getMandatoryPrototypeEndpoint(context, uri);
        }
        return endpoint;
    }

    /**
     * Gets the mandatory property of the exchange of the correct type
     *
     * @param  exchange                the exchange
     * @param  propertyName            the property name
     * @param  type                    the type
     * @return                         the property value
     * @throws TypeConversionException is thrown if error during type conversion
     * @throws NoSuchPropertyException is thrown if no property exists
     */
    public static  T getMandatoryProperty(Exchange exchange, String propertyName, Class type)
            throws NoSuchPropertyException {
        T result = exchange.getProperty(propertyName, type);
        if (result != null) {
            return result;
        }
        throw new NoSuchPropertyException(exchange, propertyName, type);
    }

    /**
     * Gets the mandatory inbound header of the correct type
     *
     * @param  exchange                the exchange
     * @param  headerName              the header name
     * @param  type                    the type
     * @return                         the header value
     * @throws TypeConversionException is thrown if error during type conversion
     * @throws NoSuchHeaderException   is thrown if no headers exists
     */
    public static  T getMandatoryHeader(Exchange exchange, String headerName, Class type)
            throws TypeConversionException, NoSuchHeaderException {
        T answer = exchange.getIn().getHeader(headerName, type);
        if (answer == null) {
            throw new NoSuchHeaderException(exchange, headerName, type);
        }
        return answer;
    }

    /**
     * Gets the mandatory inbound header of the correct type
     *
     * @param  message                 the message
     * @param  headerName              the header name
     * @param  type                    the type
     * @return                         the header value
     * @throws TypeConversionException is thrown if error during type conversion
     * @throws NoSuchHeaderException   is thrown if no headers exists
     */
    public static  T getMandatoryHeader(Message message, String headerName, Class type)
            throws TypeConversionException, NoSuchHeaderException {
        T answer = message.getHeader(headerName, type);
        if (answer == null) {
            throw new NoSuchHeaderException(message.getExchange(), headerName, type);
        }
        return answer;
    }

    /**
     * Gets an header or property of the correct type
     *
     * @param  exchange                the exchange
     * @param  name                    the name of the header or the property
     * @param  type                    the type
     * @return                         the header or property value
     * @throws TypeConversionException is thrown if error during type conversion
     * @throws NoSuchHeaderException   is thrown if no headers exists
     */
    public static  T getHeaderOrProperty(Exchange exchange, String name, Class type) throws TypeConversionException {
        T answer = exchange.getIn().getHeader(name, type);
        if (answer == null) {
            answer = exchange.getProperty(name, type);
        }
        return answer;
    }

    /**
     * Converts the value to the given expected type or throws an exception
     *
     * @return                                     the converted value
     * @throws TypeConversionException             is thrown if error during type conversion
     * @throws NoTypeConversionAvailableException} if no type converters exists to convert to the given type
     */
    public static  T convertToMandatoryType(Exchange exchange, Class type, Object value)
            throws TypeConversionException, NoTypeConversionAvailableException {
        return exchange.getContext().getTypeConverter().mandatoryConvertTo(type, exchange, value);
    }

    /**
     * Converts the value to the given expected type
     *
     * @return                                          the converted value
     * @throws org.apache.camel.TypeConversionException is thrown if error during type conversion
     */
    public static  T convertToType(Exchange exchange, Class type, Object value) throws TypeConversionException {
        return exchange.getContext().getTypeConverter().convertTo(type, exchange, value);
    }

    /**
     * Creates a new instance and copies from the current message exchange so that it can be forwarded to another
     * destination as a new instance. Unlike regular copy this operation will not share the same
     * {@link org.apache.camel.spi.UnitOfWork} so its should be used for async messaging, where the original and copied
     * exchange are independent.
     *
     * @param exchange original copy of the exchange
     * @param handover whether the on completion callbacks should be handed over to the new copy.
     */
    public static Exchange createCorrelatedCopy(Exchange exchange, boolean handover) {
        return createCorrelatedCopy(exchange, handover, false);
    }

    /**
     * Creates a new instance and copies from the current message exchange so that it can be forwarded to another
     * destination as a new instance. Unlike regular copy this operation will not share the same
     * {@link org.apache.camel.spi.UnitOfWork} so its should be used for async messaging, where the original and copied
     * exchange are independent.
     *
     * @param exchange         original copy of the exchange
     * @param handover         whether the on completion callbacks should be handed over to the new copy.
     * @param useSameMessageId whether to use same message id on the copy message.
     */
    public static Exchange createCorrelatedCopy(Exchange exchange, boolean handover, boolean useSameMessageId) {
        String id = exchange.getExchangeId();

        // make sure to do a safe copy as the correlated copy can be routed independently of the source.
        Exchange copy = exchange.copy();
        // do not reuse message id on copy
        if (!useSameMessageId) {
            if (copy.hasOut()) {
                copy.getOut().setMessageId(null);
            }
            copy.getIn().setMessageId(null);
        }
        // do not share the unit of work
        copy.getExchangeExtension().setUnitOfWork(null);
        if (handover) {
            // Need to hand over the completion for async invocation
            exchange.getExchangeExtension().handoverCompletions(copy);
        }
        // set a correlation id so we can track back the original exchange
        copy.setProperty(ExchangePropertyKey.CORRELATION_ID, id);
        return copy;
    }

    /**
     * Creates a new instance and copies from the current message exchange so that it can be forwarded to another
     * destination as a new instance.
     *
     * @param  exchange           original copy of the exchange
     * @param  preserveExchangeId whether or not the exchange id should be preserved
     * @return                    the copy
     */
    public static Exchange createCopy(Exchange exchange, boolean preserveExchangeId) {
        Exchange copy = exchange.copy();
        if (preserveExchangeId) {
            // must preserve exchange id
            copy.setExchangeId(exchange.getExchangeId());
        }
        return copy;
    }

    /**
     * Copies the results of a message exchange from the source exchange to the result exchange which will copy the
     * message contents, exchange properties and the exception. Notice the {@link ExchangePattern} is not
     * copied/altered.
     *
     * @param target the target exchange which will have the output and error state added (result)
     * @param source the source exchange which is not modified
     */
    public static void copyResults(Exchange target, Exchange source) {
        doCopyResults(target, source, false);
    }

    /**
     * Copies the source exchange to target exchange preserving the {@link ExchangePattern} of
     * target.
     *
     * @param target the target exchange which will have the output and error state added (result)
     * @param source source exchange.
     */
    public static void copyResultsPreservePattern(Exchange target, Exchange source) {
        doCopyResults(target, source, true);
    }

    private static void doCopyResults(Exchange result, Exchange source, boolean preserverPattern) {
        if (result == source) {
            copyFromOutMessageConditionally(result, source);
            return;
        }

        if (source.hasOut()) {
            copyFromOutMessage(result, source, preserverPattern);
        } else {
            copyFromInMessage(result, source, preserverPattern);
        }

        if (source.hasProperties()) {
            result.getProperties().putAll(source.getProperties());
        }

        final ExchangeExtension sourceExtension = source.getExchangeExtension();
        sourceExtension.copyInternalProperties(result);

        final ExchangeExtension resultExtension = result.getExchangeExtension();
        sourceExtension.copySafeCopyPropertiesTo(resultExtension);

        // copy over state
        result.setRouteStop(source.isRouteStop());
        result.setRollbackOnly(source.isRollbackOnly());
        result.setRollbackOnlyLast(source.isRollbackOnlyLast());
        resultExtension.setNotifyEvent(sourceExtension.isNotifyEvent());
        resultExtension.setRedeliveryExhausted(sourceExtension.isRedeliveryExhausted());
        resultExtension.setErrorHandlerHandled(sourceExtension.getErrorHandlerHandled());
        resultExtension.setFailureHandled(sourceExtension.isFailureHandled());

        result.setException(source.getException());
    }

    private static void copyFromOutMessageConditionally(Exchange result, Exchange source) {
        // we just need to ensure MEP is as expected (eg copy result to OUT if out capable)
        // and the result is not failed
        if (result.getPattern().isOutCapable() && !result.hasOut() && !result.isFailed()) {
            // copy IN to OUT as we expect a OUT response
            result.getOut().copyFrom(source.getIn());
        }
    }

    private static void copyFromInMessage(Exchange result, Exchange source, boolean preserverPattern) {
        // no results so lets copy the last input
        // as the final processor on a pipeline might not
        // have created any OUT; such as a mock:endpoint
        // so lets assume the last IN is the OUT
        if (!preserverPattern && result.getPattern().isOutCapable()) {
            // only set OUT if its OUT capable
            result.getOut().copyFrom(source.getIn());
        } else {
            // if not replace IN instead to keep the MEP
            result.getIn().copyFrom(source.getIn());
            // clear any existing OUT as the result is on the IN
            if (result.hasOut()) {
                result.setOut(null);
            }
        }
    }

    private static void copyFromOutMessage(Exchange result, Exchange source, boolean preserverPattern) {
        if (preserverPattern) {
            // exchange pattern sensitive
            Message resultMessage = getResultMessage(result);
            resultMessage.copyFrom(source.getOut());
        } else {
            result.getOut().copyFrom(source.getOut());
        }
    }

    /**
     * Returns the message where to write results in an exchange-pattern-sensitive way.
     *
     * @param  exchange message exchange.
     * @return          result message.
     */
    public static Message getResultMessage(Exchange exchange) {
        if (exchange.getPattern().isOutCapable()) {
            return exchange.getOut();
        } else {
            return exchange.getIn();
        }
    }

    /**
     * Returns true if the given exchange pattern (if defined) can support OUT messages
     *
     * @param  exchange the exchange to interrogate
     * @return          true if the exchange is defined as an {@link ExchangePattern} which supports OUT messages
     */
    public static boolean isOutCapable(Exchange exchange) {
        ExchangePattern pattern = exchange.getPattern();
        return pattern != null && pattern.isOutCapable();
    }

    /**
     * Creates a new instance of the given type from the injector
     *
     * @param  exchange the exchange
     * @param  type     the given type
     * @return          the created instance of the given type
     */
    public static  T newInstance(Exchange exchange, Class type) {
        return exchange.getContext().getInjector().newInstance(type);
    }

    /**
     * Creates a Map of the variables which are made available to a script or template
     *
     * @param  exchange           the exchange to make available
     * @param  allowContextMapAll whether to allow access to all context map or not (prefer to use false due to security
     *                            reasons preferred to only allow access to body/headers)
     * @return                    a Map populated with the require variables
     */
    public static Map createVariableMap(Exchange exchange, boolean allowContextMapAll) {
        Map answer = new HashMap<>();
        populateVariableMap(exchange, answer, allowContextMapAll);
        return answer;
    }

    /**
     * Populates the Map with the variables which are made available to a script or template
     *
     * @param exchange           the exchange to make available
     * @param map                the map to populate
     * @param allowContextMapAll whether to allow access to all context map or not (prefer to use false due to security
     *                           reasons preferred to only allow access to body/headers)
     */
    public static void populateVariableMap(Exchange exchange, Map map, boolean allowContextMapAll) {
        Message in = exchange.getIn();
        map.put("body", in.getBody());
        map.put("header", in.getHeaders());
        map.put("headers", in.getHeaders());
        var v = exchange.getVariables();
        map.put("variable", v);
        map.put("variables", v);
        map.put("exception", LanguageHelper.exception(exchange));
        if (allowContextMapAll) {
            map.put("in", in);
            map.put("request", in);
            map.put("exchange", exchange);
            var p = exchange.getAllProperties();
            map.put("exchangeProperty", p);
            map.put("exchangeProperties", p);
            if (isOutCapable(exchange)) {
                // if we are out capable then set out and response as well
                // however only grab OUT if it exists, otherwise reuse IN
                // this prevents side effects to alter the Exchange if we force creating an OUT message
                Message msg = exchange.getMessage();
                map.put("out", msg);
                map.put("response", msg);
            }
            map.put("camelContext", exchange.getContext());
        }
    }

    /**
     * Returns the MIME content type on the input message or null if one is not defined
     *
     * @param  exchange the exchange
     * @return          the MIME content type
     */
    public static String getContentType(Exchange exchange) {
        return MessageHelper.getContentType(exchange.getIn());
    }

    /**
     * Returns the MIME content encoding on the input message or null if one is not defined
     *
     * @param  exchange the exchange
     * @return          the MIME content encoding
     */
    public static String getContentEncoding(Exchange exchange) {
        return MessageHelper.getContentEncoding(exchange.getIn());
    }

    /**
     * Performs a lookup in the registry of the mandatory bean name and throws an exception if it could not be found
     *
     * @param  exchange            the exchange
     * @param  name                the bean name
     * @return                     the bean
     * @throws NoSuchBeanException if no bean could be found in the registry
     */
    public static Object lookupMandatoryBean(Exchange exchange, String name) throws NoSuchBeanException {
        Object value = lookupBean(exchange, name);
        if (value == null) {
            throw new NoSuchBeanException(name);
        }
        return value;
    }

    /**
     * Performs a lookup in the registry of the mandatory bean name and throws an exception if it could not be found
     *
     * @param  exchange            the exchange
     * @param  name                the bean name
     * @param  type                the expected bean type
     * @return                     the bean
     * @throws NoSuchBeanException if no bean could be found in the registry
     */
    public static  T lookupMandatoryBean(Exchange exchange, String name, Class type) {
        T value = lookupBean(exchange, name, type);
        if (value == null) {
            throw new NoSuchBeanException(name);
        }
        return value;
    }

    /**
     * Performs a lookup in the registry of the bean name
     *
     * @param  exchange the exchange
     * @param  name     the bean name
     * @return          the bean, or null if no bean could be found
     */
    public static Object lookupBean(Exchange exchange, String name) {
        return exchange.getContext().getRegistry().lookupByName(name);
    }

    /**
     * Performs a lookup in the registry of the bean name and type
     *
     * @param  exchange the exchange
     * @param  name     the bean name
     * @param  type     the expected bean type
     * @return          the bean, or null if no bean could be found
     */
    public static  T lookupBean(Exchange exchange, String name, Class type) {
        return exchange.getContext().getRegistry().lookupByNameAndType(name, type);
    }

    /**
     * Prepares the exchanges for aggregation.
     * 

* This implementation will copy the OUT body to the IN body so when you do aggregation the body is only in * the IN body to avoid confusing end users. * * @param oldExchange the old exchange * @param newExchange the new exchange */ public static void prepareAggregation(Exchange oldExchange, Exchange newExchange) { // move body/header from OUT to IN if (oldExchange != null) { ExchangeHelper.prepareOutToIn(oldExchange); } if (newExchange != null) { ExchangeHelper.prepareOutToIn(newExchange); } } /** * Checks whether the exchange has been failure handed * * @param exchange the exchange * @return true if failure handled, false otherwise */ public static boolean isFailureHandled(Exchange exchange) { return exchange.getExchangeExtension().isFailureHandled(); } /** * Checks whether the exchange has been error handler bridged * * @param exchange the exchange * @return true if error handler bridged, false otherwise */ public static boolean isErrorHandlerBridge(Exchange exchange) { return exchange.getProperty(ExchangePropertyKey.ERRORHANDLER_BRIDGE, false, Boolean.class); } /** * Checks whether the exchange {@link UnitOfWork} is exhausted * * @param exchange the exchange * @return true if exhausted, false otherwise */ public static boolean isUnitOfWorkExhausted(Exchange exchange) { return exchange.getProperty(ExchangePropertyKey.UNIT_OF_WORK_EXHAUSTED, false, Boolean.class); } /** * Sets the exchange to be failure handled. * * @param exchange the exchange */ public static void setFailureHandled(Exchange exchange) { // clear exception since its failure handled exchange.setException(null); exchange.getExchangeExtension().setFailureHandled(true); } /** * Checks whether the exchange {@link UnitOfWork} is redelivered * * @param exchange the exchange * @return true if redelivered, false otherwise */ public static boolean isRedelivered(Exchange exchange) { return exchange.getIn().hasHeaders() && exchange.getIn().getHeader(Exchange.REDELIVERED, false, Boolean.class); } /** * Check whether or not stream caching is enabled for the given route or globally. * * @param exchange the exchange * @return true if enabled, false otherwise */ public static boolean isStreamCachingEnabled(final Exchange exchange) { Route route = exchange.getContext().getRoute(exchange.getFromRouteId()); if (route != null) { return route.isStreamCaching(); } else { return exchange.getContext().getStreamCachingStrategy().isEnabled(); } } /** * Extracts the body from the given exchange. *

* If the exchange pattern is provided it will try to honor it and retrieve the body from either IN or OUT according * to the pattern. * * @param exchange the exchange * @param pattern exchange pattern if given, can be null * @return the result body, can be null. * @throws CamelExecutionException is thrown if the processing of the exchange failed */ public static Object extractResultBody(Exchange exchange, ExchangePattern pattern) { Object answer = null; if (exchange != null) { // rethrow if there was an exception during execution if (exchange.getException() != null) { throw CamelExecutionException.wrapCamelExecutionException(exchange, exchange.getException()); } // okay no fault then return the response according to the pattern // try to honor pattern if provided boolean notOut = pattern != null && !pattern.isOutCapable(); boolean hasOut = exchange.hasOut(); if (hasOut && !notOut) { // we have a response in out and the pattern is out capable answer = exchange.getOut().getBody(); } else { // use IN as the response answer = exchange.getIn().getBody(); } // in a very seldom situation then getBody can cause an exception to be set on the exchange // rethrow if there was an exception during execution if (exchange.getException() != null) { throw CamelExecutionException.wrapCamelExecutionException(exchange, exchange.getException()); } } return answer; } /** * Extracts the body from the given future, that represents a handle to an asynchronous exchange. *

* Will wait until the future task is complete. * * @param context the camel context * @param future the future handle * @param type the expected body response type * @return the result body, can be null. * @throws CamelExecutionException is thrown if the processing of the exchange failed */ public static T extractFutureBody(CamelContext context, Future future, Class type) { try { return doExtractFutureBody(context, future.get(), type); } catch (InterruptedException e) { Thread.currentThread().interrupt(); throw RuntimeCamelException.wrapRuntimeCamelException(e); } catch (ExecutionException e) { // execution failed due to an exception so rethrow the cause throw CamelExecutionException.wrapCamelExecutionException(null, e.getCause()); } finally { // its harmless to cancel if task is already completed // and in any case we do not want to get hold of the task a 2nd time // and its recommended to cancel according to Brian Goetz in his Java Concurrency in Practice book future.cancel(true); } } /** * Extracts the body from the given future, that represents a handle to an asynchronous exchange. *

* Will wait for the future task to complete, but waiting at most the timeout value. * * @param context the camel context * @param future the future handle * @param timeout timeout value * @param unit timeout unit * @param type the expected body response type * @return the result body, can be null. * @throws CamelExecutionException is thrown if the processing of the exchange failed * @throws java.util.concurrent.TimeoutException is thrown if a timeout triggered */ public static T extractFutureBody(CamelContext context, Future future, long timeout, TimeUnit unit, Class type) throws TimeoutException { try { if (timeout > 0) { return doExtractFutureBody(context, future.get(timeout, unit), type); } else { return doExtractFutureBody(context, future.get(), type); } } catch (InterruptedException e) { // execution failed due interruption so rethrow the cause Thread.currentThread().interrupt(); throw CamelExecutionException.wrapCamelExecutionException(null, e); } catch (ExecutionException e) { // execution failed due to an exception so rethrow the cause throw CamelExecutionException.wrapCamelExecutionException(null, e.getCause()); } finally { // its harmless to cancel if task is already completed // and in any case we do not want to get hold of the task a 2nd time // and its recommended to cancel according to Brian Goetz in his Java Concurrency in Practice book future.cancel(true); } } private static T doExtractFutureBody(CamelContext context, Object result, Class type) { if (result == null) { return null; } if (type.isAssignableFrom(result.getClass())) { return type.cast(result); } if (result instanceof Exchange exchange) { Object answer = ExchangeHelper.extractResultBody(exchange, exchange.getPattern()); return context.getTypeConverter().convertTo(type, exchange, answer); } return context.getTypeConverter().convertTo(type, result); } /** * Strategy to prepare results before next iterator or when we are complete, which is done by copying OUT to IN, so * there is only an IN as input for the next iteration. * * @param exchange the exchange to prepare */ public static void prepareOutToIn(Exchange exchange) { // we are routing using pipes and filters so we need to manually copy OUT to IN if (exchange.hasOut()) { exchange.setIn(exchange.getOut()); exchange.setOut(null); } } /** * Gets both the messageId and exchangeId to be used for logging purposes. *

* Logging both ids, can help to correlate exchanges which may be redelivered messages from for example a JMS * broker. * * @param exchange the exchange * @return a log message with both the messageId and exchangeId */ public static String logIds(Exchange exchange) { String msgId = exchange.getMessage().getMessageId(); return "(MessageId: " + msgId + " on ExchangeId: " + exchange.getExchangeId() + ")"; } /* * Safe copy message history using a defensive copy */ private static void setMessageHistory(Exchange target, Exchange source) { final Object history = source.getProperty(ExchangePropertyKey.MESSAGE_HISTORY); if (history != null) { // use thread-safe list as message history may be accessed concurrently target.setProperty(ExchangePropertyKey.MESSAGE_HISTORY, new CopyOnWriteArrayList<>((List) history)); } } /** * Copies the exchange but the copy will be tied to the given context * * @param exchange the source exchange * @return a copy with the given camel context */ public static Exchange copyExchangeWithProperties(Exchange exchange, CamelContext context) { Exchange answer = exchange.getExchangeExtension().createCopyWithProperties(context); setMessageHistory(answer, exchange); answer.setIn(exchange.getIn().copy()); if (exchange.hasOut()) { answer.setOut(exchange.getOut().copy()); } answer.setException(exchange.getException()); return answer; } /** * Replaces the existing message with the new message * * @param exchange the exchange * @param newMessage the new message * @param outOnly whether to replace the message as OUT message */ public static void replaceMessage(Exchange exchange, Message newMessage, boolean outOnly) { Message old = exchange.getMessage(); if (outOnly || exchange.hasOut()) { exchange.setOut(newMessage); } else { exchange.setIn(newMessage); } // need to de-reference old from the exchange so it can be GC if (old instanceof MessageSupport messageSupport) { messageSupport.setExchange(null); } } /** * Gets the original IN {@link Message} this Unit of Work was started with. *

* The original message is only returned if the option * {@link org.apache.camel.RuntimeConfiguration#isAllowUseOriginalMessage()} is enabled. If it is disabled, then * IllegalStateException is thrown. * * @return the original IN {@link Message} */ public static Message getOriginalInMessage(Exchange exchange) { Message answer = null; // try parent first UnitOfWork uow = exchange.getProperty(ExchangePropertyKey.PARENT_UNIT_OF_WORK, UnitOfWork.class); if (uow != null) { answer = uow.getOriginalInMessage(); } // fallback to the current exchange if (answer == null) { uow = exchange.getUnitOfWork(); if (uow != null) { answer = uow.getOriginalInMessage(); } } return answer; } /** * Resolve the component scheme (aka name) from the given endpoint uri * * @param uri the endpoint uri * @return the component scheme (name), or null if not possible to resolve */ public static String resolveScheme(String uri) { return StringHelper.before(uri, ":"); } /** * @see #getCharsetName(Exchange, boolean) */ public static String getCharsetName(Exchange exchange) { return getCharsetName(exchange, true); } /** * @see #getCharset(Exchange, boolean) */ public static Charset getCharset(Exchange exchange) { return getCharset(exchange, true); } /** * Gets the charset name if set as header or property {@link Exchange#CHARSET_NAME}. Notice: The lookup from * the header has priority over the property. * * @param exchange the exchange * @param useDefault should we fallback and use JVM default charset if no property existed? * @return the charset, or null if no found */ public static String getCharsetName(Exchange exchange, boolean useDefault) { if (exchange != null) { // header takes precedence String charsetName = exchange.getIn().getHeader(Exchange.CHARSET_NAME, String.class); if (charsetName == null) { charsetName = exchange.getProperty(ExchangePropertyKey.CHARSET_NAME, String.class); } if (charsetName != null) { return IOHelper.normalizeCharset(charsetName); } } if (useDefault) { return getDefaultCharsetName(); } else { return null; } } /** * Gets the charset if set as header or property {@link Exchange#CHARSET_NAME}. Notice: The lookup from the * header has priority over the property. * * @param exchange the exchange * @param useDefault should we fallback and use JVM default charset if no property existed? * @return the charset, or null if no found */ public static Charset getCharset(Exchange exchange, boolean useDefault) { if (exchange != null) { // header takes precedence String charsetName = exchange.getIn().getHeader(Exchange.CHARSET_NAME, String.class); if (charsetName == null) { charsetName = exchange.getProperty(ExchangePropertyKey.CHARSET_NAME, String.class); } if (charsetName != null) { charsetName = IOHelper.normalizeCharset(charsetName); return Charset.forName(charsetName); } } if (useDefault) { return getDefaultCharset(); } else { return null; } } private static String getDefaultCharsetName() { return DEFAULT_CHARSET_NAME; } private static Charset getDefaultCharset() { return DEFAULT_CHARSET; } /** * Creates a {@link Scanner} for scanning the given value. * * @param exchange the current exchange * @param value the value, typically the message IN body * @param delimiter the delimiter pattern to use * @return the scanner, is newer null */ public static Scanner getScanner(Exchange exchange, Object value, String delimiter) { if (value instanceof WrappedFile gf) { Object body = gf.getBody(); if (body != null) { // we have loaded the file content into the body so use that value = body; } else { // generic file is just a wrapper for the real file so call again with the real file value = gf.getFile(); } } Scanner scanner; if (value instanceof Readable readable) { scanner = new Scanner(readable, delimiter); } else if (value instanceof String str) { scanner = new Scanner(str, delimiter); } else { String charset = exchange.getProperty(ExchangePropertyKey.CHARSET_NAME, String.class); if (value instanceof Path path) { try { scanner = new Scanner( Files.newByteChannel(path, StandardOpenOption.READ), charset, delimiter); } catch (IOException e) { throw new RuntimeCamelException(e); } } else if (value instanceof File file) { try { scanner = new Scanner(file, charset, delimiter); } catch (IOException e) { throw new RuntimeCamelException(e); } } else if (value instanceof InputStream inputStream) { scanner = new Scanner(inputStream, charset, delimiter); } else if (value instanceof ReadableByteChannel readableByteChannel) { scanner = new Scanner(readableByteChannel, charset, delimiter); } else { // value is not a suitable type, try to convert value to a string String text = exchange.getContext().getTypeConverter().convertTo(String.class, exchange, value); scanner = new Scanner(text, delimiter); } } return scanner; } public static String getRouteId(Exchange exchange) { String answer = getAtRouteId(exchange); if (answer == null) { // fallback and get from route id on the exchange answer = exchange.getFromRouteId(); } return answer; } public static String getAtRouteId(Exchange exchange) { String answer = null; Route rc = getRoute(exchange); if (rc != null) { answer = rc.getRouteId(); } return answer; } public static String getRouteGroup(Exchange exchange) { Route rc = getRoute(exchange); if (rc != null) { return rc.getGroup(); } return null; } public static Route getRoute(Exchange exchange) { UnitOfWork uow = exchange.getUnitOfWork(); return uow != null ? uow.getRoute() : null; } /** * Sets the body in message in the exchange taking the exchange pattern into consideration. If the pattern is out * capable, then the body is set outbound message. Otherwise it is set on the inbound message. * * @param exchange the exchange containing the message to set the body * @param body the body to set */ public static void setInOutBodyPatternAware(Exchange exchange, Object body) { if (exchange.getPattern().isOutCapable()) { exchange.getOut().copyFrom(exchange.getIn()); exchange.getOut().setBody(body); } else { exchange.getIn().setBody(body); } } /** * Sets the body in message in the exchange taking the exchange pattern into consideration. If the pattern is out * capable, then the body is set outbound message. Otherwise nothing is done. * * @param exchange the exchange containing the message to set the body * @param body the body to set */ public static void setOutBodyPatternAware(Exchange exchange, Object body) { if (exchange.getPattern().isOutCapable()) { exchange.getOut().copyFrom(exchange.getIn()); exchange.getOut().setBody(body); } } /** * Sets the variable * * @param exchange the exchange * @param name the variable name. Can be prefixed with repo-id:name to use a specific repository. If no repo-id * is provided, then the variable is set on the exchange * @param value the value of the variable */ public static void setVariable(Exchange exchange, String name, Object value) { VariableRepository repo = null; final String id = getVariableRepositoryId(name); if (id != null) { repo = getVariableRepository(exchange, id); name = resolveVariableRepositoryName(exchange, name, id); } final VariableAware va = getVariableAware(exchange, repo); va.setVariable(name, value); } /** * Gets the variable repository id * * @param name the variable name * @return the repository id if any given, or null */ public static String getVariableRepositoryId(String name) { String id = StringHelper.before(name, ":"); // header and exchange is reserved if (isReserved(id)) { id = null; } return id; } /** * Resolves the variable name * * @param exchange the exchange * @param name the variable name * @param id the repository id * @return the resolved variable name */ public static String resolveVariableRepositoryName(Exchange exchange, String name, String id) { name = StringHelper.after(name, ":"); // special for route, where we need to enrich the name with current route id if none given if ("route".equals(id) && !name.contains(":")) { String prefix = getAtRouteId(exchange); if (prefix != null) { name = prefix + ":" + name; } else { // we are not currently in a given route return null; } } return name; } /** * Sets the variable from the given message body and headers * * @param exchange the exchange * @param name the variable name. Can be prefixed with repo-id:name to use a specific repository. If no repo-id * is provided, then the variable is set on the exchange * @param message the message with the body and headers as source values */ public static void setVariableFromMessageBodyAndHeaders(Exchange exchange, String name, Message message) { VariableRepository repo = null; final String id = getVariableRepositoryId(name); if (id != null) { repo = getVariableRepository(exchange, id); name = resolveVariableRepositoryName(exchange, name, id); } final VariableAware va = getVariableAware(exchange, repo); // set body and headers as variables Object body = message.getBody(); va.setVariable(name, body); for (Map.Entry header : message.getHeaders().entrySet()) { String key = "header:" + name + "." + header.getKey(); Object value = header.getValue(); va.setVariable(key, value); } } /** * Whether the processing of the {@link Exchange} was success and that the result should be stored in variable. * * @param exchange the exchange * @param name the variable name * @return true to call setVariableFromMessageBodyAndHeaders to set the result after-wards */ public static boolean shouldSetVariableResult(Exchange exchange, String name) { if (name == null) { return false; } // same logic as in Pipeline/PipelineHelper boolean stop = exchange.isRouteStop() || exchange.isFailed() || exchange.isRollbackOnly() || exchange.isRollbackOnlyLast() || exchange.getExchangeExtension().isErrorHandlerHandledSet() && exchange.getExchangeExtension().isErrorHandlerHandled(); if (stop) { return false; } // success return true; } /** * Gets the variable * * @param exchange the exchange * @param name the variable name. Can be prefixed with repo-id:name to lookup the variable from a specific * repository. If no repo-id is provided, then the variable is set on the exchange * @return the variable */ public static Object getVariable(Exchange exchange, String name) { VariableRepository repo = null; final String id = getVariableRepositoryId(name); if (id != null) { repo = getVariableRepository(exchange, id); name = resolveVariableRepositoryName(exchange, name, id); } final VariableAware va = getVariableAware(exchange, repo); return va.getVariable(name); } /** * Gets the variable repository by the given id * * @param exchange the exchange * @param id the repository id * @return the variable repository * @throws IllegalArgumentException is thrown if the repository does not eists */ public static VariableRepository getVariableRepository(Exchange exchange, String id) { VariableRepositoryFactory factory = exchange.getContext().getCamelContextExtension().getContextPlugin(VariableRepositoryFactory.class); VariableRepository repo = factory.getVariableRepository(id); if (repo == null) { throw new IllegalArgumentException("VariableRepository with id: " + id + " does not exist"); } return repo; } private static boolean isReserved(String id) { return "header".equals(id) || "exchange".equals(id); } private static VariableAware getVariableAware(Exchange exchange, VariableRepository repo) { VariableAware va = repo != null ? repo : exchange; return va; } /** * Gets the variable, converted to the given type * * @param exchange the exchange * @param name the variable name. Can be prefixed with repo-id:name to lookup the variable from a specific * repository. If no repo-id is provided, then the variable is set on the exchange * @param type the type to convert to * @return the variable */ public static T getVariable(Exchange exchange, String name, Class type) { Object answer = getVariable(exchange, name); if (answer != null) { return exchange.getContext().getTypeConverter().convertTo(type, exchange, answer); } return null; } /** * Returns the body as the specified type. If stream * caching. is enabled and the body is an instance of {@link StreamCache}, the stream is reset before converting * and returning the body. * * @param exchange the message exchange being processed * @param type the type to convert to * @return the body of the message as the specified type, or null if body does not exist */ public static T getBodyAndResetStreamCache(Exchange exchange, Class type) { Object body = exchange.getMessage().getBody(); if (body instanceof StreamCache sc) { sc.reset(); } return exchange.getMessage().getBody(type); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy