-
-
Notifications
You must be signed in to change notification settings - Fork 263
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add
Rails/UniqueValidationWithoutIndex
cop
- Loading branch information
Showing
13 changed files
with
758 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
# frozen_string_literal: true | ||
|
||
module RuboCop | ||
module Cop | ||
# A mixin to extend cops for ActiveRecord features | ||
module ActiveRecordHelper | ||
extend NodePattern::Macros | ||
|
||
def_node_search :find_set_table_name, <<-PATTERN | ||
(send self :table_name= {str sym}) | ||
PATTERN | ||
|
||
def table_name(class_node) | ||
table_name = find_set_table_name(class_node).to_a.last&.first_argument | ||
return table_name.value.to_s if table_name | ||
|
||
namespaces = class_node.each_ancestor(:class, :module) | ||
[class_node, *namespaces] | ||
.reverse | ||
.map { |klass| klass.identifier.children[1] }.join('_') | ||
.tableize | ||
end | ||
end | ||
end | ||
end |
123 changes: 123 additions & 0 deletions
123
lib/rubocop/cop/rails/unique_validation_without_index.rb
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,123 @@ | ||
# frozen_string_literal: true | ||
|
||
module RuboCop | ||
module Cop | ||
module Rails | ||
# When you define uniqueness validation in ActiveRecord model, | ||
# you also should add a unique index for the column. It has two reasons. | ||
# First, duplicated records may occur even if ActiveRecord's validation | ||
# is defined. | ||
# Second, it will cause slow queries. The validation executes a `SELECT` | ||
# statement with the target column when inserting/updating a record. | ||
# If the column does not have an index and the table is large, | ||
# the query will be heavy. | ||
# | ||
# @example | ||
# # bad - if the schema does not have a unique index | ||
# validates :account, uniqueness: true | ||
# | ||
# # good - if the schema has a unique index | ||
# validates :account, uniqueness: true | ||
# | ||
# # good - even if the schema does not have a unique index | ||
# validates :account, length: { minimum: MIN_LENGTH } | ||
# | ||
class UniqueValidationWithoutIndex < Cop | ||
include ActiveRecordHelper | ||
|
||
MSG = 'Uniqueness validation should be with a unique index' | ||
|
||
def on_send(node) | ||
return unless node.method?(:validates) | ||
return unless uniqueness_part(node) | ||
return unless schema | ||
return if with_index?(node) | ||
|
||
add_offense(node) | ||
end | ||
|
||
private | ||
|
||
def with_index?(node) | ||
klass = class_node(node) | ||
return true unless klass # Skip analysis | ||
|
||
table = schema.table_by(name: table_name(klass)) | ||
return true unless table # Skip analysis if it can't find the table | ||
|
||
names = column_names(node) | ||
return unless names | ||
|
||
table.indices.any? do |index| | ||
index.unique && index.columns.to_set == names | ||
end | ||
end | ||
|
||
def column_names(node) | ||
arg = node.first_argument | ||
return unless arg.str_type? || arg.sym_type? | ||
|
||
ret = [arg.value] | ||
names_from_scope = column_names_from_scope(node) | ||
ret.concat(names_from_scope) if names_from_scope | ||
|
||
ret.map(&:to_s).to_set | ||
end | ||
|
||
def column_names_from_scope(node) | ||
uniq = uniqueness_part(node) | ||
return unless uniq.hash_type? | ||
|
||
scope = find_scope(uniq) | ||
return unless scope | ||
|
||
case scope.type | ||
when :sym, :str | ||
[scope.value] | ||
when :array | ||
array_node_to_array(scope) | ||
end | ||
end | ||
|
||
def find_scope(pairs) | ||
pairs.each_pair.find do |pair| | ||
key = pair.key | ||
next unless key.sym_type? && key.value == :scope | ||
|
||
break pair.value | ||
end | ||
end | ||
|
||
def class_node(node) | ||
node.each_ancestor.find(&:class_type?) | ||
end | ||
|
||
def uniqueness_part(node) | ||
pairs = node.arguments.last | ||
return unless pairs.hash_type? | ||
|
||
pairs.each_pair.find do |pair| | ||
next unless pair.key.sym_type? && pair.key.value == :uniqueness | ||
|
||
break pair.value | ||
end | ||
end | ||
|
||
def array_node_to_array(node) | ||
node.values.map do |elm| | ||
case elm.type | ||
when :str, :sym | ||
elm.value | ||
else | ||
return nil | ||
end | ||
end | ||
end | ||
|
||
def schema | ||
RuboCop::Rails::SchemaLoader.load(target_ruby_version) | ||
end | ||
end | ||
end | ||
end | ||
end |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,61 @@ | ||
# frozen_string_literal: true | ||
|
||
module RuboCop | ||
module Rails | ||
# It loads db/schema.rb and return Schema object. | ||
# Cops refers database schema information with this module. | ||
module SchemaLoader | ||
extend self | ||
|
||
# It parses `db/schema.rb` and return it. | ||
# It returns `nil` if it can't find `db/schema.rb`. | ||
# So a cop that uses the loader should handle `nil` properly. | ||
# | ||
# @return [Schema, nil] | ||
def load(target_ruby_version) | ||
return @schema if defined?(@schema) | ||
|
||
@schema = load!(target_ruby_version) | ||
end | ||
|
||
def reset! | ||
return unless instance_variable_defined?(:@schema) | ||
|
||
remove_instance_variable(:@schema) | ||
end | ||
|
||
private | ||
|
||
def load!(target_ruby_version) | ||
path = db_schema_path | ||
return unless path | ||
|
||
ast = parse(path, target_ruby_version) | ||
Schema.new(ast) | ||
end | ||
|
||
def db_schema_path | ||
path = Pathname.pwd | ||
until path.root? | ||
schema_path = path.join('db/schema.rb') | ||
return schema_path if schema_path.exist? | ||
|
||
path = path.join('../').cleanpath | ||
end | ||
|
||
nil | ||
end | ||
|
||
def parse(path, target_ruby_version) | ||
klass_name = :"Ruby#{target_ruby_version.to_s.sub('.', '')}" | ||
klass = ::Parser.const_get(klass_name) | ||
parser = klass.new(RuboCop::AST::Builder.new) | ||
|
||
buffer = Parser::Source::Buffer.new(path, 1) | ||
buffer.source = path.read | ||
|
||
parser.parse(buffer) | ||
end | ||
end | ||
end | ||
end |
Oops, something went wrong.