modelscope 0.0.1 → 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.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: bce912a368e26f345cb884efa8e2a812c8ca58dae75d856cf6aba131bc3c6d9b
4
- data.tar.gz: 0fc51f66a85f939a811e851b061d95b9bc64bee134b9eb5c39dde04dca4a14a1
3
+ metadata.gz: 797e5f9be7613c91445fe7079db6889653f1807efb0c65fd571bdd4371e3dba9
4
+ data.tar.gz: 12c425ccff813c02fcf3570cbb7da963da2b933397af1931460e27b51373771c
5
5
  SHA512:
6
- metadata.gz: d1fbf5acaa0f20d6e47f44712f3d72a0f22b6ab297156fa7fce87bfe16b157166cacf3befaaa3e098c0e5005bd9e5f475ee45d15003b742523839dfa16e5985d
7
- data.tar.gz: fcbda96d45601414867abfe42d86e12b4e7441f3fd07d6a401ca4004dd7fa30b460fcf7a64af46a6be3529ba9a90fbdbb676391dd563acdf341c7a582053f9f1
6
+ metadata.gz: 78a6d989e9fe646b473b2e4518152ff77dd32b4d0b3adf2d29fe8bf6d86a395d068f8f4193d484ae748d86ecc7453fc0cb8d2376e52e3972c27791a0660a9030
7
+ data.tar.gz: 307b3ee5b3fee4df6fcd06b7a52d4606cddf409f75faafabc1a5409bd64f892441f9742f4d22fc2009585acd5c5fd4645a2b3d45921d0c2dd19fcfbfed25bb24
data/README.md CHANGED
@@ -1,3 +1,85 @@
1
+ ## Description
2
+
3
+ Build an initial implementation of Model Scope.
4
+
5
+ ### Example usage
6
+
7
+ As an end user, I'd like to install a gem and run a Rake task to get the report:
8
+
9
+ ```sh
10
+ $ bundle add modelscope --group development
11
+
12
+ $ bin/rails modelscope:callbacks
13
+
14
+ Model Scope callbacks report:
15
+
16
+ Model | Kind | Total | Own | Inherited | Rails | Gems | Conditional |
17
+ ------|--------------|-------|-----|-----------|-------|------|-------------|
18
+ User | all | 23 | 5 | 2 | 8 | 8 | 21 |
19
+ | b/validation | 3 | 1 | 0 | 1 | 1 | 2 |
20
+ | a/save | 13 | 5 | 4 | 3 | 2 | 4 |
21
+
22
+ # ...
23
+ ```
24
+
25
+ The `rails modelscope:validations` task works in a similar fashion.
26
+
27
+ The **key features** in the example:
28
+
29
+ - The library should work without requiring adding any custom code; just adding the gem to the bundle should be enough to use it.
30
+ - For each model we show the total number of callbacks in two dimensions: callback kind (`before_create`, `after_save`, etc.) and **origin distributions** (where the callback is defined). We also count conditional callbacks separately.
31
+
32
+ ### References
33
+
34
+ There is an outdated project called [arca](https://github.com/jonmagic/arca), which has some code (for older Rails versions) to get the callbacks information from models.
35
+
36
+ Feel free to re-use its parts if you find them useful.
37
+
38
+ ## Tasks
39
+
40
+ - [x] Research the possibility of extending the set of parameters to analyze callbacks (right now we have "kind", "origin", "conditional" — what else could be useful?)
41
+ - [x] Implement core functionality for callbacks (an API which accepts a model class and returns all the data)
42
+ - [x] Come up with the set of paramters for validations — what information can we get from Active Record, which could be useful?
43
+ - [x] Implement core functionality for validations
44
+ - [x] Implement a Rake task with table-like output for console (it's okay to use gems to format the output — we aim for a good UX)
45
+ - [x] Don't forget about tests/CI at all stages.
46
+
47
+ ## Future considerations
48
+
49
+ Please, keep in mind the potential future extensions when designing a library:
50
+
51
+ - Different output formats/modes support (JSON, Web UI, etc.)
52
+ - Ability to filter models by name.
53
+ - Modular architecture support (multiple `app/models` directories).
54
+
55
+ ---
56
+
57
+ # Development log
58
+
59
+ Structure: Runner runs the task, Collector collects Callback's, Callback represends a single "find", Stats does the stats for reports if needed, Report formats the existing data (several kinds of reports).
60
+
61
+ Table report done via `terminal-table`.
62
+
63
+ Added the `line` report format for debugging. Also added `github` report format for GitHub Actions (reference: Rubocop).
64
+
65
+ `model` parameter for the rake tasks supports both class names and relative file names.
66
+
67
+ Now supporting `paths` parameter for extra paths, and auto loading models from Rails Engines.
68
+
69
+ Integration specs are done using checking the output of `line` formatter.
70
+
71
+ Unit tests are simple, the actual logic is in the integration tests.
72
+
73
+ Re: "Research the possibility of extending the set of parameters to analyze callbacks" added two categories of callbacks: caused by attributes, and caused by associations.
74
+
75
+ Added `rake modelscope:validations` that heavily reuses existing code.
76
+
77
+ Using custom matchers to check for callbacks. And custom matchers for validations. Lots of integration tests.
78
+
79
+ TODO: rename. callback_hell is a nice name. modelscope is taken.
80
+
81
+ ---
82
+
1
83
  [![Gem Version](https://badge.fury.io/rb/modelscope.svg)](https://rubygems.org/gems/modelscope) [![Build](https://github.com/evilmartians/modelscope/workflows/Build/badge.svg)](https://github.com/evilmartians/modelscope/actions)
2
84
  [![JRuby Build](https://github.com/evilmartians/modelscope/workflows/JRuby%20Build/badge.svg)](https://github.com/evilmartians/modelscope/actions)
3
85
 
@@ -0,0 +1,167 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ module Analyzers
5
+ class CallbackAnalyzer
6
+ RAILS_GEMS = %w[
7
+ actioncable actionmailbox actionmailer actionpack actiontext
8
+ actionview activejob activemodel activerecord activestorage
9
+ activesupport railties
10
+ ].freeze
11
+
12
+ RAILS_ATTRIBUTE_OWNERS = [
13
+ defined?(ActiveRecord::Normalization) ? ActiveRecord::Normalization : ActiveModel::Attributes::Normalization,
14
+ ActiveRecord::Encryption::EncryptableRecord
15
+ ].freeze
16
+
17
+ def initialize(callback, model, defining_class)
18
+ @callback = callback
19
+ @model = model
20
+ @defining_class = defining_class
21
+ @filter = callback.filter
22
+ end
23
+
24
+ def origin
25
+ if rails_callback?
26
+ :rails
27
+ elsif external_class?
28
+ :gems
29
+ elsif !@filter.is_a?(Symbol)
30
+ :own
31
+ else
32
+ external_method?(callback_method) ? :gems : :own
33
+ end
34
+ end
35
+
36
+ def inherited?
37
+ owner = callback_owner
38
+ return false unless owner
39
+
40
+ if validator?(@filter)
41
+ @model != @defining_class
42
+ else
43
+ owner != @model &&
44
+ !rails_module?(owner) &&
45
+ owner != @defining_class
46
+ end
47
+ end
48
+
49
+ def conditional?
50
+ [@callback.instance_variable_get(:@if),
51
+ @callback.instance_variable_get(:@unless)].any? do |condition|
52
+ next false if condition.nil?
53
+ [*condition].any? { |c| c.is_a?(Symbol) || c.is_a?(Proc) }
54
+ end
55
+ end
56
+
57
+ def association_generated?
58
+ generated_by_module?("GeneratedAssociationMethods") ||
59
+ from_rails_path?(%r{/active_record/(autosave_association\.rb|associations/builder)}) ||
60
+ ValidationAnalyzer.belongs_to_validator?(@filter, @model)
61
+ end
62
+
63
+ def attribute_generated?
64
+ generated_by_module?("GeneratedAttributeMethods") ||
65
+ generated_by_rails_attributes? ||
66
+ from_rails_path?("active_record/attribute_methods/")
67
+ end
68
+
69
+ private
70
+
71
+ def rails_callback?
72
+ ValidationAnalyzer.belongs_to_validator?(@filter, @model) || standard_rails_callback?
73
+ end
74
+
75
+ def standard_rails_callback?
76
+ case @filter
77
+ when Symbol, Proc then from_rails_path?
78
+ else @defining_class == ApplicationRecord
79
+ end
80
+ end
81
+
82
+ def callback_owner
83
+ @callback_owner ||= determine_owner
84
+ end
85
+
86
+ def determine_owner
87
+ case @filter
88
+ when Symbol then callback_method&.owner
89
+ when Proc then nil
90
+ when ActiveModel::Validator, ActiveModel::EachValidator then @defining_class
91
+ else @filter.class
92
+ end
93
+ end
94
+
95
+ def callback_method
96
+ return nil unless @filter.is_a?(Symbol) || @filter.is_a?(String)
97
+
98
+ @callback_method ||= begin
99
+ @model.instance_method(@filter)
100
+ rescue
101
+ nil
102
+ end
103
+ end
104
+
105
+ def source_location
106
+ @source_location ||= case @filter
107
+ when Symbol, String then callback_method&.source_location&.first
108
+ when Proc then @filter.source_location&.first
109
+ end.to_s
110
+ end
111
+
112
+ def external_class?
113
+ @defining_class != @model && !@model.ancestors.include?(@defining_class)
114
+ end
115
+
116
+ def external_method?(method)
117
+ return false unless method
118
+
119
+ source = method.source_location&.first.to_s
120
+ !from_app_path?(source)
121
+ end
122
+
123
+ def from_app_path?(path)
124
+ path.start_with?(Rails.root.to_s) &&
125
+ !path.start_with?(Rails.root.join("vendor").to_s)
126
+ end
127
+
128
+ def generated_by_module?(suffix)
129
+ callback_method&.owner&.name&.end_with?("::" + suffix) || false
130
+ end
131
+
132
+ def generated_by_rails_attributes?
133
+ method = callback_method
134
+ return false unless method
135
+
136
+ RAILS_ATTRIBUTE_OWNERS.include?(method.owner)
137
+ end
138
+
139
+ def from_rails_path?(subpath = nil)
140
+ return false if source_location.empty?
141
+
142
+ rails_paths.any? do |rails_path|
143
+ case subpath
144
+ when String
145
+ source_location.include?("/#{subpath}")
146
+ when Regexp
147
+ source_location.match?(subpath)
148
+ else
149
+ source_location.include?(rails_path)
150
+ end
151
+ end
152
+ end
153
+
154
+ def rails_paths
155
+ @rails_paths ||= RAILS_GEMS.map { |name| Gem::Specification.find_by_name(name).full_gem_path }
156
+ end
157
+
158
+ def rails_module?(mod)
159
+ mod.name&.start_with?("ActiveRecord::", "ActiveModel::")
160
+ end
161
+
162
+ def validator?(obj)
163
+ obj.is_a?(ActiveModel::Validator)
164
+ end
165
+ end
166
+ end
167
+ end
@@ -0,0 +1,82 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ module Analyzers
5
+ class ValidationAnalyzer
6
+ STANDARD_VALIDATIONS = %w[
7
+ presence uniqueness format length inclusion exclusion
8
+ numericality acceptance confirmation
9
+ ].freeze
10
+
11
+ STANDARD_VALIDATION_PATTERN = /^validates?_(#{STANDARD_VALIDATIONS.join("|")})(?:_of)?$/
12
+
13
+ class << self
14
+ def belongs_to_validator?(filter, model)
15
+ presence_validator?(filter) &&
16
+ association_attribute?(filter.attributes.first, model, :belongs_to)
17
+ end
18
+
19
+ def detect_type(filter, model)
20
+ return nil unless filter
21
+
22
+ if belongs_to_validator?(filter, model)
23
+ "associated"
24
+ elsif validator?(filter)
25
+ validator_type(filter)
26
+ else
27
+ normalize_validation_name(filter.to_s)
28
+ end
29
+ end
30
+
31
+ def human_method_name(filter)
32
+ case filter
33
+ when Proc then format_proc_location(filter)
34
+ when Class, ActiveModel::Validator then format_validator(filter)
35
+ else filter.to_s
36
+ end
37
+ end
38
+
39
+ private
40
+
41
+ def presence_validator?(filter)
42
+ filter.is_a?(ActiveRecord::Validations::PresenceValidator)
43
+ end
44
+
45
+ def association_attribute?(attribute, model, macro)
46
+ model.reflect_on_association(attribute)&.macro == macro
47
+ end
48
+
49
+ def validator?(obj)
50
+ obj.class <= ActiveModel::EachValidator
51
+ end
52
+
53
+ def validator_type(validator)
54
+ validator.class.name.demodulize.sub("Validator", "").underscore
55
+ end
56
+
57
+ def normalize_validation_name(name)
58
+ case name
59
+ when STANDARD_VALIDATION_PATTERN, /^validate_(#{STANDARD_VALIDATIONS.join("|")})$/
60
+ $1
61
+ when /associated_records_for_/
62
+ "associated"
63
+ else
64
+ "custom"
65
+ end
66
+ end
67
+
68
+ def format_proc_location(proc)
69
+ location = proc.source_location
70
+ return "Proc (unknown location)" unless location
71
+
72
+ file = location.first.split("/").last(2).join("/")
73
+ "Proc (#{file}:#{location.last})"
74
+ end
75
+
76
+ def format_validator(validator)
77
+ "#{validator.class.name.split("::").last} (#{validator.attributes.join(", ")})"
78
+ end
79
+ end
80
+ end
81
+ end
82
+ end
@@ -0,0 +1,38 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ class Callback
5
+ attr_reader :model, :method_name, :conditional, :origin, :inherited, :kind,
6
+ :association_generated, :attribute_generated
7
+
8
+ def initialize(model:, rails_callback:, name:, defining_class:)
9
+ @model = model
10
+ @callback = rails_callback
11
+ @name = name
12
+ @defining_class = defining_class
13
+
14
+ analyzer = Analyzers::CallbackAnalyzer.new(@callback, model, defining_class)
15
+
16
+ @kind = @callback.kind
17
+ @method_name = @callback.filter
18
+ @conditional = analyzer.conditional?
19
+ @origin = analyzer.origin
20
+ @inherited = analyzer.inherited?
21
+ @association_generated = analyzer.association_generated?
22
+ @attribute_generated = analyzer.attribute_generated?
23
+ end
24
+
25
+ def callback_group
26
+ @name.to_s
27
+ end
28
+
29
+ def validation_type
30
+ return nil unless callback_group == "validate"
31
+ Analyzers::ValidationAnalyzer.detect_type(@callback.filter, model)
32
+ end
33
+
34
+ def human_method_name
35
+ Analyzers::ValidationAnalyzer.human_method_name(@callback.filter)
36
+ end
37
+ end
38
+ end
@@ -0,0 +1,69 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "set"
4
+
5
+ module ModelScope
6
+ class Collector
7
+ attr_reader :models
8
+
9
+ def initialize(models = nil, paths: nil, kind: :callbacks)
10
+ @paths = paths
11
+ @kind = kind
12
+
13
+ eager_load!
14
+ @models = Set.new(models ? [*models] : ApplicationRecord.descendants)
15
+ end
16
+
17
+ def collect
18
+ models.flat_map { |model| collect_for_model(model) }
19
+ end
20
+
21
+ private
22
+
23
+ def eager_load!
24
+ Rails.application.eager_load!
25
+ load_additional_paths
26
+ end
27
+
28
+ def collect_for_model(model)
29
+ model.ancestors.select { |ancestor| ancestor < ActiveRecord::Base }
30
+ .flat_map { |ancestor| collect_callbacks_for_class(model, ancestor) }
31
+ end
32
+
33
+ def collect_callbacks_for_class(model, klass)
34
+ callbacks = klass.__callbacks
35
+ callbacks = callbacks.slice(:validate) if @kind == :validations
36
+
37
+ callbacks.flat_map do |kind, chain|
38
+ chain.map { |callback| build_callback(model, callback, kind, klass) }
39
+ end
40
+ end
41
+
42
+ def build_callback(model, callback, kind, klass)
43
+ Callback.new(
44
+ model: model,
45
+ rails_callback: callback,
46
+ name: kind,
47
+ defining_class: klass
48
+ )
49
+ end
50
+
51
+ def load_additional_paths
52
+ model_paths.each do |path|
53
+ Dir[File.join(path, "**", "*.rb")].sort.each { |file| require_dependency file }
54
+ end
55
+ end
56
+
57
+ def model_paths
58
+ @model_paths ||= begin
59
+ paths = engine_paths
60
+ paths += [@paths] if @paths
61
+ paths.select(&:exist?)
62
+ end
63
+ end
64
+
65
+ def engine_paths
66
+ @engine_paths ||= Rails::Engine.subclasses.map { |engine| engine.root.join("app/models") }
67
+ end
68
+ end
69
+ end
@@ -1,6 +1,7 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module Modelscope # :nodoc:
4
- class Railtie < ::Rails::Railtie # :nodoc:
3
+ module ModelScope
4
+ class Railtie < ::Rails::Railtie
5
+ load "tasks/modelscope.rake"
5
6
  end
6
7
  end
@@ -0,0 +1,18 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ module Reports
5
+ class Base
6
+ attr_reader :callbacks
7
+
8
+ def initialize(callbacks)
9
+ @callbacks = callbacks
10
+ @stats = Stats.new(callbacks)
11
+ end
12
+
13
+ def generate
14
+ raise NotImplementedError
15
+ end
16
+ end
17
+ end
18
+ end
@@ -0,0 +1,29 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ module Reports
5
+ module Callbacks
6
+ class Github < GithubBase
7
+ private
8
+
9
+ def report_title
10
+ "Model Scope callbacks report"
11
+ end
12
+
13
+ def format_group_name(callback)
14
+ "#{timing_symbol(callback.kind)}/#{callback.callback_group}"
15
+ end
16
+
17
+ def timing_symbol(timing)
18
+ case timing
19
+ when :before, "before" then "⇥"
20
+ when :after, "after" then "↦"
21
+ when :around, "around" then "↔"
22
+ else
23
+ " "
24
+ end
25
+ end
26
+ end
27
+ end
28
+ end
29
+ end
@@ -0,0 +1,19 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ module Reports
5
+ module Callbacks
6
+ class Line < LineBase
7
+ private
8
+
9
+ def report_title
10
+ "Model Scope callbacks report:"
11
+ end
12
+
13
+ def format_callback_name(callback)
14
+ "#{callback.kind}_#{callback.callback_group}"
15
+ end
16
+ end
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,29 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ module Reports
5
+ module Callbacks
6
+ class Table < TableBase
7
+ private
8
+
9
+ def report_title
10
+ "Model Scope callbacks report"
11
+ end
12
+
13
+ def format_group_name(callback)
14
+ "#{timing_symbol(callback.kind)}/#{callback.callback_group}"
15
+ end
16
+
17
+ def timing_symbol(timing)
18
+ case timing
19
+ when :before, "before" then "⇥"
20
+ when :after, "after" then "↦"
21
+ when :around, "around" then "↔"
22
+ else
23
+ " "
24
+ end
25
+ end
26
+ end
27
+ end
28
+ end
29
+ end
@@ -0,0 +1,67 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ module Reports
5
+ class GithubBase < Base
6
+ def generate
7
+ output = ["::group::#{report_title}"]
8
+
9
+ @stats.by_model.each_with_index do |(model_name, _), index|
10
+ output << "" if index > 0
11
+ output << "::group::#{model_name}"
12
+
13
+ # Add total row first
14
+ total_stats = @stats.stats_for(model_name)
15
+ output << format_group(
16
+ "all",
17
+ total_stats[:total],
18
+ total_stats[:own],
19
+ total_stats[:inherited],
20
+ total_stats[:rails],
21
+ total_stats[:association_generated],
22
+ total_stats[:attribute_generated],
23
+ total_stats[:gems],
24
+ total_stats[:conditional]
25
+ )
26
+
27
+ # Group and sort callbacks
28
+ grouped_callbacks = @stats.by_model[model_name].group_by { |cb| format_group_name(cb) }
29
+
30
+ grouped_callbacks.keys.sort.each do |group_name|
31
+ group_callbacks = grouped_callbacks[group_name]
32
+ output << format_group(
33
+ group_name,
34
+ group_callbacks.size,
35
+ group_callbacks.count { |cb| cb.origin == :own },
36
+ group_callbacks.count(&:inherited),
37
+ group_callbacks.count { |cb| cb.origin == :rails },
38
+ group_callbacks.count(&:association_generated),
39
+ group_callbacks.count(&:attribute_generated),
40
+ group_callbacks.count { |cb| cb.origin == :gems },
41
+ group_callbacks.count(&:conditional)
42
+ )
43
+ end
44
+
45
+ output << "::endgroup::"
46
+ end
47
+
48
+ output << "::endgroup::"
49
+ output.join("\n")
50
+ end
51
+
52
+ private
53
+
54
+ def report_title
55
+ raise NotImplementedError
56
+ end
57
+
58
+ def format_group(name, total, own, inherited, rails, association_generated, attribute_generated, gems, conditional)
59
+ "::debug::kind=#{name} total=#{total} own=#{own} inherited=#{inherited} rails=#{rails} association_generated=#{association_generated} attribute_generated=#{attribute_generated} gems=#{gems} conditional=#{conditional}"
60
+ end
61
+
62
+ def format_group_name(callback)
63
+ raise NotImplementedError
64
+ end
65
+ end
66
+ end
67
+ end
@@ -0,0 +1,42 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ module Reports
5
+ class LineBase < Base
6
+ def generate
7
+ output = [report_title]
8
+
9
+ @stats.by_model.each do |model_name, model_callbacks|
10
+ output << "\n#{model_name}:"
11
+ model_callbacks.sort_by(&:kind).each do |callback|
12
+ output << format_callback(callback)
13
+ end
14
+ end
15
+
16
+ output.join("\n")
17
+ end
18
+
19
+ private
20
+
21
+ def report_title
22
+ raise NotImplementedError
23
+ end
24
+
25
+ def format_callback(callback)
26
+ [
27
+ " #{format_callback_name(callback)}",
28
+ "method_name: #{callback.human_method_name}",
29
+ "origin: #{callback.origin}",
30
+ "association_generated: #{callback.association_generated ? "yes" : "no"}",
31
+ "attribute_generated: #{callback.attribute_generated ? "yes" : "no"}",
32
+ "inherited: #{callback.inherited ? "inherited" : "own"}",
33
+ "conditional: #{callback.conditional ? "yes" : "no"}"
34
+ ].compact.join(", ")
35
+ end
36
+
37
+ def format_callback_name(callback)
38
+ raise NotImplementedError
39
+ end
40
+ end
41
+ end
42
+ end
@@ -0,0 +1,78 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "terminal-table"
4
+
5
+ module ModelScope
6
+ module Reports
7
+ class TableBase < Base
8
+ def generate
9
+ table = Terminal::Table.new do |t|
10
+ t.title = report_title
11
+ t.headings = ["Model", "Kind", "Total", "Own", "Inherited", "Rails", "Associations", "Attributes", "Gems", "Conditional"]
12
+ t.rows = generate_rows
13
+ end
14
+
15
+ table.to_s
16
+ end
17
+
18
+ private
19
+
20
+ def report_title
21
+ raise NotImplementedError
22
+ end
23
+
24
+ def generate_rows
25
+ rows = []
26
+ @stats.by_model.each_with_index do |(model_name, _), index|
27
+ rows << :separator if index > 0
28
+ rows.concat(model_rows(model_name))
29
+ end
30
+ rows
31
+ end
32
+
33
+ def model_rows(model_name)
34
+ rows = []
35
+ total_stats = @stats.stats_for(model_name)
36
+
37
+ # Add total row
38
+ rows << [
39
+ model_name,
40
+ "all",
41
+ total_stats[:total],
42
+ total_stats[:own],
43
+ total_stats[:inherited],
44
+ total_stats[:rails],
45
+ total_stats[:association_generated],
46
+ total_stats[:attribute_generated],
47
+ total_stats[:gems],
48
+ total_stats[:conditional]
49
+ ]
50
+
51
+ # Group and sort callbacks
52
+ grouped_callbacks = @stats.by_model[model_name].group_by { |cb| format_group_name(cb) }
53
+
54
+ grouped_callbacks.keys.sort.each do |group_name|
55
+ group_callbacks = grouped_callbacks[group_name]
56
+ rows << [
57
+ "",
58
+ group_name,
59
+ group_callbacks.size,
60
+ group_callbacks.count { |cb| cb.origin == :own },
61
+ group_callbacks.count(&:inherited),
62
+ group_callbacks.count { |cb| cb.origin == :rails },
63
+ group_callbacks.count(&:association_generated),
64
+ group_callbacks.count(&:attribute_generated),
65
+ group_callbacks.count { |cb| cb.origin == :gems },
66
+ group_callbacks.count(&:conditional)
67
+ ]
68
+ end
69
+
70
+ rows
71
+ end
72
+
73
+ def format_group_name(callback)
74
+ raise NotImplementedError
75
+ end
76
+ end
77
+ end
78
+ end
@@ -0,0 +1,19 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ module Reports
5
+ module Validations
6
+ class Github < GithubBase
7
+ private
8
+
9
+ def report_title
10
+ "Model Scope validations report"
11
+ end
12
+
13
+ def format_group_name(callback)
14
+ callback.validation_type
15
+ end
16
+ end
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,24 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ module Reports
5
+ module Validations
6
+ class Line < LineBase
7
+ private
8
+
9
+ def report_title
10
+ "Model Scope validations report:"
11
+ end
12
+
13
+ def format_callback_name(callback)
14
+ if callback.method_name.is_a?(Symbol) || callback.method_name.is_a?(String)
15
+ type = callback.validation_type
16
+ (type == "custom") ? "custom (#{callback.method_name})" : type
17
+ else
18
+ callback.validation_type
19
+ end
20
+ end
21
+ end
22
+ end
23
+ end
24
+ end
@@ -0,0 +1,19 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ module Reports
5
+ module Validations
6
+ class Table < TableBase
7
+ private
8
+
9
+ def report_title
10
+ "Model Scope validations report"
11
+ end
12
+
13
+ def format_group_name(callback)
14
+ callback.validation_type
15
+ end
16
+ end
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,59 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ class Runner
5
+ DEFAULT_FORMAT = :table
6
+
7
+ def self.run(format: DEFAULT_FORMAT, model: nil, paths: nil, kind: :callbacks)
8
+ new(format: format, model: model, paths: paths, kind: kind).run
9
+ end
10
+
11
+ def initialize(format:, model:, paths:, kind: :callbacks)
12
+ @format = (format || DEFAULT_FORMAT).to_sym
13
+ @model_name = model
14
+ @paths = paths
15
+ @kind = kind
16
+ end
17
+
18
+ def run
19
+ if @kind == :report
20
+ [:callbacks, :validations].map do |ckind|
21
+ generate_report(collect_callbacks(ckind), ckind)
22
+ end.join("\n\n")
23
+ else
24
+ generate_report(collect_callbacks(@kind), @kind)
25
+ end
26
+ end
27
+
28
+ private
29
+
30
+ def collect_callbacks(ckind)
31
+ Collector.new(find_model_class, paths: @paths, kind: ckind).collect
32
+ end
33
+
34
+ def generate_report(callbacks, ckind)
35
+ find_reporter_class(ckind).new(callbacks).generate
36
+ end
37
+
38
+ def find_reporter_class(ckind)
39
+ namespace = ckind.to_s.capitalize
40
+ format = @format.to_s.capitalize
41
+ class_name = "ModelScope::Reports::#{namespace}::#{format}"
42
+ class_name.constantize
43
+ rescue NameError
44
+ raise ModelScope::Error, "Unknown format: #{@format} for #{ckind}"
45
+ end
46
+
47
+ def find_model_class
48
+ return unless @model_name
49
+
50
+ if @model_name.match?(/^[A-Z]/)
51
+ @model_name.constantize
52
+ else
53
+ @model_name.classify.constantize
54
+ end
55
+ rescue NameError
56
+ raise ModelScope::Error, "Cannot find model: #{@model_name}"
57
+ end
58
+ end
59
+ end
@@ -0,0 +1,59 @@
1
+ # frozen_string_literal: true
2
+
3
+ module ModelScope
4
+ class Stats
5
+ COUNTERS = %i[
6
+ total own inherited rails gems conditional
7
+ association_generated attribute_generated
8
+ ].freeze
9
+
10
+ attr_reader :callbacks
11
+
12
+ def initialize(callbacks)
13
+ @callbacks = callbacks
14
+ @stats_cache = {}
15
+ end
16
+
17
+ def by_model
18
+ @by_model ||= callbacks.group_by { |cb| cb.model.name }
19
+ end
20
+
21
+ def stats_for(model_name)
22
+ @stats_cache[model_name] ||= begin
23
+ model_callbacks = by_model[model_name]
24
+ return {} unless model_callbacks
25
+
26
+ collect_stats(model_callbacks)
27
+ end
28
+ end
29
+
30
+ def stats_for_group(model_name, group)
31
+ key = "#{model_name}_#{group}"
32
+ @stats_cache[key] ||= begin
33
+ model_callbacks = by_model[model_name]&.select { |cb| cb.callback_group == group }
34
+ return {} unless model_callbacks
35
+
36
+ collect_stats(model_callbacks)
37
+ end
38
+ end
39
+
40
+ private
41
+
42
+ def collect_stats(callbacks)
43
+ callbacks.each_with_object(initial_stats) do |cb, stats|
44
+ stats[:total] += 1
45
+ stats[:own] += 1 if cb.origin == :own
46
+ stats[:inherited] += 1 if cb.inherited
47
+ stats[:rails] += 1 if cb.origin == :rails
48
+ stats[:gems] += 1 if cb.origin == :gems
49
+ stats[:conditional] += 1 if cb.conditional
50
+ stats[:association_generated] += 1 if cb.association_generated
51
+ stats[:attribute_generated] += 1 if cb.attribute_generated
52
+ end
53
+ end
54
+
55
+ def initial_stats
56
+ COUNTERS.to_h { |counter| [counter, 0] }
57
+ end
58
+ end
59
+ end
@@ -1,5 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- module Modelscope # :nodoc:
4
- VERSION = "0.0.1"
3
+ module ModelScope
4
+ VERSION = "0.1.0"
5
5
  end
data/lib/modelscope.rb CHANGED
@@ -1,4 +1,19 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require "modelscope/version"
3
+ require "zeitwerk"
4
+
5
+ module ModelScope
6
+ class Error < StandardError; end
7
+
8
+ def self.loader # @private
9
+ @loader ||= Zeitwerk::Loader.for_gem.tap do |loader|
10
+ loader.ignore("#{__dir__}/tasks")
11
+ loader.inflector.inflect("modelscope" => "ModelScope")
12
+ loader.setup
13
+ end
14
+ end
15
+ end
16
+
17
+ ModelScope.loader
18
+
4
19
  require "modelscope/railtie" if defined?(Rails::Railtie)
@@ -0,0 +1,80 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "rake"
4
+ require "rails"
5
+
6
+ namespace :modelscope do
7
+ desc <<~DESC
8
+ Generate callbacks report for Active Record models.
9
+
10
+ Options:
11
+ format=table|line|github Report format (default: table)
12
+ model=ModelName Filter by model name (optional). Can be
13
+ specified as constant
14
+ name (MessageThread) or file path
15
+ (message_thread, admin/message_thread)
16
+ path=DIR1,DIR2 Additional model directories (comma-separated)
17
+
18
+ Examples:
19
+ # Show all models callbacks in table format (default)
20
+ rake modelscope:callbacks
21
+ DESC
22
+ task callbacks: :environment do
23
+ paths = ENV["path"]&.split(",")&.map { |p| Rails.root.join(p) }
24
+
25
+ puts ModelScope::Runner.run(
26
+ format: ENV["format"],
27
+ model: ENV["model"],
28
+ paths: paths,
29
+ kind: :callbacks
30
+ )
31
+ end
32
+
33
+ desc <<~DESC
34
+ Generate validations report for Active Record models.
35
+
36
+ Options:
37
+ format=table|line|github Report format (default: table)
38
+ model=ModelName Filter by model name (optional). Can be
39
+ specified as constant
40
+ name (MessageThread) or file path
41
+ (message_thread, admin/message_thread)
42
+ path=DIR1,DIR2 Additional model directories (comma-separated)
43
+
44
+ Examples:
45
+ # Show all models validations in table format (default)
46
+ rake modelscope:validations
47
+ DESC
48
+ task validations: :environment do
49
+ paths = ENV["path"]&.split(",")&.map { |p| Rails.root.join(p) }
50
+
51
+ puts ModelScope::Runner.run(
52
+ format: ENV["format"],
53
+ model: ENV["model"],
54
+ paths: paths,
55
+ kind: :validations
56
+ )
57
+ end
58
+
59
+ desc <<~DESC
60
+ Generate combined report (callbacks and validations) for Active Record models.
61
+
62
+ Options:
63
+ format=table|line|github Report format (default: table)
64
+ model=ModelName Filter by model name (optional). Can be
65
+ specified as constant
66
+ name (MessageThread) or file path
67
+ (message_thread, admin/message_thread)
68
+ path=DIR1,DIR2 Additional model directories (comma-separated)
69
+ DESC
70
+ task report: :environment do
71
+ paths = ENV["path"]&.split(",")&.map { |p| Rails.root.join(p) }
72
+
73
+ puts ModelScope::Runner.run(
74
+ format: ENV["format"],
75
+ model: ENV["model"],
76
+ paths: paths,
77
+ kind: :report
78
+ )
79
+ end
80
+ end
metadata CHANGED
@@ -1,43 +1,70 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: modelscope
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.0.1
4
+ version: 0.1.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Vladimir Dementyev
8
- autorequire:
9
8
  bindir: bin
10
9
  cert_chain: []
11
- date: 2023-02-17 00:00:00.000000000 Z
10
+ date: 2025-03-26 00:00:00.000000000 Z
12
11
  dependencies:
13
12
  - !ruby/object:Gem::Dependency
14
- name: bundler
13
+ name: rails
15
14
  requirement: !ruby/object:Gem::Requirement
16
15
  requirements:
17
16
  - - ">="
18
17
  - !ruby/object:Gem::Version
19
- version: '1.15'
20
- type: :development
18
+ version: '7.0'
19
+ type: :runtime
21
20
  prerelease: false
22
21
  version_requirements: !ruby/object:Gem::Requirement
23
22
  requirements:
24
23
  - - ">="
25
24
  - !ruby/object:Gem::Version
26
- version: '1.15'
25
+ version: '7.0'
27
26
  - !ruby/object:Gem::Dependency
28
- name: combustion
27
+ name: zeitwerk
28
+ requirement: !ruby/object:Gem::Requirement
29
+ requirements:
30
+ - - ">="
31
+ - !ruby/object:Gem::Version
32
+ version: '0'
33
+ type: :runtime
34
+ prerelease: false
35
+ version_requirements: !ruby/object:Gem::Requirement
36
+ requirements:
37
+ - - ">="
38
+ - !ruby/object:Gem::Version
39
+ version: '0'
40
+ - !ruby/object:Gem::Dependency
41
+ name: terminal-table
42
+ requirement: !ruby/object:Gem::Requirement
43
+ requirements:
44
+ - - ">="
45
+ - !ruby/object:Gem::Version
46
+ version: '0'
47
+ type: :runtime
48
+ prerelease: false
49
+ version_requirements: !ruby/object:Gem::Requirement
50
+ requirements:
51
+ - - ">="
52
+ - !ruby/object:Gem::Version
53
+ version: '0'
54
+ - !ruby/object:Gem::Dependency
55
+ name: bundler
29
56
  requirement: !ruby/object:Gem::Requirement
30
57
  requirements:
31
58
  - - ">="
32
59
  - !ruby/object:Gem::Version
33
- version: '1.1'
60
+ version: '2.4'
34
61
  type: :development
35
62
  prerelease: false
36
63
  version_requirements: !ruby/object:Gem::Requirement
37
64
  requirements:
38
65
  - - ">="
39
66
  - !ruby/object:Gem::Version
40
- version: '1.1'
67
+ version: '2.4'
41
68
  - !ruby/object:Gem::Dependency
42
69
  name: rake
43
70
  requirement: !ruby/object:Gem::Requirement
@@ -52,6 +79,34 @@ dependencies:
52
79
  - - ">="
53
80
  - !ruby/object:Gem::Version
54
81
  version: '13.0'
82
+ - !ruby/object:Gem::Dependency
83
+ name: combustion
84
+ requirement: !ruby/object:Gem::Requirement
85
+ requirements:
86
+ - - ">="
87
+ - !ruby/object:Gem::Version
88
+ version: '1.5'
89
+ type: :development
90
+ prerelease: false
91
+ version_requirements: !ruby/object:Gem::Requirement
92
+ requirements:
93
+ - - ">="
94
+ - !ruby/object:Gem::Version
95
+ version: '1.5'
96
+ - !ruby/object:Gem::Dependency
97
+ name: sqlite3
98
+ requirement: !ruby/object:Gem::Requirement
99
+ requirements:
100
+ - - ">="
101
+ - !ruby/object:Gem::Version
102
+ version: '1.5'
103
+ type: :development
104
+ prerelease: false
105
+ version_requirements: !ruby/object:Gem::Requirement
106
+ requirements:
107
+ - - ">="
108
+ - !ruby/object:Gem::Version
109
+ version: '1.5'
55
110
  - !ruby/object:Gem::Dependency
56
111
  name: rspec-rails
57
112
  requirement: !ruby/object:Gem::Requirement
@@ -78,18 +133,29 @@ files:
78
133
  - LICENSE.txt
79
134
  - README.md
80
135
  - lib/modelscope.rb
136
+ - lib/modelscope/analyzers/callback_analyzer.rb
137
+ - lib/modelscope/analyzers/validation_analyzer.rb
138
+ - lib/modelscope/callback.rb
139
+ - lib/modelscope/collector.rb
81
140
  - lib/modelscope/railtie.rb
141
+ - lib/modelscope/reports/base.rb
142
+ - lib/modelscope/reports/callbacks/github.rb
143
+ - lib/modelscope/reports/callbacks/line.rb
144
+ - lib/modelscope/reports/callbacks/table.rb
145
+ - lib/modelscope/reports/github_base.rb
146
+ - lib/modelscope/reports/line_base.rb
147
+ - lib/modelscope/reports/table_base.rb
148
+ - lib/modelscope/reports/validations/github.rb
149
+ - lib/modelscope/reports/validations/line.rb
150
+ - lib/modelscope/reports/validations/table.rb
151
+ - lib/modelscope/runner.rb
152
+ - lib/modelscope/stats.rb
82
153
  - lib/modelscope/version.rb
154
+ - lib/tasks/modelscope.rake
83
155
  homepage: http://github.com/evilmartians/modelscope
84
156
  licenses:
85
157
  - MIT
86
- metadata:
87
- bug_tracker_uri: http://github.com/evilmartians/modelscope/issues
88
- changelog_uri: https://github.com/palkan/modelscope/blob/master/CHANGELOG.md
89
- documentation_uri: http://github.com/evilmartians/modelscope
90
- homepage_uri: http://github.com/evilmartians/modelscope
91
- source_code_uri: http://github.com/evilmartians/modelscope
92
- post_install_message:
158
+ metadata: {}
93
159
  rdoc_options: []
94
160
  require_paths:
95
161
  - lib
@@ -97,15 +163,14 @@ required_ruby_version: !ruby/object:Gem::Requirement
97
163
  requirements:
98
164
  - - ">="
99
165
  - !ruby/object:Gem::Version
100
- version: '2.7'
166
+ version: '3.0'
101
167
  required_rubygems_version: !ruby/object:Gem::Requirement
102
168
  requirements:
103
169
  - - ">="
104
170
  - !ruby/object:Gem::Version
105
171
  version: '0'
106
172
  requirements: []
107
- rubygems_version: 3.4.6
108
- signing_key:
173
+ rubygems_version: 3.6.5
109
174
  specification_version: 4
110
175
  summary: Rails models analyzers
111
176
  test_files: []