下载 smtplib.py smtplib.py

下载 email-6.0.0a1.tar.gz Mail-2.1.0.tar.gz

安装:

$ tar zxvf email-6.0.0a1.tar.gz
$ cd email-6.0.0a1
$ sudo python setup.py build
$ sudo python setup.py install

 

使用Python的email模块来实现带有附件的邮件的发送。

SMTP (Simple Mail Transfer Protocol) 协议只能用来发送邮件,不能用来接收邮件。

MTA (Mail Transfer Agent) 邮件传送代理程序使用SMTP协议来发送电邮到接收者的邮件服务器。

大多数的邮件发送服务器 (Outgoing Mail Server) 都是使用SMTP协议,SMTP协议的默认TCP端口号是25,加密 SSL 端口号是 465 或 587

SMTP协议的一个重要特点是它能够接力传送邮件,它工作在两种情况下:

一是电子邮件从客户机传输到服务器;

二是从某一个服务器传输到另一个服务器。

 

POP3 (Post Office Protocol) & IMAP (Internet Message Access Protocol)

POP协议和IMAP协议是用于邮件接收的最常见的两种协议,应用于邮箱客户端,如 foxmail、邮箱大师等,几乎所有的邮件客户端和服务器都支持这两种协议。

POP3协议为用户提供了一种简单、标准的方式来访问邮箱和获取电邮。使用POP3协议的电邮客户端通常的工作过程是:连接服务器、获取所有信息并保存在用户主机、从服务器删除这些消息然后断开连接。 POP3协议的默认TCP端口号是110,加密 SSL 端口号为 995

IMAP协议也提供了方便的邮件下载服务,让用户能进行离线阅读。使用IMAP协议的电邮客户端通常把信息保留在服务器上直到用户显式删除。这种特性使得多个客户端可以同时管理一个邮箱。IMAP协议提供了摘要浏览功能,可以让用户在阅读完所有的邮件到达时间、主题、发件人、大小等信息后再决定是否下载。 IMAP协议的默认TCP端口号是143

 

邮件格式 ( RFC 2822 )

每封邮件都有两个部分:邮件头(head)和邮件体(body),两者使用一个空行分隔。

邮件头每个字段 (Field) 包括两部分:字段名和字段值,两者使用冒号分隔。

有两个字段需要注意:From和Sender字段。From字段指明的是邮件的作者,Sender字段指明的是邮件的发送者。如果From字段包含多于一个的作者,必须指定Sender字段;如果From字段只有一个作者并且作者和发送者相同,那么不应该再使用Sender字段,否则From字段和Sender字段应该同时使用。

邮件体包含邮件的内容,它的类型由邮件头的Content-Type字段指明。

RFC 2822定义的邮件格式中,邮件体只是单纯的ASCII编码的字符序列。

 

MIME (Multipurpose Internet Mail Extensions) ( RFC 1341 )

MIME扩展邮件的格式,用以支持非ASCII编码的文本、非文本附件以及包含多个部分 (multi-part) 的邮件体等。

 

Python email模块
1. class email.message.Message

  • __getitem__,__setitem__实现obj[key]形式的访问。
  • Msg.attach(playload): 向当前Msg添加playload。
  • Msg.set_playload(playload): 把整个Msg对象的邮件体设成playload。
  • Msg.add_header(_name, _value, **_params): 添加邮件头字段。

2. class email.mime.base.MIMEBase(_maintype, _subtype, **_params)  

所有MIME类的基类,是email.message.Message类的子类。

3. class email.mime.multipart.MIMEMultipart()

在3.0版本的email模块 (Python 2.3-Python 2.5) 中,这个类位于email.MIMEMultipart.MIMEMultipart。

这个类是MIMEBase的直接子类,用来生成包含多个部分的邮件体的MIME对象。

4. class email.mime.text.MIMEText(_text)

使用字符串_text来生成MIME对象的主体文本。

 

smtplib模块简介

smtplib.SMTP([host[, port[, local_hostname[, timeout]]]])
smtplib.SMTP_SSL([host[, port[, local_hostname[, keyfile[, certfile[, timeout]]]]]])
smtplib.LMTP([host[, port[, local_hostname]]])

此为SMTP类构造函数,表示与SMTP服务器之间的连接,并根据这个连接向smtp服务器发送指令,执行相关操作(如:登陆、发送邮件),且每个参数都是可选的

其中最重要的参数:

host:smtp服务器主机名,例如 smtp.163.com,smtp.qq.com,smtp.exmail.qq.com 

port:smtp服务的端口,默认是25,ssl是465

如果在创建SMTP对象的时候提供了这两个参数,在初始化的时候会自动调用connect方法去连接服务器。

SMTP_SSL 是派生自SMTP的子类,通过SSL加密的套接字连接(使用此类,您需要使用SSL支持编译的套接字模块openssl)。如果未指定主机,则使用“(本地主机)”。如果省略端口,则使用标准的SMTP-over-SSL端口(465)。 local_hostname和source_address与SMTP类中的含义相同。密钥文件和证书文件也是可选的 - 它们可以包含用于SSL连接的PEM格式的私钥和证书链文件。上下文也是可选的,可以包含SSLContext,是keyfile和certfile的替代方法;如果指定keyfile和certfile都必须为None。

 

smtplib.SMTP 还提供了如下方法:

SMTP.set_debuglevel(level): 设置是否为调试模式,1 - 是, 0 - 不是

SMTP.connect([host[, port]]): 连接到指定的smtp服务器。参数分别表示 smpt主机和端口。

SMTP.docmd(cmd[, argstring]): 向smtp服务器发送指令。

SMTP.helo([hostname]) : 使用"helo"指令向服务器确认身份。

SMTP.login(user, password): 登陆到smtp服务器。现在几乎所有smtp服务器,都必须在验证用户信息合法之后才允许发送邮件。

SMTP.sendmail(from_addr,to_addrs,msg[,mail_options,rcpt_options]): 发送邮件。这里要注意一下第三个参数,msg是字符串,表示邮件。我们知道邮件一般由标题,发信人,收件人,邮件内容,附件等构成,发送邮件的时候,要注意msg的格式。这个格式就是smtp协议中定义的格式。SMTP.quit() :断开与smtp服务器的连接,相当于发送"quit"指令。(重要!)

 

smtplib.SMTP  已经定义好的异常

exception smtplib.SMTPException
exception smtplib.SMTPServerDisconnected
exception smtplib.SMTPResponseException
exception smtplib.SMTPSenderRefused
exception smtplib.SMTPRecipientsRefused
exception smtplib.SMTPDataError
exception smtplib.SMTPConnectError
exception smtplib.SMTPHeloError
exception smtplib.SMTPAuthenticationError

这么多已定义的类中,最常用的的还是smtplib.SMTP类用法:smtp实例封装一个smtp连接,它支持所有的SMTP和ESMTP操作指令,如果host和port参数被定义,则smtp会在初始化期间自动调用connect()方法,如果connect()方法失败,则会触发SMTPConnectError异常,timeout参数设置了超时时间。在一般的调用过程中,应该遵connetc()、sendmail()、quit()步骤。

 

email 模块

1. class email.message.Message

__getitem__,__setitem__ 实现obj[key]形式的访问。

Msg.attach(playload): 向当前Msg添加playload。

Msg.set_playload(playload):

Msg.add_header(_name, _value, **_params): 添加邮件头字段。

 

2. class email.mime.base.MIMEBase(_maintype, _subtype, **_params)

所有MIME类的基类,是email.message.Message类的子类。

 

3. class email.mime.multipart.MIMEMultipart()

在3.0版本的email模块 (Python 2.3-Python 2.5) 中,这个类位于email.MIMEMultipart.MIMEMultipart。

这个类是MIMEBase的直接子类,用来生成包含多个部分的邮件体的MIME对象。

 

4. class email.mime.text.MIMEText(_text)

使用字符串_text来生成MIME对象的主体文本。

获得所需要使用的邮箱的host地址和port端口号,常用邮箱的smtp服务器地址和端口号如图:

 

简单示例 1

vim send_email.py

#!/usr/bin/env python
# -*- coding:utf-8 -*-
#
# author: mimvp.com
# 2012.01.05

import smtplib
from email.mime.text import MIMEText


import sys
reload(sys)
sys.setdefaultencoding('utf-8')


msg_from = 'yanggang@163.com'      		# 发送方邮箱
passwd = 'password-auth-code'         	# 填入发送方邮箱的授权码
msg_to = 'yanggang@mimvp.com'           # 收件人邮箱
mail_host="smtp.163.com"                # 发送人邮件的stmp服务器
port= 465                               # 端口号

subject = "Python 测试 smtps ssl"       	# 主题(标题要有中文,否则容易被当做垃圾邮件,拒发退回)
content = "hello mimvp.com"             # 正文
msg = MIMEText(content)
msg['Subject'] = subject
msg['From'] = msg_from
msg['To'] = msg_to

# 创建连接对象并连接到服务器
s = smtplib.SMTP_SSL(mail_host, port)   # 邮件服务器及端口号,SMTP_SSL默认使用465端口号,端口号可省略

# 登录服务器
s.login(msg_from, passwd)
try:
    s.sendmail(msg_from, msg_to, msg.as_string())
    print("send success")
except s.SMTPException as e:
    print(e)
    print("send failed")
finally:
    s.quit()

执行发送命令:

# python send_email.py 
send success

成功发送的邮件结果:

 

简单示例 2

vim send_email2.py

#!/usr/bin/env python
# -*- coding:utf-8 -*-
#
# author: mimvp.com
# 2012.01.05

import smtplib
from email.header import Header
from email.mime.text import MIMEText

import sys
reload(sys)
sys.setdefaultencoding('utf-8')


mail_info = {
    "from": "robot@mimvp.com",
    "to": "yanggang@163.com",
    "hostname": "smtp.exmail.qq.com",
    "username": "robot@mimvp.com",
    "password": "mimvp-password",
    "mail_subject": "mimvp-subject",
    "mail_text": "hello mimvp.com",
    "mail_encoding": "utf-8"
}

if __name__ == '__main__':
    # 这里使用SMTP_SSL就是默认使用465端口
    smtp = smtplib.SMTP_SSL(mail_info["hostname"])
    smtp.set_debuglevel(1)
    
    smtp.ehlo(mail_info["hostname"])
    smtp.login(mail_info["username"], mail_info["password"])

    msg = MIMEText(mail_info["mail_text"], "plain", mail_info["mail_encoding"])
    msg["Subject"] = Header(mail_info["mail_subject"], mail_info["mail_encoding"])
    msg["from"] = mail_info["from"]
    msg["to"] = mail_info["to"]
    
    smtp.sendmail(mail_info["from"], mail_info["to"], msg.as_string())

    smtp.quit()

成功发送的邮件结果:

 

代码实现

#!/usr/bin/env python
# -*- coding:utf-8 -*-
# copyright of mimvp.com
 
import os
import smtplib 
from email.mime.multipart import MIMEMultipart
from email.mime.base import MIMEBase
from email.mime.text import MIMEText
from email.utils import formatdate
from email import encoders
 

class Mail(object):
    
    MAIL_SERVER = {
                      'server_smtp' : 'smtp.mimvp.com',
                      'from' : 'yang@mimvp.com',
                      'to' : ['463153470@qq.com', 'yanggang@163.com', 'yanggang@mi.com'],
                      'user_name' : 'yang@mimvp.com',
                      'user_pass' : '123456'
                  }    
    
    @classmethod
    def send_mail(cls, subject='', body='', attaches=[]): 
        mail_server = Mail.MAIL_SERVER.get('server_smtp')
        mail_from = Mail.MAIL_SERVER.get('from') 
        mail_to = Mail.MAIL_SERVER.get('to')
        mail_to_str = ";".join(Mail.MAIL_SERVER.get('to'))
        user_name = Mail.MAIL_SERVER.get('user_name')
        user_pass = Mail.MAIL_SERVER.get('user_pass')
        mail_body = body
     
        msg = MIMEMultipart() 
        msg['Subject'] = subject 
        msg['From'] = mail_from
        msg['To'] = mail_to_str
        msg['Date'] = formatdate(localtime=True) 
        msg.attach(MIMEText(mail_body)) 
     
        for attach in attaches: 
            part = MIMEBase('application', 'octet-stream') 
            part.set_payload(open(attach, 'rb').read()) 
            encoders.encode_base64(part) 
            part.add_header('Content-Disposition', 'attachment; filename="%s"' % os.path.basename(attach)) 
            msg.attach(part) 
    
        try:
            smtp = smtplib.SMTP(mail_server) 
            smtp.login(user_name, user_pass) 
            smtp.sendmail(mail_from, mail_to, msg.as_string()) 
            smtp.close()
            print("send email success!")
        except Exception as ex:
            print("send email fail. error: " + str(ex))
        
if __name__ == "__main__":
    
    attaches = []
    for i in range(5):
        i += 1
        filename = "file_" + str(i)
        f = open(filename, 'w+')
        f.write("i am ithomer.net " + str(i))
        f.close()
        attaches.append(filename)
    print attaches
    #files.append("getUrl.py")
    #files.append("getUrl.py")
    
    Mail.send_mail('test python email', 'body of www.mimvp.com', attaches)

 

运行结果:

$ python email_test.py
['file_0', 'file_1', 'file_2', 'file_3', 'file_4']
send email success!


发送的邮件内容:

无法import的原因(ImportError: No module named *****)

python中,每个py文件被称之为模块,每个具有__init__.py文件的目录被称为包。只要模块或者包所在的目录在sys.path中,就可以使用import 模块或import 包来使用。

如果想使用非当前模块中的代码,需要使用Import,这个大家都知道。

如果你要使用的模块(py文件)和当前模块在同一目录,只要import相应的文件名就好,比如在a.py中使用b.py:

import b

但是如果要import一个不同目录的文件(例如b.py)该怎么做呢?

首先需要使用sys.path.append方法将b.py所在目录加入到搜素目录中。然后进行import即可,例如

import sys

sys.path.append('c:\xxxx\b.py')

大多数情况,上面的代码工作的很好。但是如果你没有发现上面代码有什么问题的话,可要注意了,上面的代码有时会找不到模块或者包(ImportError: No module named xxxxxx),这是因为:

sys模块是使用c语言编写的,因此字符串支持 '\n', '\r', '\t'等来表示特殊字符。所以上面代码最好写成:

sys.path.append('c:\\xxx\\b.py')

sys.path.append('c:/xxxx/b.py')

这样可以避免因为错误的组成转义字符,而造成无效的搜索目录(sys.path)设置。

 

 

参考推荐

python发送邮件

使用python偷偷发邮件

email: Examples (官方示例)

利用Python实现邮件的发送

python 发送各类邮件的主要方法

smtplib与email模块(实现邮件的发送)

python 七种邮件内容发送方法实例 推荐

Centos7 使用 sendmail 发送邮件

POP3、SMTP、IMAP 的作用和联系

 

原文: Python学习入门(5)——发送邮件