attribute_methods.rb 12.6 KB
Newer Older
J
Jeremy Kemper 已提交
1 2
require 'active_support/core_ext/enumerable'

3
module ActiveRecord
4
  # = Active Record Attribute Methods
5
  module AttributeMethods
6
    extend ActiveSupport::Concern
7
    include ActiveModel::AttributeMethods
8

9 10 11 12 13 14 15 16 17 18 19
    included do
      include Read
      include Write
      include BeforeTypeCast
      include Query
      include PrimaryKey
      include TimeZoneConversion
      include Dirty
      include Serialization
    end

20
    module ClassMethods
21 22
      # Generates all the attribute related methods for columns in the database
      # accessors, mutators and query methods.
23
      def define_attribute_methods # :nodoc:
J
Jon Leighton 已提交
24 25 26 27 28 29 30 31
        # Use a mutex; we don't want two thread simaltaneously trying to define
        # attribute methods.
        @attribute_methods_mutex.synchronize do
          return if attribute_methods_generated?
          superclass.define_attribute_methods unless self == base_class
          super(column_names)
          @attribute_methods_generated = true
        end
32 33
      end

34
      def attribute_methods_generated? # :nodoc:
J
Jon Leighton 已提交
35
        @attribute_methods_generated ||= false
36 37
      end

38
      def undefine_attribute_methods # :nodoc:
39
        super if attribute_methods_generated?
J
Jon Leighton 已提交
40
        @attribute_methods_generated = false
41 42
      end

43 44 45 46 47 48 49 50 51 52 53 54 55 56
      # Raises a <tt>ActiveRecord::DangerousAttributeError</tt> exception when an
      # \Active \Record method is defined in the model, otherwise +false+.
      #
      #   class Person < ActiveRecord::Base
      #     def save
      #       'already defined by Active Record'
      #     end
      #   end
      #
      #   Person.instance_method_already_implemented?(:save)
      #   # => ActiveRecord::DangerousAttributeError: save is defined by ActiveRecord
      #
      #   Person.instance_method_already_implemented?(:name)
      #   # => false
57
      def instance_method_already_implemented?(method_name)
58 59 60 61
        if dangerous_attribute_method?(method_name)
          raise DangerousAttributeError, "#{method_name} is defined by ActiveRecord"
        end

62
        if [Base, Model].include?(active_record_super)
J
Jon Leighton 已提交
63 64
          super
        else
65 66 67
          # If B < A and A defines its own attribute method, then we don't want to overwrite that.
          defined = method_defined_within?(method_name, superclass, superclass.generated_attribute_methods)
          defined && !ActiveRecord::Base.method_defined?(method_name) || super
J
Jon Leighton 已提交
68
        end
69
      end
70

71 72
      # A method name is 'dangerous' if it is already defined by Active Record, but
      # not by any ancestors. (So 'puts' is not dangerous but 'save' is.)
73
      def dangerous_attribute_method?(name) # :nodoc:
J
Jon Leighton 已提交
74 75
        method_defined_within?(name, Base)
      end
76

77
      def method_defined_within?(name, klass, sup = klass.superclass) # :nodoc:
J
Jon Leighton 已提交
78 79 80 81 82 83 84 85 86
        if klass.method_defined?(name) || klass.private_method_defined?(name)
          if sup.method_defined?(name) || sup.private_method_defined?(name)
            klass.instance_method(name).owner != sup.instance_method(name).owner
          else
            true
          end
        else
          false
        end
87
      end
88

89 90 91 92 93 94 95 96 97
      # Returns +true+ if +attribute+ is an attribute method and table exists,
      # +false+ otherwise.
      #
      #   class Person < ActiveRecord::Base
      #   end
      #
      #   Person.attribute_method?('name')   # => true
      #   Person.attribute_method?(:age=)    # => true
      #   Person.attribute_method?(:nothing) # => false
98 99 100 101
      def attribute_method?(attribute)
        super || (table_exists? && column_names.include?(attribute.to_s.sub(/=$/, '')))
      end

102 103 104 105 106 107 108 109
      # Returns an array of column names as strings if it's not an abstract class and
      # table exists. Otherwise it returns an empty array.
      #
      #   class Person < ActiveRecord::Base
      #   end
      #
      #   Person.attribute_names
      #   # => ["id", "created_at", "updated_at", "name", "age"]
110 111 112 113 114 115 116
      def attribute_names
        @attribute_names ||= if !abstract_class? && table_exists?
            column_names
          else
            []
          end
      end
117
    end
J
Joshua Peek 已提交
118

119 120
    # If we haven't generated any methods yet, generate them, then
    # see if we've created the method we're looking for.
121
    def method_missing(method, *args, &block) # :nodoc:
122
      unless self.class.attribute_methods_generated?
123
        self.class.define_attribute_methods
124 125 126 127 128 129

        if respond_to_without_attributes?(method)
          send(method, *args, &block)
        else
          super
        end
130 131
      else
        super
132 133 134
      end
    end

135
    def attribute_missing(match, *args, &block) # :nodoc:
136 137 138 139 140 141 142 143 144 145 146 147 148
      if self.class.columns_hash[match.attr_name]
        ActiveSupport::Deprecation.warn(
          "The method `#{match.method_name}', matching the attribute `#{match.attr_name}' has " \
          "dispatched through method_missing. This shouldn't happen, because `#{match.attr_name}' " \
          "is a column of the table. If this error has happened through normal usage of Active " \
          "Record (rather than through your own code or external libraries), please report it as " \
          "a bug."
        )
      end

      super
    end

149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164
    # A Person object with a name attribute can ask <tt>person.respond_to?(:name)</tt>,
    # <tt>person.respond_to?(:name=)</tt>, and <tt>person.respond_to?(:name?)</tt>
    # which will all return +true+. It also define the attribute methods if they have
    # not been generated.
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.new
    #   person.respond_to(:name)    # => true
    #   person.respond_to(:name=)   # => true
    #   person.respond_to(:name?)   # => true
    #   person.respond_to('age')    # => true
    #   person.respond_to('age=')   # => true
    #   person.respond_to('age?')   # => true
    #   person.respond_to(:nothing) # => false
165
    def respond_to?(name, include_private = false)
166
      self.class.define_attribute_methods unless self.class.attribute_methods_generated?
167 168
      super
    end
169

170 171 172 173 174 175 176 177 178
    # Returns +true+ if the given attribute is in the attributes hash, otherwise +false+.
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.new
    #   person.has_attribute?(:name)    # => true
    #   person.has_attribute?('age')    # => true
    #   person.has_attribute?(:nothing) # => false
179 180 181 182 183
    def has_attribute?(attr_name)
      @attributes.has_key?(attr_name.to_s)
    end

    # Returns an array of names for the attributes available on this object.
184 185 186 187 188 189 190
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.new
    #   person.attribute_names
    #   # => ["id", "created_at", "updated_at", "name", "age"]
191 192 193 194 195
    def attribute_names
      @attributes.keys
    end

    # Returns a hash of all the attributes with their names as keys and the values of the attributes as values.
196 197 198 199 200 201 202
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.create(name: 'Francesco', age: 22)
    #   person.attributes
    #   # => {"id"=>3, "created_at"=>Sun, 21 Oct 2012 04:53:04, "updated_at"=>Sun, 21 Oct 2012 04:53:04, "name"=>"Francesco", "age"=>22}
203
    def attributes
204 205 206
      attribute_names.each_with_object({}) { |name, attrs|
        attrs[name] = read_attribute(name)
      }
207 208 209 210 211 212 213 214
    end

    # Returns an <tt>#inspect</tt>-like string for the value of the
    # attribute +attr_name+. String attributes are truncated upto 50
    # characters, and Date and Time attributes are returned in the
    # <tt>:db</tt> format. Other attributes return the value of
    # <tt>#inspect</tt> without modification.
    #
215
    #   person = Person.create!(name: 'David Heinemeier Hansson ' * 3)
216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233
    #
    #   person.attribute_for_inspect(:name)
    #   # => '"David Heinemeier Hansson David Heinemeier Hansson D..."'
    #
    #   person.attribute_for_inspect(:created_at)
    #   # => '"2009-01-12 04:48:57"'
    def attribute_for_inspect(attr_name)
      value = read_attribute(attr_name)

      if value.is_a?(String) && value.length > 50
        "#{value[0..50]}...".inspect
      elsif value.is_a?(Date) || value.is_a?(Time)
        %("#{value.to_s(:db)}")
      else
        value.inspect
      end
    end

234 235 236 237 238 239 240 241 242 243 244
    # Returns +true+ if the specified +attribute+ has been set by the user or by a
    # database load and is neither +nil+ nor <tt>empty?</tt> (the latter only applies
    # to objects that respond to <tt>empty?</tt>, most notably Strings). Otherwise, +false+.
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.new(name: '')
    #   person.attribute_present?(:name) # => false
    #   person.name = 'Francesco'
    #   person.attribute_present?(:name) # => true
245 246
    def attribute_present?(attribute)
      value = read_attribute(attribute)
247
      !value.nil? && !(value.respond_to?(:empty?) && value.empty?)
248 249
    end

250 251 252 253 254 255 256 257 258 259 260 261
    # Returns the column object for the named attribute. Returns +nil+ if the
    # named attribute not exists.
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.new
    #   person.column_for_attribute(:name) # the result depends on the ConnectionAdapter
    #   # => #<ActiveRecord::ConnectionAdapters::SQLite3Column:0x007ff4ab083980 @name="name", @sql_type="varchar(255)", @null=true, ...>
    #
    #   person.column_for_attribute(:nothing)
    #   # => nil
262
    def column_for_attribute(name)
A
Aaron Patterson 已提交
263
      # FIXME: should this return a null object for columns that don't exist?
264 265 266
      self.class.columns_hash[name.to_s]
    end

267 268
    # Returns the value of the attribute identified by <tt>attr_name</tt> after it has been typecast (for example,
    # "2004-12-12" in a data column is cast to a date object, like Date.new(2004, 12, 12)).
269 270 271 272 273 274 275 276
    # (Alias for the protected <tt>read_attribute</tt> method).
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.new(name: 'Francesco', age: '22'
    #   person[:name] # => "Francesco"
    #   person[:age]  # => 22
277 278 279 280 281
    def [](attr_name)
      read_attribute(attr_name)
    end

    # Updates the attribute identified by <tt>attr_name</tt> with the specified +value+.
282 283 284 285 286 287 288 289 290
    # (Alias for the protected <tt>write_attribute</tt> method).
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.new
    #   person[:age] = '22'
    #   person[:age] # => 22
    #   person[:age] # => Fixnum
291 292 293 294
    def []=(attr_name, value)
      write_attribute(attr_name, value)
    end

295
    protected
296

297
    def clone_attributes(reader_method = :read_attribute, attributes = {}) # :nodoc:
298 299 300 301 302 303
      attribute_names.each do |name|
        attributes[name] = clone_attribute_value(reader_method, name)
      end
      attributes
    end

304
    def clone_attribute_value(reader_method, attribute_name) # :nodoc:
305 306 307 308 309 310
      value = send(reader_method, attribute_name)
      value.duplicable? ? value.clone : value
    rescue TypeError, NoMethodError
      value
    end

311
    def arel_attributes_with_values_for_create(attribute_names) # :nodoc:
J
Jon Leighton 已提交
312
      arel_attributes_with_values(attributes_for_create(attribute_names))
R
Robin Roestenburg 已提交
313
    end
314

315
    def arel_attributes_with_values_for_update(attribute_names) # :nodoc:
R
Robin Roestenburg 已提交
316
      arel_attributes_with_values(attributes_for_update(attribute_names))
317 318
    end

319
    def attribute_method?(attr_name) # :nodoc:
320
      defined?(@attributes) && @attributes.include?(attr_name)
321 322
    end

R
Robin Roestenburg 已提交
323
    private
324

R
Robin Roestenburg 已提交
325 326 327 328 329
    # Returns a Hash of the Arel::Attributes and attribute values that have been
    # type casted for use in an Arel insert/update method.
    def arel_attributes_with_values(attribute_names)
      attrs = {}
      arel_table = self.class.arel_table
330

R
Robin Roestenburg 已提交
331 332 333
      attribute_names.each do |name|
        attrs[arel_table[name]] = typecasted_attribute_value(name)
      end
334 335 336
      attrs
    end

R
Robin Roestenburg 已提交
337 338 339 340 341 342 343 344 345
    # Filters the primary keys and readonly attributes from the attribute names.
    def attributes_for_update(attribute_names)
      attribute_names.select do |name|
        column_for_attribute(name) && !pk_attribute?(name) && !readonly_attribute?(name)
      end
    end

    # Filters out the primary keys, from the attribute names, when the primary
    # key is to be generated (e.g. the id attribute has no value).
J
Jon Leighton 已提交
346 347 348
    def attributes_for_create(attribute_names)
      attribute_names.select do |name|
        column_for_attribute(name) && !(pk_attribute?(name) && id.nil?)
R
Robin Roestenburg 已提交
349
      end
350 351 352 353 354 355
    end

    def readonly_attribute?(name)
      self.class.readonly_attributes.include?(name)
    end

R
Robin Roestenburg 已提交
356 357
    def pk_attribute?(name)
      column_for_attribute(name).primary
358 359 360 361 362 363 364 365 366 367 368
    end

    def typecasted_attribute_value(name)
      if self.class.serialized_attributes.include?(name)
        @attributes[name].serialized_value
      else
        # FIXME: we need @attributes to be used consistently.
        # If the values stored in @attributes were already typecasted, this code
        # could be simplified
        read_attribute(name)
      end
369
    end
370 371
  end
end