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 +4 -4
- data/README.md +27 -28
- data/app/models/concerns/operations/participant.rb +17 -0
- data/app/models/operations/task.rb +20 -2
- data/app/models/operations/task_participant.rb +12 -0
- data/db/migrate/20250309_create_operations_task_participants.rb +15 -0
- data/lib/operations/version.rb +1 -1
- data/lib/operations.rb +1 -1
- metadata +19 -3
- data/lib/operations/global_id_serialiser.rb +0 -29
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: d9a47b950a0831c0bf0a1be69de7b7f75c80237ec73a75673397a0b8112aa846
|
4
|
+
data.tar.gz: 62e9741fcd4053d0f4139aa956f5d51025d644188b2da460d6b7550873257030
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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
|
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)).
|
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 [
|
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.
|
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
|
-
|
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
|
-
|
12
|
-
serialize :
|
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
|
data/lib/operations/version.rb
CHANGED
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
|
+
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
|
+
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
|