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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 9cfaaef3f5470debfff84e64763fd703752ae0c70a7c62c2926468aa9b897c83
4
- data.tar.gz: 2af99adbc94c3c6f734e9272b909d42d199ddc7d7f18a866f4d14bd2ad78b9ff
3
+ metadata.gz: d19a26ed4cb26b809efc2e4d551a4895613608348f3524f1764c000aceb6aeca
4
+ data.tar.gz: 2290a93af8caeacfa00fd941e62e4785366e4efa80d59579201bfe93cd85fa0a
5
5
  SHA512:
6
- metadata.gz: 40ba366b5f54cfd1d376ac34731f1d8e462afffef0512b9011603734a3c45319e40994b7790f556e6bfd27923b0cf67aadd521bebf98f004b71d7aad5956c53f
7
- data.tar.gz: 833099833d7ef03773530f4301ee5b66434c26c94af54508e0c006aa082e6d4c812b84fd621dd9e5d9cb9a300be17292bf4be7c46679f5429c50937d5362a94b
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
- ## Installation
229
- Add this line to your application's Gemfile:
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
- gem "standard_procedure_operations"
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
- Run `bundle install`, copy and run the migrations to add the tasks table to your database:
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 app:operations:install:migrations
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.
@@ -2,4 +2,6 @@ class Operations::Task::DataCarrier < OpenStruct
2
2
  def go_to(state, message = nil) = task.go_to(state, self, message)
3
3
 
4
4
  def fail_with(message) = task.fail_with(message)
5
+
6
+ def complete(results) = task.complete(results)
5
7
  end
@@ -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(exception_message: ex.message, exception_class: ex.class.name, exception_backtrace: ex.backtrace)
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) : task.go_to(next_state, data)
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
- task.send :complete, results
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 = {}) = create!(state: initial_state).tap { |task| task.send(:process_current_state, DataCarrier.new(data.merge(task: task))) }
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
- private def complete(results) = update!(status: "completed", results: results)
24
+ def complete(results) = update!(status: "completed", results: results)
19
25
  end
20
26
  end
@@ -1,12 +1,12 @@
1
- class CreateOperationsTasks < ActiveRecord::Migration[8.0]
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, default: "{}"
9
- t.text :results, default: "{}"
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
@@ -0,0 +1,2 @@
1
+ class Operations::MissingInputsError < Operations::Error
2
+ end
@@ -1,3 +1,3 @@
1
1
  module Operations
2
- VERSION = "0.1.0"
2
+ VERSION = "0.2.0"
3
3
  end
data/lib/operations.rb CHANGED
@@ -1,7 +1,10 @@
1
1
  require "ostruct"
2
2
 
3
3
  module Operations
4
+ class Error < StandardError
5
+ end
4
6
  require "operations/version"
5
7
  require "operations/engine"
6
8
  require "operations/global_id_serialiser"
9
+ require "operations/missing_inputs_error"
7
10
  end
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.1.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/releases
58
+ changelog_uri: https://github.com/standard-procedure/operations/tags
56
59
  rdoc_options: []
57
60
  require_paths:
58
61
  - lib