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

org.kiwiproject.base.process.ProcessHelper 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.

The newest version!
package org.kiwiproject.base.process;

import static org.kiwiproject.collect.KiwiLists.first;
import static org.kiwiproject.io.KiwiIO.readLinesFromInputStreamOf;
import static org.kiwiproject.io.KiwiIO.streamLinesFromInputStreamOf;

import com.google.common.annotations.VisibleForTesting;
import org.apache.commons.lang3.tuple.Pair;
import org.checkerframework.checker.nullness.qual.Nullable;

import java.io.File;
import java.io.UncheckedIOException;
import java.nio.file.Path;
import java.util.Collection;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.TimeUnit;
import java.util.stream.Stream;

/**
 * Wrapper class around the static utility functions in {@link Processes} that requires an instance, adn thus by using
 * an instance of this class instead of {@link Processes} directly, it will make it much easier to test code that deals
 * with processes.
 * 

* Note that most of the methods are intended only for use on Unix/Linux operating systems. */ public class ProcessHelper { /** * Waits up to {@link Processes#DEFAULT_WAIT_FOR_EXIT_TIME_SECONDS} for the given process to exit. * * @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. * @see Processes#waitForExit(Process) */ public Optional waitForExit(Process process) { return Processes.waitForExit(process); } /** * Waits up to the specified {@code timeout} for the given process to exit. * * @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. * @see Processes#waitForExit(Process, long, TimeUnit) */ public Optional waitForExit(Process process, long timeout, TimeUnit unit) { return Processes.waitForExit(process, timeout, unit); } /** * Launches a new process using the specified {@code command}. * * @param command the list containing the program and its arguments * @return the new {@link Process} * @see Processes#launch(List) */ public Process launch(List command) { return Processes.launch(command); } /** * Launches a new process using the specified {@code workingDirectory} and {@code command}. * * @param workingDirectory the working directory to use * @param command the list containing the program and its arguments * @return the new {@link Process} * @see Processes#launch(File, List) */ public Process launch(@Nullable File workingDirectory, List command) { return Processes.launch(workingDirectory, command); } /** * 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 Processes#launch(String...) */ public Process launch(String... command) { return Processes.launch(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 Processes#pgrep(String) * @see Processes#wasPgrepFlagsCheckSuccessful() * @see Processes#getPgrepFlags() */ public List pgrep(String commandLine) { return Processes.pgrep(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 Processes#pgrep(String, String) * @see Processes#wasPgrepFlagsCheckSuccessful() * @see Processes#getPgrepFlags() */ public List pgrep(String user, String commandLine) { return Processes.pgrep(user, commandLine); } /** * 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 Processes#pgrepWithSingleResult(String) * @see Processes#wasPgrepFlagsCheckSuccessful() * @see Processes#getPgrepFlags() */ public Optional pgrepWithSingleResult(String commandLine) { return Processes.pgrepWithSingleResult(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 Processes#pgrepWithSingleResult(String, String) * @see Processes#wasPgrepFlagsCheckSuccessful() * @see Processes#getPgrepFlags() */ public Optional pgrepWithSingleResult(String user, String commandLine) { return Processes.pgrepWithSingleResult(user, commandLine); } /** * 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 Processes#pgrepList(String) * @see Processes#wasPgrepFlagsCheckSuccessful() * @see Processes#getPgrepFlags() */ public List pgrepList(String commandLine) { return Processes.pgrepList(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 Processes#pgrepList(String, String) * @see Processes#wasPgrepFlagsCheckSuccessful() * @see Processes#getPgrepFlags() */ public List pgrepList(String user, String commandLine) { return Processes.pgrepList(user, 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 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 Processes#pgrepParsedList(String) * @see Processes#wasPgrepFlagsCheckSuccessful() * @see Processes#getPgrepFlags() */ public List> pgrepParsedList(String commandLine) { return Processes.pgrepParsedList(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 Processes#pgrepParsedList(String, String) * @see Processes#wasPgrepFlagsCheckSuccessful() * @see Processes#getPgrepFlags() */ public List> pgrepParsedList(String user, String commandLine) { return Processes.pgrepParsedList(user, commandLine); } /** * Kill a process, waiting up to {@link Processes#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 Processes#kill(long, KillSignal, KillTimeoutAction) */ public int kill(long processId, KillSignal signal, KillTimeoutAction action) { return Processes.kill(processId, signal, 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 Processes#kill(long, KillSignal, long, TimeUnit, KillTimeoutAction) */ public int kill(long processId, KillSignal signal, long timeout, TimeUnit unit, KillTimeoutAction action) { return Processes.kill(processId, signal, timeout, unit, action); } /** * Kill a process, waiting up to {@link Processes#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 * @see Processes#kill(long, String, KillTimeoutAction) */ public int kill(long processId, String signal, KillTimeoutAction action) { return Processes.kill(processId, signal, 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 * @see Processes#kill(long, String, long, TimeUnit, KillTimeoutAction) */ public int kill(long processId, String signal, long timeout, TimeUnit unit, KillTimeoutAction action) { return Processes.kill(processId, signal, timeout, unit, action); } /** * 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 Processes#killForcibly(Process, long, TimeUnit) */ public boolean killForcibly(Process process, long timeout, TimeUnit unit) throws InterruptedException { return Processes.killForcibly(process, timeout, unit); } /** * For a given parent process id, find one and only one child process. There may be no child processes, * in which case an empty {@link Optional} is returned. *

* This method considers it an error if the process has more than one child process id. * * @param parentProcessId the parent process id * @return optional containing the child process id if there is one; otherwise an empty optional * @throws IllegalStateException if there is more than one child process found */ public Optional findChildProcessId(long parentProcessId) { return findChildProcessIdInternal(parentProcessId, this); } /** * This method is entirely for testing, specifically to be able to mock the Process object returned by * calling launch on {@code processHelper} to perform a {@code pgrep -P}. * * @see #launchPgrepWithParentPidFlag(long, ProcessHelper) */ @VisibleForTesting Optional findChildProcessIdInternal(long parentProcessId, ProcessHelper processHelper) { var process = launchPgrepWithParentPidFlag(parentProcessId, processHelper); List lines = readLinesFromInputStreamOf(process); if (lines.isEmpty()) { return Optional.empty(); } else if (lines.size() == 1) { return Optional.of(Processes.getPidOrThrow(first(lines))); } else { throw new IllegalStateException("More than one child process found for process ID " + parentProcessId); } } /** * For a given parent process id, find any child processes. There may be no child processes, in which case an * empty collection is returned. * * @param parentProcessId the parent process id * @return a collection of child pids, or an empty collection if there are no child processes */ public Collection findChildProcessIds(long parentProcessId) { return findChildProcessIdsInternal(parentProcessId, this); } /** * This method is entirely for testing, specifically to be able to mock the Process object returned by * calling launch on {@code processHelper} to perform a {@code pgrep -P}. * * @see #launchPgrepWithParentPidFlag(long, ProcessHelper) */ @VisibleForTesting Collection findChildProcessIdsInternal(long parentProcessId, ProcessHelper processHelper) { var process = launchPgrepWithParentPidFlag(parentProcessId, processHelper); Stream stream = streamLinesFromInputStreamOf(process); return stream.map(Processes::getPidOrThrow).toList(); } /** * @implNote The "-P" flag causes pgrep to "Only match processes whose parent process ID is listed" per * the man page. Also of note, "-P" is equivalent to the {@code --parent} long flag */ private Process launchPgrepWithParentPidFlag(long parentProcessId, ProcessHelper processHelper) { return processHelper.launch("pgrep", "-P", String.valueOf(parentProcessId)); } /** * 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 * @see Processes#whichAsPath(String) */ public Optional whichAsPath(String program) { return Processes.whichAsPath(program); } /** * 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 * @see Processes#which(String) */ public Optional which(String program) { return Processes.which(program); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy