2018-08-15 17:45:57 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2020-01-27 16:08:47 -05:00
|
|
|
class Notify < ApplicationMailer
|
2014-02-24 06:12:29 -05:00
|
|
|
include ActionDispatch::Routing::PolymorphicRoutes
|
2017-06-29 13:06:35 -04:00
|
|
|
include GitlabRoutingHelper
|
2018-12-13 06:15:48 -05:00
|
|
|
include EmailsHelper
|
2020-10-05 14:08:51 -04:00
|
|
|
include ReminderEmailsHelper
|
2019-04-07 14:35:16 -04:00
|
|
|
include IssuablesHelper
|
2014-02-24 06:12:29 -05:00
|
|
|
|
2013-03-19 14:00:41 -04:00
|
|
|
include Emails::Issues
|
|
|
|
include Emails::MergeRequests
|
|
|
|
include Emails::Notes
|
2018-02-06 08:25:46 -05:00
|
|
|
include Emails::PagesDomains
|
2013-03-19 14:00:41 -04:00
|
|
|
include Emails::Projects
|
2013-09-11 14:17:28 -04:00
|
|
|
include Emails::Profile
|
2016-08-25 08:01:10 -04:00
|
|
|
include Emails::Pipelines
|
2016-06-02 10:14:02 -04:00
|
|
|
include Emails::Members
|
2018-09-06 15:20:42 -04:00
|
|
|
include Emails::AutoDevops
|
2018-12-05 10:22:52 -05:00
|
|
|
include Emails::RemoteMirrors
|
2019-10-17 14:08:05 -04:00
|
|
|
include Emails::Releases
|
2020-05-13 23:07:52 -04:00
|
|
|
include Emails::Groups
|
2020-05-27 14:08:14 -04:00
|
|
|
include Emails::Reviews
|
2020-07-14 11:09:05 -04:00
|
|
|
include Emails::ServiceDesk
|
2021-01-29 10:09:40 -05:00
|
|
|
include Emails::InProductMarketing
|
2021-07-12 11:09:19 -04:00
|
|
|
include Emails::AdminNotification
|
2022-07-01 11:08:30 -04:00
|
|
|
include Emails::IdentityVerification
|
2013-01-09 00:44:05 -05:00
|
|
|
|
2020-05-27 14:08:14 -04:00
|
|
|
helper TimeboxesHelper
|
2016-11-06 14:27:51 -05:00
|
|
|
helper MergeRequestsHelper
|
|
|
|
helper DiffHelper
|
|
|
|
helper BlobHelper
|
|
|
|
helper EmailsHelper
|
2020-10-05 14:08:51 -04:00
|
|
|
helper ReminderEmailsHelper
|
2016-11-06 14:27:51 -05:00
|
|
|
helper MembersHelper
|
2018-04-26 05:27:44 -04:00
|
|
|
helper AvatarsHelper
|
2016-11-06 14:27:51 -05:00
|
|
|
helper GitlabRoutingHelper
|
2019-04-07 14:35:16 -04:00
|
|
|
helper IssuablesHelper
|
2021-01-29 10:09:40 -05:00
|
|
|
helper InProductMarketingHelper
|
2012-04-23 08:32:56 -04:00
|
|
|
|
2015-01-18 10:29:37 -05:00
|
|
|
def test_email(recipient_email, subject, body)
|
2022-09-08 14:10:47 -04:00
|
|
|
mail_with_locale(to: recipient_email,
|
|
|
|
subject: subject,
|
|
|
|
body: body.html_safe,
|
|
|
|
content_type: 'text/html'
|
|
|
|
)
|
2014-12-04 09:22:10 -05:00
|
|
|
end
|
|
|
|
|
2015-02-25 09:49:40 -05:00
|
|
|
# Splits "gitlab.corp.company.com" up into "gitlab.corp.company.com",
|
|
|
|
# "corp.company.com" and "company.com".
|
|
|
|
# Respects set tld length so "company.co.uk" won't match "somethingelse.uk"
|
|
|
|
def self.allowed_email_domains
|
|
|
|
domain_parts = Gitlab.config.gitlab.host.split(".")
|
|
|
|
allowed_domains = []
|
|
|
|
begin
|
|
|
|
allowed_domains << domain_parts.join(".")
|
|
|
|
domain_parts.shift
|
|
|
|
end while domain_parts.length > ActionDispatch::Http::URL.tld_length
|
|
|
|
|
|
|
|
allowed_domains
|
|
|
|
end
|
|
|
|
|
2015-04-14 06:52:33 -04:00
|
|
|
def can_send_from_user_email?(sender)
|
|
|
|
sender_domain = sender.email.split("@").last
|
|
|
|
self.class.allowed_email_domains.include?(sender_domain)
|
|
|
|
end
|
|
|
|
|
2015-11-18 04:12:09 -05:00
|
|
|
private
|
|
|
|
|
2014-02-17 12:49:42 -05:00
|
|
|
# Return an email address that displays the name of the sender.
|
|
|
|
# Only the displayed name changes; the actual email address is always the same.
|
2020-01-09 13:07:52 -05:00
|
|
|
def sender(sender_id, send_from_user_email: false, sender_name: nil)
|
2015-04-14 06:52:33 -04:00
|
|
|
return unless sender = User.find(sender_id)
|
2015-04-25 19:18:09 -04:00
|
|
|
|
2015-04-14 06:52:33 -04:00
|
|
|
address = default_sender_address
|
2021-03-19 08:09:03 -04:00
|
|
|
address.display_name = sender_name.presence || "#{sender.name} (#{sender.to_reference})"
|
2015-02-25 08:05:45 -05:00
|
|
|
|
2015-04-14 06:52:33 -04:00
|
|
|
if send_from_user_email && can_send_from_user_email?(sender)
|
|
|
|
address.address = sender.email
|
2014-02-17 12:49:42 -05:00
|
|
|
end
|
2015-04-14 06:52:33 -04:00
|
|
|
|
|
|
|
address.format
|
2014-02-17 12:49:42 -05:00
|
|
|
end
|
|
|
|
|
2012-08-20 23:04:53 -04:00
|
|
|
# Formats arguments into a String suitable for use as an email subject
|
|
|
|
#
|
|
|
|
# extra - Extra Strings to be inserted into the subject
|
|
|
|
#
|
|
|
|
# Examples
|
|
|
|
#
|
|
|
|
# >> subject('Lorem ipsum')
|
2014-02-17 10:45:28 -05:00
|
|
|
# => "Lorem ipsum"
|
2012-08-20 23:04:53 -04:00
|
|
|
#
|
|
|
|
# # Automatically inserts Project name when @project is set
|
|
|
|
# >> @project = Project.last
|
|
|
|
# => #<Project id: 1, name: "Ruby on Rails", path: "ruby_on_rails", ...>
|
|
|
|
# >> subject('Lorem ipsum')
|
2014-02-17 10:45:28 -05:00
|
|
|
# => "Ruby on Rails | Lorem ipsum "
|
2012-08-20 23:04:53 -04:00
|
|
|
#
|
|
|
|
# # Accepts multiple arguments
|
|
|
|
# >> subject('Lorem ipsum', 'Dolor sit amet')
|
2014-02-17 10:45:28 -05:00
|
|
|
# => "Lorem ipsum | Dolor sit amet"
|
2012-08-20 23:04:53 -04:00
|
|
|
def subject(*extra)
|
2018-08-15 17:45:57 -04:00
|
|
|
subject = []
|
|
|
|
|
|
|
|
subject << @project.name if @project
|
|
|
|
subject << @group.name if @group
|
|
|
|
subject.concat(extra) if extra.present?
|
|
|
|
subject << Gitlab.config.gitlab.email_subject_suffix if Gitlab.config.gitlab.email_subject_suffix.present?
|
|
|
|
|
|
|
|
subject.join(' | ')
|
2011-12-18 09:07:47 -05:00
|
|
|
end
|
2014-02-24 06:12:29 -05:00
|
|
|
|
|
|
|
# Return a string suitable for inclusion in the 'Message-Id' mail header.
|
|
|
|
#
|
|
|
|
# The message-id is generated from the unique URL to a model object.
|
|
|
|
def message_id(model)
|
|
|
|
model_name = model.class.model_name.singular_route_key
|
|
|
|
"<#{model_name}_#{model.id}@#{Gitlab.config.gitlab.host}>"
|
|
|
|
end
|
|
|
|
|
2015-08-19 13:17:20 -04:00
|
|
|
def mail_thread(model, headers = {})
|
2015-12-19 14:04:40 -05:00
|
|
|
add_project_headers
|
2016-10-12 13:07:36 -04:00
|
|
|
add_unsubscription_headers_and_links
|
2019-08-21 15:23:27 -04:00
|
|
|
add_model_headers(model)
|
2016-10-12 13:07:36 -04:00
|
|
|
|
2015-12-09 05:59:25 -05:00
|
|
|
headers['X-GitLab-Reply-Key'] = reply_key
|
2015-08-18 18:46:36 -04:00
|
|
|
|
2017-12-28 12:25:02 -05:00
|
|
|
@reason = headers['X-GitLab-NotificationReason']
|
|
|
|
|
2017-03-21 11:28:47 -04:00
|
|
|
if Gitlab::IncomingEmail.enabled? && @sent_notification
|
2018-05-03 17:32:20 -04:00
|
|
|
headers['Reply-To'] = Mail::Address.new(Gitlab::IncomingEmail.reply_address(reply_key)).tap do |address|
|
|
|
|
address.display_name = reply_display_name(model)
|
|
|
|
end
|
2015-08-19 13:17:32 -04:00
|
|
|
|
2018-12-11 07:33:27 -05:00
|
|
|
fallback_reply_message_id = "<reply-#{reply_key}@#{Gitlab.config.gitlab.host}>"
|
2017-12-08 16:45:57 -05:00
|
|
|
headers['References'] ||= []
|
2018-07-27 08:10:18 -04:00
|
|
|
headers['References'].unshift(fallback_reply_message_id)
|
2016-03-17 15:03:51 -04:00
|
|
|
|
2015-08-19 13:17:32 -04:00
|
|
|
@reply_by_email = true
|
2015-08-18 18:46:36 -04:00
|
|
|
end
|
|
|
|
|
2022-09-08 14:10:47 -04:00
|
|
|
mail_with_locale(headers)
|
2014-02-24 06:12:29 -05:00
|
|
|
end
|
|
|
|
|
2018-05-03 17:32:20 -04:00
|
|
|
# `model` is used on EE code
|
|
|
|
def reply_display_name(_model)
|
|
|
|
@project.full_name
|
|
|
|
end
|
|
|
|
|
2015-08-19 13:17:20 -04:00
|
|
|
# Send an email that starts a new conversation thread,
|
|
|
|
# with headers suitable for grouping by thread in email clients.
|
|
|
|
#
|
|
|
|
# See: mail_answer_thread
|
|
|
|
def mail_new_thread(model, headers = {})
|
2016-03-17 15:03:51 -04:00
|
|
|
headers['Message-ID'] = message_id(model)
|
2015-08-19 13:17:20 -04:00
|
|
|
|
|
|
|
mail_thread(model, headers)
|
|
|
|
end
|
|
|
|
|
2014-02-24 06:12:29 -05:00
|
|
|
# Send an email that responds to an existing conversation thread,
|
|
|
|
# with headers suitable for grouping by thread in email clients.
|
|
|
|
#
|
|
|
|
# For grouping emails by thread, email clients heuristics require the answers to:
|
|
|
|
#
|
|
|
|
# * have a subject that begin by 'Re: '
|
|
|
|
# * have a 'In-Reply-To' or 'References' header that references the original 'Message-ID'
|
|
|
|
#
|
2015-08-18 18:46:36 -04:00
|
|
|
def mail_answer_thread(model, headers = {})
|
2015-10-01 04:44:50 -04:00
|
|
|
headers['Message-ID'] = "<#{SecureRandom.hex}@#{Gitlab.config.gitlab.host}>"
|
2014-02-24 06:12:29 -05:00
|
|
|
headers['In-Reply-To'] = message_id(model)
|
2018-07-27 08:10:18 -04:00
|
|
|
headers['References'] = [message_id(model)]
|
2014-02-24 06:12:29 -05:00
|
|
|
|
2018-12-10 07:59:17 -05:00
|
|
|
headers[:subject] = "Re: #{headers[:subject]}" if headers[:subject]
|
2015-08-18 20:02:26 -04:00
|
|
|
|
2015-08-19 13:17:20 -04:00
|
|
|
mail_thread(model, headers)
|
2014-02-24 06:12:29 -05:00
|
|
|
end
|
2015-03-27 07:58:23 -04:00
|
|
|
|
2017-09-15 02:58:21 -04:00
|
|
|
def mail_answer_note_thread(model, note, headers = {})
|
|
|
|
headers['Message-ID'] = message_id(note)
|
2017-12-08 16:45:57 -05:00
|
|
|
headers['In-Reply-To'] = message_id(note.references.last)
|
|
|
|
headers['References'] = note.references.map { |ref| message_id(ref) }
|
2017-09-15 02:58:21 -04:00
|
|
|
|
2021-04-15 23:09:00 -04:00
|
|
|
headers['X-GitLab-Discussion-ID'] = note.discussion.id if note.part_of_discussion? || note.can_be_discussion_note?
|
2017-09-15 02:58:21 -04:00
|
|
|
|
2018-12-11 07:33:27 -05:00
|
|
|
headers[:subject] = "Re: #{headers[:subject]}" if headers[:subject]
|
2017-09-15 02:58:21 -04:00
|
|
|
|
|
|
|
mail_thread(model, headers)
|
|
|
|
end
|
|
|
|
|
2015-08-18 18:46:36 -04:00
|
|
|
def reply_key
|
2015-09-21 03:46:47 -04:00
|
|
|
@reply_key ||= SentNotification.reply_key
|
2015-08-18 18:46:36 -04:00
|
|
|
end
|
2015-12-19 14:04:40 -05:00
|
|
|
|
2019-08-21 15:23:27 -04:00
|
|
|
# This method applies threading headers to the email to identify
|
|
|
|
# the instance we are discussing.
|
|
|
|
#
|
|
|
|
# All model instances must have `#id`, and may implement `#iid`.
|
|
|
|
def add_model_headers(object)
|
|
|
|
# Use replacement so we don't strip the module.
|
|
|
|
prefix = "X-GitLab-#{object.class.name.gsub(/::/, '-')}"
|
|
|
|
|
|
|
|
headers["#{prefix}-ID"] = object.id
|
|
|
|
headers["#{prefix}-IID"] = object.iid if object.respond_to?(:iid)
|
|
|
|
end
|
|
|
|
|
2015-12-19 14:04:40 -05:00
|
|
|
def add_project_headers
|
|
|
|
return unless @project
|
|
|
|
|
|
|
|
headers['X-GitLab-Project'] = @project.name
|
|
|
|
headers['X-GitLab-Project-Id'] = @project.id
|
2017-07-20 05:34:09 -04:00
|
|
|
headers['X-GitLab-Project-Path'] = @project.full_path
|
2018-12-13 06:15:48 -05:00
|
|
|
headers['List-Id'] = "#{@project.full_path} <#{create_list_id_string(@project)}>"
|
2015-12-19 14:04:40 -05:00
|
|
|
end
|
2016-10-12 13:07:36 -04:00
|
|
|
|
|
|
|
def add_unsubscription_headers_and_links
|
|
|
|
return unless !@labels_url && @sent_notification && @sent_notification.unsubscribable?
|
|
|
|
|
|
|
|
list_unsubscribe_methods = [unsubscribe_sent_notification_url(@sent_notification, force: true)]
|
|
|
|
if Gitlab::IncomingEmail.enabled? && Gitlab::IncomingEmail.supports_wildcard?
|
|
|
|
list_unsubscribe_methods << "mailto:#{Gitlab::IncomingEmail.unsubscribe_address(reply_key)}"
|
|
|
|
end
|
|
|
|
|
|
|
|
headers['List-Unsubscribe'] = list_unsubscribe_methods.map { |e| "<#{e}>" }.join(',')
|
2017-03-21 11:28:47 -04:00
|
|
|
@unsubscribe_url = unsubscribe_sent_notification_url(@sent_notification)
|
2016-10-12 13:07:36 -04:00
|
|
|
end
|
2022-05-24 05:09:17 -04:00
|
|
|
|
|
|
|
def email_with_layout(to:, subject:, layout: 'mailer')
|
2022-09-08 14:10:47 -04:00
|
|
|
mail_with_locale(to: to, subject: subject) do |format|
|
2022-05-24 05:09:17 -04:00
|
|
|
format.html { render layout: layout }
|
|
|
|
format.text { render layout: layout }
|
|
|
|
end
|
|
|
|
end
|
2011-10-08 17:36:38 -04:00
|
|
|
end
|
2019-09-13 09:26:31 -04:00
|
|
|
|
2021-05-11 17:10:21 -04:00
|
|
|
Notify.prepend_mod_with('Notify')
|