to_sql.rb 6.7 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 31
          ("WHERE #{o.wheres.map { |x| visit x }.join ' AND '}" unless o.wheres.empty?)
        ].compact.join ' '
      end

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

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

40
          (visit o.values if o.values),
41
        ].compact.join ' '
42 43
      end

44 45
      def visit_Arel_Nodes_Values o
        "VALUES (#{o.expressions.map { |value|
46
          value.nil? ? 'NULL' : visit(value)
47 48 49
        }.join ', '})"
      end

A
Aaron Patterson 已提交
50 51 52
      def visit_Arel_Nodes_SelectStatement o
        [
          o.cores.map { |x| visit x }.join,
A
Aaron Patterson 已提交
53
          ("ORDER BY #{o.orders.map { |x| visit x }.join(', ')}" unless o.orders.empty?),
A
Aaron Patterson 已提交
54
          ("LIMIT #{o.limit}" if o.limit),
A
Aaron Patterson 已提交
55
          (visit(o.offset) if o.offset),
A
Aaron Patterson 已提交
56
          (visit(o.lock) if o.lock),
A
Aaron Patterson 已提交
57 58 59 60
        ].compact.join ' '
      end

      def visit_Arel_Nodes_SelectCore o
A
Aaron Patterson 已提交
61 62
        [
          "SELECT #{o.projections.map { |x| visit x }.join ', '}",
A
Aaron Patterson 已提交
63
          ("FROM #{o.froms.map { |x| visit x }.join ', ' }" unless o.froms.empty?),
A
Aaron Patterson 已提交
64
          ("WHERE #{o.wheres.map { |x| visit x }.join ' AND ' }" unless o.wheres.empty?),
A
Aaron Patterson 已提交
65 66
          ("GROUP BY #{o.groups.map { |x| visit x }.join ', ' }" unless o.groups.empty?),
          (visit(o.having) if o.having),
A
Aaron Patterson 已提交
67 68 69
        ].compact.join ' '
      end

A
Aaron Patterson 已提交
70 71 72 73
      def visit_Arel_Nodes_Having o
        "HAVING #{visit o.expr}"
      end

A
Aaron Patterson 已提交
74 75 76 77
      def visit_Arel_Nodes_Offset o
        "OFFSET #{visit o.value}"
      end

A
Aaron Patterson 已提交
78 79 80 81 82
      # FIXME: this does nothing on SQLLite3, but should do things on other
      # databases.
      def visit_Arel_Nodes_Lock o
      end

A
Aaron Patterson 已提交
83 84 85 86
      def visit_Arel_Nodes_Group o
        visit o.expr
      end

A
Aaron Patterson 已提交
87
      def visit_Arel_Nodes_Count o
88 89 90
        "COUNT(#{o.distinct ? 'DISTINCT ' : ''}#{o.expressions.map { |x|
          visit x
        }.join(', ')})#{o.alias ? " AS #{visit o.alias}" : ''}"
A
Aaron Patterson 已提交
91 92
      end

A
Aaron Patterson 已提交
93 94 95 96 97
      def visit_Arel_Nodes_Sum o
        "SUM(#{o.expressions.map { |x|
          visit x }.join(', ')})#{o.alias ? " AS #{visit o.alias}" : ''}"
      end

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

E
Emilio Tagua 已提交
103 104 105 106 107
      def visit_Arel_Nodes_Min o
        "MIN(#{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_Avg o
        "AVG(#{o.expressions.map { |x|
          visit x }.join(', ')})#{o.alias ? " AS #{visit o.alias}" : ''}"
      end

A
Aaron Patterson 已提交
113 114 115
      def visit_Arel_Nodes_TableAlias o
        "#{visit o.relation} #{quote_table_name o.name}"
      end
A
Aaron Patterson 已提交
116

117 118 119 120
      def visit_Arel_Nodes_GreaterThanOrEqual o
        "#{visit o.left} >= #{visit o.right}"
      end

A
Aaron Patterson 已提交
121 122 123 124
      def visit_Arel_Nodes_GreaterThan o
        "#{visit o.left} > #{visit o.right}"
      end

125 126 127 128
      def visit_Arel_Nodes_StringJoin o
        "#{visit o.left} #{visit o.right}"
      end

A
Aaron Patterson 已提交
129
      def visit_Arel_Nodes_OuterJoin o
130
        "#{visit o.left} LEFT OUTER JOIN #{visit o.right} #{visit o.constraint}"
A
Aaron Patterson 已提交
131
      end
A
Aaron Patterson 已提交
132 133

      def visit_Arel_Nodes_InnerJoin o
134
        "#{visit o.left} INNER JOIN #{visit o.right} #{visit o.constraint if o.constraint}"
A
Aaron Patterson 已提交
135 136 137 138 139 140
      end

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

A
Aaron Patterson 已提交
141 142 143 144
      def visit_Arel_Table o
        quote_table_name o.name
      end

145
      def visit_Arel_Nodes_In o
A
Aaron Patterson 已提交
146
        "#{visit o.left} IN (#{o.right.map { |x| visit x }.join ', '})"
147 148
      end

A
Aaron Patterson 已提交
149 150 151 152
      def visit_Arel_Nodes_And o
        "#{visit o.left} AND #{visit o.right}"
      end

A
Aaron Patterson 已提交
153 154 155 156
      def visit_Arel_Nodes_Or o
        "#{visit o.left} OR #{visit o.right}"
      end

157
      def visit_Arel_Nodes_Assignment o
A
Aaron Patterson 已提交
158
        right = o.right
159

160
        right = right.nil? ? 'NULL' : visit(right)
A
Aaron Patterson 已提交
161 162 163
        "#{visit o.left} = #{right}"
      end

164 165 166 167 168 169 170 171 172 173
      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

174 175 176 177 178 179 180 181 182 183
      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 已提交
184 185
      def visit_Arel_Nodes_UnqualifiedColumn o
        "#{quote_column_name o.name}"
A
Aaron Patterson 已提交
186 187
      end

188
      def visit_Arel_Attributes_Attribute o
A
Aaron Patterson 已提交
189 190
        "#{quote_table_name o.relation.name}.#{quote_column_name o.name}"
      end
191 192 193
      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 已提交
194
      alias :visit_Arel_Attributes_Boolean :visit_Arel_Attributes_Attribute
A
Aaron Patterson 已提交
195 196

      def visit_Fixnum o; o end
A
Aaron Patterson 已提交
197 198
      alias :visit_Arel_Nodes_SqlLiteral :visit_Fixnum
      alias :visit_Arel_SqlLiteral :visit_Fixnum # This is deprecated
A
Aaron Patterson 已提交
199

A
Aaron Patterson 已提交
200 201
      def visit_TrueClass o; quote(o) end
      def visit_String o; quote(o) end
E
Emilio Tagua 已提交
202
      def visit_Hash o; quote(o) end
203
      def visit_ActiveSupport_Multibyte_Chars o; quote(o) end if defined?(ActiveSupport)
204
      def visit_Symbol o; quote(o) end
A
Aaron Patterson 已提交
205
      def visit_Time o; quote(o) end
A
Aaron Patterson 已提交
206
      def visit_Date o; quote(o) end
A
Aaron Patterson 已提交
207
      def visit_DateTime o; quote(o) end
A
Aaron Patterson 已提交
208
      def visit_Float o; quote(o) end
A
Aaron Patterson 已提交
209
      def visit_BigDecimal o; quote(o) end
210
      def visit_FalseClass o; quote(o) end
A
Aaron Patterson 已提交
211

A
Aaron Patterson 已提交
212
      DISPATCH = {}
A
Aaron Patterson 已提交
213
      def visit object
A
Aaron Patterson 已提交
214 215
        send "visit_#{object.class.name.gsub('::', '_')}", object
        #send DISPATCH[object.class], object
A
Aaron Patterson 已提交
216 217 218 219 220 221 222
      end

      private_instance_methods(false).each do |method|
        method = method.to_s
        next unless method =~ /^visit_(.*)$/
        const = $1.split('_').inject(Object) { |m,s| m.const_get s }
        DISPATCH[const] = method
A
Aaron Patterson 已提交
223 224
      end

225 226 227 228
      def quote value, column = nil
        @connection.quote value, column
      end

A
Aaron Patterson 已提交
229 230 231 232 233 234 235 236 237 238
      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