ProcessBuilders.java
// Generated by delombok at Mon Jan 06 07:19:11 UTC 2025
package de.larssh.utils.io;
import static java.util.stream.Collectors.joining;
import java.io.File;
import java.lang.ProcessBuilder.Redirect;
import java.lang.ProcessBuilder.Redirect.Type;
import java.nio.file.Paths;
import java.util.function.UnaryOperator;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import de.larssh.utils.Nullables;
import de.larssh.utils.SystemUtils;
import de.larssh.utils.annotations.PackagePrivate;
import de.larssh.utils.text.Strings;
/**
* This class contains helper methods for {@link ProcessBuilder}.
*/
public final class ProcessBuilders {
/**
* Pattern describing all single characters retaining the special meaning of a
* backslash when the backslash is followed by this characters inside a Unix
* command line arguments.
*/
private static final Pattern UNIX_BACKSLASH_SPECIAL_MEANING_CHARACTER_PATTERN = Pattern.compile("[\r\n!\"$\\\\`]");
/**
* Pattern describing Unix command line arguments, which can be safely used
* without quoting and escaping.
*/
private static final Pattern UNIX_SAFE_ARGUMENT_PATTERN = Pattern.compile("[-.0-9A-Z_a-z]+");
/**
* Pattern describing a single double quote
*/
private static final Pattern WINDOWS_DOUBLE_QUOTE_PATTERN = Pattern.compile("\"");
/**
* Pattern describing all single space characters inside Windows command line
* arguments, requiring an argument to be escaped.
*/
private static final Pattern WINDOWS_SPACE_CHARACTER_PATTERN = Pattern.compile("[ \t\n\013]");
/**
* Appends the redirect {@code operation} to {@code file} to {@code builder}.
* {@code file} is converted into an absolute path and normalized.
*
* @param builder the builder to append to
* @param operation the redirect operation
* @param file the path of the redirection
*/
private static void appendRedirect(final StringBuilder builder, final String operation, final File file) {
final String normalizedPath = file.toPath().toAbsolutePath().normalize().toString();
builder.append(' ').append(operation).append(' ').append(SystemUtils.isWindows() ? escapeArgumentOnWindows(normalizedPath) : escapeArgumentOnUnix(normalizedPath));
}
/**
* Appends redirect operations of {@code processBuilder} to {@code builder}.
*
* @param builder the builder to append to
* @param processBuilder the process builder, which information to handle
*/
@SuppressWarnings("PMD.CyclomaticComplexity")
private static void appendRedirects(final StringBuilder builder, final ProcessBuilder processBuilder) {
// Standard Input
final Redirect input = processBuilder.redirectInput();
if (input.type() == Type.READ && input.file() != null) {
appendRedirect(builder, "<", input.file());
}
// Standard Output
final Redirect output = processBuilder.redirectOutput();
if (output.type() == Type.APPEND && output.file() != null) {
appendRedirect(builder, ">>", output.file());
} else if (output.type() == Type.WRITE && output.file() != null) {
appendRedirect(builder, ">", output.file());
}
// Standard Error
if (processBuilder.redirectErrorStream()) {
builder.append(" 2>&1");
} else {
final Redirect error = processBuilder.redirectError();
if (error.type() == Type.APPEND && error.file() != null) {
appendRedirect(builder, "2>>", error.file());
} else if (error.type() == Type.WRITE && error.file() != null) {
appendRedirect(builder, "2>", error.file());
}
}
}
/**
* Escapes {@code argument} as command line argument for Unix.
*
* <p>
* based on <a href=
* "https://www.gnu.org/software/bash/manual/bash.html#Double-Quotes">"3.1.2.3
* Double Quotes" of the Bash Reference Manual</a>
*
* @param argument the argument to escape
* @return the escaped argument
*/
@PackagePrivate
static String escapeArgumentOnUnix(final String argument) {
// No need to quote and escape in case of clearly safe characters
if (Strings.matches(argument, UNIX_SAFE_ARGUMENT_PATTERN)) {
return argument;
}
// Wrap in double quotes and escape characters with special meanings
return '\"' + Strings.replaceAll(argument, UNIX_BACKSLASH_SPECIAL_MEANING_CHARACTER_PATTERN, "\\\\$0") + '\"';
}
/**
* Escapes {@code argument} as command line argument for Windows.
*
* @param argument the argument to escape
* @return the escaped argument
*/
@PackagePrivate
@SuppressWarnings({"checkstyle:IllegalInstantiation", "checkstyle:XIllegalTypeCustom"})
static String escapeArgumentOnWindows(final String argument) {
final StringBuffer buffer = new StringBuffer();
// The argument needs to be wrapped in double quotes in case of space characters
final boolean quote = argument.isEmpty() || Strings.find(argument, WINDOWS_SPACE_CHARACTER_PATTERN);
if (quote) {
buffer.append('\"');
}
final Matcher matcher = WINDOWS_DOUBLE_QUOTE_PATTERN.matcher(argument);
while (matcher.find()) {
matcher.appendReplacement(buffer, "");
// In case of a double quote trailing backslashes need to be duplicated and the
// found double quote needs to be escaped and appended.
for (int index = buffer.length() - 1; index > -1 && buffer.charAt(index) == '\\'; index -= 1) {
buffer.append('\\');
}
buffer.append("\\\"");
}
matcher.appendTail(buffer);
// In case of quoting trailing backslashes need to be duplicated and the
// trailing double quote needs to be appended.
if (quote) {
for (int index = buffer.length() - 1; index > -1 && buffer.charAt(index) == '\\'; index -= 1) {
buffer.append('\\');
}
buffer.append('\"');
}
return buffer.toString();
}
/**
* Returns a command to launch {@code processBuilder} via command line.
*
* <p>
* This method is most likely used for logging and user output or debugging.
*
* <p>
* The results of this method depend on the Operating System as command line
* arguments on Unix and Windows are quoted and escaped differently.
*
* @param processBuilder the process definition
* @param prependWorkingDirectory if {@code true} the working directory of
* {@code processBuilder} is prepended, else the
* returned value consists of the command line
* only.
* @return the escaped command
*/
public static String toCommandLine(final ProcessBuilder processBuilder, final boolean prependWorkingDirectory) {
final StringBuilder builder = new StringBuilder();
// Working Directory
if (prependWorkingDirectory) {
builder.append(Nullables.orElseGet(processBuilder.directory(), () -> Paths.get(".").toAbsolutePath().normalize().toString())).append("> ");
}
// Command and Arguments
final UnaryOperator<String> escapeArgument = SystemUtils.isWindows() ? ProcessBuilders::escapeArgumentOnWindows : ProcessBuilders::escapeArgumentOnUnix;
builder.append(processBuilder.command().stream().map(escapeArgument).collect(joining(" ")));
// Redirects (Standard Input, Output, Error)
appendRedirects(builder, processBuilder);
return builder.toString();
}
@java.lang.SuppressWarnings("all")
@edu.umd.cs.findbugs.annotations.SuppressFBWarnings(justification = "generated code")
@lombok.Generated
private ProcessBuilders() {
throw new java.lang.UnsupportedOperationException("This is a utility class and cannot be instantiated");
}
}