org.hibernate.search.util.impl.Futures Maven / Gradle / Ivy
/*
* Hibernate Search, full-text search for your domain model
*
* License: GNU Lesser General Public License (LGPL), version 2.1 or later
* See the lgpl.txt file in the root directory or .
*/
package org.hibernate.search.util.impl;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CompletionException;
import java.util.concurrent.CompletionStage;
import java.util.function.BiConsumer;
import java.util.function.BiFunction;
import java.util.function.Function;
import java.util.function.Supplier;
/**
* @author Yoann Rodiere
*/
public final class Futures {
private Futures() {
}
/**
* Create a {@link CompletableFuture} using the given supplier.
*
* This method is guaranteed to never throw any exception: any exception thrown by
* the given supplier will instead complete the resulting future
* {@link CompletableFuture#completeExceptionally(Throwable) exceptionally}.
*
* This is useful in particular if you want to handle errors during the {@link CompletableFuture}
* creation the same way as errors thrown during post-processing operations
* (for instance operations passed to {@link CompletableFuture#thenApply(Function)}).
*
* @param initiator A supplier that will initiate (synchronously) the asynchronous operation.
* @return A {@link CompletableFuture} wrapping the result of both the initiation and execution of the operation.
*/
public static CompletableFuture create(Supplier> initiator) {
return CompletableFuture.completedFuture( null ).thenCompose( ignored -> initiator.get() );
}
/**
* Creates a future handler that will delegate to the given {@link Function}
* after having unwrapped the throwable passed as input if it is a {@link CompletionException}.
*
* This method is meant to be used in conjunction with {@link CompletableFuture#exceptionally(Function)}.
*
* @param delegate The exception function to delegate to
* @return The new, delegating exception function.
*/
public static Function handler(Function delegate) {
return throwable -> {
if ( throwable instanceof CompletionException ) {
throwable = throwable.getCause();
}
return delegate.apply( throwable );
};
}
/**
* Creates a future handler that will delegate to the given {@link BiFunction}
* after having unwrapped the throwable passed as input if it is a {@link CompletionException}.
*
* This method is meant to be used in conjunction with {@link CompletableFuture#handle(BiFunction)}.
*
* @param delegate The handler to delegate to
* @return The new, delegating handler.
*/
public static BiFunction handler(BiFunction delegate) {
return (result, throwable) -> {
if ( throwable instanceof CompletionException ) {
throwable = throwable.getCause();
}
return delegate.apply( result, throwable );
};
}
/**
* Creates a future handler that will delegate to the given {@link BiConsumer}
* after having unwrapped the throwable passed as input if it is a {@link CompletionException}.
*
* This method is meant to be used in conjunction with {@link CompletableFuture#whenComplete(BiConsumer)}.
*
* @param delegate The handler to delegate to
* @return The new, delegating handler.
*/
public static BiConsumer handler(BiConsumer delegate) {
return (result, throwable) -> {
if ( throwable instanceof CompletionException ) {
throwable = throwable.getCause();
}
delegate.accept( result, throwable );
};
}
/**
* Creates a future handler that will copy the state of the handled future
* to the given future.
*
* This method is meant to be used in conjunction with {@link CompletableFuture#whenComplete(BiConsumer)}.
*
* @param copyFuture The future to copy to
* @return the copy handler
*/
public static BiConsumer copyHandler(CompletableFuture copyFuture) {
return (result, throwable) -> {
if ( throwable != null ) {
copyFuture.completeExceptionally( throwable );
}
else {
copyFuture.complete( result );
}
};
}
/**
* Creates a composition function that will delegate to the given {@link Function}
* but will catch any exception during composition to return a future completed exceptionally.
*
* This method is meant to be used in conjunction with {@link CompletableFuture#thenCompose(Function)}.
* It is useful in particular when you want to apply the same error handling to the composition
* function and to the resulting future.
*
* @param delegate The composition function to delegate to.
* @return The new, delegating composition function .
*/
public static Function> safeComposer(Function super T, ? extends CompletionStage> delegate) {
return result -> {
try {
return delegate.apply( result );
}
catch (Throwable t) {
CompletableFuture future = new CompletableFuture<>();
future.completeExceptionally( t );
return future;
}
};
}
/**
* Compose the given future with another as soon as it's complete,
* regardless of errors, and return a completable future that
* will take errors of both parent futures into account
* (using {@link Throwable#addSuppressed(Throwable)} if need be).
*
* @param self The future to wait for before launching the next one
* @param action the composition consumer
* @return A completable future that will be complete once {@code self} finished executing and
* {@code action} and its resulting future finished executing.
*/
public static CompletableFuture whenCompleteExecute(CompletableFuture> self, Supplier extends CompletionStage> action) {
return self.handle( handler( (result, throwable) -> throwable ) )
.thenCompose( throwable -> {
CompletionStage stage;
try {
stage = action.get();
}
catch (Throwable otherThrowable) {
CompletableFuture future = new CompletableFuture<>();
future.completeExceptionally( otherThrowable );
stage = future;
}
if ( throwable != null ) {
return stage.handle( Futures.