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

io.quarkiverse.quinoa.deployment.packagemanager.PackageManagerRunner Maven / Gradle / Ivy

package io.quarkiverse.quinoa.deployment.packagemanager;

import static io.quarkiverse.quinoa.deployment.packagemanager.types.PackageManagerType.detectPackageManagerType;
import static io.quarkiverse.quinoa.deployment.packagemanager.types.PackageManagerType.resolveConfiguredPackageManagerType;
import static java.lang.String.format;

import java.io.BufferedReader;
import java.io.Closeable;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.net.ConnectException;
import java.net.HttpURLConnection;
import java.net.Inet6Address;
import java.net.InetAddress;
import java.net.SocketTimeoutException;
import java.net.URL;
import java.net.UnknownHostException;
import java.nio.file.Path;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.function.Predicate;

import javax.net.ssl.HostnameVerifier;
import javax.net.ssl.HttpsURLConnection;
import javax.net.ssl.SSLSession;

import org.jboss.logging.Logger;

import io.quarkiverse.quinoa.QuinoaNetworkConfiguration;
import io.quarkiverse.quinoa.deployment.SslUtil;
import io.quarkiverse.quinoa.deployment.config.PackageManagerCommandConfig;
import io.quarkiverse.quinoa.deployment.packagemanager.types.PackageManager;
import io.quarkiverse.quinoa.deployment.packagemanager.types.PackageManagerType;
import io.quarkus.deployment.console.ConsoleInstalledBuildItem;
import io.quarkus.deployment.console.StartupLogCompressor;
import io.quarkus.deployment.logging.LoggingSetupBuildItem;
import io.quarkus.deployment.util.ProcessUtil;
import io.quarkus.dev.console.QuarkusConsole;
import io.quarkus.runtime.LaunchMode;

public class PackageManagerRunner {
    private static final Logger LOG = Logger.getLogger(PackageManagerRunner.class);
    public static final Predicate DEV_PROCESS_THREAD_PREDICATE = thread -> thread.getName()
            .matches("Process (stdout|stderr) streamer");

    private final Path directory;

    private final PackageManager packageManager;
    private final Optional consoleInstalledBuildItem;
    private final LoggingSetupBuildItem loggingSetupBuildItem;

    private PackageManagerRunner(Path directory, PackageManager packageManager,
            Optional consoleInstalledBuildItem,
            LoggingSetupBuildItem loggingSetupBuildItem) {
        this.directory = directory;
        this.packageManager = packageManager;
        this.consoleInstalledBuildItem = consoleInstalledBuildItem;
        this.loggingSetupBuildItem = loggingSetupBuildItem;
    }

    public Path getDirectory() {
        return directory;
    }

    public PackageManager getPackageManager() {
        return packageManager;
    }

    public void ci() {
        final PackageManager.Command ci = packageManager.ci();
        LOG.infof("Running Quinoa package manager ci command: %s", ci.commandWithArguments);
        if (!exec(ci)) {
            throw new RuntimeException(
                    format("Error in Quinoa while running package manager ci command: %s", ci.commandWithArguments));
        }
    }

    public void install() {
        final PackageManager.Command install = packageManager.install();
        LOG.infof("Running Quinoa package manager install command: %s", install.commandWithArguments);
        if (!exec(install)) {
            throw new RuntimeException(
                    format("Error in Quinoa while running package manager install command: %s", install.commandWithArguments));
        }
    }

    public void build(LaunchMode mode) {
        final PackageManager.Command build = packageManager.build(mode);
        LOG.infof("Running Quinoa package manager build command: %s", build.commandWithArguments);
        if (!exec(build)) {
            throw new RuntimeException(
                    format("Error in Quinoa while running package manager build command: %s", build.commandWithArguments));
        }
    }

    public void test() {
        final PackageManager.Command test = packageManager.test();
        LOG.infof("Running Quinoa package manager test command: %s", test.commandWithArguments);
        if (!exec(test)) {
            throw new RuntimeException(
                    format("Error in Quinoa while running package manager test command: %s", test.commandWithArguments));
        }
    }

    public void stopDev(Process process) {
        if (process == null || !process.isAlive()) {
            return;
        }
        LOG.infof("Stopping Quinoa package manager live coding as a dev service.");
        try {
            // Kill children before because React is swallowing the signal
            killDescendants(process.toHandle(), false);
            if (process.isAlive()) {
                process.destroy();
                // Force kill descendants if needed
                killDescendants(process.toHandle(), true);
                if (!process.waitFor(10, TimeUnit.SECONDS)) {
                    process.destroyForcibly();
                }
            }
        } catch (Exception e) {
            if (e instanceof InterruptedException) {
                Thread.currentThread().interrupt();
            }
            LOG.errorf(e, "Error while waiting for Quinoa Dev Server process (#%s) to exit.", process.pid());
        } finally {
            if (process.isAlive()) {
                LOG.warnf("Quinoa was not able to stop the Dev Server process (#%s).", process.pid());
            }
        }

    }

    private static void killDescendants(ProcessHandle process, boolean force) {
        process.children().forEach(child -> {
            killDescendants(child, force);
            if (child.isAlive()) {
                if (force) {
                    child.destroyForcibly();
                } else {
                    child.destroy();
                }
            }
        });
    }

    public DevServer dev(Optional consoleInstalled, LoggingSetupBuildItem loggingSetup,
            QuinoaNetworkConfiguration network, String checkPath,
            int checkTimeout) {
        final PackageManager.Command dev = packageManager.dev();
        LOG.infof("Running Quinoa package manager live coding as a dev service: %s", dev.commandWithArguments);
        StartupLogCompressor logCompressor = new StartupLogCompressor(
                "Quinoa package manager live coding dev service starting:",
                consoleInstalled,
                loggingSetup,
                DEV_PROCESS_THREAD_PREDICATE);
        Process p = process(dev);
        Runtime.getRuntime().addShutdownHook(new Thread() {
            public void run() {
                stopDev(p);
            }
        });
        if (checkPath == null) {
            LOG.infof("Quinoa is configured to continue without check if the live coding server is up");
            return new DevServer(p, network.getHost(), logCompressor);
        }
        String ipAddress = null;
        try {
            int i = 0;
            while ((ipAddress = isDevServerUp(network, checkPath)) == null) {
                if (++i >= checkTimeout / 500) {
                    stopDev(p);
                    throw new RuntimeException(
                            "Quinoa package manager live coding port " + network.getPort()
                                    + " is still not listening after the checkTimeout.");
                }
                Thread.sleep(500);
            }
            // Add a small safety delay to make sure all logs are outputted
            Thread.sleep(500);
        } catch (InterruptedException e) {
            stopDev(p);
            Thread.currentThread().interrupt();
            throw new RuntimeException(e);
        }
        return new DevServer(p, ipAddress, logCompressor);
    }

    public static PackageManagerRunner autoDetectPackageManager(Optional configuredBinary,
            PackageManagerCommandConfig packageManagerCommands, Path directory, List paths,
            Optional consoleInstalledBuildItem,
            LoggingSetupBuildItem loggingSetupBuildItem) {
        String binary;
        PackageManagerType type = detectPackageManagerType(directory);
        if (configuredBinary.isEmpty()) {
            binary = type.getOSBinary();
        } else {
            binary = configuredBinary.get();
            type = resolveConfiguredPackageManagerType(binary, type);
        }
        return new PackageManagerRunner(directory, PackageManager.resolve(type, binary, packageManagerCommands, paths),
                consoleInstalledBuildItem, loggingSetupBuildItem);
    }

    public static boolean isWindows() {
        return QuarkusConsole.IS_WINDOWS;
    }

    private Process process(PackageManager.Command command) {
        Process process = null;
        final ProcessBuilder builder = new ProcessBuilder()
                .directory(directory.toFile())
                .command(runner(command));
        if (!command.envs.isEmpty()) {
            builder.environment().putAll(command.envs);
        }
        try {
            process = ProcessUtil.launchProcess(builder, true);
        } catch (IOException e) {
            throw new RuntimeException("Input/Output error while running process.", e);
        }
        return process;
    }

    private boolean exec(PackageManager.Command command) {
        Process process = null;
        HandleOutput handleOutput = null;
        try {
            final ProcessBuilder processBuilder = new ProcessBuilder();
            if (!command.envs.isEmpty()) {
                processBuilder.environment().putAll(command.envs);
            }
            process = processBuilder
                    .directory(directory.toFile())
                    .command(runner(command))
                    .redirectErrorStream(true)
                    .start();
            handleOutput = new HandleOutput(process.getInputStream(), consoleInstalledBuildItem, loggingSetupBuildItem);
            handleOutput.run();
            process.waitFor();
        } catch (IOException e) {
            throw new RuntimeException("Input/Output error while executing command.", e);
        } catch (InterruptedException e) {
            return false;
        } finally {
            if (handleOutput != null) {
                handleOutput.close();
            }
        }
        return process != null && process.exitValue() == 0;
    }

    private String[] runner(PackageManager.Command command) {
        if (isWindows()) {
            return new String[] { "cmd.exe", "/c", command.commandWithArguments };
        } else {
            return new String[] { "sh", "-c", command.commandWithArguments };
        }
    }

    private static class HandleOutput implements Runnable, Closeable {

        private final InputStream is;
        private final Logger.Level logLevel;
        private final StartupLogCompressor logCompressor;
        private final AtomicBoolean closed = new AtomicBoolean(false);

        HandleOutput(InputStream is,
                Optional consoleInstalledBuildItem,
                LoggingSetupBuildItem loggingSetupBuildItem) {
            this(is, Logger.Level.INFO, consoleInstalledBuildItem, loggingSetupBuildItem);
        }

        HandleOutput(InputStream is, Logger.Level logLevel,
                Optional consoleInstalledBuildItem,
                LoggingSetupBuildItem loggingSetupBuildItem) {
            this.is = is;
            this.logLevel = LOG.isEnabled(logLevel) ? logLevel : null;
            logCompressor = new StartupLogCompressor("quinoa", consoleInstalledBuildItem, loggingSetupBuildItem);
        }

        @Override
        public void run() {
            try (InputStreamReader isr = new InputStreamReader(is);
                    BufferedReader reader = new BufferedReader(isr)) {

                for (String line = reader.readLine(); line != null; line = reader.readLine()) {
                    if (logLevel != null) {
                        LOG.log(logLevel, line);
                    }
                }
            } catch (IOException e) {
                if (logLevel != null) {
                    LOG.log(logLevel, "Failed to handle output", e);
                }
                closed.set(true);
                logCompressor.closeAndDumpCaptured();
            }
        }

        @Override
        public void close() {
            if (closed.compareAndSet(false, true)) {
                logCompressor.close();
            }
        }
    }

    public static String isDevServerUp(QuinoaNetworkConfiguration network, String path) {
        if (path == null) {
            return network.getHost();
        }
        final String normalizedPath = path.indexOf("/") == 0 ? path : "/" + path;
        try {
            InetAddress[] addresses = InetAddress.getAllByName(network.getHost());
            for (InetAddress address : addresses) {
                try {
                    final String hostAddress = address.getHostAddress();
                    final String ipAddress = address instanceof Inet6Address ? "[" + hostAddress + "]" : hostAddress;
                    URL url = new URL(String.format("%s://%s:%d%s", network.isTls() ? "https" : "http", ipAddress,
                            network.getPort(), normalizedPath));
                    HttpURLConnection connection;
                    if (network.isTls()) {
                        HttpsURLConnection httpsConnection = (HttpsURLConnection) url.openConnection();
                        if (network.isTlsAllowInsecure()) {
                            httpsConnection.setSSLSocketFactory(SslUtil.createNonValidatingSslContext().getSocketFactory());
                            httpsConnection.setHostnameVerifier(new HostnameVerifier() {
                                @Override
                                public boolean verify(String hostname, SSLSession session) {
                                    return true;
                                }
                            });
                        }
                        connection = httpsConnection;
                    } else {
                        connection = (HttpURLConnection) url.openConnection();
                    }

                    connection.setRequestMethod("GET");
                    connection.setConnectTimeout(2000);
                    connection.setReadTimeout(2000);
                    connection.connect();
                    int code = connection.getResponseCode();
                    // in both cases the server is started, for 404 it might be started on another path
                    return (code == 200 || code == 404) ? ipAddress : null;
                } catch (ConnectException | SocketTimeoutException e) {
                    // Try the next address
                } catch (IOException e) {
                    throw new RuntimeException("Error while checking if package manager dev server is up", e);
                }
            }
        } catch (UnknownHostException e) {
            throw new RuntimeException(e);
        }
        return null;
    }

    public static class DevServer {
        private final Process process;
        private final String hostIPAddress;

        private final StartupLogCompressor logCompressor;

        public DevServer(Process process, String hostIPAddress, StartupLogCompressor logCompressor) {
            this.process = process;
            this.hostIPAddress = hostIPAddress;
            this.logCompressor = logCompressor;
        }

        public Process process() {
            return process;
        }

        public String hostIPAddress() {
            return hostIPAddress;
        }

        public StartupLogCompressor logCompressor() {
            return logCompressor;
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy