Gmailを使ってActionMailerでメールを送信しましょう。

restful_authentication導入の続きです。

libsの下にsmtp_tls.rb というファイルを新規作成します。
smtp_tls.rbの内容は下記の通り。

require "openssl"
require "net/smtp"

Net::SMTP.class_eval do
  private
  def do_start(helodomain, user, secret, authtype)
    raise IOError, 'SMTP session already started' if @started
    # check_auth_args user, secret, authtype if user or secret
    check_auth_args user, secret if user or secret

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

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

    if starttls
      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
      do_helo(helodomain)
    end

    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') rescue return false
    return true
  end

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


ここで、

    # check_auth_args user, secret, authtype if user or secret
    check_auth_args user, secret if user or secret

となっているのは、処理自体は正常なのに、Gmailから送信された形跡がない場合、このように修正すると無事動作しました。
こちらを参照。
http://d.hatena.ne.jp/komoeda/20090531/1243763664



次に下記の内容で、config/initializers/mail.rbファイルを新規作成します。

require "smtp_tls"

ActionMailer::Base.delivery_method = :smtp
ActionMailer::Base.smtp_settings = {
  :address => "smtp.gmail.com",
  :port => 587,
  :domain => "***",
  :authentication => :login,
  :user_name => "***@***",
  :password => "***"
}


これでアクティベーションのメールが送信されるようになりました。
ところが、メールに記載されてる認証コードと、DBのコードが違っていました。
これでは認証できません。


解決方法はこちらに書かれていました。
「Activation code does not match?」の所です。
http://wiki.github.com/technoweenie/restful-authentication


方法は二つあるようです。

http://wiki.github.com/technoweenie/restful-authentication/switching-from-stateful-to-aasm


まずは、acts_as_state_machineをAASMに入れ替える方法。

app/models/user/rbを書き換える方法です。

include Authorization::StatefulRoles

include Authorization::AasmRoles


に変えます。


でも、私はこちらの方法で対処しました。
app/models/user_observer.rbを次のように書き換えます。

class UserObserver < ActiveRecord::Observer
def after_create(user)
user.reload # ここを追加
UserMailer.deliver_signup_notification(user)
end
def after_save(user)
user.reload # ここを追加
UserMailer.deliver_activation(user) if user.recently_activated?
end
end