aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/cli/internal/util/graph.go
diff options
context:
space:
mode:
author简律纯 <hsiangnianian@outlook.com>2023-04-28 01:36:44 +0800
committer简律纯 <hsiangnianian@outlook.com>2023-04-28 01:36:44 +0800
commitdd84b9d64fb98746a230cd24233ff50a562c39c9 (patch)
treeb583261ef00b3afe72ec4d6dacb31e57779a6faf /cli/internal/util/graph.go
parent0b46fcd72ac34382387b2bcf9095233efbcc52f4 (diff)
downloadHydroRoll-dd84b9d64fb98746a230cd24233ff50a562c39c9.tar.gz
HydroRoll-dd84b9d64fb98746a230cd24233ff50a562c39c9.zip
Diffstat (limited to 'cli/internal/util/graph.go')
-rw-r--r--cli/internal/util/graph.go35
1 files changed, 35 insertions, 0 deletions
diff --git a/cli/internal/util/graph.go b/cli/internal/util/graph.go
new file mode 100644
index 0000000..89de18c
--- /dev/null
+++ b/cli/internal/util/graph.go
@@ -0,0 +1,35 @@
+package util
+
+import (
+ "fmt"
+ "strings"
+
+ "github.com/pyr-sh/dag"
+)
+
+// ValidateGraph checks that a given DAG has no cycles and no self-referential edges.
+// We differ from the underlying DAG Validate method in that we allow multiple roots.
+func ValidateGraph(graph *dag.AcyclicGraph) error {
+ // We use Cycles instead of Validate because
+ // our DAG has multiple roots (entrypoints).
+ // Validate mandates that there is only a single root node.
+ cycles := graph.Cycles()
+ if len(cycles) > 0 {
+ cycleLines := make([]string, len(cycles))
+ for i, cycle := range cycles {
+ vertices := make([]string, len(cycle))
+ for j, vertex := range cycle {
+ vertices[j] = vertex.(string)
+ }
+ cycleLines[i] = "\t" + strings.Join(vertices, ",")
+ }
+ return fmt.Errorf("cyclic dependency detected:\n%s", strings.Join(cycleLines, "\n"))
+ }
+
+ for _, e := range graph.Edges() {
+ if e.Source() == e.Target() {
+ return fmt.Errorf("%s depends on itself", e.Source())
+ }
+ }
+ return nil
+}