Skip to content

Commit

Permalink
test: add test cases for pkgCache
Browse files Browse the repository at this point in the history
  • Loading branch information
LiusCraft committed Mar 19, 2024
1 parent dbec0f6 commit 1ba3937
Show file tree
Hide file tree
Showing 3 changed files with 113 additions and 9 deletions.
95 changes: 95 additions & 0 deletions packages/cache.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
package packages

import (
"bytes"
"os/exec"
"strings"
"sync"
)

// pkgPath Caches
type Cache struct {
dirCache map[string]bool
dirCacheMutex sync.RWMutex
packageCacheMap map[string]CacheInfo
packageCacheMutex sync.RWMutex
waitCache sync.WaitGroup
}

type CacheInfo struct {
ImportPath string
PkgDir string
PkgExport string
}

// https://github.com/goplus/gop/issues/1710
// Not fully optimized
// Retrieve all imports in the specified directory and cache them
func (c *Cache) goListExportCache(dir string, pkgs ...string) {
c.dirCacheMutex.Lock()
if c.dirCache[dir] {
c.dirCacheMutex.Unlock()
return
}
c.dirCache[dir] = true
c.dirCacheMutex.Unlock()
var stdout, stderr bytes.Buffer
commandStr := []string{"list", "-f", "{{.ImportPath}},{{.Dir}},{{.Export}}", "-export", "-e"}
commandStr = append(commandStr, pkgs...)
commandStr = append(commandStr, "all")
cmd := exec.Command("go", commandStr...)
cmd.Stdout = &stdout
cmd.Stderr = &stderr
cmd.Dir = dir
err := cmd.Run()
if err == nil {
ret := stdout.String()
for _, v := range strings.Split(ret, "\n") {
s := strings.Split(v, ",")
if len(s) != 3 {
continue
}
c.packageCacheMutex.Lock()
c.packageCacheMap[s[0]] = CacheInfo{s[0], s[1], s[2]}
c.packageCacheMutex.Unlock()
}
}
}

// Reduce wait time when performing `go list` on multiple pkgDir at the same time
func (c *Cache) GoListExportCacheSync(dir string, pkgs ...string) {
c.waitCache.Add(1)
go func() {
defer c.waitCache.Done()
c.goListExportCache(dir, pkgs...)
}()
}

func (c *Cache) GetPkgCache(pkgPath string) (ret CacheInfo, ok bool) {
c.waitCache.Wait()
c.packageCacheMutex.RLock()
ret, ok = c.packageCacheMap[pkgPath]
c.packageCacheMutex.RUnlock()
return
}

func (c *Cache) DelPkgCache(pkgPath string) bool {
_, ok := c.GetPkgCache(pkgPath)
if ok {
c.packageCacheMutex.Lock()
delete(c.packageCacheMap, pkgPath)
c.packageCacheMutex.Unlock()
return true
}
return false
}

func NewGoListCache(dir string) *Cache {
c := &Cache{
dirCache: make(map[string]bool),
packageCacheMap: make(map[string]CacheInfo),
}
// get the dir pkg cache
c.GoListExportCacheSync(dir)
return c
}
18 changes: 18 additions & 0 deletions packages/cache_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,18 @@
package packages

import "testing"

func TestCacheDelCache(t *testing.T) {
p := NewImporter(nil)
if !p.GetPkgCache().DelPkgCache("fmt") {
t.Fatal("del cache should pass!")
}
if p.GetPkgCache().DelPkgCache("fmt") {
t.Fatal("del cache should fail")
}
}

func TestRepeatLoadDir(t *testing.T) {
p := NewImporter(nil)
p.GetPkgCache().goListExportCache("", "")
}
9 changes: 0 additions & 9 deletions packages/imp_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -56,13 +56,4 @@ func Test_loadByExport(t *testing.T) {
FindExport(".", "C")
}

func TestGoListExportCacheSync(t *testing.T) {
GoListExportCacheSync("..")
p := NewImporter(nil, "..")
pkg, err := p.Import("github.com/goplus/gox/internal/foo")
if err != nil {
t.Fatal("Import failed:", pkg, err)
}
}

// ----------------------------------------------------------------------------

0 comments on commit 1ba3937

Please sign in to comment.