• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

devonfw / IDEasy / 25784992341

13 May 2026 07:30AM UTC coverage: 70.671% (-0.007%) from 70.678%
25784992341

Pull #1891

github

web-flow
Merge a8af714cc into b050b96e2
Pull Request #1891: #1880: Allow reset of installed plugins when launching IDE in force mode

4427 of 6920 branches covered (63.97%)

Branch coverage included in aggregate %.

11409 of 15488 relevant lines covered (73.66%)

3.12 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

65.22
cli/src/main/java/com/devonfw/tools/ide/context/AbstractIdeContext.java
1
package com.devonfw.tools.ide.context;
2

3
import static com.devonfw.tools.ide.variable.IdeVariables.IDE_MIN_VERSION;
4

5
import java.io.ByteArrayInputStream;
6
import java.io.ByteArrayOutputStream;
7
import java.io.IOException;
8
import java.nio.file.Files;
9
import java.nio.file.Path;
10
import java.time.LocalDateTime;
11
import java.util.ArrayList;
12
import java.util.HashMap;
13
import java.util.Iterator;
14
import java.util.List;
15
import java.util.Locale;
16
import java.util.Map;
17
import java.util.Map.Entry;
18
import java.util.Objects;
19
import java.util.Properties;
20
import java.util.function.Predicate;
21
import java.util.logging.FileHandler;
22
import java.util.logging.LogManager;
23
import java.util.logging.SimpleFormatter;
24

25
import org.slf4j.Logger;
26
import org.slf4j.LoggerFactory;
27

28
import com.devonfw.tools.ide.cli.CliAbortException;
29
import com.devonfw.tools.ide.cli.CliArgument;
30
import com.devonfw.tools.ide.cli.CliArguments;
31
import com.devonfw.tools.ide.cli.CliException;
32
import com.devonfw.tools.ide.commandlet.Commandlet;
33
import com.devonfw.tools.ide.commandlet.CommandletManager;
34
import com.devonfw.tools.ide.commandlet.CommandletManagerImpl;
35
import com.devonfw.tools.ide.commandlet.ContextCommandlet;
36
import com.devonfw.tools.ide.commandlet.EnvironmentCommandlet;
37
import com.devonfw.tools.ide.commandlet.UpdateCommandlet;
38
import com.devonfw.tools.ide.common.SystemPath;
39
import com.devonfw.tools.ide.completion.CompletionCandidate;
40
import com.devonfw.tools.ide.completion.CompletionCandidateCollector;
41
import com.devonfw.tools.ide.completion.CompletionCandidateCollectorDefault;
42
import com.devonfw.tools.ide.environment.AbstractEnvironmentVariables;
43
import com.devonfw.tools.ide.environment.EnvironmentVariables;
44
import com.devonfw.tools.ide.environment.EnvironmentVariablesType;
45
import com.devonfw.tools.ide.environment.IdeSystem;
46
import com.devonfw.tools.ide.environment.IdeSystemImpl;
47
import com.devonfw.tools.ide.git.GitContext;
48
import com.devonfw.tools.ide.git.GitContextImpl;
49
import com.devonfw.tools.ide.git.GitUrl;
50
import com.devonfw.tools.ide.io.FileAccess;
51
import com.devonfw.tools.ide.io.FileAccessImpl;
52
import com.devonfw.tools.ide.log.IdeLogArgFormatter;
53
import com.devonfw.tools.ide.log.IdeLogLevel;
54
import com.devonfw.tools.ide.log.IdeLogListener;
55
import com.devonfw.tools.ide.log.JulConsoleHandler;
56
import com.devonfw.tools.ide.merge.DirectoryMerger;
57
import com.devonfw.tools.ide.migration.IdeMigrator;
58
import com.devonfw.tools.ide.network.NetworkStatus;
59
import com.devonfw.tools.ide.network.NetworkStatusImpl;
60
import com.devonfw.tools.ide.os.SystemInfo;
61
import com.devonfw.tools.ide.os.SystemInfoImpl;
62
import com.devonfw.tools.ide.os.WindowsHelper;
63
import com.devonfw.tools.ide.os.WindowsHelperImpl;
64
import com.devonfw.tools.ide.os.WindowsPathSyntax;
65
import com.devonfw.tools.ide.process.ProcessContext;
66
import com.devonfw.tools.ide.process.ProcessContextImpl;
67
import com.devonfw.tools.ide.process.ProcessResult;
68
import com.devonfw.tools.ide.property.Property;
69
import com.devonfw.tools.ide.step.Step;
70
import com.devonfw.tools.ide.step.StepImpl;
71
import com.devonfw.tools.ide.tool.custom.CustomToolRepository;
72
import com.devonfw.tools.ide.tool.custom.CustomToolRepositoryImpl;
73
import com.devonfw.tools.ide.tool.mvn.MvnRepository;
74
import com.devonfw.tools.ide.tool.npm.NpmRepository;
75
import com.devonfw.tools.ide.tool.pip.PipRepository;
76
import com.devonfw.tools.ide.tool.repository.DefaultToolRepository;
77
import com.devonfw.tools.ide.tool.repository.ToolRepository;
78
import com.devonfw.tools.ide.url.model.UrlMetadata;
79
import com.devonfw.tools.ide.util.DateTimeUtil;
80
import com.devonfw.tools.ide.util.PrivacyUtil;
81
import com.devonfw.tools.ide.validation.ValidationResult;
82
import com.devonfw.tools.ide.validation.ValidationResultValid;
83
import com.devonfw.tools.ide.validation.ValidationState;
84
import com.devonfw.tools.ide.variable.IdeVariables;
85
import com.devonfw.tools.ide.version.IdeVersion;
86
import com.devonfw.tools.ide.version.VersionIdentifier;
87

88
/**
89
 * Abstract base implementation of {@link IdeContext}.
90
 */
91
public abstract class AbstractIdeContext implements IdeContext, IdeLogArgFormatter {
92

93
  static final Logger LOG = LoggerFactory.getLogger(AbstractIdeContext.class);
3✔
94

95
  /** The default shell bash (Bourne Again SHell). */
96
  public static final String BASH = "bash";
97

98
  private static final GitUrl IDE_URLS_GIT = new GitUrl("https://github.com/devonfw/ide-urls.git", null);
7✔
99

100
  private static final String LICENSE_URL = "https://github.com/devonfw/IDEasy/blob/main/documentation/LICENSE.adoc";
101

102
  private static final String DEFAULT_WINDOWS_GIT_PATH = "C:\\Program Files\\Git\\bin\\bash.exe";
103

104
  private static final String OPTION_DETAILS_START = "([";
105

106
  private final IdeStartContextImpl startContext;
107

108
  private Path ideHome;
109

110
  private final Path ideRoot;
111

112
  private Path confPath;
113

114
  protected Path settingsPath;
115

116
  private Path settingsCommitIdPath;
117

118
  protected Path pluginsPath;
119

120
  private Path workspacePath;
121

122
  private Path workspacesBasePath;
123

124
  private String workspaceName;
125

126
  private Path cwd;
127

128
  private Path downloadPath;
129

130
  private Path userHome;
131

132
  private Path userHomeIde;
133

134
  private SystemPath path;
135

136
  private WindowsPathSyntax pathSyntax;
137

138
  private final SystemInfo systemInfo;
139

140
  private EnvironmentVariables variables;
141

142
  private final FileAccess fileAccess;
143

144
  protected CommandletManager commandletManager;
145

146
  protected ToolRepository defaultToolRepository;
147

148
  private CustomToolRepository customToolRepository;
149

150
  private MvnRepository mvnRepository;
151

152
  private NpmRepository npmRepository;
153

154
  private PipRepository pipRepository;
155

156
  private DirectoryMerger workspaceMerger;
157

158
  protected UrlMetadata urlMetadata;
159

160
  protected Path defaultExecutionDirectory;
161

162
  private StepImpl currentStep;
163

164
  private NetworkStatus networkStatus;
165

166
  protected IdeSystem system;
167

168
  private WindowsHelper windowsHelper;
169

170
  private final Map<String, String> privacyMap;
171

172
  private Path bash;
173

174
  private boolean julConfigured;
175

176
  private Path logfile;
177

178
  /**
179
   * The constructor.
180
   *
181
   * @param startContext the {@link IdeStartContextImpl}.
182
   * @param workingDirectory the optional {@link Path} to current working directory.
183
   */
184
  public AbstractIdeContext(IdeStartContextImpl startContext, Path workingDirectory) {
185

186
    super();
2✔
187
    this.startContext = startContext;
3✔
188
    this.startContext.setArgFormatter(this);
4✔
189
    this.privacyMap = new HashMap<>();
5✔
190
    this.systemInfo = SystemInfoImpl.INSTANCE;
3✔
191
    if (isTest()) {
3!
192
      configureJavaUtilLogging(null);
3✔
193
    }
194
    this.commandletManager = new CommandletManagerImpl(this);
6✔
195
    this.fileAccess = new FileAccessImpl(this);
6✔
196
    String userHomeProperty = getSystem().getProperty("user.home");
5✔
197
    if (userHomeProperty != null) {
2!
198
      this.userHome = Path.of(userHomeProperty);
×
199
    }
200
    if (workingDirectory == null) {
2!
201
      workingDirectory = Path.of(System.getProperty("user.dir"));
×
202
    }
203
    workingDirectory = workingDirectory.toAbsolutePath();
3✔
204
    if (Files.isDirectory(workingDirectory)) {
5✔
205
      workingDirectory = this.fileAccess.toCanonicalPath(workingDirectory);
6✔
206
    } else {
207
      LOG.warn("Current working directory does not exist: {}", workingDirectory);
4✔
208
    }
209
    this.cwd = workingDirectory;
3✔
210
    // detect IDE_HOME and WORKSPACE
211
    String workspace = null;
2✔
212
    Path ideHomeDir = null;
2✔
213
    IdeHomeAndWorkspace ideHomeAndWorkspace = findIdeHome(workingDirectory);
4✔
214
    if (ideHomeAndWorkspace != null) {
2!
215
      ideHomeDir = ideHomeAndWorkspace.home();
3✔
216
      workspace = ideHomeAndWorkspace.workspace();
3✔
217
    }
218

219
    // detection completed, initializing variables
220
    this.ideRoot = findIdeRoot(ideHomeDir);
5✔
221

222
    setCwd(workingDirectory, workspace, ideHomeDir);
5✔
223

224
    if (this.ideRoot != null) {
3✔
225
      Path tempDownloadPath = getTempDownloadPath();
3✔
226
      if (Files.isDirectory(tempDownloadPath)) {
6✔
227
        // TODO delete all files older than 1 day here...
228
      } else {
229
        this.fileAccess.mkdirs(tempDownloadPath);
4✔
230
      }
231
    }
232
    this.defaultToolRepository = new DefaultToolRepository(this);
6✔
233
  }
1✔
234

235
  /**
236
   * Searches for the IDE home directory by traversing up the directory tree from the given working directory. This method can be overridden in test contexts to
237
   * add additional validation or boundary checks.
238
   *
239
   * @param workingDirectory the starting directory for the search.
240
   * @return an instance of {@link IdeHomeAndWorkspace} where the IDE_HOME was found or {@code null} if not found.
241
   */
242
  protected IdeHomeAndWorkspace findIdeHome(Path workingDirectory) {
243

244
    Path currentDir = workingDirectory;
2✔
245
    String name1 = "";
2✔
246
    String name2 = "";
2✔
247
    String workspace = WORKSPACE_MAIN;
2✔
248
    Path ideRootPath = getIdeRootPathFromEnv(false);
4✔
249

250
    while (currentDir != null) {
2✔
251
      LOG.trace("Looking for IDE_HOME in {}", currentDir);
4✔
252
      if (isIdeHome(currentDir)) {
4✔
253
        if (FOLDER_WORKSPACES.equals(name1) && !name2.isEmpty()) {
7✔
254
          workspace = name2;
3✔
255
        }
256
        break;
257
      }
258
      name2 = name1;
2✔
259
      int nameCount = currentDir.getNameCount();
3✔
260
      if (nameCount >= 1) {
3✔
261
        name1 = currentDir.getName(nameCount - 1).toString();
7✔
262
      }
263
      currentDir = currentDir.getParent();
3✔
264
      if ((ideRootPath != null) && (ideRootPath.equals(currentDir))) {
2!
265
        // prevent that during tests we traverse to the real IDE project of IDEasy developer
266
        currentDir = null;
×
267
      }
268
    }
1✔
269

270
    return new IdeHomeAndWorkspace(currentDir, workspace);
6✔
271
  }
272

273
  /**
274
   * @return a new {@link MvnRepository}
275
   */
276
  protected MvnRepository createMvnRepository() {
277
    return new MvnRepository(this);
5✔
278
  }
279

280
  /**
281
   * @return a new {@link NpmRepository}
282
   */
283
  protected NpmRepository createNpmRepository() {
284
    return new NpmRepository(this);
×
285
  }
286

287
  /**
288
   * @return a new {@link PipRepository}
289
   */
290
  protected PipRepository createPipRepository() {
291
    return new PipRepository(this);
×
292
  }
293

294
  private Path findIdeRoot(Path ideHomePath) {
295

296
    Path ideRootPath = null;
2✔
297
    if (ideHomePath != null) {
2✔
298
      Path ideRootPathFromEnv = getIdeRootPathFromEnv(true);
4✔
299
      ideRootPath = ideHomePath.getParent();
3✔
300
      if ((ideRootPathFromEnv != null) && !ideRootPath.toString().equals(ideRootPathFromEnv.toString())) {
2!
301
        LOG.warn(
×
302
            "Variable IDE_ROOT is set to '{}' but for your project '{}' the path '{}' would have been expected.\n"
303
                + "Please check your 'user.dir' or working directory setting and make sure that it matches your IDE_ROOT variable.",
304
            ideRootPathFromEnv,
305
            ideHomePath.getFileName(), ideRootPath);
×
306
      }
307
    } else if (!isTest()) {
4!
308
      ideRootPath = getIdeRootPathFromEnv(true);
×
309
    }
310
    return ideRootPath;
2✔
311
  }
312

313
  /**
314
   * @return the {@link #getIdeRoot() IDE_ROOT} from the system environment.
315
   */
316
  protected Path getIdeRootPathFromEnv(boolean withSanityCheck) {
317

318
    String root = getSystem().getEnv(IdeVariables.IDE_ROOT.getName());
×
319
    if (root != null) {
×
320
      Path rootPath = Path.of(root);
×
321
      if (Files.isDirectory(rootPath)) {
×
322
        Path absoluteRootPath = getFileAccess().toCanonicalPath(rootPath);
×
323
        if (withSanityCheck) {
×
324
          int nameCount = rootPath.getNameCount();
×
325
          int absoluteNameCount = absoluteRootPath.getNameCount();
×
326
          int delta = absoluteNameCount - nameCount;
×
327
          if (delta >= 0) {
×
328
            for (int nameIndex = 0; nameIndex < nameCount; nameIndex++) {
×
329
              String rootName = rootPath.getName(nameIndex).toString();
×
330
              String absoluteRootName = absoluteRootPath.getName(nameIndex + delta).toString();
×
331
              if (!rootName.equals(absoluteRootName)) {
×
332
                LOG.warn("IDE_ROOT is set to {} but was expanded to absolute path {} and does not match for segment {} and {} - fix your IDEasy installation!",
×
333
                    rootPath, absoluteRootPath, rootName, absoluteRootName);
334
                break;
×
335
              }
336
            }
337
          } else {
338
            LOG.warn("IDE_ROOT is set to {} but was expanded to a shorter absolute path {}", rootPath,
×
339
                absoluteRootPath);
340
          }
341
        }
342
        return absoluteRootPath;
×
343
      } else if (withSanityCheck) {
×
344
        LOG.warn("IDE_ROOT is set to {} that is not an existing directory - fix your IDEasy installation!", rootPath);
×
345
      }
346
    }
347
    return null;
×
348
  }
349

350
  @Override
351
  public void setCwd(Path userDir, String workspace, Path ideHome) {
352

353
    this.cwd = userDir;
3✔
354
    this.workspaceName = workspace;
3✔
355
    this.ideHome = ideHome;
3✔
356
    if (ideHome == null) {
2✔
357
      this.workspacesBasePath = null;
3✔
358
      this.workspacePath = null;
3✔
359
      this.confPath = null;
3✔
360
      this.settingsPath = null;
3✔
361
      this.pluginsPath = null;
4✔
362
    } else {
363
      this.workspacesBasePath = this.ideHome.resolve(FOLDER_WORKSPACES);
6✔
364
      this.workspacePath = this.workspacesBasePath.resolve(this.workspaceName);
7✔
365
      this.confPath = this.ideHome.resolve(FOLDER_CONF);
6✔
366
      this.settingsPath = this.ideHome.resolve(FOLDER_SETTINGS);
6✔
367
      this.settingsCommitIdPath = this.ideHome.resolve(IdeContext.SETTINGS_COMMIT_ID);
6✔
368
      this.pluginsPath = this.ideHome.resolve(FOLDER_PLUGINS);
6✔
369
    }
370
    if (isTest()) {
3!
371
      // only for testing...
372
      if (this.ideHome == null) {
3✔
373
        this.userHome = Path.of("/non-existing-user-home-for-testing");
7✔
374
      } else {
375
        this.userHome = this.ideHome.resolve("home");
6✔
376
      }
377
    }
378
    this.userHomeIde = this.userHome.resolve(FOLDER_DOT_IDE);
6✔
379
    this.downloadPath = this.userHome.resolve("Downloads/ide");
6✔
380
    resetPrivacyMap();
2✔
381
    this.path = computeSystemPath();
4✔
382
  }
1✔
383

384
  private String getMessageIdeHomeFound() {
385

386
    String wks = this.workspaceName;
3✔
387
    if (isPrivacyMode() && !WORKSPACE_MAIN.equals(wks)) {
3!
388
      wks = "*".repeat(wks.length());
×
389
    }
390
    return "IDE environment variables have been set for " + formatArgument(this.ideHome) + " in workspace " + wks;
7✔
391
  }
392

393
  private String getMessageNotInsideIdeProject() {
394

395
    return "You are not inside an IDE project: " + formatArgument(this.cwd);
6✔
396
  }
397

398
  private String getMessageIdeRootNotFound() {
399

400
    String root = getSystem().getEnv("IDE_ROOT");
5✔
401
    if (root == null) {
2!
402
      return "The environment variable IDE_ROOT is undefined. Please reinstall IDEasy or manually repair IDE_ROOT variable.";
2✔
403
    } else {
404
      return "The environment variable IDE_ROOT is pointing to an invalid path " + formatArgument(root)
×
405
          + ". Please reinstall IDEasy or manually repair IDE_ROOT variable.";
406
    }
407
  }
408

409
  /**
410
   * @return {@code true} if this is a test context for JUnits, {@code false} otherwise.
411
   */
412
  public boolean isTest() {
413

414
    return false;
×
415
  }
416

417
  protected SystemPath computeSystemPath() {
418

419
    return new SystemPath(this);
×
420
  }
421

422
  /**
423
   * Checks if the given directory is a valid IDE home by verifying it contains both 'workspaces' and 'settings' directories.
424
   *
425
   * @param dir the directory to check.
426
   * @return {@code true} if the directory is a valid IDE home, {@code false} otherwise.
427
   */
428
  protected boolean isIdeHome(Path dir) {
429

430
    if (!Files.isDirectory(dir.resolve("workspaces"))) {
7✔
431
      return false;
2✔
432
    } else if (!Files.isDirectory(dir.resolve("settings"))) {
7!
433
      return false;
×
434
    }
435
    return true;
2✔
436
  }
437

438
  private EnvironmentVariables createVariables() {
439

440
    AbstractEnvironmentVariables system = createSystemVariables();
3✔
441
    AbstractEnvironmentVariables user = system.extend(this.userHomeIde, EnvironmentVariablesType.USER);
6✔
442
    AbstractEnvironmentVariables settings = user.extend(this.settingsPath, EnvironmentVariablesType.SETTINGS);
6✔
443
    AbstractEnvironmentVariables workspace = settings.extend(this.workspacePath, EnvironmentVariablesType.WORKSPACE);
6✔
444
    AbstractEnvironmentVariables conf = workspace.extend(this.confPath, EnvironmentVariablesType.CONF);
6✔
445
    return conf.resolved();
3✔
446
  }
447

448
  protected AbstractEnvironmentVariables createSystemVariables() {
449

450
    return EnvironmentVariables.ofSystem(this);
3✔
451
  }
452

453
  @Override
454
  public SystemInfo getSystemInfo() {
455

456
    return this.systemInfo;
3✔
457
  }
458

459
  @Override
460
  public FileAccess getFileAccess() {
461

462
    return this.fileAccess;
3✔
463
  }
464

465
  @Override
466
  public CommandletManager getCommandletManager() {
467

468
    return this.commandletManager;
3✔
469
  }
470

471
  @Override
472
  public ToolRepository getDefaultToolRepository() {
473

474
    return this.defaultToolRepository;
3✔
475
  }
476

477
  @Override
478
  public MvnRepository getMvnRepository() {
479
    if (this.mvnRepository == null) {
3✔
480
      this.mvnRepository = createMvnRepository();
4✔
481
    }
482
    return this.mvnRepository;
3✔
483
  }
484

485
  @Override
486
  public NpmRepository getNpmRepository() {
487
    if (this.npmRepository == null) {
3✔
488
      this.npmRepository = createNpmRepository();
4✔
489
    }
490
    return this.npmRepository;
3✔
491
  }
492

493
  @Override
494
  public PipRepository getPipRepository() {
495
    if (this.pipRepository == null) {
3✔
496
      this.pipRepository = createPipRepository();
4✔
497
    }
498
    return this.pipRepository;
3✔
499
  }
500

501
  @Override
502
  public CustomToolRepository getCustomToolRepository() {
503

504
    if (this.customToolRepository == null) {
3!
505
      this.customToolRepository = CustomToolRepositoryImpl.of(this);
4✔
506
    }
507
    return this.customToolRepository;
3✔
508
  }
509

510
  @Override
511
  public Path getIdeHome() {
512

513
    return this.ideHome;
3✔
514
  }
515

516
  @Override
517
  public String getProjectName() {
518

519
    if (this.ideHome != null) {
3!
520
      return this.ideHome.getFileName().toString();
5✔
521
    }
522
    return "";
×
523
  }
524

525
  @Override
526
  public VersionIdentifier getProjectVersion() {
527

528
    if (this.ideHome != null) {
3!
529
      Path versionFile = this.ideHome.resolve(IdeContext.FILE_SOFTWARE_VERSION);
5✔
530
      if (Files.exists(versionFile)) {
5✔
531
        String version = this.fileAccess.readFileContent(versionFile).trim();
6✔
532
        return VersionIdentifier.of(version);
3✔
533
      }
534
    }
535
    return IdeMigrator.START_VERSION;
2✔
536
  }
537

538
  @Override
539
  public void setProjectVersion(VersionIdentifier version) {
540

541
    if (this.ideHome == null) {
3!
542
      throw new IllegalStateException("IDE_HOME not available!");
×
543
    }
544
    Objects.requireNonNull(version);
3✔
545
    Path versionFile = this.ideHome.resolve(IdeContext.FILE_SOFTWARE_VERSION);
5✔
546
    this.fileAccess.writeFileContent(version.toString(), versionFile);
6✔
547
  }
1✔
548

549
  @Override
550
  public Path getIdeRoot() {
551

552
    return this.ideRoot;
3✔
553
  }
554

555
  @Override
556
  public Path getIdePath() {
557

558
    Path myIdeRoot = getIdeRoot();
3✔
559
    if (myIdeRoot == null) {
2✔
560
      return null;
2✔
561
    }
562
    return myIdeRoot.resolve(FOLDER_UNDERSCORE_IDE);
4✔
563
  }
564

565
  @Override
566
  public Path getCwd() {
567

568
    return this.cwd;
3✔
569
  }
570

571
  @Override
572
  public Path getTempPath() {
573

574
    Path idePath = getIdePath();
3✔
575
    if (idePath == null) {
2!
576
      return null;
×
577
    }
578
    return idePath.resolve("tmp");
4✔
579
  }
580

581
  @Override
582
  public Path getTempDownloadPath() {
583

584
    Path tmp = getTempPath();
3✔
585
    if (tmp == null) {
2!
586
      return null;
×
587
    }
588
    return tmp.resolve(FOLDER_DOWNLOADS);
4✔
589
  }
590

591
  @Override
592
  public Path getUserHome() {
593

594
    return this.userHome;
3✔
595
  }
596

597
  /**
598
   * This method should only be used for tests to mock user home.
599
   *
600
   * @param userHome the new value of {@link #getUserHome()}.
601
   */
602
  protected void setUserHome(Path userHome) {
603

604
    this.userHome = userHome;
3✔
605
    this.userHomeIde = userHome.resolve(FOLDER_DOT_IDE);
5✔
606
    this.downloadPath = userHome.resolve("Downloads/ide");
5✔
607
    this.variables = null;
3✔
608
    resetPrivacyMap();
2✔
609
  }
1✔
610

611
  @Override
612
  public Path getUserHomeIde() {
613

614
    return this.userHomeIde;
3✔
615
  }
616

617
  @Override
618
  public Path getSettingsPath() {
619

620
    return this.settingsPath;
3✔
621
  }
622

623
  @Override
624
  public Path getSettingsGitRepository() {
625

626
    Path settingsPath = getSettingsPath();
3✔
627
    // check whether the settings path has a .git folder only if its not a symbolic link or junction
628
    if ((settingsPath != null) && !Files.exists(settingsPath.resolve(".git")) && !isSettingsRepositorySymlinkOrJunction()) {
12!
629
      LOG.error("Settings repository exists but is not a git repository.");
3✔
630
      return null;
2✔
631
    }
632
    return settingsPath;
2✔
633
  }
634

635
  @Override
636
  public boolean isSettingsRepositorySymlinkOrJunction() {
637

638
    Path settingsPath = getSettingsPath();
3✔
639
    if (settingsPath == null) {
2!
640
      return false;
×
641
    }
642
    return Files.isSymbolicLink(settingsPath) || getFileAccess().isJunction(settingsPath);
10!
643
  }
644

645
  @Override
646
  public Path getSettingsCommitIdPath() {
647

648
    return this.settingsCommitIdPath;
3✔
649
  }
650

651
  @Override
652
  public Path getConfPath() {
653

654
    return this.confPath;
3✔
655
  }
656

657
  @Override
658
  public Path getSoftwarePath() {
659

660
    if (this.ideHome == null) {
3✔
661
      return null;
2✔
662
    }
663
    return this.ideHome.resolve(FOLDER_SOFTWARE);
5✔
664
  }
665

666
  @Override
667
  public Path getSoftwareExtraPath() {
668

669
    Path softwarePath = getSoftwarePath();
3✔
670
    if (softwarePath == null) {
2!
671
      return null;
×
672
    }
673
    return softwarePath.resolve(FOLDER_EXTRA);
4✔
674
  }
675

676
  @Override
677
  public Path getSoftwareRepositoryPath() {
678

679
    Path idePath = getIdePath();
3✔
680
    if (idePath == null) {
2!
681
      return null;
×
682
    }
683
    return idePath.resolve(FOLDER_SOFTWARE);
4✔
684
  }
685

686
  @Override
687
  public Path getPluginsPath() {
688

689
    return this.pluginsPath;
3✔
690
  }
691

692
  @Override
693
  public String getWorkspaceName() {
694

695
    return this.workspaceName;
3✔
696
  }
697

698
  @Override
699
  public Path getWorkspacesBasePath() {
700

701
    return this.workspacesBasePath;
3✔
702
  }
703

704
  @Override
705
  public Path getWorkspacePath() {
706

707
    return this.workspacePath;
3✔
708
  }
709

710
  @Override
711
  public Path getWorkspacePath(String workspace) {
712

713
    if (this.workspacesBasePath == null) {
3!
714
      throw new IllegalStateException("Failed to access workspace " + workspace + " without IDE_HOME in " + this.cwd);
×
715
    }
716
    return this.workspacesBasePath.resolve(workspace);
5✔
717
  }
718

719
  @Override
720
  public Path getDownloadPath() {
721

722
    return this.downloadPath;
3✔
723
  }
724

725
  @Override
726
  public Path getUrlsPath() {
727

728
    Path idePath = getIdePath();
3✔
729
    if (idePath == null) {
2!
730
      return null;
×
731
    }
732
    return idePath.resolve(FOLDER_URLS);
4✔
733
  }
734

735
  @Override
736
  public Path getToolRepositoryPath() {
737

738
    Path idePath = getIdePath();
3✔
739
    if (idePath == null) {
2!
740
      return null;
×
741
    }
742
    return idePath.resolve(FOLDER_SOFTWARE);
4✔
743
  }
744

745
  @Override
746
  public SystemPath getPath() {
747

748
    return this.path;
3✔
749
  }
750

751
  @Override
752
  public EnvironmentVariables getVariables() {
753

754
    if (this.variables == null) {
3✔
755
      this.variables = createVariables();
4✔
756
    }
757
    return this.variables;
3✔
758
  }
759

760
  @Override
761
  public UrlMetadata getUrls() {
762

763
    if (this.urlMetadata == null) {
3✔
764
      if (!isTest()) {
3!
765
        getGitContext().pullOrCloneAndResetIfNeeded(IDE_URLS_GIT, getUrlsPath(), null);
×
766
      }
767
      this.urlMetadata = new UrlMetadata(this);
6✔
768
    }
769
    return this.urlMetadata;
3✔
770
  }
771

772
  @Override
773
  public boolean isQuietMode() {
774

775
    return this.startContext.isQuietMode();
4✔
776
  }
777

778
  @Override
779
  public boolean isBatchMode() {
780

781
    return this.startContext.isBatchMode();
4✔
782
  }
783

784
  @Override
785
  public boolean isForceMode() {
786

787
    return this.startContext.isForceMode();
4✔
788
  }
789

790
  @Override
791
  public boolean isForcePull() {
792

793
    return this.startContext.isForcePull();
4✔
794
  }
795

796
  @Override
797
  public boolean isForcePlugins() {
798

799
    return this.startContext.isForcePlugins();
4✔
800
  }
801

802
  @Override
803
  public boolean isForcePluginReinstall() {
804
    
805
    return this.startContext.isForcePluginReinstall();
4✔
806
  }
807

808
  @Override
809
  public boolean isForceRepositories() {
810

811
    return this.startContext.isForceRepositories();
4✔
812
  }
813

814
  @Override
815
  public boolean isOfflineMode() {
816

817
    return this.startContext.isOfflineMode();
4✔
818
  }
819

820
  @Override
821
  public boolean isPrivacyMode() {
822
    return this.startContext.isPrivacyMode();
4✔
823
  }
824

825
  @Override
826
  public boolean isSkipUpdatesMode() {
827

828
    return this.startContext.isSkipUpdatesMode();
4✔
829
  }
830

831
  @Override
832
  public boolean isNoColorsMode() {
833

834
    return this.startContext.isNoColorsMode();
×
835
  }
836

837
  @Override
838
  public NetworkStatus getNetworkStatus() {
839

840
    if (this.networkStatus == null) {
×
841
      this.networkStatus = new NetworkStatusImpl(this);
×
842
    }
843
    return this.networkStatus;
×
844
  }
845

846
  @Override
847
  public Locale getLocale() {
848

849
    Locale locale = this.startContext.getLocale();
4✔
850
    if (locale == null) {
2✔
851
      locale = Locale.getDefault();
2✔
852
    }
853
    return locale;
2✔
854
  }
855

856
  @Override
857
  public DirectoryMerger getWorkspaceMerger() {
858

859
    if (this.workspaceMerger == null) {
3✔
860
      this.workspaceMerger = new DirectoryMerger(this);
6✔
861
    }
862
    return this.workspaceMerger;
3✔
863
  }
864

865
  /**
866
   * @return the default execution directory in which a command process is executed.
867
   */
868
  @Override
869
  public Path getDefaultExecutionDirectory() {
870

871
    return this.defaultExecutionDirectory;
×
872
  }
873

874
  /**
875
   * @param defaultExecutionDirectory new value of {@link #getDefaultExecutionDirectory()}.
876
   */
877
  public void setDefaultExecutionDirectory(Path defaultExecutionDirectory) {
878

879
    if (defaultExecutionDirectory != null) {
×
880
      this.defaultExecutionDirectory = defaultExecutionDirectory;
×
881
    }
882
  }
×
883

884
  @Override
885
  public GitContext getGitContext() {
886

887
    return new GitContextImpl(this);
×
888
  }
889

890
  @Override
891
  public ProcessContext newProcess() {
892

893
    ProcessContext processContext = createProcessContext();
3✔
894
    if (this.defaultExecutionDirectory != null) {
3!
895
      processContext.directory(this.defaultExecutionDirectory);
×
896
    }
897
    return processContext;
2✔
898
  }
899

900
  @Override
901
  public IdeSystem getSystem() {
902

903
    if (this.system == null) {
×
904
      this.system = new IdeSystemImpl();
×
905
    }
906
    return this.system;
×
907
  }
908

909
  /**
910
   * @return a new instance of {@link ProcessContext}.
911
   * @see #newProcess()
912
   */
913
  protected ProcessContext createProcessContext() {
914

915
    return new ProcessContextImpl(this);
×
916
  }
917

918
  @Override
919
  public IdeLogLevel getLogLevelConsole() {
920

921
    return this.startContext.getLogLevelConsole();
×
922
  }
923

924
  @Override
925
  public IdeLogLevel getLogLevelLogger() {
926

927
    return this.startContext.getLogLevelLogger();
×
928
  }
929

930
  @Override
931
  public IdeLogListener getLogListener() {
932

933
    return this.startContext.getLogListener();
×
934
  }
935

936
  @Override
937
  public void logIdeHomeAndRootStatus() {
938
    if (this.ideRoot != null) {
3!
939
      IdeLogLevel.SUCCESS.log(LOG, "IDE_ROOT is set to {}", this.ideRoot);
×
940
    }
941
    if (this.ideHome == null) {
3✔
942
      LOG.warn(getMessageNotInsideIdeProject());
5✔
943
    } else {
944
      IdeLogLevel.SUCCESS.log(LOG, "IDE_HOME is set to {}", this.ideHome);
11✔
945
    }
946
  }
1✔
947

948
  @Override
949
  public String formatArgument(Object argument) {
950

951
    if (argument == null) {
2✔
952
      return null;
2✔
953
    }
954
    String result = argument.toString();
3✔
955
    if (isPrivacyMode()) {
3✔
956
      if ((this.ideRoot != null) && this.privacyMap.isEmpty()) {
3!
957
        initializePrivacyMap(this.userHome, "~");
×
958
        String projectName = getProjectName();
×
959
        if (!projectName.isEmpty()) {
×
960
          this.privacyMap.put(projectName, "project");
×
961
        }
962
      }
963
      for (Entry<String, String> entry : this.privacyMap.entrySet()) {
8!
964
        result = result.replace(entry.getKey(), entry.getValue());
×
965
      }
×
966
      result = PrivacyUtil.removeSensitivePathInformation(result);
3✔
967
    }
968
    return result;
2✔
969
  }
970

971
  /**
972
   * @param path the sensitive {@link Path} to
973
   * @param replacement the replacement to mask the {@link Path} in log output.
974
   */
975
  protected void initializePrivacyMap(Path path, String replacement) {
976

977
    if (path == null) {
×
978
      return;
×
979
    }
980
    if (this.systemInfo.isWindows()) {
×
981
      this.privacyMap.put(WindowsPathSyntax.WINDOWS.format(path), replacement);
×
982
      this.privacyMap.put(WindowsPathSyntax.MSYS.format(path), replacement);
×
983
    } else {
984
      this.privacyMap.put(path.toString(), replacement);
×
985
    }
986
  }
×
987

988
  /**
989
   * Resets the privacy map in case fundamental values have changed.
990
   */
991
  private void resetPrivacyMap() {
992

993
    this.privacyMap.clear();
3✔
994
  }
1✔
995

996

997
  @Override
998
  public String askForInput(String message, String defaultValue) {
999

1000
    while (true) {
1001
      if (!message.isBlank()) {
3!
1002
        IdeLogLevel.INTERACTION.log(LOG, message);
4✔
1003
      }
1004
      if (isBatchMode()) {
3!
1005
        if (isForceMode()) {
×
1006
          return defaultValue;
×
1007
        } else {
1008
          throw new CliAbortException();
×
1009
        }
1010
      }
1011
      String input = readLine().trim();
4✔
1012
      if (!input.isEmpty()) {
3!
1013
        return input;
2✔
1014
      } else {
1015
        if (defaultValue != null) {
×
1016
          return defaultValue;
×
1017
        }
1018
      }
1019
    }
×
1020
  }
1021

1022
  @Override
1023
  public <O> O question(O[] options, String question, Object... args) {
1024

1025
    assert (options.length > 0);
4!
1026
    IdeLogLevel.INTERACTION.log(LOG, question, args);
5✔
1027
    return displayOptionsAndGetAnswer(options);
4✔
1028
  }
1029

1030
  private <O> O displayOptionsAndGetAnswer(O[] options) {
1031
    Map<String, O> mapping = new HashMap<>(options.length);
6✔
1032
    int i = 0;
2✔
1033
    for (O option : options) {
16✔
1034
      i++;
1✔
1035
      String title = "" + option;
4✔
1036
      String key = computeOptionKey(title);
3✔
1037
      addMapping(mapping, key, option);
4✔
1038
      String numericKey = Integer.toString(i);
3✔
1039
      if (numericKey.equals(key)) {
4!
1040
        LOG.trace("Options should not be numeric: {}", key);
×
1041
      } else {
1042
        addMapping(mapping, numericKey, option);
4✔
1043
      }
1044
      IdeLogLevel.INTERACTION.log(LOG, "Option {}: {}", numericKey, title);
14✔
1045
    }
1046
    if (options.length == 1) {
4✔
1047
      mapping.put("", options[0]);
7✔
1048
    }
1049
    O option = null;
2✔
1050
    if (isBatchMode()) {
3!
1051
      if (isForceMode()) {
×
1052
        option = options[0];
×
1053
        IdeLogLevel.INTERACTION.log(LOG, "" + option);
×
1054
      }
1055
    } else {
1056
      while (option == null) {
2✔
1057
        String answer = readLine();
3✔
1058
        option = mapping.get(answer);
4✔
1059
        if (option == null) {
2!
1060
          LOG.warn("Invalid answer: '{}' - please try again.", answer);
×
1061
        }
1062
      }
1✔
1063
    }
1064
    return option;
2✔
1065
  }
1066

1067
  private static String computeOptionKey(String option) {
1068
    String key = option;
2✔
1069
    int index = -1;
2✔
1070
    for (char c : OPTION_DETAILS_START.toCharArray()) {
17✔
1071
      int currentIndex = key.indexOf(c);
4✔
1072
      if (currentIndex != -1) {
3✔
1073
        if ((index == -1) || (currentIndex < index)) {
3!
1074
          index = currentIndex;
2✔
1075
        }
1076
      }
1077
    }
1078
    if (index > 0) {
2✔
1079
      key = key.substring(0, index).trim();
6✔
1080
    }
1081
    return key;
2✔
1082
  }
1083

1084
  /**
1085
   * @return the input from the end-user (e.g. read from the console).
1086
   */
1087
  protected abstract String readLine();
1088

1089
  private static <O> void addMapping(Map<String, O> mapping, String key, O option) {
1090

1091
    O duplicate = mapping.put(key, option);
5✔
1092
    if (duplicate != null) {
2!
1093
      throw new IllegalArgumentException("Duplicated option " + key);
×
1094
    }
1095
  }
1✔
1096

1097
  @Override
1098
  public Step getCurrentStep() {
1099

1100
    return this.currentStep;
×
1101
  }
1102

1103
  @Override
1104
  public StepImpl newStep(boolean silent, String name, Object... parameters) {
1105

1106
    this.currentStep = new StepImpl(this, this.currentStep, name, silent, parameters);
11✔
1107
    return this.currentStep;
3✔
1108
  }
1109

1110
  /**
1111
   * Internal method to end the running {@link Step}.
1112
   *
1113
   * @param step the current {@link Step} to end.
1114
   */
1115
  public void endStep(StepImpl step) {
1116

1117
    if (step == this.currentStep) {
4!
1118
      this.currentStep = this.currentStep.getParent();
6✔
1119
    } else {
1120
      String currentStepName = "null";
×
1121
      if (this.currentStep != null) {
×
1122
        currentStepName = this.currentStep.getName();
×
1123
      }
1124
      LOG.warn("endStep called with wrong step '{}' but expected '{}'", step.getName(), currentStepName);
×
1125
    }
1126
  }
1✔
1127

1128
  /**
1129
   * Finds the matching {@link Commandlet} to run, applies {@link CliArguments} to its {@link Commandlet#getProperties() properties} and will execute it.
1130
   *
1131
   * @param arguments the {@link CliArgument}.
1132
   * @return the return code of the execution.
1133
   */
1134
  public int run(CliArguments arguments) {
1135

1136
    CliArgument current = arguments.current();
3✔
1137
    assert (this.currentStep == null);
4!
1138
    boolean supressStepSuccess = false;
2✔
1139
    StepImpl step = newStep(true, "ide", (Object[]) current.asArray());
8✔
1140
    Iterator<Commandlet> commandletIterator = this.commandletManager.findCommandlet(arguments, null);
6✔
1141
    Commandlet cmd = null;
2✔
1142
    ValidationResult result = null;
2✔
1143
    try {
1144
      while (commandletIterator.hasNext()) {
3✔
1145
        cmd = commandletIterator.next();
4✔
1146
        result = applyAndRun(arguments.copy(), cmd);
6✔
1147
        if (result.isValid()) {
3!
1148
          supressStepSuccess = cmd.isSuppressStepSuccess();
3✔
1149
          step.success();
2✔
1150
          return ProcessResult.SUCCESS;
4✔
1151
        }
1152
      }
1153
      activateLogging(cmd);
3✔
1154
      verifyIdeMinVersion(false);
3✔
1155
      if (result != null) {
2!
1156
        LOG.error(result.getErrorMessage());
×
1157
      }
1158
      step.error("Invalid arguments: {}", current.getArgs());
10✔
1159
      IdeLogLevel.INTERACTION.log(LOG, "For additional details run ide help {}", cmd == null ? "" : cmd.getName());
13!
1160
      return 1;
4✔
1161
    } catch (Throwable t) {
1✔
1162
      activateLogging(cmd);
3✔
1163
      step.error(t, true);
4✔
1164
      if (this.logfile != null) {
3!
1165
        System.err.println("Logfile can be found at " + this.logfile); // do not use logger
×
1166
      }
1167
      throw t;
2✔
1168
    } finally {
1169
      step.close();
2✔
1170
      assert (this.currentStep == null);
4!
1171
      step.logSummary(supressStepSuccess);
3✔
1172
    }
1173
  }
1174

1175
  /**
1176
   * Ensure the logging system is initialized.
1177
   */
1178
  private void activateLogging(Commandlet cmd) {
1179

1180
    configureJavaUtilLogging(cmd);
3✔
1181
    this.startContext.activateLogging();
3✔
1182
  }
1✔
1183

1184
  /**
1185
   * Configures the logging system (JUL).
1186
   *
1187
   * @param cmd the {@link Commandlet} to be called. May be {@code null}.
1188
   */
1189
  public void configureJavaUtilLogging(Commandlet cmd) {
1190

1191
    if (this.julConfigured) {
3✔
1192
      return;
1✔
1193
    }
1194
    boolean writeLogfile = isWriteLogfile(cmd);
4✔
1195
    this.startContext.setWriteLogfile(writeLogfile);
4✔
1196
    Properties properties = createJavaUtilLoggingProperties(writeLogfile, cmd);
5✔
1197
    try {
1198
      ByteArrayOutputStream out = new ByteArrayOutputStream(512);
5✔
1199
      properties.store(out, null);
4✔
1200
      out.flush();
2✔
1201
      ByteArrayInputStream in = new ByteArrayInputStream(out.toByteArray());
6✔
1202
      LogManager.getLogManager().readConfiguration(in);
3✔
1203
      this.julConfigured = true;
3✔
1204
      this.startContext.activateLogging();
3✔
1205
    } catch (IOException e) {
×
1206
      LOG.error("Failed to configure logging: {}", e.toString(), e);
×
1207
    }
1✔
1208
  }
1✔
1209

1210
  protected boolean isWriteLogfile(Commandlet cmd) {
1211
    if ((cmd == null) || !cmd.isWriteLogFile()) {
×
1212
      return false;
×
1213
    }
1214
    Boolean writeLogfile = IdeVariables.IDE_WRITE_LOGFILE.get(this);
×
1215
    return Boolean.TRUE.equals(writeLogfile);
×
1216
  }
1217

1218
  private Properties createJavaUtilLoggingProperties(boolean writeLogfile, Commandlet cmd) {
1219

1220
    Path idePath = getIdePath();
3✔
1221
    if (writeLogfile && (idePath == null)) {
2!
1222
      writeLogfile = false;
×
1223
      LOG.error("Cannot enable log-file since IDE_ROOT is undefined.");
×
1224
    }
1225
    Properties properties = new Properties();
4✔
1226
    // prevent 3rd party (e.g. java.lang.ProcessBuilder) logging into our console via JUL
1227
    // see JulLogLevel for the trick we did to workaround JUL flaws
1228
    properties.setProperty(".level", "SEVERE");
5✔
1229
    if (writeLogfile) {
2!
1230
      this.startContext.setLogLevelLogger(IdeLogLevel.TRACE);
×
1231
      String fileHandlerName = FileHandler.class.getName();
×
1232
      properties.setProperty("handlers", JulConsoleHandler.class.getName() + "," + fileHandlerName);
×
1233
      properties.setProperty(fileHandlerName + ".formatter", SimpleFormatter.class.getName());
×
1234
      properties.setProperty(fileHandlerName + ".encoding", "UTF-8");
×
1235
      this.logfile = createLogfilePath(idePath, cmd);
×
1236
      getFileAccess().mkdirs(this.logfile.getParent());
×
1237
      properties.setProperty(fileHandlerName + ".pattern", this.logfile.toString());
×
1238
    } else {
×
1239
      properties.setProperty("handlers", JulConsoleHandler.class.getName());
6✔
1240
    }
1241
    properties.setProperty(SimpleFormatter.class.getName() + ".format", "%1$tY-%1$tm-%1$td %1$tH:%1$tM:%1$tS.%1$tL [%4$s] [%3$s] %5$s%6$s%n");
7✔
1242
    return properties;
2✔
1243
  }
1244

1245
  private Path createLogfilePath(Path idePath, Commandlet cmd) {
1246
    LocalDateTime now = LocalDateTime.now();
×
1247
    Path logsPath = idePath.resolve(FOLDER_LOGS).resolve(DateTimeUtil.formatDate(now, true));
×
1248
    StringBuilder sb = new StringBuilder(32);
×
1249
    if (this.ideHome == null || ((cmd != null) && !cmd.isIdeHomeRequired())) {
×
1250
      sb.append("_ide-");
×
1251
    } else {
1252
      sb.append(this.ideHome.getFileName().toString());
×
1253
      sb.append('-');
×
1254
    }
1255
    sb.append("ide-");
×
1256
    if (cmd != null) {
×
1257
      sb.append(cmd.getName());
×
1258
      sb.append('-');
×
1259
    }
1260
    sb.append(DateTimeUtil.formatTime(now));
×
1261
    sb.append(".log");
×
1262
    return logsPath.resolve(sb.toString());
×
1263
  }
1264

1265
  @Override
1266
  public void runWithoutLogging(Runnable lambda, IdeLogLevel threshold) {
1267

1268
    this.startContext.deactivateLogging(threshold);
4✔
1269
    lambda.run();
2✔
1270
    this.startContext.activateLogging();
3✔
1271
  }
1✔
1272

1273
  /**
1274
   * @param cmd the potential {@link Commandlet} to {@link #apply(CliArguments, Commandlet) apply} and {@link Commandlet#run() run}.
1275
   * @return {@code true} if the given {@link Commandlet} matched and did {@link Commandlet#run() run} successfully, {@code false} otherwise (the
1276
   *     {@link Commandlet} did not match and we have to try a different candidate).
1277
   */
1278
  private ValidationResult applyAndRun(CliArguments arguments, Commandlet cmd) {
1279

1280
    IdeLogLevel previousLogLevel = null;
2✔
1281
    cmd.reset();
2✔
1282
    ValidationResult result = apply(arguments, cmd);
5✔
1283
    if (result.isValid()) {
3!
1284
      result = cmd.validate();
3✔
1285
    }
1286
    if (result.isValid()) {
3!
1287
      LOG.debug("Running commandlet {}", cmd);
4✔
1288
      if (cmd.isIdeHomeRequired() && (this.ideHome == null)) {
6!
1289
        throw new CliException(getMessageNotInsideIdeProject(), ProcessResult.NO_IDE_HOME);
×
1290
      } else if (cmd.isIdeRootRequired() && (this.ideRoot == null)) {
6!
1291
        throw new CliException(getMessageIdeRootNotFound(), ProcessResult.NO_IDE_ROOT);
7✔
1292
      }
1293
      try {
1294
        if (cmd.isProcessableOutput()) {
3!
1295
          if (!LOG.isDebugEnabled()) {
×
1296
            // unless --debug or --trace was supplied, processable output commandlets will disable all log-levels except INFO to prevent other logs interfere
1297
            previousLogLevel = this.startContext.setLogLevelConsole(IdeLogLevel.PROCESSABLE);
×
1298
          }
1299
        } else {
1300
          if (cmd.isIdeHomeRequired()) {
3!
1301
            LOG.debug(getMessageIdeHomeFound());
4✔
1302
          }
1303
          Path settingsRepository = getSettingsGitRepository();
3✔
1304
          if (settingsRepository != null) {
2!
1305
            if (getGitContext().isRepositoryUpdateAvailable(settingsRepository, getSettingsCommitIdPath()) || (
×
1306
                getGitContext().fetchIfNeeded(settingsRepository) && getGitContext().isRepositoryUpdateAvailable(
×
1307
                    settingsRepository, getSettingsCommitIdPath()))) {
×
1308

1309
              // Inform the user that an update is available. The update message is suppressed if we are already running the update
1310
              String msg = determineSettingsUpdateMessage(cmd);
×
1311
              if (msg != null) {
×
1312
                IdeLogLevel.INTERACTION.log(LOG, msg);
×
1313
              }
1314
            }
1315
          }
1316
        }
1317
        boolean success = ensureLicenseAgreement(cmd);
4✔
1318
        if (!success) {
2!
1319
          return ValidationResultValid.get();
×
1320
        }
1321
        cmd.run();
2✔
1322
      } finally {
1323
        if (previousLogLevel != null) {
2!
1324
          this.startContext.setLogLevelConsole(previousLogLevel);
×
1325
        }
1326
      }
1✔
1327
    } else {
1328
      LOG.trace("Commandlet did not match");
×
1329
    }
1330
    return result;
2✔
1331
  }
1332

1333

1334
  /**
1335
   * When an update is available for the settings repository, we log a message to the console, reminding the user to run {@code ide update}.
1336
   * This method determines the correct message to log, depending on whether the settings repository is a symlink/junction, or not.
1337
   * Should the user already be running the appropriate {@code ide update} command, the message is suppressed to avoid confusion.
1338
   *
1339
   * @param cmd the {@link Commandlet}.
1340
   * @return {@code msg} to log to the console. {@code null} if the message is suppressed.
1341
   */
1342
  private String determineSettingsUpdateMessage(Commandlet cmd) {
1343
    if (isSettingsRepositorySymlinkOrJunction()) {
×
1344
      if ((cmd instanceof UpdateCommandlet) && isForceMode()) {
×
1345
        return null;
×
1346
      }
1347
      return "Updates are available for the settings repository. Please pull the latest changes by yourself or by calling \"ide -f update\" to apply them.";
×
1348
    } else {
1349
      if (cmd instanceof UpdateCommandlet) {
×
1350
        return null;
×
1351
      }
1352
      return "Updates are available for the settings repository. If you want to apply the latest changes, call \"ide update\"";
×
1353
    }
1354
  }
1355

1356
  private boolean ensureLicenseAgreement(Commandlet cmd) {
1357

1358
    if (isTest()) {
3!
1359
      return true; // ignore for tests
2✔
1360
    }
1361
    getFileAccess().mkdirs(this.userHomeIde);
×
1362
    Path licenseAgreement = this.userHomeIde.resolve(FILE_LICENSE_AGREEMENT);
×
1363
    if (Files.isRegularFile(licenseAgreement)) {
×
1364
      return true; // success, license already accepted
×
1365
    }
1366
    if (cmd instanceof EnvironmentCommandlet) {
×
1367
      // if the license was not accepted, "$(ideasy env --bash)" that is written into a variable prevents the user from seeing the question he is asked
1368
      // in such situation the user could not open a bash terminal anymore and gets blocked what would really annoy the user so we exit here without doing or
1369
      // printing anything anymore in such case.
1370
      return false;
×
1371
    }
1372
    activateLogging(cmd);
×
1373
    IdeLogLevel oldLogLevel = this.startContext.getLogLevelConsole();
×
1374
    IdeLogLevel newLogLevel = oldLogLevel;
×
1375
    if (oldLogLevel.ordinal() > IdeLogLevel.INFO.ordinal()) {
×
1376
      newLogLevel = IdeLogLevel.INFO;
×
1377
      this.startContext.setLogLevelConsole(newLogLevel);
×
1378
    }
1379
    StringBuilder sb = new StringBuilder(1180);
×
1380
    sb.append(LOGO).append("""
×
1381
        Welcome to IDEasy!
1382
        This product (with its included 3rd party components) is open-source software and can be used free (also commercially).
1383
        It supports automatic download and installation of arbitrary 3rd party tools.
1384
        By default only open-source 3rd party tools are used (downloaded, installed, executed).
1385
        But if explicitly configured, also commercial software that requires an additional license may be used.
1386
        This happens e.g. if you configure "ultimate" edition of IntelliJ or "docker" edition of Docker (Docker Desktop).
1387
        You are solely responsible for all risks implied by using this software.
1388
        Before using IDEasy you need to read and accept the license agreement with all involved licenses.
1389
        You will be able to find it online under the following URL:
1390
        """).append(LICENSE_URL);
×
1391
    if (this.ideRoot != null) {
×
1392
      sb.append("\n\nAlso it is included in the documentation that you can find here:\n").
×
1393
          append(getIdePath().resolve("IDEasy.pdf").toString()).append("\n");
×
1394
    }
1395
    LOG.info(sb.toString());
×
1396
    askToContinue("Do you accept these terms of use and all license agreements?");
×
1397

1398
    sb.setLength(0);
×
1399
    LocalDateTime now = LocalDateTime.now();
×
1400
    sb.append("On ").append(DateTimeUtil.formatDate(now, false)).append(" at ").append(DateTimeUtil.formatTime(now))
×
1401
        .append(" you accepted the IDEasy license.\n").append(LICENSE_URL);
×
1402
    try {
1403
      Files.writeString(licenseAgreement, sb);
×
1404
    } catch (Exception e) {
×
1405
      throw new RuntimeException("Failed to save license agreement!", e);
×
1406
    }
×
1407
    if (oldLogLevel != newLogLevel) {
×
1408
      this.startContext.setLogLevelConsole(oldLogLevel);
×
1409
    }
1410
    return true;
×
1411
  }
1412

1413
  @Override
1414
  public void verifyIdeMinVersion(boolean throwException) {
1415
    VersionIdentifier minVersion = IDE_MIN_VERSION.get(this);
5✔
1416
    if (minVersion == null) {
2✔
1417
      return;
1✔
1418
    }
1419
    VersionIdentifier versionIdentifier = IdeVersion.getVersionIdentifier();
2✔
1420
    if (versionIdentifier.compareVersion(minVersion).isLess() && !IdeVersion.isUndefined()) {
7!
1421
      String message = String.format("Your version of IDEasy is currently %s\n"
13✔
1422
          + "However, this is too old as your project requires at latest version %s\n"
1423
          + "Please run the following command to update to the latest version of IDEasy and fix the problem:\n"
1424
          + "ide upgrade", versionIdentifier, minVersion);
1425
      if (throwException) {
2✔
1426
        throw new CliException(message);
5✔
1427
      } else {
1428
        LOG.warn(message);
3✔
1429
      }
1430
    }
1431
  }
1✔
1432

1433
  /**
1434
   * @param arguments the {@link CliArguments#ofCompletion(String...) completion arguments}.
1435
   * @param includeContextOptions to include the options of {@link ContextCommandlet}.
1436
   * @return the {@link List} of {@link CompletionCandidate}s to suggest.
1437
   */
1438
  public List<CompletionCandidate> complete(CliArguments arguments, boolean includeContextOptions) {
1439

1440
    CompletionCandidateCollector collector = new CompletionCandidateCollectorDefault(this);
5✔
1441
    if (arguments.current().isStart()) {
4✔
1442
      arguments.next();
3✔
1443
    }
1444
    if (includeContextOptions) {
2✔
1445
      ContextCommandlet cc = new ContextCommandlet();
4✔
1446
      for (Property<?> property : cc.getProperties()) {
11✔
1447
        assert (property.isOption());
4!
1448
        property.apply(arguments, this, cc, collector);
7✔
1449
      }
1✔
1450
    }
1451
    Iterator<Commandlet> commandletIterator = this.commandletManager.findCommandlet(arguments, collector);
6✔
1452
    CliArgument current = arguments.current();
3✔
1453
    if (current.isCompletion() && current.isCombinedShortOption()) {
6✔
1454
      collector.add(current.get(), null, null, null);
7✔
1455
    }
1456
    arguments.next();
3✔
1457
    while (commandletIterator.hasNext()) {
3✔
1458
      Commandlet cmd = commandletIterator.next();
4✔
1459
      if (!arguments.current().isEnd()) {
4✔
1460
        completeCommandlet(arguments.copy(), cmd, collector);
6✔
1461
      }
1462
    }
1✔
1463
    return collector.getSortedCandidates();
3✔
1464
  }
1465

1466
  private void completeCommandlet(CliArguments arguments, Commandlet cmd, CompletionCandidateCollector collector) {
1467

1468
    LOG.trace("Trying to match arguments for auto-completion for commandlet {}", cmd.getName());
5✔
1469
    Iterator<Property<?>> valueIterator = cmd.getValues().iterator();
4✔
1470
    valueIterator.next(); // skip first property since this is the keyword property that already matched to find the commandlet
3✔
1471
    List<Property<?>> properties = cmd.getProperties();
3✔
1472
    // we are creating our own list of options and remove them when matched to avoid duplicate suggestions
1473
    List<Property<?>> optionProperties = new ArrayList<>(properties.size());
6✔
1474
    for (Property<?> property : properties) {
10✔
1475
      if (property.isOption()) {
3✔
1476
        optionProperties.add(property);
4✔
1477
      }
1478
    }
1✔
1479
    CliArgument currentArgument = arguments.current();
3✔
1480
    while (!currentArgument.isEnd()) {
3✔
1481
      LOG.trace("Trying to match argument '{}'", currentArgument);
4✔
1482
      if (currentArgument.isOption() && !arguments.isEndOptions()) {
6!
1483
        if (currentArgument.isCompletion()) {
3✔
1484
          Iterator<Property<?>> optionIterator = optionProperties.iterator();
3✔
1485
          while (optionIterator.hasNext()) {
3✔
1486
            Property<?> option = optionIterator.next();
4✔
1487
            boolean success = option.apply(arguments, this, cmd, collector);
7✔
1488
            if (success) {
2✔
1489
              optionIterator.remove();
2✔
1490
              arguments.next();
3✔
1491
            }
1492
          }
1✔
1493
        } else {
1✔
1494
          Property<?> option = cmd.getOption(currentArgument.get());
5✔
1495
          if (option != null) {
2✔
1496
            arguments.next();
3✔
1497
            boolean removed = optionProperties.remove(option);
4✔
1498
            if (!removed) {
2!
1499
              option = null;
×
1500
            }
1501
          }
1502
          if (option == null) {
2✔
1503
            LOG.trace("No such option was found.");
3✔
1504
            return;
1✔
1505
          }
1506
        }
1✔
1507
      } else {
1508
        if (valueIterator.hasNext()) {
3✔
1509
          Property<?> valueProperty = valueIterator.next();
4✔
1510
          boolean success = valueProperty.apply(arguments, this, cmd, collector);
7✔
1511
          if (!success) {
2✔
1512
            LOG.trace("Completion cannot match any further.");
3✔
1513
            return;
1✔
1514
          }
1515
        } else {
1✔
1516
          LOG.trace("No value left for completion.");
3✔
1517
          return;
1✔
1518
        }
1519
      }
1520
      currentArgument = arguments.current();
4✔
1521
    }
1522
  }
1✔
1523

1524
  /**
1525
   * @param arguments the {@link CliArguments} to apply. Will be {@link CliArguments#next() consumed} as they are matched. Consider passing a
1526
   *     {@link CliArguments#copy() copy} as needed.
1527
   * @param cmd the potential {@link Commandlet} to match.
1528
   * @return the {@link ValidationResult} telling if the {@link CliArguments} can be applied successfully or if validation errors ocurred.
1529
   */
1530
  public ValidationResult apply(CliArguments arguments, Commandlet cmd) {
1531

1532
    LOG.trace("Trying to match arguments to commandlet {}", cmd.getName());
5✔
1533
    CliArgument currentArgument = arguments.current();
3✔
1534
    Iterator<Property<?>> propertyIterator = cmd.getValues().iterator();
4✔
1535
    Property<?> property = null;
2✔
1536
    if (propertyIterator.hasNext()) {
3!
1537
      property = propertyIterator.next();
4✔
1538
    }
1539
    while (!currentArgument.isEnd()) {
3✔
1540
      LOG.trace("Trying to match argument '{}'", currentArgument);
4✔
1541
      Property<?> currentProperty = property;
2✔
1542
      if (!arguments.isEndOptions()) {
3!
1543
        Property<?> option = cmd.getOption(currentArgument.getKey());
5✔
1544
        if (option != null) {
2!
1545
          currentProperty = option;
×
1546
        }
1547
      }
1548
      if (currentProperty == null) {
2!
1549
        LOG.trace("No option or next value found");
×
1550
        ValidationState state = new ValidationState(null);
×
1551
        state.addErrorMessage("No matching property found");
×
1552
        return state;
×
1553
      }
1554
      LOG.trace("Next property candidate to match argument is {}", currentProperty);
4✔
1555
      if (currentProperty == property) {
3!
1556
        if (!property.isMultiValued()) {
3✔
1557
          if (propertyIterator.hasNext()) {
3✔
1558
            property = propertyIterator.next();
5✔
1559
          } else {
1560
            property = null;
2✔
1561
          }
1562
        }
1563
        if ((property != null) && property.isValue() && property.isMultiValued()) {
8!
1564
          arguments.stopSplitShortOptions();
2✔
1565
        }
1566
      }
1567
      boolean matches = currentProperty.apply(arguments, this, cmd, null);
7✔
1568
      if (!matches) {
2!
1569
        ValidationState state = new ValidationState(null);
×
1570
        state.addErrorMessage("No matching property found");
×
1571
        return state;
×
1572
      }
1573
      currentArgument = arguments.current();
3✔
1574
    }
1✔
1575
    return ValidationResultValid.get();
2✔
1576
  }
1577

1578
  @Override
1579
  public Path findBash() {
1580
    if (this.bash != null) {
3✔
1581
      return this.bash;
3✔
1582
    }
1583
    Path bashPath = findBashOnBashPath();
3✔
1584
    if (bashPath == null) {
2✔
1585
      bashPath = findBashInPath();
3✔
1586
      if (bashPath == null && (getSystemInfo().isWindows() || SystemInfoImpl.INSTANCE.isWindows())) {
6!
1587
        bashPath = findBashOnWindowsDefaultGitPath();
3✔
1588
        if (bashPath == null) {
2!
1589
          bashPath = findBashInWindowsRegistry();
3✔
1590
        }
1591
      }
1592
    }
1593
    if (bashPath == null) {
2✔
1594
      LOG.error("No bash executable could be found on your system.");
4✔
1595
    } else {
1596
      this.bash = bashPath;
3✔
1597
    }
1598
    return bashPath;
2✔
1599
  }
1600

1601
  private Path findBashOnBashPath() {
1602
    LOG.trace("Trying to find BASH_PATH environment variable.");
3✔
1603
    Path bash;
1604
    String bashPathVariableName = IdeVariables.BASH_PATH.getName();
3✔
1605
    String bashVariable = getVariables().get(bashPathVariableName);
5✔
1606
    if (bashVariable != null) {
2✔
1607
      bash = Path.of(bashVariable);
5✔
1608
      if (Files.exists(bash)) {
5✔
1609
        LOG.debug("{} environment variable was found and points to: {}", bashPathVariableName, bash);
5✔
1610
        return bash;
2✔
1611
      } else {
1612
        LOG.error("The environment variable {} points to a non existing file: {}", bashPathVariableName, bash);
5✔
1613
        return null;
2✔
1614
      }
1615
    } else {
1616
      LOG.debug("{} environment variable was not found", bashPathVariableName);
4✔
1617
      return null;
2✔
1618
    }
1619
  }
1620

1621
  /**
1622
   * @param path the path to check.
1623
   * @param toIgnore the String sequence which needs to be checked and ignored.
1624
   * @return {@code true} if the sequence to ignore was not found, {@code false} if the path contained the sequence to ignore.
1625
   */
1626
  private boolean checkPathToIgnoreLowercase(Path path, String toIgnore) {
1627
    String s = path.toAbsolutePath().toString().toLowerCase(Locale.ROOT);
6✔
1628
    return !s.contains(toIgnore);
7!
1629
  }
1630

1631
  /**
1632
   * Tries to find the bash.exe within the PATH environment variable.
1633
   *
1634
   * @return Path to bash.exe if found in PATH environment variable, {@code null} if bash.exe was not found.
1635
   */
1636
  private Path findBashInPath() {
1637
    LOG.trace("Trying to find bash in PATH environment variable.");
3✔
1638
    Path bash;
1639
    String pathVariableName = IdeVariables.PATH.getName();
3✔
1640
    if (pathVariableName != null) {
2!
1641
      Path plainBash = Path.of(BASH);
5✔
1642
      Predicate<Path> pathsToIgnore = p -> checkPathToIgnoreLowercase(p, "\\appdata\\local\\microsoft\\windowsapps") && checkPathToIgnoreLowercase(p,
16!
1643
          "\\windows\\system32");
1644
      Path bashPath = getPath().findBinary(plainBash, pathsToIgnore);
6✔
1645
      bash = bashPath.toAbsolutePath();
3✔
1646
      if (bashPath.equals(plainBash)) {
4✔
1647
        LOG.warn("No usable bash executable was found in your PATH environment variable!");
3✔
1648
        bash = null;
3✔
1649
      } else {
1650
        if (Files.exists(bashPath)) {
5!
1651
          LOG.debug("A proper bash executable was found in your PATH environment variable at: {}", bash);
5✔
1652
        } else {
1653
          bash = null;
×
1654
          LOG.error("A path to a bash executable was found in your PATH environment variable at: {} but the file is not existing.", bash);
×
1655
        }
1656
      }
1657
    } else {
1✔
1658
      bash = null;
×
1659
      // this should never happen...
1660
      LOG.error("PATH environment variable was not found");
×
1661
    }
1662
    return bash;
2✔
1663
  }
1664

1665
  /**
1666
   * Tries to find the bash.exe within the Windows registry.
1667
   *
1668
   * @return Path to bash.exe if found in registry, {@code null} if bash.exe was found.
1669
   */
1670
  protected Path findBashInWindowsRegistry() {
1671
    LOG.trace("Trying to find bash in Windows registry");
×
1672
    // If not found in the default location, try the registry query
1673
    String[] bashVariants = { "GitForWindows", "Cygwin\\setup" };
×
1674
    String[] registryKeys = { "HKEY_LOCAL_MACHINE", "HKEY_CURRENT_USER" };
×
1675
    for (String bashVariant : bashVariants) {
×
1676
      LOG.trace("Trying to find bash variant: {}", bashVariant);
×
1677
      for (String registryKey : registryKeys) {
×
1678
        LOG.trace("Trying to find bash from registry key: {}", registryKey);
×
1679
        String toolValueName = ("GitForWindows".equals(bashVariant)) ? "InstallPath" : "rootdir";
×
1680
        String registryPath = registryKey + "\\Software\\" + bashVariant;
×
1681

1682
        String path = getWindowsHelper().getRegistryValue(registryPath, toolValueName);
×
1683
        if (path != null) {
×
1684
          Path bashPath = Path.of(path + "\\bin\\bash.exe");
×
1685
          if (Files.exists(bashPath)) {
×
1686
            LOG.debug("Found bash at: {}", bashPath);
×
1687
            return bashPath;
×
1688
          } else {
1689
            LOG.error("Found bash at: {} but it is not pointing to an existing file", bashPath);
×
1690
            return null;
×
1691
          }
1692
        } else {
1693
          LOG.info("No bash executable could be found in the Windows registry.");
×
1694
        }
1695
      }
1696
    }
1697
    // no bash found
1698
    return null;
×
1699
  }
1700

1701
  private Path findBashOnWindowsDefaultGitPath() {
1702
    // Check if Git Bash exists in the default location
1703
    LOG.trace("Trying to find bash on the Windows default git path.");
3✔
1704
    Path defaultPath = Path.of(getDefaultWindowsGitPath());
6✔
1705
    if (!defaultPath.toString().isEmpty() && Files.exists(defaultPath)) {
4!
1706
      LOG.trace("Found default path to git bash on Windows at: {}", getDefaultWindowsGitPath());
×
1707
      return defaultPath;
×
1708
    }
1709
    LOG.debug("No bash was found on the Windows default git path.");
3✔
1710
    return null;
2✔
1711
  }
1712

1713
  @Override
1714
  public WindowsPathSyntax getPathSyntax() {
1715

1716
    return this.pathSyntax;
3✔
1717
  }
1718

1719
  /**
1720
   * @param pathSyntax new value of {@link #getPathSyntax()}.
1721
   */
1722
  public void setPathSyntax(WindowsPathSyntax pathSyntax) {
1723

1724
    this.pathSyntax = pathSyntax;
3✔
1725
  }
1✔
1726

1727
  /**
1728
   * @return the {@link IdeStartContextImpl}.
1729
   */
1730
  public IdeStartContextImpl getStartContext() {
1731

1732
    return startContext;
3✔
1733
  }
1734

1735
  /**
1736
   * @return the {@link WindowsHelper}.
1737
   */
1738
  public final WindowsHelper getWindowsHelper() {
1739

1740
    if (this.windowsHelper == null) {
3✔
1741
      this.windowsHelper = createWindowsHelper();
4✔
1742
    }
1743
    return this.windowsHelper;
3✔
1744
  }
1745

1746
  /**
1747
   * @return the new {@link WindowsHelper} instance.
1748
   */
1749
  protected WindowsHelper createWindowsHelper() {
1750

1751
    return new WindowsHelperImpl(this);
×
1752
  }
1753

1754
  /**
1755
   * Reloads this context and re-initializes the {@link #getVariables() variables}.
1756
   */
1757
  public void reload() {
1758

1759
    this.variables = null;
3✔
1760
    this.customToolRepository = null;
3✔
1761
  }
1✔
1762

1763
  @Override
1764
  public void writeVersionFile(VersionIdentifier version, Path installationPath) {
1765

1766
    assert (Files.isDirectory(installationPath));
6!
1767
    Path versionFile = installationPath.resolve(FILE_SOFTWARE_VERSION);
4✔
1768
    getFileAccess().writeFileContent(version.toString(), versionFile);
6✔
1769
  }
1✔
1770

1771
  /*
1772
   * @param home the IDE_HOME directory.
1773
   * @param workspace the name of the active workspace folder.
1774
   */
1775
  protected static record IdeHomeAndWorkspace(Path home, String workspace) {
9✔
1776

1777
  }
1778

1779
  /**
1780
   * Returns the default git path on Windows. Required to be overwritten in tests.
1781
   *
1782
   * @return default path to git on Windows.
1783
   */
1784
  public String getDefaultWindowsGitPath() {
1785
    return DEFAULT_WINDOWS_GIT_PATH;
×
1786
  }
1787

1788
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc