| // 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" |
| "io/ioutil" |
| "os" |
| |
| "golang.org/x/tools/internal/lsp/source" |
| "golang.org/x/tools/internal/lsp/telemetry" |
| "golang.org/x/tools/internal/span" |
| "golang.org/x/tools/internal/telemetry/trace" |
| ) |
| |
| // ioLimit limits the number of parallel file reads per process. |
| var ioLimit = make(chan struct{}, 128) |
| |
| // nativeFileSystem implements FileSystem reading from the normal os file system. |
| type nativeFileSystem struct{} |
| |
| // nativeFileHandle implements FileHandle for nativeFileSystem |
| type nativeFileHandle struct { |
| fs *nativeFileSystem |
| identity source.FileIdentity |
| } |
| |
| func (fs *nativeFileSystem) GetFile(uri span.URI, kind source.FileKind) source.FileHandle { |
| version := "DOES NOT EXIST" |
| if fi, err := os.Stat(uri.Filename()); err == nil { |
| version = fi.ModTime().String() |
| } |
| return &nativeFileHandle{ |
| fs: fs, |
| identity: source.FileIdentity{ |
| URI: uri, |
| Version: version, |
| Kind: kind, |
| }, |
| } |
| } |
| |
| func (h *nativeFileHandle) FileSystem() source.FileSystem { |
| return h.fs |
| } |
| |
| func (h *nativeFileHandle) Identity() source.FileIdentity { |
| return h.identity |
| } |
| |
| func (h *nativeFileHandle) Read(ctx context.Context) ([]byte, string, error) { |
| ctx, done := trace.StartSpan(ctx, "cache.nativeFileHandle.Read", telemetry.File.Of(h.identity.URI.Filename())) |
| _ = ctx |
| defer done() |
| |
| ioLimit <- struct{}{} |
| defer func() { <-ioLimit }() |
| // TODO: this should fail if the version is not the same as the handle |
| data, err := ioutil.ReadFile(h.identity.URI.Filename()) |
| if err != nil { |
| return nil, "", err |
| } |
| return data, hashContents(data), nil |
| } |