Commit df0c54ac authored by Sean McGivern's avatar Sean McGivern

Ensure all traces of squash branch are removed

1. When a branch is protected by a wildcard, it can't be removed by
   anyone. Skip hooks on branch removal after squashing so that the
   branch is always removed.
2. Pushing a branch creates an event, cluttering the activity
   stream. Manually remove this, too.
parent c8074801
...@@ -65,7 +65,7 @@ module MergeRequests ...@@ -65,7 +65,7 @@ module MergeRequests
'push squashed branch' 'push squashed branch'
) )
repository.rm_branch(current_user, temp_branch) remove_branch(temp_branch, squash_sha)
success(squash_sha: squash_sha) success(squash_sha: squash_sha)
rescue GitCommandError rescue GitCommandError
...@@ -108,5 +108,32 @@ module MergeRequests ...@@ -108,5 +108,32 @@ module MergeRequests
protected_branch protected_branch
end end
# If the branch is protected, no-one can remove it, so we have to skip hooks
# in order to remove it. We also don't want a branch creation event left
# hanging around, so we look in the user's last 10 push events for this
# repository and find it from those.
#
def remove_branch(branch, rev)
full_ref = "#{Gitlab::Git::BRANCH_REF_PREFIX}#{branch}"
blank_sha = Gitlab::Git::BLANK_SHA
events = Event.code_push
.where(project: target_project, author: current_user)
.order('created_at DESC')
.limit(10)
repository.before_remove_branch
repository.update_ref!(full_ref, blank_sha, rev)
repository.after_remove_branch
event = events.find do |event|
event.data[:before] == blank_sha &&
event.data[:after] == rev &&
event.data[:ref] == full_ref &&
event.data[:total_commits_count] == 1
end
event.destroy if event
end
end end
end end
...@@ -17,11 +17,36 @@ describe MergeRequests::SquashService do ...@@ -17,11 +17,36 @@ describe MergeRequests::SquashService do
target_branch: 'master', target_project: project) target_branch: 'master', target_project: project)
end end
def stub_git_command(command, &block) def git_command(command)
git_command = a_collection_starting_with([Gitlab.config.git.bin_path, command]) a_collection_starting_with([Gitlab.config.git.bin_path, command])
end
shared_examples 'the squashed commit' do
context 'the squashed commit' do
let(:squash_sha) { service.execute(merge_request)[:squash_sha] }
let(:squash_commit) { project.repository.commit(squash_sha) }
it 'copies the author info and message from the last commit in the source branch' do
diff_head_commit = merge_request.diff_head_commit
allow(service).to receive(:popen).and_call_original expect(squash_commit.author_name).to eq(diff_head_commit.author_name)
allow(service).to receive(:popen).with(git_command, anything, anything, &block) expect(squash_commit.author_email).to eq(diff_head_commit.author_email)
expect(squash_commit.message).to eq(diff_head_commit.message)
end
it 'sets the current user as the committer' do
expect(squash_commit.committer_name).to eq(user.name.chomp('.'))
expect(squash_commit.committer_email).to eq(user.email)
end
it 'has the same diff as the merge request' do
rugged = project.repository.rugged
mr_diff = rugged.diff(merge_request.diff_base_sha, merge_request.diff_head_sha)
squash_diff = rugged.diff(merge_request.diff_start_sha, squash_sha)
expect(squash_diff.patch).to eq(mr_diff.patch)
end
end
end end
describe '#execute' do describe '#execute' do
...@@ -39,20 +64,27 @@ describe MergeRequests::SquashService do ...@@ -39,20 +64,27 @@ describe MergeRequests::SquashService do
end end
end end
# We don't run hooks in tests, so fake this case. This does involve
# duplicating logic from the service itself, but that is worth it to test
# this case.
#
context 'when the chosen branch name is protected with a wildcard' do context 'when the chosen branch name is protected with a wildcard' do
let!(:protected_branch) { create(:protected_branch, :no_one_can_push, name: '*', project: project) } let!(:protected_branch) { create(:protected_branch, :no_one_can_push, name: '*', project: project) }
before do before do
# We don't run hooks in tests, so fake this case. This does involve
# duplicating logic from the service itself, but that is worth it to
# test this case.
user_access = Gitlab::UserAccess.new(user, project: project) user_access = Gitlab::UserAccess.new(user, project: project)
stub_git_command('push') do |cmd| # If the branch is protected, then nobody can remove it, so we need to
# ensure we aren't executing hooks.
allow(GitHooksService).to receive(:new).and_raise(GitHooksService::PreReceiveError)
allow(service).to receive(:popen).and_call_original
allow(service).to receive(:popen).with(git_command('push'), anything, anything).and_wrap_original do |meth, cmd, *args|
ref = cmd.last.split(':').last ref = cmd.last.split(':').last
if user_access.can_push_to_branch?(ref) if user_access.can_push_to_branch?(ref)
['', 0] meth.call(cmd, *args)
else else
['You are not allowed to push code to protected branches on this project', 1] ['You are not allowed to push code to protected branches on this project', 1]
end end
...@@ -62,7 +94,8 @@ describe MergeRequests::SquashService do ...@@ -62,7 +94,8 @@ describe MergeRequests::SquashService do
it 'allows the user to push to that protected branch' do it 'allows the user to push to that protected branch' do
branch_params = a_hash_including(name: a_string_starting_with('temporary-gitlab-squash-branch')) branch_params = a_hash_including(name: a_string_starting_with('temporary-gitlab-squash-branch'))
expect(ProtectedBranches::CreateService).to receive(:new).with(project, user, branch_params) expect(ProtectedBranches::CreateService)
.to receive(:new).with(project, user, branch_params).and_call_original
service.execute(merge_request) service.execute(merge_request)
end end
...@@ -82,6 +115,8 @@ describe MergeRequests::SquashService do ...@@ -82,6 +115,8 @@ describe MergeRequests::SquashService do
expect(protected_branch).to be_persisted expect(protected_branch).to be_persisted
end end
include_examples 'the squashed commit'
end end
context 'when the squash succeeds' do context 'when the squash succeeds' do
...@@ -98,31 +133,11 @@ describe MergeRequests::SquashService do ...@@ -98,31 +133,11 @@ describe MergeRequests::SquashService do
service.execute(merge_request) service.execute(merge_request)
end end
context 'the squashed commit' do it 'does not keep the branch push event' do
let(:squash_sha) { service.execute(merge_request)[:squash_sha] } expect { service.execute(merge_request) }.not_to change { Event.count }
let(:squash_commit) { project.repository.commit(squash_sha) }
it 'copies the author info and message from the last commit in the source branch' do
diff_head_commit = merge_request.diff_head_commit
expect(squash_commit.author_name).to eq(diff_head_commit.author_name)
expect(squash_commit.author_email).to eq(diff_head_commit.author_email)
expect(squash_commit.message).to eq(diff_head_commit.message)
end
it 'sets the current user as the committer' do
expect(squash_commit.committer_name).to eq(user.name.gsub('.', ''))
expect(squash_commit.committer_email).to eq(user.email)
end
it 'has the same diff as the merge request' do
rugged = project.repository.rugged
mr_diff = rugged.diff(merge_request.diff_base_sha, merge_request.diff_head_sha)
squash_diff = rugged.diff(merge_request.diff_start_sha, squash_sha)
expect(squash_diff.patch).to eq(mr_diff.patch)
end
end end
include_examples 'the squashed commit'
end end
stages = { stages = {
...@@ -138,7 +153,8 @@ describe MergeRequests::SquashService do ...@@ -138,7 +153,8 @@ describe MergeRequests::SquashService do
let(:error) { 'A test error' } let(:error) { 'A test error' }
before do before do
stub_git_command(command) { [error, 1] } allow(service).to receive(:popen).and_return(['', 0])
allow(service).to receive(:popen).with(git_command(command), anything, anything).and_return([error, 1])
end end
it 'logs the stage and output' do it 'logs the stage and output' do
......
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