/
package.go
76 lines (65 loc) 路 1.58 KB
/
package.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
package codegen
import (
"go/types"
"strconv"
"golang.org/x/exp/slices"
"golang.org/x/tools/go/packages"
)
const pkgMode = packages.NeedName |
// packages.NeedFiles |
packages.NeedImports |
packages.NeedTypes |
packages.NeedSyntax |
packages.NeedTypesInfo |
packages.NeedModule |
packages.NeedDeps
type Package struct {
pkgPath string
name string
cache map[string]*types.Package
imports []*types.Package
}
func NewPackage(path, name string) *Package {
return &Package{pkgPath: path, name: name, cache: make(map[string]*types.Package)}
}
func (p Package) PkgPath() string {
return p.pkgPath
}
func (p Package) Name() string {
return p.name
}
func (p *Package) Import(pkg *types.Package) (*types.Package, bool) {
if i := slices.IndexFunc(p.imports, func(item *types.Package) bool {
return pkg.Path() == item.Path()
}); i > -1 {
return p.imports[i], false
}
pkgs, err := packages.Load(&packages.Config{
Mode: packages.NeedName,
}, pkg.Path())
if err != nil {
return pkg, false
}
// If the import path is this package, skip to import
if pkg.Path() == p.PkgPath() {
return nil, false
} else if pkgs[0].Name != "" && pkgs[0].Name != pkg.Name() {
pkg.SetName(pkgs[0].Name)
}
alias := p.newAliasIfExists(pkg)
pkg.SetName(alias)
p.cache[alias] = pkg
p.imports = append(p.imports, pkg)
return pkg, true
}
func (p *Package) newAliasIfExists(pkg *types.Package) string {
pkgName, newPkgName := pkg.Name(), pkg.Name()
for i := 1; ; i++ {
if _, ok := p.cache[newPkgName]; ok {
newPkgName = pkgName + strconv.Itoa(i)
continue
}
break
}
return newPkgName
}