Commit 8032627e authored by Thong Kuah's avatar Thong Kuah

Merge branch '338979-remove-subtransaction-in-internal-id-generator' into 'master'

Removes subtransaction in internal id generator

See merge request gitlab-org/gitlab!68617
parents b7d435bc 44915d43
...@@ -201,30 +201,55 @@ class InternalId < ApplicationRecord ...@@ -201,30 +201,55 @@ class InternalId < ApplicationRecord
InternalId.find_by(**scope, usage: usage_value) InternalId.find_by(**scope, usage: usage_value)
end end
def initial_value(subject, scope)
raise ArgumentError, 'Cannot initialize without init!' unless init
# `init` computes the maximum based on actual records. We use the
# primary to make sure we have up to date results
Gitlab::Database::LoadBalancing::Session.current.use_primary do
instance = subject.is_a?(::Class) ? nil : subject
init.call(instance, scope) || 0
end
end
def usage_value def usage_value
@usage_value ||= InternalId.usages[usage.to_s] @usage_value ||= InternalId.usages[usage.to_s]
end end
# Create InternalId record for (scope, usage) combination, if it doesn't exist # Create InternalId record for (scope, usage) combination, if it doesn't exist
# #
# We blindly insert without synchronization. If another process # We blindly insert ignoring conflicts on the unique key constraint.
# was faster in doing this, we'll realize once we hit the unique key constraint # If another process was faster in doing this, we'll end up with that record
# violation. We can safely roll-back the nested transaction and perform # when we do the lookup after the insert.
# a lookup instead to retrieve the record.
def create_record def create_record
raise ArgumentError, 'Cannot initialize without init!' unless init if Feature.enabled?(:use_insert_all_in_internal_id, default_enabled: :yaml)
scope[:project].save! if scope[:project] && !scope[:project].persisted?
instance = subject.is_a?(::Class) ? nil : subject scope[:namespace].save! if scope[:namespace] && !scope[:namespace].persisted?
subject.transaction(requires_new: true) do attributes = {
InternalId.create!( project_id: scope[:project]&.id || scope[:project_id],
**scope, namespace_id: scope[:namespace]&.id || scope[:namespace_id],
usage: usage_value, usage: usage_value,
last_value: init.call(instance, scope) || 0 last_value: initial_value(subject, scope)
) }
InternalId.insert_all([attributes])
lookup
else
begin
subject.transaction(requires_new: true) do
InternalId.create!(
**scope,
usage: usage_value,
last_value: initial_value(subject, scope)
)
end
rescue ActiveRecord::RecordNotUnique
lookup
end
end end
rescue ActiveRecord::RecordNotUnique
lookup
end end
end end
...@@ -247,6 +272,8 @@ class InternalId < ApplicationRecord ...@@ -247,6 +272,8 @@ class InternalId < ApplicationRecord
# init: Proc that accepts the subject and the scope and returns Integer|NilClass # init: Proc that accepts the subject and the scope and returns Integer|NilClass
attr_reader :subject, :scope, :scope_attrs, :usage, :init attr_reader :subject, :scope, :scope_attrs, :usage, :init
RecordAlreadyExists = Class.new(StandardError)
def initialize(subject, scope, usage, init = nil) def initialize(subject, scope, usage, init = nil)
@subject = subject @subject = subject
@scope = scope @scope = scope
...@@ -270,10 +297,8 @@ class InternalId < ApplicationRecord ...@@ -270,10 +297,8 @@ class InternalId < ApplicationRecord
return next_iid if next_iid return next_iid if next_iid
create_record!(subject, scope, usage, init) do |iid| create_record!(subject, scope, usage, initial_value(subject, scope) + 1)
iid.last_value += 1 rescue RecordAlreadyExists
end
rescue ActiveRecord::RecordNotUnique
retry retry
end end
...@@ -302,10 +327,8 @@ class InternalId < ApplicationRecord ...@@ -302,10 +327,8 @@ class InternalId < ApplicationRecord
next_iid = update_record!(subject, scope, usage, function) next_iid = update_record!(subject, scope, usage, function)
return next_iid if next_iid return next_iid if next_iid
create_record!(subject, scope, usage, init) do |object| create_record!(subject, scope, usage, [initial_value(subject, scope), new_value].max)
object.last_value = [object.last_value, new_value].max rescue RecordAlreadyExists
end
rescue ActiveRecord::RecordNotUnique
retry retry
end end
...@@ -317,27 +340,56 @@ class InternalId < ApplicationRecord ...@@ -317,27 +340,56 @@ class InternalId < ApplicationRecord
stmt.set(arel_table[:last_value] => new_value) stmt.set(arel_table[:last_value] => new_value)
stmt.wheres = InternalId.filter_by(scope, usage).arel.constraints stmt.wheres = InternalId.filter_by(scope, usage).arel.constraints
ActiveRecord::Base.connection.insert(stmt, 'Update InternalId', 'last_value') # rubocop: disable Database/MultipleDatabases InternalId.connection.insert(stmt, 'Update InternalId', 'last_value')
end end
def create_record!(subject, scope, usage, init) def create_record!(subject, scope, usage, value)
raise ArgumentError, 'Cannot initialize without init!' unless init if Feature.enabled?(:use_insert_all_in_internal_id, default_enabled: :yaml)
scope[:project].save! if scope[:project] && !scope[:project].persisted?
scope[:namespace].save! if scope[:namespace] && !scope[:namespace].persisted?
instance = subject.is_a?(::Class) ? nil : subject attributes = {
project_id: scope[:project]&.id || scope[:project_id],
namespace_id: scope[:namespace]&.id || scope[:namespace_id],
usage: usage_value,
last_value: value
}
subject.transaction(requires_new: true) do result = InternalId.insert_all([attributes])
last_value = init.call(instance, scope) || 0
internal_id = InternalId.create!(**scope, usage: usage, last_value: last_value) do |subject| raise RecordAlreadyExists if result.empty?
yield subject if block_given?
end
internal_id.last_value value
else
begin
subject.transaction(requires_new: true) do
internal_id = InternalId.create!(**scope, usage: usage, last_value: value)
internal_id.last_value
end
rescue ActiveRecord::RecordNotUnique
raise RecordAlreadyExists
end
end end
end end
def arel_table def arel_table
InternalId.arel_table InternalId.arel_table
end end
def initial_value(subject, scope)
raise ArgumentError, 'Cannot initialize without init!' unless init
# `init` computes the maximum based on actual records. We use the
# primary to make sure we have up to date results
Gitlab::Database::LoadBalancing::Session.current.use_primary do
instance = subject.is_a?(::Class) ? nil : subject
init.call(instance, scope) || 0
end
end
def usage_value
@usage_value ||= InternalId.usages[usage.to_s]
end
end end
end end
---
name: use_insert_all_in_internal_id
introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/68617
rollout_issue_url:
milestone: '14.3'
type: development
group: group::project management
default_enabled: false
...@@ -238,16 +238,48 @@ RSpec.describe InternalId do ...@@ -238,16 +238,48 @@ RSpec.describe InternalId do
end end
end end
context 'when the feature flag is disabled' do context 'when the explicit locking feature flag is disabled' do
stub_feature_flags(generate_iids_without_explicit_locking: false) before do
stub_feature_flags(generate_iids_without_explicit_locking: false)
end
it_behaves_like 'a monotonically increasing id generator' context 'when the insert all feature flag is enabled' do
before do
stub_feature_flags(use_insert_all_in_internal_id: true)
end
it_behaves_like 'a monotonically increasing id generator'
end
context 'when the insert all feature flag is disabled' do
before do
stub_feature_flags(use_insert_all_in_internal_id: false)
end
it_behaves_like 'a monotonically increasing id generator'
end
end end
context 'when the feature flag is enabled' do context 'when the explicit locking feature flag is enabled' do
stub_feature_flags(generate_iids_without_explicit_locking: true) before do
stub_feature_flags(generate_iids_without_explicit_locking: true)
end
it_behaves_like 'a monotonically increasing id generator' context 'when the insert all feature flag is enabled' do
before do
stub_feature_flags(use_insert_all_in_internal_id: true)
end
it_behaves_like 'a monotonically increasing id generator'
end
context 'when the insert all feature flag is disabled' do
before do
stub_feature_flags(use_insert_all_in_internal_id: false)
end
it_behaves_like 'a monotonically increasing id generator'
end
end end
describe '#increment_and_save!' do describe '#increment_and_save!' 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