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

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

/*
 * Copyright (C) 2017 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.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.NoSuchElementException;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;

import com.landawn.abacus.util.Tuple.Tuple2;
import com.landawn.abacus.util.Tuple.Tuple3;
import com.landawn.abacus.util.Tuple.Tuple4;
import com.landawn.abacus.util.Tuple.Tuple5;
import com.landawn.abacus.util.Tuple.Tuple6;
import com.landawn.abacus.util.Tuple.Tuple7;

/**
 * @since 0.9
 * 
 * @author Haiyang Li 
 *
 */
public final class Futures {
    private Futures() {
        // singleton.
    }

    public static  CompletableFuture zip(final CompletableFuture cf1, final CompletableFuture cf2,
            final Try.BiFunction, ? super CompletableFuture, R, Exception> zipFunctionForGet) {
        return zip(cf1, cf2, zipFunctionForGet, new Try.Function, CompletableFuture, Long, TimeUnit>, R, Exception>() {
            @Override
            public R apply(Tuple4, CompletableFuture, Long, TimeUnit> t) throws Exception {
                return zipFunctionForGet.apply(t._1, t._2);
            }
        });
    }

    public static  CompletableFuture zip(final CompletableFuture cf1, final CompletableFuture cf2,
            final Try.BiFunction, ? super CompletableFuture, R, Exception> zipFunctionForGet,
            final Try.Function, CompletableFuture, Long, TimeUnit>, R, Exception> zipFunctionTimeoutGet) {
        final List> cfs = Arrays.asList(cf1, cf2);

        return zip(cfs, new Try.Function>, R, Exception>() {
            @Override
            public R apply(List> c) throws Exception {
                return zipFunctionForGet.apply((CompletableFuture) c.get(0), (CompletableFuture) c.get(1));
            }
        }, new Try.Function>, Long, TimeUnit>, R, Exception>() {
            @Override
            public R apply(Tuple3>, Long, TimeUnit> t) throws Exception {
                return zipFunctionTimeoutGet.apply(Tuple.of((CompletableFuture) t._1.get(0), (CompletableFuture) t._1.get(1), t._2, t._3));
            }
        });
    }

    public static  CompletableFuture zip(final CompletableFuture cf1, final CompletableFuture cf2, final CompletableFuture cf3,
            final Try.TriFunction, ? super CompletableFuture, ? super CompletableFuture, R, Exception> zipFunctionForGet) {
        return zip(cf1, cf2, cf3, zipFunctionForGet,
                new Try.Function, CompletableFuture, CompletableFuture, Long, TimeUnit>, R, Exception>() {
                    @Override
                    public R apply(Tuple5, CompletableFuture, CompletableFuture, Long, TimeUnit> t) throws Exception {
                        return zipFunctionForGet.apply(t._1, t._2, t._3);
                    }
                });
    }

    public static  CompletableFuture zip(final CompletableFuture cf1, final CompletableFuture cf2, final CompletableFuture cf3,
            final Try.TriFunction, ? super CompletableFuture, ? super CompletableFuture, R, Exception> zipFunctionForGet,
            final Try.Function, CompletableFuture, CompletableFuture, Long, TimeUnit>, R, Exception> zipFunctionTimeoutGet) {
        final List> cfs = Arrays.asList(cf1, cf2, cf3);

        return zip(cfs, new Try.Function>, R, Exception>() {
            @Override
            public R apply(List> c) throws Exception {
                return zipFunctionForGet.apply((CompletableFuture) c.get(0), (CompletableFuture) c.get(1), (CompletableFuture) c.get(2));
            }
        }, new Try.Function>, Long, TimeUnit>, R, Exception>() {
            @Override
            public R apply(Tuple3>, Long, TimeUnit> t) throws Exception {
                return zipFunctionTimeoutGet.apply(
                        Tuple.of((CompletableFuture) t._1.get(0), (CompletableFuture) t._1.get(1), (CompletableFuture) t._1.get(2), t._2, t._3));
            }
        });
    }

    public static >, R> CompletableFuture zip(final FC cfs,
            final Try.Function zipFunctionForGet) {
        return zip(cfs, zipFunctionForGet, new Try.Function, R, Exception>() {
            @Override
            public R apply(Tuple3 t) throws Exception {
                return zipFunctionForGet.apply(t._1);
            }
        });
    }

    public static >, R> CompletableFuture zip(final FC cfs,
            final Try.Function zipFunctionForGet,
            final Try.Function, R, Exception> zipFunctionTimeoutGet) {
        N.checkArgument(N.notNullOrEmpty(cfs), "'cfs' can't be null or empty");
        N.requireNonNull(zipFunctionForGet);
        N.requireNonNull(zipFunctionTimeoutGet);

        return new CompletableFuture<>(new Future() {
            @Override
            public boolean cancel(boolean mayInterruptIfRunning) {
                boolean res = true;
                RuntimeException exception = null;

                for (CompletableFuture future : cfs) {
                    try {
                        res = res & future.cancel(mayInterruptIfRunning);
                    } catch (RuntimeException e) {
                        if (exception == null) {
                            exception = e;
                        } else {
                            exception.addSuppressed(e);
                        }
                    }
                }

                if (exception != null) {
                    throw exception;
                }

                return res;
            }

            @Override
            public boolean isCancelled() {
                for (CompletableFuture future : cfs) {
                    if (future.isCancelled()) {
                        return true;
                    }
                }

                return false;
            }

            @Override
            public boolean isDone() {
                for (CompletableFuture future : cfs) {
                    if (future.isDone() == false) {
                        return false;
                    }
                }

                return true;
            }

            @Override
            public R get() throws InterruptedException, ExecutionException {
                try {
                    return zipFunctionForGet.apply(cfs);
                } catch (InterruptedException | ExecutionException e) {
                    throw e;
                } catch (Exception e) {
                    throw N.toRuntimeException(e);
                }
            }

            @Override
            public R get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
                final Tuple3 t = Tuple.of(cfs, timeout, unit);

                try {
                    return zipFunctionTimeoutGet.apply(t);
                } catch (InterruptedException | ExecutionException | TimeoutException e) {
                    throw e;
                } catch (Exception e) {
                    throw N.toRuntimeException(e);
                }
            }
        }, null, ((CompletableFuture) cfs.iterator().next()).asyncExecutor);
    }

    public static  CompletableFuture> combine(final CompletableFuture cf1,
            final CompletableFuture cf2) {
        return allOf(Arrays.asList(cf1, cf2)).thenApply(new Try.Function, Tuple2, E>() {
            @Override
            public Tuple2 apply(List t) throws E {
                return Tuple.of((T1) t.get(0), (T2) t.get(1));
            }
        });
    }

    public static  CompletableFuture> combine(final CompletableFuture cf1,
            final CompletableFuture cf2, final CompletableFuture cf3) {
        return allOf(Arrays.asList(cf1, cf2, cf3)).thenApply(new Try.Function, Tuple3, E>() {
            @Override
            public Tuple3 apply(List t) throws E {
                return Tuple.of((T1) t.get(0), (T2) t.get(1), (T3) t.get(2));
            }
        });
    }

    public static  CompletableFuture> combine(final CompletableFuture cf1,
            final CompletableFuture cf2, final CompletableFuture cf3, final CompletableFuture cf4) {
        return allOf(Arrays.asList(cf1, cf2, cf3, cf4)).thenApply(new Try.Function, Tuple4, E>() {
            @Override
            public Tuple4 apply(List t) throws E {
                return Tuple.of((T1) t.get(0), (T2) t.get(1), (T3) t.get(2), (T4) t.get(3));
            }
        });
    }

    public static  CompletableFuture> combine(final CompletableFuture cf1,
            final CompletableFuture cf2, final CompletableFuture cf3, final CompletableFuture cf4,
            final CompletableFuture cf5) {
        return allOf(Arrays.asList(cf1, cf2, cf3, cf4, cf5)).thenApply(new Try.Function, Tuple5, E>() {
            @Override
            public Tuple5 apply(List t) throws E {
                return Tuple.of((T1) t.get(0), (T2) t.get(1), (T3) t.get(2), (T4) t.get(3), (T5) t.get(4));
            }
        });
    }

    public static  CompletableFuture> combine(
            final CompletableFuture cf1, final CompletableFuture cf2, final CompletableFuture cf3,
            final CompletableFuture cf4, final CompletableFuture cf5, final CompletableFuture cf6) {
        return allOf(Arrays.asList(cf1, cf2, cf3, cf4, cf5, cf6)).thenApply(new Try.Function, Tuple6, E>() {
            @Override
            public Tuple6 apply(List t) throws E {
                return Tuple.of((T1) t.get(0), (T2) t.get(1), (T3) t.get(2), (T4) t.get(3), (T5) t.get(4), (T6) t.get(5));
            }
        });
    }

    public static  CompletableFuture> combine(
            final CompletableFuture cf1, final CompletableFuture cf2, final CompletableFuture cf3,
            final CompletableFuture cf4, final CompletableFuture cf5, final CompletableFuture cf6,
            final CompletableFuture cf7) {
        return allOf(Arrays.asList(cf1, cf2, cf3, cf4, cf5, cf6, cf7)).thenApply(new Try.Function, Tuple7, E>() {
            @Override
            public Tuple7 apply(List t) throws E {
                return Tuple.of((T1) t.get(0), (T2) t.get(1), (T3) t.get(2), (T4) t.get(3), (T5) t.get(4), (T6) t.get(5), (T7) t.get(6));
            }
        });
    }

    public static  CompletableFuture combine(final CompletableFuture cf1,
            final CompletableFuture cf2, final Try.BiFunction action) {
        return allOf(Arrays.asList(cf1, cf2)).thenApply(new Try.Function, R, E>() {
            @Override
            public R apply(List t) throws E {
                return action.apply((T1) t.get(0), (T2) t.get(1));
            }
        });
    }

    public static  CompletableFuture combine(final CompletableFuture cf1,
            final CompletableFuture cf2, final CompletableFuture cf3,
            final Try.TriFunction action) {
        return allOf(Arrays.asList(cf1, cf2, cf3)).thenApply(new Try.Function, R, E>() {
            @Override
            public R apply(List t) throws E {
                return action.apply((T1) t.get(0), (T2) t.get(1), (T3) t.get(2));
            }
        });
    }

    public static  CompletableFuture combine(final Collection> cfs,
            final Try.Function, ? extends R, E> action) {
        final CompletableFuture> f = allOf(cfs);
        return f.thenApply(action);
    }

    //    public static  CompletableFuture combine(final List> cfs, final Function, ? extends R> action) {
    //        final CompletableFuture> future = allOf(cfs);
    //        return future.thenApply(action);
    //    }

    /**
     * Returns a new CompletableFuture that is completed when all of
     * the given CompletableFutures complete. If any of the given
     * CompletableFutures complete exceptionally, then the returned
     * CompletableFuture also does so.
     * 
     * @param cfs
     * @return
     */
    @SafeVarargs
    public static  CompletableFuture> allOf(final CompletableFuture... cfs) {
        return allOf2(Arrays.asList(cfs));
    }

    /**
     * Returns a new CompletableFuture that is completed when all of
     * the given CompletableFutures complete. If any of the given
     * CompletableFutures complete exceptionally, then the returned
     * CompletableFuture also does so.
     * 
     * @param cfs
     * @return
     */
    public static  CompletableFuture> allOf(final Collection> cfs) {
        return allOf2(cfs);
    }

    private static  CompletableFuture> allOf2(final Collection> cfs) {
        N.checkArgument(N.notNullOrEmpty(cfs), "'cfs' can't be null or empty");

        return new CompletableFuture<>(new Future>() {
            @Override
            public boolean cancel(boolean mayInterruptIfRunning) {
                boolean res = true;
                RuntimeException exception = null;

                for (CompletableFuture future : cfs) {
                    try {
                        res = res & future.cancel(mayInterruptIfRunning);
                    } catch (RuntimeException e) {
                        if (exception == null) {
                            exception = e;
                        } else {
                            exception.addSuppressed(e);
                        }
                    }
                }

                if (exception != null) {
                    throw exception;
                }

                return res;
            }

            @Override
            public boolean isCancelled() {
                for (CompletableFuture future : cfs) {
                    if (future.isCancelled()) {
                        return true;
                    }
                }

                return false;
            }

            @Override
            public boolean isDone() {
                for (CompletableFuture future : cfs) {
                    if (future.isDone() == false) {
                        return false;
                    }
                }

                return true;
            }

            @Override
            public List get() throws InterruptedException, ExecutionException {
                final List result = new ArrayList<>(cfs.size());

                for (CompletableFuture future : cfs) {
                    result.add(future.get());
                }

                return result;
            }

            @Override
            public List get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
                final long timeoutInMillis = unit.toMillis(timeout);
                final long now = DateUtil.currentMillis();
                final long endTime = timeoutInMillis > Long.MAX_VALUE - now ? Long.MAX_VALUE : now + timeoutInMillis;

                final List result = new ArrayList<>(cfs.size());

                for (CompletableFuture future : cfs) {
                    result.add(future.get(N.max(0, endTime - DateUtil.currentMillis()), TimeUnit.MILLISECONDS));
                }

                return result;
            }
        }, null, ((CompletableFuture) cfs.iterator().next()).asyncExecutor);
    }

    /**
     * Returns a new CompletableFuture that, when any of the given CompletableFutures complete normally. 
     * If all of the given CompletableFutures complete exceptionally, then the returned CompletableFuture also does so.
     * 
     * @param cfs
     * @return
     */
    @SafeVarargs
    public static  CompletableFuture anyOf(final CompletableFuture... cfs) {
        return anyOf2(Arrays.asList(cfs));
    }

    /**
     * Returns a new CompletableFuture that, when any of the given CompletableFutures complete normally. 
     * If all of the given CompletableFutures complete exceptionally, then the returned CompletableFuture also does so.
     * 
     * @param cfs
     * @return
     */
    public static  CompletableFuture anyOf(final Collection> cfs) {
        return anyOf2(cfs);
    }

    private static  CompletableFuture anyOf2(final Collection> cfs) {
        N.checkArgument(N.notNullOrEmpty(cfs), "'cfs' can't be null or empty");

        return new CompletableFuture<>(new Future() {
            @Override
            public boolean cancel(boolean mayInterruptIfRunning) {
                boolean res = true;
                RuntimeException exception = null;

                for (CompletableFuture future : cfs) {
                    try {
                        res = res & future.cancel(mayInterruptIfRunning);
                    } catch (RuntimeException e) {
                        if (exception == null) {
                            exception = e;
                        } else {
                            exception.addSuppressed(e);
                        }
                    }
                }

                if (exception != null) {
                    throw exception;
                }

                return res;
            }

            @Override
            public boolean isCancelled() {
                for (CompletableFuture future : cfs) {
                    if (future.isCancelled() == false) {
                        return false;
                    }
                }

                return true;
            }

            @Override
            public boolean isDone() {
                for (CompletableFuture future : cfs) {
                    if (future.isDone()) {
                        return true;
                    }
                }

                return false;
            }

            @Override
            public T get() throws InterruptedException, ExecutionException {
                final Iterator> iter = iteratte(cfs);
                Pair result = null;

                while (iter.hasNext()) {
                    result = iter.next();

                    if (result.right == null) {
                        return result.left;
                    }
                }

                return handle(result);
            }

            @Override
            public T get(final long timeout, final TimeUnit unit) throws InterruptedException, ExecutionException, TimeoutException {
                final Iterator> iter = iteratte(cfs, timeout, unit);
                Pair result = null;

                while (iter.hasNext()) {
                    result = iter.next();

                    if (result.right == null) {
                        return result.left;
                    }
                }

                return handle(result);
            }
        }, null, ((CompletableFuture) cfs.iterator().next()).asyncExecutor);
    }

    @SafeVarargs
    public static  ObjIterator iterate(final CompletableFuture... cfs) {
        return iterate02(Arrays.asList(cfs));
    }

    public static  ObjIterator iterate(final Collection> cfs) {
        return iterate02(cfs);
    }

    public static  ObjIterator iterate(final Collection> cfs, final long timeout, final TimeUnit unit) {
        return iterate02(cfs, timeout, unit);
    }

    private static  ObjIterator iterate02(final Collection> cfs) {
        return iterate02(cfs, Long.MAX_VALUE, TimeUnit.MILLISECONDS);
    }

    private static  ObjIterator iterate02(final Collection> cfs, final long timeout, final TimeUnit unit) {
        return new ObjIterator() {
            private final Iterator> iter = iterate22(cfs, timeout, unit);

            @Override
            public boolean hasNext() {
                return iter.hasNext();
            }

            @Override
            public T next() {
                try {
                    return handle(iter.next());
                } catch (InterruptedException | ExecutionException e) {
                    throw N.toRuntimeException(e);
                }
            }
        };
    }

    @SafeVarargs
    public static  ObjIterator> iteratte(final CompletableFuture... cfs) {
        return iterate22(Arrays.asList(cfs));
    }

    public static  ObjIterator> iteratte(final Collection> cfs) {
        return iterate22(cfs);
    }

    public static  ObjIterator> iteratte(final Collection> cfs, final long timeout,
            final TimeUnit unit) {
        return iterate22(cfs, timeout, unit);
    }

    private static  ObjIterator> iterate22(final Collection> cfs) {
        return iterate22(cfs, Long.MAX_VALUE, TimeUnit.MILLISECONDS);
    }

    private static  ObjIterator> iterate22(final Collection> cfs, final long timeout,
            final TimeUnit unit) {
        final ExecutorService executor = Executors.newFixedThreadPool(cfs.size());
        final BlockingQueue> queue = new ArrayBlockingQueue<>(cfs.size());

        for (CompletableFuture e : cfs) {
            final CompletableFuture futuer = (CompletableFuture) e;

            executor.execute(new Runnable() {
                @Override
                public void run() {
                    queue.offer(futuer.gett(timeout, unit));
                }
            });
        }

        return new ObjIterator>() {
            private final int end = cfs.size();
            private int cursor = 0;

            @Override
            public boolean hasNext() {
                return cursor < end;
            }

            @Override
            public Pair next() {
                if (cursor >= end) {
                    throw new NoSuchElementException();
                }

                cursor++;

                try {
                    return queue.poll(Long.MAX_VALUE, TimeUnit.MILLISECONDS);
                } catch (InterruptedException e) {
                    throw N.toRuntimeException(e);
                }
            }
        };
    }

    private static  R handle(final Pair result) throws InterruptedException, ExecutionException {
        if (result.right != null) {
            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);
            }
        }

        return result.left;
    }
}