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

org.mule.api.FutureMessageResult Maven / Gradle / Ivy

There is a newer version: 3.9.0
Show newest version
/*
 * Copyright (c) MuleSoft, Inc.  All rights reserved.  http://www.mulesoft.com
 * The software in this package is published under the terms of the CPAL v1.0
 * license, a copy of which has been included with this distribution in the
 * LICENSE.txt file.
 */
package org.mule.api;

import org.mule.DefaultMuleMessage;
import org.mule.util.concurrent.DaemonThreadFactory;

import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.Executor;
import java.util.concurrent.Executors;
import java.util.concurrent.FutureTask;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;

/**
 * FutureMessageResult is an MuleMessage result of a remote invocation
 * on a Mule Server. This object makes the result available to the client code once
 * the request has been processed. This execution happens asynchronously.
 */
// @ThreadSafe
public class FutureMessageResult extends FutureTask
{
    /**
     * This is a simple default Executor for FutureMessageResults. Instead of
     * spawning a Thread for each invocation it uses a single daemon Thread with an
     * unbounded queue, so "truly" concurrent operation of multiple Futures or
     * otherwise customized execution behaviour requires calling the
     * {@link #setExecutor(Executor)} method and passing in a custom {@link Executor}.
     * This is strongly recommended in order to provide better control over
     * concurrency, resource consumption and possible overruns.
     * 

* Reasons for these defaults: *

    *
  • a daemon thread does not block the VM on shutdown; lifecycle control * should be done elsewhere (e.g. the provider of the custom ExecutorService), * otherwise this class would become too overloaded *
  • a single thread provides for conservative & predictable yet async * behaviour from a client's point of view *
  • the unbounded queue is not optimal but probably harmless since e.g. a * MuleClient would have to create a LOT of Futures for an OOM. Cancelled/timed * out invocations are GC'ed so the problem is rather unlikely to occur. *
*/ private static final Executor DefaultExecutor = Executors.newSingleThreadExecutor( new DaemonThreadFactory("MuleDefaultFutureMessageExecutor")); // @GuardedBy(this) private Executor executor; // @GuardedBy(this) private List transformers; protected MuleContext muleContext; public FutureMessageResult(Callable callable, MuleContext muleContext) { super(callable); this.executor = DefaultExecutor; this.muleContext = muleContext; } /** * Set an ExecutorService to run this invocation. * * @param e the executor to be used. * @throws IllegalArgumentException when the executor is null or shutdown. */ public void setExecutor(Executor e) { if (e == null) { throw new IllegalArgumentException("Executor must not be null."); } synchronized (this) { this.executor = e; } } /** * Set a post-invocation transformer. * * @param t Transformers to be applied to the result of this invocation. May be * null. */ public void setTransformers(List t) { synchronized (this) { this.transformers = t; } } public MuleMessage getMessage() throws InterruptedException, ExecutionException, MuleException { return this.getMessage(this.get()); } public MuleMessage getMessage(long timeout) throws InterruptedException, ExecutionException, TimeoutException, MuleException { return this.getMessage(this.get(timeout, TimeUnit.MILLISECONDS)); } private MuleMessage getMessage(Object obj) throws MuleException { MuleMessage result = null; if (obj != null) { if (obj instanceof MuleMessage) { result = (MuleMessage)obj; } else { result = new DefaultMuleMessage(obj, muleContext); } synchronized (this) { if (transformers != null) { result.applyTransformers(null, transformers, null); } } } return result; } /** * Start asynchronous execution of this task */ public void execute() { synchronized (this) { executor.execute(this); } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy