|
| 1 | +/* implements a dependency graph algorithm */ |
| 2 | +package dep |
| 3 | + |
| 4 | +import ( |
| 5 | + "fmt" |
| 6 | + "slices" |
| 7 | +) |
| 8 | + |
| 9 | +/////////////////////////////////////////////////////////////////////////////// |
| 10 | +// TYPES |
| 11 | + |
| 12 | +type Node any |
| 13 | + |
| 14 | +type node struct { |
| 15 | + Node |
| 16 | + edges []Node |
| 17 | +} |
| 18 | + |
| 19 | +type dep struct { |
| 20 | + // Node depends on several other nodes |
| 21 | + edges map[Node][]Node |
| 22 | +} |
| 23 | + |
| 24 | +/////////////////////////////////////////////////////////////////////////////// |
| 25 | +// LIFECYCLE |
| 26 | + |
| 27 | +func NewDep() *dep { |
| 28 | + dep := new(dep) |
| 29 | + dep.edges = make(map[Node][]Node) |
| 30 | + return dep |
| 31 | +} |
| 32 | + |
| 33 | +/////////////////////////////////////////////////////////////////////////////// |
| 34 | +// STRINGIFY |
| 35 | + |
| 36 | +func (d *dep) String() string { |
| 37 | + str := "dep{\n" |
| 38 | + for a, b := range d.edges { |
| 39 | + str += fmt.Sprint(" ", a) + " -> [" |
| 40 | + for i, c := range b { |
| 41 | + if i > 0 { |
| 42 | + str += ", " |
| 43 | + } |
| 44 | + str += fmt.Sprint(c) |
| 45 | + } |
| 46 | + str += "]\n" |
| 47 | + } |
| 48 | + str += "}" |
| 49 | + return str |
| 50 | +} |
| 51 | + |
| 52 | +/////////////////////////////////////////////////////////////////////////////// |
| 53 | +// PUBLIC METHODS |
| 54 | + |
| 55 | +// Add a node a which depends on several other nodes b |
| 56 | +func (d *dep) AddNode(a Node, b ...Node) { |
| 57 | + if _, exists := d.edges[a]; exists { |
| 58 | + d.edges[a] = append(d.edges[a], b...) |
| 59 | + } else { |
| 60 | + d.edges[a] = b |
| 61 | + } |
| 62 | + for _, c := range b { |
| 63 | + if _, exists := d.edges[c]; !exists { |
| 64 | + d.edges[c] = []Node{} |
| 65 | + } |
| 66 | + } |
| 67 | +} |
| 68 | + |
| 69 | +// Resolve returns the dependency order for a node |
| 70 | +func (d *dep) Resolve(n Node) ([]Node, error) { |
| 71 | + resolved, _, err := d.resolve(n, nil, nil) |
| 72 | + return resolved, err |
| 73 | +} |
| 74 | + |
| 75 | +/////////////////////////////////////////////////////////////////////////////// |
| 76 | +// PRIVATE METHODS |
| 77 | + |
| 78 | +// Node a depends on several other nodes b |
| 79 | +func (d *dep) resolve(n Node, resolved []Node, unresolved []Node) ([]Node, []Node, error) { |
| 80 | + var err error |
| 81 | + |
| 82 | + unresolved = append(unresolved, n) |
| 83 | + deps := d.edges[n] |
| 84 | + for _, c := range deps { |
| 85 | + if !slices.Contains(resolved, c) { |
| 86 | + if slices.Contains(unresolved, c) { |
| 87 | + return nil, nil, fmt.Errorf("circular dependency detected: %v -> %v", n, c) |
| 88 | + } |
| 89 | + resolved, unresolved, err = d.resolve(c, resolved, unresolved) |
| 90 | + if err != nil { |
| 91 | + return nil, nil, err |
| 92 | + } |
| 93 | + } |
| 94 | + } |
| 95 | + return append(resolved, n), remove(unresolved, n), nil |
| 96 | +} |
| 97 | + |
| 98 | +// Remove a node from a list of nodes and return the new list |
| 99 | +func remove(nodes []Node, n Node) []Node { |
| 100 | + i := slices.Index(nodes, n) |
| 101 | + if i < 0 { |
| 102 | + return nodes |
| 103 | + } |
| 104 | + return append(nodes[:i], nodes[i+1:]...) |
| 105 | +} |
0 commit comments