204 lines
6.1 KiB
Ruby
204 lines
6.1 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'spec_helper'
|
|
|
|
RSpec.describe Suggestions::CreateService do
|
|
let(:project_with_repo) { create(:project, :repository) }
|
|
let(:merge_request) do
|
|
create(:merge_request, source_project: project_with_repo,
|
|
target_project: project_with_repo)
|
|
end
|
|
|
|
def build_position(args = {})
|
|
default_args = { old_path: "files/ruby/popen.rb",
|
|
new_path: "files/ruby/popen.rb",
|
|
old_line: nil,
|
|
new_line: 14,
|
|
diff_refs: merge_request.diff_refs }
|
|
|
|
Gitlab::Diff::Position.new(default_args.merge(args))
|
|
end
|
|
|
|
let(:position) { build_position }
|
|
|
|
let(:markdown) do
|
|
<<-MARKDOWN.strip_heredoc
|
|
```suggestion
|
|
foo
|
|
bar
|
|
```
|
|
|
|
```
|
|
nothing
|
|
```
|
|
|
|
```suggestion
|
|
xpto
|
|
baz
|
|
```
|
|
|
|
```thing
|
|
this is not a suggestion, it's a thing
|
|
```
|
|
|
|
```suggestion:-3+2
|
|
# multi-line suggestion 1
|
|
```
|
|
|
|
```suggestion:-5
|
|
# multi-line suggestion 1
|
|
```
|
|
MARKDOWN
|
|
end
|
|
|
|
subject { described_class.new(note) }
|
|
|
|
shared_examples_for 'service not tracking add suggestion event' do
|
|
it 'does not track add suggestion event' do
|
|
expect(Gitlab::UsageDataCounters::MergeRequestActivityUniqueCounter)
|
|
.not_to receive(:track_add_suggestion_action)
|
|
|
|
subject.execute
|
|
end
|
|
end
|
|
|
|
describe '#execute' do
|
|
context 'should not try to parse suggestions' do
|
|
context 'when not a diff note for merge requests' do
|
|
let(:note) do
|
|
create(:diff_note_on_commit, project: project_with_repo,
|
|
note: markdown)
|
|
end
|
|
|
|
it 'does not try to parse suggestions' do
|
|
expect(Gitlab::Diff::SuggestionsParser).not_to receive(:parse)
|
|
|
|
subject.execute
|
|
end
|
|
|
|
it_behaves_like 'service not tracking add suggestion event'
|
|
end
|
|
|
|
context 'when diff note is not for text' do
|
|
let(:note) do
|
|
create(:diff_note_on_merge_request, project: project_with_repo,
|
|
noteable: merge_request,
|
|
position: position,
|
|
note: markdown)
|
|
end
|
|
|
|
before do
|
|
allow(note).to receive(:on_text?) { false }
|
|
end
|
|
|
|
it 'does not try to parse suggestions' do
|
|
expect(Gitlab::Diff::SuggestionsParser).not_to receive(:parse)
|
|
|
|
subject.execute
|
|
end
|
|
|
|
it_behaves_like 'service not tracking add suggestion event'
|
|
end
|
|
end
|
|
|
|
context 'when diff file is not found' do
|
|
let(:note) do
|
|
create(:diff_note_on_merge_request, project: project_with_repo,
|
|
noteable: merge_request,
|
|
position: position,
|
|
note: markdown)
|
|
end
|
|
|
|
before do
|
|
expect_next_instance_of(DiffNote) do |diff_note|
|
|
expect(diff_note).to receive(:latest_diff_file).once { nil }
|
|
end
|
|
end
|
|
|
|
it 'creates no suggestion' do
|
|
expect { subject.execute }.not_to change(Suggestion, :count)
|
|
end
|
|
|
|
it_behaves_like 'service not tracking add suggestion event'
|
|
end
|
|
|
|
context 'should create suggestions' do
|
|
let(:note) do
|
|
create(:diff_note_on_merge_request, project: project_with_repo,
|
|
noteable: merge_request,
|
|
position: position,
|
|
note: markdown)
|
|
end
|
|
|
|
let(:expected_suggestions) do
|
|
Gitlab::Diff::SuggestionsParser.parse(markdown,
|
|
project: note.project,
|
|
position: note.position)
|
|
end
|
|
|
|
it 'persists suggestion records' do
|
|
expect { subject.execute }.to change { note.suggestions.count }
|
|
.from(0).to(expected_suggestions.size)
|
|
end
|
|
|
|
it 'persists suggestions data correctly' do
|
|
subject.execute
|
|
|
|
suggestions = note.suggestions.order(:relative_order)
|
|
|
|
suggestions.zip(expected_suggestions) do |suggestion, expected_suggestion|
|
|
expected_data = expected_suggestion.to_hash
|
|
|
|
expect(suggestion.from_content).to eq(expected_data[:from_content])
|
|
expect(suggestion.to_content).to eq(expected_data[:to_content])
|
|
expect(suggestion.lines_above).to eq(expected_data[:lines_above])
|
|
expect(suggestion.lines_below).to eq(expected_data[:lines_below])
|
|
end
|
|
end
|
|
|
|
it 'tracks add suggestion event' do
|
|
expect(Gitlab::UsageDataCounters::MergeRequestActivityUniqueCounter)
|
|
.to receive(:track_add_suggestion_action)
|
|
.with(user: note.author)
|
|
|
|
subject.execute
|
|
end
|
|
|
|
context 'outdated position note' do
|
|
let!(:outdated_diff) { merge_request.merge_request_diff }
|
|
let!(:latest_diff) { merge_request.create_merge_request_diff }
|
|
let(:outdated_position) { build_position(diff_refs: outdated_diff.diff_refs) }
|
|
let(:position) { build_position(diff_refs: latest_diff.diff_refs) }
|
|
|
|
it 'uses the correct position when creating the suggestion' do
|
|
expect(Gitlab::Diff::SuggestionsParser).to receive(:parse)
|
|
.with(note.note, project: note.project, position: note.position)
|
|
.and_call_original
|
|
|
|
subject.execute
|
|
end
|
|
end
|
|
|
|
context 'when a patch removes an empty line' do
|
|
let(:markdown) do
|
|
<<-MARKDOWN.strip_heredoc
|
|
```suggestion
|
|
```
|
|
MARKDOWN
|
|
end
|
|
|
|
let(:position) { build_position(new_line: 13) }
|
|
|
|
it 'creates an appliable suggestion' do
|
|
subject.execute
|
|
|
|
suggestion = note.suggestions.last
|
|
|
|
expect(suggestion).to be_appliable
|
|
expect(suggestion.from_content).to eq("\n")
|
|
expect(suggestion.to_content).to eq("")
|
|
end
|
|
end
|
|
end
|
|
end
|
|
end
|