The purpose of this gem is to provide a simple way to enqueue background jobs in different background job clientes like Sidekiq, Faktory. (More to come). You can push jobs to the clients without actually have the client installed in your system. This is useful for distributed system or data pipelines where you want to use jobs to communicate between different services.
If you are using a monolithic application, you should use the client directly. Or in a Ruby on Rails application consider using Active Jobs. ActiveJobs integrates with a wider range of services and builtin support.
Add this line to your application's Gemfile:
gem 'background_job'
And then execute:
$ bundle install
Or install it yourself as:
$ gem install background_job
Right now the gem supports Sidekiq and Faktory. Client configurations will be covered in the next section.
You can build and push jobs using a simple DSL. The DSL is the same for all clients, but the configurations may vary.
# Enqueue the 'Accounts::ConfirmationEmailWorker' job
# with 'User', 1 arguments to the sidekiq "high_priority_mailing" queue
# to be executed as soon as possible.
BackgroundJob.sidekiq("Accounts::ConfirmationEmailWorker", queue: 'high_priority_mailing')
.with_args("User", 1)
.push
# Schedule the 'Accounts::ConfirmationEmailWorker' job
# with 'User', 1 arguments to the sidekiq "high_priority_mailing" queue
# to be executed in one hour.
BackgroundJob.sidekiq("Accounts::ConfirmationEmailWorker", queue: 'high_priority_mailing')
.with_args("User", 1)
.in(1.hour)
.push
# Enqueue the 'Accounts::ConfirmationEmailWorker' job
# with 'User', 1 arguments to the faktory "mailing" queue
# to be executed as soon as possible.
BackgroundJob.faktory("Accounts::ConfirmationEmailWorker", queue: 'mailing')
.with_args("User", 1)
.push
DSL Methods:
with_args(*args)
: Pass the arguments to the jobin(time)
: Schedule the job to be executed in the future. The time can be aTime
,DateTime
,ActiveSupport::Duration
or a number of seconds.with_job_jid(jid)
: Set the job JID. This is useful to track the job status or cancel it. Optional, it will be generated automatically.created_at(time)
: Set the job creation time. Optional, it will be generated automatically.enqueue_at(time)
: Set the job enqueue time. Optional, it will be generated automatically.push
: Push the job to the client.
Sidekiq configurations are under a BackgroundJob.config.sidekiq
config. You must set the redis
connection where Sidekiq is running.
BackgroundJob.configure do |conf|
conf.sidekiq.redis = { url: 'redis://localhost:6379/0' } # You can pass the Redis instance directly as well
# Or using a connection pool
conf.sidekiq.redis = ConnectionPool.new(size: 5, timeout: 5) do
Redis.new(url: 'redis://localhost:6379/0')
end
# config.sidekiq.namespace = 'sidekiq' # Optional, default is nil in favor of number of databases of Redis
end
From an YAML file
redis:
url: 'redis://localhost:6379/0'
jobs:
UsesJob:
queue: 'default'
retry: 3
BatchImportJob:
queue: 'import'
retry: 0
BackgroundJob.config_for(:sidekiq) do |config|
config.config_path = 'config/background_job.yml'
end
If your are using Sidekiq in a service that does not have a jobs/worker defined, you may want to specify the list of jobs and their configurations like queue
and retry
in the BackgroundJob.config.sidekiq.jobs
configuration.
BackgroundJob.configure do |conf|
conf.sidekiq.redis = { url: 'redis://localhost:6379/0' }
conf.sidekiq.jobs = {
"UsesJob" => { queue: 'default', retry: 3 },
"BatchImportJob" => { queue: 'import', retry: 0 }
}
# Default is true, it means that will raise an error if the job is
# not specified in the jobs configuration
#
# conf.sidekiq.strict = false
end
This are optional, you can keep your backend implementation in your application according to the client documentation. But if you want to get benefit of global configurations, or custom middlewares, you can use the provided mixins.
class Accounts::ConfirmationEmailWorker
+ extend BackgroundJob.mixin(:sidekiq, queue: :mailing)
- include Sidekiq::Worker
- sidekiq_options queue: :mailing
def perform(resource_type, resource_id)
# Do something
end
end
Now when you call Accounts::ConfirmationEmailWorker.perform_async
or Accounts::ConfirmationEmailWorker.perform_in
it will use this gem to push jobs to the backend server with the configurations defined in the mixin and the global configurations.
Faktory configurations are under a BackgroundJob.config.faktory
config. This is in an erly stage. It means that the faktory_worker_ruby gem must be installed in your system.
require 'faktory'
BackgroundJob.configure do |conf|
conf.faktory # Just call it to enable the Faktory client
# Default is true, it means that will raise an error if the job is
# not specified in the jobs configuration
#
# conf.faktory.strict = false
end
If your are using Faktory in a service that does not have a jobs/worker defined, you may want to specify the list of jobs and their configurations like queue
and retry
in the BackgroundJob.config.faktory.jobs
configuration.
BackgroundJob.configure do |conf|
conf.faktory
conf.faktory.jobs = {
"UsesJob" => { queue: 'default', retry: 3 },
"BatchImportJob" => { queue: 'import', retry: 0 }
}
end
This are optional, you can keep your backend implementation in your application according to the client documentation. But if you want to get benefit of global configurations, or custom middlewares, you can use the provided mixins.
class Accounts::ConfirmationEmailWorker
+ extend BackgroundJob.mixin(:faktory, queue: :mailing)
- include Faktory::Job
def perform(resource_type, resource_id)
# Do something
end
end
Now when you call Accounts::ConfirmationEmailWorker.perform_async
or Accounts::ConfirmationEmailWorker.perform_in
it will use this gem to push jobs to the backend server with the configurations defined in the mixin and the global configurations.
You can define middlewares to run before and after the job execution. This is useful to add custom logging, error handling, or any other custom behavior. The current version implements a UniqueJob middleware that prevents the job to be enqueued if it is already in the queue. More details in the next section.
This is an example of a minimal middleware, note the method must return the result or the job will not push the server
class MyMiddleware
def call(job, conn_pool)
puts "Before push"
result = yield
puts "After push"
result
end
end
BackgroundJob.config_for(:sidekiq) do |config|
config.middleware do |chain|
chain.add MyMiddleware
end
end
This library provides one experimental technology to avoid enqueue duplicated jobs. Pro versions of sidekiq and faktory provides this functionality. But this project exposes a mechanism to make this control using Redis
. It's not loaded by default. You can load this function by require and initialize the UniqueJob
middleware according to the service(:faktory
or :sidekiq
).
require 'background_job/middleware/unique_job'
BackgroundJob::Middleware::UniqueJob::bootstrap(service: :sidekiq)
# Or
BackgroundJob::Middleware::UniqueJob::bootstrap(service: :faktory)
# Make sure to add a redis connection to the configuration
BackgroundJob.configure do |conf|
conf.redis = { url: 'redis://localhost:6379/0' }
# Or using a connection pool
conf.redis = ConnectionPool.new(size: 5, timeout: 5) do
Redis.new(url: 'redis://localhost:6379/0')
end
end
After that just define the :uniq
settings by worker
BackgroundJob.sidekiq('Mailing::SignUpWorker', uniq: { across: :queue, timeout: 120 })
.with_args('User', 1)
.push
You can globally disable/enable this function with the BackgroundJob.config.sidekiq.unique_job_active = <true|false>
After checking out the repo, run bin/setup
to install dependencies. You can also run bin/console
for an interactive prompt that will allow you to experiment.
To install this gem onto your local machine, run bundle exec rake install
. To release a new version, update the version number in version.rb
, and then run bundle exec rake release
, which will create a git tag for the version, push git commits and tags, and push the .gem
file to rubygems.org.
Bug reports and pull requests are welcome on GitHub at https://github.com/marcosgz/background_job.
The gem is available as open source under the terms of the MIT License.