This repository has been archived by the owner on Mar 24, 2022. It is now read-only.
forked from cloudfoundry/bosh-cli
-
Notifications
You must be signed in to change notification settings - Fork 0
/
template.go
114 lines (86 loc) · 2.36 KB
/
template.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
package template
import (
"fmt"
"regexp"
"sort"
"strings"
"github.com/cppforlife/go-patch/patch"
"gopkg.in/yaml.v2"
)
var templateFormatRegex = regexp.MustCompile(`^\(\((!?[-\w\p{L}]+)\)\)$`)
type Template struct {
bytes []byte
}
type EvaluateOpts struct {
ExpectAllKeys bool
}
func NewTemplate(bytes []byte) Template {
return Template{bytes: bytes}
}
func (t Template) Evaluate(vars Variables, ops patch.Ops, opts EvaluateOpts) ([]byte, error) {
var obj interface{}
err := yaml.Unmarshal(t.bytes, &obj)
if err != nil {
return []byte{}, err
}
obj, err = ops.Apply(obj)
if err != nil {
return []byte{}, err
}
missingVars := map[string]struct{}{}
obj = t.interpolate(obj, vars, opts, missingVars)
if len(missingVars) > 0 {
var missingVarKeys []string
for v, _ := range missingVars {
missingVarKeys = append(missingVarKeys, v)
}
sort.Strings(missingVarKeys)
return []byte{}, fmt.Errorf("Expected to find variables: %s", strings.Join(missingVarKeys, ", "))
}
bytes, err := yaml.Marshal(obj)
if err != nil {
return []byte{}, err
}
return bytes, nil
}
func (t Template) interpolate(node interface{}, vars Variables, opts EvaluateOpts, missingVars map[string]struct{}) interface{} {
switch node.(type) {
case map[interface{}]interface{}:
nodeMap := node.(map[interface{}]interface{})
for k, v := range nodeMap {
evaluatedValue := t.interpolate(v, vars, opts, missingVars)
if keyAsString, ok := k.(string); ok {
if key, eval := t.needsEvaluation(keyAsString); eval {
if foundVarKey, exists := vars[key]; exists {
delete(nodeMap, k)
k = foundVarKey
} else if opts.ExpectAllKeys {
missingVars[key] = struct{}{}
}
}
}
nodeMap[k] = evaluatedValue
}
case []interface{}:
nodeArray := node.([]interface{})
for i, x := range nodeArray {
nodeArray[i] = t.interpolate(x, vars, opts, missingVars)
}
case string:
if key, found := t.needsEvaluation(node.(string)); found {
if foundVar, exists := vars[key]; exists {
return foundVar
} else if opts.ExpectAllKeys {
missingVars[key] = struct{}{}
}
}
}
return node
}
func (t Template) needsEvaluation(value string) (string, bool) {
found := templateFormatRegex.FindAllSubmatch([]byte(value), 1)
if len(found) != 0 && len(found[0]) != 0 {
return strings.TrimPrefix(string(found[0][1]), "!"), true
}
return "", false
}