summaryrefslogtreecommitdiff
path: root/lib/gitlab/sql
diff options
context:
space:
mode:
authorYorick Peterse <yorickpeterse@gmail.com>2015-11-12 16:16:49 +0100
committerYorick Peterse <yorickpeterse@gmail.com>2015-11-18 13:02:43 +0100
commitd769596aec7daa2ff43f86c8fad4211fbc4f607d (patch)
tree86f93b26b036cc3accb741691e7ce383f13cfa34 /lib/gitlab/sql
parent054f2f98eda60682fd796a1f66681accc6f7ce1c (diff)
downloadgitlab-ce-d769596aec7daa2ff43f86c8fad4211fbc4f607d.tar.gz
Added Gitlab::SQL::Union class
This class can be used to join multiple AcitveRecord::Relation objects together using a SQL UNION statement. ActiveRecord < 5.0 sadly doesn't support UNION and existing Gems out there don't handle prepared statements (e.g. they never incremented the variable bindings).
Diffstat (limited to 'lib/gitlab/sql')
-rw-r--r--lib/gitlab/sql/union.rb34
1 files changed, 34 insertions, 0 deletions
diff --git a/lib/gitlab/sql/union.rb b/lib/gitlab/sql/union.rb
new file mode 100644
index 00000000000..1a62eff0b31
--- /dev/null
+++ b/lib/gitlab/sql/union.rb
@@ -0,0 +1,34 @@
+module Gitlab
+ module SQL
+ # Class for building SQL UNION statements.
+ #
+ # ORDER BYs are dropped from the relations as the final sort order is not
+ # guaranteed any way.
+ #
+ # Example usage:
+ #
+ # union = Gitlab::SQL::Union.new(user.personal_projects, user.projects)
+ # sql = union.to_sql
+ #
+ # Project.where("id IN (#{sql})")
+ class Union
+ def initialize(relations)
+ @relations = relations
+ end
+
+ def to_sql
+ # Some relations may include placeholders for prepared statements, these
+ # aren't incremented properly when joining relations together this way.
+ # By using "unprepared_statements" we remove the usage of placeholders
+ # (thus fixing this problem), at a slight performance cost.
+ fragments = ActiveRecord::Base.connection.unprepared_statement do
+ @relations.map do |rel|
+ "(#{rel.reorder(nil).to_sql})"
+ end
+ end
+
+ fragments.join(' UNION ')
+ end
+ end
+ end
+end