attribute_methods.rb 15.7 KB
Newer Older
1 2 3 4
require "active_support/core_ext/enumerable"
require "active_support/core_ext/string/filters"
require "mutex_m"
require "concurrent/map"
J
Jeremy Kemper 已提交
5

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

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

      delegate :column_for_attribute, to: :class
24 25
    end

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

35
    BLACKLISTED_CLASS_METHODS = %w(private public protected allocate new name parent superclass)
36

37 38
    class GeneratedAttributeMethods < Module; end # :nodoc:

39
    module ClassMethods
40 41 42 43 44 45
      def inherited(child_class) #:nodoc:
        child_class.initialize_generated_modules
        super
      end

      def initialize_generated_modules # :nodoc:
46
        @generated_attribute_methods = GeneratedAttributeMethods.new { extend Mutex_m }
47
        @attribute_methods_generated = false
48
        include @generated_attribute_methods
49 50

        super
51 52
      end

53 54
      # Generates all the attribute related methods for columns in the database
      # accessors, mutators and query methods.
55
      def define_attribute_methods # :nodoc:
56
        return false if @attribute_methods_generated
G
Gregory Igelmund 已提交
57
        # Use a mutex; we don't want two threads simultaneously trying to define
J
Jon Leighton 已提交
58
        # attribute methods.
59
        generated_attribute_methods.synchronize do
60
          return false if @attribute_methods_generated
J
Jon Leighton 已提交
61
          superclass.define_attribute_methods unless self == base_class
62
          super(attribute_names)
J
Jon Leighton 已提交
63 64
          @attribute_methods_generated = true
        end
65
        true
66 67
      end

68
      def undefine_attribute_methods # :nodoc:
69
        generated_attribute_methods.synchronize do
70
          super if defined?(@attribute_methods_generated) && @attribute_methods_generated
71 72
          @attribute_methods_generated = false
        end
73 74
      end

75
      # Raises an ActiveRecord::DangerousAttributeError exception when an
76 77 78 79 80 81 82 83 84
      # \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)
85
      #   # => ActiveRecord::DangerousAttributeError: save is defined by Active Record. Check to make sure that you don't have an attribute or method with the same name.
86 87 88
      #
      #   Person.instance_method_already_implemented?(:name)
      #   # => false
89
      def instance_method_already_implemented?(method_name)
90
        if dangerous_attribute_method?(method_name)
91
          raise DangerousAttributeError, "#{method_name} is defined by Active Record. Check to make sure that you don't have an attribute or method with the same name."
92 93
        end

J
Jon Leighton 已提交
94
        if superclass == Base
J
Jon Leighton 已提交
95 96
          super
        else
97 98 99 100 101
          # If ThisClass < ... < SomeSuperClass < ... < Base and SomeSuperClass
          # defines its own attribute method, then we don't want to overwrite that.
          defined = method_defined_within?(method_name, superclass, Base) &&
            ! superclass.instance_method(method_name).owner.is_a?(GeneratedAttributeMethods)
          defined || super
J
Jon Leighton 已提交
102
        end
103
      end
104

105
      # A method name is 'dangerous' if it is already (re)defined by Active Record, but
106
      # not by any ancestors. (So 'puts' is not dangerous but 'save' is.)
107
      def dangerous_attribute_method?(name) # :nodoc:
J
Jon Leighton 已提交
108 109
        method_defined_within?(name, Base)
      end
110

111
      def method_defined_within?(name, klass, superklass = klass.superclass) # :nodoc:
J
Jon Leighton 已提交
112
        if klass.method_defined?(name) || klass.private_method_defined?(name)
113 114 115 116 117 118 119 120 121 122 123 124 125
          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)
126
        BLACKLISTED_CLASS_METHODS.include?(method_name.to_s) || class_method_defined_within?(method_name, Base)
127 128
      end

129
      def class_method_defined_within?(name, klass, superklass = klass.superclass) # :nodoc:
130 131 132
        if klass.respond_to?(name, true)
          if superklass.respond_to?(name, true)
            klass.method(name).owner != superklass.method(name).owner
J
Jon Leighton 已提交
133 134 135 136 137 138
          else
            true
          end
        else
          false
        end
139
      end
140

141 142 143 144 145 146 147 148 149
      # 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
150
      def attribute_method?(attribute)
151
        super || (table_exists? && column_names.include?(attribute.to_s.sub(/=$/, "")))
152 153
      end

154 155 156 157 158 159 160 161
      # 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"]
162 163
      def attribute_names
        @attribute_names ||= if !abstract_class? && table_exists?
164
            attribute_types.keys
165 166 167 168
          else
            []
          end
      end
169

170 171 172 173 174 175 176 177 178 179 180 181
      # Returns true if the given attribute exists, otherwise false.
      #
      #   class Person < ActiveRecord::Base
      #   end
      #
      #   Person.has_attribute?('name')   # => true
      #   Person.has_attribute?(:age)     # => true
      #   Person.has_attribute?(:nothing) # => false
      def has_attribute?(attr_name)
        attribute_types.key?(attr_name.to_s)
      end

182
      # Returns the column object for the named attribute.
183 184
      # Returns a +ActiveRecord::ConnectionAdapters::NullColumn+ if the
      # named attribute does not exist.
185 186 187 188 189 190
      #
      #   class Person < ActiveRecord::Base
      #   end
      #
      #   person = Person.new
      #   person.column_for_attribute(:name) # the result depends on the ConnectionAdapter
191
      #   # => #<ActiveRecord::ConnectionAdapters::Column:0x007ff4ab083980 @name="name", @sql_type="varchar(255)", @null=true, ...>
192 193
      #
      #   person.column_for_attribute(:nothing)
194
      #   # => #<ActiveRecord::ConnectionAdapters::NullColumn:0xXXX @name=nil, @sql_type=nil, @cast_type=#<Type::Value>, ...>
195
      def column_for_attribute(name)
196 197 198
        name = name.to_s
        columns_hash.fetch(name) do
          ConnectionAdapters::NullColumn.new(name)
199 200
        end
      end
201
    end
J
Joshua Peek 已提交
202

203 204
    # 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>
205
    # which will all return +true+. It also defines the attribute methods if they have
206 207 208 209 210 211
    # not been generated.
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.new
K
Kivanio Barbosa 已提交
212 213 214 215 216 217 218
    #   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
219
    def respond_to?(name, include_private = false)
220
      return false unless super
221 222 223 224 225 226 227 228 229

      case name
      when :to_partial_path
        name = "to_partial_path".freeze
      when :to_model
        name = "to_model".freeze
      else
        name = name.to_s
      end
230 231 232

      # If the result is true then check for the select case.
      # For queries selecting a subset of columns, return false for unselected columns.
233
      # We check defined?(@attributes) not to issue warnings if called on objects that
234
      # have been allocated but not yet initialized.
235
      if defined?(@attributes) && self.class.column_names.include?(name)
236 237 238 239
        return has_attribute?(name)
      end

      return true
240
    end
241

242 243 244 245 246 247 248 249 250
    # 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
251
    def has_attribute?(attr_name)
252
      @attributes.key?(attr_name.to_s)
253 254 255
    end

    # Returns an array of names for the attributes available on this object.
256 257 258 259 260 261 262
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.new
    #   person.attribute_names
    #   # => ["id", "created_at", "updated_at", "name", "age"]
263
    def attribute_names
264
      @attributes.keys
265 266 267
    end

    # Returns a hash of all the attributes with their names as keys and the values of the attributes as values.
268 269 270 271 272 273 274
    #
    #   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}
275
    def attributes
276
      @attributes.to_hash
277 278 279
    end

    # Returns an <tt>#inspect</tt>-like string for the value of the
J
jbsmith86 已提交
280
    # attribute +attr_name+. String attributes are truncated up to 50
J
Jan Bernacki 已提交
281
    # characters, Date and Time attributes are returned in the
282 283
    # <tt>:db</tt> format. Other attributes return the value of
    # <tt>#inspect</tt> without modification.
284
    #
285
    #   person = Person.create!(name: 'David Heinemeier Hansson ' * 3)
286 287
    #
    #   person.attribute_for_inspect(:name)
288
    #   # => "\"David Heinemeier Hansson David Heinemeier Hansson ...\""
289 290
    #
    #   person.attribute_for_inspect(:created_at)
291
    #   # => "\"2012-10-22 00:15:07\""
J
Jan Bernacki 已提交
292 293
    #
    #   person.attribute_for_inspect(:tag_ids)
294
    #   # => "[1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11]"
295 296 297 298
    def attribute_for_inspect(attr_name)
      value = read_attribute(attr_name)

      if value.is_a?(String) && value.length > 50
299
        "#{value[0, 50]}...".inspect
300 301 302 303 304 305 306
      elsif value.is_a?(Date) || value.is_a?(Time)
        %("#{value.to_s(:db)}")
      else
        value.inspect
      end
    end

307 308 309
    # 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+.
310
    # Note that it always returns +true+ with boolean attributes.
311
    #
312
    #   class Task < ActiveRecord::Base
313 314
    #   end
    #
315 316 317
    #   task = Task.new(title: '', is_done: false)
    #   task.attribute_present?(:title)   # => false
    #   task.attribute_present?(:is_done) # => true
G
Guillermo Iguaran 已提交
318
    #   task.title = 'Buy milk'
319 320 321
    #   task.is_done = true
    #   task.attribute_present?(:title)   # => true
    #   task.attribute_present?(:is_done) # => true
322
    def attribute_present?(attribute)
323
      value = _read_attribute(attribute)
324
      !value.nil? && !(value.respond_to?(:empty?) && value.empty?)
325 326
    end

327
    # Returns the value of the attribute identified by <tt>attr_name</tt> after it has been typecast (for example,
328
    # "2004-12-12" in a date column is cast to a date object, like Date.new(2004, 12, 12)). It raises
329 330
    # <tt>ActiveModel::MissingAttributeError</tt> if the identified attribute is missing.
    #
331 332
    # Note: +:id+ is always present.
    #
333
    #   class Person < ActiveRecord::Base
334
    #     belongs_to :organization
335 336
    #   end
    #
337
    #   person = Person.new(name: 'Francesco', age: '22')
338 339
    #   person[:name] # => "Francesco"
    #   person[:age]  # => 22
340 341 342 343
    #
    #   person = Person.select('id').first
    #   person[:name]            # => ActiveModel::MissingAttributeError: missing attribute: name
    #   person[:organization_id] # => ActiveModel::MissingAttributeError: missing attribute: organization_id
344
    def [](attr_name)
345
      read_attribute(attr_name) { |n| missing_attribute(n, caller) }
346 347 348
    end

    # Updates the attribute identified by <tt>attr_name</tt> with the specified +value+.
349
    # (Alias for the protected #write_attribute method).
350 351 352 353 354 355 356
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.new
    #   person[:age] = '22'
    #   person[:age] # => 22
357
    #   person[:age].class # => Integer
358 359 360 361
    def []=(attr_name, value)
      write_attribute(attr_name, value)
    end

362
    # Returns the name of all database fields which have been read from this
363
    # model. This can be useful in development mode to determine which fields
364 365 366 367 368 369
    # need to be selected. For performance critical pages, selecting only the
    # required fields can be an easy performance win (assuming you aren't using
    # all of the fields on the model).
    #
    # For example:
    #
A
akihiro17 已提交
370 371
    #   class PostsController < ActionController::Base
    #     after_action :print_accessed_fields, only: :index
372
    #
A
akihiro17 已提交
373 374 375
    #     def index
    #       @posts = Post.all
    #     end
376
    #
A
akihiro17 已提交
377
    #     private
378
    #
A
akihiro17 已提交
379 380 381
    #     def print_accessed_fields
    #       p @posts.first.accessed_fields
    #     end
382 383 384 385
    #   end
    #
    # Which allows you to quickly change your code to:
    #
A
akihiro17 已提交
386 387 388 389
    #   class PostsController < ActionController::Base
    #     def index
    #       @posts = Post.select(:id, :title, :author_id, :updated_at)
    #     end
390 391 392 393 394
    #   end
    def accessed_fields
      @attributes.accessed
    end

395
    protected
396

397
    def clone_attribute_value(reader_method, attribute_name) # :nodoc:
398 399 400 401 402 403
      value = send(reader_method, attribute_name)
      value.duplicable? ? value.clone : value
    rescue TypeError, NoMethodError
      value
    end

404
    def arel_attributes_with_values_for_create(attribute_names) # :nodoc:
J
Jon Leighton 已提交
405
      arel_attributes_with_values(attributes_for_create(attribute_names))
R
Robin Roestenburg 已提交
406
    end
407

408
    def arel_attributes_with_values_for_update(attribute_names) # :nodoc:
R
Robin Roestenburg 已提交
409
      arel_attributes_with_values(attributes_for_update(attribute_names))
410 411
    end

412
    def attribute_method?(attr_name) # :nodoc:
413
      # We check defined? because Syck calls respond_to? before actually calling initialize.
414
      defined?(@attributes) && @attributes.key?(attr_name)
415 416
    end

R
Robin Roestenburg 已提交
417
    private
418

R
Robin Roestenburg 已提交
419
    # Returns a Hash of the Arel::Attributes and attribute values that have been
420
    # typecasted for use in an Arel insert/update method.
R
Robin Roestenburg 已提交
421 422 423
    def arel_attributes_with_values(attribute_names)
      attrs = {}
      arel_table = self.class.arel_table
424

R
Robin Roestenburg 已提交
425 426 427
      attribute_names.each do |name|
        attrs[arel_table[name]] = typecasted_attribute_value(name)
      end
428 429 430
      attrs
    end

R
Robin Roestenburg 已提交
431 432
    # Filters the primary keys and readonly attributes from the attribute names.
    def attributes_for_update(attribute_names)
433 434
      attribute_names.reject do |name|
        readonly_attribute?(name)
R
Robin Roestenburg 已提交
435 436 437 438 439
      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 已提交
440
    def attributes_for_create(attribute_names)
441 442
      attribute_names.reject do |name|
        pk_attribute?(name) && id.nil?
R
Robin Roestenburg 已提交
443
      end
444 445 446 447 448 449
    end

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

R
Robin Roestenburg 已提交
450
    def pk_attribute?(name)
S
Sean Griffin 已提交
451
      name == self.class.primary_key
452 453 454
    end

    def typecasted_attribute_value(name)
455
      _read_attribute(name)
456
    end
457 458
  end
end