modelscope 0.0.1 → 0.1.1

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: 4e40c1dcb806b8b59a48b18a239a7160b6c816e61a173896bc1f5b233395d884
4
+ data.tar.gz: 3cb1b59b808eaa01eba216b5e53c4b7286a8aa145fdca08697835a8716c20621
5
5
  SHA512:
6
- metadata.gz: d1fbf5acaa0f20d6e47f44712f3d72a0f22b6ab297156fa7fce87bfe16b157166cacf3befaaa3e098c0e5005bd9e5f475ee45d15003b742523839dfa16e5985d
7
- data.tar.gz: fcbda96d45601414867abfe42d86e12b4e7441f3fd07d6a401ca4004dd7fa30b460fcf7a64af46a6be3529ba9a90fbdbb676391dd563acdf341c7a582053f9f1
6
+ metadata.gz: 37802c37e0d1cb44f696eaea22c8531f1433e543a15ae23a5a3e283c0bb31f5bb26b948f3558d25a37aa64a81ca9ebcb1129b18aec9a4147b04500e62812b12f
7
+ data.tar.gz: a1c81b68ae3d3b3dfe8ff9108216ce5ec554de75724d061b8d99d1f6b1af87956f04328bf020eb3a0a71723c82ee32e89853667b60865673a9b6311d4feca8f5
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,158 @@
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
+ @model != @defining_class
38
+ end
39
+
40
+ def conditional?
41
+ [@callback.instance_variable_get(:@if),
42
+ @callback.instance_variable_get(:@unless)].any? do |condition|
43
+ next false if condition.nil?
44
+ [*condition].any? { |c| c.is_a?(Symbol) || c.is_a?(Proc) }
45
+ end
46
+ end
47
+
48
+ def association_generated?
49
+ generated_by_module?("GeneratedAssociationMethods") ||
50
+ from_rails_path?(%r{/active_record/(autosave_association\.rb|associations/builder)}) ||
51
+ ValidationAnalyzer.belongs_to_validator?(@filter, @model)
52
+ end
53
+
54
+ def attribute_generated?
55
+ generated_by_module?("GeneratedAttributeMethods") ||
56
+ generated_by_rails_attributes? ||
57
+ from_rails_path?("active_record/attribute_methods/")
58
+ end
59
+
60
+ private
61
+
62
+ def rails_callback?
63
+ ValidationAnalyzer.belongs_to_validator?(@filter, @model) || standard_rails_callback?
64
+ end
65
+
66
+ def standard_rails_callback?
67
+ case @filter
68
+ when Symbol, Proc then from_rails_path?
69
+ else @defining_class == ApplicationRecord
70
+ end
71
+ end
72
+
73
+ def callback_owner
74
+ @callback_owner ||= determine_owner
75
+ end
76
+
77
+ def determine_owner
78
+ case @filter
79
+ when Symbol then callback_method&.owner
80
+ when Proc then nil
81
+ when ActiveModel::Validator, ActiveModel::EachValidator then @defining_class
82
+ else @filter.class
83
+ end
84
+ end
85
+
86
+ def callback_method
87
+ return nil unless @filter.is_a?(Symbol) || @filter.is_a?(String)
88
+
89
+ @callback_method ||= begin
90
+ @model.instance_method(@filter)
91
+ rescue
92
+ nil
93
+ end
94
+ end
95
+
96
+ def source_location
97
+ @source_location ||= case @filter
98
+ when Symbol, String then callback_method&.source_location&.first
99
+ when Proc then @filter.source_location&.first
100
+ end.to_s
101
+ end
102
+
103
+ def external_class?
104
+ @defining_class != @model && !@model.ancestors.include?(@defining_class)
105
+ end
106
+
107
+ def external_method?(method)
108
+ return false unless method
109
+
110
+ source = method.source_location&.first.to_s
111
+ !from_app_path?(source)
112
+ end
113
+
114
+ def from_app_path?(path)
115
+ path.start_with?(Rails.root.to_s) &&
116
+ !path.start_with?(Rails.root.join("vendor").to_s)
117
+ end
118
+
119
+ def generated_by_module?(suffix)
120
+ callback_method&.owner&.name&.end_with?("::" + suffix) || false
121
+ end
122
+
123
+ def generated_by_rails_attributes?
124
+ method = callback_method
125
+ return false unless method
126
+
127
+ RAILS_ATTRIBUTE_OWNERS.include?(method.owner)
128
+ end
129
+
130
+ def from_rails_path?(subpath = nil)
131
+ return false if source_location.empty?
132
+
133
+ rails_paths.any? do |rails_path|
134
+ case subpath
135
+ when String
136
+ source_location.include?("/#{subpath}")
137
+ when Regexp
138
+ source_location.match?(subpath)
139
+ else
140
+ source_location.include?(rails_path)
141
+ end
142
+ end
143
+ end
144
+
145
+ def rails_paths
146
+ @rails_paths ||= RAILS_GEMS.map { |name| Gem::Specification.find_by_name(name).full_gem_path }
147
+ end
148
+
149
+ def rails_module?(mod)
150
+ mod.name&.start_with?("ActiveRecord::", "ActiveModel::")
151
+ end
152
+
153
+ def validator?(obj)
154
+ obj.is_a?(ActiveModel::Validator)
155
+ end
156
+ end
157
+ end
158
+ 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,56 @@
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, :callback, :defining_class,
7
+ :fingerprint
8
+
9
+ def initialize(model:, rails_callback:, name:, defining_class:)
10
+ @model = model
11
+ @callback = rails_callback
12
+ @name = name
13
+ @defining_class = defining_class
14
+
15
+ analyzer = Analyzers::CallbackAnalyzer.new(@callback, model, defining_class)
16
+
17
+ @kind = @callback.kind
18
+ @method_name = @callback.filter
19
+ @conditional = analyzer.conditional?
20
+ @origin = analyzer.origin
21
+ @inherited = analyzer.inherited?
22
+ @association_generated = analyzer.association_generated?
23
+ @attribute_generated = analyzer.attribute_generated?
24
+ # fingerprint allows us to de-duplicate callbacks/validations;
25
+ # in most cases, it's just an object_id, but for named validations/callbacks,
26
+ # it's a combination of the name, kind and the method_name.
27
+ # The "0" and "1" prefixes define how to handle duplicates (1 means last write wins, 0 means first write wins)
28
+ @fingerprint = (@method_name.is_a?(Symbol) && @origin != :rails) ? ["1", @name, @kind, @method_name].join("-") : "0-#{@callback.object_id}"
29
+ end
30
+
31
+ def callback_group
32
+ @name.to_s
33
+ end
34
+
35
+ def validation_type
36
+ return nil unless callback_group == "validate"
37
+ Analyzers::ValidationAnalyzer.detect_type(@callback.filter, model)
38
+ end
39
+
40
+ def human_method_name
41
+ Analyzers::ValidationAnalyzer.human_method_name(@callback.filter)
42
+ end
43
+
44
+ def to_s
45
+ [
46
+ "#{model.name}: #{human_method_name}",
47
+ "kind=#{kind}_#{callback_group}",
48
+ "origin=#{origin}",
49
+ inherited ? "inherited=true" : nil,
50
+ conditional ? "conditional=true" : nil,
51
+ association_generated ? "association=true" : nil,
52
+ attribute_generated ? "attribute=true" : nil
53
+ ].compact.join(" ")
54
+ end
55
+ end
56
+ end
@@ -0,0 +1,87 @@
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(select_models = models)
18
+ select_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
+ # collect from parent to child to correctly handle inheritance
31
+ .reverse
32
+ .flat_map { |ancestor| collect_callbacks_for_class(model, ancestor) }
33
+ .group_by(&:fingerprint)
34
+ # merge groups
35
+ .transform_values do |callbacks|
36
+ probe = callbacks.first
37
+ if probe.fingerprint.start_with?("1")
38
+ # we must keep the last non-matching callback (i.e., if all callbacks are the same,
39
+ # we must keep the first one)
40
+ callbacks.each do |clbk|
41
+ if clbk.callback != probe.callback
42
+ probe = clbk
43
+ end
44
+ end
45
+ end
46
+ probe
47
+ end
48
+ .values
49
+ end
50
+
51
+ def collect_callbacks_for_class(model, klass)
52
+ callbacks = klass.__callbacks
53
+ callbacks = callbacks.slice(:validate) if @kind == :validations
54
+
55
+ callbacks.flat_map do |kind, chain|
56
+ chain.map { |callback| build_callback(model, callback, kind, klass) }
57
+ end
58
+ end
59
+
60
+ def build_callback(model, callback, kind, klass)
61
+ Callback.new(
62
+ model: model,
63
+ rails_callback: callback,
64
+ name: kind,
65
+ defining_class: klass
66
+ )
67
+ end
68
+
69
+ def load_additional_paths
70
+ model_paths.each do |path|
71
+ Dir[File.join(path, "**", "*.rb")].sort.each { |file| require_dependency file }
72
+ end
73
+ end
74
+
75
+ def model_paths
76
+ @model_paths ||= begin
77
+ paths = engine_paths
78
+ paths += [@paths] if @paths
79
+ paths.select(&:exist?)
80
+ end
81
+ end
82
+
83
+ def engine_paths
84
+ @engine_paths ||= Rails::Engine.subclasses.map { |engine| engine.root.join("app/models") }
85
+ end
86
+ end
87
+ 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, **opts)
9
+ @callbacks = callbacks
10
+ @stats = Stats.new(callbacks, **opts)
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,61 @@
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, **opts)
8
+ new(format: format, model: model, paths: paths, kind: kind, **opts).run
9
+ end
10
+
11
+ def initialize(format:, model:, paths:, kind: :callbacks, sort_by: :size, sort_order: :desc)
12
+ @format = (format || DEFAULT_FORMAT).to_sym
13
+ @model_name = model
14
+ @paths = paths
15
+ @kind = kind
16
+ @sort_by = sort_by
17
+ @sort_order = sort_order
18
+ end
19
+
20
+ def run
21
+ if @kind == :report
22
+ [:callbacks, :validations].map do |ckind|
23
+ generate_report(collect_callbacks(ckind), ckind)
24
+ end.join("\n\n")
25
+ else
26
+ generate_report(collect_callbacks(@kind), @kind)
27
+ end
28
+ end
29
+
30
+ private
31
+
32
+ def collect_callbacks(ckind)
33
+ Collector.new(find_model_class, paths: @paths, kind: ckind).collect
34
+ end
35
+
36
+ def generate_report(callbacks, ckind)
37
+ find_reporter_class(ckind).new(callbacks, sort_by: @sort_by, sort_order: @sort_order).generate
38
+ end
39
+
40
+ def find_reporter_class(ckind)
41
+ namespace = ckind.to_s.capitalize
42
+ format = @format.to_s.capitalize
43
+ class_name = "ModelScope::Reports::#{namespace}::#{format}"
44
+ class_name.constantize
45
+ rescue NameError
46
+ raise ModelScope::Error, "Unknown format: #{@format} for #{ckind}"
47
+ end
48
+
49
+ def find_model_class
50
+ return unless @model_name
51
+
52
+ if @model_name.match?(/^[A-Z]/)
53
+ @model_name.constantize
54
+ else
55
+ @model_name.classify.constantize
56
+ end
57
+ rescue NameError
58
+ raise ModelScope::Error, "Cannot find model: #{@model_name}"
59
+ end
60
+ end
61
+ end
@@ -0,0 +1,75 @@
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
+ SORT = %i[size name].freeze
11
+ SORT_ORDER = %i[desc asc].freeze
12
+
13
+ attr_reader :callbacks
14
+ private attr_reader :sort_by, :sort_order
15
+
16
+ def initialize(callbacks, sort_by: :size, sort_order: :desc)
17
+ @callbacks = callbacks
18
+ @stats_cache = {}
19
+ @sort_by = sort_by
20
+ raise ArgumentError, "Invalid sort_by: #{@sort_by}. Available: #{SORT.join(", ")}" unless SORT.include?(@sort_by)
21
+ @sort_order = sort_order
22
+ raise ArgumentError, "Invalid sort_order: #{@sort_order}. Available: #{SORT_ORDER.join(", ")}" unless SORT_ORDER.include?(@sort_order)
23
+ end
24
+
25
+ def by_model
26
+ @by_model ||= callbacks.group_by { |cb| cb.model.name }.sort_by do |name, callbacks|
27
+ if sort_by == :size
28
+ callbacks.size
29
+ elsif sort_by == :name
30
+ name
31
+ end
32
+ end.tap do |sorted|
33
+ sorted.reverse! if sort_order == :desc
34
+ end.to_h
35
+ end
36
+
37
+ def stats_for(model_name)
38
+ @stats_cache[model_name] ||= begin
39
+ model_callbacks = by_model[model_name]
40
+ return {} unless model_callbacks
41
+
42
+ collect_stats(model_callbacks)
43
+ end
44
+ end
45
+
46
+ def stats_for_group(model_name, group)
47
+ key = "#{model_name}_#{group}"
48
+ @stats_cache[key] ||= begin
49
+ model_callbacks = by_model[model_name]&.select { |cb| cb.callback_group == group }
50
+ return {} unless model_callbacks
51
+
52
+ collect_stats(model_callbacks)
53
+ end
54
+ end
55
+
56
+ private
57
+
58
+ def collect_stats(callbacks)
59
+ callbacks.each_with_object(initial_stats) do |cb, stats|
60
+ stats[:total] += 1
61
+ stats[:own] += 1 if cb.origin == :own
62
+ stats[:inherited] += 1 if cb.inherited
63
+ stats[:rails] += 1 if cb.origin == :rails
64
+ stats[:gems] += 1 if cb.origin == :gems
65
+ stats[:conditional] += 1 if cb.conditional
66
+ stats[:association_generated] += 1 if cb.association_generated
67
+ stats[:attribute_generated] += 1 if cb.attribute_generated
68
+ end
69
+ end
70
+
71
+ def initial_stats
72
+ COUNTERS.to_h { |counter| [counter, 0] }
73
+ end
74
+ end
75
+ 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.1"
5
5
  end
data/lib/modelscope.rb CHANGED
@@ -1,4 +1,29 @@
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
+
16
+ def self.collect_callbacks(*models)
17
+ collector = Collector.new
18
+ collector.collect(models)
19
+ end
20
+
21
+ def self.collect_validations(*models)
22
+ collector = Collector.new(kind: :validations)
23
+ collector.collect(models)
24
+ end
25
+ end
26
+
27
+ ModelScope.loader
28
+
4
29
  require "modelscope/railtie" if defined?(Rails::Railtie)
@@ -0,0 +1,113 @@
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
+ sort=total|name Sort by score or name (default: score:desc)
17
+ path=DIR1,DIR2 Additional model directories (comma-separated)
18
+
19
+ Examples:
20
+ # Show all models callbacks in table format (default)
21
+ rake modelscope:callbacks
22
+ DESC
23
+ task callbacks: :environment do
24
+ opts = {
25
+ kind: :callbacks,
26
+ format: ENV["format"],
27
+ model: ENV["model"]
28
+ }.compact
29
+
30
+ if ENV["path"]
31
+ opts[:paths] = ENV["path"]&.split(",")&.map { |p| Rails.root.join(p) }
32
+ end
33
+
34
+ if ENV["sort"]
35
+ sort_by, sort_order = ENV["sort"].split(":")
36
+ opts[:sort_by] = sort_by.to_sym
37
+ sort_order ||= ((sort_by == "size") ? :desc : :asc)
38
+ opts[:sort_order] = sort_order.to_sym
39
+ end
40
+
41
+ puts ModelScope::Runner.run(**opts)
42
+ end
43
+
44
+ desc <<~DESC
45
+ Generate validations report for Active Record models.
46
+
47
+ Options:
48
+ format=table|line|github Report format (default: table)
49
+ model=ModelName Filter by model name (optional). Can be
50
+ specified as constant
51
+ name (MessageThread) or file path
52
+ (message_thread, admin/message_thread)
53
+ sort=total|name Sort by score or name (default: score:desc)
54
+ path=DIR1,DIR2 Additional model directories (comma-separated)
55
+
56
+ Examples:
57
+ # Show all models validations in table format (default)
58
+ rake modelscope:validations
59
+ DESC
60
+ task validations: :environment do
61
+ opts = {
62
+ kind: :validations,
63
+ format: ENV["format"],
64
+ model: ENV["model"]
65
+ }.compact
66
+
67
+ if ENV["path"]
68
+ opts[:paths] = ENV["path"]&.split(",")&.map { |p| Rails.root.join(p) }
69
+ end
70
+
71
+ if ENV["sort"]
72
+ sort_by, sort_order = ENV["sort"].split(":")
73
+ opts[:sort_by] = sort_by.to_sym
74
+ sort_order ||= ((sort_by == "size") ? :desc : :asc)
75
+ opts[:sort_order] = sort_order.to_sym
76
+ end
77
+
78
+ puts ModelScope::Runner.run(**opts)
79
+ end
80
+
81
+ desc <<~DESC
82
+ Generate combined report (callbacks and validations) for Active Record models.
83
+
84
+ Options:
85
+ format=table|line|github Report format (default: table)
86
+ model=ModelName Filter by model name (optional). Can be
87
+ specified as constant
88
+ name (MessageThread) or file path
89
+ (message_thread, admin/message_thread)
90
+ sort=total|name Sort by score or name (default: score:desc)
91
+ path=DIR1,DIR2 Additional model directories (comma-separated)
92
+ DESC
93
+ task report: :environment do
94
+ opts = {
95
+ kind: :report,
96
+ format: ENV["format"],
97
+ model: ENV["model"]
98
+ }.compact
99
+
100
+ if ENV["path"]
101
+ opts[:paths] = ENV["path"]&.split(",")&.map { |p| Rails.root.join(p) }
102
+ end
103
+
104
+ if ENV["sort"]
105
+ sort_by, sort_order = ENV["sort"].split(":")
106
+ opts[:sort_by] = sort_by.to_sym
107
+ sort_order ||= ((sort_by == "size") ? :desc : :asc)
108
+ opts[:sort_order] = sort_order.to_sym
109
+ end
110
+
111
+ puts ModelScope::Runner.run(**opts)
112
+ end
113
+ 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.1
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: []