Skip to content

Commit c6fadf7

Browse files
committed
address comments
1 parent 400edcd commit c6fadf7

File tree

5 files changed

+93
-207
lines changed

5 files changed

+93
-207
lines changed

tools/depcheck/examples/origin_filters_extended.json

+9-2
Original file line numberDiff line numberDiff line change
@@ -249,11 +249,18 @@
249249
"github.com/openshift/origin/vendor/k8s.io/sample-controller",
250250
"github.com/openshift/origin/vendor/k8s.io/utils",
251251

252+
"github.com/openshift/origin/vendor/gopkg.in/asn1-ber.v1",
253+
"github.com/openshift/origin/vendor/gopkg.in/gcfg.v1",
254+
"github.com/openshift/origin/vendor/gopkg.in/inf.v0",
255+
"github.com/openshift/origin/vendor/gopkg.in/ldap.v2",
256+
"github.com/openshift/origin/vendor/gopkg.in/natefinch",
257+
"github.com/openshift/origin/vendor/gopkg.in/warnings.v0",
258+
"github.com/openshift/origin/vendor/gopkg.in/yaml.v2",
259+
252260
"github.com/openshift/origin/vendor/bitbucket.org",
253261
"github.com/openshift/origin/vendor/cloud.google.com",
254262
"github.com/openshift/origin/vendor/go4.org",
255263
"github.com/openshift/origin/vendor/golang.org",
256264
"github.com/openshift/origin/vendor/google.golang.org",
257-
"github.com/openshift/origin/vendor/gopkg.in",
258265
"github.com/openshift/origin/vendor/vbom.ml"
259-
]
266+
]

tools/depcheck/pkg/cmd/trace/filter.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ import (
1515
// packagePrefix subpackages are re-written to originate from the packagePrefix, and edges
1616
// terminating at packagePrefix subpackages are re-written to terminate at the packagePrefix.
1717
func FilterPackages(g *depgraph.MutableDirectedGraph, packagePrefixes []string) (*depgraph.MutableDirectedGraph, error) {
18-
collapsedGraph := depgraph.NewMutableDirectedGraph(concrete.NewDirectedGraph())
18+
collapsedGraph := depgraph.NewMutableDirectedGraph()
1919

2020
// copy all nodes to new graph
2121
for _, n := range g.Nodes() {

tools/depcheck/pkg/cmd/trace/pkg.go

+1-50
Original file line numberDiff line numberDiff line change
@@ -45,7 +45,7 @@ func (p *PackageList) Add(pkg Package) {
4545
// Any packages whose import path is contained within a list of "excludes" are not added to the graph.
4646
// Returns a directed graph and a map of package import paths to node ids, or an error.
4747
func BuildGraph(packages *PackageList, excludes []string) (*depgraph.MutableDirectedGraph, error) {
48-
g := depgraph.NewMutableDirectedGraph(concrete.NewDirectedGraph())
48+
g := depgraph.NewMutableDirectedGraph()
4949

5050
// contains the subset of packages from the set of given packages (and their immediate dependencies)
5151
// that will actually be included in our graph - any packages in the excludes slice, or that do not
@@ -111,55 +111,6 @@ func BuildGraph(packages *PackageList, excludes []string) (*depgraph.MutableDire
111111
return g, nil
112112
}
113113

114-
func copyGraph(g *depgraph.MutableDirectedGraph) (*depgraph.MutableDirectedGraph, error) {
115-
if g == nil {
116-
return nil, fmt.Errorf("cannot copy nil graph")
117-
}
118-
119-
newGraph := depgraph.NewMutableDirectedGraph(concrete.NewDirectedGraph())
120-
121-
// copy nodes
122-
for _, n := range g.Nodes() {
123-
newNode, ok := n.(*depgraph.Node)
124-
if !ok {
125-
continue
126-
}
127-
128-
err := newGraph.AddNode(newNode)
129-
if err != nil {
130-
return nil, err
131-
}
132-
}
133-
134-
// copy edges
135-
for _, n := range g.Nodes() {
136-
node, ok := n.(*depgraph.Node)
137-
if !ok {
138-
return nil, fmt.Errorf("expected node to be of type *depgraph.Node")
139-
}
140-
141-
// ensure node exists in new graph
142-
if _, exists := newGraph.NodeByName(node.UniqueName); !exists {
143-
return nil, fmt.Errorf("expected node %q to exist in new graph", node.UniqueName)
144-
}
145-
146-
from := g.From(n)
147-
for _, to := range from {
148-
if newGraph.HasEdgeFromTo(n, to) {
149-
continue
150-
}
151-
152-
newGraph.SetEdge(concrete.Edge{
153-
F: n,
154-
T: to,
155-
}, 0)
156-
}
157-
}
158-
159-
160-
return newGraph, nil
161-
}
162-
163114
func isExcludedPath(path string, excludes []string) bool {
164115
for _, exclude := range excludes {
165116
if strings.HasPrefix(path, exclude) {

tools/depcheck/pkg/cmd/trace/trace.go

+24-153
Original file line numberDiff line numberDiff line change
@@ -214,6 +214,10 @@ func (o *TraceImportsOpts) Run() error {
214214
}
215215
}
216216

217+
// prune orphans after filtering in order to maximize chance of deleting disconnected
218+
// trees from the graph that have been collapsed into their root node by a given filter.
219+
g.PruneOrphans()
220+
217221
if len(o.OutputFormat) > 0 {
218222
return o.outputGraph(g)
219223
}
@@ -256,173 +260,40 @@ func (o *TraceImportsOpts) analyzeGraph(g *depgraph.MutableDirectedGraph) error
256260
fmt.Printf("Analyzing a total of %v packages\n", len(g.Nodes()))
257261
fmt.Println()
258262

259-
// determine existing orphans and roots
260-
keep := []graph.Node{}
261-
roots := []graph.Node{}
262-
for _, n := range g.Nodes() {
263-
node, ok := n.(*depgraph.Node)
264-
if !ok {
265-
continue
266-
}
267-
268-
to := g.To(n)
269-
from := g.From(n)
270-
271-
// found an existing orphaned node to exclude
272-
if len(to) == 0 && len(from) == 0 {
273-
continue
274-
}
275-
276-
// found a root
277-
if len(from) > 0 && len(to) == 0 {
278-
roots = append(roots, n)
279-
continue
280-
}
281-
282-
// skip package we are analyzing
283-
if node.UniqueName == o.Analyze {
284-
continue
285-
}
286-
287-
keep = append(keep, n)
288-
}
289-
290-
yoursGraph, err := copyGraph(g)
291-
if err != nil {
292-
return err
293-
}
294-
yours, err := printYours(yoursGraph, roots, keep, you)
295-
if err != nil {
296-
return err
297-
}
298-
fmt.Println()
299-
300-
// no need to copy the graph a second time - we can mutate it
301-
// since it won't be used anymore
302-
mine, err := printMine(g, roots, keep, you)
303-
if err != nil {
304-
return err
305-
}
306-
307-
ours := []graph.Node{}
308-
yoursOrMine := append(yours, mine...)
309-
for _, n := range keep {
310-
isYoursOrMine := false
311-
for _, yom := range yoursOrMine {
312-
if n.ID() == yom.ID() {
313-
isYoursOrMine = true
314-
break
315-
}
316-
}
317-
if !isYoursOrMine {
318-
ours = append(ours, n)
319-
}
320-
}
321-
fmt.Println()
322-
323-
fmt.Printf("\"Ours\": %v packages shared between the main repo and %q\n", len(ours), you.UniqueName)
324-
for _, n := range ours {
325-
pkg, ok := n.(*depgraph.Node)
326-
if !ok {
327-
continue
328-
}
329-
330-
fmt.Printf(" - %s\n", pkg.UniqueName)
331-
}
332-
333-
return nil
334-
}
335-
336-
func printYours(g *depgraph.MutableDirectedGraph, roots, nodes []graph.Node, you *depgraph.Node) ([]graph.Node, error) {
337-
g.RemoveNode(you)
338-
339-
// find nodes not reachable from our given roots
340-
yours := findOrphans(g, roots, nodes)
341-
263+
yours := calculateOrphans(g, []*depgraph.Node{you})
342264
fmt.Printf("\"Yours\": %v packages exclusive to %q\n", len(yours), you.UniqueName)
343265
for _, n := range yours {
344-
pkg, ok := n.(*depgraph.Node)
345-
if !ok {
346-
continue
347-
}
348-
349-
fmt.Printf(" - %s\n", pkg.UniqueName)
266+
fmt.Printf(" - %s\n", n.UniqueName)
350267
}
351268

352-
return yours, nil
269+
return nil
353270
}
354271

355-
func printMine(g *depgraph.MutableDirectedGraph, roots, nodes []graph.Node, you *depgraph.Node) ([]graph.Node, error) {
356-
for _, r := range roots {
357-
g.RemoveNode(r)
358-
}
359-
360-
// find nodes not reachable from "you"
361-
mine := findOrphans(g, []graph.Node{you}, nodes)
362-
fmt.Printf("\"Mine\": %v packages exclusive to the main repo\n", len(mine))
363-
for _, n := range mine {
364-
pkg, ok := n.(*depgraph.Node)
365-
if !ok {
366-
continue
367-
}
272+
// calculateOrphans receives a graph and a set of targetNodes. The graph is cloned, and all targetNodes
273+
// are removed from it. Any nodes left orphaned are added to a "removed" set.
274+
// This operation is done recursively until we no longer cause any nodes to become orphaned after pruning the removed set.
275+
// The resulting graph is returned along with the removed set.
276+
func calculateOrphans(g *depgraph.MutableDirectedGraph, targetNodes []*depgraph.Node) []*depgraph.Node {
277+
newGraph := g.Copy()
278+
orphans := []*depgraph.Node{}
368279

369-
fmt.Printf(" - %s\n", pkg.UniqueName)
280+
for _, target := range targetNodes {
281+
newGraph.RemoveNode(target)
370282
}
371283

372-
return mine, nil
373-
}
374-
375-
// findOrphans receives a set of root nodes and a set of child nodes and
376-
// returns the set of child nodes not reachable from any root node.
377-
func findOrphans(g graph.Directed, roots []graph.Node, nodes []graph.Node) []graph.Node {
378-
orphans := []graph.Node{}
379-
380-
for _, n := range nodes {
381-
isOrphan := true
382-
for _, root := range roots {
383-
if hasPathFromTo(g, root, n) {
384-
isOrphan = false
385-
break
284+
for _, n := range newGraph.Nodes() {
285+
if len(newGraph.To(n)) == 0 {
286+
if _, ok := n.(*depgraph.Node); !ok {
287+
continue
386288
}
387-
}
388289

389-
if !isOrphan {
390-
continue
290+
orphans = append(orphans, n.(*depgraph.Node))
391291
}
392-
393-
orphans = append(orphans, n)
394292
}
395293

396-
return orphans
397-
}
398-
399-
// hasPathFromTo receives a node A and determines
400-
// if a node B can be reached from it.
401-
// Returns false if the source and destination are the same.
402-
func hasPathFromTo(g graph.Directed, A graph.Node, B graph.Node) bool {
403-
if A.ID() == B.ID() {
404-
return false
405-
}
406-
if !g.Has(A) || !g.Has(B) {
407-
return false
408-
}
409-
410-
seen := map[int]bool{}
411-
unseen := []graph.Node{B}
412-
for len(unseen) > 0 {
413-
n := unseen[0]
414-
unseen = unseen[1:]
415-
416-
if _, isSeen := seen[n.ID()]; isSeen {
417-
continue
418-
}
419-
if n.ID() == A.ID() {
420-
return true
421-
}
422-
423-
seen[n.ID()] = true
424-
unseen = append(unseen, g.To(n)...)
294+
if len(orphans) == 0 {
295+
return []*depgraph.Node{}
425296
}
426297

427-
return false
298+
return append(orphans, calculateOrphans(newGraph, orphans)...)
428299
}

tools/depcheck/pkg/graph/graph.go

+58-1
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,7 @@ func (n Node) DOTAttributes() []dot.Attribute {
3232
}
3333
}
3434

35-
func NewMutableDirectedGraph(g *concrete.DirectedGraph) *MutableDirectedGraph {
35+
func NewMutableDirectedGraph() *MutableDirectedGraph {
3636
return &MutableDirectedGraph{
3737
DirectedGraph: concrete.NewDirectedGraph(),
3838
nodesByName: make(map[string]graph.Node),
@@ -55,7 +55,64 @@ func (g *MutableDirectedGraph) AddNode(n *Node) error {
5555
return nil
5656
}
5757

58+
func (g *MutableDirectedGraph) RemoveNode(n *Node) {
59+
delete(g.nodesByName, n.UniqueName)
60+
g.DirectedGraph.RemoveNode(n)
61+
}
62+
5863
func (g *MutableDirectedGraph) NodeByName(name string) (graph.Node, bool) {
5964
n, exists := g.nodesByName[name]
6065
return n, exists && g.DirectedGraph.Has(n)
6166
}
67+
68+
// remove nodes with no inbound edges
69+
func (g *MutableDirectedGraph) PruneOrphans() {
70+
for _, n := range g.Nodes() {
71+
if len(g.To(n)) == 0 {
72+
g.RemoveNode(n.(*Node))
73+
}
74+
}
75+
}
76+
77+
func (g *MutableDirectedGraph) Copy() *MutableDirectedGraph {
78+
newGraph := NewMutableDirectedGraph()
79+
80+
// copy nodes
81+
for _, n := range g.Nodes() {
82+
newNode, ok := n.(*Node)
83+
if !ok {
84+
continue
85+
}
86+
87+
// ignore error: the only error that could happen is if a name collision occurs.
88+
// since we are simply moving nodes from an existing graph to a new one,
89+
// we are guaranteed to already have unique node names from an existing valid graph.
90+
newGraph.AddNode(newNode)
91+
}
92+
93+
// copy edges
94+
for _, n := range g.Nodes() {
95+
node, ok := n.(*Node)
96+
if !ok {
97+
continue
98+
}
99+
100+
if _, exists := newGraph.NodeByName(node.UniqueName); !exists {
101+
continue
102+
}
103+
104+
from := g.From(n)
105+
for _, to := range from {
106+
if newGraph.HasEdgeFromTo(n, to) {
107+
continue
108+
}
109+
110+
newGraph.SetEdge(concrete.Edge{
111+
F: n,
112+
T: to,
113+
}, 0)
114+
}
115+
}
116+
117+
return newGraph
118+
}

0 commit comments

Comments
 (0)