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

sirius.kernel.commons.Exec Maven / Gradle / Ivy

Go to download

Provides common core classes and the microkernel powering all Sirius applications

There is a newer version: 12.9.1
Show newest version
/*
 * Made with all the love in the world
 * by scireum in Remshalden, Germany
 *
 * Copyright by scireum GmbH
 * http://www.scireum.de - [email protected]
 */

package sirius.kernel.commons;

import sirius.kernel.async.Operation;
import sirius.kernel.health.Exceptions;
import sirius.kernel.health.Log;
import sirius.kernel.nls.NLS;

import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.time.Duration;
import java.util.concurrent.Semaphore;

/**
 * A robust wrapper around calls to external programs.
 */
@SuppressWarnings("squid:S1149")
@Explain("We actually need the thread safety probided by StringBuffer here, as we start 2 threads per call.")
public class Exec {

    /**
     * Can be used to log errors and infos when executing external programs.
     */
    public static final Log LOG = Log.get("exec");

    private Exec() {
    }

    /*
     * Reads the given stream in a separate thread.
     */
    private static class StreamEater implements Runnable {

        private final InputStream stream;
        private final StringBuffer logger;
        private final ValueHolder exHolder = new ValueHolder<>(null);
        private final Semaphore completionSynchronizer;

        StreamEater(InputStream stream, StringBuffer log, Semaphore completionSynchronizer)
                throws InterruptedException {
            this.stream = stream;
            this.logger = log;
            this.completionSynchronizer = completionSynchronizer;
            this.completionSynchronizer.acquire();
        }

        @Override
        public void run() {
            try (InputStreamReader isr = new InputStreamReader(stream); BufferedReader br = new BufferedReader(isr)) {
                Thread.currentThread()
                      .setName(StreamEater.class.getSimpleName() + "-" + Thread.currentThread().getId());
                String line = br.readLine();
                while (line != null) {
                    logger.append(line);
                    logger.append("\n");
                    line = br.readLine();
                }
            } catch (IOException e) {
                logger.append(NLS.toUserString(e));
                exHolder.set(e);
            } finally {
                this.completionSynchronizer.release();
            }
        }

        /**
         * Creates a new stream eater logging to the given buffer for the given stream.
         *
         * @param stream                 the stream to read
         * @param logger                 the target for all characters read
         * @param completionSynchronizer a semaphore where a permit is acquired and released once all output hase been processed
         * @return a new stream eater which is already running in a separate thread
         */
        static StreamEater eat(InputStream stream, StringBuffer logger, Semaphore completionSynchronizer)
                throws InterruptedException {
            StreamEater eater = new StreamEater(stream, logger, completionSynchronizer);
            new Thread(eater).start();
            return eater;
        }
    }

    /**
     * Thrown if a call to an external program fails
     */
    public static class ExecException extends Exception {

        private static final long serialVersionUID = -4736872491172480346L;
        private final String log;

        ExecException(Throwable root, String log) {
            super(root);
            this.log = log;
        }

        /**
         * Provides access to the contents of the programs stdout and stderr.
         *
         * @return a transcript of the called programs stderr and stdout
         */
        public String getLog() {
            return log;
        }
    }

    /**
     * Executes the given command and returns a transcript of stderr and stdout
     *
     * @param command the command to execute
     * @return the transcript of stderr and stdout produced by the executed command
     * @throws ExecException in case the external program fails or returns an exit code other than 0.
     */
    public static String exec(String command) throws ExecException {
        return exec(command, false);
    }

    /**
     * Executes the given command and returns a transcript of stderr and stdout.
     *
     * @param command         the command to execute
     * @param ignoreExitCodes if an exit code other than 0 should result in an exception being thrown
     * @return the transcript of stderr and stdout produced by the executed command
     * @throws ExecException in case the external program fails
     */
    public static String exec(String command, boolean ignoreExitCodes) throws ExecException {
        StringBuffer logger = new StringBuffer();
        try (Operation op = new Operation(() -> command, Duration.ofMinutes(5))) {
            Process p = Runtime.getRuntime().exec(command);
            Semaphore completionSynchronizer = new Semaphore(2);
            StreamEater errEater = StreamEater.eat(p.getErrorStream(), logger, completionSynchronizer);
            StreamEater outEater = StreamEater.eat(p.getInputStream(), logger, completionSynchronizer);
            doExec(ignoreExitCodes, logger, p);

            // Wait for the stream eaters to complete...
            completionSynchronizer.acquire(2);

            if (errEater.exHolder.get() != null) {
                throw new ExecException(errEater.exHolder.get(), logger.toString());
            }
            if (outEater.exHolder.get() != null) {
                throw new ExecException(outEater.exHolder.get(), logger.toString());
            }
            return logger.toString();
        } catch (Exception e) {
            throw new ExecException(e, logger.toString());
        }
    }

    private static void doExec(boolean ignoreExitCodes, StringBuffer logger, Process p) throws ExecException {
        try {
            int code = p.waitFor();
            if (code != 0 && !ignoreExitCodes) {
                Exception root = new Exception("Command returned with exit code " + code);
                throw new ExecException(root, logger.toString());
            }
        } catch (InterruptedException e) {
            Exceptions.ignore(e);
            Thread.currentThread().interrupt();
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy