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)
以上所述就是本文的全部内容了,希望大家能够喜欢。


猜你喜欢
- \\create by ahuinan 2009-6-22 \\up by ahuian 2009-6-23 \\up by ahuinan
- 1.安装Apache 在终端中输入下面的命令就可以安装Apache了:sudo yum install httpdsudo的意思是
- 前言本文主要给大家介绍了关于Django自定义过滤器的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍:过滤器与函数d
- 前言随着微信的普及,越来越多的人开始使用微信。微信渐渐从一款单纯的社交软件转变成了一个生活方式,人们的日常沟通需要微信,工作交流也需要微信。
- 列表1~n输出步长为3的分组print([[x for x in range(1,101)][i:i+3] for i in range(0
- 1. 横排往下会影响阅读速度。如12345678的单排单列数字,肯定是竖排阅读快。但多行多列的整块信息,横排并不见得就比竖排慢,比如所有简体
- 概述OpenCV 是一个跨平台的计算机视觉库, 支持多语言, 功能强大. 今天小白就带大家一起携手走进 OpenCV 的世界.梯度运算梯度:
- 流程,通俗来讲,就是许多人,在做一系列的事情时,怎样相互协调,安排好这一系列事情的先后顺序,有什么事先的约定,需要达到怎样的预期目标。在UE
- 看代码吧~import pymongofrom dateutil import parserdateStr = "2019-05-
- 给大家分享一下TensorFlow在MAC系统中的安装步骤以及环境搭建的操作流程。TensorFlow 底层的图模型结构清晰,容易改造;支持
- 我的大体思路是:时时监听鼠标的坐标,当鼠标移动时,透明层随着鼠标移动,大图片相对透明层的移动而移动。不废话了,看代码。 <!DOCTY
- 常用的 random 模块方法import random# random.random()用于生成一个 0 到 1 的随机浮点数: 0 &l
- 使用SQL SERVER的[导入]功能,便可将access数据转换,但要注意原来的'自增字段'需要修改,将相应字段标识修改为
- 1、登陆mysql数据库 mysql -u root -p查看user表mysql> use mysql;Database chang
- 不管是哪个版本的Ubuntu,安装mysql数据库基本上都是大同小异。下面介绍一下具体的安装步骤:1、打开终端,并取得root权限2、在终端
- 比如有一个需求,通过sql语句,返回-5至5的随机整数.如果这一个放在PHP中,则非常简单直接用print rand(-5,5);?>
- 当我们想更新一张动态表的时候(即:表中的数据不断的添加),也许我们会用数据库代理,通过写作业,然后让他定时查询动态表中最新添加的数据,然后更
- 从今天起,我将陆续将 ppk on JavaScript 的读书心得发布到这个blog上。ppk是我所景仰的一位web开发者,原因无它,只是
- re.findall()在字符串中找到正则表达式所匹配的所有子串,并返回一个列表,如果没有找到匹配的,则返回空列表。语法格式:re.find
- 随着3G的普及,越来越多的人使用手机上网。移动设备正超过桌面设备,成为访问互联网的最常见终端。于是,网页设计师不得不面对一个难题:如何才能在