2021-01-29 00:20:46 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
require 'spec_helper'
|
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
RSpec.describe BulkImports::CreateService, feature_category: :importers do
|
2021-01-29 00:20:46 +05:30
|
|
|
let(:user) { create(:user) }
|
|
|
|
let(:credentials) { { url: 'http://gitlab.example', access_token: 'token' } }
|
2023-03-04 22:38:38 +05:30
|
|
|
let(:destination_group) { create(:group, path: 'destination1') }
|
2023-03-17 16:20:25 +05:30
|
|
|
let(:migrate_projects) { true }
|
2023-03-04 22:38:38 +05:30
|
|
|
let_it_be(:parent_group) { create(:group, path: 'parent-group') }
|
2021-01-29 00:20:46 +05:30
|
|
|
let(:params) do
|
|
|
|
[
|
|
|
|
{
|
|
|
|
source_type: 'group_entity',
|
|
|
|
source_full_path: 'full/path/to/group1',
|
2022-08-27 11:52:29 +05:30
|
|
|
destination_slug: 'destination group 1',
|
2023-03-17 16:20:25 +05:30
|
|
|
destination_namespace: 'parent-group',
|
|
|
|
migrate_projects: migrate_projects
|
|
|
|
|
2021-01-29 00:20:46 +05:30
|
|
|
},
|
|
|
|
{
|
|
|
|
source_type: 'group_entity',
|
|
|
|
source_full_path: 'full/path/to/group2',
|
2022-08-27 11:52:29 +05:30
|
|
|
destination_slug: 'destination group 2',
|
2023-03-17 16:20:25 +05:30
|
|
|
destination_namespace: 'parent-group',
|
|
|
|
migrate_projects: migrate_projects
|
2021-01-29 00:20:46 +05:30
|
|
|
},
|
|
|
|
{
|
|
|
|
source_type: 'project_entity',
|
|
|
|
source_full_path: 'full/path/to/project1',
|
2022-08-27 11:52:29 +05:30
|
|
|
destination_slug: 'destination project 1',
|
2023-03-17 16:20:25 +05:30
|
|
|
destination_namespace: 'parent-group',
|
|
|
|
migrate_projects: migrate_projects
|
2021-01-29 00:20:46 +05:30
|
|
|
}
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
subject { described_class.new(user, params, credentials) }
|
|
|
|
|
|
|
|
describe '#execute' do
|
2023-03-17 16:20:25 +05:30
|
|
|
context 'when gitlab version is 15.5 or higher' do
|
|
|
|
let(:source_version) { { version: "15.6.0", enterprise: false } }
|
2021-11-18 22:05:49 +05:30
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
context 'when a BulkImports::Error is raised while validating the instance version' do
|
|
|
|
before do
|
|
|
|
allow_next_instance_of(BulkImports::Clients::HTTP) do |client|
|
|
|
|
allow(client)
|
|
|
|
.to receive(:validate_instance_version!)
|
|
|
|
.and_raise(BulkImports::Error, "This is a BulkImports error.")
|
|
|
|
end
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
it 'rescues the error and raises a ServiceResponse::Error' do
|
|
|
|
result = subject.execute
|
2021-01-29 00:20:46 +05:30
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
expect(result).to be_a(ServiceResponse)
|
|
|
|
expect(result).to be_error
|
|
|
|
expect(result.message).to eq("This is a BulkImports error.")
|
|
|
|
end
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
context 'when required scopes are not present' do
|
|
|
|
it 'returns ServiceResponse with error if token does not have api scope' do
|
|
|
|
stub_request(:get, 'http://gitlab.example/api/v4/version?private_token=token').to_return(status: 404)
|
|
|
|
stub_request(:get, 'http://gitlab.example/api/v4/metadata?private_token=token')
|
|
|
|
.to_return(
|
|
|
|
status: 200,
|
|
|
|
body: source_version.to_json,
|
|
|
|
headers: { 'Content-Type' => 'application/json' }
|
|
|
|
)
|
2021-01-29 00:20:46 +05:30
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
allow_next_instance_of(BulkImports::Clients::HTTP) do |client|
|
|
|
|
allow(client).to receive(:validate_instance_version!).and_raise(BulkImports::Error.scope_validation_failure)
|
|
|
|
end
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
result = subject.execute
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
expect(result).to be_a(ServiceResponse)
|
|
|
|
expect(result).to be_error
|
|
|
|
expect(result.message)
|
|
|
|
.to eq(
|
|
|
|
"Import aborted as the provided personal access token does not have the required 'api' scope or is " \
|
|
|
|
"no longer valid."
|
|
|
|
)
|
|
|
|
end
|
|
|
|
end
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
context 'when token validation succeeds' do
|
|
|
|
before do
|
|
|
|
stub_request(:get, 'http://gitlab.example/api/v4/version?private_token=token').to_return(status: 404)
|
|
|
|
stub_request(:get, 'http://gitlab.example/api/v4/metadata?private_token=token')
|
|
|
|
.to_return(status: 200, body: source_version.to_json, headers: { 'Content-Type' => 'application/json' })
|
|
|
|
stub_request(:get, 'http://gitlab.example/api/v4/personal_access_tokens/self?private_token=token')
|
|
|
|
.to_return(
|
|
|
|
status: 200,
|
|
|
|
body: { 'scopes' => ['api'] }.to_json,
|
|
|
|
headers: { 'Content-Type' => 'application/json' }
|
|
|
|
)
|
|
|
|
end
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
it 'creates bulk import' do
|
|
|
|
parent_group.add_owner(user)
|
|
|
|
expect { subject.execute }.to change { BulkImport.count }.by(1)
|
2021-04-17 20:07:23 +05:30
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
last_bulk_import = BulkImport.last
|
|
|
|
expect(last_bulk_import.user).to eq(user)
|
|
|
|
expect(last_bulk_import.source_version).to eq(source_version[:version])
|
|
|
|
expect(last_bulk_import.user).to eq(user)
|
|
|
|
expect(last_bulk_import.source_enterprise).to eq(false)
|
2023-03-04 22:38:38 +05:30
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
expect_snowplow_event(
|
|
|
|
category: 'BulkImports::CreateService',
|
|
|
|
action: 'create',
|
|
|
|
label: 'bulk_import_group'
|
|
|
|
)
|
2023-03-04 22:38:38 +05:30
|
|
|
|
|
|
|
expect_snowplow_event(
|
|
|
|
category: 'BulkImports::CreateService',
|
|
|
|
action: 'create',
|
|
|
|
label: 'import_access_level',
|
|
|
|
user: user,
|
2023-03-17 16:20:25 +05:30
|
|
|
extra: { user_role: 'Owner', import_type: 'bulk_import_group' }
|
2023-03-04 22:38:38 +05:30
|
|
|
)
|
|
|
|
end
|
2023-03-17 16:20:25 +05:30
|
|
|
|
|
|
|
describe 'projects migration flag' do
|
|
|
|
let(:import) { BulkImport.last }
|
|
|
|
|
|
|
|
context 'when false' do
|
|
|
|
let(:migrate_projects) { false }
|
|
|
|
|
|
|
|
it 'sets false' do
|
|
|
|
subject.execute
|
|
|
|
|
|
|
|
expect(import.entities.pluck(:migrate_projects)).to contain_exactly(false, false, false)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when true' do
|
|
|
|
let(:migrate_projects) { true }
|
|
|
|
|
|
|
|
it 'sets true' do
|
|
|
|
subject.execute
|
|
|
|
|
|
|
|
expect(import.entities.pluck(:migrate_projects)).to contain_exactly(true, true, true)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when nil' do
|
|
|
|
let(:migrate_projects) { nil }
|
|
|
|
|
|
|
|
it 'sets true' do
|
|
|
|
subject.execute
|
|
|
|
|
|
|
|
expect(import.entities.pluck(:migrate_projects)).to contain_exactly(true, true, true)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2023-03-04 22:38:38 +05:30
|
|
|
end
|
2023-03-17 16:20:25 +05:30
|
|
|
end
|
2023-03-04 22:38:38 +05:30
|
|
|
|
2023-03-17 16:20:25 +05:30
|
|
|
context 'when gitlab version is lower than 15.5' do
|
|
|
|
let(:source_version) do
|
|
|
|
Gitlab::VersionInfo.new(::BulkImport::MIN_MAJOR_VERSION,
|
|
|
|
::BulkImport::MIN_MINOR_VERSION_FOR_PROJECT)
|
|
|
|
end
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow_next_instance_of(BulkImports::Clients::HTTP) do |instance|
|
|
|
|
allow(instance).to receive(:instance_version).and_return(source_version)
|
|
|
|
allow(instance).to receive(:instance_enterprise).and_return(false)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'creates bulk import' do
|
|
|
|
parent_group.add_owner(user)
|
|
|
|
expect { subject.execute }.to change { BulkImport.count }.by(1)
|
|
|
|
|
|
|
|
last_bulk_import = BulkImport.last
|
|
|
|
|
|
|
|
expect(last_bulk_import.user).to eq(user)
|
|
|
|
expect(last_bulk_import.source_version).to eq(source_version.to_s)
|
|
|
|
expect(last_bulk_import.user).to eq(user)
|
|
|
|
expect(last_bulk_import.source_enterprise).to eq(false)
|
|
|
|
|
|
|
|
expect_snowplow_event(
|
|
|
|
category: 'BulkImports::CreateService',
|
|
|
|
action: 'create',
|
|
|
|
label: 'bulk_import_group'
|
|
|
|
)
|
|
|
|
|
|
|
|
expect_snowplow_event(
|
|
|
|
category: 'BulkImports::CreateService',
|
|
|
|
action: 'create',
|
|
|
|
label: 'import_access_level',
|
|
|
|
user: user,
|
|
|
|
extra: { user_role: 'Owner', import_type: 'bulk_import_group' }
|
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'creates bulk import entities' do
|
|
|
|
expect { subject.execute }.to change { BulkImports::Entity.count }.by(3)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'creates bulk import configuration' do
|
|
|
|
expect { subject.execute }.to change { BulkImports::Configuration.count }.by(1)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'enqueues BulkImportWorker' do
|
|
|
|
expect(BulkImportWorker).to receive(:perform_async)
|
|
|
|
|
|
|
|
subject.execute
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns success ServiceResponse' do
|
|
|
|
result = subject.execute
|
|
|
|
|
|
|
|
expect(result).to be_a(ServiceResponse)
|
|
|
|
expect(result).to be_success
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns ServiceResponse with error if validation fails' do
|
|
|
|
params[0][:source_full_path] = nil
|
|
|
|
|
|
|
|
result = subject.execute
|
|
|
|
|
|
|
|
expect(result).to be_a(ServiceResponse)
|
|
|
|
expect(result).to be_error
|
|
|
|
expect(result.message).to eq("Validation failed: Source full path can't be blank")
|
|
|
|
end
|
|
|
|
|
|
|
|
describe '#user-role' do
|
|
|
|
context 'when there is a parent_namespace and the user is a member' do
|
|
|
|
let(:group2) { create(:group, path: 'destination200', source_id: parent_group.id ) }
|
|
|
|
let(:params) do
|
|
|
|
[
|
|
|
|
{
|
|
|
|
source_type: 'group_entity',
|
|
|
|
source_full_path: 'full/path/to/group1',
|
|
|
|
destination_slug: 'destination200',
|
|
|
|
destination_namespace: 'parent-group'
|
|
|
|
}
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'defines access_level from parent namespace membership' do
|
|
|
|
parent_group.add_guest(user)
|
|
|
|
subject.execute
|
|
|
|
|
|
|
|
expect_snowplow_event(
|
|
|
|
category: 'BulkImports::CreateService',
|
|
|
|
action: 'create',
|
|
|
|
label: 'import_access_level',
|
|
|
|
user: user,
|
|
|
|
extra: { user_role: 'Guest', import_type: 'bulk_import_group' }
|
|
|
|
)
|
|
|
|
end
|
2023-03-04 22:38:38 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
it 'defines access_level as not a member' do
|
|
|
|
subject.execute
|
|
|
|
expect_snowplow_event(
|
|
|
|
category: 'BulkImports::CreateService',
|
|
|
|
action: 'create',
|
|
|
|
label: 'import_access_level',
|
|
|
|
user: user,
|
|
|
|
extra: { user_role: 'Not a member', import_type: 'bulk_import_group' }
|
|
|
|
)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when there is a destination_namespace but no parent_namespace' do
|
|
|
|
let(:params) do
|
|
|
|
[
|
|
|
|
{
|
|
|
|
source_type: 'group_entity',
|
|
|
|
source_full_path: 'full/path/to/group1',
|
|
|
|
destination_slug: 'destination-group-1',
|
|
|
|
destination_namespace: 'destination1'
|
|
|
|
}
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'defines access_level from destination_namespace' do
|
|
|
|
destination_group.add_developer(user)
|
|
|
|
subject.execute
|
|
|
|
|
|
|
|
expect_snowplow_event(
|
|
|
|
category: 'BulkImports::CreateService',
|
|
|
|
action: 'create',
|
|
|
|
label: 'import_access_level',
|
|
|
|
user: user,
|
|
|
|
extra: { user_role: 'Developer', import_type: 'bulk_import_group' }
|
|
|
|
)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when there is no destination_namespace or parent_namespace' do
|
|
|
|
let(:params) do
|
|
|
|
[
|
|
|
|
{
|
|
|
|
source_type: 'group_entity',
|
|
|
|
source_full_path: 'full/path/to/group1',
|
|
|
|
destination_slug: 'destinationational mcdestiny',
|
|
|
|
destination_namespace: 'destinational-mcdestiny'
|
|
|
|
}
|
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'defines access_level as owner' do
|
|
|
|
subject.execute
|
|
|
|
|
|
|
|
expect_snowplow_event(
|
|
|
|
category: 'BulkImports::CreateService',
|
|
|
|
action: 'create',
|
|
|
|
label: 'import_access_level',
|
|
|
|
user: user,
|
|
|
|
extra: { user_role: 'Owner', import_type: 'bulk_import_group' }
|
|
|
|
)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2021-01-29 00:20:46 +05:30
|
|
|
end
|
|
|
|
end
|