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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: d9a47b950a0831c0bf0a1be69de7b7f75c80237ec73a75673397a0b8112aa846
4
- data.tar.gz: 62e9741fcd4053d0f4139aa956f5d51025d644188b2da460d6b7550873257030
3
+ metadata.gz: c254fd322931c448af32ef7aef5bfba2f09293b5d2ab2b85d069783be09cb8e3
4
+ data.tar.gz: e329a253576b4c4756793a068eedbb275c9ddab4ce0af4059285c0d13ad5dffe
5
5
  SHA512:
6
- metadata.gz: 611b50e5912231a593187ea4dc59c5e270674f65413b1c3bbd34d3e1f40849f817954282750d729bdadf23a1bacb35b202c9fb71b2edaccfa59dddd209746234
7
- data.tar.gz: 5b4fbf04648dfe9347330385f1dfdfc32eeadfe4adbb89a015a562c79b0a7b7f6e6b0e91b8d65f3d934e7f3af19bbaf8856ea39555cdd1b649f2ae9f2937ab81
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 not be a `TaskParticipant` record with a `context` of "data", `role` of "user" and `participant` of `@alice`.
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|
@@ -0,0 +1,6 @@
1
+ class AddBecomesZombieAtField < ActiveRecord::Migration[8.0]
2
+ def change
3
+ add_column :operations_tasks, :becomes_zombie_at, :datetime, null: true
4
+ add_index :operations_tasks, :becomes_zombie_at
5
+ end
6
+ end
@@ -1,3 +1,3 @@
1
1
  module Operations
2
- VERSION = "0.5.1"
2
+ VERSION = "0.5.2"
3
3
  end
@@ -1,4 +1,4 @@
1
- # desc "Explaining what the task does"
2
- # task :operations do
3
- # # Task goes here
4
- # end
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.1
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-11 00:00:00.000000000 Z
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/20250309_create_operations_task_participants.rb
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