Commit e4f59f2e authored by Kassio Borges's avatar Kassio Borges

GithubImporter: Format diff note suggestions to the gitlab format

Github "suggestion" feature has a different markdown format, it uses:

    ```suggestion
    SUGGESTION
    ```

While Gitlab has a _range_ in the suggestion markdown, like:

    ```suggestion-0+0
    SUGGESTION
    ```

To convert the Github format to Gitlab, in the `DiffNote`
_representation_ the range will be added. The range is calculated by the
difference of the `start_line` and `line` when the start_line is
present, which indicates a multi-line suggestion.

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

Changelog: changed
MR: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/71411
parent e44c0183
...@@ -40,7 +40,9 @@ module Gitlab ...@@ -40,7 +40,9 @@ module Gitlab
note: note.body, note: note.body,
created_at: note.created_at, created_at: note.created_at,
updated_at: note.updated_at, updated_at: note.updated_at,
note_id: note.id note_id: note.id,
end_line: note.line,
start_line: note.start_line
} }
new(hash) new(hash)
...@@ -83,6 +85,14 @@ module Gitlab ...@@ -83,6 +85,14 @@ module Gitlab
} }
end end
def note
@note ||= DiffNotes::SuggestionFormatter.formatted_note_for(
note: attributes[:note],
start_line: attributes[:start_line],
end_line: attributes[:end_line]
)
end
def github_identifiers def github_identifiers
{ {
note_id: note_id, note_id: note_id,
......
# frozen_string_literal: true
# This class replaces Github markdown suggestion tag with
# a Gitlab suggestion tag. The difference between
# Github's and Gitlab's suggestion tags is that Gitlab
# includes the range of the suggestion in the tag, while Github
# uses other note attributes to position the suggestion.
module Gitlab
module GithubImport
module Representation
module DiffNotes
class SuggestionFormatter
# A github suggestion:
# - the ```suggestion tag must be the first text of the line
# - it might have up to 3 spaces before the ```suggestion tag
# - extra text on the ```suggestion tag line will be ignored
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)
@note = note
@start_line = start_line
@end_line = end_line
end
def formatted_note
if contains_suggestion?
note.gsub(
GITHUB_SUGGESTION,
"\\k<suggestion>:#{suggestion_range}\\k<eol>"
)
else
note
end
end
private
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.
# Therefore, the diff hunk will always be related to lines before
# the comment itself.
def suggestion_range
"-#{line_count}+0"
end
def line_count
if start_line.present?
end_line - start_line
else
0
end
end
end
end
end
end
end
...@@ -15,10 +15,18 @@ RSpec.describe Gitlab::GithubImport::Importer::DiffNotesImporter do ...@@ -15,10 +15,18 @@ RSpec.describe Gitlab::GithubImport::Importer::DiffNotesImporter do
original_commit_id: 'original123abc', original_commit_id: 'original123abc',
diff_hunk: "@@ -1 +1 @@\n-Hello\n+Hello world", diff_hunk: "@@ -1 +1 @@\n-Hello\n+Hello world",
user: double(:user, id: 4, login: 'alice'), user: double(:user, id: 4, login: 'alice'),
body: 'Hello world',
created_at: Time.zone.now, created_at: Time.zone.now,
updated_at: Time.zone.now, updated_at: Time.zone.now,
id: 1 line: 23,
start_line: nil,
id: 1,
body: <<~BODY
Hello World
```suggestion
sug1
```
BODY
) )
end end
......
...@@ -73,6 +73,8 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do ...@@ -73,6 +73,8 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do
body: 'Hello world', body: 'Hello world',
created_at: created_at, created_at: created_at,
updated_at: updated_at, updated_at: updated_at,
line: 23,
start_line: nil,
id: 1 id: 1
) )
end end
...@@ -90,10 +92,27 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do ...@@ -90,10 +92,27 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do
expect(note.author).to be_nil expect(note.author).to be_nil
end end
it 'formats a suggestion in the note body' do
allow(response)
.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 end
describe '.from_json_hash' do describe '.from_json_hash' do
it_behaves_like 'a DiffNote' do
let(:hash) do let(:hash) do
{ {
'noteable_type' => 'MergeRequest', 'noteable_type' => 'MergeRequest',
...@@ -110,27 +129,33 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do ...@@ -110,27 +129,33 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do
} }
end end
it_behaves_like 'a DiffNote' do
let(:note) { described_class.from_json_hash(hash) } let(:note) { described_class.from_json_hash(hash) }
end end
it 'does not convert the author if it was not specified' do it 'does not convert the author if it was not specified' do
hash = { hash.delete('author')
'noteable_type' => 'MergeRequest',
'noteable_id' => 42,
'file_path' => 'README.md',
'commit_id' => '123abc',
'original_commit_id' => 'original123abc',
'diff_hunk' => hunk,
'note' => 'Hello world',
'created_at' => created_at.to_s,
'updated_at' => updated_at.to_s,
'note_id' => 1
}
note = described_class.from_json_hash(hash) note = described_class.from_json_hash(hash)
expect(note.author).to be_nil expect(note.author).to be_nil
end end
it 'formats a suggestion in the note body' do
hash['note'] = <<~BODY
```suggestion
Hello World
```
BODY
note = described_class.from_json_hash(hash)
expect(note.note).to eq <<~BODY
```suggestion:-0+0
Hello World
```
BODY
end
end end
describe '#line_code' do describe '#line_code' do
...@@ -181,4 +206,54 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do ...@@ -181,4 +206,54 @@ RSpec.describe Gitlab::GithubImport::Representation::DiffNote do
expect(note.github_identifiers).to eq(github_identifiers) expect(note.github_identifiers).to eq(github_identifiers)
end end
end end
describe '#note' do
it 'returns the given note' do
hash = {
'note': 'simple text'
}
note = described_class.new(hash)
expect(note.note).to eq 'simple text'
end
it 'returns the suggestion formatted in the note' do
hash = {
'note': <<~BODY
```suggestion
Hello World
```
BODY
}
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
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
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe Gitlab::GithubImport::Representation::DiffNotes::SuggestionFormatter do
it 'does nothing when there is any text before the suggestion tag' do
note = <<~BODY
looks like```suggestion but it isn't
```
BODY
expect(described_class.formatted_note_for(note: note)).to eq(note)
end
it 'handles nil value for note' do
note = nil
expect(described_class.formatted_note_for(note: note)).to eq(note)
end
it 'does not allow over 3 leading spaces for valid suggestion' do
note = <<~BODY
Single-line suggestion
```suggestion
sug1
```
BODY
expect(described_class.formatted_note_for(note: note)).to eq(note)
end
it 'allows up to 3 leading spaces' do
note = <<~BODY
Single-line suggestion
```suggestion
sug1
```
BODY
expected = <<~BODY
Single-line suggestion
```suggestion:-0+0
sug1
```
BODY
expect(described_class.formatted_note_for(note: note)).to eq(expected)
end
it 'does nothing when there is any text without space after the suggestion tag' do
note = <<~BODY
```suggestionbut it isn't
```
BODY
expect(described_class.formatted_note_for(note: note)).to eq(note)
end
it 'formats single-line suggestions' do
note = <<~BODY
Single-line suggestion
```suggestion
sug1
```
BODY
expected = <<~BODY
Single-line suggestion
```suggestion:-0+0
sug1
```
BODY
expect(described_class.formatted_note_for(note: note)).to eq(expected)
end
it 'ignores text after suggestion tag on the same line' do
note = <<~BODY
looks like
```suggestion text to be ignored
suggestion
```
BODY
expected = <<~BODY
looks like
```suggestion:-0+0
suggestion
```
BODY
expect(described_class.formatted_note_for(note: note)).to eq(expected)
end
it 'formats multiple single-line suggestions' do
note = <<~BODY
Single-line suggestion
```suggestion
sug1
```
OR
```suggestion
sug2
```
BODY
expected = <<~BODY
Single-line suggestion
```suggestion:-0+0
sug1
```
OR
```suggestion:-0+0
sug2
```
BODY
expect(described_class.formatted_note_for(note: note)).to eq(expected)
end
it 'formats multi-line suggestions' do
note = <<~BODY
Multi-line suggestion
```suggestion
sug1
```
BODY
expected = <<~BODY
Multi-line suggestion
```suggestion:-2+0
sug1
```
BODY
expect(described_class.formatted_note_for(note: note, start_line: 6, end_line: 8)).to eq(expected)
end
it 'formats multiple multi-line suggestions' do
note = <<~BODY
Multi-line suggestion
```suggestion
sug1
```
OR
```suggestion
sug2
```
BODY
expected = <<~BODY
Multi-line suggestion
```suggestion:-2+0
sug1
```
OR
```suggestion:-2+0
sug2
```
BODY
expect(described_class.formatted_note_for(note: note, start_line: 6, end_line: 8)).to eq(expected)
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