From 2f11788cf1b6051735336ced1817199cf0ffba57 Mon Sep 17 00:00:00 2001 From: Sergey Toy Date: Sat, 14 Sep 2024 11:16:39 +0800 Subject: [PATCH 1/2] Automatically register checkers when inherited from the BaseChecker --- lib/database_consistency.rb | 23 +++++++----- .../checkers/base_checker.rb | 12 +++++- .../processors/associations_processor.rb | 8 ---- .../processors/base_processor.rb | 16 +++----- .../processors/columns_processor.rb | 9 ----- .../processors/enums_processor.rb | 4 -- .../processors/indexes_processor.rb | 6 --- .../processors/models_processor.rb | 4 -- .../validators_fractions_processor.rb | 4 -- .../processors/validators_processor.rb | 5 --- .../writers/simple_writer.rb | 37 +++---------------- spec/checkers/unique_index_checker_spec.rb | 2 +- 12 files changed, 38 insertions(+), 92 deletions(-) diff --git a/lib/database_consistency.rb b/lib/database_consistency.rb index 8767d92..8dc04e1 100644 --- a/lib/database_consistency.rb +++ b/lib/database_consistency.rb @@ -1,6 +1,11 @@ # frozen_string_literal: true require 'active_record' +require 'active_support/inflector' + +ActiveSupport::Inflector.inflections do |inflect| + inflect.irregular 'index', 'indexes' +end require 'database_consistency/version' require 'database_consistency/helper' @@ -60,6 +65,15 @@ require 'database_consistency/databases/types/base' require 'database_consistency/databases/types/sqlite' +require 'database_consistency/processors/base_processor' +require 'database_consistency/processors/enums_processor' +require 'database_consistency/processors/associations_processor' +require 'database_consistency/processors/validators_processor' +require 'database_consistency/processors/columns_processor' +require 'database_consistency/processors/validators_fractions_processor' +require 'database_consistency/processors/indexes_processor' +require 'database_consistency/processors/models_processor' + require 'database_consistency/checkers/base_checker' require 'database_consistency/checkers/enum_checkers/enum_checker' @@ -95,15 +109,6 @@ require 'database_consistency/checkers/index_checkers/redundant_index_checker' require 'database_consistency/checkers/index_checkers/redundant_unique_index_checker' -require 'database_consistency/processors/base_processor' -require 'database_consistency/processors/enums_processor' -require 'database_consistency/processors/associations_processor' -require 'database_consistency/processors/validators_processor' -require 'database_consistency/processors/columns_processor' -require 'database_consistency/processors/validators_fractions_processor' -require 'database_consistency/processors/indexes_processor' -require 'database_consistency/processors/models_processor' - # The root module module DatabaseConsistency class << self diff --git a/lib/database_consistency/checkers/base_checker.rb b/lib/database_consistency/checkers/base_checker.rb index f081d66..0c03afc 100644 --- a/lib/database_consistency/checkers/base_checker.rb +++ b/lib/database_consistency/checkers/base_checker.rb @@ -11,9 +11,19 @@ def self.enabled?(configuration) configuration.enabled?('DatabaseConsistencyCheckers', checker_name) end + def self.inherited(subclass) + super + + return if subclass.superclass.name == 'DatabaseConsistency::Checkers::BaseChecker' + + processor_prefix = subclass.superclass.name.demodulize.delete_suffix('Checker').pluralize + processor = "DatabaseConsistency::Processors::#{processor_prefix}Processor".constantize + processor.checkers << subclass + end + # @return [String] def self.checker_name - @checker_name ||= name.split('::').last + @checker_name ||= name.demodulize end # @param [Boolean] catch_errors diff --git a/lib/database_consistency/processors/associations_processor.rb b/lib/database_consistency/processors/associations_processor.rb index 41a49ba..d3abeb5 100644 --- a/lib/database_consistency/processors/associations_processor.rb +++ b/lib/database_consistency/processors/associations_processor.rb @@ -4,14 +4,6 @@ module DatabaseConsistency module Processors # The class to process associations class AssociationsProcessor < BaseProcessor - CHECKERS = [ - Checkers::MissingIndexChecker, - Checkers::ForeignKeyChecker, - Checkers::ForeignKeyTypeChecker, - Checkers::ForeignKeyCascadeChecker, - Checkers::MissingAssociationClassChecker - ].freeze - private def check # rubocop:disable Metrics/MethodLength diff --git a/lib/database_consistency/processors/base_processor.rb b/lib/database_consistency/processors/base_processor.rb index ad7945e..2bd1966 100644 --- a/lib/database_consistency/processors/base_processor.rb +++ b/lib/database_consistency/processors/base_processor.rb @@ -4,21 +4,17 @@ module DatabaseConsistency # The module for processors module Processors def self.reports(configuration) - [ - ColumnsProcessor, - ValidatorsProcessor, - AssociationsProcessor, - ValidatorsFractionsProcessor, - IndexesProcessor, - EnumsProcessor, - ModelsProcessor - ].flat_map do |processor| + BaseProcessor.descendants.flat_map do |processor| processor.new(configuration).reports end end # The base class for processors class BaseProcessor + def self.checkers + @checkers ||= Set.new + end + attr_reader :configuration # @param [DatabaseConsistency::Configuration] configuration @@ -38,7 +34,7 @@ def reports(catch_errors: true) # @return [Array] def enabled_checkers - self.class::CHECKERS.select { |checker| checker.enabled?(configuration) } + self.class.checkers.select { |checker| checker.enabled?(configuration) } end private diff --git a/lib/database_consistency/processors/columns_processor.rb b/lib/database_consistency/processors/columns_processor.rb index 4c91403..4f962ba 100644 --- a/lib/database_consistency/processors/columns_processor.rb +++ b/lib/database_consistency/processors/columns_processor.rb @@ -4,15 +4,6 @@ module DatabaseConsistency module Processors # The class to process columns class ColumnsProcessor < BaseProcessor - CHECKERS = [ - Checkers::NullConstraintChecker, - Checkers::LengthConstraintChecker, - Checkers::PrimaryKeyTypeChecker, - Checkers::EnumValueChecker, - Checkers::ThreeStateBooleanChecker, - Checkers::ImplicitOrderingChecker - ].freeze - private def check # rubocop:disable Metrics/MethodLength diff --git a/lib/database_consistency/processors/enums_processor.rb b/lib/database_consistency/processors/enums_processor.rb index 37037a2..e244d17 100644 --- a/lib/database_consistency/processors/enums_processor.rb +++ b/lib/database_consistency/processors/enums_processor.rb @@ -4,10 +4,6 @@ module DatabaseConsistency module Processors # The class to process enums class EnumsProcessor < BaseProcessor - CHECKERS = [ - Checkers::EnumTypeChecker - ].freeze - private def check # rubocop:disable Metrics/MethodLength diff --git a/lib/database_consistency/processors/indexes_processor.rb b/lib/database_consistency/processors/indexes_processor.rb index 191fe2a..5aedbea 100644 --- a/lib/database_consistency/processors/indexes_processor.rb +++ b/lib/database_consistency/processors/indexes_processor.rb @@ -4,12 +4,6 @@ module DatabaseConsistency module Processors # The class to process indexes class IndexesProcessor < BaseProcessor - CHECKERS = [ - Checkers::UniqueIndexChecker, - Checkers::RedundantIndexChecker, - Checkers::RedundantUniqueIndexChecker - ].freeze - private def check # rubocop:disable Metrics/AbcSize, Metrics/MethodLength diff --git a/lib/database_consistency/processors/models_processor.rb b/lib/database_consistency/processors/models_processor.rb index 5e0af85..ecfaf04 100644 --- a/lib/database_consistency/processors/models_processor.rb +++ b/lib/database_consistency/processors/models_processor.rb @@ -4,10 +4,6 @@ module DatabaseConsistency module Processors # The class to process models class ModelsProcessor < BaseProcessor - CHECKERS = [ - Checkers::MissingTableChecker - ].freeze - private def check diff --git a/lib/database_consistency/processors/validators_fractions_processor.rb b/lib/database_consistency/processors/validators_fractions_processor.rb index 82964f6..efe1a4d 100644 --- a/lib/database_consistency/processors/validators_fractions_processor.rb +++ b/lib/database_consistency/processors/validators_fractions_processor.rb @@ -4,10 +4,6 @@ module DatabaseConsistency module Processors # The class to process validators class ValidatorsFractionsProcessor < BaseProcessor - CHECKERS = [ - Checkers::ColumnPresenceChecker - ].freeze - private # @return [Array] diff --git a/lib/database_consistency/processors/validators_processor.rb b/lib/database_consistency/processors/validators_processor.rb index b55b467..227f1b1 100644 --- a/lib/database_consistency/processors/validators_processor.rb +++ b/lib/database_consistency/processors/validators_processor.rb @@ -4,11 +4,6 @@ module DatabaseConsistency module Processors # The class to process validators class ValidatorsProcessor < BaseProcessor - CHECKERS = [ - Checkers::MissingUniqueIndexChecker, - Checkers::CaseSensitiveUniqueValidationChecker - ].freeze - private # @return [Array] diff --git a/lib/database_consistency/writers/simple_writer.rb b/lib/database_consistency/writers/simple_writer.rb index 76356fb..6c0eaf7 100644 --- a/lib/database_consistency/writers/simple_writer.rb +++ b/lib/database_consistency/writers/simple_writer.rb @@ -5,36 +5,6 @@ module DatabaseConsistency module Writers # The simplest formatter class SimpleWriter < BaseWriter - SLUG_TO_WRITER = { - association_foreign_type_missing_null_constraint: Simple::AssociationForeignTypeMissingNullConstraint, - association_missing_index: Simple::AssociationMissingIndex, - association_missing_null_constraint: Simple::AssociationMissingNullConstraint, - enum_values_inconsistent_with_ar_enum: Simple::EnumValuesInconsistentWithArEnum, - enum_values_inconsistent_with_inclusion: Simple::EnumValuesInconsistentWithInclusion, - has_one_missing_unique_index: Simple::HasOneMissingUniqueIndex, - implicit_order_column_missing: Simple::ImplicitOrderColumnMissing, - inconsistent_enum_type: Simple::InconsistentEnumType, - inconsistent_types: Simple::InconsistentTypes, - length_validator_greater_limit: Simple::LengthValidatorGreaterLimit, - length_validator_lower_limit: Simple::LengthValidatorLowerLimit, - length_validator_missing: Simple::LengthValidatorMissing, - missing_association_class: Simple::MissingAssociationClass, - missing_foreign_key: Simple::MissingForeignKey, - missing_foreign_key_cascade: Simple::MissingForeignKeyCascade, - missing_table: Simple::MissingTable, - missing_unique_index: Simple::MissingUniqueIndex, - missing_uniqueness_validation: Simple::MissingUniquenessValidation, - null_constraint_association_misses_validator: Simple::NullConstraintAssociationMissesValidator, - null_constraint_misses_validator: Simple::NullConstraintMissesValidator, - null_constraint_missing: Simple::NullConstraintMissing, - possible_null: Simple::PossibleNull, - redundant_case_insensitive_option: Simple::RedundantCaseInsensitiveOption, - redundant_index: Simple::RedundantIndex, - redundant_unique_index: Simple::RedundantUniqueIndex, - small_primary_key: Simple::SmallPrimaryKey, - three_state_boolean: Simple::ThreeStateBoolean - }.freeze - def write results.select(&method(:write?)) .map(&method(:writer)) @@ -58,7 +28,12 @@ def write?(report) end def writer(report) - klass = SLUG_TO_WRITER[report.error_slug] || Simple::DefaultMessage + klass = begin + "DatabaseConsistency::Writers::Simple::#{report.error_slug.to_s.classify}".constantize + rescue NameError + Simple::DefaultMessage + end + klass.new(report, config: config) end end diff --git a/spec/checkers/unique_index_checker_spec.rb b/spec/checkers/unique_index_checker_spec.rb index 4ecf1f6..ef8ed66 100644 --- a/spec/checkers/unique_index_checker_spec.rb +++ b/spec/checkers/unique_index_checker_spec.rb @@ -6,7 +6,7 @@ let(:model) { klass } let(:index) { ActiveRecord::Base.connection.indexes(klass.table_name).first } - let(:checker_name) { described_class.to_s.split('::').last } + let(:checker_name) { described_class.name.demodulize } let(:index_name) { 'index_name' } context 'when unique index is based on association' do From 45be8e2e2e383ba6866e0ee3ab057f51d216c127 Mon Sep 17 00:00:00 2001 From: Sergey Toy Date: Sat, 14 Sep 2024 12:55:47 +0800 Subject: [PATCH 2/2] Introduce `require` directive to load custom checkers --- lib/database_consistency/configuration.rb | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/lib/database_consistency/configuration.rb b/lib/database_consistency/configuration.rb index 6fa4570..adb096f 100644 --- a/lib/database_consistency/configuration.rb +++ b/lib/database_consistency/configuration.rb @@ -16,6 +16,8 @@ def initialize(file_paths = DEFAULT_PATH) end extract_configurations(existing_paths) end + + load_custom_checkers end def debug? @@ -123,5 +125,13 @@ def log_level settings && settings['log_level'] end end + + def load_custom_checkers + return unless configuration.key?('require') && configuration['require'].is_a?(Array) + + configuration['require'].each do |path| + require path + end + end end end