blob: b5da5605391f88c34d9d20afd3edfad8b6b3a8c2 [file] [log] [blame]
// 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 mod provides core features related to go.mod file
// handling for use by Go editors and tools.
package mod
import (
"context"
"fmt"
"golang.org/x/tools/internal/event"
"golang.org/x/tools/internal/lsp/debug/tag"
"golang.org/x/tools/internal/lsp/protocol"
"golang.org/x/tools/internal/lsp/source"
)
func Diagnostics(ctx context.Context, snapshot source.Snapshot) (map[source.VersionedFileIdentity][]*source.Diagnostic, error) {
ctx, done := event.Start(ctx, "mod.Diagnostics", tag.Snapshot.Of(snapshot.ID()))
defer done()
reports := map[source.VersionedFileIdentity][]*source.Diagnostic{}
for _, uri := range snapshot.ModFiles() {
fh, err := snapshot.GetVersionedFile(ctx, uri)
if err != nil {
return nil, err
}
reports[fh.VersionedFileIdentity()] = []*source.Diagnostic{}
errors, err := ErrorsForMod(ctx, snapshot, fh)
if err != nil {
return nil, err
}
for _, e := range errors {
d := &source.Diagnostic{
Message: e.Message,
Range: e.Range,
Source: e.Category,
}
switch {
case e.Category == "syntax", e.Kind == source.ListError:
d.Severity = protocol.SeverityError
case e.Kind == source.UpgradeNotification:
d.Severity = protocol.SeverityInformation
default:
d.Severity = protocol.SeverityWarning
}
fh, err := snapshot.GetVersionedFile(ctx, e.URI)
if err != nil {
return nil, err
}
reports[fh.VersionedFileIdentity()] = append(reports[fh.VersionedFileIdentity()], d)
}
}
return reports, nil
}
func ErrorsForMod(ctx context.Context, snapshot source.Snapshot, fh source.FileHandle) ([]*source.Error, error) {
pm, err := snapshot.ParseMod(ctx, fh)
if err != nil {
if pm == nil || len(pm.ParseErrors) == 0 {
return nil, err
}
return pm.ParseErrors, nil
}
var errors []*source.Error
// Add upgrade quick fixes for individual modules if we know about them.
upgrades := snapshot.View().ModuleUpgrades()
for _, req := range pm.File.Require {
ver, ok := upgrades[req.Mod.Path]
if !ok || req.Mod.Version == ver {
continue
}
rng, err := lineToRange(pm.Mapper, fh.URI(), req.Syntax.Start, req.Syntax.End)
if err != nil {
return nil, err
}
// Upgrade to the exact version we offer the user, not the most recent.
args, err := source.MarshalArgs(fh.URI(), false, []string{req.Mod.Path + "@" + ver})
if err != nil {
return nil, err
}
errors = append(errors, &source.Error{
URI: fh.URI(),
Range: rng,
Kind: source.UpgradeNotification,
Message: fmt.Sprintf("%v can be upgraded", req.Mod.Path),
SuggestedFixes: []source.SuggestedFix{{
Title: fmt.Sprintf("Upgrade to %v", ver),
Command: &protocol.Command{
Title: fmt.Sprintf("Upgrade to %v", ver),
Command: source.CommandUpgradeDependency.ID(),
Arguments: args,
},
}},
})
}
tidied, err := snapshot.ModTidy(ctx, pm)
if source.IsNonFatalGoModError(err) {
return errors, nil
}
if err != nil {
return nil, err
}
errors = append(errors, tidied.Errors...)
return errors, nil
}