standard_procedure_operations 0.4.3 → 0.5.1

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: 04267db639d0804b8fdc04e3a6cbe07531ff5870948f0ba95d91c4383188c331
4
- data.tar.gz: 02fef265e8c2bae6ba1c5178fccb9fad3b0eca6f1bd87b4bbb0d321a47e55f40
3
+ metadata.gz: d9a47b950a0831c0bf0a1be69de7b7f75c80237ec73a75673397a0b8112aa846
4
+ data.tar.gz: 62e9741fcd4053d0f4139aa956f5d51025d644188b2da460d6b7550873257030
5
5
  SHA512:
6
- metadata.gz: 48a5aaf6c3c5f3b92bd274c62a1578b43c72212452b51ff934cdc3ff9ea8ad1eb8bd597d65db64185620a3cf1cde939bbfd19b78a8f297746d6d3b563b64a430
7
- data.tar.gz: 97dbb727aa71212ba5bc485f83f1f5f68a07f3c40a08e37fe115b45246921c249917a5eee4fc841b6d23291cfa8f99d4e1bc80ae78472e62471e1b671f42f7b5
6
+ metadata.gz: 611b50e5912231a593187ea4dc59c5e270674f65413b1c3bbd34d3e1f40849f817954282750d729bdadf23a1bacb35b202c9fb71b2edaccfa59dddd209746234
7
+ data.tar.gz: 5b4fbf04648dfe9347330385f1dfdfc32eeadfe4adbb89a015a562c79b0a7b7f6e6b0e91b8d65f3d934e7f3af19bbaf8856ea39555cdd1b649f2ae9f2937ab81
data/README.md CHANGED
@@ -48,7 +48,6 @@ class PrepareDocumentForDownload < Operations::Task
48
48
  starts_with :authorised?
49
49
 
50
50
  decision :authorised? do
51
- inputs :user
52
51
  condition { user.can?(:read, data.document) }
53
52
 
54
53
  if_true :within_download_limits?
@@ -56,7 +55,6 @@ class PrepareDocumentForDownload < Operations::Task
56
55
  end
57
56
 
58
57
  decision :within_download_limits? do
59
- inputs :user
60
58
  condition { user.within_download_limits? }
61
59
 
62
60
  if_true :use_filename_scrambler?
@@ -64,7 +62,6 @@ class PrepareDocumentForDownload < Operations::Task
64
62
  end
65
63
 
66
64
  decision :use_filename_scrambler? do
67
- inputs :use_filename_scrambler
68
65
  condition { use_filename_scrambler }
69
66
 
70
67
  if_true :scramble_filename
@@ -72,16 +69,11 @@ class PrepareDocumentForDownload < Operations::Task
72
69
  end
73
70
 
74
71
  action :scramble_filename do
75
- inputs :document
76
-
77
72
  self.filename = "#{Faker::Lorem.word}#{File.extname(document.filename.to_s)}"
78
- end
73
+ end
79
74
  go_to :return_filename
80
75
 
81
76
  result :return_filename do |results|
82
- inputs :document
83
- optional :filename
84
-
85
77
  results.filename = filename || document.filename.to_s
86
78
  end
87
79
  end
@@ -158,7 +150,9 @@ go_to :send_invitations
158
150
  You can also specify the required and optional data for your action handler using parameters or within the block. `optional` is decorative and helps with documentation. When using the block form, ensure you call `inputs` at the start of the block so that the task fails before doing any meaningful work.
159
151
 
160
152
  ```ruby
161
- action :have_a_party, inputs: [:number_of_guests], optional: [:music] do
153
+ action :have_a_party do
154
+ inputs :number_of_guests
155
+ optional :music
162
156
  self.food = task.buy_some_food_for(number_of_guests)
163
157
  self.beer = task.buy_some_beer_for(number_of_guests)
164
158
  self.music ||= task.plan_a_party_playlist
@@ -166,8 +160,6 @@ end
166
160
  go_to :send_invitations
167
161
  ```
168
162
 
169
- Defining state transitions statically with `go_to` ensures that all transitions are known when the operation class is loaded, making the flow easier to understand and analyze. The `go_to` method will automatically associate the transition with the most recently defined action handler.
170
-
171
163
  ### Waiting
172
164
  Wait handlers are very similar to decision handlers but only work within [background tasks](#background-operations-and-pauses).
173
165
 
@@ -280,17 +272,36 @@ task = CombineNames.call first_name: "Alice", last_name: "Aardvark"
280
272
  task.results[:name] # => Alice Aardvark
281
273
  ```
282
274
 
283
- Because handlers are run in the context of the data carrier, you do not have direct access to methods or properties on your task object. However, the data carrier holds a reference to your task; use `task.do_something` or `task.some_attribute` to access it. The exception is the `fail_with`, `call` and `start` methods which the data carrier understands (and are intercepted when you are [testing](#testing)). Note that `go_to` has been removed from the data carrier to enforce static state transitions with the `go_to` method.
275
+ Because handlers are run in the context of the data carrier, you do not have direct access to methods or properties on your task object. However, the data carrier holds a reference to your task; use `task.do_something` or `task.some_attribute` to access it. The exception is the `fail_with`, `call` and `start` methods which the data carrier understands (and are intercepted when you are [testing](#testing)).
284
276
 
285
277
  Both your task's `data` and its final `results` are stored in the database, so they can be examined later. The `results` because that's what you're interested in, the `data` as it can be useful for debugging or auditing purposes.
286
278
 
287
279
  They are both stored as hashes that are encoded into JSON.
288
280
 
289
- Instead of using the standard [JSON coder](https://api.rubyonrails.org/v4.2/classes/ActiveModel/Serializers/JSON.html), we use a [GlobalIDSerialiser](/lib/operations/global_id_serialiser.rb). This uses [ActiveJob::Arguments](https://guides.rubyonrails.org/active_job_basics.html#supported-types-for-arguments) to transform any models into [GlobalIDs](https://github.com/rails/globalid) before storage and convert them back to models upon retrieval.
281
+ Instead of using the standard [JSON coder](https://api.rubyonrails.org/v4.2/classes/ActiveModel/Serializers/JSON.html), we use a [GlobalIdSerialiser](https://github.com/standard-procedure/global_id_serialiser). This serialises most data into standard JSON types, as you would expect, but it also takes any [GlobalID::Identification](https://github.com/rails/globalid) objects (which includes all ActiveRecord models) and converts them to a GlobalID string. Then when the data is deserialised from the database, the GlobalID is converted back into the appropriate model.
290
282
 
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]`.
283
+ If the original database record was deleted between the time the hash was serialised and when it was retrieved, the `GlobalID::Locator` will fail. In this case, the deserialised data will contain a `nil` for the value in question.
292
284
 
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)
285
+ Also note that the GlobalIdSerialiser automatically converts all hash keys into symbols (unlike the standard JSON coder which uses strings).
286
+
287
+ #### Indexing data and results
288
+
289
+ 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.
290
+
291
+ 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.
292
+
293
+ For example, you create your task as:
294
+ ```ruby
295
+ @alice = User.find 123
296
+ @task = DoSomethingImportant.call user: @alice
297
+ ```
298
+ There will not be a `TaskParticipant` record with a `context` of "data", `role` of "user" and `participant` of `@alice`.
299
+
300
+ Likewise, you can see all the tasks that Alice was involved with using:
301
+ ```ruby
302
+ @alice.involved_in_operations_as("user") # => collection of tasks where Alice was a "user" in the "data" collection
303
+ @alice.involved_in_operations_as("user", context: "results") # => collection of tasks where Alice was a "user" in the "results" collection
304
+ ```
294
305
 
295
306
  ### Failures and exceptions
296
307
  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.
@@ -369,15 +380,11 @@ class UserRegistration < Operations::Task
369
380
  starts_with :create_user
370
381
 
371
382
  action :create_user do
372
- inputs :email
373
-
374
383
  self.user = User.create! email: email
375
384
  end
376
385
  go_to :send_verification_email
377
386
 
378
387
  action :send_verification_email do
379
- inputs :user
380
-
381
388
  UserMailer.with(user: user).verification_email.deliver_later
382
389
  end
383
390
  go_to :verified?
@@ -388,8 +395,6 @@ class UserRegistration < Operations::Task
388
395
  end
389
396
 
390
397
  action :notify_administrator do
391
- inputs :user
392
-
393
398
  AdminMailer.with(user: user).verification_completed.deliver_later
394
399
  end
395
400
  end
@@ -407,7 +412,6 @@ class ParallelTasks < Operations::Task
407
412
  starts_with :start_sub_tasks
408
413
 
409
414
  action :start_sub_tasks do
410
- inputs :number_of_sub_tasks
411
415
  self.sub_tasks = (1..number_of_sub_tasks).collect { |i| start LongRunningTask, number: i }
412
416
  end
413
417
  go_to :do_something_else
@@ -581,7 +585,6 @@ end
581
585
 
582
586
  The visualization includes:
583
587
  - Color-coded nodes by state type (decisions, actions, wait states, results)
584
- - Required and optional inputs for each state
585
588
  - Transition conditions between states with custom labels when provided
586
589
  - Special handling for custom transition blocks
587
590
 
@@ -626,12 +629,8 @@ The gem is available as open source under the terms of the [LGPL License](/LICEN
626
629
  - [x] Simplify calling sub-tasks (and testing them)
627
630
  - [ ] Figure out how to stub calling sub-tasks with known results data
628
631
  - [ ] 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
632
  - [x] Make Operations::Task work in the background using ActiveJob
631
633
  - [x] Add pause/resume capabilities (for example, when a task needs to wait for user input)
632
634
  - [x] Add wait for sub-tasks capabilities
633
635
  - [x] Add GraphViz visualization export for task flows
634
- - [ ] Add ActiveModel validations support for task parameters
635
636
  - [ ] 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
@@ -8,8 +8,12 @@ module Operations
8
8
  extend InputValidation
9
9
 
10
10
  enum :status, in_progress: 0, waiting: 10, completed: 100, failed: -1
11
- serialize :data, coder: Operations::GlobalIDSerialiser, type: Hash, default: {}
12
- serialize :results, coder: Operations::GlobalIDSerialiser, type: Hash, default: {}
11
+
12
+ serialize :data, coder: GlobalIdSerialiser, type: Hash, default: {}
13
+ serialize :results, coder: GlobalIdSerialiser, type: Hash, default: {}
14
+
15
+ has_many :task_participants, class_name: "Operations::TaskParticipant", dependent: :destroy
16
+ after_save :record_participants
13
17
 
14
18
  def call sub_task_class, **data, &result_handler
15
19
  sub_task = sub_task_class.call(**data)
@@ -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.3"
2
+ VERSION = "0.5.1"
3
3
  end
data/lib/operations.rb CHANGED
@@ -1,4 +1,5 @@
1
1
  require "ostruct"
2
+ require "global_id_serialiser"
2
3
 
3
4
  module Operations
4
5
  class Error < StandardError
@@ -10,7 +11,6 @@ module Operations
10
11
  end
11
12
  require "operations/version"
12
13
  require "operations/engine"
13
- require "operations/global_id_serialiser"
14
14
  require "operations/failure"
15
15
  require "operations/cannot_wait_in_foreground"
16
16
  require "operations/timeout"
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.4.3
4
+ version: 0.5.1
5
5
  platform: ruby
6
6
  authors:
7
7
  - Rahoul Baruah
8
8
  bindir: bin
9
9
  cert_chain: []
10
- date: 2025-03-10 00:00:00.000000000 Z
10
+ date: 2025-03-11 00:00:00.000000000 Z
11
11
  dependencies:
12
12
  - !ruby/object:Gem::Dependency
13
13
  name: rails
@@ -23,6 +23,20 @@ dependencies:
23
23
  - - ">="
24
24
  - !ruby/object:Gem::Version
25
25
  version: 7.1.3
26
+ - !ruby/object:Gem::Dependency
27
+ name: standard_procedure_global_id_serialiser
28
+ requirement: !ruby/object:Gem::Requirement
29
+ requirements:
30
+ - - ">="
31
+ - !ruby/object:Gem::Version
32
+ version: '0'
33
+ type: :runtime
34
+ prerelease: false
35
+ version_requirements: !ruby/object:Gem::Requirement
36
+ requirements:
37
+ - - ">="
38
+ - !ruby/object:Gem::Version
39
+ version: '0'
26
40
  description: Pipelines and State Machines for composable, trackable business logic
27
41
  email:
28
42
  - rahoulb@echodek.co
@@ -35,6 +49,7 @@ files:
35
49
  - Rakefile
36
50
  - app/jobs/operations/application_job.rb
37
51
  - app/jobs/operations/task_runner_job.rb
52
+ - app/models/concerns/operations/participant.rb
38
53
  - app/models/operations/task.rb
39
54
  - app/models/operations/task/background.rb
40
55
  - app/models/operations/task/data_carrier.rb
@@ -47,14 +62,15 @@ files:
47
62
  - app/models/operations/task/state_management/decision_handler.rb
48
63
  - app/models/operations/task/state_management/wait_handler.rb
49
64
  - app/models/operations/task/testing.rb
65
+ - app/models/operations/task_participant.rb
50
66
  - config/routes.rb
51
67
  - db/migrate/20250127160616_create_operations_tasks.rb
68
+ - db/migrate/20250309_create_operations_task_participants.rb
52
69
  - lib/operations.rb
53
70
  - lib/operations/cannot_wait_in_foreground.rb
54
71
  - lib/operations/engine.rb
55
72
  - lib/operations/exporters/graphviz.rb
56
73
  - lib/operations/failure.rb
57
- - lib/operations/global_id_serialiser.rb
58
74
  - lib/operations/matchers.rb
59
75
  - lib/operations/no_decision.rb
60
76
  - lib/operations/timeout.rb
@@ -1,29 +0,0 @@
1
- module Operations
2
- # Serialise and deserialise data to and from JSON
3
- # Unlike the standard JSON coder, this coder uses the ActiveJob::Arguments serializer.
4
- # This means that if the data contains an ActiveRecord model, it will be serialised as a GlobalID string
5
- #
6
- # Usage:
7
- # class MyModel < ApplicationRecord
8
- # serialize :data, coder: GlobalIDSerialiser, type: Hash, default: {}
9
- # end
10
- # @my_model = MyModel.create! data: {hello: "world", user: User.first}
11
- # puts @my_model[:data] # => {hello: "world", user: #<User id: 1>}
12
- class GlobalIDSerialiser
13
- def self.dump(data) = ActiveSupport::JSON.dump(ActiveJob::Arguments.serialize([data]))
14
-
15
- def self.load(json)
16
- ActiveJob::Arguments.deserialize(ActiveSupport::JSON.decode(json)).first
17
- rescue => ex
18
- _load_without_global_ids(json).merge exception_message: ex.message, exception_class: ex.class.name, raw_data: json.to_s
19
- end
20
-
21
- def self._load_without_global_ids(json)
22
- ActiveSupport::JSON.decode(json).first.tap do |hash|
23
- hash.delete("_aj_symbol_keys")
24
- end.transform_values do |value|
25
- (value.is_a?(Hash) && value.key?("_aj_globalid")) ? value["_aj_globalid"] : value
26
- end.transform_keys(&:to_sym)
27
- end
28
- end
29
- end