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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: e07c50d749e9aac71d11c036197e5035c7f6a1d147d0e2ab547850c6ecf74594
4
- data.tar.gz: 8930a354648c1f3ebd3efa7279957715b7dfefbe7ce8d37a0829237373faee67
3
+ metadata.gz: 2dc28e5cc40b7473deaec4d6a8c7c22d63f0674cde48b8a8b3a3b34a00da56ba
4
+ data.tar.gz: ff19b52d1d239a7918001f0c60995e92ea0fa8b67b6389e9e157d00719ff15ca
5
5
  SHA512:
6
- metadata.gz: bf93e2d4f6ecdee1f267564639c14ee96079b7bb6c173859db8671276ed123dadec41cb9897bf1faee736d252c6374a95b845b832e2d2d8f6f8a1c5924d94c7b
7
- data.tar.gz: 66dca276a2b5daa6c0d8012618406525f5666a2dc0207afbd3185c3d961e24b5c149f97ae3289da05f715bf04b66cc0b9a8f7c441ec083a5af2ba6ec4bceee86
6
+ metadata.gz: a69d2493f1d27317e8956abf57a8929bcd216c48f93ae0ca978cb6d49ddc5d274e50819f2983916bd468a93b01c63291f7c8b34b0a08cfb93de7c2923741e459
7
+ data.tar.gz: 2c67dca58835e521a6b89395008d20bde423cd8921f2216c340fad2035940d6d1cb482f44f0d91c36b4600f262c5c64bce865387b1d9951a643e7f4777d4b115
data/README.md CHANGED
@@ -1,12 +1,12 @@
1
- # Spot Feel
1
+ # FEEL
2
2
 
3
- A light-weight DMN FEEL expression evaluator and business rule engine in Ruby.
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 like [Spot Flow](https://github.com/connectedbits/spot-flow).
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
- SpotFeel.evaluate('"👋 Hello " + name', variables: { name: "World" })
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
- SpotFeel.evaluate('if person.age >= 18 then "adult" else "minor"', variables:)
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
- SpotFeel.evaluate('sum([1, 2, 3])')
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
- SpotFeel.config.functions = {
59
+ FEEL.config.functions = {
49
60
  "reverse": ->(s) { s.reverse }
50
61
  }
51
- SpotFeel.evaluate('reverse("Hello World!")', functions:)
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
- SpotFeel.test(3, '<= 10, > 50'))
69
+ FEEL.test(3, '<= 10, > 50'))
59
70
  # => true
60
71
  ```
61
72
 
62
73
  ```ruby
63
- SpotFeel.test("Eric", '"Bob", "Holly", "Eric"')
74
+ FEEL.test("Eric", '"Bob", "Holly", "Eric"')
64
75
  # => true
65
76
  ```
66
77
 
67
- ![Decision Table](docs/media/decision_table.png)
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 "spot_feel"
146
+ $ bundle add "feel"
159
147
  ```
160
148
 
161
149
  Or install it directly:
162
150
 
163
151
  ```bash
164
- $ gem install spot_feel
152
+ $ gem install feel
165
153
  ```
166
154
 
167
155
  ### Setup
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module SpotFeel
3
+ module FEEL
4
4
  class Configuration
5
5
  attr_accessor :functions, :strict
6
6
 
@@ -1,4 +1,4 @@
1
- grammar SpotFeel
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
- !('"' / "\\" / line_terminator) source_character /
374
- "\\" escape_sequence /
375
- line_continuation
373
+ !('"' / "\\") source_character /
374
+ "\\" escape_sequence
376
375
  end
377
376
 
378
377
  rule single_string_character
379
- !("'" / "\\" / line_terminator) source_character /
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
- line_terminator_sequence
400
+ "\n" / "\r" / "\r\n"
406
401
  end
407
402
 
408
403
  rule line_terminator_sequence
409
- "\n" / "\r" / "\r\n"
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