-
Notifications
You must be signed in to change notification settings - Fork 1.2k
/
Copy pathsimple_email_service.rb
443 lines (415 loc) · 15.9 KB
/
simple_email_service.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
# Copyright 2011-2013 Amazon.com, Inc. or its affiliates. All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License"). You
# may not use this file except in compliance with the License. A copy of
# the License is located at
#
# http://aws.amazon.com/apache2.0/
#
# or in the "license" file accompanying this file. This file is
# distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF
# ANY KIND, either express or implied. See the License for the specific
# language governing permissions and limitations under the License.
require 'aws/core'
require 'aws/simple_email_service/config'
module AWS
# This class is the starting point for working with Amazon
# SimpleEmailService (SES).
#
# To use Amazon SimpleEmailService you must first
# [sign up here](http://aws.amazon.com/ses/)
#
# For more information about Amazon SimpleEmailService:
#
# * [Amazon SimpleEmailService](http://aws.amazon.com/ses/)
# * [Amazon SimpleEmailService Documentation](http://aws.amazon.com/documentation/ses/)
#
# # Credentials
#
# You can setup default credentials for all AWS services via
# AWS.config:
#
# AWS.config(
# :access_key_id => 'YOUR_ACCESS_KEY_ID',
# :secret_access_key => 'YOUR_SECRET_ACCESS_KEY')
#
# Or you can set them directly on the SimpleEmailService interface:
#
# ses = AWS::SimpleEmailService.new(
# :access_key_id => 'YOUR_ACCESS_KEY_ID',
# :secret_access_key => 'YOUR_SECRET_ACCESS_KEY')
#
# # Rails
#
# If you want to use Amazon SimpleEmailService to send email from your
# Rails application you just need to do 2 things:
#
# 1. Configure your AWS credentials with {AWS.config}
# 2. Set SES as the delivery method:
#
# config.action_mailer.delivery_method = :amazon_ses
#
# This has only been tested with Rails 2.3 and Rails 3.0.
#
#
# Regions
#
# SES is not available in all regions. If you configure a default region
# where SES is not available, you will receive a network error. You
# can configure a default region and a default SES region:
#
# AWS.config(
# :region => 'ap-southeast-1',
# :ses => { :region => 'us-west-2' }
# )
#
# [Go here for a complete list of supported regions](http://docs.aws.amazon.com/general/latest/gr/rande.html#ses_region).
#
# # Identities
#
# Before you can send emails, you need to verify one or more identities.
# Identities are email addresses or domain names that you have control over.
# Until you have [requested production access](http://docs.aws.amazon.com/ses/latest/DeveloperGuide/InitialSetup.Customer.html)
# you will only be able to send emails to and from verified email addresses
# and domains.
#
# ## Verifying Email Addresses
#
# You can verify an email address for sending/receiving emails using
# the identities collection.
#
# identity = ses.identities.verify('email@yourdomain.com')
# identity.verified? #=> false
#
# You will be sent an email address with a link. Follow the link to
# verify the email address.
#
# ## Verifying Domains
#
# You can also verify an entire domain for sending and receiving emails.
#
# identity = ses.identities.verify('yourdomain.com')
# identity.verification_token
# #=> "216D+lZbhUL0zOoAkC83/0TAl5lJSzLmzsOjtXM7AeM="
#
# You will be expected to update the DNS records for your domain with
# the given verification token. See the service documentation for
# more details.
#
# ## Listing Identities
#
# You can enumerate all identities:
#
# ses.identities.map(&:identity)
# #=> ['email@foo.com', 'somedomain.com']
#
# You can filter the types of identities enumerated:
#
# domains = ses.identities.domains.map(&:identity)
# email_addresses = ses.identities.email_addresses.map(&:identity)
#
# You can get the verification status and token from identities as well.
#
# # for an email address
# identity = ses.identities['youremail@yourdomain.com']
# identity.verified? #=> true/false
#
# # for a domain
# identity = ses.identities['yourdomain.com']
# identity.verified? #=> true/false
# identity.verification_token #=> '...'
#
# # Sending Email
#
# To send a basic email you can use {#send_email}.
#
# ses.send_email(
# :subject => 'A Sample Email',
# :from => 'sender@domain.com',
# :to => 'receipient@domain.com',
# :body_text => 'Sample email text.',
# :body_html => '<h1>Sample Email</h1>')
#
# If you need to send email with attachments or have other special needs
# that send_email does not support you can use {#send_raw_email}.
#
# ses.send_raw_email(<<EMAIL)
# Subject: A Sample Email
# From: sender@domain.com
# To: receipient@domain.com
#
# Sample email text.
# EMAIL
#
# If you prefer, you can also set the sender and recipient in ruby
# when sending raw emails:
#
# ses.send_raw_email(<<EMAIL, :to => 'to@foo.com', :from => 'from@foo.com')
# Subject: A Sample Email
#
# Sample email text.
# EMAIL
#
# # Quotas
#
# Based on several factors, Amazon SES determines how much email you can
# send and how quickly you can send it. These sending limits are defined
# as follows:
#
# * `:max_send_rate` - Maximum number of emails you can send per second.
# * `:max_24_hour_send` - Maximum number of emails you can send in a
# 24-hour period.
#
# To get your current quotas (and how many emails you have sent in the last
# 24 hours):
#
# ses.quotas
# # => {:max_24_hour_send=>200, :max_send_rate=>1.0, :sent_last_24_hours=>22}
#
# # Statistics
#
# You can get statistics about individual emails:
#
# ses.statistics.each do |stats|
# puts "Sent: #{stats[:sent]}"
# puts "Delivery Attempts: #{stats[:delivery_attempts]}"
# puts "Rejects: #{stats[:rejects]}"
# puts "Bounces: #{stats[:bounces]}"
# puts "Complaints: #{stats[:complaints]}"
# end
#
# @!attribute [r] client
# @return [Client] the low-level SimpleEmailService client object
class SimpleEmailService
autoload :Client, 'aws/simple_email_service/client'
autoload :Errors, 'aws/simple_email_service/errors'
autoload :EmailAddressCollection, 'aws/simple_email_service/email_address_collection'
autoload :Identity, 'aws/simple_email_service/identity'
autoload :IdentityCollection, 'aws/simple_email_service/identity_collection'
autoload :Quotas, 'aws/simple_email_service/quotas'
include Core::ServiceInterface
endpoint_prefix 'email'
# @note This method is deprecated. Use {#identities} instead.
# @return [EmailAddressCollection] Returns a collection that represents
# all of the verified email addresses for your account.
def email_addresses
EmailAddressCollection.new(:config => config)
end
# @return [IdentityCollection]
def identities
IdentityCollection.new(:config => config)
end
# Sends an email.
#
# ses.send_email(
# :subject => 'A Sample Email',
# :to => 'john@doe.com',
# :from => 'no@reply.com',
# :body_text => 'sample text ...',
# :body_html => '<p>sample text ...</p>')
#
# You can also pass multiple email addresses for the `:to`, `:cc`,
# `:bcc` and `:reply_to` options. Email addresses can also be
# formatted with names.
#
# ses.send_email(
# :subject => 'A Sample Email',
# :to => ['"John Doe" <john@doe.com>', '"Jane Doe" <jane@doe.com>'],
# :from => 'no@reply.com',
# :body_text => 'sample text ...')
#
# @param [Hash] options
# @option options [required,String] :subject The subject of the message.
# A short summary of the content, which will appear in the #
# recipient's inbox.
# @option options [required,String] :from The sender's email address.
# @option options [String,Array] :to The address(es) to send the email to.
# @option options [String,Array] :cc The address(es) to cc (carbon copy)
# the email to.
# @option options [String,Array] :bcc The address(es) to bcc (blind
# carbon copy) the email to.
# @option options [String,Array] :reply_to The reply-to email address(es)
# for the message. If the recipient replies to the message, each
# reply-to address will receive the reply.
# @option options [String] :return_path The email address to which
# bounce notifications are to be forwarded. If the message cannot be
# delivered to the recipient, then an error message will be returned
# from the recipient's ISP; this message will then be forwarded to
# the email address specified by the `:return_path` option.
# @option options [String] :body_text The email text contents.
# You must provide `:body_text`, `:body_html` or both.
# @option options [String] :body_html The email html contents.
# You must provide `:body_text`, `:body_html` or both.
# @option options [String] :subject_charset The character set of the
# `:subject` string. If the text must contain any other characters,
# then you must also specify the character set. Examples include
# UTF-8, ISO-8859-1, and Shift_JIS. Defaults to 7-bit ASCII.
# @option options [String] :body_text_charset The character set of the
# `:body_text` string. If the text must contain any other characters,
# then you must also specify the character set. Examples include
# UTF-8, ISO-8859-1, and Shift_JIS. Defaults to 7-bit ASCII.
# @option options [String] :body_html_charset The character set of the
# `:body_html` string. If the text must contain any other characters,
# then you must also specify the character set. Examples include
# UTF-8, ISO-8859-1, and Shift_JIS. Defaults to 7-bit ASCII.
# @option options [String] :body_html
# @return [Core::Response] the SendEmail response
def send_email options = {}
require_each(options, :subject, :from)
require_one_of(options, :to, :cc, :bcc)
require_one_of(options, :body_text, :body_html)
# these three options can be passed strings or arrays of strings,
# but the service requires them in a list (array)
[:to, :cc, :bcc, :reply_to].each do |key|
if options[key]
options[key] = [options[key]].flatten
end
end
accepted_options = {
:subject => %w(message subject data),
:subject_charset => %w(message subject charset),
:to => %w(destination to_addresses),
:cc => %w(destination cc_addresses),
:bcc => %w(destination bcc_addresses),
:from => %w(source),
:reply_to => %w(reply_to_addresses),
:return_path => %w(return_path),
:body_text => %w(message body text data),
:body_text_charset => %w(message body text charset),
:body_html => %w(message body html data),
:body_html_charset => %w(message body html charset),
}
client.send_email(nest_options(options, accepted_options))
end
# Sends a raw email (email message, with header and content specified).
# Useful for sending multipart MIME emails. The raw text of the message
# must comply with Internet email standards; otherwise, the message
# cannot be sent.
#
# raw = <<-EMAIL
# Date: Wed, 1 Jun 2011 09:13:07 -0700
# Subject: A Sample Email
# From: "John Doe" <johndoe@domain.com>
# To: "Jane Doe" <janedoe@domain.com>
# Accept-Language: en-US
# Content-Language: en-US
# Content-Type: text/plain; charset="utf-8"
# Content-Transfer-Encoding: base64
# MIME-Version: 1.0
#
# c2FtcGxlIHRleHQNCg==
# EMAIL
#
# ses.send_raw_email(raw)
#
# Amazon SES has a limit on the total number of recipients per
# message: The combined number of To:, CC: and BCC: email addresses
# cannot exceed 50. If you need to send an email message to a larger
# audience, you can divide your recipient list into groups of 50 or
# fewer, and then call Amazon SES repeatedly to send the message to
# each group.
#
# @param [required, String] raw_message The raw text of the message.
# You can pass in any object whos #to_s returns a valid formatted
# email (e.g. ruby Mail gem). The raw message should:
# * Contain a header and a body, separated by a blank line
# * Contain all required internet email headers
# * Each part of a multipart MIME message must be formatted properly
# * MIME content types must be among those supported by Amazon SES.
# Refer to the Amazon SES Developer Guide for more details.
# * Use content that is base64-encoded, if MIME requires it
# @option options [String,Array] :to One or more email addresses to
# send the email to.
# @option options [String] :from The sender's email address.
# If you specify the :from option, then bounce notifications and
# complaints will be sent to this email address. This takes
# precedence over any Return-Path header that you might include in
# the `raw_message`.
# @return [Core::Response] the SendRawEmail response
def send_raw_email raw_message, options = {}
send_opts = {}
send_opts[:raw_message] = {}
send_opts[:raw_message][:data] = raw_message.to_s
send_opts[:source] = options[:from] if options[:from]
if raw_message.respond_to?(:destinations)
send_opts[:destinations] = raw_message.destinations
end
send_opts[:destinations] = [options[:to]].flatten if options[:to]
response = client.send_raw_email(send_opts)
if raw_message.respond_to?(:message_id=)
raw_message.message_id = "#{response.data[:message_id]}@email.amazonses.com"
end
response
end
# for compatability with ActionMailer
alias_method :deliver, :send_raw_email
alias_method :deliver!, :send_raw_email
def settings; {}; end
# @example
#
# ses.quotas
# # {:max_24_hour_send=>200, :max_send_rate=>1.0, :sent_last_24_hours=>22}
#
# @return [Hash] Returns a hash of SES quotas and limits.
def quotas
Quotas.new(:config => config).to_h
end
# Returns an array of email statistics. Each object in this array is a
# hash with the following keys:
#
# * `:delivery_attempts`
# * `:rejects`
# * `:bounces`
# * `:complaints`
# * `:timestamp`
#
# @return [Array of Hashes] An array of email statistic hashes.
def statistics
response = client.get_send_statistics
response.data[:send_data_points].collect do |data|
{
:sent => data[:timestamp],
:delivery_attempts => data[:delivery_attempts],
:rejects => data[:rejects],
:bounces => data[:bounces],
:complaints => data[:complaints],
}
end
end
# @api private
protected
def require_one_of options, *keys
unless keys.any?{|key| options[key] }
parts = keys.collect{|key| ":#{key}" }.join(', ')
raise ArgumentError, "you must provide at least one of #{parts}"
end
end
# @api private
protected
def require_each options, *keys
keys.each do |key|
unless options[key]
raise ArgumentError, "missing required option :#{key}"
end
end
end
# @api private
protected
def nest_options options, accepted_options
send_opts = {}
accepted_options.each_pair do |option, keys|
next unless options[option]
hash = send_opts
keys.collect{|k| k.to_sym }.each do |key|
hash[key] = {} unless hash[key]
if keys.last == key.to_s
hash[key] = options[option]
else
hash = hash[key]
end
end
end
send_opts
end
end
end