mirror of
https://github.com/golang/go
synced 2024-11-18 22:14:56 -07:00
56463cc14b
Created an analogous data structure for go.mod files when we parse them using the golang.org/x/mod package. Gopls can now access the data within a go.mod file using a parseModHandle and the corresponding parseModData object. This will help down the road when it is time to implement the lsp functions for go.mod files. Updates golang/go#31999 Change-Id: Ibd4d64569bbe3df61b203490b63399d479e7d794 Reviewed-on: https://go-review.googlesource.com/c/tools/+/211303 Run-TryBot: Rohan Challa <rohan@golang.org> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Rebecca Stambler <rstambler@golang.org>
77 lines
1.7 KiB
Go
77 lines
1.7 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 cache
|
|
|
|
import (
|
|
"context"
|
|
|
|
"golang.org/x/mod/modfile"
|
|
"golang.org/x/tools/internal/lsp/source"
|
|
"golang.org/x/tools/internal/lsp/telemetry"
|
|
"golang.org/x/tools/internal/memoize"
|
|
"golang.org/x/tools/internal/telemetry/trace"
|
|
errors "golang.org/x/xerrors"
|
|
)
|
|
|
|
type parseModHandle struct {
|
|
handle *memoize.Handle
|
|
file source.FileHandle
|
|
}
|
|
|
|
type parseModData struct {
|
|
memoize.NoCopy
|
|
|
|
modfile *modfile.File
|
|
err error
|
|
}
|
|
|
|
func (c *cache) ParseModHandle(fh source.FileHandle) source.ParseModHandle {
|
|
key := parseKey{
|
|
file: fh.Identity(),
|
|
mode: source.ParseFull,
|
|
}
|
|
h := c.store.Bind(key, func(ctx context.Context) interface{} {
|
|
data := &parseModData{}
|
|
data.modfile, data.err = parseMod(ctx, fh)
|
|
return data
|
|
})
|
|
return &parseModHandle{
|
|
handle: h,
|
|
file: fh,
|
|
}
|
|
}
|
|
|
|
func parseMod(ctx context.Context, fh source.FileHandle) (modifle *modfile.File, err error) {
|
|
ctx, done := trace.StartSpan(ctx, "cache.parseMod", telemetry.File.Of(fh.Identity().URI.Filename()))
|
|
defer done()
|
|
|
|
buf, _, err := fh.Read(ctx)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
f, err := modfile.Parse(fh.Identity().URI.Filename(), buf, nil)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return f, nil
|
|
}
|
|
|
|
func (pgh *parseModHandle) String() string {
|
|
return pgh.File().Identity().URI.Filename()
|
|
}
|
|
|
|
func (pgh *parseModHandle) File() source.FileHandle {
|
|
return pgh.file
|
|
}
|
|
|
|
func (pgh *parseModHandle) Parse(ctx context.Context) (*modfile.File, error) {
|
|
v := pgh.handle.Get(ctx)
|
|
if v == nil {
|
|
return nil, errors.Errorf("no parsed file for %s", pgh.File().Identity().URI)
|
|
}
|
|
data := v.(*parseModData)
|
|
return data.modfile, data.err
|
|
}
|