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

org.neo4j.server.startup.WindowsBootloaderOs Maven / Gradle / Ivy

There is a newer version: 5.26.1
Show newest version
/*
 * Copyright (c) "Neo4j"
 * Neo4j Sweden AB [https://neo4j.com]
 *
 * This file is part of Neo4j.
 *
 * Neo4j is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see .
 */
package org.neo4j.server.startup;

import static java.lang.String.format;
import static java.util.Arrays.stream;
import static org.apache.commons.lang3.StringUtils.isNotEmpty;
import static org.apache.commons.lang3.StringUtils.join;
import static org.neo4j.configuration.BootloaderSettings.windows_tools_directory;
import static org.neo4j.configuration.GraphDatabaseSettings.logs_directory;
import static org.neo4j.server.startup.Bootloader.EXIT_CODE_NOT_RUNNING;

import java.io.ByteArrayOutputStream;
import java.io.PrintStream;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import org.apache.commons.lang3.StringUtils;
import org.eclipse.collections.api.factory.Lists;
import org.eclipse.collections.api.list.MutableList;
import org.neo4j.cli.CommandFailedException;
import org.neo4j.configuration.BootloaderSettings;
import org.neo4j.time.Stopwatch;
import org.neo4j.util.FeatureToggles;
import org.neo4j.util.Preconditions;
import org.neo4j.util.VisibleForTesting;

class WindowsBootloaderOs extends BootloaderOsAbstraction {
    private static final boolean ESCAPE_ASTERISKS =
            FeatureToggles.flag(WindowsBootloaderOs.class, "escapeAsterisks", true);
    static final String PRUNSRV_AMD_64_EXE = "prunsrv-amd64.exe";
    static final String PRUNSRV_I_386_EXE = "prunsrv-i386.exe";
    private static final String POWERSHELL_EXE = "powershell.exe";
    private static final int WINDOWS_PATH_MAX_LENGTH = 250;

    private static final ExitCodeMessageMapper SERVICE_COMMANDS_FAILURE = e -> "Unexpected service command failure.";
    private static final ExitCodeMessageMapper POWERSHELL_COMMANDS_FAILURE =
            e -> "Unexpected powershell command failure.";

    WindowsBootloaderOs(Bootloader ctx) {
        super(ctx);
    }

    @Override
    long start() throws CommandFailedException {
        if (!serviceInstalled()) {
            throw new CommandFailedException("Neo4j service is not installed", EXIT_CODE_NOT_RUNNING);
        }
        issueServiceCommand("ES", new BlockingProcess());
        return UNKNOWN_PID;
    }

    private static class BlockingProcess implements ProcessStages {
        @Override
        public void preStart(ProcessManager processManager, ProcessBuilder processBuilder) {
            processBuilder.redirectOutput(ProcessBuilder.Redirect.DISCARD);
            processBuilder.redirectError(ProcessBuilder.Redirect.DISCARD);
        }

        @Override
        public void postStart(ProcessManager processManager, Process process) throws Exception {
            processManager.waitUntilSuccessful(process, SERVICE_COMMANDS_FAILURE);
        }
    }

    @Override
    void stop(long pid) throws CommandFailedException {
        if (serviceInstalled()) {
            issueServiceCommand("SS", ProcessStages.NO_OP);
        }
    }

    @Override
    void installService() throws CommandFailedException {
        runServiceCommand("IS");
    }

    @Override
    long console() throws CommandFailedException {
        return bootloader.processManager().run(buildStandardStartArguments(), new ConsoleProcess(false));
    }

    @Override
    long admin() throws CommandFailedException {
        if (ESCAPE_ASTERISKS) {
            bootloader.additionalArgs.replaceAll(WindowsBootloaderOs::quoteWildcards);
        }
        return super.admin();
    }

    /**
     * Replaces the wildcard asterisk (*) with an escaped variant ("*") to prevent unexpected globbing.
     */
    private static String quoteWildcards(String s) {
        if (s.equals("*")) {
            return "\"*\"";
        }
        return s;
    }

    private void runServiceCommand(String baseCommand) {
        MutableList argList = baseServiceCommandArgList(baseCommand);
        Path home = bootloader.home();
        Path logs = bootloader.config().get(logs_directory);
        Path jvmDll = Path.of(getJavaCmd()).getParent().resolve(Path.of("server", "jvm.dll"));
        Preconditions.checkState(Files.exists(jvmDll), "Couldn't find the jvm DLL file %s", jvmDll);
        List jvmOpts = getJvmOpts();
        argList.with(arg("--StartMode", "jvm"))
                .with(arg("--StartMethod", "start"))
                .with(arg("--ServiceUser", "LocalSystem"))
                .with(arg("--StartPath", home.toString()))
                .with(multiArg("--StartParams", "--config-dir=" + bootloader.confDir(), "--home-dir=" + home))
                .with(arg("--StopMode", "jvm"))
                .with(arg("--StopMethod", "stop"))
                .with(arg("--StopPath", home.toString()))
                .with(arg("--Description", "Neo4j Graph Database - " + home))
                .with(arg("--DisplayName", "Neo4j Graph Database - " + serviceName()))
                .with(arg("--Jvm", jvmDll.toString()))
                .with(arg("--LogPath", logs.toString()))
                .with(arg("--StdOutput", logs.resolve("service-out.log").toString()))
                .with(arg("--StdError", logs.resolve("service-error.log").toString()))
                .with(arg("--LogPrefix", "neo4j-service"))
                .with(arg("--Classpath", getClassPath()))
                .with(multiArg("--JvmOptions", jvmOpts.toArray(new String[0])))
                .with(arg("--Startup", "auto"))
                .with(arg("--StopClass", bootloader.entrypoint.getName()))
                .with(arg("--StartClass", bootloader.entrypoint.getName()));
        for (String additionalArg : bootloader.additionalArgs) {
            argList = argList.with(arg("++StartParams", additionalArg));
        }
        // Apparently the Xms/Xmx options are passed in a special form here too
        argList = includeMemoryOption(jvmOpts, argList, "-Xms", "--JvmMs", "Start");
        argList = includeMemoryOption(jvmOpts, argList, "-Xmx", "--JvmMx", "Max");
        runProcess(argList, new ServiceCommandProcess());
    }

    private static class ServiceCommandProcess implements ProcessStages {
        @Override
        public void preStart(ProcessManager processManager, ProcessBuilder processBuilder) {
            processBuilder.inheritIO();
        }

        @Override
        public void postStart(ProcessManager processManager, Process process) throws Exception {
            processManager.waitUntilSuccessful(process, SERVICE_COMMANDS_FAILURE);
        }
    }

    private static String multiArg(String key, String... values) {
        // Procrun expects us to split each option with `;` if these characters are used inside the actual option values
        // that will cause problems in parsing. To overcome the problem, we need to escape those characters by placing
        // them inside single quotes.
        List argsEscaped = stream(values)
                .peek(WindowsBootloaderOs::throwIfContainsSingleQuotes)
                .map(opt -> opt.replace(";", "';'"))
                .map(opt -> opt.replace("#", "'#'"))
                .toList();
        return arg(key, join(argsEscaped, ';'));
    }

    private static void throwIfContainsSingleQuotes(String s) {
        // A limitation/bug in prunsrv not parsing ' characters correctly. It is better to throw exception than fail
        // silently like before
        if (s.contains("'")) {
            var firstIndex = s.indexOf("'");
            var context = s.substring(Math.max(firstIndex - 25, 0), Math.min(s.length(), firstIndex + 25));
            throw new CommandFailedException(format(
                    "We are unable to support values that contain single quote marks ('). Single quotes found in value: %s",
                    context));
        }
    }

    private String serviceName() {
        return bootloader.config().get(BootloaderSettings.windows_service_name);
    }

    @Override
    void uninstallService() throws CommandFailedException {
        issueServiceCommand("DS", new BlockingProcess());
        Stopwatch stopwatch = Stopwatch.start();
        while (serviceInstalled()
                && !stopwatch.hasTimedOut(Bootloader.DEFAULT_NEO4J_SHUTDOWN_TIMEOUT, TimeUnit.SECONDS)) {
            try {
                Thread.sleep(300);
            } catch (InterruptedException ie) {
                Thread.currentThread().interrupt();
                break;
            }
        }
    }

    @Override
    void updateService() throws CommandFailedException {
        runServiceCommand("US");
    }

    @Override
    Optional getPidIfRunning() {
        String status = getStatus();
        boolean stopped = StringUtils.isEmpty(status) || status.startsWith("Stopped");
        if (stopped) {
            return Optional.empty();
        }
        return Optional.of(UNKNOWN_PID);
    }

    @Override
    boolean isRunning(long pid) {
        return getPidIfRunning().isPresent();
    }

    @Override
    boolean serviceInstalled() {
        return StringUtils.isNotEmpty(getStatus());
    }

    private String getStatus() {
        try {
            // These are the possible states Get-Service can reply with:
            // - Stopped
            // - StartPending
            // - StopPending
            // - Running
            // - ContinuePending
            // - PausePending
            // - Paused
            //
            // It seems plausible to interpret anything other than "Stopped" as running, at least for how the Neo4j boot
            // loader is interacting with it
            //
            // Stderr should not be part of the stream since Get-Service will write stuff like:
            //      Get-service : Cannot find any service with service name 'neo4j'
            // to stderr and we should not interpret that as the service is running.
            //
            return stream(resultFromPowerShellCommand("Get-Service", serviceName(), "|", "Format-Table", "-AutoSize"))
                    .filter(s -> s.contains(serviceName()))
                    .findFirst()
                    .orElse("");
        } catch (BootProcessFailureException e) {
            return ""; // Service did not exist
        }
    }

    // For debugging flaky behaviour on WindowsServer2019
    @VisibleForTesting
    String[] getServiceStatusResult() throws BootProcessFailureException {
        String serviceName = serviceName();
        return resultFromPowerShellCommand("Get-Service", serviceName, "|", "Format-Table", "-AutoSize");
    }

    private String[] resultFromPowerShellCommand(String... command) {
        var outBuffer = new ByteArrayOutputStream();
        try (var out = new PrintStream(outBuffer)) {
            bootloader.processManager().run(asPowershellScript(List.of(command)), new PowershellWithResult(out));
            return outBuffer.toString().split(format("%n"));
        }
    }

    private static class PowershellWithResult implements ProcessStages {
        private final PrintStream out;

        PowershellWithResult(PrintStream out) {
            this.out = out;
        }

        @Override
        public void preStart(ProcessManager processManager, ProcessBuilder processBuilder) {
            processBuilder.redirectError(ProcessBuilder.Redirect.DISCARD);
        }

        @Override
        public void postStart(ProcessManager processManager, Process process) throws Exception {
            out.write(process.getInputStream().readAllBytes());
            processManager.waitUntilSuccessful(process, POWERSHELL_COMMANDS_FAILURE);
        }
    }

    private void issueServiceCommand(String serviceCommand, ProcessStages behaviour) {
        runProcess(baseServiceCommandArgList(serviceCommand), behaviour);
    }

    private void runProcess(List command, ProcessStages behaviour) {
        List entireCommand = asExternalCommand(command);
        var powershellProcessId = bootloader.processManager().run(entireCommand, behaviour);
        if (entireCommand.stream().anyMatch(cmd -> cmd.equals(POWERSHELL_EXE))
                && command.stream()
                        .anyMatch(cmd -> cmd.endsWith(PRUNSRV_I_386_EXE) || cmd.endsWith(PRUNSRV_AMD_64_EXE))) {
            // This is special condition where we run a command with our prunsrv windows-service util and we have to run
            // it with powershell, probably because we're running a command which exceeds 2000 characters which is the
            // limit of cmd.exe. Since it seems to be really hard to make powershell wait for completion of commands
            // that it runs (we've certainly tried) then we have to try and wait the completion manually here. The
            // general idea is to see if there's any prunsrv process running and we're simply waiting until there is
            // none. Now this is somewhat risky because if there's any other process with the exact same name we'll
            // wait here for the max time. Although, knowing that the PS1 scripts that this was ported from doesn't even
            // have the option to run these prunsrv commands in powershell.exe, it always ran them in cmd.exe. The
            // main cause of a command line being too long for cmd.exe is that the classpath is too long since
            // other things are somewhat fixed and doesn't exceed this limit on any sane environment. And the main
            // reason the classpath is too long is that we're currently running in a test environment, because in
            // a real-world packaging environment the classpath is a couple of wildcard directories.
            Stopwatch stopwatch = Stopwatch.start();
            do {
                try {
                    // First check if the powershell process that should have started the "prunsrv" command still runs
                    resultFromPowerShellCommand("Get-Process", "-Id", String.valueOf(powershellProcessId));
                    // Then check if the actual prunsrv process is still running
                    resultFromPowerShellCommand(
                            "Get-Process", PRUNSRV_AMD_64_EXE + "," + PRUNSRV_I_386_EXE + "," + POWERSHELL_EXE);
                    // ... if these two commands complete normally then either the powershell process that's starting
                    // the prunsrv process still runs, or there's at least one running process containing that
                    // prunsrv name

                    try {
                        Thread.sleep(100);
                    } catch (InterruptedException ie) {
                        Thread.currentThread().interrupt();
                        break;
                    }
                } catch (BootProcessFailureException e) {
                    // If this command returns exit code != 0 it typically means that there's no processes of this name
                    // running
                    break;
                }
            } while (!stopwatch.hasTimedOut(Bootloader.DEFAULT_NEO4J_SHUTDOWN_TIMEOUT, TimeUnit.SECONDS));
        }
    }

    private MutableList baseServiceCommandArgList(String serviceCommand) {
        return Lists.mutable
                .with(format("& %s", escapeQuote(findPrunCommand().toString())))
                .with(format("//%s//%s", serviceCommand, serviceName()));
    }

    private static List asPowershellScript(List command) {
        return asExternalCommand(List.of(String.join(" ", command)));
    }

    private static List asExternalCommand(List command) {
        // We use powershell rather than cmd.exe because cmd.exe doesn't support large argument lists and will wait.
        // Powershell tolerates much longer argument lists and will not wait
        Stream argsAsOne = command.size() < 2
                ? Stream.empty()
                : Stream.of(command.stream()
                        .skip(1)
                        .map(WindowsBootloaderOs::escapeQuote)
                        .collect(Collectors.joining(" ")));
        return Stream.concat(
                        Stream.of(
                                POWERSHELL_EXE,
                                "-OutputFormat",
                                "Text",
                                "-ExecutionPolicy",
                                "Bypass",
                                "-Command",
                                command.get(0)),
                        argsAsOne)
                .toList();
    }

    private Path findPrunCommand() {
        // This is apparently a standard way of finding this out on Windows
        boolean is64bit = isNotEmpty(bootloader.getEnv("ProgramFiles(x86)"));
        // These two files are part of the Neo4j packaging
        String prunSrvName = is64bit ? PRUNSRV_AMD_64_EXE : PRUNSRV_I_386_EXE;
        Path tools = bootloader.config().get(windows_tools_directory);
        Path path = tools.resolve(prunSrvName);
        Preconditions.checkState(
                Files.exists(path),
                "Couldn't find prunsrv file for interacting with the windows service subsystem %s",
                path);

        int length = path.toString().length();
        if (length >= WINDOWS_PATH_MAX_LENGTH) {
            bootloader
                    .environment
                    .err()
                    .printf(
                            "WARNING: Path length over %s characters detected. The service may not work correctly because of limitations in"
                                    + " the Windows operating system when dealing with long file paths. Path:%s (length:%s)%n",
                            WINDOWS_PATH_MAX_LENGTH, path, length);
        }
        return path;
    }

    private MutableList includeMemoryOption(
            List jvmOpts,
            MutableList argList,
            String option,
            String serviceOption,
            String description) {
        String memory = findOptionValue(jvmOpts, option);
        if (memory != null) {
            argList = argList.with(arg(serviceOption, memory));
            bootloader.environment.out().println("Use JVM " + description + " Memory of " + memory);
        }
        return argList;
    }

    private static String findOptionValue(List opts, String option) {
        for (String opt : opts) {
            if (opt.startsWith(option)) {
                return opt.substring(option.length());
            }
        }
        return null;
    }

    private static String arg(String key, String value) {
        return value == null ? key : format("%s=%s", key, value);
    }

    private static String escapeQuote(String str) {
        // Using single quotes stops powershell from trying to evaluate the contents of the string
        // replace pre-existing single quotes with double single quotes - this is the correct escape mechanism for
        // powershell
        return format("'%s'", str.replace("'", "''"));
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy