240 lines
7.7 KiB
Ruby
240 lines
7.7 KiB
Ruby
# frozen_string_literal: true
|
|
|
|
module Emails
|
|
module ServiceDesk
|
|
extend ActiveSupport::Concern
|
|
include MarkupHelper
|
|
|
|
EMAIL_ATTACHMENTS_SIZE_LIMIT = 10.megabytes.freeze
|
|
|
|
included do
|
|
layout 'service_desk', only: [:service_desk_thank_you_email, :service_desk_new_note_email]
|
|
end
|
|
|
|
def service_desk_thank_you_email(issue_id)
|
|
setup_service_desk_mail(issue_id)
|
|
|
|
email_sender = sender(
|
|
@support_bot.id,
|
|
send_from_user_email: false,
|
|
sender_name: @service_desk_setting&.outgoing_name,
|
|
sender_email: service_desk_sender_email_address
|
|
)
|
|
|
|
options = {
|
|
from: email_sender,
|
|
to: @issue.external_author,
|
|
subject: "Re: #{subject_base}",
|
|
**service_desk_template_content_options('thank_you')
|
|
}
|
|
|
|
mail_new_thread(@issue, options)
|
|
inject_service_desk_custom_email
|
|
end
|
|
|
|
def service_desk_new_note_email(issue_id, note_id, recipient)
|
|
@note = Note.find(note_id)
|
|
|
|
setup_service_desk_mail(issue_id)
|
|
# Prepare uploads for text replacement in markdown content
|
|
setup_service_desk_attachments
|
|
|
|
email_sender = sender(
|
|
@note.author_id,
|
|
send_from_user_email: false,
|
|
sender_email: service_desk_sender_email_address
|
|
)
|
|
|
|
options = {
|
|
from: email_sender,
|
|
to: recipient,
|
|
subject: subject_base,
|
|
**service_desk_template_content_options('new_note')
|
|
}
|
|
|
|
mail_answer_thread(@issue, options)
|
|
# Add attachments after email init to guide ActiveMailer
|
|
# to choose the correct multipart content types
|
|
add_uploads_as_attachments
|
|
inject_service_desk_custom_email
|
|
end
|
|
|
|
def service_desk_custom_email_verification_email(service_desk_setting)
|
|
@service_desk_setting = service_desk_setting
|
|
|
|
email_sender = sender(
|
|
User.support_bot.id,
|
|
send_from_user_email: false,
|
|
sender_name: @service_desk_setting.outgoing_name,
|
|
sender_email: @service_desk_setting.custom_email
|
|
)
|
|
|
|
@verification_token = @service_desk_setting.custom_email_verification.token
|
|
|
|
subject = format(s_("Notify|Verify custom email address %{email} for %{project_name}"),
|
|
email: @service_desk_setting.custom_email,
|
|
project_name: @service_desk_setting.project.name
|
|
)
|
|
|
|
options = {
|
|
from: email_sender,
|
|
to: @service_desk_setting.custom_email_address_for_verification,
|
|
subject: subject,
|
|
content_type: "text/plain; charset=UTF-8"
|
|
}
|
|
# Outgoing emails from GitLab usually have this set to true.
|
|
# Service Desk email ingestion ignores auto generated emails.
|
|
headers["Auto-Submitted"] = "no"
|
|
|
|
mail_with_locale(options)
|
|
inject_service_desk_custom_email(force: true)
|
|
end
|
|
|
|
def service_desk_verification_triggered_email(service_desk_setting, recipient)
|
|
@service_desk_setting = service_desk_setting
|
|
@triggerer = @service_desk_setting.custom_email_verification.triggerer
|
|
@smtp_address = @service_desk_setting.custom_email_credential.smtp_address
|
|
|
|
subject = format(s_("Notify|Verification for custom email %{email} for %{project_name} triggered"),
|
|
email: @service_desk_setting.custom_email,
|
|
project_name: @service_desk_setting.project.name
|
|
)
|
|
|
|
email_with_layout(to: recipient, subject: subject)
|
|
end
|
|
|
|
def service_desk_verification_result_email(service_desk_setting, recipient)
|
|
@service_desk_setting = service_desk_setting
|
|
@verification = @service_desk_setting.custom_email_verification
|
|
|
|
subject = format(s_("Notify|Verification result for custom email %{email} for %{project_name}"),
|
|
email: @service_desk_setting.custom_email,
|
|
project_name: @service_desk_setting.project.name
|
|
)
|
|
|
|
email_with_layout(to: recipient, subject: subject)
|
|
end
|
|
|
|
private
|
|
|
|
def setup_service_desk_mail(issue_id)
|
|
@issue = Issue.find(issue_id)
|
|
@project = @issue.project
|
|
@support_bot = User.support_bot
|
|
|
|
@service_desk_setting = @project.service_desk_setting
|
|
|
|
@sent_notification = SentNotification.record(@issue, @support_bot.id, reply_key)
|
|
end
|
|
|
|
def service_desk_template_content_options(email_type)
|
|
return {} unless template_body = template_content(email_type)
|
|
|
|
{
|
|
body: template_body,
|
|
content_type: 'text/html; charset=UTF-8'
|
|
}
|
|
end
|
|
|
|
def inject_service_desk_custom_email(force: false)
|
|
return mail if !service_desk_custom_email_enabled? && !force
|
|
return mail unless @service_desk_setting.custom_email_credential.present?
|
|
|
|
mail.delivery_method(::Mail::SMTP, @service_desk_setting.custom_email_credential.delivery_options)
|
|
end
|
|
|
|
def service_desk_custom_email_enabled?
|
|
Feature.enabled?(:service_desk_custom_email, @project) && @service_desk_setting&.custom_email_enabled?
|
|
end
|
|
|
|
def service_desk_sender_email_address
|
|
return unless service_desk_custom_email_enabled?
|
|
|
|
@service_desk_setting.custom_email
|
|
end
|
|
|
|
def template_content(email_type)
|
|
template = Gitlab::Template::ServiceDeskTemplate.find(email_type, @project)
|
|
text = substitute_template_replacements(template.content)
|
|
|
|
context = { project: @project, pipeline: :service_desk_email, uploads_as_attachments: @uploads_as_attachments }
|
|
|
|
context[:author] = @note.author if email_type == 'new_note'
|
|
|
|
markdown(text, context)
|
|
rescue Gitlab::Template::Finders::RepoTemplateFinder::FileNotFoundError
|
|
nil
|
|
end
|
|
|
|
def substitute_template_replacements(template_body)
|
|
template_body
|
|
.gsub(/%\{\s*ISSUE_ID\s*\}/, issue_id)
|
|
.gsub(/%\{\s*ISSUE_PATH\s*\}/, issue_path)
|
|
.gsub(/%\{\s*NOTE_TEXT\s*\}/, note_text)
|
|
.gsub(/%\{\s*ISSUE_DESCRIPTION\s*\}/, issue_description)
|
|
.gsub(/%\{\s*SYSTEM_HEADER\s*\}/, text_header_message.to_s)
|
|
.gsub(/%\{\s*SYSTEM_FOOTER\s*\}/, text_footer_message.to_s)
|
|
.gsub(/%\{\s*UNSUBSCRIBE_URL\s*\}/, unsubscribe_sent_notification_url(@sent_notification))
|
|
.gsub(/%\{\s*ADDITIONAL_TEXT\s*\}/, service_desk_email_additional_text.to_s)
|
|
end
|
|
|
|
def issue_id
|
|
"#{Issue.reference_prefix}#{@issue.iid}"
|
|
end
|
|
|
|
def issue_path
|
|
@issue.to_reference(full: true)
|
|
end
|
|
|
|
def note_text
|
|
@note&.note.to_s
|
|
end
|
|
|
|
def issue_description
|
|
@issue.description_html.to_s
|
|
end
|
|
|
|
def subject_base
|
|
"#{@issue.title} (##{@issue.iid})"
|
|
end
|
|
|
|
def setup_service_desk_attachments
|
|
@uploads_to_attach = []
|
|
# Filepaths we should replace in markdown content
|
|
@uploads_as_attachments = []
|
|
|
|
return unless Feature.enabled?(:service_desk_new_note_email_native_attachments, @note.project)
|
|
|
|
uploaders = find_uploaders_for(@note)
|
|
return if uploaders.nil?
|
|
return if uploaders.sum(&:size) > EMAIL_ATTACHMENTS_SIZE_LIMIT
|
|
|
|
uploaders.each do |uploader|
|
|
@uploads_to_attach << { filename: uploader.filename, content: uploader.read }
|
|
@uploads_as_attachments << "#{uploader.secret}/#{uploader.filename}"
|
|
rescue StandardError => e
|
|
Gitlab::ErrorTracking.track_exception(e, project_id: @note.project.id)
|
|
end
|
|
end
|
|
|
|
def add_uploads_as_attachments
|
|
# We read the uploads before in setup_service_desk_attachments, so let's just add them
|
|
@uploads_to_attach.each do |upload|
|
|
mail.add_file(filename: upload[:filename], content: upload[:content])
|
|
end
|
|
end
|
|
|
|
def find_uploaders_for(note)
|
|
uploads = FileUploader::MARKDOWN_PATTERN.scan(note.note)
|
|
return unless uploads.present?
|
|
|
|
project = note.project
|
|
uploads.map do |secret, file_name|
|
|
UploaderFinder.new(project, secret, file_name).execute
|
|
end
|
|
rescue StandardError => e
|
|
Gitlab::ErrorTracking.track_exception(e, project_id: note.project.id)
|
|
nil
|
|
end
|
|
end
|
|
end
|