Skip to content

Commit

Permalink
Merge branch 'main' into feature/avoid-double-init
Browse files Browse the repository at this point in the history
  • Loading branch information
hohwille authored Jan 16, 2025
2 parents cd99616 + 93247a4 commit 5b49e04
Show file tree
Hide file tree
Showing 12 changed files with 124 additions and 84 deletions.
2 changes: 2 additions & 0 deletions CHANGELOG.adoc
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ This file documents all notable changes to https://github.com/devonfw/IDEasy[IDE

Release with new features and bugfixes:

* https://github.com/devonfw/IDEasy/issues/915[#915]: custom-tools not working
* https://github.com/devonfw/IDEasy/issues/916[#916]: download is missing status code error handling
* https://github.com/devonfw/IDEasy/issues/757[#757]: Support to allow settings in code repository
* https://github.com/devonfw/IDEasy/issues/826[#826]: Fix git settings check when settings folder is empty
Expand All @@ -31,6 +32,7 @@ Then run the `setup` and all should work fine.

Release with new features and bugfixes:

* https://github.com/devonfw/IDEasy/issues/734[#734]: Improve ProcessResult: get out and err in order
* https://github.com/devonfw/IDEasy/issues/764[#764]: Fix IDEasy in CMD
* https://github.com/devonfw/IDEasy/issues/774[#774]: HTTP proxy support not working properly
* https://github.com/devonfw/IDEasy/issues/792[#792]: Honor new variable IDE_OPTIONS in ide command wrapper
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -262,7 +262,6 @@ public void setCwd(Path userDir, String workspace, Path ideHome) {
this.downloadPath = this.userHome.resolve("Downloads/ide");

this.path = computeSystemPath();
this.customToolRepository = CustomToolRepositoryImpl.of(this);
}

private String getMessageIdeHomeFound() {
Expand Down Expand Up @@ -352,6 +351,9 @@ public ToolRepository getDefaultToolRepository() {
@Override
public CustomToolRepository getCustomToolRepository() {

if (this.customToolRepository == null) {
this.customToolRepository = CustomToolRepositoryImpl.of(this);
}
return this.customToolRepository;
}

Expand Down Expand Up @@ -1130,5 +1132,6 @@ public IdeStartContextImpl getStartContext() {
*/
public void reload() {
this.variables = null;
this.customToolRepository = null;
}
}
11 changes: 11 additions & 0 deletions cli/src/main/java/com/devonfw/tools/ide/process/OutputMessage.java
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
package com.devonfw.tools.ide.process;

/**
* Represent an output message that can be either from stdout or stderr.
*
* @param error A boolean flag that indicates whether the output message is from stdout or stderr
* @param message A string containing the outout message
*/
public record OutputMessage(boolean error, String message) {

}
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.stream.Collectors;

import com.devonfw.tools.ide.cli.CliProcessException;
Expand Down Expand Up @@ -168,17 +169,16 @@ public ProcessResult run(ProcessMode processMode) {

this.processBuilder.command(args);

List<String> out = null;
List<String> err = null;
ConcurrentLinkedQueue<OutputMessage> output = new ConcurrentLinkedQueue<>();

Process process = this.processBuilder.start();

try {
if (processMode == ProcessMode.DEFAULT_CAPTURE) {
CompletableFuture<List<String>> outFut = readInputStream(process.getInputStream());
CompletableFuture<List<String>> errFut = readInputStream(process.getErrorStream());
out = outFut.get();
err = errFut.get();
CompletableFuture<Void> outFut = readInputStream(process.getInputStream(), false, output);
CompletableFuture<Void> errFut = readInputStream(process.getErrorStream(), true, output);
outFut.get();
errFut.get();
}

int exitCode;
Expand All @@ -189,7 +189,8 @@ public ProcessResult run(ProcessMode processMode) {
exitCode = process.waitFor();
}

ProcessResult result = new ProcessResultImpl(this.executable.getFileName().toString(), command, exitCode, out, err);
List<OutputMessage> finalOutput = new ArrayList<>(output);
ProcessResult result = new ProcessResultImpl(this.executable.getFileName().toString(), command, exitCode, finalOutput);

performLogging(result, exitCode, interpreter);

Expand Down Expand Up @@ -218,14 +219,22 @@ public ProcessResult run(ProcessMode processMode) {
* "https://stackoverflow.com/questions/14165517/processbuilder-forwarding-stdout-and-stderr-of-started-processes-without-blocki/57483714#57483714">StackOverflow</a>
*
* @param is {@link InputStream}.
* @param errorStream to identify if the output came from stdout or stderr
* @return {@link CompletableFuture}.
*/
private static CompletableFuture<List<String>> readInputStream(InputStream is) {
private static CompletableFuture<Void> readInputStream(InputStream is, boolean errorStream, ConcurrentLinkedQueue<OutputMessage> outputMessages) {

return CompletableFuture.supplyAsync(() -> {

try (InputStreamReader isr = new InputStreamReader(is); BufferedReader br = new BufferedReader(isr)) {
return br.lines().toList();

String line;
while ((line = br.readLine()) != null) {
OutputMessage outputMessage = new OutputMessage(errorStream, line);
outputMessages.add(outputMessage);
}

return null;
} catch (Throwable e) {
throw new RuntimeException("There was a problem while executing the program", e);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,12 @@ default boolean isSuccessful() {
*/
List<String> getErr();

/**
* @return the {@link List} with {@link OutputMessage} that captured on standard out and standard error lines. Will be {@code null} if not captured but
* redirected.
*/
List<OutputMessage> getOutputMessages();

/**
* Logs output and error messages on the provided log level.
*
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,27 +19,23 @@ public class ProcessResultImpl implements ProcessResult {

private final int exitCode;

private final List<String> out;

private final List<String> err;
private final List<OutputMessage> outputMessages;

/**
* The constructor.
*
* @param executable the {@link #getExecutable() executable}.
* @param command the {@link #getCommand() command}.
* @param exitCode the {@link #getExitCode() exit code}.
* @param out the {@link #getOut() out}.
* @param err the {@link #getErr() err}.
* @param outputMessages {@link #getOutputMessages() output Messages}.
*/
public ProcessResultImpl(String executable, String command, int exitCode, List<String> out, List<String> err) {
public ProcessResultImpl(String executable, String command, int exitCode, List<OutputMessage> outputMessages) {

super();
this.executable = executable;
this.command = command;
this.exitCode = exitCode;
this.out = Objects.requireNonNullElse(out, Collections.emptyList());
this.err = Objects.requireNonNullElse(err, Collections.emptyList());
this.outputMessages = Objects.requireNonNullElse(outputMessages, Collections.emptyList());
}

@Override
Expand All @@ -63,13 +59,20 @@ public int getExitCode() {
@Override
public List<String> getOut() {

return this.out;
return this.outputMessages.stream().filter(outputMessage -> !outputMessage.error()).map(OutputMessage::message).toList();
}

@Override
public List<String> getErr() {

return this.err;
return this.outputMessages.stream().filter(OutputMessage::error).map(OutputMessage::message).toList();
}

@Override
public List<OutputMessage> getOutputMessages() {

return this.outputMessages;

}

@Override
Expand All @@ -80,22 +83,23 @@ public void log(IdeLogLevel level, IdeContext context) {
@Override
public void log(IdeLogLevel outLevel, IdeContext context, IdeLogLevel errorLevel) {

if (!this.out.isEmpty()) {
doLog(outLevel, this.out, context);
}
if (!this.err.isEmpty()) {
doLog(errorLevel, this.err, context);
if (!this.outputMessages.isEmpty()) {
for (OutputMessage outputMessage : this.outputMessages) {
if (outputMessage.error()) {
doLog(errorLevel, outputMessage.message(), context);
} else {
doLog(outLevel, outputMessage.message(), context);
}
}
}
}

private void doLog(IdeLogLevel level, List<String> lines, IdeContext context) {
for (String line : lines) {
// remove !MESSAGE from log message
if (line.startsWith("!MESSAGE ")) {
line = line.substring(9);
}
context.level(level).log(line);
private void doLog(IdeLogLevel level, String message, IdeContext context) {
// remove !MESSAGE from log message
if (message.startsWith("!MESSAGE ")) {
message = message.substring(9);
}
context.level(level).log(message);
}

@Override
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,9 @@
package com.devonfw.tools.ide.tool;

import com.devonfw.tools.ide.context.IdeContext;
import com.devonfw.tools.ide.process.EnvironmentContext;
import com.devonfw.tools.ide.repo.CustomToolMetadata;
import com.devonfw.tools.ide.version.GenericVersionRange;
import com.devonfw.tools.ide.version.VersionIdentifier;

/**
Expand All @@ -11,6 +13,12 @@ public class CustomToolCommandlet extends LocalToolCommandlet {

private CustomToolMetadata customTool;

/**
* The constructor.
*
* @param context the {@link IdeContext}.
* @param customTool the {@link CustomToolMetadata} to handle (e.g. install or uninstall).
*/
public CustomToolCommandlet(IdeContext context, CustomToolMetadata customTool) {

super(context, customTool.getTool(), null);
Expand All @@ -29,4 +37,9 @@ public String getConfiguredEdition() {
return this.customTool.getEdition();
}

@Override
public ToolInstallation installTool(GenericVersionRange version, EnvironmentContext environmentContext, String edition) {

return installTool(version, environmentContext, edition, this.context.getCustomToolRepository());
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@
import java.util.ArrayList;
import java.util.List;

import com.devonfw.tools.ide.process.OutputMessage;
import com.devonfw.tools.ide.process.ProcessContext;
import com.devonfw.tools.ide.process.ProcessErrorHandling;
import com.devonfw.tools.ide.process.ProcessMode;
Expand All @@ -21,59 +22,25 @@ public class ProcessContextGitMock implements ProcessContext {

private final List<String> arguments;

private final List<String> errors;

private final List<String> outs;

private final LocalDateTime now;

private int exitCode;

private final Path directory;

private List<OutputMessage> outputMessages;

/**
* @param directory the {@link Path} to the git repository.
*/
public ProcessContextGitMock(Path directory) {

this.arguments = new ArrayList<>();
this.errors = new ArrayList<>();
this.outs = new ArrayList<>();
this.exitCode = ProcessResult.SUCCESS;
this.directory = directory;
this.now = LocalDateTime.now();
this.outputMessages = new ArrayList<OutputMessage>();
}

/**
* @return the mocked {@link ProcessResult#getExitCode() exit code}.
*/
public int getExitCode() {

return this.exitCode;
}

/**
* @param exitCode the {@link #getExitCode() exit code}.
*/
public void setExitCode(int exitCode) {

this.exitCode = exitCode;
}

/**
* @return the {@link List} of mocked error messages.
*/
public List<String> getErrors() {

return errors;
}

/**
* @return the {@link List} of mocked out messages.
*/
public List<String> getOuts() {

return outs;
public void addOutputMessage(OutputMessage message) {
this.outputMessages.add(message);
}

@Override
Expand Down Expand Up @@ -121,6 +88,7 @@ public ProcessContext withPathEntry(Path path) {
@Override
public ProcessResult run(ProcessMode processMode) {

int exitCode = ProcessResult.SUCCESS;
StringBuilder command = new StringBuilder("git");
for (String arg : this.arguments) {
command.append(' ');
Expand All @@ -131,16 +99,15 @@ public ProcessResult run(ProcessMode processMode) {
if (this.arguments.contains("clean")) {
try {
Files.deleteIfExists(this.directory.resolve("new-folder"));
this.exitCode = 0;
} catch (IOException e) {
throw new RuntimeException(e);
}
}
// part of git cleanup checks if a new directory 'new-folder' exists
if (this.arguments.contains("ls-files")) {
if (Files.exists(this.directory.resolve("new-folder"))) {
this.outs.add("new-folder");
this.exitCode = 0;
OutputMessage outputMessage = new OutputMessage(false, "new-folder");
this.outputMessages.add(outputMessage);
}
}
if (this.arguments.contains("clone")) {
Expand All @@ -149,14 +116,13 @@ public ProcessResult run(ProcessMode processMode) {
Path newFile = Files.createFile(gitFolderPath.resolve("url"));
// 3rd argument = repository Url
Files.writeString(newFile, this.arguments.get(2));
this.exitCode = 0;
} catch (IOException e) {
throw new RuntimeException(e);
}
}
// always consider that files were changed
if (this.arguments.contains("diff-index")) {
this.exitCode = 1;
exitCode = 1;
}
// changes file back to initial state (uses reference file in .git folder)
if (this.arguments.contains("reset")) {
Expand All @@ -165,7 +131,7 @@ public ProcessResult run(ProcessMode processMode) {
Files.copy(gitFolderPath.resolve("objects").resolve("referenceFile"), this.directory.resolve("trackedFile"),
StandardCopyOption.REPLACE_EXISTING);
}
this.exitCode = 0;
exitCode = ProcessResult.SUCCESS;
} catch (IOException e) {
throw new RuntimeException(e);
}
Expand All @@ -175,13 +141,13 @@ public ProcessResult run(ProcessMode processMode) {
Files.createDirectories(gitFolderPath);
Path newFile = Files.createFile(gitFolderPath.resolve("update"));
Files.writeString(newFile, this.now.toString());
this.exitCode = 0;
exitCode = ProcessResult.SUCCESS;
} catch (IOException e) {
throw new RuntimeException(e);
}
}
this.arguments.clear();
return new ProcessResultImpl("git", command.toString(), this.exitCode, this.outs, this.errors);
return new ProcessResultImpl("git", command.toString(), exitCode, this.outputMessages);
}

}
Loading

0 comments on commit 5b49e04

Please sign in to comment.