Commit 7749eabc authored by Kamil Trzciński's avatar Kamil Trzciński Committed by Aleksei Lipniagov

Specify if `iid` should be generated during Import

Refactor `InternalId` generation.
Remove a hack to handle ci_pipelines.iid for imports.
parent 56987a07
...@@ -25,7 +25,7 @@ module Ci ...@@ -25,7 +25,7 @@ module Ci
belongs_to :merge_request, class_name: 'MergeRequest' belongs_to :merge_request, class_name: 'MergeRequest'
belongs_to :external_pull_request belongs_to :external_pull_request
has_internal_id :iid, scope: :project, presence: false, init: ->(s) do has_internal_id :iid, scope: :project, presence: false, ensure_if: -> { !importing? }, init: ->(s) do
s&.project&.all_pipelines&.maximum(:iid) || s&.project&.all_pipelines&.count s&.project&.all_pipelines&.maximum(:iid) || s&.project&.all_pipelines&.count
end end
......
...@@ -27,53 +27,74 @@ module AtomicInternalId ...@@ -27,53 +27,74 @@ module AtomicInternalId
extend ActiveSupport::Concern extend ActiveSupport::Concern
class_methods do class_methods do
def has_internal_id(column, scope:, init:, presence: true) # rubocop:disable Naming/PredicateName def has_internal_id(column, scope:, init:, ensure_if: nil, presence: true) # rubocop:disable Naming/PredicateName
# We require init here to retain the ability to recalculate in the absence of a # We require init here to retain the ability to recalculate in the absence of a
# InternaLId record (we may delete records in `internal_ids` for example). # InternaLId record (we may delete records in `internal_ids` for example).
raise "has_internal_id requires a init block, none given." unless init raise "has_internal_id requires a init block, none given." unless init
raise "has_internal_id needs to be defined on association." unless self.reflect_on_association(scope)
before_validation :"ensure_#{scope}_#{column}!", on: :create before_validation :"track_#{scope}_#{column}!", on: :create
before_validation :"ensure_#{scope}_#{column}!", on: :create, if: ensure_if
validates column, presence: presence validates column, presence: presence
define_method("ensure_#{scope}_#{column}!") do define_method("ensure_#{scope}_#{column}!") do
scope_value = association(scope).reader scope_value = internal_id_read_scope(scope)
value = read_attribute(column) value = read_attribute(column)
return value unless scope_value return value unless scope_value
scope_attrs = { scope_value.class.table_name.singularize.to_sym => scope_value } if value.nil?
usage = self.class.table_name.to_sym
if value.present? && (@iid_needs_tracking || Feature.enabled?(:iid_always_track, default_enabled: true))
# The value was set externally, e.g. by the user
# We update the InternalId record to keep track of the greatest value.
InternalId.track_greatest(self, scope_attrs, usage, value, init)
@iid_needs_tracking = false
elsif !value.present?
# We don't have a value yet and use a InternalId record to generate # We don't have a value yet and use a InternalId record to generate
# the next value. # the next value.
value = InternalId.generate_next(self, scope_attrs, usage, init) value = InternalId.generate_next(
self,
internal_id_scope_attrs(scope),
internal_id_scope_usage,
init)
write_attribute(column, value) write_attribute(column, value)
end end
value value
end end
define_method("track_#{scope}_#{column}!") do
if !@internal_id_needs_tracking
return unless Feature.enabled?(:iid_always_track, default_enabled: true)
end
@internal_id_needs_tracking = false
scope_value = internal_id_read_scope(scope)
value = read_attribute(column)
return unless scope_value
if value.present?
# The value was set externally, e.g. by the user
# We update the InternalId record to keep track of the greatest value.
InternalId.track_greatest(
self,
internal_id_scope_attrs(scope),
internal_id_scope_usage,
value,
init)
end
end
define_method("#{column}=") do |value| define_method("#{column}=") do |value|
super(value).tap do |v| super(value).tap do |v|
# Indicate the iid was set from externally # Indicate the iid was set from externally
@iid_needs_tracking = true @internal_id_needs_tracking = true
end end
end end
define_method("reset_#{scope}_#{column}") do define_method("reset_#{scope}_#{column}") do
if value = read_attribute(column) if value = read_attribute(column)
scope_value = association(scope).reader did_reset = InternalId.reset(
scope_attrs = { scope_value.class.table_name.singularize.to_sym => scope_value } self,
usage = self.class.table_name.to_sym internal_id_scope_attrs(scope),
internal_id_scope_usage,
value)
if InternalId.reset(self, scope_attrs, usage, value) if did_reset
write_attribute(column, nil) write_attribute(column, nil)
end end
end end
...@@ -82,4 +103,18 @@ module AtomicInternalId ...@@ -82,4 +103,18 @@ module AtomicInternalId
end end
end end
end end
def internal_id_scope_attrs(scope)
scope_value = internal_id_read_scope(scope)
{ scope_value.class.table_name.singularize.to_sym => scope_value } if scope_value
end
def internal_id_scope_usage
self.class.table_name.to_sym
end
def internal_id_read_scope(scope)
association(scope).reader
end
end end
...@@ -184,18 +184,9 @@ module Gitlab ...@@ -184,18 +184,9 @@ module Gitlab
return if tree_hash[relation_key].blank? return if tree_hash[relation_key].blank?
tree_array = [tree_hash[relation_key]].flatten tree_array = [tree_hash[relation_key]].flatten
null_iid_pipelines = []
# Avoid keeping a possible heavy object in memory once we are done with it # Avoid keeping a possible heavy object in memory once we are done with it
while relation_item = (tree_array.shift || null_iid_pipelines.shift) while relation_item = tree_array.shift
if nil_iid_pipeline?(relation_key, relation_item) && tree_array.any?
# Move pipelines with NULL IIDs to the end
# so they don't clash with existing IIDs.
null_iid_pipelines << relation_item
next
end
remove_feature_dependent_sub_relations(relation_item) remove_feature_dependent_sub_relations(relation_item)
# The transaction at this level is less speedy than one single transaction # The transaction at this level is less speedy than one single transaction
...@@ -259,10 +250,6 @@ module Gitlab ...@@ -259,10 +250,6 @@ module Gitlab
def excluded_keys_for_relation(relation) def excluded_keys_for_relation(relation)
reader.attributes_finder.find_excluded_keys(relation) reader.attributes_finder.find_excluded_keys(relation)
end end
def nil_iid_pipeline?(relation_key, relation_item)
relation_key == 'ci_pipelines' && relation_item['iid'].nil?
end
end end
end end
end end
......
...@@ -9,16 +9,10 @@ describe AtomicInternalId do ...@@ -9,16 +9,10 @@ describe AtomicInternalId do
let(:scope_attrs) { { project: milestone.project } } let(:scope_attrs) { { project: milestone.project } }
let(:usage) { :milestones } let(:usage) { :milestones }
describe '#ensure_project_iid!' do describe '#track_project_iid!' do
subject { milestone.ensure_project_iid! } subject { milestone.track_project_iid! }
it 'generates a new value if non is present' do
expect(InternalId).to receive(:generate_next).with(milestone, scope_attrs, usage, anything).and_return(iid)
expect { subject }.to change { milestone.iid }.from(nil).to(iid.to_i) it 'tracks the present value' do
end
it 'tracks the present value if not generated by InternalId.generate_next' do
milestone.iid = external_iid milestone.iid = external_iid
expect(InternalId).to receive(:track_greatest).once.with(milestone, scope_attrs, usage, external_iid, anything) expect(InternalId).to receive(:track_greatest).once.with(milestone, scope_attrs, usage, external_iid, anything)
...@@ -27,27 +21,51 @@ describe AtomicInternalId do ...@@ -27,27 +21,51 @@ describe AtomicInternalId do
subject subject
end end
it 'generates a new value if first set with iid= but later set to nil' do context 'when value is set by ensure_project_iid!' do
expect(InternalId).to receive(:generate_next).with(milestone, scope_attrs, usage, anything).and_return(iid) context 'with iid_always_track true' do
before do
milestone.iid = external_iid stub_feature_flags(iid_always_track: false)
milestone.iid = nil end
expect { subject }.to change { milestone.iid }.from(nil).to(iid.to_i) it 'does not track the value' do
end expect(InternalId).not_to receive(:track_greatest)
context 'with iid_always_track disabled' do milestone.ensure_project_iid!
before do subject
stub_feature_flags(iid_always_track: false) end
end end
it 'does not track the present value if generated by InternalId.generate_next' do context 'with iid_always_track enabled' do
milestone.ensure_project_iid! before do
stub_feature_flags(iid_always_track: true)
end
expect(InternalId).not_to receive(:track_greatest) it 'does not track the value' do
expect(InternalId).to receive(:track_greatest)
subject milestone.ensure_project_iid!
subject
end
end end
end end
end end
describe '#ensure_project_iid!' do
subject { milestone.ensure_project_iid! }
it 'generates a new value if non is present' do
expect(InternalId).to receive(:generate_next).with(milestone, scope_attrs, usage, anything).and_return(iid)
expect { subject }.to change { milestone.iid }.from(nil).to(iid.to_i)
end
it 'generates a new value if first set with iid= but later set to nil' do
expect(InternalId).to receive(:generate_next).with(milestone, scope_attrs, usage, anything).and_return(iid)
milestone.iid = external_iid
milestone.iid = nil
expect { subject }.to change { milestone.iid }.from(nil).to(iid.to_i)
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