使用Ruby收發郵件

 

1。通過SMTP發送Email

每星期Holden Glova, Pat Eyler, 和 Phil Thomson都會向Ruby Garden 網站(http://www.rubygarden.org)提交一個Ruby Weekly News (RWN)文章。一個Ruby腳本通過email接收這篇文章,將它從原來的xml格式轉換爲HTML和純文本格式,然後將HTML格式的發表到網站,然後將純文本格式的文章發到郵件列表。如果這中間出現什麼問題(比如xml文檔結構不對等),這個腳本將向發送者發送一封包含錯誤信息的email。

這個腳本用Net::SMTP (Simple Mail Transfer Protocol) 庫發送email。清單 1 是這個腳本中用來發送email的方法,這個方法接收3個參數:email地址,標題,和信件內容。因爲這個程序要在各種控制環境下使用,所以一些類似發件人,轉發email的主機等屬性都定義爲全局常量,而不是參數。

清單 1: 通過SMTP發送郵件

 1   FROM_ADDRESS = "[email protected]"
 2   SMTP_HOST = "localhost"
 3
 4   def reply(to, subject, msg)
 5     mail = "To: #{to}/r/n" +
 6            "From: #{FROM_ADDRESS}/r/n" +
 7            "Subject: #{subject}/r/n" +
 8            "/r/n" +
 9            msg
10
11     Net::SMTP.start(SMTP_HOST) do |smtp|
12       smtp.send_mail(mail, FROM_ADDRESS,
13         [ to, '[email protected]' ])
14     end
15   end

一個email消息由兩部分組成:信封(envelope)和內容(content)。信封告訴SMTP代理(sendmail或者postfix)如何投遞消息。內容包括能被人們閱讀的消息本身和一些標題(header)(比如消息subject),而一些內容中的header可能和envelope中的重複(比如"To"地址),這些重複的header用來顯示時候使用,而envelope中的則是用來投遞使用。(這也是爲什麼你會收到"To"地址不是你的垃圾郵件)

你可以看到reply方法已經分離了envelop和content,第5行到第9行生成了content,它包括3個header:To,From,Subject,然後在一個空行後面加上了消息內容。注意郵件主體內容之前的header之後必須有一個回車換行,即"/r"和"/n"的組合。

方法Net::SMTP.start來和MTA( mail transfer agent)建立連接,這個方法的一個參數是運行MTA的機器名稱,並且使用了默認端口(25),這個方法返回一個對象用來和MTA交互,並且把這個對象作爲參數傳給了block(11行到13行)。使用block,能保證block結束後連接能夠被關閉。

在我們的例子裏,這個交互過程很簡單,第12行的程序只是發送了剛纔傳劍的郵件內容。

send_mail方法的第二個參數是使用的From地址,這是一個全局變量。第三個參數是一個包含接收者地址的數組。我們這裏把這條消息發送到了兩個地方,一個參數指定的to,還有一個歸檔所有消息的本地郵箱。

 


2。用POP接收和閱讀郵件

使用Ruby從POP服務器接收郵件是非常簡單的事情。假設我們要對人們對各種語言的喜愛程度,參加調查的人可以通過發送標題爲i like xxxxx的郵件給特定的地址,xxxxx是發信者喜歡的語言的名字。清單3的Ruby腳本用來從POP服務器接收結果並進行計算,把每種語言的喜愛者的數目存在一個普通文件,每種語言一個文件。

 

清單3: Fetching email with POP

 1   require 'net/pop'
 2
 3   Net::POP3.delete_all('pop3.server.address', 110,
 4                        'YourAccount', 'YourPassword' ) do |email|
 5     hdr = email.header
 6     if hdr =~ /Subject:/s+I like/s+(/w+)/
 7       language = $1.upcase
 8     else
 9       language = "INVALID"
10     end
11
12     count = (File.read(language) rescue "0")
13     File.open(language, "w") {|f| f.puts old_count.succ}
14   end

POP服務器存放着用戶的消息,當你讀完一條消息的時候,你可以選擇刪除這封信,或者還把它放在服務器上存放,在我們的例子裏,我們讀完之後將刪除它。幸運的是,Ruby提供了一個很方便的迭代器delete_all,它將一條條的取出郵件,處理完之後刪掉這些信件。delete_all需要的參數有POP服務器的地址和端口(標準端口爲110),還有用戶名和密碼。

這個方法開始之後將用指定的參數連接服務器,一封一封的取得該用戶的郵件,然後每次將這封信(作爲一個Net::POPMail對象)傳給給定的block來處理,當block處理完這封信之後,將刪除這封信。

在這個塊內,第5行將這封信的所有header都取出來放到一個字符串當中。然後在第6行中用一個正則表達式在標題(Subject)中查找類似的包括的I like xxxx 行,找出xxxx代表的語言,然後在12和13行中對找到的投票者選擇的語言的計數進行更新。

 

第12行有一個很有意思的結構。每次我們得到一個給某種語言的投票,我們都用一個文件來存儲這個語言得到的投票數。我們可以讀取這個值,增加它,然後寫回到文件。但是第一次有人給某個語言投票時,這個文件還不存在,當我們讀取這個文件時會得到一個異常,很幸運Ruby提供了一個異常機制(關鍵字rescue),但出現異常的時候可能是因爲文件不存在,所以捕獲這個異常,返回一個默認值0,即這個語言的得票數爲0。

另一個小技巧是第13行的old_count.succ,我們用這個來增加一個字符串。在Ruby中這是允許的,如果一個字符串包含的是一個整數,那麼這個succ方法返回的是這個包含這個整數的下一個值的字符串。即aString.succ=aString.to_i.succ.to_s 。

譯者注:old_count.succ可能應該是count.succ

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章