require 'optparse'
require 'rubygems'
require 'action_mailer'
class Object # :nodoc:
unless respond_to? :path2class then
def self.path2class(path)
path.split(/::/).inject self do |k,n| k.const_get n end
end
end
end
##
# Hack in RSET
class Net::SMTP # :nodoc:
unless instance_methods.include? 'reset' then
##
# Resets the SMTP connection.
def reset
getok 'RSET'
end
end
end
##
# ActionMailer::ARSendmail delivers email from the email table to the
# configured SMTP server.
#
# See ar_sendmail -h for the full list of supported options.
#
# The interesting options are:
# * --daemon
# * --mailq
# * --create-migration
# * --create-model
# * --table-name
class ActionMailer::ARSendmail
##
# Email delivery attempts per run
attr_accessor :batch_size
##
# Seconds to delay between runs
attr_accessor :delay
##
# Be verbose
attr_accessor :verbose
##
# ActiveRecord class that holds emails
attr_reader :email_class
##
# True if only one delivery attempt will be made per call to run
attr_reader :once
##
# Creates a new migration using +table_name+ and prints it on stdout.
def self.create_migration(table_name)
puts <<-EOF
class Add#{table_name.classify} < ActiveRecord::Migration
def self.up
create_table :#{table_name.tableize} do |t|
t.column :from, :string
t.column :to, :string
t.column :last_send_attempt, :integer, :default => 0
t.column :mail, :text
end
end
def self.down
drop_table :email
end
end
EOF
end
##
# Creates a new model using +table_name+ and prints it on stdout.
def self.create_model(table_name)
puts <<-EOF
class #{table_name.classify} < ActiveRecord::Base
end
EOF
end
##
# Prints a list of unsent emails and the last delivery attempt, if any.
#
# If ActiveRecord::Timestamp is not being used the arrival time will not be
# known. See http://api.rubyonrails.org/classes/ActiveRecord/Timestamp.html
# to learn how to enable ActiveRecord::Timestamp.
def self.mailq
emails = Email.find :all
if emails.empty? then
puts "Mail queue is empty"
return
end
total_size = 0
puts "-Queue ID- --Size-- ----Arrival Time---- -Sender/Recipient-------"
emails.each do |email|
size = email.mail.length
total_size += size
create_timestamp = email.created_at rescue
email.created_on rescue
Time.at(email.created_date) rescue # for Robot Co-op
nil
created = if create_timestamp.nil? then
' Unknown'
else
create_timestamp.strftime '%a %b %d %H:%M:%S'
end
puts "%10d %8d %s %s" % [email.id, size, created, email.from]
if email.last_send_attempt > 0 then
puts "Last send attempt: #{Time.at email.last_send_attempt}"
end
puts " #{email.to}"
puts
end
puts "-- #{total_size/1024} Kbytes in #{emails.length} Requests."
end
##
# Processes command line options in +args+
def self.process_args(args)
name = File.basename $0
options = {}
options[:Daemon] = false
options[:Delay] = 60
options[:Once] = false
options[:TableName] = 'Email'
options[:Chdir] = '.'
options[:RailsEnv] = ENV['RAILS_ENV']
opts = OptionParser.new do |opts|
opts.banner = "Usage: #{name} [options]"
opts.separator ''
opts.separator "#{name} scans the email table for new messages and sends them to the"
opts.separator "website's configured SMTP host."
opts.separator ''
opts.separator "#{name} must be run from a Rails application's root."
opts.separator ''
opts.separator 'Sendmail options:'
opts.on("-b", "--batch-size BATCH_SIZE",
"Maximum number of emails to send per delay",
"Default: Deliver all available emails", Integer) do |batch_size|
options[:BatchSize] = batch_size
end
opts.on( "--delay DELAY",
"Delay between checks for new mail",
"in the database",
"Default: #{options[:Delay]}", Integer) do |delay|
options[:Delay] = delay
end
opts.on("-o", "--once",
"Only check for new mail and deliver once",
"Default: #{options[:Once]}") do |once|
options[:Once] = once
end
opts.on("-d", "--daemonize",
"Run as a daemon process",
"Default: #{options[:Daemon]}") do |daemon|
options[:Daemon] = true
end
opts.on( "--mailq",
"Display a list of emails waiting to be sent") do |mailq|
options[:MailQ] = true
end
opts.separator ''
opts.separator 'Setup Options:'
opts.on( "--create-migration",
"Prints a migration to add an Email table",
"to stdout") do |create|
options[:Migrate] = true
end
opts.on( "--create-model",
"Prints a model for an Email ActiveRecord",
"object to stdout") do |create|
options[:Model] = true
end
opts.separator ''
opts.separator 'Generic Options:'
opts.on("-c", "--chdir PATH",
"Use PATH for the application path",
"Default: #{options[:Chdir]}") do |path|
usage opts, "#{path} is not a directory" unless File.directory? path
usage opts, "#{path} is not readable" unless File.readable? path
options[:Chdir] = path
end
opts.on("-e", "--environment RAILS_ENV",
"Set the RAILS_ENV constant",
"Default: #{options[:RailsEnv]}") do |env|
options[:RailsEnv] = env
end
opts.on("-t", "--table-name TABLE_NAME",
"Name of table holding emails",
"Used for both sendmail and",
"migration creation",
"Default: #{options[:TableName]}") do |name|
options[:TableName] = name
end
opts.on("-v", "--[no-]verbose",
"Be verbose",
"Default: #{options[:Verbose]}") do |verbose|
options[:Verbose] = verbose
end
opts.on("-h", "--help",
"You're looking at it") do
usage opts
end
opts.separator ''
end
opts.parse! args
return options if options.include? :Migrate or options.include? :Model
ENV['RAILS_ENV'] = options[:RailsEnv]
Dir.chdir options[:Chdir] do
begin
require 'config/environment'
rescue LoadError
usage opts, <<-EOF
#{name} must be run from a Rails application's root to deliver email.
#{Dir.pwd} does not appear to be a Rails application root.
EOF
end
end
return options
end
##
# Processes +args+ and runs as appropriate
def self.run(args = ARGV)
options = process_args args
if options.include? :Migrate then
create_migration options[:TableName]
exit
elsif options.include? :Model then
create_model options[:TableName]
exit
elsif options.include? :MailQ then
mailq
exit
end
if options[:Daemon] then
require 'webrick/server'
WEBrick::Daemon.start
end
new(options).run
rescue Interrupt
exit
end
##
# Prints a usage message to $stderr using +opts+ and exits
def self.usage(opts, message = nil)
if message then
$stderr.puts message
$stderr.puts
end
$stderr.puts opts
exit 1
end
##
# Creates a new ARSendmail.
#
# Valid options are:
# :BatchSize:: Maximum number of emails to send per delay
# :Delay:: Delay between deliver attempts
# :TableName:: Table name that stores the emails
# :Once:: Only attempt to deliver emails once when run is called
# :Verbose:: Be verbose.
def initialize(options = {})
options[:Delay] ||= 60
options[:TableName] ||= 'Email'
@batch_size = options[:BatchSize]
@delay = options[:Delay]
@email_class = Object.path2class options[:TableName]
@once = options[:Once]
@verbose = options[:Verbose]
end
##
# Delivers +emails+ to ActionMailer's SMTP server and destroys them.
def deliver(emails)
Net::SMTP.start server_settings[:address], server_settings[:port],
server_settings[:domain], server_settings[:user],
server_settings[:password],
server_settings[:authentication] do |smtp|
until emails.empty? do
email = emails.shift
begin
res = smtp.send_message email.mail, email.from, email.to
email.destroy
log "sent email %011d from %s to %s: %p" %
[email.id, email.from, email.to, res]
rescue Net::SMTPFatalError => e
log "5xx error sending email %d, removing from queue: %p(%s):\n\t%s" %
[email.id, e.message, e.class, e.backtrace.join("\n\t")]
email.destroy
smtp.reset
rescue Net::SMTPServerBusy, Net::SMTPUnknownError,
Net::SMTPSyntaxError, TimeoutError => e
email.last_send_attempt = Time.now.to_i
email.save rescue nil
log "error sending email %d: %p(%s):\n\t%s" %
[email.id, e.message, e.class, e.backtrace.join("\n\t")]
smtp.reset
end
end
end
end
##
# Prepares ar_sendmail for exiting
def do_exit
log "caught signal, shutting down"
raise Interrupt
end
##
# Returns emails in email_class that haven't had a delivery attempt in the
# last 300 seconds.
def find_emails
options = { :conditions => ['last_send_attempt < ?', Time.now.to_i - 300] }
options[:limit] = batch_size unless batch_size.nil?
mail = @email_class.find :all, options
log "found #{mail.length} emails to send"
mail
end
##
# Installs signal handlers to gracefully exit.
def install_signal_handlers
trap 'TERM' do do_exit end
trap 'INT' do do_exit end
end
##
# Logs +message+ if verbose
def log(message)
$stderr.puts message if @verbose
ActionMailer::Base.logger.info "ar_sendmail: #{message}"
end
##
# Scans for emails and delivers them every delay seconds. Only returns if
# once is true.
def run
install_signal_handlers
loop do
now = Time.now
begin
deliver find_emails
rescue ActiveRecord::Transactions::TransactionError
end
break if @once
sleep @delay if now + @delay > Time.now
end
end
##
# Proxy to ActionMailer::Base#server_settings. See
# http://api.rubyonrails.org/classes/ActionMailer/Base.html
# for instructions on how to configure ActionMailer's SMTP server.
def server_settings
ActionMailer::Base.server_settings
end
end