2018-12-14 05:06:12 -05:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2019-01-15 10:55:42 -05:00
|
|
|
# The +Message-ID+ as specified by rfc822 is supposed to be a unique identifier for that individual email.
|
|
|
|
# That makes it an ideal tracking token for debugging and forensics, just like +X-Request-Id+ does for
|
2018-12-12 19:34:05 -05:00
|
|
|
# web request.
|
|
|
|
#
|
|
|
|
# If an inbound email does not, against the rfc822 mandate, specify a Message-ID, one will be generated
|
2019-01-15 10:55:42 -05:00
|
|
|
# using the approach from <tt>Mail::MessageIdField</tt>.
|
2018-09-28 15:19:43 -04:00
|
|
|
module ActionMailbox::InboundEmail::MessageId
|
2018-09-28 14:01:49 -04:00
|
|
|
extend ActiveSupport::Concern
|
|
|
|
|
2018-12-19 16:51:42 -05:00
|
|
|
class_methods do
|
2019-01-15 10:55:42 -05:00
|
|
|
# Create a new +InboundEmail+ from the raw +source+ of the email, which be uploaded as a Active Storage
|
|
|
|
# attachment called +raw_email+. Before the upload, extract the Message-ID from the +source+ and set
|
|
|
|
# it as an attribute on the new +InboundEmail+.
|
2018-10-18 10:23:17 -04:00
|
|
|
def create_and_extract_message_id!(source, **options)
|
2019-01-17 12:13:40 -05:00
|
|
|
message_checksum = Digest::SHA1.hexdigest(source)
|
|
|
|
message_id = extract_message_id(source) || generate_missing_message_id(message_checksum)
|
|
|
|
|
|
|
|
create! options.merge(message_id: message_id, message_checksum: message_checksum) do |inbound_email|
|
2018-10-18 10:23:17 -04:00
|
|
|
inbound_email.raw_email.attach io: StringIO.new(source), filename: "message.eml", content_type: "message/rfc822"
|
2018-10-06 22:02:08 -04:00
|
|
|
end
|
2019-01-17 12:13:40 -05:00
|
|
|
rescue ActiveRecord::RecordNotUnique
|
|
|
|
nil
|
2018-09-28 14:01:49 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
private
|
2018-10-18 10:23:17 -04:00
|
|
|
def extract_message_id(source)
|
2018-12-16 08:27:41 -05:00
|
|
|
Mail.from_source(source).message_id rescue nil
|
2018-09-28 14:01:49 -04:00
|
|
|
end
|
|
|
|
|
2019-01-17 12:13:40 -05:00
|
|
|
def generate_missing_message_id(message_checksum)
|
|
|
|
Mail::MessageIdField.new("<#{message_checksum}@#{::Socket.gethostname}.mail>").message_id.tap do |message_id|
|
|
|
|
logger.warn "Message-ID couldn't be parsed or is missing. Generated a new Message-ID: #{message_id}"
|
|
|
|
end
|
2018-12-16 08:27:41 -05:00
|
|
|
end
|
2019-01-17 12:13:40 -05:00
|
|
|
end
|
2018-09-28 14:01:49 -04:00
|
|
|
end
|