blob: 8ad8014ca5f5ce5344cedfb8206f8cf992cd4410 [file] [log] [blame]
Alan Donovan3cded4a2014-09-09 18:39:26 -04001// Copyright 2014 The Go Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
5// Package importgraph computes the forward and reverse import
6// dependency graphs for all packages in a Go workspace.
David Symonds24257c82014-12-09 15:00:58 +11007package importgraph // import "golang.org/x/tools/refactor/importgraph"
Alan Donovan3cded4a2014-09-09 18:39:26 -04008
9import (
10 "go/build"
11 "sync"
12
Andrew Gerrand5ebbcd12014-11-10 08:50:40 +110013 "golang.org/x/tools/go/buildutil"
Alan Donovan3cded4a2014-09-09 18:39:26 -040014)
15
16// A Graph is an import dependency graph, either forward or reverse.
17//
18// The graph maps each node (a package import path) to the set of its
19// successors in the graph. For a forward graph, this is the set of
20// imported packages (prerequisites); for a reverse graph, it is the set
21// of importing packages (clients).
22//
23type Graph map[string]map[string]bool
24
25func (g Graph) addEdge(from, to string) {
26 edges := g[from]
27 if edges == nil {
28 edges = make(map[string]bool)
29 g[from] = edges
30 }
31 edges[to] = true
32}
33
34// Search returns all the nodes of the graph reachable from
35// any of the specified roots, by following edges forwards.
36// Relationally, this is the reflexive transitive closure.
37func (g Graph) Search(roots ...string) map[string]bool {
38 seen := make(map[string]bool)
39 var visit func(x string)
40 visit = func(x string) {
41 if !seen[x] {
42 seen[x] = true
43 for y := range g[x] {
44 visit(y)
45 }
46 }
47 }
48 for _, root := range roots {
49 visit(root)
50 }
51 return seen
52}
53
Alan Donovane079f6c2015-01-21 14:13:33 -050054// Build scans the specified Go workspace and builds the forward and
Alan Donovan3cded4a2014-09-09 18:39:26 -040055// reverse import dependency graphs for all its packages.
56// It also returns a mapping from import paths to errors for packages
Alan Donovana9866432015-06-04 18:38:33 -040057// whose loading was not entirely successful.
58// A package may appear in the graph and in the errors mapping.
Alan Donovan3cded4a2014-09-09 18:39:26 -040059func Build(ctxt *build.Context) (forward, reverse Graph, errors map[string]error) {
Alan Donovan41f0d012014-12-03 09:56:23 -050060 type importEdge struct {
61 from, to string
62 }
Alan Donovan3cded4a2014-09-09 18:39:26 -040063 type pathError struct {
64 path string
65 err error
66 }
Alan Donovan41f0d012014-12-03 09:56:23 -050067
68 ch := make(chan interface{})
Alan Donovan3cded4a2014-09-09 18:39:26 -040069
70 var wg sync.WaitGroup
Alan Donovan66176e22014-09-11 14:33:37 -040071 buildutil.ForEachPackage(ctxt, func(path string, err error) {
Alan Donovan3cded4a2014-09-09 18:39:26 -040072 wg.Add(1)
Alan Donovan41f0d012014-12-03 09:56:23 -050073 go func() {
Alan Donovan3cded4a2014-09-09 18:39:26 -040074 defer wg.Done()
Alan Donovan41f0d012014-12-03 09:56:23 -050075 if err != nil {
76 ch <- pathError{path, err}
77 return
78 }
Alan Donovana9866432015-06-04 18:38:33 -040079
Alan Donovan3cded4a2014-09-09 18:39:26 -040080 bp, err := ctxt.Import(path, "", 0)
Alan Donovan3cded4a2014-09-09 18:39:26 -040081 if err != nil {
Alan Donovana9866432015-06-04 18:38:33 -040082 if _, ok := err.(*build.NoGoError); ok {
83 // empty directory is not an error
84 } else {
85 ch <- pathError{path, err}
86 }
87 // Even in error cases, Import usually returns a package.
Alan Donovan3cded4a2014-09-09 18:39:26 -040088 }
Alan Donovana9866432015-06-04 18:38:33 -040089 if bp != nil {
90 for _, imp := range bp.Imports {
91 ch <- importEdge{path, imp}
92 }
93 for _, imp := range bp.TestImports {
94 ch <- importEdge{path, imp}
95 }
96 for _, imp := range bp.XTestImports {
97 ch <- importEdge{path, imp}
98 }
Alan Donovan3cded4a2014-09-09 18:39:26 -040099 }
Alan Donovan41f0d012014-12-03 09:56:23 -0500100 }()
Alan Donovan3cded4a2014-09-09 18:39:26 -0400101 })
Alan Donovan41f0d012014-12-03 09:56:23 -0500102 go func() {
103 wg.Wait()
104 close(ch)
105 }()
Alan Donovan3cded4a2014-09-09 18:39:26 -0400106
Alan Donovan41f0d012014-12-03 09:56:23 -0500107 forward = make(Graph)
108 reverse = make(Graph)
109
110 for e := range ch {
111 switch e := e.(type) {
112 case pathError:
113 if errors == nil {
114 errors = make(map[string]error)
115 }
116 errors[e.path] = e.err
117
118 case importEdge:
119 if e.to == "C" {
120 continue // "C" is fake
121 }
122 forward.addEdge(e.from, e.to)
123 reverse.addEdge(e.to, e.from)
124 }
125 }
Alan Donovan3cded4a2014-09-09 18:39:26 -0400126
127 return forward, reverse, errors
128}