2021-01-29 00:20:46 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require 'spec_helper'
|
|
|
|
|
|
|
|
RSpec.describe Groups::DependencyProxyForContainersController do
|
2021-02-22 17:27:13 +05:30
|
|
|
include HttpBasicAuthHelpers
|
|
|
|
include DependencyProxyHelpers
|
|
|
|
|
|
|
|
let_it_be(:user) { create(:user) }
|
2021-10-27 15:23:28 +05:30
|
|
|
let_it_be_with_reload(:group) { create(:group, :private) }
|
2021-04-29 21:17:54 +05:30
|
|
|
|
2021-01-29 00:20:46 +05:30
|
|
|
let(:token_response) { { status: :success, token: 'abcd1234' } }
|
2021-02-22 17:27:13 +05:30
|
|
|
let(:jwt) { build_jwt(user) }
|
|
|
|
let(:token_header) { "Bearer #{jwt.encoded}" }
|
2021-10-27 15:23:28 +05:30
|
|
|
let(:snowplow_gitlab_standard_context) { { namespace: group, user: user } }
|
2021-02-22 17:27:13 +05:30
|
|
|
|
|
|
|
shared_examples 'without a token' do
|
|
|
|
before do
|
|
|
|
request.headers['HTTP_AUTHORIZATION'] = nil
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'feature flag disabled' do
|
2021-10-27 15:23:28 +05:30
|
|
|
let_it_be(:group) { create(:group) }
|
|
|
|
|
2021-02-22 17:27:13 +05:30
|
|
|
before do
|
|
|
|
stub_feature_flags(dependency_proxy_for_private_groups: false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to have_gitlab_http_status(:ok) }
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to have_gitlab_http_status(:unauthorized) }
|
|
|
|
end
|
|
|
|
|
|
|
|
shared_examples 'feature flag disabled with private group' do
|
|
|
|
before do
|
|
|
|
stub_feature_flags(dependency_proxy_for_private_groups: false)
|
|
|
|
end
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
it 'returns not found' do
|
2021-02-22 17:27:13 +05:30
|
|
|
group.update!(visibility_level: Gitlab::VisibilityLevel::PRIVATE)
|
|
|
|
|
|
|
|
subject
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
expect(response).to have_gitlab_http_status(:not_found)
|
2021-02-22 17:27:13 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
shared_examples 'without permission' do
|
|
|
|
context 'with invalid user' do
|
|
|
|
before do
|
|
|
|
user = double('bad_user', id: 999)
|
|
|
|
token_header = "Bearer #{build_jwt(user).encoded}"
|
|
|
|
request.headers['HTTP_AUTHORIZATION'] = token_header
|
|
|
|
end
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
it { is_expected.to have_gitlab_http_status(:unauthorized) }
|
2021-02-22 17:27:13 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
context 'with valid user that does not have access' do
|
|
|
|
before do
|
|
|
|
request.headers['HTTP_AUTHORIZATION'] = token_header
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to have_gitlab_http_status(:not_found) }
|
|
|
|
end
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
context 'with deploy token from a different group,' do
|
|
|
|
let_it_be(:user) { create(:deploy_token, :group, :dependency_proxy_scopes) }
|
|
|
|
|
|
|
|
it { is_expected.to have_gitlab_http_status(:not_found) }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with revoked deploy token' do
|
|
|
|
let_it_be(:user) { create(:deploy_token, :revoked, :group, :dependency_proxy_scopes) }
|
|
|
|
let_it_be(:group_deploy_token) { create(:group_deploy_token, deploy_token: user, group: group) }
|
|
|
|
|
|
|
|
it { is_expected.to have_gitlab_http_status(:unauthorized) }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with expired deploy token' do
|
|
|
|
let_it_be(:user) { create(:deploy_token, :expired, :group, :dependency_proxy_scopes) }
|
|
|
|
let_it_be(:group_deploy_token) { create(:group_deploy_token, deploy_token: user, group: group) }
|
|
|
|
|
|
|
|
it { is_expected.to have_gitlab_http_status(:unauthorized) }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'with deploy token with insufficient scopes' do
|
|
|
|
let_it_be(:user) { create(:deploy_token, :group) }
|
|
|
|
let_it_be(:group_deploy_token) { create(:group_deploy_token, deploy_token: user, group: group) }
|
|
|
|
|
|
|
|
it { is_expected.to have_gitlab_http_status(:not_found) }
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when a group is not found' do
|
|
|
|
before do
|
|
|
|
expect(Group).to receive(:find_by_full_path).and_return(nil)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to have_gitlab_http_status(:not_found) }
|
|
|
|
end
|
|
|
|
|
2021-02-22 17:27:13 +05:30
|
|
|
context 'when user is not found' do
|
|
|
|
before do
|
|
|
|
allow(User).to receive(:find).and_return(nil)
|
|
|
|
end
|
|
|
|
|
|
|
|
it { is_expected.to have_gitlab_http_status(:unauthorized) }
|
|
|
|
end
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
|
|
|
|
shared_examples 'not found when disabled' do
|
|
|
|
context 'feature disabled' do
|
|
|
|
before do
|
|
|
|
disable_dependency_proxy
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns 404' do
|
|
|
|
subject
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:not_found)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(Gitlab.config.dependency_proxy)
|
|
|
|
.to receive(:enabled).and_return(true)
|
|
|
|
|
|
|
|
allow_next_instance_of(DependencyProxy::RequestTokenService) do |instance|
|
|
|
|
allow(instance).to receive(:execute).and_return(token_response)
|
|
|
|
end
|
2021-02-22 17:27:13 +05:30
|
|
|
|
|
|
|
request.headers['HTTP_AUTHORIZATION'] = token_header
|
2021-01-29 00:20:46 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
describe 'GET #manifest' do
|
2021-02-22 17:27:13 +05:30
|
|
|
let_it_be(:manifest) { create(:dependency_proxy_manifest) }
|
2021-04-29 21:17:54 +05:30
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
let(:pull_response) { { status: :success, manifest: manifest, from_cache: false } }
|
2021-01-29 00:20:46 +05:30
|
|
|
|
|
|
|
before do
|
2021-02-22 17:27:13 +05:30
|
|
|
allow_next_instance_of(DependencyProxy::FindOrCreateManifestService) do |instance|
|
2021-01-29 00:20:46 +05:30
|
|
|
allow(instance).to receive(:execute).and_return(pull_response)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
subject { get_manifest }
|
|
|
|
|
|
|
|
context 'feature enabled' do
|
|
|
|
before do
|
|
|
|
enable_dependency_proxy
|
|
|
|
end
|
|
|
|
|
2021-02-22 17:27:13 +05:30
|
|
|
it_behaves_like 'without a token'
|
|
|
|
it_behaves_like 'without permission'
|
|
|
|
it_behaves_like 'feature flag disabled with private group'
|
|
|
|
|
2021-01-29 00:20:46 +05:30
|
|
|
context 'remote token request fails' do
|
|
|
|
let(:token_response) do
|
|
|
|
{
|
|
|
|
status: :error,
|
|
|
|
http_status: 503,
|
|
|
|
message: 'Service Unavailable'
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
before do
|
|
|
|
group.add_guest(user)
|
|
|
|
end
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
it 'proxies status from the remote token request', :aggregate_failures do
|
2021-01-29 00:20:46 +05:30
|
|
|
subject
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:service_unavailable)
|
|
|
|
expect(response.body).to eq('Service Unavailable')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'remote manifest request fails' do
|
|
|
|
let(:pull_response) do
|
|
|
|
{
|
|
|
|
status: :error,
|
|
|
|
http_status: 400,
|
|
|
|
message: ''
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
before do
|
|
|
|
group.add_guest(user)
|
|
|
|
end
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
it 'proxies status from the remote manifest request', :aggregate_failures do
|
2021-01-29 00:20:46 +05:30
|
|
|
subject
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:bad_request)
|
|
|
|
expect(response.body).to be_empty
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
context 'a valid user' do
|
|
|
|
before do
|
|
|
|
group.add_guest(user)
|
|
|
|
end
|
2021-02-22 17:27:13 +05:30
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
it_behaves_like 'a successful manifest pull'
|
|
|
|
it_behaves_like 'a package tracking event', described_class.name, 'pull_manifest'
|
|
|
|
|
|
|
|
context 'with a cache entry' do
|
|
|
|
let(:pull_response) { { status: :success, manifest: manifest, from_cache: true } }
|
|
|
|
|
|
|
|
it_behaves_like 'returning response status', :success
|
|
|
|
it_behaves_like 'a package tracking event', described_class.name, 'pull_manifest_from_cache'
|
|
|
|
end
|
2021-02-22 17:27:13 +05:30
|
|
|
end
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
context 'a valid deploy token' do
|
|
|
|
let_it_be(:user) { create(:deploy_token, :dependency_proxy_scopes, :group) }
|
|
|
|
let_it_be(:group_deploy_token) { create(:group_deploy_token, deploy_token: user, group: group) }
|
|
|
|
|
|
|
|
it_behaves_like 'a successful manifest pull'
|
|
|
|
|
|
|
|
context 'pulling from a subgroup' do
|
|
|
|
let_it_be_with_reload(:parent_group) { create(:group) }
|
|
|
|
let_it_be_with_reload(:group) { create(:group, parent: parent_group) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
parent_group.create_dependency_proxy_setting!(enabled: true)
|
|
|
|
group_deploy_token.update_column(:group_id, parent_group.id)
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
it_behaves_like 'a successful manifest pull'
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like 'not found when disabled'
|
|
|
|
|
|
|
|
def get_manifest
|
|
|
|
get :manifest, params: { group_id: group.to_param, image: 'alpine', tag: '3.9.2' }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
describe 'GET #blob' do
|
2021-02-22 17:27:13 +05:30
|
|
|
let_it_be(:blob) { create(:dependency_proxy_blob) }
|
2021-04-29 21:17:54 +05:30
|
|
|
|
2021-01-29 00:20:46 +05:30
|
|
|
let(:blob_sha) { blob.file_name.sub('.gz', '') }
|
2021-10-27 15:23:28 +05:30
|
|
|
let(:blob_response) { { status: :success, blob: blob, from_cache: false } }
|
2021-01-29 00:20:46 +05:30
|
|
|
|
|
|
|
before do
|
|
|
|
allow_next_instance_of(DependencyProxy::FindOrCreateBlobService) do |instance|
|
|
|
|
allow(instance).to receive(:execute).and_return(blob_response)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
subject { get_blob }
|
|
|
|
|
|
|
|
context 'feature enabled' do
|
|
|
|
before do
|
|
|
|
enable_dependency_proxy
|
|
|
|
end
|
|
|
|
|
2021-02-22 17:27:13 +05:30
|
|
|
it_behaves_like 'without a token'
|
|
|
|
it_behaves_like 'without permission'
|
|
|
|
it_behaves_like 'feature flag disabled with private group'
|
|
|
|
|
2021-01-29 00:20:46 +05:30
|
|
|
context 'remote blob request fails' do
|
|
|
|
let(:blob_response) do
|
|
|
|
{
|
|
|
|
status: :error,
|
|
|
|
http_status: 400,
|
|
|
|
message: ''
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
before do
|
|
|
|
group.add_guest(user)
|
|
|
|
end
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
it 'proxies status from the remote blob request', :aggregate_failures do
|
2021-01-29 00:20:46 +05:30
|
|
|
subject
|
|
|
|
|
|
|
|
expect(response).to have_gitlab_http_status(:bad_request)
|
|
|
|
expect(response.body).to be_empty
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
context 'a valid user' do
|
|
|
|
before do
|
|
|
|
group.add_guest(user)
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
it_behaves_like 'a successful blob pull'
|
|
|
|
it_behaves_like 'a package tracking event', described_class.name, 'pull_blob'
|
|
|
|
|
|
|
|
context 'with a cache entry' do
|
|
|
|
let(:blob_response) { { status: :success, blob: blob, from_cache: true } }
|
|
|
|
|
|
|
|
it_behaves_like 'returning response status', :success
|
|
|
|
it_behaves_like 'a package tracking event', described_class.name, 'pull_blob_from_cache'
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
end
|
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
context 'a valid deploy token' do
|
|
|
|
let_it_be(:user) { create(:deploy_token, :group, :dependency_proxy_scopes) }
|
|
|
|
let_it_be(:group_deploy_token) { create(:group_deploy_token, deploy_token: user, group: group) }
|
|
|
|
|
|
|
|
it_behaves_like 'a successful blob pull'
|
2021-01-29 00:20:46 +05:30
|
|
|
|
2021-10-27 15:23:28 +05:30
|
|
|
context 'pulling from a subgroup' do
|
|
|
|
let_it_be_with_reload(:parent_group) { create(:group) }
|
|
|
|
let_it_be_with_reload(:group) { create(:group, parent: parent_group) }
|
|
|
|
|
|
|
|
before do
|
|
|
|
parent_group.create_dependency_proxy_setting!(enabled: true)
|
|
|
|
group_deploy_token.update_column(:group_id, parent_group.id)
|
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like 'a successful blob pull'
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it_behaves_like 'not found when disabled'
|
|
|
|
|
|
|
|
def get_blob
|
|
|
|
get :blob, params: { group_id: group.to_param, image: 'alpine', sha: blob_sha }
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def enable_dependency_proxy
|
|
|
|
group.create_dependency_proxy_setting!(enabled: true)
|
|
|
|
end
|
|
|
|
|
|
|
|
def disable_dependency_proxy
|
|
|
|
group.create_dependency_proxy_setting!(enabled: false)
|
|
|
|
end
|
|
|
|
end
|