Rubyメールプロアクティブプッシュトリガを実現

4779 ワード

メールサーバがメールを受信すると、サービスpush通知プログラムが表示されます.何か実現する方法はありますか?
1、クライアントポーリング2、サーバーが自発的にプッシュする.
まずよく知っておきますが、メールの送受信プロトコル:Net::SMTP(メール送信)Net::POP 3(メール受信)Net::IMAP(メール受信)
ネット上ではpop 3でメールを受信する例が多いが、pop 3でメールを受信すると受信ボックス内のすべてのメールしか取得できず、メールがすでに読んでいるかどうかなどのマークが取得できない.imapプロトコルを使用すると、この気まずい思いを避けることができる.imapはメールの詳細(例えば、読んでいるかどうか、返事しているかどうか)を得るだけでなく、ユーザーがメールのマークを変更することもできる.しかし、現在imapプロトコルをサポートするメールサーバは多くありません.私が知っているのは21 cnとgmailだけです.次の例ではエージェント、SSL認証の複数の内容を使用しています.参考にしてください.
imapメールは、必要に応じて請求されます.つまり、Messageのオブジェクトを取得すると、実際には何も情報が入っていません.このオブジェクトでgetメソッドで情報を取得すると、getSubjectのようにMessageオブジェクトがメールサーバに再アクセスしてこのメッセージを得るので、必要な情報をすべて取得する前にディレクトリを閉じることはできません.接続を切断することはできません.
本当にディレクトリを閉じるか接続後にMessageオブジェクトを操作したい場合は、Folderオブジェクトのfetchメソッドを使用して必要な情報を得る必要があります.
一:クライアントポーリング
下にpop 3とimapでポーリングアクセス取得メールの例を表示します.
POP 3ポーリング:
 
  
loop do
require 'net/pop'
pop = Net::POP3.new('EMAILSERVICE')
pop.start('USENAME', 'PASSWORD')           
if pop.mails.empty?
  puts 'No mail.'
else
  pop.each_mail do |m|
    m.pop do |chunk|  
      p chunk
    end
  end
  puts "#{pop.mails.size} mails popped."
end
pop.finish
sleep(10)
end

imapポーリング:
 
  
loop do
require 'net/imap'
imap = Net::IMAP.new('EMAILSERVICE')
imap.login "USERNAME", "PASSWORD"
imap.examine('INBOX')
imap.search(["BEFORE", "29-Oct-2014", "SINCE", "28-Oct-2014"]).each do |message_id|
   envelope = imap.fetch(message_id, "ENVELOPE")[0].attr["ENVELOPE"]
   puts "#{envelope.from[0].name}: \t#{envelope.subject}"
end
sleep(10)
end

二:サーバーの自発的なプッシュ
下側は1種のサーバーの自発的なプッシュ方式を実現します:(IMAP.IDLE)
これはpullとPersistent TCP/IPの間の技術であるlong polling(ロングポーリング).原理は,クライアントがサービスに対する要求のたびにサービス側にholdされ,messageが返されるかtime outが返されると,再び自発的に要求を開始し,messageの到着を待つことである.このモードは心拍数を保つ必要もなく,TCPの占有を継続する必要もなく,ページ側のタイムリーなメッセージのプッシュに適している.
 
  
SERVER = 'EMAILSERVICE'
USERNAME = 'USERNAME'
PW = 'PASSWORD'
require 'net/imap'

# Extend support for idle command. See online.
# http://www.ruby-forum.com/topic/50828
# https://gist.github.com/jem/2783772
# but that was wrong. see /opt/ruby-1.9.1-p243/lib/net/imap.rb.
class Net::IMAP
  def idle
    cmd = "IDLE"
    synchronize do
      @idle_tag = generate_tag
      put_string(@idle_tag + " " + cmd)
      put_string(CRLF)
    end
  end

  def say_done
    cmd = "DONE"
    synchronize do
      put_string(cmd)
      put_string(CRLF)
    end
  end

  def await_done_confirmation
    synchronize do
      get_tagged_response(@idle_tag, nil)
      puts 'just got confirmation'
    end
  end
end

class Remailer
  attr_reader :imap

  public
  def initialize
    @imap = nil
    @mailer = nil
    start_imap
  end

  def tidy
    stop_imap
  end

  def print_pust
       envelope = @imap.fetch(-1, "ENVELOPE")[0].attr["ENVELOPE"]
       puts "From:#{envelope.from[0].name}\t Subject: #{envelope.subject}"
  end

  def bounce_idle
    # Bounces the idle command.
    @imap.say_done
    @imap.await_done_confirmation
    # Do a manual check, just in case things aren't working properly.
    @imap.idle
  end

  private
  def start_imap
    @imap = Net::IMAP.new('pop.i-click.com')
    @imap.login USERNAME, PW
    @imap.select 'INBOX'

    # Add handler.
    @imap.add_response_handler do |resp|
      if resp.kind_of?(Net::IMAP::UntaggedResponse) and resp.name == "EXISTS"
        @imap.say_done
        Thread.new do
          @imap.await_done_confirmation
          print_pust
          @imap.idle
        end
      end
    end
    @imap.idle
  end

  def stop_imap
    @imap.done
  end

end

begin
  Net::IMAP.debug = true
  r = Remailer.new
  loop do
    puts 'bouncing...'
    r.bounce_idle
    sleep 15*60
    # 15
  end
ensure
  r.tidy
end