Python基于smtplib实现异步发送邮件服务
作者:hebedich 发布时间:2021-01-15 07:40:37
标签:Python,smtplib,异步发送邮件
基于smtplib包制作而成,但在实践中发现一个不知道算不算是smtplib留的一个坑,在网络断开的情况下发送邮件时会抛出一个socket.gaierror的异常,但是smtplib中并没有捕获这个异常,导致程序会因这个异常终止,因此代码中针对这部分的异常进行处理,确保不会异常终止。
#!/usr/bin/env python
# -*- coding: utf-8 -*-
__author__ = 'Zoa Chou'
# see http://www.mudoom.com/Article/show/id/29.html for detail
import logging
import smtplib
import mimetypes
import socket
from email import encoders
from email.header import Header
from email.mime.text import MIMEText, MIMENonMultipart
from email.mime.base import MIMEBase
from email.utils import parseaddr, formataddr
class Mailer(object):
def __init__(self):
pass
def send_mail(self, smtp_server, from_address, to_address, subject, body, files=None):
"""
发送邮件主程序
:param smtp_server: dict 邮件服务器设置
:keyword host: string smtp服务器地址
:keyword port: int smtp服务器端口号
:keyword user: string 用户名
:keyword passwd: string 密码
:keyword ssl: bool 是否启用ssl,默认False
:keyword timeout: int 超时时间,默认10s
:param from_address: 发件人邮箱
:param to_address: 收件人邮箱
:param subject: 邮件标题
:param body: 邮件内容
:param files: 附件
:raise: NetworkError/MailerException
"""
# 格式化邮件内容
body = self._encode_utf8(body)
# 邮件类型
content_type = 'html' if body.startswith('<html>') else 'plain'
msg = MIMENonMultipart() if files else MIMEText(body, content_type, 'utf-8')
# 格式化邮件数据
msg['From'] = self._format_address(from_address)
msg['To'] = ', '.join(self._format_list(to_address))
msg['subject'] = self._encode_utf8(subject)
# 构造附件数据
if files:
msg.attach(MIMEText(body, content_type, 'utf-8'))
cid = 0
for file_name, payload in files:
file_name = self._encode_utf8(file_name)
main_type, sub_type = self._get_file_type(file_name)
if hasattr(payload, 'read'):
payload = payload.read()
f_name = self._encode_header(file_name)
mime = MIMEBase(main_type, sub_type, filename=f_name)
mime.add_header('Content-Disposition', 'attachment', filename=f_name)
mime.add_header('Content-ID', '<%s>' % cid)
mime.add_header('X-Attachment-Id', '%s' % cid)
mime.set_payload(payload)
encoders.encode_base64(mime)
msg.attach(mime)
cid += 1
host = smtp_server.get('host')
port = smtp_server.get('port')
user = smtp_server.get('user')
passwd = smtp_server.get('passwd')
ssl = smtp_server.get('ssl', False)
time_out = smtp_server.get('timeout', 10)
# 没有输入端口则使用默认端口
if port is None or port == 0:
if ssl:
port = 465
else:
port = 25
logging.debug('Send mail form %s to %s' % (msg['From'], msg['To']))
try:
if ssl:
# 开启ssl连接模式
server = smtplib.SMTP_SSL('%s:%d' % (host, port), timeout=time_out)
else:
server = smtplib.SMTP('%s:%d' % (host, port), timeout=time_out)
# 开启调试模式
# server.set_debuglevel(1)
# 如果存在用户名密码则尝试登录
if user and passwd:
server.login(user, passwd)
# 发送邮件
server.sendmail(from_address, to_address, msg.as_string())
logging.debug('Mail sent success.')
# 关闭stmp连接
server.quit()
except socket.gaierror, e:
""" 网络无法连接 """
logging.exception(e)
raise NetworkError(e)
except smtplib.SMTPServerDisconnected, e:
""" 网络连接异常 """
logging.exception(e)
raise NetworkError(e)
except smtplib.SMTPException, e:
""" 邮件发送异常 """
logging.exception(e)
raise MailerException(e)
def _format_address(self, s):
"""
格式化邮件地址
:param s:string 邮件地址
:return: string 格式化后的邮件地址
"""
name, address = parseaddr(s)
return formataddr((self._encode_header(name), self._encode_utf8(address)))
def _encode_header(self, s):
"""
格式化符合MIME的头部数据
:param s: string 待格式化数据
:return: 格式化后的数据
"""
return Header(s, 'utf-8').encode()
def _encode_utf8(self, s):
"""
格式化成utf-8编码
:param s: string 待格式化数据
:return: string 格式化后的数据
"""
if isinstance(s, unicode):
return s.encode('utf-8')
else:
return s
def _get_file_type(self, file_name):
"""
获取附件类型
:param file_name: 附件文件名
:return: dict 附件MIME
"""
s = file_name.lower()
pos = s.rfind('.')
if pos == -1:
return 'application', 'octet-stream'
ext = s[pos:]
mime = mimetypes.types_map.get(ext, 'application/octet-stream')
pos = mime.find('/')
if pos == (-1):
return mime, ''
return mime[:pos], mime[pos+1:]
def _format_list(self, address):
"""
将收件人地址格式化成list
:param address: string/list 收件人邮箱
:return: list 收件人邮箱list
"""
l = address
if isinstance(l, basestring):
l = [l]
return [self._format_address(s) for s in l]
class MailerException(Exception):
""" 邮件发送异常类 """
pass
class NetworkError(MailerException):
""" 网络异常类 """
pass
# test for @qq.com
if __name__ == '__main__':
import sys
def prompt(prompt):
"""
接收终端输入的数据
"""
sys.stdout.write(prompt + ": ")
return sys.stdin.readline().strip()
from_address = prompt("From(Only @qq.com)")
passwd = prompt("Password")
to_address = prompt("To").split(',')
subject = prompt("Subject")
print "Enter message, end with ^D:"
msg = ''
while 1:
line = sys.stdin.readline()
if not line:
break
msg = msg + line
print "Message length is %d" % len(msg)
# QQ邮箱默认设置
smtp_server = {'host': 'smtp.qq.com', 'port': None, 'user': from_address, 'passwd': passwd, 'ssl': True}
mailer = Mailer()
try:
mailer.send_mail(smtp_server, from_address, to_address, subject, msg)
except MailerException, e:
print(e)
以上所述就是本文的全部内容了,希望大家能够喜欢。
0
投稿
猜你喜欢
- 在php中获取数组长度方法很简单,php为我们提供了两个函数可以计算一维数组长度,如count,sizeof都可以直接统计数组长度哦,下面我
- Microsoft SQL server2000由一系列相互协作的组件构成。能满足最大的WEB站点和企业数据处理系统存储
- 因为正则不够完善,所以代码中不能直接出现 <? 和 ?>如果是字符串,可以拆开写 "<" + &quo
- 使用MySQL,安全问题不能不注意。以下是MySQL提示的23个注意事项:1.如果客户端和服务器端的连接需要跨越并通过不可信任的网络,那么就
- 问:怎样实现ORACLE中用一条SQL实现其它进制到十进制的转换?答:具体示例如下:-----二进制转换十进制---------------
- 本文实例为大家分享了Python基于OpenCV实现人脸检测,并保存的具体代码,供大家参考,具体内容如下安装opencv如果安装了pip的话
- 功能: 1、 允许/限制对表的修改 2、 自动生成派生列,比如自增字段 3、 强制数据一致性 4、 提供审计和日志记录 5、 防止无效的事务
- 以前跟同事开玩笑时说过,我们遇到的用户在访谈测试过程中的表现基本上就三种类型,发泄型,赞美型和实话实说型。发泄型用户通常是在产品的使用过程中
- PHP生成桌面快捷方式就是这么的简单,大家生成的时候改下你要生成的网站即可。dianji.html代码:<a href="a
- 随着互联网的飞速发展,很多主页都是越做越漂亮、越来越有个性。如果你是一个资深的网虫不会没有自己的主页吧!下面小弟就来谈谈如何使你的网上“小窝
- 内容摘要:本文介绍了使用asp来JMail v4.3发信的大部分常用方法,包括邮件基本信息、身份验证、附件等。无需很多的修改就可以
- argparse是python标准库里面用来处理命令行参数的库命令行参数分为位置参数和选项参数:位置参数就是程序根据该参数出现的位置来确定的
- Cookie是一种小型文本文件,存储在用户计算机中,用于跟踪用户在互联网上的活动。Cookie通常由网站创建,以便记住用户的偏好和登录状态,
- 在了解了JavaScript1.7的相关更新后,我们再继续关注JavaScript1.8的新特性。在W3C Group带来的JavaScri
- 如何验证日期输入是否正确?我们可以利用ASP 3.0里的On Error Resume Next来完成这项功能:<%Option&nb
- 1.JOIN和UNION区别 join 是两张表做交连后里面条件相同的部分记录产生一个记录集, union是产生的两个记录集(字段要一样的)
- 基本函数如下:/// <summary> /// 需要分页时使用,根据参数和ConditionExpress获取DataTabl
- 本文实例讲述了asp.net C#实现解压缩文件的方法。一共给大家介绍了三段代码,一个是简单的解压缩单个zip文件,后一个可以解压批量的大量
- 如果你是个赛车手,并且按一下按钮就能够立即更换引擎而不需要把车开到车库里去换,那会是什么感觉呢?MySQL数据库为开发人员所做的就好像是按按
- <html><head><meta http-equiv="Content-T