rdux 0.10.0 β 0.12.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 +74 -23
- data/app/models/rdux/action.rb +1 -1
- data/app/models/rdux/failed_action.rb +1 -1
- data/lib/rdux/version.rb +1 -1
- data/lib/rdux.rb +10 -5
- metadata +4 -4
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: bb992b109d0983534ec092280239b9eb1f689f9e92f5f011047660215ab6a39a
|
4
|
+
data.tar.gz: 22abeef261cd3d464361ed7b910ce8d033c611e5681f9228e631c99d88758f87
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 4aa7c1246cdfd68af71e4171f1e20bab160c52a1e9d891519209a3dd2308307e87ead38b3e66ffce317136de745bfedf88a10544d970bf13e89d58528ab570cf
|
7
|
+
data.tar.gz: 21b865e2527a7df5d2ab952d44fd607e3fafb8f91f116839d60de3b51f2a5ebec4660e1691e667ad70717fcc6f88fe8e233ff9473f02280fb21e9c294a29c73c
|
data/README.md
CHANGED
@@ -1,14 +1,26 @@
|
|
1
1
|
# Rdux - A Minimal Event Sourcing Plugin for Rails
|
2
2
|
|
3
|
-
|
3
|
+
<div align="center">
|
4
|
+
|
5
|
+
<div>
|
6
|
+
<img width="500px" src="docs/logo.webp">
|
7
|
+
</div>
|
8
|
+
|
9
|
+

|
10
|
+

|
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
|
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(
|
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
|
-
* `
|
61
|
-
* `payload` (Hash): The input data passed as the first argument to the `call` or `up` method of the action performer.
|
62
|
-
* `opts` (Hash): Optional parameters passed as the second argument to the `call` or `up` method, if defined. This
|
63
|
-
* `meta` (Hash): Additional metadata stored in the database alongside the `
|
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
|

|
82
94
|
|
83
|
-
###
|
95
|
+
### π΅οΈββοΈ Processing an action
|
84
96
|
|
85
|
-
|
86
|
-
This method must return
|
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
|
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
|
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
|
233
|
-
The actionβs `up` or `call` method receives the unsanitized version.
|
234
|
-
Note that
|
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
|
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
|
-
|
244
|
-
This method can
|
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
|
-
|
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
|
data/app/models/rdux/action.rb
CHANGED
data/lib/rdux/version.rb
CHANGED
data/lib/rdux.rb
CHANGED
@@ -8,14 +8,11 @@ require 'active_support/concern'
|
|
8
8
|
module Rdux
|
9
9
|
class << self
|
10
10
|
def dispatch(action_name, payload, opts = {}, meta: nil)
|
11
|
-
|
12
|
-
action = Action.new(name: action_name, up_payload: payload, meta:)
|
13
|
-
sanitize(action)
|
14
|
-
action.save!
|
11
|
+
action = create_action(action_name, payload, opts, meta)
|
15
12
|
res = call_call_or_up_on_action(action, opts)
|
16
13
|
res.up_result ||= opts[:up_result]
|
17
14
|
assign_and_persist(res, action)
|
18
|
-
res.after_save
|
15
|
+
res.after_save.call(res.action) if res.after_save && res.action
|
19
16
|
res
|
20
17
|
end
|
21
18
|
|
@@ -23,6 +20,14 @@ module Rdux
|
|
23
20
|
|
24
21
|
private
|
25
22
|
|
23
|
+
def create_action(action_name, payload, opts, meta)
|
24
|
+
(opts[:ars] || {}).each { |k, v| payload["#{k}_id"] = v.id }
|
25
|
+
action = Action.new(name: action_name, up_payload: payload, meta:)
|
26
|
+
sanitize(action)
|
27
|
+
action.save!
|
28
|
+
action
|
29
|
+
end
|
30
|
+
|
26
31
|
def call_call_or_up_on_action(action, opts)
|
27
32
|
res = action.call(opts)
|
28
33
|
if res
|
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.
|
4
|
+
version: 0.12.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:
|
11
|
+
date: 2025-03-17 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: rails
|
@@ -36,14 +36,14 @@ dependencies:
|
|
36
36
|
requirements:
|
37
37
|
- - ">="
|
38
38
|
- !ruby/object:Gem::Version
|
39
|
-
version: 1.5.
|
39
|
+
version: 1.5.9
|
40
40
|
type: :development
|
41
41
|
prerelease: false
|
42
42
|
version_requirements: !ruby/object:Gem::Requirement
|
43
43
|
requirements:
|
44
44
|
- - ">="
|
45
45
|
- !ruby/object:Gem::Version
|
46
|
-
version: 1.5.
|
46
|
+
version: 1.5.9
|
47
47
|
- !ruby/object:Gem::Dependency
|
48
48
|
name: rubocop
|
49
49
|
requirement: !ruby/object:Gem::Requirement
|