quoting.rb 2.3 KB
Newer Older
J
Jeremy Kemper 已提交
1 2
require 'active_support/core_ext/big_decimal/conversions'

3 4 5
module ActiveRecord
  module ConnectionAdapters # :nodoc:
    module Quoting
6 7
      # Quotes the column value to help prevent
      # {SQL injection attacks}[http://en.wikipedia.org/wiki/SQL_injection].
8
      def quote(value, column = nil)
9 10 11
        # records are quoted as their primary key
        return value.quoted_id if value.respond_to?(:quoted_id)

12
        case value
A
Aaron Patterson 已提交
13 14
        when String, ActiveSupport::Multibyte::Chars
          value = value.to_s
15 16
          return "'#{quote_string(value)}'" unless column

A
Aaron Patterson 已提交
17 18 19 20
          case column.type
          when :binary then "'#{quote_string(column.string_to_binary(value))}'"
          when :integer then value.to_i.to_s
          when :float then value.to_f.to_s
A
Aaron Patterson 已提交
21
          else
A
Aaron Patterson 已提交
22
            "'#{quote_string(value)}'"
A
Aaron Patterson 已提交
23
          end
24

A
Aaron Patterson 已提交
25 26 27 28 29 30
        when true, false
          if column && column.type == :integer
            value ? '1' : '0'
          else
            value ? quoted_true : quoted_false
          end
31
          # BigDecimals need to be put in a non-normalized form and quoted.
A
Aaron Patterson 已提交
32
        when nil        then "NULL"
33 34 35
        when BigDecimal then value.to_s('F')
        when Numeric    then value.to_s
        when Date, Time then "'#{quoted_date(value)}'"
P
Pratik Naik 已提交
36
        when Symbol     then "'#{quote_string(value.to_s)}'"
A
Aaron Patterson 已提交
37
        else
38
          "'#{quote_string(YAML.dump(value))}'"
39 40 41
        end
      end

42 43
      # Quotes a string, escaping any ' (single quote) and \ (backslash)
      # characters.
44 45 46 47
      def quote_string(s)
        s.gsub(/\\/, '\&\&').gsub(/'/, "''") # ' (for ruby-mode)
      end

48 49 50 51 52 53 54 55
      # Quotes the column name. Defaults to no quoting.
      def quote_column_name(column_name)
        column_name
      end

      # Quotes the table name. Defaults to column name quoting.
      def quote_table_name(table_name)
        quote_column_name(table_name)
56 57 58 59 60
      end

      def quoted_true
        "'t'"
      end
61

62 63 64
      def quoted_false
        "'f'"
      end
65

66
      def quoted_date(value)
67 68 69 70 71 72
        if value.acts_like?(:time)
          zone_conversion_method = ActiveRecord::Base.default_timezone == :utc ? :getutc : :getlocal
          value.respond_to?(zone_conversion_method) ? value.send(zone_conversion_method) : value
        else
          value
        end.to_s(:db)
73
      end
74 75
    end
  end
76
end