mirror of
https://github.com/rails/rails.git
synced 2022-11-09 12:12:34 -05:00
1ac40f16c5
In Active Record internal, `arel_table` is not directly used but `arel_attribute` is used, since `arel_table` doesn't normalize an attribute name as a string, and doesn't resolve attribute aliases. For the above reason, `arel_attribute` should be used rather than `arel_table`, but most people directly use `arel_table`, both `arel_table` and `arel_attribute` are private API though. Although I'd not recommend using private API, `arel_table` is actually widely used, and it is also problematic for unscopeable queries and hash-like relation merging friendly, as I explained at #39863. To resolve the issue, this change moves Arel attribute normalization (attribute name as a string, and attribute alias resolution) into `arel_table`.
118 lines
2.6 KiB
Ruby
118 lines
2.6 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module Arel # :nodoc: all
|
|
class Table
|
|
include Arel::Crud
|
|
include Arel::FactoryMethods
|
|
include Arel::AliasPredication
|
|
|
|
@engine = nil
|
|
class << self; attr_accessor :engine; end
|
|
|
|
attr_accessor :name, :table_alias
|
|
|
|
# TableAlias and Table both have a #table_name which is the name of the underlying table
|
|
alias :table_name :name
|
|
|
|
def initialize(name, as: nil, klass: nil, type_caster: klass&.type_caster)
|
|
@name = name.to_s
|
|
@klass = klass
|
|
@type_caster = type_caster
|
|
|
|
# Sometime AR sends an :as parameter to table, to let the table know
|
|
# that it is an Alias. We may want to override new, and return a
|
|
# TableAlias node?
|
|
if as.to_s == @name
|
|
as = nil
|
|
end
|
|
@table_alias = as
|
|
end
|
|
|
|
def alias(name = "#{self.name}_2")
|
|
Nodes::TableAlias.new(self, name)
|
|
end
|
|
|
|
def from
|
|
SelectManager.new(self)
|
|
end
|
|
|
|
def join(relation, klass = Nodes::InnerJoin)
|
|
return from unless relation
|
|
|
|
case relation
|
|
when String, Nodes::SqlLiteral
|
|
raise EmptyJoinError if relation.empty?
|
|
klass = Nodes::StringJoin
|
|
end
|
|
|
|
from.join(relation, klass)
|
|
end
|
|
|
|
def outer_join(relation)
|
|
join(relation, Nodes::OuterJoin)
|
|
end
|
|
|
|
def group(*columns)
|
|
from.group(*columns)
|
|
end
|
|
|
|
def order(*expr)
|
|
from.order(*expr)
|
|
end
|
|
|
|
def where(condition)
|
|
from.where condition
|
|
end
|
|
|
|
def project(*things)
|
|
from.project(*things)
|
|
end
|
|
|
|
def take(amount)
|
|
from.take amount
|
|
end
|
|
|
|
def skip(amount)
|
|
from.skip amount
|
|
end
|
|
|
|
def having(expr)
|
|
from.having expr
|
|
end
|
|
|
|
def [](name, table = self)
|
|
name = name.to_s if name.is_a?(Symbol)
|
|
name = @klass.attribute_aliases[name] || name if @klass
|
|
Attribute.new(table, name)
|
|
end
|
|
|
|
def hash
|
|
# Perf note: aliases and table alias is excluded from the hash
|
|
# aliases can have a loop back to this table breaking hashes in parent
|
|
# relations, for the vast majority of cases @name is unique to a query
|
|
@name.hash
|
|
end
|
|
|
|
def eql?(other)
|
|
self.class == other.class &&
|
|
self.name == other.name &&
|
|
self.table_alias == other.table_alias
|
|
end
|
|
alias :== :eql?
|
|
|
|
def type_cast_for_database(attr_name, value)
|
|
type_caster.type_cast_for_database(attr_name, value)
|
|
end
|
|
|
|
def type_for_attribute(name)
|
|
type_caster.type_for_attribute(name)
|
|
end
|
|
|
|
def able_to_type_cast?
|
|
!type_caster.nil?
|
|
end
|
|
|
|
private
|
|
attr_reader :type_caster
|
|
end
|
|
end
|