active_record_auditable 0.1.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.
Files changed (32) hide show
  1. checksums.yaml +7 -0
  2. data/MIT-LICENSE +20 -0
  3. data/README.md +41 -0
  4. data/Rakefile +8 -0
  5. data/app/assets/config/active_record_auditable_manifest.js +1 -0
  6. data/app/assets/stylesheets/active_record_auditable/application.css +15 -0
  7. data/app/controllers/active_record_auditable/application_controller.rb +4 -0
  8. data/app/helpers/active_record_auditable/application_helper.rb +4 -0
  9. data/app/jobs/active_record_auditable/application_job.rb +4 -0
  10. data/app/mailers/active_record_auditable/application_mailer.rb +6 -0
  11. data/app/models/active_record_auditable/application_record.rb +5 -0
  12. data/app/models/active_record_auditable/audit.rb +16 -0
  13. data/app/models/active_record_auditable/audit_action.rb +5 -0
  14. data/app/models/active_record_auditable/audit_auditable_type.rb +5 -0
  15. data/app/models/active_record_auditable/base_audit.rb +23 -0
  16. data/app/models/concerns/active_record_auditable/acts_as_json.rb +49 -0
  17. data/app/services/active_record_auditable/application_service.rb +2 -0
  18. data/app/services/active_record_auditable/audit_classes.rb +19 -0
  19. data/app/services/active_record_auditable/create_audits_table_for_model_class.rb +28 -0
  20. data/app/services/active_record_auditable/migrate_audits_from_shared_table.rb +70 -0
  21. data/app/views/layouts/active_record_auditable/application.html.erb +15 -0
  22. data/config/routes.rb +2 -0
  23. data/db/migrate/20240307212702_create_audit_actions.rb +8 -0
  24. data/db/migrate/20240307212756_create_audit_auditable_types.rb +8 -0
  25. data/db/migrate/20240307212826_create_audits.rb +12 -0
  26. data/db/migrate/20240609073430_add_params_to_audits.rb +5 -0
  27. data/lib/active_record_auditable/audited.rb +153 -0
  28. data/lib/active_record_auditable/engine.rb +5 -0
  29. data/lib/active_record_auditable/version.rb +3 -0
  30. data/lib/active_record_auditable.rb +7 -0
  31. data/lib/tasks/active_record_auditable_tasks.rake +4 -0
  32. metadata +104 -0
checksums.yaml ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA256:
3
+ metadata.gz: ff588e17ec1c5f91e07018334f37b9e3073fb8ebcfa0782625c3f59729758c7d
4
+ data.tar.gz: eb559e570cb5295a111bfecd26541e6d8625739578eb119aeda3d9391d9eb759
5
+ SHA512:
6
+ metadata.gz: c3ecf7ac5c7b6cae5319b6877ae5c2919e9f87e0de4ceae07a79b93f4fa2855f77a79b67aabf33716d20753dac5d755468933e16e7da04bf7474369ad61093d3
7
+ data.tar.gz: 7c419488a1cf65effe3863e0a30513fdb2bf8cb0d5c7a3ea1851afca29183733dbb311479eb0842487203cd570ae4588a6049d2cc5162afeb6bb3752c97bd01c
data/MIT-LICENSE ADDED
@@ -0,0 +1,20 @@
1
+ Copyright kaspernj
2
+
3
+ Permission is hereby granted, free of charge, to any person obtaining
4
+ a copy of this software and associated documentation files (the
5
+ "Software"), to deal in the Software without restriction, including
6
+ without limitation the rights to use, copy, modify, merge, publish,
7
+ distribute, sublicense, and/or sell copies of the Software, and to
8
+ permit persons to whom the Software is furnished to do so, subject to
9
+ the following conditions:
10
+
11
+ The above copyright notice and this permission notice shall be
12
+ included in all copies or substantial portions of the Software.
13
+
14
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,41 @@
1
+ # ActiveRecordAuditable
2
+ Short description and motivation.
3
+
4
+ ## Usage
5
+ How to use my plugin.
6
+
7
+ ## Installation
8
+ Add this line to your application's Gemfile:
9
+
10
+ ```ruby
11
+ gem "active_record_auditable"
12
+ ```
13
+
14
+ And then execute:
15
+ ```bash
16
+ $ bundle
17
+ ```
18
+
19
+ Or install it yourself as:
20
+ ```bash
21
+ $ gem install active_record_auditable
22
+ ```
23
+
24
+ ```bash
25
+ rails active_record_auditable:install:migrations
26
+ ```
27
+
28
+ Sometimes you need to do something like this in `config/initializers/active_record_auditable.rb`:
29
+ ```ruby
30
+ Rails.configuration.to_prepare do
31
+ ActiveRecordAuditable::Audit.class_eval do
32
+ belongs_to :user, default: -> { Current.user }, optional: true
33
+ end
34
+ end
35
+ ```
36
+
37
+ ## Contributing
38
+ Contribution directions go here.
39
+
40
+ ## License
41
+ The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).
data/Rakefile ADDED
@@ -0,0 +1,8 @@
1
+ require "bundler/setup"
2
+
3
+ APP_RAKEFILE = File.expand_path("spec/dummy/Rakefile", __dir__)
4
+ load "rails/tasks/engine.rake"
5
+
6
+ load "rails/tasks/statistics.rake"
7
+
8
+ require "bundler/gem_tasks"
@@ -0,0 +1 @@
1
+ //= link_directory ../stylesheets/active_record_auditable .css
@@ -0,0 +1,15 @@
1
+ /*
2
+ * This is a manifest file that'll be compiled into application.css, which will include all the files
3
+ * listed below.
4
+ *
5
+ * Any CSS and SCSS file within this directory, lib/assets/stylesheets, vendor/assets/stylesheets,
6
+ * or any plugin's vendor/assets/stylesheets directory can be referenced here using a relative path.
7
+ *
8
+ * You're free to add application-wide styles to this file and they'll appear at the bottom of the
9
+ * compiled file so the styles you add here take precedence over styles defined in any other CSS/SCSS
10
+ * files in this directory. Styles in this file should be added after the last require_* statement.
11
+ * It is generally better to create a new file per style scope.
12
+ *
13
+ *= require_tree .
14
+ *= require_self
15
+ */
@@ -0,0 +1,4 @@
1
+ module ActiveRecordAuditable
2
+ class ApplicationController < ActionController::Base
3
+ end
4
+ end
@@ -0,0 +1,4 @@
1
+ module ActiveRecordAuditable
2
+ module ApplicationHelper
3
+ end
4
+ end
@@ -0,0 +1,4 @@
1
+ module ActiveRecordAuditable
2
+ class ApplicationJob < ActiveJob::Base
3
+ end
4
+ end
@@ -0,0 +1,6 @@
1
+ module ActiveRecordAuditable
2
+ class ApplicationMailer < ActionMailer::Base
3
+ default from: "from@example.com"
4
+ layout "mailer"
5
+ end
6
+ end
@@ -0,0 +1,5 @@
1
+ module ActiveRecordAuditable
2
+ class ApplicationRecord < ActiveRecord::Base
3
+ self.abstract_class = true
4
+ end
5
+ end
@@ -0,0 +1,16 @@
1
+ class ActiveRecordAuditable::Audit < ActiveRecordAuditable::BaseAudit
2
+ self.table_name = "audits"
3
+
4
+ belongs_to :audit_auditable_type, class_name: "ActiveRecordAuditable::AuditAuditableType"
5
+ belongs_to :auditable, optional: true, polymorphic: true
6
+
7
+ scope :where_type, ->(type) { joins(:audit_auditable_type).where(audit_auditable_types: {name: type}) }
8
+
9
+ before_validation :set_audit_auditable_type
10
+
11
+ private
12
+
13
+ def set_audit_auditable_type
14
+ self.audit_auditable_type = ActiveRecordAuditable::AuditAuditableType.find_or_create_by!(name: auditable_type) if !audit_auditable_type && auditable_type?
15
+ end
16
+ end
@@ -0,0 +1,5 @@
1
+ class ActiveRecordAuditable::AuditAction < ApplicationRecord
2
+ self.table_name = "audit_actions"
3
+
4
+ has_many :audits, class_name: "ActiveRecordAuditable::Audit", dependent: :restrict_with_error
5
+ end
@@ -0,0 +1,5 @@
1
+ class ActiveRecordAuditable::AuditAuditableType < ApplicationRecord
2
+ self.table_name = "audit_auditable_types"
3
+
4
+ has_many :audits, class_name: "ActiveRecordAuditable::Audit", dependent: :restrict_with_error
5
+ end
@@ -0,0 +1,23 @@
1
+ class ActiveRecordAuditable::BaseAudit < ActiveRecordAuditable::ApplicationRecord
2
+ self.abstract_class = true
3
+
4
+ def self.inherited(child)
5
+ super
6
+
7
+ child.include ActiveRecordAuditable::ActsAsJson
8
+
9
+ child.acts_as_json :audited_changes
10
+ child.acts_as_json :extra_liquid_variables
11
+ child.acts_as_json :params
12
+ end
13
+
14
+ belongs_to :audit_action, class_name: "ActiveRecordAuditable::AuditAction"
15
+
16
+ scope :where_action, ->(action) { joins(:audit_action).where(audit_actions: {action:}) }
17
+
18
+ delegate :action, to: :audit_action
19
+
20
+ def action=(action_name)
21
+ self.audit_action = ActiveRecordAuditable::AuditAction.find_or_create_by!(action: action_name)
22
+ end
23
+ end
@@ -0,0 +1,49 @@
1
+ module ActiveRecordAuditable::ActsAsJson
2
+ def self.included(base)
3
+ base.extend ClassMethods
4
+ end
5
+
6
+ module ClassMethods
7
+ def acts_as_json(attribute_name) # rubocop:disable Metrics/PerceivedComplexity
8
+ return if ActiveRecordAuditable::Audit.connection.class.name.include?("SQLite")
9
+
10
+ serialize attribute_name, coder: JSON
11
+
12
+ validate do
13
+ value = __send__(attribute_name)
14
+
15
+ if value.is_a?(String)
16
+ begin
17
+ JSON.parse(value)
18
+ rescue JSON::ParserError
19
+ errors.add(attribute_name, :invalid)
20
+ end
21
+ end
22
+ end
23
+
24
+ define_method(attribute_name) do
25
+ value = super()
26
+
27
+ if value.is_a?(String) && value.present?
28
+ JSON.parse(value)
29
+ else
30
+ value
31
+ end
32
+ rescue JSON::ParserError
33
+ super()
34
+ end
35
+
36
+ define_method(:"#{attribute_name}=") do |new_value|
37
+ if new_value.is_a?(Hash)
38
+ super(JSON.generate(new_value))
39
+ elsif new_value.is_a?(String) && new_value.blank?
40
+ super(nil)
41
+ else
42
+ super(new_value)
43
+ end
44
+ rescue JSON::ParserError
45
+ super(new_value)
46
+ end
47
+ end
48
+ end
49
+ end
@@ -0,0 +1,2 @@
1
+ class ActiveRecordAuditable::ApplicationService < ServicePattern::Service
2
+ end
@@ -0,0 +1,19 @@
1
+ class ActiveRecordAuditable::AuditClasses < ActiveRecordAuditable::ApplicationService
2
+ def perform
3
+ Rails.application.eager_load!
4
+ succeed!(audit_classes:)
5
+ end
6
+
7
+ def audit_classes
8
+ @audit_classes ||= begin
9
+ audit_classes = []
10
+ audit_classes << ActiveRecordAuditable::Audit
11
+
12
+ ActiveRecordAuditable::BaseAudit.descendants.each do |klass|
13
+ audit_classes << klass
14
+ end
15
+
16
+ audit_classes
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,28 @@
1
+ class ActiveRecordAuditable::CreateAuditsTableForModelClass < ActiveRecordAuditable::ApplicationService
2
+ arguments :model_class
3
+ argument :create_table_args, default: nil
4
+ argument :extra_table_actions, default: nil
5
+ argument :id_type, default: nil
6
+
7
+ def perform
8
+ create_args = create_table_args || {}
9
+
10
+ type_args = {}
11
+ type_args[:type] = id_type if id_type
12
+
13
+ ActiveRecord::Migration.new.create_table table_name, **create_args do |t|
14
+ t.references model_class.model_name.param_key.to_sym, null: false, **type_args
15
+ t.json :audited_changes
16
+ t.references :audit_action, foreign_key: true, null: false, **type_args
17
+ t.json :params
18
+ extra_table_actions.call(t) if extra_table_actions
19
+ t.timestamps
20
+ end
21
+
22
+ succeed!
23
+ end
24
+
25
+ def table_name
26
+ @table_name ||= "#{model_class.table_name.singularize}_audits"
27
+ end
28
+ end
@@ -0,0 +1,70 @@
1
+ class ActiveRecordAuditable::MigrateAuditsFromSharedTable < ActiveRecordAuditable::ApplicationService
2
+ arguments :model_class
3
+
4
+ def perform
5
+ while audits_in_general_table_exists?
6
+ sql = "SELECT * FROM audits WHERE auditable_type = '#{model_class.name}' LIMIT 1000"
7
+ records = model_class.connection.execute(sql).to_a(as: :hash)
8
+
9
+ insert_sql = "INSERT INTO `#{new_table_name}` ("
10
+ insert_sql << "`#{model_class.model_name.param_key}_id`"
11
+ delete_sql = "DELETE FROM `audits` WHERE `#{id_column_name}` IN ("
12
+ records_count = 0
13
+
14
+ records.first.keys.each do |key|
15
+ next if key == "auditable_id" || key == "auditable_type" || key == "audit_auditable_type_id"
16
+
17
+ key = "current_user_id" if key == "user_id"
18
+ insert_sql << ", `#{key}`"
19
+ end
20
+
21
+
22
+ insert_sql << ") VALUES "
23
+ inserts = []
24
+
25
+ records.each_with_index do |data, data_index|
26
+ delete_sql << ", " if data_index > 0
27
+ delete_sql << model_class.connection.quote(data.fetch(id_column_name))
28
+
29
+ insert_sql << ", " if data_index > 0
30
+ insert_sql << "("
31
+ insert_sql << model_class.connection.quote(data.fetch("auditable_id"))
32
+
33
+ data.each do |key, value|
34
+ next if key == "auditable_id" || key == "auditable_type" || key == "audit_auditable_type_id"
35
+
36
+ insert_sql << ", "
37
+ insert_sql << model_class.connection.quote(value)
38
+ end
39
+
40
+ insert_sql << ")"
41
+ records_count += 1
42
+
43
+ break if insert_sql.bytesize >= 5.megabytes
44
+ end
45
+
46
+ delete_sql << ")"
47
+
48
+ ActiveRecordAuditable::Audit.transaction do
49
+ create_result = ActiveRecordAuditable::Audit.connection.execute(insert_sql)
50
+ delete_result = ActiveRecordAuditable::Audit.connection.execute(delete_sql)
51
+ end
52
+ end
53
+
54
+ succeed!
55
+ end
56
+
57
+ def audits_in_general_table_exists?
58
+ sql = "SELECT 1 FROM audits WHERE auditable_type = '#{model_class.name}' LIMIT 1"
59
+ results = model_class.connection.execute(sql).first
60
+ results&.length&.positive?
61
+ end
62
+
63
+ def id_column_name
64
+ @id_column_name ||= model_class.primary_key
65
+ end
66
+
67
+ def new_table_name
68
+ @new_table_name ||= "#{model_class.table_name.singularize}_audits"
69
+ end
70
+ end
@@ -0,0 +1,15 @@
1
+ <!DOCTYPE html>
2
+ <html>
3
+ <head>
4
+ <title>Active record auditable</title>
5
+ <%= csrf_meta_tags %>
6
+ <%= csp_meta_tag %>
7
+
8
+ <%= stylesheet_link_tag "active_record_auditable/application", media: "all" %>
9
+ </head>
10
+ <body>
11
+
12
+ <%= yield %>
13
+
14
+ </body>
15
+ </html>
data/config/routes.rb ADDED
@@ -0,0 +1,2 @@
1
+ ActiveRecordAuditable::Engine.routes.draw do
2
+ end
@@ -0,0 +1,8 @@
1
+ class CreateAuditActions < ActiveRecord::Migration[7.1]
2
+ def change
3
+ create_table :audit_actions do |t|
4
+ t.string :action, index: {unique: true}, null: false
5
+ t.timestamps
6
+ end
7
+ end
8
+ end
@@ -0,0 +1,8 @@
1
+ class CreateAuditAuditableTypes < ActiveRecord::Migration[7.1]
2
+ def change
3
+ create_table :audit_auditable_types do |t|
4
+ t.string :name, index: {unique: true}, null: false
5
+ t.timestamps
6
+ end
7
+ end
8
+ end
@@ -0,0 +1,12 @@
1
+ class CreateAudits < ActiveRecord::Migration[7.1]
2
+ def change
3
+ create_table :audits do |t|
4
+ t.references :audit_action, foreign_key: true, null: false
5
+ t.references :audit_auditable_type, foreign_key: true, null: false
6
+ t.references :auditable, null: false, polymorphic: true
7
+ t.references :user, polymorphic: true
8
+ t.json :audited_changes
9
+ t.timestamps
10
+ end
11
+ end
12
+ end
@@ -0,0 +1,5 @@
1
+ class AddParamsToAudits < ActiveRecord::Migration[7.1]
2
+ def change
3
+ add_column :audits, :params, :json
4
+ end
5
+ end
@@ -0,0 +1,153 @@
1
+ module ActiveRecordAuditable::Audited
2
+ def self.__cached_audit_table_names
3
+ @__cached_audit_table_names ||= begin
4
+ result = {}
5
+ ActiveRecordAuditable::Audit.connection.tables.filter { |table_name| table_name.ends_with?("_audits") }.each do |table_name|
6
+ result[table_name] = true
7
+ end
8
+ result
9
+ end
10
+ end
11
+
12
+ def self.included(base)
13
+ dedicated_table_name = "#{base.table_name.singularize}_audits"
14
+ dedicated_table_exists = __dedicated_table_exists?(base, dedicated_table_name)
15
+
16
+ if dedicated_table_exists
17
+ table_name = dedicated_table_name
18
+ audit_class = __dedicated_audit_class(base, table_name)
19
+ audit_class_name = "#{base.name}::Audit"
20
+ inverse_of = base.model_name.param_key.to_sym
21
+ as = inverse_of
22
+ else
23
+ table_name = "audits"
24
+ audit_class = ActiveRecordAuditable::Audit
25
+ audit_class_name = "ActiveRecordAuditable::Audit"
26
+ inverse_of = :auditable
27
+ as = :auditable
28
+ end
29
+
30
+ base.has_one :create_audit, # rubocop:disable Rails/HasManyOrHasOneDependent
31
+ -> { joins(:audit_action).where(audit_actions: {action: "create"}) },
32
+ as: as,
33
+ class_name: audit_class_name,
34
+ inverse_of: inverse_of
35
+ base.has_many :audits, # rubocop:disable Rails/HasManyOrHasOneDependent
36
+ as: as,
37
+ class_name: audit_class_name,
38
+ inverse_of: inverse_of
39
+
40
+ base.has_one :create_audit, # rubocop:disable Rails/HasManyOrHasOneDependent
41
+ -> { joins(:audit_action).where(audit_actions: {action: "create"}) },
42
+ class_name: audit_class_name,
43
+ inverse_of: inverse_of
44
+ base.has_many :audits, # rubocop:disable Rails/HasManyOrHasOneDependent
45
+ class_name: audit_class_name,
46
+ inverse_of: inverse_of
47
+
48
+ base.after_create do
49
+ create_audit!(action: :create)
50
+ end
51
+
52
+ base.after_update do
53
+ create_audit!(action: :update)
54
+ end
55
+
56
+ base.after_destroy do
57
+ create_audit!(action: :destroy, audited_changes: attributes)
58
+ end
59
+
60
+ base.scope :without_audit, lambda { |action|
61
+ audit_class = base.reflections["audits"].klass
62
+ audit_foreign_key = base.reflections["audits"].foreign_key
63
+
64
+ audit_query = audit_class
65
+ .select(1)
66
+ .joins(:audit_action)
67
+ .where(audit_actions: {action:})
68
+ .where("#{audit_class.table_name}.#{audit_foreign_key} = #{base.table_name}.#{base.primary_key}")
69
+ .limit(1)
70
+
71
+ audit_query = audit_query.where(auditable_type: base.model_name.name) unless dedicated_table_exists
72
+ where("NOT EXISTS (#{audit_query.to_sql})")
73
+ }
74
+ end
75
+
76
+ def self.__dedicated_table_exists?(base, dedicated_table_name)
77
+ ActiveRecordAuditable::Audited.__cached_audit_table_names.key?(dedicated_table_name)
78
+ rescue ActiveRecord::StatementInvalid
79
+ false
80
+ end
81
+
82
+ def self.__dedicated_audit_class(base, table_name)
83
+ if base.const_defined?("Audit")
84
+ base.const_get("Audit")
85
+ else
86
+ audit_class = Class.new(ActiveRecordAuditable::BaseAudit)
87
+ audit_class.class_eval do
88
+ self.table_name = table_name
89
+
90
+ belongs_to base.model_name.param_key.to_sym, optional: true
91
+ belongs_to :auditable, class_name: base.name, foreign_key: :"#{base.model_name.param_key}_id", optional: true
92
+
93
+ def self.base_model
94
+ reflections["auditable"].klass
95
+ end
96
+
97
+ def auditable_type
98
+ self.class.reflections["auditable"].class_name
99
+ end
100
+ end
101
+
102
+ base.const_set("Audit", audit_class)
103
+
104
+ ActiveRecordAuditable::AuditAction.has_many(audit_class.model_name.plural.to_sym, class_name: audit_class.name)
105
+ ActiveRecordAuditable::AuditAuditableType.has_many(audit_class.model_name.plural.to_sym, class_name: audit_class.name)
106
+
107
+ audit_class
108
+ end
109
+ end
110
+
111
+ def audit_monitor
112
+ @@audit_monitor ||= Monitor.new # rubocop:disable Style/ClassVars
113
+ end
114
+
115
+ def create_audit!(action:, audited_changes: saved_changes_for_audit, **args)
116
+ audit_data = {
117
+ audit_action: find_or_create_auditable_action(action),
118
+ audited_changes:
119
+ }
120
+
121
+ audit_class = self.class.reflections["audits"].klass
122
+
123
+ if audit_class == ActiveRecordAuditable::Audit
124
+ audit_data[:audit_auditable_type_id] = find_or_create_auditable_type.id
125
+ audit_data[:auditable_id] = id
126
+ audit_data[:auditable_type] = self.model_name.name
127
+ else
128
+ audit_data[:"#{self.class.model_name.param_key}_id"] = id
129
+ end
130
+
131
+ audit_class.create!(audit_data.merge(args))
132
+ end
133
+
134
+ def find_or_create_auditable_action(action)
135
+ audit_monitor.synchronize do
136
+ return ActiveRecordAuditable::AuditAction.find_or_create_by!(action:)
137
+ end
138
+ end
139
+
140
+ def find_or_create_auditable_type
141
+ audit_monitor.synchronize do
142
+ return ActiveRecordAuditable::AuditAuditableType.find_or_create_by!(name: self.class.name)
143
+ end
144
+ end
145
+
146
+ def saved_changes_for_audit
147
+ saved_changes_for_audit = {}
148
+ saved_changes.each do |attribute_name, from_and_to|
149
+ saved_changes_for_audit[attribute_name] = from_and_to[1]
150
+ end
151
+ saved_changes_for_audit
152
+ end
153
+ end
@@ -0,0 +1,5 @@
1
+ module ActiveRecordAuditable
2
+ class Engine < ::Rails::Engine
3
+ isolate_namespace ActiveRecordAuditable
4
+ end
5
+ end
@@ -0,0 +1,3 @@
1
+ module ActiveRecordAuditable
2
+ VERSION = "0.1.0"
3
+ end
@@ -0,0 +1,7 @@
1
+ require "active_record_auditable/version"
2
+ require "active_record_auditable/engine"
3
+ require "service_pattern"
4
+
5
+ module ActiveRecordAuditable
6
+ autoload :Audited, "#{__dir__}/active_record_auditable/audited"
7
+ end
@@ -0,0 +1,4 @@
1
+ # desc "Explaining what the task does"
2
+ # task :active_record_auditable do
3
+ # # Task goes here
4
+ # end
metadata ADDED
@@ -0,0 +1,104 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: active_record_auditable
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - kaspernj
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+ date: 2025-03-20 00:00:00.000000000 Z
12
+ dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: rails
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: 7.0.0
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ">="
25
+ - !ruby/object:Gem::Version
26
+ version: 7.0.0
27
+ - !ruby/object:Gem::Dependency
28
+ name: service_pattern
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: 1.0.10
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ">="
39
+ - !ruby/object:Gem::Version
40
+ version: 1.0.10
41
+ description: Audits for your records.
42
+ email:
43
+ - kasper@diestoeckels.de
44
+ executables: []
45
+ extensions: []
46
+ extra_rdoc_files: []
47
+ files:
48
+ - MIT-LICENSE
49
+ - README.md
50
+ - Rakefile
51
+ - app/assets/config/active_record_auditable_manifest.js
52
+ - app/assets/stylesheets/active_record_auditable/application.css
53
+ - app/controllers/active_record_auditable/application_controller.rb
54
+ - app/helpers/active_record_auditable/application_helper.rb
55
+ - app/jobs/active_record_auditable/application_job.rb
56
+ - app/mailers/active_record_auditable/application_mailer.rb
57
+ - app/models/active_record_auditable/application_record.rb
58
+ - app/models/active_record_auditable/audit.rb
59
+ - app/models/active_record_auditable/audit_action.rb
60
+ - app/models/active_record_auditable/audit_auditable_type.rb
61
+ - app/models/active_record_auditable/base_audit.rb
62
+ - app/models/concerns/active_record_auditable/acts_as_json.rb
63
+ - app/services/active_record_auditable/application_service.rb
64
+ - app/services/active_record_auditable/audit_classes.rb
65
+ - app/services/active_record_auditable/create_audits_table_for_model_class.rb
66
+ - app/services/active_record_auditable/migrate_audits_from_shared_table.rb
67
+ - app/views/layouts/active_record_auditable/application.html.erb
68
+ - config/routes.rb
69
+ - db/migrate/20240307212702_create_audit_actions.rb
70
+ - db/migrate/20240307212756_create_audit_auditable_types.rb
71
+ - db/migrate/20240307212826_create_audits.rb
72
+ - db/migrate/20240609073430_add_params_to_audits.rb
73
+ - lib/active_record_auditable.rb
74
+ - lib/active_record_auditable/audited.rb
75
+ - lib/active_record_auditable/engine.rb
76
+ - lib/active_record_auditable/version.rb
77
+ - lib/tasks/active_record_auditable_tasks.rake
78
+ homepage: https://github.com/kaspernj/active_record_auditable
79
+ licenses:
80
+ - MIT
81
+ metadata:
82
+ homepage_uri: https://github.com/kaspernj/active_record_auditable
83
+ source_code_uri: https://github.com/kaspernj/active_record_auditable
84
+ changelog_uri: https://github.com/kaspernj/active_record_auditable/blob/master/CHANGELOG.md
85
+ post_install_message:
86
+ rdoc_options: []
87
+ require_paths:
88
+ - lib
89
+ required_ruby_version: !ruby/object:Gem::Requirement
90
+ requirements:
91
+ - - ">="
92
+ - !ruby/object:Gem::Version
93
+ version: '0'
94
+ required_rubygems_version: !ruby/object:Gem::Requirement
95
+ requirements:
96
+ - - ">="
97
+ - !ruby/object:Gem::Version
98
+ version: '0'
99
+ requirements: []
100
+ rubygems_version: 3.4.19
101
+ signing_key:
102
+ specification_version: 4
103
+ summary: Audits for your records.
104
+ test_files: []