python的smtplib提供了一種很方便的途徑發(fā)送電子郵件。它對smtp協(xié)議進(jìn)行了簡單的封裝。
	smtp協(xié)議的基本命令包括:
	    HELO 向服務(wù)器標(biāo)識(shí)用戶身份
	    MAIL 初始化郵件傳輸 mail from:
	    RCPT 標(biāo)識(shí)單個(gè)的郵件接收人;常在MAIL命令后面,可有多個(gè)rcpt to:
	    DATA 在單個(gè)或多個(gè)RCPT命令后,表示所有的郵件接收人已標(biāo)識(shí),并初始化數(shù)據(jù)傳輸,以.結(jié)束
	    VRFY 用于驗(yàn)證指定的用戶/郵箱是否存在;由于安全方面的原因,服務(wù)器常禁止此命令
	    EXPN 驗(yàn)證給定的郵箱列表是否存在,擴(kuò)充郵箱列表,也常被禁用
	    HELP 查詢服務(wù)器支持什么命令
	    NOOP 無操作,服務(wù)器應(yīng)響應(yīng)OK
	    QUIT 結(jié)束會(huì)話
	    RSET 重置會(huì)話,當(dāng)前傳輸被取消
	    MAIL FROM 指定發(fā)送者地址
	    RCPT TO 指明的接收者地址
一般smtp會(huì)話有兩種方式,一種是郵件直接投遞,就是說,比如你要發(fā)郵件給zzz@163.com,那就直接連接163.com的郵件服務(wù)器,把信投給zzz@163.com; 另一種是驗(yàn)證過后的發(fā)信,它的過程是,比如你要發(fā)郵件給zzz@163.com,你不是直接投到163.com,而是通過自己在sina.com的另一個(gè)郵箱來發(fā)。這樣就要先連接sina.com的smtp服務(wù)器,然后認(rèn)證,之后在把要發(fā)到163.com的信件投到sina.com上,sina.com會(huì)幫你把信投遞到163.com。
第一種方式的命令流程基本是這樣:
	      1. helo
	      2. mail from
	      3. rcpt to
	      4. data
	      5. quit
但是第一種發(fā)送方式一般有限制的,就是rcpt to指定的這個(gè)郵件接收者必須在這個(gè)服務(wù)器上存在,否則是不會(huì)接收的。 先看看代碼:
 #-*- encoding: gb2312 -*-import os, sys, stringimport smtplib# 郵件服務(wù)器地址mailserver = "smtp.163.com"# smtp會(huì)話過程中的mail from地址from_addr = "asfgysg@zxsdf.com"# smtp會(huì)話過程中的rcpt to地址to_addr = "zhaoweikid@163.com"# 信件內(nèi)容msg = "test mail"svr = smtplib.SMTP(mailserver)# 設(shè)置為調(diào)試模式,就是在會(huì)話過程中會(huì)有輸出信息svr.set_debuglevel(1)# helo命令,docmd方法包括了獲取對方服務(wù)器返回信息svr.docmd("HELO server")# mail from, 發(fā)送郵件發(fā)送者svr.docmd("MAIL FROM: <%s>" % from_addr)# rcpt to, 郵件接收者svr.docmd("RCPT TO: <%s>" % to_addr)# data命令,開始發(fā)送數(shù)據(jù)svr.docmd("DATA")# 發(fā)送正文數(shù)據(jù)svr.send(msg)# 比如以 . 作為正文發(fā)送結(jié)束的標(biāo)記,用send發(fā)送的,所以要用getreply獲取返回信息svr.send(" . ")svr.getreply()# 發(fā)送結(jié)束,退出svr.quit()注意的是,163.com是有反垃圾郵件功能的,想上面的這種投遞郵件的方法不一定能通過反垃圾郵件系統(tǒng)的檢測的。所以一般不推薦個(gè)人這樣發(fā)送。
	    第二種有點(diǎn)不一樣:
	      1.ehlo
	      2.auth login
	      3.mail from
	      4.rcpt to 
	      5.data
	      6.quit
相對于第一種來說,多了一個(gè)認(rèn)證過程,就是auth login這個(gè)過程。
#-*- encoding: gb2312 -*-import os, sys, stringimport smtplibimport base64# 郵件服務(wù)器地址mailserver = "smtp.163.com"# 郵件用戶名username = "xxxxxx@163.com"# 密碼password = "xxxxxxx"# smtp會(huì)話過程中的mail from地址from_addr = "xxxxxx@163.com"# smtp會(huì)話過程中的rcpt to地址to_addr = "yyyyyy@163.com"# 信件內(nèi)容msg = "my test mail"svr = smtplib.SMTP(mailserver)# 設(shè)置為調(diào)試模式,就是在會(huì)話過程中會(huì)有輸出信息svr.set_debuglevel(1)# ehlo命令,docmd方法包括了獲取對方服務(wù)器返回信息svr.docmd("EHLO server")# auth login 命令svr.docmd("AUTH LOGIN")# 發(fā)送用戶名,是base64編碼過的,用send發(fā)送的,所以要用getreply獲取返回信息svr.send(base64.encodestring(username))svr.getreply()# 發(fā)送密碼svr.send(base64.encodestring(password))svr.getreply()# mail from, 發(fā)送郵件發(fā)送者svr.docmd("MAIL FROM: <%s>" % from_addr)# rcpt to, 郵件接收者svr.docmd("RCPT TO: <%s>" % to_addr)# data命令,開始發(fā)送數(shù)據(jù)svr.docmd("DATA")# 發(fā)送正文數(shù)據(jù)svr.send(msg)# 比如以 . 作為正文發(fā)送結(jié)束的標(biāo)記svr.send(" . ")svr.getreply()# 發(fā)送結(jié)束,退出svr.quit()上面說的是最普通的情況,但是不能忽略的是現(xiàn)在好多企業(yè)郵件是支持安全郵件的,就是通過SSL發(fā)送的郵件,這個(gè)怎么發(fā)呢?SMTP對SSL安全郵件的支持有兩種方案,一種老的是專門開啟一個(gè)465端口來接收ssl郵件,另一種更新的做法是在標(biāo)準(zhǔn)的25端口的smtp上增加一個(gè)starttls的命令來支持。
看看第一種怎么辦:
#-*- encoding: gb2312 -*-import os, sys, string, socketimport smtplibclass SMTP_SSL (smtplib.SMTP):  def __init__(self, host='', port=465, local_hostname=None, key=None, cert=None):    self.cert = cert    self.key = key    smtplib.SMTP.__init__(self, host, port, local_hostname)      def connect(self, host='localhost', port=465):    if not port and (host.find(':') == host.rfind(':')):      i = host.rfind(':')      if i >= 0:        host, port = host[:i], host[i+1:]        try: port = int(port)        except ValueError:          raise socket.error, "nonnumeric port"    if not port: port = 654    if self.debuglevel > 0: print>>stderr, 'connect:', (host, port)    msg = "getaddrinfo returns an empty list"    self.sock = None    for res in socket.getaddrinfo(host, port, 0, socket.SOCK_STREAM):      af, socktype, proto, canonname, sa = res      try:        self.sock = socket.socket(af, socktype, proto)        if self.debuglevel > 0: print>>stderr, 'connect:', (host, port)        self.sock.connect(sa)        # 新增加的創(chuàng)建ssl連接        sslobj = socket.ssl(self.sock, self.key, self.cert)      except socket.error, msg:        if self.debuglevel > 0:           print>>stderr, 'connect fail:', (host, port)        if self.sock:          self.sock.close()        self.sock = None        continue      break    if not self.sock:      raise socket.error, msg    # 設(shè)置ssl    self.sock = smtplib.SSLFakeSocket(self.sock, sslobj)    self.file = smtplib.SSLFakeFile(sslobj);    (code, msg) = self.getreply()    if self.debuglevel > 0: print>>stderr, "connect:", msg    return (code, msg)    if __name__ == '__main__':  smtp = SMTP_SSL('192.168.2.10')  smtp.set_debuglevel(1)  smtp.sendmail("zzz@xxx.com", "zhaowei@zhaowei.com", "xxxxxxxxxxxxxxxxx")  smtp.quit()這里我是從原來的smtplib.SMTP派生出了新的SMTP_SSL類,它專門來處理ssl連接。我這里測試的192.168.2.10是我自己的測試服務(wù)器.
第二種是新增加了starttls的命令,這個(gè)很簡單,smtplib里就有這個(gè)方法,叫smtplib.starttls()。當(dāng)然,不是所有的郵件系統(tǒng)都支持安全郵件的,這個(gè)需要從ehlo的返回值里來確認(rèn),如果里面有starttls,才表示支持。相對于發(fā)送普通郵件的第二種方法來說,只需要新增加一行代碼就可以了:
#-*- encoding: gb2312 -*-import os, sys, stringimport smtplibimport base64# 郵件服務(wù)器地址mailserver = "smtp.163.com"# 郵件用戶名username = "xxxxxx@163.com"# 密碼password = "xxxxxxx"# smtp會(huì)話過程中的mail from地址from_addr = "xxxxxx@163.com"# smtp會(huì)話過程中的rcpt to地址to_addr = "yyyyyy@163.com"# 信件內(nèi)容msg = "my test mail"svr = smtplib.SMTP(mailserver)# 設(shè)置為調(diào)試模式,就是在會(huì)話過程中會(huì)有輸出信息svr.set_debuglevel(1)# ehlo命令,docmd方法包括了獲取對方服務(wù)器返回信息,如果支持安全郵件,返回值里會(huì)有starttls提示svr.docmd("EHLO server")svr.starttls() # <------ 這行就是新加的支持安全郵件的代碼!# auth login 命令svr.docmd("AUTH LOGIN")# 發(fā)送用戶名,是base64編碼過的,用send發(fā)送的,所以要用getreply獲取返回信息svr.send(base64.encodestring(username))svr.getreply()# 發(fā)送密碼svr.send(base64.encodestring(password))svr.getreply()# mail from, 發(fā)送郵件發(fā)送者svr.docmd("MAIL FROM: <%s>" % from_addr)# rcpt to, 郵件接收者svr.docmd("RCPT TO: <%s>" % to_addr)# data命令,開始發(fā)送數(shù)據(jù)svr.docmd("DATA")# 發(fā)送正文數(shù)據(jù)svr.send(msg)# 比如以 . 作為正文發(fā)送結(jié)束的標(biāo)記svr.send(" . ")svr.getreply()# 發(fā)送結(jié)束,退出svr.quit()注意:以上的代碼為了方便我都沒有判斷返回值,嚴(yán)格說來,是應(yīng)該判斷一下返回的代碼的,在smtp協(xié)議中,只有返回代碼是2xx或者3xx才能繼續(xù)下一步,返回4xx或5xx的,都是出錯(cuò)了。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持VEVB武林網(wǎng)。
新聞熱點(diǎn)
疑難解答
圖片精選