@@ -214,6 +214,10 @@ func (o *TraceImportsOpts) Run() error {
214
214
}
215
215
}
216
216
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
+
217
221
if len (o .OutputFormat ) > 0 {
218
222
return o .outputGraph (g )
219
223
}
@@ -256,173 +260,40 @@ func (o *TraceImportsOpts) analyzeGraph(g *depgraph.MutableDirectedGraph) error
256
260
fmt .Printf ("Analyzing a total of %v packages\n " , len (g .Nodes ()))
257
261
fmt .Println ()
258
262
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 })
342
264
fmt .Printf ("\" Yours\" : %v packages exclusive to %q\n " , len (yours ), you .UniqueName )
343
265
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 )
350
267
}
351
268
352
- return yours , nil
269
+ return nil
353
270
}
354
271
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 {}
368
279
369
- fmt .Printf (" - %s\n " , pkg .UniqueName )
280
+ for _ , target := range targetNodes {
281
+ newGraph .RemoveNode (target )
370
282
}
371
283
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
386
288
}
387
- }
388
289
389
- if ! isOrphan {
390
- continue
290
+ orphans = append (orphans , n .(* depgraph.Node ))
391
291
}
392
-
393
- orphans = append (orphans , n )
394
292
}
395
293
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 {}
425
296
}
426
297
427
- return false
298
+ return append ( orphans , calculateOrphans ( newGraph , orphans ) ... )
428
299
}
0 commit comments