使用Python读取网易邮箱大师客户端的所有邮件以及邮箱验证码

查看 210|回复 10
作者:我真的爱发明   
[toc]
1. 前言
  • 现在绝大多数的网站注册的时候,都需要你提供手机或者邮箱的验证码。如果只是想要注册一两个账号,那么随便接一下验证码就可以了。但如果我们想要注册几十个账号,意味着你需要接受几十个验证码,这可能就是一个不小的灾难了。你首先得从几十个邮箱中找到这个特定的邮箱,点开它上面的收件箱,然后复制验证码.......


  • 可能这么说一下,大家不觉得有啥,但实际操作起来真的是太烦人类了。于是我想着能不能解析一下网易邮箱大师上面的API来直接读取邮件内容或者验证码呢?于是便有了下面的项目。

    2. 效果
  • 可以直接通过Python代码或者命令行程序直接读取网易邮箱大师的本地客户端邮件数据




  • 3. 探究过程
  • 想要读取网易邮箱上的邮件要么是通过API进行调用,要么是通过客户端本地的数据进行逆向获取
  • 经过一番百度以及一番初步尝试,发现网易邮箱并没有给用户或者开发者提供程序API接口,那这是否就意味着我们想要用程序读取网易邮箱中的邮件呢做不到呢?其实并不然,就算没有上层的API,但我们可以读取底层的保存数据(只要他没有太高难度的加密就行),于是便有了下面的尝试

    3.1. 找到本地存储的数据库

  • 那么首先需要找到网易邮箱大师的数据存储位置

  • 系统设置这里可以看到网易邮箱的数据存储位


  • 打开位置之后,是一系列以邮箱名命名的文件夹


  • 但是这里面的数据似乎是加密的数据库文件


  • 自己使用Navicat for sqlite打开发现也看不到里面的具体数据


  • 自己又换了一个db文件进行读取发现里面的数据似乎也并未完全加密,依稀可以看到一些邮件内容之前之所以显示是乱码,可能是转码的问题


  • 自己又查看了一下另外两个mail和search ,没想到邮件内容直接是在search/search_content中明文存储,看来今天的探索应该可以结束了


  • 我们来新发一份邮件,看能否实时更新



    可能是自己添加的邮箱数量太多了,这里一直处于发送过程中

    重启了一下网易邮箱客户端之后正常了
  • 刷新了一下Navicat for sqlite,然后看到了我们刚才测试的数据


    到这里证明我们的这个原理是行得通的了。那么下面就用程序来实现这个过程

    3.2. 使用Python读取数据库
    3.2.1. 代码
    #导入sqllite3模块
    import sqlite3
    # 1.硬盘上创建连接
    con = sqlite3.connect('D:\MailMasterData\[email protected]_1414\search.db')
    # 获取cursor对象
    cur = con.cursor()
    # 执行sql创建表
    sql = 'select * from Search_content'
    try:
        cur.execute(sql)
        # 获取所有数据
        person_all = cur.fetchall()
        # print(person_all)
        # 遍历
        for p in person_all:
            print(p)
    except Exception as e:
        print(e)
        print('查询失败')
    finally:
        # 关闭游标
        cur.close()
        # 关闭连接
        con.close()




  • 4. 探究结果
    4.1. 函数
    4.1.1. 找到特定邮~箱的最新一条邮件
    def sqlite3_get_last_data(db_path,sql):          找到特定邮箱最后一次的数据
        # 导入sqllite3模块
        import sqlite3
        # 1.硬盘上创建连接
        con = sqlite3.connect(db_path)
        # 获取cursor对象
        cur = con.cursor()
        # 执行sql创建表
        try:
            cur.execute(sql)
            # 获取所有数据
            person_all = cur.fetchall()
            last_data = person_all[-1]
            # print(last_data)
            # print("type(last_data):", type(last_data))
            # print("last_data:", )
            last_text = last_data[6]
            return last_text
        except Exception as e:
            print(e)
            print('查询失败')
        finally:
            # 关闭游标
            cur.close()
            # 关闭连接
            con.close()
    email_path = '[email protected]'
    db_path = 'D:\MailMasterData\{}_1414\search.db'.format(email_path)
    sql = 'select * from Search_content'

    4.1.2. 找到特定邮箱的最新一次验证码

    代码
    def find_spacial_email_last_Verification_Code(email):        #找到特定邮箱最后一次的验证码
        def sqlite3_get_last_data(db_path, sql):
            # 导入sqllite3模块
            import sqlite3
            # 1.硬盘上创建连接
            con = sqlite3.connect(db_path)
            # 获取cursor对象
            cur = con.cursor()
            # 执行sql创建表
            try:
                cur.execute(sql)
                # 获取所有数据
                person_all = cur.fetchall()
                last_data = person_all[-1]
                # print(last_data)
                # print("type(last_data):", type(last_data))
                # print("last_data:", )
                last_text = last_data[6]
                return last_text
            except Exception as e:
                print(e)
                print('查询失败')
            finally:
                # 关闭游标
                cur.close()
                # 关闭连接
                con.close()
        def find_continuous_data(string):  # 查找字符串中连续的数字
            import re
            result = re.findall(r"\d{4,6}", string)
            return result
        def find_str_in_list_full_string(list_str, find_str):  # 查找字符串列表中包含某一个字符的完整字符
            for index, str in enumerate(list_str):
                if find_str in str:
                    break
            return str
        import os
        mailbox_rootpath = 'D:\MailMasterData'
        email_name_list = os.listdir(mailbox_rootpath)
        email_dir_name = find_str_in_list_full_string(email_name_list, email)
        db_path = '{}\{}\search.db'.format(mailbox_rootpath,email_dir_name)
        sql = 'select * from Search_content'
        last_text = sqlite3_get_last_data(db_path, sql)
        print("last_text:", last_text)
        result = find_continuous_data(last_text)
        if (len(result) == 1):
            print('找到了验证码')
            print(result[0])
            pass
        else:
            from winsound import Beep
            Beep(600, 1000)
    email = '[email protected]'
    find_spacial_email_last_Verification_Code(email)


  • 4.1.3. 通过命令行调用Python代码找到特定邮箱的最新的验证码
    代码
    def find_spacial_email_last_Verification_Code(email):        #找到特定邮箱最后一次的验证码
        def sqlite3_get_last_data(db_path, sql):        #找到特定邮箱最后一次的数据
            # 导入sqllite3模块
            import sqlite3
            # 1.硬盘上创建连接
            con = sqlite3.connect(db_path)
            # 获取cursor对象
            cur = con.cursor()
            # 执行sql创建表
            try:
                cur.execute(sql)
                # 获取所有数据
                person_all = cur.fetchall()
                last_data = person_all[-1]
                # print(last_data)
                # print("type(last_data):", type(last_data))
                # print("last_data:", )
                last_text = last_data[6]
                return last_text
            except Exception as e:
                print(e)
                print('查询失败')
            finally:
                # 关闭游标
                cur.close()
                # 关闭连接
                con.close()
        def find_continuous_data(string):  # 查找字符串中连续的数字
            import re
            result = re.findall(r"\d{4,6}", string)
            return result
        def find_str_in_list_full_string(list_str, find_str):  # 查找字符串列表中包含某一个字符的完整字符
            for index, str in enumerate(list_str):
                if find_str in str:
                    break
            return str
        import os
        mailbox_rootpath = 'D:\MailMasterData'
        email_name_list = os.listdir(mailbox_rootpath)
        email_dir_name = find_str_in_list_full_string(email_name_list, email)
        db_path = '{}\{}\search.db'.format(mailbox_rootpath,email_dir_name)
        sql = 'select * from Search_content'
        last_text = sqlite3_get_last_data(db_path, sql)
        # print("last_text:", last_text)
        result = find_continuous_data(last_text)
        if (len(result) == 1):
            print('找到了验证码')
            print(result[0])
            pass
        else:
            from winsound import Beep
            Beep(600, 1000)
    import os
    import argparse
    parser = argparse.ArgumentParser(description='命令行传入参数')
    #type是要传入的参数的数据类型  help是该参数的提示信息
    parser.add_argument('--email', type=str, help='传入的邮箱')
    args = parser.parse_args()
    #获得传入的参数
    print(args.email)
    email = args.email
    find_spacial_email_last_Verification_Code(email)
  • python .\读取特定邮箱的数据库.py --email "[email protected]"


    邮箱, 验证码

  • LightswornSnow   


    我真的爱发明 发表于 2022-12-28 22:52
    这种方式是不是需要手机号获得对应的信息?

    没有吧。就在邮箱设置里开启IMAP功能时要用手机确认一下来获取授权码。以后脚本用邮箱用户名、授权码登录。
    (我以前只写过学校邮箱的脚本,学校邮箱可以直接拿用户名密码登录收发邮件,而网易QQ这些邮箱都是用授权码、不用密码。)
    另外就是多一个ID字段要加……刚刚随便搜了一下,参考这两个帖子:
    python邮件处理(1)-IMAP收取邮件_w.royee的博客-CSDN博客_imap python https://blog.csdn.net/naoyaolin0720/article/details/119851663
    【2020可用】Python使用 imaplib imapclient连接网易邮箱提示 Unsafe Login. Please contact [email protected] for help 的解决办法_去山的那边看海啊的博客-CSDN博客 https://blog.csdn.net/jony_online/article/details/108638571
    我真的爱发明
    OP
      


    cnmsh 发表于 2022-12-30 11:15
    是不是每回开机 先更新一下所有邮箱,在搜本地数据
    长时间不登录邮箱, 容易忘密码

    这个就是读取本地的数据,更新一下邮箱,可以保证邮箱中获得最新的邮件
    zzcg150   

    谢谢分享
    yulonglailo   

    谢谢分享
    jixingzi   

    感谢分享
    hui-shao   

    实用技术,感谢分享,收藏备用!
    LightswornSnow   

    感谢分享了。我是觉得可能利用邮箱的IMAP协议读取的多一点?不过解析客户端多种思路多种手段了。
    我真的爱发明
    OP
      


    LightswornSnow 发表于 2022-12-28 22:34
    感谢分享了。我是觉得可能利用邮箱的IMAP协议读取的多一点?不过解析客户端多种思路多种手段了。

    这种方式是不是需要手机号获得对应的信息?
    bennyt   

    过来学习一下,不错。
    您需要登录后才可以回帖 登录 | 立即注册

    返回顶部