热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

imap读取邮箱和附件_Python3之发送和读取邮件

01发送邮件发送邮件使用SMTP协议【SimpleMailTransferProtocol简单的邮件传输协议】,SMTP协议是SMTP客户端与SMTP服务器之间的通信
01

发送邮件

发送邮件使用SMTP协议【Simple Mail Transfer Protocol简单的邮件传输协议】,SMTP协议是SMTP客户端与SMTP服务器之间的通信协议。

python中发送邮件使用的模块有smtplib和email:
1.使用smtplib模块进行发送邮件;
2.使用email模块来添加发送的邮件内容。

1. smtplib模块

导入模块:import smtplib

1.1. 创建SMTP对象

smtplib.SMTPsmtplib.SMTP_SSL:均可以用来创建SMTP对象;smtplib.SMTP_SSL:使用安全加密的SSL协议连接到SMTP服务器;smtplib.SMTP:没有进行安全加密。
故若待测试邮箱不允许使用非SSL和非TLS频道通信时,则无法使用smtp.SMTP方式来创建客户端对象。
【查看邮箱的通信方式:邮箱设置菜单中,查看邮箱的接收服务器和发送服务器信息。】

如:腾讯企业邮箱接收服务器:imap.exmail.qq.com(使用SSL,端口号993)发送服务器:smtp.exmail.qq.com(使用SSL,端口号465)

  • smtplib.SMTP(host,port,local_hostname,timeout,source_address)

  • smtplib.SMTP_SSL(host, port, local_hostname, keyfile, certfile, timeout, source_address, context)
    创建SMTP对象。
    host:SMTP发送服务器主机
    port:SMTP服务器端口号

1.2. SMTP对象操作
  • login(user, password, *, initial_response_ok=True)
    SMTP对象登录
    user:授权登录的用户名
    password:授权登录的密码

  • sendmail(from_addr, to_addrs, msg, mail_options=[], rcpt_options=[])
    SMTP对象发送邮件
    from_addr:发件人地址,字符串类型。
    to_addr:收件人地址,包括收件人和抄送人。
    多个收件人时to_addr参数为列表,单个收件人时to_addr参数可以为列表或字符串。
    msg:要发送的信息

  • quite()
    终止SMTP会话

2. 发送邮件的实例

2.1. 添加邮件内容,包括收件人、抄送人、正文、附件

from email.mime.multipart import MIMEMultipartfrom email.header import Headerfrom email.mime.text import MIMETextfrom email.mime.image import MIMEImageimport osclass EmailContent: def __init__(self, senderAdr, emailSubject, toReceivers, ccReceivers): # 邮件对象 self.msg = MIMEMultipart() # 添加发件人头 self.msg['From'] = Header("测试" + " + senderAdr + ">", 'utf-8') # 添加收件人 if isinstance(toReceivers, str): self.msg["To"] = toReceivers elif isinstance(toReceivers, list): self.msg['To'] = ";".join(toReceivers) # 添加抄送人 if isinstance(ccReceivers, str): self.msg["Cc"] = ccReceivers elif isinstance(ccReceivers, list): self.msg["Cc"] = ";".join(ccReceivers) # 添加邮件主题 self.msg['Subject'] = Header(emailSubject, "utf-8") def addBody(self, bodyType): """ 添加不同的邮件正文的实例 1. body为字符串:(如)"这是一个邮件正文内容" 2. body为html格式的字符串:(如)"

第一段

第二段

" 3. body正文中包含有图片: """ if bodyType == "string": body = "这是一个邮件正文内容" mimeText = MIMEText(body, "plain", "utf-8") self.msg.attach(mimeText) elif bodyType == "html": body = "

第一段

第二段

" mimeText = MIMEText(body, "html", "utf-8") self.msg.attach(mimeText) elif "image" in bodyType: imageFile = "E://log//test.png" imageId = os.path.split(imageFile)[1] # 添加内容 body = '''

测试图片为:

'''.format(imageId=imageId) mimeText = MIMEText(body, "html", "utf-8") self.msg.attach(mimeText) # 读取图片,并设置图片id用于邮件正文引用 with open(imageFile, "rb") as fp: mimeImage = MIMEImage(fp.read()) mimeImage.add_header("Content-ID", imageId) self.msg.attach(mimeImage) def addAttachment(self, attachmentName): """ 添加附件 :return: """ file = "E://log//test.txt" # file = "E://log//test.zip" # file = "E://log//test.png" filePath, fileName = os.path.split(file) print("fileName =", fileName) enclosure = MIMEText(open(file, 'rb').read(), 'base64', 'utf-8') enclosure['Content-Type'] = 'application/octet-stream' if attachmentName == "英文": enclosure['Content-Disposition'] = 'attachment; filename="%s"' % fileName elif attachmentName == "中文": enclosure.add_header("Content-Disposition", "attachment", filename=("gbk", "", fileName)) self.msg.attach(enclosure)
2.2. 发送邮件

import smtplibdef SendEmail(): """发送邮件""" # SMTP的服务器信息 smtpHost = "smtp.exmail.qq.com" sslPort = 465 senderAdr = "xx@xx.cn" senderPwd = "XXXX" # 创建SMTP对象 smtpServer = smtplib.SMTP_SSL(smtpHost, sslPort) # # 设置debug模块 # smtpServer.set_debuglevel(True) # 登录 smtpServer.login(senderAdr, senderPwd) # 添加邮件内容 toReceivers = ["a@xx.cn", "b@xx.cn"] ccReceivers = ["d@xx.cn", "e@xx.cn"] toAddrs = toReceivers + ccReceivers emailSubject = "这是个编程技术圈自动发送的邮件" emailContent = EmailContent(senderAdr, emailSubject, toReceivers, ccReceivers) emailContent.addBody("html") emailContent.addAttachment("英文") message = emailContent.msg # 发送 smtpServer.sendmail(senderAdr, toAddrs, message.as_string()) # 终止SMTP会话 smtpServer.quit()SendEmail()

02

读取邮件

收取邮件使用POP3协议;

解析邮件:需要将收取的邮件转化为email.message.Message对象,再使用email模块解析内容。

1. 读取邮件的实例

1.1. 获取某封邮件的对象

import poplibfrom email.parser import Parser"""POP的服务器信息"""popHost = "pop.exmail.qq.com"userAdr = "xx@xx.cn"userPwd = "xxxxx"""" 创建POP3对象,添加用户名和密码"""pop3Server = poplib.POP3(popHost)pop3Server.user(userAdr)pop3Server.pass_(userPwd)"""获取邮件数量和占用空间"""messageCount, mailboxSize = pop3Server.stat()"""获取邮件请求返回状态码、每封邮件的字节大小(b'第几封邮件 此邮件字节大小')、"""response, msgNumOctets, octets = pop3Server.list()""" 获取任意一封邮件的邮件对象【第一封邮件的编号为1,而不是0】"""msgIndex = random.randint(1,messageCount)print(msgIndex)# 获取第msgIndex封邮件的信息response, msgLines, octets = pop3Server.retr(msgIndex)# msgLines中为该邮件的每行数据,先将内容连接成字符串,再转化为email.message.Message对象msgLinesToStr = b"\r\n".join(msgLines).decode("utf8", "ignore")messageObject = Parser().parsestr(msgLinesToStr)print(messageObject)""" 终止POP3服务"""pop3Server.quit()

1.2. 解析邮件对象

1.2.1. 获取邮件日期

msgDate = messageObject["date"]print(msgDate)

1.2.2. 获取邮件发件人实名、邮箱地址

获取邮件实名时,名称一般是加密的,此时就需要对头文件进行解码才可获取它的实际内容

from email.header import decode_headerdef decodeMsgHeader(header): """ 解码头文件 :param header: 需解码的内容 :return: """ value, charset = decode_header(header)[0] if charset: value = value.decode(charset) return value

from email.utils import parseaddrsenderContent = messageObject["From"]# parseaddr()函数返回的是一个元组(realname, emailAddress)senderRealName, senderAdr = parseaddr(senderContent) # 将加密的名称进行解码 senderRealName = decodeMsgHeader(senderRealName) print(senderRealName)print(senderAdr)

1.2.3. 获取邮件主题

获取的邮件的主题也是加密的,此时就需要对头文件进行解码才可获取它的实际内容

msgHeader = messageObject["Subject"]# 对头文件进行解码msgHeader = decodeMsgHeader(msgHeader )print(msgHeader)

1.2.4. 获取邮件正文

一封邮件的正文内容,可能是由几部分构成,每部分的格式不同。

"""获取邮件正文内容"""msgBodyContents = []if messageObject.is_multipart(): # 判断邮件是否由多个部分构成 messageParts = messageObject.get_payload() # 获取邮件附载部分 for messagePart in messageParts: bodyContent = decodeBody(messagePart) if bodyContent: msgBodyContents.append(bodyContent)else: bodyContent = decodeBody(messageObject) if bodyContent: messageBodyContents.append(bodyContent)print(msgBodyContents)

def decodeBody(msgPart): """ 解码内容 :param msgPart: 邮件某部分 """ contentType = msgPart.get_content_type() # 判断邮件内容的类型,text/html textContent = "" if contentType == 'text/plain' or contentType == 'text/html': content = msgPart.get_payload(decode=True) charset = msgPart.get_charset() if charset is None: contentType = msgPart.get('Content-Type', '').lower() position = contentType.find('charset=') if position >= 0: charset = contentType[position + 8:].strip() if charset: textContent = content.decode(charset) return textContent

1.2.5. 获取邮件附件

邮件附件名为中文时,需借助头文件解码方式进行解码,否则会为乱码。

messageAttachments = []if messageObject.is_multipart(): # 判断邮件是否由多个部分构成 messageParts = messageObject.get_payload() # 获取邮件附载部分 for messagePart in messageParts: name = messagePart.get_param("name") # 名字存在,则表示此部分为附件 if name: fileName = decodeMsgHeader(name) # 解码 messageAttachments.append(fileName)else: name = messageObject.get_param("name") if name: fileName = decodeMsgHeader(name) # 解码 messageAttachments.append(fileName)print(messageAttachments)

2. 读取邮件时遇到的问题

2.1. 提示“poplib.error_proto: line too long”

File "XXX/EmailInfo.py", line 22, in getMessageObject return parser.Parser().parsestr(b"\n".join(self.popServer.retr(i)[1]).decode("utf8", "ignore")) File "/usr/local/lib/python3.6/poplib.py", line 248, in retr return self._longcmd('RETR %s' % which) File "/usr/local/lib/python3.6/poplib.py", line 183, in _longcmd return self._getlongresp() File "/usr/local/lib/python3.6/poplib.py", line 168, in _getlongresp line, o = self._getline() File "/usr/local/lib/python3.6/poplib.py", line 130, in _getline raise error_proto('line too long')poplib.error_proto: line too long

POP3对行长度做了限制,默认为_MAXLINE = 2048,故若是邮件超过此长度就会提示“poplib.error_proto: line too long”。

解决方案:在读取邮件代码中重新定义最大行长度,即给poplib._MAXLINE设置新值。

import poplibpoplib._MAXLINE=20480

   - end -  

a693e0914e8bf4c362cf8aa51bfb9088.png




推荐阅读
  • Linux服务器密码过期策略、登录次数限制、私钥登录等配置方法
    本文介绍了在Linux服务器上进行密码过期策略、登录次数限制、私钥登录等配置的方法。通过修改配置文件中的参数,可以设置密码的有效期、最小间隔时间、最小长度,并在密码过期前进行提示。同时还介绍了如何进行公钥登录和修改默认账户用户名的操作。详细步骤和注意事项可参考本文内容。 ... [详细]
  • 本文介绍了Redis的基础数据结构string的应用场景,并以面试的形式进行问答讲解,帮助读者更好地理解和应用Redis。同时,描述了一位面试者的心理状态和面试官的行为。 ... [详细]
  • 本文介绍了Web学习历程记录中关于Tomcat的基本概念和配置。首先解释了Web静态Web资源和动态Web资源的概念,以及C/S架构和B/S架构的区别。然后介绍了常见的Web服务器,包括Weblogic、WebSphere和Tomcat。接着详细讲解了Tomcat的虚拟主机、web应用和虚拟路径映射的概念和配置过程。最后简要介绍了http协议的作用。本文内容详实,适合初学者了解Tomcat的基础知识。 ... [详细]
  • 在重复造轮子的情况下用ProxyServlet反向代理来减少工作量
    像不少公司内部不同团队都会自己研发自己工具产品,当各个产品逐渐成熟,到达了一定的发展瓶颈,同时每个产品都有着自己的入口,用户 ... [详细]
  • web.py开发web 第八章 Formalchemy 服务端验证方法
    本文介绍了在web.py开发中使用Formalchemy进行服务端表单数据验证的方法。以User表单为例,详细说明了对各字段的验证要求,包括必填、长度限制、唯一性等。同时介绍了如何自定义验证方法来实现验证唯一性和两个密码是否相等的功能。该文提供了相关代码示例。 ... [详细]
  • iOS Swift中如何实现自动登录?
    本文介绍了在iOS Swift中如何实现自动登录的方法,包括使用故事板、SWRevealViewController等技术,以及解决用户注销后重新登录自动跳转到主页的问题。 ... [详细]
  • 本文介绍了一个React Native新手在尝试将数据发布到服务器时遇到的问题,以及他的React Native代码和服务器端代码。他使用fetch方法将数据发送到服务器,但无法在服务器端读取/获取发布的数据。 ... [详细]
  • 微信官方授权及获取OpenId的方法,服务器通过SpringBoot实现
    主要步骤:前端获取到code(wx.login),传入服务器服务器通过参数AppID和AppSecret访问官方接口,获取到OpenId ... [详细]
  • 本文介绍了解决java开源项目apache commons email简单使用报错的方法,包括使用正确的JAR包和正确的代码配置,以及相关参数的设置。详细介绍了如何使用apache commons email发送邮件。 ... [详细]
  • 基于分布式锁的防止重复请求解决方案
    一、前言关于重复请求,指的是我们服务端接收到很短的时间内的多个相同内容的重复请求。而这样的重复请求如果是幂等的(每次请求的结果都相同,如查 ... [详细]
  • python+selenium十:基于原生selenium的二次封装fromseleniumimportwebdriverfromselenium.webdriv ... [详细]
  • 使用Flutternewintegration_test进行示例集成测试?回答首先在dev下的p ... [详细]
  • 参考网上搜到的结果,但仍有问题,调试后完整步骤如下配置163邮箱SMTP,需要配置授权码(作为密码使用)&#x ... [详细]
  • 分享西西弗斯自动求职系统【PHP】
    后端开发|php教程后端开发-php教程做这个系统的背景是前段时间帮女朋友找工作,大家也许都有过这样的感受,发出一份份简历都石沉大海,为什么呢?经过测试,我发现是因为邮件的阅读率并 ... [详细]
  • 由于php没有提供现成的smtp函数,却提供了一个功能不甚灵活的mail()函数,这个函数需要服务器配置上的支持,并且不支持smtp验证,在很多场合无法正常的工作,因此不建议使用。 ... [详细]
author-avatar
mobiledu2502938577
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有