Commit bcaea367 authored by Jarka Kadlecová's avatar Jarka Kadlecová

Add system notes for issue - epic association

parent c291a740
...@@ -16,6 +16,7 @@ class SystemNoteMetadata < ActiveRecord::Base ...@@ -16,6 +16,7 @@ class SystemNoteMetadata < ActiveRecord::Base
opened closed merged duplicate locked unlocked opened closed merged duplicate locked unlocked
outdated outdated
approved unapproved relate unrelate approved unapproved relate unrelate
epic_issue_added issue_added_to_epic epic_issue_removed issue_removed_from_epic
].freeze ].freeze
validates :note, presence: true validates :note, presence: true
......
...@@ -566,6 +566,36 @@ module SystemNoteService ...@@ -566,6 +566,36 @@ module SystemNoteService
create_note(NoteSummary.new(noteable, noteable.project, user, body, action: 'unrelate')) create_note(NoteSummary.new(noteable, noteable.project, user, body, action: 'unrelate'))
end end
def epic_issue(epic, issue, user, type)
return unless validate_epic_issue_action_type(type)
action = type == :added ? 'epic_issue_added' : 'epic_issue_removed'
body = "#{type} issue #{issue.to_reference(epic.group)}"
create_note(NoteSummary.new(epic, nil, user, body, action: action))
end
def issue_on_epic(issue, epic, user, type)
return unless validate_epic_issue_action_type(type)
if type == :added
direction = 'to'
action = 'issue_added_to_epic'
else
direction = 'from'
action = 'issue_removed_from_epic'
end
body = "#{type} #{direction} epic #{epic.to_reference(issue.project)}"
create_note(NoteSummary.new(issue, issue.project, user, body, action: action))
end
def validate_epic_issue_action_type(type)
[:added, :removed].include?(type)
end
# Called when the merge request is approved by user # Called when the merge request is approved by user
# #
# noteable - Noteable object # noteable - Noteable object
......
---
title: Add system notes for issue - epic association
merge_request:
author:
type: added
...@@ -8,8 +8,9 @@ module EpicIssues ...@@ -8,8 +8,9 @@ module EpicIssues
link.save! link.save!
end end
def create_notes? def create_notes(referenced_issue)
false SystemNoteService.epic_issue(issuable, referenced_issue, current_user, :added)
SystemNoteService.issue_on_epic(referenced_issue, issuable, current_user, :added)
end end
def extractor_context def extractor_context
......
...@@ -2,10 +2,6 @@ module EpicIssues ...@@ -2,10 +2,6 @@ module EpicIssues
class DestroyService < IssuableLinks::DestroyService class DestroyService < IssuableLinks::DestroyService
private private
def create_notes?
false
end
def source def source
@source ||= link.epic @source ||= link.epic
end end
...@@ -17,5 +13,10 @@ module EpicIssues ...@@ -17,5 +13,10 @@ module EpicIssues
def permission_to_remove_relation? def permission_to_remove_relation?
can?(current_user, :admin_epic_issue, target) && can?(current_user, :admin_epic, source) can?(current_user, :admin_epic_issue, target) && can?(current_user, :admin_epic, source)
end end
def create_notes
SystemNoteService.epic_issue(source, target, current_user, :removed)
SystemNoteService.issue_on_epic(target, source, current_user, :removed)
end
end end
end end
...@@ -19,7 +19,7 @@ module IssuableLinks ...@@ -19,7 +19,7 @@ module IssuableLinks
def create_issue_links def create_issue_links
referenced_issues.each do |referenced_issue| referenced_issues.each do |referenced_issue|
create_notes(referenced_issue) if relate_issues(referenced_issue) && create_notes? create_notes(referenced_issue) if relate_issues(referenced_issue)
end end
end end
...@@ -49,19 +49,10 @@ module IssuableLinks ...@@ -49,19 +49,10 @@ module IssuableLinks
extractor.issues extractor.issues
end end
def create_notes(referenced_issue)
SystemNoteService.relate_issue(issuable, referenced_issue, current_user)
SystemNoteService.relate_issue(referenced_issue, issuable, current_user)
end
def extractor_context def extractor_context
{} {}
end end
def create_notes?
true
end
def linkable_issues(issues) def linkable_issues(issues)
raise NotImplementedError raise NotImplementedError
end end
......
...@@ -11,24 +11,15 @@ module IssuableLinks ...@@ -11,24 +11,15 @@ module IssuableLinks
return error('No Issue Link found', 404) unless permission_to_remove_relation? return error('No Issue Link found', 404) unless permission_to_remove_relation?
remove_relation remove_relation
create_notes if create_notes? create_notes
success(message: 'Relation was removed') success(message: 'Relation was removed')
end end
private private
def create_notes
SystemNoteService.unrelate_issue(source, target, current_user)
SystemNoteService.unrelate_issue(target, source, current_user)
end
def remove_relation def remove_relation
link.destroy! link.destroy!
end end
def create_notes?
true
end
end end
end end
...@@ -7,5 +7,10 @@ module IssueLinks ...@@ -7,5 +7,10 @@ module IssueLinks
def linkable_issues(issues) def linkable_issues(issues)
issues.select { |issue| can?(current_user, :admin_issue_link, issue) } issues.select { |issue| can?(current_user, :admin_issue_link, issue) }
end end
def create_notes(referenced_issue)
SystemNoteService.relate_issue(issuable, referenced_issue, current_user)
SystemNoteService.relate_issue(referenced_issue, issuable, current_user)
end
end end
end end
...@@ -13,5 +13,10 @@ module IssueLinks ...@@ -13,5 +13,10 @@ module IssueLinks
def permission_to_remove_relation? def permission_to_remove_relation?
can?(current_user, :admin_issue_link, source) && can?(current_user, :admin_issue_link, target) can?(current_user, :admin_issue_link, source) && can?(current_user, :admin_issue_link, target)
end end
def create_notes
SystemNoteService.unrelate_issue(source, target, current_user)
SystemNoteService.unrelate_issue(target, source, current_user)
end
end end
end end
...@@ -25,6 +25,32 @@ describe EpicIssues::CreateService do ...@@ -25,6 +25,32 @@ describe EpicIssues::CreateService do
it 'returns success status' do it 'returns success status' do
expect(subject).to eq(status: :success) expect(subject).to eq(status: :success)
end end
it 'creates 2 system notes' do
expect { subject }.to change { Note.count }.from(0).to(2)
end
it 'creates a note for epic correctly' do
subject
note = Note.find_by(noteable_id: epic.id, noteable_type: 'Epic')
expect(note.note).to eq("added issue #{issue.to_reference(epic.group)}")
expect(note.author).to eq(user)
expect(note.project).to be_nil
expect(note.noteable_type).to eq('Epic')
expect(note.system_note_metadata.action).to eq('epic_issue_added')
end
it 'creates a note for issue correctly' do
subject
note = Note.find_by(noteable_id: issue.id, noteable_type: 'Issue')
expect(note.note).to eq("added to epic #{epic.to_reference(issue.project)}")
expect(note.author).to eq(user)
expect(note.project).to eq(issue.project)
expect(note.noteable_type).to eq('Issue')
expect(note.system_note_metadata.action).to eq('issue_added_to_epic')
end
end end
shared_examples 'returns an error' do shared_examples 'returns an error' do
...@@ -57,6 +83,10 @@ describe EpicIssues::CreateService do ...@@ -57,6 +83,10 @@ describe EpicIssues::CreateService do
it 'returns an error' do it 'returns an error' do
expect(assign_issue([])).to eq(message: 'No Issue found for given params', status: :error, http_status: 404) expect(assign_issue([])).to eq(message: 'No Issue found for given params', status: :error, http_status: 404)
end end
it 'does not create a system note' do
expect { assign_issue([]) }.not_to change { Note.count }
end
end end
context 'when there is an issue to relate' do context 'when there is an issue to relate' do
...@@ -72,6 +102,9 @@ describe EpicIssues::CreateService do ...@@ -72,6 +102,9 @@ describe EpicIssues::CreateService do
include_examples 'returns success' include_examples 'returns success'
it 'does not perofrm N + 1 queries' do it 'does not perofrm N + 1 queries' do
allow(SystemNoteService).to receive(:epic_issue)
allow(SystemNoteService).to receive(:issue_on_epic)
params = { issue_references: [valid_reference] } params = { issue_references: [valid_reference] }
control_count = ActiveRecord::QueryRecorder.new { described_class.new(epic, user, params).execute }.count control_count = ActiveRecord::QueryRecorder.new { described_class.new(epic, user, params).execute }.count
......
...@@ -38,6 +38,32 @@ describe EpicIssues::DestroyService do ...@@ -38,6 +38,32 @@ describe EpicIssues::DestroyService do
it 'returns success message' do it 'returns success message' do
is_expected.to eq(message: 'Relation was removed', status: :success) is_expected.to eq(message: 'Relation was removed', status: :success)
end end
it 'creates 2 system notes' do
expect { subject }.to change { Note.count }.from(0).to(2)
end
it 'creates a note for epic correctly' do
subject
note = Note.find_by(noteable_id: epic.id, noteable_type: 'Epic')
expect(note.note).to eq("removed issue #{issue.to_reference(epic.group)}")
expect(note.author).to eq(user)
expect(note.project).to be_nil
expect(note.noteable_type).to eq('Epic')
expect(note.system_note_metadata.action).to eq('epic_issue_removed')
end
it 'creates a note for issue correctly' do
subject
note = Note.find_by(noteable_id: issue.id, noteable_type: 'Issue')
expect(note.note).to eq("removed from epic #{epic.to_reference(issue.project)}")
expect(note.author).to eq(user)
expect(note.project).to eq(issue.project)
expect(note.noteable_type).to eq('Issue')
expect(note.system_note_metadata.action).to eq('issue_removed_from_epic')
end
end end
context 'user does not have permissions to remove associations' do context 'user does not have permissions to remove associations' do
......
...@@ -5,10 +5,11 @@ describe SystemNoteService do ...@@ -5,10 +5,11 @@ describe SystemNoteService do
include Gitlab::Routing include Gitlab::Routing
set(:group) { create(:group) } set(:group) { create(:group) }
set(:project) { create(:project, :repository, group: group) } let(:project) { create(:project, :repository, group: group) }
set(:author) { create(:user) } set(:author) { create(:user) }
let(:noteable) { create(:issue, project: project) } let(:noteable) { create(:issue, project: project) }
let(:issue) { noteable } let(:issue) { noteable }
let(:epic) { create(:epic) }
shared_examples_for 'a system note' do shared_examples_for 'a system note' do
let(:expected_noteable) { noteable } let(:expected_noteable) { noteable }
...@@ -1303,4 +1304,73 @@ describe SystemNoteService do ...@@ -1303,4 +1304,73 @@ describe SystemNoteService do
end end
end end
end end
describe '.epic_issue' do
let(:noteable) { epic }
let(:project) { nil }
context 'issue added to an epic' do
subject { described_class.epic_issue(epic, issue, author, :added) }
it_behaves_like 'a system note' do
let(:action) { 'epic_issue_added' }
end
it 'creates the note text correctly' do
expect(subject.note).to eq("added issue #{issue.to_reference(epic.group)}")
end
end
context 'issue removed from an epic' do
subject { described_class.epic_issue(epic, issue, author, :removed) }
it_behaves_like 'a system note' do
let(:action) { 'epic_issue_removed' }
end
it 'creates the note text correctly' do
expect(subject.note).to eq("removed issue #{issue.to_reference(epic.group)}")
end
end
context 'invalid type' do
it 'raises an error' do
expect { described_class.issue_on_epic(issue, epic, author, :invalid) }
.not_to change { Note.count }
end
end
end
describe '.issue_on_epic' do
context 'issue added to an epic' do
subject { described_class.issue_on_epic(issue, epic, author, :added) }
it_behaves_like 'a system note' do
let(:action) { 'issue_added_to_epic' }
end
it 'creates the note text correctly' do
expect(subject.note).to eq("added to epic #{epic.to_reference(issue.project)}")
end
end
context 'issue removed from an epic' do
subject { described_class.issue_on_epic(issue, epic, author, :removed) }
it_behaves_like 'a system note' do
let(:action) { 'issue_removed_from_epic' }
end
it 'creates the note text correctly' do
expect(subject.note).to eq("removed from epic #{epic.to_reference(issue.project)}")
end
end
context 'invalid type' do
it 'does not create a new note' do
expect { described_class.issue_on_epic(issue, epic, author, :invalid) }
.not_to change { Note.count }
end
end
end
end end
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment