2025年3月31日 星期一 乙巳(蛇)年 正月初一 设为首页 加入收藏
rss
您当前的位置:首页 > 计算机 > 编程开发 > Python

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

时间:08-21来源:作者:点击数:28
发送邮件

SMTP协议

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

  • 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() 登出,保存修改,解锁邮箱,结束连接,退出
  • 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),第二个参数不知到是干什么官方没解释
  • 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邮箱

  • # -*- 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),所以连接偶尔报错上面的信息。

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

  • server = poplib.POP3(pop3_server, 110, timeout=10)
  • or
  • server = poplib.POP3_SSL(pop3_server, 995, timeout=10)

获取信体部分

  • 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的格式,能在浏览器里查看,就像平时看到的邮件那样。

官方文档:

方便获取更多学习、工作、生活信息请关注本站微信公众号城东书院 微信服务号城东书院 微信订阅号
推荐内容
相关内容
栏目更新
栏目热门
本栏推荐