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

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

There is a newer version: 1.10.1
Show newest version
/*
 * 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.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.ExecutorService;
import java.util.concurrent.Executors;
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.Tuple.Tuple4;
import com.landawn.abacus.util.function.BiConsumer;
import com.landawn.abacus.util.function.BiFunction;
import com.landawn.abacus.util.function.Consumer;
import com.landawn.abacus.util.function.Function;

/**
 * 
 * @since 0.8
 * 
 * @author Haiyang Li
 * 
 * @see https://docs.oracle.com/javase/8/docs/api/java/util/concurrent/CompletableFuture.html
 */
public class CompletableFuture implements Future {
    private static final Logger logger = LoggerFactory.getLogger(CompletableFuture.class);

    private static final ExecutorService commonPool = Executors.newFixedThreadPool(64);

    static {
        Runtime.getRuntime().addShutdownHook(new Thread() {
            @Override
            public void run() {
                try {
                    commonPool.shutdown();
                    commonPool.awaitTermination(180, TimeUnit.SECONDS);
                } catch (Throwable e) {
                    logger.error("Failed to commit the tasks in queue in ExecutorService before shutdown", e);
                }
            }
        });
    }

    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 Runnable action) {
        return run(action, commonPool);
    }

    public static  CompletableFuture run(final Try.Callable action) {
        return run(action, commonPool);
    }

    public static CompletableFuture run(final 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, commonPool);
    }

    @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 get2() {
        try {
            return Pair.of(get(), null);
        } catch (Throwable e) {
            return Pair.of(null, e);
        }
    }

    public Pair get2(final long timeout, final TimeUnit unit) {
        try {
            return Pair.of(get(timeout, unit), null);
        } catch (Throwable e) {
            return Pair.of(null, e);
        }
    }

    public T getNow(T defaultValue) {
        try {
            return isDone() ? get() : defaultValue;
        } catch (InterruptedException | ExecutionException e) {
            throw N.toRuntimeException(e);
        }
    }

    public  U get(final Function action) {
        try {
            return action.apply(get());
        } catch (InterruptedException | ExecutionException e) {
            throw N.toRuntimeException(e);
        }
    }

    public  U get(final long timeout, final TimeUnit unit, final Function action) {
        try {
            return action.apply(get(timeout, unit));
        } catch (InterruptedException | ExecutionException | TimeoutException e) {
            throw N.toRuntimeException(e);
        }
    }

    public  U get(final BiFunction action) {
        final Pair result = get2();
        return action.apply(result.left, result.right);
    }

    public  U get(final long timeout, final TimeUnit unit, final BiFunction action) {
        final Pair result = get2(timeout, unit);
        return action.apply(result.left, result.right);
    }

    //    public void complete() throws InterruptedException, ExecutionException {
    //        get();
    //    }
    //
    //    public void complete(Consumer action) {
    //        try {
    //            action.accept(get());
    //        } catch (InterruptedException | ExecutionException e) {
    //            throw N.toRuntimeException(e);
    //        }
    //    }
    //
    //    public void complete(BiConsumer action) {
    //        final Pair result = get2();
    //        action.accept(result.left, result.right);
    //    }

     CompletableFuture thenApply(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 U get() throws InterruptedException, ExecutionException {
                return action.apply(future.get());
            }

            @Override
            public U get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
                return action.apply(future.get(timeout, unit));
            }
        }, null, asyncExecutor) {
            @Override
            public boolean cancelAll(boolean mayInterruptIfRunning) {
                return super.cancelAll(mayInterruptIfRunning);
            }

            @Override
            public boolean isAllCancelled() {
                return super.isAllCancelled();
            }
        };
    }

    //    public  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 = get2();
    //                return action.apply(result.left, result.right);
    //            }
    //
    //            @Override
    //            public U get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
    //                final Pair result = get2(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 Consumer action) {
    //        return thenApply(new Function() {
    //            @Override
    //            public Void apply(T t) {
    //                action.accept(t);
    //                return null;
    //            }
    //        });
    //    }
    //
    //    public  CompletableFuture thenAccept(final BiConsumer action) {
    //        return thenApply(new BiFunction() {
    //            @Override
    //            public Void apply(T t, Throwable 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 Function, R> 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 = get2();
    //                final Pair result2 = other.get2();
    //
    //                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.get2(timeout, unit);
    //                final Pair result2 = other.get2(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 Consumer> action) {
    //        return thenCombine(other, new Function, Void>() {
    //            @Override
    //            public Void apply(Tuple4 t) {
    //                action.accept(t);
    //                return null;
    //            }
    //        });
    //    }

    public CompletableFuture thenRun(final Runnable action) {
        return execute(new Callable() {
            @Override
            public Void call() throws Exception {
                get();
                action.run();
                return null;
            }
        });
    }

    public CompletableFuture thenRun(final Consumer action) {
        return execute(new Callable() {
            @Override
            public Void call() throws Exception {
                action.accept(get());
                return null;
            }
        });
    }

    public CompletableFuture thenRun(final BiConsumer action) {
        return execute(new Runnable() {
            @Override
            public void run() {
                final Pair result = get2();
                action.accept(result.left, result.right);
            }
        });
    }

    public  CompletableFuture thenCall(final Try.Callable action) {
        return execute(new Callable() {
            @Override
            public U call() throws Exception {
                get();
                return action.call();
            }
        });
    }

    public  CompletableFuture thenCall(final Function action) {
        return execute(new Callable() {
            @Override
            public R call() throws Exception {
                return action.apply(get());
            }
        });
    }

    public  CompletableFuture thenCall(final BiFunction action) {
        return execute(new Callable() {
            @Override
            public R call() throws Exception {
                final Pair result = get2();
                return action.apply(result.left, result.right);
            }
        });
    }

    public CompletableFuture runAfterBoth(final CompletableFuture other, final 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 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 Consumer> action) {
        return execute(new Runnable() {
            @Override
            public void run() {
                final Pair result = get2();
                final Pair result2 = other.get2();

                action.accept(Tuple.of(result.left, result.right, result2.left, result.right));
            }
        }, other);
    }

    public  CompletableFuture callAfterBoth(final CompletableFuture other, final Try.Callable action) {
        return execute(new Callable() {
            @Override
            public U call() throws Exception {
                get();
                other.get();
                return action.call();
            }
        }, other);
    }

    public  CompletableFuture callAfterBoth(final CompletableFuture other, final 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 Function, R> action) {
        return execute(new Callable() {
            @Override
            public R call() throws Exception {
                final Pair result = get2();
                final Pair result2 = other.get2();

                return action.apply(Tuple.of(result.left, result.right, result2.left, result.right));
            }
        }, other);
    }

    public CompletableFuture runAfterEither(final CompletableFuture other, final 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 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 BiConsumer action) {
        return execute(new Runnable() {
            @Override
            public void run() {
                final Pair result = Futures.anyOf(N.asList(CompletableFuture.this, other)).get2();

                action.accept(result.left, result.right);
            }
        }, other);
    }

    public  CompletableFuture callAfterEither(final CompletableFuture other, final Try.Callable action) {
        return execute(new Callable() {
            @Override
            public U call() throws Exception {
                Futures.anyOf(N.asList(CompletableFuture.this, other)).get();

                return action.call();
            }
        }, other);
    }

    public  CompletableFuture callAfterEither(final CompletableFuture other, final 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 BiFunction action) {
        return execute(new Callable() {
            @Override
            public R call() throws Exception {
                final Pair result = Futures.anyOf(N.asList(CompletableFuture.this, other)).get2();

                return action.apply(result.left, result.right);
            }
        }, other);
    }

    //    /**
    //     * 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 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 (Throwable 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 (Throwable 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 = get2();
    //
    //                if (result.right != null) {
    //                    try {
    //                        action.accept(result.left, result.right);
    //                    } catch (Throwable 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 = get2(timeout, unit);
    //
    //                if (result.right != null) {
    //                    try {
    //                        action.accept(result.left, result.right);
    //                    } catch (Throwable 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 = get2();
    //                return action.apply(result.left, result.right);
    //            }
    //
    //            @Override
    //            public U get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
    //                final Pair result = get2(timeout, unit);
    //                return action.apply(result.left, result.right);
    //            }
    //        }, asyncExecutor);
    //    }

    private CompletableFuture execute(final Runnable command) {
        return execute(command, null);
    }

    private CompletableFuture execute(final Runnable command, final CompletableFuture other) {
        return execute(new FutureTask(command, null), other);
    }

    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 = N.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 - N.currentMillis());
                }
            }
        }, null, executor) {
            @Override
            public boolean cancelAll(boolean mayInterruptIfRunning) {
                return super.cancelAll(mayInterruptIfRunning);
            }

            @Override
            public boolean isAllCancelled() {
                return super.isAllCancelled();
            }
        };
    }
}