/* * The MIT License (MIT) * * Copyright (c) 2017-2020 Yegor Bugayenko * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included * in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ package org.cactoos.experimental; import java.util.concurrent.Callable; import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.Future; import java.util.concurrent.TimeUnit; import org.cactoos.Func; import org.cactoos.Scalar; import org.cactoos.func.CallableOf; import org.cactoos.func.UncheckedFunc; import org.cactoos.iterable.IterableEnvelope; import org.cactoos.iterable.IterableOf; import org.cactoos.iterable.Mapped; import org.cactoos.list.ListOf; /** * Allows to execute the tasks concurrently within given timeout. * * @param <T> The type of task result item. * @since 1.0.0 */ public final class Timed<T> extends IterableEnvelope<T> { /** * Ctor. * @param exc The executor. * @param timeout The maximum time to wait. * @param unit The time unit of the timeout argument. * @param tasks The tasks to be executed concurrently. * @checkstyle ParameterNumberCheck (5 lines) */ @SafeVarargs public Timed( final ExecutorService exc, final long timeout, final TimeUnit unit, final Scalar<T>... tasks ) { this(exc, new IterableOf<>(tasks), timeout, unit); } /** * Ctor. * @param exc The executor. * @param tasks The tasks to be executed concurrently. * @param timeout The maximum time to wait. * @param unit The time unit of the timeout argument. * @checkstyle ParameterNumberCheck (5 lines) */ public Timed( final ExecutorService exc, final Iterable<Scalar<T>> tasks, final long timeout, final TimeUnit unit ) { this( input -> exc.invokeAll(new ListOf<>(input), timeout, unit), tasks ); } /** * Ctor. * @param threads The quantity of threads which will be used within the * {@link ExecutorService}. * @param timeout The maximum time to wait. * @param unit The time unit of the timeout argument. * @param tasks The tasks to be executed concurrently. * @see Executors#newFixedThreadPool(int) * @checkstyle ParameterNumberCheck (5 lines) */ @SafeVarargs public Timed( final int threads, final long timeout, final TimeUnit unit, final Scalar<T>... tasks ) { this(threads, new IterableOf<>(tasks), timeout, unit); } /** * Ctor. * @param threads The quantity of threads which will be used within the * {@link ExecutorService}. * @param tasks The tasks to be executed concurrently. * @param timeout The maximum time to wait. * @param unit The time unit of the timeout argument. * @checkstyle IndentationCheck (20 lines) * @checkstyle ParameterNumberCheck (5 lines) */ public Timed( final int threads, final Iterable<Scalar<T>> tasks, final long timeout, final TimeUnit unit ) { this( todo -> { final ExecutorService executor = Executors.newFixedThreadPool( threads ); try { return executor.invokeAll(new ListOf<>(todo), timeout, unit); } finally { executor.shutdown(); } }, tasks ); } /** * Ctor. * @param fnc The function to map each task into {@link Future}. * @param tasks The tasks to be executed concurrently. */ private Timed( final Func<Iterable<Callable<T>>, Iterable<Future<T>>> fnc, final Iterable<Scalar<T>> tasks ) { super( () -> new Mapped<>( Future::get, new UncheckedFunc<>(fnc).apply(new Mapped<>(CallableOf::new, tasks)) ).iterator() ); } }