rdux 0.10.0 β†’ 0.11.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: dbb84ff27b098d28ad7395b590852b257851825cef51837d01a48c353e2b528e
4
- data.tar.gz: b6bab2fc5e8685bf8ec774e1b3c76e8b533fe0f5e3cbe85d44564f4f04067202
3
+ metadata.gz: 15cd57ce4ac093218d95c279b804de8b5d2bab76b4e81697252559a848fabbd3
4
+ data.tar.gz: 41c9436b74cd2bdf9bfc9ea877b8bb266463af5d5a10e20979c3135c4ed8e974
5
5
  SHA512:
6
- metadata.gz: bc9bd6c320bbbfe0f058234dee3c4729d64aa2a56d0568f26e896fc1fd9a63116a43d8e2634dc68995cb5f4fadb1cc6b30fdeabeee486e67aeedc3eb6e3c9294
7
- data.tar.gz: 86eabdd326c7d46a3654b43f7e06faf79c4af7e00122ce25dda9b49ae1307df7436e536199c7ed31b4d6bdd08b9652abc3e7430eb18b72e936d5845b232b3467
6
+ metadata.gz: f3cf61749ab041d56909e3b77430f9d59a9277b01af2741eb18479cebcc88f7c7569820f2e8d80b75235a2744eff39d908da6ef7d89d6ff120d68e19f29b1a3b
7
+ data.tar.gz: 63eb19f6a5ca7334ff82c074d48c5d35736f45689477ce731061fe8b984e9dc2aaf22c43d052e170138473016cf9aac824e06936d15c58dbcfaaa6cd5a265b48
data/README.md CHANGED
@@ -1,14 +1,26 @@
1
1
  # Rdux - A Minimal Event Sourcing Plugin for Rails
2
2
 
3
- ![Logo](docs/logo.webp)
3
+ <div align="center">
4
+
5
+ <div>
6
+ <img width="500px" src="docs/logo.webp">
7
+ </div>
8
+
9
+ ![GitHub](https://img.shields.io/github/license/artofcodelabs/rdux)
10
+ ![GitHub tag (latest SemVer)](https://img.shields.io/github/v/tag/artofcodelabs/rdux)
11
+
12
+ </div>
4
13
 
5
14
  Rdux is a lightweight, minimalistic Rails plugin designed to introduce event sourcing and audit logging capabilities to your Rails application. With Rdux, you can efficiently track and store the history of actions performed within your app, offering transparency and traceability for key processes.
6
15
 
7
16
  **Key Features**
8
17
 
9
- * **Audit Logging** πŸ‘‰ Rdux stores sanitized input data, the name of module or class (action) responsible for processing them, processing results, and additional metadata in the database.
18
+ * **Audit Logging** πŸ‘‰ Rdux stores sanitized input data, the name of module or class (action performer) responsible for processing them, processing results, and additional metadata in the database.
10
19
  * **Model Representation** πŸ‘‰ Before action is executed it gets stored in the database through the `Rdux::Action` model. `Rdux::Action` is converted to the `Rdux::FailedAction` when it fails. These models can be nested, allowing for complex action structures.
11
- * **Revert and Retry** πŸ‘‰ `Rdux::Action` can be reverted or retried.
20
+ * **Revert and Retry** πŸ‘‰ `Rdux::Action` can be reverted. `Rdux::FailedAction` retains the input data and processing results necessary for implementing custom mechanisms to retry failed actions.
21
+ * **Exception Handling and Recovery** πŸ‘‰ Rdux automatically creates a `Rdux::FailedAction` when an exception occurs during action execution. It retains the `up_payload` and allows you to capture additional data using `opts[:up_result]`, ensuring all necessary information is available for retrying the action.
22
+ * **Metadata** πŸ‘‰ Metadata can include the ID of the authenticated resource responsible for performing a given action, as well as resource IDs from external systems related to the action. This creates a clear audit trail of who executed each action and on whose behalf.
23
+ * **Streams** πŸ‘‰ Rdux enables the identification of action chains (streams) by utilizing resource IDs stored in metadata. This makes it easy to query and track related actions.
12
24
 
13
25
  Rdux is designed to integrate seamlessly with your existing Rails application, offering a straightforward and powerful solution for managing and auditing key actions.
14
26
 
@@ -50,17 +62,17 @@ To dispatch an action using Rdux, use the `dispatch` method (aliased as `perform
50
62
  Definition:
51
63
 
52
64
  ```ruby
53
- def dispatch(action_name, payload, opts = {}, meta: nil)
65
+ def dispatch(action, payload, opts = {}, meta: nil)
54
66
 
55
67
  alias perform dispatch
56
68
  ```
57
69
 
58
70
  Arguments:
59
71
 
60
- * `action_name`: The name of the service, class, or module that will process the action. This is persisted as an instance of `Rdux::Action` in the database, with its `name` attribute set to `action_name`. The `action_name` should correspond to the class or module that implements the `call` or `up` method, referred to as "action" or β€œaction performer.”
61
- * `payload` (Hash): The input data passed as the first argument to the `call` or `up` method of the action performer. This is sanitized and stored in the database before being processed. The keys in the `payload` are stringified during deserialization.
62
- * `opts` (Hash): Optional parameters passed as the second argument to the `call` or `up` method, if defined. This is useful when you want to avoid redundant database queries (e.g., if you already have an ActiveRecord object available). There is a helper that facitilates this use case. The implementation is clear enough IMO `(opts[:ars] || {}).each { |k, v| payload["#{k}_id"] = v.id }`. `:ars` means ActiveRecords. Note that `opts` is not stored in the database and `payload` should be fully sufficient to perform an **action**. `opts` provides an optimization.
63
- * `meta` (Hash): Additional metadata stored in the database alongside the `action_name` and `payload`. The `stream` key is particularly useful for scoping actions during reversions. For example, you can construct a `stream` based on the owner of action.
72
+ * `action`: The name of the module or class (action performer) that processes the action. This is stored in the database as an instance of `Rdux::Action`, with its `name` attribute set to `action` (e.g., `Task::Create`).
73
+ * `payload` (Hash): The input data passed as the first argument to the `call` or `up` method of the action performer. The data is sanitized and stored in the database before being processed by the action performer. During deserialization, the keys in the `payload` are converted to strings.
74
+ * `opts` (Hash): Optional parameters passed as the second argument to the `call` or `up` method, if defined. This can help avoid redundant database queries (e.g., if you already have an ActiveRecord object available before calling `Rdux.perform`). A helper is available to facilitate this use case: `(opts[:ars] || {}).each { |k, v| payload["#{k}_id"] = v.id }`, where `:ars` represents ActiveRecord objects. Note that `opts` is not stored in the database, and the `payload` should be fully sufficient to perform an **action**. `opts` provides an optimization.
75
+ * `meta` (Hash): Additional metadata stored in the database alongside the `action` and `payload`. The `stream` key is particularly useful for specifying the stream of actions used during reversions. For example, a `stream` can be constructed based on the owner of the action.
64
76
 
65
77
  Example:
66
78
 
@@ -80,15 +92,15 @@ Rdux.perform(
80
92
 
81
93
  ![Flow Diagram](docs/flow.png)
82
94
 
83
- ### πŸ’ͺ Action
95
+ ### πŸ•΅οΈβ€β™€οΈ Processing an action
84
96
 
85
- An action in Rdux is a Plain Old Ruby Object (PORO) that implements a class or instance method `call` or `up`.
86
- This method must return an `Rdux::Result` `struct`.
97
+ Action in Rdux is processed by an action performer which is a Plain Old Ruby Object (PORO) that implements a class or instance method `call` or `up`.
98
+ This method must return a `Rdux::Result` `struct`.
87
99
  Optionally, an action can implement a class or instance method `down` to specify how to revert it.
88
100
 
89
101
  #### Action Structure:
90
102
 
91
- * `call` or `up` method: Accepts a required `payload` and an optional `opts` argument. This method processes the action and returns an `Rdux::Result`.
103
+ * `call` or `up` method: Accepts a required `payload` and an optional `opts` argument. This method processes the action and returns a `Rdux::Result`.
92
104
  * `down` method: Accepts the deserialized `down_payload` which is one of arguments of the `Rdux::Result` `struct` returned by the `up` method on success and saved in DB. `down` method can optionally accept the 2nd argument (Hash) which `:nested` key contains nested `Rdux::Action`s
93
105
 
94
106
  See [πŸš› Dispatching an action](#-dispatching-an-action) section.
@@ -181,8 +193,8 @@ end
181
193
  Arguments:
182
194
 
183
195
  * `ok` (Boolean): Indicates whether the action was successful. If `true`, the `Rdux::Action` is persisted in the database.
184
- * `down_payload` (Hash): Passed to the action’s `down` method during reversion (`down` method is called on `Rdux::Action`). It does not have to be defined if an action does not implement the `down` method. `down_payload` is saved in the DB.
185
- * `val` (Hash): Contains any additional data to return besides down_payload.
196
+ * `down_payload` (Hash): Passed to the action performer’s `down` method during reversion (`down` method is called on `Rdux::Action`). It does not have to be defined if an action performer does not implement the `down` method. `down_payload` is saved in the DB.
197
+ * `val` (Hash): Contains different returned data than `down_payload`.
186
198
  * `up_result` (Hash): Stores data related to the action’s execution, such as created record IDs, DB changes, responses from 3rd parties, etc.
187
199
  * `save` (Boolean): If `true` and `ok` is `false`, the action is saved as a `Rdux::FailedAction`.
188
200
  * `after_save` (Proc): Called just before the `dispatch` method returns the `Rdux::Result` with `Rdux::Action` or `Rdux::FailedAction` as an argument.
@@ -229,19 +241,19 @@ res.action.down
229
241
 
230
242
  ### 😷 Sanitization
231
243
 
232
- When calling `Rdux.perform`, the `up_payload` is sanitized using `Rails.application.config.filter_parameters` before saving to the database.
233
- The action’s `up` or `call` method receives the unsanitized version.
234
- Note that if the `up_payload` is sanitized, the `Rdux::Action` cannot be retried via calling the `#up` method.
244
+ When `Rdux.perform` is called, the `up_payload` is sanitized using `Rails.application.config.filter_parameters` before being saved to the database.
245
+ The action performer’s `up` or `call` method receives the unsanitized version.
246
+ Note that once the `up_payload` is sanitized, the `Rdux::Action` cannot be retried by calling the `#up` method.
235
247
 
236
248
  ### πŸ—£οΈ Queries
237
249
 
238
- Most likely, it won't be needed to save a `Rdux::Action` for every request a Rails app receives.
250
+ Most likely, it won't be necessary to save a `Rdux::Action` for every request a Rails app receives.
239
251
  The suggested approach is to save `Rdux::Action`s for Create, Update, and Delete (CUD) operations.
240
252
  This approach organically creates a new layer - queries in addition to actions.
241
253
  Thus, it is required to call `Rdux.perform` only for actions.
242
254
 
243
- An example approach is to create the `perform` method that calls `Rdux.perform` or a query depending on the presence of `action` or `query` keywords.
244
- This method can set `meta` attributes, fulfill params validation, etc.
255
+ One approach is to create a `perform` method that invokes either `Rdux.perform` or a query, depending on the presence of `action` or `query` keywords.
256
+ This method can also handle setting `meta` attributes, performing parameter validation, and more.
245
257
 
246
258
  Example:
247
259
 
@@ -265,10 +277,9 @@ end
265
277
 
266
278
  ### πŸ•΅οΈ Indexing
267
279
 
268
- Depending on your use case, create indices, especially when using PostgreSQL and querying based on JSONB columns.
280
+ Depending on your use case, it’s recommended to create indices, especially when using PostgreSQL and querying JSONB columns.
269
281
  Both `Rdux::Action` and `Rdux::FailedAction` are standard ActiveRecord models.
270
- You can inherit from them and extend.
271
- Depending on your use case, create indices, especially when using PostgreSQL and querying based on `JSONB` columns.
282
+ You can inherit from them and extend.
272
283
 
273
284
  Example:
274
285
  ```ruby
@@ -277,6 +288,46 @@ class Action < Rdux::Action
277
288
  end
278
289
  ```
279
290
 
291
+ ### πŸš‘ Recovering from Exceptions
292
+
293
+ Rdux creates a `Rdux::FailedAction` when an exception is raised during the execution of an action.
294
+ The `up_payload` is retained, but having only the input data is often not enough to retry an action.
295
+ It is crucial to capture data obtained during the action’s execution, up until the exception occurred.
296
+ This can be done by using `opts[:up_result]` to store all necessary data incrementally.
297
+ The assigned data will then be available as the `up_result` argument in the `Rdux::FailedAction`.
298
+
299
+ Example:
300
+ ```ruby
301
+ class CreditCard
302
+ class Charge
303
+ class << self
304
+ def call(payload, opts)
305
+ create_res = create(payload.slice('user_id', 'credit_card'), opts.slice(:user))
306
+ return create_res unless create_res.ok
307
+
308
+ opts[:up_result] = { credit_card_create_action_id: create_res.action.id }
309
+ charge_id = PaymentGateway.charge(create_res.val[:credit_card].token, payload['amount'])[:id]
310
+ if charge_id.nil?
311
+ Rdux::Result[ok: false, val: { errors: { base: 'Invalid credit card' } }, save: true,
312
+ nested: [create_res]]
313
+ else
314
+ Rdux::Result[ok: true, val: { charge_id: }, nested: [create_res]]
315
+ end
316
+ end
317
+
318
+ private
319
+
320
+ def create(payload, opts)
321
+ res = Rdux.perform(Create, payload, opts)
322
+ return res if res.ok
323
+
324
+ Rdux::Result[ok: false, val: { errors: res.val[:errors] }, save: true, nested: [res]]
325
+ end
326
+ end
327
+ end
328
+ end
329
+ ```
330
+
280
331
  ## πŸ‘©πŸ½β€πŸ”¬ Testing
281
332
 
282
333
  ### πŸ’‰ Setup
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Rdux
4
- class Action < ApplicationRecord
4
+ class Action < ActiveRecord::Base
5
5
  include Actionable
6
6
 
7
7
  attr_accessor :up_payload_unsanitized
@@ -1,7 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Rdux
4
- class FailedAction < ApplicationRecord
4
+ class FailedAction < ActiveRecord::Base
5
5
  include Actionable
6
6
 
7
7
  belongs_to :rdux_failed_action, optional: true, class_name: 'Rdux::FailedAction'
data/lib/rdux/version.rb CHANGED
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
3
  module Rdux
4
- VERSION = '0.10.0'
4
+ VERSION = '0.11.0'
5
5
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: rdux
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.10.0
4
+ version: 0.11.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Zbigniew Humeniuk
8
8
  autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2024-09-25 00:00:00.000000000 Z
11
+ date: 2025-03-12 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: rails