مراية لـ
https://github.com/postalserver/postal.git
تم المزامنة 2026-01-20 14:49:53 +00:00
276 أسطر
8.9 KiB
Ruby
276 أسطر
8.9 KiB
Ruby
require 'resolv'
|
|
|
|
module Postal
|
|
class SMTPSender < Sender
|
|
|
|
def initialize(domain, source_ip_address, options = {})
|
|
@domain = domain
|
|
@source_ip_address = source_ip_address
|
|
@options = options
|
|
@smtp_client = nil
|
|
@connection_errors = []
|
|
@hostnames = []
|
|
@log_id = Nifty::Utils::RandomString.generate(:length => 8).upcase
|
|
end
|
|
|
|
def start
|
|
servers.each do |server|
|
|
if server.is_a?(SMTPEndpoint)
|
|
hostname = server.hostname
|
|
port = server.port || 25
|
|
ssl_mode = server.ssl_mode
|
|
elsif server.is_a?(Hash)
|
|
hostname = server[:hostname]
|
|
port = server[:port] || 25
|
|
ssl_mode = server[:ssl_mode] || 'Auto'
|
|
else
|
|
hostname = server
|
|
port = 25
|
|
ssl_mode = 'Auto'
|
|
end
|
|
|
|
@hostnames << hostname
|
|
[:aaaa, :a].each do |ip_type|
|
|
|
|
if @source_ip_address && @source_ip_address.ipv6.blank? && ip_type == :aaaa
|
|
# Don't try to use IPv6 if the IP address we're sending from doesn't support it.
|
|
next
|
|
end
|
|
|
|
begin
|
|
@remote_ip = lookup_ip_address(ip_type, hostname)
|
|
if @remote_ip.nil?
|
|
if ip_type == :a
|
|
# As we can't resolve the last IP, we'll put this
|
|
@connection_errors << "Could not resolve #{hostname}"
|
|
end
|
|
next
|
|
end
|
|
smtp_client = Net::SMTP.new(hostname, port)
|
|
smtp_client.open_timeout = Postal.config.smtp_client.open_timeout
|
|
smtp_client.read_timeout = Postal.config.smtp_client.read_timeout
|
|
if @source_ip_address
|
|
# Set the source IP as appropriate
|
|
smtp_client.source_address = ip_type == :aaaa ? @source_ip_address.ipv6 : @source_ip_address.ipv4
|
|
end
|
|
case ssl_mode
|
|
when 'Auto'
|
|
smtp_client.enable_starttls_auto(self.class.ssl_context_without_verify)
|
|
when 'STARTTLS'
|
|
smtp_client.enable_starttls(self.class.ssl_context_with_verify)
|
|
when 'TLS'
|
|
smtp_client.enable_tls(self.class.ssl_context_with_verify)
|
|
else
|
|
# Nothing
|
|
end
|
|
smtp_client.start(@source_ip_address ? @source_ip_address.hostname : self.class.default_helo_hostname)
|
|
log "Connected to #{@remote_ip}:#{port} (#{hostname})"
|
|
rescue => e
|
|
log "Cannot connect to #{@remote_ip}:#{port} (#{hostname}) (#{e.class}: #{e.message})"
|
|
@connection_errors << e.message unless @connection_errors.include?(e.message)
|
|
smtp_client.disconnect rescue nil
|
|
smtp_client = nil
|
|
end
|
|
|
|
if smtp_client
|
|
@smtp_client = smtp_client
|
|
return true
|
|
end
|
|
end
|
|
end
|
|
|
|
@connection_errors
|
|
end
|
|
|
|
def reconnect
|
|
log "Reconnecting"
|
|
@smtp_client&.finish rescue nil
|
|
start
|
|
end
|
|
|
|
def safe_rset
|
|
# Something went wrong sending the last email. Reset the connection if possible, else disconnect.
|
|
begin
|
|
@smtp_client.rset
|
|
rescue
|
|
# Don't reconnect, this would be rather rude if we don't have any more emails to send.
|
|
@smtp_client.finish rescue nil
|
|
end
|
|
end
|
|
|
|
def send_message(message, force_rcpt_to = nil)
|
|
start_time = Time.now
|
|
result = SendResult.new
|
|
result.log_id = @log_id
|
|
if @smtp_client && !@smtp_client.started?
|
|
# For some reason we had an SMTP connection but it's no longer connected.
|
|
# Make a new one.
|
|
start
|
|
end
|
|
|
|
if @smtp_client
|
|
result.secure = @smtp_client.secure_socket?
|
|
end
|
|
|
|
begin
|
|
if message.bounce == 1
|
|
mail_from = ""
|
|
elsif message.domain.return_path_status == 'OK'
|
|
mail_from = "#{message.server.token}@#{message.domain.return_path_domain}"
|
|
else
|
|
mail_from = "#{message.server.token}@#{Postal.config.dns.return_path}"
|
|
end
|
|
raw_message = "Resent-Sender: #{mail_from}\r\n" + message.raw_message
|
|
tries = 0
|
|
begin
|
|
if @smtp_client.nil?
|
|
log "-> No SMTP server available for #{@domain}"
|
|
log "-> Hostnames: #{@hostnames.inspect}"
|
|
log "-> Errors: #{@connection_errors.inspect}"
|
|
result.type = 'SoftFail'
|
|
result.retry = true
|
|
result.details = "No SMTP servers were available for #{@domain}. Tried #{@hostnames.to_sentence}"
|
|
result.output = @connection_errors.join(', ')
|
|
result.connect_error = true
|
|
return result
|
|
else
|
|
@smtp_client.rset_errors
|
|
rcpt_to = force_rcpt_to || @options[:force_rcpt_to] || message.rcpt_to
|
|
log "Sending message #{message.server.id}::#{message.id} to #{rcpt_to}"
|
|
smtp_result = @smtp_client.send_message(raw_message, mail_from, [rcpt_to])
|
|
end
|
|
rescue Errno::ECONNRESET, Errno::EPIPE, OpenSSL::SSL::SSLError
|
|
if (tries += 1) < 2
|
|
reconnect
|
|
retry
|
|
else
|
|
raise
|
|
end
|
|
end
|
|
result.type = 'Sent'
|
|
result.details = "Message for #{rcpt_to} accepted by #{destination_host_description}"
|
|
if @smtp_client.source_address
|
|
result.details += " (from #{@smtp_client.source_address})"
|
|
end
|
|
result.output = smtp_result.string
|
|
log "Message sent ##{message.id} to #{destination_host_description} for #{rcpt_to}"
|
|
|
|
rescue Net::SMTPServerBusy, Net::SMTPAuthenticationError, Net::SMTPSyntaxError, Net::SMTPUnknownError, Net::ReadTimeout => e
|
|
log "#{e.class}: #{e.message}"
|
|
result.type = 'SoftFail'
|
|
result.retry = true
|
|
result.details = "Temporary SMTP delivery error when sending to #{destination_host_description}"
|
|
result.output = e.message
|
|
if e.to_s =~ /(\d+) seconds/
|
|
result.retry = $1.to_i + 10
|
|
elsif e.to_s =~ /(\d+) minutes/
|
|
result.retry = ($1.to_i * 60) + 10
|
|
end
|
|
|
|
safe_rset
|
|
rescue Net::SMTPFatalError => e
|
|
log "#{e.class}: #{e.message}"
|
|
result.type = 'HardFail'
|
|
result.details = "Permanent SMTP delivery error when sending to #{destination_host_description}"
|
|
result.output = e.message
|
|
safe_rset
|
|
rescue => e
|
|
log "#{e.class}: #{e.message}"
|
|
if defined?(Raven)
|
|
Raven.capture_exception(e, :extra => {:log_id => @log_id, :server_id => message.server.id, :message_id => message.id})
|
|
end
|
|
result.type = 'SoftFail'
|
|
result.retry = true
|
|
result.details = "An error occurred while sending the message to #{destination_host_description}"
|
|
result.output = e.message
|
|
safe_rset
|
|
end
|
|
|
|
result.time = (Time.now - start_time).to_f.round(2)
|
|
return result
|
|
ensure
|
|
end
|
|
|
|
def finish
|
|
log "Finishing up"
|
|
@smtp_client&.finish
|
|
end
|
|
|
|
private
|
|
|
|
def servers
|
|
@options[:servers] || self.class.relay_hosts || @servers ||= begin
|
|
mx_servers = []
|
|
Resolv::DNS.open do |dns|
|
|
dns.timeouts = [10,5]
|
|
mx_servers = dns.getresources(@domain, Resolv::DNS::Resource::IN::MX).map { |m| [m.preference.to_i, m.exchange.to_s] }.sort.map{ |m| m[1] }
|
|
if mx_servers.empty?
|
|
mx_servers = [@domain] # This will be resolved to an A or AAAA record later
|
|
end
|
|
end
|
|
mx_servers
|
|
end
|
|
end
|
|
|
|
def log(text)
|
|
Postal.logger_for(:smtp_sender).info "[#{@log_id}] #{text}"
|
|
end
|
|
|
|
def destination_host_description
|
|
"#{@hostnames.last} (#{@remote_ip})"
|
|
end
|
|
|
|
def lookup_ip_address(type, hostname)
|
|
records = []
|
|
Resolv::DNS.open do |dns|
|
|
dns.timeouts = [10,5]
|
|
case type
|
|
when :a
|
|
records = dns.getresources(hostname, Resolv::DNS::Resource::IN::A)
|
|
when :aaaa
|
|
records = dns.getresources(hostname, Resolv::DNS::Resource::IN::AAAA)
|
|
end
|
|
end
|
|
records.first&.address&.to_s&.downcase
|
|
end
|
|
|
|
def self.ssl_context_with_verify
|
|
@ssl_context_with_verify ||= begin
|
|
c = OpenSSL::SSL::SSLContext.new
|
|
c.verify_mode = OpenSSL::SSL::VERIFY_PEER
|
|
c.cert_store = OpenSSL::X509::Store.new
|
|
c.cert_store.set_default_paths
|
|
c
|
|
end
|
|
end
|
|
|
|
def self.ssl_context_without_verify
|
|
@ssl_context_without_verify ||= begin
|
|
c = OpenSSL::SSL::SSLContext.new
|
|
c.verify_mode = OpenSSL::SSL::VERIFY_NONE
|
|
c
|
|
end
|
|
end
|
|
|
|
def self.default_helo_hostname
|
|
Postal.config.dns.helo_hostname || Postal.config.dns.smtp_server_hostname || "localhost"
|
|
end
|
|
|
|
def self.relay_hosts
|
|
hosts = Postal.config.smtp_relays.map do |relay|
|
|
if relay.hostname.present?
|
|
{
|
|
:hostname => relay.hostname,
|
|
:port => relay.port,
|
|
:ssl_mode => relay.ssl_mode
|
|
}
|
|
else
|
|
nil
|
|
end
|
|
end.compact
|
|
hosts.empty? ? nil : hosts
|
|
end
|
|
|
|
end
|
|
end
|