Skip to content

Commit 36b30fa

Browse files
committed
add graph analysis
1 parent 6aa440a commit 36b30fa

File tree

3 files changed

+226
-0
lines changed

3 files changed

+226
-0
lines changed

tools/depcheck/pkg/analyze/analyze.go

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
package analyze
2+
3+
import (
4+
"github.com/openshift/origin/tools/depcheck/pkg/graph"
5+
)
6+
7+
func FindExclusiveDependencies(g *graph.MutableDirectedGraph, targetNodes []*graph.Node) []*graph.Node {
8+
newGraph := g.Copy()
9+
for _, target := range targetNodes {
10+
newGraph.RemoveNode(target)
11+
}
12+
13+
return newGraph.PruneOrphans()
14+
}

tools/depcheck/pkg/cmd/analyze.go

+211
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,211 @@
1+
package cmd
2+
3+
import (
4+
"errors"
5+
"fmt"
6+
"io"
7+
"strings"
8+
9+
"github.com/spf13/cobra"
10+
11+
"github.com/gonum/graph/path"
12+
13+
"github.com/openshift/origin/tools/depcheck/pkg/analyze"
14+
"github.com/openshift/origin/tools/depcheck/pkg/graph"
15+
)
16+
17+
var analyzeImportsExample = `# analyze a dependency graph against one of its vendor packages
18+
%[1]s analyze --root=github.com/openshift/origin --entry=pkg/foo/... --dep=github.com/openshift/origin/vendor/k8s.io/kubernetes
19+
`
20+
21+
type AnalyzeOptions struct {
22+
*graph.GraphOptions
23+
24+
// Packages to analyze against
25+
Dependencies []string
26+
27+
Out io.Writer
28+
ErrOut io.Writer
29+
}
30+
31+
type AnalyzeFlags struct {
32+
*graph.GraphFlags
33+
34+
Dependencies []string
35+
}
36+
37+
func (o *AnalyzeFlags) ToOptions(out, errout io.Writer) (*AnalyzeOptions, error) {
38+
graphOpts, err := o.GraphFlags.ToOptions(out, errout)
39+
if err != nil {
40+
return nil, err
41+
}
42+
43+
return &AnalyzeOptions{
44+
GraphOptions: graphOpts,
45+
Dependencies: o.Dependencies,
46+
47+
Out: out,
48+
ErrOut: errout,
49+
}, nil
50+
}
51+
52+
func NewCmdAnalyzeImports(parent string, out, errout io.Writer) *cobra.Command {
53+
analyzeFlags := &AnalyzeFlags{
54+
GraphFlags: &graph.GraphFlags{},
55+
}
56+
57+
cmd := &cobra.Command{
58+
Use: "analyze --root=github.com/openshift/origin --entry=pkg/foo/... --dep pkg/vendor/bar",
59+
Short: "Creates and analyzes a dependency graph against a specified subpackage",
60+
Long: "Creates and analyzes a dependency graph against a specified subpackage",
61+
Example: fmt.Sprintf(traceImportsExample, parent),
62+
RunE: func(c *cobra.Command, args []string) error {
63+
opts, err := analyzeFlags.ToOptions(out, errout)
64+
if err != nil {
65+
return err
66+
}
67+
68+
if err := opts.Complete(); err != nil {
69+
return err
70+
}
71+
if err := opts.Validate(); err != nil {
72+
return err
73+
}
74+
if err := opts.Run(); err != nil {
75+
return err
76+
}
77+
78+
return nil
79+
},
80+
}
81+
82+
analyzeFlags.AddFlags(cmd)
83+
cmd.Flags().StringSliceVarP(&analyzeFlags.Dependencies, "dep", "d", analyzeFlags.Dependencies, "set of dependencies to calculate \"yours\" dependencies against, in a summary report on the dependency sets of a given repository.")
84+
return cmd
85+
}
86+
87+
func (o *AnalyzeOptions) Complete() error {
88+
return o.GraphOptions.Complete()
89+
}
90+
91+
func (o *AnalyzeOptions) Validate() error {
92+
if err := o.GraphOptions.Validate(); err != nil {
93+
return err
94+
}
95+
if len(o.Dependencies) == 0 {
96+
return errors.New("at least one --dep package must be specified")
97+
}
98+
99+
return nil
100+
}
101+
102+
func (o *AnalyzeOptions) Run() error {
103+
g, err := o.GraphOptions.BuildGraph()
104+
if err != nil {
105+
return err
106+
}
107+
108+
return o.analyzeGraph(g)
109+
}
110+
111+
// analyzeGraph receives a MutableDirectedGraph and outputs
112+
// - "Yours": a list of every node in the set unique to the dependency tree of a given target node.
113+
// - "Mine": a list of every node in the set unique to the dependency tree of the root nodes (set non-overlapping with given target node)
114+
// - "Ours": a list of every node in the overlapping set between the dependency tree of the root nodes and a given target node
115+
func (o *AnalyzeOptions) analyzeGraph(g *graph.MutableDirectedGraph) error {
116+
yoursRoots := []*graph.Node{}
117+
for _, dep := range o.Dependencies {
118+
n, exists := g.NodeByName(dep)
119+
if !exists {
120+
return fmt.Errorf("unable to find dependency with import path %q", dep)
121+
}
122+
node, ok := n.(*graph.Node)
123+
if !ok {
124+
return fmt.Errorf("expected node to analyze to be of type *graph.Node. Got: %v", n)
125+
}
126+
127+
yoursRoots = append(yoursRoots, node)
128+
}
129+
130+
fmt.Printf("Analyzing a total of %v packages\n", len(g.Nodes()))
131+
fmt.Println()
132+
133+
yours := analyze.FindExclusiveDependencies(g, yoursRoots)
134+
fmt.Printf("\"Yours\": %v dependencies exclusive to %q\n", len(yours), yoursRoots)
135+
for _, n := range yours {
136+
fmt.Printf(" - %s\n", n)
137+
}
138+
fmt.Println()
139+
140+
// calculate root repo packages, as well as their
141+
// immediate vendor package dependencies
142+
unfilteredMine := map[int]*graph.Node{}
143+
for _, n := range g.Nodes() {
144+
node, ok := n.(*graph.Node)
145+
if !ok {
146+
return fmt.Errorf("expected node to analyze to be of type *graph.Node. Got: %v", n)
147+
}
148+
if isVendorPackage(node) {
149+
continue
150+
}
151+
152+
// obtain immediate vendor package deps from the current node
153+
// and aggregate those as well
154+
for _, v := range g.From(n) {
155+
if !isVendorPackage(v.(*graph.Node)) {
156+
continue
157+
}
158+
159+
unfilteredMine[v.ID()] = v.(*graph.Node)
160+
}
161+
}
162+
163+
mine := []*graph.Node{}
164+
ours := []*graph.Node{}
165+
for _, n := range unfilteredMine {
166+
// determine if the current origin node is reachable from any of the "yours" packages
167+
if isReachableFrom(g, yours, n) {
168+
ours = append(ours, n)
169+
continue
170+
}
171+
172+
mine = append(mine, n)
173+
}
174+
175+
fmt.Printf("\"Mine\": %v direct (first-level) dependencies exclusive to the origin repo\n", len(mine))
176+
for _, n := range mine {
177+
fmt.Printf(" - %s\n", n)
178+
}
179+
fmt.Println()
180+
181+
fmt.Printf("\"Ours\": %v shared dependencies between the origin repo and %v\n", len(ours), yoursRoots)
182+
for _, n := range ours {
183+
fmt.Printf(" - %s\n", n)
184+
}
185+
186+
return nil
187+
}
188+
189+
// isVendorPackage receives a *graph.Node and
190+
// returns true if the given node's unique name is in the vendor path.
191+
func isVendorPackage(n *graph.Node) bool {
192+
if strings.Contains(n.UniqueName, "/vendor/") {
193+
return true
194+
}
195+
196+
return false
197+
}
198+
199+
// isReachableFrom receives a set of root nodes and determines
200+
// if a given destination node can be reached from any of them.
201+
func isReachableFrom(g *graph.MutableDirectedGraph, roots []*graph.Node, dest *graph.Node) bool {
202+
for _, root := range roots {
203+
// no negative edge weights, use Dijkstra
204+
paths := path.DijkstraFrom(root, g)
205+
if pathTo, _ := paths.To(dest); len(pathTo) > 0 {
206+
return true
207+
}
208+
}
209+
210+
return false
211+
}

tools/depcheck/pkg/cmd/cmd.go

+1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ func NewCmdDepCheck(name string, out, errout io.Writer) *cobra.Command {
2323

2424
cmd.AddCommand(NewCmdPinImports(name, out, errout))
2525
cmd.AddCommand(NewCmdTraceImports(name, out, errout))
26+
cmd.AddCommand(NewCmdAnalyzeImports(name, out, errout))
2627

2728
// add glog flags to our global flag set
2829
pflag.CommandLine.AddGoFlagSet(flag.CommandLine)

0 commit comments

Comments
 (0)