sidekiq-batchs 0.3.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.github/FUNDING.yml +3 -0
- data/.github/dependabot.yml +8 -0
- data/.github/workflows/ci.yml +37 -0
- data/.github/workflows/codeql-analysis.yml +70 -0
- data/.github/workflows/stale.yml +19 -0
- data/.gitignore +12 -0
- data/.rspec +2 -0
- data/.travis.yml +7 -0
- data/Gemfile +8 -0
- data/Guardfile +4 -0
- data/LICENSE.txt +21 -0
- data/README.md +78 -0
- data/Rakefile +6 -0
- data/lib/sidekiq/batch/callback.rb +104 -0
- data/lib/sidekiq/batch/extension/worker.rb +15 -0
- data/lib/sidekiq/batch/middleware.rb +55 -0
- data/lib/sidekiq/batch/status.rb +61 -0
- data/lib/sidekiq/batch/version.rb +5 -0
- data/lib/sidekiq/batch.rb +331 -0
- data/sidekiq-batch.gemspec +27 -0
- metadata +125 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: 011c76443fe320ee58137aff6159d3cf4010818da75bb8c7bf1c81d5d26d4493
|
4
|
+
data.tar.gz: 5105e3e07c1cfa271cf308f8b7d9decae96985b628535f24a08295a8981e68c2
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: 7c76f4b95624e17a840f2e9863bb24ddccd7fe1bb68b1916fef40261c3fd5c572fb3731aba5e2cc894af93d061e75e13bf25717b3d7cd7d72a1bf96e7e2ca6fe
|
7
|
+
data.tar.gz: fb82c8d808d165684639fd5307135c3505ff706b00438a262e1fd0ba85ee79b52d64093705ea8700bcb73a05a3a5214e312ba6f6fac6ccdab61cd53d9b71cc47
|
data/.github/FUNDING.yml
ADDED
@@ -0,0 +1,37 @@
|
|
1
|
+
name: CI
|
2
|
+
|
3
|
+
on: [pull_request, workflow_dispatch]
|
4
|
+
|
5
|
+
jobs:
|
6
|
+
test:
|
7
|
+
env:
|
8
|
+
REDIS_HOST: 'redis'
|
9
|
+
|
10
|
+
runs-on: ubuntu-latest
|
11
|
+
services:
|
12
|
+
redis:
|
13
|
+
image: redis
|
14
|
+
# Set health checks to wait until redis has started
|
15
|
+
options: >-
|
16
|
+
--health-cmd "redis-cli ping"
|
17
|
+
--health-interval 10s
|
18
|
+
--health-timeout 5s
|
19
|
+
--health-retries 5
|
20
|
+
ports:
|
21
|
+
# Maps port 6379 on service container to the host
|
22
|
+
- 6379:6379
|
23
|
+
|
24
|
+
strategy:
|
25
|
+
fail-fast: false
|
26
|
+
matrix:
|
27
|
+
ruby: ["3.0", "3.1", "3.2", ruby-head]
|
28
|
+
|
29
|
+
steps:
|
30
|
+
- uses: actions/checkout@v2
|
31
|
+
- name: Set up Ruby
|
32
|
+
uses: ruby/setup-ruby@v1
|
33
|
+
with:
|
34
|
+
bundler-cache: true # 'bundle install' and cache gems
|
35
|
+
ruby-version: ${{ matrix.ruby }}
|
36
|
+
- name: Run tests
|
37
|
+
run: bundle exec rake
|
@@ -0,0 +1,70 @@
|
|
1
|
+
# For most projects, this workflow file will not need changing; you simply need
|
2
|
+
# to commit it to your repository.
|
3
|
+
#
|
4
|
+
# You may wish to alter this file to override the set of languages analyzed,
|
5
|
+
# or to provide custom queries or build logic.
|
6
|
+
#
|
7
|
+
# ******** NOTE ********
|
8
|
+
# We have attempted to detect the languages in your repository. Please check
|
9
|
+
# the `language` matrix defined below to confirm you have the correct set of
|
10
|
+
# supported CodeQL languages.
|
11
|
+
#
|
12
|
+
name: "CodeQL"
|
13
|
+
|
14
|
+
on:
|
15
|
+
push:
|
16
|
+
branches: [ master ]
|
17
|
+
pull_request:
|
18
|
+
# The branches below must be a subset of the branches above
|
19
|
+
branches: [ master ]
|
20
|
+
schedule:
|
21
|
+
- cron: '24 3 * * 2'
|
22
|
+
|
23
|
+
jobs:
|
24
|
+
analyze:
|
25
|
+
name: Analyze
|
26
|
+
runs-on: ubuntu-latest
|
27
|
+
permissions:
|
28
|
+
actions: read
|
29
|
+
contents: read
|
30
|
+
security-events: write
|
31
|
+
|
32
|
+
strategy:
|
33
|
+
fail-fast: false
|
34
|
+
matrix:
|
35
|
+
language: [ 'ruby' ]
|
36
|
+
# CodeQL supports [ 'cpp', 'csharp', 'go', 'java', 'javascript', 'python', 'ruby' ]
|
37
|
+
# Learn more about CodeQL language support at https://git.io/codeql-language-support
|
38
|
+
|
39
|
+
steps:
|
40
|
+
- name: Checkout repository
|
41
|
+
uses: actions/checkout@v3
|
42
|
+
|
43
|
+
# Initializes the CodeQL tools for scanning.
|
44
|
+
- name: Initialize CodeQL
|
45
|
+
uses: github/codeql-action/init@v2
|
46
|
+
with:
|
47
|
+
languages: ${{ matrix.language }}
|
48
|
+
# If you wish to specify custom queries, you can do so here or in a config file.
|
49
|
+
# By default, queries listed here will override any specified in a config file.
|
50
|
+
# Prefix the list here with "+" to use these queries and those in the config file.
|
51
|
+
# queries: ./path/to/local/query, your-org/your-repo/queries@main
|
52
|
+
|
53
|
+
# Autobuild attempts to build any compiled languages (C/C++, C#, or Java).
|
54
|
+
# If this step fails, then you should remove it and run the build manually (see below)
|
55
|
+
- name: Autobuild
|
56
|
+
uses: github/codeql-action/autobuild@v2
|
57
|
+
|
58
|
+
# ℹ️ Command-line programs to run using the OS shell.
|
59
|
+
# 📚 https://git.io/JvXDl
|
60
|
+
|
61
|
+
# ✏️ If the Autobuild fails above, remove it and uncomment the following three lines
|
62
|
+
# and modify them (or add more) to build your code if your project
|
63
|
+
# uses a compiled language
|
64
|
+
|
65
|
+
#- run: |
|
66
|
+
# make bootstrap
|
67
|
+
# make release
|
68
|
+
|
69
|
+
- name: Perform CodeQL Analysis
|
70
|
+
uses: github/codeql-action/analyze@v2
|
@@ -0,0 +1,19 @@
|
|
1
|
+
name: Mark stale issues and pull requests
|
2
|
+
|
3
|
+
on:
|
4
|
+
schedule:
|
5
|
+
- cron: "0 0 * * *"
|
6
|
+
|
7
|
+
jobs:
|
8
|
+
stale:
|
9
|
+
|
10
|
+
runs-on: ubuntu-latest
|
11
|
+
|
12
|
+
steps:
|
13
|
+
- uses: actions/stale@v1
|
14
|
+
with:
|
15
|
+
repo-token: ${{ secrets.GITHUB_TOKEN }}
|
16
|
+
stale-issue-message: 'Stale issue message'
|
17
|
+
stale-pr-message: 'Stale pull request message'
|
18
|
+
stale-issue-label: 'no-issue-activity'
|
19
|
+
stale-pr-label: 'no-pr-activity'
|
data/.gitignore
ADDED
data/.rspec
ADDED
data/.travis.yml
ADDED
data/Gemfile
ADDED
data/Guardfile
ADDED
data/LICENSE.txt
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
The MIT License (MIT)
|
2
|
+
|
3
|
+
Copyright (c) 2016 Breamware
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
7
|
+
in the Software without restriction, including without limitation the rights
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
10
|
+
furnished to do so, subject to the following conditions:
|
11
|
+
|
12
|
+
The above copyright notice and this permission notice shall be included in
|
13
|
+
all copies or substantial portions of the Software.
|
14
|
+
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
21
|
+
THE SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,78 @@
|
|
1
|
+
[gem]: https://rubygems.org/gems/sidekiq-batch
|
2
|
+
[travis]: https://travis-ci.org/breamware/sidekiq-batch
|
3
|
+
[codeclimate]: https://codeclimate.com/github/breamware/sidekiq-batch
|
4
|
+
|
5
|
+
# Sidekiq::Batch
|
6
|
+
|
7
|
+
[![Join the chat at https://gitter.im/breamware/sidekiq-batch](https://badges.gitter.im/breamware/sidekiq-batch.svg)](https://gitter.im/breamware/sidekiq-batch?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge)
|
8
|
+
|
9
|
+
[![Gem Version](https://badge.fury.io/rb/sidekiq-batch.svg)][gem]
|
10
|
+
[![Build Status](https://travis-ci.org/breamware/sidekiq-batch.svg?branch=master)][travis]
|
11
|
+
[![Code Climate](https://codeclimate.com/github/breamware/sidekiq-batch/badges/gpa.svg)][codeclimate]
|
12
|
+
[![Code Climate](https://codeclimate.com/github/breamware/sidekiq-batch/badges/coverage.svg)][codeclimate]
|
13
|
+
[![Code Climate](https://codeclimate.com/github/breamware/sidekiq-batch/badges/issue_count.svg)][codeclimate]
|
14
|
+
|
15
|
+
Simple Sidekiq Batch Job implementation.
|
16
|
+
|
17
|
+
## Installation
|
18
|
+
|
19
|
+
Add this line to your application's Gemfile:
|
20
|
+
|
21
|
+
```ruby
|
22
|
+
gem 'sidekiq-batch'
|
23
|
+
```
|
24
|
+
|
25
|
+
And then execute:
|
26
|
+
|
27
|
+
$ bundle
|
28
|
+
|
29
|
+
Or install it yourself as:
|
30
|
+
|
31
|
+
$ gem install sidekiq-batch
|
32
|
+
|
33
|
+
## Usage
|
34
|
+
|
35
|
+
Sidekiq Batch is MOSTLY a drop-in replacement for the API from Sidekiq PRO. See https://github.com/mperham/sidekiq/wiki/Batches for usage.
|
36
|
+
|
37
|
+
## Caveats/Gotchas
|
38
|
+
|
39
|
+
Consider the following workflow:
|
40
|
+
|
41
|
+
* Batch Z created
|
42
|
+
* Worker A queued in batch Z
|
43
|
+
* Worker A starts Worker B in batch Z
|
44
|
+
* Worker B completes *before* worker A does
|
45
|
+
* Worker A completes
|
46
|
+
|
47
|
+
In the standard configuration, the `on(:success)` and `on(:complete)` callbacks will be triggered when Worker B completes.
|
48
|
+
This configuration is the default, simply for legacy reasons. This gem adds the following option to the sidekiq.yml options:
|
49
|
+
|
50
|
+
```yaml
|
51
|
+
:batch_push_interval: 0
|
52
|
+
```
|
53
|
+
|
54
|
+
When this value is *absent* (aka legacy), Worker A will only push the increment of batch jobs (aka Worker B) *when it completes*
|
55
|
+
|
56
|
+
When this value is set to `0`, Worker A will increment the count as soon as `WorkerB.perform_async` is called
|
57
|
+
|
58
|
+
When this value is a positive number, Worker A will wait a maximum of value-seconds before pushing the increment to redis, or until it's done, whichever comes first.
|
59
|
+
|
60
|
+
This comes into play if Worker A is queueing thousands of WorkerB jobs, or has some other reason for WorkerB to complete beforehand.
|
61
|
+
|
62
|
+
If you are queueing many WorkerB jobs, it is recommended to set this value to something like `3` to avoid thousands of calls to redis, and call WorkerB like so:
|
63
|
+
```ruby
|
64
|
+
WorkerB.perform_in(4.seconds, some, args)
|
65
|
+
```
|
66
|
+
this will ensure that the batch callback does not get triggered until WorkerA *and* the last WorkerB job complete.
|
67
|
+
|
68
|
+
If WorkerA is just slow for whatever reason, setting to `0` will update the batch status immediately so that the callbacks don't fire.
|
69
|
+
|
70
|
+
|
71
|
+
## Contributing
|
72
|
+
|
73
|
+
Bug reports and pull requests are welcome on GitHub at https://github.com/breamware/sidekiq-batch.
|
74
|
+
|
75
|
+
|
76
|
+
## License
|
77
|
+
|
78
|
+
The gem is available as open source under the terms of the [MIT License](http://opensource.org/licenses/MIT).
|
data/Rakefile
ADDED
@@ -0,0 +1,104 @@
|
|
1
|
+
module Sidekiq
|
2
|
+
class Batch
|
3
|
+
module Callback
|
4
|
+
class Worker
|
5
|
+
include Sidekiq::Worker
|
6
|
+
|
7
|
+
def perform(clazz, event, opts, bid, parent_bid)
|
8
|
+
return unless %w(success complete).include?(event)
|
9
|
+
clazz, method = clazz.split("#") if (clazz && clazz.class == String && clazz.include?("#"))
|
10
|
+
method = "on_#{event}" if method.nil?
|
11
|
+
status = Sidekiq::Batch::Status.new(bid)
|
12
|
+
|
13
|
+
if clazz && object = Object.const_get(clazz)
|
14
|
+
instance = object.new
|
15
|
+
instance.send(method, status, opts) if instance.respond_to?(method)
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
class Finalize
|
21
|
+
def dispatch status, opts
|
22
|
+
bid = opts["bid"]
|
23
|
+
callback_bid = status.bid
|
24
|
+
event = opts["event"].to_sym
|
25
|
+
callback_batch = bid != callback_bid
|
26
|
+
|
27
|
+
Sidekiq.logger.debug {"Finalize #{event} batch id: #{opts["bid"]}, callback batch id: #{callback_bid} callback_batch #{callback_batch}"}
|
28
|
+
|
29
|
+
batch_status = Status.new bid
|
30
|
+
send(event, bid, batch_status, batch_status.parent_bid)
|
31
|
+
|
32
|
+
|
33
|
+
# Different events are run in different callback batches
|
34
|
+
Sidekiq::Batch.cleanup_redis callback_bid if callback_batch
|
35
|
+
Sidekiq::Batch.cleanup_redis bid if event == :success
|
36
|
+
end
|
37
|
+
|
38
|
+
def success(bid, status, parent_bid)
|
39
|
+
return unless parent_bid
|
40
|
+
|
41
|
+
_, _, success, _, complete, pending, children, failure = Sidekiq.redis do |r|
|
42
|
+
r.multi do |pipeline|
|
43
|
+
pipeline.sadd("BID-#{parent_bid}-success", [bid])
|
44
|
+
pipeline.expire("BID-#{parent_bid}-success", Sidekiq::Batch::BID_EXPIRE_TTL)
|
45
|
+
pipeline.scard("BID-#{parent_bid}-success")
|
46
|
+
pipeline.sadd("BID-#{parent_bid}-complete", [bid])
|
47
|
+
pipeline.scard("BID-#{parent_bid}-complete")
|
48
|
+
pipeline.hincrby("BID-#{parent_bid}", "pending", 0)
|
49
|
+
pipeline.hincrby("BID-#{parent_bid}", "children", 0)
|
50
|
+
pipeline.scard("BID-#{parent_bid}-failed")
|
51
|
+
end
|
52
|
+
end
|
53
|
+
# if job finished successfully and parent batch completed call parent complete callback
|
54
|
+
# Success callback is called after complete callback
|
55
|
+
if complete == children && pending == failure
|
56
|
+
Sidekiq.logger.debug {"Finalize parent complete bid: #{parent_bid}"}
|
57
|
+
Batch.enqueue_callbacks(:complete, parent_bid)
|
58
|
+
end
|
59
|
+
|
60
|
+
end
|
61
|
+
|
62
|
+
def complete(bid, status, parent_bid)
|
63
|
+
pending, children, success = Sidekiq.redis do |r|
|
64
|
+
r.multi do |pipeline|
|
65
|
+
pipeline.hincrby("BID-#{bid}", "pending", 0)
|
66
|
+
pipeline.hincrby("BID-#{bid}", "children", 0)
|
67
|
+
pipeline.scard("BID-#{bid}-success")
|
68
|
+
end
|
69
|
+
end
|
70
|
+
|
71
|
+
# if we batch was successful run success callback
|
72
|
+
if pending.to_i.zero? && children == success
|
73
|
+
Batch.enqueue_callbacks(:success, bid)
|
74
|
+
|
75
|
+
elsif parent_bid
|
76
|
+
# if batch was not successfull check and see if its parent is complete
|
77
|
+
# if the parent is complete we trigger the complete callback
|
78
|
+
# We don't want to run this if the batch was successfull because the success
|
79
|
+
# callback may add more jobs to the parent batch
|
80
|
+
|
81
|
+
Sidekiq.logger.debug {"Finalize parent complete bid: #{parent_bid}"}
|
82
|
+
_, complete, pending, children, failure = Sidekiq.redis do |r|
|
83
|
+
r.multi do |pipeline|
|
84
|
+
pipeline.sadd("BID-#{parent_bid}-complete", [bid])
|
85
|
+
pipeline.scard("BID-#{parent_bid}-complete")
|
86
|
+
pipeline.hincrby("BID-#{parent_bid}", "pending", 0)
|
87
|
+
pipeline.hincrby("BID-#{parent_bid}", "children", 0)
|
88
|
+
pipeline.scard("BID-#{parent_bid}-failed")
|
89
|
+
end
|
90
|
+
end
|
91
|
+
if complete == children && pending == failure
|
92
|
+
Batch.enqueue_callbacks(:complete, parent_bid)
|
93
|
+
end
|
94
|
+
end
|
95
|
+
end
|
96
|
+
|
97
|
+
def cleanup_redis bid, callback_bid=nil
|
98
|
+
Sidekiq::Batch.cleanup_redis bid
|
99
|
+
Sidekiq::Batch.cleanup_redis callback_bid if callback_bid
|
100
|
+
end
|
101
|
+
end
|
102
|
+
end
|
103
|
+
end
|
104
|
+
end
|
@@ -0,0 +1,55 @@
|
|
1
|
+
require_relative 'extension/worker'
|
2
|
+
|
3
|
+
module Sidekiq
|
4
|
+
class Batch
|
5
|
+
module Middleware
|
6
|
+
class ClientMiddleware
|
7
|
+
def call(_worker, msg, _queue, _redis_pool = nil)
|
8
|
+
if (batch = Thread.current[:batch])
|
9
|
+
batch.increment_job_queue(msg['jid']) if (msg[:bid] = batch.bid)
|
10
|
+
end
|
11
|
+
yield
|
12
|
+
end
|
13
|
+
end
|
14
|
+
|
15
|
+
class ServerMiddleware
|
16
|
+
def call(_worker, msg, _queue)
|
17
|
+
if (bid = msg['bid'])
|
18
|
+
begin
|
19
|
+
Thread.current[:batch] = Sidekiq::Batch.new(bid)
|
20
|
+
yield
|
21
|
+
Thread.current[:batch] = nil
|
22
|
+
Batch.process_successful_job(bid, msg['jid'])
|
23
|
+
rescue
|
24
|
+
Batch.process_failed_job(bid, msg['jid'])
|
25
|
+
raise
|
26
|
+
ensure
|
27
|
+
Thread.current[:batch] = nil
|
28
|
+
end
|
29
|
+
else
|
30
|
+
yield
|
31
|
+
end
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
def self.configure
|
36
|
+
Sidekiq.configure_client do |config|
|
37
|
+
config.client_middleware do |chain|
|
38
|
+
chain.add Sidekiq::Batch::Middleware::ClientMiddleware
|
39
|
+
end
|
40
|
+
end
|
41
|
+
Sidekiq.configure_server do |config|
|
42
|
+
config.client_middleware do |chain|
|
43
|
+
chain.add Sidekiq::Batch::Middleware::ClientMiddleware
|
44
|
+
end
|
45
|
+
config.server_middleware do |chain|
|
46
|
+
chain.add Sidekiq::Batch::Middleware::ServerMiddleware
|
47
|
+
end
|
48
|
+
end
|
49
|
+
Sidekiq::Worker.send(:include, Sidekiq::Batch::Extension::Worker)
|
50
|
+
end
|
51
|
+
end
|
52
|
+
end
|
53
|
+
end
|
54
|
+
|
55
|
+
Sidekiq::Batch::Middleware.configure
|
@@ -0,0 +1,61 @@
|
|
1
|
+
module Sidekiq
|
2
|
+
class Batch
|
3
|
+
class Status
|
4
|
+
attr_reader :bid
|
5
|
+
|
6
|
+
def initialize(bid)
|
7
|
+
@bid = bid
|
8
|
+
end
|
9
|
+
|
10
|
+
def join
|
11
|
+
raise "Not supported"
|
12
|
+
end
|
13
|
+
|
14
|
+
def pending
|
15
|
+
Sidekiq.redis { |r| r.hget("BID-#{bid}", 'pending') }.to_i
|
16
|
+
end
|
17
|
+
|
18
|
+
def failures
|
19
|
+
Sidekiq.redis { |r| r.scard("BID-#{bid}-failed") }.to_i
|
20
|
+
end
|
21
|
+
|
22
|
+
def created_at
|
23
|
+
Sidekiq.redis { |r| r.hget("BID-#{bid}", 'created_at') }
|
24
|
+
end
|
25
|
+
|
26
|
+
def total
|
27
|
+
Sidekiq.redis { |r| r.hget("BID-#{bid}", 'total') }.to_i
|
28
|
+
end
|
29
|
+
|
30
|
+
def parent_bid
|
31
|
+
Sidekiq.redis { |r| r.hget("BID-#{bid}", "parent_bid") }
|
32
|
+
end
|
33
|
+
|
34
|
+
def failure_info
|
35
|
+
Sidekiq.redis { |r| r.smembers("BID-#{bid}-failed") } || []
|
36
|
+
end
|
37
|
+
|
38
|
+
def complete?
|
39
|
+
'true' == Sidekiq.redis { |r| r.hget("BID-#{bid}", 'complete') }
|
40
|
+
end
|
41
|
+
|
42
|
+
def child_count
|
43
|
+
Sidekiq.redis { |r| r.hget("BID-#{bid}", 'children') }.to_i
|
44
|
+
end
|
45
|
+
|
46
|
+
def data
|
47
|
+
{
|
48
|
+
bid: bid,
|
49
|
+
total: total,
|
50
|
+
failures: failures,
|
51
|
+
pending: pending,
|
52
|
+
created_at: created_at,
|
53
|
+
complete: complete?,
|
54
|
+
failure_info: failure_info,
|
55
|
+
parent_bid: parent_bid,
|
56
|
+
child_count: child_count
|
57
|
+
}
|
58
|
+
end
|
59
|
+
end
|
60
|
+
end
|
61
|
+
end
|
@@ -0,0 +1,331 @@
|
|
1
|
+
require 'securerandom'
|
2
|
+
require 'sidekiq'
|
3
|
+
|
4
|
+
require 'sidekiq/batch/callback'
|
5
|
+
require 'sidekiq/batch/middleware'
|
6
|
+
require 'sidekiq/batch/status'
|
7
|
+
require 'sidekiq/batch/version'
|
8
|
+
|
9
|
+
module Sidekiq
|
10
|
+
class Batch
|
11
|
+
class NoBlockGivenError < StandardError; end
|
12
|
+
|
13
|
+
BID_EXPIRE_TTL = 2_592_000
|
14
|
+
|
15
|
+
attr_reader :bid, :description, :callback_queue, :created_at
|
16
|
+
|
17
|
+
def initialize(existing_bid = nil)
|
18
|
+
@bid = existing_bid || SecureRandom.urlsafe_base64(10)
|
19
|
+
@existing = !(!existing_bid || existing_bid.empty?) # Basically existing_bid.present?
|
20
|
+
@initialized = false
|
21
|
+
@created_at = Time.now.utc.to_f
|
22
|
+
@bidkey = "BID-" + @bid.to_s
|
23
|
+
@queued_jids = []
|
24
|
+
@pending_jids = []
|
25
|
+
|
26
|
+
@incremental_push = !Sidekiq.default_configuration[:batch_push_interval].nil?
|
27
|
+
@batch_push_interval = Sidekiq.default_configuration[:batch_push_interval]
|
28
|
+
end
|
29
|
+
|
30
|
+
def description=(description)
|
31
|
+
@description = description
|
32
|
+
persist_bid_attr('description', description)
|
33
|
+
end
|
34
|
+
|
35
|
+
def callback_queue=(callback_queue)
|
36
|
+
@callback_queue = callback_queue
|
37
|
+
persist_bid_attr('callback_queue', callback_queue)
|
38
|
+
end
|
39
|
+
|
40
|
+
def callback_batch=(callback_batch)
|
41
|
+
@callback_batch = callback_batch
|
42
|
+
persist_bid_attr('callback_batch', callback_batch)
|
43
|
+
end
|
44
|
+
|
45
|
+
def on(event, callback, options = {})
|
46
|
+
return unless %w(success complete).include?(event.to_s)
|
47
|
+
callback_key = "#{@bidkey}-callbacks-#{event}"
|
48
|
+
Sidekiq.redis do |r|
|
49
|
+
r.multi do |pipeline|
|
50
|
+
pipeline.sadd(callback_key, [JSON.unparse({
|
51
|
+
callback: callback,
|
52
|
+
opts: options
|
53
|
+
})])
|
54
|
+
pipeline.expire(callback_key, BID_EXPIRE_TTL)
|
55
|
+
end
|
56
|
+
end
|
57
|
+
end
|
58
|
+
|
59
|
+
def jobs
|
60
|
+
raise NoBlockGivenError unless block_given?
|
61
|
+
|
62
|
+
bid_data, Thread.current[:bid_data] = Thread.current[:bid_data], []
|
63
|
+
|
64
|
+
begin
|
65
|
+
if !@existing && !@initialized
|
66
|
+
parent_bid = Thread.current[:batch].bid if Thread.current[:batch]
|
67
|
+
|
68
|
+
Sidekiq.redis do |r|
|
69
|
+
r.multi do |pipeline|
|
70
|
+
pipeline.hset(@bidkey, "created_at", @created_at)
|
71
|
+
pipeline.expire(@bidkey, BID_EXPIRE_TTL)
|
72
|
+
if parent_bid
|
73
|
+
pipeline.hset(@bidkey, "parent_bid", parent_bid.to_s)
|
74
|
+
pipeline.hincrby("BID-#{parent_bid}", "children", 1)
|
75
|
+
end
|
76
|
+
end
|
77
|
+
end
|
78
|
+
|
79
|
+
@initialized = true
|
80
|
+
end
|
81
|
+
|
82
|
+
@queued_jids = []
|
83
|
+
@pending_jids = []
|
84
|
+
|
85
|
+
begin
|
86
|
+
parent = Thread.current[:batch]
|
87
|
+
Thread.current[:batch] = self
|
88
|
+
Thread.current[:parent_bid] = parent_bid
|
89
|
+
yield
|
90
|
+
ensure
|
91
|
+
Thread.current[:batch] = parent
|
92
|
+
Thread.current[:parent_bid] = nil
|
93
|
+
end
|
94
|
+
|
95
|
+
return [] if @queued_jids.size == 0
|
96
|
+
conditional_redis_increment!(true)
|
97
|
+
|
98
|
+
Sidekiq.redis do |r|
|
99
|
+
r.multi do |pipeline|
|
100
|
+
if parent_bid
|
101
|
+
pipeline.expire("BID-#{parent_bid}", BID_EXPIRE_TTL)
|
102
|
+
end
|
103
|
+
|
104
|
+
pipeline.expire(@bidkey, BID_EXPIRE_TTL)
|
105
|
+
|
106
|
+
pipeline.sadd(@bidkey + "-jids", @queued_jids)
|
107
|
+
pipeline.expire(@bidkey + "-jids", BID_EXPIRE_TTL)
|
108
|
+
end
|
109
|
+
end
|
110
|
+
|
111
|
+
@queued_jids
|
112
|
+
ensure
|
113
|
+
Thread.current[:bid_data] = bid_data
|
114
|
+
end
|
115
|
+
end
|
116
|
+
|
117
|
+
def increment_job_queue(jid)
|
118
|
+
@queued_jids << jid
|
119
|
+
@pending_jids << jid
|
120
|
+
conditional_redis_increment!
|
121
|
+
end
|
122
|
+
|
123
|
+
def conditional_redis_increment!(force=false)
|
124
|
+
if should_increment? || force
|
125
|
+
parent_bid = Thread.current[:parent_bid]
|
126
|
+
Sidekiq.redis do |r|
|
127
|
+
r.multi do |pipeline|
|
128
|
+
if parent_bid
|
129
|
+
pipeline.hincrby("BID-#{parent_bid}", "total", @pending_jids.length)
|
130
|
+
pipeline.expire("BID-#{parent_bid}", BID_EXPIRE_TTL)
|
131
|
+
end
|
132
|
+
|
133
|
+
pipeline.hincrby(@bidkey, "pending", @pending_jids.length)
|
134
|
+
pipeline.hincrby(@bidkey, "total", @pending_jids.length)
|
135
|
+
pipeline.expire(@bidkey, BID_EXPIRE_TTL)
|
136
|
+
end
|
137
|
+
end
|
138
|
+
@pending_jids = []
|
139
|
+
end
|
140
|
+
end
|
141
|
+
|
142
|
+
def should_increment?
|
143
|
+
return false unless @incremental_push
|
144
|
+
return true if @batch_push_interval == 0 || @queued_jids.length == 1
|
145
|
+
now = Time.now.to_f
|
146
|
+
@last_increment ||= now
|
147
|
+
if @last_increment + @batch_push_interval > now
|
148
|
+
@last_increment = now
|
149
|
+
return true
|
150
|
+
end
|
151
|
+
end
|
152
|
+
|
153
|
+
def invalidate_all
|
154
|
+
Sidekiq.redis do |r|
|
155
|
+
r.setex("invalidated-bid-#{bid}", BID_EXPIRE_TTL, 1)
|
156
|
+
end
|
157
|
+
end
|
158
|
+
|
159
|
+
def parent_bid
|
160
|
+
Sidekiq.redis do |r|
|
161
|
+
r.hget(@bidkey, "parent_bid")
|
162
|
+
end
|
163
|
+
end
|
164
|
+
|
165
|
+
def parent
|
166
|
+
if parent_bid
|
167
|
+
Sidekiq::Batch.new(parent_bid)
|
168
|
+
end
|
169
|
+
end
|
170
|
+
|
171
|
+
def valid?(batch = self)
|
172
|
+
valid = Sidekiq.redis { |r| r.exists("invalidated-bid-#{batch.bid}") }.zero?
|
173
|
+
batch.parent ? valid && valid?(batch.parent) : valid
|
174
|
+
end
|
175
|
+
|
176
|
+
private
|
177
|
+
|
178
|
+
def persist_bid_attr(attribute, value)
|
179
|
+
Sidekiq.redis do |r|
|
180
|
+
r.multi do |pipeline|
|
181
|
+
pipeline.hset(@bidkey, attribute, value)
|
182
|
+
pipeline.expire(@bidkey, BID_EXPIRE_TTL)
|
183
|
+
end
|
184
|
+
end
|
185
|
+
end
|
186
|
+
|
187
|
+
class << self
|
188
|
+
def process_failed_job(bid, jid)
|
189
|
+
_, pending, failed, children, complete, parent_bid = Sidekiq.redis do |r|
|
190
|
+
r.multi do |pipeline|
|
191
|
+
pipeline.sadd("BID-#{bid}-failed", [jid])
|
192
|
+
|
193
|
+
pipeline.hincrby("BID-#{bid}", "pending", 0)
|
194
|
+
pipeline.scard("BID-#{bid}-failed")
|
195
|
+
pipeline.hincrby("BID-#{bid}", "children", 0)
|
196
|
+
pipeline.scard("BID-#{bid}-complete")
|
197
|
+
pipeline.hget("BID-#{bid}", "parent_bid")
|
198
|
+
|
199
|
+
pipeline.expire("BID-#{bid}-failed", BID_EXPIRE_TTL)
|
200
|
+
end
|
201
|
+
end
|
202
|
+
|
203
|
+
# if the batch failed, and has a parent, update the parent to show one pending and failed job
|
204
|
+
if parent_bid
|
205
|
+
Sidekiq.redis do |r|
|
206
|
+
r.multi do |pipeline|
|
207
|
+
pipeline.hincrby("BID-#{parent_bid}", "pending", 1)
|
208
|
+
pipeline.sadd("BID-#{parent_bid}-failed", [jid])
|
209
|
+
pipeline.expire("BID-#{parent_bid}-failed", BID_EXPIRE_TTL)
|
210
|
+
end
|
211
|
+
end
|
212
|
+
end
|
213
|
+
|
214
|
+
if pending.to_i == failed.to_i && children == complete
|
215
|
+
enqueue_callbacks(:complete, bid)
|
216
|
+
end
|
217
|
+
end
|
218
|
+
|
219
|
+
def process_successful_job(bid, jid)
|
220
|
+
failed, pending, children, complete, success, total, parent_bid = Sidekiq.redis do |r|
|
221
|
+
r.multi do |pipeline|
|
222
|
+
pipeline.scard("BID-#{bid}-failed")
|
223
|
+
pipeline.hincrby("BID-#{bid}", "pending", -1)
|
224
|
+
pipeline.hincrby("BID-#{bid}", "children", 0)
|
225
|
+
pipeline.scard("BID-#{bid}-complete")
|
226
|
+
pipeline.scard("BID-#{bid}-success")
|
227
|
+
pipeline.hget("BID-#{bid}", "total")
|
228
|
+
pipeline.hget("BID-#{bid}", "parent_bid")
|
229
|
+
|
230
|
+
pipeline.srem("BID-#{bid}-failed", [jid])
|
231
|
+
pipeline.srem("BID-#{bid}-jids", [jid])
|
232
|
+
pipeline.expire("BID-#{bid}", BID_EXPIRE_TTL)
|
233
|
+
end
|
234
|
+
end
|
235
|
+
|
236
|
+
all_success = pending.to_i.zero? && children == success
|
237
|
+
# if complete or successfull call complete callback (the complete callback may then call successful)
|
238
|
+
if (pending.to_i == failed.to_i && children == complete) || all_success
|
239
|
+
enqueue_callbacks(:complete, bid)
|
240
|
+
enqueue_callbacks(:success, bid) if all_success
|
241
|
+
end
|
242
|
+
end
|
243
|
+
|
244
|
+
def enqueue_callbacks(event, bid)
|
245
|
+
event_name = event.to_s
|
246
|
+
batch_key = "BID-#{bid}"
|
247
|
+
callback_key = "#{batch_key}-callbacks-#{event_name}"
|
248
|
+
already_processed, _, callbacks, queue, parent_bid, callback_batch = Sidekiq.redis do |r|
|
249
|
+
r.multi do |pipeline|
|
250
|
+
pipeline.hget(batch_key, event_name)
|
251
|
+
pipeline.hset(batch_key, event_name, 'true')
|
252
|
+
pipeline.smembers(callback_key)
|
253
|
+
pipeline.hget(batch_key, "callback_queue")
|
254
|
+
pipeline.hget(batch_key, "parent_bid")
|
255
|
+
pipeline.hget(batch_key, "callback_batch")
|
256
|
+
end
|
257
|
+
end
|
258
|
+
|
259
|
+
return if already_processed == 'true'
|
260
|
+
|
261
|
+
queue ||= "default"
|
262
|
+
parent_bid = !parent_bid || parent_bid.empty? ? nil : parent_bid # Basically parent_bid.blank?
|
263
|
+
callback_args = callbacks.reduce([]) do |memo, jcb|
|
264
|
+
cb = Sidekiq.load_json(jcb)
|
265
|
+
memo << [cb['callback'], event_name, cb['opts'], bid, parent_bid]
|
266
|
+
end
|
267
|
+
|
268
|
+
opts = {"bid" => bid, "event" => event_name}
|
269
|
+
|
270
|
+
# Run callback batch finalize synchronously
|
271
|
+
if callback_batch
|
272
|
+
# Extract opts from cb_args or use current
|
273
|
+
# Pass in stored event as callback finalize is processed on complete event
|
274
|
+
cb_opts = callback_args.first&.at(2) || opts
|
275
|
+
|
276
|
+
Sidekiq.logger.debug {"Run callback batch bid: #{bid} event: #{event_name} args: #{callback_args.inspect}"}
|
277
|
+
# Finalize now
|
278
|
+
finalizer = Sidekiq::Batch::Callback::Finalize.new
|
279
|
+
status = Status.new bid
|
280
|
+
finalizer.dispatch(status, cb_opts)
|
281
|
+
|
282
|
+
return
|
283
|
+
end
|
284
|
+
|
285
|
+
Sidekiq.logger.debug {"Enqueue callback bid: #{bid} event: #{event_name} args: #{callback_args.inspect}"}
|
286
|
+
|
287
|
+
if callback_args.empty?
|
288
|
+
# Finalize now
|
289
|
+
finalizer = Sidekiq::Batch::Callback::Finalize.new
|
290
|
+
status = Status.new bid
|
291
|
+
finalizer.dispatch(status, opts)
|
292
|
+
else
|
293
|
+
# Otherwise finalize in sub batch complete callback
|
294
|
+
cb_batch = self.new
|
295
|
+
cb_batch.callback_batch = 'true'
|
296
|
+
Sidekiq.logger.debug {"Adding callback batch: #{cb_batch.bid} for batch: #{bid}"}
|
297
|
+
cb_batch.on(:complete, "Sidekiq::Batch::Callback::Finalize#dispatch", opts)
|
298
|
+
cb_batch.jobs do
|
299
|
+
push_callbacks callback_args, queue
|
300
|
+
end
|
301
|
+
end
|
302
|
+
end
|
303
|
+
|
304
|
+
def cleanup_redis(bid)
|
305
|
+
Sidekiq.logger.debug {"Cleaning redis of batch #{bid}"}
|
306
|
+
Sidekiq.redis do |r|
|
307
|
+
r.del(
|
308
|
+
"BID-#{bid}",
|
309
|
+
"BID-#{bid}-callbacks-complete",
|
310
|
+
"BID-#{bid}-callbacks-success",
|
311
|
+
"BID-#{bid}-failed",
|
312
|
+
|
313
|
+
"BID-#{bid}-success",
|
314
|
+
"BID-#{bid}-complete",
|
315
|
+
"BID-#{bid}-jids",
|
316
|
+
)
|
317
|
+
end
|
318
|
+
end
|
319
|
+
|
320
|
+
private
|
321
|
+
|
322
|
+
def push_callbacks args, queue
|
323
|
+
Sidekiq::Client.push_bulk(
|
324
|
+
'class' => Sidekiq::Batch::Callback::Worker,
|
325
|
+
'args' => args,
|
326
|
+
'queue' => queue
|
327
|
+
) unless args.empty?
|
328
|
+
end
|
329
|
+
end
|
330
|
+
end
|
331
|
+
end
|
@@ -0,0 +1,27 @@
|
|
1
|
+
# coding: utf-8
|
2
|
+
lib = File.expand_path('../lib', __FILE__)
|
3
|
+
$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
|
4
|
+
require 'sidekiq/batch/version'
|
5
|
+
|
6
|
+
Gem::Specification.new do |spec|
|
7
|
+
spec.name = "sidekiq-batchs"
|
8
|
+
spec.version = Sidekiq::Batch::VERSION
|
9
|
+
spec.authors = ["Marcin Naglik"]
|
10
|
+
spec.email = ["tianlu1677@gmail.com"]
|
11
|
+
|
12
|
+
spec.summary = "Sidekiq Batch Jobs"
|
13
|
+
spec.description = "Sidekiq Batch Jobs Implementation"
|
14
|
+
spec.homepage = "http://github.com/breamware/sidekiq-batch"
|
15
|
+
spec.license = "MIT"
|
16
|
+
|
17
|
+
spec.files = `git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features)/}) }
|
18
|
+
spec.bindir = "exe"
|
19
|
+
spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
|
20
|
+
spec.require_paths = ["lib"]
|
21
|
+
|
22
|
+
spec.add_dependency "sidekiq", ">= 7", "<8"
|
23
|
+
|
24
|
+
spec.add_development_dependency "bundler", "~> 2.1"
|
25
|
+
spec.add_development_dependency "rake", "~> 13.0"
|
26
|
+
spec.add_development_dependency "rspec", "~> 3.0"
|
27
|
+
end
|
metadata
ADDED
@@ -0,0 +1,125 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: sidekiq-batchs
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.3.0
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Marcin Naglik
|
8
|
+
autorequire:
|
9
|
+
bindir: exe
|
10
|
+
cert_chain: []
|
11
|
+
date: 2024-04-08 00:00:00.000000000 Z
|
12
|
+
dependencies:
|
13
|
+
- !ruby/object:Gem::Dependency
|
14
|
+
name: sidekiq
|
15
|
+
requirement: !ruby/object:Gem::Requirement
|
16
|
+
requirements:
|
17
|
+
- - ">="
|
18
|
+
- !ruby/object:Gem::Version
|
19
|
+
version: '7'
|
20
|
+
- - "<"
|
21
|
+
- !ruby/object:Gem::Version
|
22
|
+
version: '8'
|
23
|
+
type: :runtime
|
24
|
+
prerelease: false
|
25
|
+
version_requirements: !ruby/object:Gem::Requirement
|
26
|
+
requirements:
|
27
|
+
- - ">="
|
28
|
+
- !ruby/object:Gem::Version
|
29
|
+
version: '7'
|
30
|
+
- - "<"
|
31
|
+
- !ruby/object:Gem::Version
|
32
|
+
version: '8'
|
33
|
+
- !ruby/object:Gem::Dependency
|
34
|
+
name: bundler
|
35
|
+
requirement: !ruby/object:Gem::Requirement
|
36
|
+
requirements:
|
37
|
+
- - "~>"
|
38
|
+
- !ruby/object:Gem::Version
|
39
|
+
version: '2.1'
|
40
|
+
type: :development
|
41
|
+
prerelease: false
|
42
|
+
version_requirements: !ruby/object:Gem::Requirement
|
43
|
+
requirements:
|
44
|
+
- - "~>"
|
45
|
+
- !ruby/object:Gem::Version
|
46
|
+
version: '2.1'
|
47
|
+
- !ruby/object:Gem::Dependency
|
48
|
+
name: rake
|
49
|
+
requirement: !ruby/object:Gem::Requirement
|
50
|
+
requirements:
|
51
|
+
- - "~>"
|
52
|
+
- !ruby/object:Gem::Version
|
53
|
+
version: '13.0'
|
54
|
+
type: :development
|
55
|
+
prerelease: false
|
56
|
+
version_requirements: !ruby/object:Gem::Requirement
|
57
|
+
requirements:
|
58
|
+
- - "~>"
|
59
|
+
- !ruby/object:Gem::Version
|
60
|
+
version: '13.0'
|
61
|
+
- !ruby/object:Gem::Dependency
|
62
|
+
name: rspec
|
63
|
+
requirement: !ruby/object:Gem::Requirement
|
64
|
+
requirements:
|
65
|
+
- - "~>"
|
66
|
+
- !ruby/object:Gem::Version
|
67
|
+
version: '3.0'
|
68
|
+
type: :development
|
69
|
+
prerelease: false
|
70
|
+
version_requirements: !ruby/object:Gem::Requirement
|
71
|
+
requirements:
|
72
|
+
- - "~>"
|
73
|
+
- !ruby/object:Gem::Version
|
74
|
+
version: '3.0'
|
75
|
+
description: Sidekiq Batch Jobs Implementation
|
76
|
+
email:
|
77
|
+
- tianlu1677@gmail.com
|
78
|
+
executables: []
|
79
|
+
extensions: []
|
80
|
+
extra_rdoc_files: []
|
81
|
+
files:
|
82
|
+
- ".github/FUNDING.yml"
|
83
|
+
- ".github/dependabot.yml"
|
84
|
+
- ".github/workflows/ci.yml"
|
85
|
+
- ".github/workflows/codeql-analysis.yml"
|
86
|
+
- ".github/workflows/stale.yml"
|
87
|
+
- ".gitignore"
|
88
|
+
- ".rspec"
|
89
|
+
- ".travis.yml"
|
90
|
+
- Gemfile
|
91
|
+
- Guardfile
|
92
|
+
- LICENSE.txt
|
93
|
+
- README.md
|
94
|
+
- Rakefile
|
95
|
+
- lib/sidekiq/batch.rb
|
96
|
+
- lib/sidekiq/batch/callback.rb
|
97
|
+
- lib/sidekiq/batch/extension/worker.rb
|
98
|
+
- lib/sidekiq/batch/middleware.rb
|
99
|
+
- lib/sidekiq/batch/status.rb
|
100
|
+
- lib/sidekiq/batch/version.rb
|
101
|
+
- sidekiq-batch.gemspec
|
102
|
+
homepage: http://github.com/breamware/sidekiq-batch
|
103
|
+
licenses:
|
104
|
+
- MIT
|
105
|
+
metadata: {}
|
106
|
+
post_install_message:
|
107
|
+
rdoc_options: []
|
108
|
+
require_paths:
|
109
|
+
- lib
|
110
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
111
|
+
requirements:
|
112
|
+
- - ">="
|
113
|
+
- !ruby/object:Gem::Version
|
114
|
+
version: '0'
|
115
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
116
|
+
requirements:
|
117
|
+
- - ">="
|
118
|
+
- !ruby/object:Gem::Version
|
119
|
+
version: '0'
|
120
|
+
requirements: []
|
121
|
+
rubygems_version: 3.1.6
|
122
|
+
signing_key:
|
123
|
+
specification_version: 4
|
124
|
+
summary: Sidekiq Batch Jobs
|
125
|
+
test_files: []
|