Concurrency and threshold throttling for Sidekiq.
Add this line to your application's Gemfile:
gem "sidekiq-throttled"
And then execute:
$ bundle
Or install it yourself as:
$ gem install sidekiq-throttled
Add somewhere in your app's bootstrap (e.g. config/initializers/sidekiq.rb
if
you are using Rails):
require "sidekiq/throttled"
Sidekiq::Throttled.setup!
Once you've done that you can include Sidekiq::Throttled::Worker
to your
job classes and configure throttling:
class MyWorker
include Sidekiq::Worker
include Sidekiq::Throttled::Worker
sidekiq_options :queue => :my_queue
sidekiq_throttle({
# Allow maximum 10 concurrent jobs of this class at a time.
:concurrency => { :limit => 10 },
# Allow maximum 1K jobs being processed within one hour window.
:threshold => { :limit => 1_000, :period => 1.hour }
})
def perform
# ...
end
end
You can throttle jobs dynamically with :key_suffix
option:
class MyWorker
include Sidekiq::Worker
include Sidekiq::Throttled::Worker
sidekiq_options :queue => :my_queue
sidekiq_throttle({
# Allow maximum 10 concurrent jobs per user at a time.
:concurrency => { :limit => 10, :key_suffix => -> (user_id) { user_id } }
})
def perform(user_id)
# ...
end
end
You can also supply dynamic values for limits and periods by supplying a proc for these values. The proc will be evaluated at the time the job is fetched and will receive the same arguments that are passed to the job.
class MyWorker
include Sidekiq::Worker
include Sidekiq::Throttled::Worker
sidekiq_options :queue => :my_queue
sidekiq_throttle({
# Allow maximum 1000 concurrent jobs of this class at a time for VIPs and 10 for all other users.
:concurrency => {
:limit => ->(user_id) { User.vip?(user_id) ? 1_000 : 10 },
:key_suffix => ->(user_id) { User.vip?(user_id) ? "vip" : "std" }
},
# Allow 1000 jobs/hour to be processed for VIPs and 10/day for all others
:threshold => {
:limit => ->(user_id) { User.vip?(user_id) ? 1_000 : 10 },
:period => ->(user_id) { User.vip?(user_id) ? 1.hour : 1.day },
:key_suffix => ->(user_id) { User.vip?(user_id) ? "vip" : "std" }
})
def perform(user_id)
# ...
end
end
NB Don't forget to specify :key_suffix
and make it return different values
if you are using dynamic limit/period options. Otherwise you risk getting into
some trouble.
This library aims to support and is tested against the following Ruby versions:
- Ruby 2.0.0
- Ruby 2.1.x
- Ruby 2.2.x
- Ruby 2.3.x
- Fork sidekiq-throttled on GitHub
- Make your changes
- Ensure all tests pass (
bundle exec rake
) - Send a pull request
- If we like them we'll merge them
- If we've accepted a patch, feel free to ask for commit access!
Copyright (c) 2015 SensorTower Inc. See LICENSE.md for further details.