dry-logic 1.0.2 → 1.0.8

Sign up to get free protection for your applications and to get access to all the features.
Files changed (99) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +129 -26
  3. data/LICENSE +1 -1
  4. data/README.md +12 -14
  5. data/dry-logic.gemspec +27 -14
  6. data/lib/dry-logic.rb +1 -1
  7. data/lib/dry/logic.rb +2 -2
  8. data/lib/dry/logic/evaluator.rb +1 -1
  9. data/lib/dry/logic/operations.rb +11 -11
  10. data/lib/dry/logic/operations/abstract.rb +6 -6
  11. data/lib/dry/logic/operations/and.rb +4 -4
  12. data/lib/dry/logic/operations/attr.rb +1 -1
  13. data/lib/dry/logic/operations/binary.rb +1 -1
  14. data/lib/dry/logic/operations/check.rb +4 -4
  15. data/lib/dry/logic/operations/each.rb +2 -2
  16. data/lib/dry/logic/operations/implication.rb +2 -2
  17. data/lib/dry/logic/operations/key.rb +5 -5
  18. data/lib/dry/logic/operations/negation.rb +2 -2
  19. data/lib/dry/logic/operations/or.rb +2 -2
  20. data/lib/dry/logic/operations/set.rb +3 -3
  21. data/lib/dry/logic/operations/unary.rb +1 -1
  22. data/lib/dry/logic/operations/xor.rb +2 -2
  23. data/lib/dry/logic/operators.rb +4 -4
  24. data/lib/dry/logic/predicates.rb +60 -28
  25. data/lib/dry/logic/result.rb +2 -2
  26. data/lib/dry/logic/rule.rb +11 -11
  27. data/lib/dry/logic/rule/interface.rb +32 -37
  28. data/lib/dry/logic/rule/predicate.rb +3 -3
  29. data/lib/dry/logic/rule_compiler.rb +3 -3
  30. data/lib/dry/logic/version.rb +1 -1
  31. metadata +12 -133
  32. data/.codeclimate.yml +0 -15
  33. data/.gitignore +0 -9
  34. data/.rspec +0 -3
  35. data/.travis.yml +0 -31
  36. data/CONTRIBUTING.md +0 -29
  37. data/Gemfile +0 -15
  38. data/Rakefile +0 -14
  39. data/benchmarks/rule_application.rb +0 -30
  40. data/benchmarks/setup.rb +0 -13
  41. data/bin/console +0 -11
  42. data/examples/basic.rb +0 -16
  43. data/spec/integration/result_spec.rb +0 -61
  44. data/spec/integration/rule_spec.rb +0 -55
  45. data/spec/shared/predicates.rb +0 -59
  46. data/spec/shared/rule.rb +0 -74
  47. data/spec/spec_helper.rb +0 -36
  48. data/spec/support/mutant.rb +0 -11
  49. data/spec/unit/operations/and_spec.rb +0 -70
  50. data/spec/unit/operations/attr_spec.rb +0 -29
  51. data/spec/unit/operations/check_spec.rb +0 -51
  52. data/spec/unit/operations/each_spec.rb +0 -49
  53. data/spec/unit/operations/implication_spec.rb +0 -32
  54. data/spec/unit/operations/key_spec.rb +0 -135
  55. data/spec/unit/operations/negation_spec.rb +0 -51
  56. data/spec/unit/operations/or_spec.rb +0 -75
  57. data/spec/unit/operations/set_spec.rb +0 -43
  58. data/spec/unit/operations/xor_spec.rb +0 -63
  59. data/spec/unit/predicates/array_spec.rb +0 -43
  60. data/spec/unit/predicates/attr_spec.rb +0 -31
  61. data/spec/unit/predicates/bool_spec.rb +0 -36
  62. data/spec/unit/predicates/case_spec.rb +0 -35
  63. data/spec/unit/predicates/date_spec.rb +0 -33
  64. data/spec/unit/predicates/date_time_spec.rb +0 -33
  65. data/spec/unit/predicates/decimal_spec.rb +0 -34
  66. data/spec/unit/predicates/empty_spec.rb +0 -40
  67. data/spec/unit/predicates/eql_spec.rb +0 -23
  68. data/spec/unit/predicates/even_spec.rb +0 -33
  69. data/spec/unit/predicates/excluded_from_spec.rb +0 -37
  70. data/spec/unit/predicates/excludes_spec.rb +0 -58
  71. data/spec/unit/predicates/false_spec.rb +0 -37
  72. data/spec/unit/predicates/filled_spec.rb +0 -40
  73. data/spec/unit/predicates/float_spec.rb +0 -33
  74. data/spec/unit/predicates/format_spec.rb +0 -23
  75. data/spec/unit/predicates/gt_spec.rb +0 -42
  76. data/spec/unit/predicates/gteq_spec.rb +0 -42
  77. data/spec/unit/predicates/included_in_spec.rb +0 -37
  78. data/spec/unit/predicates/includes_spec.rb +0 -24
  79. data/spec/unit/predicates/int_spec.rb +0 -36
  80. data/spec/unit/predicates/key_spec.rb +0 -31
  81. data/spec/unit/predicates/lt_spec.rb +0 -42
  82. data/spec/unit/predicates/lteq_spec.rb +0 -42
  83. data/spec/unit/predicates/max_size_spec.rb +0 -51
  84. data/spec/unit/predicates/min_size_spec.rb +0 -51
  85. data/spec/unit/predicates/none_spec.rb +0 -30
  86. data/spec/unit/predicates/not_eql_spec.rb +0 -23
  87. data/spec/unit/predicates/number_spec.rb +0 -39
  88. data/spec/unit/predicates/odd_spec.rb +0 -33
  89. data/spec/unit/predicates/respond_to_spec.rb +0 -31
  90. data/spec/unit/predicates/size_spec.rb +0 -57
  91. data/spec/unit/predicates/str_spec.rb +0 -34
  92. data/spec/unit/predicates/time_spec.rb +0 -33
  93. data/spec/unit/predicates/true_spec.rb +0 -37
  94. data/spec/unit/predicates/type_spec.rb +0 -37
  95. data/spec/unit/predicates/uuid_v4_spec.rb +0 -29
  96. data/spec/unit/predicates_spec.rb +0 -25
  97. data/spec/unit/rule/predicate_spec.rb +0 -55
  98. data/spec/unit/rule_compiler_spec.rb +0 -129
  99. data/spec/unit/rule_spec.rb +0 -213
@@ -1,129 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- require 'dry/logic/rule_compiler'
4
-
5
- RSpec.describe Dry::Logic::RuleCompiler, '#call' do
6
- subject(:compiler) { RuleCompiler.new(predicates) }
7
-
8
- let(:predicates) {
9
- { key?: predicate,
10
- attr?: predicate,
11
- filled?: predicate,
12
- gt?: predicate,
13
- one: predicate }
14
- }
15
-
16
- let(:predicate) { double(:predicate, name: :test?, arity: 2).as_null_object }
17
-
18
- let(:rule) { Rule::Predicate.build(predicate) }
19
- let(:key_op) { Operations::Key.new(rule, name: :email) }
20
- let(:attr_op) { Operations::Attr.new(rule, name: :email) }
21
- let(:check_op) { Operations::Check.new(rule, keys: [:email]) }
22
- let(:not_key_op) { Operations::Negation.new(key_op) }
23
- let(:and_op) { key_op.curry(:email) & rule }
24
- let(:or_op) { key_op.curry(:email) | rule }
25
- let(:xor_op) { key_op.curry(:email) ^ rule }
26
- let(:set_op) { Operations::Set.new(rule) }
27
- let(:each_op) { Operations::Each.new(rule) }
28
-
29
- it 'compiles key rules' do
30
- ast = [[:key, [:email, [:predicate, [:filled?, [[:input, Undefined]]]]]]]
31
-
32
- rules = compiler.(ast)
33
-
34
- expect(rules).to eql([key_op])
35
- end
36
-
37
- it 'compiles attr rules' do
38
- ast = [[:attr, [:email, [:predicate, [:filled?, [[:input, Undefined]]]]]]]
39
-
40
- rules = compiler.(ast)
41
-
42
- expect(rules).to eql([attr_op])
43
- end
44
-
45
- it 'compiles check rules' do
46
- ast = [[:check, [[:email], [:predicate, [:filled?, [[:input, Undefined]]]]]]]
47
-
48
- rules = compiler.(ast)
49
-
50
- expect(rules).to eql([check_op])
51
- end
52
-
53
- it 'compiles attr rules' do
54
- ast = [[:attr, [:email, [:predicate, [:filled?, [[:input, Undefined]]]]]]]
55
-
56
- rules = compiler.(ast)
57
-
58
- expect(rules).to eql([attr_op])
59
- end
60
-
61
- it 'compiles negated rules' do
62
- ast = [[:not, [:key, [:email, [:predicate, [:filled?, [[:input, Undefined]]]]]]]]
63
-
64
- rules = compiler.(ast)
65
-
66
- expect(rules).to eql([not_key_op])
67
- end
68
-
69
- it 'compiles and rules' do
70
- ast = [
71
- [
72
- :and, [
73
- [:key, [:email, [:predicate, [:key?, [[:name, :email], [:input, Undefined]]]]]],
74
- [:predicate, [:filled?, [[:input, Undefined]]]]
75
- ]
76
- ]
77
- ]
78
-
79
- rules = compiler.(ast)
80
-
81
- expect(rules).to eql([and_op])
82
- end
83
-
84
- it 'compiles or rules' do
85
- ast = [
86
- [
87
- :or, [
88
- [:key, [:email, [:predicate, [:key?, [[:name, :email], [:input, Undefined]]]]]],
89
- [:predicate, [:filled?, [[:input, Undefined]]]]
90
- ]
91
- ]
92
- ]
93
-
94
- rules = compiler.(ast)
95
-
96
- expect(rules).to eql([or_op])
97
- end
98
-
99
- it 'compiles exclusive or rules' do
100
- ast = [
101
- [
102
- :xor, [
103
- [:key, [:email, [:predicate, [:key?, [[:name, :email], [:input, Undefined]]]]]],
104
- [:predicate, [:filled?, [[:input, Undefined]]]]
105
- ]
106
- ]
107
- ]
108
-
109
- rules = compiler.(ast)
110
-
111
- expect(rules).to eql([xor_op])
112
- end
113
-
114
- it 'compiles set rules' do
115
- ast = [[:set, [[:predicate, [:filled?, [[:input, nil]]]]]]]
116
-
117
- rules = compiler.(ast)
118
-
119
- expect(rules).to eql([set_op])
120
- end
121
-
122
- it 'compiles each rules' do
123
- ast = [[:each, [:predicate, [:filled?, [[:input, nil]]]]]]
124
-
125
- rules = compiler.(ast)
126
-
127
- expect(rules).to eql([each_op])
128
- end
129
- end
@@ -1,213 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- RSpec.describe Dry::Logic::Rule do
4
- subject(:rule) { Rule.build(predicate, options) }
5
-
6
- let(:predicate) { -> { true } }
7
- let(:options) { {} }
8
-
9
- let(:schema) do
10
- Class.new do
11
- define_method(:class, Kernel.instance_method(:class))
12
-
13
- def method_missing(m, *)
14
- if m.to_s.end_with?('?')
15
- self.class.new
16
- else
17
- super
18
- end
19
- end
20
-
21
- def to_proc
22
- -> value { value }
23
- end
24
-
25
- def arity
26
- 1
27
- end
28
-
29
- def parameters
30
- [[:req, :value]]
31
- end
32
- end.new
33
- end
34
-
35
- it_behaves_like Dry::Logic::Rule
36
-
37
- describe '.new' do
38
- it 'accepts an :id' do
39
- expect(Rule.build(predicate, id: :check_num).id).to be(:check_num)
40
- end
41
- end
42
-
43
- describe 'with a function returning truthy value' do
44
- it 'is successful for valid input' do
45
- expect(Rule.build(-> val { val }).('true')).to be_success
46
- end
47
-
48
- it 'is not successful for invalid input' do
49
- expect(Rule.build(-> val { val }).(nil)).to be_failure
50
- end
51
- end
52
-
53
- describe '#ast' do
54
- it 'returns predicate node with :id' do
55
- expect(Rule.build(-> value { true }).with(id: :email?).ast('oops')).to eql(
56
- [:predicate, [:email?, [[:value, 'oops']]]]
57
- )
58
- end
59
-
60
- it 'returns predicate node with undefined args' do
61
- expect(Rule.build(-> value { true }).with(id: :email?).ast).to eql(
62
- [:predicate, [:email?, [[:value, Undefined]]]]
63
- )
64
- end
65
- end
66
-
67
- describe '#type' do
68
- it 'returns rule type' do
69
- expect(rule.type).to be(:rule)
70
- end
71
- end
72
-
73
- describe '#bind' do
74
- let(:bound) { rule.with(id: :bound).bind(object) }
75
-
76
- context 'with an unbound method' do
77
- let(:predicate) { klass.instance_method(:test?) }
78
- let(:klass) { Class.new { def test?; true; end } }
79
- let(:object) { klass.new }
80
-
81
- it 'returns a new rule with its predicate bound to a specific object' do
82
- expect(bound.()).to be_success
83
- end
84
-
85
- it 'carries id' do
86
- expect(bound.id).to be(:bound)
87
- end
88
- end
89
-
90
- context 'with an arbitrary block' do
91
- let(:predicate) { -> value { value == expected } }
92
- let(:object) { Class.new { def expected; 'test'; end }.new }
93
-
94
- it 'returns a new with its predicate executed in the context of the provided object' do
95
- expect(bound.('test')).to be_success
96
- expect(bound.('oops')).to be_failure
97
- end
98
-
99
- it 'carries id' do
100
- expect(bound.id).to be(:bound)
101
- end
102
-
103
- it 'stores arity' do
104
- expect(bound.options[:arity]).to be(rule.arity)
105
- end
106
-
107
- it 'stores parameters' do
108
- expect(bound.options[:parameters]).to eql(rule.parameters)
109
- end
110
- end
111
-
112
- context 'with a schema instance' do
113
- let(:object) { schema }
114
- let(:predicate) { schema }
115
-
116
- it 'returns a new with its predicate executed in the context of the provided object' do
117
- expect(bound.(true)).to be_success
118
- expect(bound.(false)).to be_failure
119
- end
120
- end
121
- end
122
-
123
- describe '#eval_args' do
124
- context 'with an unbound method' do
125
- let(:options) { { args: [1, klass.instance_method(:num), :foo], arity: 3 } }
126
- let(:klass) { Class.new { def num; 7; end } }
127
- let(:object) { klass.new }
128
-
129
- it 'evaluates args in the context of the provided object' do
130
- expect(rule.eval_args(object).args).to eql([1, 7, :foo])
131
- end
132
- end
133
-
134
- context 'with a schema instance' do
135
- let(:options) { { args: [1, schema, :foo], arity: 3 } }
136
- let(:object) { Object.new }
137
-
138
- it 'returns a new with its predicate executed in the context of the provided object' do
139
- expect(rule.eval_args(object).args).to eql([1, schema, :foo])
140
- end
141
- end
142
- end
143
-
144
- describe 'arity specialization' do
145
- describe '0-arity rule' do
146
- let(:options) { { args: [1], arity: 1 } }
147
- let(:predicate) { :odd?.to_proc }
148
-
149
- it 'generates interface with the right arity' do
150
- expect(rule.method(:call).arity).to be_zero
151
- expect(rule.method(:[]).arity).to be_zero
152
- expect(rule[]).to be(true)
153
- expect(rule.()).to be_success
154
- end
155
- end
156
-
157
- describe '1-arity rule' do
158
- let(:options) { { args: [1], arity: 2 } }
159
- let(:predicate) { -> a, b { a + b } }
160
-
161
- it 'generates interface with the right arity' do
162
- expect(rule.method(:call).arity).to be(1)
163
- expect(rule.method(:[]).arity).to be(1)
164
- expect(rule[10]).to be(11)
165
- expect(rule.(1)).to be_success
166
- end
167
- end
168
-
169
- describe 'currying' do
170
- let(:options) { { args: [], arity: 2 } }
171
- let(:predicate) { -> a, b { a + b } }
172
- let(:rule) { super().curry(1) }
173
-
174
- it 'generates correct arity on currying' do
175
- expect(rule.method(:call).arity).to be(1)
176
- expect(rule.method(:[]).arity).to be(1)
177
- expect(rule[10]).to be(11)
178
- expect(rule.(1)).to be_success
179
- end
180
- end
181
-
182
- describe 'arbitrary arity' do
183
- let(:arity) { rand(1..20) }
184
- let(:curried) { rand(arity) }
185
-
186
- let(:options) { { args: [1] * curried, arity: arity } }
187
- let(:predicate) { double(:predicate) }
188
-
189
- it 'generates correct arity' do
190
- expect(rule.method(:call).arity).to be(arity - curried)
191
- expect(rule.method(:[]).arity).to be(arity - curried)
192
- end
193
- end
194
-
195
- describe '-1 arity' do
196
- let(:options) { { args: [], arity: -1 } }
197
-
198
- it 'accepts variable number of arguments' do
199
- expect(rule.method(:call).arity).to be(-1)
200
- expect(rule.method(:[]).arity).to be(-1)
201
- end
202
- end
203
-
204
- describe 'constants' do
205
- let(:options) { { args: [], arity: 0 } }
206
-
207
- fit 'accepts variable number of arguments' do
208
- expect(rule.method(:call).arity).to be(-1)
209
- expect(rule.method(:[]).arity).to be(-1)
210
- end
211
- end
212
- end
213
- end