// Copyright 2019 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. package lsp import ( "context" "golang.org/x/tools/internal/lsp/protocol" "golang.org/x/tools/internal/lsp/source" "golang.org/x/tools/internal/span" errors "golang.org/x/xerrors" ) func (s *Server) didChangeWorkspaceFolders(ctx context.Context, params *protocol.DidChangeWorkspaceFoldersParams) error { event := params.Event for _, folder := range event.Removed { view := s.session.View(folder.Name) if view != nil { view.Shutdown(ctx) } else { return errors.Errorf("view %s for %v not found", folder.Name, folder.URI) } } return s.addFolders(ctx, event.Added) } func (s *Server) addView(ctx context.Context, name string, uri span.URI) (source.Snapshot, func(), error) { s.stateMu.Lock() state := s.state s.stateMu.Unlock() if state < serverInitialized { return nil, func() {}, errors.Errorf("addView called before server initialized") } options := s.session.Options().Clone() if err := s.fetchConfig(ctx, name, uri, options); err != nil { return nil, func() {}, err } _, snapshot, release, err := s.session.NewView(ctx, name, uri, options) return snapshot, release, err } func (s *Server) didChangeConfiguration(ctx context.Context, _ *protocol.DidChangeConfigurationParams) error { // Apply any changes to the session-level settings. options := s.session.Options().Clone() semanticTokensRegistered := options.SemanticTokens if err := s.fetchConfig(ctx, "", "", options); err != nil { return err } s.session.SetOptions(options) // Go through each view, getting and updating its configuration. for _, view := range s.session.Views() { options := s.session.Options().Clone() if err := s.fetchConfig(ctx, view.Name(), view.Folder(), options); err != nil { return err } view, err := view.SetOptions(ctx, options) if err != nil { return err } go func() { snapshot, release := view.Snapshot(ctx) defer release() s.diagnoseDetached(snapshot) }() } // Update any session-specific registrations or unregistrations. if !semanticTokensRegistered && options.SemanticTokens { if err := s.client.RegisterCapability(ctx, &protocol.RegistrationParams{ Registrations: semanticTokenRegistrations(), }); err != nil { return err } } else if semanticTokensRegistered && !options.SemanticTokens { var unregistrations []protocol.Unregistration for _, r := range semanticTokenRegistrations() { unregistrations = append(unregistrations, protocol.Unregistration{ ID: r.ID, Method: r.Method, }) } if err := s.client.UnregisterCapability(ctx, &protocol.UnregistrationParams{ Unregisterations: unregistrations, }); err != nil { return err } } return nil } func semanticTokenRegistrations() []protocol.Registration { var registrations []protocol.Registration for _, method := range []string{ "textDocument/semanticTokens/full", "textDocument/semanticTokens/full/delta", "textDocument/semanticTokens/range", } { registrations = append(registrations, protocol.Registration{ ID: method, Method: method, RegisterOptions: &protocol.SemanticTokensOptions{ Legend: protocol.SemanticTokensLegend{ // TODO(pjw): trim these to what we use (and an unused one // at position 0 of TokTypes, to catch typos) TokenTypes: SemanticMemo.tokTypes, TokenModifiers: SemanticMemo.tokMods, }, }, }) } return registrations }