activerecord 7.2.3 → 8.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/CHANGELOG.md +391 -958
- data/README.rdoc +1 -1
- data/lib/active_record/association_relation.rb +1 -0
- data/lib/active_record/associations/association.rb +34 -10
- data/lib/active_record/associations/builder/association.rb +7 -6
- data/lib/active_record/associations/collection_association.rb +1 -1
- data/lib/active_record/associations/disable_joins_association_scope.rb +1 -1
- data/lib/active_record/associations/has_many_through_association.rb +3 -2
- data/lib/active_record/associations/preloader/association.rb +2 -2
- data/lib/active_record/associations/singular_association.rb +8 -3
- data/lib/active_record/associations.rb +34 -4
- data/lib/active_record/asynchronous_queries_tracker.rb +28 -24
- data/lib/active_record/attribute_methods/primary_key.rb +4 -8
- data/lib/active_record/attribute_methods/query.rb +34 -0
- data/lib/active_record/attribute_methods/time_zone_conversion.rb +2 -12
- data/lib/active_record/autosave_association.rb +69 -27
- data/lib/active_record/connection_adapters/abstract/connection_handler.rb +34 -25
- data/lib/active_record/connection_adapters/abstract/connection_pool/queue.rb +0 -1
- data/lib/active_record/connection_adapters/abstract/connection_pool/reaper.rb +0 -1
- data/lib/active_record/connection_adapters/abstract/connection_pool.rb +6 -15
- data/lib/active_record/connection_adapters/abstract/database_statements.rb +90 -43
- data/lib/active_record/connection_adapters/abstract/query_cache.rb +8 -2
- data/lib/active_record/connection_adapters/abstract/quoting.rb +1 -1
- data/lib/active_record/connection_adapters/abstract/schema_creation.rb +4 -5
- data/lib/active_record/connection_adapters/abstract/schema_definitions.rb +7 -2
- data/lib/active_record/connection_adapters/abstract/schema_statements.rb +34 -7
- data/lib/active_record/connection_adapters/abstract/transaction.rb +15 -5
- data/lib/active_record/connection_adapters/abstract_adapter.rb +31 -43
- data/lib/active_record/connection_adapters/abstract_mysql_adapter.rb +21 -40
- data/lib/active_record/connection_adapters/mysql/quoting.rb +0 -8
- data/lib/active_record/connection_adapters/mysql/schema_definitions.rb +2 -8
- data/lib/active_record/connection_adapters/mysql/schema_statements.rb +50 -45
- data/lib/active_record/connection_adapters/mysql2/database_statements.rb +84 -94
- data/lib/active_record/connection_adapters/mysql2_adapter.rb +1 -8
- data/lib/active_record/connection_adapters/pool_config.rb +7 -7
- data/lib/active_record/connection_adapters/postgresql/database_statements.rb +72 -43
- data/lib/active_record/connection_adapters/postgresql/oid/array.rb +1 -1
- data/lib/active_record/connection_adapters/postgresql/oid/point.rb +10 -0
- data/lib/active_record/connection_adapters/postgresql/referential_integrity.rb +2 -4
- data/lib/active_record/connection_adapters/postgresql/schema_creation.rb +1 -11
- data/lib/active_record/connection_adapters/postgresql/schema_definitions.rb +6 -12
- data/lib/active_record/connection_adapters/postgresql/schema_dumper.rb +2 -1
- data/lib/active_record/connection_adapters/postgresql/schema_statements.rb +59 -16
- data/lib/active_record/connection_adapters/postgresql_adapter.rb +46 -96
- data/lib/active_record/connection_adapters/schema_cache.rb +1 -3
- data/lib/active_record/connection_adapters/sqlite3/database_statements.rb +80 -100
- data/lib/active_record/connection_adapters/sqlite3/schema_creation.rb +0 -6
- data/lib/active_record/connection_adapters/sqlite3/schema_dumper.rb +13 -0
- data/lib/active_record/connection_adapters/sqlite3/schema_statements.rb +9 -1
- data/lib/active_record/connection_adapters/sqlite3_adapter.rb +53 -12
- data/lib/active_record/connection_adapters/statement_pool.rb +4 -2
- data/lib/active_record/connection_adapters/trilogy/database_statements.rb +37 -67
- data/lib/active_record/connection_adapters/trilogy_adapter.rb +0 -17
- data/lib/active_record/connection_adapters.rb +0 -56
- data/lib/active_record/connection_handling.rb +23 -1
- data/lib/active_record/core.rb +29 -14
- data/lib/active_record/database_configurations/database_config.rb +4 -0
- data/lib/active_record/database_configurations/hash_config.rb +16 -2
- data/lib/active_record/encryption/config.rb +3 -1
- data/lib/active_record/encryption/encryptable_record.rb +4 -4
- data/lib/active_record/encryption/encrypted_attribute_type.rb +10 -1
- data/lib/active_record/encryption/encryptor.rb +16 -8
- data/lib/active_record/encryption/extended_deterministic_queries.rb +4 -2
- data/lib/active_record/encryption/scheme.rb +8 -1
- data/lib/active_record/enum.rb +9 -22
- data/lib/active_record/errors.rb +13 -5
- data/lib/active_record/fixtures.rb +0 -2
- data/lib/active_record/future_result.rb +13 -9
- data/lib/active_record/gem_version.rb +3 -3
- data/lib/active_record/insert_all.rb +1 -1
- data/lib/active_record/locking/optimistic.rb +1 -1
- data/lib/active_record/log_subscriber.rb +5 -11
- data/lib/active_record/migration/command_recorder.rb +31 -11
- data/lib/active_record/migration/compatibility.rb +5 -2
- data/lib/active_record/migration.rb +38 -42
- data/lib/active_record/model_schema.rb +3 -4
- data/lib/active_record/nested_attributes.rb +4 -6
- data/lib/active_record/persistence.rb +128 -130
- data/lib/active_record/query_logs.rb +102 -50
- data/lib/active_record/query_logs_formatter.rb +17 -28
- data/lib/active_record/querying.rb +8 -8
- data/lib/active_record/railtie.rb +2 -26
- data/lib/active_record/railties/databases.rake +11 -35
- data/lib/active_record/reflection.rb +18 -21
- data/lib/active_record/relation/batches/batch_enumerator.rb +4 -3
- data/lib/active_record/relation/batches.rb +132 -72
- data/lib/active_record/relation/calculations.rb +40 -39
- data/lib/active_record/relation/delegation.rb +25 -14
- data/lib/active_record/relation/finder_methods.rb +18 -18
- data/lib/active_record/relation/merger.rb +8 -8
- data/lib/active_record/relation/predicate_builder/polymorphic_array_value.rb +1 -1
- data/lib/active_record/relation/predicate_builder/relation_handler.rb +4 -3
- data/lib/active_record/relation/predicate_builder.rb +13 -0
- data/lib/active_record/relation/query_methods.rb +105 -61
- data/lib/active_record/relation/spawn_methods.rb +7 -7
- data/lib/active_record/relation.rb +79 -61
- data/lib/active_record/result.rb +66 -4
- data/lib/active_record/sanitization.rb +7 -6
- data/lib/active_record/schema_dumper.rb +5 -0
- data/lib/active_record/schema_migration.rb +2 -1
- data/lib/active_record/scoping/named.rb +5 -2
- data/lib/active_record/statement_cache.rb +14 -14
- data/lib/active_record/store.rb +7 -3
- data/lib/active_record/table_metadata.rb +1 -3
- data/lib/active_record/tasks/database_tasks.rb +69 -60
- data/lib/active_record/tasks/mysql_database_tasks.rb +0 -2
- data/lib/active_record/tasks/postgresql_database_tasks.rb +2 -1
- data/lib/active_record/tasks/sqlite_database_tasks.rb +2 -2
- data/lib/active_record/test_databases.rb +1 -1
- data/lib/active_record/test_fixtures.rb +12 -0
- data/lib/active_record/token_for.rb +1 -1
- data/lib/active_record/transactions.rb +5 -6
- data/lib/active_record/validations/uniqueness.rb +8 -8
- data/lib/active_record.rb +21 -48
- data/lib/arel/collectors/bind.rb +2 -2
- data/lib/arel/collectors/sql_string.rb +1 -1
- data/lib/arel/collectors/substitute_binds.rb +2 -2
- data/lib/arel/nodes/binary.rb +1 -1
- data/lib/arel/nodes/node.rb +1 -1
- data/lib/arel/nodes/sql_literal.rb +1 -1
- data/lib/arel/table.rb +3 -7
- metadata +9 -10
- data/lib/active_record/relation/record_fetch_warning.rb +0 -52
|
@@ -1,8 +1,5 @@
|
|
|
1
1
|
# frozen_string_literal: true
|
|
2
2
|
|
|
3
|
-
require "benchmark"
|
|
4
|
-
require "set"
|
|
5
|
-
require "zlib"
|
|
6
3
|
require "active_support/core_ext/array/access"
|
|
7
4
|
require "active_support/core_ext/enumerable"
|
|
8
5
|
require "active_support/core_ext/module/attribute_accessors"
|
|
@@ -21,7 +18,7 @@ module ActiveRecord
|
|
|
21
18
|
# For example the following migration is not reversible.
|
|
22
19
|
# Rolling back this migration will raise an ActiveRecord::IrreversibleMigration error.
|
|
23
20
|
#
|
|
24
|
-
# class IrreversibleMigrationExample < ActiveRecord::Migration[
|
|
21
|
+
# class IrreversibleMigrationExample < ActiveRecord::Migration[8.0]
|
|
25
22
|
# def change
|
|
26
23
|
# create_table :distributors do |t|
|
|
27
24
|
# t.string :zipcode
|
|
@@ -39,7 +36,7 @@ module ActiveRecord
|
|
|
39
36
|
#
|
|
40
37
|
# 1. Define <tt>#up</tt> and <tt>#down</tt> methods instead of <tt>#change</tt>:
|
|
41
38
|
#
|
|
42
|
-
# class ReversibleMigrationExample < ActiveRecord::Migration[
|
|
39
|
+
# class ReversibleMigrationExample < ActiveRecord::Migration[8.0]
|
|
43
40
|
# def up
|
|
44
41
|
# create_table :distributors do |t|
|
|
45
42
|
# t.string :zipcode
|
|
@@ -64,7 +61,7 @@ module ActiveRecord
|
|
|
64
61
|
#
|
|
65
62
|
# 2. Use the #reversible method in <tt>#change</tt> method:
|
|
66
63
|
#
|
|
67
|
-
# class ReversibleMigrationExample < ActiveRecord::Migration[
|
|
64
|
+
# class ReversibleMigrationExample < ActiveRecord::Migration[8.0]
|
|
68
65
|
# def change
|
|
69
66
|
# create_table :distributors do |t|
|
|
70
67
|
# t.string :zipcode
|
|
@@ -151,11 +148,10 @@ module ActiveRecord
|
|
|
151
148
|
include ActiveSupport::ActionableError
|
|
152
149
|
|
|
153
150
|
action "Run pending migrations" do
|
|
154
|
-
ActiveRecord::Tasks::DatabaseTasks.
|
|
151
|
+
ActiveRecord::Tasks::DatabaseTasks.migrate_all
|
|
155
152
|
|
|
156
153
|
if ActiveRecord.dump_schema_after_migration
|
|
157
|
-
|
|
158
|
-
ActiveRecord::Tasks::DatabaseTasks.dump_schema(connection.pool.db_config)
|
|
154
|
+
ActiveRecord::Tasks::DatabaseTasks.dump_all
|
|
159
155
|
end
|
|
160
156
|
end
|
|
161
157
|
|
|
@@ -250,7 +246,7 @@ module ActiveRecord
|
|
|
250
246
|
#
|
|
251
247
|
# Example of a simple migration:
|
|
252
248
|
#
|
|
253
|
-
# class AddSsl < ActiveRecord::Migration[
|
|
249
|
+
# class AddSsl < ActiveRecord::Migration[8.0]
|
|
254
250
|
# def up
|
|
255
251
|
# add_column :accounts, :ssl_enabled, :boolean, default: true
|
|
256
252
|
# end
|
|
@@ -270,7 +266,7 @@ module ActiveRecord
|
|
|
270
266
|
#
|
|
271
267
|
# Example of a more complex migration that also needs to initialize data:
|
|
272
268
|
#
|
|
273
|
-
# class AddSystemSettings < ActiveRecord::Migration[
|
|
269
|
+
# class AddSystemSettings < ActiveRecord::Migration[8.0]
|
|
274
270
|
# def up
|
|
275
271
|
# create_table :system_settings do |t|
|
|
276
272
|
# t.string :name
|
|
@@ -357,7 +353,7 @@ module ActiveRecord
|
|
|
357
353
|
#
|
|
358
354
|
# === Deletion
|
|
359
355
|
#
|
|
360
|
-
# * <tt>drop_table(
|
|
356
|
+
# * <tt>drop_table(*names)</tt>: Drops the given tables.
|
|
361
357
|
# * <tt>drop_join_table(table_1, table_2, options)</tt>: Drops the join table
|
|
362
358
|
# specified by the given arguments.
|
|
363
359
|
# * <tt>remove_column(table_name, column_name, type, options)</tt>: Removes the column
|
|
@@ -399,7 +395,7 @@ module ActiveRecord
|
|
|
399
395
|
# $ bin/rails generate migration add_fieldname_to_tablename fieldname:string
|
|
400
396
|
#
|
|
401
397
|
# This will generate the file <tt>timestamp_add_fieldname_to_tablename.rb</tt>, which will look like this:
|
|
402
|
-
# class AddFieldnameToTablename < ActiveRecord::Migration[
|
|
398
|
+
# class AddFieldnameToTablename < ActiveRecord::Migration[8.0]
|
|
403
399
|
# def change
|
|
404
400
|
# add_column :tablenames, :fieldname, :string
|
|
405
401
|
# end
|
|
@@ -425,7 +421,7 @@ module ActiveRecord
|
|
|
425
421
|
#
|
|
426
422
|
# Not all migrations change the schema. Some just fix the data:
|
|
427
423
|
#
|
|
428
|
-
# class RemoveEmptyTags < ActiveRecord::Migration[
|
|
424
|
+
# class RemoveEmptyTags < ActiveRecord::Migration[8.0]
|
|
429
425
|
# def up
|
|
430
426
|
# Tag.all.each { |tag| tag.destroy if tag.pages.empty? }
|
|
431
427
|
# end
|
|
@@ -438,7 +434,7 @@ module ActiveRecord
|
|
|
438
434
|
#
|
|
439
435
|
# Others remove columns when they migrate up instead of down:
|
|
440
436
|
#
|
|
441
|
-
# class RemoveUnnecessaryItemAttributes < ActiveRecord::Migration[
|
|
437
|
+
# class RemoveUnnecessaryItemAttributes < ActiveRecord::Migration[8.0]
|
|
442
438
|
# def up
|
|
443
439
|
# remove_column :items, :incomplete_items_count
|
|
444
440
|
# remove_column :items, :completed_items_count
|
|
@@ -452,7 +448,7 @@ module ActiveRecord
|
|
|
452
448
|
#
|
|
453
449
|
# And sometimes you need to do something in SQL not abstracted directly by migrations:
|
|
454
450
|
#
|
|
455
|
-
# class MakeJoinUnique < ActiveRecord::Migration[
|
|
451
|
+
# class MakeJoinUnique < ActiveRecord::Migration[8.0]
|
|
456
452
|
# def up
|
|
457
453
|
# execute "ALTER TABLE `pages_linked_pages` ADD UNIQUE `page_id_linked_page_id` (`page_id`,`linked_page_id`)"
|
|
458
454
|
# end
|
|
@@ -469,7 +465,7 @@ module ActiveRecord
|
|
|
469
465
|
# <tt>Base#reset_column_information</tt> in order to ensure that the model has the
|
|
470
466
|
# latest column data from after the new column was added. Example:
|
|
471
467
|
#
|
|
472
|
-
# class AddPeopleSalary < ActiveRecord::Migration[
|
|
468
|
+
# class AddPeopleSalary < ActiveRecord::Migration[8.0]
|
|
473
469
|
# def up
|
|
474
470
|
# add_column :people, :salary, :integer
|
|
475
471
|
# Person.reset_column_information
|
|
@@ -531,7 +527,7 @@ module ActiveRecord
|
|
|
531
527
|
# To define a reversible migration, define the +change+ method in your
|
|
532
528
|
# migration like this:
|
|
533
529
|
#
|
|
534
|
-
# class TenderloveMigration < ActiveRecord::Migration[
|
|
530
|
+
# class TenderloveMigration < ActiveRecord::Migration[8.0]
|
|
535
531
|
# def change
|
|
536
532
|
# create_table(:horses) do |t|
|
|
537
533
|
# t.column :content, :text
|
|
@@ -561,7 +557,7 @@ module ActiveRecord
|
|
|
561
557
|
# can't execute inside a transaction though, and for these situations
|
|
562
558
|
# you can turn the automatic transactions off.
|
|
563
559
|
#
|
|
564
|
-
# class ChangeEnum < ActiveRecord::Migration[
|
|
560
|
+
# class ChangeEnum < ActiveRecord::Migration[8.0]
|
|
565
561
|
# disable_ddl_transaction!
|
|
566
562
|
#
|
|
567
563
|
# def up
|
|
@@ -604,7 +600,7 @@ module ActiveRecord
|
|
|
604
600
|
end
|
|
605
601
|
end
|
|
606
602
|
|
|
607
|
-
def drop_table(
|
|
603
|
+
def drop_table(*table_names, **options)
|
|
608
604
|
if block_given?
|
|
609
605
|
super { |t| yield compatible_table_definition(t) }
|
|
610
606
|
else
|
|
@@ -681,10 +677,6 @@ module ActiveRecord
|
|
|
681
677
|
paths = all_configs.flat_map { |config| config.migrations_paths || Migrator.migrations_paths }.uniq
|
|
682
678
|
@file_watcher.new([], paths.index_with(["rb"]), &block)
|
|
683
679
|
end
|
|
684
|
-
|
|
685
|
-
def connection
|
|
686
|
-
ActiveRecord::Tasks::DatabaseTasks.migration_connection
|
|
687
|
-
end
|
|
688
680
|
end
|
|
689
681
|
|
|
690
682
|
class << self
|
|
@@ -715,13 +707,7 @@ module ActiveRecord
|
|
|
715
707
|
|
|
716
708
|
def load_schema_if_pending!
|
|
717
709
|
if any_schema_needs_update?
|
|
718
|
-
|
|
719
|
-
root = defined?(ENGINE_ROOT) ? ENGINE_ROOT : Rails.root
|
|
720
|
-
|
|
721
|
-
FileUtils.cd(root) do
|
|
722
|
-
Base.connection_handler.clear_all_connections!(:all)
|
|
723
|
-
system("bin/rails db:test:prepare")
|
|
724
|
-
end
|
|
710
|
+
load_schema!
|
|
725
711
|
end
|
|
726
712
|
|
|
727
713
|
check_pending_migrations
|
|
@@ -760,7 +746,7 @@ module ActiveRecord
|
|
|
760
746
|
private
|
|
761
747
|
def any_schema_needs_update?
|
|
762
748
|
!db_configs_in_current_env.all? do |db_config|
|
|
763
|
-
Tasks::DatabaseTasks.schema_up_to_date?(db_config
|
|
749
|
+
Tasks::DatabaseTasks.schema_up_to_date?(db_config)
|
|
764
750
|
end
|
|
765
751
|
end
|
|
766
752
|
|
|
@@ -785,6 +771,16 @@ module ActiveRecord
|
|
|
785
771
|
def env
|
|
786
772
|
ActiveRecord::ConnectionHandling::DEFAULT_ENV.call
|
|
787
773
|
end
|
|
774
|
+
|
|
775
|
+
def load_schema!
|
|
776
|
+
# Roundtrip to Rake to allow plugins to hook into database initialization.
|
|
777
|
+
root = defined?(ENGINE_ROOT) ? ENGINE_ROOT : Rails.root
|
|
778
|
+
|
|
779
|
+
FileUtils.cd(root) do
|
|
780
|
+
Base.connection_handler.clear_all_connections!(:all)
|
|
781
|
+
system("bin/rails db:test:prepare")
|
|
782
|
+
end
|
|
783
|
+
end
|
|
788
784
|
end
|
|
789
785
|
|
|
790
786
|
def disable_ddl_transaction # :nodoc:
|
|
@@ -822,7 +818,7 @@ module ActiveRecord
|
|
|
822
818
|
# and create the table 'apples' on the way up, and the reverse
|
|
823
819
|
# on the way down.
|
|
824
820
|
#
|
|
825
|
-
# class FixTLMigration < ActiveRecord::Migration[
|
|
821
|
+
# class FixTLMigration < ActiveRecord::Migration[8.0]
|
|
826
822
|
# def change
|
|
827
823
|
# revert do
|
|
828
824
|
# create_table(:horses) do |t|
|
|
@@ -841,7 +837,7 @@ module ActiveRecord
|
|
|
841
837
|
#
|
|
842
838
|
# require_relative "20121212123456_tenderlove_migration"
|
|
843
839
|
#
|
|
844
|
-
# class FixupTLMigration < ActiveRecord::Migration[
|
|
840
|
+
# class FixupTLMigration < ActiveRecord::Migration[8.0]
|
|
845
841
|
# def change
|
|
846
842
|
# revert TenderloveMigration
|
|
847
843
|
#
|
|
@@ -892,7 +888,7 @@ module ActiveRecord
|
|
|
892
888
|
# when the three columns 'first_name', 'last_name' and 'full_name' exist,
|
|
893
889
|
# even when migrating down:
|
|
894
890
|
#
|
|
895
|
-
# class SplitNameMigration < ActiveRecord::Migration[
|
|
891
|
+
# class SplitNameMigration < ActiveRecord::Migration[8.0]
|
|
896
892
|
# def change
|
|
897
893
|
# add_column :users, :first_name, :string
|
|
898
894
|
# add_column :users, :last_name, :string
|
|
@@ -920,7 +916,7 @@ module ActiveRecord
|
|
|
920
916
|
# In the following example, the new column +published+ will be given
|
|
921
917
|
# the value +true+ for all existing records.
|
|
922
918
|
#
|
|
923
|
-
# class AddPublishedToPosts < ActiveRecord::Migration[
|
|
919
|
+
# class AddPublishedToPosts < ActiveRecord::Migration[8.0]
|
|
924
920
|
# def change
|
|
925
921
|
# add_column :posts, :published, :boolean, default: false
|
|
926
922
|
# up_only do
|
|
@@ -972,16 +968,16 @@ module ActiveRecord
|
|
|
972
968
|
when :down then announce "reverting"
|
|
973
969
|
end
|
|
974
970
|
|
|
975
|
-
|
|
971
|
+
time_elapsed = nil
|
|
976
972
|
ActiveRecord::Tasks::DatabaseTasks.migration_connection.pool.with_connection do |conn|
|
|
977
|
-
|
|
973
|
+
time_elapsed = ActiveSupport::Benchmark.realtime do
|
|
978
974
|
exec_migration(conn, direction)
|
|
979
975
|
end
|
|
980
976
|
end
|
|
981
977
|
|
|
982
978
|
case direction
|
|
983
|
-
when :up then announce "migrated (%.4fs)" %
|
|
984
|
-
when :down then announce "reverted (%.4fs)" %
|
|
979
|
+
when :up then announce "migrated (%.4fs)" % time_elapsed; write
|
|
980
|
+
when :down then announce "reverted (%.4fs)" % time_elapsed; write
|
|
985
981
|
end
|
|
986
982
|
end
|
|
987
983
|
|
|
@@ -1022,8 +1018,8 @@ module ActiveRecord
|
|
|
1022
1018
|
def say_with_time(message)
|
|
1023
1019
|
say(message)
|
|
1024
1020
|
result = nil
|
|
1025
|
-
|
|
1026
|
-
say "%.4fs" %
|
|
1021
|
+
time_elapsed = ActiveSupport::Benchmark.realtime { result = yield }
|
|
1022
|
+
say "%.4fs" % time_elapsed, :subitem
|
|
1027
1023
|
say("#{result} rows", :subitem) if result.is_a?(Integer)
|
|
1028
1024
|
result
|
|
1029
1025
|
end
|
|
@@ -276,15 +276,14 @@ module ActiveRecord
|
|
|
276
276
|
end
|
|
277
277
|
|
|
278
278
|
@table_name = value
|
|
279
|
-
@quoted_table_name = nil
|
|
280
279
|
@arel_table = nil
|
|
281
280
|
@sequence_name = nil unless @explicit_sequence_name
|
|
282
281
|
@predicate_builder = nil
|
|
283
282
|
end
|
|
284
283
|
|
|
285
|
-
# Returns a quoted version of the table name
|
|
284
|
+
# Returns a quoted version of the table name.
|
|
286
285
|
def quoted_table_name
|
|
287
|
-
|
|
286
|
+
adapter_class.quote_table_name(table_name)
|
|
288
287
|
end
|
|
289
288
|
|
|
290
289
|
# Computes the table name, (re)sets it internally, and returns it.
|
|
@@ -502,7 +501,7 @@ module ActiveRecord
|
|
|
502
501
|
# when just after creating a table you want to populate it with some default
|
|
503
502
|
# values, e.g.:
|
|
504
503
|
#
|
|
505
|
-
# class CreateJobLevels < ActiveRecord::Migration[
|
|
504
|
+
# class CreateJobLevels < ActiveRecord::Migration[8.0]
|
|
506
505
|
# def up
|
|
507
506
|
# create_table :job_levels do |t|
|
|
508
507
|
# t.integer :id
|
|
@@ -524,12 +524,12 @@ module ActiveRecord
|
|
|
524
524
|
unless reject_new_record?(association_name, attributes)
|
|
525
525
|
association.reader.build(attributes.except(*UNASSIGNABLE_KEYS))
|
|
526
526
|
end
|
|
527
|
-
elsif existing_record = find_record_by_id(existing_records, attributes["id"])
|
|
527
|
+
elsif existing_record = find_record_by_id(association.klass, existing_records, attributes["id"])
|
|
528
528
|
unless call_reject_if(association_name, attributes)
|
|
529
529
|
# Make sure we are operating on the actual object which is in the association's
|
|
530
530
|
# proxy_target array (either by finding it, or adding it if not found)
|
|
531
531
|
# Take into account that the proxy_target may have changed due to callbacks
|
|
532
|
-
target_record = find_record_by_id(association.target, attributes["id"])
|
|
532
|
+
target_record = find_record_by_id(association.klass, association.target, attributes["id"])
|
|
533
533
|
if target_record
|
|
534
534
|
existing_record = target_record
|
|
535
535
|
else
|
|
@@ -621,10 +621,8 @@ module ActiveRecord
|
|
|
621
621
|
model, "id", record_id)
|
|
622
622
|
end
|
|
623
623
|
|
|
624
|
-
def find_record_by_id(records, id)
|
|
625
|
-
|
|
626
|
-
|
|
627
|
-
if records.first.class.composite_primary_key?
|
|
624
|
+
def find_record_by_id(klass, records, id)
|
|
625
|
+
if klass.composite_primary_key?
|
|
628
626
|
id = Array(id).map(&:to_s)
|
|
629
627
|
records.find { |record| Array(record.id).map(&:to_s) == id }
|
|
630
628
|
else
|
|
@@ -248,18 +248,16 @@ module ActiveRecord
|
|
|
248
248
|
|
|
249
249
|
im = Arel::InsertManager.new(arel_table)
|
|
250
250
|
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
im.insert(values.transform_keys { |name| arel_table[name] })
|
|
256
|
-
end
|
|
257
|
-
|
|
258
|
-
connection.insert(
|
|
259
|
-
im, "#{self} Create", primary_key || false, primary_key_value,
|
|
260
|
-
returning: returning
|
|
261
|
-
)
|
|
251
|
+
if values.empty?
|
|
252
|
+
im.insert(connection.empty_insert_statement_value(primary_key))
|
|
253
|
+
else
|
|
254
|
+
im.insert(values.transform_keys { |name| arel_table[name] })
|
|
262
255
|
end
|
|
256
|
+
|
|
257
|
+
connection.insert(
|
|
258
|
+
im, "#{self} Create", primary_key || false, primary_key_value,
|
|
259
|
+
returning: returning
|
|
260
|
+
)
|
|
263
261
|
end
|
|
264
262
|
|
|
265
263
|
def _update_record(values, constraints) # :nodoc:
|
|
@@ -812,159 +810,159 @@ module ActiveRecord
|
|
|
812
810
|
end
|
|
813
811
|
end
|
|
814
812
|
|
|
815
|
-
|
|
816
|
-
|
|
817
|
-
|
|
818
|
-
|
|
819
|
-
|
|
820
|
-
|
|
813
|
+
private
|
|
814
|
+
def init_internals
|
|
815
|
+
super
|
|
816
|
+
@_trigger_destroy_callback = @_trigger_update_callback = nil
|
|
817
|
+
@previously_new_record = false
|
|
818
|
+
end
|
|
821
819
|
|
|
822
|
-
|
|
823
|
-
|
|
824
|
-
|
|
825
|
-
|
|
826
|
-
|
|
820
|
+
def strict_loaded_associations
|
|
821
|
+
@association_cache.find_all do |_, assoc|
|
|
822
|
+
assoc.owner.strict_loading? && !assoc.owner.strict_loading_n_plus_one_only?
|
|
823
|
+
end.map(&:first)
|
|
824
|
+
end
|
|
827
825
|
|
|
828
|
-
|
|
829
|
-
|
|
830
|
-
|
|
826
|
+
def _find_record(options)
|
|
827
|
+
all_queries = options ? options[:all_queries] : nil
|
|
828
|
+
base = self.class.all(all_queries: all_queries).preload(strict_loaded_associations)
|
|
831
829
|
|
|
832
|
-
|
|
833
|
-
|
|
834
|
-
|
|
835
|
-
|
|
830
|
+
if options && options[:lock]
|
|
831
|
+
base.lock(options[:lock]).find_by!(_in_memory_query_constraints_hash)
|
|
832
|
+
else
|
|
833
|
+
base.find_by!(_in_memory_query_constraints_hash)
|
|
834
|
+
end
|
|
836
835
|
end
|
|
837
|
-
end
|
|
838
836
|
|
|
839
|
-
|
|
840
|
-
|
|
841
|
-
|
|
842
|
-
|
|
843
|
-
|
|
844
|
-
|
|
837
|
+
def _in_memory_query_constraints_hash
|
|
838
|
+
if self.class.query_constraints_list.nil?
|
|
839
|
+
{ @primary_key => id }
|
|
840
|
+
else
|
|
841
|
+
self.class.query_constraints_list.index_with do |column_name|
|
|
842
|
+
attribute(column_name)
|
|
843
|
+
end
|
|
845
844
|
end
|
|
846
845
|
end
|
|
847
|
-
end
|
|
848
846
|
|
|
849
|
-
|
|
850
|
-
|
|
851
|
-
|
|
852
|
-
|
|
847
|
+
def apply_scoping?(options)
|
|
848
|
+
!(options && options[:unscoped]) &&
|
|
849
|
+
(self.class.default_scopes?(all_queries: true) || self.class.global_current_scope)
|
|
850
|
+
end
|
|
853
851
|
|
|
854
|
-
|
|
855
|
-
|
|
856
|
-
|
|
857
|
-
|
|
858
|
-
|
|
859
|
-
|
|
852
|
+
def _query_constraints_hash
|
|
853
|
+
if self.class.query_constraints_list.nil?
|
|
854
|
+
{ @primary_key => id_in_database }
|
|
855
|
+
else
|
|
856
|
+
self.class.query_constraints_list.index_with do |column_name|
|
|
857
|
+
attribute_in_database(column_name)
|
|
858
|
+
end
|
|
860
859
|
end
|
|
861
860
|
end
|
|
862
|
-
end
|
|
863
861
|
|
|
864
|
-
|
|
865
|
-
|
|
866
|
-
|
|
867
|
-
|
|
868
|
-
def destroy_row
|
|
869
|
-
_delete_row
|
|
870
|
-
end
|
|
871
|
-
|
|
872
|
-
def _delete_row
|
|
873
|
-
self.class._delete_record(_query_constraints_hash)
|
|
874
|
-
end
|
|
862
|
+
# A hook to be overridden by association modules.
|
|
863
|
+
def destroy_associations
|
|
864
|
+
end
|
|
875
865
|
|
|
876
|
-
|
|
877
|
-
|
|
866
|
+
def destroy_row
|
|
867
|
+
_delete_row
|
|
868
|
+
end
|
|
878
869
|
|
|
879
|
-
|
|
880
|
-
|
|
870
|
+
def _delete_row
|
|
871
|
+
self.class._delete_record(_query_constraints_hash)
|
|
881
872
|
end
|
|
882
873
|
|
|
883
|
-
|
|
884
|
-
|
|
874
|
+
def _touch_row(attribute_names, time)
|
|
875
|
+
time ||= current_time_from_proper_timezone
|
|
885
876
|
|
|
886
|
-
|
|
887
|
-
|
|
888
|
-
|
|
889
|
-
_query_constraints_hash
|
|
890
|
-
)
|
|
891
|
-
end
|
|
877
|
+
attribute_names.each do |attr_name|
|
|
878
|
+
_write_attribute(attr_name, time)
|
|
879
|
+
end
|
|
892
880
|
|
|
893
|
-
|
|
894
|
-
|
|
895
|
-
return false if destroyed?
|
|
896
|
-
result = new_record? ? _create_record(&block) : _update_record(&block)
|
|
897
|
-
result != false
|
|
898
|
-
end
|
|
881
|
+
_update_row(attribute_names, "touch")
|
|
882
|
+
end
|
|
899
883
|
|
|
900
|
-
|
|
901
|
-
|
|
902
|
-
|
|
903
|
-
|
|
884
|
+
def _update_row(attribute_names, attempted_action = "update")
|
|
885
|
+
self.class._update_record(
|
|
886
|
+
attributes_with_values(attribute_names),
|
|
887
|
+
_query_constraints_hash
|
|
888
|
+
)
|
|
889
|
+
end
|
|
904
890
|
|
|
905
|
-
|
|
906
|
-
|
|
907
|
-
|
|
908
|
-
|
|
909
|
-
|
|
910
|
-
@_trigger_update_callback = affected_rows == 1
|
|
891
|
+
def create_or_update(**, &block)
|
|
892
|
+
_raise_readonly_record_error if readonly?
|
|
893
|
+
return false if destroyed?
|
|
894
|
+
result = new_record? ? _create_record(&block) : _update_record(&block)
|
|
895
|
+
result != false
|
|
911
896
|
end
|
|
912
897
|
|
|
913
|
-
|
|
898
|
+
# Updates the associated record with values matching those of the instance attributes.
|
|
899
|
+
# Returns the number of affected rows.
|
|
900
|
+
def _update_record(attribute_names = self.attribute_names)
|
|
901
|
+
attribute_names = attributes_for_update(attribute_names)
|
|
914
902
|
|
|
915
|
-
|
|
903
|
+
if attribute_names.empty?
|
|
904
|
+
affected_rows = 0
|
|
905
|
+
@_trigger_update_callback = true
|
|
906
|
+
else
|
|
907
|
+
affected_rows = _update_row(attribute_names)
|
|
908
|
+
@_trigger_update_callback = affected_rows == 1
|
|
909
|
+
end
|
|
916
910
|
|
|
917
|
-
|
|
918
|
-
end
|
|
911
|
+
@previously_new_record = false
|
|
919
912
|
|
|
920
|
-
|
|
921
|
-
# and returns its id.
|
|
922
|
-
def _create_record(attribute_names = self.attribute_names)
|
|
923
|
-
attribute_names = attributes_for_create(attribute_names)
|
|
913
|
+
yield(self) if block_given?
|
|
924
914
|
|
|
925
|
-
|
|
926
|
-
|
|
915
|
+
affected_rows
|
|
916
|
+
end
|
|
927
917
|
|
|
928
|
-
|
|
929
|
-
|
|
930
|
-
|
|
931
|
-
|
|
932
|
-
)
|
|
918
|
+
# Creates a record with values matching those of the instance attributes
|
|
919
|
+
# and returns its id.
|
|
920
|
+
def _create_record(attribute_names = self.attribute_names)
|
|
921
|
+
attribute_names = attributes_for_create(attribute_names)
|
|
933
922
|
|
|
934
|
-
|
|
935
|
-
|
|
936
|
-
end if returning_values
|
|
937
|
-
end
|
|
923
|
+
self.class.with_connection do |connection|
|
|
924
|
+
returning_columns = self.class._returning_columns_for_insert(connection)
|
|
938
925
|
|
|
939
|
-
|
|
940
|
-
|
|
926
|
+
returning_values = self.class._insert_record(
|
|
927
|
+
connection,
|
|
928
|
+
attributes_with_values(attribute_names),
|
|
929
|
+
returning_columns
|
|
930
|
+
)
|
|
941
931
|
|
|
942
|
-
|
|
932
|
+
returning_columns.zip(returning_values).each do |column, value|
|
|
933
|
+
_write_attribute(column, type_for_attribute(column).deserialize(value)) if !_read_attribute(column)
|
|
934
|
+
end if returning_values
|
|
935
|
+
end
|
|
943
936
|
|
|
944
|
-
|
|
945
|
-
|
|
937
|
+
@new_record = false
|
|
938
|
+
@previously_new_record = true
|
|
946
939
|
|
|
947
|
-
|
|
948
|
-
raise ActiveRecordError, "#{name} is marked as readonly" if self.class.readonly_attribute?(name)
|
|
949
|
-
end
|
|
940
|
+
yield(self) if block_given?
|
|
950
941
|
|
|
951
|
-
|
|
952
|
-
|
|
953
|
-
key = self.class.primary_key
|
|
954
|
-
raise @_association_destroy_exception || RecordNotDestroyed.new("Failed to destroy #{self.class} with #{key}=#{id}", self)
|
|
955
|
-
ensure
|
|
956
|
-
@_association_destroy_exception = nil
|
|
957
|
-
end
|
|
942
|
+
id
|
|
943
|
+
end
|
|
958
944
|
|
|
959
|
-
|
|
960
|
-
|
|
961
|
-
|
|
945
|
+
def verify_readonly_attribute(name)
|
|
946
|
+
raise ActiveRecordError, "#{name} is marked as readonly" if self.class.readonly_attribute?(name)
|
|
947
|
+
end
|
|
962
948
|
|
|
963
|
-
|
|
964
|
-
|
|
965
|
-
|
|
966
|
-
|
|
967
|
-
|
|
968
|
-
|
|
949
|
+
def _raise_record_not_destroyed
|
|
950
|
+
@_association_destroy_exception ||= nil
|
|
951
|
+
key = self.class.primary_key
|
|
952
|
+
raise @_association_destroy_exception || RecordNotDestroyed.new("Failed to destroy #{self.class} with #{key}=#{id}", self)
|
|
953
|
+
ensure
|
|
954
|
+
@_association_destroy_exception = nil
|
|
955
|
+
end
|
|
956
|
+
|
|
957
|
+
def _raise_readonly_record_error
|
|
958
|
+
raise ReadOnlyRecord, "#{self.class} is marked as readonly"
|
|
959
|
+
end
|
|
960
|
+
|
|
961
|
+
def _raise_record_not_touched_error
|
|
962
|
+
raise ActiveRecordError, <<~MSG.squish
|
|
963
|
+
Cannot touch on a new or destroyed record object. Consider using
|
|
964
|
+
persisted?, new_record?, or destroyed? before touching.
|
|
965
|
+
MSG
|
|
966
|
+
end
|
|
969
967
|
end
|
|
970
968
|
end
|