|
4 | 4 |
|
5 | 5 | package ssa
|
6 | 6 |
|
| 7 | +import ( |
| 8 | + "go/ast" |
| 9 | + "go/token" |
| 10 | + "go/types" |
| 11 | + "testing" |
| 12 | + |
| 13 | + "golang.org/x/tools/go/packages" |
| 14 | + "golang.org/x/tools/internal/testfiles" |
| 15 | + "golang.org/x/tools/txtar" |
| 16 | +) |
| 17 | + |
7 | 18 | // SetNormalizeAnyForTesting is exported here for external tests.
|
8 | 19 | func SetNormalizeAnyForTesting(normalize bool) {
|
9 | 20 | normalizeAnyForTesting = normalize
|
10 | 21 | }
|
| 22 | + |
| 23 | +// PackagesFromArchive creates the packages from archive with txtar format. |
| 24 | +func PackagesFromArchive(t *testing.T, archive string) []*packages.Package { |
| 25 | + ar := txtar.Parse([]byte(archive)) |
| 26 | + |
| 27 | + fs, err := txtar.FS(ar) |
| 28 | + if err != nil { |
| 29 | + t.Fatal(err) |
| 30 | + } |
| 31 | + |
| 32 | + dir := testfiles.CopyToTmp(t, fs) |
| 33 | + if err != nil { |
| 34 | + t.Fatal(err) |
| 35 | + } |
| 36 | + |
| 37 | + var baseConfig = &packages.Config{ |
| 38 | + Mode: packages.NeedSyntax | |
| 39 | + packages.NeedTypesInfo | |
| 40 | + packages.NeedDeps | |
| 41 | + packages.NeedName | |
| 42 | + packages.NeedFiles | |
| 43 | + packages.NeedImports | |
| 44 | + packages.NeedCompiledGoFiles | |
| 45 | + packages.NeedTypes, |
| 46 | + Dir: dir, |
| 47 | + } |
| 48 | + pkgs, err := packages.Load(baseConfig, "./...") |
| 49 | + if err != nil { |
| 50 | + t.Fatal(err) |
| 51 | + } |
| 52 | + if num := packages.PrintErrors(pkgs); num > 0 { |
| 53 | + t.Fatalf("packages contained %d errors", num) |
| 54 | + } |
| 55 | + return pkgs |
| 56 | +} |
| 57 | + |
| 58 | +// CreateProgram creates a program with given initial packages for testing, |
| 59 | +// usually the packages are constructed via PackagesFromArchive. |
| 60 | +func CreateProgram(t *testing.T, initial []*packages.Package, mode BuilderMode) *Program { |
| 61 | + var fset *token.FileSet |
| 62 | + if len(initial) > 0 { |
| 63 | + fset = initial[0].Fset |
| 64 | + } |
| 65 | + |
| 66 | + prog := NewProgram(fset, mode) |
| 67 | + |
| 68 | + isInitial := make(map[*packages.Package]bool, len(initial)) |
| 69 | + for _, p := range initial { |
| 70 | + isInitial[p] = true |
| 71 | + } |
| 72 | + |
| 73 | + packages.Visit(initial, nil, func(p *packages.Package) { |
| 74 | + if p.Types != nil && !p.IllTyped { |
| 75 | + var files []*ast.File |
| 76 | + var info *types.Info |
| 77 | + if isInitial[p] { |
| 78 | + files = p.Syntax |
| 79 | + info = p.TypesInfo |
| 80 | + } |
| 81 | + prog.CreatePackage(p.Types, files, info, true) |
| 82 | + return |
| 83 | + } |
| 84 | + |
| 85 | + t.Fatalf("package %s or its any dependency contains errors", p.Name) |
| 86 | + }) |
| 87 | + |
| 88 | + return prog |
| 89 | +} |
0 commit comments