Commit d53ea02e authored by Andreas Brandl's avatar Andreas Brandl

Trigger iid logic from GitHub importer for milestones.

parent cc74fb53
...@@ -111,7 +111,7 @@ class InternalId < ActiveRecord::Base ...@@ -111,7 +111,7 @@ class InternalId < ActiveRecord::Base
# Generates next internal id and returns it # Generates next internal id and returns it
def generate def generate
subject.transaction do InternalId.transaction do
# Create a record in internal_ids if one does not yet exist # Create a record in internal_ids if one does not yet exist
# and increment its last value # and increment its last value
# #
...@@ -125,7 +125,7 @@ class InternalId < ActiveRecord::Base ...@@ -125,7 +125,7 @@ class InternalId < ActiveRecord::Base
# #
# Note this will acquire a ROW SHARE lock on the InternalId record # Note this will acquire a ROW SHARE lock on the InternalId record
def track_greatest(new_value) def track_greatest(new_value)
subject.transaction do InternalId.transaction do
(lookup || create_record).track_greatest_and_save!(new_value) (lookup || create_record).track_greatest_and_save!(new_value)
end end
end end
...@@ -148,7 +148,7 @@ class InternalId < ActiveRecord::Base ...@@ -148,7 +148,7 @@ class InternalId < ActiveRecord::Base
# violation. We can safely roll-back the nested transaction and perform # violation. We can safely roll-back the nested transaction and perform
# a lookup instead to retrieve the record. # a lookup instead to retrieve the record.
def create_record def create_record
subject.transaction(requires_new: true) do InternalId.transaction(requires_new: true) do
InternalId.create!( InternalId.create!(
**scope, **scope,
usage: usage_value, usage: usage_value,
......
...@@ -15,10 +15,12 @@ module Gitlab ...@@ -15,10 +15,12 @@ module Gitlab
end end
# Bulk inserts the given rows into the database. # Bulk inserts the given rows into the database.
def bulk_insert(model, rows, batch_size: 100) def bulk_insert(model, rows, batch_size: 100, pre_hook: nil)
rows.each_slice(batch_size) do |slice| rows.each_slice(batch_size) do |slice|
pre_hook.call(slice) if pre_hook
Gitlab::Database.bulk_insert(model.table_name, slice) Gitlab::Database.bulk_insert(model.table_name, slice)
end end
rows
end end
end end
end end
......
...@@ -17,10 +17,20 @@ module Gitlab ...@@ -17,10 +17,20 @@ module Gitlab
end end
def execute def execute
bulk_insert(Milestone, build_milestones) # We insert records in bulk, by-passing any standard model callbacks.
# The pre_hook here makes sure we track internal ids consistently.
# Note this has to be called before performing an insert of a batch
# because we're outside a transaction scope here.
bulk_insert(Milestone, build_milestones, pre_hook: method(:track_greatest_iid))
build_milestones_cache build_milestones_cache
end end
def track_greatest_iid(slice)
greatest_iid = slice.max { |e| e[:iid] }[:iid]
InternalId.track_greatest(nil, { project: project }, :milestones, greatest_iid, ->(_) { project.milestones.maximum(:iid) })
end
def build_milestones def build_milestones
build_database_rows(each_milestone) build_database_rows(each_milestone)
end end
......
...@@ -58,5 +58,17 @@ describe Gitlab::GithubImport::BulkImporting do ...@@ -58,5 +58,17 @@ describe Gitlab::GithubImport::BulkImporting do
importer.bulk_insert(model, rows, batch_size: 5) importer.bulk_insert(model, rows, batch_size: 5)
end end
it 'calls pre_hook for each slice if given' do
rows = [{ title: 'Foo' }] * 10
model = double(:model, table_name: 'kittens')
pre_hook = double('pre_hook', call: nil)
allow(Gitlab::Database).to receive(:bulk_insert)
expect(pre_hook).to receive(:call).with(rows[0..4])
expect(pre_hook).to receive(:call).with(rows[5..9])
importer.bulk_insert(model, rows, batch_size: 5, pre_hook: pre_hook)
end
end end
end end
...@@ -29,13 +29,25 @@ describe Gitlab::GithubImport::Importer::MilestonesImporter, :clean_gitlab_redis ...@@ -29,13 +29,25 @@ describe Gitlab::GithubImport::Importer::MilestonesImporter, :clean_gitlab_redis
expect(importer) expect(importer)
.to receive(:bulk_insert) .to receive(:bulk_insert)
.with(Milestone, [milestone_hash]) .with(Milestone, [milestone_hash], any_args)
expect(importer) expect(importer)
.to receive(:build_milestones_cache) .to receive(:build_milestones_cache)
importer.execute importer.execute
end end
it 'tracks internal ids' do
milestone_hash = { iid: 1, title: '1.0', project_id: project.id }
allow(importer)
.to receive(:build_milestones)
.and_return([milestone_hash])
expect(InternalId).to receive(:track_greatest)
.with(nil, { project: project }, :milestones, 1, any_args)
importer.execute
end
end end
describe '#build_milestones' do describe '#build_milestones' do
......
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