batch_reactor 0.0.1

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 ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA1:
3
+ metadata.gz: 15dc90fa2f1e38647ea6e975cb258274f4d51573
4
+ data.tar.gz: 4b4ec60ee51726b1c865274d7c7e6629e2fa3235
5
+ SHA512:
6
+ metadata.gz: a17b3a124b8556aea468dd265ac40de1bdc7f805ab39aa400a0599bb520df6f562da6ff5cf76d55b1a6e8ef2a901742e629f0c3a3f15323b5a0d53bdd9cadad6
7
+ data.tar.gz: c98f2c55832fc30d70dfae41ecfd06a86c3aeb44a5dc3f2a53577eab55a33fd9a96fee2004d7f74ac03be6ac37321c2e9a8a9954d3a2506135ca6c171ec191fa
data/LICENSE.txt ADDED
@@ -0,0 +1,13 @@
1
+ Copyright 2014-2015 Thomas RM Rogers
2
+
3
+ Licensed under the Apache License, Version 2.0 (the "License");
4
+ you may not use this file except in compliance with the License.
5
+ You may obtain a copy of the License at
6
+
7
+ http://www.apache.org/licenses/LICENSE-2.0
8
+
9
+ Unless required by applicable law or agreed to in writing, software
10
+ distributed under the License is distributed on an "AS IS" BASIS,
11
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12
+ See the License for the specific language governing permissions and
13
+ limitations under the License.
data/README.md ADDED
@@ -0,0 +1,17 @@
1
+ # batch_reactor
2
+
3
+ ## Copyright
4
+
5
+ Copyright 2015 Thomas Rogers.
6
+
7
+ Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
8
+
9
+ [http://www.apache.org/licenses/LICENSE-2.0](http://www.apache.org/licenses/LICENSE-2.0)
10
+
11
+ Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
12
+
13
+
14
+
15
+
16
+
17
+
@@ -0,0 +1,126 @@
1
+ module BatchReactor
2
+ class Reactor
3
+ include MonitorMixin
4
+ extend Forwardable
5
+
6
+ def initialize(options, &yield_batch_callback)
7
+ @yield_batch_callback = yield_batch_callback
8
+ @front_buffer = []
9
+ @back_buffer = []
10
+ @stopping_promise = Ione::Promise.new
11
+ @stopped_promise = Ione::Promise.new
12
+ @max_batch_size = options.fetch(:max_batch_size) { 100 }
13
+ @no_work_backoff = options.fetch(:no_work_backoff) { 0.1 }
14
+ super()
15
+ end
16
+
17
+ def start
18
+ return @started_promise.future if @started_promise
19
+
20
+ @started_promise = Ione::Promise.new
21
+ Thread.start do
22
+ @started_promise.fulfill(self)
23
+
24
+ last_batch_future = Ione::Future.resolved(nil)
25
+ until @stopping
26
+ swap_buffers if needs_work?
27
+ next if no_work?
28
+ last_batch_future = process_batch
29
+ end
30
+
31
+ last_batch_future.on_complete { |_, _| shutdown }
32
+ end
33
+ @started_promise.future
34
+ end
35
+
36
+ def stop
37
+ @stopping = true
38
+ @stopped_promise.future
39
+ end
40
+
41
+ def perform_within_batch(&block)
42
+ promise = Ione::Promise.new
43
+ if @stopping_promise.future.resolved?
44
+ promise.fail(StandardError.new('Reactor stopped!'))
45
+ else
46
+ synchronize { @back_buffer << Work.new(block, promise) }
47
+ end
48
+ promise.future
49
+ end
50
+
51
+ private
52
+
53
+ def_delegator :@front_buffer, :empty?, :needs_work?
54
+
55
+ Work = Struct.new(:proc, :promise, :result)
56
+
57
+ def swap_buffers
58
+ synchronize do
59
+ temp_buffer = @front_buffer
60
+ @front_buffer = @back_buffer
61
+ @back_buffer = temp_buffer
62
+ end
63
+ end
64
+
65
+ def no_work?
66
+ if @front_buffer.empty?
67
+ sleep @no_work_backoff
68
+ true
69
+ end
70
+ end
71
+
72
+ def process_batch
73
+ buffer = @front_buffer.slice!(0...@max_batch_size)
74
+ batch_future = create_batch(buffer)
75
+ batch_future.on_complete do |_, error|
76
+ error ? handle_failure(buffer, error) : handle_success(buffer)
77
+ end
78
+ batch_future
79
+ end
80
+
81
+ def create_batch(buffer)
82
+ @yield_batch_callback.call { |batch| perform_work(batch, buffer) }
83
+ end
84
+
85
+ def perform_work(batch, buffer)
86
+ buffer.each { |work| work.result = work.proc.call(batch) }
87
+ end
88
+
89
+ def handle_success(buffer)
90
+ buffer.each do |work|
91
+ result = work.result
92
+ if result.respond_to?(:on_complete)
93
+ handle_result_future(result, work)
94
+ else
95
+ work.promise.fulfill(result)
96
+ end
97
+ end
98
+ end
99
+
100
+ def handle_result_future(result, work)
101
+ result.on_complete do |value, error|
102
+ error ? work.promise.fail(error) : work.promise.fulfill(value)
103
+ end
104
+ end
105
+
106
+ def handle_failure(buffer, error)
107
+ buffer.each { |work| work.promise.fail(error) }
108
+ end
109
+
110
+ def shutdown
111
+ @stopping_promise.fulfill(self)
112
+
113
+ futures = []
114
+ finish_remaining_work(futures)
115
+ swap_buffers
116
+ finish_remaining_work(futures)
117
+
118
+ Ione::Future.all(futures).on_complete { |_, _| @stopped_promise.fulfill(self) }
119
+ end
120
+
121
+ def finish_remaining_work(futures)
122
+ futures << process_batch until @front_buffer.empty?
123
+ end
124
+
125
+ end
126
+ end
@@ -0,0 +1,30 @@
1
+ module BatchReactor
2
+ class ReactorCluster
3
+
4
+ def initialize(count, options, &batch_callback)
5
+ @reactors = count.times.map do |index|
6
+ Reactor.new(options) { |&block| batch_callback.call(index, &block) }
7
+ end
8
+ end
9
+
10
+ def define_partitioner(&partitioner_callback)
11
+ @partitioner_callback = partitioner_callback
12
+ end
13
+
14
+ def start
15
+ futures = @reactors.map(&:start)
16
+ Ione::Future.all(futures)
17
+ end
18
+
19
+ def stop
20
+ futures = @reactors.map(&:stop)
21
+ Ione::Future.all(futures)
22
+ end
23
+
24
+ def perform_within_batch(key, &block)
25
+ partition = @partitioner_callback.call(key)
26
+ @reactors[partition].perform_within_batch(&block)
27
+ end
28
+
29
+ end
30
+ end
@@ -0,0 +1,2 @@
1
+ require 'batch_reactor/reactor'
2
+ require 'batch_reactor/reactor_cluster'
metadata ADDED
@@ -0,0 +1,65 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: batch_reactor
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.0.1
5
+ platform: ruby
6
+ authors:
7
+ - Thomas RM Rogers
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2015-12-02 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: ione
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ~>
18
+ - !ruby/object:Gem::Version
19
+ version: '1.2'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ~>
25
+ - !ruby/object:Gem::Version
26
+ version: '1.2'
27
+ description: |-
28
+ A reactor pattern batching system for Ruby.
29
+ It provides simple functionality for batching work in background thread.
30
+ Also adds simple but powerful work partitioning behaviour
31
+ email: thomasrogers03@gmail.com
32
+ executables: []
33
+ extensions: []
34
+ extra_rdoc_files: []
35
+ files:
36
+ - LICENSE.txt
37
+ - README.md
38
+ - lib/batch_reactor.rb
39
+ - lib/batch_reactor/reactor.rb
40
+ - lib/batch_reactor/reactor_cluster.rb
41
+ homepage: https://www.github.com/thomasrogers03/batch_reactor
42
+ licenses:
43
+ - Apache License 2.0
44
+ metadata: {}
45
+ post_install_message:
46
+ rdoc_options: []
47
+ require_paths:
48
+ - lib
49
+ required_ruby_version: !ruby/object:Gem::Requirement
50
+ requirements:
51
+ - - '>='
52
+ - !ruby/object:Gem::Version
53
+ version: '0'
54
+ required_rubygems_version: !ruby/object:Gem::Requirement
55
+ requirements:
56
+ - - '>='
57
+ - !ruby/object:Gem::Version
58
+ version: '0'
59
+ requirements: []
60
+ rubyforge_project:
61
+ rubygems_version: 2.4.8
62
+ signing_key:
63
+ specification_version: 4
64
+ summary: A reactor pattern batching system for Ruby
65
+ test_files: []