|  | // 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" | 
|  | "slices" | 
|  |  | 
|  | "golang.org/x/tools/gopls/internal/protocol" | 
|  | "golang.org/x/tools/gopls/internal/settings" | 
|  | ) | 
|  |  | 
|  | // check implements the check verb for gopls. | 
|  | type check struct { | 
|  | app      *Application | 
|  | Severity string `flag:"severity" help:"minimum diagnostic severity (hint, info, warning, or error)"` | 
|  | } | 
|  |  | 
|  | func (c *check) Name() string      { return "check" } | 
|  | func (c *check) Parent() string    { return c.app.Name() } | 
|  | 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/cmd/check.go | 
|  | `) | 
|  | printFlagDefaults(f) | 
|  | } | 
|  |  | 
|  | // 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 { | 
|  | severityCutoff := protocol.SeverityWarning | 
|  | switch c.Severity { | 
|  | case "hint": | 
|  | severityCutoff = protocol.SeverityHint | 
|  | case "info": | 
|  | severityCutoff = protocol.SeverityInformation | 
|  | case "warning": | 
|  | // default | 
|  | case "error": | 
|  | severityCutoff = protocol.SeverityError | 
|  | default: | 
|  | return fmt.Errorf("unrecognized -severity value %q", c.Severity) | 
|  | } | 
|  |  | 
|  | if len(args) == 0 { | 
|  | return nil | 
|  | } | 
|  |  | 
|  | // TODO(adonovan): formally, we are required to set this | 
|  | // option if we want RelatedInformation, but it appears to | 
|  | // have no effect on the server, even though the default is | 
|  | // false. Investigate. | 
|  | origOptions := c.app.options | 
|  | c.app.options = func(opts *settings.Options) { | 
|  | if origOptions != nil { | 
|  | origOptions(opts) | 
|  | } | 
|  | opts.RelatedInformationSupported = true | 
|  | } | 
|  |  | 
|  | cli, _, err := c.app.connect(ctx) | 
|  | if err != nil { | 
|  | return err | 
|  | } | 
|  | defer cli.terminate(ctx) | 
|  |  | 
|  | // Open and diagnose the requested files. | 
|  | var ( | 
|  | uris     []protocol.DocumentURI | 
|  | checking = make(map[protocol.DocumentURI]*cmdFile) | 
|  | ) | 
|  | for _, arg := range args { | 
|  | uri := protocol.URIFromPath(arg) | 
|  | uris = append(uris, uri) | 
|  | file, err := cli.openFile(ctx, uri) | 
|  | if err != nil { | 
|  | return err | 
|  | } | 
|  | checking[uri] = file | 
|  | } | 
|  | if err := diagnoseFiles(ctx, cli.server, uris); err != nil { | 
|  | return err | 
|  | } | 
|  |  | 
|  | // print prints a single element of a diagnostic. | 
|  | print := func(uri protocol.DocumentURI, rng protocol.Range, message string) error { | 
|  | file, err := cli.openFile(ctx, uri) | 
|  | if err != nil { | 
|  | return err | 
|  | } | 
|  | spn, err := file.rangeSpan(rng) | 
|  | if err != nil { | 
|  | return fmt.Errorf("could not convert position %v for %q", rng, message) | 
|  | } | 
|  | fmt.Printf("%v: %v\n", spn, message) | 
|  | return nil | 
|  | } | 
|  |  | 
|  | for _, file := range checking { | 
|  | file.diagnosticsMu.Lock() | 
|  | diags := slices.Clone(file.diagnostics) | 
|  | file.diagnosticsMu.Unlock() | 
|  |  | 
|  | for _, diag := range diags { | 
|  | if diag.Severity > severityCutoff { // lower severity value => greater severity, counterintuitively | 
|  | continue | 
|  | } | 
|  | if err := print(file.uri, diag.Range, diag.Message); err != nil { | 
|  | return err | 
|  | } | 
|  | for _, rel := range diag.RelatedInformation { | 
|  | if err := print(rel.Location.URI, rel.Location.Range, "- "+rel.Message); err != nil { | 
|  | return err | 
|  | } | 
|  | } | 
|  |  | 
|  | } | 
|  | } | 
|  | return nil | 
|  | } |