to_sql.rb 7.1 KB
Newer Older
A
Aaron Patterson 已提交
1 2
require 'bigdecimal'

A
Aaron Patterson 已提交
3 4 5 6 7 8 9 10 11 12 13 14 15 16
module Arel
  module Visitors
    class ToSql
      def initialize engine
        @engine     = engine
        @connection = nil
      end

      def accept object
        @connection = @engine.connection
        visit object
      end

      private
A
Aaron Patterson 已提交
17 18 19 20 21 22 23
      def visit_Arel_Nodes_DeleteStatement o
        [
          "DELETE FROM #{visit o.relation}",
          ("WHERE #{o.wheres.map { |x| visit x }.join ' AND '}" unless o.wheres.empty?)
        ].compact.join ' '
      end

A
Aaron Patterson 已提交
24 25 26
      def visit_Arel_Nodes_UpdateStatement o
        [
          "UPDATE #{visit o.relation}",
A
Aaron Patterson 已提交
27
          ("SET #{o.values.map { |value| visit value }.join ', '}" unless o.values.empty?),
A
Aaron Patterson 已提交
28 29 30
          ("WHERE #{o.wheres.map { |x| visit x }.join ' AND '}" unless o.wheres.empty?),
          ("ORDER BY #{o.orders.map { |x| visit x }.join(', ')}" unless o.orders.empty?),
          ("LIMIT #{o.limit}" if o.limit),
A
Aaron Patterson 已提交
31 32 33
        ].compact.join ' '
      end

34
      def visit_Arel_Nodes_InsertStatement o
35 36
        [
          "INSERT INTO #{visit o.relation}",
37 38 39 40 41

          ("(#{o.columns.map { |x|
                quote_column_name x.name
            }.join ', '})" unless o.columns.empty?),

42
          (visit o.values if o.values),
43
        ].compact.join ' '
44 45
      end

46 47 48 49
      def visit_Arel_Nodes_Exists o
        "EXISTS (#{visit o.select_stmt})"
      end

50
      def visit_Arel_Nodes_Values o
51 52
        "VALUES (#{o.expressions.zip(o.columns).map { |value, column|
          quote(value, column && column.column)
53 54 55
        }.join ', '})"
      end

A
Aaron Patterson 已提交
56 57 58
      def visit_Arel_Nodes_SelectStatement o
        [
          o.cores.map { |x| visit x }.join,
A
Aaron Patterson 已提交
59
          ("ORDER BY #{o.orders.map { |x| visit x }.join(', ')}" unless o.orders.empty?),
A
Aaron Patterson 已提交
60
          ("LIMIT #{o.limit}" if o.limit),
A
Aaron Patterson 已提交
61
          (visit(o.offset) if o.offset),
A
Aaron Patterson 已提交
62
          (visit(o.lock) if o.lock),
A
Aaron Patterson 已提交
63 64 65 66
        ].compact.join ' '
      end

      def visit_Arel_Nodes_SelectCore o
A
Aaron Patterson 已提交
67 68
        [
          "SELECT #{o.projections.map { |x| visit x }.join ', '}",
69
          ("FROM #{visit o.froms}" if o.froms),
A
Aaron Patterson 已提交
70
          ("WHERE #{o.wheres.map { |x| visit x }.join ' AND ' }" unless o.wheres.empty?),
A
Aaron Patterson 已提交
71 72
          ("GROUP BY #{o.groups.map { |x| visit x }.join ', ' }" unless o.groups.empty?),
          (visit(o.having) if o.having),
A
Aaron Patterson 已提交
73 74 75
        ].compact.join ' '
      end

A
Aaron Patterson 已提交
76 77 78 79
      def visit_Arel_Nodes_Having o
        "HAVING #{visit o.expr}"
      end

A
Aaron Patterson 已提交
80 81 82 83
      def visit_Arel_Nodes_Offset o
        "OFFSET #{visit o.value}"
      end

A
Aaron Patterson 已提交
84 85 86 87 88
      # FIXME: this does nothing on SQLLite3, but should do things on other
      # databases.
      def visit_Arel_Nodes_Lock o
      end

A
Aaron Patterson 已提交
89 90 91 92
      def visit_Arel_Nodes_Grouping o
        "(#{visit o.expr})"
      end

A
Aaron Patterson 已提交
93 94 95 96
      def visit_Arel_Nodes_Group o
        visit o.expr
      end

A
Aaron Patterson 已提交
97
      def visit_Arel_Nodes_Count o
98 99 100
        "COUNT(#{o.distinct ? 'DISTINCT ' : ''}#{o.expressions.map { |x|
          visit x
        }.join(', ')})#{o.alias ? " AS #{visit o.alias}" : ''}"
A
Aaron Patterson 已提交
101 102
      end

A
Aaron Patterson 已提交
103 104 105 106 107
      def visit_Arel_Nodes_Sum o
        "SUM(#{o.expressions.map { |x|
          visit x }.join(', ')})#{o.alias ? " AS #{visit o.alias}" : ''}"
      end

A
Aaron Patterson 已提交
108 109 110 111 112
      def visit_Arel_Nodes_Max o
        "MAX(#{o.expressions.map { |x|
          visit x }.join(', ')})#{o.alias ? " AS #{visit o.alias}" : ''}"
      end

E
Emilio Tagua 已提交
113 114 115 116 117
      def visit_Arel_Nodes_Min o
        "MIN(#{o.expressions.map { |x|
          visit x }.join(', ')})#{o.alias ? " AS #{visit o.alias}" : ''}"
      end

A
Aaron Patterson 已提交
118 119 120 121 122
      def visit_Arel_Nodes_Avg o
        "AVG(#{o.expressions.map { |x|
          visit x }.join(', ')})#{o.alias ? " AS #{visit o.alias}" : ''}"
      end

A
Aaron Patterson 已提交
123 124 125
      def visit_Arel_Nodes_TableAlias o
        "#{visit o.relation} #{quote_table_name o.name}"
      end
A
Aaron Patterson 已提交
126

127 128 129 130
      def visit_Arel_Nodes_Between o
        "#{visit o.left} BETWEEN #{visit o.right}"
      end

131 132 133 134
      def visit_Arel_Nodes_GreaterThanOrEqual o
        "#{visit o.left} >= #{visit o.right}"
      end

A
Aaron Patterson 已提交
135 136 137 138
      def visit_Arel_Nodes_GreaterThan o
        "#{visit o.left} > #{visit o.right}"
      end

139 140 141 142
      def visit_Arel_Nodes_LessThan o
        "#{visit o.left} < #{visit o.right}"
      end

143 144 145 146
      def visit_Arel_Nodes_StringJoin o
        "#{visit o.left} #{visit o.right}"
      end

A
Aaron Patterson 已提交
147
      def visit_Arel_Nodes_OuterJoin o
148
        "#{visit o.left} LEFT OUTER JOIN #{visit o.right} #{visit o.constraint}"
A
Aaron Patterson 已提交
149
      end
A
Aaron Patterson 已提交
150 151

      def visit_Arel_Nodes_InnerJoin o
152
        "#{visit o.left} INNER JOIN #{visit o.right} #{visit o.constraint if o.constraint}"
A
Aaron Patterson 已提交
153 154 155 156 157 158
      end

      def visit_Arel_Nodes_On o
        "ON #{visit o.expr}"
      end

A
Aaron Patterson 已提交
159
      def visit_Arel_Table o
160 161 162 163 164
        if o.table_alias
          "#{quote_table_name o.name} #{quote_table_name o.table_alias}"
        else
          quote_table_name o.name
        end
A
Aaron Patterson 已提交
165 166
      end

167
      def visit_Arel_Nodes_In o
168 169 170
        right = o.right
        right = right.empty? ? 'NULL' : right.map { |x| visit x }.join(', ')
        "#{visit o.left} IN (#{right})"
171 172
      end

A
Aaron Patterson 已提交
173 174 175 176
      def visit_Arel_Nodes_And o
        "#{visit o.left} AND #{visit o.right}"
      end

A
Aaron Patterson 已提交
177 178 179 180
      def visit_Arel_Nodes_Or o
        "#{visit o.left} OR #{visit o.right}"
      end

181
      def visit_Arel_Nodes_Assignment o
182
        right = quote(o.right, o.left.column)
A
Aaron Patterson 已提交
183 184 185
        "#{visit o.left} = #{right}"
      end

186 187 188 189 190 191 192 193 194 195
      def visit_Arel_Nodes_Equality o
        right = o.right

        if right.nil?
          "#{visit o.left} IS NULL"
        else
          "#{visit o.left} = #{visit right}"
        end
      end

196 197 198 199 200 201 202 203 204 205
      def visit_Arel_Nodes_NotEqual o
        right = o.right

        if right.nil?
          "#{visit o.left} IS NOT NULL"
        else
          "#{visit o.left} != #{visit right}"
        end
      end

A
Aaron Patterson 已提交
206 207
      def visit_Arel_Nodes_UnqualifiedColumn o
        "#{quote_column_name o.name}"
A
Aaron Patterson 已提交
208 209
      end

210
      def visit_Arel_Attributes_Attribute o
211 212
        join_name = o.relation.table_alias || o.relation.name
        "#{quote_table_name join_name}.#{quote_column_name o.name}"
A
Aaron Patterson 已提交
213
      end
214 215 216
      alias :visit_Arel_Attributes_Integer :visit_Arel_Attributes_Attribute
      alias :visit_Arel_Attributes_String :visit_Arel_Attributes_Attribute
      alias :visit_Arel_Attributes_Time :visit_Arel_Attributes_Attribute
A
Aaron Patterson 已提交
217
      alias :visit_Arel_Attributes_Boolean :visit_Arel_Attributes_Attribute
A
Aaron Patterson 已提交
218 219

      def visit_Fixnum o; o end
A
Aaron Patterson 已提交
220 221
      alias :visit_Arel_Nodes_SqlLiteral :visit_Fixnum
      alias :visit_Arel_SqlLiteral :visit_Fixnum # This is deprecated
A
Aaron Patterson 已提交
222

A
Aaron Patterson 已提交
223
      def visit_String o; quote(o) end
224 225 226 227 228 229 230 231 232 233 234

      alias :visit_ActiveSupport_Multibyte_Chars :visit_String
      alias :visit_BigDecimal :visit_String
      alias :visit_Date :visit_String
      alias :visit_DateTime :visit_String
      alias :visit_FalseClass :visit_String
      alias :visit_Float :visit_String
      alias :visit_Hash :visit_String
      alias :visit_Symbol :visit_String
      alias :visit_Time :visit_String
      alias :visit_TrueClass :visit_String
A
Aaron Patterson 已提交
235

236 237
      DISPATCH = Hash.new do |hash, klass|
        hash[klass] = "visit_#{klass.name.gsub('::', '_')}"
A
Aaron Patterson 已提交
238 239
      end

240 241
      def visit object
        send DISPATCH[object.class], object
A
Aaron Patterson 已提交
242 243
      end

244 245 246 247
      def quote value, column = nil
        @connection.quote value, column
      end

A
Aaron Patterson 已提交
248 249 250 251 252 253 254 255 256 257
      def quote_table_name name
        @connection.quote_table_name name
      end

      def quote_column_name name
        @connection.quote_column_name name
      end
    end
  end
end