199 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			199 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			Go
		
	
	
	
| // 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 (
 | |
| 	"bytes"
 | |
| 	"context"
 | |
| 	"fmt"
 | |
| 	"os"
 | |
| 	"path"
 | |
| 
 | |
| 	"golang.org/x/tools/internal/jsonrpc2"
 | |
| 	"golang.org/x/tools/internal/lsp/protocol"
 | |
| 	"golang.org/x/tools/internal/lsp/source"
 | |
| 	"golang.org/x/tools/internal/span"
 | |
| )
 | |
| 
 | |
| func (s *Server) initialize(ctx context.Context, params *protocol.InitializeParams) (*protocol.InitializeResult, error) {
 | |
| 	s.initializedMu.Lock()
 | |
| 	defer s.initializedMu.Unlock()
 | |
| 	if s.isInitialized {
 | |
| 		return nil, jsonrpc2.NewErrorf(jsonrpc2.CodeInvalidRequest, "server already initialized")
 | |
| 	}
 | |
| 	s.isInitialized = true // mark server as initialized now
 | |
| 
 | |
| 	// TODO(iancottrell): Change this default to protocol.Incremental and remove the option
 | |
| 	s.textDocumentSyncKind = protocol.Full
 | |
| 	if opts, ok := params.InitializationOptions.(map[string]interface{}); ok {
 | |
| 		if opt, ok := opts["incrementalSync"].(bool); ok && opt {
 | |
| 			s.textDocumentSyncKind = protocol.Incremental
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	s.setClientCapabilities(params.Capabilities)
 | |
| 
 | |
| 	folders := params.WorkspaceFolders
 | |
| 	if len(folders) == 0 {
 | |
| 		if params.RootURI != "" {
 | |
| 			folders = []protocol.WorkspaceFolder{{
 | |
| 				URI:  params.RootURI,
 | |
| 				Name: path.Base(params.RootURI),
 | |
| 			}}
 | |
| 		} else {
 | |
| 			// no folders and no root, single file mode
 | |
| 			//TODO(iancottrell): not sure how to do single file mode yet
 | |
| 			//issue: golang.org/issue/31168
 | |
| 			return nil, fmt.Errorf("single file mode not supported yet")
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	for _, folder := range folders {
 | |
| 		if err := s.addView(ctx, folder.Name, span.NewURI(folder.URI)); err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return &protocol.InitializeResult{
 | |
| 		Capabilities: protocol.ServerCapabilities{
 | |
| 			CodeActionProvider: true,
 | |
| 			CompletionProvider: &protocol.CompletionOptions{
 | |
| 				TriggerCharacters: []string{"."},
 | |
| 			},
 | |
| 			DefinitionProvider:         true,
 | |
| 			DocumentFormattingProvider: true,
 | |
| 			DocumentSymbolProvider:     true,
 | |
| 			HoverProvider:              true,
 | |
| 			DocumentHighlightProvider:  true,
 | |
| 			DocumentLinkProvider:       &protocol.DocumentLinkOptions{},
 | |
| 			SignatureHelpProvider: &protocol.SignatureHelpOptions{
 | |
| 				TriggerCharacters: []string{"(", ","},
 | |
| 			},
 | |
| 			TextDocumentSync: &protocol.TextDocumentSyncOptions{
 | |
| 				Change:    s.textDocumentSyncKind,
 | |
| 				OpenClose: true,
 | |
| 			},
 | |
| 			TypeDefinitionProvider: true,
 | |
| 			Workspace: &struct {
 | |
| 				WorkspaceFolders *struct {
 | |
| 					Supported           bool   "json:\"supported,omitempty\""
 | |
| 					ChangeNotifications string "json:\"changeNotifications,omitempty\""
 | |
| 				} "json:\"workspaceFolders,omitempty\""
 | |
| 			}{
 | |
| 				WorkspaceFolders: &struct {
 | |
| 					Supported           bool   "json:\"supported,omitempty\""
 | |
| 					ChangeNotifications string "json:\"changeNotifications,omitempty\""
 | |
| 				}{
 | |
| 					Supported:           true,
 | |
| 					ChangeNotifications: "workspace/didChangeWorkspaceFolders",
 | |
| 				},
 | |
| 			},
 | |
| 		},
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| func (s *Server) setClientCapabilities(caps protocol.ClientCapabilities) {
 | |
| 	// Check if the client supports snippets in completion items.
 | |
| 	s.insertTextFormat = protocol.PlainTextTextFormat
 | |
| 	if caps.TextDocument.Completion.CompletionItem.SnippetSupport {
 | |
| 		s.insertTextFormat = protocol.SnippetTextFormat
 | |
| 	}
 | |
| 	// Check if the client supports configuration messages.
 | |
| 	s.configurationSupported = caps.Workspace.Configuration
 | |
| 	s.dynamicConfigurationSupported = caps.Workspace.DidChangeConfiguration.DynamicRegistration
 | |
| 
 | |
| 	// Check which types of content format are supported by this client.
 | |
| 	s.preferredContentFormat = protocol.PlainText
 | |
| 	if len(caps.TextDocument.Hover.ContentFormat) > 0 {
 | |
| 		s.preferredContentFormat = caps.TextDocument.Hover.ContentFormat[0]
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (s *Server) initialized(ctx context.Context, params *protocol.InitializedParams) error {
 | |
| 	if s.configurationSupported {
 | |
| 		if s.dynamicConfigurationSupported {
 | |
| 			s.client.RegisterCapability(ctx, &protocol.RegistrationParams{
 | |
| 				Registrations: []protocol.Registration{{
 | |
| 					ID:     "workspace/didChangeConfiguration",
 | |
| 					Method: "workspace/didChangeConfiguration",
 | |
| 				}, {
 | |
| 					ID:     "workspace/didChangeWorkspaceFolders",
 | |
| 					Method: "workspace/didChangeWorkspaceFolders",
 | |
| 				}},
 | |
| 			})
 | |
| 		}
 | |
| 		for _, view := range s.session.Views() {
 | |
| 			config, err := s.client.Configuration(ctx, &protocol.ConfigurationParams{
 | |
| 				Items: []protocol.ConfigurationItem{{
 | |
| 					ScopeURI: protocol.NewURI(view.Folder()),
 | |
| 					Section:  "gopls",
 | |
| 				}},
 | |
| 			})
 | |
| 			if err != nil {
 | |
| 				return err
 | |
| 			}
 | |
| 			if err := s.processConfig(view, config[0]); err != nil {
 | |
| 				return err
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 	buf := &bytes.Buffer{}
 | |
| 	PrintVersionInfo(buf, true, false)
 | |
| 	s.session.Logger().Infof(ctx, "%s", buf)
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (s *Server) processConfig(view source.View, config interface{}) error {
 | |
| 	// TODO: We should probably store and process more of the config.
 | |
| 	if config == nil {
 | |
| 		return nil // ignore error if you don't have a config
 | |
| 	}
 | |
| 	c, ok := config.(map[string]interface{})
 | |
| 	if !ok {
 | |
| 		return fmt.Errorf("invalid config gopls type %T", config)
 | |
| 	}
 | |
| 	// Get the environment for the go/packages config.
 | |
| 	if env := c["env"]; env != nil {
 | |
| 		menv, ok := env.(map[string]interface{})
 | |
| 		if !ok {
 | |
| 			return fmt.Errorf("invalid config gopls.env type %T", env)
 | |
| 		}
 | |
| 		env := view.Env()
 | |
| 		for k, v := range menv {
 | |
| 			env = append(env, fmt.Sprintf("%s=%s", k, v))
 | |
| 		}
 | |
| 		view.SetEnv(env)
 | |
| 	}
 | |
| 	// Check if placeholders are enabled.
 | |
| 	if usePlaceholders, ok := c["usePlaceholders"].(bool); ok {
 | |
| 		s.usePlaceholders = usePlaceholders
 | |
| 	}
 | |
| 	// Check if user has disabled documentation on hover.
 | |
| 	if noDocsOnHover, ok := c["noDocsOnHover"].(bool); ok {
 | |
| 		s.noDocsOnHover = noDocsOnHover
 | |
| 	}
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (s *Server) shutdown(ctx context.Context) error {
 | |
| 	s.initializedMu.Lock()
 | |
| 	defer s.initializedMu.Unlock()
 | |
| 	if !s.isInitialized {
 | |
| 		return jsonrpc2.NewErrorf(jsonrpc2.CodeInvalidRequest, "server not initialized")
 | |
| 	}
 | |
| 	// drop all the active views
 | |
| 	s.session.Shutdown(ctx)
 | |
| 	s.isInitialized = false
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| func (s *Server) exit(ctx context.Context) error {
 | |
| 	if s.isInitialized {
 | |
| 		os.Exit(1)
 | |
| 	}
 | |
| 	os.Exit(0)
 | |
| 	return nil
 | |
| }
 |