1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122
|
# frozen_string_literal: true
module Issues
class BuildService < Issues::BaseService
include ResolveDiscussions
def execute(initialize_callbacks: true)
filter_resolve_discussion_params
container_param = case container
when Project
{ project: project }
when Namespaces::ProjectNamespace
{ project: container.project }
else
{ namespace: container }
end
@issue = model_klass.new(issue_params.merge(container_param)).tap do |issue|
set_work_item_type(issue)
initialize_callbacks!(issue) if initialize_callbacks
end
end
def issue_params_with_info_from_discussions
return {} unless merge_request_to_resolve_discussions_of
{ title: title_from_merge_request, description: description_for_discussions }
end
def title_from_merge_request
"Follow-up from \"#{merge_request_to_resolve_discussions_of.title}\""
end
def description_for_discussions
if discussions_to_resolve.empty?
return "There are no unresolved discussions. "\
"Review the conversation in #{merge_request_to_resolve_discussions_of.to_reference}"
end
description = "The following #{'discussion'.pluralize(discussions_to_resolve.size)} "\
"from #{merge_request_to_resolve_discussions_of.to_reference} "\
"should be addressed:"
[description, *items_for_discussions].join("\n\n")
end
def items_for_discussions
discussions_to_resolve.map { |discussion| item_for_discussion(discussion) }
end
def item_for_discussion(discussion)
first_note_to_resolve = discussion.first_note_to_resolve || discussion.first_note
is_very_first_note = first_note_to_resolve == discussion.first_note
action = is_very_first_note ? "started" : "commented on"
note_url = Gitlab::UrlBuilder.build(first_note_to_resolve)
other_note_count = discussion.notes.size - 1
discussion_info = ["- [ ] #{first_note_to_resolve.author.to_reference} #{action} a [discussion](#{note_url}): "]
discussion_info << "(+#{other_note_count} #{'comment'.pluralize(other_note_count)})" if other_note_count > 0
spaces = ' ' * 4
quote = first_note_to_resolve.note.lines.map { |line| "#{spaces}> #{line}" }.join
[discussion_info.join(' '), quote].join("\n\n")
end
def issue_params
@issue_params ||= build_issue_params
end
private
def set_work_item_type(issue)
work_item_type = if params[:work_item_type_id].present?
params.delete(:work_item_type)
WorkItems::Type.find_by(id: params.delete(:work_item_type_id)) # rubocop: disable CodeReuse/ActiveRecord
else
params.delete(:work_item_type)
end
# We need to support the legacy input params[:issue_type] even if we don't have the issue_type column anymore.
# In the future only params[:work_item_type] should be provided
base_type = work_item_type&.base_type || params[:issue_type]
issue.work_item_type = if create_issue_type_allowed?(container, base_type)
work_item_type || WorkItems::Type.default_by_type(base_type)
else
# If no work item type was provided or not allowed, we need to set it to
# the default issue_type
WorkItems::Type.default_by_type(::Issue::DEFAULT_ISSUE_TYPE)
end
end
def model_klass
::Issue
end
def public_params
# Additional params may be assigned later (in a CreateService for example)
public_issue_params = [
:title,
:description,
:confidential
]
params.slice(*public_issue_params)
end
def build_issue_params
{ author: current_user }
.merge(issue_params_with_info_from_discussions)
.merge(public_params)
.with_indifferent_access
end
end
end
Issues::BuildService.prepend_mod_with('Issues::BuildService')
|