attribute_methods.rb 16.3 KB
Newer Older
J
Jeremy Kemper 已提交
1
require 'active_support/core_ext/enumerable'
2
require 'mutex_m'
3
require 'thread_safe'
J
Jeremy Kemper 已提交
4

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

11
    included do
12
      initialize_generated_modules
13 14 15 16 17 18 19 20
      include Read
      include Write
      include BeforeTypeCast
      include Query
      include PrimaryKey
      include TimeZoneConversion
      include Dirty
      include Serialization
21 22

      delegate :column_for_attribute, to: :class
23 24
    end

25 26 27 28
    AttrNames = Module.new {
      def self.set_name_cache(name, value)
        const_name = "ATTR_#{name}"
        unless const_defined? const_name
29
          const_set const_name, value.dup.freeze
30 31 32 33
        end
      end
    }

34 35
    BLACKLISTED_CLASS_METHODS = %w(private public protected)

36 37 38
    class AttributeMethodCache
      def initialize
        @module = Module.new
39
        @method_cache = ThreadSafe::Cache.new
40 41 42
      end

      def [](name)
43 44 45 46 47 48
        @method_cache.compute_if_absent(name) do
          safe_name = name.unpack('h*').first
          temp_method = "__temp__#{safe_name}"
          ActiveRecord::AttributeMethods::AttrNames.set_name_cache safe_name, name
          @module.module_eval method_body(temp_method, safe_name), __FILE__, __LINE__
          @module.instance_method temp_method
49 50 51 52
        end
      end

      private
B
Bin Huang 已提交
53 54 55 56 57

      # Override this method in the subclasses for method body.
      def method_body(method_name, const_name)
        raise NotImplementedError, "Subclasses must implement a method_body(method_name, const_name) method."
      end
58 59
    end

60
    module ClassMethods
61 62 63 64 65 66
      def inherited(child_class) #:nodoc:
        child_class.initialize_generated_modules
        super
      end

      def initialize_generated_modules # :nodoc:
67
        @generated_attribute_methods = Module.new { extend Mutex_m }
68
        @attribute_methods_generated = false
69 70 71
        include @generated_attribute_methods
      end

72 73
      # Generates all the attribute related methods for columns in the database
      # accessors, mutators and query methods.
74
      def define_attribute_methods # :nodoc:
75
        return false if @attribute_methods_generated
V
Vipul A M 已提交
76
        # Use a mutex; we don't want two thread simultaneously trying to define
J
Jon Leighton 已提交
77
        # attribute methods.
78
        generated_attribute_methods.synchronize do
79
          return false if @attribute_methods_generated
J
Jon Leighton 已提交
80 81 82 83
          superclass.define_attribute_methods unless self == base_class
          super(column_names)
          @attribute_methods_generated = true
        end
84
        true
85 86
      end

87
      def undefine_attribute_methods # :nodoc:
88 89 90 91
        generated_attribute_methods.synchronize do
          super if @attribute_methods_generated
          @attribute_methods_generated = false
        end
92 93
      end

94 95 96 97 98 99 100 101 102 103 104 105 106 107
      # 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
108
      def instance_method_already_implemented?(method_name)
109
        if dangerous_attribute_method?(method_name)
X
Xavier Noria 已提交
110
          raise DangerousAttributeError, "#{method_name} is defined by Active Record"
111 112
        end

J
Jon Leighton 已提交
113
        if superclass == Base
J
Jon Leighton 已提交
114 115
          super
        else
116
          # If B < A and A defines its own attribute method, then we don't want to overwrite that.
A
Arthur Neves 已提交
117 118 119
          defined = method_defined_within?(method_name, superclass, superclass.generated_attribute_methods)
          base_defined = Base.method_defined?(method_name) || Base.private_method_defined?(method_name)
          defined && !base_defined || super
J
Jon Leighton 已提交
120
        end
121
      end
122

123
      # A method name is 'dangerous' if it is already (re)defined by Active Record, but
124
      # not by any ancestors. (So 'puts' is not dangerous but 'save' is.)
125
      def dangerous_attribute_method?(name) # :nodoc:
J
Jon Leighton 已提交
126 127
        method_defined_within?(name, Base)
      end
128

129
      def method_defined_within?(name, klass, superklass = klass.superclass) # :nodoc:
J
Jon Leighton 已提交
130
        if klass.method_defined?(name) || klass.private_method_defined?(name)
131 132 133 134 135 136 137 138 139 140 141 142 143
          if superklass.method_defined?(name) || superklass.private_method_defined?(name)
            klass.instance_method(name).owner != superklass.instance_method(name).owner
          else
            true
          end
        else
          false
        end
      end

      # A class method is 'dangerous' if it is already (re)defined by Active Record, but
      # not by any ancestors. (So 'puts' is not dangerous but 'new' is.)
      def dangerous_class_method?(method_name)
144
        BLACKLISTED_CLASS_METHODS.include?(method_name.to_s) || class_method_defined_within?(method_name, Base)
145 146 147 148 149 150
      end

      def class_method_defined_within?(name, klass, superklass = klass.superclass) # :nodoc
        if klass.respond_to?(name, true)
          if superklass.respond_to?(name, true)
            klass.method(name).owner != superklass.method(name).owner
J
Jon Leighton 已提交
151 152 153 154 155 156
          else
            true
          end
        else
          false
        end
157
      end
158

159 160 161 162 163 164 165 166 167 168
      def find_generated_attribute_method(method_name) # :nodoc:
        klass = self
        until klass == Base
          gen_methods = klass.generated_attribute_methods
          return gen_methods.instance_method(method_name) if method_defined_within?(method_name, gen_methods, Object)
          klass = klass.superclass
        end
        nil
      end

169 170 171 172 173 174 175 176 177
      # 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
178 179 180 181
      def attribute_method?(attribute)
        super || (table_exists? && column_names.include?(attribute.to_s.sub(/=$/, '')))
      end

182 183 184 185 186 187 188 189
      # 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"]
190 191 192 193 194 195 196
      def attribute_names
        @attribute_names ||= if !abstract_class? && table_exists?
            column_names
          else
            []
          end
      end
197

198 199 200
      # Returns the column object for the named attribute.
      # Returns a +ActiveRecord::ConnectionAdapters::NullColumn+ if the
      # named attribute does not exist.
201 202 203 204 205 206 207 208 209
      #
      #   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)
210
      #   # => #<ActiveRecord::ConnectionAdapters::NullColumn:0xXXX @name=nil, @sql_type=nil, @cast_type=#<Type::Value>, ...>
211 212 213
      def column_for_attribute(name)
        name = name.to_s
        columns_hash.fetch(name) do
214
          ConnectionAdapters::NullColumn.new(name)
215 216
        end
      end
217
    end
J
Joshua Peek 已提交
218

219 220
    # If we haven't generated any methods yet, generate them, then
    # see if we've created the method we're looking for.
221
    def method_missing(method, *args, &block) # :nodoc:
222 223
      self.class.define_attribute_methods
      if respond_to_without_attributes?(method)
224 225 226 227 228
        # make sure to invoke the correct attribute method, as we might have gotten here via a `super`
        # call in a overwritten attribute method
        if attribute_method = self.class.find_generated_attribute_method(method)
          # this is probably horribly slow, but should only happen at most once for a given AR class
          attribute_method.bind(self).call(*args, &block)
229
        else
230
          return super unless respond_to_missing?(method, true)
231
          send(method, *args, &block)
232
        end
233 234
      else
        super
235 236 237
      end
    end

238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253
    # 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
254
    def respond_to?(name, include_private = false)
255
      name = name.to_s
256
      self.class.define_attribute_methods
257 258
      result = super

259
      # If the result is false the answer is false.
260 261 262 263
      return false unless result

      # If the result is true then check for the select case.
      # For queries selecting a subset of columns, return false for unselected columns.
264
      # We check defined?(@attributes) not to issue warnings if called on objects that
265
      # have been allocated but not yet initialized.
266
      if defined?(@attributes) && @attributes.any? && self.class.column_names.include?(name)
267 268 269 270
        return has_attribute?(name)
      end

      return true
271
    end
272

273 274 275 276 277 278 279 280 281
    # 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
282
    def has_attribute?(attr_name)
283
      @attributes.has_key?(attr_name.to_s)
284 285 286
    end

    # Returns an array of names for the attributes available on this object.
287 288 289 290 291 292 293
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.new
    #   person.attribute_names
    #   # => ["id", "created_at", "updated_at", "name", "age"]
294
    def attribute_names
295
      @attributes.keys
296 297 298
    end

    # Returns a hash of all the attributes with their names as keys and the values of the attributes as values.
299 300 301 302 303 304 305
    #
    #   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}
306
    def attributes
307 308 309
      attribute_names.each_with_object({}) { |name, attrs|
        attrs[name] = read_attribute(name)
      }
310 311 312 313
    end

    # Returns an <tt>#inspect</tt>-like string for the value of the
    # attribute +attr_name+. String attributes are truncated upto 50
J
Jan Bernacki 已提交
314 315 316 317
    # characters, Date and Time attributes are returned in the
    # <tt>:db</tt> format, Array attributes are truncated upto 10 values.
    # Other attributes return the value of <tt>#inspect</tt> without
    # modification.
318
    #
319
    #   person = Person.create!(name: 'David Heinemeier Hansson ' * 3)
320 321
    #
    #   person.attribute_for_inspect(:name)
322
    #   # => "\"David Heinemeier Hansson David Heinemeier Hansson ...\""
323 324
    #
    #   person.attribute_for_inspect(:created_at)
325
    #   # => "\"2012-10-22 00:15:07\""
J
Jan Bernacki 已提交
326 327 328
    #
    #   person.attribute_for_inspect(:tag_ids)
    #   # => "[1, 2, 3, 4, 5, 6, 7, 8, 9, 10, ...]"
329 330 331 332
    def attribute_for_inspect(attr_name)
      value = read_attribute(attr_name)

      if value.is_a?(String) && value.length > 50
333
        "#{value[0, 50]}...".inspect
334 335
      elsif value.is_a?(Date) || value.is_a?(Time)
        %("#{value.to_s(:db)}")
J
Jan Bernacki 已提交
336 337 338
      elsif value.is_a?(Array) && value.size > 10
        inspected = value.first(10).inspect
        %(#{inspected[0...-1]}, ...])
339 340 341 342 343
      else
        value.inspect
      end
    end

344 345 346
    # 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+.
347
    # Note that it always returns +true+ with boolean attributes.
348
    #
349
    #   class Task < ActiveRecord::Base
350 351
    #   end
    #
352 353 354
    #   task = Task.new(title: '', is_done: false)
    #   task.attribute_present?(:title)   # => false
    #   task.attribute_present?(:is_done) # => true
G
Guillermo Iguaran 已提交
355
    #   task.title = 'Buy milk'
356 357 358
    #   task.is_done = true
    #   task.attribute_present?(:title)   # => true
    #   task.attribute_present?(:is_done) # => true
359 360
    def attribute_present?(attribute)
      value = read_attribute(attribute)
361
      !value.nil? && !(value.respond_to?(:empty?) && value.empty?)
362 363
    end

364
    # Returns the value of the attribute identified by <tt>attr_name</tt> after it has been typecast (for example,
365
    # "2004-12-12" in a date column is cast to a date object, like Date.new(2004, 12, 12)). It raises
366 367
    # <tt>ActiveModel::MissingAttributeError</tt> if the identified attribute is missing.
    #
368 369
    # Note: +:id+ is always present.
    #
370
    # Alias for the <tt>read_attribute</tt> method.
371 372
    #
    #   class Person < ActiveRecord::Base
373
    #     belongs_to :organization
374 375
    #   end
    #
376
    #   person = Person.new(name: 'Francesco', age: '22')
377 378
    #   person[:name] # => "Francesco"
    #   person[:age]  # => 22
379 380 381 382
    #
    #   person = Person.select('id').first
    #   person[:name]            # => ActiveModel::MissingAttributeError: missing attribute: name
    #   person[:organization_id] # => ActiveModel::MissingAttributeError: missing attribute: organization_id
383
    def [](attr_name)
384
      read_attribute(attr_name) { |n| missing_attribute(n, caller) }
385 386 387
    end

    # Updates the attribute identified by <tt>attr_name</tt> with the specified +value+.
388 389 390 391 392 393 394 395 396
    # (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
397 398 399 400
    def []=(attr_name, value)
      write_attribute(attr_name, value)
    end

401
    protected
402

403 404 405
    def clone_attributes # :nodoc:
      @attributes.each_with_object({}) do |(name, attr), h|
        h[name] = attr.dup
406 407 408
      end
    end

409
    def clone_attribute_value(reader_method, attribute_name) # :nodoc:
410 411 412 413 414 415
      value = send(reader_method, attribute_name)
      value.duplicable? ? value.clone : value
    rescue TypeError, NoMethodError
      value
    end

416
    def arel_attributes_with_values_for_create(attribute_names) # :nodoc:
J
Jon Leighton 已提交
417
      arel_attributes_with_values(attributes_for_create(attribute_names))
R
Robin Roestenburg 已提交
418
    end
419

420
    def arel_attributes_with_values_for_update(attribute_names) # :nodoc:
R
Robin Roestenburg 已提交
421
      arel_attributes_with_values(attributes_for_update(attribute_names))
422 423
    end

424
    def attribute_method?(attr_name) # :nodoc:
425
      # We check defined? because Syck calls respond_to? before actually calling initialize.
426
      defined?(@attributes) && @attributes.include?(attr_name)
427 428
    end

R
Robin Roestenburg 已提交
429
    private
430

R
Robin Roestenburg 已提交
431
    # Returns a Hash of the Arel::Attributes and attribute values that have been
432
    # typecasted for use in an Arel insert/update method.
R
Robin Roestenburg 已提交
433 434 435
    def arel_attributes_with_values(attribute_names)
      attrs = {}
      arel_table = self.class.arel_table
436

R
Robin Roestenburg 已提交
437 438 439
      attribute_names.each do |name|
        attrs[arel_table[name]] = typecasted_attribute_value(name)
      end
440 441 442
      attrs
    end

R
Robin Roestenburg 已提交
443 444
    # Filters the primary keys and readonly attributes from the attribute names.
    def attributes_for_update(attribute_names)
445 446
      attribute_names.reject do |name|
        readonly_attribute?(name)
R
Robin Roestenburg 已提交
447 448 449 450 451
      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 已提交
452
    def attributes_for_create(attribute_names)
453 454
      attribute_names.reject do |name|
        pk_attribute?(name) && id.nil?
R
Robin Roestenburg 已提交
455
      end
456 457 458 459 460 461
    end

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

R
Robin Roestenburg 已提交
462
    def pk_attribute?(name)
S
Sean Griffin 已提交
463
      name == self.class.primary_key
464 465 466
    end

    def typecasted_attribute_value(name)
467
      read_attribute(name)
468
    end
469 470
  end
end