standard_procedure_operations 0.1.0 → 0.2.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/README.md +54 -16
- data/app/models/operations/task/data_carrier.rb +2 -0
- data/app/models/operations/task/state_management.rb +11 -3
- data/app/models/operations/task/testing.rb +27 -0
- data/app/models/operations/task.rb +8 -2
- data/db/migrate/20250127160616_create_operations_tasks.rb +3 -3
- data/lib/operations/matchers.rb +23 -0
- data/lib/operations/missing_inputs_error.rb +2 -0
- data/lib/operations/version.rb +1 -1
- data/lib/operations.rb +3 -0
- metadata +5 -2
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: d19a26ed4cb26b809efc2e4d551a4895613608348f3524f1764c000aceb6aeca
|
4
|
+
data.tar.gz: 2290a93af8caeacfa00fd941e62e4785366e4efa80d59579201bfe93cd85fa0a
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 898f19eb0c69a359a939f5f3b563645dd36d8d6bc2e77707ecc1099698541c02210cbc351866d255a3a6747fb957fbb5f82146a120e53c1ba7d87cd68b280e61
|
7
|
+
data.tar.gz: 8b44ab81583ce10937a02d9cc9cbe7e6d7c153a83b29661d4cb1055876b39f181de7aa022aa9a3bbb2adf0e97aa618034582512bb72465172908b388243d11b1
|
data/README.md
CHANGED
@@ -44,6 +44,7 @@ Here's how this would be represented using Operations.
|
|
44
44
|
|
45
45
|
```ruby
|
46
46
|
class PrepareDocumentForDownload < Operations::Task
|
47
|
+
inputs :user, :document, :use_filename_scrambler
|
47
48
|
starts_with :authorised?
|
48
49
|
|
49
50
|
decision :authorised? do
|
@@ -78,6 +79,8 @@ end
|
|
78
79
|
|
79
80
|
The five states are represented as three [decision](#decisions) handlers, one [action](#actions) handler and a [result](#results) handler.
|
80
81
|
|
82
|
+
The task also declares that it requires a `user`, `document` and `use_filename_scrambler` parameter to be provided, and also declares its initial state - `authorised?`.
|
83
|
+
|
81
84
|
### Decisions
|
82
85
|
A decision handler evaluates a condition, then changes state depending upon if the result is true or false.
|
83
86
|
|
@@ -194,14 +197,12 @@ Handlers can alternatively be implemented as methods on the task itself. This m
|
|
194
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`.
|
195
198
|
|
196
199
|
### Failures and exceptions
|
197
|
-
|
198
|
-
If any handlers raise an exception, the task will be terminated. It will be marked as `failed?` and the `results` hash will contain `results.exception_message`, `results.exception_class` and `results.exception_backtrace` for the exception's message, class name and backtrace respectively.
|
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.
|
199
201
|
|
200
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`.
|
201
203
|
|
202
204
|
### Task life-cycle and the database
|
203
|
-
|
204
|
-
There is an ActiveRecord migration that creates the `operations_tasks` table. Use `bin/rails app:operations:install:migrations` to copy it to your application.
|
205
|
+
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.
|
205
206
|
|
206
207
|
When you `call` a task, it is written to the database. Then whenever a state transition occurs, the task record is updated.
|
207
208
|
|
@@ -214,33 +215,69 @@ This gives you a number of possibilities:
|
|
214
215
|
However, it also means that your database table could fill up with junk that you're no longer interested in. Therefore you can specify the maximum age of a task and, periodically, clean old tasks away. Every task has a `delete_at` field that, by default, is set to `90.days.from_now`. This can be changed by calling `Operations::Task.delete_after 7.days` (or whatever value you prefer). Then, run a cron job (once per day) that calls `Operations::Task.delete_expired`, removing any tasks whose `deleted_at` date has passed.
|
215
216
|
|
216
217
|
### Status messages
|
217
|
-
|
218
218
|
Documentation coming soon.
|
219
219
|
|
220
220
|
### Child tasks
|
221
|
-
|
222
221
|
Coming soon.
|
223
222
|
|
224
223
|
### Background operations and pauses
|
225
|
-
|
226
224
|
Coming soon.
|
227
225
|
|
228
|
-
##
|
229
|
-
|
226
|
+
## Testing
|
227
|
+
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.
|
230
228
|
|
229
|
+
Instead, you can test each state handler in isolation. As the handlers are state-less, we can simulate calling one by creating a task object and then calling the appropriate handler with the data that it expects. This is done by calling `handling`, which yields a `test` object with outcomes from the handler that we can inspect
|
230
|
+
|
231
|
+
To test if we have moved on to another state (for actions or decisions):
|
231
232
|
```ruby
|
232
|
-
|
233
|
+
MyOperation.handling(:an_action_or_decision, some: "data") do |test|
|
234
|
+
assert_equal test.next_state, "new_state"
|
235
|
+
# or
|
236
|
+
expect(test).to have_moved_to "new_state"
|
237
|
+
end
|
233
238
|
```
|
239
|
+
To test if some data has been set or modified (for actions):
|
240
|
+
```ruby
|
241
|
+
MyOperation.handling(:an_action, existing_data: "some_value") do |test|
|
242
|
+
# has a new data value been added?
|
243
|
+
assert_equal test.new_data, "new_value"
|
244
|
+
# or
|
245
|
+
expect(test.new_data).to eq "new_value"
|
246
|
+
# has an existing data value been modified?
|
247
|
+
assert_equal test.existing_data, "some_other_value"
|
248
|
+
# or
|
249
|
+
expect(test.existing_data).to eq "some_other_value"
|
250
|
+
end
|
251
|
+
```
|
252
|
+
To test the results from a result handler:
|
253
|
+
```ruby
|
254
|
+
MyOperation.handling(:a_result, some: "data") do |test|
|
255
|
+
assert_equal test.outcome, "everything is as expected"
|
256
|
+
# or
|
257
|
+
expect(test.outcome).to eq "everything is as expected"
|
258
|
+
end
|
259
|
+
```
|
260
|
+
To test if a handler has failed:
|
261
|
+
```ruby
|
262
|
+
MyOperation.handling(:a_failure, some: "data") do |test|
|
263
|
+
assert_equal test.failure_message, "oh dear"
|
264
|
+
# or
|
265
|
+
expect(test).to have_failed_with "oh dear"
|
266
|
+
end
|
267
|
+
```
|
268
|
+
If you are using RSpec, you must `require "operations/matchers"` to make the matchers available to your specs.
|
234
269
|
|
235
|
-
|
236
|
-
|
270
|
+
## Installation
|
271
|
+
Step 1: Add the gem to your Rails application's Gemfile:
|
272
|
+
```ruby
|
273
|
+
gem "standard_procedure_operations"
|
274
|
+
```
|
275
|
+
Step 2: Run `bundle install`, then copy and run the migrations to add the tasks table to your database:
|
237
276
|
```sh
|
238
|
-
bin/rails
|
277
|
+
bin/rails operations:install:migrations
|
239
278
|
bin/rails db:migrate
|
240
279
|
```
|
241
|
-
|
242
|
-
Then create your own operations by inheriting from `Operations::Task`.
|
243
|
-
|
280
|
+
Step 3: Create your own operations by inheriting from `Operations::Task` and revel in the stateful flowcharts!
|
244
281
|
```ruby
|
245
282
|
class DailyLife < Operations::Task
|
246
283
|
starts_with :am_i_awake?
|
@@ -256,6 +293,7 @@ class DailyLife < Operations::Task
|
|
256
293
|
def am_i_awake? = (7..23).include?(Time.now.hour)
|
257
294
|
end
|
258
295
|
```
|
296
|
+
Step 4: If you're using RSpec for testing, add `require "operations/matchers" to your "spec/rails_helper.rb" file.
|
259
297
|
|
260
298
|
## License
|
261
299
|
The gem is available as open source under the terms of the [LGPL License](/LICENSE). This may or may not make it suitable for your needs.
|
@@ -7,6 +7,10 @@ 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
|
+
|
10
14
|
def starts_with(value) = @initial_state = value.to_sym
|
11
15
|
|
12
16
|
def initial_state = @initial_state
|
@@ -20,13 +24,17 @@ module Operations::Task::StateManagement
|
|
20
24
|
def state_handlers = @state_handlers ||= {}
|
21
25
|
|
22
26
|
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))
|
23
31
|
end
|
24
32
|
|
25
33
|
private def handler_for(state) = self.class.handler_for(state.to_sym)
|
26
34
|
private def process_current_state(data)
|
27
35
|
handler_for(state).call(self, data)
|
28
36
|
rescue => ex
|
29
|
-
update! status: "failed", results: OpenStruct.new(
|
37
|
+
update! status: "failed", results: OpenStruct.new(failure_message: ex.message, exception_class: ex.class.name, exception_backtrace: ex.backtrace)
|
30
38
|
end
|
31
39
|
private def state_is_valid
|
32
40
|
errors.add :state, :invalid if state.blank? || handler_for(state.to_sym).nil?
|
@@ -61,7 +69,7 @@ module Operations::Task::StateManagement
|
|
61
69
|
def call(task, data)
|
62
70
|
result = @condition.nil? ? task.send(@name, data) : data.instance_exec(&@condition)
|
63
71
|
next_state = result ? @true_state : @false_state
|
64
|
-
next_state.respond_to?(:call) ? data.instance_eval(&next_state) :
|
72
|
+
next_state.respond_to?(:call) ? data.instance_eval(&next_state) : data.go_to(next_state, data)
|
65
73
|
end
|
66
74
|
end
|
67
75
|
|
@@ -74,7 +82,7 @@ module Operations::Task::StateManagement
|
|
74
82
|
def call(task, data)
|
75
83
|
results = OpenStruct.new
|
76
84
|
data.instance_exec(results, &@handler) unless @handler.nil?
|
77
|
-
|
85
|
+
data.complete(results)
|
78
86
|
end
|
79
87
|
end
|
80
88
|
end
|
@@ -0,0 +1,27 @@
|
|
1
|
+
module Operations::Task::Testing
|
2
|
+
extend ActiveSupport::Concern
|
3
|
+
|
4
|
+
class_methods do
|
5
|
+
def handling state, **data, &block
|
6
|
+
task = new state: state
|
7
|
+
data = TestResultCarrier.new(data.merge(task: task))
|
8
|
+
handler_for(state).call(task, data)
|
9
|
+
data.completion_results.nil? ? block.call(data) : block.call(data.completion_results)
|
10
|
+
end
|
11
|
+
end
|
12
|
+
|
13
|
+
class TestResultCarrier < OpenStruct
|
14
|
+
def go_to(state, message = nil)
|
15
|
+
self.next_state = state
|
16
|
+
self.status_message = message || next_state.to_s
|
17
|
+
end
|
18
|
+
|
19
|
+
def fail_with(message)
|
20
|
+
self.failure_message = message
|
21
|
+
end
|
22
|
+
|
23
|
+
def complete(results)
|
24
|
+
self.completion_results = results
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
@@ -2,11 +2,17 @@ module Operations
|
|
2
2
|
class Task < ApplicationRecord
|
3
3
|
include StateManagement
|
4
4
|
include Deletion
|
5
|
+
include Testing
|
5
6
|
enum :status, in_progress: 0, completed: 1, failed: -1
|
6
7
|
composed_of :results, class_name: "OpenStruct", constructor: ->(results) { results.to_h }, converter: ->(hash) { OpenStruct.new(hash) }
|
7
8
|
serialize :results, coder: Operations::GlobalIDSerialiser, type: Hash, default: {}
|
8
9
|
|
9
|
-
def self.call(data = {})
|
10
|
+
def self.call(data = {})
|
11
|
+
raise MissingInputsError, "Missing inputs: #{missing_inputs_from(data).join(", ")}" unless required_inputs_are_present_in?(data)
|
12
|
+
create!(state: initial_state).tap do |task|
|
13
|
+
task.send(:process_current_state, DataCarrier.new(data.merge(task: task)))
|
14
|
+
end
|
15
|
+
end
|
10
16
|
|
11
17
|
def go_to(state, data = {}, message = nil)
|
12
18
|
update!(state: state, status_message: message || state.to_s)
|
@@ -15,6 +21,6 @@ module Operations
|
|
15
21
|
|
16
22
|
def fail_with(message) = update! status: "failed", results: {failure_message: message.to_s}
|
17
23
|
|
18
|
-
|
24
|
+
def complete(results) = update!(status: "completed", results: results)
|
19
25
|
end
|
20
26
|
end
|
@@ -1,12 +1,12 @@
|
|
1
|
-
class CreateOperationsTasks < ActiveRecord::Migration[
|
1
|
+
class CreateOperationsTasks < ActiveRecord::Migration[7.1]
|
2
2
|
def change
|
3
3
|
create_table :operations_tasks do |t|
|
4
4
|
t.string :type
|
5
5
|
t.integer :status, default: 0, null: false
|
6
6
|
t.string :state, null: false
|
7
7
|
t.string :status_message, default: "", null: false
|
8
|
-
t.text :data
|
9
|
-
t.text :results
|
8
|
+
t.text :data
|
9
|
+
t.text :results
|
10
10
|
t.boolean :background, default: false, null: false
|
11
11
|
t.datetime :delete_at, null: false, index: true
|
12
12
|
t.timestamps
|
@@ -0,0 +1,23 @@
|
|
1
|
+
require "rspec/expectations"
|
2
|
+
|
3
|
+
# Has the state of the task moved to the expected new state?
|
4
|
+
#
|
5
|
+
# Example:
|
6
|
+
# expect(test).to have_moved_to "new_state"
|
7
|
+
#
|
8
|
+
RSpec::Matchers.matcher :have_moved_to do |state|
|
9
|
+
match do |test_result|
|
10
|
+
test_result.next_state.to_s == state.to_s
|
11
|
+
end
|
12
|
+
end
|
13
|
+
|
14
|
+
# Has the task failed with a given failure message?
|
15
|
+
#
|
16
|
+
# Example:
|
17
|
+
# expect(test).to have_failed_with "some_error"
|
18
|
+
#
|
19
|
+
RSpec::Matchers.matcher :have_failed_with do |failure_message|
|
20
|
+
match do |test_result|
|
21
|
+
test_result.failure_message.to_s == failure_message.to_s
|
22
|
+
end
|
23
|
+
end
|
data/lib/operations/version.rb
CHANGED
data/lib/operations.rb
CHANGED
metadata
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: standard_procedure_operations
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.2.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- Rahoul Baruah
|
@@ -37,11 +37,14 @@ files:
|
|
37
37
|
- app/models/operations/task/data_carrier.rb
|
38
38
|
- app/models/operations/task/deletion.rb
|
39
39
|
- app/models/operations/task/state_management.rb
|
40
|
+
- app/models/operations/task/testing.rb
|
40
41
|
- config/routes.rb
|
41
42
|
- db/migrate/20250127160616_create_operations_tasks.rb
|
42
43
|
- lib/operations.rb
|
43
44
|
- lib/operations/engine.rb
|
44
45
|
- lib/operations/global_id_serialiser.rb
|
46
|
+
- lib/operations/matchers.rb
|
47
|
+
- lib/operations/missing_inputs_error.rb
|
45
48
|
- lib/operations/version.rb
|
46
49
|
- lib/standard_procedure_operations.rb
|
47
50
|
- lib/tasks/operations_tasks.rake
|
@@ -52,7 +55,7 @@ metadata:
|
|
52
55
|
allowed_push_host: https://rubygems.org
|
53
56
|
homepage_uri: https://theartandscienceofruby.com/
|
54
57
|
source_code_uri: https://github.com/standard-procedure/operations
|
55
|
-
changelog_uri: https://github.com/standard-procedure/operations/
|
58
|
+
changelog_uri: https://github.com/standard-procedure/operations/tags
|
56
59
|
rdoc_options: []
|
57
60
|
require_paths:
|
58
61
|
- lib
|