一尘不染

如何使用Python imaplib回复电子邮件并包含原始消息?

python

我目前正在使用imaplib从服务器获取电子邮件并处理内容和附件的方法。

我想以状态/错误消息回复这些消息,并链接到我的网站上生成的结果内容(如果可以处理)。这应包括原始消息,但应删除所有附件(附件很大),最好仅用其文件名/大小替换它们。

由于我已经遍历了MIME消息部分,因此我假设需要做的是构建一个包含原始消息副本的新MIME消息树,并删除/替换附件节点。

在我走这条路之前,我希望有人能给我一些提示。有没有任何一种库函数可以做到这一点?我应该遵守哪种标准行为?

我目前知道的/我使用的imaplibsmtplibemail模块和,但可能已经错过了在那里的东西明显。它也正在Django中运行,因此可以在其中使用任何东西,django.core.email如果这样做更容易。


阅读 297

收藏
2021-01-20

共1个答案

一尘不染

传入消息的原始MIME树结构如下(使用email.iterators._structure(msg)):

multipart/mixed
    text/html                (message)
    application/octet-stream (attachment 1)
    application/octet-stream (attachment 2)

通过GMail进行回复的结构如下:

multipart/alternative
    text/plain
    text/html

也就是说,它们并没有我想的那么聪明,只是丢弃了附件(好的),并提供了文本和HTML版本来明确重组“引用的内容”。

我开始认为这也是我应该做的,只是用一条简单的消息答复,因为丢弃附件后,保留原始消息没有多大意义。

尽管如此,还是应该回答我最初的问题,因为无论如何我现在已经弄清楚了如何解决。

首先,将原始邮件中的所有附件替换为文本/纯文本占位符:

import email

original = email.message_from_string( ... )

for part in original.walk():
    if (part.get('Content-Disposition')
        and part.get('Content-Disposition').startswith("attachment")):

        part.set_type("text/plain")
        part.set_payload("Attachment removed: %s (%s, %d bytes)"
                         %(part.get_filename(), 
                           part.get_content_type(), 
                           len(part.get_payload(decode=True))))
        del part["Content-Disposition"]
        del part["Content-Transfer-Encoding"]

然后创建一个回复消息:

from email.mime.multipart import MIMEMultipart
from email.mime.text import MIMEText
from email.mime.message import MIMEMessage

new = MIMEMultipart("mixed")
body = MIMEMultipart("alternative")
body.attach( MIMEText("reply body text", "plain") )
body.attach( MIMEText("<html>reply body text</html>", "html") )
new.attach(body)

new["Message-ID"] = email.utils.make_msgid()
new["In-Reply-To"] = original["Message-ID"]
new["References"] = original["Message-ID"]
new["Subject"] = "Re: "+original["Subject"]
new["To"] = original["Reply-To"] or original["From"]
new["From"] = "me@mysite.com"

然后附加原始的MIME消息对象并发送:

new.attach( MIMEMessage(original) )

s = smtplib.SMTP()
s.sendmail("me@mysite.com", [new["To"]], new.as_string())
s.quit()

结果结构为:

multipart/mixed
    multipart/alternative
        text/plain
        text/html
    message/rfc822
        multipart/mixed
            text/html
            text/plain
            text/plain

或者使用Django稍微简单一些:

from django.core.mail import EmailMultiAlternatives
from email.mime.message import MIMEMessage

new = EmailMultiAlternatives("Re: "+original["Subject"],
                             "reply body text", 
                             "me@mysite.com", # from
                             [original["Reply-To"] or original["From"]], # to
                             headers = {'Reply-To': "me@mysite.com",
                                        "In-Reply-To": original["Message-ID"],
                                        "References": original["Message-ID"]})
new.attach_alternative("<html>reply body text</html>", "text/html")
new.attach( MIMEMessage(original) ) # attach original message
new.send()

结果结束(至少在GMail中),将原始消息显示为“
----转发的消息----”,这与我所追求的不完全相同,但是总体思路可行,我希望这个答案可以帮助某人尝试弄清楚如何摆弄MIME消息。

2021-01-20