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

ratpack.exec.Promise Maven / Gradle / Ivy

There is a newer version: 2.0.0-rc-1
Show newest version
/*
 * Copyright 2014 the original author or authors.
 *
 * Licensed 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 ratpack.exec;

import ratpack.exec.internal.CachingUpstream;
import ratpack.exec.internal.DefaultOperation;
import ratpack.exec.internal.DefaultPromise;
import ratpack.exec.internal.DefaultExecution;
import ratpack.func.*;

import java.util.Objects;

import static ratpack.func.Action.ignoreArg;

/**
 * A promise for a single value.
 * 

* A promise is a representation of a value which will become available later. * Methods such as {@link #map(Function)}, {@link #flatMap(Function)}, {@link #cache()} etc.) allow a pipeline of “operations” to be specified, * that the value will travel through as it becomes available. * Such operations are implemented via the {@link #transform(Function)} method. * Each operation returns a new promise object, not the original promise object. *

* To create a promise, use the {@link Promise#of(Upstream)} method (or one of the variants such as {@link Promise#ofLazy(Factory)}. * To test code that uses promises, use the {@link ratpack.test.exec.ExecHarness}. *

* The promise is not “activated” until the {@link #then(Action)} method is called. * This method terminates the pipeline, and receives the final value. *

* Promise objects are multi use. * Every promise pipeline has a value producing function at its start. * Activating a promise (i.e. calling {@link #then(Action)}) invokes the function. * The {@link #cache()} operation can be used to change this behaviour. * * @param the type of promised value */ @SuppressWarnings("JavadocReference") public interface Promise { /** * Creates a promise for value that will be available later. *

* This method can be used to integrate with APIs that produce values asynchronously. * The {@link Upstream#connect(Downstream)} method will be invoked every time the value is requested. * This method should propagate the value (or error) to the given downstream object when it is available. * * @param upstream the producer of the value * @param the type of promised value * @return a promise for the asynchronously created value */ static Promise of(Upstream upstream) { return new DefaultPromise<>(DefaultExecution.upstream(upstream)); } /** * Creates a promise for the given value. *

* This method can be used when a promise is called for, but the value is immediately available. * * @param t the promised value * @param the type of promised value * @return a promise for the given item */ static Promise value(T t) { return of(f -> f.success(t)); } /** * Creates a promise for value produced by the given factory. *

* This method can be used when a promise is called for, and the value is available synchronously as the result of a function. * * @param factory the producer of the value * @param the type of promised value * @return a promise for the result of the factory */ static Promise ofLazy(Factory factory) { return of(f -> f.success(factory.create())); } /** * Creates a failed promise with the given error. *

* This method can be used when a promise is called for, but the failure is immediately available. * * @param t the error * @param the type of promised value * @return a failed promise */ static Promise error(Throwable t) { return of(f -> f.error(t)); } /** * Specifies what should be done with the promised object when it becomes available. *

* Important: this method can only be used from a Ratpack managed compute thread. * If it is called on a non Ratpack managed compute thread it will immediately throw an {@link ExecutionException}. * * @param then the receiver of the promised value * @throws ExecutionException if not called on a Ratpack managed compute thread */ void then(Action then); /** * A low level hook for consuming the promised value. *

* It is generally preferable to use {@link #then(Action)} over this method. * * @param downstream the downstream consumer */ void connect(Downstream downstream); /** * Apply a custom transform to this promise. *

* This method is the basis for the standard operations of this interface, such as {@link #map(Function)}. * The following is a non generic implementation of a map that converts the value to upper case. *

{@code
   * import ratpack.test.exec.ExecHarness;
   * import ratpack.exec.ExecResult;
   * import ratpack.exec.Promise;
   *
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     ExecResult result = ExecHarness.yieldSingle(c ->
   *       Promise.value("foo")
   *         .transform(up -> down ->
   *           up.connect(down.onSuccess(value -> {
   *             try {
   *               down.success(value.toUpperCase());
   *             } catch (Throwable e) {
   *               down.error(e);
   *             }
   *           }))
   *         )
   *     );
   *
   *     assertEquals("FOO", result.getValue());
   *   }
   * }
   * }
*

* The “upstreamTransformer” function takes an upstream data source, and returns another upstream that wraps it. * It is typical for the returned upstream to invoke the {@link Upstream#connect(Downstream)} method of the given upstream during its connect method. *

* For more examples of transform implementations, please see the implementations of the methods of this interface. * * @param upstreamTransformer a function that returns a new upstream, typically wrapping the given upstream argument * @param the type of item emitted by the transformed upstream * @return a new promise */ Promise transform(Function, ? extends Upstream> upstreamTransformer); /** * Specifies the action to take if the an error occurs trying to produce the promised value. *

* If the given action throws an exception, the original exception will be rethrown with the exception thrown * by the action added to the suppressed exceptions list. * * @param errorHandler the action to take if an error occurs * @return A promise for the successful result */ default Promise onError(Action errorHandler) { return transform(up -> down -> up.connect(down.onError(throwable -> { try { errorHandler.execute(throwable); } catch (Throwable e) { if (e != throwable) { e.addSuppressed(throwable); } down.error(e); return; } down.complete(); })) ); } /** * Consume the promised value as a {@link Result}. *

* This method is an alternative to {@link #then(Action)} and {@link #onError(Action)}. * * @param resultHandler the consumer of the result */ default void result(Action> resultHandler) { onError(t -> resultHandler.execute(Result.error(t))).then(v -> resultHandler.execute(Result.success(v))); } /** * Transforms the promised value by applying the given function to it. *

{@code
   * import ratpack.test.exec.ExecHarness;
   * import ratpack.exec.ExecResult;
   * import ratpack.exec.Promise;
   *
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     ExecResult result = ExecHarness.yieldSingle(c ->
   *         Promise.value("foo")
   *           .map(String::toUpperCase)
   *           .map(s -> s + "-BAR")
   *     );
   *
   *     assertEquals("FOO-BAR", result.getValue());
   *   }
   * }
   * }
* * @param transformer the transformation to apply to the promised value * @param the type of the transformed object * @return a promise for the transformed value */ default Promise map(Function transformer) { return transform(up -> down -> up.connect( down.onSuccess(value -> { try { O apply = transformer.apply(value); down.success(apply); } catch (Throwable e) { down.error(e); } }) ) ); } /** * Like {@link #map(Function)}, but performs the transformation on a blocking thread. *

* This is simply a more convenient form of using {@link Blocking#get(Factory)} and {@link #flatMap(Function)}. * * @param transformer the transformation to apply to the promised value, on a blocking thread * @param the type of the transformed object * @return a promise for the transformed value */ default Promise blockingMap(Function transformer) { return flatMap(t -> Blocking.get(() -> transformer.apply(t))); } /** * Executes the given action with the promise value, on a blocking thread. *

* Similar to {@link #blockingMap(Function)}, but does not provide a new value. * This can be used to do something with the value, without terminating the promise. * * @param action the action to to perform with the value, on a blocking thread * @return a promise for the same value given to the action */ default Promise blockingOp(Action action) { return flatMap(t -> Blocking.op(action.curry(t)).map(() -> t)); } default Promise next(Promise next) { return flatMap(in -> next); } default Promise> left(Promise left) { return flatMap(right -> left.map(value -> Pair.of(value, right))); } default Promise> right(Promise right) { return flatMap(left -> right.map(value -> Pair.of(left, value))); } default Operation operation() { return operation(Action.noop()); } default Operation operation(Action action) { return new DefaultOperation( map(t -> { action.execute(t); return null; }) ); } /** * Transforms the promise failure (potentially into a value) by applying the given function to it. *

* If the function returns a value, the promise will now be considered successful. *

{@code
   * import ratpack.test.exec.ExecHarness;
   * import ratpack.exec.ExecResult;
   * import ratpack.exec.Promise;
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     ExecResult result = ExecHarness.yieldSingle(c ->
   *         Promise.error(new Exception("!"))
   *           .mapError(e -> "value")
   *     );
   *
   *     assertEquals("value", result.getValue());
   *   }
   * }
   * }
*

* If the function throws an exception, that exception will now represent the promise failure. *

{@code
   * import ratpack.test.exec.ExecHarness;
   * import ratpack.exec.ExecResult;
   * import ratpack.exec.Promise;
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     ExecResult result = ExecHarness.yieldSingle(c ->
   *         Promise.error(new Exception("!"))
   *           .mapError(e -> { throw new RuntimeException("mapped", e); })
   *     );
   *
   *     assertEquals("mapped", result.getThrowable().getMessage());
   *   }
   * }
   * }
*

* The function will not be called if the promise is successful. * * @param transformer the transformation to apply to the promise failure * @return a promise */ default Promise mapError(Function transformer) { return transform(up -> down -> up.connect(down.onError(throwable -> { try { T transformed = transformer.apply(throwable); down.success(transformed); } catch (Throwable t) { down.error(t); } })) ); } /** * Applies the custom operation function to this promise. *

* This method can be used to apply custom operations without breaking the “code flow”. * It works particularly well with method references. *

{@code
   * import ratpack.exec.Promise;
   * import ratpack.test.exec.ExecHarness;
   *
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     Integer value = ExecHarness.yieldSingle(e ->
   *         Promise.value(1)
   *           .apply(Example::dubble)
   *           .apply(Example::triple)
   *     ).getValue();
   *
   *     assertEquals(Integer.valueOf(6), value);
   *   }
   *
   *   public static Promise dubble(Promise input) {
   *     return input.map(i -> i * 2);
   *   }
   *
   *   public static Promise triple(Promise input) {
   *     return input.map(i -> i * 3);
   *   }
   * }
   * }
*

* If the apply function throws an exception, the returned promise will fail. *

{@code
   * import ratpack.exec.Promise;
   * import ratpack.test.exec.ExecHarness;
   *
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     Throwable error = ExecHarness.yieldSingle(e ->
   *         Promise.value(1)
   *           .apply(Example::explode)
   *     ).getThrowable();
   *
   *     assertEquals("bang!", error.getMessage());
   *   }
   *
   *   public static Promise explode(Promise input) throws Exception {
   *     throw new Exception("bang!");
   *   }
   * }
   * }
*

* If the promise having the operation applied to fails, the operation will not be applied. *

{@code
   * import ratpack.exec.Promise;
   * import ratpack.test.exec.ExecHarness;
   *
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     Throwable error = ExecHarness.yieldSingle(e ->
   *         Promise.error(new Exception("bang!"))
   *           .apply(Example::dubble)
   *     ).getThrowable();
   *
   *     assertEquals("bang!", error.getMessage());
   *   }
   *
   *   public static Promise dubble(Promise input) {
   *     return input.map(i -> i * 2);
   *   }
   * }
   * }
* * @param the type of promised object after the operation * @param function the operation implementation * @return the transformed promise */ default Promise apply(Function, ? extends Promise> function) { try { return function.apply(this); } catch (Throwable e) { return Promise.error(e); } } /** * Applies the given function to {@code this} and returns the result. *

* This method can be useful when needing to convert a promise to another type as it facilitates doing so without breaking the “code flow”. * For example, this can be used when integrating with RxJava. *

{@code
   * import ratpack.rx.RxRatpack;
   * import ratpack.exec.Promise;
   * import ratpack.test.exec.ExecHarness;
   *
   * import java.util.Arrays;
   * import java.util.LinkedList;
   * import java.util.List;
   *
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   private static final List LOG = new LinkedList<>();
   *
   *   public static void main(String... args) throws Exception {
   *     ExecHarness.runSingle(e ->
   *         Promise.value("foo")
   *           .to(RxRatpack::observe)
   *           .doOnNext(i -> LOG.add("doOnNext"))
   *           .subscribe(LOG::add)
   *     );
   *
   *     assertEquals(Arrays.asList("doOnNext", "foo"), LOG);
   *   }
   * }
   * }
*

* The given function is executed immediately. *

* This method should only be used when converting a promise to another type. * See {@link #apply(Function)} for applying custom promise operators. * * @param function the promise conversion function * @param the type the promise will be converted to * @return the output of the given function * @throws Exception any thrown by the given function */ default O to(Function, ? extends O> function) throws Exception { return function.apply(this); } /** * Transforms the promised value by applying the given function to it that returns a promise for the transformed value. *

* This is useful when the transformation involves an asynchronous operation. *

{@code
   * import ratpack.test.exec.ExecHarness;
   * import ratpack.exec.ExecResult;
   * import ratpack.exec.Promise;
   * import ratpack.exec.Blocking;
   *
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static void main(String[] args) throws Exception {
   *     ExecResult result = ExecHarness.yieldSingle(c ->
   *         Promise.value("foo")
   *           .flatMap(s -> Blocking.get(s::toUpperCase))
   *           .map(s -> s + "-BAR")
   *     );
   *
   *     assertEquals("FOO-BAR", result.getValue());
   *   }
   * }
   * }
*

* * @param transformer the transformation to apply to the promised value * @param the type of the transformed object * @return a promise for the transformed value */ default Promise flatMap(Function> transformer) { return transform(up -> down -> up.connect(down.onSuccess(value -> { try { transformer.apply(value).onError(down::error).then(down::success); } catch (Throwable e) { down.error(e); } })) ); } /** * Allows the promised value to be handled specially if it meets the given predicate, instead of being handled by the promise subscriber. *

* This is typically used for validating values, centrally. *

{@code
   * import com.google.common.collect.Lists;
   * import ratpack.test.exec.ExecHarness;
   * import ratpack.exec.ExecResult;
   * import ratpack.exec.Promise;
   *
   * import java.util.List;
   *
   * import static org.junit.Assert.*;
   *
   * public class Example {
   *   public static ExecResult yield(int i, List collector) throws Exception {
   *     return ExecHarness.yieldSingle(c ->
   *         Promise.value(i)
   *           .route(v -> v > 5, collector::add)
   *     );
   *   }
   *
   *   public static void main(String... args) throws Exception {
   *     List routed = Lists.newLinkedList();
   *
   *     ExecResult result1 = yield(1, routed);
   *     assertEquals(new Integer(1), result1.getValue());
   *     assertFalse(result1.isComplete()); // false because promise returned a value before the execution completed
   *     assertTrue(routed.isEmpty());
   *
   *     ExecResult result10 = yield(10, routed);
   *     assertNull(result10.getValue());
   *     assertTrue(result10.isComplete()); // true because the execution completed before the promised value was returned (i.e. it was routed)
   *     assertTrue(routed.contains(10));
   *   }
   * }
   * }
*

* Be careful about using this where the eventual promise subscriber is unlikely to know that the promise * will routed as it can be surprising when neither the promised value nor an error appears. *

* It can be useful at the handler layer to provide common validation. *

{@code
   * import ratpack.exec.Promise;
   * import ratpack.handling.Context;
   * import ratpack.test.embed.EmbeddedApp;
   *
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static Promise getAge(Context ctx) {
   *     return Promise.value(10)
   *       .route(
   *         i -> i < 21,
   *         i -> ctx.render(i + " is too young to be here!")
   *       );
   *   }
   *
   *   public static void main(String... args) throws Exception {
   *     EmbeddedApp.fromHandler(ctx ->
   *         getAge(ctx).then(age -> ctx.render("welcome!"))
   *     ).test(httpClient -> {
   *       assertEquals("10 is too young to be here!", httpClient.getText());
   *     });
   *   }
   * }
   * }
*

* If the routed-to action throws an exception, it will be forwarded down the promise chain. * * @param predicate the condition under which the value should be routed * @param action the terminal action for the value * @return a routed promise */ default Promise route(Predicate predicate, Action action) { return transform(up -> down -> up.connect(down.onSuccess(value -> { boolean apply; try { apply = predicate.apply(value); } catch (Throwable e) { down.error(e); return; } if (apply) { try { action.execute(value); down.complete(); } catch (Throwable e) { down.error(e); } } else { down.success(value); } })) ); } /** * A convenience shorthand for {@link #route(Predicate, Action) routing} {@code null} values. *

* If the promised value is {@code null}, the given action will be called. * * @param action the action to route to if the promised value is null * @return a routed promise */ default Promise onNull(Block action) { return route(Objects::isNull, ignoreArg(action)); } /** * Caches the promised value (or error) and returns it to all subscribers. *

{@code
   * import ratpack.exec.Promise;
   * import ratpack.test.exec.ExecHarness;
   *
   * import java.util.concurrent.atomic.AtomicLong;
   *
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     ExecHarness.runSingle(c -> {
   *       AtomicLong counter = new AtomicLong();
   *       Promise uncached = Promise.of(f -> f.success(counter.getAndIncrement()));
   *
   *       uncached.then(i -> assertEquals(0l, i.longValue()));
   *       uncached.then(i -> assertEquals(1l, i.longValue()));
   *       uncached.then(i -> assertEquals(2l, i.longValue()));
   *
   *       Promise cached = uncached.cache();
   *
   *       cached.then(i -> assertEquals(3l, i.longValue()));
   *       cached.then(i -> assertEquals(3l, i.longValue()));
   *
   *       uncached.then(i -> assertEquals(4l, i.longValue()));
   *       cached.then(i -> assertEquals(3l, i.longValue()));
   *     });
   *   }
   * }
   * }
*

* If the cached promise fails, the same exception will be returned every time. *

{@code
   * import ratpack.exec.Promise;
   * import ratpack.test.exec.ExecHarness;
   *
   * import static org.junit.Assert.assertTrue;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     ExecHarness.runSingle(c -> {
   *       Throwable error = new Exception("bang!");
   *       Promise cached = Promise.of(f -> f.error(error)).cache();
   *       cached.onError(t -> assertTrue(t == error)).then(i -> assertTrue("not called", false));
   *       cached.onError(t -> assertTrue(t == error)).then(i -> assertTrue("not called", false));
   *       cached.onError(t -> assertTrue(t == error)).then(i -> assertTrue("not called", false));
   *     });
   *   }
   * }
   * }
   *
   * @return a caching promise.
   */
  default Promise cache() {
    return transform(CachingUpstream::new);
  }

  /**
   * Allows the execution of the promise to be deferred to a later time.
   * 

* When the returned promise is subscribed to, the given {@code releaser} action will be invoked. * The execution of {@code this} promise is deferred until the runnable given to the {@code releaser} is run. *

* It is generally more convenient to use {@link #throttled(Throttle)} or {@link #onYield(Runnable)} than this operation. * * @param releaser the action that will initiate the execution some time later * @return a deferred promise */ default Promise defer(Action releaser) { return transform(up -> down -> DefaultExecution.require().streamSubscribe(stream -> { try { releaser.execute((Runnable) () -> { stream.event(() -> up.connect(down)); stream.complete(); } ); } catch (Throwable t) { down.error(t); } }) ); } /** * Registers a listener that is invoked when {@code this} promise is initiated. *

{@code
   * import com.google.common.collect.Lists;
   * import ratpack.test.exec.ExecHarness;
   * import ratpack.exec.Promise;
   *
   * import java.util.Arrays;
   * import java.util.List;
   *
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     List events = Lists.newLinkedList();
   *     ExecHarness.runSingle(c ->
   *         Promise.of(f -> {
   *           events.add("promise");
   *           f.success("foo");
   *         })
   *           .onYield(() -> events.add("onYield"))
   *           .then(v -> events.add("then"))
   *     );
   *     assertEquals(Arrays.asList("onYield", "promise", "then"), events);
   *   }
   * }
   * }
* * @param onYield the action to take when the promise is initiated * @return effectively, {@code this} promise */ default Promise onYield(Runnable onYield) { return transform(up -> down -> { try { onYield.run(); } catch (Throwable e) { down.error(e); return; } up.connect(down); }); } /** * Registers a listener for the promise outcome. *
{@code
   * import com.google.common.collect.Lists;
   * import ratpack.test.exec.ExecHarness;
   * import ratpack.exec.Promise;
   *
   * import java.util.Arrays;
   * import java.util.List;
   *
   * import static org.junit.Assert.assertEquals;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     List events = Lists.newLinkedList();
   *     ExecHarness.runSingle(c ->
   *         Promise.of(f -> {
   *           events.add("promise");
   *           f.success("foo");
   *         })
   *           .wiretap(r -> events.add("wiretap: " + r.getValue()))
   *           .then(v -> events.add("then"))
   *     );
   *
   *     assertEquals(Arrays.asList("promise", "wiretap: foo", "then"), events);
   *   }
   * }
   * }
* * @param listener the result listener * @return effectively, {@code this} promise */ default Promise wiretap(Action> listener) { return transform(up -> down -> up.connect(down.onSuccess(value -> { try { listener.execute(Result.success(value)); } catch (Throwable t) { down.error(t); return; } down.success(value); })) ); } /** * Throttles {@code this} promise, using the given {@link Throttle throttle}. *

* Throttling can be used to limit concurrency. * Typically to limit concurrent use of an external resource, such as a HTTP API. *

* Note that the {@link Throttle} instance given defines the actual throttling semantics. *

{@code
   * import ratpack.exec.Throttle;
   * import ratpack.exec.Promise;
   * import ratpack.exec.Execution;
   * import ratpack.test.exec.ExecHarness;
   * import ratpack.exec.ExecResult;
   *
   * import java.util.concurrent.atomic.AtomicInteger;
   *
   * import static org.junit.Assert.assertTrue;
   *
   * public class Example {
   *   public static void main(String... args) throws Exception {
   *     int numJobs = 1000;
   *     int maxAtOnce = 10;
   *
   *     ExecResult result = ExecHarness.yieldSingle(exec -> {
   *       AtomicInteger maxConcurrent = new AtomicInteger();
   *       AtomicInteger active = new AtomicInteger();
   *       AtomicInteger done = new AtomicInteger();
   *
   *       Throttle throttle = Throttle.ofSize(maxAtOnce);
   *
   *       // Launch numJobs forked executions, and return the maximum number that were executing at any given time
   *       return Promise.of(outerFulfiller -> {
   *         for (int i = 0; i < numJobs; i++) {
   *           Execution.fork().start(forkedExec ->
   *             Promise.of(innerFulfiller -> {
   *               int activeNow = active.incrementAndGet();
   *               int maxConcurrentVal = maxConcurrent.updateAndGet(m -> Math.max(m, activeNow));
   *               active.decrementAndGet();
   *               innerFulfiller.success(maxConcurrentVal);
   *             })
   *             .throttled(throttle) // limit concurrency
   *             .then(max -> {
   *               if (done.incrementAndGet() == numJobs) {
   *                 outerFulfiller.success(max);
   *               }
   *             })
   *           );
   *         }
   *       });
   *     });
   *
   *     assertTrue(result.getValue() <= maxAtOnce);
   *   }
   * }
   * }
* * @param throttle the particular throttle to use to throttle the operation * @return the throttled promise */ default Promise throttled(Throttle throttle) { return throttle.throttle(this); } static Promise wrap(Factory> factory) { try { return factory.create(); } catch (Exception e) { return Promise.error(e); } } }