页面

2009年3月24日

python模块之email: 电子邮件编码解码 (二、编码邮件)

    用email模块来生成邮件也是很简单的,只是需要一些mime的基础知识。下面看看一点mime基础。
    MIME消息由消息头和消息体两大部分组成,在邮件里就是邮件头和邮件体。邮件头与邮件体之间以空行进行分隔。这点可以用文本编辑器(比如记事本)查看一个邮件的源文件就可以清除看到。outlook和foxmail自己就有查看源文件的功能。
    邮件头包含了发件人、收件人、主题、时间、MIME版本、邮件内容的类型等重要信息。每条信息称为一个域,由域名后加“: ”和信息内容构成,可以是一行,较长的也可以占用多行。域的首行必须“顶头”写,即左边不能有空白字符(空格和制表符);续行则必须以空白字符打头,且第一个空白字符不是信息本身固有的。
    邮件体包含邮件的内容,它的类型由邮件头的“Content-Type”域指出。最常见的类型有text/plain(纯文本)和text/html(超文本)。邮件体被分为多个段,每个段又包含段头和段体两部分,这两部分之间也以空行分隔。常见的multipart类型有三种:multipart/mixed, multipart/related和multipart/alternative。从它们的名称,不难推知这些类型各自的含义和用处。它们之间的层次关系可归纳为下图所示:
    可以看出,如果在邮件中要添加附件,必须定义multipart/mixed段;如果存在内嵌资源,至少要定义 multipart/related段;如果纯文本与超文本共存,至少要定义multipart/alternative段。生成邮件就是要生成这各个MIME部分。email模块对这些处理都是包装好的,看看生成方法:

#-*- encoding: gb2312 -*-
import email
import string, sys, os, email
import time

class MailCreator:
    def __init__(self):
        # 创建邮件的message对象
        self.msg = email.Message.Message()
        self.mail = ""   
       
    def create(self, mailheader, maildata, mailattachlist=[]):
        # mailheader 是dict类型,maildata是list, 且里面第一项为纯文本类型,第二项为html.
        # mailattachlist 是list, 里面为附件文件名
        if not mailheader or not maildata:
            return
       
        for k in mailheader.keys():
            # 对subject要作特殊处理,中文要转换一下。
            # 比如 "我的一个测试邮件" 就要转换为 =?gb2312?b?ztK1xNK7uPay4srU08q8/g==?=
            if k == 'subject':
                self.msg[k] = email.Header.Header(mailheader[k], 'gb2312')          
            else:
                self.msg[k] = mailheader[k]
        # 创建纯文本部分
        body_plain = email.MIMEText.MIMEText(maildata[0], _subtype='plain', _charset='gb2312')
        body_html = None
        # 创建html部分,这个是可选的
        if maildata[1]:
            body_html = email.MIMEText.MIMEText(maildata[1], _subtype='html', _charset='gb2312')
       
       
        # 创建一个multipart, 然后把前面的文本部分和html部分都附加到上面,至于为什么,可以看看mime相关内容
        attach=email.MIMEMultipart.MIMEMultipart()
        attach.attach(body_plain)
        if body_html:
            attach.attach(body_html)
        # 处理每一个附件
        for fname in mailattachlist:
            attachment=email.MIMEText.MIMEText(email.Encoders._bencode(open(fname,'rb').read()))
            # 这里设置文件类型,全部都设置为Application.当然也可以是Image,Audio什么的,这里不管那么多
            attachment.replace_header('Content-type','Application/octet-stream;name="'+os.path.basename(fname)+'"')
            # 一定要把传输编码设置为base64,因为这里默认就是用的base64
            attachment.replace_header('Content-Transfer-Encoding', 'base64')
            attachment.add_header('Content-Disposition','attachment;filename="'+os.path.basename(fname)+'"')
            attach.attach(attachment)
        # 生成最终的邮件           
        self.mail = self.msg.as_string()[:-1] + attach.as_string()
       
        return self.mail

if __name__ == '__main__':
    mc = MailCreator()
    header = {'from': 'zhaowei@163.com', 'to':'weizhao@163.com', 'subject':'我的一个测试邮件'}
    data = ['plain text information', '<font color="red">html text information</font>']
    if sys.platform == 'win32':
        attach = ['c:\\windows\\clock.avi']
    else:
        attach = ['/bin/cp']
   
    mail = mc.create(header, data, attach)
   
    f = open("test.eml", "wb")
    f.write(mail)
    f.close()

这里我自己封装了一个类来做处理,大体的过程就是:
1. 先创建message对象: email.Message.Message()
2. 创建MIMEMultipart对象:email.MIMEMultipart.MIMEMultipart()
3. 创建各个MIMEText对象,并把他们attach到MIMEMultipart里,这里的MIMEText其实不仅仅是text, 也包括image, application, audio等等。
4. 生成最终邮件。

没有评论: