2019-03-05 14:16:44 -05:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require "cases/helper"
|
|
|
|
require "models/book"
|
2019-06-21 01:34:45 -04:00
|
|
|
require "models/speedometer"
|
2019-03-05 14:16:44 -05:00
|
|
|
|
|
|
|
class ReadonlyNameBook < Book
|
|
|
|
attr_readonly :name
|
|
|
|
end
|
|
|
|
|
2019-03-06 10:20:44 -05:00
|
|
|
class InsertAllTest < ActiveRecord::TestCase
|
2019-03-05 14:16:44 -05:00
|
|
|
fixtures :books
|
|
|
|
|
|
|
|
def test_insert
|
2019-03-31 14:41:18 -04:00
|
|
|
skip unless supports_insert_on_duplicate_skip?
|
|
|
|
|
|
|
|
id = 1_000_000
|
|
|
|
|
|
|
|
assert_difference "Book.count", +1 do
|
2019-09-15 16:46:26 -04:00
|
|
|
Book.insert({ id: id, name: "Rework", author_id: 1 })
|
2019-03-31 14:41:18 -04:00
|
|
|
end
|
|
|
|
|
2019-09-15 16:46:26 -04:00
|
|
|
Book.upsert({ id: id, name: "Remote", author_id: 1 })
|
2019-03-31 14:41:18 -04:00
|
|
|
|
|
|
|
assert_equal "Remote", Book.find(id).name
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert!
|
2019-03-05 14:16:44 -05:00
|
|
|
assert_difference "Book.count", +1 do
|
2019-09-15 16:46:26 -04:00
|
|
|
Book.insert!({ name: "Rework", author_id: 1 })
|
2019-03-05 14:16:44 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert_all
|
|
|
|
assert_difference "Book.count", +10 do
|
|
|
|
Book.insert_all! [
|
|
|
|
{ name: "Rework", author_id: 1 },
|
|
|
|
{ name: "Patterns of Enterprise Application Architecture", author_id: 1 },
|
|
|
|
{ name: "Design of Everyday Things", author_id: 1 },
|
|
|
|
{ name: "Practical Object-Oriented Design in Ruby", author_id: 1 },
|
|
|
|
{ name: "Clean Code", author_id: 1 },
|
|
|
|
{ name: "Ruby Under a Microscope", author_id: 1 },
|
|
|
|
{ name: "The Principles of Product Development Flow", author_id: 1 },
|
|
|
|
{ name: "Peopleware", author_id: 1 },
|
|
|
|
{ name: "About Face", author_id: 1 },
|
|
|
|
{ name: "Eloquent Ruby", author_id: 1 },
|
|
|
|
]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert_all_should_handle_empty_arrays
|
2019-03-06 10:19:53 -05:00
|
|
|
assert_raise ArgumentError do
|
2019-03-05 14:16:44 -05:00
|
|
|
Book.insert_all! []
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert_all_raises_on_duplicate_records
|
|
|
|
assert_raise ActiveRecord::RecordNotUnique do
|
|
|
|
Book.insert_all! [
|
|
|
|
{ name: "Rework", author_id: 1 },
|
|
|
|
{ name: "Patterns of Enterprise Application Architecture", author_id: 1 },
|
|
|
|
{ name: "Agile Web Development with Rails", author_id: 1 },
|
|
|
|
]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert_all_returns_ActiveRecord_Result
|
|
|
|
result = Book.insert_all! [{ name: "Rework", author_id: 1 }]
|
|
|
|
assert_kind_of ActiveRecord::Result, result
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert_all_returns_primary_key_if_returning_is_supported
|
|
|
|
skip unless supports_insert_returning?
|
2019-03-06 10:29:22 -05:00
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
result = Book.insert_all! [{ name: "Rework", author_id: 1 }]
|
|
|
|
assert_equal %w[ id ], result.columns
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert_all_returns_nothing_if_returning_is_empty
|
|
|
|
skip unless supports_insert_returning?
|
2019-03-06 10:29:22 -05:00
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
result = Book.insert_all! [{ name: "Rework", author_id: 1 }], returning: []
|
|
|
|
assert_equal [], result.columns
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert_all_returns_nothing_if_returning_is_false
|
|
|
|
skip unless supports_insert_returning?
|
2019-03-06 10:29:22 -05:00
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
result = Book.insert_all! [{ name: "Rework", author_id: 1 }], returning: false
|
|
|
|
assert_equal [], result.columns
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert_all_returns_requested_fields
|
|
|
|
skip unless supports_insert_returning?
|
2019-03-06 10:29:22 -05:00
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
result = Book.insert_all! [{ name: "Rework", author_id: 1 }], returning: [:id, :name]
|
|
|
|
assert_equal %w[ Rework ], result.pluck("name")
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert_all_can_skip_duplicate_records
|
|
|
|
skip unless supports_insert_on_duplicate_skip?
|
2019-03-06 10:29:22 -05:00
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
assert_no_difference "Book.count" do
|
|
|
|
Book.insert_all [{ id: 1, name: "Agile Web Development with Rails" }]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-04-04 08:35:04 -04:00
|
|
|
def test_insert_all_with_skip_duplicates_and_autonumber_id_not_given
|
|
|
|
skip unless supports_insert_on_duplicate_skip?
|
|
|
|
|
|
|
|
assert_difference "Book.count", 1 do
|
|
|
|
# These two books are duplicates according to an index on %i[author_id name]
|
|
|
|
# but their IDs are not specified so they will be assigned different IDs
|
|
|
|
# by autonumber. We will get an exception from MySQL if we attempt to skip
|
|
|
|
# one of these records by assigning its ID.
|
|
|
|
Book.insert_all [
|
|
|
|
{ author_id: 8, name: "Refactoring" },
|
|
|
|
{ author_id: 8, name: "Refactoring" }
|
|
|
|
]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert_all_with_skip_duplicates_and_autonumber_id_given
|
|
|
|
skip unless supports_insert_on_duplicate_skip?
|
|
|
|
|
|
|
|
assert_difference "Book.count", 1 do
|
|
|
|
Book.insert_all [
|
|
|
|
{ id: 200, author_id: 8, name: "Refactoring" },
|
|
|
|
{ id: 201, author_id: 8, name: "Refactoring" }
|
|
|
|
]
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_skip_duplicates_strategy_does_not_secretly_upsert
|
|
|
|
skip unless supports_insert_on_duplicate_skip?
|
|
|
|
|
|
|
|
book = Book.create!(author_id: 8, name: "Refactoring", format: "EXPECTED")
|
|
|
|
|
|
|
|
assert_no_difference "Book.count" do
|
2019-09-15 16:46:26 -04:00
|
|
|
Book.insert({ author_id: 8, name: "Refactoring", format: "UNEXPECTED" })
|
2019-04-04 08:35:04 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
assert_equal "EXPECTED", book.reload.format
|
|
|
|
end
|
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
def test_insert_all_will_raise_if_duplicates_are_skipped_only_for_a_certain_conflict_target
|
|
|
|
skip unless supports_insert_on_duplicate_skip? && supports_insert_conflict_target?
|
2019-03-06 10:29:22 -05:00
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
assert_raise ActiveRecord::RecordNotUnique do
|
|
|
|
Book.insert_all [{ id: 1, name: "Agile Web Development with Rails" }],
|
2019-03-08 19:15:17 -05:00
|
|
|
unique_by: :index_books_on_author_id_and_name
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert_all_and_upsert_all_with_index_finding_options
|
|
|
|
skip unless supports_insert_conflict_target?
|
|
|
|
|
2019-09-25 12:42:06 -04:00
|
|
|
assert_difference "Book.count", +4 do
|
2019-03-08 19:15:17 -05:00
|
|
|
Book.insert_all [{ name: "Rework", author_id: 1 }], unique_by: :isbn
|
|
|
|
Book.insert_all [{ name: "Remote", author_id: 1 }], unique_by: %i( author_id name )
|
|
|
|
Book.insert_all [{ name: "Renote", author_id: 1 }], unique_by: :index_books_on_isbn
|
2019-09-25 12:42:06 -04:00
|
|
|
Book.insert_all [{ name: "Recoat", author_id: 1 }], unique_by: :id
|
2019-03-08 19:15:17 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
assert_raise ActiveRecord::RecordNotUnique do
|
|
|
|
Book.upsert_all [{ name: "Rework", author_id: 1 }], unique_by: :isbn
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert_all_and_upsert_all_raises_when_index_is_missing
|
|
|
|
skip unless supports_insert_conflict_target?
|
|
|
|
|
|
|
|
[ :cats, %i( author_id isbn ), :author_id ].each do |missing_or_non_unique_by|
|
|
|
|
error = assert_raises ArgumentError do
|
|
|
|
Book.insert_all [{ name: "Rework", author_id: 1 }], unique_by: missing_or_non_unique_by
|
|
|
|
end
|
|
|
|
assert_match "No unique index", error.message
|
|
|
|
|
|
|
|
error = assert_raises ArgumentError do
|
|
|
|
Book.upsert_all [{ name: "Rework", author_id: 1 }], unique_by: missing_or_non_unique_by
|
|
|
|
end
|
|
|
|
assert_match "No unique index", error.message
|
2019-03-05 14:16:44 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-04-07 23:22:54 -04:00
|
|
|
def test_insert_logs_message_including_model_name
|
|
|
|
skip unless supports_insert_conflict_target?
|
|
|
|
|
|
|
|
capture_log_output do |output|
|
2019-09-15 16:46:26 -04:00
|
|
|
Book.insert({ name: "Rework", author_id: 1 })
|
2019-04-07 23:22:54 -04:00
|
|
|
assert_match "Book Insert", output.string
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_insert_all_logs_message_including_model_name
|
|
|
|
skip unless supports_insert_conflict_target?
|
|
|
|
|
|
|
|
capture_log_output do |output|
|
|
|
|
Book.insert_all [{ name: "Remote", author_id: 1 }, { name: "Renote", author_id: 1 }]
|
|
|
|
assert_match "Book Bulk Insert", output.string
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_upsert_logs_message_including_model_name
|
|
|
|
skip unless supports_insert_on_duplicate_update?
|
|
|
|
|
|
|
|
capture_log_output do |output|
|
2019-09-15 16:46:26 -04:00
|
|
|
Book.upsert({ name: "Remote", author_id: 1 })
|
2019-04-07 23:22:54 -04:00
|
|
|
assert_match "Book Upsert", output.string
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_upsert_all_logs_message_including_model_name
|
|
|
|
skip unless supports_insert_on_duplicate_update?
|
|
|
|
|
|
|
|
capture_log_output do |output|
|
|
|
|
Book.upsert_all [{ name: "Remote", author_id: 1 }, { name: "Renote", author_id: 1 }]
|
|
|
|
assert_match "Book Bulk Upsert", output.string
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
def test_upsert_all_updates_existing_records
|
|
|
|
skip unless supports_insert_on_duplicate_update?
|
2019-03-06 10:29:22 -05:00
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
new_name = "Agile Web Development with Rails, 4th Edition"
|
|
|
|
Book.upsert_all [{ id: 1, name: new_name }]
|
|
|
|
assert_equal new_name, Book.find(1).name
|
|
|
|
end
|
|
|
|
|
2019-06-21 01:34:45 -04:00
|
|
|
def test_upsert_all_updates_existing_record_by_primary_key
|
2019-09-25 12:42:06 -04:00
|
|
|
skip unless supports_insert_conflict_target?
|
2019-06-21 01:34:45 -04:00
|
|
|
|
|
|
|
Book.upsert_all [{ id: 1, name: "New edition" }], unique_by: :id
|
|
|
|
|
|
|
|
assert_equal "New edition", Book.find(1).name
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_upsert_all_updates_existing_record_by_configured_primary_key
|
|
|
|
skip unless supports_insert_on_duplicate_update?
|
|
|
|
|
|
|
|
error = assert_raises ArgumentError do
|
|
|
|
Speedometer.upsert_all [{ speedometer_id: "s1", name: "New Speedometer" }]
|
|
|
|
end
|
|
|
|
assert_match "No unique index found for speedometer_id", error.message
|
|
|
|
end
|
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
def test_upsert_all_does_not_update_readonly_attributes
|
|
|
|
skip unless supports_insert_on_duplicate_update?
|
2019-03-06 10:29:22 -05:00
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
new_name = "Agile Web Development with Rails, 4th Edition"
|
|
|
|
ReadonlyNameBook.upsert_all [{ id: 1, name: new_name }]
|
|
|
|
assert_not_equal new_name, Book.find(1).name
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_upsert_all_does_not_update_primary_keys
|
|
|
|
skip unless supports_insert_on_duplicate_update? && supports_insert_conflict_target?
|
2019-03-06 10:29:22 -05:00
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
Book.upsert_all [{ id: 101, name: "Perelandra", author_id: 7 }]
|
|
|
|
Book.upsert_all [{ id: 103, name: "Perelandra", author_id: 7, isbn: "1974522598" }],
|
2019-03-08 19:15:17 -05:00
|
|
|
unique_by: :index_books_on_author_id_and_name
|
2019-03-06 10:29:22 -05:00
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
book = Book.find_by(name: "Perelandra")
|
|
|
|
assert_equal 101, book.id, "Should not have updated the ID"
|
|
|
|
assert_equal "1974522598", book.isbn, "Should have updated the isbn"
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_upsert_all_does_not_perform_an_upsert_if_a_partial_index_doesnt_apply
|
|
|
|
skip unless supports_insert_on_duplicate_update? && supports_insert_conflict_target? && supports_partial_index?
|
2019-03-06 10:29:22 -05:00
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
Book.upsert_all [{ name: "Out of the Silent Planet", author_id: 7, isbn: "1974522598", published_on: Date.new(1938, 4, 1) }]
|
|
|
|
Book.upsert_all [{ name: "Perelandra", author_id: 7, isbn: "1974522598" }],
|
2019-03-08 19:15:17 -05:00
|
|
|
unique_by: :index_books_on_isbn
|
2019-03-06 10:29:22 -05:00
|
|
|
|
2019-03-05 14:16:44 -05:00
|
|
|
assert_equal ["Out of the Silent Planet", "Perelandra"], Book.where(isbn: "1974522598").order(:name).pluck(:name)
|
|
|
|
end
|
2019-03-15 22:33:48 -04:00
|
|
|
|
2019-11-02 19:37:48 -04:00
|
|
|
def test_upsert_all_does_not_touch_updated_at_when_values_do_not_change
|
|
|
|
skip unless supports_insert_on_duplicate_update?
|
|
|
|
|
|
|
|
updated_at = Time.now.utc - 5.years
|
|
|
|
Book.insert_all [{ id: 101, name: "Out of the Silent Planet", published_on: Date.new(1938, 4, 1), updated_at: updated_at }]
|
|
|
|
Book.upsert_all [{ id: 101, name: "Out of the Silent Planet", published_on: Date.new(1938, 4, 1) }]
|
|
|
|
|
|
|
|
assert_in_delta updated_at, Book.find(101).updated_at, 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_upsert_all_touches_updated_at_and_updated_on_when_values_change
|
|
|
|
skip unless supports_insert_on_duplicate_update?
|
|
|
|
|
|
|
|
Book.insert_all [{ id: 101, name: "Out of the Silent Planet", published_on: Date.new(1938, 4, 1), updated_at: 5.years.ago, updated_on: 5.years.ago }]
|
|
|
|
Book.upsert_all [{ id: 101, name: "Out of the Silent Planet", published_on: Date.new(1938, 4, 8) }]
|
|
|
|
|
|
|
|
assert_equal Time.now.year, Book.find(101).updated_at.year
|
|
|
|
assert_equal Time.now.year, Book.find(101).updated_on.year
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_upsert_all_uses_given_updated_at_over_implicit_updated_at
|
|
|
|
skip unless supports_insert_on_duplicate_update?
|
|
|
|
|
|
|
|
updated_at = Time.now.utc - 1.year
|
|
|
|
Book.insert_all [{ id: 101, name: "Out of the Silent Planet", published_on: Date.new(1938, 4, 1), updated_at: 5.years.ago }]
|
|
|
|
Book.upsert_all [{ id: 101, name: "Out of the Silent Planet", published_on: Date.new(1938, 4, 8), updated_at: updated_at }]
|
|
|
|
|
|
|
|
assert_in_delta updated_at, Book.find(101).updated_at, 1
|
|
|
|
end
|
|
|
|
|
|
|
|
def test_upsert_all_uses_given_updated_on_over_implicit_updated_on
|
|
|
|
skip unless supports_insert_on_duplicate_update?
|
|
|
|
|
|
|
|
updated_on = Time.now.utc.to_date - 30
|
|
|
|
Book.insert_all [{ id: 101, name: "Out of the Silent Planet", published_on: Date.new(1938, 4, 1), updated_on: 5.years.ago }]
|
|
|
|
Book.upsert_all [{ id: 101, name: "Out of the Silent Planet", published_on: Date.new(1938, 4, 8), updated_on: updated_on }]
|
|
|
|
|
|
|
|
assert_equal updated_on, Book.find(101).updated_on
|
|
|
|
end
|
|
|
|
|
2019-03-15 22:33:48 -04:00
|
|
|
def test_insert_all_raises_on_unknown_attribute
|
|
|
|
assert_raise ActiveRecord::UnknownAttributeError do
|
|
|
|
Book.insert_all! [{ unknown_attribute: "Test" }]
|
|
|
|
end
|
|
|
|
end
|
2019-04-07 23:22:54 -04:00
|
|
|
|
2019-10-29 07:54:03 -04:00
|
|
|
def test_upsert_all_works_with_partitioned_indexes
|
|
|
|
skip unless supports_insert_on_duplicate_update? && supports_insert_conflict_target? && supports_partitioned_indexes?
|
|
|
|
|
|
|
|
require "models/measurement"
|
|
|
|
|
|
|
|
Measurement.upsert_all([{ city_id: "1", logdate: 1.days.ago, peaktemp: 1, unitsales: 1 },
|
|
|
|
{ city_id: "2", logdate: 2.days.ago, peaktemp: 2, unitsales: 2 },
|
|
|
|
{ city_id: "2", logdate: 3.days.ago, peaktemp: 0, unitsales: 0 }],
|
|
|
|
unique_by: %i[logdate city_id])
|
|
|
|
assert_equal [[1.day.ago.to_date, 1, 1]],
|
|
|
|
Measurement.where(city_id: 1).pluck(:logdate, :peaktemp, :unitsales)
|
|
|
|
assert_equal [[2.days.ago.to_date, 2, 2], [3.days.ago.to_date, 0, 0]],
|
|
|
|
Measurement.where(city_id: 2).pluck(:logdate, :peaktemp, :unitsales)
|
|
|
|
end
|
|
|
|
|
2020-03-18 14:40:25 -04:00
|
|
|
def test_insert_all_with_enum_values
|
|
|
|
Book.insert_all! [{ status: :published, isbn: "1234566", name: "Rework", author_id: 1 },
|
|
|
|
{ status: :proposed, isbn: "1234567", name: "Remote", author_id: 2 }]
|
|
|
|
assert_equal ["published", "proposed"], Book.where(isbn: ["1234566", "1234567"]).order(:id).pluck(:status)
|
|
|
|
end
|
|
|
|
|
2019-04-07 23:22:54 -04:00
|
|
|
private
|
|
|
|
def capture_log_output
|
|
|
|
output = StringIO.new
|
|
|
|
old_logger, ActiveRecord::Base.logger = ActiveRecord::Base.logger, ActiveSupport::Logger.new(output)
|
|
|
|
|
|
|
|
begin
|
|
|
|
yield output
|
|
|
|
ensure
|
|
|
|
ActiveRecord::Base.logger = old_logger
|
|
|
|
end
|
|
|
|
end
|
2019-03-05 14:16:44 -05:00
|
|
|
end
|