standard_procedure_operations 0.5.1 → 0.5.2
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 +4 -4
- data/README.md +11 -1
- data/app/models/operations/task/background.rb +11 -0
- data/app/models/operations/task.rb +2 -1
- data/db/migrate/20250403075414_add_becomes_zombie_at_field.rb +6 -0
- data/lib/operations/version.rb +1 -1
- data/lib/tasks/operations_tasks.rake +4 -4
- metadata +4 -3
- /data/db/migrate/{20250309_create_operations_task_participants.rb → 20250309160616_create_operations_task_participants.rb} +0 -0
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: c254fd322931c448af32ef7aef5bfba2f09293b5d2ab2b85d069783be09cb8e3
|
4
|
+
data.tar.gz: e329a253576b4c4756793a068eedbb275c9ddab4ce0af4059285c0d13ad5dffe
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 920f49215031fce81eabe5dd4aec50df392d1dfd93cc497eb47aafbb4544ab9defd378faf0bb99d88b0c5c5fbf326e2dd2e77df6f2d8c737589944e174c0e740
|
7
|
+
data.tar.gz: 2c5c2cc235ed70113c7881fb75f46a9962548bd333d589cc73cbcbc5fbeb50f030b5212e2e2bbd16b7b4b3509284aa1b8e04d151b84cfde07780c4da5f9a9e5c
|
data/README.md
CHANGED
@@ -295,7 +295,7 @@ For example, you create your task as:
|
|
295
295
|
@alice = User.find 123
|
296
296
|
@task = DoSomethingImportant.call user: @alice
|
297
297
|
```
|
298
|
-
There will
|
298
|
+
There will be a `TaskParticipant` record with a `context` of "data", `role` of "user" and `participant` of `@alice`.
|
299
299
|
|
300
300
|
Likewise, you can see all the tasks that Alice was involved with using:
|
301
301
|
```ruby
|
@@ -470,6 +470,16 @@ class WaitForSomething < Operations::Task
|
|
470
470
|
end
|
471
471
|
```
|
472
472
|
|
473
|
+
#### Zombie tasks
|
474
|
+
|
475
|
+
There's a chance that the `Operations::TaskRunnerJob` might get lost - maybe there's a crash in some process and the job does not restart correctly. As the process for handling background tasks relies on the task "waking up", performing the next action, then queuing up the next task-runner, if the background job does not queue as expected, the task will sit there, waiting forever.
|
476
|
+
|
477
|
+
To monitor for this, every task can be checked to see if it is a `zombie?`. This means that the current time is more than 3 times the expected delay, compared to the `updated_at` field. So if the `delay` is set to 1 minute and the task last woke up more than 3 minutes ago, it is classed as a zombie.
|
478
|
+
|
479
|
+
There are two ways to handle zombies.
|
480
|
+
- Manually; add a user interface listing your tasks with a "Restart" button. The "Restart" button calls `restart` on the task (which internally schedules a new task runner job).
|
481
|
+
- Automatically; set up a cron job which calls the `operations:restart_zombie_tasks` rake task. This rake task searches for zombie jobs and calls `restart` on them. Note that cron jobs have a minimum resolution of 1 minute so this will cause pauses in tasks with a delay measured in seconds. Also be aware that a cron job that calls a rake task will load the entire Rails stack as a new process, so be sure that your server has sufficient memory to cope. If you're using [SolidQueue](https://github.com/rails/solid_queue/), the job runner already sets up a separate "supervisor" process and allows you to define [recurring jobs](https://github.com/rails/solid_queue/#recurring-tasks) with a resolution of 1 second. This may be a suitable solution, but I've not tried it yet.
|
482
|
+
|
473
483
|
## Testing
|
474
484
|
Because operations are intended to model long, complex, flowcharts of decisions and actions, it can be a pain coming up with the combinations of inputs to test every path through the sequence.
|
475
485
|
|
@@ -1,6 +1,11 @@
|
|
1
1
|
module Operations::Task::Background
|
2
2
|
extend ActiveSupport::Concern
|
3
3
|
|
4
|
+
included do
|
5
|
+
scope :zombies, -> { zombies_at(Time.now) }
|
6
|
+
scope :zombies_at, ->(time) { where(becomes_zombie_at: ..time) }
|
7
|
+
end
|
8
|
+
|
4
9
|
class_methods do
|
5
10
|
def delay(value) = @background_delay = value
|
6
11
|
|
@@ -15,9 +20,15 @@ module Operations::Task::Background
|
|
15
20
|
def timeout_handler = @on_timeout
|
16
21
|
|
17
22
|
def with_timeout(data) = data.merge(_execution_timeout: execution_timeout.from_now.utc)
|
23
|
+
|
24
|
+
def restart_zombie_tasks = zombies.find_each { |t| t.restart! }
|
18
25
|
end
|
19
26
|
|
27
|
+
def zombie? = Time.now > (updated_at + zombie_delay)
|
28
|
+
|
20
29
|
private def background_delay = self.class.background_delay
|
30
|
+
private def zombie_delay = background_delay * 3
|
31
|
+
private def zombie_time = becomes_zombie_at || Time.now
|
21
32
|
private def execution_timeout = self.class.execution_timeout
|
22
33
|
private def timeout_handler = self.class.timeout_handler
|
23
34
|
private def timeout!
|
@@ -35,9 +35,10 @@ module Operations
|
|
35
35
|
end
|
36
36
|
|
37
37
|
def perform_later
|
38
|
-
waiting
|
38
|
+
update! status: "waiting", becomes_zombie_at: Time.now + zombie_delay
|
39
39
|
TaskRunnerJob.set(wait_until: background_delay.from_now).perform_later self
|
40
40
|
end
|
41
|
+
alias_method :restart!, :perform_later
|
41
42
|
|
42
43
|
def self.call(**)
|
43
44
|
build(background: false, **).tap do |task|
|
data/lib/operations/version.rb
CHANGED
@@ -1,4 +1,4 @@
|
|
1
|
-
|
2
|
-
|
3
|
-
|
4
|
-
|
1
|
+
desc "Restart any zombie tasks"
|
2
|
+
task :restart_zombie_tasks do
|
3
|
+
Operations::Task.restart_zombie_tasks
|
4
|
+
end
|
metadata
CHANGED
@@ -1,13 +1,13 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: standard_procedure_operations
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.5.
|
4
|
+
version: 0.5.2
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Rahoul Baruah
|
8
8
|
bindir: bin
|
9
9
|
cert_chain: []
|
10
|
-
date: 2025-03
|
10
|
+
date: 2025-04-03 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: rails
|
@@ -65,7 +65,8 @@ files:
|
|
65
65
|
- app/models/operations/task_participant.rb
|
66
66
|
- config/routes.rb
|
67
67
|
- db/migrate/20250127160616_create_operations_tasks.rb
|
68
|
-
- db/migrate/
|
68
|
+
- db/migrate/20250309160616_create_operations_task_participants.rb
|
69
|
+
- db/migrate/20250403075414_add_becomes_zombie_at_field.rb
|
69
70
|
- lib/operations.rb
|
70
71
|
- lib/operations/cannot_wait_in_foreground.rb
|
71
72
|
- lib/operations/engine.rb
|