-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathvariable.go
248 lines (213 loc) · 8.16 KB
/
variable.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
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
package skipper
import (
"fmt"
"regexp"
"strings"
)
// valid variables: ${foo:bar} ${foo:bar:baz} ${something}
// invalid variables: ${foo:} ${bar::} ${:bar}
var variableRegex = regexp.MustCompile(`\$\{((\w*)(\:\w+)*)\}`)
// Variable is a keyword which self-references the Data map it is defined in.
// A Variable has the form ${key:key}.
type Variable struct {
// Name of the variable is whatever string is between ${}.
// + For dynamic variables, this can be a ':' separated string which points somewhere into the Data map.
// The reason we use ':' is to improve readability between curly braces.
// + For predefined variables, this can be any string and must not be a path into the Data map.
Name string
// Identifier is the list of keys which point to the variable itself within the data set in which it is used.
Identifier []interface{}
}
func (v Variable) FullName() string {
return fmt.Sprintf("${%s}", v.Name)
}
func (v Variable) Path() string {
var segments []string
for _, seg := range v.Identifier {
segments = append(segments, fmt.Sprint(seg))
}
return strings.Join(segments, ".")
}
func (v Variable) NameAsIdentifier() (id []interface{}) {
tmp := strings.Split(v.Name, ":")
id = make([]interface{}, len(tmp))
for i := 0; i < len(tmp); i++ {
id[i] = tmp[i]
}
return id
}
// FindVariables leverages the [FindValues] function of the given Data to extract
// all variables by using the [variableFindValueFunc] as callback.
func FindVariables(data Data) ([]Variable, error) {
var foundValues []interface{}
err := data.FindValues(variableFindValueFunc(), &foundValues)
if err != nil {
return nil, err
}
var foundVariables []Variable
for _, val := range foundValues {
// variableFindValueFunc returns []Variable so we need to ensure that matches
vars, ok := val.([]Variable)
if !ok {
return nil, fmt.Errorf("unexpected error during variable detection, file a bug report")
}
foundVariables = append(foundVariables, vars...)
}
return foundVariables, nil
}
// ReplaceVariables searches and replaces variables defined in data.
// The classFiles are used for local referencing variables (class internal references).
// predefinedVariables can be used to provide global user-defined variables.
func ReplaceVariables(data Data, classFiles []*Class, predefinedVariables map[string]interface{}) (err error) {
isPredefinedVariable := func(variable Variable) bool {
for name := range predefinedVariables {
if strings.EqualFold(variable.Name, name) {
return true
}
}
return false
}
// variables can be ignored, they are stored here :)
ignoredVariables := []Variable{}
// TODO: gosh, make this a standalone function already
replaceVariable := func(variable Variable) error {
var targetValue interface{}
if isPredefinedVariable(variable) {
targetValue = predefinedVariables[variable.Name]
} else {
// targetValue is the value on which the variable points to.
// This is the value we need to replace the variable with
targetValue, err = data.GetPath(variable.NameAsIdentifier()...)
if err != nil {
// for any other error than a 'key not found' there is nothing we can do
if !strings.Contains(err.Error(), "key not found") {
return fmt.Errorf("reference to invalid variable '%s': %w", variable.FullName(), err)
}
// Local variable handling
//
// at this point we have failed to resolve the variable using 'absolute' paths
// but the variable may be only locally defined which means we need to change the lookup path.
// We iterate over all classes and attempt to resolve the variable within that limited scope.
for i, class := range classFiles {
// if the value to which the variable points is valid inside the class scope, we just need to add the class identifier
// if the combination works this means we have found ourselves a local variable and we can set the targetValue
fullPath := []interface{}{}
fullPath = append(fullPath, class.NameAsIdentifier()...)
// edge case: the class root key is 'foo', and the variable used references it like ${foo:bar:baz}
// this would result in the full path being 'foo foo bar baz', hence we need to strip the class name from the variable reference.
if strings.EqualFold(class.RootKey(), variable.NameAsIdentifier()[0].(string)) {
fullPath = append(fullPath, variable.NameAsIdentifier()[1:]...)
} else {
// default case: the class root key is not used in the variable, we can add the full variable identifier
fullPath = append(fullPath, variable.NameAsIdentifier()...)
}
targetValue, err = data.GetPath(fullPath...)
// as long as not all classes have been checked, we cannot be sure that the variable is undefined (aka. key not found error)
if targetValue == nil &&
i < len(classFiles) &&
strings.Contains(err.Error(), "key not found") {
continue
}
// the local variable is really not defined at this point
if err != nil {
return fmt.Errorf("reference to invalid variable '%s': %w", variable.FullName(), err)
}
break
}
}
}
// sourceValue is the value where the variable is located. It needs to be replaced with the 'targetValue'
sourceValue, err := data.GetPath(variable.Identifier...)
if err != nil {
return err
}
// an inline variable is a variable which occurs with a context and not alone
// inline variable: "foo ${my_variable} bar"
// not inline: "${my_variable}"
isInlineVariable := func() bool {
return variable.FullName() != sourceValue
}
// At this point, the variable might still point to something unknown (targetValue == nil).
// This case might occur if one defines an inventory value which uses bash / environment variables, which
// look like skipper variables, but actually aren't.
// In this case, we ignore this variable.
if targetValue == nil {
ignoredVariables = append(ignoredVariables, variable)
return nil
}
// if the variable is not 'inline', we are going to 'attach' whatever the variable points to
// with the variable. This allows you to import a list from a different class for example.
// class-file:
// ```
// myclass:
// foo:
// - somewhere
// - over
// - the rainbow
// ```
//
// target file:
// ```
// target:
// something: ${myclass:foo} // <-- Non-Inline import of the list under `myclass.foo`
// something_else: "hello ${myclass:foo:2}" // <-- inline variable which will be 'string replaced'
// ```
if isInlineVariable() {
sourceValue = strings.ReplaceAll(fmt.Sprint(sourceValue), variable.FullName(), fmt.Sprint(targetValue))
} else {
sourceValue = targetValue
}
// replace variable in Data
return data.SetPath(sourceValue, variable.Identifier...)
}
// Replace variables in an undefined amount of iterations.
// This needs to be done because one variable can be replaced by another, which will only be replaced in the next iteration.
var variables []Variable
for {
variables, err = FindVariables(data)
if err != nil {
return err
}
// remove the ignored variables from the found variables
// otherwise we would end up in an endless loop.
for _, ignored := range ignoredVariables {
for i, variable := range variables {
if variable.Name == ignored.Name {
variables[i] = variables[len(variables)-1]
variables = variables[:len(variables)-1]
}
}
}
if len(variables) == 0 {
break
}
for _, variable := range variables {
err = replaceVariable(variable)
if err != nil {
return err
}
}
}
return nil
}
// variableFindValueFunc implements the [FindValueFunc] and searches for variables inside [Data].
// Variables are extracted by matching the values to the [variableRegex].
// All found variables are initialized and added to the output.
// The function returns `[]Variable`.
func variableFindValueFunc() FindValueFunc {
return func(value string, path []interface{}) (interface{}, error) {
var variables []Variable
matches := variableRegex.FindAllStringSubmatch(value, -1)
if len(matches) > 0 {
for _, variable := range matches {
if len(variable) >= 2 {
variables = append(variables, Variable{
Name: variable[1],
Identifier: path,
})
}
}
}
return variables, nil
}
}