用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. 生成最终邮件。

posted on 2008-12-01 13:13  sislcb  阅读(6996)  评论(0编辑  收藏  举报