aboutsummaryrefslogtreecommitdiffstats
path: root/lib/arel/engines
diff options
context:
space:
mode:
authorEmilio Tagua <miloops@gmail.com>2009-06-23 17:37:10 -0300
committerEmilio Tagua <miloops@gmail.com>2009-06-23 17:37:10 -0300
commita9486193c8c6250d56bfb0258be50025f7f62f5e (patch)
tree9bea10577f8a606371745f7fede93a3fd04da116 /lib/arel/engines
parentae0c58fed1cd3aca1edd75a249fa99bc9571a8a3 (diff)
downloadrails-a9486193c8c6250d56bfb0258be50025f7f62f5e.tar.gz
rails-a9486193c8c6250d56bfb0258be50025f7f62f5e.tar.bz2
rails-a9486193c8c6250d56bfb0258be50025f7f62f5e.zip
Build valid SQL query when using PostreSQL with given order and DISTINCT
ON clause. TODO: refactoring to clean up components.
Diffstat (limited to 'lib/arel/engines')
-rw-r--r--lib/arel/engines/sql/relations/relation.rb24
1 files changed, 24 insertions, 0 deletions
diff --git a/lib/arel/engines/sql/relations/relation.rb b/lib/arel/engines/sql/relations/relation.rb
index 4cfb83a601..ed7f19ddbd 100644
--- a/lib/arel/engines/sql/relations/relation.rb
+++ b/lib/arel/engines/sql/relations/relation.rb
@@ -5,6 +5,26 @@ module Arel
end
def select_sql
+ if engine.adapter_name == "PostgreSQL" && !orders.blank? && using_distinct_on?
+ # PostgreSQL does not allow arbitrary ordering when using DISTINCT ON, so we work around this
+ # by wrapping the +sql+ string as a sub-select and ordering in that query.
+ order = order_clauses.join(', ').split(',').map { |s| s.strip }.reject(&:blank?)
+ order = order.zip((0...order.size).to_a).map { |s,i| "id_list.alias_#{i} #{'DESC' if s =~ /\bdesc$/i}" }.join(', ')
+
+ query = build_query \
+ "SELECT #{select_clauses.to_s}",
+ "FROM #{table_sql(Sql::TableReference.new(self))}",
+ (joins(self) unless joins(self).blank? ),
+ ("WHERE #{where_clauses.join("\n\tAND ")}" unless wheres.blank? ),
+ ("GROUP BY #{group_clauses.join(', ')}" unless groupings.blank? )
+
+ build_query \
+ "SELECT * FROM (#{query}) AS id_list",
+ "ORDER BY #{order}",
+ ("LIMIT #{taken}" unless taken.blank? ),
+ ("OFFSET #{skipped}" unless skipped.blank? )
+
+ else
build_query \
"SELECT #{select_clauses.join(', ')}",
"FROM #{table_sql(Sql::TableReference.new(self))}",
@@ -14,6 +34,7 @@ module Arel
("ORDER BY #{order_clauses.join(', ')}" unless orders.blank? ),
("LIMIT #{taken}" unless taken.blank? ),
("OFFSET #{skipped}" unless skipped.blank? )
+ end
end
def inclusion_predicate_sql
@@ -46,5 +67,8 @@ module Arel
orders.collect { |o| o.to_sql(Sql::OrderClause.new(self)) }
end
+ def using_distinct_on?
+ select_clauses.any? { |x| x =~ /DISTINCT ON/ }
+ end
end
end