2018-06-27 03:23:28 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2020-02-19 13:09:10 -05:00
|
|
|
class EmailReceiverWorker # rubocop:disable Scalability/IdempotentWorker
|
2017-11-28 11:08:30 -05:00
|
|
|
include ApplicationWorker
|
2015-08-18 18:46:36 -04:00
|
|
|
|
2021-07-21 08:09:35 -04:00
|
|
|
data_consistency :always
|
|
|
|
|
2021-04-30 14:10:09 -04:00
|
|
|
sidekiq_options retry: 3
|
|
|
|
|
2019-10-18 07:11:44 -04:00
|
|
|
feature_category :issue_tracking
|
2020-03-02 13:07:42 -05:00
|
|
|
urgency :high
|
2020-01-24 13:09:00 -05:00
|
|
|
weight 2
|
2019-10-18 07:11:44 -04:00
|
|
|
|
2021-07-27 02:09:01 -04:00
|
|
|
# https://gitlab.com/gitlab-com/gl-infra/scalability/-/issues/1087#jobs-written-to-redis-without-passing-through-the-application
|
|
|
|
tags :needs_own_queue
|
|
|
|
|
2021-05-09 20:10:37 -04:00
|
|
|
attr_accessor :raw
|
|
|
|
|
2015-08-18 18:46:36 -04:00
|
|
|
def perform(raw)
|
2021-05-09 20:10:37 -04:00
|
|
|
return unless should_perform?
|
2015-08-18 18:46:36 -04:00
|
|
|
|
2021-05-09 20:10:37 -04:00
|
|
|
@raw = raw
|
|
|
|
execute_receiver
|
|
|
|
end
|
|
|
|
|
|
|
|
def should_perform?
|
|
|
|
Gitlab::IncomingEmail.enabled?
|
2015-08-18 18:46:36 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
2021-05-09 20:10:37 -04:00
|
|
|
def execute_receiver
|
|
|
|
receiver.execute
|
|
|
|
log_success
|
|
|
|
rescue StandardError => e
|
|
|
|
log_error(e)
|
|
|
|
handle_failure(e)
|
|
|
|
end
|
|
|
|
|
|
|
|
def receiver
|
|
|
|
@receiver ||= Gitlab::Email::Receiver.new(raw)
|
|
|
|
end
|
|
|
|
|
|
|
|
def logger
|
|
|
|
Sidekiq.logger
|
|
|
|
end
|
|
|
|
|
|
|
|
def log_success
|
|
|
|
logger.info(build_message('Successfully processed message', receiver.mail_metadata))
|
|
|
|
end
|
|
|
|
|
|
|
|
def log_error(error)
|
|
|
|
payload =
|
|
|
|
case error
|
|
|
|
# Unparsable e-mails don't have metadata we can use
|
|
|
|
when Gitlab::Email::EmailUnparsableError, Gitlab::Email::EmptyEmailError
|
|
|
|
{}
|
|
|
|
else
|
|
|
|
mail_metadata
|
|
|
|
end
|
|
|
|
|
|
|
|
# We don't need the backtrace and more details if the e-mail couldn't be processed
|
|
|
|
if error.is_a?(Gitlab::Email::ProcessingError)
|
|
|
|
payload['exception.class'] = error.class.name
|
|
|
|
else
|
|
|
|
Gitlab::ExceptionLogFormatter.format!(error, payload)
|
|
|
|
Gitlab::ErrorTracking.track_exception(error)
|
|
|
|
end
|
|
|
|
|
|
|
|
logger.error(build_message('Error processing message', payload))
|
|
|
|
end
|
|
|
|
|
|
|
|
def build_message(message, params = {})
|
|
|
|
{
|
|
|
|
class: self.class.name,
|
|
|
|
Labkit::Correlation::CorrelationId::LOG_KEY => Labkit::Correlation::CorrelationId.current_id,
|
|
|
|
message: message
|
|
|
|
}.merge(params)
|
|
|
|
end
|
|
|
|
|
|
|
|
def mail_metadata
|
|
|
|
receiver.mail_metadata
|
|
|
|
rescue StandardError => e
|
|
|
|
# We should never get here as long as we check EmailUnparsableError, but
|
|
|
|
# let's be defensive in case we did something wrong.
|
|
|
|
Gitlab::ErrorTracking.track_exception(e)
|
|
|
|
{}
|
|
|
|
end
|
2015-08-19 14:10:21 -04:00
|
|
|
|
2021-05-09 20:10:37 -04:00
|
|
|
def handle_failure(error)
|
2015-08-24 13:57:35 -04:00
|
|
|
return unless raw.present?
|
|
|
|
|
2015-08-19 14:10:21 -04:00
|
|
|
can_retry = false
|
2016-05-24 03:16:16 -04:00
|
|
|
reason =
|
2018-07-04 10:02:01 -04:00
|
|
|
case error
|
2016-05-24 03:16:16 -04:00
|
|
|
when Gitlab::Email::UnknownIncomingEmail
|
2019-04-10 09:30:09 -04:00
|
|
|
s_("EmailError|We couldn't figure out what the email is for. Please create your issue or comment through the web interface.")
|
2016-05-24 03:16:16 -04:00
|
|
|
when Gitlab::Email::SentNotificationNotFoundError
|
2019-04-10 09:30:09 -04:00
|
|
|
s_("EmailError|We couldn't figure out what the email is in reply to. Please create your comment through the web interface.")
|
2016-05-24 03:16:16 -04:00
|
|
|
when Gitlab::Email::ProjectNotFound
|
2019-04-10 09:30:09 -04:00
|
|
|
s_("EmailError|We couldn't find the project. Please check if there's any typo.")
|
2016-05-24 03:16:16 -04:00
|
|
|
when Gitlab::Email::EmptyEmailError
|
|
|
|
can_retry = true
|
2019-04-10 09:30:09 -04:00
|
|
|
s_("EmailError|It appears that the email is blank. Make sure your reply is at the top of the email, we can't process inline replies.")
|
2016-05-24 03:16:16 -04:00
|
|
|
when Gitlab::Email::UserNotFoundError
|
2019-04-10 09:30:09 -04:00
|
|
|
s_("EmailError|We couldn't figure out what user corresponds to the email. Please create your comment through the web interface.")
|
2016-05-24 03:16:16 -04:00
|
|
|
when Gitlab::Email::UserBlockedError
|
2019-04-10 09:30:09 -04:00
|
|
|
s_("EmailError|Your account has been blocked. If you believe this is in error, contact a staff member.")
|
2016-05-24 03:16:16 -04:00
|
|
|
when Gitlab::Email::UserNotAuthorizedError
|
2019-04-10 09:30:09 -04:00
|
|
|
s_("EmailError|You are not allowed to perform this action. If you believe this is in error, contact a staff member.")
|
2016-05-24 03:16:16 -04:00
|
|
|
when Gitlab::Email::NoteableNotFoundError
|
2019-04-10 09:30:09 -04:00
|
|
|
s_("EmailError|The thread you are replying to no longer exists, perhaps it was deleted? If you believe this is in error, contact a staff member.")
|
2018-10-24 12:01:44 -04:00
|
|
|
when Gitlab::Email::InvalidAttachment
|
|
|
|
error.message
|
2017-08-24 02:20:36 -04:00
|
|
|
when Gitlab::Email::InvalidRecordError
|
2016-05-24 03:16:16 -04:00
|
|
|
can_retry = true
|
2018-07-04 10:02:01 -04:00
|
|
|
error.message
|
2016-05-24 03:16:16 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
if reason
|
|
|
|
EmailRejectionMailer.rejection(reason, raw, can_retry).deliver_later
|
2015-08-19 14:10:21 -04:00
|
|
|
end
|
2015-08-18 18:46:36 -04:00
|
|
|
end
|
|
|
|
end
|