debian-mirror-gitlab/lib/gitlab/sql/set_operator.rb

60 lines
1.8 KiB
Ruby
Raw Normal View History

2020-11-24 15:15:51 +05:30
# frozen_string_literal: true
module Gitlab
module SQL
# Class for building SQL set operator statements (UNION, INTERSECT, and
# EXCEPT).
#
# ORDER BYs are dropped from the relations as the final sort order is not
# guaranteed any way.
#
2021-04-29 21:17:54 +05:30
# remove_order: false option can be used in special cases where the
# ORDER BY is necessary for the query.
#
2020-11-24 15:15:51 +05:30
# Example usage:
#
# union = Gitlab::SQL::Union.new([user.personal_projects, user.projects])
# sql = union.to_sql
#
# Project.where("id IN (#{sql})")
class SetOperator
2021-04-29 21:17:54 +05:30
def initialize(relations, remove_duplicates: true, remove_order: true)
2020-11-24 15:15:51 +05:30
@relations = relations
@remove_duplicates = remove_duplicates
2021-04-29 21:17:54 +05:30
@remove_order = remove_order
2020-11-24 15:15:51 +05:30
end
def self.operator_keyword
raise NotImplementedError
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
2021-04-29 21:17:54 +05:30
relations.map do |rel|
remove_order ? rel.reorder(nil).to_sql : rel.to_sql
end.reject(&:blank?)
2020-11-24 15:15:51 +05:30
end
if fragments.any?
"(" + fragments.join(")\n#{operator_keyword_fragment}\n(") + ")"
else
'NULL'
end
end
# UNION [ALL] | INTERSECT [ALL] | EXCEPT [ALL]
def operator_keyword_fragment
remove_duplicates ? self.class.operator_keyword : "#{self.class.operator_keyword} ALL"
end
private
2021-04-29 21:17:54 +05:30
attr_reader :relations, :remove_duplicates, :remove_order
2020-11-24 15:15:51 +05:30
end
end
end