質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.35%

  • Ruby

    10209questions

    Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

Ruby net/smtp TLSを使えない

解決済

回答 2

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 71

kazuyakazuya

score 89

RubyのNet/SMTPでメールを送りたいです。
smtp4devでダミーのSMTPサーバー(localhost)を使うので実際にメールが外に流れることはありません。
やりたいことは
SMTP AUTH認証 と TLS の2つが整っているSMTPサーバーにアクセスしてメールを送るプログラムを作成
するということです。

一応、SMTP AUTH認証だけが実装されているパータンのものは成功しました。

require 'net/smtp'
a = Net::SMTP.new("localhost",50002)
a.start('localhost'){|smtp|
  smtp.authenticate("sample_user_name","sample_password")
  smtp.send_message(<<-EndOfMail, 'from@example.com', 'to@example.net')
 From: Your Name <from@example.com>
 To: Dest Address <to@example.net>
 Subject: test mail
 Date: Sat, 23 Jun 2019 18:20:50 +0900
 Message-Id: <unique.message.id.string@yourhost.example.com>

 This is a test mail.
   EndOfMail
}


smtp4devの仕様なのか
startの引数に指定するSMTP AUTHの情報はなんでも通過できます。(なにも指定しなければ無論エラーになる。)
とりあえずこれでメールを送ることはできました。

つぎはTLS機能もsmtp4devに設定して(TLSじゃないと受け付けない・・・はず)

プログラムコードには

starttlsメソッド


を追加
ver2.4リファレンス

require 'net/smtp'
a = Net::SMTP.new("localhost",50002)
a.start('localhost'){|smtp|
  smtp.starttls
  smtp.authenticate("sample_user_name","sample_password")
  smtp.send_message(<<-EndOfMail, 'from@example.com', 'to@example.net')
 From: Your Name <from@example.com>
 To: Dest Address <to@example.net>
 Subject: test mail
 Date: Sat, 23 Jun 2019 18:20:50 +0900
 Message-Id: <unique.message.id.string@yourhost.example.com>

 This is a test mail.
   EndOfMail
}


実行結果

:\Users\matsu\Desktop\ruby_lesson>ruby sample.rb
<internal:prelude>:78:in `__read_nonblock': An established connection was aborted by the software in your host machine. (Errno::ECONNABORTED)
        from <internal:prelude>:78:in `read_nonblock'
        from C:/Ruby24-x64/lib/ruby/2.4.0/net/protocol.rb:172:in `rbuf_fill'
        from C:/Ruby24-x64/lib/ruby/2.4.0/net/protocol.rb:154:in `readuntil'
        from C:/Ruby24-x64/lib/ruby/2.4.0/net/protocol.rb:164:in `readline'
        from C:/Ruby24-x64/lib/ruby/2.4.0/net/smtp.rb:950:in `recv_response'
        from C:/Ruby24-x64/lib/ruby/2.4.0/net/smtp.rb:944:in `get_response'
        from C:/Ruby24-x64/lib/ruby/2.4.0/net/smtp.rb:734:in `block in auth_plain'
        from C:/Ruby24-x64/lib/ruby/2.4.0/net/smtp.rb:960:in `critical'
        from C:/Ruby24-x64/lib/ruby/2.4.0/net/smtp.rb:733:in `auth_plain'
        from C:/Ruby24-x64/lib/ruby/2.4.0/net/smtp.rb:728:in `authenticate'
        from sample.rb:5:in `block in <main>'
        from C:/Ruby24-x64/lib/ruby/2.4.0/net/smtp.rb:519:in `start'
        from sample.rb:3:in `<main>'
確立された接続は、ホストマシンのソフトウェアによって中止されました。


みたいなことが書かれてありますが
コード自体には問題はないのでしょうか?

ファイアウォールが原因なのかと思い一回すべて無効にしてから
実行してみたのですが結果は全く変わらなかったので
ファイアウォールは関係ないと思います。

何が原因なのかがわかりません。
リファレンスを見た限り違うと思いますが
Rubyのバージョン自体の問題なのでしょうか?

分からないので参考になるリンクまたは説明をお願いします。


解決方法

smtp4devではできませんでしたがsendgridを経由したらいけました。
smtp4dev自体に問題があったようです。

require 'net/smtp'
a = Net::SMTP.new("smtp.sendgrid.net", 587)
a.enable_starttls
a.start('matsukazu0409@gmail.com'){|smtp|
  smtp.authenticate("sendgridユーザー","パスワード")
  smtp.send_message(<<-EndOfMail, 'matsukazu0409@gmail.com', 'matsu-kazu4@ezweb.ne.jp')
  From: Your Name <from@example.com>
  To: Dest Address <to@example.net>
  Subject: test mail
  Date: Sat, 23 Jun 2019 18:20:50 +0900
  Message-Id: <unique.message.id.string@yourhost.example.com>

  This is a test mail.
    EndOfMail


 }
  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • asm

    2019/08/09 01:24

    gmailだとポート開放が必要で若干めんどくさいですね
    Thunderbird
    Outlook
    Becky!
    等での動作を確認してはどうでしょうか?

    キャンセル

  • kazuyakazuya

    2019/08/09 09:04

    ありがとうございます
    一回それで試してみます。

    キャンセル

  • kazuyakazuya

    2019/08/09 16:36

    Thunderbirdで試行していますが
    タイムアウトになってしまいます。
    送信するときに必ず宛先を指定しなければいけないのですが
    この宛先がなにであれエラーには関係ない・・・のでしょうか?

    キャンセル

回答 2

checkベストアンサー

+1

Rubyコードの問題点

サーバーの設定次第ですがa.startの時点でEHLOが平文で送られてしまうので
starttlsを送信する前にエラーになる可能性があります。

start前にenable_starttls_autoもしくはenable_starttlsを使うとよいでしょう。

RFCを確認しましたが

  smtp.send_message(<<-EndOfMail, 'from@example.com', 'to@example.net')
 From: Your Name <from@example.com>
 To: Dest Address <to@example.net>
 Subject: test mail
 Date: Sat, 23 Jun 2019 18:20:50 +0900
 Message-Id: <unique.message.id.string@yourhost.example.com>

 This is a test mail.
   EndOfMail

では、フィールド名の前の空白が問題を引き起こす可能性があります。<<~を使い先頭の空白を消したほうがよいでしょう。

仕様の問題点

使いたいのがSTARTTLSなのかSMTP over SSL/TLSなのかが不明です。
net/smtpはSTARTTLSとSMTP over SSL/TLSの併用は不可能な仕様になっています。
それぞれを確認するとよいでしょう。

検証方法の問題点

smtp4devは私が検証した限り.cerファイルを渡さないとTLS機能が使えない
かつ、.cerファイルの形式が不明な上、動作報告が見当たりません。
いくつか試してみましたが、正常動作させる方法は不明です。

代替として、OrigamiSMTPを利用しstarttlsの検証ができました。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2019/08/14 00:51

    ありがとうございます。

    キャンセル

+1

a.start('localhost'){|smtp| の段階ですでに接続が始まっているので、その後で starttls をするのは手遅れのようです。
上記のマニュアルにも、 starttls のところには、

通常は Net::SMTP#start で STARTTLS が送られるため 利用する必要はないはずです。

と書いてありますよね。

TLSになってなさそうであれば、下記のように enable_starttls を追加するのでどうでしょうか。
ついでにアカウント・パスワードもstartメソッドにまとめています。

require 'net/smtp'
a = Net::SMTP.new("localhost", 50002)
a.enable_starttls
a.start('localhost', "sample_user_name", "sample_password"){|smtp|
  smtp.send_message(<<-EndOfMail, 'from@example.com', 'to@example.net')
# (以下省略)

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2019/08/14 00:52

    ありがとうございます。

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 90.35%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

同じタグがついた質問を見る

  • Ruby

    10209questions

    Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。