blob: 46a3ad4c786be5eb8e5b4bf0e46ea9bbda72ba7c (
plain) (
blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
|
module Arel
module Visitors
class Oracle < Arel::Visitors::ToSql
private
def visit_Arel_Nodes_SelectStatement o
order_hacks(o)
if o.limit && o.orders.empty? && !o.offset
o.cores.last.wheres.push Nodes::LessThanOrEqual.new(
Nodes::SqlLiteral.new('ROWNUM'), o.limit
)
o.limit = nil
return super
end
if o.limit && o.offset
limit = o.limit.to_i
offset = o.offset
o.limit = nil
o.offset = nil
sql = super
return <<-eosql
SELECT * FROM (
SELECT raw_sql_.*, rownum raw_rnum_
FROM (#{sql}) raw_sql_
WHERE rownum <= #{offset.value.to_i + limit}
)
WHERE #{visit offset}
eosql
end
if o.limit && !o.orders.empty?
limit = o.limit
o.limit = nil
return "SELECT * FROM (#{super}) WHERE ROWNUM <= #{limit}"
end
super
end
def visit_Arel_Nodes_Offset o
"raw_rnum_ > #{visit o.value}"
end
###
# Hacks for the order clauses specific to Oracle
def order_hacks o
return if o.orders.empty?
return unless o.cores.any? do |core|
core.projections.any? do |projection|
/DISTINCT.*FIRST_VALUE/ === projection
end
end
orders = o.orders
o.orders = []
orders.each_with_index do |order, i|
o.orders <<
Nodes::SqlLiteral.new("alias_#{i}__ #{'DESC' if /\bdesc$/i === order}")
end
end
end
end
end
|