From ed1e84c39392c3bac72829c4fc097e50ad128558 Mon Sep 17 00:00:00 2001 From: Knize <62898885+kirill-knize-sonarsource@users.noreply.github.com> Date: Tue, 12 Mar 2024 13:58:34 +0100 Subject: [PATCH] SLLS-227 Remove temporary workaround endpoints for config scopes (#339) * SLLS-227 Remove temporary workaround endpoints for config scopes * Adjust after rebase * Fix (?) flakiness of test assertion --------- Co-authored-by: Jean-Baptiste Lievremont --- .../ls/SonarLintExtendedLanguageServer.java | 18 ----- .../sonarlint/ls/SonarLintLanguageServer.java | 19 ----- .../AbstractLanguageServerMediumTests.java | 12 ++- .../mediumtests/ConnectedModeMediumTests.java | 74 +++++++++---------- 4 files changed, 39 insertions(+), 84 deletions(-) diff --git a/src/main/java/org/sonarsource/sonarlint/ls/SonarLintExtendedLanguageServer.java b/src/main/java/org/sonarsource/sonarlint/ls/SonarLintExtendedLanguageServer.java index 732e94d80..61b905a05 100644 --- a/src/main/java/org/sonarsource/sonarlint/ls/SonarLintExtendedLanguageServer.java +++ b/src/main/java/org/sonarsource/sonarlint/ls/SonarLintExtendedLanguageServer.java @@ -730,22 +730,4 @@ public boolean isBindingSuggestionDisabled() { } } - @JsonNotification("didAddConfigurationScopes") - CompletableFuture didAddConfigurationScopes(DidAddConfigurationScopes params); - - class DidRemoveConfigurationScopeParams { - private final String removedId; - - public DidRemoveConfigurationScopeParams(String removedId) { - this.removedId = removedId; - } - - public String getRemovedId() { - return removedId; - } - } - - @JsonNotification("didRemoveConfigurationScope") - CompletableFuture didRemoveConfigurationScope(DidRemoveConfigurationScopeParams params); - } diff --git a/src/main/java/org/sonarsource/sonarlint/ls/SonarLintLanguageServer.java b/src/main/java/org/sonarsource/sonarlint/ls/SonarLintLanguageServer.java index e0a5ebf12..3033c81ec 100644 --- a/src/main/java/org/sonarsource/sonarlint/ls/SonarLintLanguageServer.java +++ b/src/main/java/org/sonarsource/sonarlint/ls/SonarLintLanguageServer.java @@ -92,8 +92,6 @@ import org.sonarsource.sonarlint.core.rpc.protocol.backend.analysis.GetSupportedFilePatternsParams; import org.sonarsource.sonarlint.core.rpc.protocol.backend.analysis.GetSupportedFilePatternsResponse; import org.sonarsource.sonarlint.core.rpc.protocol.backend.binding.GetBindingSuggestionParams; -import org.sonarsource.sonarlint.core.rpc.protocol.backend.config.binding.BindingConfigurationDto; -import org.sonarsource.sonarlint.core.rpc.protocol.backend.config.scope.ConfigurationScopeDto; import org.sonarsource.sonarlint.core.rpc.protocol.backend.connection.auth.HelpGenerateUserTokenResponse; import org.sonarsource.sonarlint.core.rpc.protocol.backend.connection.validate.ValidateConnectionParams; import org.sonarsource.sonarlint.core.rpc.protocol.backend.hotspot.CheckStatusChangePermittedParams; @@ -917,23 +915,6 @@ public CompletableFuture getAllowedHotspotSta }); } - @Override - public CompletableFuture didAddConfigurationScopes(DidAddConfigurationScopes params) { - var bindingConfigurationDto = new BindingConfigurationDto(params.getConnectionId(), params.getSonarProjectKey(), - params.isBindingSuggestionDisabled()); - var configurationDto = new ConfigurationScopeDto(params.getId(), null, params.isBindable(), params.getName(), bindingConfigurationDto); - var arg = new org.sonarsource.sonarlint.core.rpc.protocol.backend.config.scope.DidAddConfigurationScopesParams(List.of(configurationDto)); - backendServiceFacade.getBackendService().didAddConfigurationScopes(arg); - return CompletableFuture.completedFuture(null); - } - - @Override - public CompletableFuture didRemoveConfigurationScope(DidRemoveConfigurationScopeParams params) { - var arg = new org.sonarsource.sonarlint.core.rpc.protocol.backend.config.scope.DidRemoveConfigurationScopeParams(params.getRemovedId()); - backendServiceFacade.getBackendService().didRemoveConfigurationScope(arg); - return CompletableFuture.completedFuture(null); - } - @Override public void reopenResolvedLocalIssues(ReopenAllIssuesForFileParams params) { var reopenAllIssuesParams = new org.sonarsource.sonarlint.core.rpc.protocol.backend.issue.ReopenAllIssuesForFileParams( diff --git a/src/test/java/org/sonarsource/sonarlint/ls/mediumtests/AbstractLanguageServerMediumTests.java b/src/test/java/org/sonarsource/sonarlint/ls/mediumtests/AbstractLanguageServerMediumTests.java index d9d2f4ac0..9bac0d62f 100644 --- a/src/test/java/org/sonarsource/sonarlint/ls/mediumtests/AbstractLanguageServerMediumTests.java +++ b/src/test/java/org/sonarsource/sonarlint/ls/mediumtests/AbstractLanguageServerMediumTests.java @@ -46,7 +46,6 @@ import java.util.concurrent.atomic.AtomicInteger; import java.util.function.Function; import java.util.regex.Pattern; -import java.util.stream.Collectors; import java.util.stream.Stream; import javax.annotation.Nullable; import org.apache.commons.io.FileUtils; @@ -96,7 +95,6 @@ import org.junit.jupiter.api.extension.ExtendWith; import org.sonarsource.sonarlint.core.rpc.protocol.client.binding.AssistBindingParams; import org.sonarsource.sonarlint.core.rpc.protocol.client.binding.SuggestBindingParams; -import org.sonarsource.sonarlint.core.rpc.protocol.client.hotspot.HotspotDetailsDto; import org.sonarsource.sonarlint.ls.EnginesFactory; import org.sonarsource.sonarlint.ls.ServerMain; import org.sonarsource.sonarlint.ls.SonarLintExtendedLanguageClient; @@ -276,11 +274,11 @@ final void closeFiles() { for (var uri : notebooksToBeClosed) { lsProxy.getNotebookDocumentService().didClose(new DidCloseNotebookDocumentParams(new NotebookDocumentIdentifier(uri), List.of())); } - var changeWorkspaceFoldersParams = new DidChangeWorkspaceFoldersParams(); - var event = new WorkspaceFoldersChangeEvent(); - event.setRemoved(foldersToRemove.stream().map(WorkspaceFolder::new).collect(Collectors.toList())); - changeWorkspaceFoldersParams.setEvent(event); - lsProxy.getWorkspaceService().didChangeWorkspaceFolders(changeWorkspaceFoldersParams); + foldersToRemove.forEach(folderUri -> { + lsProxy.getWorkspaceService().didChangeWorkspaceFolders(new DidChangeWorkspaceFoldersParams( + new WorkspaceFoldersChangeEvent(List.of(), List.of(new WorkspaceFolder(folderUri))))); + awaitUntilAsserted(() -> assertLogContains("Configuration scope '" + folderUri + "' removed, clearing matched branch")); + }); instanceTempDirs.forEach(tempDirPath -> FileUtils.deleteQuietly(tempDirPath.toFile())); instanceTempDirs.clear(); } diff --git a/src/test/java/org/sonarsource/sonarlint/ls/mediumtests/ConnectedModeMediumTests.java b/src/test/java/org/sonarsource/sonarlint/ls/mediumtests/ConnectedModeMediumTests.java index a3a709177..598ca9ab1 100644 --- a/src/test/java/org/sonarsource/sonarlint/ls/mediumtests/ConnectedModeMediumTests.java +++ b/src/test/java/org/sonarsource/sonarlint/ls/mediumtests/ConnectedModeMediumTests.java @@ -25,7 +25,6 @@ import java.nio.file.Path; import java.nio.file.Paths; import java.time.Instant; -import java.util.ArrayList; import java.util.Collections; import java.util.HashMap; import java.util.List; @@ -53,7 +52,6 @@ import org.eclipse.lsp4j.jsonrpc.messages.Either; import org.jetbrains.annotations.NotNull; import org.junit.jupiter.api.AfterAll; -import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.BeforeAll; import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Disabled; @@ -110,7 +108,6 @@ class ConnectedModeMediumTests extends AbstractLanguageServerMediumTests { private static final String PROJECT_NAME2 = "Project Two"; private static final long CURRENT_TIME = System.currentTimeMillis(); private static Path folder1BaseDir; - private List addedConfigScopeIds; @BeforeAll public static void initialize() throws Exception { @@ -121,11 +118,7 @@ public static void initialize() throws Exception { "productVersion", "0.1", "productKey", "productKey"), new WorkspaceFolder(folder1BaseDir.toUri().toString(), "My Folder 1")); - } - @AfterEach - void removeBoundedScopes() { - addedConfigScopeIds.forEach(addedConfigScopeId -> lsProxy.didRemoveConfigurationScope(new SonarLintExtendedLanguageServer.DidRemoveConfigurationScopeParams(addedConfigScopeId))); } @BeforeEach @@ -198,8 +191,6 @@ public void mockSonarQube() { .setType(Common.BranchType.BRANCH) .build()) .build()); - - addedConfigScopeIds = new ArrayList<>(); } @NotNull @@ -240,7 +231,7 @@ public static void cleanUp() { void analysisConnected_find_hotspot() { mockNoIssuesNoHotspotsForProject(); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); var uriInFolder = folder1BaseDir.resolve("hotspot.py").toUri().toString(); didOpen(uriInFolder, "python", "IP_ADDRESS = '12.34.56.78'\n"); @@ -325,7 +316,7 @@ void analysisConnected_find_tracked_hotspot_before_sq_10_1() { .build() ); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); awaitUntilAsserted(() -> assertThat(client.logs).anyMatch(messageParams -> messageParams.getMessage().contains("Synchronizing project branches for project 'myProject'"))); lsProxy.didLocalBranchNameChange(new SonarLintExtendedLanguageServer.DidLocalBranchNameChangeParams(folder1BaseDir.toUri().toString(), "master")); @@ -401,7 +392,7 @@ void analysisConnected_find_tracked_hotspot_after_sq_10_1() { ); var uriInFolder = folder1BaseDir.resolve("hotspot.py").toUri().toString(); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); awaitUntilAsserted(() -> assertThat(client.logs).anyMatch(messageParams -> messageParams.getMessage().contains("Synchronizing project branches for project 'myProject'"))); lsProxy.didLocalBranchNameChange(new SonarLintExtendedLanguageServer.DidLocalBranchNameChangeParams(folder1BaseDir.toUri().toString(), "master")); @@ -443,7 +434,7 @@ void analysisConnected_scan_all_hotspot_then_forget() { List documents = List.of(doc1, doc2); var scanParams = new SonarLintExtendedLanguageServer.ScanFolderForHotspotsParams(folder1BaseDir.toUri().toString(), documents); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); awaitUntilAsserted(() -> assertThat(client.logs).anyMatch(messageParams -> messageParams.getMessage().contains("Synchronizing project branches for project 'myProject'"))); lsProxy.didLocalBranchNameChange(new SonarLintExtendedLanguageServer.DidLocalBranchNameChangeParams(folder1BaseDir.toUri().toString(), "master")); @@ -524,7 +515,7 @@ void analysisConnected_no_matching_server_issues() { .setQueryTimestamp(CURRENT_TIME) .build()); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); var uriInFolder = folder1BaseDir.resolve("inFolder.py").toUri().toString(); didOpen(uriInFolder, "python", "def foo():\n toto = 0\n plouf = 0\n"); @@ -536,10 +527,12 @@ void analysisConnected_no_matching_server_issues() { tuple(2, 2, 2, 7, PYTHON_S1481, "sonarlint", "Remove the unused local variable \"plouf\".", DiagnosticSeverity.Warning))); } - private void addConfigScope(String connectionId, String projectKey, String configScopeId) { - lsProxy.didAddConfigurationScopes(new SonarLintExtendedLanguageServer.DidAddConfigurationScopes(configScopeId, true, "someName", - connectionId, projectKey, false)); - addedConfigScopeIds.add(configScopeId); + private void addConfigScope(String configScopeId) { + lsProxy.getWorkspaceService() + .didChangeWorkspaceFolders( + new DidChangeWorkspaceFoldersParams( + new WorkspaceFoldersChangeEvent(List.of(new WorkspaceFolder(configScopeId)), Collections.emptyList()))); + foldersToRemove.add(configScopeId); awaitUntilAsserted(() -> assertThat(client) .satisfiesAnyOf( c -> c.scopeReadyForAnalysis.containsKey(configScopeId), @@ -590,8 +583,8 @@ void analysisConnected_matching_server_issues() throws Exception { mockWebServerExtension.addProtobufResponse("/api/issues/search.protobuf?statuses=OPEN,CONFIRMED,REOPENED,RESOLVED&types=VULNERABILITY&componentKeys=myProject&rules=&branch=master&ps=500&p=2", Issues.SearchWsResponse.newBuilder().addComponents(Issues.Component.newBuilder().setKey("componentKey").setPath("componentPath").build()).build()); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); - awaitUntilAsserted(() -> assertThat(client.logs).anyMatch(messageParams -> messageParams.getMessage().contains("Synchronizing project branches for project 'myProject'"))); + addConfigScope(folder1BaseDir.toUri().toString()); + awaitUntilAsserted(() -> assertLogContains("Synchronizing project branches for project 'myProject'")); lsProxy.didLocalBranchNameChange(new SonarLintExtendedLanguageServer.DidLocalBranchNameChangeParams(folder1BaseDir.toUri().toString(), "master")); var uriInFolder = folder1BaseDir.resolve("inFolder.py").toUri().toString(); @@ -649,7 +642,6 @@ void analysisConnected_matching_server_issues_on_sq_with_pull_issues() { .setQueryTimestamp(System.currentTimeMillis()) .build()); - addConfigScope(CONNECTION_ID, "myProject", CONFIG_SCOPE_ID); var uriInFolder = folder1BaseDir.resolve("pythonFile.py").toUri().toString(); didOpen(uriInFolder, "python", "def foo():\n toto = 0\n plouf = 0\n"); @@ -739,16 +731,22 @@ void shouldReturnErrorForInvalidUrl() { @Test void openHotspotInBrowserShouldLogIfBranchNotFound() { + lsProxy.getWorkspaceService() + .didChangeWorkspaceFolders( + new DidChangeWorkspaceFoldersParams( + new WorkspaceFoldersChangeEvent(List.of(new WorkspaceFolder(folder1BaseDir.toUri().toString())), Collections.emptyList()))); + foldersToRemove.add(folder1BaseDir.toUri().toString()); + lsProxy.openHotspotInBrowser(new SonarLintExtendedLanguageServer.OpenHotspotInBrowserLsParams("id", folder1BaseDir.toUri().toString())); - assertLogContains("Can't find branch for workspace folder " + folder1BaseDir.toUri().getPath() - + " during attempt to open hotspot in browser."); + awaitUntilAsserted(() -> assertLogContains("Can't find branch for workspace folder " + folder1BaseDir.toUri().getPath() + + " during attempt to open hotspot in browser.")); } @Test void shouldOpenHotspotDescription() { mockNoIssuesNoHotspotsForProject(); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); var uriInFolder = folder1BaseDir.resolve("hotspot.py").toUri().toString(); didOpen(uriInFolder, "python", "IP_ADDRESS = '12.34.56.78'\n"); awaitUntilAsserted(() -> assertThat(client.getHotspots(uriInFolder)).hasSize(1)); @@ -820,7 +818,7 @@ void shouldChangeIssueStatus() { mockWebServerExtension.addResponse("/api/issues/do_transition", new MockResponse().setResponseCode(200)); mockWebServerExtension.addResponse("/api/issues/add_comment", new MockResponse().setResponseCode(200)); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); awaitUntilAsserted(() -> assertThat(client.logs.stream().anyMatch(messageParams -> messageParams.getMessage().contains("Synchronizing project branches for project 'myProject'"))).isTrue()); lsProxy.didLocalBranchNameChange(new SonarLintExtendedLanguageServer.DidLocalBranchNameChangeParams(folder1BaseDir.toUri().toString(), "some/branch/name")); @@ -865,7 +863,7 @@ void shouldNotChangeStatusWhenServerIsDown() throws IOException { .build()); mockWebServerExtension.addResponse("/api/issues/do_transition", new MockResponse().setResponseCode(200)); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); lsProxy.didLocalBranchNameChange(new SonarLintExtendedLanguageServer.DidLocalBranchNameChangeParams(folder1BaseDir.toUri().toString(), "some/branch/name")); var fileUri = folder1BaseDir.resolve("changeIssueStatus.py").toUri().toString(); var content = "def foo():\n toto = 0\n plouf = 0\n"; @@ -946,7 +944,7 @@ void change_hotspot_status_to_resolved() { var uriInFolder = folder1BaseDir.resolve(analyzedFileName).toUri().toString(); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); awaitUntilAsserted(() -> assertThat(client.logs).anyMatch(messageParams -> messageParams.getMessage().contains("Synchronizing project branches for project 'myProject'"))); lsProxy.didLocalBranchNameChange(new SonarLintExtendedLanguageServer.DidLocalBranchNameChangeParams(folder1BaseDir.toUri().toString(), "master")); @@ -1040,7 +1038,7 @@ void change_hotspot_status_to_acknowledged() { var uriInFolder = folder1BaseDir.resolve(analyzedFileName).toUri().toString(); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); awaitUntilAsserted(() -> assertThat(client.logs).anyMatch(messageParams -> messageParams.getMessage().contains("Synchronizing project branches for project 'myProject'"))); lsProxy.didLocalBranchNameChange(new SonarLintExtendedLanguageServer.DidLocalBranchNameChangeParams(folder1BaseDir.toUri().toString(), "master")); @@ -1135,7 +1133,7 @@ void change_hotspot_status_permission_check() throws ExecutionException, Interru var uriInFolder = folder1BaseDir.resolve(analyzedFileName).toUri().toString(); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); awaitUntilAsserted(() -> assertThat(client.logs).anyMatch(messageParams -> messageParams.getMessage().contains("Synchronizing project branches for project 'myProject'"))); didOpen(uriInFolder, "python", "IP_ADDRESS = '12.34.56.78'\n"); @@ -1195,7 +1193,7 @@ void change_issue_status_permission_check() throws ExecutionException, Interrupt mockWebServerExtension.addResponse("/api/issues/do_transition", new MockResponse().setResponseCode(200)); mockWebServerExtension.addResponse("/api/issues/add_comment", new MockResponse().setResponseCode(200)); - + addConfigScope(folder1BaseDir.toUri().toString()); lsProxy.didLocalBranchNameChange(new SonarLintExtendedLanguageServer.DidLocalBranchNameChangeParams(folder1BaseDir.toUri().toString(), "some/branch/name")); var fileUri = folder1BaseDir.resolve("changeIssueStatus.py").toUri().toString(); var content = "def foo():\n toto = 0\n plouf = 0\n"; @@ -1219,7 +1217,8 @@ void change_issue_status_permission_check() throws ExecutionException, Interrupt @Test void change_issue_status_permission_check_exceptionally() throws ExecutionException, InterruptedException { - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); + awaitUntilAsserted(() -> assertThat(client.logs.stream().anyMatch(messageParams -> messageParams.getMessage().contains("Synchronizing project branches for project 'myProject'"))).isTrue()); var issueKey = "malformed issue UUID"; var result = lsProxy.checkIssueStatusChangePermitted(new SonarLintExtendedLanguageServer.CheckIssueStatusChangePermittedParams(folder1BaseDir.toUri().toString(), issueKey)).get(); @@ -1323,7 +1322,7 @@ private void assertLocalIssuesStatusChanged(String fileUri) { mockWebServerExtension.addResponse("/api/issues/anticipated_transitions?projectKey=myProject", new MockResponse().setResponseCode(200)); mockWebServerExtension.addResponse("/api/issues/add_comment", new MockResponse().setResponseCode(200)); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); awaitUntilAsserted(() -> assertThat(client.logs.stream().anyMatch(messageParams -> messageParams.getMessage().contains("Synchronizing project branches for project 'myProject'"))).isTrue()); lsProxy.didLocalBranchNameChange(new SonarLintExtendedLanguageServer.DidLocalBranchNameChangeParams(folder1BaseDir.toUri().toString(), "some/branch/name")); @@ -1398,7 +1397,7 @@ void shouldReportTaintIssues() { .setCreationDate(Instant.now().toEpochMilli()) .build()); - addConfigScope(CONNECTION_ID, "myProject", folder1BaseDir.toUri().toString()); + addConfigScope(folder1BaseDir.toUri().toString()); var content = "def foo():\n toto = 0\n plouf = 0\n"; didOpen(fileUri, "python", content); @@ -1408,15 +1407,12 @@ void shouldReportTaintIssues() { .contains(tuple(0, 1, 0, 2, "ruleKey", "Latest SonarQube Analysis", "message", DiagnosticSeverity.Warning))); } - @Test void should_automatically_suggest_bindings_to_client() throws Exception { client.suggestBindingLatch = new CountDownLatch(1); var basedir = Paths.get("path/to/base/auto-suggest").toAbsolutePath(); var workspaceUri = basedir.toUri().toString(); - addedConfigScopeIds.add(workspaceUri); var workspaceFolder = new WorkspaceFolder(workspaceUri); - getFolderSettings(workspaceUri); foldersToRemove.add(workspaceUri); lsProxy.getWorkspaceService().didChangeWorkspaceFolders(new DidChangeWorkspaceFoldersParams( new WorkspaceFoldersChangeEvent(List.of(workspaceFolder), Collections.emptyList()))); @@ -1428,17 +1424,15 @@ void should_automatically_suggest_bindings_to_client() throws Exception { assertThat(client.suggestedBindings.getSuggestions().get(workspaceUri)).isNotNull(); } - @Test void should_allow_client_to_explicitly_ask_for_binding_suggestions() throws ExecutionException, InterruptedException { - var basedir = Paths.get("path/to/base/explicit-request").toAbsolutePath(); - var workspaceUri = basedir.toUri().toString(); - addedConfigScopeIds.add(workspaceUri); + var workspaceUri = folder1BaseDir.resolve("foo-bar").toUri().toString(); var workspaceFolder = new WorkspaceFolder(workspaceUri, "foo-bar"); client.folderSettings = new HashMap<>(); client.folderSettings.put(workspaceUri, new HashMap<>()); lsProxy.getWorkspaceService().didChangeWorkspaceFolders(new DidChangeWorkspaceFoldersParams( new WorkspaceFoldersChangeEvent(List.of(workspaceFolder), Collections.emptyList()))); + foldersToRemove.add(workspaceUri); // Availability of binding suggestions for the added folder can take some time awaitUntilAsserted(() -> {