cleanup_tags_service.rb 2.9 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?
K
Kamil Trzciński 已提交
9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28

        tags = container_repository.tags
        tags_by_digest = group_by_digest(tags)

        tags = without_latest(tags)
        tags = filter_by_name(tags)
        tags = with_manifest(tags)
        tags = order_by_date(tags)
        tags = filter_keep_n(tags)
        tags = filter_by_older_than(tags)

        deleted_tags = delete_tags(tags, tags_by_digest)

        success(deleted: deleted_tags.map(&:name))
      end

      private

      def delete_tags(tags_to_delete, tags_by_digest)
        deleted_digests = group_by_digest(tags_to_delete).select do |digest, tags|
29
          delete_tag_digest(tags, tags_by_digest[digest])
K
Kamil Trzciński 已提交
30 31 32 33 34
        end

        deleted_digests.values.flatten
      end

35
      def delete_tag_digest(tags, other_tags)
36
        # Issue: https://gitlab.com/gitlab-org/gitlab-foss/issues/21405
K
Kamil Trzciński 已提交
37 38 39 40 41 42
        # we have to remove all tags due
        # to Docker Distribution bug unable
        # to delete single tag
        return unless tags.count == other_tags.count

        # delete all tags
43
        tags.map(&:unsafe_delete)
K
Kamil Trzciński 已提交
44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63
      end

      def group_by_digest(tags)
        tags.group_by(&:digest)
      end

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

      def with_manifest(tags)
        tags.select(&:valid?)
      end

      def order_by_date(tags)
        now = DateTime.now
        tags.sort_by { |tag| tag.created_at || now }.reverse
      end

      def filter_by_name(tags)
64 65 66 67 68
        # 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
        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 已提交
69 70

        tags.select do |tag|
71 72
          # regex_retain will override any overlapping matches by regex_delete
          regex_delete.match?(tag.name) && !regex_retain.match?(tag.name)
K
Kamil Trzciński 已提交
73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89
        end
      end

      def filter_keep_n(tags)
        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

90 91 92 93
      def can_destroy?
        return true if params['container_expiration_policy']

        can?(current_user, :destroy_container_image, project)
K
Kamil Trzciński 已提交
94 95 96 97 98 99 100 101
      end

      def can_use?
        Feature.enabled?(:container_registry_cleanup, project, default_enabled: true)
      end
    end
  end
end