Class: EmailFuse::Mailer

Inherits:
Object
  • Object
show all
Defined in:
lib/email_fuse/mailer.rb

Overview

Mailer class used by railtie

Constant Summary collapse

IGNORED_HEADERS =

These are set as headers by the Rails API, but these will be filtered out when constructing the EmailFuse API payload, since they’re are sent as post params. resend.com/docs/api-reference/emails/send-email

%w[
  cc bcc
  from reply-to to subject mime-version
  html text
  content-type tags scheduled_at
  headers options
].freeze
SUPPORTED_OPTIONS =
%w[idempotency_key].freeze

Instance Attribute Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(config) ⇒ Mailer

Returns a new instance of Mailer.

Raises:



23
24
25
26
27
28
29
# File 'lib/email_fuse/mailer.rb', line 23

def initialize(config)
  @config = config
  @settings = config.is_a?(Hash) ? config : {}

  # Check for API key in settings first, then fall back to global config
  raise EmailFuse::Error.new("Make sure your API Key is set", @config) unless api_key
end

Instance Attribute Details

#configObject

Returns the value of attribute config.



8
9
10
# File 'lib/email_fuse/mailer.rb', line 8

def config
  @config
end

#settingsObject

Returns the value of attribute settings.



8
9
10
# File 'lib/email_fuse/mailer.rb', line 8

def settings
  @settings
end

Instance Method Details

#api_keyObject

Returns the API key from settings or global config



32
33
34
# File 'lib/email_fuse/mailer.rb', line 32

def api_key
  @settings[:api_key] || EmailFuse.api_key
end

#base_urlObject

Returns the base URL from settings or global config



37
38
39
# File 'lib/email_fuse/mailer.rb', line 37

def base_url
  @settings[:host] || @settings[:base_url] || EmailFuse.base_url
end

#build_resend_params(mail) ⇒ Object

Builds the payload for sending

Parameters:

  • Mail

    mail rails mail object

Returns:

  • Hash hash with all EmailFuse params



69
70
71
72
73
74
75
76
77
78
79
80
81
# File 'lib/email_fuse/mailer.rb', line 69

def build_resend_params(mail)
  params = {
    from: get_from(mail),
    to: mail.to,
    subject: mail.subject
  }
  params.merge!(get_addons(mail))
  params.merge!(get_headers(mail))
  params.merge!(get_tags(mail))
  params[:attachments] = get_attachments(mail) if mail.attachments.present?
  params.merge!(get_contents(mail))
  params
end

#cleanup_headers(headers) ⇒ Object

Remove nils from header values



131
132
133
# File 'lib/email_fuse/mailer.rb', line 131

def cleanup_headers(headers)
  headers.delete_if { |_k, v| v.nil? }
end

#deliver!(mail) ⇒ Object

Overwritten deliver! method

Parameters:

  • Mail

    mail

Returns:

  • Object resend response



48
49
50
51
52
53
54
55
56
57
58
59
60
# File 'lib/email_fuse/mailer.rb', line 48

def deliver!(mail)
  params = build_resend_params(mail)
  options = get_options(mail) if mail[:options].present?
  options ||= {}

  # Pass api_key and base_url from settings to the request
  options[:api_key] = api_key
  options[:base_url] = base_url

  resp = EmailFuse::Emails.send(params, options: options)
  mail.message_id = resp[:id] if resp[:error].nil?
  resp
end

#get_addons(mail) ⇒ Object

Add cc, bcc, reply_to fields

Parameters:

  • Mail

    mail Rails Mail Object

Returns:

  • Hash hash containing cc/bcc/reply_to attrs



181
182
183
184
185
186
187
# File 'lib/email_fuse/mailer.rb', line 181

def get_addons(mail)
  params = {}
  params[:cc] = mail.cc if mail.cc.present?
  params[:bcc] = mail.bcc if mail.bcc.present?
  params[:reply_to] = mail.reply_to if mail.reply_to.present?
  params
end

#get_attachments(mail) ⇒ Object

Handle attachments when present

Returns:

  • Array attachments array



231
232
233
234
235
236
237
238
239
240
241
242
243
244
# File 'lib/email_fuse/mailer.rb', line 231

def get_attachments(mail)
  attachments = []
  mail.attachments.each do |part|
    attachment = {
      filename: part.filename,
      content: part.body.decoded.bytes
    }

    # Rails uses the auto generated cid for inline attachments
    attachment[:content_id] = part.cid if part.inline?
    attachments.append(attachment)
  end
  attachments
end

#get_contents(mail) ⇒ Object

Gets the body of the email

Parameters:

  • Mail

    mail Rails Mail Object

Returns:

  • Hash hash containing html/text or both attrs



196
197
198
199
200
201
202
203
204
205
206
207
208
# File 'lib/email_fuse/mailer.rb', line 196

def get_contents(mail)
  params = {}
  case mail.mime_type
  when "text/plain"
    params[:text] = mail.body.decoded
  when "text/html"
    params[:html] = mail.body.decoded
  when "multipart/alternative", "multipart/mixed", "multipart/related"
    params[:text] = mail.text_part.decoded if mail.text_part
    params[:html] = mail.html_part.decoded if mail.html_part
  end
  params
end

#get_from(input) ⇒ Object

Properly gets the from attr

Parameters:

  • Mail

    input object

Returns:

  • String from string



217
218
219
220
221
222
223
224
# File 'lib/email_fuse/mailer.rb', line 217

def get_from(input)
  return input.from.first if input[:from].nil?

  from = input[:from].formatted
  return from.first if from.is_a? Array

  from.to_s
end

#get_headers(mail) ⇒ Object

Add custom headers fields.

Both ways are supported:

1. Through the `#mail()` method ie:
  mail(headers: { "X-Custom-Header" => "value" })

2. Through the Rails `#headers` method ie:
  headers["X-Custom-Header"] = "value"

setting the header values through the #mail method will overwrite values set through the #headers method using the same key.

Parameters:

  • Mail

    mail Rails Mail object

Returns:

  • Hash hash with headers param



102
103
104
105
106
107
108
109
110
111
112
# File 'lib/email_fuse/mailer.rb', line 102

def get_headers(mail)
  params = {}

  if mail[:headers].present? || unignored_headers(mail).present?
    params[:headers] = {}
    params[:headers].merge!(headers_values(mail)) if unignored_headers(mail).present?
    params[:headers].merge!(mail_headers_values(mail)) if mail[:headers].present?
  end

  params
end

#get_options(mail) ⇒ Object

Adds additional options fields. Currently supports only :idempotency_key

Parameters:

  • Mail

    mail Rails Mail object

Returns:

  • Hash hash with headers param



121
122
123
124
125
126
127
128
# File 'lib/email_fuse/mailer.rb', line 121

def get_options(mail)
  opts = {}
  if mail[:options].present?
    opts.merge!(mail[:options].unparsed_value)
    opts.delete_if { |k, _v| !SUPPORTED_OPTIONS.include?(k.to_s) }
  end
  opts
end

#get_tags(mail) ⇒ Object

Add tags fields

Parameters:

  • Mail

    mail Rails Mail object

Returns:

  • Hash hash with tags param



168
169
170
171
172
# File 'lib/email_fuse/mailer.rb', line 168

def get_tags(mail)
  params = {}
  params[:tags] = mail[:tags].unparsed_value if mail[:tags].present?
  params
end

#headers_values(mail) ⇒ Object

Gets the values of the headers that are set through the #headers method

Parameters:

  • Mail

    mail Rails Mail object

Returns:

  • Hash hash with headers values



152
153
154
155
156
157
158
159
# File 'lib/email_fuse/mailer.rb', line 152

def headers_values(mail)
  params = {}
  unignored_headers(mail).each do |h|
    params[h.name.to_s] = h.unparsed_value
  end
  cleanup_headers(params)
  params
end

#mail_headers_values(mail) ⇒ Object

Gets the values of the headers that are set through the #mail method

Parameters:

  • Mail

    mail Rails Mail object

Returns:

  • Hash hash with mail headers values



139
140
141
142
143
144
145
146
# File 'lib/email_fuse/mailer.rb', line 139

def mail_headers_values(mail)
  params = {}
  mail[:headers].unparsed_value.each do |k, v|
    params[k.to_s] = v
  end
  cleanup_headers(params)
  params
end

#unignored_headers(mail) ⇒ Object

Get all headers that are not ignored

Parameters:

  • Mail

    mail

Returns:

  • Array headers



253
254
255
# File 'lib/email_fuse/mailer.rb', line 253

def unignored_headers(mail)
  @unignored_headers ||= mail.header_fields.reject { |h| IGNORED_HEADERS.include?(h.name.downcase) }
end