From 1355887e70cda64d7540d62959f9bf46e05fad63 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?J=C3=B6rg=20Hohwiller?= Date: Fri, 20 Dec 2024 19:23:05 +0100 Subject: [PATCH] #898: constructive review and cleanup + fix of existing code --- ...AbstractVersionOrEditionGetCommandlet.java | 119 ++++++++++++++++++ .../ide/commandlet/EditionGetCommandlet.java | 68 ++-------- .../ide/commandlet/VersionGetCommandlet.java | 70 ++--------- .../com/devonfw/tools/ide/io/FileAccess.java | 8 ++ .../devonfw/tools/ide/io/FileAccessImpl.java | 13 ++ .../com/devonfw/tools/ide/os/MacOsHelper.java | 2 +- .../tools/ide/tool/LocalToolCommandlet.java | 56 +++++---- .../tools/ide/tool/ToolCommandlet.java | 2 +- 8 files changed, 192 insertions(+), 146 deletions(-) create mode 100644 cli/src/main/java/com/devonfw/tools/ide/commandlet/AbstractVersionOrEditionGetCommandlet.java diff --git a/cli/src/main/java/com/devonfw/tools/ide/commandlet/AbstractVersionOrEditionGetCommandlet.java b/cli/src/main/java/com/devonfw/tools/ide/commandlet/AbstractVersionOrEditionGetCommandlet.java new file mode 100644 index 000000000..5efed8c7a --- /dev/null +++ b/cli/src/main/java/com/devonfw/tools/ide/commandlet/AbstractVersionOrEditionGetCommandlet.java @@ -0,0 +1,119 @@ +package com.devonfw.tools.ide.commandlet; + +import java.util.Objects; + +import com.devonfw.tools.ide.context.IdeContext; +import com.devonfw.tools.ide.log.IdeLogLevel; +import com.devonfw.tools.ide.log.IdeSubLogger; +import com.devonfw.tools.ide.property.FlagProperty; +import com.devonfw.tools.ide.property.ToolProperty; +import com.devonfw.tools.ide.tool.ToolCommandlet; + +/** + * An internal {@link Commandlet} to get the installed version for a tool. + * + * @see ToolCommandlet#getInstalledVersion() + */ +public abstract class AbstractVersionOrEditionGetCommandlet extends Commandlet { + + /** The tool to get the version of. */ + public final ToolProperty tool; + + /** Flag to get the configured version. */ + public final FlagProperty configured; + + /** Flag to get the installed version. */ + public final FlagProperty installed; + + /** + * The constructor. + * + * @param context the {@link IdeContext}. + */ + public AbstractVersionOrEditionGetCommandlet(IdeContext context) { + + super(context); + addKeyword(getName()); + this.tool = add(new ToolProperty("", true, "tool")); + this.configured = add(new FlagProperty("--configured")); + this.installed = add(new FlagProperty("--installed")); + } + + @Override + public boolean isProcessableOutput() { + + return true; + } + + /** + * @return the property to get (e.g. "version" or "edition"). + */ + protected abstract String getPropertyToGet(); + + /** + * @param commandlet the {@link ToolCommandlet} to get the value from. + * @return the configured value. + * @see ToolCommandlet#getConfiguredVersion() + * @see ToolCommandlet#getConfiguredEdition() + */ + protected abstract Object getConfiguredValue(ToolCommandlet commandlet); + + /** + * @param commandlet the {@link ToolCommandlet} to get the value from. + * @return the installed value or {@code null} if the tool is not installed. + * @see ToolCommandlet#getInstalledVersion() + * @see ToolCommandlet#getInstalledEdition() + */ + protected abstract Object getInstalledValue(ToolCommandlet commandlet); + + @Override + public void run() { + + ToolCommandlet commandlet = this.tool.getValue(); + IdeSubLogger logger = this.context.level(IdeLogLevel.PROCESSABLE); + Object configuredValue = getConfiguredValue(commandlet); + Object installedValue = getInstalledValue(commandlet); + boolean getInstalledValue = this.installed.isTrue(); + boolean getConfiguredValue = this.configured.isTrue(); + if (getInstalledValue == getConfiguredValue) { + if (getInstalledValue) { // both --configured and --installed + logToolInfo(logger, commandlet, configuredValue, installedValue); + } else if (this.context.debug().isEnabled()) { + logToolInfo(logger, commandlet, configuredValue, installedValue); + } else { + if (installedValue == null) { + logger.log(configuredValue.toString()); + } else { + logger.log(installedValue.toString()); + } + } + } else { + if (getInstalledValue) { + if (installedValue == null) { + logToolInfo(logger, commandlet, configuredValue, null); + } else { + logger.log(installedValue.toString()); + } + } else { + logger.log(configuredValue.toString()); + } + } + } + + private void logToolInfo(IdeSubLogger logger, ToolCommandlet commandlet, Object configuredValue, Object installedValue) { + + String property = getPropertyToGet(); + String toolName = commandlet.getName(); + if (installedValue == null) { + logger.log("No installation of tool {} was found.", toolName); + } else { + logger.log("The installed {} for tool {} is {}", property, toolName, installedValue); + } + logger.log("The configured {} for tool {} is {}", property, toolName, configuredValue); + if (!Objects.equals(configuredValue, installedValue)) { + logger.log("To install that {} call the following command:", property); + logger.log("ide install {}", toolName); + } + } + +} diff --git a/cli/src/main/java/com/devonfw/tools/ide/commandlet/EditionGetCommandlet.java b/cli/src/main/java/com/devonfw/tools/ide/commandlet/EditionGetCommandlet.java index c7f063b0b..981e6f662 100644 --- a/cli/src/main/java/com/devonfw/tools/ide/commandlet/EditionGetCommandlet.java +++ b/cli/src/main/java/com/devonfw/tools/ide/commandlet/EditionGetCommandlet.java @@ -1,10 +1,6 @@ package com.devonfw.tools.ide.commandlet; import com.devonfw.tools.ide.context.IdeContext; -import com.devonfw.tools.ide.log.IdeLogLevel; -import com.devonfw.tools.ide.log.IdeSubLogger; -import com.devonfw.tools.ide.property.FlagProperty; -import com.devonfw.tools.ide.property.ToolProperty; import com.devonfw.tools.ide.tool.ToolCommandlet; /** @@ -12,16 +8,7 @@ * * @see ToolCommandlet#getInstalledEdition() */ -public class EditionGetCommandlet extends Commandlet { - - /** The tool to get the edition of. */ - public final ToolProperty tool; - - /** Flag to get the configured version. */ - public final FlagProperty configured; - - /** Flag to get the installed version. */ - public final FlagProperty installed; +public class EditionGetCommandlet extends AbstractVersionOrEditionGetCommandlet { /** * The constructor. @@ -31,10 +18,6 @@ public class EditionGetCommandlet extends Commandlet { public EditionGetCommandlet(IdeContext context) { super(context); - addKeyword(getName()); - this.tool = add(new ToolProperty("", true, "tool")); - this.configured = add(new FlagProperty("--configured")); - this.installed = add(new FlagProperty("--installed")); } @Override @@ -44,55 +27,20 @@ public String getName() { } @Override - public boolean isProcessableOutput() { + protected String getPropertyToGet() { - return true; + return "edition"; } @Override - public void run() { + protected Object getConfiguredValue(ToolCommandlet commandlet) { - ToolCommandlet commandlet = this.tool.getValue(); - String configuredEdition = commandlet.getConfiguredEdition(); - String installedEdition = commandlet.getInstalledEdition(); - IdeSubLogger logger = this.context.level(IdeLogLevel.PROCESSABLE); - if (this.installed.isTrue() && !this.configured.isTrue()) { // get installed edition - if (commandlet.getInstalledVersion() == null) { - // note: getInstalledEdition() will fallback to configured edition and not return null, therefore we use getInstalledVersion() - toolInstallInfo(commandlet.getName(), configuredEdition, null, commandlet); - } else { - logger.log(installedEdition); - } - } else if (!this.installed.isTrue() && this.configured.isTrue()) { // get configured edition - logger.log(configuredEdition); - } else if (this.installed.isTrue() && this.configured.isTrue()) { // get both configured and installed edition - logger.log(configuredEdition); - if (!configuredEdition.equals(installedEdition)) { - if (commandlet.getInstalledVersion() != null) { - logger.log(installedEdition); - } else { - logger.log("No installed edition detected"); - } - } - } else { // get configured or installed depending on if the tool is installed or not - if (commandlet.getInstalledVersion() == null) { - logger.log(configuredEdition); - } else { - logger.log(installedEdition); - } - } + return commandlet.getConfiguredEdition(); } - private void toolInstallInfo(String toolName, String configuredEdition, String installedEdition, ToolCommandlet commandlet) { + @Override + protected Object getInstalledValue(ToolCommandlet commandlet) { - IdeSubLogger logger = this.context.level(IdeLogLevel.PROCESSABLE); - if (installedEdition == null) { - logger.log("No installation of tool {} was found.", commandlet.getName()); - } else { - logger.log("The installed edition for tool {} is {}", commandlet.getName(), installedEdition); - } - logger.log("The configured edition for tool {} is {}", toolName, configuredEdition); - logger.log("To install that edition call the following command:"); - logger.log("ide install {}", toolName); + return commandlet.getInstalledEdition(); } } diff --git a/cli/src/main/java/com/devonfw/tools/ide/commandlet/VersionGetCommandlet.java b/cli/src/main/java/com/devonfw/tools/ide/commandlet/VersionGetCommandlet.java index 6d914073d..8b9ddeea4 100644 --- a/cli/src/main/java/com/devonfw/tools/ide/commandlet/VersionGetCommandlet.java +++ b/cli/src/main/java/com/devonfw/tools/ide/commandlet/VersionGetCommandlet.java @@ -1,28 +1,14 @@ package com.devonfw.tools.ide.commandlet; import com.devonfw.tools.ide.context.IdeContext; -import com.devonfw.tools.ide.log.IdeLogLevel; -import com.devonfw.tools.ide.log.IdeSubLogger; -import com.devonfw.tools.ide.property.FlagProperty; -import com.devonfw.tools.ide.property.ToolProperty; import com.devonfw.tools.ide.tool.ToolCommandlet; -import com.devonfw.tools.ide.version.VersionIdentifier; /** * An internal {@link Commandlet} to get the installed version for a tool. * * @see ToolCommandlet#getInstalledVersion() */ -public class VersionGetCommandlet extends Commandlet { - - /** The tool to get the version of. */ - public final ToolProperty tool; - - /** Flag to get the configured version. */ - public final FlagProperty configured; - - /** Flag to get the installed version. */ - public final FlagProperty installed; +public class VersionGetCommandlet extends AbstractVersionOrEditionGetCommandlet { /** * The constructor. @@ -32,10 +18,6 @@ public class VersionGetCommandlet extends Commandlet { public VersionGetCommandlet(IdeContext context) { super(context); - addKeyword(getName()); - this.tool = add(new ToolProperty("", true, "tool")); - this.configured = add(new FlagProperty("--configured")); - this.installed = add(new FlagProperty("--installed")); } @Override @@ -45,56 +27,20 @@ public String getName() { } @Override - public boolean isProcessableOutput() { + protected String getPropertyToGet() { - return true; + return "version"; } @Override - public void run() { + protected Object getConfiguredValue(ToolCommandlet commandlet) { - ToolCommandlet commandlet = this.tool.getValue(); - VersionIdentifier configuredVersion = commandlet.getConfiguredVersion(); - VersionIdentifier installedVersion = commandlet.getInstalledVersion(); - IdeSubLogger logger = this.context.level(IdeLogLevel.PROCESSABLE); - if (this.installed.isTrue() && !this.configured.isTrue()) {// get installed version - if (installedVersion == null) { - toolInstallInfo(commandlet.getName(), configuredVersion, null, commandlet); - } else { - logger.log(installedVersion.toString()); - } - } else if (!this.installed.isTrue() && this.configured.isTrue()) {// get configured version - logger.log(configuredVersion.toString()); - } else if (this.installed.isTrue() && this.configured.isTrue()) {// get both configured and installed version - logger.log(configuredVersion.toString()); - if (!configuredVersion.matches(installedVersion)) { - if (installedVersion != null) { - logger.log(installedVersion.toString()); - } else { - logger.log("No installed version detected"); - } - } - } else { - if (installedVersion == null) { - logger.log(configuredVersion.toString()); - } else { - logger.log(installedVersion.toString()); - } - } + return commandlet.getConfiguredVersion(); } - private void toolInstallInfo(String toolName, VersionIdentifier configuredVersion, VersionIdentifier installedVersion, ToolCommandlet commandlet) { - - IdeSubLogger logger = this.context.level(IdeLogLevel.PROCESSABLE); - if (installedVersion == null) { - logger.log("No installation of tool {} was found.", commandlet.getName()); - } else { - logger.log("The installed version for tool {} is {}", commandlet.getName(), installedVersion); - } - logger.log("The configured version for tool {} is {}", toolName, configuredVersion); - logger.log("To install that version call the following command:"); - logger.log("ide install {}", toolName); + @Override + protected Object getInstalledValue(ToolCommandlet commandlet) { + return commandlet.getInstalledVersion(); } - } diff --git a/cli/src/main/java/com/devonfw/tools/ide/io/FileAccess.java b/cli/src/main/java/com/devonfw/tools/ide/io/FileAccess.java index d3f46e900..5568bbebd 100644 --- a/cli/src/main/java/com/devonfw/tools/ide/io/FileAccess.java +++ b/cli/src/main/java/com/devonfw/tools/ide/io/FileAccess.java @@ -271,4 +271,12 @@ default void extract(Path archiveFile, Path targetDir, Consumer postExtrac * @param filePath the {@link Path} to the file or folder. */ void touch(Path filePath); + + /** + * @param file the {@link Path} to the file to read. + * @return the content of the specified file (in UTF-8 encoding). + * @see java.nio.file.Files#readString(Path) + */ + String readFileContent(Path file); + } diff --git a/cli/src/main/java/com/devonfw/tools/ide/io/FileAccessImpl.java b/cli/src/main/java/com/devonfw/tools/ide/io/FileAccessImpl.java index 87a3896f3..be18c6fb5 100644 --- a/cli/src/main/java/com/devonfw/tools/ide/io/FileAccessImpl.java +++ b/cli/src/main/java/com/devonfw/tools/ide/io/FileAccessImpl.java @@ -952,4 +952,17 @@ public void touch(Path filePath) { } } } + + @Override + public String readFileContent(Path file) { + + this.context.trace("Reading content of file from {}", file); + try { + String content = Files.readString(file); + this.context.trace("Read content of file {} as {}", file, content); + return content; + } catch (IOException e) { + throw new IllegalStateException("Failed to read file " + file, e); + } + } } diff --git a/cli/src/main/java/com/devonfw/tools/ide/os/MacOsHelper.java b/cli/src/main/java/com/devonfw/tools/ide/os/MacOsHelper.java index 2f2b9b2ea..7f210b590 100644 --- a/cli/src/main/java/com/devonfw/tools/ide/os/MacOsHelper.java +++ b/cli/src/main/java/com/devonfw/tools/ide/os/MacOsHelper.java @@ -94,7 +94,7 @@ public Path findLinkDir(Path rootDir, String tool) { */ public Path findRootToolPath(ToolCommandlet commandlet, IdeContext context) { return context.getSoftwareRepositoryPath().resolve(ToolRepository.ID_DEFAULT).resolve(commandlet.getName()) - .resolve(commandlet.getInstalledEdition().toString()) + .resolve(commandlet.getInstalledEdition()) .resolve(commandlet.getInstalledVersion().toString()); } diff --git a/cli/src/main/java/com/devonfw/tools/ide/tool/LocalToolCommandlet.java b/cli/src/main/java/com/devonfw/tools/ide/tool/LocalToolCommandlet.java index 53f04d5c0..0f0c8dba5 100644 --- a/cli/src/main/java/com/devonfw/tools/ide/tool/LocalToolCommandlet.java +++ b/cli/src/main/java/com/devonfw/tools/ide/tool/LocalToolCommandlet.java @@ -312,20 +312,14 @@ protected VersionIdentifier getInstalledVersion(Path toolPath) { return null; } } - this.context.trace("Reading tool version from {}", toolVersionFile); - try { - String version = Files.readString(toolVersionFile).trim(); - this.context.trace("Read tool version {} from {}", version, toolVersionFile); - return VersionIdentifier.of(version); - } catch (IOException e) { - throw new IllegalStateException("Failed to read file " + toolVersionFile, e); - } + String version = this.context.getFileAccess().readFileContent(toolVersionFile).trim(); + return VersionIdentifier.of(version); } @Override public String getInstalledEdition() { - return getInstalledEdition(this.context.getSoftwarePath().resolve(getName())); + return getInstalledEdition(this.context.getSoftwarePath().resolve(this.tool)); } /** @@ -333,25 +327,43 @@ public String getInstalledEdition() { * to the passed {@link Path path} must be the name of the edition. * @return the installed edition of this tool or {@code null} if not installed. */ - public String getInstalledEdition(Path toolPath) { + private String getInstalledEdition(Path toolPath) { if (!Files.isDirectory(toolPath)) { - this.context.debug("Tool {} not installed in {}", getName(), toolPath); + this.context.debug("Tool {} not installed in {}", this.tool, toolPath); return null; } - try { - String edition = toolPath.toRealPath().getParent().getFileName().toString(); - if (!this.context.getUrls().getSortedEditions(getName()).contains(edition)) { - edition = getConfiguredEdition(); + Path realPath = this.context.getFileAccess().toRealPath(toolPath); + // if the realPath changed, a link has been resolved + if (realPath.equals(toolPath)) { + if (!isIgnoreSoftwareRepo()) { + this.context.warning("Tool {} is not installed via software repository (maybe from devonfw-ide). Please consider reinstalling it.", this.tool); } - return edition; - } catch (IOException e) { - throw new IllegalStateException( - "Couldn't determine the edition of " + getName() + " from the directory structure of its software path " - + toolPath - + ", assuming the name of the parent directory of the real path of the software path to be the edition " - + "of the tool.", e); + // I do not see any reliable way how we could determine the edition of a tool that does not use software repo or that was installed by devonfw-ide + return getConfiguredEdition(); + } + Path toolRepoFolder = context.getSoftwareRepositoryPath().resolve(ToolRepository.ID_DEFAULT).resolve(this.tool); + String edition = getEdition(toolRepoFolder, realPath); + if (!this.context.getUrls().getSortedEditions(this.tool).contains(edition)) { + this.context.warning("Undefined edition {} of tool {}", edition, this.tool); + } + return edition; + } + + private String getEdition(Path toolRepoFolder, Path toolInstallFolder) { + + int toolRepoNameCount = toolRepoFolder.getNameCount(); + int toolInstallNameCount = toolInstallFolder.getNameCount(); + if (toolRepoNameCount < toolInstallNameCount) { + // ensure toolInstallFolder starts with $IDE_ROOT/_ide/software/default/«tool» + for (int i = 0; i < toolRepoNameCount; i++) { + if (!toolRepoFolder.getName(i).toString().equals(toolInstallFolder.getName(i).toString())) { + return null; + } + } + return toolInstallFolder.getName(toolRepoNameCount).toString(); } + return null; } @Override diff --git a/cli/src/main/java/com/devonfw/tools/ide/tool/ToolCommandlet.java b/cli/src/main/java/com/devonfw/tools/ide/tool/ToolCommandlet.java index a20bc0eb9..a2aef5d61 100644 --- a/cli/src/main/java/com/devonfw/tools/ide/tool/ToolCommandlet.java +++ b/cli/src/main/java/com/devonfw/tools/ide/tool/ToolCommandlet.java @@ -67,7 +67,7 @@ protected void initProperties() { * @return the name of the tool (e.g. "java", "mvn", "npm", "node"). */ @Override - public String getName() { + public final String getName() { return this.tool; }