mirror of
https://gitlab.com/gitlab-org/gitlab-foss.git
synced 2025-08-13 13:31:19 +00:00
404 lines
13 KiB
Ruby
404 lines
13 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'spec_helper'
|
|
|
|
RSpec.describe Gitlab::QuickActions::Extractor, feature_category: :team_planning do
|
|
using RSpec::Parameterized::TableSyntax
|
|
|
|
let(:definitions) do
|
|
Class.new do
|
|
include Gitlab::QuickActions::Dsl
|
|
|
|
command(:reopen, :open, :close) {}
|
|
command(:assign) {}
|
|
command(:label) {}
|
|
command(:power) {}
|
|
command(:noop_command)
|
|
substitution(:substitution) { 'foo' }
|
|
substitution :shrug do
|
|
"SHRUG"
|
|
end
|
|
end.command_definitions
|
|
end
|
|
|
|
let(:extractor) { described_class.new(definitions, keep_actions: keep_actions) }
|
|
let(:keep_actions) { false }
|
|
|
|
shared_examples 'command with no argument' do
|
|
it 'extracts command' do
|
|
msg, commands = extractor.extract_commands(original_msg)
|
|
|
|
expect(commands).to eq [['reopen']]
|
|
expect(msg).to eq final_msg
|
|
end
|
|
end
|
|
|
|
shared_examples 'command with a single argument' do
|
|
it 'extracts command' do
|
|
msg, commands = extractor.extract_commands(original_msg)
|
|
|
|
expect(commands).to eq [['assign', '@joe']]
|
|
expect(msg).to eq final_msg
|
|
end
|
|
end
|
|
|
|
shared_examples 'command with multiple arguments' do
|
|
it 'extracts command' do
|
|
msg, commands = extractor.extract_commands(original_msg)
|
|
|
|
expect(commands).to match_array [['label', '~foo ~"bar baz" label']]
|
|
expect(msg).to eq final_msg
|
|
end
|
|
end
|
|
|
|
describe '#extract_commands' do
|
|
describe 'command with no argument' do
|
|
context 'at the start of content' do
|
|
it_behaves_like 'command with no argument' do
|
|
let(:original_msg) { "/reopen\nworld" }
|
|
let(:final_msg) { "world" }
|
|
end
|
|
end
|
|
|
|
context 'in the middle of content' do
|
|
it_behaves_like 'command with no argument' do
|
|
let(:original_msg) { "hello\n/reopen\nworld" }
|
|
let(:final_msg) { "hello\nworld" }
|
|
end
|
|
end
|
|
|
|
context 'in the middle of a line' do
|
|
it 'does not extract command' do
|
|
msg = "hello\nworld /reopen"
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to be_empty
|
|
expect(msg).to eq "hello\nworld /reopen"
|
|
end
|
|
end
|
|
|
|
context 'at the end of content' do
|
|
it_behaves_like 'command with no argument' do
|
|
let(:original_msg) { "hello\n/reopen" }
|
|
let(:final_msg) { "hello" }
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'command with a single argument' do
|
|
context 'at the start of content' do
|
|
it_behaves_like 'command with a single argument' do
|
|
let(:original_msg) { "/assign @joe\nworld" }
|
|
let(:final_msg) { "world" }
|
|
end
|
|
|
|
it 'allows slash in command arguments' do
|
|
msg = "/assign @joe / @jane\nworld"
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to eq [['assign', '@joe / @jane']]
|
|
expect(msg).to eq 'world'
|
|
end
|
|
end
|
|
|
|
context 'in the middle of content' do
|
|
it_behaves_like 'command with a single argument' do
|
|
let(:original_msg) { "hello\n/assign @joe\nworld" }
|
|
let(:final_msg) { "hello\nworld" }
|
|
end
|
|
end
|
|
|
|
context 'in the middle of a line' do
|
|
it 'does not extract command' do
|
|
msg = "hello\nworld /assign @joe"
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to be_empty
|
|
expect(msg).to eq "hello\nworld /assign @joe"
|
|
end
|
|
end
|
|
|
|
context 'at the end of content' do
|
|
it_behaves_like 'command with a single argument' do
|
|
let(:original_msg) { "hello\n/assign @joe" }
|
|
let(:final_msg) { "hello" }
|
|
end
|
|
end
|
|
|
|
context 'when argument is not separated with a space' do
|
|
it 'does not extract command' do
|
|
msg = "hello\n/assign@joe\nworld"
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to be_empty
|
|
expect(msg).to eq "hello\n/assign@joe\nworld"
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'command with multiple arguments' do
|
|
context 'at the start of content' do
|
|
it_behaves_like 'command with multiple arguments' do
|
|
let(:original_msg) { %(/label ~foo ~"bar baz" label\nworld) }
|
|
let(:final_msg) { "world" }
|
|
end
|
|
end
|
|
|
|
context 'in the middle of content' do
|
|
it_behaves_like 'command with multiple arguments' do
|
|
let(:original_msg) { %(hello\n/label ~foo ~"bar baz" label\nworld) }
|
|
let(:final_msg) { "hello\nworld" }
|
|
end
|
|
end
|
|
|
|
context 'in the middle of a line' do
|
|
it 'does not extract command' do
|
|
msg = %(hello\nworld /label ~foo ~"bar baz" label)
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to be_empty
|
|
expect(msg).to eq %(hello\nworld /label ~foo ~"bar baz" label)
|
|
end
|
|
end
|
|
|
|
context 'at the end of content' do
|
|
it_behaves_like 'command with multiple arguments' do
|
|
let(:original_msg) { %(hello\n/label ~foo ~"bar baz" label) }
|
|
let(:final_msg) { "hello" }
|
|
end
|
|
end
|
|
|
|
context 'when argument is not separated with a space' do
|
|
it 'does not extract command' do
|
|
msg = %(hello\n/label~foo ~"bar baz" label\nworld)
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to be_empty
|
|
expect(msg).to eq %(hello\n/label~foo ~"bar baz" label\nworld)
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'command with keep_actions' do
|
|
let(:keep_actions) { true }
|
|
|
|
context 'at the start of content' do
|
|
it_behaves_like 'command with a single argument' do
|
|
let(:original_msg) { "/assign @joe\nworld" }
|
|
let(:final_msg) { "<p>/assign @joe</p>\nworld" }
|
|
end
|
|
end
|
|
|
|
context 'in the middle of content' do
|
|
it_behaves_like 'command with a single argument' do
|
|
let(:original_msg) { "hello\n/assign @joe\nworld" }
|
|
let(:final_msg) { "hello\n<p>/assign @joe</p>\nworld" }
|
|
end
|
|
end
|
|
|
|
context 'at the end of content' do
|
|
it_behaves_like 'command with a single argument' do
|
|
let(:original_msg) { "hello\n/assign @joe" }
|
|
let(:final_msg) { "hello\n<p>/assign @joe</p>" }
|
|
end
|
|
end
|
|
end
|
|
|
|
it 'extracts command with multiple arguments and various prefixes' do
|
|
msg = %(hello\n/power @user.name %9.10 ~"bar baz.2"\nworld)
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to eq [['power', '@user.name %9.10 ~"bar baz.2"']]
|
|
expect(msg).to eq "hello\nworld"
|
|
end
|
|
|
|
it 'extracts command case insensitive' do
|
|
msg = %(hello\n/PoWer @user.name %9.10 ~"bar baz.2"\nworld)
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to eq [['power', '@user.name %9.10 ~"bar baz.2"']]
|
|
expect(msg).to eq "hello\nworld"
|
|
end
|
|
|
|
it 'does not extract noop commands' do
|
|
msg = %(hello\nworld\n/reopen\n/noop_command)
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to eq [['reopen']]
|
|
expect(msg).to eq "hello\nworld\n/noop_command"
|
|
end
|
|
|
|
it 'extracts and performs substitution commands' do
|
|
msg = %(hello\nworld\n/reopen\n/substitution)
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to eq [['reopen'], ['substitution']]
|
|
expect(msg).to eq "hello\nworld\nfoo"
|
|
end
|
|
|
|
it 'extracts and performs substitution commands' do
|
|
msg = %(hello\nworld\n/reopen\n/shrug this is great?)
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to eq [['reopen'], ['shrug', 'this is great?']]
|
|
expect(msg).to eq "hello\nworld\nSHRUG"
|
|
end
|
|
|
|
it 'extracts and performs multiple substitution commands' do
|
|
msg = %(hello\nworld\n/reopen\n/shrug this is great?\n/shrug)
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to eq [['reopen'], ['shrug', 'this is great?'], ['shrug']]
|
|
expect(msg).to eq "hello\nworld\nSHRUG\nSHRUG"
|
|
end
|
|
|
|
it 'does not extract substitution command in inline code' do
|
|
msg = %(hello\nworld\n/reopen\n`/tableflip this is great`?)
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to eq [['reopen']]
|
|
expect(msg).to eq "hello\nworld\n`/tableflip this is great`?"
|
|
end
|
|
|
|
it 'extracts and performs substitution commands case insensitive' do
|
|
msg = %(hello\nworld\n/reOpen\n/sHRuG this is great?)
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to eq [['reopen'], ['shrug', 'this is great?']]
|
|
expect(msg).to eq "hello\nworld\nSHRUG"
|
|
end
|
|
|
|
it 'extracts and performs substitution commands with comments' do
|
|
msg = %(hello\nworld\n/reopen\n/substitution wow this is a thing.)
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to match_array [['reopen'], ['substitution', 'wow this is a thing.']]
|
|
expect(msg).to eq "hello\nworld\nfoo"
|
|
end
|
|
|
|
it 'extracts and performs substitution commands with keep_actions' do
|
|
extractor = described_class.new(definitions, keep_actions: true)
|
|
msg = %(hello\nworld\n/reopen\n/substitution wow this is a thing.)
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to match_array [['reopen'], ['substitution', 'wow this is a thing.']]
|
|
expect(msg).to eq "hello\nworld\n<p>/reopen</p>\nfoo"
|
|
end
|
|
|
|
it 'extracts multiple commands' do
|
|
msg = %(hello\n/power @user.name %9.10 ~"bar baz.2" label\nworld\n/reopen)
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to eq [['power', '@user.name %9.10 ~"bar baz.2" label'], ['reopen']]
|
|
expect(msg).to eq "hello\nworld"
|
|
end
|
|
|
|
it 'extracts command when between HTML comment and HTML tags' do
|
|
msg = <<~MSG.strip
|
|
<!-- this is a comment -->
|
|
|
|
/label ~bug
|
|
|
|
<p>
|
|
</p>
|
|
MSG
|
|
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to match_array [['label', '~bug']]
|
|
expect(msg).to eq "<!-- this is a comment -->\n\n<p>\n</p>"
|
|
end
|
|
|
|
it 'does not alter original content if no command is found' do
|
|
msg = 'Fixes #123'
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to be_empty
|
|
expect(msg).to eq 'Fixes #123'
|
|
end
|
|
|
|
it 'does not get confused if command comes before an inline code' do
|
|
msg = "/reopen\n`some inline code`\n/label ~a\n`more inline code`"
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to match_array([['reopen'], ['label', '~a']])
|
|
expect(msg).to eq "`some inline code`\n`more inline code`"
|
|
end
|
|
|
|
it 'does not get confused if command comes before a code block' do
|
|
msg = "/reopen\n```\nsome blockcode\n```\n/label ~a\n```\nmore blockcode\n```"
|
|
msg, commands = extractor.extract_commands(msg)
|
|
|
|
expect(commands).to match_array([['reopen'], ['label', '~a']])
|
|
expect(msg).to eq "```\nsome blockcode\n```\n```\nmore blockcode\n```"
|
|
end
|
|
|
|
context 'does not extract commands inside' do
|
|
where(:description, :text) do
|
|
'block HTML tags' | "Hello\r\n<div>\r\nText\r\n/close\r\n/assign @user\r\n</div>\r\n\r\nWorld"
|
|
'raw HTML with sourcepos' | "<p data-sourcepos=\"0:1-2:10\">\r\n/close\r\n</p>"
|
|
'inline html on seperated rows' | "Text\r\n<b>\r\n/close\r\n</b>"
|
|
'HTML comments' | "<!--\n/assign @user\n-->"
|
|
'blockquotes' | "> Text\r\n/reopen"
|
|
'multiline blockquotes' | "Hello\r\n\r\n>>>\r\nText\r\n/close\r\n/assign @user\r\n>>>\r\n\r\nWorld"
|
|
'code blocks' | "Hello\r\n```\r\nText\r\n/close\r\n/assign @user\r\n```\r\n\r\nWorld"
|
|
'inline code on seperated rows' | "Hello `Text\r\n/close\r\n/assign @user\r\n`\r\n\r\nWorld"
|
|
end
|
|
|
|
with_them do
|
|
specify do
|
|
expected = text.delete("\r")
|
|
msg, commands = extractor.extract_commands(text)
|
|
|
|
expect(commands).to be_empty
|
|
expect(msg).to eq expected
|
|
end
|
|
end
|
|
end
|
|
|
|
it 'limits to passed commands when they are passed' do
|
|
msg = <<~MSG.strip
|
|
Hello, we should only extract the commands passed
|
|
/reopen
|
|
/label hello world
|
|
/power
|
|
MSG
|
|
expected_msg = <<~EXPECTED.strip
|
|
Hello, we should only extract the commands passed
|
|
/power
|
|
EXPECTED
|
|
expected_commands = [['reopen'], ['label', 'hello world']]
|
|
|
|
msg, commands = extractor.extract_commands(msg, only: [:open, :label])
|
|
|
|
expect(commands).to eq(expected_commands)
|
|
expect(msg).to eq expected_msg
|
|
end
|
|
|
|
it 'fails fast for strings with many newlines' do
|
|
msg = '`' + ("\n" * 100_000)
|
|
|
|
expect do
|
|
Timeout.timeout(3.seconds) { extractor.extract_commands(msg) }
|
|
end.not_to raise_error
|
|
end
|
|
end
|
|
|
|
describe '#redact_commands' do
|
|
where(:text, :expected) do
|
|
"hello\n/label ~label1 ~label2\nworld" | "hello\n`/label ~label1 ~label2`\nworld"
|
|
"hello\n/open\n/label ~label1\nworld" | "hello\n`/open`\n`/label ~label1`\nworld"
|
|
"hello\n/reopen\nworld" | "hello\n`/reopen`\nworld"
|
|
"/reopen\nworld" | "`/reopen`\nworld"
|
|
"hello\n/open" | "hello\n`/open`"
|
|
"<!--\n/assign @user\n-->" | "<!--\n/assign @user\n-->"
|
|
end
|
|
|
|
with_them do
|
|
it 'encloses quick actions with code span markdown' do
|
|
expect(extractor.redact_commands(text)).to eq(expected)
|
|
end
|
|
end
|
|
end
|
|
end
|