Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >python实现邮件接收、附件下载

python实现邮件接收、附件下载

作者头像
周小董
发布于 2019-03-25 02:29:04
发布于 2019-03-25 02:29:04
6.4K00
代码可运行
举报
文章被收录于专栏:python前行者python前行者
运行总次数:0
代码可运行

发送邮件

SMTP协议

SMTP(Simple Mail Transfer Protocol)即简单邮件传输协议,它是一组用于由源地址到目的地址传送邮件的规则,由它来控制信件的中转方式。SMTP协议属于TCP/IP协议簇,它帮助每台计算机在发送或中转信件时找到下一个目的地。通过SMTP协议所指定的服务器,就可以把E-mail寄到收信人的服务器上了。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import smtplib
from email import encoders
from email.header import Header
from email.mime.text import MIMEText
from email.utils import parseaddr, formataddr

def send_email(from_addr, to_addr, subject, password):
    msg = MIMEText("邮件正文",'html','utf-8')
    msg['From'] = u'<%s>' % from_addr
    msg['To'] = u'<%s>' % to_addr
    msg['Subject'] = subject

    smtp = smtplib.SMTP_SSL('smtp.163.com', 465)
    smtp.set_debuglevel(1)
    smtp.ehlo("smtp.163.com")
    smtp.login(from_addr, password)
    smtp.sendmail(from_addr, [to_addr], msg.as_string())

if __name__ == "__main__":
    # 这里的密码是开启smtp服务时输入的客户端登录授权码,并不是邮箱密码
    # 现在很多邮箱都需要先开启smtp才能这样发送邮件
    send_email(u"from_addr",u"to_addr",u"主题",u"password")

邮箱

SMTP服务器

SSL协议端口

非SSL协议端口

163

smtp.163.com

465或者994

25

qq

smtp.qq.com

465或587

25

接收邮件

POP3和IMAP

POP是指邮局协议,目的是让用户可以访问邮箱服务器中的邮件,允许用户从服务器上把邮件存储到本地主机(即自己的计算机)上,同时删除保存在邮件服务器上的邮件,而POP3服务器则是遵循POP3协议的接收邮件服务器,用来接收电子邮件的。

后来又出现了IMAP协议(Interactive Mail Access Protocol),即交互式邮件访问协议,与POP3的不同在于:开启了IMAP后,在电子邮件客户端收取的邮件仍然保留在服务器上,同时在客户端上的操作都会反馈到服务器上,如:删除邮件,标记已读等,服务器上的邮件也会做相应的动作。

  • poplib的常用方法:

方法

描述

POP3(server)

实例化POP3对象,server是pop服务器地址

user(username)

发送用户名到服务器,等待服务器返回信息

pass_(password)

密码

stat()

返回邮箱的状态,返回2元祖(消息的数量,消息的总字节)

list([msgnum])

stat()的扩展,返回一个3元祖(返回信息, 消息列表, 消息的大小),如果指定msgnum,就只返回指定消息的数据

retr(msgnum)

获取详细msgnum,设置为已读,返回3元组(返回信息, 消息msgnum的所以内容, 消息的字节数),如果指定msgnum,就只返回指定消息的数据

dele(msgnum)

将指定消息标记为删除

quit()

登出,保存修改,解锁邮箱,结束连接,退出

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
from poplib import POP3

p = POP3('pop.163.com')
p.user('xxxxxxx@163.com')
p.pass_('xxxxxxxx')

p.stat()
...
p.quit()

使用IMAP

python中的imaplib包支持IMAP4

常用方法:

方法

描述

IMAP4(server)

与IMAP服务器建立连接

login(user, pass)

用户密码登录

list()

查看所有的文件夹(IMAP可以支持创建文件夹)

select()

选择文件夹默认是"INBOX"

search()

三个参数,第一的是CHARSET,通常为None(ASCII),第二个参数不知到是干什么官方没解释

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import getpass, imaplib

M = imaplib.IMAP4()
M.login(getpass.getuser(), getpass.getpass())
M.select()
typ, data = M.search(None, 'ALL')
for num in data[0].split():
    typ, data = M.fetch(num, '(RFC822)')
    print 'Message %s\n%s\n' % (num, data[0][1])
M.close()
M.logout()

邮箱: 网易163邮箱,或qq邮箱

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
# -*- coding: utf-8 -*-
import poplib,email,telnetlib
import datetime,time,sys,traceback
from email.parser import Parser
from email.header import decode_header
from email.utils import parseaddr


class down_email():

    def __init__(self,user,password,eamil_server):
        # 输入邮件地址, 口令和POP3服务器地址:
        self.user = user
        # 此处密码是授权码,用于登录第三方邮件客户端
        self.password = password
        self.pop3_server = eamil_server

    # 获得msg的编码
    def guess_charset(self,msg):
        charset = msg.get_charset()
        if charset is None:
            content_type = msg.get('Content-Type', '').lower()
            pos = content_type.find('charset=')
            if pos >= 0:
                charset = content_type[pos + 8:].strip()
        return charset

    #获取邮件内容
    def get_content(self,msg):
        content=''
        content_type = msg.get_content_type()
        # print('content_type:',content_type)
        if content_type == 'text/plain': # or content_type == 'text/html'
            content = msg.get_payload(decode=True)
            charset = self.guess_charset(msg)
            if charset:
                content = content.decode(charset)
        return content

    # 字符编码转换
    # @staticmethod
    def decode_str(self,str_in):
        value, charset = decode_header(str_in)[0]
        if charset:
            value = value.decode(charset)
        return value

    # 解析邮件,获取附件
    def get_att(self,msg_in, str_day):
        attachment_files = []
        for part in msg_in.walk():
            # 获取附件名称类型
            file_name = part.get_param("name")  # 如果是附件,这里就会取出附件的文件名
            # file_name = part.get_filename() #获取file_name的第2中方法
            # contType = part.get_content_type()
            if file_name:
                h = email.header.Header(file_name)
                # 对附件名称进行解码
                dh = email.header.decode_header(h)
                filename = dh[0][0]
                if dh[0][1]:
                    # 将附件名称可读化
                    filename = self.decode_str(str(filename, dh[0][1]))
                    # print(filename)
                    # filename = filename.encode("utf-8")
                # 下载附件
                data = part.get_payload(decode=True)
                # 在指定目录下创建文件,注意二进制文件需要用wb模式打开
                att_file = open('./test/' + filename, 'wb')
                att_file.write(data)  # 保存附件
                att_file.close()
                attachment_files.append(filename)
            else:
                # 不是附件,是文本内容
                print(self.get_content(part))
                # # 如果ture的话内容是没用的
                # if not part.is_multipart():
                #     # 解码出文本内容,直接输出来就可以了。
                #     print(part.get_payload(decode=True).decode('utf-8'))

        return attachment_files

    def run_ing(self):
        str_day = str(datetime.date.today())# 日期赋值
        # 连接到POP3服务器,有些邮箱服务器需要ssl加密,可以使用poplib.POP3_SSL
        try:
            telnetlib.Telnet(self.pop3_server, 995)
            self.server = poplib.POP3_SSL(self.pop3_server, 995, timeout=10)
        except:
            time.sleep(5)
            self.server = poplib.POP3(self.pop3_server, 110, timeout=10)

        # server.set_debuglevel(1) # 可以打开或关闭调试信息
        # 打印POP3服务器的欢迎文字:
        print(self.server.getwelcome().decode('utf-8'))
        # 身份认证:
        self.server.user(self.user)
        self.server.pass_(self.password)
        # 返回邮件数量和占用空间:
        print('Messages: %s. Size: %s' % self.server.stat())
        # list()返回所有邮件的编号:
        resp, mails, octets = self.server.list()
        # 可以查看返回的列表类似[b'1 82923', b'2 2184', ...]
        print(mails)
        index = len(mails)
        for i in range(index, 0, -1):# 倒序遍历邮件
        # for i in range(1, index + 1):# 顺序遍历邮件
            resp, lines, octets = self.server.retr(i)
            # lines存储了邮件的原始文本的每一行,
            # 邮件的原始文本:
            msg_content = b'\r\n'.join(lines).decode('utf-8')
            # 解析邮件:
            msg = Parser().parsestr(msg_content)
            #获取邮件的发件人,收件人, 抄送人,主题
            # hdr, addr = parseaddr(msg.get('From'))
            # From = self.decode_str(hdr)
            # hdr, addr = parseaddr(msg.get('To'))
            # To = self.decode_str(hdr)
            # 方法2:from or Form均可
            From = parseaddr(msg.get('from'))[1]
            To = parseaddr(msg.get('To'))[1]
            Cc=parseaddr(msg.get_all('Cc'))[1]# 抄送人
            Subject = self.decode_str(msg.get('Subject'))
            print('from:%s,to:%s,Cc:%s,subject:%s'%(From,To,Cc,Subject))
            # 获取邮件时间,格式化收件时间
            date1 = time.strptime(msg.get("Date")[0:24], '%a, %d %b %Y %H:%M:%S')
            # 邮件时间格式转换
            date2 = time.strftime("%Y-%m-%d",date1)
            if date2 < str_day:
                break # 倒叙用break
                # continue # 顺叙用continue
            else:
                # 获取附件
                attach_file=self.get_att(msg,str_day)
                print(attach_file)

        # 可以根据邮件索引号直接从服务器删除邮件:
        # self.server.dele(7)
        self.server.quit()


if __name__ == '__main__':
    #把打印内容输出到文件
    # origin = sys.stdout
    # f = open('./test/log.txt', 'w')
    # sys.stdout = f
    try:
        # 输入邮件地址, 口令和POP3服务器地址:
        user = '8284@163.com'
        # 此处密码是授权码,用于登录第三方邮件客户端
        password = '8284'
        eamil_server = 'pop.163.com'
        # user='xinfei@.com'
        # password = 'f67h2'
        # eamil_server = 'pop.exmail.qq.com'
        email_class=down_email(user=user,password=password,eamil_server=eamil_server)
        email_class.run_ing()
    except Exception as e:
        import traceback
        ex_msg = '{exception}'.format(exception=traceback.format_exc())
        print(ex_msg)
        # traceback.print_exc()
    # sys.stdout = origin
    # f.close()

遇到的并且待验证问题:

1、报错pop3 ConnectionRefusedError: [WinError 10061] 由于目标计算机积极拒绝,无法连接。

一开始直接用的 server = poplib.POP3(pop3_server),所以连接偶尔报错上面的信息。

之后改成指定端口和超时时间之后就不再有报错信息了

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
server = poplib.POP3(pop3_server, 110, timeout=10)
or 
server = poplib.POP3_SSL(pop3_server, 995, timeout=10)

获取信体部分

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
for part in msg.walk():
        # 如果ture的话内容是没用的
        if not part.is_multipart():            
            print(part.get_payload(decode=True).decode('utf-8')) 
            # 解码出文本内容,直接输出来就可以了。

walk()函数能历遍邮件所有部分,所以通常都把它放到for循环里面使用。然后再使用is_multipart()函数来判断内容是否有用,打印出有用内容最后用get_payload(decode=True).decode(‘utf-8’)解码并且打印到控制台。通常这个循环有两次,第一次是单纯的字符串格式的,能在控制台显示出来的,第二次循环打印的是像HTML的格式,能在浏览器里查看,就像平时看到的邮件那样。

官方文档:

参考:https://www.cnblogs.com/itogo/p/5910681.html https://blog.csdn.net/u012209894/article/details/82384987

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2019年01月25日,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
cellchat-(1)单数据集的细胞通讯分析
接下来需要选择配体受体数据库,cellchat的数据库是基于文献手动生成的,包含的物种和数量为:
R小白
2023/05/20
1.6K0
cellchat-(1)单数据集的细胞通讯分析
CellChat:细胞间相互作用分析利器
我们知道,细胞间信息传递方式一个是细胞表面配受体的相互作用,另一个通过细胞产生的可溶性小分子,即细胞因子。在单细胞数据分析中下游,有时候我们想看某几种细胞类型之间的相互作用,就有人推荐我们做一个配受体分析。那什么是配受体?我们在文章Cell-Cell Interaction Database|| 单细胞配受体库你还在文章的附录里找吗?中提到配受体其实是细胞的特定蛋白,蛋白追溯到基因表达上就是基因对。
生信技能树jimmy
2020/07/29
6.3K0
CellChat:细胞间相互作用分析利器
单细胞实战之cellchat——入门到进阶(高级篇1)
在中级篇中已经完成了细胞亚群的细分流程,并回顾了几个重要的分析工具。接下来将进入高级篇的内容。首先将回顾最常用的细胞通讯工具之一——CellChat。在该讲中会按照单一及两样本进行分析。
凑齐六个字吧
2025/04/13
6960
单细胞实战之cellchat——入门到进阶(高级篇1)
CellChat三部曲1:使用CellChat对单个数据集进行细胞间通讯分析
自从周运来写了一篇cellchat的中文介绍教程《CellChat:细胞间相互作用分析利器》,然后R包作者也在B站做了直播介绍,cellchat作为一个细胞通讯分析的新兴R包,受到了广泛关注。教程也如雨后春笋般涌现。
生信技能树jimmy
2021/08/20
35.3K4
单细胞转录组之使用CellChat对单个数据集进行细胞间通讯分析
细胞通讯分析可以提供细胞亚群互调控的信息,细胞间的调控主要是通过受体配体结合来实现信号传递的。
青青青山
2022/07/13
6.4K1
单细胞转录组之使用CellChat对单个数据集进行细胞间通讯分析
通过空间行为(optimal transport)推断空间细胞间通讯信号方向(COMMOT)
配体和受体通常在有限的空间范围内与多种复合物相互作用。考虑到这一点,作者提出了具有三个重要特征的collective optimal transport:首先,the use of non-probability mass distributions to control the marginals of the transport plan to maintain comparability between species(需要一点数学背景知识);其次,对CCC实施空间距离约束,以避免连接空间上相距较远的细胞;最后,将多种配体分布结合到多中受体分布以解释多种相互作用。
追风少年i
2023/02/23
1.1K0
通过空间行为(optimal transport)推断空间细胞间通讯信号方向(COMMOT)
空转细胞通讯分析之COMMOT
空间转录组以及单细胞转录组技术,为解析细胞-细胞通讯(Cell–Cell Communication, CCC)提供了前所未有的机会。然而,如何在重建细胞通讯网络的过程中充分整合细胞间的空间位置信息和复杂的生化过程,仍然是一个重大挑战。
生信菜鸟团
2025/03/24
2280
空转细胞通讯分析之COMMOT
空转 | CellChat-V2,揭秘空间转录组数据的细胞通讯分析
仍然使用空转 | 结合scRNA完成空转spot注释(Seurat Mapping) & 彩蛋(封面的空转主图代码)推文中的空转数据进行示例展示。
生信补给站
2023/11/09
5.1K0
空转 | CellChat-V2,揭秘空间转录组数据的细胞通讯分析
scRNA分析|使用CellChat完成细胞通讯分析-简单且可视化出众,代码自取
之前介绍过使用cellphoneDB 进行细胞通讯分析scRNA分析 | 解决可能的报错,从0开始教你完成细胞通讯分析-cellphoneDB,可能会遇到一些报错。这次介绍另一款细胞通讯分析的常见方法CellChat 。CellChat是一款R包,使用更容易且可视化结果也非常不错。
生信补给站
2023/08/25
2.9K0
scRNA分析|使用CellChat完成细胞通讯分析-简单且可视化出众,代码自取
CellChat细胞通讯
细胞通讯是单细胞数据高级分析中比较常见的一个,我们习惯使用的R包是CellChat。
生信技能树
2024/06/21
6000
CellChat细胞通讯
🤩 Cellchat | 空间转录组也可以做细胞通讯啦!~(一)(单个数据集篇)
CellChat 2 现在支持多种数据类型,除了单细胞转录组数据(scRNA-seq),还扩展支持 空间转录组数据(spatial transcriptomics)。😘
生信漫卷
2024/11/23
5680
🤩 Cellchat | 空间转录组也可以做细胞通讯啦!~(一)(单个数据集篇)
单样本Cellchat(V2)细胞通讯分析学习和整理
细胞通讯分析是一种研究不同细胞类型之间如何通过信号分子(如配体和受体)进行相互交流和调控的分析方法。它在揭示细胞间相互作用的机制,理解组织和器官如何协调运作方面具有重要意义。
凑齐六个字吧
2024/09/21
5500
单样本Cellchat(V2)细胞通讯分析学习和整理
🤩 Cellchat | 空间转录组也可以做细胞通讯啦!~(二)(多个数据集篇)
CellChat是一个专门用于分析细胞间通信的强大工具,现在也是支持空间转录组的分析了。🧐
生信漫卷
2025/01/07
3340
🤩 Cellchat | 空间转录组也可以做细胞通讯啦!~(二)(多个数据集篇)
胞间互作工具——Cellchat(一)
最初接触这个R包是去年年中,想做细胞间相互作用,又不会python,正好看到周老师的推文,就跟着学了学,CellChat:细胞间相互作用分析利器,当时CellChat包还是0.0.1版本,里面有不少小bug,文章放在预印版上,而今年二月份他终于见刊,发表在NC,现在R包也来到了1.1.2版本,并且在github上持续更新,今天我们来重新学习一次。
生信菜鸟团
2021/07/29
8.5K0
胞间互作工具——Cellchat(一)
单细胞分析十八般武艺7:CellChat
单细胞初级8讲和高级分析8讲 单细胞分析十八般武艺1:harmony 单细胞分析十八般武艺2:LIGER 单细胞分析十八般武艺3:fastMNN 单细胞分析十八般武艺4:velocyto 单细胞分析十八般武艺5:monocle3 单细胞分析十八般武艺6:NicheNet
生信技能树jimmy
2021/04/29
4.4K0
单细胞分析十八般武艺7:CellChat
day 10 GSVA和CellChat
昆兰
2024/07/01
1100
GSVA和细胞通讯
教程:https://htmlpreview.github.io/?https://github.com/jinworks/CellChat/blob/master/tutorial/CellChat-vignette.html #再次感叹生信真牛
用户10300557
2024/06/27
2450
CellChat分析结果可视化
前面的帖子(CellChat学习笔记【一】——通讯网络构建)中我们已经成功地进行了细胞间通讯网络的构建,总的来看借助下面简易的分析流程即可完成:
生信技能树jimmy
2023/02/10
2K0
CellChat分析结果可视化
细胞相互作用在单细胞转录组中的应用(一)
许多生物进程和细胞通讯息息相关,如胚胎发育,器官形成,癌症发生发展,炎症反应,药物作用和耐药研究等。当细胞不能正确地相互作用或不正确地解码分子信息时,就会引起疾病。因此,细胞间信号通路的鉴定和定量已成为跨不同学科进行的常见分析。蛋白质间的相互作用数据库的扩展以及RNA测序技术的最新进展,使从大量和单细胞数据集的基因表达测量中进行细胞间信号传导的常规分析成为了可能。特别是,配体-受体对可用于从其同源基因的协调表达中推断细胞间的通讯。
生信交流平台
2022/09/21
1.9K0
细胞相互作用在单细胞转录组中的应用(一)
单细胞转录组 | 细胞互作分析
细胞互作(Cell-cell interaction)指的是不同细胞类型之间通过配体-受体对进行的信号交流过程。在复杂的组织和器官中,细胞间的通讯网络对于维持组织功能、调节生理过程以及疾病的发生发展具有重要作用。通过单细胞测序技术,可以在单细胞分辨率上推断这些细胞间的通讯模式。
天意生信云
2025/03/14
1310
单细胞转录组 | 细胞互作分析
推荐阅读
相关推荐
cellchat-(1)单数据集的细胞通讯分析
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验