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

com.powsybl.computation.local.LocalComputationManager Maven / Gradle / Ivy

/**
 * Copyright (c) 2016, All partners of the iTesla project (http://www.itesla-project.eu/consortium)
 * Copyright (c) 2016-2019, RTE (http://www.rte-france.com)
 * This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
 * SPDX-License-Identifier: MPL-2.0
 */
package com.powsybl.computation.local;

import com.google.common.base.Stopwatch;
import com.google.common.io.ByteStreams;
import com.powsybl.commons.PowsyblException;
import com.powsybl.commons.config.PlatformConfig;
import com.powsybl.commons.io.WorkingDirectory;
import com.powsybl.computation.*;
import org.apache.commons.compress.archivers.zip.ZipArchiveEntry;
import org.apache.commons.compress.archivers.zip.ZipFile;
import org.apache.commons.lang3.SystemUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.UncheckedIOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.*;
import java.util.concurrent.*;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import java.util.stream.IntStream;
import java.util.zip.GZIPInputStream;
import java.util.zip.GZIPOutputStream;

import static java.nio.file.StandardCopyOption.REPLACE_EXISTING;

/**
 *
 * @author Geoffroy Jamgotchian {@literal }
 */
public class LocalComputationManager implements ComputationManager {

    private static final Logger LOGGER = LoggerFactory.getLogger(LocalComputationManager.class);

    private final LocalComputationConfig config;

    private final WorkingDirectory commonDir;

    private final LocalComputationResourcesStatus status;

    private final Semaphore permits;

    private final Executor threadPool;

    private final LocalCommandExecutor localCommandExecutor;

    private static final Lock LOCK = new ReentrantLock();

    private static LocalComputationManager defaultInstance;

    public static ComputationManager getDefault() {
        LOCK.lock();
        try {
            if (defaultInstance == null) {
                try {
                    defaultInstance = new LocalComputationManager();
                    Runtime.getRuntime().addShutdownHook(new Thread(() -> defaultInstance.close()));
                } catch (IOException e) {
                    throw new UncheckedIOException(e);
                }
            }
            return defaultInstance;
        } finally {
            LOCK.unlock();
        }
    }

    private static LocalCommandExecutor getLocalCommandExecutor() {
        if (SystemUtils.IS_OS_WINDOWS) {
            return new WindowsLocalCommandExecutor();
        } else if (SystemUtils.IS_OS_UNIX) {
            return new UnixLocalCommandExecutor();
        } else {
            throw new UnsupportedOperationException("OS not supported for local execution");
        }
    }

    public LocalComputationManager() throws IOException {
        this(LocalComputationConfig.load());
    }

    public LocalComputationManager(Executor executor) throws IOException {
        this(LocalComputationConfig.load(), executor);
    }

    public LocalComputationManager(PlatformConfig platformConfig) throws IOException {
        this(LocalComputationConfig.load(platformConfig));
    }

    public LocalComputationManager(Path localDir) throws IOException {
        this(new LocalComputationConfig(localDir));
    }

    public LocalComputationManager(LocalComputationConfig config) throws IOException {
        this(config, ForkJoinPool.commonPool());
    }

    public LocalComputationManager(LocalComputationConfig config, Executor executor) throws IOException {
        this(config, getLocalCommandExecutor(), executor);
    }

    public LocalComputationManager(LocalComputationConfig config, LocalCommandExecutor localCommandExecutor, Executor executor) throws IOException {
        this.config = Objects.requireNonNull(config);
        this.localCommandExecutor = Objects.requireNonNull(localCommandExecutor);
        this.threadPool = Objects.requireNonNull(executor);
        status = new LocalComputationResourcesStatus(config.getAvailableCore());
        permits = new Semaphore(config.getAvailableCore());
        //make sure the localdir exists
        Files.createDirectories(config.getLocalDir());
        commonDir = new WorkingDirectory(config.getLocalDir(), "itools_common_", false);
        if (LOGGER.isInfoEnabled()) {
            LOGGER.info(config.toString());
        }
    }

    @Override
    public String getVersion() {
        return "none (local mode)";
    }

    @Override
    public Path getLocalDir() {
        return config.getLocalDir();
    }

    @Override
    public OutputStream newCommonFile(String fileName) throws IOException {
        return Files.newOutputStream(commonDir.toPath().resolve(fileName));
    }

    private interface ExecutionMonitor {

        void onProgress(CommandExecution execution, int executionIndex);

    }

    private ExecutionReport execute(Path workingDir, List commandExecutionList, Map variables, ComputationParameters computationParameters, ExecutionMonitor monitor)
            throws InterruptedException {
        // TODO concurrent
        List errors = new ArrayList<>();
        ExecutorService executionSubmitter = Executors.newCachedThreadPool();

        for (CommandExecution commandExecution : commandExecutionList) {
            Command command = commandExecution.getCommand();
            CountDownLatch latch = new CountDownLatch(commandExecution.getExecutionCount());
            ExecutionParameters executionParameters = new ExecutionParameters(workingDir, commandExecution, variables, computationParameters, executionSubmitter,
                command, latch, errors, monitor);
            IntStream.range(0, commandExecution.getExecutionCount()).forEach(idx -> performSingleExecution(executionParameters, idx));
            latch.await();
        }

        // TODO remove duplicated code
        executionSubmitter.shutdown();
        if (!executionSubmitter.awaitTermination(20, TimeUnit.SECONDS)) {
            executionSubmitter.shutdownNow();
            if (!executionSubmitter.awaitTermination(20, TimeUnit.SECONDS)) {
                LOGGER.error("Thread pool did not terminate");
            }
        }

        return new DefaultExecutionReport(workingDir, errors);
    }

    private record ExecutionParameters(Path workingDir, CommandExecution commandExecution,
                                       Map variables, ComputationParameters computationParameters,
                                       ExecutorService executionSubmitter, Command command, CountDownLatch latch,
                                       List errors, ExecutionMonitor monitor) {
    }

    private void performSingleExecution(ExecutionParameters executionParameters, int idx) {
        executionParameters.executionSubmitter.execute(() -> {
            try {
                enter();
                if (LOGGER.isDebugEnabled()) {
                    LOGGER.debug("Executing command {} in working directory {}",
                        executionParameters.command.toString(idx), executionParameters.workingDir);
                }
                preProcess(executionParameters.workingDir, executionParameters.command, idx);
                Stopwatch stopwatch = null;
                if (LOGGER.isDebugEnabled()) {
                    stopwatch = Stopwatch.createStarted();
                }
                int exitValue = process(executionParameters.workingDir, executionParameters.commandExecution, idx,
                    executionParameters.variables, executionParameters.computationParameters);
                if (stopwatch != null) {
                    stopwatch.stop();
                    LOGGER.debug("Command {} executed in {} ms",
                        executionParameters.command.toString(idx), stopwatch.elapsed(TimeUnit.MILLISECONDS));
                }
                postProcess(executionParameters.workingDir, executionParameters.commandExecution, idx, exitValue,
                    executionParameters.errors, executionParameters.monitor);
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
                LOGGER.warn(e.getMessage(), e);
            } catch (Exception e) {
                LOGGER.warn(e.getMessage(), e);
            } finally {
                executionParameters.latch.countDown();
                exit();
            }
        });
    }

    private void preProcess(Path workingDir, Command command, int executionIndex) throws IOException {
        // pre-processing
        for (InputFile file : command.getInputFiles()) {
            String fileName = file.getName(executionIndex);

            Path path = checkInputFileExistsInWorkingAndCommons(workingDir, fileName, file);
            if (file.getPreProcessor() != null) {
                switch (file.getPreProcessor()) {
                    case FILE_GUNZIP:
                        // gunzip the file
                        try (InputStream is = new GZIPInputStream(Files.newInputStream(path));
                             OutputStream os = Files.newOutputStream(workingDir.resolve(fileName.substring(0, fileName.length() - 3)))) {
                            ByteStreams.copy(is, os);
                        }
                        break;
                    case ARCHIVE_UNZIP:
                        // extract the archive
                        try (ZipFile zipFile = ZipFile.builder()
                            .setSeekableByteChannel(Files.newByteChannel(path))
                            .get()) {
                            for (ZipArchiveEntry ze : Collections.list(zipFile.getEntries())) {
                                Files.copy(zipFile.getInputStream(zipFile.getEntry(ze.getName())), workingDir.resolve(ze.getName()), REPLACE_EXISTING);
                            }
                        }
                        break;

                    default:
                        throw new IllegalStateException("Unexpected FilePreProcessor value: " + file.getPreProcessor());
                }
            }
        }
    }

    private int process(Path workingDir, CommandExecution commandExecution, int executionIndex, Map variables, ComputationParameters computationParameters) throws IOException, InterruptedException {
        Command command = commandExecution.getCommand();
        int exitValue = 0;
        long timeout = -1;
        Path outFile = workingDir.resolve(command.getId() + "_" + executionIndex + ".out");
        Path errFile = workingDir.resolve(command.getId() + "_" + executionIndex + ".err");
        Map executionVariables = CommandExecution.getExecutionVariables(variables, commandExecution);
        switch (command.getType()) {
            case SIMPLE:
                SimpleCommand simpleCmd = (SimpleCommand) command;
                timeout = computationParameters.getTimeout(simpleCmd.getId()).orElse(-1);
                exitValue = localCommandExecutor.execute(simpleCmd.getProgram(), timeout,
                        simpleCmd.getArgs(executionIndex),
                        outFile,
                        errFile,
                        workingDir,
                        executionVariables);
                break;
            case GROUP:
                // TODO timeout for group
                for (GroupCommand.SubCommand subCmd : ((GroupCommand) command).getSubCommands()) {
                    exitValue = localCommandExecutor.execute(subCmd.getProgram(),
                            subCmd.getArgs(executionIndex),
                            outFile,
                            errFile,
                            workingDir,
                            executionVariables);
                    if (exitValue != 0) {
                        break;
                    }
                }
                break;
            default:
                throw new IllegalStateException("Unexpected CommandType value: " + command.getType());
        }
        return exitValue;
    }

    private void postProcess(Path workingDir, CommandExecution commandExecution, int executionIndex, int exitValue, List errors, ExecutionMonitor monitor) throws IOException {
        Command command = commandExecution.getCommand();
        if (exitValue != 0) {
            errors.add(new ExecutionError(command, executionIndex, exitValue));
        } else {
            // post processing
            for (OutputFile file : command.getOutputFiles()) {
                String fileName = file.getName(executionIndex);
                Path path = workingDir.resolve(fileName);
                if (file.getPostProcessor() != null && Files.isRegularFile(path)) {
                    if (file.getPostProcessor() == FilePostProcessor.FILE_GZIP) { // gzip the file
                        try (InputStream is = Files.newInputStream(path);
                             OutputStream os = new GZIPOutputStream(Files.newOutputStream(workingDir.resolve(fileName + ".gz")))) {
                            ByteStreams.copy(is, os);
                        }

                    } else {
                        throw new IllegalStateException("Unexpected FilePostProcessor value: " + file.getPostProcessor());
                    }
                }
            }
        }

        if (monitor != null) {
            monitor.onProgress(commandExecution, executionIndex);
        }
    }

    private Path checkInputFileExistsInWorkingAndCommons(Path workingDir, String fileName, InputFile file) throws IOException {
        // first check if the file exists in the working directory
        Path path = workingDir.resolve(fileName);
        if (!Files.exists(path)) {
            // if not check if the file exists in the common directory
            path = commonDir.toPath().resolve(fileName);
            if (!Files.exists(path)) {
                throw new PowsyblException("Input file '" + fileName + "' not found in the working and common directory");
            }
            if (file.getPreProcessor() == null) {
                Files.copy(path, workingDir.resolve(path.getFileName()));
            }
        }
        return path;
    }

    private void enter() throws InterruptedException {
        permits.acquire();
        status.incrementNumberOfBusyCores();
    }

    private void exit() {
        status.decrementNumberOfBusyCores();
        permits.release();
    }

    @Override
    public  CompletableFuture execute(ExecutionEnvironment environment, ExecutionHandler handler) {
        return execute(environment, handler, ComputationParameters.empty());
    }

    @Override
    public  CompletableFuture execute(ExecutionEnvironment environment, ExecutionHandler handler, ComputationParameters parameters) {
        Objects.requireNonNull(environment);
        Objects.requireNonNull(handler);

        return CompletableFutureTask.runAsync(() -> doExecute(environment, handler, parameters), threadPool);
    }

    /**
     * Executes commands described by the specified handler. If the executing thread is interrupted,
     * for example by a call to {@link CompletableFutureTask#cancel(boolean)}, the underlying process
     * execution will be stopped.
     */
    private  R doExecute(ExecutionEnvironment environment, ExecutionHandler handler, ComputationParameters parameters) throws IOException, InterruptedException {

        try (WorkingDirectory workingDir = new WorkingDirectory(config.getLocalDir(), environment.getWorkingDirPrefix(), environment.isDebug())) {

            List commandExecutionList = handler.before(workingDir.toPath());

            ExecutionReport report;
            try {
                report = execute(workingDir.toPath(), commandExecutionList, environment.getVariables(), parameters, handler::onExecutionCompletion);
            } catch (InterruptedException exc) {
                localCommandExecutor.stop(workingDir.toPath());
                throw exc;
            }
            return handler.after(workingDir.toPath(), report);
        }
    }

    @Override
    public ComputationResourcesStatus getResourcesStatus() {
        return status;
    }

    @Override
    public Executor getExecutor() {
        return threadPool;
    }

    @Override
    public void close() {
        try {
            commonDir.close();
        } catch (IOException e) {
            throw new UncheckedIOException(e);
        }
    }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy