2019-12-21 20:55:43 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
module SystemNotes
|
|
|
|
class IssuablesService < ::SystemNotes::BaseService
|
2020-11-24 15:15:51 +05:30
|
|
|
#
|
|
|
|
# noteable_ref - Referenced noteable object
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "marked this issue as related to gitlab-foss#9001"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def relate_issue(noteable_ref)
|
|
|
|
body = "marked this issue as related to #{noteable_ref.to_reference(noteable.project)}"
|
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
issue_activity_counter.track_issue_related_action(author: author) if noteable.is_a?(Issue)
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'relate'))
|
|
|
|
end
|
|
|
|
|
|
|
|
#
|
|
|
|
# noteable_ref - Referenced noteable object
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "removed the relation with gitlab-foss#9001"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def unrelate_issue(noteable_ref)
|
|
|
|
body = "removed the relation with #{noteable_ref.to_reference(noteable.project)}"
|
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
issue_activity_counter.track_issue_unrelated_action(author: author) if noteable.is_a?(Issue)
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'unrelate'))
|
|
|
|
end
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
# Called when the assignee of a Noteable is changed or removed
|
|
|
|
#
|
|
|
|
# assignee - User being assigned, or nil
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "removed assignee"
|
|
|
|
#
|
|
|
|
# "assigned to @rspeicher"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def change_assignee(assignee)
|
|
|
|
body = assignee.nil? ? 'removed assignee' : "assigned to #{assignee.to_reference}"
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
issue_activity_counter.track_issue_assignee_changed_action(author: author) if noteable.is_a?(Issue)
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'assignee'))
|
|
|
|
end
|
|
|
|
|
|
|
|
# Called when the assignees of an issuable is changed or removed
|
|
|
|
#
|
|
|
|
# assignees - Users being assigned, or nil
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "removed all assignees"
|
|
|
|
#
|
|
|
|
# "assigned to @user1 additionally to @user2"
|
|
|
|
#
|
|
|
|
# "assigned to @user1, @user2 and @user3 and unassigned from @user4 and @user5"
|
|
|
|
#
|
|
|
|
# "assigned to @user1 and @user2"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def change_issuable_assignees(old_assignees)
|
|
|
|
unassigned_users = old_assignees - noteable.assignees
|
|
|
|
added_users = noteable.assignees.to_a - old_assignees
|
|
|
|
text_parts = []
|
|
|
|
|
|
|
|
Gitlab::I18n.with_default_locale do
|
|
|
|
text_parts << "assigned to #{added_users.map(&:to_reference).to_sentence}" if added_users.any?
|
|
|
|
text_parts << "unassigned #{unassigned_users.map(&:to_reference).to_sentence}" if unassigned_users.any?
|
|
|
|
end
|
|
|
|
|
|
|
|
body = text_parts.join(' and ')
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
issue_activity_counter.track_issue_assignee_changed_action(author: author) if noteable.is_a?(Issue)
|
2019-12-21 20:55:43 +05:30
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'assignee'))
|
2019-12-21 20:55:43 +05:30
|
|
|
end
|
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
# Called when the reviewers of an issuable is changed or removed
|
|
|
|
#
|
|
|
|
# reviewers - Users being requested to review, or nil
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "requested review from @user1 and @user2"
|
|
|
|
#
|
|
|
|
# "requested review from @user1, @user2 and @user3 and removed review request for @user4 and @user5"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def change_issuable_reviewers(old_reviewers)
|
|
|
|
unassigned_users = old_reviewers - noteable.reviewers
|
|
|
|
added_users = noteable.reviewers - old_reviewers
|
|
|
|
text_parts = []
|
|
|
|
|
|
|
|
Gitlab::I18n.with_default_locale do
|
|
|
|
text_parts << "requested review from #{added_users.map(&:to_reference).to_sentence}" if added_users.any?
|
|
|
|
text_parts << "removed review request for #{unassigned_users.map(&:to_reference).to_sentence}" if unassigned_users.any?
|
|
|
|
end
|
|
|
|
|
|
|
|
body = text_parts.join(' and ')
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'reviewer'))
|
|
|
|
end
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
# Called when the title of a Noteable is changed
|
|
|
|
#
|
|
|
|
# old_title - Previous String title
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "changed title from **Old** to **New**"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def change_title(old_title)
|
|
|
|
new_title = noteable.title.dup
|
|
|
|
|
|
|
|
old_diffs, new_diffs = Gitlab::Diff::InlineDiff.new(old_title, new_title).inline_diffs
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
marked_old_title = Gitlab::Diff::InlineDiffMarkdownMarker.new(old_title).mark(old_diffs)
|
|
|
|
marked_new_title = Gitlab::Diff::InlineDiffMarkdownMarker.new(new_title).mark(new_diffs)
|
2019-12-21 20:55:43 +05:30
|
|
|
|
|
|
|
body = "changed title from **#{marked_old_title}** to **#{marked_new_title}**"
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
issue_activity_counter.track_issue_title_changed_action(author: author) if noteable.is_a?(Issue)
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'title'))
|
|
|
|
end
|
|
|
|
|
|
|
|
# Called when the description of a Noteable is changed
|
|
|
|
#
|
|
|
|
# noteable - Noteable object that responds to `description`
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "changed the description"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def change_description
|
|
|
|
body = 'changed the description'
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
issue_activity_counter.track_issue_description_changed_action(author: author) if noteable.is_a?(Issue)
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'description'))
|
|
|
|
end
|
|
|
|
|
|
|
|
# Called when a Mentionable references a Noteable
|
|
|
|
#
|
|
|
|
# mentioner - Mentionable object
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "mentioned in #1"
|
|
|
|
#
|
|
|
|
# "mentioned in !2"
|
|
|
|
#
|
|
|
|
# "mentioned in 54f7727c"
|
|
|
|
#
|
|
|
|
# See cross_reference_note_content.
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def cross_reference(mentioner)
|
|
|
|
return if cross_reference_disallowed?(mentioner)
|
|
|
|
|
|
|
|
gfm_reference = mentioner.gfm_reference(noteable.project || noteable.group)
|
|
|
|
body = cross_reference_note_content(gfm_reference)
|
|
|
|
|
|
|
|
if noteable.is_a?(ExternalIssue)
|
2020-06-23 00:09:42 +05:30
|
|
|
noteable.project.external_issue_tracker.create_cross_reference_note(noteable, mentioner, author)
|
2019-12-21 20:55:43 +05:30
|
|
|
else
|
2021-06-08 01:23:25 +05:30
|
|
|
track_cross_reference_action
|
2019-12-21 20:55:43 +05:30
|
|
|
create_note(NoteSummary.new(noteable, noteable.project, author, body, action: 'cross_reference'))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Check if a cross-reference is disallowed
|
|
|
|
#
|
|
|
|
# This method prevents adding a "mentioned in !1" note on every single commit
|
|
|
|
# in a merge request. Additionally, it prevents the creation of references to
|
|
|
|
# external issues (which would fail).
|
|
|
|
#
|
|
|
|
# mentioner - Mentionable object
|
|
|
|
#
|
|
|
|
# Returns Boolean
|
|
|
|
def cross_reference_disallowed?(mentioner)
|
2020-04-08 14:13:33 +05:30
|
|
|
return true if noteable.is_a?(ExternalIssue) && !noteable.project&.external_references_supported?
|
2019-12-21 20:55:43 +05:30
|
|
|
return false unless mentioner.is_a?(MergeRequest)
|
|
|
|
return false unless noteable.is_a?(Commit)
|
|
|
|
|
|
|
|
mentioner.commits.include?(noteable)
|
|
|
|
end
|
|
|
|
|
|
|
|
# Called when the status of a Task has changed
|
|
|
|
#
|
|
|
|
# new_task - TaskList::Item object.
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "marked the task Whatever as completed."
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def change_task_status(new_task)
|
|
|
|
status_label = new_task.complete? ? Taskable::COMPLETED : Taskable::INCOMPLETE
|
|
|
|
body = "marked the task **#{new_task.source}** as #{status_label}"
|
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
issue_activity_counter.track_issue_description_changed_action(author: author) if noteable.is_a?(Issue)
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'task'))
|
|
|
|
end
|
|
|
|
|
|
|
|
# Called when noteable has been moved to another project
|
|
|
|
#
|
|
|
|
# noteable_ref - Referenced noteable
|
|
|
|
# direction - symbol, :to or :from
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "moved to some_namespace/project_new#11"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def noteable_moved(noteable_ref, direction)
|
|
|
|
unless [:to, :from].include?(direction)
|
|
|
|
raise ArgumentError, "Invalid direction `#{direction}`"
|
|
|
|
end
|
|
|
|
|
|
|
|
cross_reference = noteable_ref.to_reference(project)
|
|
|
|
body = "moved #{direction} #{cross_reference}"
|
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
issue_activity_counter.track_issue_moved_action(author: author) if noteable.is_a?(Issue)
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'moved'))
|
|
|
|
end
|
|
|
|
|
2021-02-22 17:27:13 +05:30
|
|
|
# Called when noteable has been cloned
|
|
|
|
#
|
|
|
|
# noteable_ref - Referenced noteable
|
|
|
|
# direction - symbol, :to or :from
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "cloned to some_namespace/project_new#11"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def noteable_cloned(noteable_ref, direction)
|
|
|
|
unless [:to, :from].include?(direction)
|
|
|
|
raise ArgumentError, "Invalid direction `#{direction}`"
|
|
|
|
end
|
|
|
|
|
|
|
|
cross_reference = noteable_ref.to_reference(project)
|
|
|
|
body = "cloned #{direction} #{cross_reference}"
|
|
|
|
|
|
|
|
issue_activity_counter.track_issue_cloned_action(author: author) if noteable.is_a?(Issue) && direction == :to
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'cloned'))
|
|
|
|
end
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
# Called when the confidentiality changes
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "made the issue confidential"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def change_issue_confidentiality
|
|
|
|
if noteable.confidential
|
|
|
|
body = 'made the issue confidential'
|
|
|
|
action = 'confidential'
|
2020-11-24 15:15:51 +05:30
|
|
|
|
|
|
|
issue_activity_counter.track_issue_made_confidential_action(author: author) if noteable.is_a?(Issue)
|
2019-12-21 20:55:43 +05:30
|
|
|
else
|
|
|
|
body = 'made the issue visible to everyone'
|
|
|
|
action = 'visible'
|
2020-11-24 15:15:51 +05:30
|
|
|
|
|
|
|
issue_activity_counter.track_issue_made_visible_action(author: author) if noteable.is_a?(Issue)
|
2019-12-21 20:55:43 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: action))
|
|
|
|
end
|
|
|
|
|
|
|
|
# Called when the status of a Noteable is changed
|
|
|
|
#
|
|
|
|
# status - String status
|
|
|
|
# source - Mentionable performing the change, or nil
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "merged"
|
|
|
|
#
|
|
|
|
# "closed via bc17db76"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def change_status(status, source = nil)
|
2021-01-03 14:25:43 +05:30
|
|
|
create_resource_state_event(status: status, mentionable_source: source)
|
2019-12-21 20:55:43 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
# Check if a cross reference to a noteable from a mentioner already exists
|
|
|
|
#
|
|
|
|
# This method is used to prevent multiple notes being created for a mention
|
|
|
|
# when a issue is updated, for example. The method also calls notes_for_mentioner
|
|
|
|
# to check if the mentioner is a commit, and return matches only on commit hash
|
|
|
|
# instead of project + commit, to avoid repeated mentions from forks.
|
|
|
|
#
|
|
|
|
# mentioner - Mentionable object
|
|
|
|
#
|
|
|
|
# Returns Boolean
|
|
|
|
def cross_reference_exists?(mentioner)
|
|
|
|
notes = noteable.notes.system
|
|
|
|
notes_for_mentioner(mentioner, noteable, notes).exists?
|
|
|
|
end
|
|
|
|
|
|
|
|
# Called when a Noteable has been marked as a duplicate of another Issue
|
|
|
|
#
|
|
|
|
# canonical_issue - Issue that this is a duplicate of
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "marked this issue as a duplicate of #1234"
|
|
|
|
#
|
|
|
|
# "marked this issue as a duplicate of other_project#5678"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def mark_duplicate_issue(canonical_issue)
|
|
|
|
body = "marked this issue as a duplicate of #{canonical_issue.to_reference(project)}"
|
2021-01-03 14:25:43 +05:30
|
|
|
|
|
|
|
issue_activity_counter.track_issue_marked_as_duplicate_action(author: author) if noteable.is_a?(Issue)
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'duplicate'))
|
|
|
|
end
|
|
|
|
|
|
|
|
# Called when a Noteable has been marked as the canonical Issue of a duplicate
|
|
|
|
#
|
|
|
|
# duplicate_issue - Issue that was a duplicate of this
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "marked #1234 as a duplicate of this issue"
|
|
|
|
#
|
|
|
|
# "marked other_project#5678 as a duplicate of this issue"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def mark_canonical_issue_of_duplicate(duplicate_issue)
|
|
|
|
body = "marked #{duplicate_issue.to_reference(project)} as a duplicate of this issue"
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'duplicate'))
|
|
|
|
end
|
|
|
|
|
2021-04-17 20:07:23 +05:30
|
|
|
def add_email_participants(body)
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body))
|
|
|
|
end
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
def discussion_lock
|
|
|
|
action = noteable.discussion_locked? ? 'locked' : 'unlocked'
|
|
|
|
body = "#{action} this #{noteable.class.to_s.titleize.downcase}"
|
|
|
|
|
2021-01-03 14:25:43 +05:30
|
|
|
if noteable.is_a?(Issue)
|
|
|
|
if action == 'locked'
|
|
|
|
issue_activity_counter.track_issue_locked_action(author: author)
|
|
|
|
else
|
|
|
|
issue_activity_counter.track_issue_unlocked_action(author: author)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: action))
|
|
|
|
end
|
|
|
|
|
2020-03-13 15:44:24 +05:30
|
|
|
def close_after_error_tracking_resolve
|
2021-01-03 14:25:43 +05:30
|
|
|
create_resource_state_event(status: 'closed', close_after_error_tracking_resolve: true)
|
2020-03-13 15:44:24 +05:30
|
|
|
end
|
|
|
|
|
2020-04-08 14:13:33 +05:30
|
|
|
def auto_resolve_prometheus_alert
|
2021-01-03 14:25:43 +05:30
|
|
|
create_resource_state_event(status: 'closed', close_auto_resolve_prometheus_alert: true)
|
2020-04-08 14:13:33 +05:30
|
|
|
end
|
|
|
|
|
2021-11-11 11:23:49 +05:30
|
|
|
def change_issue_type
|
|
|
|
body = "changed issue type to #{noteable.issue_type.humanize(capitalize: false)}"
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'issue_type'))
|
|
|
|
end
|
|
|
|
|
2019-12-21 20:55:43 +05:30
|
|
|
private
|
|
|
|
|
|
|
|
def cross_reference_note_content(gfm_reference)
|
|
|
|
"#{self.class.cross_reference_note_prefix}#{gfm_reference}"
|
|
|
|
end
|
|
|
|
|
|
|
|
def notes_for_mentioner(mentioner, noteable, notes)
|
|
|
|
if mentioner.is_a?(Commit)
|
|
|
|
text = "#{self.class.cross_reference_note_prefix}%#{mentioner.to_reference(nil)}"
|
|
|
|
notes.like_note_or_capitalized_note(text)
|
|
|
|
else
|
|
|
|
gfm_reference = mentioner.gfm_reference(noteable.project || noteable.group)
|
|
|
|
text = cross_reference_note_content(gfm_reference)
|
|
|
|
notes.for_note_or_capitalized_note(text)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.cross_reference_note_prefix
|
|
|
|
'mentioned in '
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.cross_reference?(note_text)
|
|
|
|
note_text =~ /\A#{cross_reference_note_prefix}/i
|
|
|
|
end
|
2020-06-23 00:09:42 +05:30
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
def create_resource_state_event(params)
|
|
|
|
ResourceEvents::ChangeStateService.new(resource: noteable, user: author)
|
|
|
|
.execute(params)
|
|
|
|
end
|
|
|
|
|
2020-11-24 15:15:51 +05:30
|
|
|
def issue_activity_counter
|
|
|
|
Gitlab::UsageDataCounters::IssueActivityUniqueCounter
|
|
|
|
end
|
2021-06-08 01:23:25 +05:30
|
|
|
|
|
|
|
def track_cross_reference_action
|
|
|
|
issue_activity_counter.track_issue_cross_referenced_action(author: author) if noteable.is_a?(Issue)
|
|
|
|
end
|
2019-12-21 20:55:43 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-06-08 01:23:25 +05:30
|
|
|
SystemNotes::IssuablesService.prepend_mod_with('SystemNotes::IssuablesService')
|