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

org.kiwiproject.base.process.Processes Maven / Gradle / Ivy

Go to download

Kiwi is a utility library. We really like Google's Guava, and also use Apache Commons. But if they don't have something we need, and we think it is useful, this is where we put it.

There is a newer version: 4.5.2
Show newest version
package org.kiwiproject.base.process;

import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkState;
import static java.util.Objects.isNull;
import static java.util.Objects.nonNull;
import static org.kiwiproject.base.KiwiPreconditions.checkArgumentNotNull;
import static org.kiwiproject.base.KiwiStrings.SPACE;
import static org.kiwiproject.base.KiwiStrings.format;
import static org.kiwiproject.base.KiwiStrings.splitToList;
import static org.kiwiproject.collect.KiwiLists.first;
import static org.kiwiproject.collect.KiwiLists.second;
import static org.kiwiproject.io.KiwiIO.readLinesFromErrorStreamOf;
import static org.kiwiproject.io.KiwiIO.readLinesFromInputStreamOf;
import static org.kiwiproject.io.KiwiIO.streamLinesFromInputStreamOf;

import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.Lists;
import lombok.experimental.UtilityClass;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import org.apache.commons.lang3.tuple.Pair;
import org.checkerframework.checker.nullness.qual.Nullable;
import org.kiwiproject.base.UncheckedInterruptedException;

import java.io.File;
import java.io.IOException;
import java.io.UncheckedIOException;
import java.nio.file.Path;
import java.util.List;
import java.util.Optional;
import java.util.OptionalLong;
import java.util.concurrent.TimeUnit;

/**
 * Utility class for working with operating system processes.
 * 

* Note that most of the methods are intended only for use on Unix/Linux operating systems. * * @see ProcessHelper */ @UtilityClass @Slf4j public class Processes { /** * Exit code that indicates a {@link Process} terminated normally. * * @see Process#exitValue() */ public static final int SUCCESS_EXIT_CODE = 0; /** * Default number of seconds to wait for termination of a process. * * @see #kill(long, KillSignal, KillTimeoutAction) * @see #kill(long, String, KillTimeoutAction) */ public static final int DEFAULT_KILL_TIMEOUT_SECONDS = 5; /** * Default number of seconds to wait for a process to exit. * * @see #waitForExit(Process) */ public static final long DEFAULT_WAIT_FOR_EXIT_TIME_SECONDS = 5; /** * The command flags to use with the {@code pgrep} command for matching and printing the full command line. * For example, to find the "sleep 25" process with pid 32332, we want pgrep to return results in the * format: "[pid] [full command]". For this example the expected result is: {@code 32332 sleep 25}. *

* However, there are differences in pgrep command line arguments between BSD-based systems (including macOS) and * Linux systems. There are also differences between older and newer Linux distributions such as * CentOS/Red Hat 6 and 7. Specifically, some systems require {@code -fl} to match against and print out the full * command line with {@code pgrep}. Other implementations require {@code -fa} (or {@code -f --list-full}). *

* See, for example, Linux pgrep and * BSD pgrep for more information. */ private static final String PGREP_FULL_COMMAND_MATCH_AND_PRINT_FLAGS; private static final boolean PGREP_CHECK_SUCCESSFUL; static { var result = choosePgrepFlags(); PGREP_FULL_COMMAND_MATCH_AND_PRINT_FLAGS = result.getLeft(); PGREP_CHECK_SUCCESSFUL = result.getRight(); } private static final String PGREP_COMMAND = "pgrep"; private static Pair choosePgrepFlags() { String flagsOrNull = findPgrepFlags().orElse(null); return choosePgrepFlags(flagsOrNull); } @VisibleForTesting static Pair choosePgrepFlags(@Nullable String flagsOrNull) { if (isNull(flagsOrNull)) { logPgrepFlagWarnings(); return Pair.of("-fa", false); } return Pair.of(flagsOrNull, true); } private static Optional findPgrepFlags() { return tryPgrepForSleepCommand("-fa", "123") .or(() -> tryPgrepForSleepCommand("-fl", "124")); } private static Optional tryPgrepForSleepCommand(String flags, String sleepTime) { Process sleeperProc = null; try { sleeperProc = launch("sleep", sleepTime); var pid = String.valueOf(sleeperProc.pid()); var pgrepCheckerProc = launch(PGREP_COMMAND, flags, "sleep"); var stdOutLines = readLinesFromInputStreamOf(pgrepCheckerProc); var stdErrLines = readLinesFromErrorStreamOf(pgrepCheckerProc); var expectedCommand = "sleep " + sleepTime; logPgrepCheckInfo(flags, pid, stdOutLines, stdErrLines, expectedCommand); if (linesContainPidAndFullCommand(stdOutLines, pid, expectedCommand)) { LOG.info("Will use [{}] as flags for pgrep full-command listing", flags); return Optional.of(flags); } LOG.trace("Flags [{}] did not produce pgrep full-command listing", flags); return Optional.empty(); } catch (Exception e) { LOG.error("Error checking pgrep flags. pgrep calls may fail in unexpected ways!", e); return Optional.empty(); } finally { if (nonNull(sleeperProc)) { killSilently(sleeperProc.pid()); } } } @VisibleForTesting static void logPgrepCheckInfo(String flags, String pid, List stdOutLines, List stdErrLines, String expectedCommand) { LOG.trace("Checking pgrep flags [{}] for command [{}] with pid {}", flags, expectedCommand, pid); LOG.trace("pid {} stdOut: {}", pid, stdOutLines); if (stdErrLines.isEmpty()) { LOG.trace("pid {} stdErr: {}", pid, stdErrLines); } else { LOG.warn("stdErr checking pgrep flags for pid {} (command: {}): {}", pid, expectedCommand, stdErrLines); } } private static boolean linesContainPidAndFullCommand(List lines, String pid, String expectedCommand) { return lines.stream().anyMatch(line -> line.contains(pid) && line.contains(expectedCommand)); } private static void killSilently(long processId) { try { LOG.trace("Killing sleeper process ({}) used to determine pgrep flags", processId); kill(processId, KillSignal.SIGTERM, KillTimeoutAction.NO_OP); } catch (Exception e) { LOG.warn("Error killing sleeper process ({}) used to determine pgrep flags", processId, e); } } @VisibleForTesting static void logPgrepFlagWarnings() { LOG.warn("Neither -fa nor -fl flags produced PID and full command line, so pgrep commands will behave (or fail) in unexpected ways!"); LOG.warn("If you see this warning, DO NOT use any of the pgrep-related methods in Processes or ProcessHelper and submit a bug report."); LOG.warn("Turn on TRACE-level logging to see standard output and error for pgrep commands"); LOG.warn("We will use -fa even though we know this will not work, instead of throwing an exception"); } /** * Use this method to determine if calling any of the pgrep methods in this class will work as expected. * * @return true if the pgrep check to determine the flags to use for full command matching was successful; false * otherwise. If false, you should NOT use any of the pgrep methods. * @see #getPgrepFlags() */ public static boolean wasPgrepFlagsCheckSuccessful() { return PGREP_CHECK_SUCCESSFUL; } /** * Returns the pgrep flags that {@link Processes} will use in all pgrep methods. * Use {@link #wasPgrepFlagsCheckSuccessful()} to check if the flags were chosen successfully * to ensure that pgrep commands will work as expected on your OS. * * @return the flags that will be used in pgrep commands */ public static String getPgrepFlags() { return PGREP_FULL_COMMAND_MATCH_AND_PRINT_FLAGS; } /** * Get a process id, or "pid". *

* Earlier versions of Kiwi used to perform some nasty things (checking if the actual process class was * {@code java.lang.UNIXProcess}, then using reflection to get the value of the private {@code pid} field) * to obtain Process pids, but since JDK 9 offers the {@link Process#pid()} method, that is no longer necessary. * This method is now a simple delegation to {@link Process#pid()} and you should now prefer that method. * * @param process the process to get the process id (pid) from * @return the process id of {@code process} * @throws UnsupportedOperationException if the Process implementation does not support getting the pid * @see Process#pid() */ public static long processId(Process process) { checkArgumentNotNull(process); return process.pid(); } /** * Get a process id, or "pid", if it is available from the {@link Process} implementation, wrapped inside * an OptionalLong. If the pid is not available for whatever reason, return an empty OptionalLong. * * @param process the process to get the process id (pid) from * @return an OptionalLong containing the process if of {@code process} or an empty OptionalLong if the * {@link Process} implementation does not support getting the pid for whatever reason. * @implNote the {@link Process#pid()} method says it can throw {@link UnsupportedOperationException} if the * "implementation does not support this operation" but does not specify under what circumstances that can * happen, and I have not been able to find this information using Google, Bing, or DuckDuckGo. This method * logs a warning along with the exception, so if this occurs, check your logs for a possible reason. */ public static OptionalLong processIdOrEmpty(Process process) { checkArgumentNotNull(process); try { return OptionalLong.of(process.pid()); } catch (UnsupportedOperationException e) { LOG.warn("The JDK cannot get the PID of the given Process. Check stack trace for a possible reason", e); return OptionalLong.empty(); } } /** * Check if the given {@link Process} has an exit code representing successful termination. * * @param process the Process, assumed to have exited * @return true if the process terminated successfully * @see #isSuccessfulExitCode(int) * @see #SUCCESS_EXIT_CODE * @see Process#exitValue() */ public static boolean hasSuccessfulExitCode(Process process) { return isSuccessfulExitCode(process.exitValue()); } /** * Check if the given exit code represents successful termination. * * @param exitCode the exit code to check * @return true if the exit code represents success * @see #SUCCESS_EXIT_CODE * @see Process#exitValue() */ public static boolean isSuccessfulExitCode(int exitCode) { return exitCode == SUCCESS_EXIT_CODE; } /** * Check if the given exit code represents anything other than successful termination. * In other words, is the exit code nonzero? * * @param exitCode the exit code to check * @return true if the exit code is nonzero * @see Process#exitValue() * @implNote This method is specifically named to indicate that the exit code does not * represent success, leaving the possibility open that a nonzero exit code can indicate * some condition other than an error. */ public static boolean isNonzeroExitCode(int exitCode) { return !isSuccessfulExitCode(exitCode); } /** * Waits up to {@link #DEFAULT_WAIT_FOR_EXIT_TIME_SECONDS} for the given process to exit. *

* Note that this method does not destroy the process if it times out waiting. * * @param process the process to wait for * @return an {@link Optional} that will contain the exit code if the process exited before the timeout, or * empty if the process did not exit before the timeout expired. */ public static Optional waitForExit(Process process) { return waitForExit(process, DEFAULT_WAIT_FOR_EXIT_TIME_SECONDS, TimeUnit.SECONDS); } /** * Waits up to the specified {@code timeout} for the given process to exit. *

* Note that this method does not destroy the process if it times out waiting. * * @param process the process to wait for * @param timeout the value of the time to wait * @param unit the unit of time to wait * @return an {@link Optional} that will contain the exit code if the process exited before the timeout, or * empty if the process did not exit before the timeout expired. */ public static Optional waitForExit(Process process, long timeout, TimeUnit unit) { try { boolean exited = process.waitFor(timeout, unit); return exited ? Optional.of(process.exitValue()) : Optional.empty(); } catch (InterruptedException e) { LOG.warn("Interrupted waiting for process to exit", e); Thread.currentThread().interrupt(); return Optional.empty(); } } /** * Launches a new process using the specified {@code command}. This is just a convenience wrapper around * creating a new {@link ProcessBuilder} and calling {@link ProcessBuilder#start()}. *

* This wrapper also converts any thrown {@link IOException} to an {@link UncheckedIOException}. *

* If you need more flexibility than provided in this simple wrapper, use {@link ProcessBuilder} directly. * * @param command the list containing the program and its arguments * @return the new {@link Process} * @see ProcessBuilder#ProcessBuilder(List) * @see ProcessBuilder#start() */ public static Process launch(List command) { return launch(null, command); } /** * Launches a new process using the specified {@code command} with the given working directory. * This is just a convenience wrapper around creating a new {@link ProcessBuilder}, setting the * {@link ProcessBuilder#directory(File) working directory}, and calling {@link ProcessBuilder#start()}. *

* This wrapper converts any thrown {@link IOException} to an {@link UncheckedIOException}. *

* If you need more flexibility than provided in this simple wrapper, use {@link ProcessBuilder} directly. * * @param workingDirectory the working directory to use * @param command the list containing the program and its arguments * @return the new {@link Process} * @see ProcessBuilder#ProcessBuilder(List) * @see ProcessBuilder#directory(File) * @see ProcessBuilder#start() */ public static Process launch(@Nullable File workingDirectory, List command) { try { return launchProcessInternal(workingDirectory, command); } catch (IOException e) { throw new UncheckedIOException("Error launching command: " + command, e); } } /** * Launches a new process using the specified {@code command}. * * @param command a list containing the program and its arguments * @return the new {@link Process} * @see #launch(List) */ public static Process launch(String... command) { return launch(Lists.newArrayList(command)); } /** * Does a {@code pgrep} with the specified full command. * * @param commandLine the full command to match * @return a list of matching process ids (pids) * @see #pgrep(String, String) * @see #wasPgrepFlagsCheckSuccessful() * @see #getPgrepFlags() */ public static List pgrep(String commandLine) { return pgrep(null, commandLine); } /** * Does a {@code pgrep} with the specified full command. * * @param user the OS user (passed to the {@code -u} option) * @param commandLine the full command to match * @return list of matching process ids (pids) * @see #wasPgrepFlagsCheckSuccessful() * @see #getPgrepFlags() */ public static List pgrep(String user, String commandLine) { try { List command = buildPgrepCommand(user, commandLine); var process = launchProcessInternal(command); return streamLinesFromInputStreamOf(process) .map(Processes::getPidOrThrow) .toList(); } catch (IOException e) { throw new UncheckedIOException( format("Error executing pgrep with user [%s] and command [%s]", user, commandLine), e); } } /** * Does a {@code pgrep} against the specified full command, expecting a single result, or no result. * * @param commandLine the full command line * @return an optional either containing a process id, or an empty optional * @see #wasPgrepFlagsCheckSuccessful() * @see #getPgrepFlags() */ public static Optional pgrepWithSingleResult(String commandLine) { return pgrepWithSingleResult(null, commandLine); } /** * Does a {@code pgrep} against the specified full command, expecting a single result for a specific user, or no result. * * @param user the OS user (passed to the {@code -u} option) * @param commandLine the full command to match * @return an optional either containing a process id, or an empty optional * @see #wasPgrepFlagsCheckSuccessful() * @see #getPgrepFlags() */ public static Optional pgrepWithSingleResult(String user, String commandLine) { List pids = pgrep(user, commandLine); if (pids.isEmpty()) { return Optional.empty(); } checkState(pids.size() == 1, "Expecting exactly one result pid for command [%s], but received %s: %s", commandLine, pids.size(), pids); return Optional.of(first(pids)); } /** * Does a {@code pgrep} with the specified full command. * * @param commandLine the full command line to match * @return a list of pgrep output, with each line in format "{pid} {command}" * @see #pgrepList(String, String) * @see #wasPgrepFlagsCheckSuccessful() * @see #getPgrepFlags() */ public static List pgrepList(String commandLine) { return pgrepList(null, commandLine); } /** * Does a {@code pgrep} with the specified full command. * * @param user the OS user (passed to the {@code -u} option) * @param commandLine the full command line to match * @return a list of pgrep output, with each line in format "{pid} {command}" * @see #wasPgrepFlagsCheckSuccessful() * @see #getPgrepFlags() */ public static List pgrepList(String user, String commandLine) { try { List command = buildPgrepListCommand(user, commandLine); var process = launchProcessInternal(command); return readLinesFromInputStreamOf(process); } catch (IOException e) { throw new UncheckedIOException( format("Error executing pgrep with user [%s] and command [%s]", user, commandLine), e); } } /** * Does a {@code pgrep} for the specified full command, returning a list of pairs containing the * process id (pid) and the matched command line. * * @param commandLine the full command line to match * @return a list of {@link Pair} objects; each pair contains the pid as a {@code Long} and the associated full command * @see #pgrepParsedList(String, String) * @see #wasPgrepFlagsCheckSuccessful() * @see #getPgrepFlags() */ public static List> pgrepParsedList(String commandLine) { return pgrepParsedList(null, commandLine); } /** * Does a {@code pgrep} for the specified full command, returning a list of pairs containing the * process id (pid) and the matched command line. * * @param user the OS user (passed to the {@code -u} option) * @param commandLine the full command line to match * @return a list of {@link Pair} objects; each pair contains the pid as a {@code Long} and the associated full command * @see #wasPgrepFlagsCheckSuccessful() * @see #getPgrepFlags() */ public static List> pgrepParsedList(String user, String commandLine) { List lines = pgrepList(user, commandLine); return lines.stream().map(Processes::pairFromPgrepLine).toList(); } private static Pair pairFromPgrepLine(String line) { List splat = splitToList(line, SPACE, 2); var pid = getPidOrThrow(first(splat)); var command = second(splat); return Pair.of(pid, command); } static Long getPidOrThrow(String pidString) { try { return Long.valueOf(pidString); } catch (NumberFormatException e) { throw new IllegalArgumentException("pid must be a number", e); } } private static List buildPgrepCommand(String user, String commandLine) { return buildPgrepCommand(user, commandLine, "-f"); } private static List buildPgrepListCommand(String user, String commandLine) { return buildPgrepCommand(user, commandLine, PGREP_FULL_COMMAND_MATCH_AND_PRINT_FLAGS); } private static List buildPgrepCommand(String user, String commandLine, String flags) { checkArgument(doesNotContainWhitespace(flags), "Currently only short flags specified together with no whitespace" + " are supported, e.g. -fl and NOT -f -l. Offending flags: %s", flags); if (StringUtils.isBlank(user)) { return Lists.newArrayList(PGREP_COMMAND, flags, commandLine); } return Lists.newArrayList(PGREP_COMMAND, "-u", user, flags, commandLine); } private static boolean doesNotContainWhitespace(String string) { return !string.contains(" "); } /** * Kill a process, waiting up to {@link #DEFAULT_KILL_TIMEOUT_SECONDS} seconds for it to terminate. * * @param processId the pid of the process to kill * @param signal the kill signal; this could be the signal number (e.g. "1") or name (e.g. "SIGHUP") * @param action the {@link KillTimeoutAction} to take if the process doesn't terminate within the allotted time * @return the exit code from the {@code kill} command, or {@code -1} if {@code action} is * @see #kill(long, String, long, TimeUnit, KillTimeoutAction) */ public static int kill(long processId, KillSignal signal, KillTimeoutAction action) { return kill(processId, signal.number(), action); } /** * Kill a process, waiting up to {@link #DEFAULT_KILL_TIMEOUT_SECONDS} seconds for it to terminate. * * @param processId the pid of the process to kill * @param signal the kill signal; this could be the signal number (e.g. "1") or name (e.g. "SIGHUP") * @param action the {@link KillTimeoutAction} to take if the process doesn't terminate within the allotted time * @return the exit code from the {@code kill} command, or {@code -1} if {@code action} is * {@link KillTimeoutAction#NO_OP} and the kill command times out * @throws UncheckedIOException if an I/O error occurs while killing the process */ public static int kill(long processId, String signal, KillTimeoutAction action) { return kill(processId, signal, DEFAULT_KILL_TIMEOUT_SECONDS, TimeUnit.SECONDS, action); } /** * Kill a process, waiting up to {@code timeout} in the specified {@link TimeUnit} for it to terminate. * * @param processId the pid of the process to kill * @param signal the kill signal enum * @param timeout the time to wait for the process to be killed * @param unit the time unit associated with {@code timeout} * @param action the {@link KillTimeoutAction} to take if the process doesn't terminate within the allotted time * @return the exit code from the {@code kill} command, or {@code -1} if {@code action} is * @see #kill(long, String, long, TimeUnit, KillTimeoutAction) */ public static int kill(long processId, KillSignal signal, long timeout, TimeUnit unit, KillTimeoutAction action) { return kill(processId, signal.number(), timeout, unit, action); } /** * Kill a process, waiting up to {@code timeout} in the specified {@link TimeUnit} for it to terminate. * * @param processId the pid of the process to kill * @param signal the kill signal; this could be the signal number (e.g. "1") or name (e.g. "SIGHUP") * @param timeout the time to wait for the process to be killed * @param unit the time unit associated with {@code timeout} * @param action the {@link KillTimeoutAction} to take if the process doesn't terminate within the allotted time * @return the exit code from the {@code kill} command, or {@code -1} if {@code action} is * {@link KillTimeoutAction#NO_OP} and the kill command times out * @throws UncheckedIOException if an I/O error occurs while killing the process */ public static int kill(long processId, String signal, long timeout, TimeUnit unit, KillTimeoutAction action) { try { var killProcess = launchProcessInternal("kill", KillSignal.withLeadingDash(signal), String.valueOf(processId)); return killInternal(processId, killProcess, timeout, unit, action); } catch (IOException e) { throw new UncheckedIOException(e); } } private static Process launchProcessInternal(String... commandLine) throws IOException { return launchProcessInternal(Lists.newArrayList(commandLine)); } private static Process launchProcessInternal(List command) throws IOException { return launchProcessInternal(null, command); } private static Process launchProcessInternal(@Nullable File workingDirectory, List command) throws IOException { return new ProcessBuilder(command) .directory(workingDirectory) .start(); } /** * Equivalent to a {@code kill -9} (i.e., a {@code SIGKILL}). * * @param process the process to kill forcibly * @param timeout the time to wait for the process to be forcibly killed * @param unit the time unit associated with the {@code timeout} * @return {@code true} if {@code process} was killed before the timeout period elapsed; {@code false} otherwise * @throws InterruptedException if the current thread is interrupted while waiting * @see Process#destroyForcibly() * @see Process#waitFor(long, TimeUnit) */ public static boolean killForcibly(Process process, long timeout, TimeUnit unit) throws InterruptedException { return process.destroyForcibly().waitFor(timeout, unit); } @VisibleForTesting static int killInternal(long processId, Process killProcess, long timeout, TimeUnit unit, KillTimeoutAction action) { try { boolean exitedBeforeWaitTimeout = killProcess.waitFor(timeout, unit); if (exitedBeforeWaitTimeout) { return killProcess.exitValue(); } return doTimeoutAction(action, killProcess, processId); } catch (InterruptedException e) { Thread.currentThread().interrupt(); throw new UncheckedInterruptedException(e); } } private static int doTimeoutAction(KillTimeoutAction action, Process killProcess, long processId) throws InterruptedException { switch (action) { case FORCE_KILL -> { boolean killedBeforeWaitTimeout = killForcibly(killProcess, 1L, TimeUnit.SECONDS); validateKilledBeforeTimeout(processId, killedBeforeWaitTimeout); return killProcess.exitValue(); } case THROW_EXCEPTION -> throw new IllegalStateException( format("Process %s did not end before timeout (and exception was requested)", processId)); case NO_OP -> { LOG.warn("Process {} did not end before timeout and no-op action requested, so doing nothing", processId); return -1; } default -> throw new IllegalStateException("Unaccounted for action: " + action); } } private static void validateKilledBeforeTimeout(long processId, boolean killedBeforeWaitTimeout) { if (!killedBeforeWaitTimeout) { throw new IllegalStateException( format("Process %s was not killed before 1 second timeout expired", processId)); } } /** * Locate a program in the user's path, returning the result as a {@link Path}. * * @param program the program to locate * @return an Optional containing the full {@link Path} to the program, or an empty Optional if not found * @implNote If there is more than program found, only the first one is returned */ public static Optional whichAsPath(String program) { return which(program).map(Path::of); } /** * Locate a program in the user's path. * * @param program the program to locate * @return an Optional containing the full path to the program, or an empty Optional if not found * @implNote If there is more than program found, only the first one is returned */ public static Optional which(String program) { var whichProc = launch("which", program); var stdOut = readLinesFromInputStreamOf(whichProc); return stdOut.stream().findFirst(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy