Skip to content

Commit

Permalink
Enable per job class job_iteration_max_job_runtime
Browse files Browse the repository at this point in the history
This allows incremental adoption of the setting, without applying the setting
globally. Alternatively, it allows applications to set a conservative global
setting, and a more aggressive setting per jobs.

In order to prevent rogue jobs from causing trouble, the per-job override can
only be set to a value less than the inherited value.
  • Loading branch information
sambostock committed Jul 19, 2022
1 parent 318fa3a commit a1ec6f7
Show file tree
Hide file tree
Showing 5 changed files with 154 additions and 1 deletion.
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

- [241](https://github.com/Shopify/job-iteration/pull/241) - Require Ruby 2.7+, dropping 2.6 support
- [241](https://github.com/Shopify/job-iteration/pull/241) - Require Rails 6.0+, dropping 5.2 support
- [240](https://github.com/Shopify/job-iteration/pull/240) - Allow setting inheritable per-job `job_iteration_max_job_runtime`

## v1.3.6 (Mar 9, 2022)

Expand Down
15 changes: 15 additions & 0 deletions guides/best-practices.md
Original file line number Diff line number Diff line change
Expand Up @@ -66,3 +66,18 @@ JobIteration.max_job_runtime = 5.minutes # nil by default
```

Use this accessor to tweak how often you'd like the job to interrupt itself.

### Per job max job runtime

For more granular control, `job_iteration_max_job_runtime` can be set **per-job class**. This allows both incremental adoption, as well as using a conservative global setting, and an aggressive setting on a per-job basis.

```ruby
class MyJob < ApplicationJob
include JobIteration::Iteration

self.job_iteration_max_job_runtime = 3.minutes

# ...
```

This setting will be inherited by any child classes, although it can be further overridden. Note that no class can **increase** the `max_job_runtime` it has inherited; it can only be **decreased**. No job can increase its `max_job_runtime` beyond the global limit.
9 changes: 9 additions & 0 deletions lib/job-iteration.rb
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,15 @@ module JobIteration
#
# This setting will make it to always interrupt a job after it's been iterating for 5 minutes.
# Defaults to nil which means that jobs will not be interrupted except on termination signal.
#
# This setting can be further reduced (but not increased) by using the inheritable per-class
# job_iteration_max_job_runtime setting.
# @example
#
# class MyJob < ActiveJob::Base
# include JobIteration::Iteration
# self.job_iteration_max_job_runtime = 1.minute
# # ...
attr_accessor :max_job_runtime

# Used internally for hooking into job processing frameworks like Sidekiq and Resque.
Expand Down
29 changes: 28 additions & 1 deletion lib/job-iteration/iteration.rb
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,33 @@ def inspected_cursor
define_callbacks :start
define_callbacks :shutdown
define_callbacks :complete

class_attribute(
:job_iteration_max_job_runtime,
instance_writer: false,
instance_predicate: false,
default: JobIteration.max_job_runtime,
)

singleton_class.prepend(PrependedClassMethods)
end

module PrependedClassMethods
def job_iteration_max_job_runtime=(new)
existing = job_iteration_max_job_runtime

if existing && (!new || new > existing)
existing_label = existing.inspect
new_label = new ? new.inspect : "#{new.inspect} (no limit)"
raise(
ArgumentError,
"job_iteration_max_job_runtime may only decrease; " \
"#{self} tried to increase it from #{existing_label} to #{new_label}",
)
end

super
end
end

module ClassMethods
Expand Down Expand Up @@ -262,7 +289,7 @@ def output_interrupt_summary
end

def job_should_exit?
if ::JobIteration.max_job_runtime && start_time && (Time.now.utc - start_time) > ::JobIteration.max_job_runtime
if job_iteration_max_job_runtime && start_time && (Time.now.utc - start_time) > job_iteration_max_job_runtime
return true
end

Expand Down
101 changes: 101 additions & 0 deletions test/unit/iteration_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -251,6 +251,107 @@ def test_global_max_job_runtime
end
end

def test_per_class_max_job_runtime_with_default_global
freeze_time
parent = build_slow_job_class(iterations: 3, iteration_duration: 30.seconds)
child = Class.new(parent) do
self.job_iteration_max_job_runtime = 1.minute
end

parent.perform_now
assert_empty ActiveJob::Base.queue_adapter.enqueued_jobs

child.perform_now
assert_partially_completed_job(cursor_position: 2)
end

def test_per_class_max_job_runtime_with_global_set_to_nil
freeze_time
with_global_max_job_runtime(nil) do
parent = build_slow_job_class(iterations: 3, iteration_duration: 30.seconds)
child = Class.new(parent) do
self.job_iteration_max_job_runtime = 1.minute
end

parent.perform_now
assert_empty ActiveJob::Base.queue_adapter.enqueued_jobs

child.perform_now
assert_partially_completed_job(cursor_position: 2)
end
end

def test_per_class_max_job_runtime_with_global_set
freeze_time
with_global_max_job_runtime(1.minute) do
parent = build_slow_job_class(iterations: 3, iteration_duration: 30.seconds)
child = Class.new(parent) do
self.job_iteration_max_job_runtime = 30.seconds
end

parent.perform_now
assert_partially_completed_job(cursor_position: 2)
ActiveJob::Base.queue_adapter.enqueued_jobs = []

child.perform_now
assert_partially_completed_job(cursor_position: 1)
end
end

def test_max_job_runtime_cannot_unset_global
with_global_max_job_runtime(30.seconds) do
klass = Class.new(ActiveJob::Base) do
include JobIteration::Iteration
end

error = assert_raises(ArgumentError) do
klass.job_iteration_max_job_runtime = nil
end

assert_equal(
"job_iteration_max_job_runtime may only decrease; " \
"#{klass} tried to increase it from 30 seconds to nil (no limit)",
error.message,
)
end
end

def test_max_job_runtime_cannot_be_higher_than_global
with_global_max_job_runtime(30.seconds) do
klass = Class.new(ActiveJob::Base) do
include JobIteration::Iteration
end

error = assert_raises(ArgumentError) do
klass.job_iteration_max_job_runtime = 1.minute
end

assert_equal(
"job_iteration_max_job_runtime may only decrease; #{klass} tried to increase it from 30 seconds to 1 minute",
error.message,
)
end
end

def test_max_job_runtime_cannot_be_higher_than_parent
with_global_max_job_runtime(1.minute) do
parent = Class.new(ActiveJob::Base) do
include JobIteration::Iteration
self.job_iteration_max_job_runtime = 30.seconds
end
child = Class.new(parent)

error = assert_raises(ArgumentError) do
child.job_iteration_max_job_runtime = 45.seconds
end

assert_equal(
"job_iteration_max_job_runtime may only decrease; #{child} tried to increase it from 30 seconds to 45 seconds",
error.message,
)
end
end

private

# Allows building job classes that read max_job_runtime during the test,
Expand Down

0 comments on commit a1ec6f7

Please sign in to comment.