285 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			285 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
package packages_test
 | 
						|
 | 
						|
import (
 | 
						|
	"io/ioutil"
 | 
						|
	"os"
 | 
						|
	"path/filepath"
 | 
						|
	"sort"
 | 
						|
	"strings"
 | 
						|
	"testing"
 | 
						|
 | 
						|
	"github.com/golangci/golangci-lint/pkg/fsutils"
 | 
						|
	"github.com/golangci/golangci-lint/pkg/logutils"
 | 
						|
	"github.com/golangci/golangci-lint/pkg/packages"
 | 
						|
	"github.com/stretchr/testify/assert"
 | 
						|
)
 | 
						|
 | 
						|
type fsPreparer struct {
 | 
						|
	t      *testing.T
 | 
						|
	root   string
 | 
						|
	prevWD string
 | 
						|
}
 | 
						|
 | 
						|
func (fp fsPreparer) clean() {
 | 
						|
	err := os.Chdir(fp.prevWD)
 | 
						|
	assert.NoError(fp.t, err)
 | 
						|
 | 
						|
	err = os.RemoveAll(fp.root)
 | 
						|
	assert.NoError(fp.t, err)
 | 
						|
}
 | 
						|
 | 
						|
func prepareFS(t *testing.T, paths ...string) *fsPreparer {
 | 
						|
	root, err := ioutil.TempDir("/tmp", "golangci.test.path_resolver")
 | 
						|
	assert.NoError(t, err)
 | 
						|
 | 
						|
	prevWD, err := fsutils.Getwd()
 | 
						|
	assert.NoError(t, err)
 | 
						|
 | 
						|
	err = os.Chdir(root)
 | 
						|
	assert.NoError(t, err)
 | 
						|
 | 
						|
	for _, p := range paths {
 | 
						|
		err = os.MkdirAll(filepath.Dir(p), os.ModePerm)
 | 
						|
		assert.NoError(t, err)
 | 
						|
 | 
						|
		if strings.HasSuffix(p, "/") {
 | 
						|
			continue
 | 
						|
		}
 | 
						|
 | 
						|
		goFile := "package p\n"
 | 
						|
		err = ioutil.WriteFile(p, []byte(goFile), os.ModePerm)
 | 
						|
		assert.NoError(t, err)
 | 
						|
	}
 | 
						|
 | 
						|
	return &fsPreparer{
 | 
						|
		root:   root,
 | 
						|
		t:      t,
 | 
						|
		prevWD: prevWD,
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func newTestResolver(t *testing.T, excludeDirs []string) *packages.Resolver {
 | 
						|
	r, err := packages.NewResolver(nil, excludeDirs, logutils.NewStderrLog(""))
 | 
						|
	assert.NoError(t, err)
 | 
						|
 | 
						|
	return r
 | 
						|
}
 | 
						|
 | 
						|
func TestPathResolverNotExistingPath(t *testing.T) {
 | 
						|
	fp := prepareFS(t)
 | 
						|
	defer fp.clean()
 | 
						|
 | 
						|
	_, err := newTestResolver(t, nil).Resolve("a")
 | 
						|
	assert.EqualError(t, err, "can't eval symlinks for path a: lstat a: no such file or directory")
 | 
						|
}
 | 
						|
 | 
						|
func TestPathResolverCommonCases(t *testing.T) {
 | 
						|
	type testCase struct {
 | 
						|
		name         string
 | 
						|
		prepare      []string
 | 
						|
		resolve      []string
 | 
						|
		expFiles     []string
 | 
						|
		expDirs      []string
 | 
						|
		includeTests bool
 | 
						|
	}
 | 
						|
 | 
						|
	testCases := []testCase{
 | 
						|
		{
 | 
						|
			name:    "empty root recursively",
 | 
						|
			resolve: []string{"./..."},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:    "empty root",
 | 
						|
			resolve: []string{"./"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:    "vendor is excluded recursively",
 | 
						|
			prepare: []string{"vendor/a/b.go"},
 | 
						|
			resolve: []string{"./..."},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:    "vendor is excluded",
 | 
						|
			prepare: []string{"vendor/a.go"},
 | 
						|
			resolve: []string{"./..."},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:    "nested vendor is excluded",
 | 
						|
			prepare: []string{"d/vendor/a.go"},
 | 
						|
			resolve: []string{"./..."},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "vendor dir is excluded by regexp, not the exact match",
 | 
						|
			prepare:  []string{"vendors/a.go", "novendor/b.go"},
 | 
						|
			resolve:  []string{"./..."},
 | 
						|
			expDirs:  []string{"vendors"},
 | 
						|
			expFiles: []string{"vendors/a.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "vendor explicitly resolved",
 | 
						|
			prepare:  []string{"vendor/a.go"},
 | 
						|
			resolve:  []string{"./vendor"},
 | 
						|
			expDirs:  []string{"vendor"},
 | 
						|
			expFiles: []string{"vendor/a.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "nested vendor explicitly resolved",
 | 
						|
			prepare:  []string{"d/vendor/a.go"},
 | 
						|
			resolve:  []string{"d/vendor"},
 | 
						|
			expDirs:  []string{"d/vendor"},
 | 
						|
			expFiles: []string{"d/vendor/a.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "extensions filter recursively",
 | 
						|
			prepare:  []string{"a/b.go", "a/c.txt", "d.go", "e.csv"},
 | 
						|
			resolve:  []string{"./..."},
 | 
						|
			expDirs:  []string{".", "a"},
 | 
						|
			expFiles: []string{"a/b.go", "d.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "extensions filter",
 | 
						|
			prepare:  []string{"a/b.go", "a/c.txt", "d.go"},
 | 
						|
			resolve:  []string{"a"},
 | 
						|
			expDirs:  []string{"a"},
 | 
						|
			expFiles: []string{"a/b.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "one level dirs exclusion",
 | 
						|
			prepare:  []string{"a/b/d.go", "a/c.go"},
 | 
						|
			resolve:  []string{"./a"},
 | 
						|
			expDirs:  []string{"a"},
 | 
						|
			expFiles: []string{"a/c.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "explicitly resolved files",
 | 
						|
			prepare:  []string{"a/b/c.go", "a/d.txt"},
 | 
						|
			resolve:  []string{"./a/...", "a/d.txt"},
 | 
						|
			expDirs:  []string{"a/b"},
 | 
						|
			expFiles: []string{"a/b/c.go", "a/d.txt"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:    ".* dotfiles are always ignored",
 | 
						|
			prepare: []string{".git/a.go", ".circleci/b.go"},
 | 
						|
			resolve: []string{"./..."},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "exclude dirs on any depth level",
 | 
						|
			prepare:  []string{"ok/.git/a.go", "ok/b.go"},
 | 
						|
			resolve:  []string{"./..."},
 | 
						|
			expDirs:  []string{"ok"},
 | 
						|
			expFiles: []string{"ok/b.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "exclude path, not name",
 | 
						|
			prepare:  []string{"ex/clude/me/a.go", "c/d.go"},
 | 
						|
			resolve:  []string{"./..."},
 | 
						|
			expDirs:  []string{"c"},
 | 
						|
			expFiles: []string{"c/d.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "exclude partial path",
 | 
						|
			prepare:  []string{"prefix/ex/clude/me/a.go", "prefix/ex/clude/me/subdir/c.go", "prefix/b.go"},
 | 
						|
			resolve:  []string{"./..."},
 | 
						|
			expDirs:  []string{"prefix"},
 | 
						|
			expFiles: []string{"prefix/b.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "don't exclude file instead of dir",
 | 
						|
			prepare:  []string{"a/exclude.go"},
 | 
						|
			resolve:  []string{"a"},
 | 
						|
			expDirs:  []string{"a"},
 | 
						|
			expFiles: []string{"a/exclude.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:    "don't exclude file instead of dir: check dir is excluded",
 | 
						|
			prepare: []string{"a/exclude.go/b.go"},
 | 
						|
			resolve: []string{"a/..."},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:    "ignore _*",
 | 
						|
			prepare: []string{"_any/a.go"},
 | 
						|
			resolve: []string{"./..."},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:         "include tests",
 | 
						|
			prepare:      []string{"a/b.go", "a/b_test.go"},
 | 
						|
			resolve:      []string{"./..."},
 | 
						|
			expDirs:      []string{"a"},
 | 
						|
			expFiles:     []string{"a/b.go", "a/b_test.go"},
 | 
						|
			includeTests: true,
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "exclude tests",
 | 
						|
			prepare:  []string{"a/b.go", "a/b_test.go"},
 | 
						|
			resolve:  []string{"./..."},
 | 
						|
			expDirs:  []string{"a"},
 | 
						|
			expFiles: []string{"a/b.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "exclude tests except explicitly set",
 | 
						|
			prepare:  []string{"a/b.go", "a/b_test.go", "a/c_test.go"},
 | 
						|
			resolve:  []string{"./...", "a/c_test.go"},
 | 
						|
			expDirs:  []string{"a"},
 | 
						|
			expFiles: []string{"a/b.go", "a/c_test.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "exclude dirs with no go files",
 | 
						|
			prepare:  []string{"a/b.txt", "a/c/d.go"},
 | 
						|
			resolve:  []string{"./..."},
 | 
						|
			expDirs:  []string{"a/c"},
 | 
						|
			expFiles: []string{"a/c/d.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "exclude dirs with no go files with root dir",
 | 
						|
			prepare:  []string{"a/b.txt", "a/c/d.go", "e.go"},
 | 
						|
			resolve:  []string{"./..."},
 | 
						|
			expDirs:  []string{".", "a/c"},
 | 
						|
			expFiles: []string{"a/c/d.go", "e.go"},
 | 
						|
		},
 | 
						|
		{
 | 
						|
			name:     "resolve absolute paths",
 | 
						|
			prepare:  []string{"a/b.go", "a/c.txt", "d.go", "e.csv"},
 | 
						|
			resolve:  []string{"${CWD}/..."},
 | 
						|
			expDirs:  []string{".", "a"},
 | 
						|
			expFiles: []string{"a/b.go", "d.go"},
 | 
						|
		},
 | 
						|
	}
 | 
						|
 | 
						|
	fsutils.UseWdCache(false)
 | 
						|
	for _, tc := range testCases {
 | 
						|
		t.Run(tc.name, func(t *testing.T) {
 | 
						|
			fp := prepareFS(t, tc.prepare...)
 | 
						|
			defer fp.clean()
 | 
						|
 | 
						|
			for i, rp := range tc.resolve {
 | 
						|
				tc.resolve[i] = strings.Replace(rp, "${CWD}", fp.root, -1)
 | 
						|
			}
 | 
						|
 | 
						|
			r := newTestResolver(t, []string{"vendor$", "ex/clude/me", "exclude"})
 | 
						|
 | 
						|
			prog, err := r.Resolve(tc.resolve...)
 | 
						|
			assert.NoError(t, err)
 | 
						|
			assert.NotNil(t, prog)
 | 
						|
 | 
						|
			progFiles := prog.Files(tc.includeTests)
 | 
						|
			sort.StringSlice(progFiles).Sort()
 | 
						|
			sort.StringSlice(tc.expFiles).Sort()
 | 
						|
 | 
						|
			progDirs := prog.Dirs()
 | 
						|
			sort.StringSlice(progDirs).Sort()
 | 
						|
			sort.StringSlice(tc.expDirs).Sort()
 | 
						|
 | 
						|
			if tc.expFiles == nil {
 | 
						|
				assert.Empty(t, progFiles)
 | 
						|
			} else {
 | 
						|
				assert.Equal(t, tc.expFiles, progFiles, "files")
 | 
						|
			}
 | 
						|
 | 
						|
			if tc.expDirs == nil {
 | 
						|
				assert.Empty(t, progDirs)
 | 
						|
			} else {
 | 
						|
				assert.Equal(t, tc.expDirs, progDirs, "dirs")
 | 
						|
			}
 | 
						|
		})
 | 
						|
	}
 | 
						|
}
 |