77 lines
2.4 KiB
Ruby
77 lines
2.4 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module Ci
|
|
class RetryBuildService < ::BaseService
|
|
include Gitlab::OptimisticLocking
|
|
|
|
def self.clone_accessors
|
|
%i[pipeline project ref tag options name
|
|
allow_failure stage stage_id stage_idx trigger_request
|
|
yaml_variables when environment coverage_regex
|
|
description tag_list protected needs_attributes
|
|
resource_group scheduling_type].freeze
|
|
end
|
|
|
|
def execute(build)
|
|
build.ensure_scheduling_type!
|
|
|
|
reprocess!(build).tap do |new_build|
|
|
mark_subsequent_stages_as_processable(build)
|
|
build.pipeline.reset_ancestor_bridges!
|
|
|
|
Gitlab::OptimisticLocking.retry_lock(new_build, name: 'retry_build', &:enqueue)
|
|
|
|
MergeRequests::AddTodoWhenBuildFailsService
|
|
.new(project, current_user)
|
|
.close(new_build)
|
|
end
|
|
end
|
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
|
def reprocess!(build)
|
|
unless can?(current_user, :update_build, build)
|
|
raise Gitlab::Access::AccessDeniedError
|
|
end
|
|
|
|
attributes = self.class.clone_accessors.map do |attribute|
|
|
[attribute, build.public_send(attribute)] # rubocop:disable GitlabSecurity/PublicSend
|
|
end.to_h
|
|
|
|
attributes[:user] = current_user
|
|
|
|
Ci::Build.transaction do
|
|
# mark all other builds of that name as retried
|
|
build.pipeline.builds.latest
|
|
.where(name: build.name)
|
|
.update_all(retried: true, processed: true)
|
|
|
|
create_build!(attributes).tap do
|
|
# mark existing object as retried/processed without a reload
|
|
build.retried = true
|
|
build.processed = true
|
|
end
|
|
end
|
|
end
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
|
|
|
private
|
|
|
|
def create_build!(attributes)
|
|
build = project.builds.new(attributes)
|
|
build.assign_attributes(::Gitlab::Ci::Pipeline::Seed::Build.environment_attributes_for(build))
|
|
build.retried = false
|
|
BulkInsertableAssociations.with_bulk_insert do
|
|
build.save!
|
|
end
|
|
build
|
|
end
|
|
|
|
def mark_subsequent_stages_as_processable(build)
|
|
build.pipeline.processables.skipped.after_stage(build.stage_idx).find_each do |skipped|
|
|
retry_optimistic_lock(skipped, name: 'ci_retry_build_mark_subsequent_stages') { |build| build.process(current_user) }
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
Ci::RetryBuildService.prepend_if_ee('EE::Ci::RetryBuildService')
|