python接口自动化(三十三)-python自动发邮件总结及实例说明番外篇下(详解)

简介

  发邮件前我们需要了解的是邮件是怎么一个形式去发送到对方手上的,通俗点来说就是你写好一封信,然后装进信封,写上地址,贴上邮票,然后就近找个邮局,把信仍进去,其他的就不关心了,只是关心时间,而电子邮件不像日常发送邮件的按天算,时间都是按

秒算的。

电子邮件的发送流程:

1、你使用某款电子邮件软件MUA:mail user agent --邮件用户代理,填写完成点击发送

2、在你点击发送的时候电子邮件软件发出去,到MTA:mail transfer agent--邮件传输代理,即email服务提供商,如网易等

3、MTA--邮箱传输代理会把邮箱投递到邮件的最终目的地MDA:mail delivery agent---邮箱投递服务

4、email到达MDA后就会放在某个服务器上,存在数据库里,收件人必须通过MUA从MDA中把邮箱放到自己电脑上

发件人 -> MUA -> MTA -> MTA -> 若干个MTA -> MDA<-MUA<-收件人

python发邮件需要掌握两个模块的用法,smtplib和email,这俩模块是python自带的,只需import即可使用。smtplib模块主要负责发送邮件,email模块主要负责构造邮件。

smtplib模块主要负责发送邮件:是一个发送邮件的动作,连接邮箱服务器,登录邮箱,发送邮件(有发件人,收信人,邮件内容)。

email模块主要负责构造邮件:指的是邮箱页面显示的一些构造,如发件人,收件人,主题,正文,附件等。

1.smtplib模块

smtplib使用较为简单。以下是最基本的语法。

导入及使用方法如下:

1 import smtplib
2
3 smtp = smtplib.SMTP()
4 smtp.connect(‘smtp.163.com,25‘)
5 smtp.login(username, password)
6 smtp.sendmail(sender, receiver, msg.as_string())
7 smtp.quit()

说明:

smtplib.SMTP():实例化SMTP()

connect(host,port):

host:指定连接的邮箱服务器。常用邮箱的smtp服务器地址如下:

新浪邮箱:smtp.sina.com,新浪VIP:smtp.vip.sina.com,搜狐邮箱:smtp.sohu.com,126邮箱:smtp.126.com,139邮箱:smtp.139.com,163网易邮箱:smtp.163.com。

port:指定连接服务器的端口号,默认为25.

login(user,password):

user:登录邮箱的用户名。

password:登录邮箱的密码,像笔者用的是网易邮箱,网易邮箱一般是网页版,需要用到客户端密码,需要在网页版的网易邮箱中设置授权码,该授权码即为客户端密码。

sendmail(from_addr,to_addrs,msg,...):

from_addr:邮件发送者地址

to_addrs:邮件接收者地址。发送邮件,多人时用list,字符串列表[‘接收地址1‘,‘接收地址2‘,‘接收地址3‘,...]或‘接收地址‘

msg:发送消息:邮件内容。邮件正文是一个str,一般是msg.as_string():as_string()是将msg(MIMEText对象或者MIMEMultipart对象)变为str。

quit(): 退出关闭邮箱,用于结束SMTP会话。

2.email模块

email模块下有mime包,mime英文全称为“Multipurpose Internet Mail Extensions”,即多用途互联网邮件扩展,是目前互联网电子邮件普遍遵循的邮件技术规范。

该mime包下常用的有三个模块:text,image,multpart

导入方法如下:

from email.mime.multipart import MIMEMultipart
from email.mime.text import MIMEText
from email.mime.image import MIMEImage

构造一个邮件对象就是一个Message对象,如果构造一个MIMEText对象,就表示一个文本邮件对象,如果构造一个MIMEImage对象,就表示一个作为附件的图片,要把多个对象组合起来,就用MIMEMultipart对象,而MIMEBase可以表示任何对象。它们的继承关系如下:

Message
+- MIMEBase
   +- MIMEMultipart
   +- MIMENonMultipart
      +- MIMEMessage
      +- MIMEText
      +- MIMEImage

2.1 text说明

邮件发送程序为了防止有些邮件阅读软件不能显示处理HTML格式的数据,通常都会用两类型分别为"text/plain"和"text/html"

构造MIMEText对象时,第一个参数是邮件正文,第二个参数是MIME的subtype,最后一定要用utf-8编码保证多语言兼容性。

2.1.1添加普通文本

方法很简单,在构造MIMEText对象时,把TEXT字符串传进去,再把第二个参数plain和第三个参数编码传进去即可。plain表示纯文本 ,后面的则是编译,保证多语言兼容

text = "Hi!\nHow are you?\nHere is the link you wanted:\nhttps://www.cnblogs.com/du-hong"
text_plain = MIMEText(text,‘plain‘, ‘utf-8‘)    

查看MIMEText属性:可以观察到MIMEText,MIMEImage和MIMEMultipart的属性都一样。

print dir(text_plain)

[‘__contains__‘, ‘__delitem__‘, ‘__doc__‘, ‘__getitem__‘, ‘__init__‘, ‘__len__‘, ‘__module__‘, ‘__setitem__‘, ‘__str__‘, ‘_charset‘, ‘_default_type‘, ‘_get_params_preserve‘, ‘_headers‘, ‘_payload‘, ‘_unixfrom‘, ‘add_header‘, ‘as_string‘, ‘attach‘, ‘defects‘, ‘del_param‘, ‘epilogue‘, ‘get‘, ‘get_all‘, ‘get_boundary‘, ‘get_charset‘, ‘get_charsets‘, ‘get_content_charset‘, ‘get_content_maintype‘, ‘get_content_subtype‘, ‘get_content_type‘, ‘get_default_type‘, ‘get_filename‘, ‘get_param‘, ‘get_params‘, ‘get_payload‘, ‘get_unixfrom‘, ‘has_key‘, ‘is_multipart‘, ‘items‘, ‘keys‘, ‘preamble‘, ‘replace_header‘, ‘set_boundary‘, ‘set_charset‘, ‘set_default_type‘, ‘set_param‘, ‘set_payload‘, ‘set_type‘, ‘set_unixfrom‘, ‘values‘, ‘walk‘]

2.1.2添加超文本(HTML)

方法也很简单,在构造MIMEText对象时,把HTML字符串传进去,再把第二个参数由plain变为html就可以了

 1 html = """
 2 <html>
 3   <body>
 4     <p>
 5        Here is the <a href="https://www.cnblogs.com/du-hong">link</a> you wanted.
 6     </p>
 7   </body>
 8 </html>
 9 """
10 text_html = MIMEText(html,‘html‘, ‘utf-8‘)  

2.1.3添加附件

首先要创建MIMEMultipart()实例,然后构造附件,如果有多个附件,可依次构造,最后利用smtplib.smtp发送

sendfile=open(r‘D:\pythontest\text.txt‘,‘rb‘).read()
text_att = MIMEText(sendfile, ‘base64‘, ‘utf-8‘)
text_att["Content-Type"] = ‘application/octet-stream‘
text_att["Content-Disposition"] = ‘attachment; filename="显示的名字.txt"‘  

2.2 image说明

添加图片:

sendimagefile=open(r‘D:\pythontest\testimage.png‘,‘rb‘).read()
image = MIMEImage(sendimagefile)
image.add_header(‘Content-ID‘,‘<image1>‘)

查看MIMEImage属性:

print dir(image)

[‘__contains__‘, ‘__delitem__‘, ‘__doc__‘, ‘__getitem__‘, ‘__init__‘, ‘__len__‘, ‘__module__‘, ‘__setitem__‘, ‘__str__‘, ‘_charset‘, ‘_default_type‘, ‘_get_params_preserve‘, ‘_headers‘, ‘_payload‘, ‘_unixfrom‘, ‘add_header‘, ‘as_string‘, ‘attach‘, ‘defects‘, ‘del_param‘, ‘epilogue‘, ‘get‘, ‘get_all‘, ‘get_boundary‘, ‘get_charset‘, ‘get_charsets‘, ‘get_content_charset‘, ‘get_content_maintype‘, ‘get_content_subtype‘, ‘get_content_type‘, ‘get_default_type‘, ‘get_filename‘, ‘get_param‘, ‘get_params‘, ‘get_payload‘, ‘get_unixfrom‘, ‘has_key‘, ‘is_multipart‘, ‘items‘, ‘keys‘, ‘preamble‘, ‘replace_header‘, ‘set_boundary‘, ‘set_charset‘, ‘set_default_type‘, ‘set_param‘, ‘set_payload‘, ‘set_type‘, ‘set_unixfrom‘, ‘values‘, ‘walk‘]

2.3 multpart说明

常见的multipart类型有三种:multipart/alternative, multipart/related和multipart/mixed。

邮件类型为"multipart/alternative"的邮件包括纯文本正文(text/plain)和超文本正文(text/html)。

邮件类型为"multipart/related"的邮件正文中包括图片,声音等内嵌资源。

邮件类型为"multipart/mixed"的邮件包含附件。向上兼容,如果一个邮件有纯文本正文,超文本正文,内嵌资源,附件,则选择mixed类型。

msg = MIMEMultipart(‘mixed‘)

我们必须把Subject,From,To,Date添加到MIMEText对象或者MIMEMultipart对象中,邮件中才会显示主题,发件人,收件人,时间(若无时间,就默认一般为当前时间,该值一般不设置)。

msg = MIMEMultipart(‘mixed‘)
msg[‘Subject‘] = ‘Python email test‘
msg[‘From‘] = ‘[email protected] <[email protected]>‘
msg[‘To‘] = ‘[email protected]‘
msg[‘Date‘]=‘2019-5-10‘

查看MIMEMultipart属性:

msg = MIMEMultipart(‘mixed‘)
print dir(msg) 

结果:

[‘__contains__‘, ‘__delitem__‘, ‘__doc__‘, ‘__getitem__‘, ‘__init__‘, ‘__len__‘, ‘__module__‘, ‘__setitem__‘, ‘__str__‘, ‘_charset‘, ‘_default_type‘, ‘_get_params_preserve‘, ‘_headers‘, ‘_payload‘, ‘_unixfrom‘, ‘add_header‘, ‘as_string‘, ‘attach‘, ‘defects‘, ‘del_param‘, ‘epilogue‘, ‘get‘, ‘get_all‘, ‘get_boundary‘, ‘get_charset‘, ‘get_charsets‘, ‘get_content_charset‘, ‘get_content_maintype‘, ‘get_content_subtype‘, ‘get_content_type‘, ‘get_default_type‘, ‘get_filename‘, ‘get_param‘, ‘get_params‘, ‘get_payload‘, ‘get_unixfrom‘, ‘has_key‘, ‘is_multipart‘, ‘items‘, ‘keys‘, ‘preamble‘, ‘replace_header‘, ‘set_boundary‘, ‘set_charset‘, ‘set_default_type‘, ‘set_param‘, ‘set_payload‘, ‘set_type‘, ‘set_unixfrom‘, ‘values‘, ‘walk‘]

说明:

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

msg.as_string():是将msg(MIMEText对象或者MIMEMultipart对象)变为str,如果只有一个html超文本正文或者plain普通文本正文的话,一般msg的类型可以是MIMEText;如果是多个的话,就都添加到MIMEMultipart,msg类型就变为MIMEMultipart。

msg.attach(MIMEText对象或MIMEImage对象):将MIMEText对象或MIMEImage对象添加到MIMEMultipart对象中。MIMEMultipart对象代表邮件本身,MIMEText对象或MIMEImage对象代表邮件正文。

以上的构造的文本,超文本,附件,图片都何以添加到MIMEMultipart(‘mixed‘)中:

msg.attach(text_plain)
msg.attach(text_html)
msg.attach(text_att)
msg.attach(image)

3.文字,html,图片,附件实现实例

3.1实例

3.2运行结果

3.3参考代码

 1 # coding=utf-8
 2 #1.先设置编码,utf-8可支持中英文,如上,一般放在第一行
 3
 4 #2.注释:包括记录创建时间,创建人,项目名称。
 5 ‘‘‘
 6 Created on 2019-5-10
 7 @author: 北京-宏哥
 8 Project:学习和使用邮箱发文字,html,图片,附件实现实例邮件
 9 ‘‘‘
10 #3.导入模块
11 import smtplib
12 from email.mime.multipart import MIMEMultipart
13 from email.mime.text import MIMEText
14 from email.mime.image import MIMEImage
15 from email.header import Header
16
17 # 设置smtplib所需的参数
18 # 下面的发件人,收件人是用于邮件传输的。
19 smtpserver = ‘smtp.mxhichina.com‘
20 username = ‘[email protected]‘
21 password = ‘[email protected]@xx3‘
22 sender = ‘[email protected]‘
23 # 收件人为一个收件人
24 # receiver=‘[email protected]‘
25 # 收件人为多个收件人
26 receiver = ["[email protected]","[email protected]","[email protected]"]
27
28 subject = ‘Python email test‘
29 # 通过Header对象编码的文本,包含utf-8编码信息和Base64编码信息。以下中文名测试ok
30 # subject = ‘中文标题‘
31 # subject=Header(subject, ‘utf-8‘).encode()
32
33 # 构造邮件对象MIMEMultipart对象
34 # 下面的主题,发件人,收件人,日期是显示在邮件页面上的。
35 msg = MIMEMultipart(‘mixed‘)
36 msg[‘Subject‘] = subject
37 msg[‘From‘] = ‘北京-宏哥<[email protected]>‘
38 # msg[‘To‘] = "[email protected]","[email protected]","[email protected]"
39 # 收件人为多个收件人,通过join将列表转换为以;为间隔的字符串
40 msg[‘To‘] = ";".join(receiver)
41 # msg[‘Date‘]=‘2019-5-10‘
42
43 # 构造文字内容
44 text = "Hi!\nHow are you?\nHere is the link you wanted:\nhttps://www.cnblogs.com/du-hong"
45 text_plain = MIMEText(text, ‘plain‘, ‘utf-8‘)
46 msg.attach(text_plain)
47
48 # 构造图片链接
49 sendimagefile = open(r‘E:\pythontest\testimage.png‘, ‘rb‘).read()
50 image = MIMEImage(sendimagefile)
51 image.add_header(‘Content-ID‘, ‘<image1>‘)
52 image["Content-Disposition"] = ‘attachment; filename="honggeimage.png"‘
53 msg.attach(image)
54
55 # 构造html
56 # 发送正文中的图片:由于包含未被许可的信息,网易邮箱定义为垃圾邮件,报554 DT:SPM :<p><img src="cid:image1"></p>
57 html = """
58 <html>
59   <head></head>
60   <body>
61     <p>Hi!<br>
62        How are you?<br>
63        Here is the <a href="https://www.cnblogs.com/du-hong/">link 北京-宏哥</a> you wanted.<br>
64     </p>
65   </body>
66 </html>
67 """
68 text_html = MIMEText(html, ‘html‘, ‘utf-8‘)
69 text_html["Content-Disposition"] = ‘attachment; filename="texthtml.html"‘
70 msg.attach(text_html)
71
72 # 构造附件
73 sendfile = open(r‘E:\pythontest\text.txt‘, ‘rb‘).read()
74 text_att = MIMEText(sendfile, ‘base64‘, ‘utf-8‘)
75 text_att["Content-Type"] = ‘application/octet-stream‘
76 # 以下附件可以重命名成aaa.txt
77 # text_att["Content-Disposition"] = ‘attachment; filename="aaa.txt"‘
78 # 另一种实现方式
79 text_att.add_header(‘Content-Disposition‘, ‘attachment‘, filename=‘hongge.txt‘)
80 # 以下中文测试不ok
81 # text_att["Content-Disposition"] = u‘attachment; filename="中文附件.txt"‘.decode(‘utf-8‘)
82 msg.attach(text_att)
83
84 # 发送邮件
85 smtp = smtplib.SMTP()
86 smtp.connect(‘smtp.mxhichina.com‘)
87 # 我们用set_debuglevel(1)就可以打印出和SMTP服务器交互的所有信息。
88 # smtp.set_debuglevel(1)
89 smtp.login(username, password)
90 smtp.sendmail(sender, receiver, msg.as_string())
91 smtp.quit()

说明:

对有些 header 要特别留意,服务器会针对这些 header 做检查

User-Agent : 有些服务器或 Proxy 会通过该值来判断是否是浏览器发出的请求

Content-Type : 在使用 REST 接口时,服务器会检查该值,用来确定 HTTP Body 中的内容该怎样解析。常见的取值有:

application/xml : 在 XML RPC,如 RESTful/SOAP 调用时使用

application/json : 在 JSON RPC 调用时使用

application/x-www-form-urlencoded : 浏览器提交 Web 表单时使用

在使用服务器提供的 RESTful 或 SOAP 服务时, Content-Type 设置错误会导致服务器拒绝服务

小结

  好了,哈哈,到这里把python发邮件的十八辈祖宗都被我们挖出来了,自己都感觉到有点缺德,但是我们学习就是需要这种精神,但是也不要过于钻牛角筋,要适可而止。想必小伙伴们对python发邮件有了更深刻的认识了,以后遇到类似的问题,

往上套就可以了,但是要注意方式和方法,不要生搬硬套,生搬硬套又会出现问题,要灵活有技巧的套。

原文地址:https://www.cnblogs.com/du-hong/p/10832048.html

时间: 2024-10-11 22:50:47

python接口自动化(三十三)-python自动发邮件总结及实例说明番外篇下(详解)的相关文章

python接口自动化(四十一)- 发xml格式参数的post请求(超详解)

简介 最近在工作中,遇到一种奇葩的接口,它的参数数据是通过xml,进行传递的,不要大惊小怪的,林子大了什么鸟都有,每个人的思路想法不一样,开发的接口也是各式各样的,如果想要统一的话,必须是提前团队已经做好沟通定好规则,这样就像在产品线上生成一 样规格大小一致.就不会出现前边的问题了,如果出现了怎么办?而且项目进度比较急,没有时间修改,一期上线就只能凑合着用这组接口了,那么作为QA的你,会不会测试,如何用代码.工具实现呢???等等问题跟着就产生了. 前边就有介绍到有关:post请求相对于get请求

python接口自动化9-https请求(SSL)

前言 本来最新的requests库V2.13.0是支持https请求的,但是一般写脚本时候,我们会用抓包工具fiddler,这时候会报:requests.exceptions.SSLError: [SSL: CERTIFICATE_VERIFY_FAILED] certificate verify failed (_ssl.c:590) 小编环境: python:2.7.12 requests:2.13.0 fiddler:v4.6.2.0 一.SSL问题 1.不启用fiddler,直接发htt

python接口自动化2-发送post请求

前言 发送post的请求参考例子很简单,实际遇到的情况却是很复杂的,首先第一个post请求肯定是登录了,但登录是最难处理的.登录问题解决了,后面都简单了. 一.查看官方文档 1.学习一个新的模块,其实不用去百度什么的,直接用help函数就能查看相关注释和案例内容. >>import requests >>help(requests) 2.查看python发送get和post请求的案例 >>> import requests       >>> r

python接口自动化7-参数关联

前言 我们用自动化发帖之后,要想接着对这篇帖子操作,那就需要用参数关联了,发帖之后会有一个帖子的id,获取到这个id,继续操作传这个帖子id就可以了 一.删除草稿箱 1.我们前面讲过登录后保存草稿箱,那可以继续接着操作:删除刚才保存的草稿 2.用fiddler抓包,抓到删除帖子的请求,从抓包结果可以看出,传的json参数是postId 3.这个postId哪里来的呢?可以看上个请求url地址 4.也就是说保存草稿箱成功之后,重定向一个url地址,里面带有postId这个参数.那接下来我们提取出来

python接口自动化4-绕过验证码登录(cookie)

前言 有些登录的接口会有验证码:短信验证码,图形验证码等,这种登录的话验证码参数可以从后台获取的(或者查数据库最直接). 获取不到也没关系,可以通过添加cookie的方式绕过验证码. 一.抓登录cookie 1.登录后会生成一个已登录状态的cookie,那么只需要直接把这个值添加到cookies里面就可以了. 2.可以先手动登录一次,然后抓取这个cookie,这里就需要用抓包工具fiddler了 3.先打开博客园登录界面,手动输入账号和密码(勾选下次自动登录) 4.打开fiddler抓包工具,刷

python接口自动化3-自动发帖(session)

前言 上一篇模拟登录博客园,但这只是第一步,一般登录后,还会有其它的操作,如发帖,评论等,这时候如何保持会话呢? 一.session简介 1.查看帮助文档,贴了一部分,后面省略了 >>import requests>>help(requests.session()) class Session(SessionRedirectMixin) |  A Requests session. |   |  Provides cookie persistence, connection-poo

python接口自动化5-Json数据处理

前言 有些post的请求参数是json格式的,这个前面第二篇post请求里面提到过,需要导入json模块处理. 一般常见的接口返回数据也是json格式的,我们在做判断时候,往往只需要提取其中几个关键的参数就行,这时候就需要json来解析返回的数据了. 一.json模块简介 1.Json简介:Json,全名 JavaScript Object Notation,是一种轻量级的数据交换格式,常用于http请求中 2.可以用help(json),查看对应的源码注释内容 Encoding basic P

python接口自动化10-token登录

前言 有些登录不是用cookie来验证的,是用token参数来判断是否登录. token传参有两种一种是放在请求头里,本质上是跟cookie是一样的,只是换个单词而已:另外一种是在url请求参数里,这种更直观. 一.登录返回token 1.如下图的这个登录,无cookies 2.但是登录成功后有返回token 二.请求头带token 1.登录成功后继续操作其它页面,发现post请求的请求头,都会带token参数 2.这种请求其实比cookie更简单,直接把登录后的token放到头部就行 三.to

python接口自动化8-参数化

前言 前面一篇实现了参数的关联,那种只是记流水账的完成功能,不便于维护,也没什么可读性,接下来这篇可以把每一个动作写成一个函数,这样更方便了. 参数化的思维只需记住一点:不要写死 一.登录函数 1.s参数是session的一个实例类,先放这里,方便写后面代码 2.登录函数传三个参数,s是需要调用前面的session类,所以必传,可以传个登录的url,然后payload是账号和密码 二.保存草稿 1.编辑内容的标题title和正文body_data参数化了,这样后面可以方便传不同值 2.这里返回了