attribute_methods.rb 16.6 KB
Newer Older
1 2
# frozen_string_literal: true

3
require "mutex_m"
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
    RESTRICTED_CLASS_METHODS = %w(private public protected allocate new name parent superclass)
26

27
    class GeneratedAttributeMethods < Module #:nodoc:
28 29
      include Mutex_m
    end
30

31
    module ClassMethods
32 33 34 35 36 37
      def inherited(child_class) #:nodoc:
        child_class.initialize_generated_modules
        super
      end

      def initialize_generated_modules # :nodoc:
38
        @generated_attribute_methods = const_set(:GeneratedAttributeMethods, GeneratedAttributeMethods.new)
39
        private_constant :GeneratedAttributeMethods
40
        @attribute_methods_generated = false
41
        include @generated_attribute_methods
42 43

        super
44 45
      end

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

60
      def undefine_attribute_methods # :nodoc:
61
        generated_attribute_methods.synchronize do
62
          super if defined?(@attribute_methods_generated) && @attribute_methods_generated
63 64
          @attribute_methods_generated = false
        end
65 66
      end

67
      # Raises an ActiveRecord::DangerousAttributeError exception when an
68 69 70 71 72 73 74 75 76
      # \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)
77
      #   # => 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.
78 79 80
      #
      #   Person.instance_method_already_implemented?(:name)
      #   # => false
81
      def instance_method_already_implemented?(method_name)
82
        if dangerous_attribute_method?(method_name)
83
          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."
84 85
        end

J
Jon Leighton 已提交
86
        if superclass == Base
J
Jon Leighton 已提交
87 88
          super
        else
89 90 91
          # 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) &&
92
            ! superclass.instance_method(method_name).owner.is_a?(GeneratedAttributeMethods)
93
          defined || super
J
Jon Leighton 已提交
94
        end
95
      end
96

97
      # A method name is 'dangerous' if it is already (re)defined by Active Record, but
98
      # not by any ancestors. (So 'puts' is not dangerous but 'save' is.)
99
      def dangerous_attribute_method?(name) # :nodoc:
J
Jon Leighton 已提交
100 101
        method_defined_within?(name, Base)
      end
102

103
      def method_defined_within?(name, klass, superklass = klass.superclass) # :nodoc:
J
Jon Leighton 已提交
104
        if klass.method_defined?(name) || klass.private_method_defined?(name)
105 106 107 108 109 110 111 112 113 114 115 116 117
          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)
118
        RESTRICTED_CLASS_METHODS.include?(method_name.to_s) || class_method_defined_within?(method_name, Base)
119 120
      end

121
      def class_method_defined_within?(name, klass, superklass = klass.superclass) # :nodoc:
122 123 124
        if klass.respond_to?(name, true)
          if superklass.respond_to?(name, true)
            klass.method(name).owner != superklass.method(name).owner
J
Jon Leighton 已提交
125 126 127 128 129 130
          else
            true
          end
        else
          false
        end
131
      end
132

133 134 135 136 137 138 139 140 141
      # 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
142
      def attribute_method?(attribute)
143
        super || (table_exists? && column_names.include?(attribute.to_s.sub(/=$/, "")))
144 145
      end

146 147 148 149 150 151 152 153
      # 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"]
154 155
      def attribute_names
        @attribute_names ||= if !abstract_class? && table_exists?
156
          attribute_types.keys
157 158 159
        else
          []
        end
160
      end
161

162 163
      # Regexp for column names (with or without a table name prefix). Matches
      # the following:
B
Ben Toews 已提交
164 165
      #   "#{table_name}.#{column_name}"
      #   "#{column_name}"
166
      COLUMN_NAME = /\A(?:\w+\.)?\w+\z/i
B
Ben Toews 已提交
167

168 169
      # Regexp for column names with order (with or without a table name
      # prefix, with or without various order modifiers). Matches the following:
B
Ben Toews 已提交
170 171
      #   "#{table_name}.#{column_name}"
      #   "#{table_name}.#{column_name} #{direction}"
172 173
      #   "#{table_name}.#{column_name} #{direction} NULLS FIRST"
      #   "#{table_name}.#{column_name} NULLS LAST"
B
Ben Toews 已提交
174 175
      #   "#{column_name}"
      #   "#{column_name} #{direction}"
176 177
      #   "#{column_name} #{direction} NULLS FIRST"
      #   "#{column_name} NULLS LAST"
178
      COLUMN_NAME_WITH_ORDER = /
179 180 181 182 183 184 185
        \A
        (?:\w+\.)?
        \w+
        (?:\s+asc|\s+desc)?
        (?:\s+nulls\s+(?:first|last))?
        \z
      /ix
B
Ben Toews 已提交
186

187
      def disallow_raw_sql!(args, permit: COLUMN_NAME) # :nodoc:
B
Ben Toews 已提交
188
        unexpected = args.reject do |arg|
189
          Arel.arel_node?(arg) ||
190
            arg.to_s.split(/\s*,\s*/).all? { |part| permit.match?(part) }
B
Ben Toews 已提交
191 192 193 194 195 196
        end

        return if unexpected.none?

        if allow_unsafe_raw_sql == :deprecated
          ActiveSupport::Deprecation.warn(
B
Ben Toews 已提交
197 198 199
            "Dangerous query method (method whose arguments are used as raw " \
            "SQL) called with non-attribute argument(s): " \
            "#{unexpected.map(&:inspect).join(", ")}. Non-attribute " \
200
            "arguments will be disallowed in Rails 6.1. This method should " \
B
Ben Toews 已提交
201 202 203
            "not be called with user-provided values, such as request " \
            "parameters or model attributes. Known-safe values can be passed " \
            "by wrapping them in Arel.sql()."
B
Ben Toews 已提交
204 205 206 207 208 209 210 211 212
          )
        else
          raise(ActiveRecord::UnknownAttributeReference,
            "Query method called with non-attribute argument(s): " +
            unexpected.map(&:inspect).join(", ")
          )
        end
      end

213 214 215 216 217 218 219 220 221 222 223 224
      # 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

225
      # Returns the column object for the named attribute.
226 227
      # Returns a +ActiveRecord::ConnectionAdapters::NullColumn+ if the
      # named attribute does not exist.
228 229 230 231 232 233
      #
      #   class Person < ActiveRecord::Base
      #   end
      #
      #   person = Person.new
      #   person.column_for_attribute(:name) # the result depends on the ConnectionAdapter
234
      #   # => #<ActiveRecord::ConnectionAdapters::Column:0x007ff4ab083980 @name="name", @sql_type="varchar(255)", @null=true, ...>
235 236
      #
      #   person.column_for_attribute(:nothing)
237
      #   # => #<ActiveRecord::ConnectionAdapters::NullColumn:0xXXX @name=nil, @sql_type=nil, @cast_type=#<Type::Value>, ...>
238
      def column_for_attribute(name)
239 240 241
        name = name.to_s
        columns_hash.fetch(name) do
          ConnectionAdapters::NullColumn.new(name)
242 243
        end
      end
244
    end
J
Joshua Peek 已提交
245

246 247
    # 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>
248
    # which will all return +true+. It also defines the attribute methods if they have
249 250 251 252 253 254
    # not been generated.
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.new
K
Kivanio Barbosa 已提交
255 256 257 258 259 260 261
    #   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
262
    def respond_to?(name, include_private = false)
263
      return false unless super
264

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

S
Shuhei Kitagawa 已提交
275
      true
276
    end
277

278 279 280 281 282 283 284 285 286
    # 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
287
    def has_attribute?(attr_name)
288
      @attributes.key?(attr_name.to_s)
289 290 291
    end

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

    # Returns a hash of all the attributes with their names as keys and the values of the attributes as values.
304 305 306 307 308 309 310
    #
    #   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}
311
    def attributes
312
      @attributes.to_hash
313 314 315
    end

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

336 337 338
    # 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+.
339
    # Note that it always returns +true+ with boolean attributes.
340
    #
341
    #   class Task < ActiveRecord::Base
342 343
    #   end
    #
344 345 346
    #   task = Task.new(title: '', is_done: false)
    #   task.attribute_present?(:title)   # => false
    #   task.attribute_present?(:is_done) # => true
G
Guillermo Iguaran 已提交
347
    #   task.title = 'Buy milk'
348 349 350
    #   task.is_done = true
    #   task.attribute_present?(:title)   # => true
    #   task.attribute_present?(:is_done) # => true
351
    def attribute_present?(attribute)
352
      value = _read_attribute(attribute)
353
      !value.nil? && !(value.respond_to?(:empty?) && value.empty?)
354 355
    end

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

    # Updates the attribute identified by <tt>attr_name</tt> with the specified +value+.
378
    # (Alias for the protected #write_attribute method).
379 380 381 382 383 384 385
    #
    #   class Person < ActiveRecord::Base
    #   end
    #
    #   person = Person.new
    #   person[:age] = '22'
    #   person[:age] # => 22
386
    #   person[:age].class # => Integer
387 388 389 390
    def []=(attr_name, value)
      write_attribute(attr_name, value)
    end

391
    # Returns the name of all database fields which have been read from this
392
    # model. This can be useful in development mode to determine which fields
393 394 395 396 397 398
    # 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 已提交
399 400
    #   class PostsController < ActionController::Base
    #     after_action :print_accessed_fields, only: :index
401
    #
A
akihiro17 已提交
402 403 404
    #     def index
    #       @posts = Post.all
    #     end
405
    #
A
akihiro17 已提交
406
    #     private
407
    #
A
akihiro17 已提交
408 409 410
    #     def print_accessed_fields
    #       p @posts.first.accessed_fields
    #     end
411 412 413 414
    #   end
    #
    # Which allows you to quickly change your code to:
    #
A
akihiro17 已提交
415 416 417 418
    #   class PostsController < ActionController::Base
    #     def index
    #       @posts = Post.select(:id, :title, :author_id, :updated_at)
    #     end
419 420 421 422 423
    #   end
    def accessed_fields
      @attributes.accessed
    end

R
Ryuta Kamizono 已提交
424 425
    private
      def attribute_method?(attr_name)
426 427 428
        # We check defined? because Syck calls respond_to? before actually calling initialize.
        defined?(@attributes) && @attributes.key?(attr_name)
      end
429

430 431 432
      def attributes_with_values(attribute_names)
        attribute_names.each_with_object({}) do |name, attrs|
          attrs[name] = _read_attribute(name)
433
        end
R
Robin Roestenburg 已提交
434
      end
435

436
      # Filters the primary keys and readonly attributes from the attribute names.
437
      def attributes_for_update(attribute_names)
438 439
        attribute_names &= self.class.column_names
        attribute_names.delete_if do |name|
440 441
          readonly_attribute?(name)
        end
R
Robin Roestenburg 已提交
442 443
      end

444 445
      # 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).
446
      def attributes_for_create(attribute_names)
447 448
        attribute_names &= self.class.column_names
        attribute_names.delete_if do |name|
449 450
          pk_attribute?(name) && id.nil?
        end
R
Robin Roestenburg 已提交
451
      end
452

453 454 455 456 457 458 459 460 461 462
      def format_for_inspect(value)
        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

463 464 465
      def readonly_attribute?(name)
        self.class.readonly_attributes.include?(name)
      end
466

467 468 469
      def pk_attribute?(name)
        name == self.class.primary_key
      end
470 471
  end
end