Support `returning` for `BulkInsertSafe.bulk_insert!`

This allows us to get an array with a hash containing
a primary key and its value for all successfully
inserted records.
parent e27d8f25
...@@ -68,6 +68,9 @@ module BulkInsertSafe ...@@ -68,6 +68,9 @@ module BulkInsertSafe
# @param [Boolean] validate Whether validations should run on [items] # @param [Boolean] validate Whether validations should run on [items]
# @param [Integer] batch_size How many items should at most be inserted at once # @param [Integer] batch_size How many items should at most be inserted at once
# @param [Boolean] skip_duplicates Marks duplicates as allowed, and skips inserting them # @param [Boolean] skip_duplicates Marks duplicates as allowed, and skips inserting them
# @param [Symbol] returns Pass :ids to return an array with the primary key values
# for all inserted records or nil to omit the underlying
# RETURNING SQL clause entirely.
# @param [Proc] handle_attributes Block that will receive each item attribute hash # @param [Proc] handle_attributes Block that will receive each item attribute hash
# prior to insertion for further processing # prior to insertion for further processing
# #
...@@ -78,10 +81,11 @@ module BulkInsertSafe ...@@ -78,10 +81,11 @@ module BulkInsertSafe
# #
# @return true if operation succeeded, throws otherwise. # @return true if operation succeeded, throws otherwise.
# #
def bulk_insert!(items, validate: true, skip_duplicates: false, batch_size: DEFAULT_BATCH_SIZE, &handle_attributes) def bulk_insert!(items, validate: true, skip_duplicates: false, returns: nil, batch_size: DEFAULT_BATCH_SIZE, &handle_attributes)
_bulk_insert_all!(items, _bulk_insert_all!(items,
validate: validate, validate: validate,
on_duplicate: skip_duplicates ? :skip : :raise, on_duplicate: skip_duplicates ? :skip : :raise,
returns: returns,
unique_by: nil, unique_by: nil,
batch_size: batch_size, batch_size: batch_size,
&handle_attributes) &handle_attributes)
...@@ -94,6 +98,9 @@ module BulkInsertSafe ...@@ -94,6 +98,9 @@ module BulkInsertSafe
# @param [Boolean] validate Whether validations should run on [items] # @param [Boolean] validate Whether validations should run on [items]
# @param [Integer] batch_size How many items should at most be inserted at once # @param [Integer] batch_size How many items should at most be inserted at once
# @param [Symbol/Array] unique_by Defines index or columns to use to consider item duplicate # @param [Symbol/Array] unique_by Defines index or columns to use to consider item duplicate
# @param [Symbol] returns Pass :ids to return an array with the primary key values
# for all inserted or updated records or nil to omit the
# underlying RETURNING SQL clause entirely.
# @param [Proc] handle_attributes Block that will receive each item attribute hash # @param [Proc] handle_attributes Block that will receive each item attribute hash
# prior to insertion for further processing # prior to insertion for further processing
# #
...@@ -109,10 +116,11 @@ module BulkInsertSafe ...@@ -109,10 +116,11 @@ module BulkInsertSafe
# #
# @return true if operation succeeded, throws otherwise. # @return true if operation succeeded, throws otherwise.
# #
def bulk_upsert!(items, unique_by:, validate: true, batch_size: DEFAULT_BATCH_SIZE, &handle_attributes) def bulk_upsert!(items, unique_by:, returns: nil, validate: true, batch_size: DEFAULT_BATCH_SIZE, &handle_attributes)
_bulk_insert_all!(items, _bulk_insert_all!(items,
validate: validate, validate: validate,
on_duplicate: :update, on_duplicate: :update,
returns: returns,
unique_by: unique_by, unique_by: unique_by,
batch_size: batch_size, batch_size: batch_size,
&handle_attributes) &handle_attributes)
...@@ -120,21 +128,30 @@ module BulkInsertSafe ...@@ -120,21 +128,30 @@ module BulkInsertSafe
private private
def _bulk_insert_all!(items, on_duplicate:, unique_by:, validate:, batch_size:, &handle_attributes) def _bulk_insert_all!(items, on_duplicate:, returns:, unique_by:, validate:, batch_size:, &handle_attributes)
return true if items.empty? return [] if items.empty?
returning =
case returns
when :ids
[primary_key]
when nil
false
else
raise ArgumentError, "returns needs to be :ids or nil"
end
transaction do transaction do
items.each_slice(batch_size) do |item_batch| items.each_slice(batch_size).flat_map do |item_batch|
attributes = _bulk_insert_item_attributes( attributes = _bulk_insert_item_attributes(
item_batch, validate, &handle_attributes) item_batch, validate, &handle_attributes)
ActiveRecord::InsertAll ActiveRecord::InsertAll
.new(self, attributes, on_duplicate: on_duplicate, unique_by: unique_by) .new(self, attributes, on_duplicate: on_duplicate, returning: returning, unique_by: unique_by)
.execute .execute
.pluck(primary_key)
end end
end end
true
end end
def _bulk_insert_item_attributes(items, validate_items) def _bulk_insert_item_attributes(items, validate_items)
......
...@@ -129,10 +129,37 @@ describe BulkInsertSafe do ...@@ -129,10 +129,37 @@ describe BulkInsertSafe do
end.not_to change { described_class.count } end.not_to change { described_class.count }
end end
it 'does nothing and returns true when items are empty' do it 'does nothing and returns an empty array when items are empty' do
expect(described_class.bulk_insert!([])).to be(true) expect(described_class.bulk_insert!([])).to eq([])
expect(described_class.count).to eq(0) expect(described_class.count).to eq(0)
end end
context 'with returns option set' do
context 'when is set to :ids' do
it 'return an array with the primary key values for all inserted records' do
items = described_class.valid_list(1)
expect(described_class.bulk_insert!(items, returns: :ids)).to contain_exactly(a_kind_of(Integer))
end
end
context 'when is set to nil' do
it 'returns an empty array' do
items = described_class.valid_list(1)
expect(described_class.bulk_insert!(items, returns: nil)).to eq([])
end
end
context 'when is set to anything else' do
it 'raises an error' do
items = described_class.valid_list(1)
expect { described_class.bulk_insert!([items], returns: [:id, :name]) }
.to raise_error(ArgumentError, "returns needs to be :ids or nil")
end
end
end
end end
context 'when duplicate items are to be inserted' do context 'when duplicate items are to be inserted' do
......
...@@ -45,11 +45,11 @@ RSpec.shared_examples 'a BulkInsertSafe model' do |klass| ...@@ -45,11 +45,11 @@ RSpec.shared_examples 'a BulkInsertSafe model' do |klass|
expect { target_class.bulk_insert!(items) }.to change { target_class.count }.by(items.size) expect { target_class.bulk_insert!(items) }.to change { target_class.count }.by(items.size)
end end
it 'returns true' do it 'returns an empty array' do
items = valid_items_for_bulk_insertion items = valid_items_for_bulk_insertion
expect(items).not_to be_empty expect(items).not_to be_empty
expect(target_class.bulk_insert!(items)).to be true expect(target_class.bulk_insert!(items)).to eq([])
end end
end end
...@@ -69,7 +69,7 @@ RSpec.shared_examples 'a BulkInsertSafe model' do |klass| ...@@ -69,7 +69,7 @@ RSpec.shared_examples 'a BulkInsertSafe model' do |klass|
# it is not always possible to create invalid items # it is not always possible to create invalid items
if items.any? if items.any?
expect(target_class.bulk_insert!(items, validate: false)).to be(true) expect(target_class.bulk_insert!(items, validate: false)).to eq([])
expect(target_class.count).to eq(items.size) expect(target_class.count).to eq(items.size)
end end
end end
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment