feel 0.0.1 → 0.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/README.md +23 -35
- data/lib/{spot_feel → feel}/configuration.rb +1 -1
- data/lib/{spot_feel/spot_feel.treetop → feel/feel.treetop} +10 -15
- data/lib/feel/literal_expression.rb +372 -0
- data/lib/{spot_feel → feel}/nodes.rb +51 -22
- data/lib/{spot_feel → feel}/parser.rb +7 -7
- data/lib/feel/unary_tests.rb +25 -0
- data/lib/feel/version.rb +5 -0
- data/lib/feel.rb +41 -0
- metadata +19 -46
- data/lib/spot_feel/dmn/decision.rb +0 -50
- data/lib/spot_feel/dmn/decision_table.rb +0 -53
- data/lib/spot_feel/dmn/definitions.rb +0 -68
- data/lib/spot_feel/dmn/information_requirement.rb +0 -29
- data/lib/spot_feel/dmn/input.rb +0 -28
- data/lib/spot_feel/dmn/literal_expression.rb +0 -374
- data/lib/spot_feel/dmn/output.rb +0 -29
- data/lib/spot_feel/dmn/rule.rb +0 -63
- data/lib/spot_feel/dmn/unary_tests.rb +0 -27
- data/lib/spot_feel/dmn/variable.rb +0 -27
- data/lib/spot_feel/dmn.rb +0 -17
- data/lib/spot_feel/version.rb +0 -5
- data/lib/spot_feel.rb +0 -63
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 2dc28e5cc40b7473deaec4d6a8c7c22d63f0674cde48b8a8b3a3b34a00da56ba
|
4
|
+
data.tar.gz: ff19b52d1d239a7918001f0c60995e92ea0fa8b67b6389e9e157d00719ff15ca
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: a69d2493f1d27317e8956abf57a8929bcd216c48f93ae0ca978cb6d49ddc5d274e50819f2983916bd468a93b01c63291f7c8b34b0a08cfb93de7c2923741e459
|
7
|
+
data.tar.gz: 2c67dca58835e521a6b89395008d20bde423cd8921f2216c340fad2035940d6d1cb482f44f0d91c36b4600f262c5c64bce865387b1d9951a643e7f4777d4b115
|
data/README.md
CHANGED
@@ -1,12 +1,12 @@
|
|
1
|
-
#
|
1
|
+
# FEEL
|
2
2
|
|
3
|
-
A light-weight
|
3
|
+
A light-weight FEEL expression evaluator ruby gem.
|
4
4
|
|
5
5
|
This gem implements a subset of FEEL (Friendly Enough Expression Language) as defined in the [DMN 1.3 specification](https://www.omg.org/spec/DMN/1.3/PDF) with some additional extensions.
|
6
6
|
|
7
7
|
FEEL expressions are parsed into an abstract syntax tree (AST) and then evaluated in a context. The context is a hash of variables and functions to be resolved inside the expression.
|
8
8
|
|
9
|
-
Expressions are safe, side-effect free, and deterministic. They are ideal for capturing business logic for storage in a database or embedded in DMN, BPMN, or Form documents for execution in a workflow engine
|
9
|
+
Expressions are safe, side-effect free, and deterministic. They are ideal for capturing business logic for storage in a database or embedded in DMN, BPMN, or Form documents for execution in a workflow engine.
|
10
10
|
|
11
11
|
This project was inspired by these excellent libraries:
|
12
12
|
|
@@ -18,7 +18,7 @@ This project was inspired by these excellent libraries:
|
|
18
18
|
To evaluate an expression:
|
19
19
|
|
20
20
|
```ruby
|
21
|
-
|
21
|
+
FEEL.evaluate('"👋 Hello " + name', variables: { name: "World" })
|
22
22
|
# => "👋 Hello World"
|
23
23
|
```
|
24
24
|
|
@@ -31,55 +31,50 @@ variables = {
|
|
31
31
|
age: 59,
|
32
32
|
}
|
33
33
|
}
|
34
|
-
|
34
|
+
FEEL.evaluate('if person.age >= 18 then "adult" else "minor"', variables:)
|
35
35
|
# => "adult"
|
36
36
|
```
|
37
37
|
|
38
|
+
Strict mode will raise an exception if an expression references a variable that is not defined in the context.
|
39
|
+
|
40
|
+
```ruby
|
41
|
+
FEEL.configure do |config|
|
42
|
+
config.strict = true
|
43
|
+
end
|
44
|
+
|
45
|
+
LiteralExpression.new(text: "person.agx").evaluate({ "person": { "name": "Bob", "age": 32 } })
|
46
|
+
# => raises EvaluationError("Identifier person.agx not found. Did you mean person.age?")
|
47
|
+
```
|
48
|
+
|
38
49
|
Calling a built-in function:
|
39
50
|
|
40
51
|
```ruby
|
41
|
-
|
52
|
+
FEEL.evaluate('sum([1, 2, 3])')
|
42
53
|
# => 6
|
43
54
|
```
|
44
55
|
|
45
56
|
Calling a user-defined function:
|
46
57
|
|
47
58
|
```ruby
|
48
|
-
|
59
|
+
FEEL.config.functions = {
|
49
60
|
"reverse": ->(s) { s.reverse }
|
50
61
|
}
|
51
|
-
|
62
|
+
DMN.evaluate('reverse("Hello World!")', functions:)
|
52
63
|
# => "!dlroW olleH"
|
53
64
|
```
|
54
65
|
|
55
66
|
To evaluate a unary tests:
|
56
67
|
|
57
68
|
```ruby
|
58
|
-
|
69
|
+
FEEL.test(3, '<= 10, > 50'))
|
59
70
|
# => true
|
60
71
|
```
|
61
72
|
|
62
73
|
```ruby
|
63
|
-
|
74
|
+
FEEL.test("Eric", '"Bob", "Holly", "Eric"')
|
64
75
|
# => true
|
65
76
|
```
|
66
77
|
|
67
|
-

|
68
|
-
|
69
|
-
To evaluate a DMN decision table:
|
70
|
-
|
71
|
-
```ruby
|
72
|
-
variables = {
|
73
|
-
violation: {
|
74
|
-
type: "speed",
|
75
|
-
actual_speed: 100,
|
76
|
-
speed_limit: 65,
|
77
|
-
}
|
78
|
-
}
|
79
|
-
result = SpotFeel.decide('fine_decision', definitions_xml: fixture_source("fine.dmn"), variables:)
|
80
|
-
# => { "amount" => 1000, "points" => 7 })
|
81
|
-
```
|
82
|
-
|
83
78
|
To get a list of variables or functions used in an expression:
|
84
79
|
|
85
80
|
```ruby
|
@@ -143,25 +138,18 @@ UnaryTests.new(text: '> speed - speed_limit').variable_names
|
|
143
138
|
- [x] Context: `get entries`, `get value`, `get keys`
|
144
139
|
- [x] Temporal: `now`, `today`, `day of week`, `day of year`, `month of year`, `week of year`
|
145
140
|
|
146
|
-
### DMN
|
147
|
-
|
148
|
-
- [x] Parse DMN XML documents
|
149
|
-
- [x] Evaluate DMN Decision Tables
|
150
|
-
- [x] Evaluate dependent DMN Decision Tables
|
151
|
-
- [x] Evaluate Expression Decisions
|
152
|
-
|
153
141
|
## Installation
|
154
142
|
|
155
143
|
Execute:
|
156
144
|
|
157
145
|
```bash
|
158
|
-
$ bundle add "
|
146
|
+
$ bundle add "feel"
|
159
147
|
```
|
160
148
|
|
161
149
|
Or install it directly:
|
162
150
|
|
163
151
|
```bash
|
164
|
-
$ gem install
|
152
|
+
$ gem install feel
|
165
153
|
```
|
166
154
|
|
167
155
|
### Setup
|
@@ -1,4 +1,4 @@
|
|
1
|
-
grammar
|
1
|
+
grammar FEEL
|
2
2
|
|
3
3
|
rule start
|
4
4
|
expression_or_tests
|
@@ -370,27 +370,22 @@ grammar SpotFeel
|
|
370
370
|
end
|
371
371
|
|
372
372
|
rule double_string_character
|
373
|
-
!('"' / "\\"
|
374
|
-
"\\" escape_sequence
|
375
|
-
line_continuation
|
373
|
+
!('"' / "\\") source_character /
|
374
|
+
"\\" escape_sequence
|
376
375
|
end
|
377
376
|
|
378
377
|
rule single_string_character
|
379
|
-
!("'" / "\\"
|
380
|
-
"\\" escape_sequence
|
381
|
-
line_continuation
|
378
|
+
!("'" / "\\") source_character /
|
379
|
+
"\\" escape_sequence
|
382
380
|
end
|
383
381
|
|
384
382
|
rule source_character
|
385
383
|
.
|
386
384
|
end
|
387
385
|
|
388
|
-
rule line_continuation
|
389
|
-
"\\" __ line_terminator_sequence __
|
390
|
-
end
|
391
|
-
|
392
386
|
rule escape_sequence
|
393
|
-
character_escape_sequence
|
387
|
+
character_escape_sequence /
|
388
|
+
line_terminator_sequence
|
394
389
|
end
|
395
390
|
|
396
391
|
rule character_escape_sequence
|
@@ -398,15 +393,15 @@ grammar SpotFeel
|
|
398
393
|
end
|
399
394
|
|
400
395
|
rule single_escape_character
|
401
|
-
"'" / '"' / "\\"
|
396
|
+
"'" / '"' / "\\" / "n" / "r" / "t"
|
402
397
|
end
|
403
398
|
|
404
399
|
rule line_terminator
|
405
|
-
|
400
|
+
"\n" / "\r" / "\r\n"
|
406
401
|
end
|
407
402
|
|
408
403
|
rule line_terminator_sequence
|
409
|
-
|
404
|
+
line_terminator
|
410
405
|
end
|
411
406
|
|
412
407
|
#
|
@@ -0,0 +1,372 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module FEEL
|
4
|
+
class LiteralExpression
|
5
|
+
attr_reader :id, :text
|
6
|
+
|
7
|
+
def self.from_json(json)
|
8
|
+
LiteralExpression.new(id: json[:id], text: json[:text])
|
9
|
+
end
|
10
|
+
|
11
|
+
def initialize(id: nil, text:)
|
12
|
+
@id = id
|
13
|
+
@text = text&.strip
|
14
|
+
end
|
15
|
+
|
16
|
+
def tree
|
17
|
+
@tree ||= FEEL::Parser.parse(text)
|
18
|
+
end
|
19
|
+
|
20
|
+
def valid?
|
21
|
+
return false if text.blank?
|
22
|
+
tree.present?
|
23
|
+
end
|
24
|
+
|
25
|
+
def evaluate(variables = {})
|
26
|
+
tree.eval(functions.merge(variables))
|
27
|
+
end
|
28
|
+
|
29
|
+
def functions
|
30
|
+
builtins = LiteralExpression.builtin_functions
|
31
|
+
custom = (FEEL.config.functions || {})
|
32
|
+
ActiveSupport::HashWithIndifferentAccess.new(builtins.merge(custom))
|
33
|
+
end
|
34
|
+
|
35
|
+
def named_functions
|
36
|
+
# Initialize a set to hold the qualified names
|
37
|
+
function_names = Set.new
|
38
|
+
|
39
|
+
# Define a lambda for the recursive function
|
40
|
+
walk_tree = lambda do |node|
|
41
|
+
# If the node is a qualified name, add it to the set
|
42
|
+
if node.is_a?(FEEL::FunctionInvocation)
|
43
|
+
function_names << node.fn_name.text_value
|
44
|
+
end
|
45
|
+
|
46
|
+
# Recursively walk the child nodes
|
47
|
+
node.elements&.each do |child|
|
48
|
+
walk_tree.call(child)
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
# Start walking the tree from the root
|
53
|
+
walk_tree.call(tree)
|
54
|
+
|
55
|
+
# Return the array of functions
|
56
|
+
function_names.to_a
|
57
|
+
end
|
58
|
+
|
59
|
+
def named_variables
|
60
|
+
# Initialize a set to hold the qualified names
|
61
|
+
qualified_names = Set.new
|
62
|
+
|
63
|
+
# Define a lambda for the recursive function
|
64
|
+
walk_tree = lambda do |node|
|
65
|
+
# If the node is a qualified name, add it to the set
|
66
|
+
if node.is_a?(FEEL::QualifiedName)
|
67
|
+
qualified_names << node.text_value
|
68
|
+
end
|
69
|
+
|
70
|
+
# Recursively walk the child nodes
|
71
|
+
node.elements&.each do |child|
|
72
|
+
walk_tree.call(child)
|
73
|
+
end
|
74
|
+
end
|
75
|
+
|
76
|
+
# Start walking the tree from the root
|
77
|
+
walk_tree.call(tree)
|
78
|
+
|
79
|
+
# Return the array of qualified names
|
80
|
+
qualified_names.to_a
|
81
|
+
end
|
82
|
+
|
83
|
+
def self.builtin_functions
|
84
|
+
HashWithIndifferentAccess.new({
|
85
|
+
# Conversion functions
|
86
|
+
"string": ->(from) {
|
87
|
+
return if from.nil?
|
88
|
+
from.to_s
|
89
|
+
},
|
90
|
+
"number": ->(from) {
|
91
|
+
return if from.nil?
|
92
|
+
from.include?(".") ? from.to_f : from.to_i
|
93
|
+
},
|
94
|
+
# Boolean functions
|
95
|
+
"not": ->(value) {
|
96
|
+
if value == true || value == false
|
97
|
+
!value
|
98
|
+
end
|
99
|
+
},
|
100
|
+
"is defined": ->(value) {
|
101
|
+
return if value.nil?
|
102
|
+
!value.nil?
|
103
|
+
},
|
104
|
+
"get or else": ->(value, default) {
|
105
|
+
value.nil? ? default : value
|
106
|
+
},
|
107
|
+
# String functions
|
108
|
+
"substring": ->(string, start, length) {
|
109
|
+
return if string.nil? || start.nil?
|
110
|
+
return "" if length.nil?
|
111
|
+
string[start - 1, length]
|
112
|
+
},
|
113
|
+
"substring before": ->(string, match) {
|
114
|
+
return if string.nil? || match.nil?
|
115
|
+
string.split(match).first
|
116
|
+
},
|
117
|
+
"substring after": ->(string, match) {
|
118
|
+
return if string.nil? || match.nil?
|
119
|
+
string.split(match).last
|
120
|
+
},
|
121
|
+
"string length": ->(string) {
|
122
|
+
return if string.nil?
|
123
|
+
string.length
|
124
|
+
},
|
125
|
+
"upper case": ->(string) {
|
126
|
+
return if string.nil?
|
127
|
+
string.upcase
|
128
|
+
},
|
129
|
+
"lower case": -> (string) {
|
130
|
+
return if string.nil?
|
131
|
+
string.downcase
|
132
|
+
},
|
133
|
+
"contains": ->(string, match) {
|
134
|
+
return if string.nil? || match.nil?
|
135
|
+
string.include?(match)
|
136
|
+
},
|
137
|
+
"starts with": ->(string, match) {
|
138
|
+
return if string.nil? || match.nil?
|
139
|
+
string.start_with?(match)
|
140
|
+
},
|
141
|
+
"ends with": ->(string, match) {
|
142
|
+
return if string.nil? || match.nil?
|
143
|
+
string.end_with?(match)
|
144
|
+
},
|
145
|
+
"matches": ->(string, match) {
|
146
|
+
return if string.nil? || match.nil?
|
147
|
+
string.match?(match)
|
148
|
+
},
|
149
|
+
"replace": ->(string, match, replacement) {
|
150
|
+
return if string.nil? || match.nil? || replacement.nil?
|
151
|
+
string.gsub(match, replacement)
|
152
|
+
},
|
153
|
+
"split": ->(string, match) {
|
154
|
+
return if string.nil? || match.nil?
|
155
|
+
string.split(match)
|
156
|
+
},
|
157
|
+
"strip": -> (string) {
|
158
|
+
return if string.nil?
|
159
|
+
string.strip
|
160
|
+
},
|
161
|
+
"extract": -> (string, pattern) {
|
162
|
+
return if string.nil? || pattern.nil?
|
163
|
+
string.match(pattern).captures
|
164
|
+
},
|
165
|
+
# Numeric functions
|
166
|
+
"decimal": ->(n, scale) {
|
167
|
+
return if n.nil? || scale.nil?
|
168
|
+
n.round(scale)
|
169
|
+
},
|
170
|
+
"floor": ->(n) {
|
171
|
+
return if n.nil?
|
172
|
+
n.floor
|
173
|
+
},
|
174
|
+
"ceiling": ->(n) {
|
175
|
+
return if n.nil?
|
176
|
+
n.ceil
|
177
|
+
},
|
178
|
+
"round up": ->(n) {
|
179
|
+
return if n.nil?
|
180
|
+
n.ceil
|
181
|
+
},
|
182
|
+
"round down": ->(n) {
|
183
|
+
return if n.nil?
|
184
|
+
n.floor
|
185
|
+
},
|
186
|
+
"abs": ->(n) {
|
187
|
+
return if n.nil?
|
188
|
+
n.abs
|
189
|
+
},
|
190
|
+
"modulo": ->(n, divisor) {
|
191
|
+
return if n.nil? || divisor.nil?
|
192
|
+
n % divisor
|
193
|
+
},
|
194
|
+
"sqrt": ->(n) {
|
195
|
+
return if n.nil?
|
196
|
+
Math.sqrt(n)
|
197
|
+
},
|
198
|
+
"log": ->(n) {
|
199
|
+
return if n.nil?
|
200
|
+
Math.log(n)
|
201
|
+
},
|
202
|
+
"exp": ->(n) {
|
203
|
+
return if n.nil?
|
204
|
+
Math.exp(n)
|
205
|
+
},
|
206
|
+
"odd": ->(n) {
|
207
|
+
return if n.nil?
|
208
|
+
n.odd?
|
209
|
+
},
|
210
|
+
"even": ->(n) {
|
211
|
+
return if n.nil?
|
212
|
+
n.even?
|
213
|
+
},
|
214
|
+
"random number": ->(n) {
|
215
|
+
return if n.nil?
|
216
|
+
rand(n)
|
217
|
+
},
|
218
|
+
# List functions
|
219
|
+
"list contains": ->(list, match) {
|
220
|
+
return if list.nil?
|
221
|
+
return false if match.nil?
|
222
|
+
list.include?(match)
|
223
|
+
},
|
224
|
+
"count": ->(list) {
|
225
|
+
return if list.nil?
|
226
|
+
return 0 if list.empty?
|
227
|
+
list.length
|
228
|
+
},
|
229
|
+
"min": ->(list) {
|
230
|
+
return if list.nil?
|
231
|
+
list.min
|
232
|
+
},
|
233
|
+
"max": ->(list) {
|
234
|
+
return if list.nil?
|
235
|
+
list.max
|
236
|
+
},
|
237
|
+
"sum": ->(list) {
|
238
|
+
return if list.nil?
|
239
|
+
list.sum
|
240
|
+
},
|
241
|
+
"product": ->(list) {
|
242
|
+
return if list.nil?
|
243
|
+
list.inject(:*)
|
244
|
+
},
|
245
|
+
"mean": ->(list) {
|
246
|
+
return if list.nil?
|
247
|
+
list.sum / list.length
|
248
|
+
},
|
249
|
+
"median": ->(list) {
|
250
|
+
return if list.nil?
|
251
|
+
list.sort[list.length / 2]
|
252
|
+
},
|
253
|
+
"stddev": ->(list) {
|
254
|
+
return if list.nil?
|
255
|
+
mean = list.sum / list.length.to_f
|
256
|
+
Math.sqrt(list.map { |n| (n - mean)**2 }.sum / list.length)
|
257
|
+
},
|
258
|
+
"mode": ->(list) {
|
259
|
+
return if list.nil?
|
260
|
+
list.group_by(&:itself).values.max_by(&:size).first
|
261
|
+
},
|
262
|
+
"all": ->(list) {
|
263
|
+
return if list.nil?
|
264
|
+
list.all?
|
265
|
+
},
|
266
|
+
"any": ->(list) {
|
267
|
+
return if list.nil?
|
268
|
+
list.any?
|
269
|
+
},
|
270
|
+
"sublist": ->(list, start, length) {
|
271
|
+
return if list.nil? || start.nil?
|
272
|
+
return [] if length.nil?
|
273
|
+
list[start - 1, length]
|
274
|
+
},
|
275
|
+
"append": ->(list, item) {
|
276
|
+
return if list.nil?
|
277
|
+
list + [item]
|
278
|
+
},
|
279
|
+
"concatenate": ->(list1, list2) {
|
280
|
+
return [nil, nil] if list1.nil? && list2.nil?
|
281
|
+
return [nil] + list2 if list1.nil?
|
282
|
+
return list1 + [nil] if list2.nil?
|
283
|
+
Array.wrap(list1) + Array.wrap(list2)
|
284
|
+
},
|
285
|
+
"insert before": ->(list, position, item) {
|
286
|
+
return if list.nil? || position.nil?
|
287
|
+
list.insert(position - 1, item)
|
288
|
+
},
|
289
|
+
"remove": ->(list, position) {
|
290
|
+
return if list.nil? || position.nil?
|
291
|
+
list.delete_at(position - 1); list
|
292
|
+
},
|
293
|
+
"reverse": ->(list) {
|
294
|
+
return if list.nil?
|
295
|
+
list.reverse
|
296
|
+
},
|
297
|
+
"index of": ->(list, match) {
|
298
|
+
return if list.nil?
|
299
|
+
return [] if match.nil?
|
300
|
+
list.index(match) + 1
|
301
|
+
},
|
302
|
+
"union": ->(list1, list2) {
|
303
|
+
return if list1.nil? || list2.nil?
|
304
|
+
list1 | list2
|
305
|
+
},
|
306
|
+
"distinct values": ->(list) {
|
307
|
+
return if list.nil?
|
308
|
+
list.uniq
|
309
|
+
},
|
310
|
+
"duplicate values": ->(list) {
|
311
|
+
return if list.nil?
|
312
|
+
list.select { |e| list.count(e) > 1 }.uniq
|
313
|
+
},
|
314
|
+
"flatten": ->(list) {
|
315
|
+
return if list.nil?
|
316
|
+
list.flatten
|
317
|
+
},
|
318
|
+
"sort": ->(list) {
|
319
|
+
return if list.nil?
|
320
|
+
list.sort
|
321
|
+
},
|
322
|
+
"string join": ->(list, separator) {
|
323
|
+
return if list.nil?
|
324
|
+
list.join(separator)
|
325
|
+
},
|
326
|
+
# Context functions
|
327
|
+
"get value": ->(context, name) {
|
328
|
+
return if context.nil? || name.nil?
|
329
|
+
context[name]
|
330
|
+
},
|
331
|
+
"context put": ->(context, name, value) {
|
332
|
+
return if context.nil? || name.nil?
|
333
|
+
context[name] = value; context
|
334
|
+
},
|
335
|
+
"context merge": ->(context1, context2) {
|
336
|
+
return if context1.nil? || context2.nil?
|
337
|
+
context1.merge(context2)
|
338
|
+
},
|
339
|
+
"get entries": ->(context) {
|
340
|
+
return if context.nil?
|
341
|
+
context.entries
|
342
|
+
},
|
343
|
+
# Temporal functions
|
344
|
+
"now": ->() { Time.now },
|
345
|
+
"today": ->() { Date.today },
|
346
|
+
"day of week": ->(date) {
|
347
|
+
return if date.nil?
|
348
|
+
date.wday
|
349
|
+
},
|
350
|
+
"day of year": ->(date) {
|
351
|
+
return if date.nil?
|
352
|
+
date.yday
|
353
|
+
},
|
354
|
+
"week of year": ->(date) {
|
355
|
+
return if date.nil?
|
356
|
+
date.cweek
|
357
|
+
},
|
358
|
+
"month of year": ->(date) {
|
359
|
+
return if date.nil?
|
360
|
+
date.month
|
361
|
+
},
|
362
|
+
})
|
363
|
+
end
|
364
|
+
|
365
|
+
def as_json
|
366
|
+
{
|
367
|
+
id: id,
|
368
|
+
text: text,
|
369
|
+
}
|
370
|
+
end
|
371
|
+
end
|
372
|
+
end
|