From 5974258e689a4f8a93448a0d181737afa4506e3f Mon Sep 17 00:00:00 2001 From: Robert Findley Date: Fri, 26 May 2023 12:59:48 -0400 Subject: [PATCH] gopls/internal/lsp: clear vuln diagnostics on config changes Clear vuln diagnostics before diagnosing views, during handling of config changes. This should mostly avoid the symptoms of golang/go#60465, but as noted in the code is not a proper fix. Unfortunately there is no way to write a test for this behavior that is not flaky, because the operation itself is flaky. I have scheduled a proper fix for gopls@v0.12.1. Fixes golang/go#60465 Change-Id: If41f5420c24dfa15a7d83e89988488619a2dd857 Reviewed-on: https://go-review.googlesource.com/c/tools/+/498560 gopls-CI: kokoro Run-TryBot: Robert Findley Reviewed-by: Alan Donovan TryBot-Result: Gopher Robot --- gopls/internal/lsp/cache/snapshot.go | 4 ++++ gopls/internal/lsp/workspace.go | 22 +++++++++++++++++++++- 2 files changed, 25 insertions(+), 1 deletion(-) diff --git a/gopls/internal/lsp/cache/snapshot.go b/gopls/internal/lsp/cache/snapshot.go index de9524bf0ae..7e0a9ba196b 100644 --- a/gopls/internal/lsp/cache/snapshot.go +++ b/gopls/internal/lsp/cache/snapshot.go @@ -2156,6 +2156,10 @@ func (s *snapshot) clone(ctx, bgCtx context.Context, changes map[span.URI]*fileC if !change.exists { result.files.Delete(uri) } else { + // TODO(golang/go#57558): the line below is strictly necessary to ensure + // that snapshots have each overlay, but it is problematic that we must + // set any content in snapshot.clone: if the file has changed, let it be + // re-read. result.files.Set(uri, change.fileHandle) } diff --git a/gopls/internal/lsp/workspace.go b/gopls/internal/lsp/workspace.go index 818135e94a2..e5f813e730c 100644 --- a/gopls/internal/lsp/workspace.go +++ b/gopls/internal/lsp/workspace.go @@ -61,10 +61,30 @@ func (s *Server) didChangeConfiguration(ctx context.Context, _ *protocol.DidChan if err := s.fetchConfig(ctx, view.Name(), view.Folder(), options); err != nil { return err } - view, err := s.session.SetViewOptions(ctx, view, options) + _, err := s.session.SetViewOptions(ctx, view, options) if err != nil { return err } + } + + // Now that all views have been updated: reset vulncheck diagnostics, rerun + // diagnostics, and hope for the best... + // + // TODO(golang/go#60465): this not a reliable way to ensure the correctness + // of the resulting diagnostics below. A snapshot could still be in the + // process of diagnosing the workspace, and not observe the configuration + // changes above. + // + // The real fix is golang/go#42814: we should create a new snapshot on any + // change that could affect the derived results in that snapshot. However, we + // are currently (2023-05-26) on the verge of a release, and the proper fix + // is too risky a change. Since in the common case a configuration change is + // only likely to occur during a period of quiescence on the server, it is + // likely that the clearing below will have the desired effect. + s.clearDiagnosticSource(modVulncheckSource) + + for _, view := range s.session.Views() { + view := view go func() { snapshot, release, err := view.Snapshot() if err != nil {