109 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Go
		
	
	
	
			
		
		
	
	
			109 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Go
		
	
	
	
| // Copyright 2018 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"
 | |
| 	"fmt"
 | |
| 	"strings"
 | |
| 
 | |
| 	"golang.org/x/tools/internal/lsp/protocol"
 | |
| 	"golang.org/x/tools/internal/lsp/source"
 | |
| 	"golang.org/x/tools/internal/span"
 | |
| )
 | |
| 
 | |
| func (s *Server) codeAction(ctx context.Context, params *protocol.CodeActionParams) ([]protocol.CodeAction, error) {
 | |
| 	uri := span.NewURI(params.TextDocument.URI)
 | |
| 	view := s.session.ViewOf(uri)
 | |
| 	_, m, err := getSourceFile(ctx, view, uri)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	spn, err := m.RangeSpan(params.Range)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	var codeActions []protocol.CodeAction
 | |
| 	// TODO(rstambler): Handle params.Context.Only when VSCode-Go uses a
 | |
| 	// version of vscode-languageclient that fixes
 | |
| 	// https://github.com/Microsoft/vscode-languageserver-node/issues/442.
 | |
| 	edits, err := organizeImports(ctx, view, spn)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	if len(edits) > 0 {
 | |
| 		codeActions = append(codeActions, protocol.CodeAction{
 | |
| 			Title: "Organize Imports",
 | |
| 			Kind:  protocol.SourceOrganizeImports,
 | |
| 			Edit: &protocol.WorkspaceEdit{
 | |
| 				Changes: &map[string][]protocol.TextEdit{
 | |
| 					string(spn.URI()): edits,
 | |
| 				},
 | |
| 			},
 | |
| 		})
 | |
| 		// If we also have diagnostics, we can associate them with quick fixes.
 | |
| 		if findImportErrors(params.Context.Diagnostics) {
 | |
| 			// TODO(rstambler): Separate this into a set of codeActions per diagnostic,
 | |
| 			// where each action is the addition or removal of one import.
 | |
| 			// This can only be done when https://golang.org/issue/31493 is resolved.
 | |
| 			codeActions = append(codeActions, protocol.CodeAction{
 | |
| 				Title: "Organize All Imports", // clarify that all imports will change
 | |
| 				Kind:  protocol.QuickFix,
 | |
| 				Edit: &protocol.WorkspaceEdit{
 | |
| 					Changes: &map[string][]protocol.TextEdit{
 | |
| 						string(uri): edits,
 | |
| 					},
 | |
| 				},
 | |
| 			})
 | |
| 		}
 | |
| 	}
 | |
| 	return codeActions, nil
 | |
| }
 | |
| 
 | |
| func organizeImports(ctx context.Context, v source.View, s span.Span) ([]protocol.TextEdit, error) {
 | |
| 	f, m, err := getGoFile(ctx, v, s.URI())
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	rng, err := s.Range(m.Converter)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	if rng.Start == rng.End {
 | |
| 		// If we have a single point, assume we want the whole file.
 | |
| 		tok := f.GetToken(ctx)
 | |
| 		if tok == nil {
 | |
| 			return nil, fmt.Errorf("no file information for %s", f.URI())
 | |
| 		}
 | |
| 		rng.End = tok.Pos(tok.Size())
 | |
| 	}
 | |
| 	edits, err := source.Imports(ctx, f, rng)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	return ToProtocolEdits(m, edits)
 | |
| }
 | |
| 
 | |
| // findImports determines if a given diagnostic represents an error that could
 | |
| // be fixed by organizing imports.
 | |
| // TODO(rstambler): We need a better way to check this than string matching.
 | |
| func findImportErrors(diagnostics []protocol.Diagnostic) bool {
 | |
| 	for _, diagnostic := range diagnostics {
 | |
| 		// "undeclared name: X" may be an unresolved import.
 | |
| 		if strings.HasPrefix(diagnostic.Message, "undeclared name: ") {
 | |
| 			return true
 | |
| 		}
 | |
| 		// "could not import: X" may be an invalid import.
 | |
| 		if strings.HasPrefix(diagnostic.Message, "could not import: ") {
 | |
| 			return true
 | |
| 		}
 | |
| 		// "X imported but not used" is an unused import.
 | |
| 		if strings.HasSuffix(diagnostic.Message, " imported but not used") {
 | |
| 			return true
 | |
| 		}
 | |
| 	}
 | |
| 	return false
 | |
| }
 |