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

devonfw / IDEasy / 24844919410

23 Apr 2026 03:50PM UTC coverage: 70.658% (+0.05%) from 70.61%
24844919410

push

github

web-flow
#1270: Load global user settings outside project (#1813)

Co-authored-by: Jörg Hohwiller <hohwille@users.noreply.github.com>

4338 of 6786 branches covered (63.93%)

Branch coverage included in aggregate %.

11221 of 15234 relevant lines covered (73.66%)

3.11 hits per line

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

65.19
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 isForceRepositories() {
804

805
    return this.startContext.isForceRepositories();
4✔
806
  }
807

808
  @Override
809
  public boolean isOfflineMode() {
810

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

814
  @Override
815
  public boolean isPrivacyMode() {
816
    return this.startContext.isPrivacyMode();
4✔
817
  }
818

819
  @Override
820
  public boolean isSkipUpdatesMode() {
821

822
    return this.startContext.isSkipUpdatesMode();
4✔
823
  }
824

825
  @Override
826
  public boolean isNoColorsMode() {
827

828
    return this.startContext.isNoColorsMode();
×
829
  }
830

831
  @Override
832
  public NetworkStatus getNetworkStatus() {
833

834
    if (this.networkStatus == null) {
×
835
      this.networkStatus = new NetworkStatusImpl(this);
×
836
    }
837
    return this.networkStatus;
×
838
  }
839

840
  @Override
841
  public Locale getLocale() {
842

843
    Locale locale = this.startContext.getLocale();
4✔
844
    if (locale == null) {
2✔
845
      locale = Locale.getDefault();
2✔
846
    }
847
    return locale;
2✔
848
  }
849

850
  @Override
851
  public DirectoryMerger getWorkspaceMerger() {
852

853
    if (this.workspaceMerger == null) {
3✔
854
      this.workspaceMerger = new DirectoryMerger(this);
6✔
855
    }
856
    return this.workspaceMerger;
3✔
857
  }
858

859
  /**
860
   * @return the default execution directory in which a command process is executed.
861
   */
862
  @Override
863
  public Path getDefaultExecutionDirectory() {
864

865
    return this.defaultExecutionDirectory;
×
866
  }
867

868
  /**
869
   * @param defaultExecutionDirectory new value of {@link #getDefaultExecutionDirectory()}.
870
   */
871
  public void setDefaultExecutionDirectory(Path defaultExecutionDirectory) {
872

873
    if (defaultExecutionDirectory != null) {
×
874
      this.defaultExecutionDirectory = defaultExecutionDirectory;
×
875
    }
876
  }
×
877

878
  @Override
879
  public GitContext getGitContext() {
880

881
    return new GitContextImpl(this);
×
882
  }
883

884
  @Override
885
  public ProcessContext newProcess() {
886

887
    ProcessContext processContext = createProcessContext();
3✔
888
    if (this.defaultExecutionDirectory != null) {
3!
889
      processContext.directory(this.defaultExecutionDirectory);
×
890
    }
891
    return processContext;
2✔
892
  }
893

894
  @Override
895
  public IdeSystem getSystem() {
896

897
    if (this.system == null) {
×
898
      this.system = new IdeSystemImpl();
×
899
    }
900
    return this.system;
×
901
  }
902

903
  /**
904
   * @return a new instance of {@link ProcessContext}.
905
   * @see #newProcess()
906
   */
907
  protected ProcessContext createProcessContext() {
908

909
    return new ProcessContextImpl(this);
×
910
  }
911

912
  @Override
913
  public IdeLogLevel getLogLevelConsole() {
914

915
    return this.startContext.getLogLevelConsole();
×
916
  }
917

918
  @Override
919
  public IdeLogLevel getLogLevelLogger() {
920

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

924
  @Override
925
  public IdeLogListener getLogListener() {
926

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

930
  @Override
931
  public void logIdeHomeAndRootStatus() {
932
    if (this.ideRoot != null) {
3!
933
      IdeLogLevel.SUCCESS.log(LOG, "IDE_ROOT is set to {}", this.ideRoot);
×
934
    }
935
    if (this.ideHome == null) {
3✔
936
      LOG.warn(getMessageNotInsideIdeProject());
5✔
937
    } else {
938
      IdeLogLevel.SUCCESS.log(LOG, "IDE_HOME is set to {}", this.ideHome);
11✔
939
    }
940
  }
1✔
941

942
  @Override
943
  public String formatArgument(Object argument) {
944

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

965
  /**
966
   * @param path the sensitive {@link Path} to
967
   * @param replacement the replacement to mask the {@link Path} in log output.
968
   */
969
  protected void initializePrivacyMap(Path path, String replacement) {
970

971
    if (path == null) {
×
972
      return;
×
973
    }
974
    if (this.systemInfo.isWindows()) {
×
975
      this.privacyMap.put(WindowsPathSyntax.WINDOWS.format(path), replacement);
×
976
      this.privacyMap.put(WindowsPathSyntax.MSYS.format(path), replacement);
×
977
    } else {
978
      this.privacyMap.put(path.toString(), replacement);
×
979
    }
980
  }
×
981

982
  /**
983
   * Resets the privacy map in case fundamental values have changed.
984
   */
985
  private void resetPrivacyMap() {
986

987
    this.privacyMap.clear();
3✔
988
  }
1✔
989

990

991
  @Override
992
  public String askForInput(String message, String defaultValue) {
993

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

1016
  @Override
1017
  public <O> O question(O[] options, String question, Object... args) {
1018

1019
    assert (options.length > 0);
4!
1020
    IdeLogLevel.INTERACTION.log(LOG, question, args);
5✔
1021
    return displayOptionsAndGetAnswer(options);
4✔
1022
  }
1023

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

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

1078
  /**
1079
   * @return the input from the end-user (e.g. read from the console).
1080
   */
1081
  protected abstract String readLine();
1082

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

1085
    O duplicate = mapping.put(key, option);
5✔
1086
    if (duplicate != null) {
2!
1087
      throw new IllegalArgumentException("Duplicated option " + key);
×
1088
    }
1089
  }
1✔
1090

1091
  @Override
1092
  public Step getCurrentStep() {
1093

1094
    return this.currentStep;
×
1095
  }
1096

1097
  @Override
1098
  public StepImpl newStep(boolean silent, String name, Object... parameters) {
1099

1100
    this.currentStep = new StepImpl(this, this.currentStep, name, silent, parameters);
11✔
1101
    return this.currentStep;
3✔
1102
  }
1103

1104
  /**
1105
   * Internal method to end the running {@link Step}.
1106
   *
1107
   * @param step the current {@link Step} to end.
1108
   */
1109
  public void endStep(StepImpl step) {
1110

1111
    if (step == this.currentStep) {
4!
1112
      this.currentStep = this.currentStep.getParent();
6✔
1113
    } else {
1114
      String currentStepName = "null";
×
1115
      if (this.currentStep != null) {
×
1116
        currentStepName = this.currentStep.getName();
×
1117
      }
1118
      LOG.warn("endStep called with wrong step '{}' but expected '{}'", step.getName(), currentStepName);
×
1119
    }
1120
  }
1✔
1121

1122
  /**
1123
   * Finds the matching {@link Commandlet} to run, applies {@link CliArguments} to its {@link Commandlet#getProperties() properties} and will execute it.
1124
   *
1125
   * @param arguments the {@link CliArgument}.
1126
   * @return the return code of the execution.
1127
   */
1128
  public int run(CliArguments arguments) {
1129

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

1169
  /**
1170
   * Ensure the logging system is initialized.
1171
   */
1172
  private void activateLogging(Commandlet cmd) {
1173

1174
    configureJavaUtilLogging(cmd);
3✔
1175
    this.startContext.activateLogging();
3✔
1176
  }
1✔
1177

1178
  /**
1179
   * Configures the logging system (JUL).
1180
   *
1181
   * @param cmd the {@link Commandlet} to be called. May be {@code null}.
1182
   */
1183
  public void configureJavaUtilLogging(Commandlet cmd) {
1184

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

1204
  protected boolean isWriteLogfile(Commandlet cmd) {
1205
    if ((cmd == null) || !cmd.isWriteLogFile()) {
×
1206
      return false;
×
1207
    }
1208
    Boolean writeLogfile = IdeVariables.IDE_WRITE_LOGFILE.get(this);
×
1209
    return Boolean.TRUE.equals(writeLogfile);
×
1210
  }
1211

1212
  private Properties createJavaUtilLoggingProperties(boolean writeLogfile, Commandlet cmd) {
1213

1214
    Path idePath = getIdePath();
3✔
1215
    if (writeLogfile && (idePath == null)) {
2!
1216
      writeLogfile = false;
×
1217
      LOG.error("Cannot enable log-file since IDE_ROOT is undefined.");
×
1218
    }
1219
    Properties properties = new Properties();
4✔
1220
    // prevent 3rd party (e.g. java.lang.ProcessBuilder) logging into our console via JUL
1221
    // see JulLogLevel for the trick we did to workaround JUL flaws
1222
    properties.setProperty(".level", "SEVERE");
5✔
1223
    if (writeLogfile) {
2!
1224
      this.startContext.setLogLevelLogger(IdeLogLevel.TRACE);
×
1225
      String fileHandlerName = FileHandler.class.getName();
×
1226
      properties.setProperty("handlers", JulConsoleHandler.class.getName() + "," + fileHandlerName);
×
1227
      properties.setProperty(fileHandlerName + ".formatter", SimpleFormatter.class.getName());
×
1228
      properties.setProperty(fileHandlerName + ".encoding", "UTF-8");
×
1229
      this.logfile = createLogfilePath(idePath, cmd);
×
1230
      getFileAccess().mkdirs(this.logfile.getParent());
×
1231
      properties.setProperty(fileHandlerName + ".pattern", this.logfile.toString());
×
1232
    } else {
×
1233
      properties.setProperty("handlers", JulConsoleHandler.class.getName());
6✔
1234
    }
1235
    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✔
1236
    return properties;
2✔
1237
  }
1238

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

1259
  @Override
1260
  public void runWithoutLogging(Runnable lambda, IdeLogLevel threshold) {
1261

1262
    this.startContext.deactivateLogging(threshold);
4✔
1263
    lambda.run();
2✔
1264
    this.startContext.activateLogging();
3✔
1265
  }
1✔
1266

1267
  /**
1268
   * @param cmd the potential {@link Commandlet} to {@link #apply(CliArguments, Commandlet) apply} and {@link Commandlet#run() run}.
1269
   * @return {@code true} if the given {@link Commandlet} matched and did {@link Commandlet#run() run} successfully, {@code false} otherwise (the
1270
   *     {@link Commandlet} did not match and we have to try a different candidate).
1271
   */
1272
  private ValidationResult applyAndRun(CliArguments arguments, Commandlet cmd) {
1273

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

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

1327

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

1350
  private boolean ensureLicenseAgreement(Commandlet cmd) {
1351

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

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

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

1427
  /**
1428
   * @param arguments the {@link CliArguments#ofCompletion(String...) completion arguments}.
1429
   * @param includeContextOptions to include the options of {@link ContextCommandlet}.
1430
   * @return the {@link List} of {@link CompletionCandidate}s to suggest.
1431
   */
1432
  public List<CompletionCandidate> complete(CliArguments arguments, boolean includeContextOptions) {
1433

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

1460
  private void completeCommandlet(CliArguments arguments, Commandlet cmd, CompletionCandidateCollector collector) {
1461

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

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

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

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

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

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

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

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

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

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

1707
  @Override
1708
  public WindowsPathSyntax getPathSyntax() {
1709

1710
    return this.pathSyntax;
3✔
1711
  }
1712

1713
  /**
1714
   * @param pathSyntax new value of {@link #getPathSyntax()}.
1715
   */
1716
  public void setPathSyntax(WindowsPathSyntax pathSyntax) {
1717

1718
    this.pathSyntax = pathSyntax;
3✔
1719
  }
1✔
1720

1721
  /**
1722
   * @return the {@link IdeStartContextImpl}.
1723
   */
1724
  public IdeStartContextImpl getStartContext() {
1725

1726
    return startContext;
3✔
1727
  }
1728

1729
  /**
1730
   * @return the {@link WindowsHelper}.
1731
   */
1732
  public final WindowsHelper getWindowsHelper() {
1733

1734
    if (this.windowsHelper == null) {
3✔
1735
      this.windowsHelper = createWindowsHelper();
4✔
1736
    }
1737
    return this.windowsHelper;
3✔
1738
  }
1739

1740
  /**
1741
   * @return the new {@link WindowsHelper} instance.
1742
   */
1743
  protected WindowsHelper createWindowsHelper() {
1744

1745
    return new WindowsHelperImpl(this);
×
1746
  }
1747

1748
  /**
1749
   * Reloads this context and re-initializes the {@link #getVariables() variables}.
1750
   */
1751
  public void reload() {
1752

1753
    this.variables = null;
3✔
1754
    this.customToolRepository = null;
3✔
1755
  }
1✔
1756

1757
  @Override
1758
  public void writeVersionFile(VersionIdentifier version, Path installationPath) {
1759

1760
    assert (Files.isDirectory(installationPath));
6!
1761
    Path versionFile = installationPath.resolve(FILE_SOFTWARE_VERSION);
4✔
1762
    getFileAccess().writeFileContent(version.toString(), versionFile);
6✔
1763
  }
1✔
1764

1765
  /*
1766
   * @param home the IDE_HOME directory.
1767
   * @param workspace the name of the active workspace folder.
1768
   */
1769
  protected static record IdeHomeAndWorkspace(Path home, String workspace) {
9✔
1770

1771
  }
1772

1773
  /**
1774
   * Returns the default git path on Windows. Required to be overwritten in tests.
1775
   *
1776
   * @return default path to git on Windows.
1777
   */
1778
  public String getDefaultWindowsGitPath() {
1779
    return DEFAULT_WINDOWS_GIT_PATH;
×
1780
  }
1781

1782
}
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