dependent_option_checker 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 ADDED
@@ -0,0 +1,7 @@
1
+ ---
2
+ SHA256:
3
+ metadata.gz: e03c96889e5fc45953006a43fd53e44293774a8843aaeedb44dcbe00ef06c434
4
+ data.tar.gz: 83d1adf802a900300caf508526299c5090a800dca52ad12acc3bda30ab539180
5
+ SHA512:
6
+ metadata.gz: 854c450ac46eb74d7463202b0eedd7662261d919bf5403692962e22e10e3cfd9070fd1c52e50227bf5b5369fefb49b2efad99618f3c75722294cda9550ff428f
7
+ data.tar.gz: 6bce7b79d4f319c1c0bf77d40aba16d5066ea17b75c057d6a31784e179951c4837f5830864871ebadbe4a1caa5db5b4ba7265f1b45d9d2bc8958c9a11e39f92c
data/.rubocop.yml ADDED
@@ -0,0 +1,10 @@
1
+ AllCops:
2
+ TargetRubyVersion: 3.2
3
+ NewCops: enable
4
+
5
+ Style/Documentation:
6
+ Enabled: false
7
+
8
+ Metrics/BlockLength:
9
+ Exclude:
10
+ - 'spec/**/*_spec.rb'
data/LICENSE.txt ADDED
@@ -0,0 +1,21 @@
1
+ The MIT License (MIT)
2
+
3
+ Copyright (c) 2025 muryoimpl
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy
6
+ of this software and associated documentation files (the "Software"), to deal
7
+ in the Software without restriction, including without limitation the rights
8
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9
+ copies of the Software, and to permit persons to whom the Software is
10
+ furnished to do so, subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in
13
+ all copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21
+ THE SOFTWARE.
data/README.md ADDED
@@ -0,0 +1,64 @@
1
+ # DependentOptionChecker
2
+
3
+ `dependent_option_checker` is a simple gem that provides a Rake task to detect missing `dependent` options in `has_many` / `has_one` associations in ActiveRecord models. It also helps identify missing `has_many` / `has_one` associations themselves.
4
+
5
+ ## Features
6
+
7
+ - Detects associations lacking a dependent: ... option.
8
+ - Identifies missing has_many / has_one associations.
9
+ - Outputs the names of models and the specific missing configurations.
10
+ - Allows excluding specific tables from the check via a YAML config file.
11
+
12
+ ## Installation
13
+
14
+ Add this line to your application's Gemfile:
15
+ ```ruby
16
+ gem 'dependent_option_checker'
17
+ ```
18
+
19
+ And then execute:
20
+ ```console
21
+ bundle install
22
+
23
+ bin/rails g dependent_option_checker:install
24
+ ```
25
+
26
+ This will generate a configuration file at config/dependent_option_checker.yml, which you can edit to specify tables to ignore during checks.
27
+
28
+ ## Usage
29
+
30
+ Run the following command:
31
+
32
+ ```console
33
+ bin/rails dependent_option_checker:check
34
+ ```
35
+
36
+ If any missing configuration is detected, the task will output the corresponding model names and the details of what is missing.
37
+
38
+
39
+ ## Configuration
40
+
41
+ You can create a `dependent_option_checker.yml` file in your Rails `config` directory to exclude specific tables and relations from the check:
42
+
43
+ ```yaml
44
+ ignored_tables:
45
+ - users
46
+
47
+ ignored_relations:
48
+ Organization:
49
+ - employees # alert: this is table name
50
+ ```
51
+
52
+ ## Development
53
+
54
+ After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake spec` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
55
+
56
+ To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and the created tag, and push the `.gem` file to [rubygems.org](https://rubygems.org).
57
+
58
+ ## Contributing
59
+
60
+ Bug reports and pull requests are welcome on GitHub at https://github.com/muryoimpl/dependent_option_checker.
61
+
62
+ ## License
63
+
64
+ 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,12 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'bundler/gem_tasks'
4
+ require 'rspec/core/rake_task'
5
+
6
+ RSpec::Core::RakeTask.new(:spec)
7
+
8
+ require 'rubocop/rake_task'
9
+
10
+ RuboCop::RakeTask.new
11
+
12
+ task default: %i[spec rubocop]
@@ -0,0 +1,44 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'active_record/reflection'
4
+
5
+ module DependentOptionChecker
6
+ class Checker
7
+ class DependentChecker
8
+ DEPENDENT_OPTION_VALUES = %i[
9
+ destroy
10
+ destroy_async
11
+ delete_all
12
+ nullify
13
+ restrict_with_exception
14
+ restrict_with_error
15
+ ].freeze
16
+
17
+ def initialize(model)
18
+ @model = model
19
+ end
20
+
21
+ def extract_unspecified_relations
22
+ reflections.filter_map do |reflection|
23
+ option_specified?(reflection.options) ? nil : reflection.name.to_s
24
+ end
25
+ end
26
+
27
+ private
28
+
29
+ def reflections
30
+ @model.reflections.select do |_, reflection|
31
+ ::DependentOptionChecker::Checker::TARGET_CLASSES.include?(reflection.class)
32
+ end.values
33
+ end
34
+
35
+ def option_specified?(options)
36
+ options.key?(:dependent) &&
37
+ (
38
+ DEPENDENT_OPTION_VALUES.include?(options[:dependent]) ||
39
+ options[:dependent].nil?
40
+ )
41
+ end
42
+ end
43
+ end
44
+ end
@@ -0,0 +1,39 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'active_record/reflection'
4
+ require 'active_support/core_ext/string'
5
+
6
+ module DependentOptionChecker
7
+ class Checker
8
+ class RelationDeclarationChecker
9
+ def initialize(model:, table_cache:, ignored_relations:)
10
+ @model = model
11
+ @table_cache = table_cache
12
+ @ignored_relations = ignored_relations || []
13
+ end
14
+
15
+ def extract_undeclared_tables
16
+ tables = table_names_having_attribute
17
+ tables - relation_table_names - @ignored_relations
18
+ end
19
+
20
+ private
21
+
22
+ def table_names_having_attribute
23
+ @table_cache.select do |_, attributes|
24
+ attributes.include?(attribute_name)
25
+ end.keys
26
+ end
27
+
28
+ def attribute_name
29
+ @attribute_name ||= "#{@model.table_name.singularize.underscore}_id"
30
+ end
31
+
32
+ def relation_table_names
33
+ @model.reflections.select do |_, relation|
34
+ ::DependentOptionChecker::Checker::TARGET_CLASSES.include?(relation.class)
35
+ end.values.map(&:plural_name)
36
+ end
37
+ end
38
+ end
39
+ end
@@ -0,0 +1,61 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'active_support/concern'
4
+ require 'active_support/core_ext/module/delegation'
5
+ require 'active_record'
6
+
7
+ require_relative 'configuration'
8
+ require_relative 'model_loader'
9
+ require_relative 'checker/dependent_checker'
10
+ require_relative 'checker/relation_declaration_checker'
11
+
12
+ module DependentOptionChecker
13
+ class Checker
14
+ TARGET_CLASSES = [
15
+ ActiveRecord::Reflection::HasOneReflection,
16
+ ActiveRecord::Reflection::HasManyReflection
17
+ ].freeze
18
+
19
+ def initialize
20
+ super
21
+ @config = Configuration.load
22
+ end
23
+
24
+ def execute
25
+ ModelLoader.load_files!
26
+
27
+ model_loader.load_table_attributes
28
+
29
+ model_loader.application_record_classes.each_with_object([]) do |model, acc|
30
+ unspecified = dependent_checker(model).extract_unspecified_relations
31
+ undeclared = relation_declaration_checker(model).extract_undeclared_tables
32
+
33
+ next if unspecified.empty? && undeclared.empty?
34
+
35
+ acc << result.new(table_name: model.table_name, unspecified:, undeclared:)
36
+ end
37
+ end
38
+
39
+ private
40
+
41
+ def model_loader
42
+ @model_loader ||= ModelLoader.new(@config)
43
+ end
44
+
45
+ def dependent_checker(model)
46
+ DependentChecker.new(model)
47
+ end
48
+
49
+ def relation_declaration_checker(model)
50
+ RelationDeclarationChecker.new(
51
+ model:,
52
+ table_cache: model_loader.cache_table_attributes,
53
+ ignored_relations: @config.ignored_relations[model.name]
54
+ )
55
+ end
56
+
57
+ def result
58
+ @result ||= Data.define(:table_name, :unspecified, :undeclared)
59
+ end
60
+ end
61
+ end
@@ -0,0 +1,24 @@
1
+ # frozen_string_literal: true
2
+
3
+ module DependentOptionChecker
4
+ class Configuration
5
+ def self.load = new
6
+
7
+ def initialize
8
+ @config = if File.exist?(config_file_path)
9
+ load_config
10
+ else
11
+ {}
12
+ end
13
+ end
14
+
15
+ def ignored_tables = @config['ignored_tables'] || []
16
+ def ignored_relations = @config['ignored_relations'] || {}
17
+
18
+ private
19
+
20
+ def config_file_path = Rails.root.join('config/dependent_option_checker.yml')
21
+
22
+ def load_config = YAML.safe_load_file(config_file_path)
23
+ end
24
+ end
@@ -0,0 +1,33 @@
1
+ # frozen_string_literal: true
2
+
3
+ module DependentOptionChecker
4
+ class ModelLoader
5
+ def self.load_files!
6
+ Rails.autoloaders.main.eager_load_dir(Rails.root.join('app', 'models')) unless Rails.env.production?
7
+ end
8
+
9
+ attr_reader :cache_table_attributes
10
+
11
+ def initialize(config)
12
+ @config = config
13
+ end
14
+
15
+ def application_record_classes
16
+ @application_record_classes ||= ::ApplicationRecord.descendants.filter(&:base_class?)
17
+ end
18
+
19
+ def application_table_models
20
+ @application_table_models ||= application_record_classes.reject do |klass|
21
+ next if @config.nil?
22
+
23
+ @config.ignored_tables.include?(klass.table_name)
24
+ end
25
+ end
26
+
27
+ def load_table_attributes
28
+ @cache_table_attributes = application_table_models.each_with_object({}) do |klass, acc|
29
+ acc[klass.table_name] = klass.attribute_names
30
+ end
31
+ end
32
+ end
33
+ end
@@ -0,0 +1,11 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'rails/railtie'
4
+
5
+ module DependentOptionChecker
6
+ class Railtie < ::Rails::Railtie
7
+ rake_tasks do
8
+ load 'dependent_option_checker/tasks/dependent_option_checker.rake'
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,31 @@
1
+ # frozen_string_literal: true
2
+
3
+ namespace :dependent_option_checker do
4
+ desc 'Detect omissions of the dependent option and related definitions in the AR models'
5
+ task check: :environment do
6
+ data = DependentOptionChecker::Checker.new.execute
7
+
8
+ if data.empty?
9
+ puts 'No omission detected.'
10
+ else
11
+ puts 'Detected `dependent` option or omissions of has_many/has_one denifition.'
12
+ puts
13
+
14
+ data.each do |d|
15
+ puts "# Model: \e[32m#{d.table_name.classify}\e[0m"
16
+
17
+ puts ' * dependent option omission' if d.unspecified.size.positive?
18
+ d.unspecified.each do |relation_name|
19
+ puts " + \e[31m#{relation_name}\e[0m"
20
+ end
21
+
22
+ puts ' - has_many/has_one omission' if d.undeclared.size.positive?
23
+ d.undeclared.each do |table_name|
24
+ puts " + \e[31m#{table_name}\e[0m"
25
+ end
26
+ end
27
+
28
+ exit(1)
29
+ end
30
+ end
31
+ end
@@ -0,0 +1,5 @@
1
+ # frozen_string_literal: true
2
+
3
+ module DependentOptionChecker
4
+ VERSION = '0.1.0'
5
+ end
@@ -0,0 +1,10 @@
1
+ # frozen_string_literal: true
2
+
3
+ require_relative 'dependent_option_checker/version'
4
+ require_relative 'dependent_option_checker/checker'
5
+
6
+ module DependentOptionChecker
7
+ class Error < StandardError; end
8
+ end
9
+
10
+ require_relative 'dependent_option_checker/railtie'
@@ -0,0 +1,13 @@
1
+ # frozen_string_literal: true
2
+
3
+ require 'rails/generators'
4
+
5
+ module DependentOptionChecker
6
+ class InstallGenerator < Rails::Generators::Base
7
+ source_root File.expand_path('templates', __dir__)
8
+
9
+ def create_yml_file
10
+ copy_file 'dependent_option_checker.yml', 'config/dependent_option_checker.yml'
11
+ end
12
+ end
13
+ end
@@ -0,0 +1,6 @@
1
+ ignored_tables:
2
+ # - users
3
+
4
+ ignored_relations:
5
+ # Organization:
6
+ # - employees # this is table name, not relation name
@@ -0,0 +1,4 @@
1
+ module DependentOptionChecker
2
+ VERSION: String
3
+ # See the writing guide of rbs: https://github.com/ruby/rbs#guides
4
+ end
metadata ADDED
@@ -0,0 +1,90 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: dependent_option_checker
3
+ version: !ruby/object:Gem::Version
4
+ version: 0.1.0
5
+ platform: ruby
6
+ authors:
7
+ - muryoimpl
8
+ autorequire:
9
+ bindir: exe
10
+ cert_chain: []
11
+ date: 2025-04-02 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.1'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ">="
25
+ - !ruby/object:Gem::Version
26
+ version: '7.1'
27
+ - !ruby/object:Gem::Dependency
28
+ name: rake
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: 13.0.6
34
+ type: :runtime
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - ">="
39
+ - !ruby/object:Gem::Version
40
+ version: 13.0.6
41
+ description: ActiveRecord has_many/has_one dependent option checker
42
+ email:
43
+ - muryoimpl@gmail.com
44
+ executables: []
45
+ extensions: []
46
+ extra_rdoc_files: []
47
+ files:
48
+ - ".rubocop.yml"
49
+ - LICENSE.txt
50
+ - README.md
51
+ - Rakefile
52
+ - lib/dependent_option_checker.rb
53
+ - lib/dependent_option_checker/checker.rb
54
+ - lib/dependent_option_checker/checker/dependent_checker.rb
55
+ - lib/dependent_option_checker/checker/relation_declaration_checker.rb
56
+ - lib/dependent_option_checker/configuration.rb
57
+ - lib/dependent_option_checker/model_loader.rb
58
+ - lib/dependent_option_checker/railtie.rb
59
+ - lib/dependent_option_checker/tasks/dependent_option_checker.rake
60
+ - lib/dependent_option_checker/version.rb
61
+ - lib/generators/dependent_option_checker/install_generator.rb
62
+ - lib/generators/dependent_option_checker/templates/dependent_option_checker.yml
63
+ - sig/dependent_option_checker.rbs
64
+ homepage: https://github.com/muryoimpl/dependent_option_checker
65
+ licenses:
66
+ - MIT
67
+ metadata:
68
+ homepage_uri: https://github.com/muryoimpl/dependent_option_checker
69
+ source_code_uri: https://github.com/muryoimpl/dependent_option_checker
70
+ rubygems_mfa_required: 'true'
71
+ post_install_message:
72
+ rdoc_options: []
73
+ require_paths:
74
+ - lib
75
+ required_ruby_version: !ruby/object:Gem::Requirement
76
+ requirements:
77
+ - - ">="
78
+ - !ruby/object:Gem::Version
79
+ version: 3.2.0
80
+ required_rubygems_version: !ruby/object:Gem::Requirement
81
+ requirements:
82
+ - - ">="
83
+ - !ruby/object:Gem::Version
84
+ version: '0'
85
+ requirements: []
86
+ rubygems_version: 3.4.19
87
+ signing_key:
88
+ specification_version: 4
89
+ summary: ActiveRecord has_many/has_one dependent option checker
90
+ test_files: []