This repository was archived by the owner on Aug 4, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtransform_connect_refs.go
131 lines (102 loc) · 3.58 KB
/
transform_connect_refs.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
package core
import (
"github.com/hashicorp/hcl/v2"
"github.com/zclconf/go-cty/cty"
"bridgedl/config/addr"
"bridgedl/graph"
)
// AddressableVertex is implemented by all types used as graph.Vertex that can
// expose an address for receiving events.
type AddressableVertex interface {
// Address for receiving events.
// The returned boolean value indicates whether all expressions from
// the component's configuration could be decoded without injecting
// placeholders into the evaluation context.
EventAddress(*Evaluator) (cty.Value, bool, hcl.Diagnostics)
// This interface is embedded for convenience. During a graph
// evaluation, knowledge about the category of the component
// represented by the vertex is required to be able to populate the
// Evaluator (in addition to the event address).
MessagingComponentVertex
}
// ReferenceableVertex is implemented by all types used as graph.Vertex that
// can be referenced by other vertices.
type ReferenceableVertex interface {
Referenceable() addr.Referenceable
// In the current version of the Bridge Description Language, a
// Referenceable vertex must also expose an address and accept events.
// This may change in the future.
AddressableVertex
}
// ReferencerVertex is implemented by all types used as graph.Vertex that can
// reference other vertices.
type ReferencerVertex interface {
References() ([]*addr.Reference, hcl.Diagnostics)
}
// EventSenderVertex is implemented by all types used as graph.Vertex that may
// have a main event destination configured ("to" top-level HCL attribute).
type EventSenderVertex interface {
// Event destination.
// The returned boolean value indicates whether all expressions from
// the component's configuration could be decoded without injecting
// placeholders into the evaluation context.
EventDestination(*Evaluator) (cty.Value, bool, hcl.Diagnostics)
// If a component can send events, it can also have at least one
// reference to other components.
ReferencerVertex
}
// ConnectReferencesTransformer is a GraphTransformer that connects vertices of
// a graph based on how they reference each other.
type ConnectReferencesTransformer struct{}
var _ GraphTransformer = (*ConnectReferencesTransformer)(nil)
// Transform implements GraphTransformer.
func (t *ConnectReferencesTransformer) Transform(g *graph.DirectedGraph) hcl.Diagnostics {
var diags hcl.Diagnostics
vs := g.Vertices()
rm := NewReferenceMap(vs)
for _, v := range vs {
refs, refDiags := rm.References(v)
diags = diags.Extend(refDiags)
for _, ref := range refs {
g.Connect(v, ref)
}
}
return diags
}
// ReferenceMap is a lookup map of Referenceable vertices indexed by address.
type ReferenceMap map[string]graph.Vertex
// NewReferenceMap returns a ReferenceMap initialized from the given vertices.
func NewReferenceMap(vs graph.IndexedVertices) ReferenceMap {
rm := make(ReferenceMap)
for _, v := range vs {
ref, ok := v.(ReferenceableVertex)
if !ok {
continue
}
key := ref.Referenceable().Addr()
rm[key] = v
}
return rm
}
// References returns all the graph vertices the given vertex refers to.
func (rm ReferenceMap) References(v graph.Vertex) ([]graph.Vertex, hcl.Diagnostics) {
rfr, ok := v.(ReferencerVertex)
if !ok {
return nil, nil
}
var diags hcl.Diagnostics
var vs []graph.Vertex
refs, refDiags := rfr.References()
diags = diags.Extend(refDiags)
for _, ref := range refs {
key := ref.Subject.Addr()
v, exists := rm[key]
if !exists {
diags = diags.Append(unknownReferenceDiagnostic(ref.Subject, ref.SourceRange))
}
if v != nil {
vs = append(vs, v)
}
}
return vs, diags
}