From f6aaabeff28e897d7f91aa5aed448e7b82f91160 Mon Sep 17 00:00:00 2001 From: Laura Montemayor <lmontemayor@gitlab.com> Date: Tue, 9 Jul 2024 04:22:59 +0000 Subject: [PATCH] Remove EnsureStageService and prep for removal of `stage` * Add the ignore_column for `stage` * Remove the service from rubocop_todo * Remove `stage` from clone_accessors in job models Changelog: removed --- .../naming/heredoc_delimiter_naming.yml | 1 - .../style/inline_disable_annotation.yml | 1 - app/models/commit_status.rb | 17 +---- app/services/ci/ensure_stage_service.rb | 56 --------------- .../ci_remove_ensure_stage_service.yml | 8 --- spec/factories/ci/processable.rb | 4 +- spec/models/commit_status_spec.rb | 70 ------------------- spec/services/ci/ensure_stage_service_spec.rb | 57 --------------- spec/support/rspec_order_todo.yml | 1 - 9 files changed, 3 insertions(+), 212 deletions(-) delete mode 100644 app/services/ci/ensure_stage_service.rb delete mode 100644 config/feature_flags/development/ci_remove_ensure_stage_service.yml delete mode 100644 spec/services/ci/ensure_stage_service_spec.rb diff --git a/.rubocop_todo/naming/heredoc_delimiter_naming.yml b/.rubocop_todo/naming/heredoc_delimiter_naming.yml index 09e03a992dd0..cb457fd73794 100644 --- a/.rubocop_todo/naming/heredoc_delimiter_naming.yml +++ b/.rubocop_todo/naming/heredoc_delimiter_naming.yml @@ -4,7 +4,6 @@ Naming/HeredocDelimiterNaming: - 'app/models/ci/build_trace_chunks/redis_base.rb' - 'app/models/concerns/legacy_bulk_insert.rb' - 'app/models/trending_project.rb' - - 'app/services/ci/ensure_stage_service.rb' - 'app/services/packages/debian/generate_distribution_key_service.rb' - 'app/workers/concerns/limited_capacity/job_tracker.rb' - 'config/initializers/01_secret_token.rb' diff --git a/.rubocop_todo/style/inline_disable_annotation.yml b/.rubocop_todo/style/inline_disable_annotation.yml index 99822bee577f..5044797c8363 100644 --- a/.rubocop_todo/style/inline_disable_annotation.yml +++ b/.rubocop_todo/style/inline_disable_annotation.yml @@ -578,7 +578,6 @@ Style/InlineDisableAnnotation: - 'app/services/ci/create_pipeline_service.rb' - 'app/services/ci/delete_objects_service.rb' - 'app/services/ci/delete_unit_tests_service.rb' - - 'app/services/ci/ensure_stage_service.rb' - 'app/services/ci/expire_pipeline_cache_service.rb' - 'app/services/ci/job_artifacts/bulk_delete_by_project_service.rb' - 'app/services/ci/job_artifacts/destroy_batch_service.rb' diff --git a/app/models/commit_status.rb b/app/models/commit_status.rb index 2b4feb80435c..6bfdc906bb8a 100644 --- a/app/models/commit_status.rb +++ b/app/models/commit_status.rb @@ -51,7 +51,7 @@ class CommitStatus < Ci::ApplicationRecord validates :pipeline, presence: true, unless: :importing? validates :name, presence: true, unless: :importing? - validates :ci_stage, presence: true, on: :create, unless: :importing?, if: -> { Feature.enabled?(:ci_remove_ensure_stage_service, project) } + validates :ci_stage, presence: true, on: :create, unless: :importing? validates :ref, :target_url, :description, length: { maximum: 255 } alias_attribute :author, :user @@ -120,21 +120,6 @@ class CommitStatus < Ci::ApplicationRecord merge(or_conditions) end - ## - # We still create some CommitStatuses outside of CreatePipelineService. - # - # These are pages deployments and external statuses. - # - before_create unless: :importing? do - next if Feature.enabled?(:ci_remove_ensure_stage_service, project) - - # rubocop: disable CodeReuse/ServiceClass - Ci::EnsureStageService.new(project, user).execute(self) do |stage| - self.run_after_commit { StageUpdateWorker.perform_async(stage.id) } - end - # rubocop: enable CodeReuse/ServiceClass - end - before_save if: :status_changed?, unless: :importing? do # we mark `processed` as always changed: # another process might change its value and our object diff --git a/app/services/ci/ensure_stage_service.rb b/app/services/ci/ensure_stage_service.rb deleted file mode 100644 index 9d5ccecbe338..000000000000 --- a/app/services/ci/ensure_stage_service.rb +++ /dev/null @@ -1,56 +0,0 @@ -# frozen_string_literal: true - -module Ci - ## - # We call this service everytime we persist a CI/CD job. - # - # In most cases a job should already have a stage assigned, but in cases it - # doesn't have we need to either find existing one or create a brand new - # stage. - # - class EnsureStageService < BaseService - EnsureStageError = Class.new(StandardError) - - def execute(build) - @build = build - - return if build.stage_id.present? - return if build.invalid? - - ensure_stage.tap do |stage| - build.stage_id = stage.id - - yield stage if block_given? - end - end - - private - - def ensure_stage(attempts: 2) - find_stage || create_stage - rescue ActiveRecord::RecordNotUnique - retry if (attempts -= 1) > 0 - - raise EnsureStageError, <<~EOS - We failed to find or create a unique pipeline stage after 2 retries. - This should never happen and is most likely the result of a bug in - the database load balancing code. - EOS - end - - # rubocop: disable CodeReuse/ActiveRecord - def find_stage - @build.pipeline.stages.find_by(name: @build.stage) - end - # rubocop: enable CodeReuse/ActiveRecord - - def create_stage - Ci::Stage.create!( - name: @build.stage, - position: @build.stage_idx, - pipeline: @build.pipeline, - project: @build.project - ) - end - end -end diff --git a/config/feature_flags/development/ci_remove_ensure_stage_service.yml b/config/feature_flags/development/ci_remove_ensure_stage_service.yml deleted file mode 100644 index a742891d3475..000000000000 --- a/config/feature_flags/development/ci_remove_ensure_stage_service.yml +++ /dev/null @@ -1,8 +0,0 @@ ---- -name: ci_remove_ensure_stage_service -introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/107389 -rollout_issue_url: https://gitlab.com/gitlab-org/gitlab/-/issues/389076 -milestone: '15.9' -type: development -group: group::pipeline authoring -default_enabled: true diff --git a/spec/factories/ci/processable.rb b/spec/factories/ci/processable.rb index cfe736bce500..221abcadc459 100644 --- a/spec/factories/ci/processable.rb +++ b/spec/factories/ci/processable.rb @@ -25,7 +25,7 @@ end after(:build) do |processable, evaluator| - next if processable.ci_stage || Feature.disabled?(:ci_remove_ensure_stage_service, processable.project) + next if processable.ci_stage processable.stage = evaluator.stage pipeline = processable.pipeline @@ -57,7 +57,7 @@ end before(:create) do |processable, evaluator| - next if processable.ci_stage || Feature.disabled?(:ci_remove_ensure_stage_service, processable.project) + next if processable.ci_stage processable.ci_stage = if ci_stage = processable.pipeline.stages.find_by(name: evaluator.stage) diff --git a/spec/models/commit_status_spec.rb b/spec/models/commit_status_spec.rb index 52f3aaae3e75..c3cba6a45d33 100644 --- a/spec/models/commit_status_spec.rb +++ b/spec/models/commit_status_spec.rb @@ -832,76 +832,6 @@ def create_status(**opts) end end - describe 'ensure stage assignment' do - before do - stub_feature_flags(ci_remove_ensure_stage_service: false) - end - - context 'when the feature flag ci_remove_ensure_stage_service is disabled' do - context 'when commit status has a stage_id assigned' do - let!(:stage) do - create(:ci_stage, project: project, pipeline: pipeline) - end - - let(:commit_status) do - create(:commit_status, stage_id: stage.id, name: 'rspec', stage: 'test') - end - - it 'does not create a new stage' do - expect { commit_status }.not_to change { Ci::Stage.count } - expect(commit_status.stage_id).to eq stage.id - end - end - - context 'when commit status does not have a stage_id assigned' do - let(:commit_status) do - FactoryBot.build(:commit_status, name: 'rspec', stage: 'test', status: :success) - end - - let(:stage) { Ci::Stage.first } - - it 'creates a new stage', :sidekiq_might_not_need_inline do - expect { commit_status.save! }.to change { Ci::Stage.count }.by(1) - - expect(stage.name).to eq 'test' - expect(stage.project).to eq commit_status.project - expect(stage.pipeline).to eq commit_status.pipeline - expect(stage.status).to eq commit_status.status - expect(commit_status.stage_id).to eq stage.id - end - end - - context 'when commit status does not have stage but it exists' do - let!(:stage) do - create(:ci_stage, project: project, pipeline: pipeline, name: 'test') - end - - let(:commit_status) do - FactoryBot.build(:commit_status, project: project, pipeline: pipeline, name: 'rspec', stage: 'test', - status: :success) - end - - it 'uses existing stage', :sidekiq_might_not_need_inline do - expect { commit_status.save! }.not_to change { Ci::Stage.count } - - expect(commit_status.stage_id).to eq stage.id - expect(stage.reload.status).to eq commit_status.status - end - end - - context 'when commit status is being imported' do - let(:commit_status) do - FactoryBot.build(:commit_status, name: 'rspec', stage: 'test', importing: true) - end - - it 'does not create a new stage' do - expect { commit_status.save! }.not_to change { Ci::Stage.count } - expect(commit_status.stage_id).not_to be_present - end - end - end - end - describe '#all_met_to_become_pending?' do subject { commit_status.all_met_to_become_pending? } diff --git a/spec/services/ci/ensure_stage_service_spec.rb b/spec/services/ci/ensure_stage_service_spec.rb deleted file mode 100644 index 8debc5eab73b..000000000000 --- a/spec/services/ci/ensure_stage_service_spec.rb +++ /dev/null @@ -1,57 +0,0 @@ -# frozen_string_literal: true - -require 'spec_helper' - -RSpec.describe Ci::EnsureStageService, '#execute', feature_category: :continuous_integration do - let_it_be(:project) { create(:project) } - let_it_be(:user) { create(:user) } - - let(:stage) { create(:ci_stage) } - let(:job) { build(:ci_build) } - - let(:service) { described_class.new(project, user) } - - before do - stub_feature_flags(ci_remove_ensure_stage_service: false) - end - - context 'when build has a stage assigned' do - it 'does not create a new stage' do - job.assign_attributes(stage_id: stage.id) - - expect { service.execute(job) }.not_to change { Ci::Stage.count } - end - end - - context 'when build does not have a stage assigned' do - it 'creates a new stage' do - job.assign_attributes(stage_id: nil, stage: 'test') - - expect { service.execute(job) }.to change { Ci::Stage.count }.by(1) - end - end - - context 'when build is invalid' do - it 'does not create a new stage' do - job.assign_attributes(stage_id: nil, ref: nil) - - expect { service.execute(job) }.not_to change { Ci::Stage.count } - end - end - - context 'when new stage can not be created because of an exception' do - before do - allow(Ci::Stage).to receive(:create!) - .and_raise(ActiveRecord::RecordNotUnique.new('Duplicates!')) - end - - it 'retries up to two times' do - job.assign_attributes(stage_id: nil) - - expect(service).to receive(:find_stage).twice - - expect { service.execute(job) } - .to raise_error(Ci::EnsureStageService::EnsureStageError) - end - end -end diff --git a/spec/support/rspec_order_todo.yml b/spec/support/rspec_order_todo.yml index c093558f1c52..2556b0b57721 100644 --- a/spec/support/rspec_order_todo.yml +++ b/spec/support/rspec_order_todo.yml @@ -8153,7 +8153,6 @@ - './spec/services/ci/destroy_pipeline_service_spec.rb' - './spec/services/ci/destroy_secure_file_service_spec.rb' - './spec/services/ci/drop_pipeline_service_spec.rb' -- './spec/services/ci/ensure_stage_service_spec.rb' - './spec/services/ci/expire_pipeline_cache_service_spec.rb' - './spec/services/ci/external_pull_requests/create_pipeline_service_spec.rb' - './spec/services/ci/find_exposed_artifacts_service_spec.rb' -- GitLab