blob: 90471ed44014a973797faaeb78f6279de1e5c4ec [file] [log] [blame]
// Copyright 2020 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"
"encoding/json"
"io/ioutil"
"os"
"path/filepath"
"testing"
"time"
"github.com/google/go-cmp/cmp"
"golang.org/x/tools/gopls/internal/govulncheck"
"golang.org/x/tools/gopls/internal/lsp/fake"
"golang.org/x/tools/gopls/internal/lsp/source"
"golang.org/x/tools/gopls/internal/span"
)
func TestCaseInsensitiveFilesystem(t *testing.T) {
base, err := ioutil.TempDir("", t.Name())
if err != nil {
t.Fatal(err)
}
inner := filepath.Join(base, "a/B/c/DEFgh")
if err := os.MkdirAll(inner, 0777); err != nil {
t.Fatal(err)
}
file := filepath.Join(inner, "f.go")
if err := ioutil.WriteFile(file, []byte("hi"), 0777); err != nil {
t.Fatal(err)
}
if _, err := os.Stat(filepath.Join(inner, "F.go")); err != nil {
t.Skip("filesystem is case-sensitive")
}
tests := []struct {
path string
err bool
}{
{file, false},
{filepath.Join(inner, "F.go"), true},
{filepath.Join(base, "a/b/c/defgh/f.go"), true},
}
for _, tt := range tests {
err := checkPathCase(tt.path)
if err != nil != tt.err {
t.Errorf("checkPathCase(%q) = %v, wanted error: %v", tt.path, err, tt.err)
}
}
}
func TestFindWorkspaceModFile(t *testing.T) {
workspace := `
-- a/go.mod --
module a
-- a/x/x.go
package x
-- a/x/y/y.go
package x
-- b/go.mod --
module b
-- b/c/go.mod --
module bc
-- d/gopls.mod --
module d-goplsworkspace
-- d/e/go.mod --
module de
-- f/g/go.mod --
module fg
`
dir, err := fake.Tempdir(fake.UnpackTxt(workspace))
if err != nil {
t.Fatal(err)
}
defer os.RemoveAll(dir)
tests := []struct {
folder, want string
}{
{"", ""}, // no module at root, and more than one nested module
{"a", "a/go.mod"},
{"a/x", "a/go.mod"},
{"a/x/y", "a/go.mod"},
{"b/c", "b/c/go.mod"},
{"d", "d/e/go.mod"},
{"d/e", "d/e/go.mod"},
{"f", "f/g/go.mod"},
}
for _, test := range tests {
ctx := context.Background()
rel := fake.RelativeTo(dir)
folderURI := span.URIFromPath(rel.AbsPath(test.folder))
excludeNothing := func(string) bool { return false }
got, err := findWorkspaceModFile(ctx, folderURI, New(nil), excludeNothing)
if err != nil {
t.Fatal(err)
}
want := span.URI("")
if test.want != "" {
want = span.URIFromPath(rel.AbsPath(test.want))
}
if got != want {
t.Errorf("findWorkspaceModFile(%q) = %q, want %q", test.folder, got, want)
}
}
}
func TestInVendor(t *testing.T) {
for _, tt := range []struct {
path string
inVendor bool
}{
{"foo/vendor/x.go", false},
{"foo/vendor/x/x.go", true},
{"foo/x.go", false},
{"foo/vendor/foo.txt", false},
{"foo/vendor/modules.txt", false},
} {
if got := inVendor(span.URIFromPath(tt.path)); got != tt.inVendor {
t.Errorf("expected %s inVendor %v, got %v", tt.path, tt.inVendor, got)
}
}
}
func TestFilters(t *testing.T) {
tests := []struct {
filters []string
included []string
excluded []string
}{
{
included: []string{"x"},
},
{
filters: []string{"-"},
excluded: []string{"x", "x/a"},
},
{
filters: []string{"-x", "+y"},
included: []string{"y", "y/a", "z"},
excluded: []string{"x", "x/a"},
},
{
filters: []string{"-x", "+x/y", "-x/y/z"},
included: []string{"x/y", "x/y/a", "a"},
excluded: []string{"x", "x/a", "x/y/z/a"},
},
{
filters: []string{"+foobar", "-foo"},
included: []string{"foobar", "foobar/a"},
excluded: []string{"foo", "foo/a"},
},
}
for _, tt := range tests {
filterer := source.NewFilterer(tt.filters)
for _, inc := range tt.included {
if pathExcludedByFilter(inc, filterer) {
t.Errorf("filters %q excluded %v, wanted included", tt.filters, inc)
}
}
for _, exc := range tt.excluded {
if !pathExcludedByFilter(exc, filterer) {
t.Errorf("filters %q included %v, wanted excluded", tt.filters, exc)
}
}
}
}
func TestSuffixes(t *testing.T) {
type file struct {
path string
want bool
}
type cases struct {
option []string
files []file
}
tests := []cases{
{[]string{"tmpl", "gotmpl"}, []file{ // default
{"foo", false},
{"foo.tmpl", true},
{"foo.gotmpl", true},
{"tmpl", false},
{"tmpl.go", false}},
},
{[]string{"tmpl", "gotmpl", "html", "gohtml"}, []file{
{"foo.gotmpl", true},
{"foo.html", true},
{"foo.gohtml", true},
{"html", false}},
},
{[]string{"tmpl", "gotmpl", ""}, []file{ // possible user mistake
{"foo.gotmpl", true},
{"foo.go", false},
{"foo", false}},
},
}
for _, a := range tests {
suffixes := a.option
for _, b := range a.files {
got := fileHasExtension(b.path, suffixes)
if got != b.want {
t.Errorf("got %v, want %v, option %q, file %q (%+v)",
got, b.want, a.option, b.path, b)
}
}
}
}
func TestView_Vulnerabilities(t *testing.T) {
// TODO(hyangah): use t.Cleanup when we get rid of go1.13 legacy CI.
defer func() { timeNow = time.Now }()
now := time.Now()
view := &View{
vulns: make(map[span.URI]*govulncheck.Result),
}
file1, file2 := span.URIFromPath("f1/go.mod"), span.URIFromPath("f2/go.mod")
vuln1 := &govulncheck.Result{AsOf: now.Add(-(maxGovulncheckResultAge * 3) / 4)} // already ~3/4*maxGovulncheckResultAge old
view.SetVulnerabilities(file1, vuln1)
vuln2 := &govulncheck.Result{AsOf: now} // fresh.
view.SetVulnerabilities(file2, vuln2)
t.Run("fresh", func(t *testing.T) {
got := view.Vulnerabilities()
want := map[span.URI]*govulncheck.Result{
file1: vuln1,
file2: vuln2,
}
if diff := cmp.Diff(toJSON(want), toJSON(got)); diff != "" {
t.Errorf("view.Vulnerabilities() mismatch (-want +got):\n%s", diff)
}
})
// maxGovulncheckResultAge/2 later
timeNow = func() time.Time { return now.Add(maxGovulncheckResultAge / 2) }
t.Run("after30min", func(t *testing.T) {
got := view.Vulnerabilities()
want := map[span.URI]*govulncheck.Result{
file1: nil, // expired.
file2: vuln2,
}
if diff := cmp.Diff(toJSON(want), toJSON(got)); diff != "" {
t.Errorf("view.Vulnerabilities() mismatch (-want +got):\n%s", diff)
}
})
// maxGovulncheckResultAge later
timeNow = func() time.Time { return now.Add(maxGovulncheckResultAge + time.Minute) }
t.Run("after1hr", func(t *testing.T) {
got := view.Vulnerabilities()
want := map[span.URI]*govulncheck.Result{
file1: nil,
file2: nil,
}
if diff := cmp.Diff(toJSON(want), toJSON(got)); diff != "" {
t.Errorf("view.Vulnerabilities() mismatch (-want +got):\n%s", diff)
}
})
}
func toJSON(x interface{}) string {
b, _ := json.MarshalIndent(x, "", " ")
return string(b)
}
func TestIgnoreFilter(t *testing.T) {
tests := []struct {
dirs []string
path string
want bool
}{
{[]string{"a"}, "a/testdata/foo", true},
{[]string{"a"}, "a/_ignore/foo", true},
{[]string{"a"}, "a/.ignore/foo", true},
{[]string{"a"}, "b/testdata/foo", false},
{[]string{"a"}, "testdata/foo", false},
{[]string{"a", "b"}, "b/testdata/foo", true},
{[]string{"a"}, "atestdata/foo", false},
}
for _, test := range tests {
// convert to filepaths, for convenience
for i, dir := range test.dirs {
test.dirs[i] = filepath.FromSlash(dir)
}
test.path = filepath.FromSlash(test.path)
f := newIgnoreFilter(test.dirs)
if got := f.ignored(test.path); got != test.want {
t.Errorf("newIgnoreFilter(%q).ignore(%q) = %t, want %t", test.dirs, test.path, got, test.want)
}
}
}