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

com.landawn.abacus.android.util.CompletableFuture Maven / Gradle / Ivy

/*
 * Copyright (C) 2016 HaiYang Li
 *
 * 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 com.landawn.abacus.android.util;

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

import com.landawn.abacus.logging.Logger;
import com.landawn.abacus.logging.LoggerFactory;
import com.landawn.abacus.util.DateUtil;
import com.landawn.abacus.util.N;
import com.landawn.abacus.util.Pair;
import com.landawn.abacus.util.Try;
import com.landawn.abacus.util.Tuple;
import com.landawn.abacus.util.Tuple.Tuple4;

/**
 * 
* TP -> Thread Pool * * @since 0.8 * * @author Haiyang Li */ public class CompletableFuture implements Future { static final Logger logger = LoggerFactory.getLogger(CompletableFuture.class); final Future future; final List> upFutures; final Executor asyncExecutor; CompletableFuture(final Future future, final List> upFutures, final Executor asyncExecutor) { this.future = future; this.upFutures = upFutures; this.asyncExecutor = asyncExecutor; } // public static CompletableFuture run(final Try.Runnable action) { // return run(action, Async.SERIAL_EXECUTOR); // } // // public static CompletableFuture run(final Try.Callable action) { // return run(action, Async.SERIAL_EXECUTOR); // } // // public static CompletableFuture run(final Try.Runnable action, final Executor executor) { // final FutureTask futureTask = new FutureTask<>(action, null); // // executor.execute(futureTask); // // return new CompletableFuture<>(futureTask, null, executor); // } // // public static CompletableFuture run(final Try.Callable action, final Executor executor) { // final FutureTask futureTask = new FutureTask<>(action); // // executor.execute(futureTask); // // return new CompletableFuture<>(futureTask, null, executor); // } /** * * @param result * @param asyncExecutor * @return a CompletableFuture which is already done by passing the result to it directly. */ public static CompletableFuture completed(final T result) { return new CompletableFuture<>(new Future() { @Override public boolean cancel(boolean mayInterruptIfRunning) { return false; } @Override public boolean isCancelled() { return false; } @Override public boolean isDone() { return true; } @Override public T get() { return result; } @Override public T get(final long timeout, final TimeUnit unit) { return result; } }, null, Async.SERIAL_EXECUTOR); } @Override public boolean cancel(boolean mayInterruptIfRunning) { return future.cancel(mayInterruptIfRunning); } @Override public boolean isCancelled() { return future.isCancelled(); } /** * Cancel this future and all the previous stage future recursively. * * @param mayInterruptIfRunning * @return */ public boolean cancelAll(boolean mayInterruptIfRunning) { boolean res = true; if (N.notNullOrEmpty(upFutures)) { for (CompletableFuture preFuture : upFutures) { res = res & preFuture.cancelAll(mayInterruptIfRunning); } } return cancel(mayInterruptIfRunning) && res; } /** * Returns true if this future and all previous stage futures have been recursively cancelled, otherwise false is returned. * * @return */ public boolean isAllCancelled() { boolean res = true; if (N.notNullOrEmpty(upFutures)) { for (CompletableFuture preFuture : upFutures) { res = res & preFuture.isAllCancelled(); } } return isCancelled() && res; } @Override public boolean isDone() { return future.isDone(); } @Override public T get() throws InterruptedException, ExecutionException { return future.get(); } @Override public T get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { return future.get(timeout, unit); } public Pair gett() { try { return Pair.of(get(), null); } catch (Exception e) { return Pair.of(null, e); } } public Pair gett(final long timeout, final TimeUnit unit) { try { return Pair.of(get(timeout, unit), null); } catch (Exception e) { return Pair.of(null, e); } } public U get(final Try.Function action) throws E { try { return action.apply(get()); } catch (InterruptedException | ExecutionException e) { throw N.toRuntimeException(e); } } public U get(final long timeout, final TimeUnit unit, final Try.Function action) throws E { try { return action.apply(get(timeout, unit)); } catch (InterruptedException | ExecutionException | TimeoutException e) { throw N.toRuntimeException(e); } } public U get(final Try.BiFunction action) throws E { final Pair result = gett(); return action.apply(result.left, result.right); } public U get(final long timeout, final TimeUnit unit, final Try.BiFunction action) throws E { final Pair result = gett(timeout, unit); return action.apply(result.left, result.right); } public T getNow(T defaultValue) { try { return isDone() ? get() : defaultValue; } catch (InterruptedException | ExecutionException e) { throw N.toRuntimeException(e); } } public void getAndThen(final Try.Consumer action) throws InterruptedException, ExecutionException, E { action.accept(get()); } public void getAndThen(final long timeout, final TimeUnit unit, final Try.Consumer action) throws InterruptedException, ExecutionException, TimeoutException, E { action.accept(get(timeout, unit)); } // public void complete() throws InterruptedException, ExecutionException { // get(); // } // // public void complete(Try.Consumer action) { // try { // action.accept(get()); // } catch (InterruptedException | ExecutionException e) { // throw N.toRuntimeException(e); // } // } // // public void complete(Try.BiConsumer action) { // final Pair result = gett(); // action.accept(result.left, result.right); // } CompletableFuture thenApply(final Try.Function action) { return new CompletableFuture(new Future() { @Override public boolean cancel(boolean mayInterruptIfRunning) { return future.cancel(mayInterruptIfRunning); } @Override public boolean isCancelled() { return future.isCancelled(); } @Override public boolean isDone() { return future.isDone(); } @Override public U get() throws InterruptedException, ExecutionException { try { return action.apply(future.get()); } catch (InterruptedException | ExecutionException e) { throw e; } catch (Exception e) { throw N.toRuntimeException(e); } } @Override public U get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { try { return action.apply(future.get(timeout, unit)); } catch (InterruptedException | ExecutionException | TimeoutException e) { throw e; } catch (Exception e) { throw N.toRuntimeException(e); } } }, null, asyncExecutor) { @Override public boolean cancelAll(boolean mayInterruptIfRunning) { return super.cancelAll(mayInterruptIfRunning); } @Override public boolean isAllCancelled() { return super.isAllCancelled(); } }; } // CompletableFuture thenApply(final BiFunction action) { // return new CompletableFuture(new Future() { // @Override // public boolean cancel(boolean mayInterruptIfRunning) { // return future.cancel(mayInterruptIfRunning); // } // // @Override // public boolean isCancelled() { // return future.isCancelled(); // } // // @Override // public boolean isDone() { // return future.isDone(); // } // // @Override // public U get() throws InterruptedException, ExecutionException { // final Pair result = gett(); // // return action.apply(result.left, result.right); // } // // @Override // public U get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { // final Pair result = gett(timeout, unit); // // return action.apply(result.left, result.right); // } // }, null, asyncExecutor) { // @Override // public boolean cancelAll(boolean mayInterruptIfRunning) { // return super.cancelAll(mayInterruptIfRunning); // } // // @Override // public boolean isAllCancelled() { // return super.isAllCancelled(); // } // }; // } // public CompletableFuture thenAccept(final Try.Consumer action) { // return thenApply(new Function() { // @Override // public Void apply(T t) { // action.accept(t); // return null; // } // }); // } // // public CompletableFuture thenAccept(final Try.BiConsumer action) { // return thenApply(new BiFunction() { // @Override // public Void apply(T t, Exception e) { // action.accept(t, e); // return null; // } // }); // } // // public CompletableFuture thenCombine(final CompletableFuture other, final BiFunction action) { // return new CompletableFuture(new Future() { // @Override // public boolean cancel(boolean mayInterruptIfRunning) { // return future.cancel(mayInterruptIfRunning) && other.future.cancel(mayInterruptIfRunning); // } // // @Override // public boolean isCancelled() { // return future.isCancelled() || other.future.isCancelled(); // } // // @Override // public boolean isDone() { // return future.isDone() && other.future.isDone(); // } // // @Override // public R get() throws InterruptedException, ExecutionException { // return action.apply(future.get(), other.future.get()); // } // // @Override // public R get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { // final long timeoutInMillis = unit.toMillis(timeout); // final long now = N.currentMillis(); // final long endTime = timeoutInMillis > Long.MAX_VALUE - now ? Long.MAX_VALUE : now + timeoutInMillis; // // final T result = future.get(timeout, unit); // final U otherResult = other.future.get(N.max(0, endTime - N.currentMillis()), TimeUnit.MILLISECONDS); // // return action.apply(result, otherResult); // } // }, null, asyncExecutor) { // @Override // public boolean cancelAll(boolean mayInterruptIfRunning) { // return super.cancelAll(mayInterruptIfRunning) && other.cancelAll(mayInterruptIfRunning); // } // // @Override // public boolean isAllCancelled() { // return super.isAllCancelled() && other.isAllCancelled(); // } // }; // } // // public CompletableFuture thenCombine(final CompletableFuture other, final Try.Function, R, E> action) { // return new CompletableFuture(new Future() { // @Override // public boolean cancel(boolean mayInterruptIfRunning) { // return future.cancel(mayInterruptIfRunning) && other.future.cancel(mayInterruptIfRunning); // } // // @Override // public boolean isCancelled() { // return future.isCancelled() || other.future.isCancelled(); // } // // @Override // public boolean isDone() { // return future.isDone() && other.future.isDone(); // } // // @Override // public R get() throws InterruptedException, ExecutionException { // final Pair result = gett(); // final Pair result2 = other.gett(); // // return action.apply(Tuple.of(result.left, result.right, (U) result2.left, result.right)); // } // // @Override // public R get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { // final long timeoutInMillis = unit.toMillis(timeout); // final long now = N.currentMillis(); // final long endTime = timeoutInMillis > Long.MAX_VALUE - now ? Long.MAX_VALUE : now + timeoutInMillis; // // final Pair result = CompletableFuture.this.gett(timeout, unit); // final Pair result2 = other.gett(N.max(0, endTime - N.currentMillis()), TimeUnit.MILLISECONDS); // // return action.apply(Tuple.of(result.left, result.right, (U) result2.left, result.right)); // } // }, null, asyncExecutor) { // @Override // public boolean cancelAll(boolean mayInterruptIfRunning) { // return super.cancelAll(mayInterruptIfRunning) && other.cancelAll(mayInterruptIfRunning); // } // // @Override // public boolean isAllCancelled() { // return super.isAllCancelled() && other.isAllCancelled(); // } // }; // } // // public CompletableFuture thenAcceptBoth(final CompletableFuture other, final BiConsumer action) { // return thenCombine(other, new BiFunction() { // @Override // public Void apply(T t, U u) { // action.accept(t, u); // return null; // } // }); // } // // public CompletableFuture thenAcceptBoth(final CompletableFuture other, final Try.Consumer, E> action) { // return thenCombine(other, new Function, Void>() { // @Override // public Void apply(Tuple4 t) { // action.accept(t); // return null; // } // }); // } public CompletableFuture thenRun(final Try.Runnable action) { return execute(new Callable() { @Override public Void call() throws Exception { get(); action.run(); return null; } }); } public CompletableFuture thenRun(final Try.Consumer action) { return execute(new Callable() { @Override public Void call() throws Exception { action.accept(get()); return null; } }); } public CompletableFuture thenRun(final Try.BiConsumer action) { return execute(new Callable() { @Override public Void call() throws Exception { final Pair result = gett(); action.accept(result.left, result.right); return null; } }); } public CompletableFuture thenCall(final Callable action) { return execute(new Callable() { @Override public R call() throws Exception { get(); return action.call(); } }); } public CompletableFuture thenCall(final Try.Function action) { return execute(new Callable() { @Override public R call() throws Exception { return action.apply(get()); } }); } public CompletableFuture thenCall(final Try.BiFunction action) { return execute(new Callable() { @Override public R call() throws Exception { final Pair result = gett(); return action.apply(result.left, result.right); } }); } public CompletableFuture thenRunOnUI(final Try.Runnable action) { return withUIExecutor().thenRun(action); } public CompletableFuture thenRunOnUI(final Try.Consumer action) { return withUIExecutor().thenRun(action); } public CompletableFuture thenRunOnUI(final Try.BiConsumer action) { return withUIExecutor().thenRun(action); } public CompletableFuture thenCallOnUI(final Callable action) { return withUIExecutor().thenCall(action); } public CompletableFuture thenCallOnUI(final Try.Function action) { return withUIExecutor().thenCall(action); } public CompletableFuture thenCallOnUI(final Try.BiFunction action) { return withUIExecutor().thenCall(action); } public CompletableFuture thenRunByTP(final Try.Runnable action) { return withTPExecutor().thenRun(action); } public CompletableFuture thenRunByTP(final Try.Consumer action) { return withTPExecutor().thenRun(action); } public CompletableFuture thenRunByTP(final Try.BiConsumer action) { return withTPExecutor().thenRun(action); } public CompletableFuture thenCallByTP(final Callable action) { return withTPExecutor().thenCall(action); } public CompletableFuture thenCallByTP(final Try.Function action) { return withTPExecutor().thenCall(action); } public CompletableFuture thenCallByTP(final Try.BiFunction action) { return withTPExecutor().thenCall(action); } public CompletableFuture runAfterBoth(final CompletableFuture other, final Try.Runnable action) { return execute(new Callable() { @Override public Void call() throws Exception { get(); other.get(); action.run(); return null; } }, other); } public CompletableFuture runAfterBoth(final CompletableFuture other, final Try.BiConsumer action) { return execute(new Callable() { @Override public Void call() throws Exception { action.accept(get(), other.get()); return null; } }, other); } public CompletableFuture runAfterBoth(final CompletableFuture other, final Try.Consumer, E> action) { return execute(new Callable() { @Override public Void call() throws Exception { final Pair result = gett(); final Pair result2 = other.gett(); action.accept(Tuple.of(result.left, result.right, result2.left, result.right)); return null; } }, other); } public CompletableFuture callAfterBoth(final CompletableFuture other, final Callable action) { return execute(new Callable() { @Override public R call() throws Exception { get(); other.get(); return action.call(); } }, other); } public CompletableFuture callAfterBoth(final CompletableFuture other, final Try.BiFunction action) { return execute(new Callable() { @Override public R call() throws Exception { return action.apply(get(), other.get()); } }, other); } public CompletableFuture callAfterBoth(final CompletableFuture other, final Try.Function, R, E> action) { return execute(new Callable() { @Override public R call() throws Exception { final Pair result = gett(); final Pair result2 = other.gett(); return action.apply(Tuple.of(result.left, result.right, result2.left, result.right)); } }, other); } public CompletableFuture runOnUIAfterBoth(final CompletableFuture other, final Try.Runnable action) { return withUIExecutor().runAfterBoth(other, action); } public CompletableFuture runOnUIAfterBoth(final CompletableFuture other, final Try.BiConsumer action) { return withUIExecutor().runAfterBoth(other, action); } public CompletableFuture runOnUIAfterBoth(final CompletableFuture other, final Try.Consumer, E> action) { return withUIExecutor().runAfterBoth(other, action); } public CompletableFuture callOnUIAfterBoth(final CompletableFuture other, final Callable action) { return withUIExecutor().callAfterBoth(other, action); } public CompletableFuture callOnUIAfterBoth(final CompletableFuture other, final Try.BiFunction action) { return withUIExecutor().callAfterBoth(other, action); } public CompletableFuture callOnUIAfterBoth(final CompletableFuture other, final Try.Function, R, E> action) { return withUIExecutor().callAfterBoth(other, action); } public CompletableFuture runByTPAfterBoth(final CompletableFuture other, final Try.Runnable action) { return withTPExecutor().runAfterBoth(other, action); } public CompletableFuture runByTPAfterBoth(final CompletableFuture other, final Try.BiConsumer action) { return withTPExecutor().runAfterBoth(other, action); } public CompletableFuture runByTPAfterBoth(final CompletableFuture other, final Try.Consumer, E> action) { return withTPExecutor().runAfterBoth(other, action); } public CompletableFuture callByTPAfterBoth(final CompletableFuture other, final Callable action) { return withTPExecutor().callAfterBoth(other, action); } public CompletableFuture callByTPAfterBoth(final CompletableFuture other, final Try.BiFunction action) { return withTPExecutor().callAfterBoth(other, action); } public CompletableFuture callByTPAfterBoth(final CompletableFuture other, final Try.Function, R, E> action) { return withTPExecutor().callAfterBoth(other, action); } public CompletableFuture runAfterEither(final CompletableFuture other, final Try.Runnable action) { return execute(new Callable() { @Override public Void call() throws Exception { Futures.anyOf(N.asList(CompletableFuture.this, other)).get(); action.run(); return null; } }, other); } public CompletableFuture runAfterEither(final CompletableFuture other, final Try.Consumer action) { return execute(new Callable() { @Override public Void call() throws Exception { final T result = Futures.anyOf(N.asList(CompletableFuture.this, other)).get(); action.accept(result); return null; } }, other); } public CompletableFuture runAfterEither(final CompletableFuture other, final Try.BiConsumer action) { return execute(new Callable() { @Override public Void call() throws Exception { final Pair result = Futures.anyOf(N.asList(CompletableFuture.this, other)).gett(); action.accept(result.left, result.right); return null; } }, other); } public CompletableFuture callAfterEither(final CompletableFuture other, final Callable action) { return execute(new Callable() { @Override public R call() throws Exception { Futures.anyOf(N.asList(CompletableFuture.this, other)).get(); return action.call(); } }, other); } public CompletableFuture callAfterEither(final CompletableFuture other, final Try.Function action) { return execute(new Callable() { @Override public R call() throws Exception { final T result = Futures.anyOf(N.asList(CompletableFuture.this, other)).get(); return action.apply(result); } }, other); } public CompletableFuture callAfterEither(final CompletableFuture other, final Try.BiFunction action) { return execute(new Callable() { @Override public R call() throws Exception { final Pair result = Futures.anyOf(N.asList(CompletableFuture.this, other)).gett(); return action.apply(result.left, result.right); } }, other); } public CompletableFuture runOnUIAfterEither(final CompletableFuture other, final Try.Runnable action) { return withUIExecutor().runAfterEither(other, action); } public CompletableFuture runOnUIAfterEither(final CompletableFuture other, final Try.Consumer action) { return withUIExecutor().runAfterEither(other, action); } public CompletableFuture runOnUIAfterEither(final CompletableFuture other, final Try.BiConsumer action) { return withUIExecutor().runAfterEither(other, action); } public CompletableFuture callOnUIAfterEither(final CompletableFuture other, final Callable action) { return withUIExecutor().callAfterEither(other, action); } public CompletableFuture callOnUIAfterEither(final CompletableFuture other, final Try.Function action) { return withUIExecutor().callAfterEither(other, action); } public CompletableFuture callOnUIAfterEither(final CompletableFuture other, final Try.BiFunction action) { return withUIExecutor().callAfterEither(other, action); } public CompletableFuture runByTPAfterEither(final CompletableFuture other, final Try.Runnable action) { return withTPExecutor().runAfterEither(other, action); } public CompletableFuture runByTPAfterEither(final CompletableFuture other, final Try.Consumer action) { return withTPExecutor().runAfterEither(other, action); } public CompletableFuture runByTPAfterEither(final CompletableFuture other, final Try.BiConsumer action) { return withTPExecutor().runAfterEither(other, action); } public CompletableFuture callByTPAfterEither(final CompletableFuture other, final Callable action) { return withTPExecutor().callAfterEither(other, action); } public CompletableFuture callByTPAfterEither(final CompletableFuture other, final Try.Function action) { return withTPExecutor().callAfterEither(other, action); } public CompletableFuture callByTPAfterEither(final CompletableFuture other, final Try.BiFunction action) { return withTPExecutor().callAfterEither(other, action); } // /** // * Returns a new CompletableFuture that, when either this or the // * other given CompletableFuture complete normally. If both of the given // * CompletableFutures complete exceptionally, then the returned // * CompletableFuture also does so. // * // * @param other // * @param action // * @return // */ // public CompletableFuture applyToEither(final CompletableFuture other, final Function action) { // return Futures.anyOf(N.asList(this, other)).thenApply(action); // } // // /** // * Returns a new CompletableFuture that, when either this or the // * other given CompletableFuture complete normally. If both of the given // * CompletableFutures complete exceptionally, then the returned // * CompletableFuture also does so. // * // * @param other // * @param action // * @return // */ // public CompletableFuture acceptEither(final CompletableFuture other, final Try.Consumer action) { // return Futures.anyOf(N.asList(this, other)).thenAccept(action); // } // /** // * Returns a new CompletableFuture that, when this CompletableFuture completes // * exceptionally, is executed with this CompletableFuture's exception as the // * argument to the supplied function. Otherwise, if this CompletableFuture // * completes normally, then the returned CompletableFuture also completes // * normally with the same value. // * // * @param action // * @return // */ // public CompletableFuture exceptionally(final Function action) { // return new CompletableFuture(new Future() { // @Override // public boolean cancel(boolean mayInterruptIfRunning) { // return future.cancel(mayInterruptIfRunning); // } // // @Override // public boolean isCancelled() { // return future.isCancelled(); // } // // @Override // public boolean isDone() { // return future.isDone(); // } // // @Override // public T get() throws InterruptedException, ExecutionException { // try { // return future.get(); // } catch (Exception e) { // return action.apply(e); // } // } // // @Override // public T get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { // try { // return future.get(timeout, unit); // } catch (Exception e) { // return action.apply(e); // } // } // }, null, asyncExecutor) { // @Override // public boolean cancelAll(boolean mayInterruptIfRunning) { // return super.cancelAll(mayInterruptIfRunning); // } // // @Override // public boolean isAllCancelled() { // return super.isAllCancelled(); // } // }; // } // public CompletableFuture whenComplete(final BiConsumer action) { // return new CompletableFuture<>(new Future() { // @Override // public boolean cancel(boolean mayInterruptIfRunning) { // return future.cancel(mayInterruptIfRunning); // } // // @Override // public boolean isCancelled() { // return future.isCancelled(); // } // // @Override // public boolean isDone() { // return future.isDone(); // } // // @Override // public T get() throws InterruptedException, ExecutionException { // final Pair result = gett(); // // if (result.right != null) { // try { // action.accept(result.left, result.right); // } catch (Exception e) { // // ignore. // } // // if (result.right instanceof InterruptedException) { // throw ((InterruptedException) result.right); // } else if (result.right instanceof ExecutionException) { // throw ((ExecutionException) result.right); // } else { // throw N.toRuntimeException(result.right); // } // } else { // action.accept(result.left, result.right); // return result.left; // } // } // // @Override // public T get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { // final Pair result = gett(timeout, unit); // // if (result.right != null) { // try { // action.accept(result.left, result.right); // } catch (Exception e) { // // ignore. // } // // if (result.right instanceof InterruptedException) { // throw ((InterruptedException) result.right); // } else if (result.right instanceof ExecutionException) { // throw ((ExecutionException) result.right); // } else { // throw N.toRuntimeException(result.right); // } // } else { // action.accept(result.left, result.right); // return result.left; // } // } // }, asyncExecutor); // } // // public CompletableFuture handle(final BiFunction action) { // return new CompletableFuture<>(new Future() { // @Override // public boolean cancel(boolean mayInterruptIfRunning) { // return future.cancel(mayInterruptIfRunning); // } // // @Override // public boolean isCancelled() { // return future.isCancelled(); // } // // @Override // public boolean isDone() { // return future.isDone(); // } // // @Override // public U get() throws InterruptedException, ExecutionException { // final Pair result = gett(); // return action.apply(result.left, result.right); // } // // @Override // public U get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { // final Pair result = gett(timeout, unit); // return action.apply(result.left, result.right); // } // }, asyncExecutor); // } private CompletableFuture execute(final Callable command) { return execute(command, null); } private CompletableFuture execute(final Callable command, final CompletableFuture other) { return execute(new FutureTask<>(command), other); } private CompletableFuture execute(final FutureTask futureTask, final CompletableFuture other) { asyncExecutor.execute(futureTask); @SuppressWarnings("rawtypes") final List> upFutures = other == null ? (List) Arrays.asList(this) : Arrays.asList(this, other); return new CompletableFuture<>(futureTask, upFutures, asyncExecutor); } public CompletableFuture delayed(long delay, TimeUnit unit) { if (delay <= 0) { return this; } return with(asyncExecutor, delay, TimeUnit.MILLISECONDS); } public CompletableFuture with(Executor executor) { return with(executor, 0, TimeUnit.MILLISECONDS); } public CompletableFuture with(final Executor executor, final long delay, final TimeUnit unit) { N.requireNonNull(executor); return new CompletableFuture(new Future() { private final long delayEndTime = DateUtil.currentMillis() + unit.toMillis(delay); private volatile boolean isDelayed = false; @Override public boolean cancel(boolean mayInterruptIfRunning) { return future.cancel(mayInterruptIfRunning); } @Override public boolean isCancelled() { return future.isCancelled(); } @Override public boolean isDone() { boolean isDone = future.isDone(); if (isDone) { delay(); } return isDone; } @Override public T get() throws InterruptedException, ExecutionException { delay(); return future.get(); } @Override public T get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException { delay(); return future.get(timeout, unit); } private void delay() { if (isDelayed == false) { isDelayed = true; N.sleep(delayEndTime - DateUtil.currentMillis()); } } }, null, executor) { @Override public boolean cancelAll(boolean mayInterruptIfRunning) { return super.cancelAll(mayInterruptIfRunning); } @Override public boolean isAllCancelled() { return super.isAllCancelled(); } }; } public CompletableFuture withUIExecutor() { return with(Async.UI_EXECUTOR); } public CompletableFuture withUIExecutor(final long delay) { return with(Async.UI_EXECUTOR, delay, TimeUnit.MILLISECONDS); } public CompletableFuture withSerialExecutor() { return with(Async.SERIAL_EXECUTOR); } public CompletableFuture withSerialExecutor(final long delay) { return with(Async.SERIAL_EXECUTOR, delay, TimeUnit.MILLISECONDS); } /** * With Thread Pool Executor. * @return */ public CompletableFuture withTPExecutor() { return with(Async.TP_EXECUTOR); } public CompletableFuture withTPExecutor(final long delay) { return with(Async.TP_EXECUTOR, delay, TimeUnit.MILLISECONDS); } }