Как отправлять почту с ruby ​​через smtp с помощью ssl (не с rails, без TLS для gmail) - PullRequest
9 голосов
/ 02 апреля 2009

Все, что я хочу - это отправлять электронные письма из моих скриптов ruby ​​через SMTP с использованием SSL.

Я нахожу примеры только из Rails или для Gmail с TLS.

Я нашел людей, говорящих о поддержке SMTPS с ruby ​​1.8.5, но libdoc не упоминает об этом.

Кто-нибудь с примером отправки почты по SMTP с SSL по порту 465?

ruby -v
ruby 1.8.7 (2008-08-11 patchlevel 72) [i486-linux]

Ответы [ 6 ]

10 голосов
/ 11 января 2013

Я решаю эту проблему с помощью этой конфигурации ниже:

config.action_mailer.perform_deliveries = true
config.action_mailer.raise_delivery_errors = false
config.action_mailer.delivery_method = :smtp
config.action_mailer.smtp_settings = {
    :address              => 'mail.domain.com',
    :port                 => '465',
    :domain               => 'yourdomain.com',
    :user_name            => 'email@yourdomain.com',
    :password             => 'yourpassword',
    :authentication       => :login,
    :ssl                  => true,
    :openssl_verify_mode  => 'none' #Use this because ssl is activated but we have no certificate installed. So clients need to confirm to use the untrusted url.
}

Это очень хорошо работает для меня.

3 голосов
/ 29 июля 2009

Как насчет пони?
драгоценный камень установить пони.
http://github.com/adamwiggins/pony/tree/master
или я не понял вашего вопроса?

Я надеюсь, что это поможет вам.
Благодаря
tknv /

0 голосов
/ 19 февраля 2010

Если вам нужно использовать SSL вместо TLS, вы можете использовать Monkey-patch Net :: SMTP следующим образом:

require "openssl"
require "net/smtp"

Net::SMTP.class_eval do

  def self.start( address, port = nil,
                  helo = 'localhost.localdomain',
                  user = nil, secret = nil, authtype = nil, use_tls = false,
                  use_ssl = false, &block) # :yield: smtp
    new(address, port).start(helo, user, secret, authtype, use_tls, use_ssl, &block)
  end

  def start( helo = 'localhost.localdomain',
             user = nil, secret = nil, authtype = nil, use_tls = false, use_ssl = false ) # :yield: smtp
    start_method = use_tls ? :do_tls_start : use_ssl ? :do_ssl_start : :do_start
    if block_given?
      begin
        send start_method, helo, user, secret, authtype
        return yield(self)
      ensure
        do_finish
      end
    else
      send start_method, helo, user, secret, authtype
      return self
    end
  end

  private

  def do_tls_start(helodomain, user, secret, authtype)
    raise IOError, 'SMTP session already started' if @started

    if VERSION == '1.8.6'
      check_auth_args user, secret, authtype if user or secret
    elsif VERSION == '1.8.7'
      check_auth_args user, secret
    end

    sock = timeout(@open_timeout) { TCPSocket.open(@address, @port) }
    @socket = Net::InternetMessageIO.new(sock)
    @socket.read_timeout = 60 #@read_timeout
    @socket.debug_output = STDERR #@debug_output

    check_response(critical { recv_response() })
    do_helo(helodomain)

    raise 'openssl library not installed' unless defined?(OpenSSL)
    starttls
    ssl = OpenSSL::SSL::SSLSocket.new(sock)
    ssl.sync_close = true
    ssl.connect
    @socket = Net::InternetMessageIO.new(ssl)
    @socket.read_timeout = 60 #@read_timeout
    @socket.debug_output = STDERR #@debug_output
    do_helo(helodomain)

    authenticate user, secret, authtype if user
    @started = true
  ensure
    unless @started
      # authentication failed, cancel connection.
        @socket.close if not @started and @socket and not @socket.closed?
      @socket = nil
    end
  end

  def do_ssl_start(helodomain, user, secret, authtype)
    raise IOError, 'SMTP session already started' if @started

    if VERSION == '1.8.6'
      check_auth_args user, secret, authtype if user or secret
    elsif VERSION == '1.8.7'
      check_auth_args user, secret
    end

    sock = timeout(@open_timeout) { TCPSocket.open(@address, @port) }
    raise 'openssl library not installed' unless defined?(OpenSSL)
    ssl = OpenSSL::SSL::SSLSocket.new(sock)
    ssl.sync_close = true
    ssl.connect
    @socket = Net::InternetMessageIO.new(ssl)
    @socket.read_timeout = 60 #@read_timeout
    @socket.debug_output = STDERR #@debug_output

    check_response(critical { recv_response() })
    do_helo(helodomain)

    do_helo(helodomain)

    authenticate user, secret, authtype if user
    @started = true
  ensure
    unless @started
      # authentication failed, cancel connection.
        @socket.close if not @started and @socket and not @socket.closed?
      @socket = nil
    end
  end

  def do_helo(helodomain)
     begin
      if @esmtp
        ehlo helodomain
      else
        helo helodomain
      end
    rescue Net::ProtocolError
      if @esmtp
        @esmtp = false
        @error_occured = false
        retry
      end
      raise
    end
  end

  def starttls
    getok('STARTTLS')
  end

  def quit
    begin
      getok('QUIT')
    rescue EOFError, OpenSSL::SSL::SSLError
    end
  end
end

См. http://github.com/collectiveidea/action_mailer_optional_tls/blob/master/lib/smtp_tls.rb

0 голосов
/ 24 июля 2009

Вы можете использовать стороннюю программу командной строки с открытым исходным кодом, такую ​​как mailsend (http://www.muquit.com/muquit/software/mailsend/mailsend.html)), чтобы сделать вашу грязную работу за вас. Просто передайте какой-то вывод в него в ожидаемом формате.

0 голосов
/ 24 июля 2009

Единственная интересная вещь, о которой я недавно слышал в программной электронной почте, это Лэмсон: http://lamsonproject.org/

Это Python, а не Ruby, но вы можете вызвать Python из Ruby, если хотите (вот один из способов: http://www.goto.info.waseda.ac.jp/~fukusima/ruby/python-e.html)

0 голосов
/ 02 апреля 2009

Вы, наверное, уже знаете о стандартной библиотеке Net :: SMTP

Что касается части SSL, которая, кажется, не поддерживается "из коробки", я нашел несколько возможных указателей:

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...