2017-07-09 13:41:28 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2011-12-15 15:07:41 -05:00
|
|
|
module ActiveRecord
|
|
|
|
module Querying
|
2019-03-07 04:30:41 -05:00
|
|
|
QUERYING_METHODS = [
|
|
|
|
:find, :find_by, :find_by!, :take, :take!, :first, :first!, :last, :last!,
|
|
|
|
:second, :second!, :third, :third!, :fourth, :fourth!, :fifth, :fifth!,
|
|
|
|
:forty_two, :forty_two!, :third_to_last, :third_to_last!, :second_to_last, :second_to_last!,
|
|
|
|
:exists?, :any?, :many?, :none?, :one?,
|
|
|
|
:first_or_create, :first_or_create!, :first_or_initialize,
|
|
|
|
:find_or_create_by, :find_or_create_by!, :find_or_initialize_by,
|
|
|
|
:create_or_find_by, :create_or_find_by!,
|
2019-04-05 03:13:40 -04:00
|
|
|
:destroy_all, :delete_all, :update_all, :touch_all, :destroy_by, :delete_by,
|
2019-03-07 04:30:41 -05:00
|
|
|
:find_each, :find_in_batches, :in_batches,
|
|
|
|
:select, :reselect, :order, :reorder, :group, :limit, :offset, :joins, :left_joins, :left_outer_joins,
|
2019-03-29 19:01:45 -04:00
|
|
|
:where, :rewhere, :preload, :extract_associated, :eager_load, :includes, :from, :lock, :readonly, :extending, :or,
|
2019-03-12 20:29:00 -04:00
|
|
|
:having, :create_with, :distinct, :references, :none, :unscope, :optimizer_hints, :merge, :except, :only,
|
2019-03-14 15:57:14 -04:00
|
|
|
:count, :average, :minimum, :maximum, :sum, :calculate, :annotate,
|
2019-03-07 04:30:41 -05:00
|
|
|
:pluck, :pick, :ids
|
|
|
|
].freeze # :nodoc:
|
|
|
|
delegate(*QUERYING_METHODS, to: :all)
|
2011-12-15 15:07:41 -05:00
|
|
|
|
|
|
|
# Executes a custom SQL query against your database and returns all the results. The results will
|
2018-11-28 17:45:27 -05:00
|
|
|
# be returned as an array, with the requested columns encapsulated as attributes of the model you call
|
|
|
|
# this method from. For example, if you call <tt>Product.find_by_sql</tt>, then the results will be returned in
|
2013-05-14 02:15:32 -04:00
|
|
|
# a +Product+ object with the attributes you specified in the SQL query.
|
2011-12-15 15:07:41 -05:00
|
|
|
#
|
2018-11-28 17:45:27 -05:00
|
|
|
# If you call a complicated SQL query which spans multiple tables, the columns specified by the
|
2011-12-15 15:07:41 -05:00
|
|
|
# SELECT will be attributes of the model, whether or not they are columns of the corresponding
|
|
|
|
# table.
|
|
|
|
#
|
2018-11-28 17:45:27 -05:00
|
|
|
# The +sql+ parameter is a full SQL query as a string. It will be called as is; there will be
|
|
|
|
# no database agnostic conversions performed. This should be a last resort because using
|
|
|
|
# database-specific terms will lock you into using that particular database engine, or require you to
|
2011-12-15 15:07:41 -05:00
|
|
|
# change your call if you switch engines.
|
|
|
|
#
|
|
|
|
# # A simple SQL query spanning multiple tables
|
|
|
|
# Post.find_by_sql "SELECT p.title, c.author FROM posts p, comments c WHERE p.id = c.post_id"
|
2012-12-01 15:35:04 -05:00
|
|
|
# # => [#<Post:0x36bff9c @attributes={"title"=>"Ruby Meetup", "first_name"=>"Quentin"}>, ...]
|
2011-12-15 15:07:41 -05:00
|
|
|
#
|
2013-05-14 02:15:32 -04:00
|
|
|
# You can use the same string replacement techniques as you can with <tt>ActiveRecord::QueryMethods#where</tt>:
|
|
|
|
#
|
2011-12-15 15:07:41 -05:00
|
|
|
# Post.find_by_sql ["SELECT title FROM posts WHERE author = ? AND created > ?", author_id, start_date]
|
2013-05-14 02:15:32 -04:00
|
|
|
# Post.find_by_sql ["SELECT body FROM comments WHERE author = :user_id OR approved_by = :user_id", { :user_id => user_id }]
|
2016-08-31 08:54:38 -04:00
|
|
|
def find_by_sql(sql, binds = [], preparable: nil, &block)
|
2016-02-11 10:54:58 -05:00
|
|
|
result_set = connection.select_all(sanitize_sql(sql), "#{name} Load", binds, preparable: preparable)
|
2014-09-27 03:00:58 -04:00
|
|
|
column_types = result_set.column_types.dup
|
2019-02-12 14:03:55 -05:00
|
|
|
attribute_types.each_key { |k| column_types.delete k }
|
2014-10-13 17:22:24 -04:00
|
|
|
message_bus = ActiveSupport::Notifications.instrumenter
|
|
|
|
|
|
|
|
payload = {
|
|
|
|
record_count: result_set.length,
|
|
|
|
class_name: name
|
|
|
|
}
|
|
|
|
|
2016-08-06 12:24:04 -04:00
|
|
|
message_bus.instrument("instantiation.active_record", payload) do
|
2018-06-25 18:52:59 -04:00
|
|
|
if result_set.includes_column?(inheritance_column)
|
|
|
|
result_set.map { |record| instantiate(record, column_types, &block) }
|
|
|
|
else
|
|
|
|
# Instantiate a homogeneous set
|
|
|
|
result_set.map { |record| instantiate_instance_of(self, record, column_types, &block) }
|
|
|
|
end
|
2014-10-13 17:22:24 -04:00
|
|
|
end
|
2011-12-15 15:07:41 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
# Returns the result of an SQL statement that should only include a COUNT(*) in the SELECT part.
|
|
|
|
# The use of this method should be restricted to complicated SQL queries that can't be executed
|
2018-11-28 17:45:27 -05:00
|
|
|
# using the ActiveRecord::Calculations class methods. Look into those before using this method,
|
|
|
|
# as it could lock you into a specific database engine or require a code change to switch
|
|
|
|
# database engines.
|
2011-12-15 15:07:41 -05:00
|
|
|
#
|
2014-12-31 12:05:51 -05:00
|
|
|
# Product.count_by_sql "SELECT COUNT(*) FROM sales s, customers c WHERE s.customer_id = c.id"
|
|
|
|
# # => 12
|
2011-12-15 15:07:41 -05:00
|
|
|
#
|
2014-12-31 12:05:51 -05:00
|
|
|
# ==== Parameters
|
2011-12-15 15:07:41 -05:00
|
|
|
#
|
2014-12-31 12:05:51 -05:00
|
|
|
# * +sql+ - An SQL statement which should return a count query from the database, see the example above.
|
2011-12-15 15:07:41 -05:00
|
|
|
def count_by_sql(sql)
|
2016-07-31 00:11:30 -04:00
|
|
|
connection.select_value(sanitize_sql(sql), "#{name} Count").to_i
|
2011-12-15 15:07:41 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|