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.
@@ -1,12 +1,12 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module SpotFeel
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 = '', qualified_names = Set.new)
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 SpotFeel.config.strict && !context.key?(head.text_value.to_sym)
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('.') }.inject(context[head.text_value.to_sym]) do |hash, key|
232
- raise_evaluation_error("#{head.text_value}#{tail.text_value}", context) if SpotFeel.config.strict && (hash.blank? || !hash.key?(key.to_sym))
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 < Node
334
- def eval(_context = {})
335
- text_value[1..-2]
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 SpotFeel.config.strict
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 '<' then left.eval(context) < right.eval(context)
496
- when '<=' then left.eval(context) <= right.eval(context)
497
- when '>=' then left.eval(context) >= right.eval(context)
498
- when '>' then left.eval(context) > right.eval(context)
499
- when '!=' then left.eval(context) != right.eval(context)
500
- when '=' then left.eval(context) == right.eval(context)
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(context = {})
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 SpotFeel
3
+ module FEEL
4
4
  class Parser
5
- # Load the Treetop grammar from the 'spot_feel' file, and create a new
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__), 'spot_feel.treetop')))
9
- @@parser = SpotFeelParser.new
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 || '-', root: :simple_unary_tests).tap do |ast|
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
@@ -0,0 +1,5 @@
1
+ # frozen_string_literal: true
2
+
3
+ module FEEL
4
+ VERSION = "0.0.4"
5
+ end
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.1
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: 2024-12-30 00:00:00.000000000 Z
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: awesome_print
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: xmlhasher
55
+ name: treetop
71
56
  requirement: !ruby/object:Gem::Requirement
72
57
  requirements:
73
- - - "~>"
58
+ - - '='
74
59
  - !ruby/object:Gem::Version
75
- version: 1.0.7
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.0.7
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/spot_feel.rb
275
- - lib/spot_feel/configuration.rb
276
- - lib/spot_feel/dmn.rb
277
- - lib/spot_feel/dmn/decision.rb
278
- - lib/spot_feel/dmn/decision_table.rb
279
- - lib/spot_feel/dmn/definitions.rb
280
- - lib/spot_feel/dmn/information_requirement.rb
281
- - lib/spot_feel/dmn/input.rb
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.4.19
313
- signing_key:
287
+ rubygems_version: 3.6.5
314
288
  specification_version: 4
315
- summary: A light-weight DMN FEEL expression evaluator and business rule engine in
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
@@ -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