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 pathtranslate.go
188 lines (149 loc) · 5.45 KB
/
translate.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
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
package core
import (
"github.com/hashicorp/hcl/v2"
"github.com/zclconf/go-cty/cty"
"bridgedl/config"
"bridgedl/graph"
"bridgedl/k8s"
"bridgedl/translation"
)
// BridgeTranslator translates a Bridge into a collection of Kubernetes API
// objects for deploying that Bridge.
type BridgeTranslator struct {
Impls *componentImpls
Bridge *config.Bridge
}
// Translate performs the translation.
func (t *BridgeTranslator) Translate(g *graph.DirectedGraph) ([]interface{}, hcl.Diagnostics) {
var diags hcl.Diagnostics
var bridgeManifests []interface{}
eval := NewEvaluator()
// The returned SCCs are topologically sorted as a byproduct of the
// cycle detection algorithm.
//
// If the Bridge is acyclic, there is no strongly connected component
// with a size greater than 1. The topological order guarantees that
// each evaluated component has access to all the variables (event
// addresses) it needs, because those were evaluated ahead of time.
//
// If the Bridge contains cycles, it exists strongly connected
// components with a size greater than 1 within the graph. In this
// case, the first component evaluated in each cycle will always be
// missing the event address of its successor, and this gap is
// compensated for by injecting a temporary placeholder value into the
// evaluation context. The "incomplete" component is re-evaluated once
// all addresses in the cycle have been determined.
sccs := g.StronglyConnectedComponents()
for _, scc := range sccs {
manifests, translDiags := translateComponents(eval, scc)
diags = diags.Extend(translDiags)
bridgeManifests = append(bridgeManifests, manifests...)
}
return bridgeManifests, diags
}
// translateComponents translates all components from a list of graph vertices.
func translateComponents(e *Evaluator, vs []graph.Vertex) ([]interface{}, hcl.Diagnostics) {
var diags hcl.Diagnostics
var manifests []interface{}
var incompleteDecodeQueue []MessagingComponentVertex
// first pass: push addresses of visited components to the Evaluator,
// translate immediately if the addresses of all successors allowed the
// config body to be fully decoded
for _, v := range vs {
cmp, ok := v.(MessagingComponentVertex)
if !ok {
continue
}
ref, ok := v.(ReferenceableVertex)
if ok {
evalDiags := appendToEvaluator(e, ref)
diags = diags.Extend(evalDiags)
}
cfg, evDst, complete, cfgDiags := configAndDestination(e, cmp)
diags = diags.Extend(cfgDiags)
if cfgDiags.HasErrors() {
continue
}
// We are in a cycle and not all successors have been evaluated
// yet. Push the component to a queue and delay its second
// evaluation to a point where the Evaluator contains all
// necessary addresses.
if !complete {
incompleteDecodeQueue = append(incompleteDecodeQueue, cmp)
continue
}
res, translDiags := translate(e, cmp, cfg, evDst)
diags = diags.Extend(translDiags)
manifests = append(manifests, res...)
}
// second pass: remaining components which evaluation was delayed due
// to cycles (incomplete evaluation context)
for _, cmp := range incompleteDecodeQueue {
cfg, evDst, complete, cfgDiags := configAndDestination(e, cmp)
diags = diags.Extend(cfgDiags)
if cfgDiags.HasErrors() {
continue
}
// It is not acceptable to end up with an incomplete decoding
// on a second visit. After all vertices in the strongly
// connected componenent have been evaluated, the Evaluator
// should contain all addresses.
if !complete {
diags = diags.Append(undecodableDiagnostic(cmp.ComponentAddr()))
continue
}
res, translDiags := translate(e, cmp, cfg, evDst)
diags = diags.Extend(translDiags)
manifests = append(manifests, res...)
}
return manifests, diags
}
// translate invokes the translator of the given component.
func translate(e *Evaluator, cmp MessagingComponentVertex, cfg, evDst cty.Value) (
[]interface{}, hcl.Diagnostics) {
var diags hcl.Diagnostics
cmpAddr := cmp.ComponentAddr()
transl, ok := cmp.Implementation().(translation.Translatable)
if !ok {
diags = diags.Append(noTranslatableDiagnostic(cmpAddr))
return nil, diags
}
return transl.Manifests(cmpAddr.Identifier, cfg, evDst), diags
}
// appendToEvaluator appends the event address of the given referenceable
// Bridge component to an Evaluator.
func appendToEvaluator(e *Evaluator, cmp ReferenceableVertex) hcl.Diagnostics {
var diags hcl.Diagnostics
cmpAddr := cmp.ComponentAddr()
if e.HasVariable(cmpAddr.Category.String(), cmpAddr.Identifier) {
return diags
}
evAddr, _, evAddrDiags := cmp.EventAddress(e)
diags = diags.Extend(evAddrDiags)
if evAddrDiags.HasErrors() {
return diags
}
e.InsertVariable(cmpAddr.Category.String(), cmpAddr.Identifier, evAddr)
return diags
}
// configAndDestination returns the decoded configuration of the given
// component as well as its event destination, if applicable.
func configAndDestination(e *Evaluator, v MessagingComponentVertex) (
cfg, evDst cty.Value, complete bool, diags hcl.Diagnostics) {
cfgComplete := true
dstComplete := true
cfg = cty.NullVal(cty.DynamicPseudoType)
if dec, ok := v.(DecodableConfigVertex); ok {
var cfgDiags hcl.Diagnostics
cfg, cfgComplete, cfgDiags = dec.DecodedConfig(e)
diags = diags.Extend(cfgDiags)
}
evDst = cty.NullVal(k8s.DestinationCty)
if sdr, ok := v.(EventSenderVertex); ok {
var dstDiags hcl.Diagnostics
evDst, dstComplete, dstDiags = sdr.EventDestination(e)
diags = diags.Extend(dstDiags)
}
complete = cfgComplete && dstComplete
return cfg, evDst, complete, diags
}