首页 > 技术文章 > python模块之email: 电子邮件编码

smallgou 2016-08-01 16:30 原文

转载自:http://www.cnblogs.com/sislcb/archive/2008/12/01/1344861.html

用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模块对这些处理都是包装好的,看看生成方法:

 1 #-*- encoding: gb2312 -*-
 2 
 3 import email
 4 import string, sys, os, email
 5 import time
 6 
 7 class MailCreator:
 8     def __init__(self):
 9         # 创建邮件的message对象
10         self.msg = email.Message.Message()
11         self.mail = ""    
12         
13     def create(self, mailheader, maildata, mailattachlist=[]):
14         # mailheader 是dict类型,maildata是list, 且里面第一项为纯文本类型,第二项为html.
15         # mailattachlist 是list, 里面为附件文件名
16         if not mailheader or not maildata:
17             return
18         
19         for k in mailheader.keys():
20             # 对subject要作特殊处理,中文要转换一下。
21             # 比如 "我的一个测试邮件" 就要转换为 =?gb2312?b?ztK1xNK7uPay4srU08q8/g==?=
22             if k == 'subject':
23                 self.msg[k] = email.Header.Header(mailheader[k], 'gb2312')           
24             else:
25                 self.msg[k] = mailheader[k]
26         # 创建纯文本部分
27         body_plain = email.MIMEText.MIMEText(maildata[0], _subtype='plain', _charset='gb2312')
28         body_html = None
29         # 创建html部分,这个是可选的
30         if maildata[1]:
31             body_html = email.MIMEText.MIMEText(maildata[1], _subtype='html', _charset='gb2312')
32         
33         
34         # 创建一个multipart, 然后把前面的文本部分和html部分都附加到上面,至于为什么,可以看看mime相关内容
35         attach=email.MIMEMultipart.MIMEMultipart()
36         attach.attach(body_plain)
37         if body_html:
38             attach.attach(body_html)
39         # 处理每一个附件
40         for fname in mailattachlist:
41             attachment=email.MIMEText.MIMEText(email.Encoders._bencode(open(fname,'rb').read()))
42             # 这里设置文件类型,全部都设置为Application.当然也可以是Image,Audio什么的,这里不管那么多
43             attachment.replace_header('Content-type','Application/octet-stream;name="'+os.path.basename(fname)+'"')
44             # 一定要把传输编码设置为base64,因为这里默认就是用的base64
45             attachment.replace_header('Content-Transfer-Encoding', 'base64')
46             attachment.add_header('Content-Disposition','attachment;filename="'+os.path.basename(fname)+'"')
47             attach.attach(attachment)
48         # 生成最终的邮件            
49         self.mail = self.msg.as_string()[:-1] + attach.as_string()
50         
51         return self.mail
52 
53 if __name__ == '__main__':
54     mc = MailCreator()
55     header = {'from': 'zhaowei@163.com', 'to':'weizhao@163.com', 'subject':'我的一个测试邮件'}
56     data = ['plain text information', '<font color="red">html text information</font>']
57     if sys.platform == 'win32':
58         attach = ['c:\windows\clock.avi']
59     else:
60         attach = ['/bin/cp']
61     
62     mail = mc.create(header, data, attach)
63     
64     f = open("test.eml", "wb")
65     f.write(mail)
66     f.close()

 

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

 

推荐阅读