mail-trunk 2.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/CHANGELOG.rdoc +555 -0
- data/Dependencies.txt +3 -0
- data/Gemfile +29 -0
- data/README.mkd +583 -0
- data/Rakefile +66 -0
- data/TODO.rdoc +9 -0
- data/lib/VERSION +4 -0
- data/lib/mail.rb +89 -0
- data/lib/mail/attachments_list.rb +105 -0
- data/lib/mail/body.rb +292 -0
- data/lib/mail/configuration.rb +73 -0
- data/lib/mail/core_extensions/nil.rb +17 -0
- data/lib/mail/core_extensions/object.rb +13 -0
- data/lib/mail/core_extensions/shellwords.rb +57 -0
- data/lib/mail/core_extensions/smtp.rb +25 -0
- data/lib/mail/core_extensions/string.rb +31 -0
- data/lib/mail/core_extensions/string/access.rb +104 -0
- data/lib/mail/core_extensions/string/multibyte.rb +78 -0
- data/lib/mail/elements.rb +14 -0
- data/lib/mail/elements/address.rb +306 -0
- data/lib/mail/elements/address_list.rb +74 -0
- data/lib/mail/elements/content_disposition_element.rb +30 -0
- data/lib/mail/elements/content_location_element.rb +25 -0
- data/lib/mail/elements/content_transfer_encoding_element.rb +24 -0
- data/lib/mail/elements/content_type_element.rb +35 -0
- data/lib/mail/elements/date_time_element.rb +26 -0
- data/lib/mail/elements/envelope_from_element.rb +34 -0
- data/lib/mail/elements/message_ids_element.rb +29 -0
- data/lib/mail/elements/mime_version_element.rb +26 -0
- data/lib/mail/elements/phrase_list.rb +21 -0
- data/lib/mail/elements/received_element.rb +30 -0
- data/lib/mail/encodings.rb +266 -0
- data/lib/mail/encodings/7bit.rb +31 -0
- data/lib/mail/encodings/8bit.rb +31 -0
- data/lib/mail/encodings/base64.rb +33 -0
- data/lib/mail/encodings/binary.rb +31 -0
- data/lib/mail/encodings/quoted_printable.rb +38 -0
- data/lib/mail/encodings/transfer_encoding.rb +58 -0
- data/lib/mail/envelope.rb +35 -0
- data/lib/mail/field.rb +224 -0
- data/lib/mail/field_list.rb +33 -0
- data/lib/mail/fields.rb +35 -0
- data/lib/mail/fields/bcc_field.rb +56 -0
- data/lib/mail/fields/cc_field.rb +55 -0
- data/lib/mail/fields/comments_field.rb +41 -0
- data/lib/mail/fields/common/address_container.rb +16 -0
- data/lib/mail/fields/common/common_address.rb +125 -0
- data/lib/mail/fields/common/common_date.rb +42 -0
- data/lib/mail/fields/common/common_field.rb +51 -0
- data/lib/mail/fields/common/common_message_id.rb +44 -0
- data/lib/mail/fields/common/parameter_hash.rb +58 -0
- data/lib/mail/fields/content_description_field.rb +19 -0
- data/lib/mail/fields/content_disposition_field.rb +69 -0
- data/lib/mail/fields/content_id_field.rb +63 -0
- data/lib/mail/fields/content_location_field.rb +42 -0
- data/lib/mail/fields/content_transfer_encoding_field.rb +50 -0
- data/lib/mail/fields/content_type_field.rb +198 -0
- data/lib/mail/fields/date_field.rb +57 -0
- data/lib/mail/fields/from_field.rb +55 -0
- data/lib/mail/fields/in_reply_to_field.rb +55 -0
- data/lib/mail/fields/keywords_field.rb +44 -0
- data/lib/mail/fields/message_id_field.rb +83 -0
- data/lib/mail/fields/mime_version_field.rb +53 -0
- data/lib/mail/fields/optional_field.rb +13 -0
- data/lib/mail/fields/received_field.rb +75 -0
- data/lib/mail/fields/references_field.rb +55 -0
- data/lib/mail/fields/reply_to_field.rb +55 -0
- data/lib/mail/fields/resent_bcc_field.rb +55 -0
- data/lib/mail/fields/resent_cc_field.rb +55 -0
- data/lib/mail/fields/resent_date_field.rb +35 -0
- data/lib/mail/fields/resent_from_field.rb +55 -0
- data/lib/mail/fields/resent_message_id_field.rb +34 -0
- data/lib/mail/fields/resent_sender_field.rb +62 -0
- data/lib/mail/fields/resent_to_field.rb +55 -0
- data/lib/mail/fields/return_path_field.rb +65 -0
- data/lib/mail/fields/sender_field.rb +67 -0
- data/lib/mail/fields/structured_field.rb +51 -0
- data/lib/mail/fields/subject_field.rb +16 -0
- data/lib/mail/fields/to_field.rb +55 -0
- data/lib/mail/fields/unstructured_field.rb +182 -0
- data/lib/mail/header.rb +265 -0
- data/lib/mail/indifferent_hash.rb +146 -0
- data/lib/mail/mail.rb +255 -0
- data/lib/mail/message.rb +2017 -0
- data/lib/mail/multibyte.rb +42 -0
- data/lib/mail/multibyte/chars.rb +474 -0
- data/lib/mail/multibyte/exceptions.rb +8 -0
- data/lib/mail/multibyte/unicode.rb +392 -0
- data/lib/mail/multibyte/utils.rb +60 -0
- data/lib/mail/network.rb +13 -0
- data/lib/mail/network/delivery_methods/file_delivery.rb +40 -0
- data/lib/mail/network/delivery_methods/sendmail.rb +62 -0
- data/lib/mail/network/delivery_methods/smtp.rb +137 -0
- data/lib/mail/network/delivery_methods/smtp_connection.rb +74 -0
- data/lib/mail/network/delivery_methods/test_mailer.rb +40 -0
- data/lib/mail/network/retriever_methods/base.rb +63 -0
- data/lib/mail/network/retriever_methods/imap.rb +158 -0
- data/lib/mail/network/retriever_methods/pop3.rb +140 -0
- data/lib/mail/network/retriever_methods/test_retriever.rb +47 -0
- data/lib/mail/parsers/address_lists.rb +64 -0
- data/lib/mail/parsers/address_lists.treetop +19 -0
- data/lib/mail/parsers/content_disposition.rb +535 -0
- data/lib/mail/parsers/content_disposition.treetop +46 -0
- data/lib/mail/parsers/content_location.rb +139 -0
- data/lib/mail/parsers/content_location.treetop +20 -0
- data/lib/mail/parsers/content_transfer_encoding.rb +162 -0
- data/lib/mail/parsers/content_transfer_encoding.treetop +20 -0
- data/lib/mail/parsers/content_type.rb +967 -0
- data/lib/mail/parsers/content_type.treetop +68 -0
- data/lib/mail/parsers/date_time.rb +114 -0
- data/lib/mail/parsers/date_time.treetop +11 -0
- data/lib/mail/parsers/envelope_from.rb +194 -0
- data/lib/mail/parsers/envelope_from.treetop +32 -0
- data/lib/mail/parsers/message_ids.rb +45 -0
- data/lib/mail/parsers/message_ids.treetop +15 -0
- data/lib/mail/parsers/mime_version.rb +144 -0
- data/lib/mail/parsers/mime_version.treetop +19 -0
- data/lib/mail/parsers/phrase_lists.rb +45 -0
- data/lib/mail/parsers/phrase_lists.treetop +15 -0
- data/lib/mail/parsers/received.rb +71 -0
- data/lib/mail/parsers/received.treetop +11 -0
- data/lib/mail/parsers/rfc2045.rb +464 -0
- data/lib/mail/parsers/rfc2045.treetop +36 -0
- data/lib/mail/parsers/rfc2822.rb +5341 -0
- data/lib/mail/parsers/rfc2822.treetop +410 -0
- data/lib/mail/parsers/rfc2822_obsolete.rb +3757 -0
- data/lib/mail/parsers/rfc2822_obsolete.treetop +241 -0
- data/lib/mail/part.rb +116 -0
- data/lib/mail/parts_list.rb +51 -0
- data/lib/mail/patterns.rb +35 -0
- data/lib/mail/utilities.rb +215 -0
- data/lib/mail/version.rb +24 -0
- data/lib/mail/version_specific/ruby_1_8.rb +98 -0
- data/lib/mail/version_specific/ruby_1_9.rb +113 -0
- data/lib/tasks/corpus.rake +125 -0
- data/lib/tasks/treetop.rake +10 -0
- metadata +221 -0
data/lib/mail/network.rb
ADDED
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
require 'mail/network/retriever_methods/base'
|
|
2
|
+
|
|
3
|
+
module Mail
|
|
4
|
+
autoload :SMTP, 'mail/network/delivery_methods/smtp'
|
|
5
|
+
autoload :FileDelivery, 'mail/network/delivery_methods/file_delivery'
|
|
6
|
+
autoload :Sendmail, 'mail/network/delivery_methods/sendmail'
|
|
7
|
+
autoload :SMTPConnection, 'mail/network/delivery_methods/smtp_connection'
|
|
8
|
+
autoload :TestMailer, 'mail/network/delivery_methods/test_mailer'
|
|
9
|
+
|
|
10
|
+
autoload :POP3, 'mail/network/retriever_methods/pop3'
|
|
11
|
+
autoload :IMAP, 'mail/network/retriever_methods/imap'
|
|
12
|
+
autoload :TestRetriever, 'mail/network/retriever_methods/test_retriever'
|
|
13
|
+
end
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
module Mail
|
|
2
|
+
|
|
3
|
+
# FileDelivery class delivers emails into multiple files based on the destination
|
|
4
|
+
# address. Each file is appended to if it already exists.
|
|
5
|
+
#
|
|
6
|
+
# So if you have an email going to fred@test, bob@test, joe@anothertest, and you
|
|
7
|
+
# set your location path to /path/to/mails then FileDelivery will create the directory
|
|
8
|
+
# if it does not exist, and put one copy of the email in three files, called
|
|
9
|
+
# "fred@test", "bob@test" and "joe@anothertest"
|
|
10
|
+
#
|
|
11
|
+
# Make sure the path you specify with :location is writable by the Ruby process
|
|
12
|
+
# running Mail.
|
|
13
|
+
class FileDelivery
|
|
14
|
+
|
|
15
|
+
if RUBY_VERSION >= '1.9.1'
|
|
16
|
+
require 'fileutils'
|
|
17
|
+
else
|
|
18
|
+
require 'ftools'
|
|
19
|
+
end
|
|
20
|
+
|
|
21
|
+
def initialize(values)
|
|
22
|
+
self.settings = { :location => './mails' }.merge!(values)
|
|
23
|
+
end
|
|
24
|
+
|
|
25
|
+
attr_accessor :settings
|
|
26
|
+
|
|
27
|
+
def deliver!(mail)
|
|
28
|
+
if ::File.respond_to?(:makedirs)
|
|
29
|
+
::File.makedirs settings[:location]
|
|
30
|
+
else
|
|
31
|
+
::FileUtils.mkdir_p settings[:location]
|
|
32
|
+
end
|
|
33
|
+
|
|
34
|
+
mail.destinations.uniq.each do |to|
|
|
35
|
+
::File.open(::File.join(settings[:location], to), 'a') { |f| "#{f.write(mail.encoded)}\r\n\r\n" }
|
|
36
|
+
end
|
|
37
|
+
end
|
|
38
|
+
|
|
39
|
+
end
|
|
40
|
+
end
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
module Mail
|
|
2
|
+
# A delivery method implementation which sends via sendmail.
|
|
3
|
+
#
|
|
4
|
+
# To use this, first find out where the sendmail binary is on your computer,
|
|
5
|
+
# if you are on a mac or unix box, it is usually in /usr/sbin/sendmail, this will
|
|
6
|
+
# be your sendmail location.
|
|
7
|
+
#
|
|
8
|
+
# Mail.defaults do
|
|
9
|
+
# delivery_method :sendmail
|
|
10
|
+
# end
|
|
11
|
+
#
|
|
12
|
+
# Or if your sendmail binary is not at '/usr/sbin/sendmail'
|
|
13
|
+
#
|
|
14
|
+
# Mail.defaults do
|
|
15
|
+
# delivery_method :sendmail, :location => '/absolute/path/to/your/sendmail'
|
|
16
|
+
# end
|
|
17
|
+
#
|
|
18
|
+
# Then just deliver the email as normal:
|
|
19
|
+
#
|
|
20
|
+
# Mail.deliver do
|
|
21
|
+
# to 'mikel@test.lindsaar.net'
|
|
22
|
+
# from 'ada@test.lindsaar.net'
|
|
23
|
+
# subject 'testing sendmail'
|
|
24
|
+
# body 'testing sendmail'
|
|
25
|
+
# end
|
|
26
|
+
#
|
|
27
|
+
# Or by calling deliver on a Mail message
|
|
28
|
+
#
|
|
29
|
+
# mail = Mail.new do
|
|
30
|
+
# to 'mikel@test.lindsaar.net'
|
|
31
|
+
# from 'ada@test.lindsaar.net'
|
|
32
|
+
# subject 'testing sendmail'
|
|
33
|
+
# body 'testing sendmail'
|
|
34
|
+
# end
|
|
35
|
+
#
|
|
36
|
+
# mail.deliver!
|
|
37
|
+
class Sendmail
|
|
38
|
+
|
|
39
|
+
def initialize(values)
|
|
40
|
+
self.settings = { :location => '/usr/sbin/sendmail',
|
|
41
|
+
:arguments => '-i -t' }.merge(values)
|
|
42
|
+
end
|
|
43
|
+
|
|
44
|
+
attr_accessor :settings
|
|
45
|
+
|
|
46
|
+
def deliver!(mail)
|
|
47
|
+
envelope_from = mail.return_path || mail.sender || mail.from_addrs.first
|
|
48
|
+
return_path = "-f \"#{envelope_from.to_s.shellescape}\"" if envelope_from
|
|
49
|
+
|
|
50
|
+
arguments = [settings[:arguments], return_path].compact.join(" ")
|
|
51
|
+
|
|
52
|
+
Sendmail.call(settings[:location], arguments, mail.destinations.collect(&:shellescape).join(" "), mail)
|
|
53
|
+
end
|
|
54
|
+
|
|
55
|
+
def Sendmail.call(path, arguments, destinations, mail)
|
|
56
|
+
IO.popen("#{path} #{arguments} #{destinations}", "w+") do |io|
|
|
57
|
+
io.puts mail.encoded.to_lf
|
|
58
|
+
io.flush
|
|
59
|
+
end
|
|
60
|
+
end
|
|
61
|
+
end
|
|
62
|
+
end
|
|
@@ -0,0 +1,137 @@
|
|
|
1
|
+
module Mail
|
|
2
|
+
# == Sending Email with SMTP
|
|
3
|
+
#
|
|
4
|
+
# Mail allows you to send emails using SMTP. This is done by wrapping Net::SMTP in
|
|
5
|
+
# an easy to use manner.
|
|
6
|
+
#
|
|
7
|
+
# === Sending via SMTP server on Localhost
|
|
8
|
+
#
|
|
9
|
+
# Sending locally (to a postfix or sendmail server running on localhost) requires
|
|
10
|
+
# no special setup. Just to Mail.deliver &block or message.deliver! and it will
|
|
11
|
+
# be sent in this method.
|
|
12
|
+
#
|
|
13
|
+
# === Sending via MobileMe
|
|
14
|
+
#
|
|
15
|
+
# Mail.defaults do
|
|
16
|
+
# delivery_method :smtp, { :address => "smtp.me.com",
|
|
17
|
+
# :port => 587,
|
|
18
|
+
# :domain => 'your.host.name',
|
|
19
|
+
# :user_name => '<username>',
|
|
20
|
+
# :password => '<password>',
|
|
21
|
+
# :authentication => 'plain',
|
|
22
|
+
# :enable_starttls_auto => true }
|
|
23
|
+
# end
|
|
24
|
+
#
|
|
25
|
+
# === Sending via GMail
|
|
26
|
+
#
|
|
27
|
+
# Mail.defaults do
|
|
28
|
+
# delivery_method :smtp, { :address => "smtp.gmail.com",
|
|
29
|
+
# :port => 587,
|
|
30
|
+
# :domain => 'your.host.name',
|
|
31
|
+
# :user_name => '<username>',
|
|
32
|
+
# :password => '<password>',
|
|
33
|
+
# :authentication => 'plain',
|
|
34
|
+
# :enable_starttls_auto => true }
|
|
35
|
+
# end
|
|
36
|
+
#
|
|
37
|
+
# === Certificate verification
|
|
38
|
+
#
|
|
39
|
+
# When using TLS, some mail servers provide certificates that are self-signed
|
|
40
|
+
# or whose names do not exactly match the hostname given in the address.
|
|
41
|
+
# OpenSSL will reject these by default. The best remedy is to use the correct
|
|
42
|
+
# hostname or update the certificate authorities trusted by your ruby. If
|
|
43
|
+
# that isn't possible, you can control this behavior with
|
|
44
|
+
# an :openssl_verify_mode setting. Its value may be either an OpenSSL
|
|
45
|
+
# verify mode constant (OpenSSL::SSL::VERIFY_NONE), or a string containing
|
|
46
|
+
# the name of an OpenSSL verify mode (none, peer, client_once,
|
|
47
|
+
# fail_if_no_peer_cert).
|
|
48
|
+
#
|
|
49
|
+
# === Others
|
|
50
|
+
#
|
|
51
|
+
# Feel free to send me other examples that were tricky
|
|
52
|
+
#
|
|
53
|
+
# === Delivering the email
|
|
54
|
+
#
|
|
55
|
+
# Once you have the settings right, sending the email is done by:
|
|
56
|
+
#
|
|
57
|
+
# Mail.deliver do
|
|
58
|
+
# to 'mikel@test.lindsaar.net'
|
|
59
|
+
# from 'ada@test.lindsaar.net'
|
|
60
|
+
# subject 'testing sendmail'
|
|
61
|
+
# body 'testing sendmail'
|
|
62
|
+
# end
|
|
63
|
+
#
|
|
64
|
+
# Or by calling deliver on a Mail message
|
|
65
|
+
#
|
|
66
|
+
# mail = Mail.new do
|
|
67
|
+
# to 'mikel@test.lindsaar.net'
|
|
68
|
+
# from 'ada@test.lindsaar.net'
|
|
69
|
+
# subject 'testing sendmail'
|
|
70
|
+
# body 'testing sendmail'
|
|
71
|
+
# end
|
|
72
|
+
#
|
|
73
|
+
# mail.deliver!
|
|
74
|
+
class SMTP
|
|
75
|
+
|
|
76
|
+
def initialize(values)
|
|
77
|
+
self.settings = { :address => "localhost",
|
|
78
|
+
:port => 25,
|
|
79
|
+
:domain => 'localhost.localdomain',
|
|
80
|
+
:user_name => nil,
|
|
81
|
+
:password => nil,
|
|
82
|
+
:authentication => nil,
|
|
83
|
+
:enable_starttls_auto => true,
|
|
84
|
+
:openssl_verify_mode => nil
|
|
85
|
+
}.merge!(values)
|
|
86
|
+
end
|
|
87
|
+
|
|
88
|
+
attr_accessor :settings
|
|
89
|
+
|
|
90
|
+
# Send the message via SMTP.
|
|
91
|
+
# The from and to attributes are optional. If not set, they are retrieve from the Message.
|
|
92
|
+
def deliver!(mail)
|
|
93
|
+
|
|
94
|
+
# Set the envelope from to be either the return-path, the sender or the first from address
|
|
95
|
+
envelope_from = mail.return_path || mail.sender || mail.from_addrs.first
|
|
96
|
+
if envelope_from.blank?
|
|
97
|
+
raise ArgumentError.new('A sender (Return-Path, Sender or From) required to send a message')
|
|
98
|
+
end
|
|
99
|
+
|
|
100
|
+
destinations ||= mail.destinations if mail.respond_to?(:destinations) && mail.destinations
|
|
101
|
+
if destinations.blank?
|
|
102
|
+
raise ArgumentError.new('At least one recipient (To, Cc or Bcc) is required to send a message')
|
|
103
|
+
end
|
|
104
|
+
|
|
105
|
+
message ||= mail.encoded if mail.respond_to?(:encoded)
|
|
106
|
+
if message.blank?
|
|
107
|
+
raise ArgumentError.new('A encoded content is required to send a message')
|
|
108
|
+
end
|
|
109
|
+
|
|
110
|
+
smtp = Net::SMTP.new(settings[:address], settings[:port])
|
|
111
|
+
if settings[:enable_starttls_auto]
|
|
112
|
+
if smtp.respond_to?(:enable_starttls_auto)
|
|
113
|
+
unless settings[:openssl_verify_mode]
|
|
114
|
+
smtp.enable_starttls_auto
|
|
115
|
+
else
|
|
116
|
+
openssl_verify_mode = settings[:openssl_verify_mode]
|
|
117
|
+
if openssl_verify_mode.kind_of?(String)
|
|
118
|
+
openssl_verify_mode = "OpenSSL::SSL::VERIFY_#{openssl_verify_mode.upcase}".constantize
|
|
119
|
+
end
|
|
120
|
+
context = Net::SMTP.default_ssl_context
|
|
121
|
+
context.verify_mode = openssl_verify_mode
|
|
122
|
+
smtp.enable_starttls_auto(context)
|
|
123
|
+
end
|
|
124
|
+
end
|
|
125
|
+
end
|
|
126
|
+
|
|
127
|
+
response = nil
|
|
128
|
+
smtp.start(settings[:domain], settings[:user_name], settings[:password], settings[:authentication]) do |smtp|
|
|
129
|
+
response = smtp.sendmail(message, envelope_from, destinations)
|
|
130
|
+
end
|
|
131
|
+
|
|
132
|
+
return settings[:return_response] ? response : self
|
|
133
|
+
end
|
|
134
|
+
|
|
135
|
+
|
|
136
|
+
end
|
|
137
|
+
end
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
module Mail
|
|
2
|
+
# == Sending Email with SMTP
|
|
3
|
+
#
|
|
4
|
+
# Mail allows you to send emails using an open SMTP connection. This is done by
|
|
5
|
+
# passing a created Net::SMTP object. This way we can get better performance to
|
|
6
|
+
# our local mail server by reducing the number of connections at any one time.
|
|
7
|
+
#
|
|
8
|
+
# === Sending via SMTP server on Localhost
|
|
9
|
+
#
|
|
10
|
+
# To send mail open a connection with Net::Smtp using any options you like
|
|
11
|
+
# === Delivering the email
|
|
12
|
+
#
|
|
13
|
+
# Once you have the settings right, sending the email is done by:
|
|
14
|
+
#
|
|
15
|
+
# smtp_conn = Net::SMTP.start(settings[:address], settings[:port])
|
|
16
|
+
# Mail.defaults do
|
|
17
|
+
# delivery_method :smtp_connection, { :connection => smtp_conn }
|
|
18
|
+
# end
|
|
19
|
+
#
|
|
20
|
+
# Mail.deliver do
|
|
21
|
+
# to 'mikel@test.lindsaar.net'
|
|
22
|
+
# from 'ada@test.lindsaar.net'
|
|
23
|
+
# subject 'testing sendmail'
|
|
24
|
+
# body 'testing sendmail'
|
|
25
|
+
# end
|
|
26
|
+
#
|
|
27
|
+
# Or by calling deliver on a Mail message
|
|
28
|
+
#
|
|
29
|
+
# mail = Mail.new do
|
|
30
|
+
# to 'mikel@test.lindsaar.net'
|
|
31
|
+
# from 'ada@test.lindsaar.net'
|
|
32
|
+
# subject 'testing sendmail'
|
|
33
|
+
# body 'testing sendmail'
|
|
34
|
+
# end
|
|
35
|
+
#
|
|
36
|
+
# mail.deliver!
|
|
37
|
+
class SMTPConnection
|
|
38
|
+
|
|
39
|
+
def initialize(values)
|
|
40
|
+
raise ArgumentError.new('A Net::SMTP object is required for this delivery method') if values[:connection].nil?
|
|
41
|
+
self.smtp = values[:connection]
|
|
42
|
+
self.settings = values
|
|
43
|
+
end
|
|
44
|
+
|
|
45
|
+
attr_accessor :smtp
|
|
46
|
+
attr_accessor :settings
|
|
47
|
+
|
|
48
|
+
# Send the message via SMTP.
|
|
49
|
+
# The from and to attributes are optional. If not set, they are retrieve from the Message.
|
|
50
|
+
def deliver!(mail)
|
|
51
|
+
|
|
52
|
+
# Set the envelope from to be either the return-path, the sender or the first from address
|
|
53
|
+
envelope_from = mail.return_path || mail.sender || mail.from_addrs.first
|
|
54
|
+
if envelope_from.blank?
|
|
55
|
+
raise ArgumentError.new('A sender (Return-Path, Sender or From) required to send a message')
|
|
56
|
+
end
|
|
57
|
+
|
|
58
|
+
destinations ||= mail.destinations if mail.respond_to?(:destinations) && mail.destinations
|
|
59
|
+
if destinations.blank?
|
|
60
|
+
raise ArgumentError.new('At least one recipient (To, Cc or Bcc) is required to send a message')
|
|
61
|
+
end
|
|
62
|
+
|
|
63
|
+
message ||= mail.encoded if mail.respond_to?(:encoded)
|
|
64
|
+
if message.blank?
|
|
65
|
+
raise ArgumentError.new('A encoded content is required to send a message')
|
|
66
|
+
end
|
|
67
|
+
|
|
68
|
+
response = smtp.sendmail(message, envelope_from, destinations)
|
|
69
|
+
|
|
70
|
+
settings[:return_response] ? response : self
|
|
71
|
+
end
|
|
72
|
+
|
|
73
|
+
end
|
|
74
|
+
end
|
|
@@ -0,0 +1,40 @@
|
|
|
1
|
+
module Mail
|
|
2
|
+
# The TestMailer is a bare bones mailer that does nothing. It is useful
|
|
3
|
+
# when you are testing.
|
|
4
|
+
#
|
|
5
|
+
# It also provides a template of the minimum methods you require to implement
|
|
6
|
+
# if you want to make a custom mailer for Mail
|
|
7
|
+
class TestMailer
|
|
8
|
+
|
|
9
|
+
# Provides a store of all the emails sent with the TestMailer so you can check them.
|
|
10
|
+
def TestMailer.deliveries
|
|
11
|
+
@@deliveries ||= []
|
|
12
|
+
end
|
|
13
|
+
|
|
14
|
+
# Allows you to over write the default deliveries store from an array to some
|
|
15
|
+
# other object. If you just want to clear the store,
|
|
16
|
+
# call TestMailer.deliveries.clear.
|
|
17
|
+
#
|
|
18
|
+
# If you place another object here, please make sure it responds to:
|
|
19
|
+
#
|
|
20
|
+
# * << (message)
|
|
21
|
+
# * clear
|
|
22
|
+
# * length
|
|
23
|
+
# * size
|
|
24
|
+
# * and other common Array methods
|
|
25
|
+
def TestMailer.deliveries=(val)
|
|
26
|
+
@@deliveries = val
|
|
27
|
+
end
|
|
28
|
+
|
|
29
|
+
def initialize(values)
|
|
30
|
+
@settings = {}
|
|
31
|
+
end
|
|
32
|
+
|
|
33
|
+
attr_accessor :settings
|
|
34
|
+
|
|
35
|
+
def deliver!(mail)
|
|
36
|
+
Mail::TestMailer.deliveries << mail
|
|
37
|
+
end
|
|
38
|
+
|
|
39
|
+
end
|
|
40
|
+
end
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
# encoding: utf-8
|
|
2
|
+
|
|
3
|
+
module Mail
|
|
4
|
+
|
|
5
|
+
class Retriever
|
|
6
|
+
|
|
7
|
+
# Get the oldest received email(s)
|
|
8
|
+
#
|
|
9
|
+
# Possible options:
|
|
10
|
+
# count: number of emails to retrieve. The default value is 1.
|
|
11
|
+
# order: order of emails returned. Possible values are :asc or :desc. Default value is :asc.
|
|
12
|
+
#
|
|
13
|
+
def first(options = {}, &block)
|
|
14
|
+
options ||= {}
|
|
15
|
+
options[:what] = :first
|
|
16
|
+
options[:count] ||= 1
|
|
17
|
+
find(options, &block)
|
|
18
|
+
end
|
|
19
|
+
|
|
20
|
+
# Get the most recent received email(s)
|
|
21
|
+
#
|
|
22
|
+
# Possible options:
|
|
23
|
+
# count: number of emails to retrieve. The default value is 1.
|
|
24
|
+
# order: order of emails returned. Possible values are :asc or :desc. Default value is :asc.
|
|
25
|
+
#
|
|
26
|
+
def last(options = {}, &block)
|
|
27
|
+
options ||= {}
|
|
28
|
+
options[:what] = :last
|
|
29
|
+
options[:count] ||= 1
|
|
30
|
+
find(options, &block)
|
|
31
|
+
end
|
|
32
|
+
|
|
33
|
+
# Get all emails.
|
|
34
|
+
#
|
|
35
|
+
# Possible options:
|
|
36
|
+
# order: order of emails returned. Possible values are :asc or :desc. Default value is :asc.
|
|
37
|
+
#
|
|
38
|
+
def all(options = {}, &block)
|
|
39
|
+
options ||= {}
|
|
40
|
+
options[:count] = :all
|
|
41
|
+
find(options, &block)
|
|
42
|
+
end
|
|
43
|
+
|
|
44
|
+
# Find emails in the mailbox, and then deletes them. Without any options, the
|
|
45
|
+
# five last received emails are returned.
|
|
46
|
+
#
|
|
47
|
+
# Possible options:
|
|
48
|
+
# what: last or first emails. The default is :first.
|
|
49
|
+
# order: order of emails returned. Possible values are :asc or :desc. Default value is :asc.
|
|
50
|
+
# count: number of emails to retrieve. The default value is 10. A value of 1 returns an
|
|
51
|
+
# instance of Message, not an array of Message instances.
|
|
52
|
+
# delete_after_find: flag for whether to delete each retreived email after find. Default
|
|
53
|
+
# is true. Call #find if you would like this to default to false.
|
|
54
|
+
#
|
|
55
|
+
def find_and_delete(options = {}, &block)
|
|
56
|
+
options ||= {}
|
|
57
|
+
options[:delete_after_find] ||= true
|
|
58
|
+
find(options, &block)
|
|
59
|
+
end
|
|
60
|
+
|
|
61
|
+
end
|
|
62
|
+
|
|
63
|
+
end
|
|
@@ -0,0 +1,158 @@
|
|
|
1
|
+
# encoding: utf-8
|
|
2
|
+
|
|
3
|
+
module Mail
|
|
4
|
+
# The IMAP retriever allows to get the last, first or all emails from a IMAP server.
|
|
5
|
+
# Each email retrieved (RFC2822) is given as an instance of +Message+.
|
|
6
|
+
#
|
|
7
|
+
# While being retrieved, emails can be yielded if a block is given.
|
|
8
|
+
#
|
|
9
|
+
# === Example of retrieving Emails from GMail:
|
|
10
|
+
#
|
|
11
|
+
# Mail.defaults do
|
|
12
|
+
# retriever_method :imap, { :address => "imap.googlemail.com",
|
|
13
|
+
# :port => 993,
|
|
14
|
+
# :user_name => '<username>',
|
|
15
|
+
# :password => '<password>',
|
|
16
|
+
# :enable_ssl => true }
|
|
17
|
+
# end
|
|
18
|
+
#
|
|
19
|
+
# Mail.all #=> Returns an array of all emails
|
|
20
|
+
# Mail.first #=> Returns the first unread email
|
|
21
|
+
# Mail.last #=> Returns the first unread email
|
|
22
|
+
#
|
|
23
|
+
# You can also pass options into Mail.find to locate an email in your imap mailbox
|
|
24
|
+
# with the following options:
|
|
25
|
+
#
|
|
26
|
+
# mailbox: name of the mailbox used for email retrieval. The default is 'INBOX'.
|
|
27
|
+
# what: last or first emails. The default is :first.
|
|
28
|
+
# order: order of emails returned. Possible values are :asc or :desc. Default value is :asc.
|
|
29
|
+
# count: number of emails to retrieve. The default value is 10. A value of 1 returns an
|
|
30
|
+
# instance of Message, not an array of Message instances.
|
|
31
|
+
#
|
|
32
|
+
# Mail.find(:what => :first, :count => 10, :order => :asc)
|
|
33
|
+
# #=> Returns the first 10 emails in ascending order
|
|
34
|
+
#
|
|
35
|
+
class IMAP < Retriever
|
|
36
|
+
require 'net/imap'
|
|
37
|
+
|
|
38
|
+
def initialize(values)
|
|
39
|
+
self.settings = { :address => "localhost",
|
|
40
|
+
:port => 143,
|
|
41
|
+
:user_name => nil,
|
|
42
|
+
:password => nil,
|
|
43
|
+
:authentication => nil,
|
|
44
|
+
:enable_ssl => false }.merge!(values)
|
|
45
|
+
end
|
|
46
|
+
|
|
47
|
+
attr_accessor :settings
|
|
48
|
+
|
|
49
|
+
# Find emails in a IMAP mailbox. Without any options, the 10 last received emails are returned.
|
|
50
|
+
#
|
|
51
|
+
# Possible options:
|
|
52
|
+
# mailbox: mailbox to search the email(s) in. The default is 'INBOX'.
|
|
53
|
+
# what: last or first emails. The default is :first.
|
|
54
|
+
# order: order of emails returned. Possible values are :asc or :desc. Default value is :asc.
|
|
55
|
+
# count: number of emails to retrieve. The default value is 10. A value of 1 returns an
|
|
56
|
+
# instance of Message, not an array of Message instances.
|
|
57
|
+
# delete_after_find: flag for whether to delete each retreived email after find. Default
|
|
58
|
+
# is false. Use #find_and_delete if you would like this to default to true.
|
|
59
|
+
#
|
|
60
|
+
def find(options={}, &block)
|
|
61
|
+
options = validate_options(options)
|
|
62
|
+
|
|
63
|
+
start do |imap|
|
|
64
|
+
imap.select(options[:mailbox])
|
|
65
|
+
|
|
66
|
+
message_ids = imap.uid_search(options[:keys])
|
|
67
|
+
message_ids.reverse! if options[:what].to_sym == :last
|
|
68
|
+
message_ids = message_ids.first(options[:count]) if options[:count].is_a?(Integer)
|
|
69
|
+
message_ids.reverse! if (options[:what].to_sym == :last && options[:order].to_sym == :asc) ||
|
|
70
|
+
(options[:what].to_sym != :last && options[:order].to_sym == :desc)
|
|
71
|
+
|
|
72
|
+
if block_given?
|
|
73
|
+
message_ids.each do |message_id|
|
|
74
|
+
fetchdata = imap.uid_fetch(message_id, ['RFC822'])[0]
|
|
75
|
+
new_message = Mail.new(fetchdata.attr['RFC822'])
|
|
76
|
+
new_message.mark_for_delete = true if options[:delete_after_find]
|
|
77
|
+
if block.arity == 3
|
|
78
|
+
yield new_message, imap, message_id
|
|
79
|
+
else
|
|
80
|
+
yield new_message
|
|
81
|
+
end
|
|
82
|
+
imap.uid_store(message_id, "+FLAGS", [Net::IMAP::DELETED]) if options[:delete_after_find] && new_message.is_marked_for_delete?
|
|
83
|
+
end
|
|
84
|
+
imap.expunge if options[:delete_after_find]
|
|
85
|
+
else
|
|
86
|
+
emails = []
|
|
87
|
+
message_ids.each do |message_id|
|
|
88
|
+
fetchdata = imap.uid_fetch(message_id, ['RFC822'])[0]
|
|
89
|
+
emails << Mail.new(fetchdata.attr['RFC822'])
|
|
90
|
+
imap.uid_store(message_id, "+FLAGS", [Net::IMAP::DELETED]) if options[:delete_after_find]
|
|
91
|
+
end
|
|
92
|
+
imap.expunge if options[:delete_after_find]
|
|
93
|
+
emails.size == 1 && options[:count] == 1 ? emails.first : emails
|
|
94
|
+
end
|
|
95
|
+
end
|
|
96
|
+
end
|
|
97
|
+
|
|
98
|
+
# Delete all emails from a IMAP mailbox
|
|
99
|
+
def delete_all(mailbox='INBOX')
|
|
100
|
+
mailbox ||= 'INBOX'
|
|
101
|
+
mailbox = Net::IMAP.encode_utf7(mailbox)
|
|
102
|
+
|
|
103
|
+
start do |imap|
|
|
104
|
+
imap.select(mailbox)
|
|
105
|
+
imap.uid_search(['ALL']).each do |message_id|
|
|
106
|
+
imap.uid_store(message_id, "+FLAGS", [Net::IMAP::DELETED])
|
|
107
|
+
end
|
|
108
|
+
imap.expunge
|
|
109
|
+
end
|
|
110
|
+
end
|
|
111
|
+
|
|
112
|
+
# Returns the connection object of the retrievable (IMAP or POP3)
|
|
113
|
+
def connection(&block)
|
|
114
|
+
raise ArgumentError.new('Mail::Retrievable#connection takes a block') unless block_given?
|
|
115
|
+
|
|
116
|
+
start do |imap|
|
|
117
|
+
yield imap
|
|
118
|
+
end
|
|
119
|
+
end
|
|
120
|
+
|
|
121
|
+
private
|
|
122
|
+
|
|
123
|
+
# Set default options
|
|
124
|
+
def validate_options(options)
|
|
125
|
+
options ||= {}
|
|
126
|
+
options[:mailbox] ||= 'INBOX'
|
|
127
|
+
options[:count] ||= 10
|
|
128
|
+
options[:order] ||= :asc
|
|
129
|
+
options[:what] ||= :first
|
|
130
|
+
options[:keys] ||= 'ALL'
|
|
131
|
+
options[:delete_after_find] ||= false
|
|
132
|
+
options[:mailbox] = Net::IMAP.encode_utf7(options[:mailbox])
|
|
133
|
+
|
|
134
|
+
options
|
|
135
|
+
end
|
|
136
|
+
|
|
137
|
+
# Start an IMAP session and ensures that it will be closed in any case.
|
|
138
|
+
def start(config=Mail::Configuration.instance, &block)
|
|
139
|
+
raise ArgumentError.new("Mail::Retrievable#imap_start takes a block") unless block_given?
|
|
140
|
+
|
|
141
|
+
imap = Net::IMAP.new(settings[:address], settings[:port], settings[:enable_ssl], nil, false)
|
|
142
|
+
if settings[:authentication].nil?
|
|
143
|
+
imap.login(settings[:user_name], settings[:password])
|
|
144
|
+
else
|
|
145
|
+
# Note that Net::IMAP#authenticate('LOGIN', ...) is not equal with Net::IMAP#login(...)!
|
|
146
|
+
# (see also http://www.ensta.fr/~diam/ruby/online/ruby-doc-stdlib/libdoc/net/imap/rdoc/classes/Net/IMAP.html#M000718)
|
|
147
|
+
imap.authenticate(settings[:authentication], settings[:user_name], settings[:password])
|
|
148
|
+
end
|
|
149
|
+
|
|
150
|
+
yield imap
|
|
151
|
+
ensure
|
|
152
|
+
if defined?(imap) && imap && !imap.disconnected?
|
|
153
|
+
imap.disconnect
|
|
154
|
+
end
|
|
155
|
+
end
|
|
156
|
+
|
|
157
|
+
end
|
|
158
|
+
end
|