rdux 0.9.1 → 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: fb70d8633e88f42179223f60bfe5634273591875f0e3d104fb076128af8811ba
4
- data.tar.gz: 375487c3f51b78803b9d610877edb5e5fcec70918e86e376cf2601188a9e4358
3
+ metadata.gz: 15cd57ce4ac093218d95c279b804de8b5d2bab76b4e81697252559a848fabbd3
4
+ data.tar.gz: 41c9436b74cd2bdf9bfc9ea877b8bb266463af5d5a10e20979c3135c4ed8e974
5
5
  SHA512:
6
- metadata.gz: 4fbd330495ba4b140f828dc2e5184e27f4abd653c9e8d314d8a576c4612294c8baca8aceeeeccdab1e8be184a566544daaf1a75bc2dab2f1bba1801aab7f11cd
7
- data.tar.gz: da3820133a5d277e1b5a20eee3876efb13e7cd434105585a3b2da5a014062072e17a77a93b35f35bb601d1f7d6c30dc4b930adb9e0f56e99aaeb358f1b9629e8
6
+ metadata.gz: f3cf61749ab041d56909e3b77430f9d59a9277b01af2741eb18479cebcc88f7c7569820f2e8d80b75235a2744eff39d908da6ef7d89d6ff120d68e19f29b1a3b
7
+ data.tar.gz: 63eb19f6a5ca7334ff82c074d48c5d35736f45689477ce731061fe8b984e9dc2aaf22c43d052e170138473016cf9aac824e06936d15c58dbcfaaa6cd5a265b48
data/README.md CHANGED
@@ -1,54 +1,336 @@
1
- # Rdux
2
- Minimal take on event sourcing.
1
+ # Rdux - A Minimal Event Sourcing Plugin for Rails
3
2
 
4
- ## Usage
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>
13
+
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.
15
+
16
+ **Key Features**
17
+
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.
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.
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.
24
+
25
+ Rdux is designed to integrate seamlessly with your existing Rails application, offering a straightforward and powerful solution for managing and auditing key actions.
26
+
27
+ ## 📲 Instalation
28
+
29
+ Add this line to your application's Gemfile:
30
+
31
+ ```ruby
32
+ gem 'rdux'
33
+ ```
34
+
35
+ And then execute:
36
+
37
+ ```bash
38
+ $ bundle
39
+ ```
40
+
41
+ Or install it yourself as:
42
+
43
+ ```bash
44
+ $ gem install rdux
45
+ ```
46
+
47
+ Then install and run migrations:
5
48
 
6
49
  ```bash
7
50
  $ bin/rails rdux:install:migrations
8
51
  $ bin/rails db:migrate
9
52
  ```
10
53
 
11
- ### Code structure
54
+ ⚠️ Note: Rdux uses `JSONB` datatype instead of `text` for Postgres.
55
+
56
+ ## 🎮 Usage
57
+
58
+ ### 🚛 Dispatching an action
59
+
60
+ To dispatch an action using Rdux, use the `dispatch` method (aliased as `perform`).
12
61
 
13
- #### Dispatch action
62
+ Definition:
63
+
64
+ ```ruby
65
+ def dispatch(action, payload, opts = {}, meta: nil)
66
+
67
+ alias perform dispatch
68
+ ```
69
+
70
+ Arguments:
71
+
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.
76
+
77
+ Example:
14
78
 
15
79
  ```ruby
16
80
  Rdux.perform(
17
- Activity::Stop,
18
- { activity_id: current_activity.id },
19
- { activity: current_activity },
81
+ Task::Create,
82
+ { task: { name: 'Foo bar baz' } },
83
+ { ars: { user: current_user } },
20
84
  meta: {
21
- stream: { user_id: 123, context: 'foo' }, bar: 'baz'
85
+ stream: { user_id: current_user.id, context: 'foo' },
86
+ bar: 'baz'
22
87
  }
23
88
  )
24
89
  ```
25
90
 
26
- #### Return
91
+ ### 📈 Flow diagram
92
+
93
+ ![Flow Diagram](docs/flow.png)
94
+
95
+ ### 🕵️‍♀️ Processing an action
96
+
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`.
99
+ Optionally, an action can implement a class or instance method `down` to specify how to revert it.
100
+
101
+ #### Action Structure:
102
+
103
+ * `call` or `up` method: Accepts a required `payload` and an optional `opts` argument. This method processes the action and returns a `Rdux::Result`.
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
105
+
106
+ See [🚛 Dispatching an action](#-dispatching-an-action) section.
107
+
108
+ Examples:
27
109
 
28
110
  ```ruby
29
- Rdux::Result[true, { activity: activity }]
111
+ # app/actions/task/create.rb
112
+
113
+ class Task
114
+ class Create
115
+ def up(payload, opts)
116
+ user = opts.dig(:ars, :user) || User.find(payload['user_id'])
117
+ task = user.tasks.new(payload['task'])
118
+ if task.save
119
+ Rdux::Result[ok: true, down_payload: { user_id: user.id, task_id: task.id }, val: { task: }]
120
+ else
121
+ Rdux::Result[false, { errors: task.errors }]
122
+ end
123
+ end
124
+
125
+ def down(payload)
126
+ Delete.up(payload)
127
+ end
128
+ end
129
+ end
30
130
  ```
31
131
 
32
- ## Installation
33
- Add this line to your application's Gemfile:
132
+ ```ruby
133
+ # app/actions/task/delete.rb
134
+
135
+ class Task
136
+ module Delete
137
+ def self.up(payload)
138
+ user = User.find(payload['user_id'])
139
+ task = user.tasks.find(payload['task_id'])
140
+ task.destroy
141
+ Rdux::Result[true, { task: task.attributes }]
142
+ end
143
+ end
144
+ end
145
+ ```
146
+
147
+ #### Suggested Directory Structure:
148
+
149
+ The location that is often used for entities like actions accross code bases is `app/services`.
150
+ This directory is de facto the bag of random objects.
151
+ I'd recomment to place actions inside `app/actions` for better organization and consistency.
152
+ Actions are consistent in terms of structure, input and output data.
153
+ They are good canditates to create a new layer in Rails apps.
154
+
155
+ Structure:
156
+
157
+ ```
158
+ .
159
+ └── app/actions/
160
+ ├── activity/
161
+ │ ├── common/
162
+ │ │ └── fetch.rb
163
+ │ ├── create.rb
164
+ │ ├── stop.rb
165
+ │ └── switch.rb
166
+ ├── task/
167
+ │ ├── create.rb
168
+ │ └── delete.rb
169
+ └── misc/
170
+ └── create_attachment.rb
171
+ ```
172
+
173
+ The [dedicated page about actions](docs/ACTIONS.md) contains more arguments in favor of actions.
174
+
175
+ ### ⛩️ Returned `struct` `Rdux::Result`
176
+
177
+ Definition:
34
178
 
35
179
  ```ruby
36
- gem 'rdux'
180
+ module Rdux
181
+ Result = Struct.new(:ok, :down_payload, :val, :up_result, :save, :after_save, :nested, :action) do
182
+ def val
183
+ self[:val] || down_payload
184
+ end
185
+
186
+ def save_failed?
187
+ ok == false && save
188
+ end
189
+ end
190
+ end
37
191
  ```
38
192
 
39
- And then execute:
40
- ```bash
41
- $ bundle
193
+ Arguments:
194
+
195
+ * `ok` (Boolean): Indicates whether the action was successful. If `true`, the `Rdux::Action` is persisted in the database.
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`.
198
+ * `up_result` (Hash): Stores data related to the action’s execution, such as created record IDs, DB changes, responses from 3rd parties, etc.
199
+ * `save` (Boolean): If `true` and `ok` is `false`, the action is saved as a `Rdux::FailedAction`.
200
+ * `after_save` (Proc): Called just before the `dispatch` method returns the `Rdux::Result` with `Rdux::Action` or `Rdux::FailedAction` as an argument.
201
+ * `nested` (Array of `Rdux::Result`): `Rdux::Action` can be connected with other `rdux_actions`. `Rdux::FailedAction` can be connected with other `rdux_actions` and `rdux_failed_actions`. To establish an association, a given action must `Rdux.dispatch` other actions in the `up` or `call` method and add the returned by the `dispatch` value (`Rdux::Result`) to the `:nested` array
202
+ * `action`: Rdux assigns `Rdux::Action` or `Rdux::FailedAction` to this argument
203
+
204
+ ### ⏮️ Reverting an Action
205
+
206
+ To revert an action, call the `down` method on the persisted in DB `Rdux::Action` instance.
207
+ The `Rdux::Action` must have a `down_payload` defined and the action (action performer) must have the `down` method implemented.
208
+
209
+ ![Revert action](docs/down.png)
210
+
211
+ The `down_at` attribute is set upon successful reversion. Actions cannot be reverted if there are newer, unreverted actions in the same stream (if defined) or in general. See `meta` in [🚛 Dispatching an action](#-dispatching-an-action) section.
212
+
213
+ ### 🗿 Data model
214
+
215
+ ```ruby
216
+ payload = {
217
+ task: { 'name' => 'Foo bar baz' },
218
+ user_id: 159163583
219
+ }
220
+
221
+ res = Rdux.dispatch(Task::Create, payload)
222
+
223
+ res.action
224
+ # #<Rdux::Action:0x000000011c4d8e98
225
+ # id: 1,
226
+ # name: "Task::Create",
227
+ # up_payload: {"task"=>{"name"=>"Foo bar baz"}, "user_id"=>159163583},
228
+ # down_payload: {"task_id"=>207620945},
229
+ # down_at: nil,
230
+ # up_payload_sanitized: false,
231
+ # up_result: nil,
232
+ # meta: {},
233
+ # stream_hash: nil,
234
+ # rdux_action_id: nil,
235
+ # rdux_failed_action_id: nil,
236
+ # created_at: Fri, 28 Jun 2024 21:35:36.838898000 UTC +00:00,
237
+ # updated_at: Fri, 28 Jun 2024 21:35:36.839728000 UTC +00:00>>
238
+
239
+ res.action.down
42
240
  ```
43
241
 
44
- Or install it yourself as:
45
- ```bash
46
- $ gem install rdux
242
+ ### 😷 Sanitization
243
+
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.
247
+
248
+ ### 🗣️ Queries
249
+
250
+ Most likely, it won't be necessary to save a `Rdux::Action` for every request a Rails app receives.
251
+ The suggested approach is to save `Rdux::Action`s for Create, Update, and Delete (CUD) operations.
252
+ This approach organically creates a new layer - queries in addition to actions.
253
+ Thus, it is required to call `Rdux.perform` only for actions.
254
+
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.
257
+
258
+ Example:
259
+
260
+ ```ruby
261
+ class TasksController < ApiController
262
+ def show
263
+ perform(
264
+ query: Task::Show,
265
+ payload: { id: params[:id] }
266
+ )
267
+ end
268
+
269
+ def create
270
+ perform(
271
+ action: Task::Create,
272
+ payload: create_task_params
273
+ )
274
+ end
275
+ end
276
+ ```
277
+
278
+ ### 🕵️ Indexing
279
+
280
+ Depending on your use case, it’s recommended to create indices, especially when using PostgreSQL and querying JSONB columns.
281
+ Both `Rdux::Action` and `Rdux::FailedAction` are standard ActiveRecord models.
282
+ You can inherit from them and extend.
283
+
284
+ Example:
285
+ ```ruby
286
+ class Action < Rdux::Action
287
+ include Actionable
288
+ end
47
289
  ```
48
290
 
49
- ## Test
291
+ ### 🚑 Recovering from Exceptions
50
292
 
51
- ### Setup
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
+
331
+ ## 👩🏽‍🔬 Testing
332
+
333
+ ### 💉 Setup
52
334
 
53
335
  ```bash
54
336
  $ cd test/dummy
@@ -57,12 +339,17 @@ $ DB=all bin/rails db:prepare
57
339
  $ cd ../..
58
340
  ```
59
341
 
60
- ### Run tests
342
+ ### 🧪 Run tests
61
343
 
62
344
  ```bash
63
345
  $ DB=postgres bin/rails test
64
346
  $ DB=sqlite bin/rails test
65
347
  ```
66
348
 
67
- ## License
349
+ ## 📄 License
350
+
68
351
  The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
352
+
353
+ ## 👨‍🏭 Author
354
+
355
+ Zbigniew Humeniuk from [Art of Code](https://artofcode.co)
@@ -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
@@ -63,15 +63,15 @@ module Rdux
63
63
  return if performer.nil?
64
64
 
65
65
  if opts.any? || performer.method(meth).arity.abs == 2
66
- performer.public_send(meth, payload, opts.merge(action: self))
66
+ performer.public_send(meth, payload, opts.merge!(action: self))
67
67
  else
68
68
  performer.public_send(meth, payload)
69
69
  end
70
70
  end
71
71
 
72
72
  def build_opts
73
+ nested = rdux_actions.order(:created_at)
73
74
  {}.tap do |h|
74
- nested = rdux_actions.order(:created_at)
75
75
  h[:nested] = nested if nested.any?
76
76
  end
77
77
  end
@@ -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.9.1'
4
+ VERSION = '0.11.0'
5
5
  end
data/lib/rdux.rb CHANGED
@@ -13,6 +13,7 @@ module Rdux
13
13
  sanitize(action)
14
14
  action.save!
15
15
  res = call_call_or_up_on_action(action, opts)
16
+ res.up_result ||= opts[:up_result]
16
17
  assign_and_persist(res, action)
17
18
  res.after_save&.call(res.action)
18
19
  res
@@ -31,7 +32,7 @@ module Rdux
31
32
 
32
33
  action.up(opts)
33
34
  rescue StandardError => e
34
- handle_exception(e, action)
35
+ handle_exception(e, action, opts[:up_result])
35
36
  end
36
37
 
37
38
  def no_down(res)
@@ -80,14 +81,13 @@ module Rdux
80
81
  action.up_payload = up_payload_sanitized
81
82
  end
82
83
 
83
- def handle_exception(exc, action)
84
+ def handle_exception(exc, action, up_result)
84
85
  failed_action = action.to_failed_action
85
- failed_action.up_result = {
86
- 'Exception' => {
87
- class: exc.class.name,
88
- message: exc.message
89
- }
90
- }
86
+ failed_action.up_result ||= up_result || {}
87
+ failed_action.up_result.merge!({ 'Exception' => {
88
+ class: exc.class.name,
89
+ message: exc.message
90
+ } })
91
91
  failed_action.save!
92
92
  action.destroy
93
93
  raise exc
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.9.1
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-07-08 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
@@ -36,59 +36,60 @@ dependencies:
36
36
  requirements:
37
37
  - - ">="
38
38
  - !ruby/object:Gem::Version
39
- version: 1.5.4
39
+ version: 1.5.8
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.4
46
+ version: 1.5.8
47
47
  - !ruby/object:Gem::Dependency
48
48
  name: rubocop
49
49
  requirement: !ruby/object:Gem::Requirement
50
50
  requirements:
51
51
  - - ">="
52
52
  - !ruby/object:Gem::Version
53
- version: 1.59.0
53
+ version: 1.66.1
54
54
  type: :development
55
55
  prerelease: false
56
56
  version_requirements: !ruby/object:Gem::Requirement
57
57
  requirements:
58
58
  - - ">="
59
59
  - !ruby/object:Gem::Version
60
- version: 1.59.0
60
+ version: 1.66.1
61
61
  - !ruby/object:Gem::Dependency
62
62
  name: rubocop-rails
63
63
  requirement: !ruby/object:Gem::Requirement
64
64
  requirements:
65
65
  - - ">="
66
66
  - !ruby/object:Gem::Version
67
- version: 2.23.1
67
+ version: 2.26.0
68
68
  type: :development
69
69
  prerelease: false
70
70
  version_requirements: !ruby/object:Gem::Requirement
71
71
  requirements:
72
72
  - - ">="
73
73
  - !ruby/object:Gem::Version
74
- version: 2.23.1
74
+ version: 2.26.0
75
75
  - !ruby/object:Gem::Dependency
76
76
  name: sqlite3
77
77
  requirement: !ruby/object:Gem::Requirement
78
78
  requirements:
79
79
  - - ">="
80
80
  - !ruby/object:Gem::Version
81
- version: 1.7.0
81
+ version: 2.1.0
82
82
  type: :development
83
83
  prerelease: false
84
84
  version_requirements: !ruby/object:Gem::Requirement
85
85
  requirements:
86
86
  - - ">="
87
87
  - !ruby/object:Gem::Version
88
- version: 1.7.0
89
- description: |
90
- Write apps that are easy to test.
91
- It makes it easy to trace when, where, why, and how your application's state changed.
88
+ version: 2.1.0
89
+ description: "Rdux is a lightweight, minimalistic Rails plugin designed to introduce
90
+ event sourcing and audit logging capabilities to your Rails application. \nWith
91
+ Rdux, you can efficiently track and store the history of actions performed within
92
+ your app, offering transparency and traceability for key processes.\n"
92
93
  email:
93
94
  - hello@artofcode.co
94
95
  executables: []
@@ -128,8 +129,8 @@ required_rubygems_version: !ruby/object:Gem::Requirement
128
129
  - !ruby/object:Gem::Version
129
130
  version: '0'
130
131
  requirements: []
131
- rubygems_version: 3.5.6
132
+ rubygems_version: 3.5.16
132
133
  signing_key:
133
134
  specification_version: 4
134
- summary: Rdux adds a new layer to Rails apps - actions.
135
+ summary: A Minimal Event Sourcing Plugin for Rails
135
136
  test_files: []