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