# frozen_string_literal: true require 'spec_helper' describe QuickActions::InterpretService do let(:project) { create(:project, :public) } let(:developer) { create(:user) } let(:developer2) { create(:user) } let(:issue) { create(:issue, project: project) } let(:milestone) { create(:milestone, project: project, title: '9.10') } let(:commit) { create(:commit, project: project) } let(:inprogress) { create(:label, project: project, title: 'In Progress') } let(:helmchart) { create(:label, project: project, title: 'Helm Chart Registry') } let(:bug) { create(:label, project: project, title: 'Bug') } let(:note) { build(:note, commit_id: merge_request.diff_head_sha) } let(:service) { described_class.new(project, developer) } before do stub_licensed_features(multiple_issue_assignees: false, multiple_merge_request_assignees: false) project.add_developer(developer) 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 helmchart # populate the label _, 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 helmchart # populate the label 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 'wip command' do it 'returns wip_event: "wip" if content contains /wip' 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 Work In Progress.") end end shared_examples 'unwip command' do it 'returns wip_event: "unwip" if content contains /wip' 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 Work In Progress.") 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 _, updates, _ = service.execute(content, issuable) expect(updates).to eq(spend_time: { duration: 3600, user_id: developer.id, spent_at: DateTime.current.to_date }) end it 'returns the spend_time message including the formatted duration and verb' do _, _, message = service.execute('/spend -120m', issuable) expect(message).to eq('Subtracted 2h spent time.') end end shared_examples 'spend command with negative time' do it 'populates spend_time: -1800 if content contains /spend -30m' do _, updates, _ = service.execute(content, issuable) expect(updates).to eq(spend_time: { duration: -1800, user_id: developer.id, spent_at: DateTime.current.to_date }) 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 'empty 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) { create(:project, :repository) } 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) { create(:project, :repository) } 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 'does not return the success message' do _, _, message = service.execute(content, issuable) expect(message).to be_empty 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 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 'empty 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 'empty command' do let(:content) { "/merge" } let(:issuable) { merge_request } end end context 'when sha is missing' do let(:project) { create(:project, :repository) } 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 'empty command' do let(:content) { "/merge" } let(:issuable) { issue } end end context 'non persisted merge request cant be merged' do it_behaves_like 'empty 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 'empty 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 'empty 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) { merge_request } end end # CE does not have multiple assignees context 'assign command with multiple assignees' do before do project.add_developer(developer2) end it_behaves_like 'assign command' do let(:content) { "/assign @#{developer.username} @#{developer2.username}" } let(:issuable) { issue } end it_behaves_like 'assign command', :quarantine 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 'empty command', "Failed to assign a user because no user was found." do let(:content) { '/assign @abcd1234' } let(:issuable) { issue } end it_behaves_like 'empty command', "Failed to assign a user because no user was found." do let(:content) { '/assign' } let(:issuable) { issue } 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 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 context 'only group milestones available' do let(:ancestor_group) { create(:group) } let(:group) { create(:group, parent: ancestor_group) } let(:project) { create(:project, :public, namespace: group) } let(:milestone) { create(:milestone, group: ancestor_group, title: '10.0') } 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 'empty 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 'wip command' do let(:content) { '/wip' } let(:issuable) { merge_request } end it_behaves_like 'unwip command' do let(:content) { '/wip' } let(:issuable) { merge_request } end it_behaves_like 'empty 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 'empty command' do let(:content) { '/estimate' } let(:issuable) { issue } end it_behaves_like 'empty 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 with negative time' do let(:content) { '/spend -30m' } 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 invalid date' do let(:content) { '/spend 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 'empty command' do let(:content) { '/spend' } let(:issuable) { issue } end it_behaves_like 'empty command' do let(:content) { '/spend 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 '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 'empty 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 'empty 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 'empty 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 'empty command' do let(:content) { "/copy_metadata imaginary##{non_existing_record_iid}" } let(:issuable) { issue } end it_behaves_like 'empty 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 'empty 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 'empty 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 'empty 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 'empty command' do let(:content) { "/assign @#{developer.username}" } let(:issuable) { issue } end it_behaves_like 'empty command' do let(:content) { '/unassign' } let(:issuable) { issue } end it_behaves_like 'empty command' do let(:content) { "/milestone %#{milestone.title}" } let(:issuable) { issue } end it_behaves_like 'empty command' do let(:content) { '/remove_milestone' } let(:issuable) { issue } end it_behaves_like 'empty command' do let(:content) { %(/label ~"#{inprogress.title}" ~#{bug.title} ~unknown) } let(:issuable) { issue } end it_behaves_like 'empty command' do let(:content) { %(/unlabel ~"#{inprogress.title}") } let(:issuable) { issue } end it_behaves_like 'empty command' do let(:content) { %(/relabel ~"#{inprogress.title}") } let(:issuable) { issue } end it_behaves_like 'empty command' do let(:content) { '/due tomorrow' } let(:issuable) { issue } end it_behaves_like 'empty command' do let(:content) { '/remove_due_date' } let(:issuable) { issue } end it_behaves_like 'empty command' do let(:content) { '/confidential' } let(:issuable) { issue } end it_behaves_like 'empty command' do let(:content) { '/lock' } let(:issuable) { issue } end it_behaves_like 'empty 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 'empty command' do let(:content) { '/award' } let(:issuable) { issue } end end context 'ignores non-existing / invalid emojis' do it_behaves_like 'empty command' do let(:content) { '/award noop' } let(:issuable) { issue } end it_behaves_like 'empty 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 'empty 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 'empty command' do let(:content) { '/target_branch' } let(:issuable) { another_merge_request } end end context 'ignores non-existing target branch' do it_behaves_like 'empty 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(:todo) { create(:label, project: project, title: 'To Do') } let(:inreview) { create(:label, project: project, title: 'In Review') } let(:content) { %{/board_move ~"#{inreview.title}"} } let!(:board) { create(:board, project: project) } let!(:todo_list) { create(:list, board: board, label: todo) } let!(:inreview_list) { create(:list, board: board, label: inreview) } let!(: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 'empty command' end context 'if the given label does not exist' do let(:issuable) { issue } let(:content) { '/board_move ~"Fake Label"' } it_behaves_like 'empty 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 'empty 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 'empty 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 'empty command' end end context '/tag command' do let(:issuable) { commit } context 'ignores command with no argument' do it_behaves_like 'empty 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 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 'empty command' end context "when logged user cannot create_merge_requests in the project" do let(:project) { create(:project, :archived) } it_behaves_like 'empty 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 'empty 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 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 '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' } let!(:bug) { create(:label, project: project, title: 'Bug') } 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 'wip command' do let(:content) { '/wip' } it 'includes the new status' do _, explanations = service.explain(content, merge_request) expect(explanations).to eq(['Marks this merge request as Work In Progress.']) 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 let(:content) { '/spend -120m' } it 'includes the formatted duration and proper verb' do _, explanations = service.explain(content, 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' } let!(:bug) { create(:label, project: project, title: 'bug') } 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