通过python发送时,附件不会显示在Outlook中

时间:2013-04-30 22:42:48

标签: python email outlook email-attachments

我们正在尝试发送带附件的电子邮件,但出于某种原因,附件不会显示给使用Outlook的用户。

如果他们使用gmail将电子邮件转发给某人,则可以在Gmail中显示 如果gmail用户将电子邮件转发给outlook用户,则会显示在outlook中(可能是因为gmail重建了该消息)。

这是我们用来发送电子邮件的代码:

def send_email(headers={}, attachments=[], body={}):
    ADDRESS_HEADERS = set(['from', 'to', 'cc', 'bcc', 'reply-to'])
    msg = MIMEMultipart('alternative')
    msg.preamble = "You need a MIME-aware email client to read this email.\n"

    def add_headers():
        def encode_address(v):
            (name, address) = parseaddr(v)
            name = str(Header(unicode(name), 'utf-8'))
            address = address.encode('ascii')
            return formataddr((name, address))

        for key, value in headers.iteritems():
            if not isinstance(value, list):
                value = [value]
            if key.lower() in ADDRESS_HEADERS:
                value = map(encode_address, value)
            msg[key.title()] = u';'.join(value)

    def set_body():
        msg.attach(MIMEText(body.get('text', ''), 'plain', _charset='utf-8'))
        if 'html' in body:
            msg.attach(MIMEText(body['html'], 'html', _charset='utf-8'))

    def attach_file(attachment):
        maintype, subtype = attachment['mimetype'].split("/", 1)
        part = MIMEBase(maintype, subtype)
        filename = attachment['filename']
        name = attachment.get('name', os.path.basename(filename))
        with open(filename, 'rb') as f:
            part.set_payload(f.read())
        encoders.encode_base64(part)
        part.add_header('Content-Disposition', 'attachment', filename=name)
        msg.attach(part)

    add_headers()
    map(attach_file, attachments)
    set_body()
    composed = msg.as_string()

    p = subprocess.Popen("sendmail -t", shell=True, stdin=subprocess.PIPE, stderr=subprocess.PIPE, stdout=subprocess.PIPE)
    (stdout, stderr) = p.communicate(composed)
    if p.returncode != 0:
        raise IOError(u'{}\n\n{}'.format(stdout, stderr).strip())

由于电子邮件实施的碎片化,很难找到任何相关信息。

我们附加的文件是mime类型为application/vnd.openxmlformats-officedocument.spreadsheetml.sheet

的excel文件

我正在尝试查找有关正在使用的Outlook版本的更多信息。

3 个答案:

答案 0 :(得分:2)

你可以尝试两件事。首先,我将mimetype设置为标准的bag-of bit,无论你真正发送什么类型,我都有更好的可靠性:

part = MIMEBase('application', "octet-stream")

其次,看看更改附件标题是否有帮助:

part.add_header('Content-Disposition', 'attachment; filename="%s"' % name) 

像这样设置标题适用于我们发送到Outlook。这是使用电子邮件版本'4.0.3'。我不知道你使用的是什么版本。如你所知,其中有很多。

答案 1 :(得分:2)

我也有这个问题。您可以通过更改以下行来解决它

msg = MIMEMultipart('alternative') 

msg = MIMEMultipart("multipart")

答案 2 :(得分:0)

使用 win32com.client

时,这非常简单

创建SendEmail对象后,它非常简单。

import win32com.client
outlook_obj = win32com.client.Dispatch ("Outlook Application")
sendEmail_obj = outlook_obj.CreateItem( 0x0 )

创建字符串列表,每个字符串是您要附加到当前SendEmail对象的文件的完整路径(即“C:/User/SubFolder/Filename.pdf”)。

添加相关字符串,例如收件人的电子邮件地址,主题和正文,如下所示:

sendEmail_obj.To ( "RecipientsEmail@gmail.com" )
sendEmail_obj.Subject ( "Subject Title String" )
sendEmail_obj.Body ("Dear Friend:\n \t I wanted to...") 

您的附件路径字符串项列表应各自代表您要附加的项的完整文件系统路径。让我们将这个路径字符串列表称为attachment_pathlist。

for CurrentAttachmentPath in attachment_pathlist : 
    sendEmail_obj.Attachments.Add ( CurrentAttachmentPath )

那应该为发送准备所有附件。然后剩下的就是......

sendEmail_obj.Send()