2019-07-31 22:56:46 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
require 'spec_helper'
|
|
|
|
|
2023-05-27 22:25:52 +05:30
|
|
|
RSpec.describe Branches::CreateService, :use_clean_rails_redis_caching, feature_category: :source_code_management do
|
2020-01-01 13:55:28 +05:30
|
|
|
subject(:service) { described_class.new(project, user) }
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
let_it_be(:project) { create(:project_empty_repo) }
|
|
|
|
let_it_be(:user) { create(:user) }
|
|
|
|
|
2022-08-27 11:52:29 +05:30
|
|
|
describe '#bulk_create' do
|
|
|
|
subject { service.bulk_create(branches) }
|
|
|
|
|
|
|
|
let_it_be(:project) { create(:project, :custom_repo, files: { 'foo/a.txt' => 'foo' }) }
|
|
|
|
|
|
|
|
let(:branches) { { 'branch' => 'master', 'another_branch' => 'master' } }
|
|
|
|
|
|
|
|
it 'creates two branches' do
|
|
|
|
expect(subject[:status]).to eq(:success)
|
|
|
|
expect(subject[:branches].map(&:name)).to match_array(%w[branch another_branch])
|
|
|
|
|
|
|
|
expect(project.repository.branch_exists?('branch')).to be_truthy
|
|
|
|
expect(project.repository.branch_exists?('another_branch')).to be_truthy
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when branches are empty' do
|
|
|
|
let(:branches) { {} }
|
|
|
|
|
|
|
|
it 'is successful' do
|
|
|
|
expect(subject[:status]).to eq(:success)
|
|
|
|
expect(subject[:branches]).to eq([])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when incorrect reference is provided' do
|
|
|
|
let(:branches) { { 'new-feature' => 'unknown' } }
|
|
|
|
|
|
|
|
before do
|
|
|
|
allow(project.repository).to receive(:add_branch).and_return(false)
|
|
|
|
end
|
|
|
|
|
|
|
|
it 'returns an error with a reference name' do
|
|
|
|
err_msg = 'Failed to create branch \'new-feature\': invalid reference name \'unknown\''
|
|
|
|
|
|
|
|
expect(subject[:status]).to eq(:error)
|
|
|
|
expect(subject[:message]).to match_array([err_msg])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when branch already exists' do
|
|
|
|
let(:branches) { { 'master' => 'master' } }
|
|
|
|
|
|
|
|
it 'returns an error' do
|
|
|
|
expect(subject[:status]).to eq(:error)
|
|
|
|
expect(subject[:message]).to match_array(['Branch already exists'])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when PreReceiveError exception' do
|
|
|
|
let(:branches) { { 'error' => 'master' } }
|
|
|
|
|
|
|
|
it 'logs and returns an error if there is a PreReceiveError exception' do
|
|
|
|
error_message = 'pre receive error'
|
|
|
|
raw_message = "GitLab: #{error_message}"
|
|
|
|
pre_receive_error = Gitlab::Git::PreReceiveError.new(raw_message)
|
|
|
|
|
|
|
|
allow(project.repository).to receive(:add_branch).and_raise(pre_receive_error)
|
|
|
|
|
|
|
|
expect(Gitlab::ErrorTracking).to receive(:log_exception).with(
|
|
|
|
pre_receive_error,
|
|
|
|
pre_receive_message: raw_message,
|
|
|
|
branch_name: 'error',
|
|
|
|
ref: 'master'
|
|
|
|
)
|
|
|
|
|
|
|
|
expect(subject[:status]).to eq(:error)
|
|
|
|
expect(subject[:message]).to match_array([error_message])
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when multiple errors occur' do
|
|
|
|
let(:branches) { { 'master' => 'master', '' => 'master', 'failed_branch' => 'master' } }
|
|
|
|
|
|
|
|
it 'returns all errors' do
|
|
|
|
allow(project.repository).to receive(:add_branch).with(
|
|
|
|
user,
|
|
|
|
'failed_branch',
|
|
|
|
'master',
|
|
|
|
expire_cache: false
|
|
|
|
).and_return(false)
|
|
|
|
|
|
|
|
expect(subject[:status]).to eq(:error)
|
|
|
|
expect(subject[:message]).to match_array(
|
|
|
|
[
|
|
|
|
'Branch already exists',
|
|
|
|
'Branch name is invalid',
|
|
|
|
"Failed to create branch 'failed_branch': invalid reference name 'master'"
|
|
|
|
]
|
|
|
|
)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'without N+1 for Redis cache' do
|
|
|
|
let(:branches) { { 'branch1' => 'master', 'branch2' => 'master', 'branch3' => 'master' } }
|
|
|
|
|
|
|
|
it 'does not trigger Redis recreation' do
|
|
|
|
project.repository.expire_branches_cache
|
|
|
|
|
|
|
|
control = RedisCommands::Recorder.new(pattern: ':branch_names:') { subject }
|
|
|
|
|
2023-05-27 22:25:52 +05:30
|
|
|
expect(control).not_to exceed_redis_command_calls_limit(:sadd, 1)
|
2022-08-27 11:52:29 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'without N+1 branch cache expiration' do
|
|
|
|
let(:branches) { { 'branch_1' => 'master', 'branch_2' => 'master', 'branch_3' => 'master' } }
|
|
|
|
|
|
|
|
it 'triggers branch cache expiration only once' do
|
|
|
|
expect(project.repository).to receive(:expire_branches_cache).once
|
|
|
|
|
|
|
|
subject
|
|
|
|
end
|
|
|
|
|
|
|
|
context 'when branches were not added' do
|
|
|
|
let(:branches) { { 'master' => 'master' } }
|
|
|
|
|
|
|
|
it 'does not trigger branch expiration' do
|
|
|
|
expect(project.repository).not_to receive(:expire_branches_cache)
|
|
|
|
|
|
|
|
subject
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
describe '#execute' do
|
|
|
|
context 'when repository is empty' do
|
|
|
|
it 'creates master branch' do
|
2022-08-27 11:52:29 +05:30
|
|
|
result = service.execute('my-feature', 'master')
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2022-08-27 11:52:29 +05:30
|
|
|
expect(result[:status]).to eq(:success)
|
|
|
|
expect(result[:branch].name).to eq('my-feature')
|
2017-08-17 22:00:37 +05:30
|
|
|
expect(project.repository.branch_exists?('master')).to be_truthy
|
|
|
|
end
|
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
it 'creates another-feature branch' do
|
|
|
|
service.execute('another-feature', 'master')
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
expect(project.repository.branch_exists?('another-feature')).to be_truthy
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
end
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
context 'when branch already exists' do
|
|
|
|
it 'returns an error' do
|
|
|
|
result = service.execute('master', 'master')
|
|
|
|
|
|
|
|
expect(result[:status]).to eq(:error)
|
|
|
|
expect(result[:message]).to eq('Branch already exists')
|
|
|
|
end
|
|
|
|
end
|
2019-12-26 22:10:19 +05:30
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
context 'when incorrect reference is provided' do
|
2019-12-26 22:10:19 +05:30
|
|
|
before do
|
|
|
|
allow(project.repository).to receive(:add_branch).and_return(false)
|
|
|
|
end
|
|
|
|
|
2020-05-24 23:13:21 +05:30
|
|
|
it 'returns an error with a reference name' do
|
2021-09-30 23:02:18 +05:30
|
|
|
err_msg = 'Failed to create branch \'new-feature\': invalid reference name \'unknown\''
|
2020-05-24 23:13:21 +05:30
|
|
|
result = service.execute('new-feature', 'unknown')
|
2019-12-26 22:10:19 +05:30
|
|
|
|
|
|
|
expect(result[:status]).to eq(:error)
|
2021-09-30 23:02:18 +05:30
|
|
|
expect(result[:message]).to eq(err_msg)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-10-24 23:57:45 +05:30
|
|
|
it 'logs and returns an error if there is a PreReceiveError exception' do
|
|
|
|
error_message = 'pre receive error'
|
|
|
|
raw_message = "GitLab: #{error_message}"
|
|
|
|
pre_receive_error = Gitlab::Git::PreReceiveError.new(raw_message)
|
|
|
|
|
|
|
|
allow(project.repository).to receive(:add_branch).and_raise(pre_receive_error)
|
|
|
|
|
2022-04-04 11:22:00 +05:30
|
|
|
expect(Gitlab::ErrorTracking).to receive(:log_exception).with(
|
2020-10-24 23:57:45 +05:30
|
|
|
pre_receive_error,
|
|
|
|
pre_receive_message: raw_message,
|
|
|
|
branch_name: 'new-feature',
|
|
|
|
ref: 'unknown'
|
|
|
|
)
|
|
|
|
|
|
|
|
result = service.execute('new-feature', 'unknown')
|
|
|
|
|
|
|
|
expect(result[:status]).to eq(:error)
|
|
|
|
expect(result[:message]).to eq(error_message)
|
|
|
|
end
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
end
|