-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathimporter.go
133 lines (110 loc) · 3.16 KB
/
importer.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
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
// importer provides mechanisms for loading go/build.Package
// structures from source packages on disk.
package importer
import (
"fmt"
"go/build"
"os"
"path"
"path/filepath"
"runtime"
"strings"
"github.com/pkg/errors"
)
// Importer imports a package.
type Importer interface {
// Import imports the package from importpath.
Import(importpath string) (*build.Package, error)
// Before searchs this importer before the child.
Before(child Importer) Importer
// After searches this importer after the child.
After(child Importer) Importer
}
// ImporterFn converts a func(string) (*build.Package, error) to an Importer.
type ImporterFn func(importpath string) (*build.Package, error)
func (fn ImporterFn) Import(importpath string) (*build.Package, error) {
return fn(importpath)
}
// GOROOT returns an Importer which loads packages from the standard library.
func GOROOT(ctx *build.Context) Importer {
return &srcdirImporter{
Context: ctx,
root: filepath.Join(runtime.GOROOT()),
}
}
type srcdirImporter struct {
*build.Context
root string
}
func (i *srcdirImporter) Import(importpath string) (*build.Package, error) {
if importpath == "" {
return nil, errors.WithStack(&importErr{path: importpath, msg: "invalid import path"})
}
if importpath == "." || importpath == ".." || strings.HasPrefix(importpath, "./") || strings.HasPrefix(importpath, "../") {
return nil, errors.WithStack(&importErr{path: importpath, msg: "relative import not supported"})
}
if strings.HasPrefix(importpath, "/") {
return nil, errors.WithStack(&importErr{path: importpath, msg: "cannot import absolute path"})
}
var p *build.Package
loadPackage := func(importpath, dir string) error {
pkg, err := i.ImportDir(dir, 0)
if err != nil {
return err
}
p = pkg
p.ImportPath = importpath
return nil
}
// if this is the stdlib, then search vendor first.
// this isn't real vendor support, just enough to make net/http compile.
if i.root == runtime.GOROOT() {
importpath := path.Join("vendor", importpath)
dir := filepath.Join(i.root, "src", filepath.FromSlash(importpath))
fi, err := os.Stat(dir)
if err == nil && fi.IsDir() {
err := loadPackage(importpath, dir)
return p, err
}
}
dir := filepath.Join(i.root, "src", filepath.FromSlash(importpath))
fi, err := os.Stat(dir)
if err != nil {
return nil, err
}
if !fi.IsDir() {
return nil, errors.WithStack(&importErr{path: importpath, msg: "not a directory"})
}
err = loadPackage(importpath, dir)
return p, err
}
func (i *srcdirImporter) Before(child Importer) Importer {
return &delegateImporter{
Importer: i,
subordinate: child,
}
}
func (i *srcdirImporter) After(child Importer) Importer {
return &delegateImporter{
Importer: child,
subordinate: i,
}
}
type delegateImporter struct {
Importer // first
subordinate Importer // second
}
func (i *delegateImporter) Import(importpath string) (*build.Package, error) {
pkg, err := i.Importer.Import(importpath)
if err != nil {
return i.subordinate.Import(importpath)
}
return pkg, nil
}
type importErr struct {
path string
msg string
}
func (e *importErr) Error() string {
return fmt.Sprintf("import %q: %v", e.path, e.msg)
}