-
Notifications
You must be signed in to change notification settings - Fork 20
/
results.go
156 lines (135 loc) · 4.76 KB
/
results.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
package flows
import (
"encoding/json"
"time"
"github.com/nyaruka/goflow/excellent/types"
"github.com/nyaruka/goflow/utils"
)
// Result describes a value captured during a run's execution. It might have been implicitly created by a router, or explicitly
// created by a [set_run_result](#action:set_run_result) action.It renders as its value in a template, and has the following
// properties which can be accessed:
//
// * `value` the value of the result
// * `category` the category of the result
// * `category_localized` the localized category of the result
// * `input` the input associated with the result
// * `node_uuid` the UUID of the node where the result was created
// * `created_on` the time when the result was created
//
// Examples:
//
// @results.favorite_color -> red
// @results.favorite_color.value -> red
// @results.favorite_color.category -> Red
//
// @context result
type Result struct {
Name string `json:"name"`
Value string `json:"value"`
Category string `json:"category,omitempty"`
CategoryLocalized string `json:"category_localized,omitempty"`
NodeUUID NodeUUID `json:"node_uuid"`
Input *string `json:"input,omitempty"`
Extra json.RawMessage `json:"extra,omitempty"`
CreatedOn time.Time `json:"created_on"`
}
// Resolve resolves the passed in key to a value. Result values have a name, value, category, node and created_on
func (r *Result) Resolve(env utils.Environment, key string) types.XValue {
switch key {
case "name":
return types.NewXText(r.Name)
case "value":
return types.NewXText(r.Value)
case "category":
return types.NewXText(r.Category)
case "category_localized":
if r.CategoryLocalized == "" {
return types.NewXText(r.Category)
}
return types.NewXText(r.CategoryLocalized)
case "input":
if r.Input != nil {
return types.NewXText(*r.Input)
}
return nil
case "extra":
return types.JSONToXValue(r.Extra)
case "node_uuid":
return types.NewXText(string(r.NodeUUID))
case "created_on":
return types.NewXDateTime(r.CreatedOn)
}
return types.NewXResolveError(r, key)
}
// Describe returns a representation of this type for error messages
func (r *Result) Describe() string { return "run result" }
// Reduce is called when this object needs to be reduced to a primitive
func (r *Result) Reduce(env utils.Environment) types.XPrimitive {
return types.NewXText(r.Value)
}
// ToXJSON is called when this type is passed to @(json(...))
func (r *Result) ToXJSON(env utils.Environment) types.XText {
return types.ResolveKeys(env, r, "name", "value", "category", "category_localized", "input", "node_uuid", "created_on").ToXJSON(env)
}
var _ types.XValue = (*Result)(nil)
var _ types.XResolvable = (*Result)(nil)
// Results is our wrapper around a map of snakified result names to result objects
type Results map[string]*Result
// NewResults creates a new empty set of results
func NewResults() Results {
return make(Results, 0)
}
// Clone returns a clone of this results set
func (r Results) Clone() Results {
clone := make(Results, len(r))
for k, v := range r {
clone[k] = v
}
return clone
}
// Save saves a new result in our map. The key is saved in a snakified format
func (r Results) Save(name string, value string, category string, categoryLocalized string, nodeUUID NodeUUID, input *string, extra json.RawMessage, createdOn time.Time) {
r[utils.Snakify(name)] = &Result{
Name: name,
Value: value,
Category: category,
CategoryLocalized: categoryLocalized,
NodeUUID: nodeUUID,
Input: input,
Extra: extra,
CreatedOn: createdOn,
}
}
func (r Results) Get(key string) *Result {
return r[key]
}
// Length is called to get the length of this object
func (r Results) Length() int {
return len(r)
}
// Resolve resolves the passed in key, which is snakified before lookup
func (r Results) Resolve(env utils.Environment, key string) types.XValue {
key = utils.Snakify(key)
result, exists := r[key]
if !exists {
return types.NewXErrorf("no such run result '%s'", key)
}
return result
}
// Describe returns a representation of this type for error messages
func (r Results) Describe() string { return "run results" }
// Reduce is called when this object needs to be reduced to a primitive
func (r Results) Reduce(env utils.Environment) types.XPrimitive {
results := types.NewEmptyXMap()
for k, v := range r {
results.Put(k, v)
}
return results
}
// ToXJSON is called when this type is passed to @(json(...))
func (r Results) ToXJSON(env utils.Environment) types.XText {
return r.Reduce(env).ToXJSON(env)
}
var _ types.XValue = (Results)(nil)
var _ types.XLengthable = (Results)(nil)
var _ types.XResolvable = (Results)(nil)