123456789_123456789_123456789_123456789_123456789_

Class: ActiveRecord::Validations::UniquenessValidator

Do not use. This class is for internal use only.
Relationships & Source Files
Super Chains via Extension / Inclusion / Inheritance
Class Chain:
Instance Chain:
Inherits: ActiveModel::EachValidator
Defined in: activerecord/lib/active_record/validations/uniqueness.rb

Class Method Summary

::ActiveModel::EachValidator - Inherited

.new

Returns a new validator instance.

::ActiveModel::Validator - Inherited

.kind

Returns the kind of the validator.

.new

Accepts options that will be made available through the options reader.

Instance Attribute Summary

Instance Method Summary

::ActiveModel::EachValidator - Inherited

#check_validity!

Hook method that gets called by the initializer allowing verification that the arguments supplied are valid.

#validate

Performs validation on the supplied record.

#validate_each

Override this method in subclasses with the validation logic, adding errors to the records errors array where necessary.

#prepare_value_for_validation

::ActiveModel::Validator - Inherited

#kind

Returns the kind for this validator.

#validate

Override this method in subclasses with validation logic, adding errors to the records errors array where necessary.

Constructor Details

.new(options) ⇒ UniquenessValidator

[ GitHub ]

  
# File 'activerecord/lib/active_record/validations/uniqueness.rb', line 6

def initialize(options)
  if options[:conditions] && !options[:conditions].respond_to?(:call)
    raise ArgumentError, "#{options[:conditions]} was passed as :conditions but is not callable. " \
                         "Pass a callable instead: `conditions: -> { where(approved: true) }`"
  end
  unless Array(options[:scope]).all? { |scope| scope.respond_to?(:to_sym) }
    raise ArgumentError, "#{options[:scope]} is not supported format for :scope option. " \
      "Pass a symbol or an array of symbols instead: `scope: :user_id`"
  end
  super
  @klass = options[:class]
  @klass = @klass.superclass if @klass.singleton_class?
end

Instance Method Details

#build_relation(klass, attribute, value) (private)

[ GitHub ]

  
# File 'activerecord/lib/active_record/validations/uniqueness.rb', line 112

def build_relation(klass, attribute, value)
  relation = klass.unscoped
  # TODO: Add case-sensitive / case-insensitive operators to Arel
  # to no longer need to checkout a connection here.
  comparison = klass.with_connection do |connection|
    relation.bind_attribute(attribute, value) do |attr, bind|
      return relation.none! if bind.unboundable?

      if !options.key?(:case_sensitive) || bind.nil?
        connection.default_uniqueness_comparison(attr, bind)
      elsif options[:case_sensitive]
        connection.case_sensitive_comparison(attr, bind)
      else
        # will use SQL LOWER function before comparison, unless it detects a case insensitive collation
        connection.case_insensitive_comparison(attr, bind)
      end
    end
  end

  relation.where!(comparison)
end

#covered_by_unique_index?(klass, record, attribute, scope) ⇒ Boolean (private)

[ GitHub ]

  
# File 'activerecord/lib/active_record/validations/uniqueness.rb', line 83

def covered_by_unique_index?(klass, record, attribute, scope)
  @covered ||= self.attributes.map(&:to_s).select do |attr|
    attributes = scope + [attr]
    attributes = resolve_attributes(record, attributes)

    klass.schema_cache.indexes(klass.table_name).any? do |index|
      index.unique &&
        index.where.nil? &&
        (Array(index.columns) - attributes).empty?
    end
  end

  @covered.include?(attribute.to_s)
end

#find_finder_class_for(record) (private)

The check for an existing value should be run from a class that isn’t abstract. This means working down from the current class (self), to the first non-abstract class. Since classes don’t know their subclasses, we have to build the hierarchy between self and the record’s class.

[ GitHub ]

  
# File 'activerecord/lib/active_record/validations/uniqueness.rb', line 60

def find_finder_class_for(record)
  class_hierarchy = [record.class]

  while class_hierarchy.first != @klass
    class_hierarchy.unshift(class_hierarchy.first.superclass)
  end

  class_hierarchy.detect { |klass| !klass.abstract_class? }
end

#map_enum_attribute(klass, attribute, value) (private)

[ GitHub ]

  
# File 'activerecord/lib/active_record/validations/uniqueness.rb', line 147

def map_enum_attribute(klass, attribute, value)
  mapping = klass.defined_enums[attribute.to_s]
  value = mapping[value] if value && mapping
  value
end

#resolve_attributes(record, attributes) (private)

[ GitHub ]

  
# File 'activerecord/lib/active_record/validations/uniqueness.rb', line 98

def resolve_attributes(record, attributes)
  attributes.flat_map do |attribute|
    reflection = record.class._reflect_on_association(attribute)

    if reflection.nil?
      attribute.to_s
    elsif reflection.polymorphic?
      [reflection.foreign_key, reflection.foreign_type]
    else
      reflection.foreign_key
    end
  end
end

#scope_relation(record, relation) (private)

[ GitHub ]

  
# File 'activerecord/lib/active_record/validations/uniqueness.rb', line 134

def scope_relation(record, relation)
  Array(options[:scope]).each do |scope_item|
    scope_value = if record.class._reflect_on_association(scope_item)
      record.association(scope_item).reader
    else
      record.read_attribute(scope_item)
    end
    relation = relation.where(scope_item => scope_value)
  end

  relation
end

#validate_each(record, attribute, value)

[ GitHub ]

  
# File 'activerecord/lib/active_record/validations/uniqueness.rb', line 20

def validate_each(record, attribute, value)
  finder_class = find_finder_class_for(record)
  value = map_enum_attribute(finder_class, attribute, value)

  return if record.persisted? && !validation_needed?(finder_class, record, attribute)

  relation = build_relation(finder_class, attribute, value)
  if record.persisted?
    if finder_class.primary_key
      relation = relation.where.not(finder_class.primary_key => [record.id_in_database])
    else
      raise UnknownPrimaryKey.new(finder_class, "Cannot validate uniqueness for persisted record without primary key.")
    end
  end
  relation = scope_relation(record, relation)

  if options[:conditions]
    conditions = options[:conditions]

    relation = if conditions.arity.zero?
      relation.instance_exec(&conditions)
    else
      relation.instance_exec(record, &conditions)
    end
  end

  if relation.exists?
    error_options = options.except(:case_sensitive, :scope, :conditions)
    error_options[:value] = value

    record.errors.add(attribute, :taken, **error_options)
  end
end

#validation_needed?(klass, record, attribute) ⇒ Boolean (private)

[ GitHub ]

  
# File 'activerecord/lib/active_record/validations/uniqueness.rb', line 70

def validation_needed?(klass, record, attribute)
  return true if options[:conditions] || options.key?(:case_sensitive)

  scope = Array(options[:scope])
  attributes = scope + [attribute]
  attributes = resolve_attributes(record, attributes)

  return true if attributes.any? { |attr| record.attribute_changed?(attr) ||
                                          record.read_attribute(attr).nil? }

  !covered_by_unique_index?(klass, record, attribute, scope)
end