2009-10-17 09:31:11 -04:00
require 'active_support/core_ext/class'
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>,
2009-10-28 05:17:59 -04:00
# and <tt>:file</tt>. Or you may provide a custom delivery method object eg. MyOwnDeliveryMethodClass.new
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
2009-11-12 00:08:50 -05:00
include AdvAttrAccessor , Quoting , Utils
2009-10-17 09:31:11 -04:00
include AbstractController :: RenderingController
2009-10-28 08:33:05 -04:00
include AbstractController :: LocalizedCache
2009-10-17 09:31:11 -04:00
include AbstractController :: Layouts
2009-09-20 10:09:17 -04:00
2009-10-18 18:20:14 -04:00
include AbstractController :: Helpers
helper MailHelper
2008-09-09 18:25:09 -04:00
if Object . const_defined? ( :ActionController )
include ActionController :: UrlWriter
end
2005-05-29 12:36:22 -04:00
2009-10-21 19:05:55 -04:00
include ActionMailer :: DeprecatedBody
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
@@raise_delivery_errors = true
cattr_accessor :raise_delivery_errors
@@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
2009-10-21 19:05:55 -04:00
@@protected_instance_variables = [ ]
2009-10-28 07:08:55 -04:00
cattr_reader :protected_instance_variables
2008-09-28 13:31:45 -04:00
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
# 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
2009-10-21 19:05:55 -04:00
# The mail and action_name instances referenced by this mailer.
attr_reader :mail , :action_name
# Where the response body is stored.
attr_internal :response_body
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.
2009-10-21 19:05:55 -04:00
attr_writer :mailer_name
2007-11-25 22:36:28 -05:00
def mailer_name ( value = nil )
if value
2009-10-21 19:05:55 -04:00
@mailer_name = value
2007-11-25 22:36:28 -05:00
else
2009-10-21 19:05:55 -04:00
@mailer_name || self . class . mailer_name
2007-11-25 22:36:28 -05:00
end
end
2008-05-14 15:09:49 -04:00
2009-10-21 19:05:55 -04:00
# Alias controller_path to mailer_name so render :partial in views work.
alias :controller_path :mailer_name
2009-10-18 20:52:36 -04:00
2009-11-19 22:10:57 -05:00
# Add a part to a multipart message, with the given content-type. The
# part itself is yielded to the block so that other properties (charset,
# body, headers, etc.) can be set on it.
2009-11-12 00:08:50 -05:00
def part ( params )
2009-11-19 22:10:57 -05:00
params = { :content_type = > params } if String === params
if custom_headers = params . delete ( :headers )
STDERR . puts ( " Passing custom headers with :headers => {} is deprecated. Please just pass in custom headers directly. " )
params = params . merge ( custom_headers )
end
2009-11-12 00:08:50 -05:00
part = Mail :: Part . new ( params )
2009-11-19 22:10:57 -05:00
yield part if block_given?
@parts << part
end
# Add an attachment to a multipart message. This is simply a part with the
# content-disposition set to "attachment".
def attachment ( params , & block )
params = { :content_type = > params } if String === params
params = { :disposition = > " attachment " ,
:transfer_encoding = > " base64 " } . merge ( params )
part ( params , & block )
2009-11-12 00:08:50 -05:00
end
2005-07-31 04:26:32 -04:00
class << self
2007-11-25 22:36:28 -05:00
attr_writer :mailer_name
2009-10-28 07:08:55 -04:00
delegate :settings , :settings = , :to = > ActionMailer :: DeliveryMethod :: File , :prefix = > :file
delegate :settings , :settings = , :to = > ActionMailer :: DeliveryMethod :: Sendmail , :prefix = > :sendmail
delegate :settings , :settings = , :to = > ActionMailer :: DeliveryMethod :: Smtp , :prefix = > :smtp
2007-11-25 22:36:28 -05:00
def mailer_name
@mailer_name || = name . underscore
end
2009-10-28 07:08:55 -04:00
def delivery_method = ( method_name )
@delivery_method = ActionMailer :: DeliveryMethod . lookup_method ( method_name )
end
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?
2009-11-19 22:10:57 -05:00
mail = Mail . new ( raw_email )
2005-07-31 04:26:32 -04:00
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
2009-10-18 18:20:14 -04:00
# Should template root overwrite the whole view_paths?
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
2009-10-28 07:08:55 -04:00
# Configure delivery method. Check ActionMailer::DeliveryMethod for more
# instructions.
superclass_delegating_reader :delivery_method
self . delivery_method = :smtp
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
2009-11-12 00:08:50 -05:00
# rendered and a new Mail object created.
2005-07-06 04:22:56 -04:00
def create! ( method_name , * parameters ) #:nodoc:
2009-10-18 20:52:36 -04:00
initialize_defaults ( method_name )
__send__ ( method_name , * parameters )
2009-10-21 19:05:55 -04:00
# Create e-mail parts
create_parts
2009-09-20 10:09:17 -04:00
2009-10-28 07:22:07 -04:00
# Set the subject if not set yet
@subject || = I18n . t ( method_name , :scope = > [ :actionmailer , :subjects , mailer_name ] ,
:default = > method_name . humanize )
2009-10-18 20:52:36 -04:00
# build the mail object itself
@mail = create_mail
2005-05-29 12:36:22 -04:00
end
2004-11-23 20:04:44 -05:00
2009-11-12 00:08:50 -05:00
# Delivers a 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-09-20 10:09:17 -04:00
2009-10-18 20:52:36 -04:00
if logger
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
2009-10-15 17:51:51 -04:00
ActiveSupport :: Notifications . instrument ( :deliver_mail , :mail = > @mail ) do
2009-09-20 10:09:17 -04:00
begin
2009-10-28 05:17:59 -04:00
self . delivery_method . perform_delivery ( mail ) if perform_deliveries
2009-09-20 10:09:17 -04:00
rescue Exception = > e # Net::SMTP errors or sendmail pipe errors
raise e if raise_delivery_errors
end
2005-05-29 12:36:22 -04:00
end
2005-02-19 16:51:16 -05:00
2009-09-20 10:09:17 -04:00
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
2009-10-21 19:05:55 -04:00
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 )
2009-10-28 07:08:55 -04:00
@charset || = @@default_charset . dup
@content_type || = @@default_content_type . dup
2005-10-26 09:02:58 -04:00
@implicit_parts_order || = @@default_implicit_parts_order . dup
2009-10-28 07:08:55 -04:00
@mime_version || = @@default_mime_version . dup if @@default_mime_version
@mailer_name || = self . class . mailer_name
@template || = method_name
@action_name = @template
@parts || = [ ]
2005-10-26 09:02:58 -04:00
@headers || = { }
2009-08-07 21:37:21 -04:00
@sent_on || = Time . now
2005-08-22 16:53:27 -04:00
2009-10-21 19:05:55 -04:00
super # Run deprecation hooks
2008-09-09 18:25:09 -04:00
end
2009-10-21 19:05:55 -04:00
def create_parts
super # Run deprecation hooks
2008-07-31 21:09:10 -04:00
2009-10-21 19:05:55 -04:00
if String === response_body
2009-11-12 00:08:50 -05:00
@parts . unshift Mail :: Part . new (
:content_type = > [ " text " , " plain " , { :charset = > charset } ] ,
:content_disposition = > " inline " ,
2009-10-21 19:05:55 -04:00
:body = > response_body
)
else
self . class . template_root . find_all ( @template , { } , mailer_name ) . each do | template |
2009-11-12 00:08:50 -05:00
ct = template . mime_type ? template . mime_type . to_s : " text/plain "
main_type , sub_type = ct . split ( " / " )
@parts << Mail :: Part . new (
:content_type = > [ main_type , sub_type , { :charset = > charset } ] ,
:content_disposition = > " inline " ,
2009-10-21 19:05:55 -04:00
:body = > render_to_body ( :_template = > template )
)
end
2008-07-31 21:09:10 -04:00
2009-10-21 19:05:55 -04:00
if @parts . size > 1
@content_type = " multipart/alternative " if @content_type !~ / ^multipart /
@parts = sort_parts ( @parts , @implicit_parts_order )
end
2009-10-28 07:22:07 -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?
2009-10-21 19:05:55 -04:00
end
2004-11-23 20:04:44 -05:00
end
2005-07-01 16:43:40 -04:00
def sort_parts ( parts , order = [ ] )
order = order . collect { | s | s . downcase }
2009-10-21 19:05:55 -04:00
2005-07-01 16:43:40 -04:00
parts = parts . sort do | a , b |
2009-11-20 05:19:10 -05:00
a_ct = a . content_type . string . downcase
b_ct = b . content_type . string . downcase
2005-07-01 16:43:40 -04:00
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
2009-11-12 00:08:50 -05:00
m = 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-10-15 17:06:15 -04:00
2005-05-29 12:36:22 -04:00
if @parts . empty?
2009-11-19 22:10:57 -05:00
main_type , sub_type = split_content_type ( real_content_type )
2009-11-20 05:19:10 -05:00
m . content_type ( [ main_type , sub_type , 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?
2009-11-19 22:10:57 -05:00
main_type , sub_type = split_content_type ( real_content_type )
2009-11-20 05:19:10 -05:00
m . content_type ( [ main_type , sub_type , ctype_attrs ] )
2009-04-22 20:16:28 -04:00
m . body = normalize_new_lines ( @parts . first . body )
2005-05-29 12:36:22 -04:00
else
@parts . each do | p |
2009-11-19 22:10:57 -05:00
m . parts << p
2005-05-29 12:36:22 -04:00
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 "
2009-11-19 22:10:57 -05:00
main_type , sub_type = split_content_type ( real_content_type )
2009-11-20 05:19:10 -05:00
m . content_type ( [ main_type , sub_type , ctype_attrs ] )
2006-03-18 19:18:50 -05:00
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
2009-11-19 22:10:57 -05:00
def split_content_type ( ct )
ct . to_s . split ( " / " )
end
2005-04-11 06:29:15 -04:00
2009-11-12 00:08:50 -05:00
def parse_content_type ( defaults = nil )
if content_type . blank?
return defaults ?
[ defaults . content_type , { 'charset' = > defaults . charset } ] :
[ nil , { } ]
end
ctype , * attrs = content_type . split ( / ; \ s* / )
attrs = attrs . inject ( { } ) { | h , s | k , v = s . split ( / = / , 2 ) ; h [ k ] = v ; h }
[ ctype , { " charset " = > charset || defaults && defaults . charset } . merge ( attrs ) ]
end
2004-11-23 20:04:44 -05:00
end
end