Commit d816489a authored by Stan Hu's avatar Stan Hu

Merge branch '31832-improve-performance-of-the-container-registry-delete-tags-api' into 'master'

Improve performance of the Container Registry delete tags API

See merge request gitlab-org/gitlab!23325
parents 1e32c494 76ab17ae
...@@ -77,7 +77,11 @@ class ContainerRepository < ApplicationRecord ...@@ -77,7 +77,11 @@ class ContainerRepository < ApplicationRecord
end end
def delete_tag_by_digest(digest) def delete_tag_by_digest(digest)
client.delete_repository_tag(self.path, digest) client.delete_repository_tag_by_digest(self.path, digest)
end
def delete_tag_by_name(name)
client.delete_repository_tag_by_name(self.path, name)
end end
def self.build_from_path(path) def self.build_from_path(path)
......
...@@ -14,12 +14,25 @@ module Projects ...@@ -14,12 +14,25 @@ module Projects
private private
# Delete tags by name with a single DELETE request. This is only supported
# by the GitLab Container Registry fork. See
# https://gitlab.com/gitlab-org/gitlab/-/merge_requests/23325 for details.
def fast_delete(container_repository, tag_names)
deleted_tags = tag_names.select do |name|
container_repository.delete_tag_by_name(name)
end
deleted_tags.any? ? success(deleted: deleted_tags) : error('could not delete tags')
end
# Replace a tag on the registry with a dummy tag. # Replace a tag on the registry with a dummy tag.
# This is a hack as the registry doesn't support deleting individual # This is a hack as the registry doesn't support deleting individual
# tags. This code effectively pushes a dummy image and assigns the tag to it. # tags. This code effectively pushes a dummy image and assigns the tag to it.
# This way when the tag is deleted only the dummy image is affected. # This way when the tag is deleted only the dummy image is affected.
# This is used to preverse compatibility with third-party registries that
# don't support fast delete.
# See https://gitlab.com/gitlab-org/gitlab/issues/15737 for a discussion # See https://gitlab.com/gitlab-org/gitlab/issues/15737 for a discussion
def smart_delete(container_repository, tag_names) def slow_delete(container_repository, tag_names)
# generates the blobs for the dummy image # generates the blobs for the dummy image
dummy_manifest = container_repository.client.generate_empty_manifest(container_repository.path) dummy_manifest = container_repository.client.generate_empty_manifest(container_repository.path)
return error('could not generate manifest') if dummy_manifest.nil? return error('could not generate manifest') if dummy_manifest.nil?
...@@ -36,6 +49,15 @@ module Projects ...@@ -36,6 +49,15 @@ module Projects
end end
end end
def smart_delete(container_repository, tag_names)
fast_delete_enabled = Feature.enabled?(:container_registry_fast_tag_delete, default_enabled: true)
if fast_delete_enabled && container_repository.client.supports_tag_delete?
fast_delete(container_repository, tag_names)
else
slow_delete(container_repository, tag_names)
end
end
# update the manifests of the tags with the new dummy image # update the manifests of the tags with the new dummy image
def replace_tag_manifests(container_repository, dummy_manifest, tag_names) def replace_tag_manifests(container_repository, dummy_manifest, tag_names)
deleted_tags = {} deleted_tags = {}
......
---
title: Improve performance of the Container Registry delete tags API
merge_request: 23325
author:
type: performance
...@@ -6,6 +6,8 @@ require 'digest' ...@@ -6,6 +6,8 @@ require 'digest'
module ContainerRegistry module ContainerRegistry
class Client class Client
include Gitlab::Utils::StrongMemoize
attr_accessor :uri attr_accessor :uri
DOCKER_DISTRIBUTION_MANIFEST_V2_TYPE = 'application/vnd.docker.distribution.manifest.v2+json' DOCKER_DISTRIBUTION_MANIFEST_V2_TYPE = 'application/vnd.docker.distribution.manifest.v2+json'
...@@ -35,10 +37,25 @@ module ContainerRegistry ...@@ -35,10 +37,25 @@ module ContainerRegistry
response.headers['docker-content-digest'] if response.success? response.headers['docker-content-digest'] if response.success?
end end
def delete_repository_tag(name, reference) def delete_repository_tag_by_digest(name, reference)
result = faraday.delete("/v2/#{name}/manifests/#{reference}") delete_if_exists("/v2/#{name}/manifests/#{reference}")
end
result.success? || result.status == 404 def delete_repository_tag_by_name(name, reference)
delete_if_exists("/v2/#{name}/tags/reference/#{reference}")
end
# Check if the registry supports tag deletion. This is only supported by the
# GitLab registry fork. The fastest and safest way to check this is to send
# an OPTIONS request to /v2/<name>/tags/reference/<tag>, using a random
# repository name and tag (the registry won't check if they exist).
# Registries that support tag deletion will reply with a 200 OK and include
# the DELETE method in the Allow header. Others reply with an 404 Not Found.
def supports_tag_delete?
strong_memoize(:supports_tag_delete) do
response = faraday.run_request(:options, '/v2/name/tags/reference/tag', '', {})
response.success? && response.headers['allow']&.include?('DELETE')
end
end end
def upload_raw_blob(path, blob) def upload_raw_blob(path, blob)
...@@ -86,9 +103,7 @@ module ContainerRegistry ...@@ -86,9 +103,7 @@ module ContainerRegistry
end end
def delete_blob(name, digest) def delete_blob(name, digest)
result = faraday.delete("/v2/#{name}/blobs/#{digest}") delete_if_exists("/v2/#{name}/blobs/#{digest}")
result.success? || result.status == 404
end end
def put_tag(name, reference, manifest) def put_tag(name, reference, manifest)
...@@ -163,6 +178,12 @@ module ContainerRegistry ...@@ -163,6 +178,12 @@ module ContainerRegistry
conn.adapter :net_http conn.adapter :net_http
end end
end end
def delete_if_exists(path)
result = faraday.delete(path)
result.success? || result.status == 404
end
end end
end end
......
...@@ -118,7 +118,7 @@ module ContainerRegistry ...@@ -118,7 +118,7 @@ module ContainerRegistry
def unsafe_delete def unsafe_delete
return unless digest return unless digest
client.delete_repository_tag(repository.path, digest) client.delete_repository_tag_by_digest(repository.path, digest)
end end
end end
end end
...@@ -146,4 +146,57 @@ describe ContainerRegistry::Client do ...@@ -146,4 +146,57 @@ describe ContainerRegistry::Client do
expect(subject).to eq 'sha256:123' expect(subject).to eq 'sha256:123'
end end
end end
describe '#delete_repository_tag_by_name' do
subject { client.delete_repository_tag_by_name('group/test', 'a') }
context 'when the tag exists' do
before do
stub_request(:delete, "http://container-registry/v2/group/test/tags/reference/a")
.to_return(status: 200, body: "")
end
it { is_expected.to be_truthy }
end
context 'when the tag does not exist' do
before do
stub_request(:delete, "http://container-registry/v2/group/test/tags/reference/a")
.to_return(status: 404, body: "")
end
it { is_expected.to be_truthy }
end
context 'when an error occurs' do
before do
stub_request(:delete, "http://container-registry/v2/group/test/tags/reference/a")
.to_return(status: 500, body: "")
end
it { is_expected.to be_falsey }
end
end
describe '#supports_tag_delete?' do
subject { client.supports_tag_delete? }
context 'when the server supports tag deletion' do
before do
stub_request(:options, "http://container-registry/v2/name/tags/reference/tag")
.to_return(status: 200, body: "", headers: { 'Allow' => 'DELETE' })
end
it { is_expected.to be_truthy }
end
context 'when the server does not support tag deletion' do
before do
stub_request(:options, "http://container-registry/v2/name/tags/reference/tag")
.to_return(status: 404, body: "")
end
it { is_expected.to be_falsey }
end
end
end end
...@@ -85,7 +85,7 @@ describe ContainerRepository do ...@@ -85,7 +85,7 @@ describe ContainerRepository do
context 'when action succeeds' do context 'when action succeeds' do
it 'returns status that indicates success' do it 'returns status that indicates success' do
expect(repository.client) expect(repository.client)
.to receive(:delete_repository_tag) .to receive(:delete_repository_tag_by_digest)
.twice .twice
.and_return(true) .and_return(true)
...@@ -96,7 +96,7 @@ describe ContainerRepository do ...@@ -96,7 +96,7 @@ describe ContainerRepository do
context 'when action fails' do context 'when action fails' do
it 'returns status that indicates failure' do it 'returns status that indicates failure' do
expect(repository.client) expect(repository.client)
.to receive(:delete_repository_tag) .to receive(:delete_repository_tag_by_digest)
.twice .twice
.and_return(false) .and_return(false)
...@@ -105,6 +105,36 @@ describe ContainerRepository do ...@@ -105,6 +105,36 @@ describe ContainerRepository do
end end
end end
describe '#delete_tag_by_name' do
let(:repository) do
create(:container_repository, name: 'my_image',
tags: { latest: '123', rc1: '234' },
project: project)
end
context 'when action succeeds' do
it 'returns status that indicates success' do
expect(repository.client)
.to receive(:delete_repository_tag_by_name)
.with(repository.path, "latest")
.and_return(true)
expect(repository.delete_tag_by_name('latest')).to be_truthy
end
end
context 'when action fails' do
it 'returns status that indicates failure' do
expect(repository.client)
.to receive(:delete_repository_tag_by_name)
.with(repository.path, "latest")
.and_return(false)
expect(repository.delete_tag_by_name('latest')).to be_falsey
end
end
end
describe '#location' do describe '#location' do
context 'when registry is running on a custom port' do context 'when registry is running on a custom port' do
before do before do
......
...@@ -41,7 +41,7 @@ describe Projects::ContainerRepository::CleanupTagsService do ...@@ -41,7 +41,7 @@ describe Projects::ContainerRepository::CleanupTagsService do
let(:params) { {} } let(:params) { {} }
it 'does not remove anything' do it 'does not remove anything' do
expect_any_instance_of(ContainerRegistry::Client).not_to receive(:delete_repository_tag) expect_any_instance_of(ContainerRegistry::Client).not_to receive(:delete_repository_tag_by_digest)
is_expected.to include(status: :success, deleted: []) is_expected.to include(status: :success, deleted: [])
end end
...@@ -156,7 +156,7 @@ describe Projects::ContainerRepository::CleanupTagsService do ...@@ -156,7 +156,7 @@ describe Projects::ContainerRepository::CleanupTagsService do
def expect_delete(digest) def expect_delete(digest)
expect_any_instance_of(ContainerRegistry::Client) expect_any_instance_of(ContainerRegistry::Client)
.to receive(:delete_repository_tag) .to receive(:delete_repository_tag_by_digest)
.with(repository.path, digest) { true } .with(repository.path, digest) { true }
end end
end end
...@@ -18,10 +18,6 @@ describe Projects::ContainerRepository::DeleteTagsService do ...@@ -18,10 +18,6 @@ describe Projects::ContainerRepository::DeleteTagsService do
stub_container_registry_tags( stub_container_registry_tags(
repository: repository.path, repository: repository.path,
tags: %w(latest A Ba Bb C D E)) tags: %w(latest A Ba Bb C D E))
stub_tag_digest('latest', 'sha256:configA')
stub_tag_digest('A', 'sha256:configA')
stub_tag_digest('Ba', 'sha256:configB')
end end
describe '#execute' do describe '#execute' do
...@@ -38,82 +34,178 @@ describe Projects::ContainerRepository::DeleteTagsService do ...@@ -38,82 +34,178 @@ describe Projects::ContainerRepository::DeleteTagsService do
project.add_developer(user) project.add_developer(user)
end end
context 'when no params are specified' do context 'when the registry supports fast delete' do
let(:params) { {} } context 'and the feature is enabled' do
let_it_be(:project) { create(:project, :private) }
let_it_be(:repository) { create(:container_repository, :root, project: project) }
it 'does not remove anything' do before do
expect_any_instance_of(ContainerRegistry::Client).not_to receive(:delete_repository_tag) allow(repository.client).to receive(:supports_tag_delete?).and_return(true)
end
is_expected.to include(status: :error) context 'with tags to delete' do
end let_it_be(:tags) { %w[A Ba] }
end
context 'with empty tags' do it 'deletes the tags by name' do
let(:tags) { [] } stub_request(:delete, "http://registry.gitlab/v2/#{repository.path}/tags/reference/A")
.to_return(status: 200, body: "")
it 'does not remove anything' do stub_request(:delete, "http://registry.gitlab/v2/#{repository.path}/tags/reference/Ba")
expect_any_instance_of(ContainerRegistry::Client).not_to receive(:delete_repository_tag) .to_return(status: 200, body: "")
is_expected.to include(status: :error) expect_delete_tag_by_name('A')
end expect_delete_tag_by_name('Ba')
end
is_expected.to include(status: :success)
end
it 'succeeds when tag delete returns 404' do
stub_request(:delete, "http://registry.gitlab/v2/#{repository.path}/tags/reference/A")
.to_return(status: 200, body: "")
stub_request(:delete, "http://registry.gitlab/v2/#{repository.path}/tags/reference/Ba")
.to_return(status: 404, body: "")
is_expected.to include(status: :success)
end
context 'with failures' do
context 'when the delete request fails' do
before do
stub_request(:delete, "http://registry.gitlab/v2/#{repository.path}/tags/reference/A")
.to_return(status: 500, body: "")
stub_request(:delete, "http://registry.gitlab/v2/#{repository.path}/tags/reference/Ba")
.to_return(status: 500, body: "")
end
context 'with tags to delete' do it { is_expected.to include(status: :error) }
let(:tags) { %w[A Ba] } end
end
end
context 'when no params are specified' do
let_it_be(:params) { {} }
it 'does not remove anything' do
expect_any_instance_of(ContainerRegistry::Client).not_to receive(:delete_repository_tag_by_name)
is_expected.to include(status: :error)
end
end
it 'deletes the tags using a dummy image' do context 'with empty tags' do
stub_upload("{\n \"config\": {\n }\n}", 'sha256:4435000728ee66e6a80e55637fc22725c256b61de344a2ecdeaac6bdb36e8bc3') let_it_be(:tags) { [] }
stub_request(:put, "http://registry.gitlab/v2/#{repository.path}/manifests/A") it 'does not remove anything' do
.to_return(status: 200, body: "", headers: { 'docker-content-digest' => 'sha256:dummy' }) expect_any_instance_of(ContainerRegistry::Client).not_to receive(:delete_repository_tag_by_name)
stub_request(:put, "http://registry.gitlab/v2/#{repository.path}/manifests/Ba") is_expected.to include(status: :error)
.to_return(status: 200, body: "", headers: { 'docker-content-digest' => 'sha256:dummy' }) end
end
end
context 'and the feature is disabled' do
before do
stub_feature_flags(container_registry_fast_tag_delete: false)
end
expect_delete_tag('sha256:dummy') it 'fallbacks to slow delete' do
expect(service).not_to receive(:fast_delete)
expect(service).to receive(:slow_delete).with(repository, tags)
is_expected.to include(status: :success) subject
end
end end
end
context 'when the registry does not support fast delete' do
let_it_be(:project) { create(:project, :private) }
let_it_be(:repository) { create(:container_repository, :root, project: project) }
it 'succedes when tag delete returns 404' do before do
stub_upload("{\n \"config\": {\n }\n}", 'sha256:4435000728ee66e6a80e55637fc22725c256b61de344a2ecdeaac6bdb36e8bc3') stub_tag_digest('latest', 'sha256:configA')
stub_tag_digest('A', 'sha256:configA')
stub_tag_digest('Ba', 'sha256:configB')
stub_request(:put, "http://registry.gitlab/v2/#{repository.path}/manifests/A") allow(repository.client).to receive(:supports_tag_delete?).and_return(false)
.to_return(status: 200, body: "", headers: { 'docker-content-digest' => 'sha256:dummy' }) end
stub_request(:put, "http://registry.gitlab/v2/#{repository.path}/manifests/Ba") context 'when no params are specified' do
.to_return(status: 200, body: "", headers: { 'docker-content-digest' => 'sha256:dummy' }) let_it_be(:params) { {} }
stub_request(:delete, "http://registry.gitlab/v2/#{repository.path}/manifests/sha256:dummy") it 'does not remove anything' do
.to_return(status: 404, body: "", headers: {}) expect_any_instance_of(ContainerRegistry::Client).not_to receive(:delete_repository_tag_by_digest)
is_expected.to include(status: :success) is_expected.to include(status: :error)
end
end end
context 'with failures' do context 'with empty tags' do
context 'when the dummy manifest generation fails' do let_it_be(:tags) { [] }
before do
stub_upload("{\n \"config\": {\n }\n}", 'sha256:4435000728ee66e6a80e55637fc22725c256b61de344a2ecdeaac6bdb36e8bc3', success: false) it 'does not remove anything' do
end expect_any_instance_of(ContainerRegistry::Client).not_to receive(:delete_repository_tag_by_digest)
is_expected.to include(status: :error)
end
end
context 'with tags to delete' do
let_it_be(:tags) { %w[A Ba] }
it { is_expected.to include(status: :error) } it 'deletes the tags using a dummy image' do
stub_upload("{\n \"config\": {\n }\n}", 'sha256:4435000728ee66e6a80e55637fc22725c256b61de344a2ecdeaac6bdb36e8bc3')
stub_request(:put, "http://registry.gitlab/v2/#{repository.path}/manifests/A")
.to_return(status: 200, body: "", headers: { 'docker-content-digest' => 'sha256:dummy' })
stub_request(:put, "http://registry.gitlab/v2/#{repository.path}/manifests/Ba")
.to_return(status: 200, body: "", headers: { 'docker-content-digest' => 'sha256:dummy' })
expect_delete_tag_by_digest('sha256:dummy')
is_expected.to include(status: :success)
end end
context 'when updating the tags fails' do it 'succeeds when tag delete returns 404' do
before do stub_upload("{\n \"config\": {\n }\n}", 'sha256:4435000728ee66e6a80e55637fc22725c256b61de344a2ecdeaac6bdb36e8bc3')
stub_upload("{\n \"config\": {\n }\n}", 'sha256:4435000728ee66e6a80e55637fc22725c256b61de344a2ecdeaac6bdb36e8bc3')
stub_request(:put, "http://registry.gitlab/v2/#{repository.path}/manifests/A") stub_request(:put, "http://registry.gitlab/v2/#{repository.path}/manifests/A")
.to_return(status: 500, body: "", headers: { 'docker-content-digest' => 'sha256:dummy' }) .to_return(status: 200, body: "", headers: { 'docker-content-digest' => 'sha256:dummy' })
stub_request(:put, "http://registry.gitlab/v2/#{repository.path}/manifests/Ba") stub_request(:put, "http://registry.gitlab/v2/#{repository.path}/manifests/Ba")
.to_return(status: 500, body: "", headers: { 'docker-content-digest' => 'sha256:dummy' }) .to_return(status: 200, body: "", headers: { 'docker-content-digest' => 'sha256:dummy' })
stub_request(:delete, "http://registry.gitlab/v2/#{repository.path}/manifests/sha256:dummy")
.to_return(status: 404, body: "", headers: {})
is_expected.to include(status: :success)
end
stub_request(:delete, "http://registry.gitlab/v2/#{repository.path}/manifests/sha256:4435000728ee66e6a80e55637fc22725c256b61de344a2ecdeaac6bdb36e8bc3") context 'with failures' do
.to_return(status: 200, body: "", headers: {}) context 'when the dummy manifest generation fails' do
before do
stub_upload("{\n \"config\": {\n }\n}", 'sha256:4435000728ee66e6a80e55637fc22725c256b61de344a2ecdeaac6bdb36e8bc3', success: false)
end
it { is_expected.to include(status: :error) }
end end
it { is_expected.to include(status: :error) } context 'when updating the tags fails' do
before do
stub_upload("{\n \"config\": {\n }\n}", 'sha256:4435000728ee66e6a80e55637fc22725c256b61de344a2ecdeaac6bdb36e8bc3')
stub_request(:put, "http://registry.gitlab/v2/#{repository.path}/manifests/A")
.to_return(status: 500, body: "", headers: { 'docker-content-digest' => 'sha256:dummy' })
stub_request(:put, "http://registry.gitlab/v2/#{repository.path}/manifests/Ba")
.to_return(status: 500, body: "", headers: { 'docker-content-digest' => 'sha256:dummy' })
stub_request(:delete, "http://registry.gitlab/v2/#{repository.path}/manifests/sha256:4435000728ee66e6a80e55637fc22725c256b61de344a2ecdeaac6bdb36e8bc3")
.to_return(status: 200, body: "", headers: {})
end
it { is_expected.to include(status: :error) }
end
end end
end end
end end
...@@ -141,9 +233,21 @@ describe Projects::ContainerRepository::DeleteTagsService do ...@@ -141,9 +233,21 @@ describe Projects::ContainerRepository::DeleteTagsService do
.with(repository.path, content, digest) { double(success?: success ) } .with(repository.path, content, digest) { double(success?: success ) }
end end
def expect_delete_tag(digest) def expect_delete_tag_by_digest(digest)
expect_any_instance_of(ContainerRegistry::Client) expect_any_instance_of(ContainerRegistry::Client)
.to receive(:delete_repository_tag) .to receive(:delete_repository_tag_by_digest)
.with(repository.path, digest) { true } .with(repository.path, digest) { true }
expect_any_instance_of(ContainerRegistry::Client)
.not_to receive(:delete_repository_tag_by_name)
end
def expect_delete_tag_by_name(name)
expect_any_instance_of(ContainerRegistry::Client)
.to receive(:delete_repository_tag_by_name)
.with(repository.path, name) { true }
expect_any_instance_of(ContainerRegistry::Client)
.not_to receive(:delete_repository_tag_by_digest)
end end
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