Commit 869875fe authored by Piotr Stankowski's avatar Piotr Stankowski

Allow squashing in MRs with single commit

This change:
- Allows checking "Squash commits" in MRs with 1 commit for
  projects with allowed/encouraged squash.
- Performs squash when merging MR for projects with enforced
  squash in MRs with 1 commit.
- Skips performing squash only when there's 1 commit in MR and
  squash commit message is the same as existing commit message.
  When no message is passed, it's compared to default squash
  commit message.

Changelog: changed
parent 3f51bfe0
...@@ -287,7 +287,7 @@ export default { ...@@ -287,7 +287,7 @@ export default {
return false; return false;
} }
return enableSquashBeforeMerge && this.commitsCount > 1; return enableSquashBeforeMerge;
}, },
shouldShowMergeControls() { shouldShowMergeControls() {
if (this.glFeatures.restructuredMrWidget) { if (this.glFeatures.restructuredMrWidget) {
......
...@@ -5,7 +5,7 @@ module MergeRequests ...@@ -5,7 +5,7 @@ module MergeRequests
def execute def execute
# If performing a squash would result in no change, then # If performing a squash would result in no change, then
# immediately return a success message without performing a squash # immediately return a success message without performing a squash
if merge_request.commits_count < 2 && message.nil? if merge_request.commits_count == 1 && message == merge_request.first_commit.safe_message
return success(squash_sha: merge_request.diff_head_sha) return success(squash_sha: merge_request.diff_head_sha)
end end
...@@ -17,7 +17,7 @@ module MergeRequests ...@@ -17,7 +17,7 @@ module MergeRequests
private private
def squash! def squash!
squash_sha = repository.squash(current_user, merge_request, message || merge_request.default_squash_commit_message) squash_sha = repository.squash(current_user, merge_request, message)
success(squash_sha: squash_sha) success(squash_sha: squash_sha)
rescue StandardError => e rescue StandardError => e
...@@ -39,7 +39,7 @@ module MergeRequests ...@@ -39,7 +39,7 @@ module MergeRequests
end end
def message def message
params[:squash_commit_message].presence params[:squash_commit_message].presence || merge_request.default_squash_commit_message
end end
end end
end end
...@@ -22,7 +22,7 @@ RSpec.describe 'User squashes a merge request', :js do ...@@ -22,7 +22,7 @@ RSpec.describe 'User squashes a merge request', :js do
committer_name: user.name) committer_name: user.name)
merge_commit = an_object_having_attributes(sha: a_string_matching(/\h{40}/), merge_commit = an_object_having_attributes(sha: a_string_matching(/\h{40}/),
message: a_string_starting_with("Merge branch 'csv' into 'master'"), message: a_string_starting_with("Merge branch '#{source_branch}' into 'master'"),
author_name: user.name, author_name: user.name,
committer_name: user.name) committer_name: user.name)
...@@ -57,34 +57,34 @@ RSpec.describe 'User squashes a merge request', :js do ...@@ -57,34 +57,34 @@ RSpec.describe 'User squashes a merge request', :js do
end end
context 'when the MR has only one commit' do context 'when the MR has only one commit' do
let(:source_branch) { 'master' } let(:source_branch) { 'feature' }
let(:target_branch) { 'branch-merged' } let(:target_branch) { 'master' }
let(:protected_source_branch) { true }
let(:source_sha) { project.commit(source_branch).sha } let(:source_sha) { project.commit(source_branch).sha }
let(:target_sha) { project.commit(target_branch).sha } let(:target_sha) { project.commit(target_branch).sha }
before do before do
merge_request = create(:merge_request, source_project: project, target_project: project, source_branch: source_branch, target_branch: target_branch, squash: true) visit project_new_merge_request_path(project, merge_request: { target_branch: target_branch, source_branch: source_branch })
check 'merge_request[squash]'
visit project_merge_request_path(project, merge_request) click_on 'Create merge request'
wait_for_requests
end end
it 'accepts the merge request without issuing a squash request', :sidekiq_inline do context 'when squash message differs from existing commit message' do
expect_next_instance_of(Gitlab::GitalyClient::OperationService) do |instance| before do
expect(instance).not_to receive(:user_squash) accept_mr
end end
expect(project.repository.ancestor?(source_branch, target_branch)).to be_falsey include_examples 'squash'
expect(page).not_to have_field('squash') end
context 'when squash message is the same as existing commit message' do
before do
click_button("Modify commit messages")
fill_in('Squash commit message', with: project.commit(source_branch).safe_message)
accept_mr accept_mr
end
expect(page).to have_content('Merged') include_examples 'no squash'
latest_target_commits = project.repository.commits_between(source_sha, target_sha).map(&:raw)
expect(latest_target_commits.count).to eq(1)
expect(project.repository.ancestor?(source_branch, target_branch)).to be_truthy
end end
end end
......
...@@ -503,10 +503,10 @@ describe('ReadyToMerge', () => { ...@@ -503,10 +503,10 @@ describe('ReadyToMerge', () => {
expect(findCheckboxElement().exists()).toBeFalsy(); expect(findCheckboxElement().exists()).toBeFalsy();
}); });
it('should not be rendered when there is only 1 commit', () => { it('should be rendered when there is only 1 commit', () => {
createComponent({ mr: { commitsCount: 1, enableSquashBeforeMerge: true } }); createComponent({ mr: { commitsCount: 1, enableSquashBeforeMerge: true } });
expect(findCheckboxElement().exists()).toBeFalsy(); expect(findCheckboxElement().exists()).toBe(true);
}); });
describe('squash options', () => { describe('squash options', () => {
......
...@@ -150,7 +150,10 @@ RSpec.describe MergeRequests::MergeToRefService do ...@@ -150,7 +150,10 @@ RSpec.describe MergeRequests::MergeToRefService do
merge_request.update!(squash: true) merge_request.update!(squash: true)
end end
it_behaves_like 'MergeService for target ref' it_behaves_like 'successfully merges to ref with merge method' do
let(:first_parent_ref) { 'refs/heads/master' }
let(:target_ref) { merge_request.merge_ref_path }
end
it 'does not squash before merging' do it 'does not squash before merging' do
expect(MergeRequests::SquashService).not_to receive(:new) expect(MergeRequests::SquashService).not_to receive(:new)
......
...@@ -55,20 +55,28 @@ RSpec.describe MergeRequests::SquashService do ...@@ -55,20 +55,28 @@ RSpec.describe MergeRequests::SquashService do
expect(merge_request).to receive(:commits_count).at_least(:once).and_return(1) expect(merge_request).to receive(:commits_count).at_least(:once).and_return(1)
end end
it 'will skip performing the squash, as the outcome would be the same' do it 'will still perform the squash' do
expect(merge_request.target_project.repository).not_to receive(:squash) expect(merge_request.target_project.repository).to receive(:squash).and_return('sha')
service.execute service.execute
end end
it 'will still perform the squash when a custom squash commit message has been provided' do context 'when squash message matches commit message' do
service = described_class.new(project: project, current_user: user, params: { merge_request: merge_request, squash_commit_message: 'A custom commit message' }) let(:service) { described_class.new(project: project, current_user: user, params: { merge_request: merge_request, squash_commit_message: merge_request.first_commit.safe_message }) }
expect(merge_request.target_project.repository).to receive(:squash).and_return('sha') it 'returns that commit SHA' do
result = service.execute
expect(result).to match(status: :success, squash_sha: merge_request.diff_head_sha)
end
it 'does not perform any git actions' do
expect(repository).not_to receive(:squash)
service.execute service.execute
end end
end end
end
context 'the squashed commit' do context 'the squashed commit' do
let(:squash_sha) { service.execute[:squash_sha] } let(:squash_sha) { service.execute[:squash_sha] }
...@@ -113,17 +121,7 @@ RSpec.describe MergeRequests::SquashService do ...@@ -113,17 +121,7 @@ RSpec.describe MergeRequests::SquashService do
context 'when there is only one commit in the merge request' do context 'when there is only one commit in the merge request' do
let(:merge_request) { merge_request_with_one_commit } let(:merge_request) { merge_request_with_one_commit }
it 'returns that commit SHA' do include_examples 'the squash succeeds'
result = service.execute
expect(result).to match(status: :success, squash_sha: merge_request.diff_head_sha)
end
it 'does not perform any git actions' do
expect(repository).not_to receive(:popen)
service.execute
end
end end
context 'when squashing only new files' do context 'when squashing only new files' 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