standard_procedure_operations 0.2.2 → 0.2.4

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: 247457067fee3dc2c60698254bcb9f9d51474762d9414547f525f84e0d76be1b
4
- data.tar.gz: b17839f7c5e0e25891c53551d419611c7cabfc2a235ea850a16c6742a47ca49a
3
+ metadata.gz: 227d7875d414fd181b6f620f403dbdd3d4fb098e86853563c328d871f521cc33
4
+ data.tar.gz: 1512b85b79a129784ecfdea2d2c82c3ea842aca78ce69c4c6b1a1e65662b9ea4
5
5
  SHA512:
6
- metadata.gz: ca73b99af337d62e70cc542e2c63cb401f1f269d349008bc01fea0856a4147a3b5eed1458504e3fbb9f0bf9b1dc5b6aa4ec420c594f7741d22bec1fbd39d963c
7
- data.tar.gz: 315a25627f2c5ce4798f2efa6e26375a44ffcbef30981f1df4f87946373bd3d6513cf51c7b9ce6202fa8b8325278471473d9591b3d9380cdfa626d44c9727fa8
6
+ metadata.gz: e4f1e35fb4ffffbf58cae1e98ead209656542115c9a05cc6b6d7122a8659a63ae82599700fa804650c8539fe38f980ad77521b2ad4744734326813f6fa44aaaf
7
+ data.tar.gz: 53c72faa3640bc85f6283ae4bb8ff5aaac2d462994cff1b4654d84c88642039fb62082dcdf212d7c643edd4d2d84d37f85ea34c0f1259333bb44a0cca1e28b89
data/README.md CHANGED
@@ -48,27 +48,38 @@ class PrepareDocumentForDownload < Operations::Task
48
48
  starts_with :authorised?
49
49
 
50
50
  decision :authorised? do
51
+ inputs :user
52
+
51
53
  if_true :within_download_limits?
52
54
  if_false { fail_with "unauthorised" }
53
55
  end
54
56
 
55
57
  decision :within_download_limits? do
58
+ inputs :user
59
+
56
60
  if_true :use_filename_scrambler?
57
61
  if_false { fail_with "download_limit_reached" }
58
62
  end
59
63
 
60
64
  decision :use_filename_scrambler? do
65
+ inputs :use_filename_scrambler
61
66
  condition { use_filename_scrambler }
67
+
62
68
  if_true :scramble_filename
63
69
  if_false :return_filename
64
70
  end
65
71
 
66
72
  action :scramble_filename do
73
+ inputs :document
74
+
67
75
  self.filename = "#{Faker::Lorem.word}#{File.extname(document.filename.to_s)}"
68
76
  go_to :return_filename
69
77
  end
70
78
 
71
79
  result :return_filename do |results|
80
+ inputs :document
81
+ optional :filename
82
+
72
83
  results.filename = filename || document.filename.to_s
73
84
  end
74
85
 
@@ -89,6 +100,7 @@ It's up to you whether you define the condition as a block, as part of the decis
89
100
  ```ruby
90
101
  decision :is_it_the_weekend? do
91
102
  condition { Date.today.wday.in? [0, 6] }
103
+
92
104
  if_true :have_a_party
93
105
  if_false :go_to_work
94
106
  end
@@ -104,6 +116,7 @@ def is_it_the_weekend?(data)
104
116
  Date.today.wday.in? [0, 6]
105
117
  end
106
118
  ```
119
+
107
120
  A decision can also mark a failure, which will terminate the task.
108
121
  ```ruby
109
122
  decision :authorised? do
@@ -113,6 +126,20 @@ decision :authorised? do
113
126
  end
114
127
  ```
115
128
 
129
+ You can specify the data that is required for a decision handler to run by specifying `inputs` and `optionals`:
130
+ ```ruby
131
+ decision :authorised? do
132
+ inputs :user
133
+ optionals :override
134
+
135
+ condition { override || user.administrator? }
136
+
137
+ if_true :do_some_work
138
+ if_false { fail_with "Unauthorised" }
139
+ end
140
+ ```
141
+ In this case, the task will fail if there is no `user` specified. However, `override` is optional (and in fact the `optional` method is just there to help you document your operations).
142
+
116
143
  ### Actions
117
144
  An action handler does some work, then moves to another state.
118
145
 
@@ -124,7 +151,21 @@ action :have_a_party do
124
151
  go_to :send_invitations
125
152
  end
126
153
  ```
127
- Again, instead of using a block in the action handler, you could provide a method to do the work.
154
+ You can specify the required and optional data for your action handler within the block. `optional` is decorative and to help with your documentation. Ensure you call `inputs` at the start of the block.
155
+
156
+ ```ruby
157
+ action :have_a_party do
158
+ inputs :task
159
+ optional :music
160
+
161
+ self.food = task.buy_some_food_for(number_of_guests)
162
+ self.beer = task.buy_some_beer_for(number_of_guests)
163
+ self.music ||= task.plan_a_party_playlist
164
+ go_to :send_invitations
165
+ end
166
+ ```
167
+
168
+ Again, instead of using a block in the action handler, you could provide a method to do the work. However, you cannot specify `inputs` or `optional` data when using a method.
128
169
 
129
170
  ```ruby
130
171
  action :have_a_party
@@ -143,6 +184,8 @@ Do not forget to call `go_to` from your action handler, otherwise the operation
143
184
  ### Results
144
185
  A result handler marks the end of an operation, optionally returning some results. You need to copy your desired results from your [data](#data-and-results) to your results object. This is so only the information that matters to you is stored in the database (as many operations may have a large set of working data).
145
186
 
187
+ There is no method equivalent to a block handler.
188
+
146
189
  ```ruby
147
190
  action :send_invitations do
148
191
  self.invited_friends = (0..number_of_guests).collect do |i|
@@ -157,7 +200,7 @@ result :ready_to_party do |results|
157
200
  results.invited_friends = invited_friends
158
201
  end
159
202
  ```
160
- After this result handler has executed, the task will then be marked as `completed?`, the task's state will be `ready_to_party` and `results.invited_friends` will contain an array of the people you sent invitations to.
203
+ After this result handler has executed, the task will then be marked as `completed?`, the task's state will be `ready_to_party` and `results[:invited_friends]` will contain an array of the people you sent invitations to.
161
204
 
162
205
  If you don't have any meaningful results, you can omit the block on your result handler.
163
206
  ```ruby
@@ -165,6 +208,24 @@ result :go_to_work
165
208
  ```
166
209
  In this case, the task will be marked as `completed?`, the task's state will be `go_to_work` and `results` will be empty.
167
210
 
211
+ You can also specify the required and optional data for your result handler within the block. `optional` is decorative and to help with your documentation. Ensure you call `inputs` at the start of the block.
212
+ ```ruby
213
+ action :send_invitations do
214
+ inputs :number_of_guests
215
+ self.invited_friends = (0..number_of_guests).collect do |i|
216
+ friend = friends.pop
217
+ FriendsMailer.with(recipient: friend).party_invitation.deliver_later
218
+ friend
219
+ end
220
+ go_to :ready_to_party
221
+ end
222
+
223
+ result :ready_to_party do |results|
224
+ inputs :invited_friends
225
+
226
+ results.invited_friends = invited_friends
227
+ end
228
+
168
229
  ### Calling an operation
169
230
  You would use the earlier [PrepareDocumentForDownload](spec/examples/prepare_document_for_download_spec.rb) operation in a controller like this:
170
231
 
@@ -190,16 +251,22 @@ Each operation carries its own, mutable, data for the duration of the operation.
190
251
 
191
252
  For example, in the [DownloadsController](#calling-an-operation) shown above, the `user`, `document` and `use_filename_scrambler` are set within the data object when the operation is started. But if the `scramble_filename` action is called, it generates a new filename and adds that to the data object as well. Finally the `return_filename` result handler then returns either the scrambled or the original filename to the caller.
192
253
 
193
- Within handlers implemented as blocks, you can read the data directly - for example, `condition { use_filename_scrambler }` from the `use_filename_scrambler?` decision shown earlier. If you want to modify a value, or add a new one, you must use `self` - `self.my_data = "something important"`. This is because the data is carried using a [DataCarrier](/app/models/operations/task/data_carrier.rb) object and `instance_eval` is used within your block handlers. This also means that block handlers must use `task.method` to access methods or data on the task object itself (as you are not actually within the context of the task object itself). The exceptions are the `go_to` and `fail_with` methods which the data carrier forwards to the task.
254
+ Within handlers implemented as blocks, you can read the data directly - for example, `condition { use_filename_scrambler }` from the `use_filename_scrambler?` decision shown earlier. If you want to modify a value, or add a new one, you must use `self` - `self.my_data = "something important"`.
255
+
256
+ This is because the data is carried using a [DataCarrier](/app/models/operations/task/data_carrier.rb) object and `instance_eval` is used within your block handlers.
257
+
258
+ This also means that block handlers must use `task.method` to access methods or data on the task object itself (as you are not actually within the context of the task object itself). The exceptions are the `go_to` and `fail_with` methods which the data carrier forwards to the task.
194
259
 
195
260
  Handlers can alternatively be implemented as methods on the task itself. This means that they are executed within the context of the task and can methods and variables belonging to the task. Each handler method receives a `data` parameter which is the data carrier for that task. Individual items can be accessed as a hash - `data[:my_item]` - or as an attribute - `data.my_item`.
196
261
 
197
- The final `results` data from any `result` handlers is stored, along with the task, in the database, so it can be examined later. It is accessed as an OpenStruct that is encoded into JSON. But any ActiveRecord models are translated using a [GlobalID](https://github.com/rails/globalid) using [ActiveJob::Arguments](https://guides.rubyonrails.org/active_job_basics.html#supported-types-for-arguments). Be aware that if you do store an ActiveRecord model into your `results` and that model is later deleted from the database, your task's `results` will be unavailable, as the `GlobalID::Locator` will fail when it tries to load the record. The data is not lost though - if the deserialisation fails, the routine will return the JSON string as `results.raw_data`.
262
+ The final `results` data from any `result` handlers is stored, along with the task, in the database, so it can be examined later. It is a Hash that is encoded into JSON with any ActiveRecord models translated using a [GlobalID](https://github.com/rails/globalid) (this uses [ActiveJob::Arguments](https://guides.rubyonrails.org/active_job_basics.html#supported-types-for-arguments) so works the same way as passing models to ActiveJob).
263
+
264
+ Be aware that if you do store an ActiveRecord model into your `results` and that model is later deleted from the database, your task's `results` will be unavailable (as `GlobalID::Locator` will fail when it tries to load the record). The data is not lost though - if the deserialisation fails, the routine will return the JSON string as `results.raw_data`.
198
265
 
199
266
  ### Failures and exceptions
200
- 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.
267
+ 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.
201
268
 
202
- You can also stop a task at any point by calling `fail_with message`. This will mark the task as `failed?` and the `reeults` has will contain `results.failure_message`.
269
+ You can also stop a task at any point by calling `fail_with message`. This will mark the task as `failed?` and the `reeults` has will contain `results[:failure_message]`.
203
270
 
204
271
  ### Task life-cycle and the database
205
272
  There is an ActiveRecord migration that creates the `operations_tasks` table. Use `bin/rails operations:install:migrations` to copy it to your application, then run `bin/rails db:migrate` to add the table to your application's database.
@@ -254,9 +321,15 @@ To test the results from a result handler:
254
321
  MyOperation.handling(:a_result, some: "data") do |test|
255
322
  assert_equal test.outcome, "everything is as expected"
256
323
  # or
324
+ assert_equal test[:outcome], "everything is as expected"
325
+ # or
257
326
  expect(test.outcome).to eq "everything is as expected"
327
+ # or
328
+ expect(test[:outcome]).to eq "everything is as expected"
258
329
  end
259
330
  ```
331
+ (Note - although results are stored in the database as a Hash, within your test, the results object is still carried as an OpenStruct, so you can access it using either notation).
332
+
260
333
  To test if a handler has failed:
261
334
  ```ruby
262
335
  MyOperation.handling(:a_failure, some: "data") do |test|
@@ -300,6 +373,7 @@ The gem is available as open source under the terms of the [LGPL License](/LICEN
300
373
 
301
374
  ## Roadmap
302
375
 
376
+ - [x] Specify inputs (required and optional) per-state, not just at the start
303
377
  - [ ] Always raise errors instead of just recording a failure (will be useful when dealing with sub-tasks)
304
378
  - [ ] Simplify calling sub-tasks (and testing the same)
305
379
  - [ ] Split out the state-management definition stuff from the task class (so you can use it without subclassing Operations::Task)
@@ -4,4 +4,11 @@ class Operations::Task::DataCarrier < OpenStruct
4
4
  def fail_with(message) = task.fail_with(message)
5
5
 
6
6
  def complete(results) = task.complete(results)
7
+
8
+ def inputs(*names)
9
+ missing_inputs = (names.map(&:to_sym) - to_h.keys)
10
+ raise ArgumentError.new("Missing inputs: #{missing_inputs.join(", ")}") if missing_inputs.any?
11
+ end
12
+
13
+ def optional(*names) = nil
7
14
  end
@@ -0,0 +1,17 @@
1
+ module Operations::Task::InputValidation
2
+ def inputs(*names) = @required_inputs = names.map(&:to_sym)
3
+
4
+ def optional(*names) = @optional_inputs = names.map(&:to_sym)
5
+
6
+ def optional_inputs = @optional_inputs ||= []
7
+
8
+ def required_inputs = @required_inputs ||= []
9
+
10
+ def required_inputs_are_present_in?(hash) = missing_inputs_from(hash).empty?
11
+
12
+ def missing_inputs_from(hash) = (required_inputs - hash.keys.map(&:to_sym))
13
+
14
+ def validate_inputs! hash
15
+ raise ArgumentError, "Missing inputs: #{missing_inputs_from(hash).join(", ")}" unless required_inputs_are_present_in?(hash)
16
+ end
17
+ end
@@ -0,0 +1,12 @@
1
+ class Operations::Task::StateManagement::ActionHandler
2
+ def initialize name, inputs = [], optional = [], &action
3
+ @name = name.to_sym
4
+ @required_inputs = inputs
5
+ @optional_inputs = optional
6
+ @action = action
7
+ end
8
+
9
+ def call(task, data)
10
+ @action.nil? ? task.send(@name, data) : data.instance_exec(&@action)
11
+ end
12
+ end
@@ -0,0 +1,14 @@
1
+ class Operations::Task::StateManagement::CompletionHandler
2
+ def initialize name, inputs = [], optional = [], &handler
3
+ @name = name.to_sym
4
+ @required_inputs = inputs
5
+ @optional_inputs = optional
6
+ @handler = handler
7
+ end
8
+
9
+ def call(task, data)
10
+ results = OpenStruct.new
11
+ data.instance_exec(results, &@handler) unless @handler.nil?
12
+ data.complete(results)
13
+ end
14
+ end
@@ -0,0 +1,24 @@
1
+ class Operations::Task::StateManagement::DecisionHandler
2
+ include Operations::Task::InputValidation
3
+
4
+ def initialize name, &config
5
+ @name = name.to_sym
6
+ @condition = nil
7
+ @true_state = nil
8
+ @false_state = nil
9
+ instance_eval(&config)
10
+ end
11
+
12
+ def condition(&condition) = @condition = condition
13
+
14
+ def if_true(state = nil, &handler) = @true_state = state || handler
15
+
16
+ def if_false(state = nil, &handler) = @false_state = state || handler
17
+
18
+ def call(task, data)
19
+ validate_inputs! data.to_h
20
+ result = @condition.nil? ? task.send(@name, data) : data.instance_exec(&@condition)
21
+ next_state = result ? @true_state : @false_state
22
+ next_state.respond_to?(:call) ? data.instance_eval(&next_state) : data.go_to(next_state, data)
23
+ end
24
+ end
@@ -7,82 +7,28 @@ module Operations::Task::StateManagement
7
7
  end
8
8
 
9
9
  class_methods do
10
- def inputs(*names) = @required_inputs = names.map(&:to_sym)
11
-
12
- def required_inputs = @required_inputs ||= []
13
-
14
10
  def starts_with(value) = @initial_state = value.to_sym
15
11
 
16
12
  def initial_state = @initial_state
17
13
 
18
14
  def decision(name, &config) = state_handlers[name.to_sym] = DecisionHandler.new(name, &config)
19
15
 
20
- def action(name, &handler) = state_handlers[name.to_sym] = ActionHandler.new(name, &handler)
16
+ def action(name, inputs: [], optional: [], &handler) = state_handlers[name.to_sym] = ActionHandler.new(name, inputs, optional, &handler)
21
17
 
22
- def result(name, &results) = state_handlers[name.to_sym] = CompletionHandler.new(name, &results)
18
+ def result(name, inputs: [], optional: [], &results) = state_handlers[name.to_sym] = CompletionHandler.new(name, inputs, optional, &results)
23
19
 
24
20
  def state_handlers = @state_handlers ||= {}
25
21
 
26
22
  def handler_for(state) = state_handlers[state.to_sym]
27
-
28
- def required_inputs_are_present_in?(data) = missing_inputs_from(data).empty?
29
-
30
- def missing_inputs_from(data) = (required_inputs - data.keys.map(&:to_sym))
31
23
  end
32
24
 
33
25
  private def handler_for(state) = self.class.handler_for(state.to_sym)
34
26
  private def process_current_state(data)
35
27
  handler_for(state).call(self, data)
36
28
  rescue => ex
37
- update! status: "failed", status_message: ex.message.to_s.truncate(240), results: OpenStruct.new(failure_message: ex.message, exception_class: ex.class.name, exception_backtrace: ex.backtrace)
29
+ update! status: "failed", status_message: ex.message.to_s.truncate(240), results: {failure_message: ex.message, exception_class: ex.class.name, exception_backtrace: ex.backtrace}
38
30
  end
39
31
  private def state_is_valid
40
32
  errors.add :state, :invalid if state.blank? || handler_for(state.to_sym).nil?
41
33
  end
42
-
43
- class ActionHandler
44
- def initialize name, &action
45
- @name = name.to_sym
46
- @action = action
47
- end
48
-
49
- def call(task, data)
50
- @action.nil? ? task.send(@name, data) : data.instance_exec(&@action)
51
- end
52
- end
53
-
54
- class DecisionHandler
55
- def initialize name, &config
56
- @name = name.to_sym
57
- @condition = nil
58
- @true_state = nil
59
- @false_state = nil
60
- instance_eval(&config)
61
- end
62
-
63
- def condition(&condition) = @condition = condition
64
-
65
- def if_true(state = nil, &handler) = @true_state = state || handler
66
-
67
- def if_false(state = nil, &handler) = @false_state = state || handler
68
-
69
- def call(task, data)
70
- result = @condition.nil? ? task.send(@name, data) : data.instance_exec(&@condition)
71
- next_state = result ? @true_state : @false_state
72
- next_state.respond_to?(:call) ? data.instance_eval(&next_state) : data.go_to(next_state, data)
73
- end
74
- end
75
-
76
- class CompletionHandler
77
- def initialize name, &handler
78
- @name = name.to_sym
79
- @handler = handler
80
- end
81
-
82
- def call(task, data)
83
- results = OpenStruct.new
84
- data.instance_exec(results, &@handler) unless @handler.nil?
85
- data.complete(results)
86
- end
87
- end
88
34
  end
@@ -3,12 +3,13 @@ module Operations
3
3
  include StateManagement
4
4
  include Deletion
5
5
  include Testing
6
+ extend InputValidation
7
+
6
8
  enum :status, in_progress: 0, completed: 1, failed: -1
7
- composed_of :results, class_name: "OpenStruct", constructor: ->(results) { results.to_h }, converter: ->(hash) { OpenStruct.new(hash) }
8
9
  serialize :results, coder: Operations::GlobalIDSerialiser, type: Hash, default: {}
9
10
 
10
11
  def self.call(data = {})
11
- raise MissingInputsError, "Missing inputs: #{missing_inputs_from(data).join(", ")}" unless required_inputs_are_present_in?(data)
12
+ validate_inputs! data
12
13
  create!(state: initial_state, status_message: "").tap do |task|
13
14
  task.send(:process_current_state, DataCarrier.new(data.merge(task: task)))
14
15
  end
@@ -1,3 +1,3 @@
1
1
  module Operations
2
- VERSION = "0.2.2"
2
+ VERSION = "0.2.4"
3
3
  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.2.2
4
+ version: 0.2.4
5
5
  platform: ruby
6
6
  authors:
7
7
  - Rahoul Baruah
8
8
  bindir: bin
9
9
  cert_chain: []
10
- date: 2025-01-31 00:00:00.000000000 Z
10
+ date: 2025-02-03 00:00:00.000000000 Z
11
11
  dependencies:
12
12
  - !ruby/object:Gem::Dependency
13
13
  name: rails
@@ -36,7 +36,11 @@ files:
36
36
  - app/models/operations/task.rb
37
37
  - app/models/operations/task/data_carrier.rb
38
38
  - app/models/operations/task/deletion.rb
39
+ - app/models/operations/task/input_validation.rb
39
40
  - app/models/operations/task/state_management.rb
41
+ - app/models/operations/task/state_management/action_handler.rb
42
+ - app/models/operations/task/state_management/completion_handler.rb
43
+ - app/models/operations/task/state_management/decision_handler.rb
40
44
  - app/models/operations/task/testing.rb
41
45
  - config/routes.rb
42
46
  - db/migrate/20250127160616_create_operations_tasks.rb