Commit 3dcbfee2 authored by Bob Van Landuyt's avatar Bob Van Landuyt

Merge branch '263110-improve-cleanup-policies-selection-during-their-execution' into 'master'

Improve cleanup policies selection during their execution

See merge request gitlab-org/gitlab!44748
parents 284ad184 50e767df
...@@ -3,6 +3,7 @@ ...@@ -3,6 +3,7 @@
class ContainerExpirationPolicy < ApplicationRecord class ContainerExpirationPolicy < ApplicationRecord
include Schedulable include Schedulable
include UsageStatistics include UsageStatistics
include EachBatch
belongs_to :project, inverse_of: :container_expiration_policy belongs_to :project, inverse_of: :container_expiration_policy
...@@ -19,6 +20,16 @@ class ContainerExpirationPolicy < ApplicationRecord ...@@ -19,6 +20,16 @@ class ContainerExpirationPolicy < ApplicationRecord
scope :active, -> { where(enabled: true) } scope :active, -> { where(enabled: true) }
scope :preloaded, -> { preload(project: [:route]) } scope :preloaded, -> { preload(project: [:route]) }
def self.executable
runnable_schedules.where(
'EXISTS (?)',
ContainerRepository.select(1)
.where(
'container_repositories.project_id = container_expiration_policies.project_id'
)
)
end
def self.keep_n_options def self.keep_n_options
{ {
1 => _('%{tags} tag per image name') % { tags: 1 }, 1 => _('%{tags} tag per image name') % { tags: 1 },
......
...@@ -7,13 +7,15 @@ class ContainerExpirationPolicyWorker # rubocop:disable Scalability/IdempotentWo ...@@ -7,13 +7,15 @@ class ContainerExpirationPolicyWorker # rubocop:disable Scalability/IdempotentWo
feature_category :container_registry feature_category :container_registry
def perform def perform
ContainerExpirationPolicy.runnable_schedules.preloaded.find_each do |container_expiration_policy| ContainerExpirationPolicy.executable.preloaded.each_batch do |relation|
with_context(project: container_expiration_policy.project, relation.each do |container_expiration_policy|
user: container_expiration_policy.project.owner) do |project:, user:| with_context(project: container_expiration_policy.project,
ContainerExpirationPolicyService.new(project, user) user: container_expiration_policy.project.owner) do |project:, user:|
.execute(container_expiration_policy) ContainerExpirationPolicyService.new(project, user)
rescue ContainerExpirationPolicyService::InvalidPolicyError => e .execute(container_expiration_policy)
Gitlab::ErrorTracking.log_exception(e, container_expiration_policy_id: container_expiration_policy.id) rescue ContainerExpirationPolicyService::InvalidPolicyError => e
Gitlab::ErrorTracking.log_exception(e, container_expiration_policy_id: container_expiration_policy.id)
end
end end
end end
end end
......
---
title: Exclude policies with no container repositories when executing them
merge_request: 44748
author:
type: fixed
# frozen_string_literal: true
class AddIndexWithProjectIdToContainerExpirationPolicies < ActiveRecord::Migration[6.0]
include Gitlab::Database::MigrationHelpers
DOWNTIME = false
INDEX_NAME = 'idx_container_exp_policies_on_project_id_next_run_at_enabled'
disable_ddl_transaction!
def up
add_concurrent_index :container_expiration_policies, [:project_id, :next_run_at, :enabled], name: INDEX_NAME
end
def down
remove_concurrent_index_by_name :container_expiration_policies, INDEX_NAME
end
end
d0944a864a1a89e9339eb1f8ffab683df1a5bb90f7b7a16cabd4871f34d1cd48
\ No newline at end of file
...@@ -19622,6 +19622,8 @@ CREATE INDEX idx_audit_events_on_entity_id_desc_author_id_created_at ON audit_ev ...@@ -19622,6 +19622,8 @@ CREATE INDEX idx_audit_events_on_entity_id_desc_author_id_created_at ON audit_ev
CREATE INDEX idx_ci_pipelines_artifacts_locked ON ci_pipelines USING btree (ci_ref_id, id) WHERE (locked = 1); CREATE INDEX idx_ci_pipelines_artifacts_locked ON ci_pipelines USING btree (ci_ref_id, id) WHERE (locked = 1);
CREATE INDEX idx_container_exp_policies_on_project_id_next_run_at_enabled ON container_expiration_policies USING btree (project_id, next_run_at, enabled);
CREATE INDEX idx_deployment_clusters_on_cluster_id_and_kubernetes_namespace ON deployment_clusters USING btree (cluster_id, kubernetes_namespace); CREATE INDEX idx_deployment_clusters_on_cluster_id_and_kubernetes_namespace ON deployment_clusters USING btree (cluster_id, kubernetes_namespace);
CREATE UNIQUE INDEX idx_deployment_merge_requests_unique_index ON deployment_merge_requests USING btree (deployment_id, merge_request_id); CREATE UNIQUE INDEX idx_deployment_merge_requests_unique_index ON deployment_merge_requests USING btree (deployment_id, merge_request_id);
......
...@@ -104,6 +104,18 @@ RSpec.describe ContainerExpirationPolicy, type: :model do ...@@ -104,6 +104,18 @@ RSpec.describe ContainerExpirationPolicy, type: :model do
end end
end end
describe '.executable' do
subject { described_class.executable }
let_it_be(:policy1) { create(:container_expiration_policy, :runnable) }
let_it_be(:container_repository1) { create(:container_repository, project: policy1.project) }
let_it_be(:policy2) { create(:container_expiration_policy, :runnable) }
let_it_be(:container_repository2) { create(:container_repository, project: policy2.project) }
let_it_be(:policy3) { create(:container_expiration_policy, :runnable) }
it { is_expected.to contain_exactly(policy1, policy2) }
end
describe '#disable!' do describe '#disable!' do
let_it_be(:container_expiration_policy) { create(:container_expiration_policy) } let_it_be(:container_expiration_policy) { create(:container_expiration_policy) }
......
...@@ -7,19 +7,24 @@ RSpec.describe ContainerExpirationPolicyWorker do ...@@ -7,19 +7,24 @@ RSpec.describe ContainerExpirationPolicyWorker do
subject { described_class.new.perform } subject { described_class.new.perform }
context 'With no container expiration policies' do RSpec.shared_examples 'not executing any policy' do
it 'Does not execute any policies' do it 'does not run any policy' do
expect(ContainerExpirationPolicyService).not_to receive(:new) expect(ContainerExpirationPolicyService).not_to receive(:new)
subject subject
end end
end end
context 'With no container expiration policies' do
it_behaves_like 'not executing any policy'
end
context 'With container expiration policies' do context 'With container expiration policies' do
context 'a valid policy' do let_it_be(:container_expiration_policy, reload: true) { create(:container_expiration_policy, :runnable) }
let!(:container_expiration_policy) { create(:container_expiration_policy, :runnable) } let_it_be(:container_repository) { create(:container_repository, project: container_expiration_policy.project) }
let(:user) { container_expiration_policy.project.owner } let_it_be(:user) { container_expiration_policy.project.owner }
context 'a valid policy' do
it 'runs the policy' do it 'runs the policy' do
service = instance_double(ContainerExpirationPolicyService, execute: true) service = instance_double(ContainerExpirationPolicyService, execute: true)
...@@ -31,33 +36,30 @@ RSpec.describe ContainerExpirationPolicyWorker do ...@@ -31,33 +36,30 @@ RSpec.describe ContainerExpirationPolicyWorker do
end end
context 'a disabled policy' do context 'a disabled policy' do
let!(:container_expiration_policy) { create(:container_expiration_policy, :runnable, :disabled) } before do
let(:user) {container_expiration_policy.project.owner } container_expiration_policy.disable!
it 'does not run the policy' do
expect(ContainerExpirationPolicyService)
.not_to receive(:new).with(container_expiration_policy, user)
subject
end end
it_behaves_like 'not executing any policy'
end end
context 'a policy that is not due for a run' do context 'a policy that is not due for a run' do
let!(:container_expiration_policy) { create(:container_expiration_policy) } before do
let(:user) {container_expiration_policy.project.owner } container_expiration_policy.update_column(:next_run_at, 2.minutes.from_now)
end
it 'does not run the policy' do it_behaves_like 'not executing any policy'
expect(ContainerExpirationPolicyService) end
.not_to receive(:new).with(container_expiration_policy, user)
subject context 'a policy linked to no container repository' do
before do
container_expiration_policy.container_repositories.delete_all
end end
it_behaves_like 'not executing any policy'
end end
context 'an invalid policy' do context 'an invalid policy' do
let_it_be(:container_expiration_policy) { create(:container_expiration_policy, :runnable) }
let_it_be(:user) {container_expiration_policy.project.owner }
before do before do
container_expiration_policy.update_column(:name_regex, '*production') container_expiration_policy.update_column(:name_regex, '*production')
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