跳到主要內容

JavaMail - 如何刪除Hotmail上的信件?

應觀眾要求開始著手於砍facebook垃圾信件的程式開發,當然順便砍掉之前做問卷所寄來每天上百封的垃圾信。在我還不懂事的時候,有透過JavaMail[1]寫GMail與北科信箱的自動寄信程式(當時純粹是為了惡搞),而在我Google大法後,有文章提及到hotmail是利用WebDav的方式去存取信件,另外包含一篇2010年3月30日的開發教學[2]。我嘗試照它的程式碼去執行過一次,跳出了帳號認證錯誤的例外情形。但我相信,在7-11吹著冷氣並喝著City,順便看著可愛的便利商店美眉,我的腦袋並不至於會昏迷到連自己的帳號密碼都打錯。於是我又開始努力的尋找解答。終於讓我看到一篇文章提到M$在2009年3月12日就提供hotmail POP3的存取方式,這讓我非常的開心。另外在別篇文章也找尋到hotmail也提供SMTP的存取方式,以下為設定資訊:

POP 伺服器:pop3.live.com (port:995)SSL安全連線:是
SMTP 伺服器:smtp.live.com (port:25)TLS/SSL安全連線:是; 是否需要驗證:是

首先我要聲明,我有一堆看不完得書、學不完的技術(與打不完的電動)要殺我的時間,這篇文章中僅提供POP3的撰寫方式與我嘗試過後的經驗分享。其實在[3]中,已經有許多的範例與教學。要使用JavaMail API首先要到它的網站中下載mail.jar和activation.jar。在我們開始寫程式以前,我們先介紹JavaMail幾個重要的Concept,分別為、Message、Folder、Store、Transport、Session。 Message:表達的是Mail中的信件。其包含了寄件者(From)、收件者(To)、主旨(Subject)、內容(Text)等屬性。 Folder:表達的是資料夾的概念,包含了Sub-Folder與Message。在POP3協定中,僅能得到名稱為INBOX的資料夾。 Store:表達信件系統中資料庫的概念。除了連結與關閉信件資料庫外,並提供存取Folder與Message之功能。如果要寫一個屬於自己定義的郵件Protocol,實做這個就對了。 Transport:中文翻譯叫運輸工具,我英文比較破,如果我在設計可能會把它叫做MailSender…,我們會告訴Transport目的地在哪,接著它會攜帶著訊息帶往目的地,也許是POP3或SMTP的Transport。簡而言之,Transport會幫你把信件送到目的地啦,至於空難車禍什麼的,就…。 Session:用來表達用戶對信件系統的存取資訊,會根據用戶的需求,去產生對應的Store與Transport。也會透過Session去做信件系統的認證。(感覺這個Session的責任有點重,一般Session應只負責儲存資料。但在[3]中的第二章有提到,它希望提供一個便於使用的API,所以別怪它了。)

聽我說完以上基本的Concept後,我們可以了解到:要寄信就找Transport,要去操作郵件系統就找Store。有沒有覺得了解Concept比看程式碼簡單多了?接下的範例是要敘述我們該如何刪除郵件系統上的一個訊息。 

Step1: 

產生SSL POP3的Store並連線 hotmail提供的POP3協定要進行SSL的安全連線,因此要透過POP3SSLStore去存取。我們要給它session與連線位置,其中連線位置包含了POP3伺服器位置、Port、使用者帳號(xxxx@hotmail.com)與密碼。

	Properties pop3_properties = new Properties();
	URLName url = new URLName("pop3", "pop3.live.com", 995, "",
			username, password);
	Session session = Session.getInstance(pop3_properties, null);
	Store pop3_store = new POP3SSLStore(session, url); 
	pop3_store.connect();

我們可以透過isConnected去確認是否連線成功。

	if( pop3_store.isConnected() ){
		System.out.println("Hotmail pop3Service connection: success");
	}

Step2: 

刪除INBOX資料夾中的”某個”訊息 首先我們透過pop3_store去取得INBOX,pop3協定也僅能讀取INBOX。接著使用讀寫的權限開啟這個資料夾,如果使用READ_ONLY,那我們的修改就沒用了。

	// Get the inbox in the pop3 store.
	Folder inbox = pop3_store.getFolder("INBOX");
	inbox.open(Folder.READ_WRITE);

在開啟資料夾後,我們去取得所有的訊息內容。日期越早的訊息編號會在越前面,我們從後面開始讀取訊息內容。在這裡我們透過getForm的API去取得寄件者資訊,若包含Facebook的字串就進行訊息的刪除。message刪除方式是透過設定delete flag,但設定完後並不會直接刪除,我們繼續看Step3。

	// Get messages from inbox.
	Message []messages = inbox.getMessages();
	int messageCount = inbox.getMessageCount();
	for( int i = messageCount - 1 ; i >= 0 ; i-- ){
		Message message = messages[i];
		//String subject = message.getSubject();
		String address = message.getFrom()[0].toString();
		if( address.contains("Facebook") ){
			// delete the message.
			message.setFlag(Flag.DELETED, true);
		}
	}

Step3:

關閉資料夾與Store連線 僅僅兩行資源釋放的程式碼,是最容易被大家忽略的。在Step2中設定完delete flag後,並不會直接刪除,必須在inbox關閉並丟入為true的expunge參數後,訊息才會真正的消失於人間之中。最後就是Store的連線關閉了。

	inbox.close(true);
	pop3_store.close();

以上三個步驟就可以做到POP3刪除郵件的功能了。但在Step2中,要去確認所有的訊息不是很花時間嗎?其實我已經嘗試過郵件是否被讀取的flag(seen),僅去確認未讀取的信件增加效能,但可惜POP3僅提供delete的flag。因此如果要讓效能更好與撰寫更多的功能,應該要往SMTP的方向走。

留言

這個網誌中的熱門文章

Show NIC selection when setting the network command with the device option

 Problem  在answer file中設定網卡名稱後,安裝時會停在以下畫面: 所使用的command參數如下: network --onboot = yes --bootproto =dhcp --ipv6 =auto --device =eth1 Diagnostic Result 這樣的參數,以前試驗過是可以安裝完成的。因此在發生這個問題後,我檢查了它的debug console: 從console得知,eth1可能是沒有連接網路線或者是網路太慢而導致的問題。後來和Ivy再三確認,有問題的是有接網路線的網卡,且問題是發生在activate階段: Solution 我想既然有retry應該就有次數或者timeout限制,因此發現在Anaconda的說明文件中( link ),有提到dhcptimeout這個boot參數。看了一些人的使用範例,應該是可以直接串在isolinux.cfg中,如下: default linux ksdevice = link ip =dhcp ks =cdrom: / ks.cfg dhcptimeout = 90 然而我在RHEL/CentOS 6.7與6.8試驗後都無效。 因此我就拿了顯示的錯誤字串,問問Google大師,想找一下Anaconda source code來看一下。最後找到別人根據Anaconda code修改的版本: link ,關鍵在於setupIfaceStruct函式中的setupIfaceStruct與readNetConfig: setupIfaceStruct: 會在dhcp時設定dhcptimeout。 readNetConfig: 在writeEnabledNetInfo將timeout寫入dhclient config中;在wait_for_iface_activation內會根據timeout做retry。 再來從log與code可以得知,它讀取的檔案是answer file而不是boot command line。因此我接下來的測試,就是在answer file的network command上加入dhcptimeout: network --onboot = yes --bootproto =dhcp --ipv6 =auto --device =eth1 --dhcptimeo

解決RobotFramework從3.1.2升級到3.2.2之後,Choose File突然會整個Hand住的問題

考慮到自動測試環境的維護,我們很久以前就使用java去執行robot framework。前陣子開始處理從3.1.2升級到3.2.2的事情,主要先把明確的runtime語法錯誤與deprecate item處理好,這部分內容可以參考: link 。 直到最近才發現,透過SeleniumLibrary執行Choose File去上傳檔案的動作,會導致測試案例timeout。本篇文章主要分享心路歷程與解決方法,我也送了一條issue給robot framework: link 。 我的環境如下: RobotFramework: 3.2.2 Selenium: 3.141.0 SeleniumLibrary: 3.3.1 Remote Selenium Version: selenium-server-standalone-3.141.59 首先並非所有Choose File的動作都會hang住,有些測試案例是可以執行的,但是上傳一個作業系統ISO檔案一定會發生問題。後來我透過wireshark去比對新舊版本的上傳動作,因為我使用 Remote Selenium ,所以Selenium會先把檔案透過REST API發送到Remote Selenium Server上。從下圖我們可以發現,在3.2.2的最後一個TCP封包,比3.1.2大概少了500個bytes。 於是就開始了我trace code之路。包含SeleniumLibrary產生要送給Remote Selenium Server的request內容,還有HTTP Content-Length的計算,我都確認過沒有問題。 最後發現問題是出在socket API的使用上,就是下圖的這支code: 最後發現可能因為開始使用nio的方式送資料,但沒處理到尚未送完的資料內容,而導致發生問題。加一個loop去做計算就可以解決了。 最後我有把解法提供給robot framework官方,在他們出新的版本之前,我是將改完的_socket.py放在我們自己的Lib底下,好讓我們測試可以正常進行。(shutil.py應該也是為了解某個bug而產生的樣子..)

PostgreSQL - Unattended installation on windows

Introduction 要將別人軟體包裝到自己軟體中,不可或缺的東西就是Unattended installation。以Unattended installation來說,我們可以選擇透過Installer的silent mode安裝,也可以透過把目標軟體做成portable的版本。本篇文章分享這兩種方法,教導大家如何將PostgreSQL透過Unattended installation方式安裝到目標系統成為service。 Note. 本篇以PostgreSQL 10.7為例。 Install with installer Tips 安裝程式或反安裝程式的參數,除了可以直接上官網搜尋Installation User Guide以外,也可以直接使用help參數查詢: postgresql- 10.7 - 2 -windows-x64.exe --help Windows安裝程式主要有EnterpriseDB與BigSQL兩種。BigSQL版本安裝元件是透過網路下載且支援參數不如EnterpriseDB版本多,以我們需求來說,我們傾向於使用EnterpriseDB版本。接下來分享給大家安裝與反安裝方法。 Installation @ echo off set INSTALL_DIR =C:\postgres10 set INSTALLER =postgresql- 10.7 - 2 -windows-x64.exe   rem options for installation set SSMDB_SERVICE =postgresql- 10 set MODE =--unattendedmodeui none --mode unattended   set DB_PASSWD =--superpassword postgres set DB_PORT =--serverport 5432   set SERVICE_NAME =--servicename % SSMDB_SERVICE %   set PREFIX =--prefix "%INSTALL_DIR%" set DATA_DIR =--datadir "%INSTALL_DIR%\data"   set OPTIONS =