Linux自動下傳送HTML格式並帶附件的郵件

liangxichen發表於2011-07-18

引:

進入BEIDOU組的第一個專案就是實現一個統計報表自動傳送郵件的應用,利用Shell指令碼來做,期間回顧了awk,sed等文字過濾工具,crontab計劃任務,還學會了在Linux下傳送HTML郵件附帶MS WORD/EXCEL/PPT格式附件的方法,在春節前圓滿的完成了任務也算是可以踏踏實實過年了,活雖然小但畢竟可以算作一個小Milestone )

遇到問題:

統計報表實現基本思想,按處理流程順序
1) 利用scp下載遠端線上機器的Log日誌檔案
2) 利用awk,sed,sort等Linux下命令過濾並且分析日誌,生成基本的模板(template)文字。
3) 根據該模板(template)文字統計資訊生成HTML格式的郵件正文。
4) 根據該模板(template)文字統計資訊生成CVS、TXT、XLS格式的統計資訊作為郵件附件。
5) 利用sendmail或者mutt命令傳送郵件。
6) 利用crontab計劃任務定時傳送日報、週報、月報。
問題就出現在步驟5)。開始我嘗試利用來實現傳送HTML格式正文郵件並且附帶附件:
mutt -e "my_hdr content-type:text/html" -s "郵件標題" -a 附件.xls receiver@123.com < mail.html
用outlook做客戶端接收郵件,發現附件丟失了,變成了正文裡的亂碼,如果不加-e "my_hdr content-type:text/html"引數,附件成功又不能顯示HTML格式郵件,期間google了各種mutt相關問題官方FAQ都無從知曉為什麼,現在看來既有可能是mutt版本沒有升級到1.5的一個bug,但自己不是admin也沒法裝最新版本的mutt,最終選擇放棄使用mutt。

解決方法:

編寫以下兩個函式,其中sendmail()函式配好引數,就可以直接呼叫了。這樣就可以傳送帶多媒體附件的HTML格式正文的郵件了。在此感謝@lingbing同學的幫助。
#傳送多媒體附件的HTML格式正文的函式 (多媒體附件指非txt或者cvs格式的檔案,例如excel的xls)
#$1: mail_from
#$2: mail_to
#$3: subject
#$4: content mimetype, such as "text/plain"
#$5: content
#$6: attach mimetype, such as "text/csv"
#$7: attach display name
#$8: attach file path
function SendMailMultiMediaAttach(){
local MSG_FILE="/tmp/mail.tmp"

echo "From: $1" > $MSG_FILE
echo "To: $2" >> $MSG_FILE
echo "Subject: $3" >> $MSG_FILE
echo "Mime-Version: 1.0" >> $MSG_FILE
echo 'Content-Type: multipart/mixed; boundary="GvXjxJ+pjyke8COw"' >> $MSG_FILE
echo "Content-Disposition: inline" >> $MSG_FILE
echo "" >> $MSG_FILE
echo "--GvXjxJ+pjyke8COw" >> $MSG_FILE
echo "Content-Type: $4" >> $MSG_FILE
echo "Content-Disposition: inline" >> $MSG_FILE
echo "" >> $MSG_FILE
echo "$5" >> $MSG_FILE
echo "" >> $MSG_FILE
echo "" >> $MSG_FILE
echo "--GvXjxJ+pjyke8COw" >> $MSG_FILE
echo "Content-Type: $6" >> $MSG_FILE
echo "Content-Transfer-Encoding: base64" >> $MSG_FILE
echo "Content-Disposition: attachement; filename=$7" >> $MSG_FILE
echo "" >> $MSG_FILE
echo "" >> $MSG_FILE
${BIN_PATH}/base64 -e $8 >> $MSG_FILE

cat $MSG_FILE | /usr/lib/sendmail -t
}

##! @TODO: 傳送郵件
##! @AUTHOR: zhangxu
##! @VERSION: 1.0
##! @IN:
##! @OUT:
function sendMail()
{
echo "Sending $Subject mail from $From to $To"

from="from@123.com"
to="receiver@123.com"
subject="${Subject}"
content_type="text/html"
body=`cat $MAIL_HTML`
attach_type="application/vnd.ms-excel"
attach_name="${file_title}.xls"
attach_path="${TEMP_DIR}/${file_title}.xls"
SendMailMultiMediaAttach "$from" "$to" "$subject" "$content_type" "$body" "$attach_type" "$attach_name" "$attach_path"

echo "Send mail done."
}
要注意以下幾點:
1) 多媒體檔案對應的格式可以從下面的連結參考,用於替換引數$6的mimetype。
2) 如何判斷自己的附件是不是純文字的呢?Windows下如果可以用notepad記事本開啟,或者Linux下可以用cat顯示正常的都是可以用 text/plain的MIME TYPE的,其他的一律需要用1)中提到的對應的編碼格式,還要保證又base64編碼再傳送出去,郵件客戶端或者接受者可以base64解碼還原。這就是之所以Content-Transfer-Encoding: base64用並且要用base64 -e 編碼的原因,base64命令可以Google下並下載。
附:曾經幫助啟發過我的連結


[@more@]

搞web開發的同志可能碰到過需要在頁面裡嵌入傳送郵件的功能,如果是普通的純文字的郵件還好,沒問題,用asp有好多元件,用cgi也有好工具,比如perl。在perl中使用unix平臺下的sendmail可以實現這個目的。Perl中傳送純文字郵件的典型例子如下:

#!/usr/lib/perl
use strict;

my($r_mail) = ;
my($s_mail) = ;
my($subject) = 'subject';

open(MAIL,'|/usr/lib/sendmail -t');
select(MAIL);

print<To: $r_mail
From: $s_mail
Subject: $subject

郵件內容

END_TAG

有幾點要注意,在傳送郵件裡To, From和接受者郵件地址變數$r_mail以及傳送者郵件$s_mail之間
要留一個空格,避免不必要的報錯問題(我遇到過,不知道你有沒有碰到)。還有那個結束標記
END_TAG如果是檔案的最後一行,最好在後面加一兩個空行,我曾經碰到沒後面的空行perl找不到
END_TAG的情況。還有,不要忘了subject之後的那個空行是必須的,它分開了郵件頭和郵件內容。

好,進入正題,如果我們需要傳送html格式的郵件呢?如果寫成這樣

#!/usr/lib/perl
use strict;

my($r_mail) = ;
my($s_mail) = ;
my($subject) = 'subject';

open(MAIL,'|/usr/lib/sendmail -t');
select(MAIL);

print<To: $r_mail
From: $s_mail
Subject: $subject

郵件內容

END_TAG

試試看,在263裡原始碼全顯示出來了,在hotmail中好點,如果郵件是個完整的html郵件,基本上
能夠完整的呈現html頁面。其實這裡頭有個MIME型別的問題。詳細的MIME資料大家自己上網找吧,
否則扯得太遠,我這點水不夠倒的。如果這個html郵件沒有連線任何圖片以及此類的外部內容,那好辦,
在郵件頭部分加一句Content-type:text/html就可以了。如果使用了中文需要指定一下內碼表,直接
在後面在添上charset="gb2312",中間用分號格開。完整程式碼如下:

#!/usr/lib/perl
use strict;

my($r_mail) = ;
my($s_mail) = ;
my($subject) = 'subject';

open(MAIL,'|/usr/lib/sendmail -t');
select(MAIL);

print<To: $r_mail
From: $s_mail
Subject: $subject
Content-type:text/html;charset="gb2312"

郵件內容

END_TAG

這樣一般使用的接收郵件的工具都能看到html格式的郵件了。如果問題再複雜一點,
這個html頁面裡有圖,還有flash,那怎麼辦?會有辦法:把這些圖片放在網上,
頁面的圖片都寫全路徑連結,這樣就根本不需要在郵件裡真的帶上這些累贅了,並且
還減小了郵件的大小,一舉兩得!我嚴重贊同。但是總有碰到不能這樣乾的時候,所以
繼續。html頁面的對這些圖的連結並不能夠讓使用者收到的郵件裡有這些圖和flash檔案。
看到的html頁面是開了天窗的頁面。看看MIME型別,有個multipart/mixed的型別能夠完
成我們的最終目的,讓使用者收到的郵件是圖文並舉的完整頁面。首先需要按一定的編碼
方法對圖片或者flash等檔案編碼,電子郵件中最常用的是base64編碼,還有
quoted-printable編碼。找個工具,把圖片等需要鏈入hmtl郵件的檔案使用base64編碼,
對html郵件則使用quoted-printable編碼。然後,在郵件頭寫
Content-Type: multipart/mixed;boundary="----=_NextPart_000_0008_01C2BCB0.9CF9AE70" name="thanks.gif"
這裡的multipart/mixed表示本郵件是混合型別的郵件。接下來的boundary是指定分隔
郵件內容裡各不同各部分的標記是什麼。這裡就是----=_NextPart_000_0008_01C2BCB0.9CF9AE70了。
這個值必須要怎樣我不是很清楚,我的理解是在本郵件中能夠不與任何編碼後的某段
內容相同就可以了。後面那個name可以不要。說起來比較羅索,還是先看程式碼吧。
下面就是個完整的傳送hmtl郵件的例子。

#!/usr/lib/perl
use strict;

my($r_mail) = ;
my($s_mail) = ;
my($subject) = 'subject';

open(MAIL,'|/usr/lib/sendmail -t');
select(MAIL);

print<To: $r_mail
From: $s_mail
Subject: $subject
Content-Type: multipart/mixed;boundary="----=_NextPart_000_0008_01C2BCB0.9CF9AE70"

This is a multi-part message in MIME format.

------=_NextPart_000_0008_01C2BCB0.9CF9AE70
Content-Type: text/html;charset="gb2312"
Content-Transfer-Encoding: quoted-printable

=D0=BB=D0=BB=C4=FA=B5=C4=B2=CE=D3=EB=A3=A1


Linux自動下傳送HTML格式並帶附件的郵件height=3D400=20
src=3D"file:///C:/DEV/perl/images/popup_thanks.gif" =
width=3D400=20
border=3D0>

------=_NextPart_000_0008_01C2BCB0.9CF9AE70
Content-Type: image/gif
Content-Transfer-Encoding: base64
Content-Disposition: attachment;filename="thanks.gif"
Content-Location: file:///C:/DEV/perl/images/popup_thanks.gif

R0lGODlhkAGQAfcAAPCyTdvr0cvbCJ8tTt3nAEeRGarPkRSmULrVqrRtLtbeB7CRNdTZ0LGtNaXJ
EaLR7ejz23e1GFir2NNMayaR0VWaLfjMRtnINsfWuYi6aGWsNevKN5PHdLZKMyZLC6smNs3nu8mI
NWetGBIqCOqXNdutkWyRRzZvEqvIK9fJSZGMNrSxSlalGO/36rS5revt6MySScswSperNZWtT9C3

------=_NextPart_000_0008_01C2BCB0.9CF9AE70--

END_TAG

有點長了,慢慢解釋吧。這封要傳送的hmtl郵件裡只有一張圖片popup_thanks.gif. 裡面有一句話"This is a multi-part message in MIME format.", 放在第一個boundary出現
之前,這是個描述資訊,不用管它。然後就是第一個boundary:
------=_NextPart_000_0008_01C2BCB0.9CF9AE70,它告訴使用者的郵件程式這裡有一部份的內容。注意這裡是--boundary,就是說在boundary前面加了兩個-,大家還請注意看最後一個boundary,它的前後都加了兩個-,表示整個郵件結束。
Content-type:text/html;charset="gb2312" 說明本部分內容的文件型別是html格式的,
Content-Transfer-Encoding: quoted-printable 說明本部分內容使用 quoted-printable 方法
編碼的,當然,下面的內容要確實是 quoted-printable 編碼的,否則使用者就看不到正確的內容了。
郵件內容沒什麼好說的,然後是下一個 boundary,這裡的東西就是我們要的那個popup_thanks.gif了。
看MIME型別是:Content-Type: image/gif 圖片一般就用base64編碼,所以這裡是
Content-Transfer-Encoding: base64 再看下面是一行
Content-Disposition: attachment;filename="thanks.gif"
這裡的attahment表示此圖片作為附件,它還可以是inline,那樣的話這個圖片就會直接在收件人的
郵件程式的郵件顯示區域裡顯示了。filename指定了在附件區域顯示什麼樣的檔名,這裡就把
popup_thanks.gif改成了thanks.gif.下面還有一句
Content-Location: file:///C:/DEV/perl/images/popup_thanks.gif
指定檔案的原始路徑。好像沒用啊?其實很重要,注意html檔案裡連線這個圖片的標籤裡的src是怎麼樣寫
的?這兩個之間要是對不上,那末郵件顯示的時候,附件裡有圖,但郵件還是開了天窗了。好了,基本
就是這樣。不,還有個問題,做程式的時候,怎樣才能得到需要的編碼後的檔案啊?perl裡怎麼樣做
我不知道,CPAN裡也許有這樣的package吧,那位對編碼熟悉,也可以自己寫,不過我做得時候取巧了。
大家用過IE5的另存為.mht檔案嗎?對了,就是它!把需要傳送的html郵件用IE5在本地開啟,再另存為
mht檔案,所有的編碼都得到了,而且圖片的連結關係也都是現成的了,其他的按需要調整一下,帖到你的
程式裡就萬事大吉。更進一步,如果需要做到像263那樣,從頁面上接收包括正文,接受者以及各種可能的附件
等資訊再傳送呢?有點複雜了,也不是這裡要討論的,那位高手做過這些東西,可以把經驗貢獻出來,讓
我們一起學習,這篇就是拋磚之作了。


書面版權所有,書刊轉載請與本人聯絡
參考了系列好文《用PHP傳送MIME郵件》,裡面有較為詳細的MIME介紹,強烈建議閱讀
致謝此文作者:Kartic Krishnamurthy 和譯者:limodou


本篇文章來源於 黑基網-中國最大的網路安全站點 原文連結:

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/59792/viewspace-1052646/,如需轉載,請註明出處,否則將追究法律責任。

相關文章