quoting.rb 3.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 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68
      # Cast a +value+ to a type that the database understands. For example,
      # SQLite does not understand dates, so this method will convert a Date
      # to a String.
      def type_cast(value, column)
        return value.id if value.respond_to?(:quoted_id)

        case value
        when String, ActiveSupport::Multibyte::Chars
          value = value.to_s
          return value unless column

          case column.type
          when :binary then value
          when :integer then value.to_i
          when :float then value.to_f
          else
            value
          end

        when true, false
          if column && column.type == :integer
            value ? 1 : 0
          else
            value ? 't' : 'f'
          end
          # BigDecimals need to be put in a non-normalized form and quoted.
        when nil        then nil
69
        when BigDecimal then value.to_s('F')
70 71 72 73 74 75 76 77
        when Numeric    then value
        when Date, Time then quoted_date(value)
        when Symbol     then value.to_s
        else
          YAML.dump(value)
        end
      end

78 79
      # Quotes a string, escaping any ' (single quote) and \ (backslash)
      # characters.
80 81 82 83
      def quote_string(s)
        s.gsub(/\\/, '\&\&').gsub(/'/, "''") # ' (for ruby-mode)
      end

84 85 86 87 88 89 90 91
      # 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)
92 93 94 95 96
      end

      def quoted_true
        "'t'"
      end
97

98 99 100
      def quoted_false
        "'f'"
      end
101

102
      def quoted_date(value)
103 104 105 106 107 108
        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)
109
      end
110 111
    end
  end
112
end