cleanup_tags_service.rb 2.8 KB
Newer Older
K
Kamil Trzciński 已提交
1 2 3 4 5 6 7
# frozen_string_literal: true

module Projects
  module ContainerRepository
    class CleanupTagsService < BaseService
      def execute(container_repository)
        return error('feature disabled') unless can_use?
8
        return error('access denied') unless can_destroy?
9
        return error('invalid regex') unless valid_regex?
K
Kamil Trzciński 已提交
10 11 12 13 14 15 16

        tags = container_repository.tags
        tags = without_latest(tags)
        tags = filter_by_name(tags)
        tags = filter_keep_n(tags)
        tags = filter_by_older_than(tags)

17
        delete_tags(container_repository, tags)
K
Kamil Trzciński 已提交
18 19 20 21
      end

      private

22 23
      def delete_tags(container_repository, tags)
        return success(deleted: []) unless tags.any?
K
Kamil Trzciński 已提交
24

25
        tag_names = tags.map(&:name)
K
Kamil Trzciński 已提交
26

27 28 29
        Projects::ContainerRepository::DeleteTagsService
          .new(container_repository.project, current_user, tags: tag_names)
          .execute(container_repository)
K
Kamil Trzciński 已提交
30 31 32 33 34 35 36
      end

      def without_latest(tags)
        tags.reject(&:latest?)
      end

      def order_by_date(tags)
37
        now = DateTime.current
K
Kamil Trzciński 已提交
38 39 40 41
        tags.sort_by { |tag| tag.created_at || now }.reverse
      end

      def filter_by_name(tags)
42 43 44
        # Technical Debt: https://gitlab.com/gitlab-org/gitlab/issues/207267
        # name_regex to be removed when container_expiration_policies is updated
        # to have both regex columns
45 46
        regex_delete = ::Gitlab::UntrustedRegexp.new("\\A#{params['name_regex_delete'] || params['name_regex']}\\z")
        regex_retain = ::Gitlab::UntrustedRegexp.new("\\A#{params['name_regex_keep']}\\z")
K
Kamil Trzciński 已提交
47 48

        tags.select do |tag|
49 50
          # regex_retain will override any overlapping matches by regex_delete
          regex_delete.match?(tag.name) && !regex_retain.match?(tag.name)
K
Kamil Trzciński 已提交
51 52 53 54
        end
      end

      def filter_keep_n(tags)
55 56 57
        return tags unless params['keep_n']

        tags = order_by_date(tags)
K
Kamil Trzciński 已提交
58 59 60 61 62 63 64 65 66 67 68 69 70
        tags.drop(params['keep_n'].to_i)
      end

      def filter_by_older_than(tags)
        return tags unless params['older_than']

        older_than = ChronicDuration.parse(params['older_than']).seconds.ago

        tags.select do |tag|
          tag.created_at && tag.created_at < older_than
        end
      end

71 72 73 74
      def can_destroy?
        return true if params['container_expiration_policy']

        can?(current_user, :destroy_container_image, project)
K
Kamil Trzciński 已提交
75 76 77 78 79
      end

      def can_use?
        Feature.enabled?(:container_registry_cleanup, project, default_enabled: true)
      end
80 81 82 83

      def valid_regex?
        %w(name_regex_delete name_regex name_regex_keep).each do |param_name|
          regex = params[param_name]
84
          ::Gitlab::UntrustedRegexp.new(regex) unless regex.blank?
85 86 87
        end
        true
      rescue RegexpError => e
88
        ::Gitlab::ErrorTracking.log_exception(e, project_id: project.id)
89 90
        false
      end
K
Kamil Trzciński 已提交
91 92 93
    end
  end
end