mirror of
https://github.com/golang/go
synced 2024-11-19 00:34:40 -07:00
f558378bf8
instead of driving the source pacakge directly, it indirects through the lsp protocol (the same way check does) We are normalizing on all the command lines doing this, so that server mode is more viable in the future. Change-Id: Ib5f2a059a44a5c60a53129c554e3cc14ca72c4a8 Reviewed-on: https://go-review.googlesource.com/c/tools/+/170577 Run-TryBot: Ian Cottrell <iancottrell@google.com> Reviewed-by: Rebecca Stambler <rstambler@golang.org>
98 lines
2.2 KiB
Go
98 lines
2.2 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 cmd
|
|
|
|
import (
|
|
"context"
|
|
"flag"
|
|
"fmt"
|
|
|
|
"golang.org/x/tools/internal/lsp/protocol"
|
|
"golang.org/x/tools/internal/span"
|
|
)
|
|
|
|
// check implements the check verb for gopls.
|
|
type check struct {
|
|
app *Application
|
|
}
|
|
|
|
type checkClient struct {
|
|
baseClient
|
|
diagnostics chan entry
|
|
}
|
|
|
|
type entry struct {
|
|
uri span.URI
|
|
diagnostics []protocol.Diagnostic
|
|
}
|
|
|
|
func (c *check) Name() string { return "check" }
|
|
func (c *check) Usage() string { return "<filename>" }
|
|
func (c *check) ShortHelp() string { return "show diagnostic results for the specified file" }
|
|
func (c *check) DetailedHelp(f *flag.FlagSet) {
|
|
fmt.Fprint(f.Output(), `
|
|
Example: show the diagnostic results of this file:
|
|
|
|
$ gopls check internal/lsp/cmd/check.go
|
|
|
|
gopls check flags are:
|
|
`)
|
|
f.PrintDefaults()
|
|
}
|
|
|
|
// Run performs the check on the files specified by args and prints the
|
|
// results to stdout.
|
|
func (c *check) Run(ctx context.Context, args ...string) error {
|
|
if len(args) == 0 {
|
|
// no files, so no results
|
|
return nil
|
|
}
|
|
client := &checkClient{
|
|
diagnostics: make(chan entry),
|
|
}
|
|
checking := map[span.URI]*protocol.ColumnMapper{}
|
|
// now we ready to kick things off
|
|
_, err := c.app.connect(ctx, client)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
for _, arg := range args {
|
|
uri := span.FileURI(arg)
|
|
m, err := client.AddFile(ctx, uri)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
checking[uri] = m
|
|
}
|
|
// now wait for results
|
|
for entry := range client.diagnostics {
|
|
//TODO:timeout?
|
|
m, found := checking[entry.uri]
|
|
if !found {
|
|
continue
|
|
}
|
|
for _, d := range entry.diagnostics {
|
|
spn, err := m.RangeSpan(d.Range)
|
|
if err != nil {
|
|
return fmt.Errorf("Could not convert position %v for %q", d.Range, d.Message)
|
|
}
|
|
fmt.Printf("%v: %v\n", spn, d.Message)
|
|
}
|
|
delete(checking, entry.uri)
|
|
if len(checking) == 0 {
|
|
return nil
|
|
}
|
|
}
|
|
return fmt.Errorf("did not get all results")
|
|
}
|
|
|
|
func (c *checkClient) PublishDiagnostics(ctx context.Context, p *protocol.PublishDiagnosticsParams) error {
|
|
c.diagnostics <- entry{
|
|
uri: span.URI(p.URI),
|
|
diagnostics: p.Diagnostics,
|
|
}
|
|
return nil
|
|
}
|