com.github.writethemfirst.approvals.utils.CommandFinder Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of approvals-java Show documentation
Show all versions of approvals-java Show documentation
Approval testing library for Java. Alleviates the burden of hand-writing assertions.
/*
* Approvals-Java - Approval testing library for Java. Alleviates the burden of hand-writing assertions.
* Copyright © 2018 Write Them First!
*
* This program 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 com.github.writethemfirst.approvals.utils;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.Map;
import java.util.Optional;
import java.util.stream.Stream;
import static java.lang.Runtime.getRuntime;
import static java.lang.System.getenv;
import static java.nio.file.FileVisitOption.FOLLOW_LINKS;
import static java.nio.file.Paths.get;
import static java.util.Comparator.naturalOrder;
import static java.util.stream.Stream.concat;
import static java.util.stream.Stream.of;
/**
* Wrapper around an executable command outside the JVM.
*
* It enables to look for the latest version based on folder names.
*/
public class CommandFinder {
private static final int MAX_FOLDERS_DEPTH = 5;
static String WINDOWS_ENV_PROGRAM_FILES = "ProgramFiles";
static String WINDOWS_ENV_PROGRAM_FILES_X86 = "ProgramFiles(x86)";
static final String PROGRAM_FILES_KEY = "%programFiles%";
private final Runtime runtime;
private final String path;
private final String executable;
private ExecutableCommand cachedExecutableCommand;
private boolean searched = false;
private final String programFilesFolder;
private final String programFilesX86Folder;
/**
* Represents the latest version of the executable found by scanning subfolders of path. The path will have
* %programFiles% replaced by the actual value in the environment variable `ProgramFiles`.
*/
public CommandFinder(final String path, final String executable) {
this(path, executable, getRuntime(), getenv());
}
/**
* Only use this constructor from test code so the environment Map and Runtime can be mocked.
*/
CommandFinder(final String path, final String executable, final Runtime runtime, final Map env) {
this.path = path;
this.executable = executable;
this.runtime = runtime;
programFilesFolder = env.get(WINDOWS_ENV_PROGRAM_FILES);
programFilesX86Folder = env.get(WINDOWS_ENV_PROGRAM_FILES_X86);
}
public Optional executableCommand() {
searchForLatest();
return Optional.ofNullable(cachedExecutableCommand);
}
private void searchForLatest() {
if (!searched) {
searchForExe().ifPresent(s -> cachedExecutableCommand = new ExecutableCommand(s, runtime));
searched = true;
}
}
/**
* Finds the latest version of an installed software.
*
* Sort order is based on folder names, assuming that latest version have a greater version number.
*/
public Optional searchForExe() {
return searchForAllExe().max(naturalOrder());
}
/**
* Finds all versions of an installed software.
*/
public Stream searchForAllExe() {
final Stream programFilesFolders = concat(replaced(programFilesFolder), replaced(programFilesX86Folder));
final Stream possiblePaths = concat(programFilesFolders, notReplaced());
try {
return possiblePaths
.flatMap(this::matchingCommandInPath)
.map(Path::toString);
} catch (final Exception e) {
// can occur when there is a file system loop, see https://bugs.openjdk.java.net/browse/JDK-8039910
e.printStackTrace();
return Stream.empty();
}
}
private Stream matchingCommandInPath(final Path possiblePath) {
try {
return Files.find(
possiblePath,
MAX_FOLDERS_DEPTH,
(p, a) -> p.endsWith(executable),
FOLLOW_LINKS);
} catch (IOException e) {
e.printStackTrace();
return Stream.empty();
}
}
private Stream replaced(final String folder) {
if (folder == null) {
return Stream.empty();
} else {
final Path path = get(this.path.replace(PROGRAM_FILES_KEY, folder));
return path.toFile().isDirectory()
? of(path)
: Stream.empty();
}
}
private Stream notReplaced() {
if (path.contains(PROGRAM_FILES_KEY)) {
return Stream.empty();
} else {
final Path pat = get(path);
return pat.toFile().isDirectory()
? of(pat)
: Stream.empty();
}
}
}