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
@@ -1,12 +1,12 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
3
|
+
module FEEL
|
4
4
|
class Node < Treetop::Runtime::SyntaxNode
|
5
5
|
#
|
6
6
|
# Takes a context hash and returns an array of qualified names
|
7
7
|
# { "person": { "name": { "first": "Eric", "last": "Carlson" }, "age": 60 } } => ["person", "person.name.first", "person.name.last", "person.age"]
|
8
8
|
#
|
9
|
-
def qualified_names_in_context(hash = {}, prefix =
|
9
|
+
def qualified_names_in_context(hash = {}, prefix = "", qualified_names = Set.new)
|
10
10
|
hash.each do |key, value|
|
11
11
|
new_prefix = prefix.empty? ? "#{key}" : "#{prefix}.#{key}"
|
12
12
|
if value.is_a?(Hash)
|
@@ -114,13 +114,13 @@ module SpotFeel
|
|
114
114
|
second_val = second.eval(context)
|
115
115
|
|
116
116
|
case [start, finish]
|
117
|
-
when [
|
117
|
+
when ["(", ")"]
|
118
118
|
->(input) { first_val < input && input < second_val }
|
119
|
-
when [
|
119
|
+
when ["[", "]"]
|
120
120
|
->(input) { first_val <= input && input <= second_val }
|
121
|
-
when [
|
121
|
+
when ["(", "]"]
|
122
122
|
->(input) { first_val < input && input <= second_val }
|
123
|
-
when [
|
123
|
+
when ["[", ")"]
|
124
124
|
->(input) { first_val <= input && input < second_val }
|
125
125
|
end
|
126
126
|
end
|
@@ -225,11 +225,11 @@ module SpotFeel
|
|
225
225
|
class QualifiedName < Node
|
226
226
|
def eval(context = {})
|
227
227
|
if tail.empty?
|
228
|
-
raise_evaluation_error(head.text_value, context) if
|
228
|
+
raise_evaluation_error(head.text_value, context) if FEEL.config.strict && !context.key?(head.text_value.to_sym)
|
229
229
|
context[head.text_value.to_sym]
|
230
230
|
else
|
231
|
-
tail.elements.flat_map { |element| element.name.text_value.split(
|
232
|
-
raise_evaluation_error("#{head.text_value}#{tail.text_value}", context) if
|
231
|
+
tail.elements.flat_map { |element| element.name.text_value.split(".") }.inject(context[head.text_value.to_sym]) do |hash, key|
|
232
|
+
raise_evaluation_error("#{head.text_value}#{tail.text_value}", context) if FEEL.config.strict && (hash.blank? || !hash.key?(key.to_sym))
|
233
233
|
return nil unless hash
|
234
234
|
hash[key.to_sym]
|
235
235
|
end
|
@@ -330,9 +330,40 @@ module SpotFeel
|
|
330
330
|
#
|
331
331
|
# 35. string literal = '"' , { character – ('"' | vertical space) }, '"' ;
|
332
332
|
#
|
333
|
-
class StringLiteral <
|
334
|
-
def eval(
|
335
|
-
|
333
|
+
class StringLiteral < Treetop::Runtime::SyntaxNode
|
334
|
+
def eval(context={})
|
335
|
+
# Collect all characters and process escape sequences
|
336
|
+
string_value = chars.elements.map do |char|
|
337
|
+
if char.respond_to?(:text_value) && char.text_value.start_with?('\\')
|
338
|
+
process_escape_sequence(char.text_value)
|
339
|
+
else
|
340
|
+
char.text_value
|
341
|
+
end
|
342
|
+
end.join
|
343
|
+
|
344
|
+
string_value
|
345
|
+
end
|
346
|
+
|
347
|
+
private
|
348
|
+
|
349
|
+
def process_escape_sequence(escape_seq)
|
350
|
+
case escape_seq
|
351
|
+
when '\\n'
|
352
|
+
"\n"
|
353
|
+
when '\\r'
|
354
|
+
"\r"
|
355
|
+
when '\\t'
|
356
|
+
"\t"
|
357
|
+
when '\\"'
|
358
|
+
'"'
|
359
|
+
when '\\\''
|
360
|
+
"'"
|
361
|
+
when '\\\\'
|
362
|
+
'\\'
|
363
|
+
else
|
364
|
+
# Return the character after the backslash for unknown escape sequences
|
365
|
+
escape_seq[1..-1]
|
366
|
+
end
|
336
367
|
end
|
337
368
|
end
|
338
369
|
|
@@ -385,7 +416,7 @@ module SpotFeel
|
|
385
416
|
fn = context[fn_name.text_value.to_sym]
|
386
417
|
|
387
418
|
unless fn
|
388
|
-
raise_evaluation_error(fn_name.text_value, context) if
|
419
|
+
raise_evaluation_error(fn_name.text_value, context) if FEEL.config.strict
|
389
420
|
return nil
|
390
421
|
end
|
391
422
|
|
@@ -492,12 +523,12 @@ module SpotFeel
|
|
492
523
|
class Comparison < Node
|
493
524
|
def eval(context = {})
|
494
525
|
case operator.text_value
|
495
|
-
when
|
496
|
-
when
|
497
|
-
when
|
498
|
-
when
|
499
|
-
when
|
500
|
-
when
|
526
|
+
when "<" then left.eval(context) < right.eval(context)
|
527
|
+
when "<=" then left.eval(context) <= right.eval(context)
|
528
|
+
when ">=" then left.eval(context) >= right.eval(context)
|
529
|
+
when ">" then left.eval(context) > right.eval(context)
|
530
|
+
when "!=" then left.eval(context) != right.eval(context)
|
531
|
+
when "=" then left.eval(context) == right.eval(context)
|
501
532
|
end
|
502
533
|
end
|
503
534
|
end
|
@@ -522,7 +553,7 @@ module SpotFeel
|
|
522
553
|
# 53. instance of = expression , "instance" , "of" , type ;
|
523
554
|
#
|
524
555
|
class InstanceOf < Node
|
525
|
-
def eval(
|
556
|
+
def eval(_context = {})
|
526
557
|
case type.text_value
|
527
558
|
when "string"
|
528
559
|
->(input) { input.is_a?(String) }
|
@@ -556,8 +587,6 @@ module SpotFeel
|
|
556
587
|
->(input) { input.is_a?(ActiveSupport::Duration) && input.parts.keys.sort == [:seconds] }
|
557
588
|
when "time duration"
|
558
589
|
->(input) { input.is_a?(ActiveSupport::Duration) && input.parts.keys.sort == [:hours, :minutes, :seconds] }
|
559
|
-
when "years and months duration"
|
560
|
-
->(input) { input.is_a?(ActiveSupport::Duration) && input.parts.keys.sort == [:months, :years] }
|
561
590
|
when "list"
|
562
591
|
->(input) { input.is_a?(Array) }
|
563
592
|
when "interval"
|
@@ -1,12 +1,12 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module
|
3
|
+
module FEEL
|
4
4
|
class Parser
|
5
|
-
# Load the Treetop grammar from the '
|
5
|
+
# Load the Treetop grammar from the 'feel' file, and create a new
|
6
6
|
# instance of that parser as a class variable so we don't have to re-create
|
7
7
|
# it every time we need to parse a string
|
8
|
-
Treetop.load(File.expand_path(File.join(File.dirname(__FILE__),
|
9
|
-
@@parser =
|
8
|
+
Treetop.load(File.expand_path(File.join(File.dirname(__FILE__), "feel.treetop")))
|
9
|
+
@@parser = FEELParser.new
|
10
10
|
|
11
11
|
def self.parse(expression, root: nil)
|
12
12
|
@@parser.parse(expression, root:).tap do |ast|
|
@@ -15,15 +15,15 @@ module SpotFeel
|
|
15
15
|
end
|
16
16
|
|
17
17
|
def self.parse_test(expression)
|
18
|
-
@@parser.parse(expression ||
|
18
|
+
@@parser.parse(expression || "-", root: :simple_unary_tests).tap do |ast|
|
19
19
|
raise SyntaxError, "Invalid unary test: #{expression.inspect}" unless ast
|
20
20
|
end
|
21
21
|
end
|
22
22
|
|
23
23
|
def self.clean_tree(root_node)
|
24
24
|
return if(root_node.elements.nil?)
|
25
|
-
root_node.elements.delete_if{|node| node.class.name == "Treetop::Runtime::SyntaxNode" }
|
26
|
-
root_node.elements.each {|node| self.clean_tree(node) }
|
25
|
+
root_node.elements.delete_if{ |node| node.class.name == "Treetop::Runtime::SyntaxNode" }
|
26
|
+
root_node.elements.each { |node| self.clean_tree(node) }
|
27
27
|
end
|
28
28
|
end
|
29
29
|
end
|
@@ -0,0 +1,25 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module FEEL
|
4
|
+
class UnaryTests < LiteralExpression
|
5
|
+
attr_reader :id, :text
|
6
|
+
|
7
|
+
def self.from_json(json)
|
8
|
+
UnaryTests.new(id: json[:id], text: json[:text])
|
9
|
+
end
|
10
|
+
|
11
|
+
def tree
|
12
|
+
@tree ||= Parser.parse_test(text)
|
13
|
+
end
|
14
|
+
|
15
|
+
def valid?
|
16
|
+
return true if text.nil? || text == "-"
|
17
|
+
tree.present?
|
18
|
+
end
|
19
|
+
|
20
|
+
def test(input, variables = {})
|
21
|
+
return true if text.nil? || text == "-"
|
22
|
+
tree.eval(functions.merge(variables)).call(input)
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|
data/lib/feel/version.rb
ADDED
data/lib/feel.rb
ADDED
@@ -0,0 +1,41 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require_relative "feel/version"
|
4
|
+
|
5
|
+
require "active_support"
|
6
|
+
require "active_support/time"
|
7
|
+
require "active_support/core_ext/hash"
|
8
|
+
|
9
|
+
require "treetop"
|
10
|
+
|
11
|
+
require "feel/configuration"
|
12
|
+
require "feel/nodes"
|
13
|
+
require "feel/parser"
|
14
|
+
|
15
|
+
require "feel/literal_expression"
|
16
|
+
require "feel/unary_tests"
|
17
|
+
|
18
|
+
module FEEL
|
19
|
+
class SyntaxError < StandardError; end
|
20
|
+
class EvaluationError < StandardError; end
|
21
|
+
|
22
|
+
def self.evaluate(expression_text, variables: {})
|
23
|
+
literal_expression = FEEL::LiteralExpression.new(text: expression_text)
|
24
|
+
raise SyntaxError, "Expression is not valid" unless literal_expression.valid?
|
25
|
+
literal_expression.evaluate(variables)
|
26
|
+
end
|
27
|
+
|
28
|
+
def self.test(input, unary_tests_text, variables: {})
|
29
|
+
unary_tests = FEEL::UnaryTests.new(text: unary_tests_text)
|
30
|
+
raise SyntaxError, "Unary tests are not valid" unless unary_tests.valid?
|
31
|
+
unary_tests.test(input, variables)
|
32
|
+
end
|
33
|
+
|
34
|
+
def self.config
|
35
|
+
@config ||= Configuration.new
|
36
|
+
end
|
37
|
+
|
38
|
+
def self.configure
|
39
|
+
yield(config)
|
40
|
+
end
|
41
|
+
end
|
metadata
CHANGED
@@ -1,14 +1,13 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: feel
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.0.
|
4
|
+
version: 0.0.4
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Connected Bits
|
8
|
-
autorequire:
|
9
8
|
bindir: bin
|
10
9
|
cert_chain: []
|
11
|
-
date:
|
10
|
+
date: 2025-04-02 00:00:00.000000000 Z
|
12
11
|
dependencies:
|
13
12
|
- !ruby/object:Gem::Dependency
|
14
13
|
name: activemodel
|
@@ -39,21 +38,7 @@ dependencies:
|
|
39
38
|
- !ruby/object:Gem::Version
|
40
39
|
version: 7.0.2.3
|
41
40
|
- !ruby/object:Gem::Dependency
|
42
|
-
name:
|
43
|
-
requirement: !ruby/object:Gem::Requirement
|
44
|
-
requirements:
|
45
|
-
- - "~>"
|
46
|
-
- !ruby/object:Gem::Version
|
47
|
-
version: '1.9'
|
48
|
-
type: :runtime
|
49
|
-
prerelease: false
|
50
|
-
version_requirements: !ruby/object:Gem::Requirement
|
51
|
-
requirements:
|
52
|
-
- - "~>"
|
53
|
-
- !ruby/object:Gem::Version
|
54
|
-
version: '1.9'
|
55
|
-
- !ruby/object:Gem::Dependency
|
56
|
-
name: treetop
|
41
|
+
name: ostruct
|
57
42
|
requirement: !ruby/object:Gem::Requirement
|
58
43
|
requirements:
|
59
44
|
- - ">="
|
@@ -67,19 +52,19 @@ dependencies:
|
|
67
52
|
- !ruby/object:Gem::Version
|
68
53
|
version: '0'
|
69
54
|
- !ruby/object:Gem::Dependency
|
70
|
-
name:
|
55
|
+
name: treetop
|
71
56
|
requirement: !ruby/object:Gem::Requirement
|
72
57
|
requirements:
|
73
|
-
- -
|
58
|
+
- - '='
|
74
59
|
- !ruby/object:Gem::Version
|
75
|
-
version: 1.
|
60
|
+
version: 1.6.12
|
76
61
|
type: :runtime
|
77
62
|
prerelease: false
|
78
63
|
version_requirements: !ruby/object:Gem::Requirement
|
79
64
|
requirements:
|
80
|
-
- -
|
65
|
+
- - '='
|
81
66
|
- !ruby/object:Gem::Version
|
82
|
-
version: 1.
|
67
|
+
version: 1.6.12
|
83
68
|
- !ruby/object:Gem::Dependency
|
84
69
|
name: rake
|
85
70
|
requirement: !ruby/object:Gem::Requirement
|
@@ -271,30 +256,20 @@ extra_rdoc_files: []
|
|
271
256
|
files:
|
272
257
|
- README.md
|
273
258
|
- Rakefile
|
274
|
-
- lib/
|
275
|
-
- lib/
|
276
|
-
- lib/
|
277
|
-
- lib/
|
278
|
-
- lib/
|
279
|
-
- lib/
|
280
|
-
- lib/
|
281
|
-
- lib/
|
282
|
-
- lib/spot_feel/dmn/literal_expression.rb
|
283
|
-
- lib/spot_feel/dmn/output.rb
|
284
|
-
- lib/spot_feel/dmn/rule.rb
|
285
|
-
- lib/spot_feel/dmn/unary_tests.rb
|
286
|
-
- lib/spot_feel/dmn/variable.rb
|
287
|
-
- lib/spot_feel/nodes.rb
|
288
|
-
- lib/spot_feel/parser.rb
|
289
|
-
- lib/spot_feel/spot_feel.treetop
|
290
|
-
- lib/spot_feel/version.rb
|
259
|
+
- lib/feel.rb
|
260
|
+
- lib/feel/configuration.rb
|
261
|
+
- lib/feel/feel.treetop
|
262
|
+
- lib/feel/literal_expression.rb
|
263
|
+
- lib/feel/nodes.rb
|
264
|
+
- lib/feel/parser.rb
|
265
|
+
- lib/feel/unary_tests.rb
|
266
|
+
- lib/feel/version.rb
|
291
267
|
homepage: https://www.connectedbits.com
|
292
268
|
licenses:
|
293
269
|
- MIT
|
294
270
|
metadata:
|
295
271
|
homepage_uri: https://www.connectedbits.com
|
296
|
-
source_code_uri: https://github.com/connectedbits/feel
|
297
|
-
post_install_message:
|
272
|
+
source_code_uri: https://github.com/connectedbits/bpmn/feel
|
298
273
|
rdoc_options: []
|
299
274
|
require_paths:
|
300
275
|
- lib
|
@@ -309,9 +284,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
|
|
309
284
|
- !ruby/object:Gem::Version
|
310
285
|
version: '0'
|
311
286
|
requirements: []
|
312
|
-
rubygems_version: 3.
|
313
|
-
signing_key:
|
287
|
+
rubygems_version: 3.6.5
|
314
288
|
specification_version: 4
|
315
|
-
summary: A light-weight
|
316
|
-
Ruby.
|
289
|
+
summary: A light-weight FEEL expression evaluator in Ruby.
|
317
290
|
test_files: []
|
@@ -1,50 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module SpotFeel
|
4
|
-
module Dmn
|
5
|
-
class Decision
|
6
|
-
attr_reader :id, :name, :decision_table, :variable, :literal_expression, :information_requirements
|
7
|
-
|
8
|
-
def self.from_json(json)
|
9
|
-
information_requirements = Array.wrap(json[:information_requirement]).map { |ir| InformationRequirement.from_json(ir) } if json[:information_requirement]
|
10
|
-
decision_table = DecisionTable.from_json(json[:decision_table]) if json[:decision_table]
|
11
|
-
literal_expression = LiteralExpression.from_json(json[:literal_expression]) if json[:literal_expression]
|
12
|
-
variable = Variable.from_json(json[:variable]) if json[:variable]
|
13
|
-
Decision.new(id: json[:id], name: json[:name], decision_table:, variable:, literal_expression:, information_requirements:)
|
14
|
-
end
|
15
|
-
|
16
|
-
def initialize(id:, name:, decision_table:, variable:, literal_expression:, information_requirements:)
|
17
|
-
@id = id
|
18
|
-
@name = name
|
19
|
-
@decision_table = decision_table
|
20
|
-
@variable = variable
|
21
|
-
@literal_expression = literal_expression
|
22
|
-
@information_requirements = information_requirements
|
23
|
-
end
|
24
|
-
|
25
|
-
def evaluate(variables = {})
|
26
|
-
if literal_expression.present?
|
27
|
-
result = literal_expression.evaluate(variables)
|
28
|
-
variable.present? ? { variable.name => result } : result
|
29
|
-
elsif decision_table.present?
|
30
|
-
decision_table.evaluate(variables)
|
31
|
-
end
|
32
|
-
end
|
33
|
-
|
34
|
-
def required_decision_ids
|
35
|
-
information_requirements&.map(&:required_decision_id)
|
36
|
-
end
|
37
|
-
|
38
|
-
def as_json
|
39
|
-
{
|
40
|
-
id: id,
|
41
|
-
name: name,
|
42
|
-
decision_table: decision_table.as_json,
|
43
|
-
variable: variable.as_json,
|
44
|
-
literal_expression: literal_expression.as_json,
|
45
|
-
information_requirements: information_requirements&.map(&:as_json),
|
46
|
-
}
|
47
|
-
end
|
48
|
-
end
|
49
|
-
end
|
50
|
-
end
|
@@ -1,53 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module SpotFeel
|
4
|
-
module Dmn
|
5
|
-
class DecisionTable
|
6
|
-
attr_reader :id, :hit_policy, :inputs, :outputs, :rules
|
7
|
-
|
8
|
-
def self.from_json(json)
|
9
|
-
inputs = Array.wrap(json[:input]).map { |input| Input.from_json(input) }
|
10
|
-
outputs = Array.wrap(json[:output]).map { |output| Output.from_json(output) }
|
11
|
-
rules = Array.wrap(json[:rule]).map { |rule| Rule.from_json(rule) }
|
12
|
-
DecisionTable.new(id: json[:id], hit_policy: json[:hit_policy], inputs: inputs, outputs: outputs, rules: rules)
|
13
|
-
end
|
14
|
-
|
15
|
-
def initialize(id:, hit_policy:, inputs:, outputs:, rules:)
|
16
|
-
@id = id
|
17
|
-
@hit_policy = hit_policy&.downcase&.to_sym || :unique
|
18
|
-
@inputs = inputs
|
19
|
-
@outputs = outputs
|
20
|
-
@rules = rules
|
21
|
-
end
|
22
|
-
|
23
|
-
def evaluate(variables = {})
|
24
|
-
output_values = []
|
25
|
-
|
26
|
-
input_values = inputs.map do |input|
|
27
|
-
input.input_expression.evaluate(variables)
|
28
|
-
end
|
29
|
-
|
30
|
-
rules.each do |rule|
|
31
|
-
results = rule.evaluate(input_values, variables)
|
32
|
-
if results.all?
|
33
|
-
output_value = rule.output_value(outputs, variables)
|
34
|
-
return output_value if hit_policy == :first || hit_policy == :unique
|
35
|
-
output_values << output_value
|
36
|
-
end
|
37
|
-
end
|
38
|
-
|
39
|
-
output_values.empty? ? nil : output_values
|
40
|
-
end
|
41
|
-
|
42
|
-
def as_json
|
43
|
-
{
|
44
|
-
id: id,
|
45
|
-
hit_policy: hit_policy,
|
46
|
-
inputs: inputs.map(&:as_json),
|
47
|
-
outputs: outputs.map(&:as_json),
|
48
|
-
rules: rules.map(&:as_json),
|
49
|
-
}
|
50
|
-
end
|
51
|
-
end
|
52
|
-
end
|
53
|
-
end
|
@@ -1,68 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module SpotFeel
|
4
|
-
module Dmn
|
5
|
-
class Definitions
|
6
|
-
attr_reader :id, :name, :namespace, :exporter, :exporter_version, :execution_platform, :execution_platform_version
|
7
|
-
attr_reader :decisions
|
8
|
-
|
9
|
-
def self.from_xml(xml)
|
10
|
-
XmlHasher.configure do |config|
|
11
|
-
config.snakecase = true
|
12
|
-
config.ignore_namespaces = true
|
13
|
-
config.string_keys = false
|
14
|
-
end
|
15
|
-
json = XmlHasher.parse(xml)
|
16
|
-
Definitions.from_json(json[:definitions])
|
17
|
-
end
|
18
|
-
|
19
|
-
def self.from_json(json)
|
20
|
-
decisions = Array.wrap(json[:decision]).map { |decision| Decision.from_json(decision) }
|
21
|
-
Definitions.new(id: json[:id], name: json[:name], namespace: json[:namespace], exporter: json[:exporter], exporter_version: json[:exporter_version], execution_platform: json[:execution_platform], execution_platform_version: json[:execution_platform_version], decisions: decisions)
|
22
|
-
end
|
23
|
-
|
24
|
-
def initialize(id:, name:, namespace:, exporter:, exporter_version:, execution_platform:, execution_platform_version:, decisions:)
|
25
|
-
@id = id
|
26
|
-
@name = name
|
27
|
-
@namespace = namespace
|
28
|
-
@exporter = exporter
|
29
|
-
@exporter_version = exporter_version
|
30
|
-
@execution_platform = execution_platform
|
31
|
-
@execution_platform_version = execution_platform_version
|
32
|
-
@decisions = decisions
|
33
|
-
end
|
34
|
-
|
35
|
-
def evaluate(decision_id, variables: {}, already_evaluated_decisions: {})
|
36
|
-
decision = decisions.find { |d| d.id == decision_id }
|
37
|
-
raise EvaluationError, "Decision #{decision_id} not found" unless decision
|
38
|
-
|
39
|
-
# Evaluate required decisions recursively
|
40
|
-
decision.required_decision_ids&.each do |required_decision_id|
|
41
|
-
next if already_evaluated_decisions[required_decision_id]
|
42
|
-
next if decisions.find { |d| d.id == required_decision_id }.nil?
|
43
|
-
|
44
|
-
result = evaluate(required_decision_id, variables:, already_evaluated_decisions:)
|
45
|
-
|
46
|
-
variables.merge!(result) if result.is_a?(Hash)
|
47
|
-
|
48
|
-
already_evaluated_decisions[required_decision_id] = true
|
49
|
-
end
|
50
|
-
|
51
|
-
decision.evaluate(variables)
|
52
|
-
end
|
53
|
-
|
54
|
-
def as_json
|
55
|
-
{
|
56
|
-
id: id,
|
57
|
-
name: name,
|
58
|
-
namespace: namespace,
|
59
|
-
exporter: exporter,
|
60
|
-
exporter_version: exporter_version,
|
61
|
-
execution_platform: execution_platform,
|
62
|
-
execution_platform_version: execution_platform_version,
|
63
|
-
decisions: decisions.map(&:as_json),
|
64
|
-
}
|
65
|
-
end
|
66
|
-
end
|
67
|
-
end
|
68
|
-
end
|
@@ -1,29 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module SpotFeel
|
4
|
-
module Dmn
|
5
|
-
class InformationRequirement
|
6
|
-
attr_reader :id, :required_input_id, :required_decision_id
|
7
|
-
|
8
|
-
def self.from_json(json)
|
9
|
-
required_input_id = json[:required_input][:href].delete_prefix("#") if json[:required_input]
|
10
|
-
required_decision_id = json[:required_decision][:href].delete_prefix("#") if json[:required_decision]
|
11
|
-
InformationRequirement.new(id: json[:id], required_input_id: required_input_id, required_decision_id: required_decision_id)
|
12
|
-
end
|
13
|
-
|
14
|
-
def initialize(id:, required_input_id:, required_decision_id:)
|
15
|
-
@id = id
|
16
|
-
@required_input_id = required_input_id
|
17
|
-
@required_decision_id = required_decision_id
|
18
|
-
end
|
19
|
-
|
20
|
-
def as_json
|
21
|
-
{
|
22
|
-
id: id,
|
23
|
-
required_decision_id: required_decision_id,
|
24
|
-
required_input_id: required_input_id,
|
25
|
-
}
|
26
|
-
end
|
27
|
-
end
|
28
|
-
end
|
29
|
-
end
|
data/lib/spot_feel/dmn/input.rb
DELETED
@@ -1,28 +0,0 @@
|
|
1
|
-
# frozen_string_literal: true
|
2
|
-
|
3
|
-
module SpotFeel
|
4
|
-
module Dmn
|
5
|
-
class Input
|
6
|
-
attr_reader :id, :label, :input_expression
|
7
|
-
|
8
|
-
def self.from_json(json)
|
9
|
-
input_expression = LiteralExpression.from_json(json[:input_expression]) if json[:input_expression]
|
10
|
-
Input.new(id: json[:id], label: json[:label], input_expression:)
|
11
|
-
end
|
12
|
-
|
13
|
-
def initialize(id:, label:, input_expression:)
|
14
|
-
@id = id
|
15
|
-
@label = label
|
16
|
-
@input_expression = input_expression
|
17
|
-
end
|
18
|
-
|
19
|
-
def as_json
|
20
|
-
{
|
21
|
-
id: id,
|
22
|
-
label: label,
|
23
|
-
input_expression: input_expression.as_json,
|
24
|
-
}
|
25
|
-
end
|
26
|
-
end
|
27
|
-
end
|
28
|
-
end
|