Commit 2db9d635 authored by Kassio Borges's avatar Kassio Borges

GithubImport: Enable users to import _modern_ diff notes

To enable users to import _modern_ diff notes features, like
suggestions, the diff note must be created with `DiffNote` model,
instead of `LegacyDiffNote`.

This commit enables the GithubImporter to use `DiffNotes` when the
`:github_importer_replace_legacy_diffnote` feature flag is enabled.

With the intent to get better performance, and given that Github doesn't
support `quick_actions`, the quick actions parsing is disabled for notes
created by the importer.

Related to: https://gitlab.com/gitlab-org/gitlab/-/issues/340624

MR: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/71765
parent fbbcaddf
# frozen_string_literal: true
module Import
module Github
module Notes
class CreateService < ::Notes::CreateService
# Github does not have support to quick actions in notes (like /assign)
# Therefore, when importing notes we skip the quick actions processing
def quick_actions_supported?(_note)
false
end
end
end
end
end
...@@ -43,7 +43,7 @@ module Notes ...@@ -43,7 +43,7 @@ module Notes
private private
def execute_quick_actions(note) def execute_quick_actions(note)
return yield(false) unless quick_actions_service.supported?(note) return yield(false) unless quick_actions_supported?(note)
content, update_params, message = quick_actions_service.execute(note, quick_action_options) content, update_params, message = quick_actions_service.execute(note, quick_action_options)
only_commands = content.empty? only_commands = content.empty?
...@@ -54,6 +54,10 @@ module Notes ...@@ -54,6 +54,10 @@ module Notes
do_commands(note, update_params, message, only_commands) do_commands(note, update_params, message, only_commands)
end end
def quick_actions_supported?(note)
quick_actions_service.supported?(note)
end
def quick_actions_service def quick_actions_service
@quick_actions_service ||= QuickActionsService.new(project, current_user) @quick_actions_service ||= QuickActionsService.new(project, current_user)
end end
......
---
name: github_importer_use_diff_note_with_suggestions
introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/71765
rollout_issue_url: https://gitlab.com/gitlab-org/gitlab/-/issues/344309
milestone: '14.5'
type: development
group: group::import
default_enabled: false
...@@ -4,58 +4,132 @@ module Gitlab ...@@ -4,58 +4,132 @@ module Gitlab
module GithubImport module GithubImport
module Importer module Importer
class DiffNoteImporter class DiffNoteImporter
attr_reader :note, :project, :client, :user_finder # note - An instance of `Gitlab::GithubImport::Representation::DiffNote`
# project - An instance of `Project`
# note - An instance of `Gitlab::GithubImport::Representation::DiffNote`. # client - An instance of `Gitlab::GithubImport::Client`
# project - An instance of `Project`.
# client - An instance of `Gitlab::GithubImport::Client`.
def initialize(note, project, client) def initialize(note, project, client)
@note = note @note = note
@project = project @project = project
@client = client @client = client
@user_finder = GithubImport::UserFinder.new(project, client)
end end
def execute def execute
return unless (mr_id = find_merge_request_id) return if merge_request_id.blank?
author_id, author_found = user_finder.author_id_for(note) build_author_attributes
note_body = MarkdownText.format(note.note, note.author, author_found) # Diff notes with suggestions are imported with DiffNote, which is
# slower to import than LegacyDiffNote. Importing DiffNote is slower
# because it cannot use the BulkImporting strategy, which skips
# callbacks and validations. For this reason, notes that don't have
# suggestions are still imported with LegacyDiffNote
if import_with_diff_note?
import_with_diff_note
else
import_with_legacy_diff_note
end
rescue ActiveRecord::InvalidForeignKey => e
# It's possible the project and the issue have been deleted since
# scheduling this job. In this case we'll just skip creating the note
Logger.info(
message: e.message,
github_identifiers: note.github_identifiers
)
end
attributes = { private
discussion_id: Discussion.discussion_id(note),
attr_reader :note, :project, :client, :author_id, :author_found
def import_with_diff_note?
note.contains_suggestion? && use_diff_note_with_suggestions_enabled?
end
def use_diff_note_with_suggestions_enabled?
Feature.enabled?(
:github_importer_use_diff_note_with_suggestions,
default_enabled: :yaml
)
end
def build_author_attributes
@author_id, @author_found = user_finder.author_id_for(note)
end
# rubocop:disable Gitlab/BulkInsert
def import_with_legacy_diff_note
log_diff_note_creation('LegacyDiffNote')
# It's possible that during an import we'll insert tens of thousands
# of diff notes. If we were to use the Note/LegacyDiffNote model here
# we'd also have to run additional queries for both validations and
# callbacks, putting a lot of pressure on the database.
#
# To work around this we're using bulk_insert with a single row. This
# allows us to efficiently insert data (even if it's just 1 row)
# without having to use all sorts of hacks to disable callbacks.
Gitlab::Database.main.bulk_insert(LegacyDiffNote.table_name, [{
noteable_type: 'MergeRequest', noteable_type: 'MergeRequest',
noteable_id: mr_id, system: false,
type: 'LegacyDiffNote',
discussion_id: Discussion.discussion_id(note),
noteable_id: merge_request_id,
project_id: project.id, project_id: project.id,
author_id: author_id, author_id: author_id,
note: note_body, note: note_body,
system: false,
commit_id: note.original_commit_id, commit_id: note.original_commit_id,
line_code: note.line_code, line_code: note.line_code,
type: 'LegacyDiffNote',
created_at: note.created_at, created_at: note.created_at,
updated_at: note.updated_at, updated_at: note.updated_at,
st_diff: note.diff_hash.to_yaml st_diff: note.diff_hash.to_yaml
} }])
end
# rubocop:enabled Gitlab/BulkInsert
# It's possible that during an import we'll insert tens of thousands def import_with_diff_note
# of diff notes. If we were to use the Note/LegacyDiffNote model here log_diff_note_creation('DiffNote')
# we'd also have to run additional queries for both validations and
# callbacks, putting a lot of pressure on the database. ::Import::Github::Notes::CreateService.new(project, author, {
# noteable_type: 'MergeRequest',
# To work around this we're using bulk_insert with a single row. This system: false,
# allows us to efficiently insert data (even if it's just 1 row) type: 'DiffNote',
# without having to use all sorts of hacks to disable callbacks. noteable_id: merge_request_id,
Gitlab::Database.main.bulk_insert(LegacyDiffNote.table_name, [attributes]) # rubocop:disable Gitlab/BulkInsert project_id: project.id,
rescue ActiveRecord::InvalidForeignKey note: note_body,
# It's possible the project and the issue have been deleted since commit_id: note.original_commit_id,
# scheduling this job. In this case we'll just skip creating the note. created_at: note.created_at,
updated_at: note.updated_at,
position: note.diff_position(merge_request)
}).execute
end
def note_body
@note_body ||= MarkdownText.format(note.note, note.author, author_found)
end
def author
@author ||= User.find(author_id)
end
def merge_request
@merge_request ||= MergeRequest.find(merge_request_id)
end end
# Returns the ID of the merge request this note belongs to. # Returns the ID of the merge request this note belongs to.
def find_merge_request_id def merge_request_id
GithubImport::IssuableFinder.new(project, note).database_id @merge_request_id ||= GithubImport::IssuableFinder.new(project, note).database_id
end
def user_finder
@user_finder ||= GithubImport::UserFinder.new(project, client)
end
def log_diff_note_creation(model)
Logger.info(
project_id: project.id,
importer: self.class.name,
github_identifiers: note.github_identifiers,
model: model
)
end end
end end
end end
......
...@@ -10,8 +10,9 @@ module Gitlab ...@@ -10,8 +10,9 @@ module Gitlab
attr_reader :attributes attr_reader :attributes
expose_attribute :noteable_type, :noteable_id, :commit_id, :file_path, expose_attribute :noteable_type, :noteable_id, :commit_id, :file_path,
:diff_hunk, :author, :note, :created_at, :updated_at, :diff_hunk, :author, :created_at, :updated_at,
:original_commit_id, :note_id :original_commit_id, :note_id, :end_line, :start_line,
:side
NOTEABLE_ID_REGEX = %r{/pull/(?<iid>\d+)}i.freeze NOTEABLE_ID_REGEX = %r{/pull/(?<iid>\d+)}i.freeze
...@@ -42,7 +43,8 @@ module Gitlab ...@@ -42,7 +43,8 @@ module Gitlab
updated_at: note.updated_at, updated_at: note.updated_at,
note_id: note.id, note_id: note.id,
end_line: note.line, end_line: note.line,
start_line: note.start_line start_line: note.start_line,
side: note.side
} }
new(hash) new(hash)
...@@ -60,17 +62,31 @@ module Gitlab ...@@ -60,17 +62,31 @@ module Gitlab
# Hash must be Symbols. # Hash must be Symbols.
def initialize(attributes) def initialize(attributes)
@attributes = attributes @attributes = attributes
@note_formatter = DiffNotes::SuggestionFormatter.new(
note: attributes[:note],
start_line: attributes[:start_line],
end_line: attributes[:end_line]
)
end
def contains_suggestion?
@note_formatter.contains_suggestion?
end
def note
@note_formatter.formatted_note
end end
def line_code def line_code
diff_line = Gitlab::Diff::Parser.new.parse(diff_hunk.lines).to_a.last diff_line = Gitlab::Diff::Parser.new.parse(diff_hunk.lines).to_a.last
Gitlab::Git Gitlab::Git.diff_line_code(file_path, diff_line.new_pos, diff_line.old_pos)
.diff_line_code(file_path, diff_line.new_pos, diff_line.old_pos)
end end
# Returns a Hash that can be used to populate `notes.st_diff`, removing # Returns a Hash that can be used to populate `notes.st_diff`, removing
# the need for requesting Git data for every diff note. # the need for requesting Git data for every diff note.
# Used when importing with LegacyDiffNote
def diff_hash def diff_hash
{ {
diff: diff_hunk, diff: diff_hunk,
...@@ -85,12 +101,15 @@ module Gitlab ...@@ -85,12 +101,15 @@ module Gitlab
} }
end end
def note # Used when importing with DiffNote
@note ||= DiffNotes::SuggestionFormatter.formatted_note_for( def diff_position(merge_request)
note: attributes[:note], position_params = {
start_line: attributes[:start_line], diff_refs: merge_request.diff_refs,
end_line: attributes[:end_line] old_path: file_path,
) new_path: file_path
}
Gitlab::Diff::Position.new(position_params.merge(diff_line_params))
end end
def github_identifiers def github_identifiers
...@@ -100,6 +119,20 @@ module Gitlab ...@@ -100,6 +119,20 @@ module Gitlab
noteable_type: noteable_type noteable_type: noteable_type
} }
end end
private
def diff_line_params
if addition?
{ new_line: end_line, old_line: nil }
else
{ new_line: nil, old_line: end_line }
end
end
def addition?
side == 'RIGHT'
end
end end
end end
end end
......
...@@ -10,30 +10,38 @@ module Gitlab ...@@ -10,30 +10,38 @@ module Gitlab
module Representation module Representation
module DiffNotes module DiffNotes
class SuggestionFormatter class SuggestionFormatter
include Gitlab::Utils::StrongMemoize
# A github suggestion: # A github suggestion:
# - the ```suggestion tag must be the first text of the line # - the ```suggestion tag must be the first text of the line
# - it might have up to 3 spaces before the ```suggestion tag # - it might have up to 3 spaces before the ```suggestion tag
# - extra text on the ```suggestion tag line will be ignored # - extra text on the ```suggestion tag line will be ignored
GITHUB_SUGGESTION = /^\ {,3}(?<suggestion>```suggestion\b).*(?<eol>\R)/.freeze GITHUB_SUGGESTION = /^\ {,3}(?<suggestion>```suggestion\b).*(?<eol>\R)/.freeze
def self.formatted_note_for(...)
new(...).formatted_note
end
def initialize(note:, start_line: nil, end_line: nil) def initialize(note:, start_line: nil, end_line: nil)
@note = note @note = note
@start_line = start_line @start_line = start_line
@end_line = end_line @end_line = end_line
end end
# Returns a tuple with:
# - a boolean indicating if the note has suggestions
# - the note with the suggestion formatted for Gitlab
def formatted_note def formatted_note
if contains_suggestion? @formatted_note ||=
note.gsub( if contains_suggestion?
GITHUB_SUGGESTION, note.gsub(
"\\k<suggestion>:#{suggestion_range}\\k<eol>" GITHUB_SUGGESTION,
) "\\k<suggestion>:#{suggestion_range}\\k<eol>"
else )
note else
note
end
end
def contains_suggestion?
strong_memoize(:contain_suggestion) do
note.to_s.match?(GITHUB_SUGGESTION)
end end
end end
...@@ -41,10 +49,6 @@ module Gitlab ...@@ -41,10 +49,6 @@ module Gitlab
attr_reader :note, :start_line, :end_line attr_reader :note, :start_line, :end_line
def contains_suggestion?
note.to_s.match?(GITHUB_SUGGESTION)
end
# Github always saves the comment on the _last_ line of the range. # Github always saves the comment on the _last_ line of the range.
# Therefore, the diff hunk will always be related to lines before # Therefore, the diff hunk will always be related to lines before
# the comment itself. # the comment itself.
......
...@@ -2,156 +2,224 @@ ...@@ -2,156 +2,224 @@
require 'spec_helper' require 'spec_helper'
RSpec.describe Gitlab::GithubImport::Importer::DiffNoteImporter do RSpec.describe Gitlab::GithubImport::Importer::DiffNoteImporter, :aggregate_failures do
let(:project) { create(:project) } let_it_be(:project) { create(:project, :repository) }
let(:client) { double(:client) } let_it_be(:user) { create(:user) }
let(:user) { create(:user) }
let(:created_at) { Time.new(2017, 1, 1, 12, 00) }
let(:updated_at) { Time.new(2017, 1, 1, 12, 15) }
let(:hunk) do let(:client) { double(:client) }
'@@ -1 +1 @@ let(:discussion_id) { 'b0fa404393eeebb4e82becb8104f238812bb1fe6' }
let(:created_at) { Time.new(2017, 1, 1, 12, 00).utc }
let(:updated_at) { Time.new(2017, 1, 1, 12, 15).utc }
let(:note_body) { 'Hello' }
let(:file_path) { 'files/ruby/popen.rb' }
let(:diff_hunk) do
'@@ -14 +14 @@
-Hello -Hello
+Hello world' +Hello world'
end end
let(:note) do let(:note_representation) do
Gitlab::GithubImport::Representation::DiffNote.new( Gitlab::GithubImport::Representation::DiffNote.new(
noteable_type: 'MergeRequest', noteable_type: 'MergeRequest',
noteable_id: 1, noteable_id: 1,
commit_id: '123abc', commit_id: '123abc',
original_commit_id: 'original123abc', original_commit_id: 'original123abc',
file_path: 'README.md', file_path: file_path,
diff_hunk: hunk, author: Gitlab::GithubImport::Representation::User.new(id: user.id, login: user.username),
author: Gitlab::GithubImport::Representation::User note: note_body,
.new(id: user.id, login: user.username),
note: 'Hello',
created_at: created_at, created_at: created_at,
updated_at: updated_at, updated_at: updated_at,
github_id: 1 start_line: nil,
end_line: 15,
github_id: 1,
diff_hunk: diff_hunk,
side: 'RIGHT'
) )
end end
let(:importer) { described_class.new(note, project, client) } subject(:importer) { described_class.new(note_representation, project, client) }
shared_examples 'diff notes without suggestion' do
it 'imports the note as legacy diff note' do
stub_user_finder(user.id, true)
expect { subject.execute }
.to change(LegacyDiffNote, :count)
.by(1)
note = project.notes.diff_notes.take
expect(note).to be_valid
expect(note.author_id).to eq(user.id)
expect(note.commit_id).to eq('original123abc')
expect(note.created_at).to eq(created_at)
expect(note.diff).to be_an_instance_of(Gitlab::Git::Diff)
expect(note.discussion_id).to eq(discussion_id)
expect(note.line_code).to eq(note_representation.line_code)
expect(note.note).to eq('Hello')
expect(note.noteable_id).to eq(merge_request.id)
expect(note.noteable_type).to eq('MergeRequest')
expect(note.project_id).to eq(project.id)
expect(note.st_diff).to eq(note_representation.diff_hash)
expect(note.system).to eq(false)
expect(note.type).to eq('LegacyDiffNote')
expect(note.updated_at).to eq(updated_at)
end
it 'adds a "created by:" note when the author cannot be found' do
stub_user_finder(project.creator_id, false)
expect { subject.execute }
.to change(LegacyDiffNote, :count)
.by(1)
note = project.notes.diff_notes.take
expect(note).to be_valid
expect(note.author_id).to eq(project.creator_id)
expect(note.note).to eq("*Created by: #{user.username}*\n\nHello")
end
it 'does not import the note when a foreign key error is raised' do
stub_user_finder(project.creator_id, false)
expect(Gitlab::Database.main)
.to receive(:bulk_insert)
.and_raise(ActiveRecord::InvalidForeignKey, 'invalid foreign key')
expect { subject.execute }
.not_to change(LegacyDiffNote, :count)
end
end
describe '#execute' do describe '#execute' do
context 'when the merge request no longer exists' do context 'when the merge request no longer exists' do
it 'does not import anything' do it 'does not import anything' do
expect(Gitlab::Database.main).not_to receive(:bulk_insert) expect { subject.execute }
.to not_change(DiffNote, :count)
importer.execute .and not_change(LegacyDiffNote, :count)
end end
end end
context 'when the merge request exists' do context 'when the merge request exists' do
let!(:merge_request) do let_it_be(:merge_request) do
create(:merge_request, source_project: project, target_project: project) create(:merge_request, source_project: project, target_project: project)
end end
before do before do
allow(importer) expect_next_instance_of(Gitlab::GithubImport::IssuableFinder) do |finder|
.to receive(:find_merge_request_id) expect(finder)
.and_return(merge_request.id) .to receive(:database_id)
end .and_return(merge_request.id)
end
it 'imports the note' do
allow(importer.user_finder) expect(Discussion)
.to receive(:author_id_for) .to receive(:discussion_id)
.and_return([user.id, true]) .and_return(discussion_id)
expect(Gitlab::Database.main)
.to receive(:bulk_insert)
.with(
LegacyDiffNote.table_name,
[
{
discussion_id: anything,
noteable_type: 'MergeRequest',
noteable_id: merge_request.id,
project_id: project.id,
author_id: user.id,
note: 'Hello',
system: false,
commit_id: 'original123abc',
line_code: note.line_code,
type: 'LegacyDiffNote',
created_at: created_at,
updated_at: updated_at,
st_diff: note.diff_hash.to_yaml
}
]
)
.and_call_original
importer.execute
end end
it 'imports the note when the author could not be found' do context 'when github_importer_use_diff_note_with_suggestions is disabled' do
allow(importer.user_finder) before do
.to receive(:author_id_for) stub_feature_flags(github_importer_use_diff_note_with_suggestions: false)
.and_return([project.creator_id, false]) end
expect(Gitlab::Database.main) it_behaves_like 'diff notes without suggestion'
.to receive(:bulk_insert)
.with( context 'when the note has suggestions' do
LegacyDiffNote.table_name, let(:note_body) do
[ <<~EOB
{ Suggestion:
discussion_id: anything, ```suggestion
noteable_type: 'MergeRequest', what do you think to do it like this
noteable_id: merge_request.id, ```
project_id: project.id, EOB
author_id: project.creator_id, end
note: "*Created by: #{user.username}*\n\nHello",
system: false, it 'imports the note' do
commit_id: 'original123abc', stub_user_finder(user.id, true)
line_code: note.line_code,
type: 'LegacyDiffNote', expect { subject.execute }
created_at: created_at, .to change(LegacyDiffNote, :count)
updated_at: updated_at, .and not_change(DiffNote, :count)
st_diff: note.diff_hash.to_yaml
} note = project.notes.diff_notes.take
] expect(note).to be_valid
) expect(note.note)
.and_call_original .to eq <<~NOTE
Suggestion:
importer.execute ```suggestion:-0+0
end what do you think to do it like this
```
it 'produces a valid LegacyDiffNote' do NOTE
allow(importer.user_finder) end
.to receive(:author_id_for) end
.and_return([user.id, true])
importer.execute
note = project.notes.diff_notes.take
expect(note).to be_valid
expect(note.diff).to be_an_instance_of(Gitlab::Git::Diff)
end end
it 'does not import the note when a foreign key error is raised' do context 'when github_importer_use_diff_note_with_suggestions is enabled' do
allow(importer.user_finder) before do
.to receive(:author_id_for) stub_feature_flags(github_importer_use_diff_note_with_suggestions: true)
.and_return([project.creator_id, false]) end
expect(Gitlab::Database.main) it_behaves_like 'diff notes without suggestion'
.to receive(:bulk_insert)
.and_raise(ActiveRecord::InvalidForeignKey, 'invalid foreign key') context 'when the note has suggestions' do
let(:note_body) do
expect { importer.execute }.not_to raise_error <<~EOB
Suggestion:
```suggestion
what do you think to do it like this
```
EOB
end
it 'imports the note as diff note' do
stub_user_finder(user.id, true)
expect { subject.execute }
.to change(DiffNote, :count)
.by(1)
note = project.notes.diff_notes.take
expect(note).to be_valid
expect(note.noteable_type).to eq('MergeRequest')
expect(note.noteable_id).to eq(merge_request.id)
expect(note.project_id).to eq(project.id)
expect(note.author_id).to eq(user.id)
expect(note.system).to eq(false)
expect(note.discussion_id).to eq(discussion_id)
expect(note.commit_id).to eq('original123abc')
expect(note.line_code).to eq(note_representation.line_code)
expect(note.type).to eq('DiffNote')
expect(note.created_at).to eq(created_at)
expect(note.updated_at).to eq(updated_at)
expect(note.position.to_h).to eq({
base_sha: merge_request.diffs.diff_refs.base_sha,
head_sha: merge_request.diffs.diff_refs.head_sha,
start_sha: merge_request.diffs.diff_refs.start_sha,
new_line: 15,
old_line: nil,
new_path: file_path,
old_path: file_path,
position_type: 'text',
line_range: nil
})
expect(note.note)
.to eq <<~NOTE
Suggestion:
```suggestion:-0+0
what do you think to do it like this
```
NOTE
end
end
end end
end end
end end
describe '#find_merge_request_id' do def stub_user_finder(user, found)
it 'returns a merge request ID' do expect_next_instance_of(Gitlab::GithubImport::UserFinder) do |finder|
expect_next_instance_of(Gitlab::GithubImport::IssuableFinder) do |instance| expect(finder)
expect(instance).to receive(:database_id).and_return(10) .to receive(:author_id_for)
end .and_return([user, found])
expect(importer.find_merge_request_id).to eq(10)
end end
end end
end end
...@@ -20,6 +20,7 @@ RSpec.describe Gitlab::GithubImport::Importer::DiffNotesImporter do ...@@ -20,6 +20,7 @@ RSpec.describe Gitlab::GithubImport::Importer::DiffNotesImporter do
line: 23, line: 23,
start_line: nil, start_line: nil,
id: 1, id: 1,
side: 'RIGHT',
body: <<~BODY body: <<~BODY
Hello World Hello World
......
...@@ -9,16 +9,21 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do ...@@ -9,16 +9,21 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do
+Hello world' +Hello world'
end end
let(:note_body) { 'Hello world' }
let(:start_line) { nil }
let(:end_line) { 23 }
let(:user_data) { { 'id' => 4, 'login' => 'alice' } }
let(:side) { 'RIGHT' }
let(:created_at) { Time.new(2017, 1, 1, 12, 00) } let(:created_at) { Time.new(2017, 1, 1, 12, 00) }
let(:updated_at) { Time.new(2017, 1, 1, 12, 15) } let(:updated_at) { Time.new(2017, 1, 1, 12, 15) }
shared_examples 'a DiffNote' do shared_examples 'a DiffNote representation' do
it 'returns an instance of DiffNote' do it 'returns an instance of DiffNote' do
expect(note).to be_an_instance_of(described_class) expect(note).to be_an_instance_of(described_class)
end end
context 'the returned DiffNote' do context 'the returned DiffNote' do
it 'includes the number of the note' do it 'includes the number of the merge request' do
expect(note.noteable_id).to eq(42) expect(note.noteable_id).to eq(42)
end end
...@@ -30,18 +35,6 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do ...@@ -30,18 +35,6 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do
expect(note.commit_id).to eq('123abc') expect(note.commit_id).to eq('123abc')
end end
it 'includes the user details' do
expect(note.author)
.to be_an_instance_of(Gitlab::GithubImport::Representation::User)
expect(note.author.id).to eq(4)
expect(note.author.login).to eq('alice')
end
it 'includes the note body' do
expect(note.note).to eq('Hello world')
end
it 'includes the created timestamp' do it 'includes the created timestamp' do
expect(note.created_at).to eq(created_at) expect(note.created_at).to eq(created_at)
end end
...@@ -57,203 +50,192 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do ...@@ -57,203 +50,192 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do
it 'returns the noteable type' do it 'returns the noteable type' do
expect(note.noteable_type).to eq('MergeRequest') expect(note.noteable_type).to eq('MergeRequest')
end end
end
end
describe '.from_api_response' do
let(:response) do
double(
:response,
html_url: 'https://github.com/foo/bar/pull/42',
path: 'README.md',
commit_id: '123abc',
original_commit_id: 'original123abc',
diff_hunk: hunk,
user: double(:user, id: 4, login: 'alice'),
body: 'Hello world',
created_at: created_at,
updated_at: updated_at,
line: 23,
start_line: nil,
id: 1
)
end
it_behaves_like 'a DiffNote' do
let(:note) { described_class.from_api_response(response) }
end
it 'does not set the user if the response did not include a user' do describe '#diff_hash' do
allow(response) it 'returns a Hash containing the diff details' do
.to receive(:user) expect(note.diff_hash).to eq(
.and_return(nil) diff: hunk,
new_path: 'README.md',
note = described_class.from_api_response(response) old_path: 'README.md',
a_mode: '100644',
expect(note.author).to be_nil b_mode: '100644',
end new_file: false
)
it 'formats a suggestion in the note body' do end
allow(response) end
.to receive(:body)
.and_return <<~BODY
```suggestion
Hello World
```
BODY
note = described_class.from_api_response(response)
expect(note.note).to eq <<~BODY
```suggestion:-0+0
Hello World
```
BODY
end
end
describe '.from_json_hash' do
let(:hash) do
{
'noteable_type' => 'MergeRequest',
'noteable_id' => 42,
'file_path' => 'README.md',
'commit_id' => '123abc',
'original_commit_id' => 'original123abc',
'diff_hunk' => hunk,
'author' => { 'id' => 4, 'login' => 'alice' },
'note' => 'Hello world',
'created_at' => created_at.to_s,
'updated_at' => updated_at.to_s,
'note_id' => 1
}
end
it_behaves_like 'a DiffNote' do
let(:note) { described_class.from_json_hash(hash) }
end
it 'does not convert the author if it was not specified' do
hash.delete('author')
note = described_class.from_json_hash(hash)
expect(note.author).to be_nil
end
it 'formats a suggestion in the note body' do describe '#diff_position' do
hash['note'] = <<~BODY let(:diff_refs) { double(:refs, base_sha: 'base', start_sha: 'start', head_sha: 'head') }
```suggestion let(:merge_request) { double(:merge_request, diff_refs: diff_refs) }
Hello World
``` context 'when the diff is an addition' do
BODY it 'returns a Gitlab::Diff::Position' do
expect(note.diff_position(merge_request).to_h).to eq(
base_sha: 'base',
head_sha: 'head',
line_range: nil,
new_line: 23,
new_path: 'README.md',
old_line: nil,
old_path: 'README.md',
position_type: 'text',
start_sha: 'start'
)
end
end
context 'when the diff is an deletion' do
let(:side) { 'LEFT' }
it 'returns a Gitlab::Diff::Position' do
expect(note.diff_position(merge_request).to_h).to eq(
base_sha: 'base',
head_sha: 'head',
line_range: nil,
old_line: 23,
new_path: 'README.md',
new_line: nil,
old_path: 'README.md',
position_type: 'text',
start_sha: 'start'
)
end
end
end
note = described_class.from_json_hash(hash) describe '#github_identifiers' do
it 'returns a hash with needed identifiers' do
expect(note.github_identifiers).to eq(
noteable_id: 42,
noteable_type: 'MergeRequest',
note_id: 1
)
end
end
expect(note.note).to eq <<~BODY describe '#line_code' do
```suggestion:-0+0 it 'generates the proper line code' do
Hello World note = described_class.new(diff_hunk: hunk, file_path: 'README.md')
```
BODY
end
end
describe '#line_code' do expect(note.line_code).to eq('8ec9a00bfd09b3190ac6b22251dbb1aa95a0579d_2_2')
it 'returns a String' do end
note = described_class.new(diff_hunk: hunk, file_path: 'README.md') end
expect(note.line_code).to be_an_instance_of(String) describe '#note and #contains_suggestion?' do
it 'includes the note body' do
expect(note.note).to eq('Hello world')
expect(note.contains_suggestion?).to eq(false)
end
context 'when the note have a suggestion' do
let(:note_body) do
<<~BODY
```suggestion
Hello World
```
BODY
end
it 'returns the suggestion formatted in the note' do
expect(note.note).to eq <<~BODY
```suggestion:-0+0
Hello World
```
BODY
expect(note.contains_suggestion?).to eq(true)
end
end
context 'when the note have a multiline suggestion' do
let(:start_line) { 20 }
let(:end_line) { 23 }
let(:note_body) do
<<~BODY
```suggestion
Hello World
```
BODY
end
it 'returns the multi-line suggestion formatted in the note' do
expect(note.note).to eq <<~BODY
```suggestion:-3+0
Hello World
```
BODY
expect(note.contains_suggestion?).to eq(true)
end
end
describe '#author' do
it 'includes the user details' do
expect(note.author).to be_an_instance_of(
Gitlab::GithubImport::Representation::User
)
expect(note.author.id).to eq(4)
expect(note.author.login).to eq('alice')
end
context 'when the author is empty' do
let(:user_data) { nil }
it 'does not set the user if the response did not include a user' do
expect(note.author).to be_nil
end
end
end
end
end end
end end
describe '#diff_hash' do describe '.from_api_response' do
it 'returns a Hash containing the diff details' do it_behaves_like 'a DiffNote representation' do
note = described_class.from_json_hash( let(:response) do
'noteable_type' => 'MergeRequest', double(
'noteable_id' => 42, :response,
'file_path' => 'README.md', id: 1,
'commit_id' => '123abc', html_url: 'https://github.com/foo/bar/pull/42',
'original_commit_id' => 'original123abc', path: 'README.md',
'diff_hunk' => hunk, commit_id: '123abc',
'author' => { 'id' => 4, 'login' => 'alice' }, original_commit_id: 'original123abc',
'note' => 'Hello world', side: side,
'created_at' => created_at.to_s, user: user_data && double(:user, user_data),
'updated_at' => updated_at.to_s, diff_hunk: hunk,
'note_id' => 1 body: note_body,
) created_at: created_at,
updated_at: updated_at,
expect(note.diff_hash).to eq( line: end_line,
diff: hunk, start_line: start_line
new_path: 'README.md', )
old_path: 'README.md', end
a_mode: '100644',
b_mode: '100644',
new_file: false
)
end
end
describe '#github_identifiers' do subject(:note) { described_class.from_api_response(response) }
it 'returns a hash with needed identifiers' do
github_identifiers = {
noteable_id: 42,
noteable_type: 'MergeRequest',
note_id: 1
}
other_attributes = { something_else: '_something_else_' }
note = described_class.new(github_identifiers.merge(other_attributes))
expect(note.github_identifiers).to eq(github_identifiers)
end end
end end
describe '#note' do describe '.from_json_hash' do
it 'returns the given note' do it_behaves_like 'a DiffNote representation' do
hash = { let(:hash) do
'note': 'simple text' {
} 'note_id' => 1,
'noteable_type' => 'MergeRequest',
note = described_class.new(hash) 'noteable_id' => 42,
'file_path' => 'README.md',
expect(note.note).to eq 'simple text' 'commit_id' => '123abc',
end 'original_commit_id' => 'original123abc',
'side' => side,
it 'returns the suggestion formatted in the note' do 'author' => user_data,
hash = { 'diff_hunk' => hunk,
'note': <<~BODY 'note' => note_body,
```suggestion 'created_at' => created_at.to_s,
Hello World 'updated_at' => updated_at.to_s,
``` 'end_line' => end_line,
BODY 'start_line' => start_line
} }
end
note = described_class.new(hash)
expect(note.note).to eq <<~BODY
```suggestion:-0+0
Hello World
```
BODY
end
it 'returns the multi-line suggestion formatted in the note' do subject(:note) { described_class.from_json_hash(hash) }
hash = {
'start_line': 20,
'end_line': 23,
'note': <<~BODY
```suggestion
Hello World
```
BODY
}
note = described_class.new(hash)
expect(note.note).to eq <<~BODY
```suggestion:-3+0
Hello World
```
BODY
end end
end end
end end
...@@ -9,13 +9,19 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat ...@@ -9,13 +9,19 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat
``` ```
BODY BODY
expect(described_class.formatted_note_for(note: note)).to eq(note) note_formatter = described_class.new(note: note)
expect(note_formatter.formatted_note).to eq(note)
expect(note_formatter.contains_suggestion?).to eq(false)
end end
it 'handles nil value for note' do it 'handles nil value for note' do
note = nil note = nil
expect(described_class.formatted_note_for(note: note)).to eq(note) note_formatter = described_class.new(note: note)
expect(note_formatter.formatted_note).to eq(note)
expect(note_formatter.contains_suggestion?).to eq(false)
end end
it 'does not allow over 3 leading spaces for valid suggestion' do it 'does not allow over 3 leading spaces for valid suggestion' do
...@@ -26,7 +32,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat ...@@ -26,7 +32,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat
``` ```
BODY BODY
expect(described_class.formatted_note_for(note: note)).to eq(note) note_formatter = described_class.new(note: note)
expect(note_formatter.formatted_note).to eq(note)
expect(note_formatter.contains_suggestion?).to eq(false)
end end
it 'allows up to 3 leading spaces' do it 'allows up to 3 leading spaces' do
...@@ -44,7 +53,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat ...@@ -44,7 +53,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat
``` ```
BODY BODY
expect(described_class.formatted_note_for(note: note)).to eq(expected) note_formatter = described_class.new(note: note)
expect(note_formatter.formatted_note).to eq(expected)
expect(note_formatter.contains_suggestion?).to eq(true)
end end
it 'does nothing when there is any text without space after the suggestion tag' do it 'does nothing when there is any text without space after the suggestion tag' do
...@@ -53,7 +65,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat ...@@ -53,7 +65,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat
``` ```
BODY BODY
expect(described_class.formatted_note_for(note: note)).to eq(note) note_formatter = described_class.new(note: note)
expect(note_formatter.formatted_note).to eq(note)
expect(note_formatter.contains_suggestion?).to eq(false)
end end
it 'formats single-line suggestions' do it 'formats single-line suggestions' do
...@@ -71,7 +86,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat ...@@ -71,7 +86,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat
``` ```
BODY BODY
expect(described_class.formatted_note_for(note: note)).to eq(expected) note_formatter = described_class.new(note: note)
expect(note_formatter.formatted_note).to eq(expected)
expect(note_formatter.contains_suggestion?).to eq(true)
end end
it 'ignores text after suggestion tag on the same line' do it 'ignores text after suggestion tag on the same line' do
...@@ -89,7 +107,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat ...@@ -89,7 +107,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat
``` ```
BODY BODY
expect(described_class.formatted_note_for(note: note)).to eq(expected) note_formatter = described_class.new(note: note)
expect(note_formatter.formatted_note).to eq(expected)
expect(note_formatter.contains_suggestion?).to eq(true)
end end
it 'formats multiple single-line suggestions' do it 'formats multiple single-line suggestions' do
...@@ -115,7 +136,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat ...@@ -115,7 +136,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat
``` ```
BODY BODY
expect(described_class.formatted_note_for(note: note)).to eq(expected) note_formatter = described_class.new(note: note)
expect(note_formatter.formatted_note).to eq(expected)
expect(note_formatter.contains_suggestion?).to eq(true)
end end
it 'formats multi-line suggestions' do it 'formats multi-line suggestions' do
...@@ -133,7 +157,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat ...@@ -133,7 +157,10 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat
``` ```
BODY BODY
expect(described_class.formatted_note_for(note: note, start_line: 6, end_line: 8)).to eq(expected) note_formatter = described_class.new(note: note, start_line: 6, end_line: 8)
expect(note_formatter.formatted_note).to eq(expected)
expect(note_formatter.contains_suggestion?).to eq(true)
end end
it 'formats multiple multi-line suggestions' do it 'formats multiple multi-line suggestions' do
...@@ -159,6 +186,9 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat ...@@ -159,6 +186,9 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormat
``` ```
BODY BODY
expect(described_class.formatted_note_for(note: note, start_line: 6, end_line: 8)).to eq(expected) note_formatter = described_class.new(note: note, start_line: 6, end_line: 8)
expect(note_formatter.formatted_note).to eq(expected)
expect(note_formatter.contains_suggestion?).to eq(true)
end end
end end
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe Import::Github::Notes::CreateService do
it 'does not support quick actions' do
project = create(:project, :repository)
user = create(:user)
merge_request = create(:merge_request, source_project: project)
project.add_maintainer(user)
note = described_class.new(
project,
user,
note: '/close',
noteable_type: 'MergeRequest',
noteable_id: merge_request.id
).execute
expect(note.note).to eq('/close')
expect(note.noteable.closed?).to be(false)
end
end
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