quoting.rb 5.9 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
        if column
13 14 15 16 17 18
          ActiveSupport::Deprecation.warn(<<-MSG.squish)
            Passing a column to `quote` has been deprecated. It is only used
            for type casting, which should be handled elsewhere. See
            https://github.com/rails/arel/commit/6160bfbda1d1781c3b08a33ec4955f170e95be11
            for more information.
          MSG
19
          value = type_cast_from_column(column, value)
20 21
        end

22
        _quote(value)
23 24
      end

25 26 27
      # 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.
28
      def type_cast(value, column = nil)
29 30 31
        if value.respond_to?(:quoted_id) && value.respond_to?(:id)
          return value.id
        end
32

33
        if column
34
          value = type_cast_from_column(column, value)
35 36
        end

37 38 39 40
        _type_cast(value)
      rescue TypeError
        to_type = column ? " to #{column.type}" : ""
        raise TypeError, "can't cast #{value.class}#{to_type}"
41 42
      end

43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65
      # If you are having to call this function, you are likely doing something
      # wrong. The column does not have sufficient type information if the user
      # provided a custom type on the class level either explicitly (via
      # `attribute`) or implicitly (via `serialize`,
      # `time_zone_aware_attributes`). In almost all cases, the sql type should
      # only be used to change quoting behavior, when the primitive to
      # represent the type doesn't sufficiently reflect the differences
      # (varchar vs binary) for example. The type used to get this primitive
      # should have been provided before reaching the connection adapter.
      def type_cast_from_column(column, value) # :nodoc:
        if column
          type = lookup_cast_type_from_column(column)
          type.type_cast_for_database(value)
        else
          value
        end
      end

      # See docs for +type_cast_from_column+
      def lookup_cast_type_from_column(column) # :nodoc:
        lookup_cast_type(column.sql_type)
      end

S
Sean Griffin 已提交
66 67 68 69 70 71 72 73 74 75 76
      def fetch_type_metadata(sql_type)
        cast_type = lookup_cast_type(sql_type)
        SqlTypeMetadata.new(
          sql_type: sql_type,
          type: cast_type.type,
          limit: cast_type.limit,
          precision: cast_type.precision,
          scale: cast_type.scale,
        )
      end

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

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

93 94 95 96 97 98
      # Override to return the quoted table name for assignment. Defaults to
      # table quoting.
      #
      # This works for mysql and mysql2 where table.column can be used to
      # resolve ambiguity.
      #
99
      # We override this in the sqlite3 and postgresql adapters to use only
100 101 102 103 104
      # the column name (as per syntax requirements).
      def quote_table_name_for_assignment(table, attr)
        quote_table_name("#{table}.#{attr}")
      end

105 106 107 108 109
      def quote_default_expression(value, column) #:nodoc:
        value = lookup_cast_type(column.sql_type).type_cast_for_database(value)
        quote(value)
      end

110 111 112
      def quoted_true
        "'t'"
      end
113

114 115 116 117
      def unquoted_true
        't'
      end

118 119 120
      def quoted_false
        "'f'"
      end
121

122 123 124 125
      def unquoted_false
        'f'
      end

126
      def quoted_date(value)
127 128
        if value.acts_like?(:time)
          zone_conversion_method = ActiveRecord::Base.default_timezone == :utc ? :getutc : :getlocal
J
Jon Leighton 已提交
129 130 131 132 133 134

          if value.respond_to?(zone_conversion_method)
            value = value.send(zone_conversion_method)
          end
        end

135 136 137 138 139 140
        result = value.to_s(:db)
        if value.respond_to?(:usec) && value.usec > 0
          "#{result}.#{sprintf("%06d", value.usec)}"
        else
          result
        end
141
      end
142

143
      def prepare_binds_for_database(binds) # :nodoc:
S
Sean Griffin 已提交
144
        binds.map(&:value_for_database)
145 146
      end

147 148
      private

149 150 151
      def types_which_need_no_typecasting
        [nil, Numeric, String]
      end
152 153 154 155 156 157 158 159 160 161 162 163 164

      def _quote(value)
        case value
        when String, ActiveSupport::Multibyte::Chars, Type::Binary::Data
          "'#{quote_string(value.to_s)}'"
        when true       then quoted_true
        when false      then quoted_false
        when nil        then "NULL"
        # BigDecimals need to be put in a non-normalized form and quoted.
        when BigDecimal then value.to_s('F')
        when Numeric, ActiveSupport::Duration then value.to_s
        when Date, Time then "'#{quoted_date(value)}'"
        when Symbol     then "'#{quote_string(value.to_s)}'"
165
        when Class      then "'#{value}'"
166
        else raise TypeError, "can't quote #{value.class.name}"
167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183
        end
      end

      def _type_cast(value)
        case value
        when Symbol, ActiveSupport::Multibyte::Chars, Type::Binary::Data
          value.to_s
        when true       then unquoted_true
        when false      then unquoted_false
        # BigDecimals need to be put in a non-normalized form and quoted.
        when BigDecimal then value.to_s('F')
        when Date, Time then quoted_date(value)
        when *types_which_need_no_typecasting
          value
        else raise TypeError
        end
      end
184 185
    end
  end
186
end