mirror of
https://github.com/gitlabhq/gitlabhq.git
synced 2025-07-25 16:00:50 +00:00
904 lines
30 KiB
Ruby
904 lines
30 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'spec_helper'
|
|
|
|
RSpec.describe DiffHelper, feature_category: :code_review_workflow do
|
|
include RepoHelpers
|
|
|
|
let(:project) { create(:project, :repository) }
|
|
let(:repository) { project.repository }
|
|
let(:commit) { project.commit(sample_commit.id) }
|
|
let(:diffs) { commit.raw_diffs }
|
|
let(:diff) { diffs.first }
|
|
let(:diff_refs) { commit.diff_refs }
|
|
let(:diff_file) { Gitlab::Diff::File.new(diff, diff_refs: diff_refs, repository: repository) }
|
|
|
|
before do
|
|
allow(helper).to receive(:current_user).and_return(project.owner)
|
|
end
|
|
|
|
describe 'diff_view' do
|
|
it 'uses the view param over the cookie' do
|
|
controller.params[:view] = 'parallel'
|
|
helper.request.cookies[:diff_view] = 'inline'
|
|
|
|
expect(helper.diff_view).to eq :parallel
|
|
end
|
|
|
|
it 'returns the default value when the view param is invalid' do
|
|
controller.params[:view] = 'invalid'
|
|
|
|
expect(helper.diff_view).to eq :inline
|
|
end
|
|
|
|
it 'returns a valid value when cookie is set' do
|
|
helper.request.cookies[:diff_view] = 'parallel'
|
|
|
|
expect(helper.diff_view).to eq :parallel
|
|
end
|
|
|
|
it 'returns the default value when cookie is invalid' do
|
|
helper.request.cookies[:diff_view] = 'invalid'
|
|
|
|
expect(helper.diff_view).to eq :inline
|
|
end
|
|
|
|
it 'returns the default value when cookie is nil' do
|
|
expect(helper.request.cookies).to be_empty
|
|
|
|
expect(helper.diff_view).to eq :inline
|
|
end
|
|
end
|
|
|
|
describe 'diff_options' do
|
|
it 'returns no collapse false' do
|
|
expect(helper.diff_options).to include(expanded: false)
|
|
end
|
|
|
|
it 'returns no collapse true if expanded' do
|
|
allow(controller).to receive(:params) { { expanded: true } }
|
|
expect(helper.diff_options).to include(expanded: true)
|
|
end
|
|
|
|
context 'when action name is diff_for_path' do
|
|
before do
|
|
allow(controller).to receive(:action_name) { 'diff_for_path' }
|
|
end
|
|
|
|
it 'returns expanded true' do
|
|
expect(helper.diff_options).to include(expanded: true)
|
|
end
|
|
|
|
it 'returns paths if param old path' do
|
|
allow(controller).to receive(:params) { { old_path: 'lib/wadus.rb' } }
|
|
expect(helper.diff_options[:paths]).to include('lib/wadus.rb')
|
|
end
|
|
|
|
it 'returns paths if param new path' do
|
|
allow(controller).to receive(:params) { { new_path: 'lib/wadus.rb' } }
|
|
expect(helper.diff_options[:paths]).to include('lib/wadus.rb')
|
|
end
|
|
|
|
it 'does not set max_patch_bytes_for_file_extension' do
|
|
expect(helper.diff_options[:max_patch_bytes_for_file_extension]).to be_nil
|
|
end
|
|
|
|
context 'when file_identifier include .ipynb' do
|
|
before do
|
|
allow(controller).to receive(:params) { { file_identifier: 'something.ipynb' } }
|
|
end
|
|
|
|
it 'sets max_patch_bytes_for_file_extension' do
|
|
expect(helper.diff_options[:max_patch_bytes_for_file_extension]).to eq({ '.ipynb' => 1.megabyte })
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#diff_line_content' do
|
|
context 'when the line is empty' do
|
|
it 'returns a non breaking space' do
|
|
expect(diff_line_content(nil)).to eq(' ')
|
|
end
|
|
|
|
it 'returns an HTML-safe string' do
|
|
expect(diff_line_content(nil)).to be_html_safe
|
|
end
|
|
end
|
|
|
|
context 'when the line is not empty' do
|
|
context 'when the line starts with +, -, or a space' do
|
|
it 'strips the first character' do
|
|
expect(diff_line_content('+new line')).to eq('new line')
|
|
expect(diff_line_content('-new line')).to eq('new line')
|
|
expect(diff_line_content(' new line')).to eq('new line')
|
|
end
|
|
|
|
context 'when the line is HTML-safe' do
|
|
it 'returns an HTML-safe string' do
|
|
expect(diff_line_content('+new line'.html_safe)).to be_html_safe
|
|
expect(diff_line_content('-new line'.html_safe)).to be_html_safe
|
|
expect(diff_line_content(' new line'.html_safe)).to be_html_safe
|
|
end
|
|
end
|
|
|
|
context 'when the line is not HTML-safe' do
|
|
it 'returns a non-HTML-safe string' do
|
|
expect(diff_line_content('+new line')).not_to be_html_safe
|
|
expect(diff_line_content('-new line')).not_to be_html_safe
|
|
expect(diff_line_content(' new line')).not_to be_html_safe
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'when the line does not start with a +, -, or a space' do
|
|
it 'returns the string' do
|
|
expect(diff_line_content('@@ -6,12 +6,18 @@ module Popen')).to eq('@@ -6,12 +6,18 @@ module Popen')
|
|
end
|
|
|
|
context 'when the line is HTML-safe' do
|
|
it 'returns an HTML-safe string' do
|
|
expect(diff_line_content('@@ -6,12 +6,18 @@ module Popen'.html_safe)).to be_html_safe
|
|
end
|
|
end
|
|
|
|
context 'when the line is not HTML-safe' do
|
|
it 'returns a non-HTML-safe string' do
|
|
expect(diff_line_content('@@ -6,12 +6,18 @@ module Popen')).not_to be_html_safe
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "#diff_link_number" do
|
|
using RSpec::Parameterized::TableSyntax
|
|
|
|
let(:line) do
|
|
double(:line, type: line_type)
|
|
end
|
|
|
|
# This helper is used to generate the line numbers on the
|
|
# diff lines. It essentially just returns a blank string
|
|
# on the old/new lines. The following table tests all the
|
|
# possible permutations for clarity.
|
|
|
|
where(:line_type, :match, :line_number, :expected_return_value) do
|
|
"new" | "new" | 1 | " "
|
|
"new" | "old" | 2 | 2
|
|
"old" | "new" | 3 | 3
|
|
"old" | "old" | 4 | " "
|
|
"new-nonewline" | "new" | 5 | 5
|
|
"new-nonewline" | "old" | 6 | 6
|
|
"old-nonewline" | "new" | 7 | 7
|
|
"old-nonewline" | "old" | 8 | 8
|
|
"match" | "new" | 9 | 9
|
|
"match" | "old" | 10 | 10
|
|
end
|
|
|
|
with_them do
|
|
it "returns the expected value" do
|
|
expect(helper.diff_link_number(line.type, match, line_number)).to eq(expected_return_value)
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#parallel_diff_btn' do
|
|
let(:params) do
|
|
ActionController::Parameters.new({
|
|
controller: "projects/commit",
|
|
action: "show",
|
|
namespace_id: "foo",
|
|
project_id: "bar",
|
|
id: commit.sha,
|
|
view: 'parallel'
|
|
}).permit!
|
|
end
|
|
|
|
before do
|
|
allow(helper).to receive(:params).and_return(params)
|
|
end
|
|
|
|
subject(:diff_btn) { helper.parallel_diff_btn }
|
|
|
|
it 'renders button' do
|
|
expect(diff_btn).to include('Side-by-side')
|
|
expect(diff_btn).to include('gl-button btn btn-md btn-default selected')
|
|
end
|
|
end
|
|
|
|
describe "#mark_inline_diffs" do
|
|
let(:old_line) { %(abc 'def') }
|
|
let(:new_line) { %(abc "def") }
|
|
|
|
it "returns strings with marked inline diffs" do
|
|
marked_old_line, marked_new_line = mark_inline_diffs(old_line, new_line)
|
|
|
|
expect(marked_old_line).to eq(%q(abc <span class="idiff left deletion">'</span>def<span class="idiff right deletion">'</span>))
|
|
expect(marked_old_line).to be_html_safe
|
|
expect(marked_new_line).to eq(%q(abc <span class="idiff left addition">"</span>def<span class="idiff right addition">"</span>))
|
|
expect(marked_new_line).to be_html_safe
|
|
end
|
|
|
|
context 'when given HTML' do
|
|
it 'sanitizes it' do
|
|
old_line = %(test.txt)
|
|
new_line = %{<img src=x onerror=alert(document.domain)>}
|
|
|
|
marked_old_line, marked_new_line = mark_inline_diffs(old_line, new_line)
|
|
|
|
expect(marked_old_line).to eq(%q(<span class="idiff left right deletion">test.txt</span>))
|
|
expect(marked_old_line).to be_html_safe
|
|
expect(marked_new_line).to eq(%q{<span class="idiff left right addition"><img src=x onerror=alert(document.domain)></span>})
|
|
expect(marked_new_line).to be_html_safe
|
|
end
|
|
|
|
it 'sanitizes the entire line, not just the changes' do
|
|
old_line = %{<img src=x onerror=alert(document.domain)>}
|
|
new_line = %{<img src=y onerror=alert(document.domain)>}
|
|
|
|
marked_old_line, marked_new_line = mark_inline_diffs(old_line, new_line)
|
|
|
|
expect(marked_old_line).to eq(%q{<img src=<span class="idiff left right deletion">x</span> onerror=alert(document.domain)>})
|
|
expect(marked_old_line).to be_html_safe
|
|
expect(marked_new_line).to eq(%q{<img src=<span class="idiff left right addition">y</span> onerror=alert(document.domain)>})
|
|
expect(marked_new_line).to be_html_safe
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#parallel_diff_discussions' do
|
|
let(:discussion) { { 'abc_3_3' => 'comment' } }
|
|
let(:diff_file) { double(line_code: 'abc_3_3') }
|
|
|
|
before do
|
|
helper.instance_variable_set(:@grouped_diff_discussions, discussion)
|
|
end
|
|
|
|
it 'does not put comments on nonewline lines' do
|
|
left = Gitlab::Diff::Line.new('\\nonewline', 'old-nonewline', 3, 3, 3)
|
|
right = Gitlab::Diff::Line.new('\\nonewline', 'new-nonewline', 3, 3, 3)
|
|
|
|
result = helper.parallel_diff_discussions(left, right, diff_file)
|
|
|
|
expect(result).to eq([nil, nil])
|
|
end
|
|
|
|
it 'puts comments on added lines' do
|
|
left = Gitlab::Diff::Line.new('\\nonewline', 'old-nonewline', 3, 3, 3)
|
|
right = Gitlab::Diff::Line.new('new line', 'new', 3, 3, 3)
|
|
|
|
result = helper.parallel_diff_discussions(left, right, diff_file)
|
|
|
|
expect(result).to eq([nil, 'comment'])
|
|
end
|
|
|
|
it 'puts comments on unchanged lines' do
|
|
left = Gitlab::Diff::Line.new('unchanged line', nil, 3, 3, 3)
|
|
right = Gitlab::Diff::Line.new('unchanged line', nil, 3, 3, 3)
|
|
|
|
result = helper.parallel_diff_discussions(left, right, diff_file)
|
|
|
|
expect(result).to eq(['comment', nil])
|
|
end
|
|
end
|
|
|
|
describe "#diff_match_line" do
|
|
let(:old_pos) { 40 }
|
|
let(:new_pos) { 50 }
|
|
let(:text) { 'some_text' }
|
|
|
|
it "generates foldable top match line for inline view with empty text by default" do
|
|
output = diff_match_line old_pos, new_pos
|
|
|
|
expect(output).to be_html_safe
|
|
expect(output).to have_css "td:nth-child(1):not(.js-unfold-bottom).diff-line-num.unfold.js-unfold.old_line[data-linenumber='#{old_pos}']", text: '...'
|
|
expect(output).to have_css "td:nth-child(2):not(.js-unfold-bottom).diff-line-num.unfold.js-unfold.new_line[data-linenumber='#{new_pos}']", text: '...'
|
|
expect(output).to have_css 'td:nth-child(3):not(.parallel).line_content.match', text: ''
|
|
end
|
|
|
|
it "allows to define text and bottom option" do
|
|
output = diff_match_line old_pos, new_pos, text: text, bottom: true
|
|
|
|
expect(output).to be_html_safe
|
|
expect(output).to have_css "td:nth-child(1).diff-line-num.unfold.js-unfold.js-unfold-bottom.old_line[data-linenumber='#{old_pos}']", text: '...'
|
|
expect(output).to have_css "td:nth-child(2).diff-line-num.unfold.js-unfold.js-unfold-bottom.new_line[data-linenumber='#{new_pos}']", text: '...'
|
|
expect(output).to have_css 'td:nth-child(3):not(.parallel).line_content.match', text: text
|
|
end
|
|
|
|
it "generates match line for parallel view" do
|
|
output = diff_match_line old_pos, new_pos, text: text, view: :parallel
|
|
|
|
expect(output).to be_html_safe
|
|
expect(output).to have_css "td:nth-child(1):not(.js-unfold-bottom).diff-line-num.unfold.js-unfold.old_line[data-linenumber='#{old_pos}']", text: '...'
|
|
expect(output).to have_css 'td:nth-child(2).line_content.match.parallel', text: text
|
|
expect(output).to have_css "td:nth-child(3):not(.js-unfold-bottom).diff-line-num.unfold.js-unfold.new_line[data-linenumber='#{new_pos}']", text: '...'
|
|
expect(output).to have_css 'td:nth-child(4).line_content.match.parallel', text: text
|
|
end
|
|
|
|
it "allows to generate only left match line for parallel view" do
|
|
output = diff_match_line old_pos, nil, text: text, view: :parallel
|
|
|
|
expect(output).to be_html_safe
|
|
expect(output).to have_css "td:nth-child(1):not(.js-unfold-bottom).diff-line-num.unfold.js-unfold.old_line[data-linenumber='#{old_pos}']", text: '...'
|
|
expect(output).to have_css 'td:nth-child(2).line_content.match.parallel', text: text
|
|
expect(output).not_to have_css 'td:nth-child(3)'
|
|
end
|
|
|
|
it "allows to generate only right match line for parallel view" do
|
|
output = diff_match_line nil, new_pos, text: text, view: :parallel
|
|
|
|
expect(output).to be_html_safe
|
|
expect(output).to have_css "td:nth-child(1):not(.js-unfold-bottom).diff-line-num.unfold.js-unfold.new_line[data-linenumber='#{new_pos}']", text: '...'
|
|
expect(output).to have_css 'td:nth-child(2).line_content.match.parallel', text: text
|
|
expect(output).not_to have_css 'td:nth-child(3)'
|
|
end
|
|
end
|
|
|
|
describe "#diff_nomappinginraw_line" do
|
|
using RSpec::Parameterized::TableSyntax
|
|
|
|
let(:line) { double("line") }
|
|
let(:line_type) { 'line_type' }
|
|
|
|
before do
|
|
allow(line).to receive(:rich_text).and_return('line_text')
|
|
allow(line).to receive(:type).and_return(line_type)
|
|
end
|
|
|
|
it 'generates only single line num' do
|
|
output = diff_nomappinginraw_line(line, ['line_num_1'], nil, ['line_content'])
|
|
|
|
expect(output).to be_html_safe
|
|
expect(output).to have_css 'td:nth-child(1).line_num_1'
|
|
expect(output).to have_css 'td:nth-child(2).line_content', text: 'line_text'
|
|
expect(output).not_to have_css 'td:nth-child(3)'
|
|
end
|
|
|
|
it 'generates only both line nums' do
|
|
output = diff_nomappinginraw_line(line, ['line_num_1'], ['line_num_2'], ['line_content'])
|
|
|
|
expect(output).to be_html_safe
|
|
expect(output).to have_css 'td:nth-child(1).line_num_1'
|
|
expect(output).to have_css 'td:nth-child(2).line_num_2'
|
|
expect(output).to have_css 'td:nth-child(3).line_content', text: 'line_text'
|
|
end
|
|
|
|
where(:line_type, :added_class) do
|
|
'old-nomappinginraw' | '.old'
|
|
'new-nomappinginraw' | '.new'
|
|
'unchanged-nomappinginraw' | ''
|
|
end
|
|
|
|
with_them do
|
|
it "appends the correct class" do
|
|
output = diff_nomappinginraw_line(line, ['line_num_1'], ['line_num_2'], ['line_content'])
|
|
|
|
expect(output).to be_html_safe
|
|
expect(output).to have_css 'td:nth-child(1).line_num_1' + added_class
|
|
expect(output).to have_css 'td:nth-child(2).line_num_2' + added_class
|
|
expect(output).to have_css 'td:nth-child(3).line_content' + added_class, text: 'line_text'
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#render_overflow_warning?' do
|
|
using RSpec::Parameterized::TableSyntax
|
|
|
|
let(:diffs_collection) { instance_double(Gitlab::Diff::FileCollection::MergeRequestDiff, raw_diff_files: diff_files, overflow?: false) }
|
|
let(:diff_files) { Gitlab::Git::DiffCollection.new(files) }
|
|
let(:safe_file) { { too_large: false, diff: '' } }
|
|
let(:large_file) { { too_large: true, diff: '' } }
|
|
let(:files) { [safe_file, safe_file] }
|
|
|
|
context 'when no limits are hit' do
|
|
before do
|
|
allow(diff_files).to receive(:overflow_max_bytes?).and_return(false)
|
|
allow(diff_files).to receive(:overflow_max_files?).and_return(false)
|
|
allow(diff_files).to receive(:overflow_max_lines?).and_return(false)
|
|
allow(diff_files).to receive(:collapsed_safe_bytes?).and_return(false)
|
|
allow(diff_files).to receive(:collapsed_safe_files?).and_return(false)
|
|
allow(diff_files).to receive(:collapsed_safe_lines?).and_return(false)
|
|
end
|
|
|
|
it 'returns false and does not log any overflow events' do
|
|
expect(Gitlab::Metrics).not_to receive(:add_event).with(:diffs_overflow_collection_limits)
|
|
expect(Gitlab::Metrics).not_to receive(:add_event).with(:diffs_overflow_single_file_limits)
|
|
expect(Gitlab::Metrics).not_to receive(:add_event).with(:diffs_overflow_max_bytes_limits)
|
|
expect(Gitlab::Metrics).not_to receive(:add_event).with(:diffs_overflow_max_files_limits)
|
|
expect(Gitlab::Metrics).not_to receive(:add_event).with(:diffs_overflow_max_lines_limits)
|
|
expect(Gitlab::Metrics).not_to receive(:add_event).with(:diffs_overflow_collapsed_bytes_limits)
|
|
expect(Gitlab::Metrics).not_to receive(:add_event).with(:diffs_overflow_collapsed_files_limits)
|
|
expect(Gitlab::Metrics).not_to receive(:add_event).with(:diffs_overflow_collapsed_lines_limits)
|
|
|
|
expect(render_overflow_warning?(diffs_collection)).to be false
|
|
end
|
|
end
|
|
|
|
where(:overflow_method, :event_name) do
|
|
:overflow_max_bytes? | :diffs_overflow_max_bytes_limits
|
|
:overflow_max_files? | :diffs_overflow_max_files_limits
|
|
:overflow_max_lines? | :diffs_overflow_max_lines_limits
|
|
:collapsed_safe_bytes? | :diffs_overflow_collapsed_bytes_limits
|
|
:collapsed_safe_files? | :diffs_overflow_collapsed_files_limits
|
|
:collapsed_safe_lines? | :diffs_overflow_collapsed_lines_limits
|
|
end
|
|
|
|
with_them do
|
|
it 'returns false and only logs the correct collection overflow event' do
|
|
allow(diff_files).to receive(overflow_method).and_return(true)
|
|
expect(Gitlab::Metrics).to receive(:add_event).with(event_name).once
|
|
expect(render_overflow_warning?(diffs_collection)).to be false
|
|
end
|
|
end
|
|
|
|
context 'when the file collection has an overflow' do
|
|
before do
|
|
allow(diffs_collection).to receive(:overflow?).and_return(true)
|
|
end
|
|
|
|
it 'returns true and only logs all the correct collection overflow event' do
|
|
expect(Gitlab::Metrics).to receive(:add_event).with(:diffs_overflow_collection_limits).once
|
|
|
|
expect(render_overflow_warning?(diffs_collection)).to be true
|
|
end
|
|
end
|
|
|
|
context 'when two individual files are too big' do
|
|
let(:files) { [safe_file, large_file, large_file] }
|
|
|
|
it 'returns false and only logs single file overflow events' do
|
|
expect(Gitlab::Metrics).to receive(:add_event).with(:diffs_overflow_single_file_limits).exactly(:once)
|
|
expect(Gitlab::Metrics).not_to receive(:add_event).with(:diffs_overflow_collection_limits)
|
|
|
|
expect(render_overflow_warning?(diffs_collection)).to be false
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#diff_file_html_data' do
|
|
let(:project) { build(:project) }
|
|
let(:path) { 'path/to/file' }
|
|
let(:sha) { '1234567890' }
|
|
|
|
subject do
|
|
helper.diff_file_html_data(project, path, sha)
|
|
end
|
|
|
|
it 'returns data for project files' do
|
|
expect(subject).to include(blob_diff_path: helper.project_blob_diff_path(project, "#{sha}/#{path}"))
|
|
end
|
|
end
|
|
|
|
describe "#collapsed_diff_url" do
|
|
let(:params) do
|
|
{
|
|
controller: "projects/commit",
|
|
action: "show",
|
|
namespace_id: "foo",
|
|
project_id: "bar",
|
|
id: commit.sha
|
|
}
|
|
end
|
|
|
|
subject { helper.collapsed_diff_url(diff_file) }
|
|
|
|
it "returns a valid URL" do
|
|
allow(helper).to receive(:safe_params).and_return(params)
|
|
|
|
expect(subject).to match(%r{foo/bar/-/commit/#{commit.sha}/diff_for_path})
|
|
end
|
|
end
|
|
|
|
describe '#params_with_whitespace' do
|
|
context 'with safe params' do
|
|
before do
|
|
controller.params[:protocol] = 'HACKED!'
|
|
controller.params[:host] = 'HACKED!'
|
|
end
|
|
|
|
it "filters with safe_params" do
|
|
expect(helper.params_with_whitespace.to_h).to eq({ 'w' => 1 })
|
|
end
|
|
end
|
|
|
|
context 'with hide_whitespace' do
|
|
before do
|
|
allow(helper).to receive(:safe_params).and_return({ key: 'value' })
|
|
end
|
|
|
|
context 'when w parameter is present' do
|
|
it 'removes the parameter when w = 1' do
|
|
controller.params[:w] = '1'
|
|
expect(helper.params_with_whitespace).to eq({ key: 'value' })
|
|
end
|
|
|
|
it 'adds w = 1 when w is any other value' do
|
|
controller.params[:w] = '0'
|
|
expect(helper.params_with_whitespace).to eq({ key: 'value', w: 1 })
|
|
end
|
|
end
|
|
|
|
context 'when w parameter is not present' do
|
|
context 'when user is not logged in' do
|
|
before do
|
|
allow(helper).to receive(:current_user).and_return(nil)
|
|
end
|
|
|
|
it 'removes the w parameter' do
|
|
expect(helper.params_with_whitespace).to eq({ key: 'value' })
|
|
end
|
|
end
|
|
|
|
context 'when user has show_whitespace_in_diffs = false' do
|
|
before do
|
|
allow(helper.current_user).to receive(:show_whitespace_in_diffs).and_return(false)
|
|
end
|
|
|
|
it 'removes the w parameter' do
|
|
expect(helper.params_with_whitespace).to eq({ key: 'value' })
|
|
end
|
|
end
|
|
|
|
context 'when user has show_whitespace_in_diffs = true' do
|
|
before do
|
|
allow(helper.current_user).to receive(:show_whitespace_in_diffs).and_return(true)
|
|
end
|
|
|
|
it 'adds w = 1 parameter' do
|
|
expect(helper.params_with_whitespace).to eq({ key: 'value', w: 1 })
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "#render_fork_suggestion" do
|
|
subject { helper.render_fork_suggestion }
|
|
|
|
before do
|
|
allow(helper).to receive(:current_user).and_return(current_user)
|
|
end
|
|
|
|
context "user signed in" do
|
|
let(:current_user) { build(:user) }
|
|
|
|
it "renders the partial" do
|
|
expect(helper).to receive(:render).with(partial: "projects/fork_suggestion").exactly(:once)
|
|
|
|
5.times { subject }
|
|
end
|
|
end
|
|
|
|
context "guest" do
|
|
let(:current_user) { nil }
|
|
|
|
it { is_expected.to be_nil }
|
|
end
|
|
end
|
|
|
|
describe '#conflicts' do
|
|
let(:merge_request) do
|
|
instance_double(
|
|
MergeRequest,
|
|
cannot_be_merged?: cannot_be_merged?,
|
|
source_branch_exists?: source_branch_exists?,
|
|
target_branch_exists?: target_branch_exists?
|
|
)
|
|
end
|
|
|
|
let(:cannot_be_merged?) { true }
|
|
let(:source_branch_exists?) { true }
|
|
let(:target_branch_exists?) { true }
|
|
let(:can_be_resolved_in_ui?) { true }
|
|
let(:allow_tree_conflicts) { false }
|
|
let(:files) { [instance_double(Gitlab::Conflict::File, path: 'a')] }
|
|
let(:exception) { nil }
|
|
|
|
before do
|
|
allow(helper).to receive(:merge_request).and_return(merge_request)
|
|
|
|
allow_next_instance_of(MergeRequests::Conflicts::ListService, merge_request, allow_tree_conflicts: allow_tree_conflicts) do |svc|
|
|
allow(svc).to receive(:can_be_resolved_in_ui?).and_return(can_be_resolved_in_ui?)
|
|
|
|
if exception.present?
|
|
allow(svc).to receive_message_chain(:conflicts, :files).and_raise(exception)
|
|
else
|
|
allow(svc).to receive_message_chain(:conflicts, :files).and_return(files)
|
|
end
|
|
end
|
|
end
|
|
|
|
it 'returns list of conflicts indexed by path' do
|
|
expect(helper.conflicts).to eq('a' => files.first)
|
|
end
|
|
|
|
context 'when merge request can be merged' do
|
|
let(:cannot_be_merged?) { false }
|
|
|
|
it 'returns nil' do
|
|
expect(helper.conflicts).to be_nil
|
|
end
|
|
end
|
|
|
|
context 'when source branch does not exist' do
|
|
let(:source_branch_exists?) { false }
|
|
|
|
it 'returns nil' do
|
|
expect(helper.conflicts).to be_nil
|
|
end
|
|
end
|
|
|
|
context 'when target branch does not exist' do
|
|
let(:target_branch_exists?) { false }
|
|
|
|
it 'returns nil' do
|
|
expect(helper.conflicts).to be_nil
|
|
end
|
|
end
|
|
|
|
context 'when conflicts cannot be resolved in UI' do
|
|
let(:can_be_resolved_in_ui?) { false }
|
|
|
|
it 'returns nil' do
|
|
expect(helper.conflicts).to be_nil
|
|
end
|
|
|
|
context 'when allow_tree_conflicts is true' do
|
|
let(:allow_tree_conflicts) { true }
|
|
|
|
it 'returns list of conflicts' do
|
|
expect(helper.conflicts(allow_tree_conflicts: allow_tree_conflicts)).to eq('a' => files.first)
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'when Gitlab::Git::Conflict::Resolver::ConflictSideMissing exception is raised' do
|
|
let(:exception) { Gitlab::Git::Conflict::Resolver::ConflictSideMissing }
|
|
|
|
it 'returns an empty hash' do
|
|
expect(helper.conflicts).to eq({})
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#show_only_context_commits?' do
|
|
let(:params) { {} }
|
|
let(:merge_request) { build_stubbed(:merge_request) }
|
|
let(:has_no_commits) { true }
|
|
|
|
subject(:result) { helper.show_only_context_commits? }
|
|
|
|
before do
|
|
assign(:merge_request, merge_request)
|
|
allow(helper).to receive(:params).and_return(params)
|
|
allow(merge_request).to receive(:has_no_commits?).and_return(has_no_commits)
|
|
end
|
|
|
|
context 'when only_context_commits param is set to true' do
|
|
let(:params) { { only_context_commits: true } }
|
|
|
|
it { is_expected.to be_truthy }
|
|
|
|
context 'when merge request has commits' do
|
|
let(:has_no_commits) { false }
|
|
|
|
it { is_expected.to be_truthy }
|
|
end
|
|
end
|
|
|
|
context 'when only_context_commits param is set to false' do
|
|
let(:params) { { only_context_commits: false } }
|
|
|
|
it { is_expected.to be_truthy }
|
|
|
|
context 'when merge request has commits' do
|
|
let(:has_no_commits) { false }
|
|
|
|
it { is_expected.to be_falsey }
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#submodule_diff_compare_link' do
|
|
context 'when the diff includes submodule changes' do
|
|
it 'generates a link to compare a diff for a submodule' do
|
|
allow(helper).to receive(:submodule_links).and_return(
|
|
Gitlab::SubmoduleLinks::Urls.new(nil, nil, '/comparison-path')
|
|
)
|
|
|
|
output = helper.submodule_diff_compare_link(diff_file)
|
|
expect(output).to match(%r{href="/comparison-path"})
|
|
expect(output).to match(
|
|
%r{Compare <span class="commit-sha">5b812ff1</span>...<span class="commit-sha">7e3e39eb</span>}
|
|
)
|
|
end
|
|
end
|
|
|
|
context 'when the diff does not include submodule changes' do
|
|
it 'returns an empty string' do
|
|
output = helper.submodule_diff_compare_link(diff_file)
|
|
expect(output).to eq('')
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#conflicts_with_types', :use_clean_rails_redis_caching do
|
|
let(:merge_request) do
|
|
create(
|
|
:merge_request,
|
|
:conflict,
|
|
merge_status: 'cannot_be_merged',
|
|
source_branch_sha: 'abc123',
|
|
target_branch_sha: 'def456'
|
|
)
|
|
end
|
|
|
|
let(:exception) { nil }
|
|
let(:conflict_file) { instance_double(Gitlab::Conflict::File, path: 'a') }
|
|
let(:files) { [conflict_file] }
|
|
|
|
before do
|
|
allow(helper).to receive(:merge_request).and_return(merge_request)
|
|
|
|
allow(conflict_file)
|
|
.to receive(:conflict_type)
|
|
.and_return(:removed_target_renamed_source)
|
|
|
|
allow(conflict_file)
|
|
.to receive(:conflict_type)
|
|
.with(when_renamed: true)
|
|
.and_return(:renamed_same_file)
|
|
|
|
allow_next_instance_of(
|
|
MergeRequests::Conflicts::ListService,
|
|
merge_request,
|
|
allow_tree_conflicts: true,
|
|
skip_content: true
|
|
) do |svc|
|
|
if exception.present?
|
|
allow(svc).to receive_message_chain(:conflicts, :files).and_raise(exception)
|
|
else
|
|
allow(svc).to receive_message_chain(:conflicts, :files).and_return(files)
|
|
end
|
|
end
|
|
end
|
|
|
|
it 'returns list of conflicts indexed by path' do
|
|
expect(helper.conflicts_with_types).to eq(
|
|
'a' => {
|
|
conflict_type: :removed_target_renamed_source,
|
|
conflict_type_when_renamed: :renamed_same_file
|
|
}
|
|
)
|
|
end
|
|
|
|
context 'when merge request can be merged' do
|
|
let(:merge_request) { create(:merge_request, merge_status: 'can_be_merged') }
|
|
|
|
it 'returns nil' do
|
|
expect(helper.conflicts_with_types).to be_nil
|
|
end
|
|
end
|
|
|
|
context 'when source branch does not exist' do
|
|
let(:merge_request) do
|
|
create(
|
|
:merge_request,
|
|
source_branch: 'i-do-no-exist',
|
|
merge_status: 'cannot_be_merged'
|
|
)
|
|
end
|
|
|
|
it 'returns nil' do
|
|
expect(helper.conflicts_with_types).to be_nil
|
|
end
|
|
end
|
|
|
|
context 'when target branch does not exist' do
|
|
let(:merge_request) do
|
|
create(
|
|
:merge_request,
|
|
target_branch: 'i-do-no-exist',
|
|
merge_status: 'cannot_be_merged'
|
|
)
|
|
end
|
|
|
|
it 'returns nil' do
|
|
expect(helper.conflicts_with_types).to be_nil
|
|
end
|
|
end
|
|
|
|
context 'when Gitlab::Git::Conflict::Resolver::ConflictSideMissing exception is raised' do
|
|
let(:exception) { Gitlab::Git::Conflict::Resolver::ConflictSideMissing }
|
|
|
|
it 'returns an empty hash' do
|
|
expect(helper.conflicts_with_types).to eq({})
|
|
end
|
|
end
|
|
|
|
context 'when cached' do
|
|
before do
|
|
helper.conflicts_with_types # Cache the result
|
|
end
|
|
|
|
it 'does not make a call to MergeRequests::Conflicts::ListService' do
|
|
expect(MergeRequests::Conflicts::ListService).not_to receive(:new)
|
|
|
|
expect(helper.conflicts_with_types).to eq(
|
|
'a' => {
|
|
conflict_type: :removed_target_renamed_source,
|
|
conflict_type_when_renamed: :renamed_same_file
|
|
}
|
|
)
|
|
end
|
|
|
|
context 'when source branch SHA changes' do
|
|
before do
|
|
allow(merge_request).to receive(:source_branch_sha).and_return('123abc')
|
|
end
|
|
|
|
it 'calls MergeRequests::Conflicts::ListService' do
|
|
expect(MergeRequests::Conflicts::ListService).to receive(:new)
|
|
|
|
helper.conflicts_with_types
|
|
end
|
|
end
|
|
|
|
context 'when target branch SHA changes' do
|
|
before do
|
|
allow(merge_request).to receive(:target_branch_sha).and_return('456def')
|
|
end
|
|
|
|
it 'calls MergeRequests::Conflicts::ListService' do
|
|
expect(MergeRequests::Conflicts::ListService).to receive(:new)
|
|
|
|
helper.conflicts_with_types
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "#file_heading_id" do
|
|
subject { helper.file_heading_id(diff_file) }
|
|
|
|
it { is_expected.to eq("#{diff_file.file_hash[0..8]}-heading") }
|
|
end
|
|
|
|
describe "#hide_whitespace?" do
|
|
subject { helper.hide_whitespace? }
|
|
|
|
context 'when request has w param set' do
|
|
before do
|
|
allow(controller).to receive(:params) { { w: '1' } }
|
|
end
|
|
|
|
it { is_expected.to be(true) }
|
|
end
|
|
|
|
context 'when user is guest' do
|
|
before do
|
|
allow(helper).to receive(:current_user).and_return(nil)
|
|
end
|
|
|
|
it { is_expected.to be(true) }
|
|
end
|
|
|
|
context 'when user has preference' do
|
|
before do
|
|
allow(helper).to receive_message_chain(:current_user, :show_whitespace_in_diffs).and_return(true)
|
|
end
|
|
|
|
it { is_expected.to be(false) }
|
|
|
|
context 'when request has w param set' do
|
|
before do
|
|
allow(controller).to receive(:params) { { w: '1' } }
|
|
end
|
|
|
|
it { is_expected.to be(true) }
|
|
end
|
|
end
|
|
end
|
|
end
|