# frozen_string_literal: true require 'spec_helper' RSpec.describe ::SystemNotes::IssuablesService do include ProjectForksHelper let_it_be(:group) { create(:group) } let_it_be(:project) { create(:project, :repository, group: group) } let_it_be(:author) { create(:user) } let(:noteable) { create(:issue, project: project) } let(:issue) { noteable } let(:service) { described_class.new(noteable: noteable, project: project, author: author) } describe '#relate_issue' do let(:noteable_ref) { create(:issue) } subject { service.relate_issue(noteable_ref) } it_behaves_like 'a system note' do let(:action) { 'relate' } end context 'when issue marks another as related' do it 'sets the note text' do expect(subject.note).to eq "marked this issue as related to #{noteable_ref.to_reference(project)}" end end end describe '#unrelate_issue' do let(:noteable_ref) { create(:issue) } subject { service.unrelate_issue(noteable_ref) } it_behaves_like 'a system note' do let(:action) { 'unrelate' } end context 'when issue relation is removed' do it 'sets the note text' do expect(subject.note).to eq "removed the relation with #{noteable_ref.to_reference(project)}" end end end describe '#change_assignee' do subject { service.change_assignee(assignee) } let(:assignee) { create(:user) } it_behaves_like 'a system note' do let(:action) { 'assignee' } end context 'when assignee added' do it_behaves_like 'a note with overridable created_at' it 'sets the note text' do expect(subject.note).to eq "assigned to @#{assignee.username}" end end context 'when assignee removed' do let(:assignee) { nil } it_behaves_like 'a note with overridable created_at' it 'sets the note text' do expect(subject.note).to eq 'removed assignee' end end end describe '#change_issuable_assignees' do subject { service.change_issuable_assignees([assignee]) } let(:assignee) { create(:user) } let(:assignee1) { create(:user) } let(:assignee2) { create(:user) } let(:assignee3) { create(:user) } it_behaves_like 'a system note' do let(:action) { 'assignee' } end def build_note(old_assignees, new_assignees) issue.assignees = new_assignees service.change_issuable_assignees(old_assignees).note end it_behaves_like 'a note with overridable created_at' it 'builds a correct phrase when an assignee is added to a non-assigned issue' do expect(build_note([], [assignee1])).to eq "assigned to @#{assignee1.username}" end it 'builds a correct phrase when assignee removed' do expect(build_note([assignee1], [])).to eq "unassigned @#{assignee1.username}" end it 'builds a correct phrase when assignees changed' do expect(build_note([assignee1], [assignee2])).to eq \ "assigned to @#{assignee2.username} and unassigned @#{assignee1.username}" end it 'builds a correct phrase when three assignees removed and one added' do expect(build_note([assignee, assignee1, assignee2], [assignee3])).to eq \ "assigned to @#{assignee3.username} and unassigned @#{assignee.username}, @#{assignee1.username}, and @#{assignee2.username}" end it 'builds a correct phrase when one assignee changed from a set' do expect(build_note([assignee, assignee1], [assignee, assignee2])).to eq \ "assigned to @#{assignee2.username} and unassigned @#{assignee1.username}" end it 'builds a correct phrase when one assignee removed from a set' do expect(build_note([assignee, assignee1, assignee2], [assignee, assignee1])).to eq \ "unassigned @#{assignee2.username}" end it 'builds a correct phrase when the locale is different' do Gitlab::I18n.with_locale('pt-BR') do expect(build_note([assignee, assignee1, assignee2], [assignee3])).to eq \ "assigned to @#{assignee3.username} and unassigned @#{assignee.username}, @#{assignee1.username}, and @#{assignee2.username}" end end end describe '#change_issuable_reviewers' do subject { service.change_issuable_reviewers([reviewer]) } let_it_be(:noteable) { create(:merge_request, :simple, source_project: project) } let_it_be(:reviewer) { create(:user) } let_it_be(:reviewer1) { create(:user) } let_it_be(:reviewer2) { create(:user) } let_it_be(:reviewer3) { create(:user) } it_behaves_like 'a system note' do let(:action) { 'reviewer' } end def build_note(old_reviewers, new_reviewers) noteable.reviewers = new_reviewers service.change_issuable_reviewers(old_reviewers).note end it 'builds a correct phrase when a reviewer is added to a non-assigned merge request' do expect(build_note([], [reviewer1])).to eq "requested review from @#{reviewer1.username}" end it 'builds a correct phrase when reviewer is removed' do expect(build_note([reviewer], [])).to eq "removed review request for @#{reviewer.username}" end it 'builds a correct phrase when reviewers changed' do expect(build_note([reviewer1], [reviewer2])).to( eq("requested review from @#{reviewer2.username} and removed review request for @#{reviewer1.username}") ) end it 'builds a correct phrase when three reviewers removed and one added' do expect(build_note([reviewer, reviewer1, reviewer2], [reviewer3])).to( eq("requested review from @#{reviewer3.username} and removed review request for @#{reviewer.username}, @#{reviewer1.username}, and @#{reviewer2.username}") ) end it 'builds a correct phrase when one reviewer is changed from a set' do expect(build_note([reviewer, reviewer1], [reviewer, reviewer2])).to( eq("requested review from @#{reviewer2.username} and removed review request for @#{reviewer1.username}") ) end it 'builds a correct phrase when one reviewer removed from a set' do expect(build_note([reviewer, reviewer1, reviewer2], [reviewer, reviewer1])).to( eq( "removed review request for @#{reviewer2.username}") ) end it 'builds a correct phrase when the locale is different' do Gitlab::I18n.with_locale('pt-BR') do expect(build_note([reviewer, reviewer1, reviewer2], [reviewer3])).to( eq("requested review from @#{reviewer3.username} and removed review request for @#{reviewer.username}, @#{reviewer1.username}, and @#{reviewer2.username}") ) end end end describe '#change_status' do subject { service.change_status(status, source) } let(:status) { 'reopened' } let(:source) { nil } it 'creates a resource state event' do expect { subject }.to change { ResourceStateEvent.count }.by(1) end end describe '#request_attention' do subject { service.request_attention(user) } let(:user) { create(:user) } it_behaves_like 'a system note' do let(:action) { 'attention_requested' } end context 'when attention requested' do it_behaves_like 'a note with overridable created_at' it 'sets the note text' do expect(subject.note).to eq "requested attention from @#{user.username}" end end end describe '#remove_attention_request' do subject { service.remove_attention_request(user) } let(:user) { create(:user) } it_behaves_like 'a system note' do let(:action) { 'attention_request_removed' } end context 'when attention request is removed' do it_behaves_like 'a note with overridable created_at' it 'sets the note text' do expect(subject.note).to eq "removed attention request from @#{user.username}" end end end describe '#change_title' do let(:noteable) { create(:issue, project: project, title: 'Lorem ipsum') } subject { service.change_title('Old title') } context 'when noteable responds to `title`' do it_behaves_like 'a system note' do let(:action) { 'title' } end it_behaves_like 'a note with overridable created_at' it 'sets the note text' do expect(subject.note) .to eq "changed title from **{-Old title-}** to **{+Lorem ipsum+}**" end end end describe '#change_description' do subject { service.change_description } context 'when noteable responds to `description`' do it_behaves_like 'a system note' do let(:action) { 'description' } end it_behaves_like 'a note with overridable created_at' it 'sets the note text' do expect(subject.note).to eq('changed the description') end it 'associates the related description version' do noteable.update!(description: 'New description') description_version_id = subject.system_note_metadata.description_version_id expect(description_version_id).not_to be_nil expect(description_version_id).to eq(noteable.saved_description_version.id) end end end describe '#change_issue_confidentiality' do subject { service.change_issue_confidentiality } context 'issue has been made confidential' do before do noteable.update_attribute(:confidential, true) end it_behaves_like 'a system note' do let(:action) { 'confidential' } end it 'sets the note text' do expect(subject.note).to eq 'made the issue confidential' end end context 'issue has been made visible' do it_behaves_like 'a system note' do let(:action) { 'visible' } end it 'sets the note text' do expect(subject.note).to eq 'made the issue visible to everyone' end end end describe '#cross_reference' do let(:service) { described_class.new(noteable: noteable, author: author) } let(:mentioned_in) { create(:issue, project: project) } subject { service.cross_reference(mentioned_in) } it_behaves_like 'a system note' do let(:action) { 'cross_reference' } end context 'when cross-reference disallowed' do before do expect_next_instance_of(described_class) do |instance| expect(instance).to receive(:cross_reference_disallowed?).and_return(true) end end it 'returns nil' do expect(subject).to be_nil end it 'does not create a system note metadata record' do expect { subject }.not_to change { SystemNoteMetadata.count } end end context 'when cross-reference allowed' do before do expect_next_instance_of(described_class) do |instance| expect(instance).to receive(:cross_reference_disallowed?).and_return(false) end end it_behaves_like 'a system note' do let(:action) { 'cross_reference' } end it_behaves_like 'a note with overridable created_at' describe 'note_body' do context 'cross-project' do let(:project2) { create(:project, :repository) } let(:mentioned_in) { create(:issue, project: project2) } context 'from Commit' do let(:mentioned_in) { project2.repository.commit } it 'references the mentioning commit' do expect(subject.note).to eq "mentioned in commit #{mentioned_in.to_reference(project)}" end end context 'from non-Commit' do it 'references the mentioning object' do expect(subject.note).to eq "mentioned in issue #{mentioned_in.to_reference(project)}" end end end context 'within the same project' do context 'from Commit' do let(:mentioned_in) { project.repository.commit } it 'references the mentioning commit' do expect(subject.note).to eq "mentioned in commit #{mentioned_in.to_reference}" end end context 'from non-Commit' do it 'references the mentioning object' do expect(subject.note).to eq "mentioned in issue #{mentioned_in.to_reference}" end end end end context 'with external issue' do let(:noteable) { ExternalIssue.new('JIRA-123', project) } let(:mentioned_in) { project.commit } it 'queues a background worker' do expect(Integrations::CreateExternalCrossReferenceWorker).to receive(:perform_async).with( project.id, 'JIRA-123', 'Commit', mentioned_in.id, author.id ) subject end end end end describe '#cross_reference_exists?' do let(:commit0) { project.commit } let(:commit1) { project.commit('HEAD~2') } context 'issue from commit' do before do # Mention issue (noteable) from commit0 service.cross_reference(commit0) end it 'is truthy when already mentioned' do expect(service.cross_reference_exists?(commit0)) .to be_truthy end it 'is falsey when not already mentioned' do expect(service.cross_reference_exists?(commit1)) .to be_falsey end context 'legacy capitalized cross reference' do before do # Mention issue (noteable) from commit0 system_note = service.cross_reference(commit0) system_note.update!(note: system_note.note.capitalize) end it 'is truthy when already mentioned' do expect(service.cross_reference_exists?(commit0)) .to be_truthy end end end context 'commit from commit' do let(:service) { described_class.new(noteable: commit0, author: author) } before do # Mention commit1 from commit0 service.cross_reference(commit1) end it 'is truthy when already mentioned' do expect(service.cross_reference_exists?(commit1)) .to be_truthy end it 'is falsey when not already mentioned' do service = described_class.new(noteable: commit1, author: author) expect(service.cross_reference_exists?(commit0)) .to be_falsey end context 'legacy capitalized cross reference' do before do # Mention commit1 from commit0 system_note = service.cross_reference(commit1) system_note.update!(note: system_note.note.capitalize) end it 'is truthy when already mentioned' do expect(service.cross_reference_exists?(commit1)) .to be_truthy end end end context 'commit with cross-reference from fork', :sidekiq_might_not_need_inline do let(:author2) { create(:project_member, :reporter, user: create(:user), project: project).user } let(:forked_project) { fork_project(project, author2, repository: true) } let(:commit2) { forked_project.commit } let(:service) { described_class.new(noteable: noteable, author: author2) } before do service.cross_reference(commit0) end it 'is true when a fork mentions an external issue' do expect(service.cross_reference_exists?(commit2)) .to be true end context 'legacy capitalized cross reference' do before do system_note = service.cross_reference(commit0) system_note.update!(note: system_note.note.capitalize) end it 'is true when a fork mentions an external issue' do expect(service.cross_reference_exists?(commit2)) .to be true end end end end describe '#change_task_status' do let(:noteable) { create(:issue, project: project) } let(:task) { double(:task, complete?: true, source: 'task') } subject { service.change_task_status(task) } it_behaves_like 'a system note' do let(:action) { 'task' } end it "posts the 'marked the task as complete' system note" do expect(subject.note).to eq("marked the task **task** as completed") end end describe '#noteable_moved' do let(:new_project) { create(:project) } let(:new_noteable) { create(:issue, project: new_project) } subject do # service = described_class.new(noteable: noteable, project: project, author: author) service.noteable_moved(new_noteable, direction) end shared_examples 'cross project mentionable' do include MarkupHelper it 'contains cross reference to new noteable' do expect(subject.note).to include cross_project_reference(new_project, new_noteable) end it 'mentions referenced noteable' do expect(subject.note).to include new_noteable.to_reference end it 'mentions referenced project' do expect(subject.note).to include new_project.full_path end end context 'moved to' do let(:direction) { :to } it_behaves_like 'cross project mentionable' it_behaves_like 'a system note' do let(:action) { 'moved' } end it 'notifies about noteable being moved to' do expect(subject.note).to match('moved to') end end context 'moved from' do let(:direction) { :from } it_behaves_like 'cross project mentionable' it_behaves_like 'a system note' do let(:action) { 'moved' } end it 'notifies about noteable being moved from' do expect(subject.note).to match('moved from') end end context 'invalid direction' do let(:direction) { :invalid } it 'raises error' do expect { subject }.to raise_error StandardError, /Invalid direction/ end end end describe '#noteable_cloned' do let(:new_project) { create(:project) } let(:new_noteable) { create(:issue, project: new_project) } subject do service.noteable_cloned(new_noteable, direction) end shared_examples 'cross project mentionable' do include MarkupHelper it 'contains cross reference to new noteable' do expect(subject.note).to include cross_project_reference(new_project, new_noteable) end it 'mentions referenced noteable' do expect(subject.note).to include new_noteable.to_reference end it 'mentions referenced project' do expect(subject.note).to include new_project.full_path end end context 'cloned to' do let(:direction) { :to } it_behaves_like 'cross project mentionable' it_behaves_like 'a system note' do let(:action) { 'cloned' } end it 'notifies about noteable being cloned to' do expect(subject.note).to match('cloned to') end end context 'cloned from' do let(:direction) { :from } it_behaves_like 'cross project mentionable' it_behaves_like 'a system note' do let(:action) { 'cloned' } end it 'notifies about noteable being cloned from' do expect(subject.note).to match('cloned from') end end context 'invalid direction' do let(:direction) { :invalid } it 'raises error' do expect { subject }.to raise_error StandardError, /Invalid direction/ end end context 'metrics' do context 'cloned from' do let(:direction) { :from } it 'does not tracks usage' do expect(Gitlab::UsageDataCounters::IssueActivityUniqueCounter) .not_to receive(:track_issue_cloned_action).with(author: author) subject end end context 'cloned to' do let(:direction) { :to } it 'tracks usage' do expect(Gitlab::UsageDataCounters::IssueActivityUniqueCounter) .to receive(:track_issue_cloned_action).with(author: author) subject end end end end describe '#mark_duplicate_issue' do subject { service.mark_duplicate_issue(canonical_issue) } context 'within the same project' do let(:canonical_issue) { create(:issue, project: project) } it_behaves_like 'a system note' do let(:action) { 'duplicate' } end it { expect(subject.note).to eq "marked this issue as a duplicate of #{canonical_issue.to_reference}" } end context 'across different projects' do let(:other_project) { create(:project) } let(:canonical_issue) { create(:issue, project: other_project) } it_behaves_like 'a system note' do let(:action) { 'duplicate' } end it { expect(subject.note).to eq "marked this issue as a duplicate of #{canonical_issue.to_reference(project)}" } end end describe '#mark_canonical_issue_of_duplicate' do subject { service.mark_canonical_issue_of_duplicate(duplicate_issue) } context 'within the same project' do let(:duplicate_issue) { create(:issue, project: project) } it_behaves_like 'a system note' do let(:action) { 'duplicate' } end it { expect(subject.note).to eq "marked #{duplicate_issue.to_reference} as a duplicate of this issue" } end context 'across different projects' do let(:other_project) { create(:project) } let(:duplicate_issue) { create(:issue, project: other_project) } it_behaves_like 'a system note' do let(:action) { 'duplicate' } end it { expect(subject.note).to eq "marked #{duplicate_issue.to_reference(project)} as a duplicate of this issue" } end end describe '#discussion_lock' do subject { service.discussion_lock } context 'discussion unlocked' do it_behaves_like 'a system note' do let(:action) { 'unlocked' } end it 'creates the note text correctly' do [:issue, :merge_request].each do |type| issuable = create(type) # rubocop:disable Rails/SaveBang service = described_class.new(noteable: issuable, author: author) expect(service.discussion_lock.note) .to eq("unlocked this #{type.to_s.titleize.downcase}") end end end context 'discussion locked' do before do noteable.update_attribute(:discussion_locked, true) end it_behaves_like 'a system note' do let(:action) { 'locked' } end it 'creates the note text correctly' do [:issue, :merge_request].each do |type| issuable = create(type, discussion_locked: true) service = described_class.new(noteable: issuable, author: author) expect(service.discussion_lock.note) .to eq("locked this #{type.to_s.titleize.downcase}") end end end end describe '#cross_reference_disallowed?' do context 'when mentioned_in is not a MergeRequest' do it 'is falsey' do mentioned_in = noteable.dup expect(service.cross_reference_disallowed?(mentioned_in)).to be_falsey end end context 'when mentioned_in is a MergeRequest' do let(:mentioned_in) { create(:merge_request, :simple, source_project: project) } let(:noteable) { project.commit } it 'is truthy when noteable is in commits' do expect(mentioned_in).to receive(:commits).and_return([noteable]) expect(service.cross_reference_disallowed?(mentioned_in)).to be_truthy end it 'is falsey when noteable is not in commits' do expect(mentioned_in).to receive(:commits).and_return([]) expect(service.cross_reference_disallowed?(mentioned_in)).to be_falsey end end context 'when notable is an ExternalIssue' do let(:project) { create(:project) } let(:noteable) { ExternalIssue.new('EXT-1234', project) } it 'is false with issue tracker supporting referencing' do create(:jira_integration, project: project) project.reload expect(service.cross_reference_disallowed?(noteable)).to be_falsey end it 'is true with issue tracker not supporting referencing' do create(:bugzilla_integration, project: project) project.reload expect(service.cross_reference_disallowed?(noteable)).to be_truthy end it 'is true without issue tracker' do expect(service.cross_reference_disallowed?(noteable)).to be_truthy end end end describe '#close_after_error_tracking_resolve' do subject { service.close_after_error_tracking_resolve } it 'creates the expected state event' do subject event = ResourceStateEvent.last expect(event.close_after_error_tracking_resolve).to eq(true) expect(event.state).to eq('closed') end end describe '#auto_resolve_prometheus_alert' do subject { service.auto_resolve_prometheus_alert } it 'creates the expected state event' do subject event = ResourceStateEvent.last expect(event.close_auto_resolve_prometheus_alert).to eq(true) expect(event.state).to eq('closed') end end describe '#change_issue_type' do let(:noteable) { create(:incident, project: project) } subject { service.change_issue_type } it_behaves_like 'a system note' do let(:action) { 'issue_type' } end it { expect(subject.note).to eq "changed issue type to incident" } end end