trailblazer-context 0.3.0 → 0.5.0
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/.github/workflows/ci.yml +17 -0
- data/.gitignore +1 -0
- data/CHANGES.md +40 -0
- data/README.md +0 -1
- data/Rakefile +6 -0
- data/lib/trailblazer/context.rb +13 -84
- data/lib/trailblazer/context/container.rb +100 -0
- data/lib/trailblazer/context/container/with_aliases.rb +102 -0
- data/lib/trailblazer/context/store/indifferent_access.rb +36 -0
- data/lib/trailblazer/context/version.rb +2 -2
- data/test/benchmark/benchmark_helper.rb +32 -0
- data/test/benchmark/indifferent_access_test.rb +89 -0
- data/test/benchmark/indifferent_access_with_aliasing_test.rb +73 -0
- data/test/context_test.rb +123 -11
- data/trailblazer-context.gemspec +3 -1
- metadata +31 -13
- data/.travis.yml +0 -12
- data/lib/trailblazer/context/aliasing.rb +0 -38
- data/lib/trailblazer/context/indifferent_access.rb +0 -38
- data/lib/trailblazer/option.rb +0 -77
- data/test/option_test.rb +0 -186
data/.travis.yml
DELETED
@@ -1,38 +0,0 @@
|
|
1
|
-
module Trailblazer
|
2
|
-
class Context
|
3
|
-
module Aliasing
|
4
|
-
def initialize(wrapped_options, mutable_options, context_alias: {}, **)
|
5
|
-
super(wrapped_options, mutable_options)
|
6
|
-
|
7
|
-
@aliases = context_alias.invert
|
8
|
-
end
|
9
|
-
|
10
|
-
def [](key)
|
11
|
-
return super unless (aka = @aliases[key]) # yepp, nil/false won't work
|
12
|
-
|
13
|
-
super(aka)
|
14
|
-
end
|
15
|
-
|
16
|
-
def key?(key)
|
17
|
-
return super unless (aka = @aliases[key]) # yepp, nil/false won't work
|
18
|
-
|
19
|
-
super(aka)
|
20
|
-
end
|
21
|
-
|
22
|
-
# @private ?
|
23
|
-
def merge(hash)
|
24
|
-
original, mutable_options = decompose
|
25
|
-
|
26
|
-
self.class.new(
|
27
|
-
original,
|
28
|
-
mutable_options.merge(hash),
|
29
|
-
context_alias: @aliases.invert # DISCUSS: maybe we can speed up by remembering the original options?
|
30
|
-
)
|
31
|
-
end
|
32
|
-
|
33
|
-
def to_hash
|
34
|
-
super.merge(Hash[@aliases.collect { |aka, k| key?(k) ? [aka, self[k]] : nil }.compact]) # FIXME: performance!
|
35
|
-
end
|
36
|
-
end
|
37
|
-
end
|
38
|
-
end
|
@@ -1,38 +0,0 @@
|
|
1
|
-
require "trailblazer/context/aliasing"
|
2
|
-
|
3
|
-
module Trailblazer
|
4
|
-
class Context
|
5
|
-
class IndifferentAccess < Context
|
6
|
-
module InstanceMethods
|
7
|
-
def [](name)
|
8
|
-
# TODO: well...
|
9
|
-
@mutable_options.key?(name.to_sym) and return @mutable_options[name.to_sym]
|
10
|
-
@mutable_options.key?(name.to_s) and return @mutable_options[name.to_s]
|
11
|
-
@wrapped_options.key?(name.to_sym) and return @wrapped_options[name.to_sym]
|
12
|
-
@wrapped_options[name.to_s]
|
13
|
-
end
|
14
|
-
|
15
|
-
def self.build(wrapped_options, mutable_options, (_ctx, flow_options), **)
|
16
|
-
new(wrapped_options, mutable_options, flow_options)
|
17
|
-
end
|
18
|
-
end
|
19
|
-
include InstanceMethods
|
20
|
-
|
21
|
-
def key?(name)
|
22
|
-
super(name.to_sym) || super(name.to_s)
|
23
|
-
end
|
24
|
-
|
25
|
-
include Aliasing # FIXME
|
26
|
-
|
27
|
-
|
28
|
-
class << self
|
29
|
-
# This also builds IndifferentAccess::Aliasing.
|
30
|
-
# The {#build} method is designed to take all args from {for_circuit} and then
|
31
|
-
# translate that to the constructor.
|
32
|
-
def build(wrapped_options, mutable_options, (_ctx, flow_options), **)
|
33
|
-
new(wrapped_options, mutable_options, **flow_options)
|
34
|
-
end
|
35
|
-
end
|
36
|
-
end
|
37
|
-
end
|
38
|
-
end
|
data/lib/trailblazer/option.rb
DELETED
@@ -1,77 +0,0 @@
|
|
1
|
-
module Trailblazer
|
2
|
-
class Option
|
3
|
-
# A call implementation invoking `proc.(*args)` and plainly forwarding all arguments.
|
4
|
-
# Override this for your own step strategy (see KW#call!).
|
5
|
-
# @private
|
6
|
-
def self.call!(proc, *args, &block)
|
7
|
-
proc.(*args, &block)
|
8
|
-
end
|
9
|
-
|
10
|
-
# Note that both #evaluate_callable and #evaluate_method drop most of the args.
|
11
|
-
# If you need those, override this class.
|
12
|
-
# @private
|
13
|
-
def self.evaluate_callable(proc, *args, **, &block)
|
14
|
-
call!(proc, *args, &block)
|
15
|
-
end
|
16
|
-
|
17
|
-
# Make the context's instance method a "lambda" and reuse #call!.
|
18
|
-
# @private
|
19
|
-
def self.evaluate_method(proc, *args, exec_context: raise("No :exec_context given."), **, &block)
|
20
|
-
call!(exec_context.method(proc), *args, &block)
|
21
|
-
end
|
22
|
-
|
23
|
-
# Generic builder for a callable "option".
|
24
|
-
# @param call_implementation [Class, Module] implements the process of calling the proc
|
25
|
-
# while passing arguments/options to it in a specific style (e.g. kw args, step interface).
|
26
|
-
# @return [Proc] when called, this proc will evaluate its option (at run-time).
|
27
|
-
def self.build(call_implementation, proc)
|
28
|
-
if proc.is_a? Symbol
|
29
|
-
->(*args, &block) { call_implementation.evaluate_method(proc, *args, &block) }
|
30
|
-
else
|
31
|
-
->(*args, &block) { call_implementation.evaluate_callable(proc, *args, &block) }
|
32
|
-
end
|
33
|
-
end
|
34
|
-
|
35
|
-
# Returns a {Proc} that, when called, invokes the `proc` argument with keyword arguments.
|
36
|
-
# This is known as "step (call) interface".
|
37
|
-
#
|
38
|
-
# This is commonly used by `Operation::step` to wrap the argument and make it
|
39
|
-
# callable in the circuit.
|
40
|
-
#
|
41
|
-
# my_proc = ->(options, **kws) { options["i got called"] = true }
|
42
|
-
# task = Trailblazer::Option::KW(my_proc)
|
43
|
-
# task.(options = {})
|
44
|
-
# options["i got called"] #=> true
|
45
|
-
#
|
46
|
-
# Alternatively, you can pass a symbol and an `:exec_context`.
|
47
|
-
#
|
48
|
-
# my_proc = :some_method
|
49
|
-
# task = Trailblazer::Option::KW(my_proc)
|
50
|
-
#
|
51
|
-
# class A
|
52
|
-
# def some_method(options, **kws)
|
53
|
-
# options["i got called"] = true
|
54
|
-
# end
|
55
|
-
# end
|
56
|
-
#
|
57
|
-
# task.(options = {}, exec_context: A.new)
|
58
|
-
# options["i got called"] #=> true
|
59
|
-
def self.KW(proc)
|
60
|
-
Option.build(KW, proc)
|
61
|
-
end
|
62
|
-
|
63
|
-
# TODO: It would be cool if call! was typed and had `options SymbolizedHash` or something.
|
64
|
-
class KW < Option
|
65
|
-
# A different call implementation that calls `proc` with a "step interface".
|
66
|
-
# your_code.(options, **options)
|
67
|
-
# @private
|
68
|
-
def self.call!(proc, options, *)
|
69
|
-
proc.(options, **options.to_hash) # Step interface: (options, **)
|
70
|
-
end
|
71
|
-
end
|
72
|
-
end
|
73
|
-
# @note This might go to trailblazer-args along with `Context` at some point.
|
74
|
-
def self.Option(proc)
|
75
|
-
Option.build(Option, proc)
|
76
|
-
end
|
77
|
-
end
|
data/test/option_test.rb
DELETED
@@ -1,186 +0,0 @@
|
|
1
|
-
require "test_helper"
|
2
|
-
|
3
|
-
class OptionTest < Minitest::Spec
|
4
|
-
def assert_result(result, block = nil)
|
5
|
-
_(result).must_equal([{a: 1}, 2, {b: 3}, block])
|
6
|
-
|
7
|
-
_(positional.inspect).must_equal %({:a=>1})
|
8
|
-
_(keywords.inspect).must_equal %({:a=>2, :b=>3})
|
9
|
-
end
|
10
|
-
|
11
|
-
describe "positional and kws" do
|
12
|
-
class Step
|
13
|
-
def with_positional_and_keywords(options, a: nil, **more_options, &block)
|
14
|
-
[options, a, more_options, block]
|
15
|
-
end
|
16
|
-
end
|
17
|
-
|
18
|
-
WITH_POSITIONAL_AND_KEYWORDS = ->(options, a: nil, **more_options, &block) do
|
19
|
-
[options, a, more_options, block]
|
20
|
-
end
|
21
|
-
|
22
|
-
class WithPositionalAndKeywords
|
23
|
-
def self.call(options, a: nil, **more_options, &block)
|
24
|
-
[options, a, more_options, block]
|
25
|
-
end
|
26
|
-
end
|
27
|
-
|
28
|
-
let(:positional) { {a: 1} }
|
29
|
-
let(:keywords) { {a: 2, b: 3} }
|
30
|
-
|
31
|
-
let(:block) { ->(*) { snippet } }
|
32
|
-
|
33
|
-
describe ":method" do
|
34
|
-
let(:option) { Trailblazer::Option(:with_positional_and_keywords) }
|
35
|
-
|
36
|
-
it "passes through all args" do
|
37
|
-
step = Step.new
|
38
|
-
|
39
|
-
# positional = { a: 1 }
|
40
|
-
# keywords = { a: 2, b: 3 }
|
41
|
-
assert_result option.(positional, keywords, exec_context: step)
|
42
|
-
end
|
43
|
-
|
44
|
-
it "allows passing a block, too" do
|
45
|
-
step = Step.new
|
46
|
-
|
47
|
-
assert_result option.(positional, keywords, {exec_context: step}, &block), block
|
48
|
-
end
|
49
|
-
end
|
50
|
-
|
51
|
-
describe "lambda" do
|
52
|
-
let(:option) { Trailblazer::Option(WITH_POSITIONAL_AND_KEYWORDS) }
|
53
|
-
|
54
|
-
it "-> {} lambda" do
|
55
|
-
assert_result option.(positional, keywords, {})
|
56
|
-
end
|
57
|
-
|
58
|
-
it "allows passing a block, too" do
|
59
|
-
assert_result option.(positional, keywords, {}, &block), block
|
60
|
-
end
|
61
|
-
|
62
|
-
it "doesn't mind :exec_context" do
|
63
|
-
assert_result option.(positional, keywords, exec_context: "bogus")
|
64
|
-
end
|
65
|
-
end
|
66
|
-
|
67
|
-
describe "Callable" do
|
68
|
-
let(:option) { Trailblazer::Option(WithPositionalAndKeywords) }
|
69
|
-
|
70
|
-
it "passes through all args" do
|
71
|
-
assert_result option.(positional, keywords, exec_context: nil)
|
72
|
-
end
|
73
|
-
|
74
|
-
it "allows passing a block, too" do
|
75
|
-
assert_result option.(positional, keywords, {exec_context: nil}, &block), block
|
76
|
-
end
|
77
|
-
end
|
78
|
-
end
|
79
|
-
|
80
|
-
describe "positionals" do
|
81
|
-
def assert_result_pos(result)
|
82
|
-
_(result).must_equal([1, 2, [3, 4]])
|
83
|
-
_(positionals).must_equal [1, 2, 3, 4]
|
84
|
-
end
|
85
|
-
|
86
|
-
class Step
|
87
|
-
def with_positionals(a, b, *args)
|
88
|
-
[a, b, args]
|
89
|
-
end
|
90
|
-
end
|
91
|
-
|
92
|
-
WITH_POSITIONALS = ->(a, b, *args) do
|
93
|
-
[a, b, args]
|
94
|
-
end
|
95
|
-
|
96
|
-
class WithPositionals
|
97
|
-
def self.call(a, b, *args)
|
98
|
-
[a, b, args]
|
99
|
-
end
|
100
|
-
end
|
101
|
-
|
102
|
-
let(:positionals) { [1, 2, 3, 4] }
|
103
|
-
|
104
|
-
it ":method" do
|
105
|
-
step = Step.new
|
106
|
-
|
107
|
-
option = Trailblazer::Option(:with_positionals)
|
108
|
-
|
109
|
-
assert_result_pos option.(*positionals, exec_context: step)
|
110
|
-
end
|
111
|
-
|
112
|
-
it "-> {} lambda" do
|
113
|
-
option = Trailblazer::Option(WITH_POSITIONALS)
|
114
|
-
|
115
|
-
assert_result_pos option.(*positionals, exec_context: "something")
|
116
|
-
end
|
117
|
-
|
118
|
-
it "callable" do
|
119
|
-
option = Trailblazer::Option(WithPositionals)
|
120
|
-
|
121
|
-
assert_result_pos option.(*positionals, exec_context: "something")
|
122
|
-
end
|
123
|
-
end
|
124
|
-
|
125
|
-
describe "Option::KW" do
|
126
|
-
def assert_result_kws(result)
|
127
|
-
_(result).must_equal([{a: 1, b: 2, c: 3}, 1, 2, {c: 3}])
|
128
|
-
end
|
129
|
-
|
130
|
-
class Step
|
131
|
-
def with_kws(options, a: nil, b: nil, **rest)
|
132
|
-
[options, a, b, rest]
|
133
|
-
end
|
134
|
-
end
|
135
|
-
|
136
|
-
module Task
|
137
|
-
def self.with_kws(options, a: nil, b: nil, **rest)
|
138
|
-
[options, a, b, rest]
|
139
|
-
end
|
140
|
-
end
|
141
|
-
|
142
|
-
WITH_KWS = ->(options, a: nil, b: nil, **rest) do
|
143
|
-
[options, a, b, rest]
|
144
|
-
end
|
145
|
-
|
146
|
-
class WithKWs
|
147
|
-
def self.call(options, a: nil, b: nil, **rest)
|
148
|
-
[options, a, b, rest]
|
149
|
-
end
|
150
|
-
end
|
151
|
-
|
152
|
-
let(:options) { {a: 1, b: 2, c: 3} }
|
153
|
-
|
154
|
-
it ":method" do
|
155
|
-
step = Step.new
|
156
|
-
|
157
|
-
option = Trailblazer::Option::KW(:with_kws)
|
158
|
-
|
159
|
-
assert_result_kws option.(options, exec_context: step)
|
160
|
-
end
|
161
|
-
|
162
|
-
it "Method instance" do
|
163
|
-
option = Trailblazer::Option::KW(Task.method(:with_kws))
|
164
|
-
|
165
|
-
assert_result_kws option.(options, {})
|
166
|
-
end
|
167
|
-
|
168
|
-
it "-> {} lambda" do
|
169
|
-
option = Trailblazer::Option::KW(WITH_KWS)
|
170
|
-
|
171
|
-
assert_result_kws option.(options, {})
|
172
|
-
end
|
173
|
-
|
174
|
-
it "lambda ignores :exec_context" do
|
175
|
-
option = Trailblazer::Option::KW(WITH_KWS)
|
176
|
-
|
177
|
-
assert_result_kws option.(options, exec_context: "something")
|
178
|
-
end
|
179
|
-
|
180
|
-
it "callable" do
|
181
|
-
option = Trailblazer::Option::KW(WithKWs)
|
182
|
-
|
183
|
-
assert_result_kws option.(options, {})
|
184
|
-
end
|
185
|
-
end
|
186
|
-
end
|