2019-10-12 21:52:04 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
RSpec.describe ContainerRegistry::Client do
|
2017-09-10 17:25:29 +05:30
|
|
|
let(:token) { '12345' }
|
|
|
|
let(:options) { { token: token } }
|
|
|
|
let(:client) { described_class.new("http://container-registry", options) }
|
2020-04-22 19:07:51 +05:30
|
|
|
let(:push_blob_headers) do
|
|
|
|
{
|
|
|
|
'Accept' => 'application/vnd.docker.distribution.manifest.v2+json, application/vnd.oci.image.manifest.v1+json',
|
|
|
|
'Authorization' => "bearer #{token}",
|
|
|
|
'Content-Type' => 'application/octet-stream',
|
|
|
|
'User-Agent' => "GitLab/#{Gitlab::VERSION}"
|
|
|
|
}
|
|
|
|
end
|
2020-10-24 23:57:45 +05:30
|
|
|
|
2020-04-22 19:07:51 +05:30
|
|
|
let(:headers_with_accept_types) do
|
|
|
|
{
|
|
|
|
'Accept' => 'application/vnd.docker.distribution.manifest.v2+json, application/vnd.oci.image.manifest.v1+json',
|
|
|
|
'Authorization' => "bearer #{token}",
|
|
|
|
'User-Agent' => "GitLab/#{Gitlab::VERSION}"
|
|
|
|
}
|
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
shared_examples '#repository_manifest' do |manifest_type|
|
|
|
|
let(:manifest) do
|
|
|
|
{
|
|
|
|
"schemaVersion" => 2,
|
|
|
|
"config" => {
|
|
|
|
"mediaType" => manifest_type,
|
|
|
|
"digest" =>
|
|
|
|
"sha256:4a3ef0786dd241be6000311e1503869b320be433b9cba84cfafeb512d1720c95",
|
|
|
|
"size" => 6608
|
|
|
|
},
|
|
|
|
"layers" => [
|
|
|
|
{
|
|
|
|
"mediaType" => manifest_type,
|
|
|
|
"digest" =>
|
|
|
|
"sha256:83ef92b73cf4595aa7fe214ec6747228283d585f373d8f6bc08d66bebab531b7",
|
|
|
|
"size" => 2828661
|
|
|
|
}
|
|
|
|
]
|
2020-04-22 19:07:51 +05:30
|
|
|
}
|
2019-09-30 21:07:59 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'GET /v2/:name/manifests/mytag' do
|
|
|
|
stub_request(:get, "http://container-registry/v2/group/test/manifests/mytag")
|
|
|
|
.with(headers: {
|
2020-04-22 19:07:51 +05:30
|
|
|
'Accept' => 'application/vnd.docker.distribution.manifest.v2+json, application/vnd.oci.image.manifest.v1+json',
|
|
|
|
'Authorization' => "bearer #{token}",
|
|
|
|
'User-Agent' => "GitLab/#{Gitlab::VERSION}"
|
2019-09-30 21:07:59 +05:30
|
|
|
})
|
|
|
|
.to_return(status: 200, body: manifest.to_json, headers: { content_type: manifest_type })
|
|
|
|
|
|
|
|
expect(client.repository_manifest('group/test', 'mytag')).to eq(manifest)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like '#repository_manifest', described_class::DOCKER_DISTRIBUTION_MANIFEST_V2_TYPE
|
|
|
|
it_behaves_like '#repository_manifest', described_class::OCI_MANIFEST_V1_TYPE
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
describe '#blob' do
|
2020-04-22 19:07:51 +05:30
|
|
|
let(:blob_headers) do
|
|
|
|
{
|
|
|
|
'Accept' => 'application/octet-stream',
|
|
|
|
'Authorization' => "bearer #{token}",
|
|
|
|
'User-Agent' => "GitLab/#{Gitlab::VERSION}"
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
let(:redirect_header) do
|
|
|
|
{
|
|
|
|
'User-Agent' => "GitLab/#{Gitlab::VERSION}"
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
it 'GET /v2/:name/blobs/:digest' do
|
|
|
|
stub_request(:get, "http://container-registry/v2/group/test/blobs/sha256:0123456789012345")
|
2020-04-22 19:07:51 +05:30
|
|
|
.with(headers: blob_headers)
|
2017-09-10 17:25:29 +05:30
|
|
|
.to_return(status: 200, body: "Blob")
|
|
|
|
|
|
|
|
expect(client.blob('group/test', 'sha256:0123456789012345')).to eq('Blob')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'follows 307 redirect for GET /v2/:name/blobs/:digest' do
|
|
|
|
stub_request(:get, "http://container-registry/v2/group/test/blobs/sha256:0123456789012345")
|
2020-04-22 19:07:51 +05:30
|
|
|
.with(headers: blob_headers)
|
2020-05-24 23:13:21 +05:30
|
|
|
.to_return(status: 307, body: '', headers: { Location: 'http://redirected' })
|
2017-09-10 17:25:29 +05:30
|
|
|
# We should probably use hash_excluding here, but that requires an update to WebMock:
|
|
|
|
# https://github.com/bblimke/webmock/blob/master/lib/webmock/matchers/hash_excluding_matcher.rb
|
|
|
|
stub_request(:get, "http://redirected/")
|
2020-04-22 19:07:51 +05:30
|
|
|
.with(headers: redirect_header) do |request|
|
|
|
|
!request.headers.include?('Authorization')
|
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
.to_return(status: 200, body: "Successfully redirected")
|
|
|
|
|
|
|
|
response = client.blob('group/test', 'sha256:0123456789012345')
|
|
|
|
|
|
|
|
expect(response).to eq('Successfully redirected')
|
|
|
|
end
|
|
|
|
end
|
2019-12-21 20:55:43 +05:30
|
|
|
|
|
|
|
def stub_upload(path, content, digest, status = 200)
|
|
|
|
stub_request(:post, "http://container-registry/v2/#{path}/blobs/uploads/")
|
2020-04-22 19:07:51 +05:30
|
|
|
.with(headers: headers_with_accept_types)
|
2019-12-21 20:55:43 +05:30
|
|
|
.to_return(status: status, body: "", headers: { 'location' => 'http://container-registry/next_upload?id=someid' })
|
|
|
|
|
|
|
|
stub_request(:put, "http://container-registry/next_upload?digest=#{digest}&id=someid")
|
2020-04-22 19:07:51 +05:30
|
|
|
.with(body: content, headers: push_blob_headers)
|
2019-12-21 20:55:43 +05:30
|
|
|
.to_return(status: status, body: "", headers: {})
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#upload_blob' do
|
|
|
|
subject { client.upload_blob('path', 'content', 'sha256:123') }
|
|
|
|
|
|
|
|
context 'with successful uploads' do
|
|
|
|
it 'starts the upload and posts the blob' do
|
|
|
|
stub_upload('path', 'content', 'sha256:123')
|
|
|
|
|
|
|
|
expect(subject).to be_success
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with a failed upload' do
|
|
|
|
before do
|
|
|
|
stub_upload('path', 'content', 'sha256:123', 400)
|
|
|
|
end
|
|
|
|
|
2019-12-26 22:10:19 +05:30
|
|
|
it 'returns a failure' do
|
|
|
|
expect(subject).not_to be_success
|
2019-12-21 20:55:43 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#generate_empty_manifest' do
|
|
|
|
subject { client.generate_empty_manifest('path') }
|
|
|
|
|
|
|
|
let(:result_manifest) do
|
|
|
|
{
|
|
|
|
schemaVersion: 2,
|
|
|
|
mediaType: 'application/vnd.docker.distribution.manifest.v2+json',
|
|
|
|
config: {
|
|
|
|
mediaType: 'application/vnd.docker.container.image.v1+json',
|
|
|
|
size: 21,
|
|
|
|
digest: 'sha256:4435000728ee66e6a80e55637fc22725c256b61de344a2ecdeaac6bdb36e8bc3'
|
|
|
|
}
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'uploads a random image and returns the manifest' do
|
|
|
|
stub_upload('path', "{\n \"config\": {\n }\n}", 'sha256:4435000728ee66e6a80e55637fc22725c256b61de344a2ecdeaac6bdb36e8bc3')
|
|
|
|
|
|
|
|
expect(subject).to eq(result_manifest)
|
|
|
|
end
|
2019-12-26 22:10:19 +05:30
|
|
|
|
|
|
|
context 'when upload fails' do
|
|
|
|
before do
|
|
|
|
stub_upload('path', "{\n \"config\": {\n }\n}", 'sha256:4435000728ee66e6a80e55637fc22725c256b61de344a2ecdeaac6bdb36e8bc3', 500)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to be nil }
|
|
|
|
end
|
2019-12-21 20:55:43 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
describe '#put_tag' do
|
2020-04-22 19:07:51 +05:30
|
|
|
let(:manifest_headers) do
|
|
|
|
{
|
|
|
|
'Accept' => 'application/vnd.docker.distribution.manifest.v2+json, application/vnd.oci.image.manifest.v1+json',
|
|
|
|
'Authorization' => "bearer #{token}",
|
|
|
|
'Content-Type' => 'application/vnd.docker.distribution.manifest.v2+json',
|
|
|
|
'User-Agent' => "GitLab/#{Gitlab::VERSION}"
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
subject { client.put_tag('path', 'tagA', { foo: :bar }) }
|
|
|
|
|
|
|
|
it 'uploads the manifest and returns the digest' do
|
|
|
|
stub_request(:put, "http://container-registry/v2/path/manifests/tagA")
|
2020-04-22 19:07:51 +05:30
|
|
|
.with(body: "{\n \"foo\": \"bar\"\n}", headers: manifest_headers)
|
2019-12-21 20:55:43 +05:30
|
|
|
.to_return(status: 200, body: "", headers: { 'docker-content-digest' => 'sha256:123' })
|
|
|
|
|
|
|
|
expect(subject).to eq 'sha256:123'
|
|
|
|
end
|
|
|
|
end
|
2020-03-13 15:44:24 +05:30
|
|
|
|
|
|
|
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")
|
2020-04-22 19:07:51 +05:30
|
|
|
.with(headers: headers_with_accept_types)
|
2020-03-13 15:44:24 +05:30
|
|
|
.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")
|
2020-04-22 19:07:51 +05:30
|
|
|
.with(headers: headers_with_accept_types)
|
2020-03-13 15:44:24 +05:30
|
|
|
.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")
|
2020-04-22 19:07:51 +05:30
|
|
|
.with(headers: headers_with_accept_types)
|
2020-03-13 15:44:24 +05:30
|
|
|
.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
|
2020-05-24 23:13:21 +05:30
|
|
|
|
|
|
|
def stub_registry_info(headers: {}, status: 200)
|
|
|
|
stub_request(:get, 'http://container-registry/v2/')
|
|
|
|
.to_return(status: status, body: "", headers: headers)
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#registry_info' do
|
|
|
|
subject { client.registry_info }
|
|
|
|
|
|
|
|
context 'when the check is successful' do
|
|
|
|
context 'when using the GitLab container registry' do
|
|
|
|
before do
|
|
|
|
stub_registry_info(headers: {
|
|
|
|
'GitLab-Container-Registry-Version' => '2.9.1-gitlab',
|
|
|
|
'GitLab-Container-Registry-Features' => 'a,b,c'
|
|
|
|
})
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'identifies the vendor as "gitlab"' do
|
|
|
|
expect(subject).to include(vendor: 'gitlab')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'identifies version and features' do
|
|
|
|
expect(subject).to include(version: '2.9.1-gitlab', features: %w[a b c])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when using a third-party container registry' do
|
|
|
|
before do
|
|
|
|
stub_registry_info
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'identifies the vendor as "other"' do
|
|
|
|
expect(subject).to include(vendor: 'other')
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'does not identify version or features' do
|
|
|
|
expect(subject).to include(version: nil, features: [])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when the check is not successful' do
|
|
|
|
it 'does not identify vendor, version or features' do
|
|
|
|
stub_registry_info(status: 500)
|
|
|
|
|
|
|
|
expect(subject).to eq({})
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2020-11-24 15:15:51 +05:30
|
|
|
|
|
|
|
describe '.supports_tag_delete?' do
|
|
|
|
let(:registry_enabled) { true }
|
|
|
|
let(:registry_api_url) { 'http://sandbox.local' }
|
|
|
|
let(:registry_tags_support_enabled) { true }
|
|
|
|
let(:is_on_dot_com) { false }
|
|
|
|
|
|
|
|
subject { described_class.supports_tag_delete? }
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(::Gitlab).to receive(:com?).and_return(is_on_dot_com)
|
|
|
|
stub_container_registry_config(enabled: registry_enabled, api_url: registry_api_url, key: 'spec/fixtures/x509_certificate_pk.key')
|
|
|
|
stub_registry_tags_support(registry_tags_support_enabled)
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with the registry enabled' do
|
|
|
|
it { is_expected.to be true }
|
|
|
|
|
|
|
|
context 'without an api url' do
|
|
|
|
let(:registry_api_url) { '' }
|
|
|
|
|
|
|
|
it { is_expected.to be false }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'on .com' do
|
|
|
|
let(:is_on_dot_com) { true }
|
|
|
|
|
|
|
|
it { is_expected.to be true }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when registry server does not support tag deletion' do
|
|
|
|
let(:registry_tags_support_enabled) { false }
|
|
|
|
|
|
|
|
it { is_expected.to be false }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with the registry disabled' do
|
|
|
|
let(:registry_enabled) { false }
|
|
|
|
|
|
|
|
it { is_expected.to be false }
|
|
|
|
end
|
|
|
|
|
|
|
|
def stub_registry_tags_support(supported = true)
|
|
|
|
status_code = supported ? 200 : 404
|
|
|
|
stub_request(:options, "#{registry_api_url}/v2/name/tags/reference/tag")
|
|
|
|
.to_return(
|
|
|
|
status: status_code,
|
|
|
|
body: '',
|
|
|
|
headers: { 'Allow' => 'DELETE' }
|
|
|
|
)
|
|
|
|
end
|
|
|
|
end
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|