standard_procedure_operations 0.2.1 → 0.2.3
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 +26 -5
- data/app/models/operations/task/state_management.rb +1 -1
- data/app/models/operations/task.rb +1 -2
- data/lib/operations/version.rb +1 -1
- metadata +1 -1
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 27f6593a2e233607d6247450ca043995e23d33dcb4af69129c314d4e89d2dab1
|
4
|
+
data.tar.gz: 6ba1683666c67b27de61f3d99ed61a4f2badb962622f06e7174cb71a09ccee93
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: b40b93fd31799d31f53294a234733c492520731332aeeea6bc6e1fa747fa19c550a5d407a4573f0fd0161404ba4e63a046454ce28b423ddb0b384700139812cf
|
7
|
+
data.tar.gz: 0d4e28d7e9a5d4d77343d8bcb56b0deff90f76c79d1da0bb980e8134354302e68bdf745d71f97e0d86f43ade3c419ed8486db444e47a92ba03597e33605b4f78
|
data/README.md
CHANGED
@@ -157,7 +157,7 @@ result :ready_to_party do |results|
|
|
157
157
|
results.invited_friends = invited_friends
|
158
158
|
end
|
159
159
|
```
|
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
|
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.
|
161
161
|
|
162
162
|
If you don't have any meaningful results, you can omit the block on your result handler.
|
163
163
|
```ruby
|
@@ -190,16 +190,22 @@ Each operation carries its own, mutable, data for the duration of the operation.
|
|
190
190
|
|
191
191
|
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
192
|
|
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"`.
|
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"`.
|
194
|
+
|
195
|
+
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.
|
196
|
+
|
197
|
+
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
198
|
|
195
199
|
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
200
|
|
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
|
201
|
+
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).
|
202
|
+
|
203
|
+
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
204
|
|
199
205
|
### 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
|
206
|
+
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
207
|
|
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
|
208
|
+
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
209
|
|
204
210
|
### Task life-cycle and the database
|
205
211
|
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 +260,15 @@ To test the results from a result handler:
|
|
254
260
|
MyOperation.handling(:a_result, some: "data") do |test|
|
255
261
|
assert_equal test.outcome, "everything is as expected"
|
256
262
|
# or
|
263
|
+
assert_equal test[:outcome], "everything is as expected"
|
264
|
+
# or
|
257
265
|
expect(test.outcome).to eq "everything is as expected"
|
266
|
+
# or
|
267
|
+
expect(test[:outcome]).to eq "everything is as expected"
|
258
268
|
end
|
259
269
|
```
|
270
|
+
(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).
|
271
|
+
|
260
272
|
To test if a handler has failed:
|
261
273
|
```ruby
|
262
274
|
MyOperation.handling(:a_failure, some: "data") do |test|
|
@@ -297,3 +309,12 @@ Step 4: If you're using RSpec for testing, add `require "operations/matchers" to
|
|
297
309
|
|
298
310
|
## License
|
299
311
|
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.
|
312
|
+
|
313
|
+
## Roadmap
|
314
|
+
|
315
|
+
- [ ] Always raise errors instead of just recording a failure (will be useful when dealing with sub-tasks)
|
316
|
+
- [ ] Simplify calling sub-tasks (and testing the same)
|
317
|
+
- [ ] Split out the state-management definition stuff from the task class (so you can use it without subclassing Operations::Task)
|
318
|
+
- [ ] Make Operations::Task work in the background using ActiveJob
|
319
|
+
- [ ] Add pause/resume capabilities (for example, when a task needs to wait for user input)
|
320
|
+
- [ ] Add wait for sub-tasks capabilities
|
@@ -34,7 +34,7 @@ module Operations::Task::StateManagement
|
|
34
34
|
private def process_current_state(data)
|
35
35
|
handler_for(state).call(self, data)
|
36
36
|
rescue => ex
|
37
|
-
update! status: "failed", status_message: ex.message.to_s.truncate(240), results:
|
37
|
+
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
38
|
end
|
39
39
|
private def state_is_valid
|
40
40
|
errors.add :state, :invalid if state.blank? || handler_for(state.to_sym).nil?
|
@@ -4,7 +4,6 @@ module Operations
|
|
4
4
|
include Deletion
|
5
5
|
include Testing
|
6
6
|
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
7
|
serialize :results, coder: Operations::GlobalIDSerialiser, type: Hash, default: {}
|
9
8
|
|
10
9
|
def self.call(data = {})
|
@@ -21,6 +20,6 @@ module Operations
|
|
21
20
|
|
22
21
|
def fail_with(message) = update! status: "failed", status_message: message.to_s.truncate(240), results: {failure_message: message.to_s}
|
23
22
|
|
24
|
-
def complete(results) = update!(status: "completed", status_message: "completed", results: results)
|
23
|
+
def complete(results) = update!(status: "completed", status_message: "completed", results: results.to_h)
|
25
24
|
end
|
26
25
|
end
|
data/lib/operations/version.rb
CHANGED