core.rb 18.6 KB
Newer Older
X
Xavier Noria 已提交
1
require 'thread'
2
require 'active_support/core_ext/hash/indifferent_access'
J
Jon Leighton 已提交
3
require 'active_support/core_ext/object/duplicable'
X
Xavier Noria 已提交
4
require 'active_support/core_ext/string/filters'
5 6

module ActiveRecord
J
Jon Leighton 已提交
7 8
  module Core
    extend ActiveSupport::Concern
9

J
Jon Leighton 已提交
10
    included do
J
Jon Leighton 已提交
11 12
      ##
      # :singleton-method:
J
Jon Leighton 已提交
13 14 15 16 17
      #
      # Accepts a logger conforming to the interface of Log4r which is then
      # passed on to any new database connections made and which can be
      # retrieved on both a class and instance level by calling +logger+.
      mattr_accessor :logger, instance_writer: false
18

J
Jon Leighton 已提交
19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44
      ##
      # Contains the database configuration - as is typically stored in config/database.yml -
      # as a Hash.
      #
      # For example, the following database.yml...
      #
      #   development:
      #     adapter: sqlite3
      #     database: db/development.sqlite3
      #
      #   production:
      #     adapter: sqlite3
      #     database: db/production.sqlite3
      #
      # ...would result in ActiveRecord::Base.configurations to look like this:
      #
      #   {
      #      'development' => {
      #         'adapter'  => 'sqlite3',
      #         'database' => 'db/development.sqlite3'
      #      },
      #      'production' => {
      #         'adapter'  => 'sqlite3',
      #         'database' => 'db/production.sqlite3'
      #      }
      #   }
45 46 47
      def self.configurations=(config)
        @@configurations = ActiveRecord::ConnectionHandling::MergeAndResolveDefaultUrlConfig.new(config).resolve
      end
J
Jon Leighton 已提交
48 49
      self.configurations = {}

50 51 52 53 54
      # Returns fully resolved configurations hash
      def self.configurations
        @@configurations
      end

J
Jon Leighton 已提交
55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
      ##
      # :singleton-method:
      # Determines whether to use Time.utc (using :utc) or Time.local (using :local) when pulling
      # dates and times from the database. This is set to :utc by default.
      mattr_accessor :default_timezone, instance_writer: false
      self.default_timezone = :utc

      ##
      # :singleton-method:
      # Specifies the format to use when dumping the database schema with Rails'
      # Rakefile. If :sql, the schema is dumped as (potentially database-
      # specific) SQL statements. If :ruby, the schema is dumped as an
      # ActiveRecord::Schema file which can be loaded into any database that
      # supports migrations. Use :ruby if you want to have different database
      # adapters for, e.g., your development and test environments.
      mattr_accessor :schema_format, instance_writer: false
      self.schema_format = :ruby

73 74 75 76 77 78 79 80
      ##
      # :singleton-method:
      # Specifies if an error should be raised on query limit or order being
      # ignored when doing batch queries. Useful in applications where the
      # limit or scope being ignored is error-worthy, rather than a warning.
      mattr_accessor :error_on_ignored_order_or_limit, instance_writer: false
      self.error_on_ignored_order_or_limit = false

J
Jon Leighton 已提交
81 82 83 84 85 86
      ##
      # :singleton-method:
      # Specify whether or not to use timestamps for migration versions
      mattr_accessor :timestamped_migrations, instance_writer: false
      self.timestamped_migrations = true

87 88 89 90 91 92 93 94 95
      ##
      # :singleton-method:
      # Specify whether schema dump should happen at the end of the
      # db:migrate rake task. This is true by default, which is useful for the
      # development environment. This should ideally be false in the production
      # environment where dumping schema is rarely needed.
      mattr_accessor :dump_schema_after_migration, instance_writer: false
      self.dump_schema_after_migration = true

96 97 98
      ##
      # :singleton-method:
      # Specifies which database schemas to dump when calling db:structure:dump.
99 100 101 102
      # If the value is :schema_search_path (the default), any schemas listed in
      # schema_search_path are dumped. Use :all to dump all schemas regardless
      # of schema_search_path, or a string of comma separated schemas for a
      # custom list.
103 104 105
      mattr_accessor :dump_schemas, instance_writer: false
      self.dump_schemas = :schema_search_path

106 107
      ##
      # :singleton-method:
108 109 110 111
      # Specify a threshold for the size of query result sets. If the number of
      # records in the set exceeds the threshold, a warning is logged. This can
      # be used to identify queries which load thousands of records and
      # potentially cause memory bloat.
112 113 114
      mattr_accessor :warn_on_records_fetched_greater_than, instance_writer: false
      self.warn_on_records_fetched_greater_than = nil

115 116
      mattr_accessor :maintain_test_schema, instance_accessor: false

117 118
      mattr_accessor :belongs_to_required_by_default, instance_accessor: false

119
      class_attribute :default_connection_handler, instance_writer: false
120

121
      def self.connection_handler
122
        ActiveRecord::RuntimeRegistry.connection_handler || default_connection_handler
123 124 125
      end

      def self.connection_handler=(handler)
126
        ActiveRecord::RuntimeRegistry.connection_handler = handler
127 128 129
      end

      self.default_connection_handler = ConnectionAdapters::ConnectionHandler.new
130 131 132
    end

    module ClassMethods
133 134 135 136 137
      def allocate
        define_attribute_methods
        super
      end

138
      def initialize_find_by_cache # :nodoc:
139
        @find_by_statement_cache = { true => {}.extend(Mutex_m), false => {}.extend(Mutex_m) }
140 141
      end

142
      def inherited(child_class) # :nodoc:
143
        # initialize cache at class definition for thread safety
144 145 146 147
        child_class.initialize_find_by_cache
        super
      end

148
      def find(*ids) # :nodoc:
149 150 151 152
        # We don't have cache keys for this stuff yet
        return super unless ids.length == 1
        return super if block_given? ||
                        primary_key.nil? ||
153
                        scope_attributes? ||
154 155 156 157
                        columns_hash.include?(inheritance_column) ||
                        ids.first.kind_of?(Array)

        id  = ids.first
158 159
        if ActiveRecord::Base === id
          id = id.id
X
Xavier Noria 已提交
160 161
          ActiveSupport::Deprecation.warn(<<-MSG.squish)
            You are passing an instance of ActiveRecord::Base to `find`.
162
            Please pass the id of the object by calling `.id`.
X
Xavier Noria 已提交
163
          MSG
164
        end
165 166 167

        key = primary_key

168 169
        statement = cached_find_by_statement(key) { |params|
          where(key => params.bind).limit(1)
170
        }
171
        record = statement.execute([id], self, connection).first
172
        unless record
173 174
          raise RecordNotFound.new("Couldn't find #{name} with '#{primary_key}'=#{id}",
                                   name, primary_key, id)
175 176
        end
        record
177
      rescue RangeError
178 179
        raise RecordNotFound.new("Couldn't find #{name} with an out of range value for '#{primary_key}'",
                                 name, primary_key)
180 181
      end

182
      def find_by(*args) # :nodoc:
183
        return super if scope_attributes? || !(Hash === args.first) || reflect_on_all_aggregations.any?
184 185 186

        hash = args.first

187
        return super if hash.values.any? { |v|
188
          v.nil? || Array === v || Hash === v || Relation === v
189
        }
190

191 192 193
        # We can't cache Post.find_by(author: david) ...yet
        return super unless hash.keys.all? { |k| columns_hash.has_key?(k.to_s) }

194
        keys = hash.keys
195

196 197
        statement = cached_find_by_statement(keys) { |params|
          wheres = keys.each_with_object({}) { |param, o|
198
            o[param] = params.bind
199
          }
200
          where(wheres).limit(1)
201 202
        }
        begin
203
          statement.execute(hash.values, self, connection).first
204 205
        rescue TypeError
          raise ActiveRecord::StatementInvalid
206 207
        rescue RangeError
          nil
208 209 210
        end
      end

211
      def find_by!(*args) # :nodoc:
212
        find_by(*args) or raise RecordNotFound.new("Couldn't find #{name}", name)
213 214
      end

215
      def initialize_generated_modules # :nodoc:
216
        generated_association_methods
217 218
      end

219 220 221
      def generated_association_methods
        @generated_association_methods ||= begin
          mod = const_set(:GeneratedAssociationMethods, Module.new)
222 223 224 225 226 227 228 229 230 231 232
          include mod
          mod
        end
      end

      # Returns a string like 'Post(id:integer, title:string, body:text)'
      def inspect
        if self == Base
          super
        elsif abstract_class?
          "#{super}(abstract)"
233
        elsif !connected?
A
Arun Agrawal 已提交
234
          "#{super} (call '#{super}.connection' to establish a connection)"
235
        elsif table_exists?
236
          attr_list = attribute_types.map { |name, type| "#{name}: #{type.type}" } * ', '
237 238 239 240 241 242 243 244 245 246 247
          "#{super}(#{attr_list})"
        else
          "#{super}(Table doesn't exist)"
        end
      end

      # Overwrite the default class equality method to provide support for association proxies.
      def ===(object)
        object.is_a?(self)
      end

P
Peter Suschlik 已提交
248
      # Returns an instance of <tt>Arel::Table</tt> loaded with the current table name.
O
Oscar Del Ben 已提交
249 250
      #
      #   class Post < ActiveRecord::Base
251
      #     scope :published_and_commented, -> { published.and(self.arel_table[:comments_count].gt(0)) }
O
Oscar Del Ben 已提交
252
      #   end
253
      def arel_table # :nodoc:
254
        @arel_table ||= Arel::Table.new(table_name, type_caster: type_caster)
255 256
      end

257
      # Returns the Arel engine.
258
      def arel_engine # :nodoc:
259
        @arel_engine ||=
260
          if Base == self || connection_handler.retrieve_connection_pool(connection_specification_name)
J
Jon Leighton 已提交
261 262 263 264
            self
          else
            superclass.arel_engine
          end
265 266
      end

267
      def arel_attribute(name, table = arel_table) # :nodoc:
268 269 270 271
        name = attribute_alias(name) if attribute_alias?(name)
        table[name]
      end

272
      def predicate_builder # :nodoc:
273
        @predicate_builder ||= PredicateBuilder.new(table_metadata)
274 275
      end

276 277 278 279
      def type_caster # :nodoc:
        TypeCaster::Map.new(self)
      end

280 281
      private

282
      def cached_find_by_statement(key, &block) # :nodoc:
283 284 285
        cache = @find_by_statement_cache[connection.prepared_statements]
        cache[key] || cache.synchronize {
          cache[key] ||= StatementCache.create(connection, &block)
286 287 288
        }
      end

289
      def relation # :nodoc:
290
        relation = Relation.create(self, arel_table, predicate_builder)
291

292
        if finder_needs_type_condition? && !ignore_default_scope?
293
          relation.where(type_condition).create_with(inheritance_column.to_sym => sti_name)
294
        else
295
          relation
296 297
        end
      end
298 299 300 301

      def table_metadata # :nodoc:
        TableMetadata.new(self, arel_table)
      end
302 303 304 305 306 307 308
    end

    # New objects can be instantiated as either empty (pass no construction parameter) or pre-set with
    # attributes but not yet saved (pass a hash with key names matching the associated table column names).
    # In both instances, valid attribute keys are determined by the column names of the associated table --
    # hence you can't have attributes that aren't part of the table columns.
    #
309
    # ==== Example:
310
    #   # Instantiates a single new object
A
AvnerCohen 已提交
311
    #   User.new(first_name: 'Jamie')
312
    def initialize(attributes = nil)
313
      @attributes = self.class._default_attributes.deep_dup
314
      self.class.define_attribute_methods
315 316

      init_internals
317
      initialize_internals_callback
318

319
      assign_attributes(attributes) if attributes
320 321

      yield self if block_given?
322
      _run_initialize_callbacks
323 324
    end

325 326
    # Initialize an empty model object from +coder+. +coder+ should be
    # the result of previously encoding an Active Record model, using
327
    # #encode_with.
328 329 330 331
    #
    #   class Post < ActiveRecord::Base
    #   end
    #
332 333 334 335
    #   old_post = Post.new(title: "hello world")
    #   coder = {}
    #   old_post.encode_with(coder)
    #
336
    #   post = Post.allocate
337
    #   post.init_with(coder)
338 339
    #   post.title # => 'hello world'
    def init_with(coder)
340
      coder = LegacyYamlAdapter.convert(self.class, coder)
341
      @attributes = self.class.yaml_encoder.decode(coder)
342

343 344
      init_internals

345
      @new_record = coder['new_record']
346

347 348
      self.class.define_attribute_methods

349 350
      _run_find_callbacks
      _run_initialize_callbacks
351 352 353

      self
    end
354

355 356 357 358
    ##
    # :method: clone
    # Identical to Ruby's clone method.  This is a "shallow" copy.  Be warned that your attributes are not copied.
    # That means that modifying attributes of the clone will modify the original, since they will both point to the
V
Vijay Dev 已提交
359
    # same attributes hash. If you need a copy of your attributes hash, please use the #dup method.
360 361 362 363 364 365 366 367 368
    #
    #   user = User.first
    #   new_user = user.clone
    #   user.name               # => "Bob"
    #   new_user.name = "Joe"
    #   user.name               # => "Joe"
    #
    #   user.object_id == new_user.object_id            # => false
    #   user.name.object_id == new_user.name.object_id  # => true
369
    #
370
    #   user.name.object_id == user.dup.name.object_id  # => false
371

372 373
    ##
    # :method: dup
374 375 376 377 378 379
    # Duped objects have no id assigned and are treated as new records. Note
    # that this is a "shallow" copy as it copies the object's attributes
    # only, not its associations. The extent of a "deep" copy is application
    # specific and is therefore left to the application to implement according
    # to its need.
    # The dup method does not preserve the timestamps (created|updated)_(at|on).
380

381
    ##
382
    def initialize_dup(other) # :nodoc:
383
      @attributes = @attributes.deep_dup
384
      @attributes.reset(self.class.primary_key)
385

386
      _run_initialize_callbacks
387 388

      @new_record  = true
389
      @destroyed   = false
390 391 392 393 394 395

      super
    end

    # Populate +coder+ with attributes about this record that should be
    # serialized. The structure of +coder+ defined in this method is
396
    # guaranteed to match the structure of +coder+ passed to the #init_with
397 398 399 400 401 402 403 404
    # method.
    #
    # Example:
    #
    #   class Post < ActiveRecord::Base
    #   end
    #   coder = {}
    #   Post.new.encode_with(coder)
405
    #   coder # => {"attributes" => {"id" => nil, ... }}
406
    def encode_with(coder)
407
      self.class.yaml_encoder.encode(@attributes, coder)
408
      coder['new_record'] = new_record?
409
      coder['active_record_yaml_version'] = 2
410 411 412 413 414 415 416 417 418 419 420 421 422 423
    end

    # Returns true if +comparison_object+ is the same exact object, or +comparison_object+
    # is of the same type and +self+ has an ID and it is equal to +comparison_object.id+.
    #
    # Note that new records are different from any other record by definition, unless the
    # other record is the receiver itself. Besides, if you fetch existing records with
    # +select+ and leave the ID out, you're on your own, this predicate will return false.
    #
    # Note also that destroying a record preserves its ID in the model instance, so deleted
    # models are still comparable.
    def ==(comparison_object)
      super ||
        comparison_object.instance_of?(self.class) &&
424
        !id.nil? &&
425 426 427 428 429 430 431
        comparison_object.id == id
    end
    alias :eql? :==

    # Delegates to id in order to allow two records of the same type and id to work with something like:
    #   [ Person.find(1), Person.find(2), Person.find(3) ] & [ Person.find(1), Person.find(4) ] # => [ Person.find(1) ]
    def hash
432
      if id
433
        [self.class, id].hash
434 435 436
      else
        super
      end
437 438
    end

439 440 441
    # Clone and freeze the attributes hash such that associations are still
    # accessible, even on destroyed records, but cloned models will not be
    # frozen.
442
    def freeze
443
      @attributes = @attributes.clone.freeze
444
      self
445 446 447 448
    end

    # Returns +true+ if the attributes hash has been frozen.
    def frozen?
449
      @attributes.frozen?
450 451
    end

452 453 454 455 456 457 458 459 460
    # Allows sort on objects
    def <=>(other_object)
      if other_object.is_a?(self.class)
        self.to_key <=> other_object.to_key
      else
        super
      end
    end

461 462 463 464 465 466 467 468 469 470 471
    # Returns +true+ if the record is read only. Records loaded through joins with piggy-back
    # attributes will be marked as read only since they cannot be saved.
    def readonly?
      @readonly
    end

    # Marks this record as read only.
    def readonly!
      @readonly = true
    end

472 473 474 475
    def connection_handler
      self.class.connection_handler
    end

476 477
    # Returns the contents of the record as a nicely formatted string.
    def inspect
478
      # We check defined?(@attributes) not to issue warnings if the object is
479
      # allocated but not initialized.
480
      inspection = if defined?(@attributes) && @attributes
481 482 483 484 485 486 487 488 489 490 491
                     self.class.column_names.collect { |name|
                       if has_attribute?(name)
                         "#{name}: #{attribute_for_inspect(name)}"
                       end
                     }.compact.join(", ")
                   else
                     "not initialized"
                   end
      "#<#{self.class} #{inspection}>"
    end

492
    # Takes a PP and prettily prints this record to it, allowing you to get a nice result from <tt>pp record</tt>
493 494
    # when pp is required.
    def pretty_print(pp)
495
      return super if custom_inspect_method_defined?
496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515
      pp.object_address_group(self) do
        if defined?(@attributes) && @attributes
          column_names = self.class.column_names.select { |name| has_attribute?(name) || new_record? }
          pp.seplist(column_names, proc { pp.text ',' }) do |column_name|
            column_value = read_attribute(column_name)
            pp.breakable ' '
            pp.group(1) do
              pp.text column_name
              pp.text ':'
              pp.breakable
              pp.pp column_value
            end
          end
        else
          pp.breakable ' '
          pp.text 'not initialized'
        end
      end
    end

516 517
    # Returns a hash of the given methods with their names as keys and returned values as values.
    def slice(*methods)
518
      Hash[methods.map! { |method| [method, public_send(method)] }].with_indifferent_access
519 520
    end

521 522
    private

523 524 525 526 527 528 529
    # Under Ruby 1.9, Array#flatten will call #to_ary (recursively) on each of the elements
    # of the array, and then rescues from the possible NoMethodError. If those elements are
    # ActiveRecord::Base's, then this triggers the various method_missing's that we have,
    # which significantly impacts upon performance.
    #
    # So we can avoid the method_missing hit by explicitly defining #to_ary as nil here.
    #
530
    # See also http://tenderlovemaking.com/2011/06/28/til-its-ok-to-return-nil-from-to_ary.html
531 532 533
    def to_ary # :nodoc:
      nil
    end
534 535

    def init_internals
536 537 538
      @readonly                 = false
      @destroyed                = false
      @marked_for_destruction   = false
539
      @destroyed_by_association = nil
540 541 542
      @new_record               = true
      @txn                      = nil
      @_start_transaction_state = {}
543
      @transaction_state        = nil
544
    end
545

546
    def initialize_internals_callback
547
    end
548

549 550
    def thaw
      if frozen?
551
        @attributes = @attributes.dup
552 553
      end
    end
554 555 556 557

    def custom_inspect_method_defined?
      self.class.instance_method(:inspect).owner != ActiveRecord::Base.instance_method(:inspect).owner
    end
558 559
  end
end