2009-08-07 23:56:54 -04:00
require 'tmpdir'
2009-06-15 14:44:45 -04:00
require " active_support/core_ext/class "
2009-06-15 14:59:28 -04:00
# Use the old layouts until actionmailer gets refactored
2009-06-15 19:29:45 -04:00
require " action_controller/legacy/layout "
2009-06-15 14:44:45 -04:00
2006-03-27 22:22:58 -05:00
module ActionMailer #:nodoc:
2008-05-25 07:29:00 -04:00
# Action Mailer allows you to send email from your application using a mailer model and views.
2004-11-23 20:04:44 -05:00
#
2006-09-02 15:32:45 -04:00
#
2006-05-10 01:38:43 -04:00
# = Mailer Models
2006-09-02 15:32:45 -04:00
#
2008-05-25 07:29:00 -04:00
# To use Action Mailer, you need to create a mailer model.
2008-05-14 15:09:49 -04:00
#
2006-05-10 01:38:43 -04:00
# $ script/generate mailer Notifier
#
2008-05-14 15:09:49 -04:00
# The generated model inherits from ActionMailer::Base. Emails are defined by creating methods within the model which are then
# used to set variables to be used in the mail template, to change options on the mail, or
2006-05-10 01:38:43 -04:00
# to add attachments.
#
# Examples:
#
# class Notifier < ActionMailer::Base
# def signup_notification(recipient)
# recipients recipient.email_address_with_name
2008-12-19 09:27:43 -05:00
# bcc ["bcc@example.com", "Order Watcher <watcher@example.com>"]
2006-05-10 01:38:43 -04:00
# from "system@example.com"
# subject "New account information"
2006-09-02 15:32:45 -04:00
# body :account => recipient
2006-05-10 01:38:43 -04:00
# end
# end
#
# Mailer methods have the following configuration methods available.
#
# * <tt>recipients</tt> - Takes one or more email addresses. These addresses are where your email will be delivered to. Sets the <tt>To:</tt> header.
# * <tt>subject</tt> - The subject of your email. Sets the <tt>Subject:</tt> header.
# * <tt>from</tt> - Who the email you are sending is from. Sets the <tt>From:</tt> header.
# * <tt>cc</tt> - Takes one or more email addresses. These addresses will receive a carbon copy of your email. Sets the <tt>Cc:</tt> header.
2008-05-23 13:40:36 -04:00
# * <tt>bcc</tt> - Takes one or more email addresses. These addresses will receive a blind carbon copy of your email. Sets the <tt>Bcc:</tt> header.
# * <tt>reply_to</tt> - Takes one or more email addresses. These addresses will be listed as the default recipients when replying to your email. Sets the <tt>Reply-To:</tt> header.
2006-05-10 01:38:43 -04:00
# * <tt>sent_on</tt> - The date on which the message was sent. If not set, the header wil be set by the delivery agent.
# * <tt>content_type</tt> - Specify the content type of the message. Defaults to <tt>text/plain</tt>.
# * <tt>headers</tt> - Specify additional headers to be set for the message, e.g. <tt>headers 'X-Mail-Count' => 107370</tt>.
#
2008-03-27 13:54:02 -04:00
# When a <tt>headers 'return-path'</tt> is specified, that value will be used as the 'envelope from'
# address. Setting this is useful when you want delivery notifications sent to a different address than
# the one in <tt>from</tt>.
#
2006-05-10 01:38:43 -04:00
# The <tt>body</tt> method has special behavior. It takes a hash which generates an instance variable
# named after each key in the hash containing the value that that key points to.
#
2008-03-27 13:54:02 -04:00
# So, for example, <tt>body :account => recipient</tt> would result
2008-05-14 15:09:49 -04:00
# in an instance variable <tt>@account</tt> with the value of <tt>recipient</tt> being accessible in the
2006-05-10 01:38:43 -04:00
# view.
#
2006-09-02 15:32:45 -04:00
#
# = Mailer views
#
2008-05-25 07:29:00 -04:00
# Like Action Controller, each mailer class has a corresponding view directory
2006-05-10 01:38:43 -04:00
# in which each method of the class looks for a template with its name.
2007-02-20 19:29:44 -05:00
# To define a template to be used with a mailing, create an <tt>.erb</tt> file with the same name as the method
2008-05-14 15:09:49 -04:00
# in your mailer model. For example, in the mailer defined above, the template at
2007-02-20 19:29:44 -05:00
# <tt>app/views/notifier/signup_notification.erb</tt> would be used to generate the email.
2006-05-10 01:38:43 -04:00
#
# Variables defined in the model are accessible as instance variables in the view.
#
# Emails by default are sent in plain text, so a sample view for our model example might look like this:
#
# Hi <%= @account.name %>,
# Thanks for joining our service! Please check back often.
#
2006-09-02 15:32:45 -04:00
# You can even use Action Pack helpers in these views. For example:
#
# You got a new note!
# <%= truncate(note.body, 25) %>
2008-05-14 15:09:49 -04:00
#
2006-09-02 15:32:45 -04:00
#
2008-04-21 15:31:54 -04:00
# = Generating URLs
2008-05-14 15:09:49 -04:00
#
2008-04-21 15:31:54 -04:00
# URLs can be generated in mailer views using <tt>url_for</tt> or named routes.
2008-05-14 15:09:49 -04:00
# Unlike controllers from Action Pack, the mailer instance doesn't have any context about the incoming request,
# so you'll need to provide all of the details needed to generate a URL.
2006-09-02 15:32:45 -04:00
#
2008-04-21 15:31:54 -04:00
# When using <tt>url_for</tt> you'll need to provide the <tt>:host</tt>, <tt>:controller</tt>, and <tt>:action</tt>:
2008-05-14 15:09:49 -04:00
#
2008-04-21 15:31:54 -04:00
# <%= url_for(:host => "example.com", :controller => "welcome", :action => "greeting") %>
2006-09-02 15:32:45 -04:00
#
2008-04-21 15:31:54 -04:00
# When using named routes you only need to supply the <tt>:host</tt>:
2008-05-14 15:09:49 -04:00
#
2008-04-21 15:31:54 -04:00
# <%= users_url(:host => "example.com") %>
#
# You will want to avoid using the <tt>name_of_route_path</tt> form of named routes because it doesn't make sense to
# generate relative URLs in email messages.
#
2008-05-14 15:09:49 -04:00
# It is also possible to set a default host that will be used in all mailers by setting the <tt>:host</tt> option in
2008-04-21 15:31:54 -04:00
# the <tt>ActionMailer::Base.default_url_options</tt> hash as follows:
#
# ActionMailer::Base.default_url_options[:host] = "example.com"
2008-05-14 15:09:49 -04:00
#
2008-05-09 05:38:02 -04:00
# This can also be set as a configuration option in <tt>config/environment.rb</tt>:
2008-04-21 15:31:54 -04:00
#
# config.action_mailer.default_url_options = { :host => "example.com" }
2006-09-02 15:32:45 -04:00
#
2008-04-21 15:31:54 -04:00
# If you do decide to set a default <tt>:host</tt> for your mailers you will want to use the
# <tt>:only_path => false</tt> option when using <tt>url_for</tt>. This will ensure that absolute URLs are generated because
2008-05-14 15:09:49 -04:00
# the <tt>url_for</tt> view helper will, by default, generate relative URLs when a <tt>:host</tt> option isn't
2008-04-21 15:31:54 -04:00
# explicitly provided.
2006-09-02 15:32:45 -04:00
#
# = Sending mail
#
2008-05-14 15:09:49 -04:00
# Once a mailer action and template are defined, you can deliver your message or create it and save it
2006-05-10 01:38:43 -04:00
# for delivery later:
#
# Notifier.deliver_signup_notification(david) # sends the email
# mail = Notifier.create_signup_notification(david) # => a tmail object
# Notifier.deliver(mail)
2008-05-14 15:09:49 -04:00
#
2006-05-10 01:38:43 -04:00
# You never instantiate your mailer class. Rather, your delivery instance
# methods are automatically wrapped in class methods that start with the word
# <tt>deliver_</tt> followed by the name of the mailer method that you would
# like to deliver. The <tt>signup_notification</tt> method defined above is
# delivered by invoking <tt>Notifier.deliver_signup_notification</tt>.
#
2006-09-02 15:32:45 -04:00
#
# = HTML email
#
2007-02-20 19:29:44 -05:00
# To send mail as HTML, make sure your view (the <tt>.erb</tt> file) generates HTML and
2006-05-10 01:38:43 -04:00
# set the content type to html.
#
# class MyMailer < ActionMailer::Base
2005-05-29 12:36:22 -04:00
# def signup_notification(recipient)
2008-05-16 18:01:32 -04:00
# recipients recipient.email_address_with_name
# subject "New account information"
# from "system@example.com"
# body :account => recipient
# content_type "text/html"
2005-05-29 12:36:22 -04:00
# end
2008-05-14 15:09:49 -04:00
# end
2006-05-10 01:38:43 -04:00
#
2006-09-02 15:32:45 -04:00
#
# = Multipart email
#
2006-05-10 01:38:43 -04:00
# You can explicitly specify multipart messages:
2005-05-29 12:36:22 -04:00
#
2006-05-10 01:38:43 -04:00
# class ApplicationMailer < ActionMailer::Base
2005-05-29 12:36:22 -04:00
# def signup_notification(recipient)
# recipients recipient.email_address_with_name
# subject "New account information"
# from "system@example.com"
2008-05-16 18:01:32 -04:00
# content_type "multipart/alternative"
2005-05-29 12:36:22 -04:00
#
# part :content_type => "text/html",
# :body => render_message("signup-as-html", :account => recipient)
#
# part "text/plain" do |p|
# p.body = render_message("signup-as-plain", :account => recipient)
# p.transfer_encoding = "base64"
# end
# end
2006-05-10 01:38:43 -04:00
# end
2008-05-14 15:09:49 -04:00
#
2008-05-25 07:29:00 -04:00
# Multipart messages can also be used implicitly because Action Mailer will automatically
2006-05-10 01:38:43 -04:00
# detect and use multipart templates, where each template is named after the name of the action, followed
# by the content type. Each such detected template will be added as separate part to the message.
2008-05-14 15:09:49 -04:00
#
2006-05-10 01:38:43 -04:00
# For example, if the following templates existed:
2007-02-20 19:29:44 -05:00
# * signup_notification.text.plain.erb
# * signup_notification.text.html.erb
# * signup_notification.text.xml.builder
# * signup_notification.text.x-yaml.erb
2008-05-14 15:09:49 -04:00
#
2006-05-10 01:38:43 -04:00
# Each would be rendered and added as a separate part to the message,
2008-05-16 18:01:32 -04:00
# with the corresponding content type. The content type for the entire
# message is automatically set to <tt>multipart/alternative</tt>, which indicates
# that the email contains multiple different representations of the same email
# body. The same body hash is passed to each template.
2005-05-29 12:36:22 -04:00
#
2008-05-16 18:01:32 -04:00
# Implicit template rendering is not performed if any attachments or parts have been added to the email.
# This means that you'll have to manually add each part to the email and set the content type of the email
# to <tt>multipart/alternative</tt>.
2006-09-02 15:32:45 -04:00
#
2006-05-10 01:38:43 -04:00
# = Attachments
2006-09-02 15:32:45 -04:00
#
2006-05-10 01:38:43 -04:00
# Attachments can be added by using the +attachment+ method.
#
# Example:
#
# class ApplicationMailer < ActionMailer::Base
2005-05-29 12:36:22 -04:00
# # attachments
# def signup_notification(recipient)
# recipients recipient.email_address_with_name
# subject "New account information"
# from "system@example.com"
#
# attachment :content_type => "image/jpeg",
# :body => File.read("an-image.jpg")
#
# attachment "application/pdf" do |a|
# a.body = generate_your_pdf_here()
# end
2004-11-23 20:04:44 -05:00
# end
2008-05-14 15:09:49 -04:00
# end
2004-12-16 12:53:19 -05:00
#
2006-09-02 15:32:45 -04:00
#
2004-12-16 12:53:19 -05:00
# = Configuration options
#
# These options are specified on the class level, like <tt>ActionMailer::Base.template_root = "/my/templates"</tt>
#
2008-05-02 09:45:23 -04:00
# * <tt>template_root</tt> - Determines the base from which template references will be made.
2004-12-16 12:53:19 -05:00
#
# * <tt>logger</tt> - the logger is used for generating information on the mailing run if available.
# Can be set to nil for no logging. Compatible with both Ruby's own Logger and Log4r loggers.
#
2008-05-02 09:45:23 -04:00
# * <tt>smtp_settings</tt> - Allows detailed configuration for <tt>:smtp</tt> delivery method:
2008-05-09 05:38:02 -04:00
# * <tt>:address</tt> - Allows you to use a remote mail server. Just change it from its default "localhost" setting.
# * <tt>:port</tt> - On the off chance that your mail server doesn't run on port 25, you can change it.
# * <tt>:domain</tt> - If you need to specify a HELO domain, you can do it here.
# * <tt>:user_name</tt> - If your mail server requires authentication, set the username in this setting.
# * <tt>:password</tt> - If your mail server requires authentication, set the password in this setting.
2008-08-06 21:08:27 -04:00
# * <tt>:authentication</tt> - If your mail server requires authentication, you need to specify the authentication type here.
2008-05-16 18:01:32 -04:00
# This is a symbol and one of <tt>:plain</tt>, <tt>:login</tt>, <tt>:cram_md5</tt>.
2009-01-11 05:02:54 -05:00
# * <tt>:enable_starttls_auto</tt> - When set to true, detects if STARTTLS is enabled in your SMTP server and starts to use it.
# It works only on Ruby >= 1.8.7 and Ruby >= 1.9. Default is true.
2004-12-16 12:53:19 -05:00
#
2008-05-16 18:01:32 -04:00
# * <tt>sendmail_settings</tt> - Allows you to override options for the <tt>:sendmail</tt> delivery method.
# * <tt>:location</tt> - The location of the sendmail executable. Defaults to <tt>/usr/sbin/sendmail</tt>.
# * <tt>:arguments</tt> - The command line arguments. Defaults to <tt>-i -t</tt>.
2008-05-09 05:38:02 -04:00
#
2009-08-07 23:56:54 -04:00
# * <tt>file_settings</tt> - Allows you to override options for the <tt>:file</tt> delivery method.
# * <tt>:location</tt> - The directory into which emails will be written. Defaults to the application <tt>tmp/mails</tt>.
#
2008-05-09 05:38:02 -04:00
# * <tt>raise_delivery_errors</tt> - Whether or not errors should be raised if the email fails to be delivered.
2004-12-16 12:53:19 -05:00
#
2009-08-07 23:56:54 -04:00
# * <tt>delivery_method</tt> - Defines a delivery method. Possible values are <tt>:smtp</tt> (default), <tt>:sendmail</tt>, <tt>:test</tt>,
# and <tt>:file</tt>.
2004-12-16 12:53:19 -05:00
#
2008-05-09 05:38:02 -04:00
# * <tt>perform_deliveries</tt> - Determines whether <tt>deliver_*</tt> methods are actually carried out. By default they are,
2004-12-16 12:53:19 -05:00
# but this can be turned off to help functional testing.
#
2008-05-02 09:45:23 -04:00
# * <tt>deliveries</tt> - Keeps an array of all the emails sent out through the Action Mailer with <tt>delivery_method :test</tt>. Most useful
2004-12-16 12:53:19 -05:00
# for unit and functional testing.
2005-02-23 20:48:29 -05:00
#
2008-08-06 21:08:27 -04:00
# * <tt>default_charset</tt> - The default charset used for the body and to encode the subject. Defaults to UTF-8. You can also
2008-05-16 18:01:32 -04:00
# pick a different charset from inside a method with +charset+.
2009-01-11 05:02:54 -05:00
#
2005-10-26 09:04:20 -04:00
# * <tt>default_content_type</tt> - The default content type used for the main part of the message. Defaults to "text/plain". You
2008-08-06 21:08:27 -04:00
# can also pick a different content type from inside a method with +content_type+.
2009-01-11 05:02:54 -05:00
#
2008-05-16 18:01:32 -04:00
# * <tt>default_mime_version</tt> - The default mime version used for the message. Defaults to <tt>1.0</tt>. You
# can also pick a different value from inside a method with +mime_version+.
2009-01-11 05:02:54 -05:00
#
2005-10-26 09:04:20 -04:00
# * <tt>default_implicit_parts_order</tt> - When a message is built implicitly (i.e. multiple parts are assembled from templates
2005-07-01 16:43:40 -04:00
# which specify the content type in their filenames) this variable controls how the parts are ordered. Defaults to
2008-05-16 18:01:32 -04:00
# <tt>["text/html", "text/enriched", "text/plain"]</tt>. Items that appear first in the array have higher priority in the mail client
2005-07-01 16:43:40 -04:00
# and appear last in the mime encoded message. You can also pick a different order from inside a method with
2008-05-16 18:01:32 -04:00
# +implicit_parts_order+.
2004-11-23 20:04:44 -05:00
class Base
2008-11-23 13:27:25 -05:00
include AdvAttrAccessor , PartContainer , Quoting , Utils
2008-09-09 18:25:09 -04:00
if Object . const_defined? ( :ActionController )
include ActionController :: UrlWriter
2009-06-15 14:59:28 -04:00
include ActionController :: Layout
2008-09-09 18:25:09 -04:00
end
2005-05-29 12:36:22 -04:00
2004-12-16 12:53:19 -05:00
private_class_method :new #:nodoc:
2004-11-23 20:04:44 -05:00
2008-07-31 21:09:10 -04:00
class_inheritable_accessor :view_paths
2009-02-22 11:39:56 -05:00
self . view_paths = [ ]
2004-11-23 20:04:44 -05:00
cattr_accessor :logger
2008-05-14 15:09:49 -04:00
@@smtp_settings = {
2009-01-11 05:02:54 -05:00
:address = > " localhost " ,
:port = > 25 ,
:domain = > 'localhost.localdomain' ,
:user_name = > nil ,
:password = > nil ,
:authentication = > nil ,
:enable_starttls_auto = > true ,
2004-11-23 20:04:44 -05:00
}
2007-01-30 21:09:45 -05:00
cattr_accessor :smtp_settings
2008-05-14 15:09:49 -04:00
2007-01-30 21:09:45 -05:00
@@sendmail_settings = {
:location = > '/usr/sbin/sendmail' ,
:arguments = > '-i -t'
}
cattr_accessor :sendmail_settings
2004-11-23 20:04:44 -05:00
2009-08-07 23:56:54 -04:00
@@file_settings = {
:location = > defined? ( Rails ) ? " #{ Rails . root } /tmp/mails " : " #{ Dir . tmpdir } /mails "
}
cattr_accessor :file_settings
2004-11-23 20:04:44 -05:00
@@raise_delivery_errors = true
cattr_accessor :raise_delivery_errors
2007-11-07 11:05:17 -05:00
superclass_delegating_accessor :delivery_method
self . delivery_method = :smtp
2008-05-14 15:09:49 -04:00
2004-11-23 20:04:44 -05:00
@@perform_deliveries = true
cattr_accessor :perform_deliveries
2008-05-14 15:09:49 -04:00
2004-11-23 20:04:44 -05:00
@@deliveries = [ ]
cattr_accessor :deliveries
2005-02-19 16:51:16 -05:00
@@default_charset = " utf-8 "
cattr_accessor :default_charset
2005-06-03 06:57:06 -04:00
@@default_content_type = " text/plain "
cattr_accessor :default_content_type
2008-05-14 15:09:49 -04:00
2006-09-09 17:56:38 -04:00
@@default_mime_version = " 1.0 "
2005-08-22 19:53:04 -04:00
cattr_accessor :default_mime_version
2005-06-03 06:57:06 -04:00
2005-07-01 16:43:40 -04:00
@@default_implicit_parts_order = [ " text/html " , " text/enriched " , " text/plain " ]
cattr_accessor :default_implicit_parts_order
2008-09-28 13:31:45 -04:00
cattr_reader :protected_instance_variables
@@protected_instance_variables = %w( @body )
2005-10-16 11:00:27 -04:00
# Specify the BCC addresses for the message
adv_attr_accessor :bcc
2008-05-14 15:09:49 -04:00
2005-10-16 11:00:27 -04:00
# Define the body of the message. This is either a Hash (in which case it
# specifies the variables to pass to the template when it is rendered),
# or a string, in which case it specifies the actual text of the message.
adv_attr_accessor :body
2008-05-14 15:09:49 -04:00
2005-10-16 11:00:27 -04:00
# Specify the CC addresses for the message.
adv_attr_accessor :cc
2008-05-14 15:09:49 -04:00
2005-10-16 11:00:27 -04:00
# Specify the charset to use for the message. This defaults to the
# +default_charset+ specified for ActionMailer::Base.
adv_attr_accessor :charset
2008-05-14 15:09:49 -04:00
2005-10-16 11:00:27 -04:00
# Specify the content type for the message. This defaults to <tt>text/plain</tt>
# in most cases, but can be automatically set in some situations.
adv_attr_accessor :content_type
2008-05-14 15:09:49 -04:00
2005-10-16 11:00:27 -04:00
# Specify the from address for the message.
adv_attr_accessor :from
2008-05-14 15:09:49 -04:00
2008-05-23 13:40:36 -04:00
# Specify the address (if different than the "from" address) to direct
# replies to this message.
adv_attr_accessor :reply_to
2005-10-16 11:00:27 -04:00
# Specify additional headers to be added to the message.
adv_attr_accessor :headers
2008-05-14 15:09:49 -04:00
2005-10-16 11:00:27 -04:00
# Specify the order in which parts should be sorted, based on content-type.
# This defaults to the value for the +default_implicit_parts_order+.
adv_attr_accessor :implicit_parts_order
2008-05-14 15:09:49 -04:00
2005-10-16 11:00:27 -04:00
# Defaults to "1.0", but may be explicitly given if needed.
adv_attr_accessor :mime_version
2008-05-14 15:09:49 -04:00
2005-10-16 11:00:27 -04:00
# The recipient addresses for the message, either as a string (for a single
# address) or an array (for multiple addresses).
adv_attr_accessor :recipients
2008-05-14 15:09:49 -04:00
2005-10-16 11:00:27 -04:00
# The date on which the message was sent. If not set (the default), the
# header will be set by the delivery agent.
adv_attr_accessor :sent_on
2008-05-14 15:09:49 -04:00
2005-10-16 11:00:27 -04:00
# Specify the subject of the message.
adv_attr_accessor :subject
2008-05-14 15:09:49 -04:00
2005-10-16 11:00:27 -04:00
# Specify the template name to use for current message. This is the "base"
# template name, without the extension or directory, and may be used to
# have multiple mailer methods share the same template.
adv_attr_accessor :template
2004-12-07 04:10:50 -05:00
2007-11-25 22:36:28 -05:00
# Override the mailer name, which defaults to an inflected version of the
# mailer's class name. If you want to use a template in a non-standard
# location, you can use this to specify that location.
def mailer_name ( value = nil )
if value
self . mailer_name = value
else
self . class . mailer_name
end
end
2008-05-14 15:09:49 -04:00
2007-11-25 22:36:28 -05:00
def mailer_name = ( value )
self . class . mailer_name = value
end
2005-10-16 11:00:27 -04:00
# The mail object instance referenced by this mailer.
attr_reader :mail
2008-09-09 18:25:09 -04:00
attr_reader :template_name , :default_template_name , :action_name
2005-05-29 12:36:22 -04:00
2009-01-22 17:18:10 -05:00
def controller_path
self . class . controller_path
end
def formats
@template . formats
end
2005-07-31 04:26:32 -04:00
class << self
2007-11-25 22:36:28 -05:00
attr_writer :mailer_name
def mailer_name
@mailer_name || = name . underscore
end
# for ActionView compatibility
alias_method :controller_name , :mailer_name
alias_method :controller_path , :mailer_name
2008-08-29 16:08:16 -04:00
def respond_to? ( method_symbol , include_private = false ) #:nodoc:
matches_dynamic_method? ( method_symbol ) || super
end
def method_missing ( method_symbol , * parameters ) #:nodoc:
2008-11-11 10:39:50 -05:00
if match = matches_dynamic_method? ( method_symbol )
case match [ 1 ]
when 'create' then new ( match [ 2 ] , * parameters ) . mail
when 'deliver' then new ( match [ 2 ] , * parameters ) . deliver!
when 'new' then nil
2008-11-11 10:45:53 -05:00
else super
2008-11-11 10:39:50 -05:00
end
else
super
2005-07-31 04:26:32 -04:00
end
end
2005-10-16 11:00:27 -04:00
# Receives a raw email, parses it into an email object, decodes it,
# instantiates a new mailer, and passes the email object to the mailer
2008-05-25 07:29:00 -04:00
# object's +receive+ method. If you want your mailer to be able to
# process incoming messages, you'll need to implement a +receive+
2005-10-16 11:00:27 -04:00
# method that accepts the email object as a parameter:
#
# class MyMailer < ActionMailer::Base
# def receive(mail)
# ...
# end
# end
def receive ( raw_email )
2005-07-31 04:26:32 -04:00
logger . info " Received mail: \n #{ raw_email } " unless logger . nil?
mail = TMail :: Mail . parse ( raw_email )
mail . base64_decode
new . receive ( mail )
end
# Deliver the given mail object directly. This can be used to deliver
# a preconstructed mail object, like:
#
# email = MyMailer.create_some_mail(parameters)
# email.set_some_obscure_header "frobnicate"
# MyMailer.deliver(email)
def deliver ( mail )
new . deliver! ( mail )
end
2007-06-07 19:42:47 -04:00
2008-07-31 21:09:10 -04:00
def template_root
self . view_paths && self . view_paths . first
end
2008-01-21 15:45:04 -05:00
def template_root = ( root )
2008-07-31 21:09:10 -04:00
self . view_paths = ActionView :: Base . process_view_paths ( root )
2008-01-21 15:45:04 -05:00
end
2008-08-29 16:08:16 -04:00
private
def matches_dynamic_method? ( method_name ) #:nodoc:
method_name = method_name . to_s
2008-11-11 10:45:53 -05:00
/ ^(create|deliver)_([_a-z] \ w*) / . match ( method_name ) || / ^(new)$ / . match ( method_name )
2008-08-29 16:08:16 -04:00
end
2005-07-31 04:26:32 -04:00
end
2005-05-29 12:36:22 -04:00
# Instantiate a new mailer object. If +method_name+ is not +nil+, the mailer
# will be initialized according to the named method. If not, the mailer will
# remain uninitialized (useful when you only need to invoke the "receive"
# method, for instance).
2005-07-06 04:22:56 -04:00
def initialize ( method_name = nil , * parameters ) #:nodoc:
2008-05-14 15:09:49 -04:00
create! ( method_name , * parameters ) if method_name
2005-05-29 12:36:22 -04:00
end
# Initialize the mailer via the given +method_name+. The body will be
# rendered and a new TMail::Mail object created.
2005-07-06 04:22:56 -04:00
def create! ( method_name , * parameters ) #:nodoc:
2005-08-22 16:53:27 -04:00
initialize_defaults ( method_name )
2006-11-13 00:03:48 -05:00
__send__ ( method_name , * parameters )
2009-04-22 20:16:28 -04:00
2005-05-29 12:36:22 -04:00
# If an explicit, textual body has not been set, we check assumptions.
unless String === @body
# First, we look to see if there are any likely templates that match,
# which include the content-type in their file name (i.e.,
2007-02-20 19:29:44 -05:00
# "the_template_file.text.html.erb", etc.). Only do this if parts
2005-08-22 19:53:04 -04:00
# have not already been specified manually.
2009-04-22 20:16:28 -04:00
# if @parts.empty?
2009-09-02 18:00:22 -04:00
template_root . find_all ( @template , { } , template_path ) . each do | template |
2008-08-06 21:08:27 -04:00
@parts << Part . new (
2009-04-27 15:34:25 -04:00
:content_type = > template . mime_type ? template . mime_type . to_s : " text/plain " ,
2008-08-06 21:08:27 -04:00
:disposition = > " inline " ,
:charset = > charset ,
2009-01-22 17:18:10 -05:00
:body = > render_template ( template , @body )
2008-08-06 21:08:27 -04:00
)
2005-05-29 12:36:22 -04:00
end
2009-04-22 20:16:28 -04:00
if @parts . size > 1
2009-03-10 12:23:52 -04:00
@content_type = " multipart/alternative " if @content_type !~ / ^multipart /
2005-07-01 16:43:40 -04:00
@parts = sort_parts ( @parts , @implicit_parts_order )
end
2009-04-22 20:16:28 -04:00
# end
2005-05-29 12:36:22 -04:00
# Then, if there were such templates, we check to see if we ought to
# also render a "normal" template (without the content type). If a
# normal template exists (or if there were no implicit parts) we render
# it.
2009-04-22 20:16:28 -04:00
# ====
# TODO: Revisit this
# template_exists = @parts.empty?
2009-08-07 14:00:12 -04:00
# template_exists ||= template_root.find("#{mailer_name}/#{@template}")
2009-04-22 20:16:28 -04:00
# @body = render_message(@template, @body) if template_exists
2005-05-29 12:36:22 -04:00
# Finally, if there are other message parts and a textual body exists,
# we shift it onto the front of the parts and set the body to nil (so
# that create_mail doesn't try to render it in addition to the parts).
2009-04-22 20:16:28 -04:00
# ====
# TODO: Revisit this
# if !@parts.empty? && String === @body
# @parts.unshift Part.new(:charset => charset, :body => @body)
# @body = nil
# end
2004-11-23 20:04:44 -05:00
end
2005-08-22 19:53:04 -04:00
# If this is a multipart e-mail add the mime_version if it is not
# already set.
@mime_version || = " 1.0 " if ! @parts . empty?
2005-05-29 12:36:22 -04:00
# build the mail object itself
@mail = create_mail
end
2004-11-23 20:04:44 -05:00
2005-07-31 04:26:32 -04:00
# Delivers a TMail::Mail object. By default, it delivers the cached mail
2008-05-25 07:29:00 -04:00
# object (from the <tt>create!</tt> method). If no cached mail object exists, and
2005-07-31 04:26:32 -04:00
# no alternate has been given as the parameter, this will fail.
2005-10-16 11:00:27 -04:00
def deliver! ( mail = @mail )
2005-07-31 04:26:32 -04:00
raise " no mail object available for delivery! " unless mail
2009-01-22 17:18:10 -05:00
2008-02-02 00:32:15 -05:00
unless logger . nil?
2008-03-31 01:31:29 -04:00
logger . info " Sent mail to #{ Array ( recipients ) . join ( ', ' ) } "
2008-02-02 00:32:15 -05:00
logger . debug " \n #{ mail . encoded } "
end
2005-04-11 06:29:15 -04:00
2005-05-29 12:36:22 -04:00
begin
2006-11-13 00:03:48 -05:00
__send__ ( " perform_delivery_ #{ delivery_method } " , mail ) if perform_deliveries
2006-08-30 23:16:28 -04:00
rescue Exception = > e # Net::SMTP errors or sendmail pipe errors
2005-05-29 12:36:22 -04:00
raise e if raise_delivery_errors
end
2005-02-19 16:51:16 -05:00
2005-07-31 04:26:32 -04:00
return mail
2005-05-29 12:36:22 -04:00
end
2005-02-19 16:51:16 -05:00
2005-05-29 12:36:22 -04:00
private
2005-08-22 16:53:27 -04:00
# Set up the default values for the various instance variables of this
# mailer. Subclasses may override this method to provide different
# defaults.
def initialize_defaults ( method_name )
2005-10-26 09:02:58 -04:00
@charset || = @@default_charset . dup
@content_type || = @@default_content_type . dup
@implicit_parts_order || = @@default_implicit_parts_order . dup
@template || = method_name
2008-09-09 18:25:09 -04:00
@default_template_name = @action_name = @template
2008-05-14 15:09:49 -04:00
@mailer_name || = self . class . name . underscore
2005-10-26 09:02:58 -04:00
@parts || = [ ]
@headers || = { }
@body || = { }
2005-08-22 19:53:04 -04:00
@mime_version = @@default_mime_version . dup if @@default_mime_version
2009-08-07 21:37:21 -04:00
@sent_on || = Time . now
2005-08-22 16:53:27 -04:00
end
2009-01-22 17:18:10 -05:00
def render_template ( template , body )
2009-04-27 15:34:25 -04:00
if template . respond_to? ( :mime_type )
@current_template_content_type = template . mime_type && template . mime_type . to_sym . to_s
2008-11-20 16:39:34 -05:00
end
2009-01-22 17:18:10 -05:00
@template = initialize_template_class ( body )
2009-04-27 15:34:25 -04:00
layout = _pick_layout ( layout , true ) unless
ActionController :: Base . exempt_from_layout . include? ( template . handler )
2009-08-07 02:18:45 -04:00
@template . _render_template ( template , layout , { } )
2009-01-22 17:18:10 -05:00
ensure
@current_template_content_type = nil
end
def render_message ( method_name , body )
2008-07-19 13:23:08 -04:00
render :file = > method_name , :body = > body
2008-11-20 16:39:34 -05:00
ensure
@current_template_content_type = nil
2005-05-29 12:36:22 -04:00
end
2005-08-22 17:17:01 -04:00
def render ( opts )
2009-01-22 17:18:10 -05:00
layout , file = opts . delete ( :layout ) , opts [ :file ]
2008-09-09 18:25:09 -04:00
begin
2009-01-22 17:18:10 -05:00
@template = initialize_template_class ( opts . delete ( :body ) )
if file
prefix = mailer_name unless file =~ / \/ /
2009-08-07 14:00:12 -04:00
template = view_paths . find ( file , { :formats = > formats } , prefix )
2009-01-22 17:18:10 -05:00
end
layout = _pick_layout ( layout ,
2009-04-27 15:34:25 -04:00
! template || ActionController :: Base . exempt_from_layout . include? ( template . handler ) )
2009-01-22 17:18:10 -05:00
if template
2009-08-07 02:18:45 -04:00
@template . _render_template ( template , layout , opts )
2009-01-22 17:18:10 -05:00
elsif inline = opts [ :inline ]
@template . _render_inline ( inline , layout , opts )
end
2008-09-09 18:25:09 -04:00
end
end
def default_template_format
2008-11-20 16:39:34 -05:00
if @current_template_content_type
Mime :: Type . lookup ( @current_template_content_type ) . to_sym
else
:html
end
2008-09-09 18:25:09 -04:00
end
2008-07-31 21:09:10 -04:00
def template_root
self . class . template_root
end
def template_root = ( root )
self . class . template_root = root
end
2005-05-29 12:36:22 -04:00
def template_path
2009-04-22 20:16:28 -04:00
" #{ mailer_name } "
2004-11-23 20:04:44 -05:00
end
2005-06-16 07:39:29 -04:00
def initialize_template_class ( assigns )
2009-02-22 11:39:56 -05:00
template = ActionView :: Base . new ( self . class . view_paths , assigns , self )
2009-01-22 17:18:10 -05:00
template . formats = [ default_template_format ]
2008-11-20 16:39:34 -05:00
template
2005-06-16 07:39:29 -04:00
end
2005-07-01 16:43:40 -04:00
def sort_parts ( parts , order = [ ] )
order = order . collect { | s | s . downcase }
2009-04-22 20:16:28 -04:00
2005-07-01 16:43:40 -04:00
parts = parts . sort do | a , b |
a_ct = a . content_type . downcase
b_ct = b . content_type . downcase
a_in = order . include? a_ct
b_in = order . include? b_ct
s = case
when a_in && b_in
order . index ( a_ct ) < = > order . index ( b_ct )
when a_in
- 1
when b_in
1
else
a_ct < = > b_ct
end
# reverse the ordering because parts that come last are displayed
# first in mail clients
( s * - 1 )
end
parts
end
2005-05-29 12:36:22 -04:00
def create_mail
m = TMail :: Mail . new
2005-05-02 11:03:13 -04:00
2008-05-23 13:40:36 -04:00
m . subject , = quote_any_if_necessary ( charset , subject )
m . to , m . from = quote_any_address_if_necessary ( charset , recipients , from )
m . bcc = quote_address_if_necessary ( bcc , charset ) unless bcc . nil?
m . cc = quote_address_if_necessary ( cc , charset ) unless cc . nil?
m . reply_to = quote_address_if_necessary ( reply_to , charset ) unless reply_to . nil?
2005-08-22 19:53:04 -04:00
m . mime_version = mime_version unless mime_version . nil?
2008-05-23 13:40:36 -04:00
m . date = sent_on . to_time rescue sent_on if sent_on
2005-05-29 12:36:22 -04:00
headers . each { | k , v | m [ k ] = v }
2004-11-23 20:04:44 -05:00
2006-03-18 18:53:07 -05:00
real_content_type , ctype_attrs = parse_content_type
2009-04-22 20:16:28 -04:00
2005-05-29 12:36:22 -04:00
if @parts . empty?
2006-03-18 18:53:07 -05:00
m . set_content_type ( real_content_type , nil , ctype_attrs )
2008-11-23 13:27:25 -05:00
m . body = normalize_new_lines ( body )
2009-04-22 20:16:28 -04:00
elsif @parts . size == 1 && @parts . first . parts . empty?
m . set_content_type ( real_content_type , nil , ctype_attrs )
m . body = normalize_new_lines ( @parts . first . body )
2005-05-29 12:36:22 -04:00
else
if String === body
part = TMail :: Mail . new
2008-11-23 13:27:25 -05:00
part . body = normalize_new_lines ( body )
2006-03-18 18:53:07 -05:00
part . set_content_type ( real_content_type , nil , ctype_attrs )
2005-05-29 12:36:22 -04:00
part . set_content_disposition " inline "
m . parts << part
end
2005-02-19 16:51:16 -05:00
2005-05-29 12:36:22 -04:00
@parts . each do | p |
part = ( TMail :: Mail === p ? p : p . to_mail ( self ) )
m . parts << part
end
2008-05-14 15:09:49 -04:00
2006-03-18 19:18:50 -05:00
if real_content_type =~ / multipart /
ctype_attrs . delete " charset "
m . set_content_type ( real_content_type , nil , ctype_attrs )
end
2005-05-29 12:36:22 -04:00
end
2005-04-11 06:29:15 -04:00
2005-05-29 12:36:22 -04:00
@mail = m
2005-04-11 06:29:15 -04:00
end
2005-05-29 12:36:22 -04:00
def perform_delivery_smtp ( mail )
2005-06-06 05:32:52 -04:00
destinations = mail . destinations
mail . ready_to_send
2009-06-09 04:22:20 -04:00
sender = ( mail [ 'return-path' ] && mail [ 'return-path' ] . spec ) || mail [ 'from' ]
2005-06-06 05:32:52 -04:00
2008-11-05 22:54:37 -05:00
smtp = Net :: SMTP . new ( smtp_settings [ :address ] , smtp_settings [ :port ] )
2009-01-11 05:02:54 -05:00
smtp . enable_starttls_auto if smtp_settings [ :enable_starttls_auto ] && smtp . respond_to? ( :enable_starttls_auto )
2008-11-05 22:54:37 -05:00
smtp . start ( smtp_settings [ :domain ] , smtp_settings [ :user_name ] , smtp_settings [ :password ] ,
smtp_settings [ :authentication ] ) do | smtp |
2008-03-27 13:54:02 -04:00
smtp . sendmail ( mail . encoded , sender , destinations )
2005-05-29 12:36:22 -04:00
end
2005-04-11 06:29:15 -04:00
end
2005-05-29 12:36:22 -04:00
def perform_delivery_sendmail ( mail )
2008-04-05 07:28:15 -04:00
sendmail_args = sendmail_settings [ :arguments ]
sendmail_args += " -f \" #{ mail [ 'return-path' ] } \" " if mail [ 'return-path' ]
IO . popen ( " #{ sendmail_settings [ :location ] } #{ sendmail_args } " , " w+ " ) do | sm |
2005-07-17 05:27:02 -04:00
sm . print ( mail . encoded . gsub ( / \ r / , '' ) )
2005-05-29 12:36:22 -04:00
sm . flush
2005-04-11 06:29:15 -04:00
end
end
2005-05-29 12:36:22 -04:00
def perform_delivery_test ( mail )
deliveries << mail
end
2009-08-07 23:56:54 -04:00
def perform_delivery_file ( mail )
FileUtils . mkdir_p file_settings [ :location ]
( mail . to + mail . cc + mail . bcc ) . uniq . each do | to |
File . open ( File . join ( file_settings [ :location ] , to ) , 'a' ) { | f | f . write ( mail ) }
end
end
2004-11-23 20:04:44 -05:00
end
2008-11-23 13:27:25 -05:00
Base . class_eval do
include Helpers
helper MailHelper
end
2004-11-23 20:04:44 -05:00
end