python smtplib模块发送SSL/TLS安全邮件实例


Posted in Python onApril 08, 2015

python的smtplib提供了一种很方便的途径发送电子邮件。它对smtp协议进行了简单的封装。

smtp协议的基本命令包括:

HELO 向服务器标识用户身份
MAIL 初始化邮件传输 mail from:
RCPT 标识单个的邮件接收人;常在MAIL命令后面,可有多个rcpt to:
DATA 在单个或多个RCPT命令后,表示所有的邮件接收人已标识,并初始化数据传输,以.结束
VRFY 用于验证指定的用户/邮箱是否存在;由于安全方面的原因,服务器常禁止此命令
EXPN 验证给定的邮箱列表是否存在,扩充邮箱列表,也常被禁用
HELP 查询服务器支持什么命令
NOOP 无操作,服务器应响应OK
QUIT 结束会话
RSET 重置会话,当前传输被取消
MAIL FROM 指定发送者地址
RCPT TO 指明的接收者地址

一般smtp会话有两种方式,一种是邮件直接投递,就是说,比如你要发邮件?zzz@163.com,那就直接连接163.com的邮件服务器,把信投?zzz@163.com; 另一种是验证过后的发信,它的过程是,比如你要发邮件?zzz@163.com,你不是直接投到163.com,而是通过自己在sina.com的另一个邮箱来发。这样就要先连接sina.com的smtp服务器,然后认证,之后在把要发到163.com的信件投到sina.com上,sina.com会帮你把信投递到163.com。

第一种方式的命令流程基本是这样:
1. helo
2. mail from
3. rcpt to
4. data
5. quit

但是第一种发送方式一般有限制的,就是rcpt to指定的这个邮件接收者必须在这个服务器上存在,否则是不会接收的。 先看看代码:

#-*- encoding: gb2312 -*-

import os, sys, string

import smtplib
# 邮件服务器地址

mailserver = "smtp.163.com"

# smtp会话过程中的mail from地址

from_addr = "asfgysg@zxsdf.com"

# smtp会话过程中的rcpt to地址

to_addr = "zhaoweikid@163.com"

# 信件内容

msg = "test mail"
svr = smtplib.SMTP(mailserver)

# 设置为调试模式,就是在会话过程中会有输出信息

svr.set_debuglevel(1)

# helo命令,docmd方法包括了获取对方服务器返回信息

svr.docmd("HELO server")

# mail from, 发送邮件发送者

svr.docmd("MAIL FROM: <%s>" % from_addr)

# rcpt to, 邮件接收者

svr.docmd("RCPT TO: <%s>" % to_addr)

# data命令,开始发送数据

svr.docmd("DATA")

# 发送正文数据

svr.send(msg)

# 比如以 . 作为正文发送结束的标记,用send发送的,所以要用getreply获取返回信息

svr.send(" . ")

svr.getreply()

# 发送结束,退出

svr.quit()

注意的是,163.com是有反垃圾邮件功能的,想上面的这种投递邮件的方法不一定能通过反垃圾邮件系统的检测的。所以一般不推荐个人这样发送。

 第二种有点不一样:

1.ehlo
2.auth login
3.mail from
4.rcpt to
5.data
6.quit

相对于第一种来说,多了一个认证过程,就是auth login这个过程。

#-*- encoding: gb2312 -*-

import os, sys, string

import smtplib

import base64
# 邮件服务器地址

mailserver = "smtp.163.com"

# 邮件用户名

username = "xxxxxx@163.com"

# 密码

password = "xxxxxxx"

# smtp会话过程中的mail from地址

from_addr = "xxxxxx@163.com"

# smtp会话过程中的rcpt to地址

to_addr = "yyyyyy@163.com"

# 信件内容

msg = "my test mail"
svr = smtplib.SMTP(mailserver)

# 设置为调试模式,就是在会话过程中会有输出信息

svr.set_debuglevel(1)

# ehlo命令,docmd方法包括了获取对方服务器返回信息

svr.docmd("EHLO server")

# auth login 命令

svr.docmd("AUTH LOGIN")

# 发送用户名,是base64编码过的,用send发送的,所以要用getreply获取返回信息

svr.send(base64.encodestring(username))

svr.getreply()

# 发送密码

svr.send(base64.encodestring(password))

svr.getreply()

# mail from, 发送邮件发送者

svr.docmd("MAIL FROM: <%s>" % from_addr)

# rcpt to, 邮件接收者

svr.docmd("RCPT TO: <%s>" % to_addr)

# data命令,开始发送数据

svr.docmd("DATA")

# 发送正文数据

svr.send(msg)

# 比如以 . 作为正文发送结束的标记

svr.send(" . ")

svr.getreply()

# 发送结束,退出

svr.quit()
   
上面说的是最普通的情况,但是不能忽略的是现在好多企业邮件是支持安全邮件的,就是通过SSL发送的邮件,这个怎么发呢?SMTP对SSL安全邮件的支持有两种方案,一种老的是专门开启一个465端口来接收ssl邮件,另一种更新的做法是在标准的25端口的smtp上增加一个starttls的命令来支持。

看看第一种怎么办:

#-*- encoding: gb2312 -*-

import os, sys, string, socket

import smtplib


class SMTP_SSL (smtplib.SMTP):

    def __init__(self, host='', port=465, local_hostname=None, key=None, cert=None):

        self.cert = cert

        self.key = key

        smtplib.SMTP.__init__(self, host, port, local_hostname)

        

    def connect(self, host='localhost', port=465):

        if not port and (host.find(':') == host.rfind(':')):

            i = host.rfind(':')

            if i >= 0:

                host, port = host[:i], host[i+1:]

                try: port = int(port)

                except ValueError:

                    raise socket.error, "nonnumeric port"

        if not port: port = 654

        if self.debuglevel > 0: print>>stderr, 'connect:', (host, port)

        msg = "getaddrinfo returns an empty list"

        self.sock = None

        for res in socket.getaddrinfo(host, port, 0, socket.SOCK_STREAM):

            af, socktype, proto, canonname, sa = res

            try:

                self.sock = socket.socket(af, socktype, proto)

                if self.debuglevel > 0: print>>stderr, 'connect:', (host, port)

                self.sock.connect(sa)

                # 新增加的创建ssl连接

                sslobj = socket.ssl(self.sock, self.key, self.cert)

            except socket.error, msg:

                if self.debuglevel > 0: 

                    print>>stderr, 'connect fail:', (host, port)

                if self.sock:

                    self.sock.close()

                self.sock = None

                continue

            break

        if not self.sock:

            raise socket.error, msg
        # 设置ssl

        self.sock = smtplib.SSLFakeSocket(self.sock, sslobj)

        self.file = smtplib.SSLFakeFile(sslobj);
        (code, msg) = self.getreply()

        if self.debuglevel > 0: print>>stderr, "connect:", msg

        return (code, msg)

        

if __name__ == '__main__':

    smtp = SMTP_SSL('192.168.2.10')

    smtp.set_debuglevel(1)

    smtp.sendmail("zzz@xxx.com", "zhaowei@zhaowei.com", "xxxxxxxxxxxxxxxxx")

    smtp.quit()
   

这里我是从原来的smtplib.SMTP派生出了新的SMTP_SSL类,它专门来处理ssl连接。我这里测试的192.168.2.10是我自己的测试服务器.

第二种是新增加了starttls的命令,这个很简单,smtplib里就有这个方法,叫smtplib.starttls()。当然,不是所有的邮件系统都支持安全邮件的,这个需要从ehlo的返回值里来确认,如果里面有starttls,才表示支持。相对于发送普通邮件的第二种方法来说,只需要新增加一行代码就可以了:

#-*- encoding: gb2312 -*-

import os, sys, string

import smtplib

import base64
# 邮件服务器地址

mailserver = "smtp.163.com"

# 邮件用户名

username = "xxxxxx@163.com"

# 密码

password = "xxxxxxx"

# smtp会话过程中的mail from地址

from_addr = "xxxxxx@163.com"

# smtp会话过程中的rcpt to地址

to_addr = "yyyyyy@163.com"

# 信件内容

msg = "my test mail"
svr = smtplib.SMTP(mailserver)

# 设置为调试模式,就是在会话过程中会有输出信息

svr.set_debuglevel(1)

# ehlo命令,docmd方法包括了获取对方服务器返回信息,如果支持安全邮件,返回值里会有starttls提示

svr.docmd("EHLO server")

svr.starttls()  # <------ 这行就是新加的支持安全邮件的代码!

# auth login 命令

svr.docmd("AUTH LOGIN")

# 发送用户名,是base64编码过的,用send发送的,所以要用getreply获取返回信息

svr.send(base64.encodestring(username))

svr.getreply()

# 发送密码

svr.send(base64.encodestring(password))

svr.getreply()

# mail from, 发送邮件发送者

svr.docmd("MAIL FROM: <%s>" % from_addr)

# rcpt to, 邮件接收者

svr.docmd("RCPT TO: <%s>" % to_addr)

# data命令,开始发送数据

svr.docmd("DATA")

# 发送正文数据

svr.send(msg)

# 比如以 . 作为正文发送结束的标记

svr.send(" . ")

svr.getreply()

# 发送结束,退出

svr.quit()

注意: 以上的代码为了方便我都没有判断返回值,严格说来,是应该判断一下返回的代码的,在smtp协议中,只有返回代码是2xx或者3xx才能继续下一步,返回4xx或5xx的,都是出错了。
Python 相关文章推荐
Linux 下 Python 实现按任意键退出的实现方法
Sep 25 Python
numpy使用技巧之数组过滤实例代码
Feb 03 Python
Pipenv一键搭建python虚拟环境的方法
May 22 Python
python 定义n个变量方法 (变量声明自动化)
Nov 10 Python
对Python3.x版本print函数左右对齐详解
Dec 22 Python
Django中create和save方法的不同
Aug 13 Python
Python实现线性判别分析(LDA)的MATLAB方式
Dec 09 Python
pytorch中nn.Conv1d的用法详解
Dec 31 Python
Python 实现将某一列设置为str类型
Jul 14 Python
python在一个范围内取随机数的简单实例
Aug 16 Python
python3代码中实现加法重载的实例
Dec 03 Python
Python卷积神经网络图片分类框架详解分析
Nov 07 Python
python复制与引用用法分析
Apr 08 #Python
Python导入txt数据到mysql的方法
Apr 08 #Python
python集合类型用法分析
Apr 08 #Python
在Python中使用Mako模版库的简单教程
Apr 08 #Python
python中requests模块的使用方法
Apr 08 #Python
介绍Python中几个常用的类方法
Apr 08 #Python
python自然语言编码转换模块codecs介绍
Apr 08 #Python
You might like
php 安全过滤函数代码
2011/05/07 PHP
php二维数组排序与默认自然排序的方法介绍
2013/04/27 PHP
windows7下安装php的imagick和imagemagick扩展教程
2014/07/04 PHP
destoon调用企业会员公司形象图片的实现方法
2014/08/21 PHP
PHP生成图片缩略图类示例
2017/01/12 PHP
php微信公众号开发之微信企业付款给个人
2018/10/04 PHP
PHP实现给定一列字符,生成指定长度的所有可能组合示例
2019/06/22 PHP
JSON 和 JavaScript eval使用说明
2010/06/13 Javascript
Google AJAX 搜索 API实现代码
2010/11/17 Javascript
jQuery中将函数赋值给变量的调用方法
2012/03/23 Javascript
JavaScript比较两个对象是否相等的方法
2015/02/06 Javascript
JavaScript判断数组是否包含指定元素的方法
2015/07/01 Javascript
JS检测移动端横竖屏的代码
2016/05/30 Javascript
AngularJS过滤器详解及示例代码
2016/08/16 Javascript
Bootstrap复选框和单选按钮美化插件(推荐)
2016/11/23 Javascript
用headjs来管理和加载js 提高网站加载速度
2016/11/29 Javascript
使用jQuery和ajax代替iframe的方法(详解)
2017/04/12 jQuery
vue中进入详情页记住滚动位置的方法(keep-alive)
2018/09/21 Javascript
react使用antd表单赋值,用于修改弹框的操作
2020/10/29 Javascript
Python中的自省(反射)详解
2015/06/02 Python
Python+matplotlib+numpy实现在不同平面的二维条形图
2018/01/02 Python
python中正则表达式与模式匹配
2019/05/07 Python
pip指定python位置安装软件包的方法
2019/07/12 Python
关于python中plt.hist参数的使用详解
2019/11/28 Python
Python中flatten( ),matrix.A用法说明
2020/07/05 Python
python re的findall和finditer的区别详解
2020/11/15 Python
css3 利用transform打造走动的2D时钟
2020/10/20 HTML / CSS
Paradigit比利时电脑卖场:购买笔记本、电脑、平板和外围设备
2016/11/28 全球购物
100%法国制造的游戏和玩具:Les Jouets Français
2021/03/02 全球购物
敬老院献爱心活动总结
2014/07/08 职场文书
2014企业年终工作总结
2014/12/23 职场文书
党课主持词大全
2015/06/30 职场文书
2016年第十四个公民道德宣传日活动总
2016/04/01 职场文书
感谢信
2019/04/11 职场文书
教你漂亮打印Pandas DataFrames和Series
2021/05/29 Python
SpringBoot+Vue+JWT的前后端分离登录认证详细步骤
2021/09/25 Java/Android