2020-07-28 23:09:34 +05:30
|
|
|
# frozen_string_literal: true
|
|
|
|
|
|
|
|
# handles service desk issue creation emails with these formats:
|
|
|
|
# incoming+gitlab-org-gitlab-ce-20-issue-@incoming.gitlab.com
|
|
|
|
# incoming+gitlab-org/gitlab-ce@incoming.gitlab.com (legacy)
|
|
|
|
module Gitlab
|
|
|
|
module Email
|
|
|
|
module Handler
|
|
|
|
class ServiceDeskHandler < BaseHandler
|
|
|
|
include ReplyProcessing
|
|
|
|
include Gitlab::Utils::StrongMemoize
|
|
|
|
|
|
|
|
HANDLER_REGEX = /\A#{HANDLER_ACTION_BASE_REGEX}-issue-\z/.freeze
|
|
|
|
HANDLER_REGEX_LEGACY = /\A(?<project_path>[^\+]*)\z/.freeze
|
|
|
|
PROJECT_KEY_PATTERN = /\A(?<slug>.+)-(?<key>[a-z0-9_]+)\z/.freeze
|
|
|
|
|
|
|
|
def initialize(mail, mail_key, service_desk_key: nil)
|
2021-12-11 22:18:48 +05:30
|
|
|
if service_desk_key
|
|
|
|
mail_key ||= service_desk_key
|
2020-07-28 23:09:34 +05:30
|
|
|
@service_desk_key = service_desk_key
|
|
|
|
end
|
2021-12-11 22:18:48 +05:30
|
|
|
|
|
|
|
super(mail, mail_key)
|
|
|
|
|
|
|
|
match_project_slug || match_legacy_project_slug
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def can_handle?
|
|
|
|
Gitlab::ServiceDesk.supported? && (project_id || can_handle_legacy_format? || service_desk_key)
|
|
|
|
end
|
|
|
|
|
|
|
|
def execute
|
|
|
|
raise ProjectNotFound if project.nil?
|
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
create_issue_or_note
|
2021-03-11 19:13:27 +05:30
|
|
|
|
2022-05-07 20:08:51 +05:30
|
|
|
if issue_creator_address
|
2021-03-11 19:13:27 +05:30
|
|
|
add_email_participant
|
2022-01-26 12:08:38 +05:30
|
|
|
send_thank_you_email unless reply_email?
|
2021-03-11 19:13:27 +05:30
|
|
|
end
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
|
2021-12-11 22:18:48 +05:30
|
|
|
def match_project_slug
|
|
|
|
return if mail_key&.include?('/')
|
|
|
|
return unless matched = HANDLER_REGEX.match(mail_key.to_s)
|
|
|
|
|
|
|
|
@project_slug = matched[:project_slug]
|
|
|
|
@project_id = matched[:project_id]&.to_i
|
|
|
|
end
|
|
|
|
|
|
|
|
def match_legacy_project_slug
|
|
|
|
return unless matched = HANDLER_REGEX_LEGACY.match(mail_key.to_s)
|
|
|
|
|
|
|
|
@project_path = matched[:project_path]
|
|
|
|
end
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
def metrics_event
|
|
|
|
:receive_email_service_desk
|
|
|
|
end
|
|
|
|
|
|
|
|
def project
|
|
|
|
strong_memoize(:project) do
|
2021-12-11 22:18:48 +05:30
|
|
|
project_record = super
|
|
|
|
project_record ||= project_from_key if service_desk_key
|
|
|
|
project_record&.service_desk_enabled? ? project_record : nil
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2021-09-30 23:02:18 +05:30
|
|
|
private
|
|
|
|
|
|
|
|
attr_reader :project_id, :project_path, :service_desk_key
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
def project_from_key
|
|
|
|
return unless match = service_desk_key.match(PROJECT_KEY_PATTERN)
|
|
|
|
|
2021-09-04 01:27:46 +05:30
|
|
|
Project.with_service_desk_key(match[:key]).find do |project|
|
|
|
|
valid_project_key?(project, match[:slug])
|
|
|
|
end
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def valid_project_key?(project, slug)
|
2021-03-08 18:12:59 +05:30
|
|
|
project.present? && slug == project.full_path_slug
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
def create_issue_or_note
|
|
|
|
if reply_email?
|
|
|
|
create_note_from_reply_email
|
|
|
|
else
|
|
|
|
create_issue!
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
def create_issue!
|
2021-11-11 11:23:49 +05:30
|
|
|
@issue = ::Issues::CreateService.new(
|
2021-06-08 01:23:25 +05:30
|
|
|
project: project,
|
|
|
|
current_user: User.support_bot,
|
|
|
|
params: {
|
|
|
|
title: mail.subject,
|
|
|
|
description: message_including_template,
|
|
|
|
confidential: true,
|
2022-05-07 20:08:51 +05:30
|
|
|
external_author: external_author
|
2021-09-30 23:02:18 +05:30
|
|
|
},
|
|
|
|
spam_params: nil
|
2020-07-28 23:09:34 +05:30
|
|
|
).execute
|
|
|
|
|
|
|
|
raise InvalidIssueError unless @issue.persisted?
|
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
begin
|
|
|
|
::Issue::Email.create!(issue: @issue, email_message_id: mail.message_id)
|
|
|
|
rescue StandardError => e
|
|
|
|
Gitlab::ErrorTracking.log_exception(e)
|
|
|
|
end
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
if service_desk_setting&.issue_template_missing?
|
2022-01-26 12:08:38 +05:30
|
|
|
create_template_not_found_note
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def issue_from_reply_to
|
|
|
|
strong_memoize(:issue_from_reply_to) do
|
|
|
|
next unless mail.in_reply_to
|
|
|
|
|
|
|
|
Issue::Email.find_by_email_message_id(mail.in_reply_to)&.issue
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
def reply_email?
|
|
|
|
issue_from_reply_to.present?
|
|
|
|
end
|
|
|
|
|
|
|
|
def create_note_from_reply_email
|
|
|
|
@issue = issue_from_reply_to
|
|
|
|
|
|
|
|
create_note(message_including_reply)
|
|
|
|
end
|
|
|
|
|
2021-06-08 01:23:25 +05:30
|
|
|
def send_thank_you_email
|
|
|
|
Notify.service_desk_thank_you_email(@issue.id).deliver_later
|
2021-09-30 23:02:18 +05:30
|
|
|
Gitlab::Metrics::BackgroundTransaction.current&.add_event(:service_desk_thank_you_email)
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def message_including_template
|
2022-01-26 12:08:38 +05:30
|
|
|
description = message_including_reply_or_only_quotes
|
2020-07-28 23:09:34 +05:30
|
|
|
template_content = service_desk_setting&.issue_template_content
|
|
|
|
|
|
|
|
if template_content.present?
|
|
|
|
description += " \n" + template_content
|
|
|
|
end
|
|
|
|
|
|
|
|
description
|
|
|
|
end
|
|
|
|
|
|
|
|
def service_desk_setting
|
|
|
|
strong_memoize(:service_desk_setting) do
|
|
|
|
project.service_desk_setting
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
def create_template_not_found_note
|
2020-07-28 23:09:34 +05:30
|
|
|
issue_template_key = service_desk_setting&.issue_template_key
|
|
|
|
|
|
|
|
warning_note = <<-MD.strip_heredoc
|
|
|
|
WARNING: The template file #{issue_template_key}.md used for service desk issues is empty or could not be found.
|
|
|
|
Please check service desk settings and update the file to be used.
|
|
|
|
MD
|
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
create_note(warning_note)
|
|
|
|
end
|
2020-07-28 23:09:34 +05:30
|
|
|
|
2022-01-26 12:08:38 +05:30
|
|
|
def create_note(note)
|
2020-07-28 23:09:34 +05:30
|
|
|
::Notes::CreateService.new(
|
|
|
|
project,
|
|
|
|
User.support_bot,
|
2022-01-26 12:08:38 +05:30
|
|
|
noteable: @issue,
|
|
|
|
note: note
|
2020-07-28 23:09:34 +05:30
|
|
|
).execute
|
|
|
|
end
|
|
|
|
|
2022-05-07 20:08:51 +05:30
|
|
|
def issue_creator_address
|
|
|
|
reply_to_address || from_address
|
|
|
|
end
|
|
|
|
|
2020-07-28 23:09:34 +05:30
|
|
|
def from_address
|
2022-05-07 20:08:51 +05:30
|
|
|
mail.from.first || mail.sender
|
|
|
|
end
|
|
|
|
|
|
|
|
def reply_to_address
|
|
|
|
(mail.reply_to || []).first
|
|
|
|
end
|
|
|
|
|
|
|
|
def external_author
|
|
|
|
return issue_creator_address unless reply_to_address && from_address
|
|
|
|
|
|
|
|
_("%{from_address} (reply to: %{reply_to_address})") % { from_address: from_address, reply_to_address: reply_to_address }
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
|
|
|
|
def can_handle_legacy_format?
|
|
|
|
project_path && project_path.include?('/') && !mail_key.include?('+')
|
|
|
|
end
|
|
|
|
|
|
|
|
def author
|
|
|
|
User.support_bot
|
|
|
|
end
|
2021-03-11 19:13:27 +05:30
|
|
|
|
|
|
|
def add_email_participant
|
2022-01-26 12:08:38 +05:30
|
|
|
return if reply_email? && !Feature.enabled?(:issue_email_participants, @issue.project)
|
|
|
|
|
2022-05-07 20:08:51 +05:30
|
|
|
@issue.issue_email_participants.create(email: issue_creator_address)
|
2021-03-11 19:13:27 +05:30
|
|
|
end
|
2020-07-28 23:09:34 +05:30
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|