python的smtplib模塊主要是用來發送郵件的,使用起來比較方便。
使用程序發送郵件只需要寫以下幾行代碼就OK了:
#!/usr/bin/env python import smtplib s = smtplib.SMTP(mail server, port) s.login(username, passwd) s.sendmail(fromaddr, toaddrs, msg)
不過使用這種方法不一定總是可行,昨天用這種方式發送郵件的時候程序總是會拋異常:
File "/usr/lib64/python2.7/smtplib.py", line 617, in login raise SMTPException("No suitable authentication method found.") smtplib.SMTPException: No suitable authentication method found.
查看python smtplib.py代碼
if not self.has_extn("auth"): raise SMTPException("SMTP AUTH extension not supported by server.") # Authentication methods the server supports: authlist = self.esmtp_features["auth"].split() # List of authentication methods we support: from preferred to # less preferred methods. Except for the purpose of testing the weaker # ones, we prefer stronger methods like CRAM-MD5: preferred_auths = [AUTH_CRAM_MD5, AUTH_PLAIN, AUTH_LOGIN] # Determine the authentication method we'll use authmethod = None for method in preferred_auths: if method in authlist: authmethod = method break if authmethod == AUTH_CRAM_MD5: (code, resp) = self.docmd("AUTH", AUTH_CRAM_MD5) if code == 503: # 503 == 'Error: already authenticated' return (code, resp) (code, resp) = self.docmd(encode_cram_md5(resp, user, password)) elif authmethod == AUTH_PLAIN: (code, resp) = self.docmd("AUTH", AUTH_PLAIN + " " + encode_plain(user, password)) elif authmethod == AUTH_LOGIN: (code, resp) = self.docmd("AUTH", "%s %s" % (AUTH_LOGIN, encode_base64(user, eol=""))) if code != 334: raise SMTPAuthenticationError(code, resp) (code, resp) = self.docmd(encode_base64(password, eol="")) elif authmethod is None: raise SMTPException("No suitable authentication method found.")
拋出異常的地方是上面代碼中加粗的地方,主要是當前連接支持server並不支持[AUTH_CRAM_MD5, AUTH_PLAIN, AUTH_LOGIN]
中的任何一種認證方式,導致程序運行出現問題。
解決方案是:在初始化SMTP和login之間調用starttls()方法就可以了,完整的代碼如下:
#!/usr/bin/env python import smtplib s = smtplib.SMTP(mail server, port) s.starttls() s.login(username, passwd) s.sendmail(fromaddr, toaddrs, msg)