dry-logic 0.3.0 → 0.4.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.travis.yml +1 -5
- data/CHANGELOG.md +28 -0
- data/Gemfile +7 -4
- data/dry-logic.gemspec +1 -0
- data/lib/dry/logic.rb +2 -3
- data/lib/dry/logic/appliable.rb +33 -0
- data/lib/dry/logic/evaluator.rb +2 -0
- data/lib/dry/logic/operations.rb +13 -0
- data/lib/dry/logic/operations/abstract.rb +44 -0
- data/lib/dry/logic/operations/and.rb +35 -0
- data/lib/dry/logic/operations/attr.rb +17 -0
- data/lib/dry/logic/operations/binary.rb +26 -0
- data/lib/dry/logic/operations/check.rb +52 -0
- data/lib/dry/logic/operations/each.rb +32 -0
- data/lib/dry/logic/operations/implication.rb +37 -0
- data/lib/dry/logic/operations/key.rb +66 -0
- data/lib/dry/logic/operations/negation.rb +18 -0
- data/lib/dry/logic/operations/or.rb +35 -0
- data/lib/dry/logic/operations/set.rb +35 -0
- data/lib/dry/logic/operations/unary.rb +24 -0
- data/lib/dry/logic/operations/xor.rb +27 -0
- data/lib/dry/logic/operators.rb +25 -0
- data/lib/dry/logic/predicates.rb +143 -136
- data/lib/dry/logic/result.rb +76 -33
- data/lib/dry/logic/rule.rb +62 -46
- data/lib/dry/logic/rule/predicate.rb +28 -0
- data/lib/dry/logic/rule_compiler.rb +16 -17
- data/lib/dry/logic/version.rb +1 -1
- data/spec/integration/result_spec.rb +59 -0
- data/spec/integration/rule_spec.rb +53 -0
- data/spec/shared/predicates.rb +6 -0
- data/spec/shared/rule.rb +67 -0
- data/spec/spec_helper.rb +10 -3
- data/spec/support/mutant.rb +9 -0
- data/spec/unit/operations/and_spec.rb +64 -0
- data/spec/unit/operations/attr_spec.rb +27 -0
- data/spec/unit/operations/check_spec.rb +49 -0
- data/spec/unit/operations/each_spec.rb +47 -0
- data/spec/unit/operations/implication_spec.rb +30 -0
- data/spec/unit/operations/key_spec.rb +119 -0
- data/spec/unit/operations/negation_spec.rb +40 -0
- data/spec/unit/operations/or_spec.rb +73 -0
- data/spec/unit/operations/set_spec.rb +41 -0
- data/spec/unit/operations/xor_spec.rb +61 -0
- data/spec/unit/predicates_spec.rb +23 -0
- data/spec/unit/rule/predicate_spec.rb +53 -0
- data/spec/unit/rule_compiler_spec.rb +38 -38
- data/spec/unit/rule_spec.rb +94 -0
- metadata +67 -40
- data/lib/dry/logic/predicate.rb +0 -100
- data/lib/dry/logic/predicate_set.rb +0 -23
- data/lib/dry/logic/result/each.rb +0 -20
- data/lib/dry/logic/result/multi.rb +0 -14
- data/lib/dry/logic/result/named.rb +0 -17
- data/lib/dry/logic/result/set.rb +0 -10
- data/lib/dry/logic/result/value.rb +0 -17
- data/lib/dry/logic/rule/attr.rb +0 -13
- data/lib/dry/logic/rule/check.rb +0 -40
- data/lib/dry/logic/rule/composite.rb +0 -91
- data/lib/dry/logic/rule/each.rb +0 -13
- data/lib/dry/logic/rule/key.rb +0 -37
- data/lib/dry/logic/rule/negation.rb +0 -15
- data/lib/dry/logic/rule/set.rb +0 -31
- data/lib/dry/logic/rule/value.rb +0 -48
- data/spec/unit/predicate_spec.rb +0 -115
- data/spec/unit/rule/attr_spec.rb +0 -29
- data/spec/unit/rule/check_spec.rb +0 -44
- data/spec/unit/rule/conjunction_spec.rb +0 -30
- data/spec/unit/rule/disjunction_spec.rb +0 -38
- data/spec/unit/rule/each_spec.rb +0 -31
- data/spec/unit/rule/exclusive_disjunction_spec.rb +0 -19
- data/spec/unit/rule/implication_spec.rb +0 -16
- data/spec/unit/rule/key_spec.rb +0 -121
- data/spec/unit/rule/set_spec.rb +0 -30
- data/spec/unit/rule/value_spec.rb +0 -99
@@ -1,30 +0,0 @@
|
|
1
|
-
RSpec.describe Rule::Composite::Conjunction do
|
2
|
-
include_context 'predicates'
|
3
|
-
|
4
|
-
subject(:rule) { Rule::Composite::Conjunction.new(left, right) }
|
5
|
-
|
6
|
-
let(:left) { Rule::Value.new(int?) }
|
7
|
-
let(:right) { Rule::Value.new(gt?.curry(18)) }
|
8
|
-
|
9
|
-
describe '#call' do
|
10
|
-
it 'calls left and right' do
|
11
|
-
expect(rule.(18)).to be_failure
|
12
|
-
end
|
13
|
-
end
|
14
|
-
|
15
|
-
describe '#and' do
|
16
|
-
let(:other) { Rule::Value.new(lt?.curry(30)) }
|
17
|
-
|
18
|
-
it 'creates conjunction with the other' do
|
19
|
-
expect(rule.and(other).(31)).to be_failure
|
20
|
-
end
|
21
|
-
end
|
22
|
-
|
23
|
-
describe '#or' do
|
24
|
-
let(:other) { Rule::Value.new(lt?.curry(14)) }
|
25
|
-
|
26
|
-
it 'creates disjunction with the other' do
|
27
|
-
expect(rule.or(other).(13)).to be_success
|
28
|
-
end
|
29
|
-
end
|
30
|
-
end
|
@@ -1,38 +0,0 @@
|
|
1
|
-
RSpec.describe Rule::Composite::Disjunction do
|
2
|
-
include_context 'predicates'
|
3
|
-
|
4
|
-
subject(:rule) { Rule::Composite::Disjunction.new(left, right) }
|
5
|
-
|
6
|
-
let(:left) { Rule::Value.new(none?) }
|
7
|
-
let(:right) { Rule::Value.new(gt?.curry(18)) }
|
8
|
-
|
9
|
-
let(:other) do
|
10
|
-
Rule::Value.new(int?) & Rule::Value.new(lt?.curry(14))
|
11
|
-
end
|
12
|
-
|
13
|
-
describe '#call' do
|
14
|
-
it 'calls left and right' do
|
15
|
-
expect(rule.(nil)).to be_success
|
16
|
-
expect(rule.(19)).to be_success
|
17
|
-
expect(rule.(18)).to be_failure
|
18
|
-
end
|
19
|
-
end
|
20
|
-
|
21
|
-
describe '#and' do
|
22
|
-
it 'creates conjunction with the other' do
|
23
|
-
expect(rule.and(other).(nil)).to be_failure
|
24
|
-
expect(rule.and(other).(19)).to be_failure
|
25
|
-
expect(rule.and(other).(13)).to be_failure
|
26
|
-
expect(rule.and(other).(14)).to be_failure
|
27
|
-
end
|
28
|
-
end
|
29
|
-
|
30
|
-
describe '#or' do
|
31
|
-
it 'creates disjunction with the other' do
|
32
|
-
expect(rule.or(other).(nil)).to be_success
|
33
|
-
expect(rule.or(other).(19)).to be_success
|
34
|
-
expect(rule.or(other).(13)).to be_success
|
35
|
-
expect(rule.or(other).(14)).to be_failure
|
36
|
-
end
|
37
|
-
end
|
38
|
-
end
|
data/spec/unit/rule/each_spec.rb
DELETED
@@ -1,31 +0,0 @@
|
|
1
|
-
require 'dry/logic/rule'
|
2
|
-
|
3
|
-
RSpec.describe Dry::Logic::Rule::Each do
|
4
|
-
include_context 'predicates'
|
5
|
-
|
6
|
-
subject(:address_rule) do
|
7
|
-
Dry::Logic::Rule::Each.new(is_string)
|
8
|
-
end
|
9
|
-
|
10
|
-
let(:is_string) { Dry::Logic::Rule::Value.new(str?) }
|
11
|
-
|
12
|
-
describe '#call' do
|
13
|
-
it 'applies its rules to all elements in the input' do
|
14
|
-
expect(address_rule.(['Address'])).to be_success
|
15
|
-
|
16
|
-
expect(address_rule.([nil, 'Address'])).to be_failure
|
17
|
-
expect(address_rule.([:Address, 'Address'])).to be_failure
|
18
|
-
end
|
19
|
-
|
20
|
-
it 'returns result ast' do
|
21
|
-
expect(address_rule.([nil, nil]).to_ast).to eql([
|
22
|
-
:result, [[nil, nil], [
|
23
|
-
:each, [
|
24
|
-
[:el, [0, [:result, [nil, [:val, [:predicate, [:str?, [[:input, nil]]]]]]]]],
|
25
|
-
[:el, [1, [:result, [nil, [:val, [:predicate, [:str?, [[:input, nil]]]]]]]]]
|
26
|
-
]
|
27
|
-
]]
|
28
|
-
])
|
29
|
-
end
|
30
|
-
end
|
31
|
-
end
|
@@ -1,19 +0,0 @@
|
|
1
|
-
RSpec.describe Rule::ExclusiveDisjunction do
|
2
|
-
include_context 'predicates'
|
3
|
-
|
4
|
-
subject(:rule) do
|
5
|
-
Rule::ExclusiveDisjunction.new(left, right)
|
6
|
-
end
|
7
|
-
|
8
|
-
let(:left) { Rule::Key.new(true?, name: :eat_cake) }
|
9
|
-
let(:right) { Rule::Key.new(true?, name: :have_cake) }
|
10
|
-
|
11
|
-
describe '#call' do
|
12
|
-
it 'calls left and right' do
|
13
|
-
expect(rule.(eat_cake: true, have_cake: false)).to be_success
|
14
|
-
expect(rule.(eat_cake: false, have_cake: true)).to be_success
|
15
|
-
expect(rule.(eat_cake: false, have_cake: false)).to be_failure
|
16
|
-
expect(rule.(eat_cake: true, have_cake: true)).to be_failure
|
17
|
-
end
|
18
|
-
end
|
19
|
-
end
|
@@ -1,16 +0,0 @@
|
|
1
|
-
RSpec.describe Rule::Composite::Implication do
|
2
|
-
include_context 'predicates'
|
3
|
-
|
4
|
-
subject(:rule) { Rule::Composite::Implication.new(left, right) }
|
5
|
-
|
6
|
-
let(:left) { Rule::Value.new(int?) }
|
7
|
-
let(:right) { Rule::Value.new(gt?.curry(18)) }
|
8
|
-
|
9
|
-
describe '#call' do
|
10
|
-
it 'calls left and right' do
|
11
|
-
expect(rule.('19')).to be_success
|
12
|
-
expect(rule.(19)).to be_success
|
13
|
-
expect(rule.(18)).to be_failure
|
14
|
-
end
|
15
|
-
end
|
16
|
-
end
|
data/spec/unit/rule/key_spec.rb
DELETED
@@ -1,121 +0,0 @@
|
|
1
|
-
require 'dry/logic/rule'
|
2
|
-
|
3
|
-
RSpec.describe Rule::Key do
|
4
|
-
include_context 'predicates'
|
5
|
-
|
6
|
-
subject(:rule) do
|
7
|
-
Rule::Key.new(predicate, name: :user)
|
8
|
-
end
|
9
|
-
|
10
|
-
let(:predicate) do
|
11
|
-
key?.curry(:name)
|
12
|
-
end
|
13
|
-
|
14
|
-
describe '#call' do
|
15
|
-
context 'with a plain predicate' do
|
16
|
-
it 'applies predicate to the value' do
|
17
|
-
expect(rule.(user: { name: 'Jane' })).to be_success
|
18
|
-
expect(rule.(user: {})).to be_failure
|
19
|
-
end
|
20
|
-
|
21
|
-
context 'with a custom predicate' do
|
22
|
-
let(:predicate) { -> input { double(success?: true, to_ast: [:foo]) } }
|
23
|
-
|
24
|
-
let(:result) { rule.(test: true) }
|
25
|
-
|
26
|
-
it 'delegates to_ast to response' do
|
27
|
-
expect(result.to_ast).to eql([:foo])
|
28
|
-
end
|
29
|
-
end
|
30
|
-
end
|
31
|
-
|
32
|
-
context 'with a set rule as predicate' do
|
33
|
-
subject(:rule) do
|
34
|
-
Rule::Key.new(predicate, name: :address)
|
35
|
-
end
|
36
|
-
|
37
|
-
let(:predicate) do
|
38
|
-
Rule::Set.new(
|
39
|
-
[Rule::Value.new(key?.curry(:city)), Rule::Value.new(key?.curry(:zipcode))]
|
40
|
-
)
|
41
|
-
end
|
42
|
-
|
43
|
-
it 'applies set rule to the value that passes' do
|
44
|
-
result = rule.(address: { city: 'NYC', zipcode: '123' })
|
45
|
-
|
46
|
-
expect(result).to be_success
|
47
|
-
end
|
48
|
-
|
49
|
-
it 'applies set rule to the value that fails' do
|
50
|
-
result = rule.(address: { city: 'NYC' })
|
51
|
-
|
52
|
-
expect(result).to be_failure
|
53
|
-
|
54
|
-
expect(result.to_ast).to eql([
|
55
|
-
:input, [
|
56
|
-
:address,
|
57
|
-
[:result, [
|
58
|
-
{ city: "NYC" },
|
59
|
-
[:set, [[:result, [{ city: 'NYC' }, [:val, [:predicate, [:key?, [[:name, :zipcode], [:input, {:city=>"NYC"}]]]]]]]]]
|
60
|
-
]]
|
61
|
-
]
|
62
|
-
])
|
63
|
-
end
|
64
|
-
end
|
65
|
-
|
66
|
-
context 'with an each rule as predicate' do
|
67
|
-
subject(:rule) do
|
68
|
-
Rule::Key.new(predicate, name: :nums)
|
69
|
-
end
|
70
|
-
|
71
|
-
let(:predicate) do
|
72
|
-
Rule::Each.new(Rule::Value.new(str?))
|
73
|
-
end
|
74
|
-
|
75
|
-
it 'applies each rule to the value that passses' do
|
76
|
-
result = rule.(nums: %w(1 2 3))
|
77
|
-
|
78
|
-
expect(result).to be_success
|
79
|
-
|
80
|
-
expect(result.to_ast).to eql([
|
81
|
-
:input, [:nums, [:result, [%w(1 2 3), [:each, []]]]]
|
82
|
-
])
|
83
|
-
end
|
84
|
-
|
85
|
-
it 'applies each rule to the value that fails' do
|
86
|
-
failure = rule.(nums: [1, '3', 3])
|
87
|
-
|
88
|
-
expect(failure).to be_failure
|
89
|
-
|
90
|
-
expect(failure.to_ast).to eql([
|
91
|
-
:input, [
|
92
|
-
:nums, [
|
93
|
-
:result, [
|
94
|
-
[1, '3', 3],
|
95
|
-
[:each, [
|
96
|
-
[:el, [0, [:result, [1, [:val, [:predicate, [:str?, [[:input, 1]]]]]]]]],
|
97
|
-
[:el, [2, [:result, [3, [:val, [:predicate, [:str?, [[:input, 3]]]]]]]]]
|
98
|
-
]]
|
99
|
-
]
|
100
|
-
]
|
101
|
-
]
|
102
|
-
])
|
103
|
-
end
|
104
|
-
end
|
105
|
-
end
|
106
|
-
|
107
|
-
describe '#and' do
|
108
|
-
let(:other) do
|
109
|
-
Rule::Key.new(str?, name: [:user, :name])
|
110
|
-
end
|
111
|
-
|
112
|
-
it 'returns conjunction rule where value is passed to the right' do
|
113
|
-
present_and_string = rule.and(other)
|
114
|
-
|
115
|
-
expect(present_and_string.(user: { name: 'Jane' })).to be_success
|
116
|
-
|
117
|
-
expect(present_and_string.(user: {})).to be_failure
|
118
|
-
expect(present_and_string.(user: { name: 1 })).to be_failure
|
119
|
-
end
|
120
|
-
end
|
121
|
-
end
|
data/spec/unit/rule/set_spec.rb
DELETED
@@ -1,30 +0,0 @@
|
|
1
|
-
require 'dry/logic/rule'
|
2
|
-
|
3
|
-
RSpec.describe Dry::Logic::Rule::Set do
|
4
|
-
include_context 'predicates'
|
5
|
-
|
6
|
-
subject(:rule) do
|
7
|
-
Dry::Logic::Rule::Set.new([is_string, min_size.curry(6)])
|
8
|
-
end
|
9
|
-
|
10
|
-
let(:is_string) { Dry::Logic::Rule::Value.new(str?) }
|
11
|
-
let(:min_size) { Dry::Logic::Rule::Value.new(min_size?) }
|
12
|
-
|
13
|
-
describe '#call' do
|
14
|
-
it 'applies its rules to the input' do
|
15
|
-
expect(rule.('Address')).to be_success
|
16
|
-
expect(rule.('Addr')).to be_failure
|
17
|
-
end
|
18
|
-
end
|
19
|
-
|
20
|
-
describe '#to_ast' do
|
21
|
-
it 'returns an array representation' do
|
22
|
-
expect(rule.to_ast).to eql([
|
23
|
-
:set, [
|
24
|
-
[:val, [:predicate, [:str?, [[:input, Predicate::Undefined]]]]],
|
25
|
-
[:val, [:predicate, [:min_size?, [[:num, 6], [:input, Predicate::Undefined]]]]]
|
26
|
-
]
|
27
|
-
])
|
28
|
-
end
|
29
|
-
end
|
30
|
-
end
|
@@ -1,99 +0,0 @@
|
|
1
|
-
require 'dry/logic/rule'
|
2
|
-
|
3
|
-
RSpec.describe Dry::Logic::Rule::Value do
|
4
|
-
include_context 'predicates'
|
5
|
-
|
6
|
-
let(:is_nil) { Dry::Logic::Rule::Value.new(none?) }
|
7
|
-
|
8
|
-
let(:is_string) { Dry::Logic::Rule::Value.new(str?) }
|
9
|
-
|
10
|
-
let(:min_size) { Dry::Logic::Rule::Value.new(min_size?) }
|
11
|
-
|
12
|
-
describe '#call' do
|
13
|
-
it 'returns result of a predicate' do
|
14
|
-
expect(is_string.(1)).to be_failure
|
15
|
-
expect(is_string.('1')).to be_success
|
16
|
-
end
|
17
|
-
|
18
|
-
context 'with a composite rule' do
|
19
|
-
subject(:rule) { Dry::Logic::Rule::Value.new(is_nil | is_string) }
|
20
|
-
|
21
|
-
it 'returns a success for valid input' do
|
22
|
-
expect(rule.(nil)).to be_success
|
23
|
-
expect(rule.('foo')).to be_success
|
24
|
-
end
|
25
|
-
|
26
|
-
it 'returns a failure for invalid input' do
|
27
|
-
expect(rule.(312)).to be_failure
|
28
|
-
end
|
29
|
-
|
30
|
-
it 'returns a failure result with curried args' do
|
31
|
-
expect(rule.(312).to_ast).to eql(
|
32
|
-
[:result, [312, [:val, [:predicate, [:str?, [[:input, 312]]]]]]]
|
33
|
-
)
|
34
|
-
end
|
35
|
-
end
|
36
|
-
|
37
|
-
context 'with a custom predicate' do
|
38
|
-
subject(:rule) { Dry::Logic::Rule::Value.new(predicate) }
|
39
|
-
|
40
|
-
let(:response) { double("response", success?: true) }
|
41
|
-
let(:predicate) { double("predicate", arity: 1, curry: curried, call: Result.new(response, double("rule"), test: true)) }
|
42
|
-
let(:curried) { double("curried", arity: 1, call: Result.new(response, double("rule"), test: true)) }
|
43
|
-
|
44
|
-
let(:result) { rule.(test: true) }
|
45
|
-
|
46
|
-
it 'calls its predicate returning custom result' do
|
47
|
-
expect(result).to be_success
|
48
|
-
end
|
49
|
-
|
50
|
-
it 'exposes access to nested result' do
|
51
|
-
expect(response).to receive(:[]).with(:foo).and_return(:bar)
|
52
|
-
expect(result[:foo]).to be(:bar)
|
53
|
-
end
|
54
|
-
|
55
|
-
it 'returns nil from [] when response does not respond to it' do
|
56
|
-
expect(result[:foo]).to be(nil)
|
57
|
-
end
|
58
|
-
|
59
|
-
it 'has no name by default' do
|
60
|
-
expect(result.name).to be(nil)
|
61
|
-
end
|
62
|
-
|
63
|
-
context "works with predicates.arity == 0" do
|
64
|
-
subject(:rule) { Dry::Logic::Rule::Value.new(predicate) }
|
65
|
-
|
66
|
-
let(:predicate) { Dry::Logic::Predicate.new(:without_args) { true } }
|
67
|
-
let(:result) { rule.('sutin') }
|
68
|
-
|
69
|
-
it "calls its predicate without any args" do
|
70
|
-
expect(result).to be_success
|
71
|
-
end
|
72
|
-
end
|
73
|
-
end
|
74
|
-
end
|
75
|
-
|
76
|
-
describe '#and' do
|
77
|
-
it 'returns a conjunction' do
|
78
|
-
string_and_min_size = is_string.and(min_size.curry(3))
|
79
|
-
|
80
|
-
expect(string_and_min_size.('abc')).to be_success
|
81
|
-
expect(string_and_min_size.('abcd')).to be_success
|
82
|
-
|
83
|
-
expect(string_and_min_size.(1)).to be_failure
|
84
|
-
expect(string_and_min_size.('ab')).to be_failure
|
85
|
-
end
|
86
|
-
end
|
87
|
-
|
88
|
-
describe '#or' do
|
89
|
-
it 'returns a disjunction' do
|
90
|
-
nil_or_string = is_nil.or(is_string)
|
91
|
-
|
92
|
-
expect(nil_or_string.(nil)).to be_success
|
93
|
-
expect(nil_or_string.('abcd')).to be_success
|
94
|
-
|
95
|
-
expect(nil_or_string.(true)).to be_failure
|
96
|
-
expect(nil_or_string.(1)).to be_failure
|
97
|
-
end
|
98
|
-
end
|
99
|
-
end
|