standard_procedure_operations 0.2.6 → 0.3.5
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 +170 -35
- data/app/jobs/operations/application_job.rb +9 -0
- data/app/jobs/operations/task_runner_job.rb +11 -0
- data/app/models/operations/task/background.rb +32 -0
- data/app/models/operations/task/data_carrier.rb +3 -1
- data/app/models/operations/task/state_management/action_handler.rb +1 -3
- data/app/models/operations/task/state_management/decision_handler.rb +20 -5
- data/app/models/operations/task/state_management/wait_handler.rb +27 -0
- data/app/models/operations/task/state_management.rb +2 -6
- data/app/models/operations/task/testing.rb +11 -2
- data/app/models/operations/task.rb +46 -9
- data/lib/operations/cannot_wait_in_foreground.rb +2 -0
- data/lib/operations/global_id_serialiser.rb +9 -1
- data/lib/operations/no_decision.rb +2 -0
- data/lib/operations/timeout.rb +2 -0
- data/lib/operations/version.rb +1 -1
- data/lib/operations.rb +3 -0
- metadata +9 -3
- data/app/models/operations/task/sub_tasks.rb +0 -7
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 71a0ca8e718085505be26fcb316ed4a1e2f9338a12ea55901626696fe5e15269
|
4
|
+
data.tar.gz: 3b94981a2c8b8193bc01ee4f414db2c34161ac111c85a59f68eb3c88dd3b7aa6
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 970184f8761aa729ceebf0cc5fa00a0c8c24a8ef9b1afaa3203b990d92c7c992441a51ef07579efcc93f4948f94d4da4b21c5df07df2adea700523cdb290d13d
|
7
|
+
data.tar.gz: 970e42aecaf9a622c259e27eb830a283ff1be70a4545ab01a4a42debe65ccf2cfcf68aa817a3778c2cfee503957cbf0c96de59e0ace254e2d773e1bb41d5c96e
|
data/README.md
CHANGED
@@ -116,6 +116,20 @@ end
|
|
116
116
|
```
|
117
117
|
(In theory the block used in the `fail_with` case can do anything within the [DataCarrier context](#data-and-results) - so you could set internal state or call methods on the containing task - but I've not tried this yet).
|
118
118
|
|
119
|
+
Alternatively, you can evaluate multiple conditions in your decision handler.
|
120
|
+
|
121
|
+
```ruby
|
122
|
+
decision :is_the_weather_good? do
|
123
|
+
condition { weather_forecast.sunny? }
|
124
|
+
go_to :the_beach
|
125
|
+
condition { weather_forecast.rainy? }
|
126
|
+
go_to :grab_an_umbrella
|
127
|
+
condition { weather_forecast.snowing? }
|
128
|
+
go_to :build_a_snowman
|
129
|
+
end
|
130
|
+
```
|
131
|
+
If no conditions are matched then the task fails with a `NoDecision` exception.
|
132
|
+
|
119
133
|
You can specify the data that is required for a decision handler to run by specifying `inputs` and `optionals`:
|
120
134
|
```ruby
|
121
135
|
decision :authorised? do
|
@@ -157,8 +171,24 @@ end
|
|
157
171
|
```
|
158
172
|
Do not forget to call `go_to` from your action handler, otherwise the operation will just stop whilst still being marked as in progress. (TODO: don't let this happen).
|
159
173
|
|
174
|
+
### Waiting
|
175
|
+
Wait handlers are very similar to decision handlers but only work within [background tasks](#background-operations-and-pauses).
|
176
|
+
|
177
|
+
```ruby
|
178
|
+
wait_until :weather_forecast_available? do
|
179
|
+
condition { weather_forecast.sunny? }
|
180
|
+
go_to :the_beach
|
181
|
+
condition { weather_forecast.rainy? }
|
182
|
+
go_to :grab_an_umbrella
|
183
|
+
condition { weather_forecast.snowing? }
|
184
|
+
go_to :build_a_snowman
|
185
|
+
end
|
186
|
+
```
|
187
|
+
|
188
|
+
If no conditions are met, then, unlike a decision handler, the task continues waiting in the same state.
|
189
|
+
|
160
190
|
### Results
|
161
|
-
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 as the results
|
191
|
+
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 as the results.
|
162
192
|
|
163
193
|
```ruby
|
164
194
|
action :send_invitations do
|
@@ -207,22 +237,6 @@ end
|
|
207
237
|
### Calling an operation
|
208
238
|
You would use the earlier [PrepareDocumentForDownload](spec/examples/prepare_document_for_download_spec.rb) operation in a controller like this:
|
209
239
|
|
210
|
-
```ruby
|
211
|
-
class DownloadsController < ApplicationController
|
212
|
-
def show
|
213
|
-
@document = Document.includes(:account).find(params[:id])
|
214
|
-
@task = PrepareDocumentForDownload.call(user: Current.user, document: @document, use_filename_scrambler: @document.account.use_filename_scrambler?)
|
215
|
-
if @task.completed?
|
216
|
-
@filename = @task.results[:filename]
|
217
|
-
send_data @document.contents, filename: @filename, disposition: "attachment"
|
218
|
-
else
|
219
|
-
render action: "error", message: @task.results[:failure_message], status: 401
|
220
|
-
end
|
221
|
-
end
|
222
|
-
end
|
223
|
-
```
|
224
|
-
|
225
|
-
Future dev: I'm going to change this so it raises an exception on failure so it will become:
|
226
240
|
```ruby
|
227
241
|
class DownloadsController < ApplicationController
|
228
242
|
def show
|
@@ -237,12 +251,16 @@ class DownloadsController < ApplicationController
|
|
237
251
|
end
|
238
252
|
```
|
239
253
|
|
240
|
-
OK - so that's a pretty longwinded way of performing a simple task.
|
254
|
+
OK - so that's a pretty longwinded way of performing a simple task.
|
255
|
+
|
256
|
+
But many operations end up as complex flows of conditionals and actions, often spread across multiple classes and objects. This means that someone trying to understand the rules for an operation can spend a lot of time tracing through code, understanding that flow.
|
257
|
+
|
258
|
+
In Collabor8Online, the actual operation for handling downloads has over twenty states, with half of them being decisions (there are a number of feature flags and per-account configuration options which need to be considered). Now they are defined as four ruby classes (using [composition](#sub-tasks) to split the workload) and the logic is extremely easy to follow.
|
241
259
|
|
242
260
|
### Data and results
|
243
261
|
Each operation carries its own, mutable, [data](/app/models/operations/task/data_carrier.rb) for the duration of the operation.
|
244
262
|
|
245
|
-
This is provided when you `call` the operation to start it and is passed through to each decision, action and result.
|
263
|
+
This is provided when you `call` the operation to start it and is passed through to each decision, action and result. If you modify the data then that modification is passed on to the next handler.
|
246
264
|
|
247
265
|
Within handlers you can read the data directly (the implementation uses `instance_eval`/`instance_exec`). Here the `build_name` action knows the `first_name` and `last_name` provided and adds in a new property of `name`.
|
248
266
|
|
@@ -265,18 +283,22 @@ task = CombineNames.call first_name: "Alice", last_name: "Aardvark"
|
|
265
283
|
task.results[:name] # => Alice Aardvark
|
266
284
|
```
|
267
285
|
|
268
|
-
Because handlers are run in the context of the data carrier,
|
286
|
+
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 exceptions are the `go_to`, `fail_with`, `call` and `start` methods which the data carrier understands (and are intercepted when you are [testing](#testing)).
|
287
|
+
|
288
|
+
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.
|
289
|
+
|
290
|
+
They are both stored as hashes that are encoded into JSON.
|
269
291
|
|
270
|
-
|
292
|
+
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.
|
271
293
|
|
272
|
-
|
294
|
+
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]`.
|
295
|
+
|
296
|
+
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)
|
273
297
|
|
274
298
|
### Failures and exceptions
|
275
299
|
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.
|
276
300
|
|
277
|
-
You can also stop a task at any point by calling `fail_with message`. This will
|
278
|
-
|
279
|
-
(Future dev - going to change this so it raises an exception (or passes it on to the caller) so you don't need to test for `failed?` every time - this will simplify managing sub-tasks).
|
301
|
+
You can also stop a task at any point by calling `fail_with message`. This will raise an `Operations::Failure` exception, marking the task as `failed?` and the `results` has will contain `results[:failure_message]`.
|
280
302
|
|
281
303
|
### Task life-cycle and the database
|
282
304
|
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.
|
@@ -332,8 +354,122 @@ class PrepareDownload < Operations::Task
|
|
332
354
|
end
|
333
355
|
```
|
334
356
|
|
357
|
+
If you want the sub-task to be a [background task](#background-operations-and-pauses), use `start` instead of `call`. This will return the newly created sub-task immediately. As the sub-task will not have completed, you cannot access its results unless you [wait](#waiting-for-sub-tasks-to-complete) (which is only possible if the parent task is a background task as well).
|
358
|
+
|
335
359
|
### Background operations and pauses
|
336
|
-
|
360
|
+
If you have ActiveJob configured, you can run your operations in the background.
|
361
|
+
|
362
|
+
Instead of using `call`, use `start` to initiate the operation. This takes the same data parameters and returns a task object that you can refer back to. But it will be `waiting?` instead of `in_progress?` or `completed?`. An `Operations::TaskRunnerJob` will be queued and it will mark the task as `in_progress?`, then call a _single_ state handler. Instead of handling the next state immediately another `Operations::TaskRunnerJob` is queued. And so on, until the task either fails or is completed.
|
363
|
+
|
364
|
+
By itself, this is not particularly useful - it just makes your operation take even longer to complete.
|
365
|
+
|
366
|
+
But if your operation may need to wait for something else to happen, background tasks are perfect.
|
367
|
+
|
368
|
+
#### Waiting for user input
|
369
|
+
For example, maybe you're handling a user registration process and you need to wait until the verification link has been clicked. The verification link goes to another controller and updates the user record in question. Once clicked, you can then notify the administrator.
|
370
|
+
|
371
|
+
```ruby
|
372
|
+
class UserRegistration < Operations::Task
|
373
|
+
inputs :email
|
374
|
+
starts_with :create_user
|
375
|
+
|
376
|
+
action :create_user do
|
377
|
+
inputs :email
|
378
|
+
|
379
|
+
self.user = User.create! email: email
|
380
|
+
go_to :send_verification_email
|
381
|
+
end
|
382
|
+
|
383
|
+
action :send_verification_email do
|
384
|
+
inputs :user
|
385
|
+
|
386
|
+
UserMailer.with(user: user).verification_email.deliver_later
|
387
|
+
go_to :verified?
|
388
|
+
end
|
389
|
+
|
390
|
+
wait_until :verified? do
|
391
|
+
condition { user.verified? }
|
392
|
+
go_to :notify_administrator
|
393
|
+
end
|
394
|
+
|
395
|
+
action :notify_administrator do
|
396
|
+
inputs :user
|
397
|
+
|
398
|
+
AdminMailer.with(user: user).verification_completed.deliver_later
|
399
|
+
end
|
400
|
+
end
|
401
|
+
|
402
|
+
@task = UserRegistration.start email: "someone@example.com"
|
403
|
+
```
|
404
|
+
Because background tasks use ActiveJobs, every time the `verified?` condition is evaluated, the task object (and hence its data) will be reloaded from the database. So the `user.verified?` property will be refreshed on each evaluation.
|
405
|
+
|
406
|
+
#### Waiting for sub-tasks to complete
|
407
|
+
Alternatively, you may have a number of sub-tasks that you want to run in parallel then continue once they have all completed. This allows you to spread their execution across multiple processes or even servers (depending upon how your job queue processes are configured).
|
408
|
+
|
409
|
+
```ruby
|
410
|
+
class ParallelTasks < Operations::Task
|
411
|
+
inputs :number_of_sub_tasks
|
412
|
+
starts_with :start_sub_tasks
|
413
|
+
|
414
|
+
action :start_sub_tasks do
|
415
|
+
inputs :number_of_sub_tasks
|
416
|
+
self.sub_tasks = (1..number_of_sub_tasks).collect { |i| start LongRunningTask, number: i }
|
417
|
+
go_to :do_something_else
|
418
|
+
end
|
419
|
+
|
420
|
+
action :do_something_else do
|
421
|
+
# do something else while the sub-tasks do their thing
|
422
|
+
go_to :sub_tasks_completed?
|
423
|
+
end
|
424
|
+
|
425
|
+
wait_until :sub_tasks_completed? do
|
426
|
+
condition { sub_tasks.all? { |t| t.completed? } }
|
427
|
+
go_to :done
|
428
|
+
end
|
429
|
+
|
430
|
+
result :done
|
431
|
+
end
|
432
|
+
|
433
|
+
@task = ParallelTasks.start number_of_sub_tasks: 5
|
434
|
+
```
|
435
|
+
The principle is the same as above; we store the newly created sub-tasks in our own `data` property. As they are ActiveRecord models, they get reloaded each time `sub_tasks_completed?` is evaluated - and we check to see that they have all completed before moving on.
|
436
|
+
|
437
|
+
#### Delays and Timeouts
|
438
|
+
When you run an operation in the background, it schedules an [ActiveJob](app/jobs/operations/task_runner_job.rb) which performs the individual state handler, scheduling a follow-up job for subsequent states. By default, these jobs are scheduled to run after 1 second - so a five state operation will take a minimum of five seconds to complete (depending upon the backlog in your job queues). This is to prevent a single operation from starving the job process of resources and allow other ActiveJobs the time to execute.
|
439
|
+
|
440
|
+
However, if you know that your `wait_until` condition may take a while you can change the default delay to something longer. In your operations definition, declare the required delay:
|
441
|
+
|
442
|
+
```ruby
|
443
|
+
class ParallelTasks < Operations::Tasks
|
444
|
+
delay 1.minute
|
445
|
+
...
|
446
|
+
end
|
447
|
+
```
|
448
|
+
|
449
|
+
Likewise, it's possible for a background task to get stuck. In the sub-tasks example above, if one of the sub-tasks fails, waiting for them _all_ to complete will never happen. Every operation has a default timeout of 5 minutes - if the operation has not completed or failed 5 minutes after it was initially started, it will fail with an `Operations::Timeout` exception.
|
450
|
+
|
451
|
+
If you need to change this (such as the user verification example above), you can declare the timeout when defining the operation. Long timeouts fit well with longer delays, so you're not filling the job queues with jobs that are meaninglessly evaluating your conditions.
|
452
|
+
|
453
|
+
```ruby
|
454
|
+
class UserRegistration < Operations::Task
|
455
|
+
timeout 24.hours
|
456
|
+
delay 15.minutes
|
457
|
+
...
|
458
|
+
end
|
459
|
+
```
|
460
|
+
|
461
|
+
Instead of failing with an `Operations::Timeout` exception, you define an `on_timeout` handler for any special processing should the time-out occur.
|
462
|
+
|
463
|
+
```ruby
|
464
|
+
class WaitForSomething < Operations::Task
|
465
|
+
timeout 10.minutes
|
466
|
+
delay 1.minute
|
467
|
+
|
468
|
+
on_timeout do
|
469
|
+
Notifier.send_timeout_notification
|
470
|
+
end
|
471
|
+
end>
|
472
|
+
```
|
337
473
|
|
338
474
|
## Testing
|
339
475
|
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.
|
@@ -407,13 +543,9 @@ end
|
|
407
543
|
### Testing failures
|
408
544
|
To test if a handler has failed:
|
409
545
|
```ruby
|
410
|
-
|
411
|
-
|
412
|
-
# or
|
413
|
-
expect(test).to have_failed_with "oh dear"
|
414
|
-
end
|
546
|
+
|
547
|
+
expect { MyOperation.handling(:a_failure, some: "data") }.to raise_error(SomeException)
|
415
548
|
```
|
416
|
-
(Future dev: exceptions will not be silently caught so this will probably become `expect { MyOperation.handling(:a_failure, some: "data") }.to raise_exception(Whatever)).
|
417
549
|
|
418
550
|
If you are using RSpec, you must `require "operations/matchers"` to make the matchers available to your specs.
|
419
551
|
|
@@ -457,7 +589,10 @@ The gem is available as open source under the terms of the [LGPL License](/LICEN
|
|
457
589
|
- [ ] Figure out how to stub calling sub-tasks with known results data
|
458
590
|
- [ ] Figure out how to test the parameters passed to sub-tasks when they are called
|
459
591
|
- [ ] Split out the state-management definition stuff from the task class (so you can use it without subclassing Operations::Task)
|
460
|
-
- [
|
461
|
-
- [
|
462
|
-
- [
|
592
|
+
- [x] Make Operations::Task work in the background using ActiveJob
|
593
|
+
- [x] Add pause/resume capabilities (for example, when a task needs to wait for user input)
|
594
|
+
- [x] Add wait for sub-tasks capabilities
|
595
|
+
- [ ] Add ActiveModel validations support for task parameters
|
596
|
+
- [ ] Option to change background job queue and priority settings
|
597
|
+
- [ ] 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)
|
463
598
|
- [ ] 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,9 @@
|
|
1
|
+
module Operations
|
2
|
+
class ApplicationJob < ActiveJob::Base
|
3
|
+
# Automatically retry jobs that encountered a deadlock
|
4
|
+
# retry_on ActiveRecord::Deadlocked
|
5
|
+
|
6
|
+
# Most jobs are safe to ignore if the underlying records are no longer available
|
7
|
+
# discard_on ActiveJob::DeserializationError
|
8
|
+
end
|
9
|
+
end
|
@@ -0,0 +1,32 @@
|
|
1
|
+
module Operations::Task::Background
|
2
|
+
extend ActiveSupport::Concern
|
3
|
+
|
4
|
+
class_methods do
|
5
|
+
def delay(value)
|
6
|
+
@background_delay = value
|
7
|
+
end
|
8
|
+
|
9
|
+
def timeout(value)
|
10
|
+
@execution_timeout = value
|
11
|
+
end
|
12
|
+
|
13
|
+
def on_timeout(&handler) = @on_timeout = handler
|
14
|
+
|
15
|
+
def background_delay = @background_delay ||= 1.second
|
16
|
+
|
17
|
+
def execution_timeout = @execution_timeout ||= 5.minutes
|
18
|
+
|
19
|
+
def timeout_handler = @on_timeout
|
20
|
+
|
21
|
+
def with_timeout(data) = data.merge(_execution_timeout: execution_timeout.from_now.utc)
|
22
|
+
end
|
23
|
+
|
24
|
+
private def background_delay = self.class.background_delay
|
25
|
+
private def execution_timeout = self.class.execution_timeout
|
26
|
+
private def timeout_handler = self.class.timeout_handler
|
27
|
+
private def timeout!
|
28
|
+
return unless timeout_expired?
|
29
|
+
timeout_handler.nil? ? raise(Operations::Timeout.new("Timeout expired", self)) : timeout_handler.call
|
30
|
+
end
|
31
|
+
private def timeout_expired? = data[:_execution_timeout].present? && data[:_execution_timeout] < Time.now.utc
|
32
|
+
end
|
@@ -1,10 +1,12 @@
|
|
1
1
|
class Operations::Task::DataCarrier < OpenStruct
|
2
|
-
def go_to(state, message
|
2
|
+
def go_to(state, message: nil) = task.go_to(state, self, message: message)
|
3
3
|
|
4
4
|
def fail_with(message) = task.fail_with(message)
|
5
5
|
|
6
6
|
def call(sub_task_class, **data, &result_handler) = task.call(sub_task_class, **data, &result_handler)
|
7
7
|
|
8
|
+
def start(sub_task_class, **data, &result_handler) = task.start(sub_task_class, **data, &result_handler)
|
9
|
+
|
8
10
|
def complete(results) = task.complete(results)
|
9
11
|
|
10
12
|
def inputs(*names)
|
@@ -3,13 +3,16 @@ class Operations::Task::StateManagement::DecisionHandler
|
|
3
3
|
|
4
4
|
def initialize name, &config
|
5
5
|
@name = name.to_sym
|
6
|
-
@
|
6
|
+
@conditions = []
|
7
|
+
@destinations = []
|
7
8
|
@true_state = nil
|
8
9
|
@false_state = nil
|
9
10
|
instance_eval(&config)
|
10
11
|
end
|
11
12
|
|
12
|
-
def condition(&condition) = @
|
13
|
+
def condition(&condition) = @conditions << condition
|
14
|
+
|
15
|
+
def go_to(destination) = @destinations << destination
|
13
16
|
|
14
17
|
def if_true(state = nil, &handler) = @true_state = state || handler
|
15
18
|
|
@@ -17,8 +20,20 @@ class Operations::Task::StateManagement::DecisionHandler
|
|
17
20
|
|
18
21
|
def call(task, data)
|
19
22
|
validate_inputs! data.to_h
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
+
has_true_false_handlers? ? handle_single_condition(task, data) : handle_multiple_conditions(task, data)
|
24
|
+
end
|
25
|
+
|
26
|
+
private def has_true_false_handlers? = !@true_state.nil? || !@false_state.nil?
|
27
|
+
|
28
|
+
private def handle_single_condition(task, data)
|
29
|
+
next_state = data.instance_eval(&@conditions.first) ? @true_state : @false_state
|
30
|
+
next_state.respond_to?(:call) ? data.instance_eval(&next_state) : data.go_to(next_state)
|
31
|
+
end
|
32
|
+
|
33
|
+
private def handle_multiple_conditions(task, data)
|
34
|
+
condition = @conditions.find { |condition| data.instance_eval(&condition) }
|
35
|
+
raise Operations::NoDecision.new("No conditions matched #{@name}") if condition.nil?
|
36
|
+
index = @conditions.index condition
|
37
|
+
data.go_to @destinations[index]
|
23
38
|
end
|
24
39
|
end
|
@@ -0,0 +1,27 @@
|
|
1
|
+
class Operations::Task::StateManagement::WaitHandler
|
2
|
+
def initialize name, &config
|
3
|
+
@name = name.to_sym
|
4
|
+
@conditions = []
|
5
|
+
@destinations = []
|
6
|
+
instance_eval(&config)
|
7
|
+
puts "Configured"
|
8
|
+
end
|
9
|
+
|
10
|
+
def condition(&condition) = @conditions << condition
|
11
|
+
|
12
|
+
def go_to(state) = @destinations << state
|
13
|
+
|
14
|
+
def call(task, data)
|
15
|
+
raise Operations::CannotWaitInForeground.new("#{task.class} cannot wait in the foreground", task) unless task.background?
|
16
|
+
puts "Searching"
|
17
|
+
condition = @conditions.find { |condition| data.instance_eval(&condition) }
|
18
|
+
if condition.nil?
|
19
|
+
puts "None"
|
20
|
+
data.go_to task.state
|
21
|
+
else
|
22
|
+
index = @conditions.index condition
|
23
|
+
puts "Found #{@destinations[index]}"
|
24
|
+
data.go_to @destinations[index]
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
@@ -15,6 +15,8 @@ module Operations::Task::StateManagement
|
|
15
15
|
|
16
16
|
def action(name, inputs: [], optional: [], &handler) = state_handlers[name.to_sym] = ActionHandler.new(name, inputs, optional, &handler)
|
17
17
|
|
18
|
+
def wait_until(name, &config) = state_handlers[name.to_sym] = WaitHandler.new(name, &config)
|
19
|
+
|
18
20
|
def result(name, inputs: [], optional: [], &results) = state_handlers[name.to_sym] = CompletionHandler.new(name, inputs, optional, &results)
|
19
21
|
|
20
22
|
def state_handlers = @state_handlers ||= {}
|
@@ -23,12 +25,6 @@ module Operations::Task::StateManagement
|
|
23
25
|
end
|
24
26
|
|
25
27
|
private def handler_for(state) = self.class.handler_for(state.to_sym)
|
26
|
-
private def process_current_state(data)
|
27
|
-
handler_for(state).call(self, data)
|
28
|
-
rescue => ex
|
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}
|
30
|
-
raise ex
|
31
|
-
end
|
32
28
|
private def state_is_valid
|
33
29
|
errors.add :state, :invalid if state.blank? || handler_for(state.to_sym).nil?
|
34
30
|
end
|
@@ -21,13 +21,22 @@ module Operations::Task::Testing
|
|
21
21
|
end
|
22
22
|
|
23
23
|
def call(sub_task_class, **data, &result_handler)
|
24
|
-
|
25
|
-
|
24
|
+
record_sub_task sub_task_class
|
25
|
+
super
|
26
|
+
end
|
27
|
+
|
28
|
+
def start(sub_task_class, **data, &result_handler)
|
29
|
+
record_sub_task sub_task_class
|
26
30
|
super
|
27
31
|
end
|
28
32
|
|
29
33
|
def complete(results)
|
30
34
|
self.completion_results = results
|
31
35
|
end
|
36
|
+
|
37
|
+
private def record_sub_task sub_task_class
|
38
|
+
self.sub_tasks ||= []
|
39
|
+
self.sub_tasks << sub_task_class
|
40
|
+
end
|
32
41
|
end
|
33
42
|
end
|
@@ -1,24 +1,54 @@
|
|
1
1
|
module Operations
|
2
2
|
class Task < ApplicationRecord
|
3
3
|
include StateManagement
|
4
|
-
include SubTasks
|
5
4
|
include Deletion
|
6
5
|
include Testing
|
6
|
+
include Background
|
7
7
|
extend InputValidation
|
8
8
|
|
9
|
-
enum :status, in_progress: 0, completed:
|
9
|
+
enum :status, in_progress: 0, waiting: 10, completed: 100, failed: -1
|
10
|
+
serialize :data, coder: Operations::GlobalIDSerialiser, type: Hash, default: {}
|
10
11
|
serialize :results, coder: Operations::GlobalIDSerialiser, type: Hash, default: {}
|
11
12
|
|
12
|
-
def
|
13
|
-
|
14
|
-
|
15
|
-
|
13
|
+
def call sub_task_class, **data, &result_handler
|
14
|
+
sub_task = sub_task_class.call(**data)
|
15
|
+
result_handler&.call(sub_task.results)
|
16
|
+
sub_task.results
|
17
|
+
end
|
18
|
+
|
19
|
+
def start sub_task_class, **data, &result_handler
|
20
|
+
sub_task_class.start(**data)
|
21
|
+
end
|
22
|
+
|
23
|
+
def perform
|
24
|
+
timeout!
|
25
|
+
in_progress!
|
26
|
+
handler_for(state).call(self, carrier_for(data))
|
27
|
+
rescue => ex
|
28
|
+
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}
|
29
|
+
raise ex
|
30
|
+
end
|
31
|
+
|
32
|
+
def perform_later
|
33
|
+
waiting!
|
34
|
+
TaskRunnerJob.set(wait_until: background_delay.from_now).perform_later self
|
35
|
+
end
|
36
|
+
|
37
|
+
def self.call(**)
|
38
|
+
build(background: false, **).tap do |task|
|
39
|
+
task.perform
|
40
|
+
end
|
41
|
+
end
|
42
|
+
|
43
|
+
def self.start(**data)
|
44
|
+
build(background: true, **with_timeout(data)).tap do |task|
|
45
|
+
task.perform_later
|
16
46
|
end
|
17
47
|
end
|
18
48
|
|
19
|
-
def go_to(state, data = {}, message
|
20
|
-
update!(state: state, status_message: (message || state).to_s.truncate(240))
|
21
|
-
|
49
|
+
def go_to(state, data = {}, message: nil)
|
50
|
+
update!(state: state, data: data.to_h, status_message: (message || state).to_s.truncate(240))
|
51
|
+
background? ? perform_later : perform
|
22
52
|
end
|
23
53
|
|
24
54
|
def fail_with(message)
|
@@ -27,5 +57,12 @@ module Operations
|
|
27
57
|
end
|
28
58
|
|
29
59
|
def complete(results) = update!(status: "completed", status_message: "completed", results: results.to_h)
|
60
|
+
|
61
|
+
private def carrier_for(data) = data.is_a?(DataCarrier) ? data : DataCarrier.new(data.merge(task: self))
|
62
|
+
|
63
|
+
def self.build(background:, **data)
|
64
|
+
validate_inputs! data
|
65
|
+
create!(state: initial_state, status: background ? "waiting" : "in_progress", data: data, status_message: "", background: background)
|
66
|
+
end
|
30
67
|
end
|
31
68
|
end
|
@@ -15,7 +15,15 @@ module Operations
|
|
15
15
|
def self.load(json)
|
16
16
|
ActiveJob::Arguments.deserialize(ActiveSupport::JSON.decode(json)).first
|
17
17
|
rescue => ex
|
18
|
-
|
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)
|
19
27
|
end
|
20
28
|
end
|
21
29
|
end
|
data/lib/operations/version.rb
CHANGED
data/lib/operations.rb
CHANGED
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.3.5
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Rahoul Baruah
|
8
8
|
bindir: bin
|
9
9
|
cert_chain: []
|
10
|
-
date: 2025-
|
10
|
+
date: 2025-03-04 00:00:00.000000000 Z
|
11
11
|
dependencies:
|
12
12
|
- !ruby/object:Gem::Dependency
|
13
13
|
name: rails
|
@@ -33,7 +33,10 @@ files:
|
|
33
33
|
- LICENSE
|
34
34
|
- README.md
|
35
35
|
- Rakefile
|
36
|
+
- app/jobs/operations/application_job.rb
|
37
|
+
- app/jobs/operations/task_runner_job.rb
|
36
38
|
- app/models/operations/task.rb
|
39
|
+
- app/models/operations/task/background.rb
|
37
40
|
- app/models/operations/task/data_carrier.rb
|
38
41
|
- app/models/operations/task/deletion.rb
|
39
42
|
- app/models/operations/task/input_validation.rb
|
@@ -41,15 +44,18 @@ files:
|
|
41
44
|
- app/models/operations/task/state_management/action_handler.rb
|
42
45
|
- app/models/operations/task/state_management/completion_handler.rb
|
43
46
|
- app/models/operations/task/state_management/decision_handler.rb
|
44
|
-
- app/models/operations/task/
|
47
|
+
- app/models/operations/task/state_management/wait_handler.rb
|
45
48
|
- app/models/operations/task/testing.rb
|
46
49
|
- config/routes.rb
|
47
50
|
- db/migrate/20250127160616_create_operations_tasks.rb
|
48
51
|
- lib/operations.rb
|
52
|
+
- lib/operations/cannot_wait_in_foreground.rb
|
49
53
|
- lib/operations/engine.rb
|
50
54
|
- lib/operations/failure.rb
|
51
55
|
- lib/operations/global_id_serialiser.rb
|
52
56
|
- lib/operations/matchers.rb
|
57
|
+
- lib/operations/no_decision.rb
|
58
|
+
- lib/operations/timeout.rb
|
53
59
|
- lib/operations/version.rb
|
54
60
|
- lib/standard_procedure_operations.rb
|
55
61
|
- lib/tasks/operations_tasks.rake
|