blob: c045c33fdfdb6508ca974586f8856bffd0dd49f4 [file] [log] [blame]
package parser
import (
"bytes"
"fmt"
"go/build"
"io/ioutil"
"os"
"os/exec"
"path"
"path/filepath"
"strings"
"sync"
)
func getPkgPath(fname string, isDir bool) (string, error) {
if !filepath.IsAbs(fname) {
pwd, err := os.Getwd()
if err != nil {
return "", err
}
fname = filepath.Join(pwd, fname)
}
goModPath, _ := goModPath(fname, isDir)
if strings.Contains(goModPath, "go.mod") {
pkgPath, err := getPkgPathFromGoMod(fname, isDir, goModPath)
if err != nil {
return "", err
}
return pkgPath, nil
}
return getPkgPathFromGOPATH(fname, isDir)
}
var goModPathCache = struct {
paths map[string]string
sync.RWMutex
}{
paths: make(map[string]string),
}
// empty if no go.mod, GO111MODULE=off or go without go modules support
func goModPath(fname string, isDir bool) (string, error) {
root := fname
if !isDir {
root = filepath.Dir(fname)
}
goModPathCache.RLock()
goModPath, ok := goModPathCache.paths[root]
goModPathCache.RUnlock()
if ok {
return goModPath, nil
}
defer func() {
goModPathCache.Lock()
goModPathCache.paths[root] = goModPath
goModPathCache.Unlock()
}()
cmd := exec.Command("go", "env", "GOMOD")
cmd.Dir = root
stdout, err := cmd.Output()
if err != nil {
return "", err
}
goModPath = string(bytes.TrimSpace(stdout))
return goModPath, nil
}
func getPkgPathFromGoMod(fname string, isDir bool, goModPath string) (string, error) {
modulePath := getModulePath(goModPath)
if modulePath == "" {
return "", fmt.Errorf("cannot determine module path from %s", goModPath)
}
rel := path.Join(modulePath, filePathToPackagePath(strings.TrimPrefix(fname, filepath.Dir(goModPath))))
if !isDir {
return path.Dir(rel), nil
}
return path.Clean(rel), nil
}
var pkgPathFromGoModCache = struct {
paths map[string]string
sync.RWMutex
}{
paths: make(map[string]string),
}
func getModulePath(goModPath string) string {
pkgPathFromGoModCache.RLock()
pkgPath, ok := pkgPathFromGoModCache.paths[goModPath]
pkgPathFromGoModCache.RUnlock()
if ok {
return pkgPath
}
defer func() {
pkgPathFromGoModCache.Lock()
pkgPathFromGoModCache.paths[goModPath] = pkgPath
pkgPathFromGoModCache.Unlock()
}()
data, err := ioutil.ReadFile(goModPath)
if err != nil {
return ""
}
pkgPath = modulePath(data)
return pkgPath
}
func getPkgPathFromGOPATH(fname string, isDir bool) (string, error) {
gopath := os.Getenv("GOPATH")
if gopath == "" {
gopath = build.Default.GOPATH
}
for _, p := range strings.Split(gopath, string(filepath.ListSeparator)) {
prefix := filepath.Join(p, "src") + string(filepath.Separator)
rel, err := filepath.Rel(prefix, fname)
if err == nil && !strings.HasPrefix(rel, ".."+string(filepath.Separator)) {
if !isDir {
return path.Dir(filePathToPackagePath(rel)), nil
} else {
return path.Clean(filePathToPackagePath(rel)), nil
}
}
}
return "", fmt.Errorf("file '%v' is not in GOPATH '%v'", fname, gopath)
}
func filePathToPackagePath(path string) string {
return filepath.ToSlash(path)
}