2458 lines
79 KiB
Ruby
2458 lines
79 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
require 'spec_helper'
|
|
|
|
RSpec.describe QuickActions::InterpretService do
|
|
let_it_be(:public_project) { create(:project, :public) }
|
|
let_it_be(:repository_project) { create(:project, :repository) }
|
|
let_it_be(:project) { public_project }
|
|
let_it_be(:developer) { create(:user) }
|
|
let_it_be(:developer2) { create(:user) }
|
|
let_it_be(:developer3) { create(:user) }
|
|
let_it_be_with_reload(:issue) { create(:issue, project: project) }
|
|
let(:milestone) { create(:milestone, project: project, title: '9.10') }
|
|
let(:commit) { create(:commit, project: project) }
|
|
let_it_be(:inprogress) { create(:label, project: project, title: 'In Progress') }
|
|
let_it_be(:helmchart) { create(:label, project: project, title: 'Helm Chart Registry') }
|
|
let_it_be(:bug) { create(:label, project: project, title: 'Bug') }
|
|
|
|
let(:service) { described_class.new(project, developer) }
|
|
|
|
before_all do
|
|
public_project.add_developer(developer)
|
|
repository_project.add_developer(developer)
|
|
end
|
|
|
|
before do
|
|
stub_licensed_features(multiple_issue_assignees: false,
|
|
multiple_merge_request_reviewers: false,
|
|
multiple_merge_request_assignees: false)
|
|
end
|
|
|
|
describe '#execute' do
|
|
let(:merge_request) { create(:merge_request, source_project: project) }
|
|
|
|
shared_examples 'reopen command' do
|
|
it 'returns state_event: "reopen" if content contains /reopen' do
|
|
issuable.close!
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(state_event: 'reopen')
|
|
end
|
|
|
|
it 'returns the reopen message' do
|
|
issuable.close!
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Reopened this #{issuable.to_ability_name.humanize(capitalize: false)}.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'close command' do
|
|
it 'returns state_event: "close" if content contains /close' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(state_event: 'close')
|
|
end
|
|
|
|
it 'returns the close message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Closed this #{issuable.to_ability_name.humanize(capitalize: false)}.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'title command' do
|
|
it 'populates title: "A brand new title" if content contains /title A brand new title' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(title: 'A brand new title')
|
|
end
|
|
|
|
it 'returns the title message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq(%{Changed the title to "A brand new title".})
|
|
end
|
|
end
|
|
|
|
shared_examples 'milestone command' do
|
|
it 'fetches milestone and populates milestone_id if content contains /milestone' do
|
|
milestone # populate the milestone
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(milestone_id: milestone.id)
|
|
end
|
|
|
|
it 'returns the milestone message' do
|
|
milestone # populate the milestone
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Set the milestone to #{milestone.to_reference}.")
|
|
end
|
|
|
|
it 'returns empty milestone message when milestone is wrong' do
|
|
_, _, message = service.execute('/milestone %wrong-milestone', issuable)
|
|
|
|
expect(message).to be_empty
|
|
end
|
|
end
|
|
|
|
shared_examples 'remove_milestone command' do
|
|
it 'populates milestone_id: nil if content contains /remove_milestone' do
|
|
issuable.update!(milestone_id: milestone.id)
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(milestone_id: nil)
|
|
end
|
|
|
|
it 'returns removed milestone message' do
|
|
issuable.update!(milestone_id: milestone.id)
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Removed #{milestone.to_reference} milestone.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'label command' do
|
|
it 'fetches label ids and populates add_label_ids if content contains /label' do
|
|
bug # populate the label
|
|
inprogress # populate the label
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(add_label_ids: [bug.id, inprogress.id])
|
|
end
|
|
|
|
it 'returns the label message' do
|
|
bug # populate the label
|
|
inprogress # populate the label
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Added #{bug.to_reference(format: :name)} #{inprogress.to_reference(format: :name)} labels.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'multiple label command' do
|
|
it 'fetches label ids and populates add_label_ids if content contains multiple /label' do
|
|
bug # populate the label
|
|
inprogress # populate the label
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(add_label_ids: [inprogress.id, bug.id])
|
|
end
|
|
end
|
|
|
|
shared_examples 'multiple label with same argument' do
|
|
it 'prevents duplicate label ids and populates add_label_ids if content contains multiple /label' do
|
|
inprogress # populate the label
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(add_label_ids: [inprogress.id])
|
|
end
|
|
end
|
|
|
|
shared_examples 'multiword label name starting without ~' do
|
|
it 'fetches label ids and populates add_label_ids if content contains /label' do
|
|
_, updates = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(add_label_ids: [helmchart.id])
|
|
end
|
|
end
|
|
|
|
shared_examples 'label name is included in the middle of another label name' do
|
|
it 'ignores the sublabel when the content contains the includer label name' do
|
|
create(:label, project: project, title: 'Chart')
|
|
|
|
_, updates = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(add_label_ids: [helmchart.id])
|
|
end
|
|
end
|
|
|
|
shared_examples 'unlabel command' do
|
|
it 'fetches label ids and populates remove_label_ids if content contains /unlabel' do
|
|
issuable.update!(label_ids: [inprogress.id]) # populate the label
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(remove_label_ids: [inprogress.id])
|
|
end
|
|
|
|
it 'returns the unlabel message' do
|
|
issuable.update!(label_ids: [inprogress.id]) # populate the label
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Removed #{inprogress.to_reference(format: :name)} label.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'multiple unlabel command' do
|
|
it 'fetches label ids and populates remove_label_ids if content contains mutiple /unlabel' do
|
|
issuable.update!(label_ids: [inprogress.id, bug.id]) # populate the label
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(remove_label_ids: [inprogress.id, bug.id])
|
|
end
|
|
end
|
|
|
|
shared_examples 'unlabel command with no argument' do
|
|
it 'populates label_ids: [] if content contains /unlabel with no arguments' do
|
|
issuable.update!(label_ids: [inprogress.id]) # populate the label
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(label_ids: [])
|
|
end
|
|
end
|
|
|
|
shared_examples 'relabel command' do
|
|
it 'populates label_ids: [] if content contains /relabel' do
|
|
issuable.update!(label_ids: [bug.id]) # populate the label
|
|
inprogress # populate the label
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(label_ids: [inprogress.id])
|
|
end
|
|
|
|
it 'returns the relabel message' do
|
|
issuable.update!(label_ids: [bug.id]) # populate the label
|
|
inprogress # populate the label
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Replaced all labels with #{inprogress.to_reference(format: :name)} label.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'todo command' do
|
|
it 'populates todo_event: "add" if content contains /todo' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(todo_event: 'add')
|
|
end
|
|
|
|
it 'returns the todo message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq('Added a to do.')
|
|
end
|
|
end
|
|
|
|
shared_examples 'done command' do
|
|
it 'populates todo_event: "done" if content contains /done' do
|
|
TodoService.new.mark_todo(issuable, developer)
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(todo_event: 'done')
|
|
end
|
|
|
|
it 'returns the done message' do
|
|
TodoService.new.mark_todo(issuable, developer)
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq('Marked to do as done.')
|
|
end
|
|
end
|
|
|
|
shared_examples 'subscribe command' do
|
|
it 'populates subscription_event: "subscribe" if content contains /subscribe' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(subscription_event: 'subscribe')
|
|
end
|
|
|
|
it 'returns the subscribe message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Subscribed to this #{issuable.to_ability_name.humanize(capitalize: false)}.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'unsubscribe command' do
|
|
it 'populates subscription_event: "unsubscribe" if content contains /unsubscribe' do
|
|
issuable.subscribe(developer, project)
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(subscription_event: 'unsubscribe')
|
|
end
|
|
|
|
it 'returns the unsubscribe message' do
|
|
issuable.subscribe(developer, project)
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Unsubscribed from this #{issuable.to_ability_name.humanize(capitalize: false)}.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'due command' do
|
|
let(:expected_date) { Date.new(2016, 8, 28) }
|
|
|
|
it 'populates due_date: Date.new(2016, 8, 28) if content contains /due 2016-08-28' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(due_date: expected_date)
|
|
end
|
|
|
|
it 'returns due_date message: Date.new(2016, 8, 28) if content contains /due 2016-08-28' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Set the due date to #{expected_date.to_s(:medium)}.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'remove_due_date command' do
|
|
before do
|
|
issuable.update!(due_date: Date.today)
|
|
end
|
|
|
|
it 'populates due_date: nil if content contains /remove_due_date' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(due_date: nil)
|
|
end
|
|
|
|
it 'returns Removed the due date' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq('Removed the due date.')
|
|
end
|
|
end
|
|
|
|
shared_examples 'draft command' do
|
|
it 'returns wip_event: "wip" if content contains /draft' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(wip_event: 'wip')
|
|
end
|
|
|
|
it 'returns the wip message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Marked this #{issuable.to_ability_name.humanize(capitalize: false)} as a draft.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'undraft command' do
|
|
it 'returns wip_event: "unwip" if content contains /draft' do
|
|
issuable.update!(title: issuable.wip_title)
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(wip_event: 'unwip')
|
|
end
|
|
|
|
it 'returns the unwip message' do
|
|
issuable.update!(title: issuable.wip_title)
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Unmarked this #{issuable.to_ability_name.humanize(capitalize: false)} as a draft.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'estimate command' do
|
|
it 'populates time_estimate: 3600 if content contains /estimate 1h' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(time_estimate: 3600)
|
|
end
|
|
|
|
it 'returns the time_estimate formatted message' do
|
|
_, _, message = service.execute('/estimate 79d', issuable)
|
|
|
|
expect(message).to eq('Set time estimate to 3mo 3w 4d.')
|
|
end
|
|
end
|
|
|
|
shared_examples 'spend command' do
|
|
it 'populates spend_time: 3600 if content contains /spend 1h' do
|
|
freeze_time do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(spend_time: {
|
|
duration: 3600,
|
|
user_id: developer.id,
|
|
spent_at: DateTime.current
|
|
})
|
|
end
|
|
end
|
|
end
|
|
|
|
shared_examples 'spend command with negative time' do
|
|
it 'populates spend_time: -7200 if content contains -120m' do
|
|
freeze_time do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(spend_time: {
|
|
duration: -7200,
|
|
user_id: developer.id,
|
|
spent_at: DateTime.current
|
|
})
|
|
end
|
|
end
|
|
|
|
it 'returns the spend_time message including the formatted duration and verb' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq('Subtracted 2h spent time.')
|
|
end
|
|
end
|
|
|
|
shared_examples 'spend command with valid date' do
|
|
it 'populates spend time: 1800 with date in date type format' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(spend_time: {
|
|
duration: 1800,
|
|
user_id: developer.id,
|
|
spent_at: Date.parse(date)
|
|
})
|
|
end
|
|
end
|
|
|
|
shared_examples 'spend command with invalid date' do
|
|
it 'will not create any note and timelog' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq({})
|
|
end
|
|
end
|
|
|
|
shared_examples 'spend command with future date' do
|
|
it 'will not create any note and timelog' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq({})
|
|
end
|
|
end
|
|
|
|
shared_examples 'remove_estimate command' do
|
|
it 'populates time_estimate: 0 if content contains /remove_estimate' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(time_estimate: 0)
|
|
end
|
|
|
|
it 'returns the remove_estimate message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq('Removed time estimate.')
|
|
end
|
|
end
|
|
|
|
shared_examples 'remove_time_spent command' do
|
|
it 'populates spend_time: :reset if content contains /remove_time_spent' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(spend_time: { duration: :reset, user_id: developer.id })
|
|
end
|
|
|
|
it 'returns the remove_time_spent message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq('Removed spent time.')
|
|
end
|
|
end
|
|
|
|
shared_examples 'lock command' do
|
|
let(:issue) { create(:issue, project: project, discussion_locked: false) }
|
|
let(:merge_request) { create(:merge_request, source_project: project, discussion_locked: false) }
|
|
|
|
it 'returns discussion_locked: true if content contains /lock' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(discussion_locked: true)
|
|
end
|
|
|
|
it 'returns the lock discussion message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq('Locked the discussion.')
|
|
end
|
|
end
|
|
|
|
shared_examples 'unlock command' do
|
|
let(:issue) { create(:issue, project: project, discussion_locked: true) }
|
|
let(:merge_request) { create(:merge_request, source_project: project, discussion_locked: true) }
|
|
|
|
it 'returns discussion_locked: true if content contains /unlock' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(discussion_locked: false)
|
|
end
|
|
|
|
it 'returns the unlock discussion message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq('Unlocked the discussion.')
|
|
end
|
|
end
|
|
|
|
shared_examples 'failed command' do |error_msg|
|
|
it 'populates {} if content contains an unsupported command' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to be_empty
|
|
end
|
|
|
|
it "returns #{error_msg || 'an empty'} message" do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
if error_msg
|
|
expect(message).to eq(error_msg)
|
|
else
|
|
expect(message).to be_empty
|
|
end
|
|
end
|
|
end
|
|
|
|
shared_examples 'merge immediately command' do
|
|
let(:project) { repository_project }
|
|
|
|
it 'runs merge command if content contains /merge' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(merge: merge_request.diff_head_sha)
|
|
end
|
|
|
|
it 'returns them merge message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq('Merged this merge request.')
|
|
end
|
|
end
|
|
|
|
shared_examples 'merge automatically command' do
|
|
let(:project) { repository_project }
|
|
|
|
it 'runs merge command if content contains /merge and returns merge message' do
|
|
_, updates, message = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(merge: merge_request.diff_head_sha)
|
|
expect(message).to eq('Scheduled to merge this merge request (Merge when pipeline succeeds).')
|
|
end
|
|
end
|
|
|
|
shared_examples 'award command' do
|
|
it 'toggle award 100 emoji if content contains /award :100:' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(emoji_award: "100")
|
|
end
|
|
|
|
it 'returns the award message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq('Toggled :100: emoji award.')
|
|
end
|
|
end
|
|
|
|
shared_examples 'duplicate command' do
|
|
it 'fetches issue and populates canonical_issue_id if content contains /duplicate issue_reference' do
|
|
issue_duplicate # populate the issue
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(canonical_issue_id: issue_duplicate.id)
|
|
end
|
|
|
|
it 'returns the duplicate message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Marked this issue as a duplicate of #{issue_duplicate.to_reference(project)}.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'copy_metadata command' do
|
|
it 'fetches issue or merge request and copies labels and milestone if content contains /copy_metadata reference' do
|
|
source_issuable # populate the issue
|
|
todo_label # populate this label
|
|
inreview_label # populate this label
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates[:add_label_ids]).to match_array([inreview_label.id, todo_label.id])
|
|
|
|
if source_issuable.milestone
|
|
expect(updates[:milestone_id]).to eq(source_issuable.milestone.id)
|
|
else
|
|
expect(updates).not_to have_key(:milestone_id)
|
|
end
|
|
end
|
|
|
|
it 'returns the copy metadata message' do
|
|
_, _, message = service.execute("/copy_metadata #{source_issuable.to_reference}", issuable)
|
|
|
|
expect(message).to eq("Copied labels and milestone from #{source_issuable.to_reference}.")
|
|
end
|
|
end
|
|
|
|
describe 'move issue command' do
|
|
it 'returns the move issue message' do
|
|
_, _, message = service.execute("/move #{project.full_path}", issue)
|
|
|
|
expect(message).to eq("Moved this issue to #{project.full_path}.")
|
|
end
|
|
|
|
it 'returns move issue failure message when the referenced issue is not found' do
|
|
_, _, message = service.execute('/move invalid', issue)
|
|
|
|
expect(message).to eq(_("Failed to move this issue because target project doesn't exist."))
|
|
end
|
|
end
|
|
|
|
shared_examples 'confidential command' do
|
|
it 'marks issue as confidential if content contains /confidential' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(confidential: true)
|
|
end
|
|
|
|
it 'returns the confidential message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq('Made this issue confidential.')
|
|
end
|
|
|
|
context 'when issuable is already confidential' do
|
|
before do
|
|
issuable.update!(confidential: true)
|
|
end
|
|
|
|
it 'returns an error message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq('Could not apply confidential command.')
|
|
end
|
|
|
|
it 'is not part of the available commands' do
|
|
expect(service.available_commands(issuable)).not_to include(a_hash_including(name: :confidential))
|
|
end
|
|
end
|
|
end
|
|
|
|
shared_examples 'shrug command' do
|
|
it 'appends ¯\_(ツ)_/¯ to the comment' do
|
|
new_content, _, _ = service.execute(content, issuable)
|
|
|
|
expect(new_content).to end_with(described_class::SHRUG)
|
|
end
|
|
end
|
|
|
|
shared_examples 'tableflip command' do
|
|
it 'appends (╯°□°)╯︵ ┻━┻ to the comment' do
|
|
new_content, _, _ = service.execute(content, issuable)
|
|
|
|
expect(new_content).to end_with(described_class::TABLEFLIP)
|
|
end
|
|
end
|
|
|
|
shared_examples 'tag command' do
|
|
it 'tags a commit' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(tag_name: tag_name, tag_message: tag_message)
|
|
end
|
|
|
|
it 'returns the tag message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
if tag_message.present?
|
|
expect(message).to eq(%{Tagged this commit to #{tag_name} with "#{tag_message}".})
|
|
else
|
|
expect(message).to eq("Tagged this commit to #{tag_name}.")
|
|
end
|
|
end
|
|
end
|
|
|
|
shared_examples 'assign command' do
|
|
it 'assigns to a single user' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(assignee_ids: [developer.id])
|
|
end
|
|
|
|
it 'returns the assign message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Assigned #{developer.to_reference}.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'assign_reviewer command' do
|
|
it 'assigns a reviewer to a single user' do
|
|
_, updates, message = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(reviewer_ids: [developer.id])
|
|
expect(message).to eq("Assigned #{developer.to_reference} as reviewer.")
|
|
end
|
|
end
|
|
|
|
shared_examples 'unassign_reviewer command' do
|
|
it 'removes a single reviewer' do
|
|
_, updates, message = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(reviewer_ids: [])
|
|
expect(message).to eq("Removed reviewer #{developer.to_reference}.")
|
|
end
|
|
end
|
|
|
|
it_behaves_like 'reopen command' do
|
|
let(:content) { '/reopen' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'reopen command' do
|
|
let(:content) { '/reopen' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'close command' do
|
|
let(:content) { '/close' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'close command' do
|
|
let(:content) { '/close' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
context 'merge command' do
|
|
let(:service) { described_class.new(project, developer, { merge_request_diff_head_sha: merge_request.diff_head_sha }) }
|
|
|
|
it_behaves_like 'merge immediately command' do
|
|
let(:content) { '/merge' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
context 'when the head pipeline of merge request is running' do
|
|
before do
|
|
create(:ci_pipeline, :detached_merge_request_pipeline, merge_request: merge_request)
|
|
merge_request.update_head_pipeline
|
|
end
|
|
|
|
it_behaves_like 'merge automatically command' do
|
|
let(:content) { '/merge' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
end
|
|
|
|
context 'can not be merged when logged user does not have permissions' do
|
|
let(:service) { described_class.new(project, create(:user)) }
|
|
|
|
it_behaves_like 'failed command', 'Could not apply merge command.' do
|
|
let(:content) { "/merge" }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
end
|
|
|
|
context 'can not be merged when sha does not match' do
|
|
let(:service) { described_class.new(project, developer, { merge_request_diff_head_sha: 'othersha' }) }
|
|
|
|
it_behaves_like 'failed command', 'Could not apply merge command.' do
|
|
let(:content) { "/merge" }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
end
|
|
|
|
context 'when sha is missing' do
|
|
let(:project) { repository_project }
|
|
let(:service) { described_class.new(project, developer, {}) }
|
|
|
|
it 'precheck passes and returns merge command' do
|
|
_, updates, _ = service.execute('/merge', merge_request)
|
|
|
|
expect(updates).to eq(merge: nil)
|
|
end
|
|
end
|
|
|
|
context 'issue can not be merged' do
|
|
it_behaves_like 'failed command', 'Could not apply merge command.' do
|
|
let(:content) { "/merge" }
|
|
let(:issuable) { issue }
|
|
end
|
|
end
|
|
|
|
context 'non persisted merge request cant be merged' do
|
|
it_behaves_like 'failed command', 'Could not apply merge command.' do
|
|
let(:content) { "/merge" }
|
|
let(:issuable) { build(:merge_request) }
|
|
end
|
|
end
|
|
|
|
context 'not persisted merge request can not be merged' do
|
|
it_behaves_like 'failed command', 'Could not apply merge command.' do
|
|
let(:content) { "/merge" }
|
|
let(:issuable) { build(:merge_request, source_project: project) }
|
|
end
|
|
end
|
|
end
|
|
|
|
it_behaves_like 'title command' do
|
|
let(:content) { '/title A brand new title' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'title command' do
|
|
let(:content) { '/title A brand new title' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { '/title' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
context 'assign command with one user' do
|
|
it_behaves_like 'assign command' do
|
|
let(:content) { "/assign @#{developer.username}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'assign command' do
|
|
let(:content) { "/assign @#{developer.username}" }
|
|
let(:issuable) { create(:incident, project: project) }
|
|
end
|
|
|
|
it_behaves_like 'assign command' do
|
|
let(:content) { "/assign @#{developer.username}" }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
end
|
|
|
|
# CE does not have multiple assignees
|
|
context 'assign command with multiple assignees' do
|
|
before do
|
|
project.add_developer(developer2)
|
|
end
|
|
|
|
# There's no guarantee that the reference extractor will preserve
|
|
# the order of the mentioned users since this is dependent on the
|
|
# order in which rows are returned. We just ensure that at least
|
|
# one of the mentioned users is assigned.
|
|
shared_examples 'assigns to one of the two users' do
|
|
let(:content) { "/assign @#{developer.username} @#{developer2.username}" }
|
|
|
|
it 'assigns to a single user' do
|
|
_, updates, message = service.execute(content, issuable)
|
|
|
|
expect(updates[:assignee_ids].count).to eq(1)
|
|
assignee = updates[:assignee_ids].first
|
|
expect([developer.id, developer2.id]).to include(assignee)
|
|
|
|
user = assignee == developer.id ? developer : developer2
|
|
|
|
expect(message).to match("Assigned #{user.to_reference}.")
|
|
end
|
|
end
|
|
|
|
it_behaves_like 'assigns to one of the two users' do
|
|
let(:content) { "/assign @#{developer.username} @#{developer2.username}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'assigns to one of the two users' do
|
|
let(:content) { "/assign @#{developer.username} @#{developer2.username}" }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
end
|
|
|
|
context 'assign command with me alias' do
|
|
it_behaves_like 'assign command' do
|
|
let(:content) { '/assign me' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'assign command' do
|
|
let(:content) { '/assign me' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
end
|
|
|
|
context 'assign command with me alias and whitespace' do
|
|
it_behaves_like 'assign command' do
|
|
let(:content) { '/assign me ' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'assign command' do
|
|
let(:content) { '/assign me ' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
end
|
|
|
|
it_behaves_like 'failed command', "Failed to assign a user because no user was found." do
|
|
let(:content) { '/assign @abcd1234' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', "Failed to assign a user because no user was found." do
|
|
let(:content) { '/assign' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
describe 'assign_reviewer command' do
|
|
let(:content) { "/assign_reviewer @#{developer.username}" }
|
|
let(:issuable) { merge_request }
|
|
|
|
context 'with one user' do
|
|
it_behaves_like 'assign_reviewer command'
|
|
end
|
|
|
|
context 'with an issue instead of a merge request' do
|
|
let(:issuable) { issue }
|
|
|
|
it_behaves_like 'failed command', 'Could not apply assign_reviewer command.'
|
|
end
|
|
|
|
# CE does not have multiple reviewers
|
|
context 'assign command with multiple assignees' do
|
|
before do
|
|
project.add_developer(developer2)
|
|
end
|
|
|
|
# There's no guarantee that the reference extractor will preserve
|
|
# the order of the mentioned users since this is dependent on the
|
|
# order in which rows are returned. We just ensure that at least
|
|
# one of the mentioned users is assigned.
|
|
context 'assigns to one of the two users' do
|
|
let(:content) { "/assign_reviewer @#{developer.username} @#{developer2.username}" }
|
|
|
|
it 'assigns to a single reviewer' do
|
|
_, updates, message = service.execute(content, issuable)
|
|
|
|
expect(updates[:reviewer_ids].count).to eq(1)
|
|
reviewer = updates[:reviewer_ids].first
|
|
expect([developer.id, developer2.id]).to include(reviewer)
|
|
|
|
user = reviewer == developer.id ? developer : developer2
|
|
|
|
expect(message).to match("Assigned #{user.to_reference} as reviewer.")
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'with "me" alias' do
|
|
let(:content) { '/assign_reviewer me' }
|
|
|
|
it_behaves_like 'assign_reviewer command'
|
|
end
|
|
|
|
context 'with an alias and whitespace' do
|
|
let(:content) { '/assign_reviewer me ' }
|
|
|
|
it_behaves_like 'assign_reviewer command'
|
|
end
|
|
|
|
context 'with an incorrect user' do
|
|
let(:content) { '/assign_reviewer @abcd1234' }
|
|
|
|
it_behaves_like 'failed command', "Failed to assign a reviewer because no user was found."
|
|
end
|
|
|
|
context 'with the "reviewer" alias' do
|
|
let(:content) { "/reviewer @#{developer.username}" }
|
|
|
|
it_behaves_like 'assign_reviewer command'
|
|
end
|
|
|
|
context 'with the "request_review" alias' do
|
|
let(:content) { "/request_review @#{developer.username}" }
|
|
|
|
it_behaves_like 'assign_reviewer command'
|
|
end
|
|
|
|
context 'with no user' do
|
|
let(:content) { '/assign_reviewer' }
|
|
|
|
it_behaves_like 'failed command', "Failed to assign a reviewer because no user was found."
|
|
end
|
|
|
|
context 'includes only the user reference with extra text' do
|
|
let(:content) { "/assign_reviewer @#{developer.username} do it!" }
|
|
|
|
it_behaves_like 'assign_reviewer command'
|
|
end
|
|
end
|
|
|
|
describe 'unassign_reviewer command' do
|
|
# CE does not have multiple reviewers, so basically anything
|
|
# after /unassign_reviewer (including whitespace) will remove
|
|
# all the current reviewers.
|
|
let(:issuable) { create(:merge_request, reviewers: [developer]) }
|
|
let(:content) { "/unassign_reviewer @#{developer.username}" }
|
|
|
|
context 'with one user' do
|
|
it_behaves_like 'unassign_reviewer command'
|
|
end
|
|
|
|
context 'with an issue instead of a merge request' do
|
|
let(:issuable) { issue }
|
|
|
|
it_behaves_like 'failed command', 'Could not apply unassign_reviewer command.'
|
|
end
|
|
|
|
context 'with anything after the command' do
|
|
let(:content) { '/unassign_reviewer supercalifragilisticexpialidocious' }
|
|
|
|
it_behaves_like 'unassign_reviewer command'
|
|
end
|
|
|
|
context 'with the "remove_reviewer" alias' do
|
|
let(:content) { "/remove_reviewer @#{developer.username}" }
|
|
|
|
it_behaves_like 'unassign_reviewer command'
|
|
end
|
|
|
|
context 'with no user' do
|
|
let(:content) { '/unassign_reviewer' }
|
|
|
|
it_behaves_like 'unassign_reviewer command'
|
|
end
|
|
end
|
|
|
|
context 'unassign command' do
|
|
let(:content) { '/unassign' }
|
|
|
|
context 'Issue' do
|
|
it 'populates assignee_ids: [] if content contains /unassign' do
|
|
issue.update!(assignee_ids: [developer.id])
|
|
_, updates, _ = service.execute(content, issue)
|
|
|
|
expect(updates).to eq(assignee_ids: [])
|
|
end
|
|
|
|
it 'returns the unassign message for all the assignee if content contains /unassign' do
|
|
issue.update!(assignee_ids: [developer.id, developer2.id])
|
|
_, _, message = service.execute(content, issue)
|
|
|
|
expect(message).to eq("Removed assignees #{developer.to_reference} and #{developer2.to_reference}.")
|
|
end
|
|
end
|
|
|
|
context 'Merge Request' do
|
|
it 'populates assignee_ids: [] if content contains /unassign' do
|
|
merge_request.update!(assignee_ids: [developer.id])
|
|
_, updates, _ = service.execute(content, merge_request)
|
|
|
|
expect(updates).to eq(assignee_ids: [])
|
|
end
|
|
|
|
it 'returns the unassign message for all the assignee if content contains /unassign' do
|
|
merge_request.update!(assignee_ids: [developer.id, developer2.id])
|
|
_, _, message = service.execute(content, merge_request)
|
|
|
|
expect(message).to eq("Removed assignees #{developer.to_reference} and #{developer2.to_reference}.")
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'project milestones' do
|
|
before do
|
|
milestone
|
|
end
|
|
|
|
it_behaves_like 'milestone command' do
|
|
let(:content) { "/milestone %#{milestone.title}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'milestone command' do
|
|
let(:content) { "/milestone %#{milestone.title}" }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
end
|
|
|
|
context 'only group milestones available' do
|
|
let_it_be(:ancestor_group) { create(:group) }
|
|
let_it_be(:group) { create(:group, parent: ancestor_group) }
|
|
let_it_be(:project) { create(:project, :public, namespace: group) }
|
|
let_it_be(:milestone) { create(:milestone, group: ancestor_group, title: '10.0') }
|
|
|
|
before_all do
|
|
project.add_developer(developer)
|
|
end
|
|
|
|
it_behaves_like 'milestone command' do
|
|
let(:content) { "/milestone %#{milestone.title}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'milestone command' do
|
|
let(:content) { "/milestone %#{milestone.title}" }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
end
|
|
|
|
it_behaves_like 'remove_milestone command' do
|
|
let(:content) { '/remove_milestone' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'remove_milestone command' do
|
|
let(:content) { '/remove_milestone' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'label command' do
|
|
let(:content) { %(/label ~"#{inprogress.title}" ~#{bug.title} ~unknown) }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'label command' do
|
|
let(:content) { %(/label ~"#{inprogress.title}" ~#{bug.title} ~unknown) }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'multiple label command' do
|
|
let(:content) { %(/label ~"#{inprogress.title}" \n/label ~#{bug.title}) }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'multiple label with same argument' do
|
|
let(:content) { %(/label ~"#{inprogress.title}" \n/label ~#{inprogress.title}) }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'multiword label name starting without ~' do
|
|
let(:content) { %(/label "#{helmchart.title}") }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'multiword label name starting without ~' do
|
|
let(:content) { %(/label "#{helmchart.title}") }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'label name is included in the middle of another label name' do
|
|
let(:content) { %(/label ~"#{helmchart.title}") }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'label name is included in the middle of another label name' do
|
|
let(:content) { %(/label ~"#{helmchart.title}") }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'unlabel command' do
|
|
let(:content) { %(/unlabel ~"#{inprogress.title}") }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'unlabel command' do
|
|
let(:content) { %(/unlabel ~"#{inprogress.title}") }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'multiple unlabel command' do
|
|
let(:content) { %(/unlabel ~"#{inprogress.title}" \n/unlabel ~#{bug.title}) }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'unlabel command with no argument' do
|
|
let(:content) { %(/unlabel) }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'unlabel command with no argument' do
|
|
let(:content) { %(/unlabel) }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'relabel command' do
|
|
let(:content) { %(/relabel ~"#{inprogress.title}") }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'relabel command' do
|
|
let(:content) { %(/relabel ~"#{inprogress.title}") }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'done command' do
|
|
let(:content) { '/done' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'done command' do
|
|
let(:content) { '/done' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'subscribe command' do
|
|
let(:content) { '/subscribe' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'subscribe command' do
|
|
let(:content) { '/subscribe' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'unsubscribe command' do
|
|
let(:content) { '/unsubscribe' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'unsubscribe command' do
|
|
let(:content) { '/unsubscribe' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply due command.' do
|
|
let(:content) { '/due 2016-08-28' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'remove_due_date command' do
|
|
let(:content) { '/remove_due_date' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'draft command' do
|
|
let(:content) { '/draft' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'undraft command' do
|
|
let(:content) { '/draft' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply remove_due_date command.' do
|
|
let(:content) { '/remove_due_date' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'estimate command' do
|
|
let(:content) { '/estimate 1h' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { '/estimate' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { '/estimate abc' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'spend command' do
|
|
let(:content) { '/spend 1h' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'spend command' do
|
|
let(:content) { '/spent 1h' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'spend command with negative time' do
|
|
let(:content) { '/spend -120m' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'spend command with negative time' do
|
|
let(:content) { '/spent -120m' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'spend command with valid date' do
|
|
let(:date) { '2016-02-02' }
|
|
let(:content) { "/spend 30m #{date}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'spend command with valid date' do
|
|
let(:date) { '2016-02-02' }
|
|
let(:content) { "/spent 30m #{date}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'spend command with invalid date' do
|
|
let(:content) { '/spend 30m 17-99-99' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'spend command with invalid date' do
|
|
let(:content) { '/spent 30m 17-99-99' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'spend command with future date' do
|
|
let(:content) { '/spend 30m 6017-10-10' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'spend command with future date' do
|
|
let(:content) { '/spent 30m 6017-10-10' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { '/spend' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { '/spent' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { '/spend abc' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { '/spent abc' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'remove_estimate command' do
|
|
let(:content) { '/remove_estimate' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'remove_time_spent command' do
|
|
let(:content) { '/remove_time_spent' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'confidential command' do
|
|
let(:content) { '/confidential' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'confidential command' do
|
|
let(:content) { '/confidential' }
|
|
let(:issuable) { create(:incident, project: project) }
|
|
end
|
|
|
|
it_behaves_like 'lock command' do
|
|
let(:content) { '/lock' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'lock command' do
|
|
let(:content) { '/lock' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
it_behaves_like 'unlock command' do
|
|
let(:content) { '/unlock' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'unlock command' do
|
|
let(:content) { '/unlock' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
context '/todo' do
|
|
let(:content) { '/todo' }
|
|
|
|
context 'if issuable is an Issue' do
|
|
it_behaves_like 'todo command' do
|
|
let(:issuable) { issue }
|
|
end
|
|
end
|
|
|
|
context 'if issuable is a MergeRequest' do
|
|
it_behaves_like 'todo command' do
|
|
let(:issuable) { merge_request }
|
|
end
|
|
end
|
|
|
|
context 'if issuable is a Commit' do
|
|
it_behaves_like 'failed command', 'Could not apply todo command.' do
|
|
let(:issuable) { commit }
|
|
end
|
|
end
|
|
end
|
|
|
|
context '/due command' do
|
|
it 'returns invalid date format message when the due date is invalid' do
|
|
issue = build(:issue, project: project)
|
|
|
|
_, _, message = service.execute('/due invalid date', issue)
|
|
|
|
expect(message).to eq(_('Failed to set due date because the date format is invalid.'))
|
|
end
|
|
|
|
it_behaves_like 'due command' do
|
|
let(:content) { '/due 2016-08-28' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'due command' do
|
|
let(:content) { '/due tomorrow' }
|
|
let(:issuable) { issue }
|
|
let(:expected_date) { Date.tomorrow }
|
|
end
|
|
|
|
it_behaves_like 'due command' do
|
|
let(:content) { '/due 5 days from now' }
|
|
let(:issuable) { issue }
|
|
let(:expected_date) { 5.days.from_now.to_date }
|
|
end
|
|
|
|
it_behaves_like 'due command' do
|
|
let(:content) { '/due in 2 days' }
|
|
let(:issuable) { issue }
|
|
let(:expected_date) { 2.days.from_now.to_date }
|
|
end
|
|
end
|
|
|
|
context '/copy_metadata command' do
|
|
let(:todo_label) { create(:label, project: project, title: 'To Do') }
|
|
let(:inreview_label) { create(:label, project: project, title: 'In Review') }
|
|
|
|
it 'is available when the user is a developer' do
|
|
expect(service.available_commands(issue)).to include(a_hash_including(name: :copy_metadata))
|
|
end
|
|
|
|
context 'when the user does not have permission' do
|
|
let(:guest) { create(:user) }
|
|
let(:service) { described_class.new(project, guest) }
|
|
|
|
it 'is not available' do
|
|
expect(service.available_commands(issue)).not_to include(a_hash_including(name: :copy_metadata))
|
|
end
|
|
end
|
|
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { '/copy_metadata' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'copy_metadata command' do
|
|
let(:source_issuable) { create(:labeled_issue, project: project, labels: [inreview_label, todo_label]) }
|
|
|
|
let(:content) { "/copy_metadata #{source_issuable.to_reference}" }
|
|
let(:issuable) { build(:issue, project: project) }
|
|
end
|
|
|
|
it_behaves_like 'copy_metadata command' do
|
|
let(:source_issuable) { create(:labeled_issue, project: project, labels: [inreview_label, todo_label]) }
|
|
|
|
let(:content) { "/copy_metadata #{source_issuable.to_reference}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
context 'when the parent issuable has a milestone' do
|
|
it_behaves_like 'copy_metadata command' do
|
|
let(:source_issuable) { create(:labeled_issue, project: project, labels: [todo_label, inreview_label], milestone: milestone) }
|
|
|
|
let(:content) { "/copy_metadata #{source_issuable.to_reference(project)}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
end
|
|
|
|
context 'when more than one issuable is passed' do
|
|
it_behaves_like 'copy_metadata command' do
|
|
let(:source_issuable) { create(:labeled_issue, project: project, labels: [inreview_label, todo_label]) }
|
|
let(:other_label) { create(:label, project: project, title: 'Other') }
|
|
let(:other_source_issuable) { create(:labeled_issue, project: project, labels: [other_label]) }
|
|
|
|
let(:content) { "/copy_metadata #{source_issuable.to_reference} #{other_source_issuable.to_reference}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
end
|
|
|
|
context 'cross project references' do
|
|
it_behaves_like 'failed command' do
|
|
let(:other_project) { create(:project, :public) }
|
|
let(:source_issuable) { create(:labeled_issue, project: other_project, labels: [todo_label, inreview_label]) }
|
|
let(:content) { "/copy_metadata #{source_issuable.to_reference(project)}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { "/copy_metadata imaginary##{non_existing_record_iid}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command' do
|
|
let(:other_project) { create(:project, :private) }
|
|
let(:source_issuable) { create(:issue, project: other_project) }
|
|
|
|
let(:content) { "/copy_metadata #{source_issuable.to_reference(project)}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
end
|
|
end
|
|
|
|
context '/duplicate command' do
|
|
it_behaves_like 'duplicate command' do
|
|
let(:issue_duplicate) { create(:issue, project: project) }
|
|
let(:content) { "/duplicate #{issue_duplicate.to_reference}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { '/duplicate' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
context 'cross project references' do
|
|
it_behaves_like 'duplicate command' do
|
|
let(:other_project) { create(:project, :public) }
|
|
let(:issue_duplicate) { create(:issue, project: other_project) }
|
|
let(:content) { "/duplicate #{issue_duplicate.to_reference(project)}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', _('Failed to mark this issue as a duplicate because referenced issue was not found.') do
|
|
let(:content) { "/duplicate imaginary##{non_existing_record_iid}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', _('Failed to mark this issue as a duplicate because referenced issue was not found.') do
|
|
let(:other_project) { create(:project, :private) }
|
|
let(:issue_duplicate) { create(:issue, project: other_project) }
|
|
|
|
let(:content) { "/duplicate #{issue_duplicate.to_reference(project)}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'when current_user cannot :admin_issue' do
|
|
let(:visitor) { create(:user) }
|
|
let(:issue) { create(:issue, project: project, author: visitor) }
|
|
let(:service) { described_class.new(project, visitor) }
|
|
|
|
it_behaves_like 'failed command', 'Could not apply assign command.' do
|
|
let(:content) { "/assign @#{developer.username}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply unassign command.' do
|
|
let(:content) { '/unassign' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply milestone command.' do
|
|
let(:content) { "/milestone %#{milestone.title}" }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply remove_milestone command.' do
|
|
let(:content) { '/remove_milestone' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply label command.' do
|
|
let(:content) { %(/label ~"#{inprogress.title}" ~#{bug.title} ~unknown) }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply unlabel command.' do
|
|
let(:content) { %(/unlabel ~"#{inprogress.title}") }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply relabel command.' do
|
|
let(:content) { %(/relabel ~"#{inprogress.title}") }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply due command.' do
|
|
let(:content) { '/due tomorrow' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply remove_due_date command.' do
|
|
let(:content) { '/remove_due_date' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply confidential command.' do
|
|
let(:content) { '/confidential' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply lock command.' do
|
|
let(:content) { '/lock' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply unlock command.' do
|
|
let(:content) { '/unlock' }
|
|
let(:issuable) { issue }
|
|
end
|
|
end
|
|
|
|
context '/award command' do
|
|
it_behaves_like 'award command' do
|
|
let(:content) { '/award :100:' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'award command' do
|
|
let(:content) { '/award :100:' }
|
|
let(:issuable) { merge_request }
|
|
end
|
|
|
|
context 'ignores command with no argument' do
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { '/award' }
|
|
let(:issuable) { issue }
|
|
end
|
|
end
|
|
|
|
context 'ignores non-existing / invalid emojis' do
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { '/award noop' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { '/award :lorem_ipsum:' }
|
|
let(:issuable) { issue }
|
|
end
|
|
end
|
|
|
|
context 'if issuable is a Commit' do
|
|
let(:content) { '/award :100:' }
|
|
let(:issuable) { commit }
|
|
|
|
it_behaves_like 'failed command', 'Could not apply award command.'
|
|
end
|
|
end
|
|
|
|
context '/shrug command' do
|
|
it_behaves_like 'shrug command' do
|
|
let(:content) { '/shrug people are people' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'shrug command' do
|
|
let(:content) { '/shrug' }
|
|
let(:issuable) { issue }
|
|
end
|
|
end
|
|
|
|
context '/tableflip command' do
|
|
it_behaves_like 'tableflip command' do
|
|
let(:content) { '/tableflip curse your sudden but enviable betrayal' }
|
|
let(:issuable) { issue }
|
|
end
|
|
|
|
it_behaves_like 'tableflip command' do
|
|
let(:content) { '/tableflip' }
|
|
let(:issuable) { issue }
|
|
end
|
|
end
|
|
|
|
context '/target_branch command' do
|
|
let(:non_empty_project) { create(:project, :repository) }
|
|
let(:another_merge_request) { create(:merge_request, author: developer, source_project: non_empty_project) }
|
|
let(:service) { described_class.new(non_empty_project, developer)}
|
|
|
|
it 'updates target_branch if /target_branch command is executed' do
|
|
_, updates, _ = service.execute('/target_branch merge-test', merge_request)
|
|
|
|
expect(updates).to eq(target_branch: 'merge-test')
|
|
end
|
|
|
|
it 'handles blanks around param' do
|
|
_, updates, _ = service.execute('/target_branch merge-test ', merge_request)
|
|
|
|
expect(updates).to eq(target_branch: 'merge-test')
|
|
end
|
|
|
|
context 'ignores command with no argument' do
|
|
it_behaves_like 'failed command', 'Could not apply target_branch command.' do
|
|
let(:content) { '/target_branch' }
|
|
let(:issuable) { another_merge_request }
|
|
end
|
|
end
|
|
|
|
context 'ignores non-existing target branch' do
|
|
it_behaves_like 'failed command', 'Could not apply target_branch command.' do
|
|
let(:content) { '/target_branch totally_non_existing_branch' }
|
|
let(:issuable) { another_merge_request }
|
|
end
|
|
end
|
|
|
|
it 'returns the target_branch message' do
|
|
_, _, message = service.execute('/target_branch merge-test', merge_request)
|
|
|
|
expect(message).to eq('Set target branch to merge-test.')
|
|
end
|
|
end
|
|
|
|
context '/board_move command' do
|
|
let_it_be(:todo) { create(:label, project: project, title: 'To Do') }
|
|
let_it_be(:inreview) { create(:label, project: project, title: 'In Review') }
|
|
let(:content) { %{/board_move ~"#{inreview.title}"} }
|
|
|
|
let_it_be(:board) { create(:board, project: project) }
|
|
let_it_be(:todo_list) { create(:list, board: board, label: todo) }
|
|
let_it_be(:inreview_list) { create(:list, board: board, label: inreview) }
|
|
let_it_be(:inprogress_list) { create(:list, board: board, label: inprogress) }
|
|
|
|
it 'populates remove_label_ids for all current board columns' do
|
|
issue.update!(label_ids: [todo.id, inprogress.id])
|
|
|
|
_, updates, _ = service.execute(content, issue)
|
|
|
|
expect(updates[:remove_label_ids]).to match_array([todo.id, inprogress.id])
|
|
end
|
|
|
|
it 'populates add_label_ids with the id of the given label' do
|
|
_, updates, _ = service.execute(content, issue)
|
|
|
|
expect(updates[:add_label_ids]).to eq([inreview.id])
|
|
end
|
|
|
|
it 'does not include the given label id in remove_label_ids' do
|
|
issue.update!(label_ids: [todo.id, inreview.id])
|
|
|
|
_, updates, _ = service.execute(content, issue)
|
|
|
|
expect(updates[:remove_label_ids]).to match_array([todo.id])
|
|
end
|
|
|
|
it 'does not remove label ids that are not lists on the board' do
|
|
issue.update!(label_ids: [todo.id, bug.id])
|
|
|
|
_, updates, _ = service.execute(content, issue)
|
|
|
|
expect(updates[:remove_label_ids]).to match_array([todo.id])
|
|
end
|
|
|
|
it 'returns board_move message' do
|
|
issue.update!(label_ids: [todo.id, inprogress.id])
|
|
|
|
_, _, message = service.execute(content, issue)
|
|
|
|
expect(message).to eq("Moved issue to ~#{inreview.id} column in the board.")
|
|
end
|
|
|
|
context 'if the project has multiple boards' do
|
|
let(:issuable) { issue }
|
|
|
|
before do
|
|
create(:board, project: project)
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply board_move command.'
|
|
end
|
|
|
|
context 'if the given label does not exist' do
|
|
let(:issuable) { issue }
|
|
let(:content) { '/board_move ~"Fake Label"' }
|
|
|
|
it_behaves_like 'failed command', 'Failed to move this issue because label was not found.'
|
|
end
|
|
|
|
context 'if multiple labels are given' do
|
|
let(:issuable) { issue }
|
|
let(:content) { %{/board_move ~"#{inreview.title}" ~"#{todo.title}"} }
|
|
|
|
it_behaves_like 'failed command', 'Failed to move this issue because only a single label can be provided.'
|
|
end
|
|
|
|
context 'if the given label is not a list on the board' do
|
|
let(:issuable) { issue }
|
|
let(:content) { %{/board_move ~"#{bug.title}"} }
|
|
|
|
it_behaves_like 'failed command', 'Failed to move this issue because label was not found.'
|
|
end
|
|
|
|
context 'if issuable is not an Issue' do
|
|
let(:issuable) { merge_request }
|
|
|
|
it_behaves_like 'failed command', 'Could not apply board_move command.'
|
|
end
|
|
end
|
|
|
|
context '/tag command' do
|
|
let(:issuable) { commit }
|
|
|
|
context 'ignores command with no argument' do
|
|
it_behaves_like 'failed command' do
|
|
let(:content) { '/tag' }
|
|
end
|
|
end
|
|
|
|
context 'tags a commit with a tag name' do
|
|
it_behaves_like 'tag command' do
|
|
let(:tag_name) { 'v1.2.3' }
|
|
let(:tag_message) { nil }
|
|
let(:content) { "/tag #{tag_name}" }
|
|
end
|
|
end
|
|
|
|
context 'tags a commit with a tag name and message' do
|
|
it_behaves_like 'tag command' do
|
|
let(:tag_name) { 'v1.2.3' }
|
|
let(:tag_message) { 'Stable release' }
|
|
let(:content) { "/tag #{tag_name} #{tag_message}" }
|
|
end
|
|
end
|
|
end
|
|
|
|
it 'limits to commands passed' do
|
|
content = "/shrug test\n/close"
|
|
|
|
text, commands = service.execute(content, issue, only: [:shrug])
|
|
|
|
expect(commands).to be_empty
|
|
expect(text).to eq("test #{described_class::SHRUG}\n/close")
|
|
end
|
|
|
|
it 'preserves leading whitespace' do
|
|
content = " - list\n\n/close\n\ntest\n\n"
|
|
|
|
text, _ = service.execute(content, issue)
|
|
|
|
expect(text).to eq(" - list\n\ntest")
|
|
end
|
|
|
|
it 'tracks MAU for commands' do
|
|
content = "/shrug test\n/assign me\n/milestone %4"
|
|
|
|
expect(Gitlab::UsageDataCounters::QuickActionActivityUniqueCounter)
|
|
.to receive(:track_unique_action)
|
|
.with('shrug', args: 'test', user: developer)
|
|
|
|
expect(Gitlab::UsageDataCounters::QuickActionActivityUniqueCounter)
|
|
.to receive(:track_unique_action)
|
|
.with('assign', args: 'me', user: developer)
|
|
|
|
expect(Gitlab::UsageDataCounters::QuickActionActivityUniqueCounter)
|
|
.to receive(:track_unique_action)
|
|
.with('milestone', args: '%4', user: developer)
|
|
|
|
service.execute(content, issue)
|
|
end
|
|
|
|
context '/create_merge_request command' do
|
|
let(:branch_name) { '1-feature' }
|
|
let(:content) { "/create_merge_request #{branch_name}" }
|
|
let(:issuable) { issue }
|
|
|
|
context 'if issuable is not an Issue' do
|
|
let(:issuable) { merge_request }
|
|
|
|
it_behaves_like 'failed command', 'Could not apply create_merge_request command.'
|
|
end
|
|
|
|
context "when logged user cannot create_merge_requests in the project" do
|
|
let(:project) { create(:project, :archived) }
|
|
|
|
before do
|
|
project.add_developer(developer)
|
|
end
|
|
|
|
it_behaves_like 'failed command', 'Could not apply create_merge_request command.'
|
|
end
|
|
|
|
context 'when logged user cannot push code to the project' do
|
|
let(:project) { create(:project, :private) }
|
|
let(:service) { described_class.new(project, create(:user)) }
|
|
|
|
it_behaves_like 'failed command', 'Could not apply create_merge_request command.'
|
|
end
|
|
|
|
it 'populates create_merge_request with branch_name and issue iid' do
|
|
_, updates, _ = service.execute(content, issuable)
|
|
|
|
expect(updates).to eq(create_merge_request: { branch_name: branch_name, issue_iid: issuable.iid })
|
|
end
|
|
|
|
it 'returns the create_merge_request message' do
|
|
_, _, message = service.execute(content, issuable)
|
|
|
|
expect(message).to eq("Created branch '#{branch_name}' and a merge request to resolve this issue.")
|
|
end
|
|
end
|
|
|
|
context 'submit_review command' do
|
|
using RSpec::Parameterized::TableSyntax
|
|
|
|
where(:note) do
|
|
[
|
|
'I like it',
|
|
'/submit_review'
|
|
]
|
|
end
|
|
|
|
with_them do
|
|
let(:content) { '/submit_review' }
|
|
let!(:draft_note) { create(:draft_note, note: note, merge_request: merge_request, author: developer) }
|
|
|
|
it 'submits the users current review' do
|
|
_, _, message = service.execute(content, merge_request)
|
|
|
|
expect { draft_note.reload }.to raise_error(ActiveRecord::RecordNotFound)
|
|
expect(message).to eq('Submitted the current review.')
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'relate command' do
|
|
let_it_be_with_refind(:group) { create(:group) }
|
|
|
|
shared_examples 'relate command' do
|
|
it 'relates issues' do
|
|
service.execute(content, issue)
|
|
|
|
expect(IssueLink.where(source: issue).map(&:target)).to match_array(issues_related)
|
|
end
|
|
end
|
|
|
|
context 'user is member of group' do
|
|
before do
|
|
group.add_developer(developer)
|
|
end
|
|
|
|
context 'relate a single issue' do
|
|
let(:other_issue) { create(:issue, project: project) }
|
|
let(:issues_related) { [other_issue] }
|
|
let(:content) { "/relate #{other_issue.to_reference}" }
|
|
|
|
it_behaves_like 'relate command'
|
|
end
|
|
|
|
context 'relate multiple issues at once' do
|
|
let(:second_issue) { create(:issue, project: project) }
|
|
let(:third_issue) { create(:issue, project: project) }
|
|
let(:issues_related) { [second_issue, third_issue] }
|
|
let(:content) { "/relate #{second_issue.to_reference} #{third_issue.to_reference}" }
|
|
|
|
it_behaves_like 'relate command'
|
|
end
|
|
|
|
context 'empty relate command' do
|
|
let(:issues_related) { [] }
|
|
let(:content) { '/relate' }
|
|
|
|
it_behaves_like 'relate command'
|
|
end
|
|
|
|
context 'already having related issues' do
|
|
let(:second_issue) { create(:issue, project: project) }
|
|
let(:third_issue) { create(:issue, project: project) }
|
|
let(:issues_related) { [second_issue, third_issue] }
|
|
let(:content) { "/relate #{third_issue.to_reference(project)}" }
|
|
|
|
before do
|
|
create(:issue_link, source: issue, target: second_issue)
|
|
end
|
|
|
|
it_behaves_like 'relate command'
|
|
end
|
|
|
|
context 'cross project' do
|
|
let(:another_group) { create(:group, :public) }
|
|
let(:other_project) { create(:project, group: another_group) }
|
|
|
|
before do
|
|
another_group.add_developer(developer)
|
|
end
|
|
|
|
context 'relate a cross project issue' do
|
|
let(:other_issue) { create(:issue, project: other_project) }
|
|
let(:issues_related) { [other_issue] }
|
|
let(:content) { "/relate #{other_issue.to_reference(project)}" }
|
|
|
|
it_behaves_like 'relate command'
|
|
end
|
|
|
|
context 'relate multiple cross projects issues at once' do
|
|
let(:second_issue) { create(:issue, project: other_project) }
|
|
let(:third_issue) { create(:issue, project: other_project) }
|
|
let(:issues_related) { [second_issue, third_issue] }
|
|
let(:content) { "/relate #{second_issue.to_reference(project)} #{third_issue.to_reference(project)}" }
|
|
|
|
it_behaves_like 'relate command'
|
|
end
|
|
|
|
context 'relate a non-existing issue' do
|
|
let(:issues_related) { [] }
|
|
let(:content) { "/relate imaginary##{non_existing_record_iid}" }
|
|
|
|
it_behaves_like 'relate command'
|
|
end
|
|
|
|
context 'relate a private issue' do
|
|
let(:private_project) { create(:project, :private) }
|
|
let(:other_issue) { create(:issue, project: private_project) }
|
|
let(:issues_related) { [] }
|
|
let(:content) { "/relate #{other_issue.to_reference(project)}" }
|
|
|
|
it_behaves_like 'relate command'
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'invite_email command' do
|
|
let_it_be(:issuable) { issue }
|
|
|
|
it_behaves_like 'failed command', "No email participants were added. Either none were provided, or they already exist." do
|
|
let(:content) { '/invite_email' }
|
|
end
|
|
|
|
context 'with existing email participant' do
|
|
let(:content) { '/invite_email a@gitlab.com' }
|
|
|
|
before do
|
|
issuable.issue_email_participants.create!(email: "a@gitlab.com")
|
|
end
|
|
|
|
it_behaves_like 'failed command', "No email participants were added. Either none were provided, or they already exist."
|
|
end
|
|
|
|
context 'with new email participants' do
|
|
let(:content) { '/invite_email a@gitlab.com b@gitlab.com' }
|
|
|
|
subject(:add_emails) { service.execute(content, issuable) }
|
|
|
|
it 'returns message' do
|
|
_, _, message = add_emails
|
|
|
|
expect(message).to eq('Added a@gitlab.com and b@gitlab.com.')
|
|
end
|
|
|
|
it 'adds 2 participants' do
|
|
expect { add_emails }.to change { issue.issue_email_participants.count }.by(2)
|
|
end
|
|
|
|
context 'with mixed case email' do
|
|
let(:content) { '/invite_email FirstLast@GitLab.com' }
|
|
|
|
it 'returns correctly cased message' do
|
|
_, _, message = add_emails
|
|
|
|
expect(message).to eq('Added FirstLast@GitLab.com.')
|
|
end
|
|
end
|
|
|
|
context 'with invalid email' do
|
|
let(:content) { '/invite_email a@gitlab.com bad_email' }
|
|
|
|
it 'only adds valid emails' do
|
|
expect { add_emails }.to change { issue.issue_email_participants.count }.by(1)
|
|
end
|
|
end
|
|
|
|
context 'with existing email' do
|
|
let(:content) { '/invite_email a@gitlab.com existing@gitlab.com' }
|
|
|
|
it 'only adds new emails' do
|
|
issue.issue_email_participants.create!(email: 'existing@gitlab.com')
|
|
|
|
expect { add_emails }.to change { issue.issue_email_participants.count }.by(1)
|
|
end
|
|
|
|
it 'only adds new (case insensitive) emails' do
|
|
issue.issue_email_participants.create!(email: 'EXISTING@gitlab.com')
|
|
|
|
expect { add_emails }.to change { issue.issue_email_participants.count }.by(1)
|
|
end
|
|
end
|
|
|
|
context 'with duplicate email' do
|
|
let(:content) { '/invite_email a@gitlab.com a@gitlab.com' }
|
|
|
|
it 'only adds unique new emails' do
|
|
expect { add_emails }.to change { issue.issue_email_participants.count }.by(1)
|
|
end
|
|
end
|
|
|
|
context 'with more than 6 emails' do
|
|
let(:content) { '/invite_email a@gitlab.com b@gitlab.com c@gitlab.com d@gitlab.com e@gitlab.com f@gitlab.com g@gitlab.com' }
|
|
|
|
it 'only adds 6 new emails' do
|
|
expect { add_emails }.to change { issue.issue_email_participants.count }.by(6)
|
|
end
|
|
end
|
|
|
|
context 'with feature flag disabled' do
|
|
before do
|
|
stub_feature_flags(issue_email_participants: false)
|
|
end
|
|
|
|
it 'does not add any participants' do
|
|
expect { add_emails }.not_to change { issue.issue_email_participants.count }
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
context 'severity command' do
|
|
let_it_be_with_reload(:issuable) { create(:incident, project: project) }
|
|
|
|
subject(:set_severity) { service.execute(content, issuable) }
|
|
|
|
it_behaves_like 'failed command', 'No severity matches the provided parameter' do
|
|
let(:content) { '/severity something' }
|
|
end
|
|
|
|
shared_examples 'updates the severity' do |new_severity|
|
|
it do
|
|
expect { set_severity }.to change { issuable.severity }.from('unknown').to(new_severity)
|
|
end
|
|
end
|
|
|
|
context 'when quick action is used on creation' do
|
|
let(:content) { '/severity s3' }
|
|
let(:issuable) { build(:incident, project: project) }
|
|
|
|
it_behaves_like 'updates the severity', 'medium'
|
|
|
|
context 'issuable does not support severity' do
|
|
let(:issuable) { build(:issue, project: project) }
|
|
|
|
it_behaves_like 'failed command', ''
|
|
end
|
|
end
|
|
|
|
context 'severity given with S format' do
|
|
let(:content) { '/severity s3' }
|
|
|
|
it_behaves_like 'updates the severity', 'medium'
|
|
end
|
|
|
|
context 'severity given with number format' do
|
|
let(:content) { '/severity 3' }
|
|
|
|
it_behaves_like 'updates the severity', 'medium'
|
|
end
|
|
|
|
context 'severity given with text format' do
|
|
let(:content) { '/severity medium' }
|
|
|
|
it_behaves_like 'updates the severity', 'medium'
|
|
end
|
|
|
|
context 'an issuable that does not support severity' do
|
|
let_it_be_with_reload(:issuable) { create(:issue, project: project) }
|
|
|
|
it_behaves_like 'failed command', 'Could not apply severity command.' do
|
|
let(:content) { '/severity s3' }
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
describe '#explain' do
|
|
let(:service) { described_class.new(project, developer) }
|
|
let(:merge_request) { create(:merge_request, source_project: project) }
|
|
|
|
describe 'close command' do
|
|
let(:content) { '/close' }
|
|
|
|
it 'includes issuable name' do
|
|
_, explanations = service.explain(content, issue)
|
|
|
|
expect(explanations).to eq(['Closes this issue.'])
|
|
end
|
|
end
|
|
|
|
describe 'reopen command' do
|
|
let(:content) { '/reopen' }
|
|
let(:merge_request) { create(:merge_request, :closed, source_project: project) }
|
|
|
|
it 'includes issuable name' do
|
|
_, explanations = service.explain(content, merge_request)
|
|
|
|
expect(explanations).to eq(['Reopens this merge request.'])
|
|
end
|
|
end
|
|
|
|
describe 'title command' do
|
|
let(:content) { '/title This is new title' }
|
|
|
|
it 'includes new title' do
|
|
_, explanations = service.explain(content, issue)
|
|
|
|
expect(explanations).to eq(['Changes the title to "This is new title".'])
|
|
end
|
|
end
|
|
|
|
describe 'assign command' do
|
|
let(:content) { "/assign @#{developer.username} do it!" }
|
|
|
|
it 'includes only the user reference' do
|
|
_, explanations = service.explain(content, merge_request)
|
|
|
|
expect(explanations).to eq(["Assigns @#{developer.username}."])
|
|
end
|
|
end
|
|
|
|
describe 'unassign command' do
|
|
let(:content) { '/unassign' }
|
|
let(:issue) { create(:issue, project: project, assignees: [developer]) }
|
|
|
|
it 'includes current assignee reference' do
|
|
_, explanations = service.explain(content, issue)
|
|
|
|
expect(explanations).to eq(["Removes assignee @#{developer.username}."])
|
|
end
|
|
end
|
|
|
|
describe 'unassign_reviewer command' do
|
|
let(:content) { '/unassign_reviewer' }
|
|
let(:merge_request) { create(:merge_request, source_project: project, reviewers: [developer]) }
|
|
|
|
it 'includes current assignee reference' do
|
|
_, explanations = service.explain(content, merge_request)
|
|
|
|
expect(explanations).to eq(["Removes reviewer @#{developer.username}."])
|
|
end
|
|
end
|
|
|
|
describe 'assign_reviewer command' do
|
|
let(:content) { "/assign_reviewer #{developer.to_reference}" }
|
|
let(:merge_request) { create(:merge_request, source_project: project, assignees: [developer]) }
|
|
|
|
it 'includes only the user reference' do
|
|
_, explanations = service.explain(content, merge_request)
|
|
|
|
expect(explanations).to eq(["Assigns #{developer.to_reference} as reviewer."])
|
|
end
|
|
end
|
|
|
|
describe 'milestone command' do
|
|
let(:content) { '/milestone %wrong-milestone' }
|
|
let!(:milestone) { create(:milestone, project: project, title: '9.10') }
|
|
|
|
it 'is empty when milestone reference is wrong' do
|
|
_, explanations = service.explain(content, issue)
|
|
|
|
expect(explanations).to eq([])
|
|
end
|
|
end
|
|
|
|
describe 'remove milestone command' do
|
|
let(:content) { '/remove_milestone' }
|
|
let(:merge_request) { create(:merge_request, source_project: project, milestone: milestone) }
|
|
|
|
it 'includes current milestone name' do
|
|
_, explanations = service.explain(content, merge_request)
|
|
|
|
expect(explanations).to eq(['Removes %"9.10" milestone.'])
|
|
end
|
|
end
|
|
|
|
describe 'label command' do
|
|
let(:content) { '/label ~missing' }
|
|
let!(:label) { create(:label, project: project) }
|
|
|
|
it 'is empty when there are no correct labels' do
|
|
_, explanations = service.explain(content, issue)
|
|
|
|
expect(explanations).to eq([])
|
|
end
|
|
end
|
|
|
|
describe 'unlabel command' do
|
|
let(:content) { '/unlabel' }
|
|
|
|
it 'says all labels if no parameter provided' do
|
|
merge_request.update!(label_ids: [bug.id])
|
|
_, explanations = service.explain(content, merge_request)
|
|
|
|
expect(explanations).to eq([_('Removes all labels.')])
|
|
end
|
|
end
|
|
|
|
describe 'relabel command' do
|
|
let(:content) { "/relabel #{bug.title}" }
|
|
let(:feature) { create(:label, project: project, title: 'Feature') }
|
|
|
|
it 'includes label name' do
|
|
issue.update!(label_ids: [feature.id])
|
|
_, explanations = service.explain(content, issue)
|
|
|
|
expect(explanations).to eq(["Replaces all labels with ~#{bug.id} label."])
|
|
end
|
|
end
|
|
|
|
describe 'subscribe command' do
|
|
let(:content) { '/subscribe' }
|
|
|
|
it 'includes issuable name' do
|
|
_, explanations = service.explain(content, issue)
|
|
|
|
expect(explanations).to eq(['Subscribes to this issue.'])
|
|
end
|
|
end
|
|
|
|
describe 'unsubscribe command' do
|
|
let(:content) { '/unsubscribe' }
|
|
|
|
it 'includes issuable name' do
|
|
merge_request.subscribe(developer, project)
|
|
_, explanations = service.explain(content, merge_request)
|
|
|
|
expect(explanations).to eq(['Unsubscribes from this merge request.'])
|
|
end
|
|
end
|
|
|
|
describe 'due command' do
|
|
let(:content) { '/due April 1st 2016' }
|
|
|
|
it 'includes the date' do
|
|
_, explanations = service.explain(content, issue)
|
|
|
|
expect(explanations).to eq(['Sets the due date to Apr 1, 2016.'])
|
|
end
|
|
end
|
|
|
|
describe 'draft command' do
|
|
let(:content) { '/draft' }
|
|
|
|
it 'includes the new status' do
|
|
_, explanations = service.explain(content, merge_request)
|
|
|
|
expect(explanations).to eq(['Marks this merge request as a draft.'])
|
|
end
|
|
end
|
|
|
|
describe 'award command' do
|
|
let(:content) { '/award :confetti_ball: ' }
|
|
|
|
it 'includes the emoji' do
|
|
_, explanations = service.explain(content, issue)
|
|
|
|
expect(explanations).to eq(['Toggles :confetti_ball: emoji award.'])
|
|
end
|
|
end
|
|
|
|
describe 'estimate command' do
|
|
let(:content) { '/estimate 79d' }
|
|
|
|
it 'includes the formatted duration' do
|
|
_, explanations = service.explain(content, merge_request)
|
|
|
|
expect(explanations).to eq(['Sets time estimate to 3mo 3w 4d.'])
|
|
end
|
|
end
|
|
|
|
describe 'spend command' do
|
|
it 'includes the formatted duration and proper verb when using /spend' do
|
|
_, explanations = service.explain('/spend -120m', issue)
|
|
|
|
expect(explanations).to eq(['Subtracts 2h spent time.'])
|
|
end
|
|
|
|
it 'includes the formatted duration and proper verb when using /spent' do
|
|
_, explanations = service.explain('/spent -120m', issue)
|
|
|
|
expect(explanations).to eq(['Subtracts 2h spent time.'])
|
|
end
|
|
end
|
|
|
|
describe 'target branch command' do
|
|
let(:content) { '/target_branch my-feature ' }
|
|
|
|
it 'includes the branch name' do
|
|
_, explanations = service.explain(content, merge_request)
|
|
|
|
expect(explanations).to eq(['Sets target branch to my-feature.'])
|
|
end
|
|
end
|
|
|
|
describe 'board move command' do
|
|
let(:content) { "/board_move ~#{bug.title}" }
|
|
let!(:board) { create(:board, project: project) }
|
|
|
|
it 'includes the label name' do
|
|
_, explanations = service.explain(content, issue)
|
|
|
|
expect(explanations).to eq(["Moves issue to ~#{bug.id} column in the board."])
|
|
end
|
|
end
|
|
|
|
describe 'move issue to another project command' do
|
|
let(:content) { '/move test/project' }
|
|
|
|
it 'includes the project name' do
|
|
_, explanations = service.explain(content, issue)
|
|
|
|
expect(explanations).to eq(["Moves this issue to test/project."])
|
|
end
|
|
end
|
|
|
|
describe 'tag a commit' do
|
|
describe 'with a tag name' do
|
|
context 'without a message' do
|
|
let(:content) { '/tag v1.2.3' }
|
|
|
|
it 'includes the tag name only' do
|
|
_, explanations = service.explain(content, commit)
|
|
|
|
expect(explanations).to eq(["Tags this commit to v1.2.3."])
|
|
end
|
|
end
|
|
|
|
context 'with an empty message' do
|
|
let(:content) { '/tag v1.2.3 ' }
|
|
|
|
it 'includes the tag name only' do
|
|
_, explanations = service.explain(content, commit)
|
|
|
|
expect(explanations).to eq(["Tags this commit to v1.2.3."])
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'with a tag name and message' do
|
|
let(:content) { '/tag v1.2.3 Stable release' }
|
|
|
|
it 'includes the tag name and message' do
|
|
_, explanations = service.explain(content, commit)
|
|
|
|
expect(explanations).to eq(["Tags this commit to v1.2.3 with \"Stable release\"."])
|
|
end
|
|
end
|
|
end
|
|
|
|
describe 'create a merge request' do
|
|
context 'with no branch name' do
|
|
let(:content) { '/create_merge_request' }
|
|
|
|
it 'uses the default branch name' do
|
|
_, explanations = service.explain(content, issue)
|
|
|
|
expect(explanations).to eq([_('Creates a branch and a merge request to resolve this issue.')])
|
|
end
|
|
|
|
it 'returns the execution message using the default branch name' do
|
|
_, _, message = service.execute(content, issue)
|
|
|
|
expect(message).to eq(_('Created a branch and a merge request to resolve this issue.'))
|
|
end
|
|
end
|
|
|
|
context 'with a branch name' do
|
|
let(:content) { '/create_merge_request foo' }
|
|
|
|
it 'uses the given branch name' do
|
|
_, explanations = service.explain(content, issue)
|
|
|
|
expect(explanations).to eq(["Creates branch 'foo' and a merge request to resolve this issue."])
|
|
end
|
|
|
|
it 'returns the execution message using the given branch name' do
|
|
_, _, message = service.execute(content, issue)
|
|
|
|
expect(message).to eq("Created branch 'foo' and a merge request to resolve this issue.")
|
|
end
|
|
end
|
|
end
|
|
|
|
describe "#commands_executed_count" do
|
|
it 'counts commands executed' do
|
|
content = "/close and \n/assign me and \n/title new title"
|
|
|
|
service.execute(content, issue)
|
|
|
|
expect(service.commands_executed_count).to eq(3)
|
|
end
|
|
end
|
|
end
|
|
end
|