2022-10-11 01:57:18 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
module Ci
|
|
|
|
##
|
|
|
|
# This module implements a way to set the `partion_id` value on a dependent
|
|
|
|
# resource from a parent record.
|
|
|
|
# Usage:
|
|
|
|
#
|
|
|
|
# class PipelineVariable < Ci::ApplicationRecord
|
|
|
|
# include Ci::Partitionable
|
|
|
|
#
|
|
|
|
# belongs_to :pipeline
|
|
|
|
# partitionable scope: :pipeline
|
|
|
|
# # Or
|
|
|
|
# partitionable scope: ->(record) { record.partition_value }
|
|
|
|
#
|
|
|
|
#
|
|
|
|
module Partitionable
|
|
|
|
extend ActiveSupport::Concern
|
|
|
|
include ::Gitlab::Utils::StrongMemoize
|
|
|
|
|
2022-11-25 23:54:43 +05:30
|
|
|
module Testing
|
|
|
|
InclusionError = Class.new(StandardError)
|
|
|
|
|
|
|
|
PARTITIONABLE_MODELS = %w[
|
|
|
|
CommitStatus
|
|
|
|
Ci::BuildMetadata
|
|
|
|
Ci::Stage
|
|
|
|
Ci::JobArtifact
|
|
|
|
Ci::PipelineVariable
|
|
|
|
Ci::Pipeline
|
|
|
|
].freeze
|
|
|
|
|
|
|
|
def self.check_inclusion(klass)
|
|
|
|
return if PARTITIONABLE_MODELS.include?(klass.name)
|
|
|
|
|
|
|
|
raise Partitionable::Testing::InclusionError,
|
|
|
|
"#{klass} must be included in PARTITIONABLE_MODELS"
|
|
|
|
|
|
|
|
rescue InclusionError => e
|
|
|
|
Gitlab::ErrorTracking.track_and_raise_for_dev_exception(e)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-10-11 01:57:18 +05:30
|
|
|
included do
|
2022-11-25 23:54:43 +05:30
|
|
|
Partitionable::Testing.check_inclusion(self)
|
|
|
|
|
2022-10-11 01:57:18 +05:30
|
|
|
before_validation :set_partition_id, on: :create
|
|
|
|
validates :partition_id, presence: true
|
|
|
|
|
|
|
|
def set_partition_id
|
|
|
|
return if partition_id_changed? && partition_id.present?
|
|
|
|
return unless partition_scope_value
|
|
|
|
|
|
|
|
self.partition_id = partition_scope_value
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
class_methods do
|
|
|
|
private
|
|
|
|
|
|
|
|
def partitionable(scope:)
|
|
|
|
define_method(:partition_scope_value) do
|
|
|
|
strong_memoize(:partition_scope_value) do
|
2022-11-25 23:54:43 +05:30
|
|
|
next Ci::Pipeline.current_partition_value if respond_to?(:importing?) && importing?
|
|
|
|
|
2022-10-11 01:57:18 +05:30
|
|
|
record = scope.to_proc.call(self)
|
|
|
|
record.respond_to?(:partition_id) ? record.partition_id : record
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|