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

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> 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> 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.handler( (ignored, otherThrowable) -> { throw wrap( Throwables.combine( throwable, otherThrowable) ); }) ); } else { return stage; } } ); } private static RuntimeException wrap(Throwable throwable) { if ( throwable instanceof RuntimeException ) { return (RuntimeException) throwable; } else { return new CompletionException( throwable ); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy