如何使用python的imaplib获取邮件正文?
我想从IMAP4服务器获取完整的消息。在Python的文档中,我找到了一段可以工作的代码:
>>> t, data = M.fetch('1', '(RFC822)')
>>> body = data[0][1]
我在想,data[0][1]是否总是能返回消息的主体内容。当我运行'RFC822.SIZE'时,我得到的只是一个字符串,而不是一个元组。
我浏览了一下rfc1730,但没能弄清楚'RFC822'的正确响应结构。从imaplib的文档中也很难看出获取结果的结构。
这是我在获取RFC822
时得到的结果:
('OK', [('1 (RFC822 {858569}', 'body of the message', ')')])
但是当我获取RFC822.SIZE
时,我得到的是:
('OK', ['1 (RFC822.SIZE 847403)'])
我应该如何正确处理data[0]这个列表?我能相信当它是一个元组列表时,这些元组恰好有3个部分,并且第二部分是有效载荷吗?
也许你知道更好的IMAP4库吗?
4 个回答
8
试试我的这个包:
https://pypi.org/project/imap-tools/
示例:
from imap_tools import MailBox
# get list of email bodies from INBOX folder
with MailBox('imap.mail.com').login('test@mail.com', 'password', 'INBOX') as mailbox:
bodies = [msg.text or msg.html for msg in mailbox.fetch()]
功能:
基本的邮件操作:获取邮件、获取唯一标识符、获取邮件数量
解析邮件的属性
搜索条件的查询构建器
对邮件的操作:复制、删除、标记、移动、添加
对文件夹的操作:列出、设置、获取、创建、检查是否存在、重命名、订阅、删除、状态
IDLE命令:开始、轮询、停止、等待
在IMAP操作失败时抛出异常
没有外部依赖,经过测试
15
IMAPClient这个包用起来简单很多。从它的描述来看:
这是一个易于使用、符合Python风格的完整IMAP客户端库。
42
不... imaplib
是一个相当不错的库,问题在于 imap 本身太难懂了。
你可以检查一下 t == 'OK'
,不过我用过的情况下,data[0][1]
的表现还是很正常的。
这里有个我用来提取我通过邮件收到的签名证书的简单例子,虽然不是特别安全,但对我来说足够用了:
import getpass, os, imaplib, email
from OpenSSL.crypto import load_certificate, FILETYPE_PEM
def getMsgs(servername="myimapserverfqdn"):
usernm = getpass.getuser()
passwd = getpass.getpass()
subject = 'Your SSL Certificate'
conn = imaplib.IMAP4_SSL(servername)
conn.login(usernm,passwd)
conn.select('Inbox')
typ, data = conn.search(None,'(UNSEEN SUBJECT "%s")' % subject)
for num in data[0].split():
typ, data = conn.fetch(num,'(RFC822)')
msg = email.message_from_string(data[0][1])
typ, data = conn.store(num,'-FLAGS','\\Seen')
yield msg
def getAttachment(msg,check):
for part in msg.walk():
if part.get_content_type() == 'application/octet-stream':
if check(part.get_filename()):
return part.get_payload(decode=1)
if __name__ == '__main__':
for msg in getMsgs():
payload = getAttachment(msg,lambda x: x.endswith('.pem'))
if not payload:
continue
try:
cert = load_certificate(FILETYPE_PEM,payload)
except:
cert = None
if cert:
cn = cert.get_subject().commonName
filename = "%s.pem" % cn
if not os.path.exists(filename):
open(filename,'w').write(payload)
print "Writing to %s" % filename
else:
print "%s already exists" % filename