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

org.apache.pulsar.shade.com.spotify.futures.CompletableFutures Maven / Gradle / Ivy

The newest version!
/*-
 * -\-\-
 * completable-futures
 * --
 * Copyright (C) 2016 - 2020 Spotify AB
 * --
 * 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 org.apache.pulsar.shade.com.spotify.futures;

import static java.util.stream.Collectors.collectingAndThen;
import static java.util.stream.Collectors.toList;
import static java.util.stream.Collectors.toMap;

import java.time.Duration;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.concurrent.CancellationException;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CompletionException;
import java.util.concurrent.CompletionStage;
import java.util.concurrent.Executor;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.ScheduledFuture;
import java.util.concurrent.TimeUnit;
import java.util.function.BiFunction;
import java.util.function.Function;
import java.util.function.Supplier;
import java.util.stream.Collector;

/**
 * A collection of static utility methods that extend the
 * {@link java.util.concurrent.CompletableFuture Java completable future} API.
 *
 * @since 0.1.0
 */
public final class CompletableFutures {

  private CompletableFutures() {
    throw new IllegalAccessError("This class must not be instantiated.");
  }

  /**
   * Returns a new {@link CompletableFuture} which completes to a list of all values of its input
   * stages, if all succeed.  The list of results is in the same order as the input stages.
   *
   * 

As soon as any of the given stages complete exceptionally, then the returned future also does so, * with a {@link CompletionException} holding this exception as its cause. * *

If no stages are provided, returns a future holding an empty list. * * @param stages the stages to combine * @param the common super-type of all of the input stages, that determines the monomorphic * type of the output future * @return a future that completes to a list of the results of the supplied stages * @throws NullPointerException if the stages list or any of its elements are {@code null} * @since 0.1.0 */ public static CompletableFuture> allAsList( List> stages) { // We use traditional for-loops instead of streams here for performance reasons, // see AllAsListBenchmark @SuppressWarnings("unchecked") // generic array creation final CompletableFuture[] all = new CompletableFuture[stages.size()]; for (int i = 0; i < stages.size(); i++) { all[i] = stages.get(i).toCompletableFuture(); } CompletableFuture allOf = CompletableFuture.allOf(all); for (int i = 0; i < all.length; i++) { all[i].exceptionally(throwable -> { if (!allOf.isDone()) { allOf.completeExceptionally(throwable); } return null; // intentionally unused }); } return allOf .thenApply(ignored -> { final List result = new ArrayList<>(all.length); for (int i = 0; i < all.length; i++) { result.add(all[i].join()); } return result; }); } /** * Returns a new {@link CompletableFuture} which completes to a map of all values of its input * stages, if all succeed. * *

If any of the given stages complete exceptionally, then the returned future also does so, * with a {@link CompletionException} holding this exception as its cause. * *

If no stages are provided, returns a future holding an empty map. * * @param map the map of stages to combine * @param the common super-type of the keys * @param the common super-type of all of the input value-stages, that determines the * monomorphic type of the output future * @return a future that completes to a map of the results of the supplied keys and value-stages * @throws NullPointerException if value-stages or any of its elements are {@code null} * @since 0.3.3 */ public static CompletableFuture> allAsMap( Map> map) { final List keys = new ArrayList<>(map.keySet()); @SuppressWarnings("unchecked") // generic array creation final CompletableFuture[] values = new CompletableFuture[keys.size()]; for (int i = 0; i < keys.size(); i++) { values[i] = map.get(keys.get(i)).toCompletableFuture(); } return CompletableFuture.allOf(values) .thenApply( ignored -> { final Map result = new HashMap<>(values.length); for (int i = 0; i < values.length; i++) { result.put(keys.get(i), values[i].join()); } return result; }); } /** * Returns a new {@link CompletableFuture} which completes to a list of values of those input * stages that succeeded. The list of results is in the same order as the input stages. For failed * stages, the defaultValueMapper will be called, and the value returned from that function will * be put in the resulting list. * *

If no stages are provided, returns a future holding an empty list. * * @param stages the stages to combine. * @param defaultValueMapper a function that will be called when a future completes exceptionally * to provide a default value to place in the resulting list * @param the common type of all of the input stages, that determines the type of the * output future * @return a future that completes to a list of the results of the supplied stages * @throws NullPointerException if the stages list or any of its elements are {@code null} */ public static CompletableFuture> successfulAsList( List> stages, Function defaultValueMapper) { return stages.stream() .map(f -> f.exceptionally(defaultValueMapper)) .collect(joinList()); } /** * Returns a new {@code CompletableFuture} that is already exceptionally completed with * the given exception. * * @param throwable the exception * @param an arbitrary type for the returned future; can be anything since the future * will be exceptionally completed and thus there will never be a value of type * {@code T} * @return a future that exceptionally completed with the supplied exception * @throws NullPointerException if the supplied throwable is {@code null} * @since 0.1.0 */ public static CompletableFuture exceptionallyCompletedFuture(Throwable throwable) { final CompletableFuture future = new CompletableFuture<>(); future.completeExceptionally(throwable); return future; } /** * Collect a stream of {@link CompletionStage}s into a single future holding a list of the * joined entities. * *

Usage: * *

{@code
   * collection.stream()
   *     .map(this::someAsyncFunc)
   *     .collect(joinList())
   *     .thenApply(this::consumeList)
   * }
* *

The generated {@link CompletableFuture} will complete to a list of all entities, in the * order they were encountered in the original stream. Similar to * {@link CompletableFuture#allOf(CompletableFuture[])}, if any of the input futures complete * exceptionally, then the returned CompletableFuture also does so, with a * {@link CompletionException} holding this exception as its cause. * * @param the common super-type of all of the input stages, that determines the monomorphic * type of the output future * @param the implementation of {@link CompletionStage} that the stream contains * @return a new {@link CompletableFuture} according to the rules outlined in the method * description * @throws NullPointerException if any future in the stream is {@code null} * @since 0.1.0 */ public static > Collector>> joinList() { return collectingAndThen(toList(), CompletableFutures::allAsList); } /** * Collect a stream of Objects into a single future holding a Map whose keys are the result of * applying the provided mapping function to the input elements, and whose values are the * corresponding joined entities. * *

Usage: * *

{@code
   * collection.stream()
   *     .collect(joinMap(this::toKey, this::someAsyncFunc))
   *     .thenApply(this::consumeMap)
   * }
* *

The generated {@link CompletableFuture} will complete to a Map of all entities. Similar * to {@link CompletableFuture#allOf(CompletableFuture[])}, if any of the input futures * complete exceptionally, then the returned CompletableFuture also does so, with a * {@link CompletionException} holding this exception as its cause. * * @param keyMapper a mapping function to produce keys * @param valueFutureMapper a mapping function to produce futures that will eventually produce * the values * @param the type of the input elements * @param the output type of the key mapping function * @param the common super-type of the stages returned by the value future mapping function * @return a new {@link CompletableFuture} according to the rules outlined in the method * description * @throws NullPointerException if valueFutureMapper returns {@code null} for any input element * @since 0.3.4 */ public static Collector>> joinMap( Function keyMapper, Function> valueFutureMapper){ Collector>> collector = toMap(keyMapper, valueFutureMapper); return collectingAndThen(collector, CompletableFutures::allAsMap); } /** * Checks that a stage is completed. * * @param stage the {@link CompletionStage} to check * @throws IllegalStateException if the stage is not completed * @since 0.1.0 */ public static void checkCompleted(CompletionStage stage) { if (!stage.toCompletableFuture().isDone()) { throw new IllegalStateException("future was not completed"); } } /** * Gets the value of a completed stage. * * @param stage a completed {@link CompletionStage} * @param the type of the value that the stage completes into * @return the value of the stage if it has one * @throws IllegalStateException if the stage is not completed * @since 0.1.0 */ public static T getCompleted(CompletionStage stage) { CompletableFuture future = stage.toCompletableFuture(); checkCompleted(future); return future.join(); } /** * Gets the exception from an exceptionally completed future * @param stage an exceptionally completed {@link CompletionStage} * @param the type of the value that the stage completes into * @return the exception the stage has completed with * @throws IllegalStateException if the stage is not completed exceptionally * @throws CancellationException if the stage was cancelled * @throws UnsupportedOperationException if the {@link CompletionStage} does not * support the {@link CompletionStage#toCompletableFuture()} operation */ public static Throwable getException(CompletionStage stage) { CompletableFuture future = stage.toCompletableFuture(); if (!future.isCompletedExceptionally()) { throw new IllegalStateException("future was not completed exceptionally"); } try { future.join(); return null; } catch (CompletionException x) { return x.getCause(); } } /** * Returns a new stage that, when this stage completes either normally or exceptionally, is * executed with this stage's result and exception as arguments to the supplied function. * *

When this stage is complete, the given function is invoked with the result (or {@code null} * if none) and the exception (or {@code null} if none) of this stage as arguments, and the * function's result is used to complete the returned stage. * *

This differs from * {@link java.util.concurrent.CompletionStage#handle(java.util.function.BiFunction)} in that the * function should return a {@link java.util.concurrent.CompletionStage} rather than the value * directly. * * @param stage the {@link CompletionStage} to compose * @param fn the function to use to compute the value of the * returned {@link CompletionStage} * @param the type of the input stage's value. * @param the function's return type * @return the new {@link CompletionStage} * @since 0.1.0 */ public static CompletionStage handleCompose( CompletionStage stage, BiFunction> fn) { return dereference(stage.handle(fn)); } /** * Returns a new stage that, when this stage completes * exceptionally, is executed with this stage's exception as the * argument to the supplied function. Otherwise, if this stage * completes normally, then the returned stage also completes * normally with the same value. * *

This differs from * {@link java.util.concurrent.CompletionStage#exceptionally(java.util.function.Function)} * in that the function should return a {@link java.util.concurrent.CompletionStage} rather than * the value directly. * * @param stage the {@link CompletionStage} to compose * @param fn the function to use to compute the value of the * returned {@link CompletionStage} if this stage completed * exceptionally * @param the type of the input stage's value. * @return the new {@link CompletionStage} * @since 0.1.0 */ public static CompletionStage exceptionallyCompose( CompletionStage stage, Function> fn) { return dereference(wrap(stage).exceptionally(fn)); } /** * This allows for a stage to be the return type of supplier when using supplyAsync and returns a plain stage. * *

This differs from {@link * java.util.concurrent.CompletableFuture#supplyAsync(Supplier)} in that the * function should return a {@link java.util.concurrent.CompletionStage} rather than the {@link CompletionStage} of * a {@link CompletionStage} of a value when the return value of the {@link Supplier} is a {@link CompletionStage}. * * @param supplier a {@link Supplier} with a return value of {@link CompletionStage} * @param the type of the supplied stage's value * @return the new {@link CompletionStage} * @since 0.3.6 */ public static CompletionStage supplyAsyncCompose(Supplier> supplier) { return dereference(CompletableFuture.supplyAsync(supplier)); } /** * This allows for a stage to be the return type of supplier when using supplyAsync and returns a plain stage. * *

This differs from {@link * java.util.concurrent.CompletableFuture#supplyAsync(Supplier, Executor)} in that the * function should return a {@link java.util.concurrent.CompletionStage} rather than the {@link CompletionStage} of * a {@link CompletionStage} of a value when the return value of the {@link Supplier} is a {@link CompletionStage}. * * @param supplier a {@link Supplier} with a return value of {@link CompletionStage} * @param executor a {@link Executor} the executor to use for asynchronous execution * @param the type of the supplied stage's value * @return the new {@link CompletionStage} * @since 0.3.6 */ public static CompletionStage supplyAsyncCompose( Supplier> supplier, Executor executor) { return dereference(CompletableFuture.supplyAsync(supplier, executor)); } /** * This takes a stage of a stage of a value and returns a plain stage of a value. * * @param stage a {@link CompletionStage} of a {@link CompletionStage} of a value * @param the type of the inner stage's value. * @return the {@link CompletionStage} of the value * @since 0.1.0 */ public static CompletionStage dereference( CompletionStage> stage) { return stage.thenCompose(Function.identity()); } private static CompletionStage> wrap(CompletionStage future) { //noinspection unchecked return future.thenApply(CompletableFuture::completedFuture); } /** * Combines multiple stages by applying a function. * * @param a the first stage. * @param b the second stage. * @param function the combining function. * @param the type of the combining function's return value. * @param the type of the first stage's value. * @param the type of the second stage's value. * @return a stage that completes into the return value of the supplied function. * @since 0.1.0 */ public static CompletionStage combine( CompletionStage a, CompletionStage b, BiFunction function) { return a.thenCombine(b, function); } /** * Combines multiple stages by applying a function. * * @param a the first stage. * @param b the second stage. * @param c the third stage. * @param function the combining function. * @param the type of the combining function's return value. * @param the type of the first stage's value. * @param the type of the second stage's value. * @param the type of the third stage's value. * @return a stage that completes into the return value of the supplied function. * @since 0.1.0 */ public static CompletionStage combine( CompletionStage a, CompletionStage b, CompletionStage c, Function3 function) { final CompletableFuture af = a.toCompletableFuture(); final CompletableFuture bf = b.toCompletableFuture(); final CompletableFuture cf = c.toCompletableFuture(); return CompletableFuture.allOf(af, bf, cf) .thenApply(ignored -> function.apply(af.join(), bf.join(), cf.join())); } /** * Combines multiple stages by applying a function. * * @param a the first stage. * @param b the second stage. * @param c the third stage. * @param d the fourth stage. * @param function the combining function. * @param the type of the combining function's return value. * @param the type of the first stage's value. * @param the type of the second stage's value. * @param the type of the third stage's value. * @param the type of the fourth stage's value. * @return a stage that completes into the return value of the supplied function. * @since 0.1.0 */ public static CompletionStage combine( CompletionStage a, CompletionStage b, CompletionStage c, CompletionStage d, Function4 function) { final CompletableFuture af = a.toCompletableFuture(); final CompletableFuture bf = b.toCompletableFuture(); final CompletableFuture cf = c.toCompletableFuture(); final CompletableFuture df = d.toCompletableFuture(); return CompletableFuture.allOf(af, bf, cf, df) .thenApply(ignored -> function.apply(af.join(), bf.join(), cf.join(), df.join())); } /** * Combines multiple stages by applying a function. * * @param a the first stage. * @param b the second stage. * @param c the third stage. * @param d the fourth stage. * @param e the fifth stage. * @param function the combining function. * @param the type of the combining function's return value. * @param the type of the first stage's value. * @param the type of the second stage's value. * @param the type of the third stage's value. * @param the type of the fourth stage's value. * @param the type of the fifth stage's value. * @return a stage that completes into the return value of the supplied function. * @since 0.1.0 */ public static CompletionStage combine( CompletionStage a, CompletionStage b, CompletionStage c, CompletionStage d, CompletionStage e, Function5 function) { final CompletableFuture af = a.toCompletableFuture(); final CompletableFuture bf = b.toCompletableFuture(); final CompletableFuture cf = c.toCompletableFuture(); final CompletableFuture df = d.toCompletableFuture(); final CompletableFuture ef = e.toCompletableFuture(); return CompletableFuture.allOf(af, bf, cf, df, ef) .thenApply(ignored -> function.apply(af.join(), bf.join(), cf.join(), df.join(), ef.join())); } /** * Combines multiple stages by applying a function. * * @param a the first stage. * @param b the second stage. * @param c the third stage. * @param d the fourth stage. * @param e the fifth stage. * @param f the sixth stage. * @param function the combining function. * @param the type of the combining function's return value. * @param the type of the first stage's value. * @param the type of the second stage's value. * @param the type of the third stage's value. * @param the type of the fourth stage's value. * @param the type of the fifth stage's value. * @param the type of the sixth stage's value. * @return a stage that completes into the return value of the supplied function. * @since 0.3.2 */ public static CompletionStage combine( CompletionStage a, CompletionStage b, CompletionStage c, CompletionStage d, CompletionStage e, CompletionStage f, Function6 function) { final CompletableFuture af = a.toCompletableFuture(); final CompletableFuture bf = b.toCompletableFuture(); final CompletableFuture cf = c.toCompletableFuture(); final CompletableFuture df = d.toCompletableFuture(); final CompletableFuture ef = e.toCompletableFuture(); final CompletableFuture ff = f.toCompletableFuture(); return CompletableFuture.allOf(af, bf, cf, df, ef, ff) .thenApply(ignored -> function.apply(af.join(), bf.join(), cf.join(), df.join(), ef.join(), ff.join())); } /** * Combines multiple stages by applying a function. * * @param function the combining function. * @param stages the stages to combine * @param the type of the combining function's return value. * @return a stage that completes into the return value of the supplied function. * @since 0.4.0 */ public static CompletionStage combine( Function function, CompletionStage... stages) { return combine(function, Arrays.asList(stages)); } /** * Combines multiple stages by applying a function. * * @param function the combining function. * @param stages the stages to combine * @param the type of the combining function's return value. * @return a stage that completes into the return value of the supplied function. * @since 0.4.0 */ public static CompletionStage combine( Function function, List> stages) { @SuppressWarnings("unchecked") // generic array creation final CompletableFuture[] all = new CompletableFuture[stages.size()]; for (int i = 0; i < stages.size(); i++) { all[i] = stages.get(i).toCompletableFuture(); } return CompletableFuture.allOf(all).thenApply(ignored -> function.apply(new CombinedFutures(stages))); } /** * Composes multiple stages into another stage using a function. * * @param a the first stage. * @param b the second stage. * @param function the combining function. * @param the type of the composed {@link CompletionStage}. * @param the type of the first stage's value. * @param the type of the second stage's value. * @return a stage that is composed from the input stages using the function. * @throws UnsupportedOperationException if any of the {@link CompletionStage}s * do not interoperate with CompletableFuture */ public static CompletionStage combineFutures( CompletionStage a, CompletionStage b, BiFunction> function) { final CompletableFuture af = a.toCompletableFuture(); final CompletableFuture bf = b.toCompletableFuture(); return CompletableFuture.allOf(af, bf) .thenCompose(ignored -> function.apply(af.join(), bf.join())); } /** * Composes multiple stages into another stage using a function. * * @param a the first stage. * @param b the second stage. * @param c the third stage. * @param function the combining function. * @param the type of the composed {@link CompletionStage}. * @param the type of the first stage's value. * @param the type of the second stage's value. * @param the type of the third stage's value. * @return a stage that is composed from the input stages using the function. * @throws UnsupportedOperationException if any of the {@link CompletionStage}s * do not interoperate with CompletableFuture */ public static CompletionStage combineFutures( CompletionStage a, CompletionStage b, CompletionStage c, Function3> function) { final CompletableFuture af = a.toCompletableFuture(); final CompletableFuture bf = b.toCompletableFuture(); final CompletableFuture cf = c.toCompletableFuture(); return CompletableFuture.allOf(af, bf, cf) .thenCompose(ignored -> function.apply(af.join(), bf.join(), cf.join())); } /** * Composes multiple stages into another stage using a function. * * @param a the first stage. * @param b the second stage. * @param c the third stage. * @param d the fourth stage. * @param function the combining function. * @param the type of the composed {@link CompletionStage}. * @param the type of the first stage's value. * @param the type of the second stage's value. * @param the type of the third stage's value. * @param the type of the fourth stage's value. * @return a stage that is composed from the input stages using the function. * @throws UnsupportedOperationException if any of the {@link CompletionStage}s * do not interoperate with CompletableFuture */ public static CompletionStage combineFutures( CompletionStage a, CompletionStage b, CompletionStage c, CompletionStage d, Function4> function) { final CompletableFuture af = a.toCompletableFuture(); final CompletableFuture bf = b.toCompletableFuture(); final CompletableFuture cf = c.toCompletableFuture(); final CompletableFuture df = d.toCompletableFuture(); return CompletableFuture.allOf(af, bf, cf, df) .thenCompose(ignored -> function.apply(af.join(), bf.join(), cf.join(), df.join())); } /** * Composes multiple stages into another stage using a function. * * @param a the first stage. * @param b the second stage. * @param c the third stage. * @param d the fourth stage. * @param e the fifth stage. * @param function the combining function. * @param the type of the composed {@link CompletionStage}. * @param the type of the first stage's value. * @param the type of the second stage's value. * @param the type of the third stage's value. * @param the type of the fourth stage's value. * @param the type of the fifth stage's value. * @return a stage that is composed from the input stages using the function. * @throws UnsupportedOperationException if any of the {@link CompletionStage}s * do not interoperate with CompletableFuture */ public static CompletionStage combineFutures( CompletionStage a, CompletionStage b, CompletionStage c, CompletionStage d, CompletionStage e, Function5> function) { final CompletableFuture af = a.toCompletableFuture(); final CompletableFuture bf = b.toCompletableFuture(); final CompletableFuture cf = c.toCompletableFuture(); final CompletableFuture df = d.toCompletableFuture(); final CompletableFuture ef = e.toCompletableFuture(); return CompletableFuture.allOf(af, bf, cf, df, ef) .thenCompose(ignored -> function.apply(af.join(), bf.join(), cf.join(), df.join(), ef.join())); } /** * Composes multiple stages into another stage using a function. * * @param a the first stage. * @param b the second stage. * @param c the third stage. * @param d the fourth stage. * @param e the fifth stage. * @param f the sixth stage. * @param function the combining function. * @param the type of the composed {@link CompletionStage}. * @param the type of the first stage's value. * @param the type of the second stage's value. * @param the type of the third stage's value. * @param the type of the fourth stage's value. * @param the type of the fifth stage's value. * @param the type of the sixth stage's value. * @return a stage that is composed from the input stages using the function. * @throws UnsupportedOperationException if any of the {@link CompletionStage}s * do not interoperate with CompletableFuture */ public static CompletionStage combineFutures( CompletionStage a, CompletionStage b, CompletionStage c, CompletionStage d, CompletionStage e, CompletionStage f, Function6> function) { final CompletableFuture af = a.toCompletableFuture(); final CompletableFuture bf = b.toCompletableFuture(); final CompletableFuture cf = c.toCompletableFuture(); final CompletableFuture df = d.toCompletableFuture(); final CompletableFuture ef = e.toCompletableFuture(); final CompletableFuture ff = f.toCompletableFuture(); return CompletableFuture.allOf(af, bf, cf, df, ef, ff) .thenCompose(ignored -> function.apply(af.join(), bf.join(), cf.join(), df.join(), ef.join(), ff.join())); } /** * Polls an external resource periodically until it returns a non-empty result. * *

The polling task should return {@code Optional.empty()} until it becomes available, and * then {@code Optional.of(result)}. If the polling task throws an exception or returns null, * that will cause the result future to complete exceptionally. * *

Canceling the returned future will cancel the scheduled polling task as well. * *

Note that on a ScheduledThreadPoolExecutor the polling task might remain allocated for up * to {@code frequency} time after completing or being cancelled. If you have lots of polling * operations or a long polling frequency, consider setting {@code removeOnCancelPolicy} to true. * See {@link java.util.concurrent.ScheduledThreadPoolExecutor#setRemoveOnCancelPolicy(boolean)}. * * @param pollingTask the polling task * @param frequency the frequency to run the polling task at * @param executorService the executor service to schedule the polling task on * @param the type of the result of the polling task, that will be returned when * the task succeeds. * @return a future completing to the result of the polling task once that becomes available */ public static CompletableFuture poll( final Supplier> pollingTask, final Duration frequency, final ScheduledExecutorService executorService) { final CompletableFuture result = new CompletableFuture<>(); final ScheduledFuture scheduled = executorService.scheduleAtFixedRate( () -> pollTask(pollingTask, result), 0, frequency.toMillis(), TimeUnit.MILLISECONDS); result.whenComplete((r, ex) -> scheduled.cancel(true)); return result; } private static void pollTask( final Supplier> pollingTask, final CompletableFuture resultFuture) { try { pollingTask.get().ifPresent(resultFuture::complete); } catch (Exception ex) { resultFuture.completeExceptionally(ex); } } }