2018-11-18 11:00:15 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2015-09-11 14:41:01 +05:30
|
|
|
# SystemNoteService
|
|
|
|
#
|
|
|
|
# Used for creating system notes (e.g., when a user references a merge request
|
|
|
|
# from an issue, an issue's assignee changes, an issue is closed, etc.)
|
2016-09-13 17:45:13 +05:30
|
|
|
module SystemNoteService
|
|
|
|
extend self
|
|
|
|
|
2015-09-11 14:41:01 +05:30
|
|
|
# Called when commits are added to a Merge Request
|
|
|
|
#
|
|
|
|
# noteable - Noteable object
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
# new_commits - Array of Commits added since last push
|
|
|
|
# existing_commits - Array of Commits added in a previous push
|
|
|
|
# oldrev - Optional String SHA of a previous Commit
|
|
|
|
#
|
|
|
|
# See new_commit_summary and existing_commit_summary.
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
2016-09-13 17:45:13 +05:30
|
|
|
def add_commits(noteable, project, author, new_commits, existing_commits = [], oldrev = nil)
|
2015-09-11 14:41:01 +05:30
|
|
|
total_count = new_commits.length + existing_commits.length
|
|
|
|
commits_text = "#{total_count} commit".pluralize(total_count)
|
|
|
|
|
2018-11-18 11:00:15 +05:30
|
|
|
text_parts = ["added #{commits_text}"]
|
|
|
|
text_parts << commits_list(noteable, new_commits, existing_commits, oldrev)
|
2019-09-04 21:01:54 +05:30
|
|
|
text_parts << "[Compare with previous version](#{diff_comparison_path(noteable, project, oldrev)})"
|
2018-11-18 11:00:15 +05:30
|
|
|
|
|
|
|
body = text_parts.join("\n\n")
|
2015-09-11 14:41:01 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'commit', commit_count: total_count))
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|
|
|
|
|
2018-11-20 20:47:30 +05:30
|
|
|
# Called when a commit was tagged
|
|
|
|
#
|
|
|
|
# noteable - Noteable object
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the tag
|
|
|
|
# tag_name - The created tag name
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def tag_commit(noteable, project, author, tag_name)
|
2019-09-04 21:01:54 +05:30
|
|
|
link = url_helpers.project_tag_path(project, id: tag_name)
|
2018-11-20 20:47:30 +05:30
|
|
|
body = "tagged commit #{noteable.sha} to [`#{tag_name}`](#{link})"
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'tag'))
|
|
|
|
end
|
|
|
|
|
2015-09-11 14:41:01 +05:30
|
|
|
# Called when the assignee of a Noteable is changed or removed
|
|
|
|
#
|
|
|
|
# noteable - Noteable object
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
# assignee - User being assigned, or nil
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "removed assignee"
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "assigned to @rspeicher"
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
|
|
|
# Returns the created Note object
|
2016-09-13 17:45:13 +05:30
|
|
|
def change_assignee(noteable, project, author, assignee)
|
2017-08-17 22:00:37 +05:30
|
|
|
body = assignee.nil? ? 'removed assignee' : "assigned to #{assignee.to_reference}"
|
2015-09-11 14:41:01 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'assignee'))
|
|
|
|
end
|
|
|
|
|
|
|
|
# Called when the assignees of an Issue is changed or removed
|
|
|
|
#
|
2019-07-31 22:56:46 +05:30
|
|
|
# issuable - Issuable object (responds to assignees)
|
2017-08-17 22:00:37 +05:30
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
# 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
|
2019-07-31 22:56:46 +05:30
|
|
|
def change_issuable_assignees(issuable, project, author, old_assignees)
|
|
|
|
unassigned_users = old_assignees - issuable.assignees
|
|
|
|
added_users = issuable.assignees.to_a - old_assignees
|
2018-03-17 18:26:18 +05:30
|
|
|
|
|
|
|
text_parts = []
|
|
|
|
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?
|
|
|
|
|
|
|
|
body = text_parts.join(' and ')
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
create_note(NoteSummary.new(issuable, project, author, body, action: 'assignee'))
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|
|
|
|
|
2018-11-20 20:47:30 +05:30
|
|
|
# Called when the milestone of a Noteable is changed
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
2018-11-20 20:47:30 +05:30
|
|
|
# noteable - Noteable object
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
# milestone - Milestone being assigned, or nil
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
2018-11-20 20:47:30 +05:30
|
|
|
# "removed milestone"
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
2018-11-20 20:47:30 +05:30
|
|
|
# "changed milestone to 7.11"
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
|
|
|
# Returns the created Note object
|
2018-11-20 20:47:30 +05:30
|
|
|
def change_milestone(noteable, project, author, milestone)
|
|
|
|
format = milestone&.group_milestone? ? :name : :iid
|
|
|
|
body = milestone.nil? ? 'removed milestone' : "changed milestone to #{milestone.to_reference(project, format: format)}"
|
2015-09-11 14:41:01 +05:30
|
|
|
|
2018-11-20 20:47:30 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'milestone'))
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|
|
|
|
|
2018-11-20 20:47:30 +05:30
|
|
|
# Called when the due_date of a Noteable is changed
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
|
|
|
# noteable - Noteable object
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
2018-11-20 20:47:30 +05:30
|
|
|
# due_date - Due date being assigned, or nil
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
2018-11-20 20:47:30 +05:30
|
|
|
# "removed due date"
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
2018-11-20 20:47:30 +05:30
|
|
|
# "changed due date to September 20, 2018"
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
|
|
|
# Returns the created Note object
|
2018-11-20 20:47:30 +05:30
|
|
|
def change_due_date(noteable, project, author, due_date)
|
|
|
|
body = due_date ? "changed due date to #{due_date.to_s(:long)}" : 'removed due date'
|
2015-09-11 14:41:01 +05:30
|
|
|
|
2018-11-20 20:47:30 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'due_date'))
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
# Called when the estimated time of a Noteable is changed
|
|
|
|
#
|
|
|
|
# noteable - Noteable object
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
# time_estimate - Estimated time
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "removed time estimate"
|
|
|
|
#
|
|
|
|
# "changed time estimate to 3d 5h"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def change_time_estimate(noteable, project, author)
|
|
|
|
parsed_time = Gitlab::TimeTrackingFormatter.output(noteable.time_estimate)
|
|
|
|
body = if noteable.time_estimate == 0
|
|
|
|
"removed time estimate"
|
|
|
|
else
|
|
|
|
"changed time estimate to #{parsed_time}"
|
|
|
|
end
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'time_tracking'))
|
|
|
|
end
|
|
|
|
|
|
|
|
# Called when the spent time of a Noteable is changed
|
|
|
|
#
|
|
|
|
# noteable - Noteable object
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
# time_spent - Spent time
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
|
|
|
# "removed time spent"
|
|
|
|
#
|
|
|
|
# "added 2h 30m of time spent"
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def change_time_spent(noteable, project, author)
|
|
|
|
time_spent = noteable.time_spent
|
|
|
|
|
|
|
|
if time_spent == :reset
|
|
|
|
body = "removed time spent"
|
|
|
|
else
|
2018-03-17 18:26:18 +05:30
|
|
|
spent_at = noteable.spent_at
|
2017-08-17 22:00:37 +05:30
|
|
|
parsed_time = Gitlab::TimeTrackingFormatter.output(time_spent.abs)
|
|
|
|
action = time_spent > 0 ? 'added' : 'subtracted'
|
2018-11-18 11:00:15 +05:30
|
|
|
|
|
|
|
text_parts = ["#{action} #{parsed_time} of time spent"]
|
|
|
|
text_parts << "at #{spent_at}" if spent_at
|
|
|
|
body = text_parts.join(' ')
|
2017-08-17 22:00:37 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'time_tracking'))
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
# Called when the status of a Noteable is changed
|
|
|
|
#
|
|
|
|
# noteable - Noteable object
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
# status - String status
|
|
|
|
# source - Mentionable performing the change, or nil
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "merged"
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "closed via bc17db76"
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
|
|
|
# Returns the created Note object
|
2018-12-13 13:39:08 +05:30
|
|
|
def change_status(noteable, project, author, status, source = nil)
|
2017-08-17 22:00:37 +05:30
|
|
|
body = status.dup
|
|
|
|
body << " via #{source.gfm_reference(project)}" if source
|
|
|
|
|
|
|
|
action = status == 'reopened' ? 'opened' : status
|
2015-12-23 02:04:40 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: action))
|
2015-12-23 02:04:40 +05:30
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
# Called when 'merge when pipeline succeeds' is executed
|
2019-09-30 21:07:59 +05:30
|
|
|
def merge_when_pipeline_succeeds(noteable, project, author, sha)
|
|
|
|
body = "enabled an automatic merge when the pipeline for #{sha} succeeds"
|
2015-12-23 02:04:40 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'merge'))
|
2015-12-23 02:04:40 +05:30
|
|
|
end
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
# Called when 'merge when pipeline succeeds' is canceled
|
|
|
|
def cancel_merge_when_pipeline_succeeds(noteable, project, author)
|
|
|
|
body = 'canceled the automatic merge'
|
2015-09-11 14:41:01 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'merge'))
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|
|
|
|
|
2019-09-30 21:07:59 +05:30
|
|
|
# Called when 'merge when pipeline succeeds' is aborted
|
|
|
|
def abort_merge_when_pipeline_succeeds(noteable, project, author, reason)
|
|
|
|
body = "aborted the automatic merge because #{reason}"
|
|
|
|
|
|
|
|
##
|
|
|
|
# TODO: Abort message should be sent by the system, not a particular user.
|
|
|
|
# See https://gitlab.com/gitlab-org/gitlab-ce/issues/63187.
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'merge'))
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
def handle_merge_request_wip(noteable, project, author)
|
|
|
|
prefix = noteable.work_in_progress? ? "marked" : "unmarked"
|
2016-06-02 11:05:42 +05:30
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
body = "#{prefix} as a **Work In Progress**"
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'title'))
|
|
|
|
end
|
|
|
|
|
|
|
|
def add_merge_request_wip_from_commit(noteable, project, author, commit)
|
|
|
|
body = "marked as a **Work In Progress** from #{commit.to_reference(project)}"
|
2016-06-02 11:05:42 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'title'))
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
def resolve_all_discussions(merge_request, project, author)
|
2019-09-30 21:07:59 +05:30
|
|
|
body = "resolved all threads"
|
2016-09-13 17:45:13 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
create_note(NoteSummary.new(merge_request, project, author, body, action: 'discussion'))
|
|
|
|
end
|
|
|
|
|
|
|
|
def discussion_continued_in_issue(discussion, project, author, issue)
|
|
|
|
body = "created #{issue.to_reference} to continue this discussion"
|
|
|
|
note_attributes = discussion.reply_attributes.merge(project: project, author: author, note: body)
|
|
|
|
|
2019-07-31 22:56:46 +05:30
|
|
|
note = Note.create(note_attributes.merge(system: true, created_at: issue.system_note_timestamp))
|
2017-08-17 22:00:37 +05:30
|
|
|
note.system_note_metadata = SystemNoteMetadata.new(action: 'discussion')
|
|
|
|
|
|
|
|
note
|
2016-09-13 17:45:13 +05:30
|
|
|
end
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
def diff_discussion_outdated(discussion, project, author, change_position)
|
|
|
|
merge_request = discussion.noteable
|
|
|
|
diff_refs = change_position.diff_refs
|
|
|
|
version_index = merge_request.merge_request_diffs.viewable.count
|
2019-09-30 21:07:59 +05:30
|
|
|
position_on_text = change_position.on_text?
|
|
|
|
text_parts = ["changed this #{position_on_text ? 'line' : 'file'} in"]
|
2017-09-10 17:25:29 +05:30
|
|
|
|
|
|
|
if version_params = merge_request.version_params_for(diff_refs)
|
2019-09-30 21:07:59 +05:30
|
|
|
repository = project.repository
|
|
|
|
anchor = position_on_text ? change_position.line_code(repository) : change_position.file_hash
|
|
|
|
url = url_helpers.diffs_project_merge_request_path(project, merge_request, version_params.merge(anchor: anchor))
|
2017-09-10 17:25:29 +05:30
|
|
|
|
2018-11-18 11:00:15 +05:30
|
|
|
text_parts << "[version #{version_index} of the diff](#{url})"
|
2017-09-10 17:25:29 +05:30
|
|
|
else
|
2018-11-18 11:00:15 +05:30
|
|
|
text_parts << "version #{version_index} of the diff"
|
2017-09-10 17:25:29 +05:30
|
|
|
end
|
|
|
|
|
2018-11-18 11:00:15 +05:30
|
|
|
body = text_parts.join(' ')
|
2017-09-10 17:25:29 +05:30
|
|
|
note_attributes = discussion.reply_attributes.merge(project: project, author: author, note: body)
|
2018-11-18 11:00:15 +05:30
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
note = Note.create(note_attributes.merge(system: true))
|
|
|
|
note.system_note_metadata = SystemNoteMetadata.new(action: 'outdated')
|
|
|
|
|
|
|
|
note
|
|
|
|
end
|
|
|
|
|
2015-09-11 14:41:01 +05:30
|
|
|
# Called when the title of a Noteable is changed
|
|
|
|
#
|
|
|
|
# noteable - Noteable object that responds to `title`
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
# old_title - Previous String title
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "changed title from **Old** to **New**"
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
|
|
|
# Returns the created Note object
|
2016-09-13 17:45:13 +05:30
|
|
|
def change_title(noteable, project, author, old_title)
|
2016-06-02 11:05:42 +05:30
|
|
|
new_title = noteable.title.dup
|
|
|
|
|
|
|
|
old_diffs, new_diffs = Gitlab::Diff::InlineDiff.new(old_title, new_title).inline_diffs
|
2015-09-11 14:41:01 +05:30
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
marked_old_title = Gitlab::Diff::InlineDiffMarkdownMarker.new(old_title).mark(old_diffs, mode: :deletion)
|
|
|
|
marked_new_title = Gitlab::Diff::InlineDiffMarkdownMarker.new(new_title).mark(new_diffs, mode: :addition)
|
2016-06-02 11:05:42 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
body = "changed title from **#{marked_old_title}** to **#{marked_new_title}**"
|
|
|
|
|
|
|
|
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(noteable, project, author)
|
|
|
|
body = 'changed the description'
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'description'))
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|
|
|
|
|
2016-06-02 11:05:42 +05:30
|
|
|
# Called when the confidentiality changes
|
|
|
|
#
|
|
|
|
# issue - Issue object
|
|
|
|
# project - Project owning the issue
|
|
|
|
# author - User performing the change
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "made the issue confidential"
|
2016-06-02 11:05:42 +05:30
|
|
|
#
|
|
|
|
# Returns the created Note object
|
2016-09-13 17:45:13 +05:30
|
|
|
def change_issue_confidentiality(issue, project, author)
|
2017-08-17 22:00:37 +05:30
|
|
|
if issue.confidential
|
|
|
|
body = 'made the issue confidential'
|
|
|
|
action = 'confidential'
|
|
|
|
else
|
|
|
|
body = 'made the issue visible to everyone'
|
|
|
|
action = 'visible'
|
|
|
|
end
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(issue, project, author, body, action: action))
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|
|
|
|
|
2015-09-11 14:41:01 +05:30
|
|
|
# Called when a branch in Noteable is changed
|
|
|
|
#
|
|
|
|
# noteable - Noteable object
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
# branch_type - 'source' or 'target'
|
|
|
|
# old_branch - old branch name
|
2019-07-07 11:18:12 +05:30
|
|
|
# new_branch - new branch name
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "changed target branch from `Old` to `New`"
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
|
|
|
# Returns the created Note object
|
2016-09-13 17:45:13 +05:30
|
|
|
def change_branch(noteable, project, author, branch_type, old_branch, new_branch)
|
2017-08-17 22:00:37 +05:30
|
|
|
body = "changed #{branch_type} branch from `#{old_branch}` to `#{new_branch}`"
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'branch'))
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|
|
|
|
|
2015-10-24 18:46:33 +05:30
|
|
|
# Called when a branch in Noteable is added or deleted
|
|
|
|
#
|
|
|
|
# noteable - Noteable object
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
# branch_type - :source or :target
|
|
|
|
# branch - branch name
|
|
|
|
# presence - :add or :delete
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "restored target branch `feature`"
|
2015-10-24 18:46:33 +05:30
|
|
|
#
|
|
|
|
# Returns the created Note object
|
2016-09-13 17:45:13 +05:30
|
|
|
def change_branch_presence(noteable, project, author, branch_type, branch, presence)
|
2015-10-24 18:46:33 +05:30
|
|
|
verb =
|
|
|
|
if presence == :add
|
|
|
|
'restored'
|
|
|
|
else
|
|
|
|
'deleted'
|
|
|
|
end
|
2016-08-24 12:49:21 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
body = "#{verb} #{branch_type} branch `#{branch}`"
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'branch'))
|
2015-10-24 18:46:33 +05:30
|
|
|
end
|
|
|
|
|
2016-06-02 11:05:42 +05:30
|
|
|
# Called when a branch is created from the 'new branch' button on a issue
|
|
|
|
# Example note text:
|
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "created branch `201-issue-branch-button`"
|
2019-09-30 21:07:59 +05:30
|
|
|
def new_issue_branch(issue, project, author, branch, branch_project: nil)
|
|
|
|
branch_project ||= project
|
|
|
|
link = url_helpers.project_compare_path(branch_project, from: branch_project.default_branch, to: branch)
|
2016-06-02 11:05:42 +05:30
|
|
|
|
2019-02-15 15:39:39 +05:30
|
|
|
body = "created branch [`#{branch}`](#{link}) to address this issue"
|
2017-08-17 22:00:37 +05:30
|
|
|
|
|
|
|
create_note(NoteSummary.new(issue, project, author, body, action: 'branch'))
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|
|
|
|
|
2019-02-15 15:39:39 +05:30
|
|
|
def new_merge_request(issue, project, author, merge_request)
|
2019-09-30 21:07:59 +05:30
|
|
|
body = "created merge request #{merge_request.to_reference(project)} to address this issue"
|
2019-02-15 15:39:39 +05:30
|
|
|
|
|
|
|
create_note(NoteSummary.new(issue, project, author, body, action: 'merge'))
|
|
|
|
end
|
|
|
|
|
2015-09-11 14:41:01 +05:30
|
|
|
# Called when a Mentionable references a Noteable
|
|
|
|
#
|
|
|
|
# noteable - Noteable object being referenced
|
|
|
|
# mentioner - Mentionable object
|
|
|
|
# author - User performing the reference
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "mentioned in #1"
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "mentioned in !2"
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "mentioned in 54f7727c"
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
|
|
|
# See cross_reference_note_content.
|
|
|
|
#
|
|
|
|
# Returns the created Note object
|
2016-09-13 17:45:13 +05:30
|
|
|
def cross_reference(noteable, mentioner, author)
|
2015-09-11 14:41:01 +05:30
|
|
|
return if cross_reference_disallowed?(noteable, mentioner)
|
|
|
|
|
2018-05-09 12:01:36 +05:30
|
|
|
gfm_reference = mentioner.gfm_reference(noteable.project || noteable.group)
|
2017-08-17 22:00:37 +05:30
|
|
|
body = cross_reference_note_content(gfm_reference)
|
2015-09-11 14:41:01 +05:30
|
|
|
|
2015-12-23 02:04:40 +05:30
|
|
|
if noteable.is_a?(ExternalIssue)
|
|
|
|
noteable.project.issues_tracker.create_cross_reference_note(noteable, mentioner, author)
|
|
|
|
else
|
2017-08-17 22:00:37 +05:30
|
|
|
create_note(NoteSummary.new(noteable, noteable.project, author, body, action: 'cross_reference'))
|
2015-12-23 02:04:40 +05:30
|
|
|
end
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
# Check if a cross-reference is disallowed
|
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# This method prevents adding a "mentioned in !1" note on every single commit
|
2015-09-11 14:41:01 +05:30
|
|
|
# in a merge request. Additionally, it prevents the creation of references to
|
|
|
|
# external issues (which would fail).
|
|
|
|
#
|
|
|
|
# noteable - Noteable object being referenced
|
|
|
|
# mentioner - Mentionable object
|
|
|
|
#
|
|
|
|
# Returns Boolean
|
2016-09-13 17:45:13 +05:30
|
|
|
def cross_reference_disallowed?(noteable, mentioner)
|
2015-12-23 02:04:40 +05:30
|
|
|
return true if noteable.is_a?(ExternalIssue) && !noteable.project.jira_tracker_active?
|
2015-09-11 14:41:01 +05:30
|
|
|
return false unless mentioner.is_a?(MergeRequest)
|
|
|
|
return false unless noteable.is_a?(Commit)
|
|
|
|
|
|
|
|
mentioner.commits.include?(noteable)
|
|
|
|
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
|
2016-04-02 18:10:28 +05:30
|
|
|
# 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.
|
2015-09-11 14:41:01 +05:30
|
|
|
#
|
|
|
|
# noteable - Noteable object being referenced
|
|
|
|
# mentioner - Mentionable object
|
|
|
|
#
|
|
|
|
# Returns Boolean
|
2016-09-13 17:45:13 +05:30
|
|
|
def cross_reference_exists?(noteable, mentioner)
|
2018-03-17 18:26:18 +05:30
|
|
|
notes = noteable.notes.system
|
2016-11-03 12:29:30 +05:30
|
|
|
notes_for_mentioner(mentioner, noteable, notes).exists?
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|
|
|
|
|
2016-09-13 17:45:13 +05:30
|
|
|
# Build an Array of lines detailing each commit added in a merge request
|
|
|
|
#
|
|
|
|
# new_commits - Array of new Commit objects
|
|
|
|
#
|
|
|
|
# Returns an Array of Strings
|
|
|
|
def new_commit_summary(new_commits)
|
|
|
|
new_commits.collect do |commit|
|
2018-03-17 18:26:18 +05:30
|
|
|
content_tag('li', "#{commit.short_id} - #{commit.title}")
|
2016-09-13 17:45:13 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Called when the status of a Task has changed
|
|
|
|
#
|
|
|
|
# noteable - Noteable object.
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
# new_task - TaskList::Item object.
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "marked the task Whatever as completed."
|
2016-09-13 17:45:13 +05:30
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def change_task_status(noteable, project, author, new_task)
|
|
|
|
status_label = new_task.complete? ? Taskable::COMPLETED : Taskable::INCOMPLETE
|
2017-08-17 22:00:37 +05:30
|
|
|
body = "marked the task **#{new_task.source}** as #{status_label}"
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'task'))
|
2016-09-13 17:45:13 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
# Called when noteable has been moved to another project
|
|
|
|
#
|
|
|
|
# direction - symbol, :to or :from
|
|
|
|
# noteable - Noteable object
|
|
|
|
# noteable_ref - Referenced noteable
|
|
|
|
# author - User performing the move
|
|
|
|
#
|
|
|
|
# Example Note text:
|
|
|
|
#
|
2017-08-17 22:00:37 +05:30
|
|
|
# "moved to some_namespace/project_new#11"
|
2016-09-13 17:45:13 +05:30
|
|
|
#
|
|
|
|
# Returns the created Note object
|
|
|
|
def noteable_moved(noteable, project, noteable_ref, author, direction:)
|
|
|
|
unless [:to, :from].include?(direction)
|
|
|
|
raise ArgumentError, "Invalid direction `#{direction}`"
|
|
|
|
end
|
|
|
|
|
|
|
|
cross_reference = noteable_ref.to_reference(project)
|
2017-08-17 22:00:37 +05:30
|
|
|
body = "moved #{direction} #{cross_reference}"
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(noteable, project, author, body, action: 'moved'))
|
2016-09-13 17:45:13 +05:30
|
|
|
end
|
|
|
|
|
2017-09-10 17:25:29 +05:30
|
|
|
# Called when a Noteable has been marked as a duplicate of another Issue
|
|
|
|
#
|
|
|
|
# noteable - Noteable object
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
# 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(noteable, project, author, canonical_issue)
|
|
|
|
body = "marked this issue as a duplicate of #{canonical_issue.to_reference(project)}"
|
|
|
|
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
|
|
|
|
#
|
|
|
|
# noteable - Noteable object
|
|
|
|
# project - Project owning noteable
|
|
|
|
# author - User performing the change
|
|
|
|
# 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(noteable, project, author, 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
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
def discussion_lock(issuable, author)
|
|
|
|
action = issuable.discussion_locked? ? 'locked' : 'unlocked'
|
|
|
|
body = "#{action} this #{issuable.class.to_s.titleize.downcase}"
|
|
|
|
|
|
|
|
create_note(NoteSummary.new(issuable, issuable.project, author, body, action: action))
|
|
|
|
end
|
|
|
|
|
|
|
|
def cross_reference?(note_text)
|
|
|
|
note_text =~ /\A#{cross_reference_note_prefix}/i
|
|
|
|
end
|
|
|
|
|
2019-10-12 21:52:04 +05:30
|
|
|
def zoom_link_added(issue, project, author)
|
|
|
|
create_note(NoteSummary.new(issue, project, author, _('a Zoom call was added to this issue'), action: 'pinned_embed'))
|
|
|
|
end
|
|
|
|
|
|
|
|
def zoom_link_removed(issue, project, author)
|
|
|
|
create_note(NoteSummary.new(issue, project, author, _('a Zoom call was removed from this issue'), action: 'pinned_embed'))
|
|
|
|
end
|
|
|
|
|
2015-09-11 14:41:01 +05:30
|
|
|
private
|
|
|
|
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: disable CodeReuse/ActiveRecord
|
2016-09-13 17:45:13 +05:30
|
|
|
def notes_for_mentioner(mentioner, noteable, notes)
|
2016-04-02 18:10:28 +05:30
|
|
|
if mentioner.is_a?(Commit)
|
2017-08-17 22:00:37 +05:30
|
|
|
text = "#{cross_reference_note_prefix}%#{mentioner.to_reference(nil)}"
|
|
|
|
notes.where('(note LIKE ? OR note LIKE ?)', text, text.capitalize)
|
2016-04-02 18:10:28 +05:30
|
|
|
else
|
2018-05-09 12:01:36 +05:30
|
|
|
gfm_reference = mentioner.gfm_reference(noteable.project || noteable.group)
|
2017-08-17 22:00:37 +05:30
|
|
|
text = cross_reference_note_content(gfm_reference)
|
|
|
|
notes.where(note: [text, text.capitalize])
|
2016-04-02 18:10:28 +05:30
|
|
|
end
|
|
|
|
end
|
2018-12-05 23:21:45 +05:30
|
|
|
# rubocop: enable CodeReuse/ActiveRecord
|
2016-04-02 18:10:28 +05:30
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
def create_note(note_summary)
|
|
|
|
note = Note.create(note_summary.note.merge(system: true))
|
|
|
|
note.system_note_metadata = SystemNoteMetadata.new(note_summary.metadata) if note_summary.metadata?
|
|
|
|
|
|
|
|
note
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|
|
|
|
|
2016-09-13 17:45:13 +05:30
|
|
|
def cross_reference_note_prefix
|
2017-08-17 22:00:37 +05:30
|
|
|
'mentioned in '
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|
|
|
|
|
2016-09-13 17:45:13 +05:30
|
|
|
def cross_reference_note_content(gfm_reference)
|
2015-09-11 14:41:01 +05:30
|
|
|
"#{cross_reference_note_prefix}#{gfm_reference}"
|
|
|
|
end
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
# Builds a list of existing and new commits according to existing_commits and
|
|
|
|
# new_commits methods.
|
|
|
|
# Returns a String wrapped in `ul` and `li` tags.
|
|
|
|
def commits_list(noteable, new_commits, existing_commits, oldrev)
|
|
|
|
existing_commit_summary = existing_commit_summary(noteable, existing_commits, oldrev)
|
|
|
|
new_commit_summary = new_commit_summary(new_commits).join
|
|
|
|
|
|
|
|
content_tag('ul', "#{existing_commit_summary}#{new_commit_summary}".html_safe)
|
|
|
|
end
|
|
|
|
|
2015-09-11 14:41:01 +05:30
|
|
|
# Build a single line summarizing existing commits being added in a merge
|
|
|
|
# request
|
|
|
|
#
|
|
|
|
# noteable - MergeRequest object
|
|
|
|
# existing_commits - Array of existing Commit objects
|
|
|
|
# oldrev - Optional String SHA of a previous Commit
|
|
|
|
#
|
|
|
|
# Examples:
|
|
|
|
#
|
|
|
|
# "* ea0f8418...2f4426b7 - 24 commits from branch `master`"
|
|
|
|
#
|
|
|
|
# "* ea0f8418..4188f0ea - 15 commits from branch `fork:master`"
|
|
|
|
#
|
|
|
|
# "* ea0f8418 - 1 commit from branch `feature`"
|
|
|
|
#
|
|
|
|
# Returns a newline-terminated String
|
2016-09-13 17:45:13 +05:30
|
|
|
def existing_commit_summary(noteable, existing_commits, oldrev = nil)
|
2015-09-11 14:41:01 +05:30
|
|
|
return '' if existing_commits.empty?
|
|
|
|
|
|
|
|
count = existing_commits.size
|
|
|
|
|
|
|
|
commit_ids = if count == 1
|
|
|
|
existing_commits.first.short_id
|
|
|
|
else
|
2015-10-24 18:46:33 +05:30
|
|
|
if oldrev && !Gitlab::Git.blank_ref?(oldrev)
|
2015-09-11 14:41:01 +05:30
|
|
|
"#{Commit.truncate_sha(oldrev)}...#{existing_commits.last.short_id}"
|
|
|
|
else
|
|
|
|
"#{existing_commits.first.short_id}..#{existing_commits.last.short_id}"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
commits_text = "#{count} commit".pluralize(count)
|
|
|
|
|
|
|
|
branch = noteable.target_branch
|
|
|
|
branch = "#{noteable.target_project_namespace}:#{branch}" if noteable.for_fork?
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
branch_name = content_tag('code', branch)
|
|
|
|
content_tag('li', "#{commit_ids} - #{commits_text} from branch #{branch_name}".html_safe)
|
2016-06-02 11:05:42 +05:30
|
|
|
end
|
2016-11-03 12:29:30 +05:30
|
|
|
|
|
|
|
def url_helpers
|
|
|
|
@url_helpers ||= Gitlab::Routing.url_helpers
|
|
|
|
end
|
|
|
|
|
2019-09-04 21:01:54 +05:30
|
|
|
def diff_comparison_path(merge_request, project, oldrev)
|
2016-11-03 12:29:30 +05:30
|
|
|
diff_id = merge_request.merge_request_diff.id
|
|
|
|
|
2019-09-04 21:01:54 +05:30
|
|
|
url_helpers.diffs_project_merge_request_path(
|
2016-11-03 12:29:30 +05:30
|
|
|
project,
|
2017-09-10 17:25:29 +05:30
|
|
|
merge_request,
|
2016-11-03 12:29:30 +05:30
|
|
|
diff_id: diff_id,
|
|
|
|
start_sha: oldrev
|
|
|
|
)
|
|
|
|
end
|
2018-03-17 18:26:18 +05:30
|
|
|
|
|
|
|
def content_tag(*args)
|
|
|
|
ActionController::Base.helpers.content_tag(*args)
|
|
|
|
end
|
2015-09-11 14:41:01 +05:30
|
|
|
end
|