standard_procedure_operations 0.4.2 → 0.5.0

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: 40faff52f3486a0ce989b2ea1fdb8350d8872827391331d0e5d8ad78c4e86a0a
4
- data.tar.gz: b3667d26ebea3598955b0d4bf71eed3e22774c4aaf589ea3851c02454144a539
3
+ metadata.gz: a14742a79230259d8cb55462cac7682b6adfe8993cacc0ec4f35ed3832910858
4
+ data.tar.gz: 939c54f7cb755794ecd71ee920bfc1ff65f8e925187f6ccd9c1e98515d5a1b3d
5
5
  SHA512:
6
- metadata.gz: a84bbf1163fbc9515450cf129bc9385043bb21f5d9d1e5c6135318f44caec11c1971801acdcd7e541915815e067898f36f7ffdbdb4e133b38f4a3a50156276aa
7
- data.tar.gz: 4e6bbab0ababc9eb6a24f9b9dacbf4e755e08c44fd8a6da01502f59a5e763945df8cccd9198e0ef5532a80d31cccec81fb4eaafb5fa390fd682daf8e408fd66e
6
+ metadata.gz: 6c3ec1e16dada2b58d48aa72a11e64aaa7701b7a481836cc9b35ac268f752257b9dabf619eb43ecc6e4541fbc6ed50ae1b51d6ee4b4ed60fe4ad2ec302b1f8b8
7
+ data.tar.gz: 67d54e8449c264dc2fb1dd4883b64a810cabd47a2b976e931f1060d1573d0579c4964336159f6f8acf1d6d599ae3edf4de64b818f08c4c46c963035534264500
data/README.md CHANGED
@@ -290,7 +290,24 @@ Instead of using the standard [JSON coder](https://api.rubyonrails.org/v4.2/clas
290
290
 
291
291
  If the original database record was deleted between the time the hash was serialised and when it was retrieved, the `GlobalID::Locator` will fail. With ActiveJob, this means that the job cannot run and is discarded. For Operations, we attempt to deserialise a second time, returning the GlobalID string instead of the model. So be aware that when you access `data` or `results` you may receive a string (similar to `"gid://test-app/User/1"`) instead of the models you were expecting. And the error handling deserialiser is very simple so you may get format changes in some of the data as well. If serialisation fails you can access the original JSON string as `data.raw_data` or `results[:raw_data]`.
292
292
 
293
- TODO: Replace the ActiveJob::Arguments deserialiser with the [transporter](https://github.com/standard-procedure/plumbing/blob/main/lib/plumbing/actor/transporter.rb) from [plumbing](https://github.com/standard-procedure/plumbing)
293
+ #### Indexing data and results
294
+
295
+ If you need to search through existing tasks by a model that is stored in the `data` or `results` fields - for example, you might want to list all operations that were started by a particular `User` - the models can be indexed alongside the task.
296
+
297
+ If your ActiveRecord model (in this example, `User`) includes the `Operations::Participant` module, it will be linked with any task that references that model. A polymorphic join table, `operations_task_participants` is used for this. Whenever a task is saved, any `Operations::Participant` records are located in the `data` and `results` collections and a `Operations::TaskParticipant` record created to join the model to the task. The `context` attribute records whether the association is in the `data` or `results` collection and the `role` attribute is the name of the hash key.
298
+
299
+ For example, you create your task as:
300
+ ```ruby
301
+ @alice = User.find 123
302
+ @task = DoSomethingImportant.call user: @alice
303
+ ```
304
+ There will not be a `TaskParticipant` record with a `context` of "data", `role` of "user" and `participant` of `@alice`.
305
+
306
+ Likewise, you can see all the tasks that Alice was involved with using:
307
+ ```ruby
308
+ @alice.involved_in_operations_as("user") # => collection of tasks where Alice was a "user" in the "data" collection
309
+ @alice.involved_in_operations_as("user", context: "results") # => collection of tasks where Alice was a "user" in the "results" collection
310
+ ```
294
311
 
295
312
  ### Failures and exceptions
296
313
  If any handlers raise an exception, the task will be terminated. It will be marked as `failed?` and the `results` hash will contain `results[:failure_message]`, `results[:exception_class]` and `results[:exception_backtrace]` for the exception's message, class name and backtrace respectively.
@@ -626,12 +643,8 @@ The gem is available as open source under the terms of the [LGPL License](/LICEN
626
643
  - [x] Simplify calling sub-tasks (and testing them)
627
644
  - [ ] Figure out how to stub calling sub-tasks with known results data
628
645
  - [ ] Figure out how to test the parameters passed to sub-tasks when they are called
629
- - [ ] Split out the state-management definition stuff from the task class (so you can use it without subclassing Operations::Task)
630
646
  - [x] Make Operations::Task work in the background using ActiveJob
631
647
  - [x] Add pause/resume capabilities (for example, when a task needs to wait for user input)
632
648
  - [x] Add wait for sub-tasks capabilities
633
649
  - [x] Add GraphViz visualization export for task flows
634
- - [ ] Add ActiveModel validations support for task parameters
635
650
  - [ ] Option to change background job queue and priority settings
636
- - [ ] Replace the ActiveJob::Arguments deserialiser with the [transporter](https://github.com/standard-procedure/plumbing/blob/main/lib/plumbing/actor/transporter.rb) from [plumbing](https://github.com/standard-procedure/plumbing)
637
- - [ ] Maybe? Split this out into two gems - one defining an Operation (pure ruby) and another defining the Task (using ActiveJob as part of a Rails Engine)
@@ -0,0 +1,17 @@
1
+ module Operations
2
+ module Participant
3
+ extend ActiveSupport::Concern
4
+
5
+ included do
6
+ has_many :operations_task_participants, class_name: "Operations::TaskParticipant", as: :participant, dependent: :destroy
7
+ has_many :operations_tasks, class_name: "Operations::Task", through: :operations_task_participants, source: :task
8
+
9
+ scope :involved_in_operation_as, ->(role:, context: "data") do
10
+ joins(:operations_task_participants).tap do |scope|
11
+ scope.where(operations_task_participants: {role: role}) if role
12
+ scope.where(operations_task_participants: {context: context}) if context
13
+ end
14
+ end
15
+ end
16
+ end
17
+ end
@@ -6,9 +6,8 @@ class Operations::Task::StateManagement::WaitHandler
6
6
  instance_eval(&config)
7
7
  end
8
8
 
9
- def condition(destination = nil, options = {}, &condition)
9
+ def condition(options = {}, &condition)
10
10
  @conditions << condition
11
- @destinations << destination if destination
12
11
  @condition_labels ||= {}
13
12
  condition_index = @conditions.size - 1
14
13
  @condition_labels[condition_index] = options[:label] if options[:label]
@@ -16,18 +15,12 @@ class Operations::Task::StateManagement::WaitHandler
16
15
 
17
16
  def go_to(state) = @destinations << state
18
17
 
19
- def condition_labels
20
- @condition_labels ||= {}
21
- end
18
+ def condition_labels = @condition_labels ||= {}
22
19
 
23
20
  def call(task, data)
24
21
  raise Operations::CannotWaitInForeground.new("#{task.class} cannot wait in the foreground", task) unless task.background?
25
22
  condition = @conditions.find { |condition| data.instance_eval(&condition) }
26
- if condition.nil?
27
- task.go_to(task.state, data.to_h)
28
- else
29
- index = @conditions.index condition
30
- task.go_to(@destinations[index], data.to_h)
31
- end
23
+ next_state = (condition.nil? || @conditions.index(condition).nil?) ? task.state : @destinations[@conditions.index(condition)]
24
+ data.go_to next_state
32
25
  end
33
26
  end
@@ -2,9 +2,9 @@ module Operations::Task::Testing
2
2
  extend ActiveSupport::Concern
3
3
 
4
4
  class_methods do
5
- def handling state, **data, &block
5
+ def handling state, background: false, **data, &block
6
6
  # Create a task specifically for testing - avoid serialization issues
7
- task = new(state: state)
7
+ task = new(state: state, background: background)
8
8
  # Use our own test-specific data carrier so we can examine results
9
9
  data = TestResultCarrier.new(data.merge(task: task))
10
10
 
@@ -22,7 +22,7 @@ module Operations::Task::Testing
22
22
 
23
23
  # Instead of extending DataCarrier (which no longer has go_to),
24
24
  # create a new class with similar functionality but keeps the go_to method for testing
25
- class TestResultCarrier < OpenStruct
25
+ class TestResultCarrier < Operations::Task::DataCarrier
26
26
  def go_to(state, message = nil)
27
27
  self.next_state = state
28
28
  self.status_message = message || next_state.to_s
@@ -41,10 +41,7 @@ module Operations::Task::Testing
41
41
 
42
42
  def call(sub_task_class, **data, &result_handler)
43
43
  record_sub_task sub_task_class
44
- # Return mock data for testing
45
- result = {answer: 42}
46
- result_handler&.call(result)
47
- result
44
+ super
48
45
  end
49
46
 
50
47
  def start(sub_task_class, **data, &result_handler)
@@ -8,9 +8,13 @@ module Operations
8
8
  extend InputValidation
9
9
 
10
10
  enum :status, in_progress: 0, waiting: 10, completed: 100, failed: -1
11
+
11
12
  serialize :data, coder: Operations::GlobalIDSerialiser, type: Hash, default: {}
12
13
  serialize :results, coder: Operations::GlobalIDSerialiser, type: Hash, default: {}
13
14
 
15
+ has_many :task_participants, class_name: "Operations::TaskParticipant", dependent: :destroy
16
+ after_save :record_participants
17
+
14
18
  def call sub_task_class, **data, &result_handler
15
19
  sub_task = sub_task_class.call(**data)
16
20
  result_handler&.call(sub_task.results)
@@ -61,6 +65,20 @@ module Operations
61
65
 
62
66
  private def carrier_for(data) = data.is_a?(DataCarrier) ? data : DataCarrier.new(data.merge(task: self))
63
67
 
68
+ private def record_participants
69
+ record_participants_in :data, data.select { |key, value| value.is_a? Participant }
70
+ record_participants_in :results, results.select { |key, value| value.is_a? Participant }
71
+ end
72
+
73
+ private def record_participants_in context, participants
74
+ task_participants.where(context: context).where.not(role: participants.keys).delete_all
75
+ participants.each do |role, participant|
76
+ task_participants.where(context: context, role: role).first_or_initialize.tap do |task_participant|
77
+ task_participant.update! participant: participant
78
+ end
79
+ end
80
+ end
81
+
64
82
  def self.build(background:, **data)
65
83
  validate_inputs! data
66
84
  create!(state: initial_state, status: background ? "waiting" : "in_progress", data: data, status_message: "", background: background)
@@ -0,0 +1,12 @@
1
+ module Operations
2
+ class TaskParticipant < ApplicationRecord
3
+ belongs_to :task
4
+ belongs_to :participant, polymorphic: true
5
+
6
+ validates :role, presence: true
7
+ validates :context, presence: true
8
+ validates :task_id, uniqueness: {scope: [:participant_type, :participant_id, :role, :context]}
9
+
10
+ scope :in, ->(context) { where(context: context) }
11
+ end
12
+ end
@@ -0,0 +1,15 @@
1
+ class CreateOperationsTaskParticipants < ActiveRecord::Migration[7.1]
2
+ def change
3
+ create_table :operations_task_participants do |t|
4
+ t.references :task, null: false, foreign_key: {to_table: :operations_tasks}
5
+ t.references :participant, polymorphic: true, null: false
6
+ t.string :role, null: false
7
+ t.string :context, null: false, default: "data"
8
+ t.timestamps
9
+ end
10
+
11
+ add_index :operations_task_participants, [:task_id, :participant_type, :participant_id, :role, :context],
12
+ name: "index_operations_task_participants_on_full_identity",
13
+ unique: true
14
+ end
15
+ end
@@ -1,3 +1,3 @@
1
1
  module Operations
2
- VERSION = "0.4.2"
2
+ VERSION = "0.5.0"
3
3
  end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: standard_procedure_operations
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.4.2
4
+ version: 0.5.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Rahoul Baruah
@@ -35,6 +35,7 @@ files:
35
35
  - Rakefile
36
36
  - app/jobs/operations/application_job.rb
37
37
  - app/jobs/operations/task_runner_job.rb
38
+ - app/models/concerns/operations/participant.rb
38
39
  - app/models/operations/task.rb
39
40
  - app/models/operations/task/background.rb
40
41
  - app/models/operations/task/data_carrier.rb
@@ -47,8 +48,10 @@ files:
47
48
  - app/models/operations/task/state_management/decision_handler.rb
48
49
  - app/models/operations/task/state_management/wait_handler.rb
49
50
  - app/models/operations/task/testing.rb
51
+ - app/models/operations/task_participant.rb
50
52
  - config/routes.rb
51
53
  - db/migrate/20250127160616_create_operations_tasks.rb
54
+ - db/migrate/20250309_create_operations_task_participants.rb
52
55
  - lib/operations.rb
53
56
  - lib/operations/cannot_wait_in_foreground.rb
54
57
  - lib/operations/engine.rb