python实现电子邮件附件指定时间段,批量下载以及C#小程序集成实现

Posted nanke_yh

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了python实现电子邮件附件指定时间段,批量下载以及C#小程序集成实现相关的知识,希望对你有一定的参考价值。

目录

一、需求背景

二、需求分析

三、需求实现

 3.1 python邮箱附件指定时间下载

3.2 python外部传参

3.3 C#界面设计

四、功能实现

 4.1 邮箱设置

4.2 邮件附件下载实现

1、依赖的模块

2、核心代码

3、其他

4、异常处理

5、python代码调用测试

4.3 C#内部代码实现

1、保存路径中“选择路径”

2、运行结束后清空控件中输入的数据

3、各控件内容判断和处理

4、C#调用外部多参数的exe

五、问题或求教

5.1 超大附件无法下载问题

5.2 C#调用exe代码中的弊端

5.3 C#程序运行依赖.NET环境

六、结语

参考:


一、需求背景

       由于女朋友做助教,每周需要通过邮箱收数学作业,一个班五十几号人给她发送邮件,一个一个的点击下载非常耗时。所以想使用程序实现一下附件下载功能。

       根据网上相关资料,使用python实现邮箱附件批量下载,然后打包生成exe后,虽是脱离了python环境便于她使用,可对于她们这些没有接触编程的人员来说,cmd命令行调用exe的操作不便于理解和操作。为此,界面化程序开发选择了C#,从而实现了C#调用python开发的exe来完成邮箱附件批量下载功能。

二、需求分析

1、指定时间的附件下载:因为作业的收发是每周都要进行的,不可每次邮件全部下载,需要指定起始和终止日期来接收一段时间内的邮箱附件;

2、基于界面化的操作,特别涉及邮箱登录等用户名、密码等信息,需要外部参数传参操作,即python核心代码中的参数输入需要留下外部接口;

3、C#界面开发,需要提供基础的参数入口:邮箱类型、邮箱账号、密码、保存路径、起止时间;

三、需求实现

 3.1 python邮箱附件指定时间下载

       该部分需要在邮件遍历过程中,设置起止时间根据邮件的接收时间来判断是否下载当前邮件内附件(参考[1]):

    # 解析邮件:
    msg = Parser().parsestr(msg_content)
    # 获取邮件日期,格式化收件时间
    date = time.strptime(msg.get('Date')[:24], '%a, %d %b %Y %H:%M:%S')
    date = time.strftime('%Y%m%d', date)#邮件时间格式转换
		
    # 只下载发送日期在start_date 到end_date 之间的邮件附件
    if date < start_date :
        break
    if date > end_date :
        continue

3.2 python外部传参

       结合需求3,需要在邮件附件下载代码中实现外部参数的传入,传入数量为邮箱账号、密码、保存路径、起止时间和邮箱类型(参考[2]): 

import sys

#------------------------------------------------------------------	
arg1 = sys.argv[1]
arg2 = sys.argv[2]
arg3 = sys.argv[3]
arg4 = sys.argv[4]
arg5 = sys.argv[5]
arg6 = sys.argv[6] 

host = "pop."+arg6     #163.com" 
#host = "pop.163.com"  # 邮箱域名,其他邮箱类似
server = poplib.POP3(host)  # 建立链接
username = arg1  # 用户名#"xxxxxxx@163.com"
password = arg2  # 密码IMAP/SMTP的密码#"AAAAAAAAAAAAAAAA"
start_date = arg4 #起始时间#'20210929'
end_date = arg5  #终止时间#'20211008' 
# 设置储存路径
mail_att_dir = arg3    #r'D:\\att_file'

3.3 C#界面设计

      提供基础的参数入口:邮箱类型、邮箱账号、密码、保存路径、起止时间

 

四、功能实现

 4.1 邮箱设置

       在实现邮件附件下载功能之前,首先需要开启邮箱的pop协议,一般在设置里,以163邮箱为例,设置后会给个密码,这个就是程序访问邮箱的登录密码:

邮箱内测试邮件:

4.2 邮件附件下载实现

1、依赖的模块

import email
import os
import poplib
import time
from email.header import decode_header
from email.parser import Parser
from email.utils import parseaddr
import sys

2、核心代码

python遍历邮件批量下载指定时间段的邮件附件代码(参考[1、3]):

def decode_str(s):
    """
    字符编码转换
    """
    value, charset = decode_header(s)[0]
    if charset:
        value = value.decode(charset)
    return value

# 登录
server.user(username)
server.pass_(password)
resp, mails, octets = server.list()  #获取所有邮件编号,mails的格式为['mesg_num octets', ...]

# 倒序遍历邮件
index = len(mails)
for i in range(index, 0, -1):
    # lines存储了邮件的原始文本的每一行
    resp, lines, octets = server.retr(i)
    # 邮件的原始文本:
    msg_content = b'\\r\\n'.join(lines).decode('utf-8')     #⭐

    # 解析邮件:
    msg = Parser().parsestr(msg_content)
    # 获取邮件日期,格式化收件时间
    date = time.strptime(msg.get('Date')[:24], '%a, %d %b %Y %H:%M:%S')
    date = time.strftime('%Y%m%d', date)#邮件时间格式转换
		
    # 只下载发送日期在arg1到arg2之间的邮件附件
    if date < start_date :
        break
    if date > end_date :
        continue
			
    # 下载附件
    attachment_files = []
    for part in msg.walk():
        file_name = part.get_filename()  # 获取附件名称类型
        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 = decode_str(str(filename, dh[0][1]))  # 将附件名称可读化
            data = part.get_payload(decode=True)  # 下载附件
            # 创建附件存储文件夹
            if not os.path.isdir(mail_att_dir ):
                os.mkdir(mail_att_dir )
            # 在指定目录下创建文件,注意二进制文件需要用wb模式打开
            att_file = open(mail_att_dir + os.sep + filename, 'wb')
            attachment_files.append(filename)
            att_file.write(data)  # 保存附件
            att_file.close()
server.quit()

3、其他

若需要获取邮件其他信息可以在循环中添加(参考[3-1]):

    #获取邮件的发件人,收件人, 抄送人,主题
    # 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 = decode_str(msg.get('Subject'))
    print('from:%s,to:%s,Cc:%s,subject:%s'%(From,To,Cc,Subject))

这一块还有很多的功能或者细节可以完善(比如函数封装等等),具体的可以自己研究。

4、异常处理

     对于有的同学,复制粘贴过去的代码,提示报错TabError: inconsistent use of tabs and spaces in indentation:参考[4-1],主要是代码中应该为空的缩进而没有为空,具体自己重新检查一下。

    对于在遍历下载中有可能出现UnicodeDecodeError: 'utf-8' codec can't decode byte..报错,参考[4-2],具体解决将代码中注释为⭐的一行代码如下修改(加上‘ignore’参数):

#修改字符集参数,一般这种情况出现得较多是在国标码(GBK)和utf8之间选择出现了问题。
#出现异常报错UnicodeDecodeError是由于设置了decode()方法的第二个参数errors为严格(strict)形式造成的,因为默认就是这个参数,将其更改为ignore等即可。
msg_content = b'\\r\\n'.join(lines).decode('utf-8','ignore')

5、python代码调用测试

     以上编写后的.py文件在python中的调用测试情况:

如此,则可以将其打包成exe,这个主要参考[2],通过Pyinstaller -F py_word.p打包成exe。

4.3 C#内部代码实现

      根据需求设计出界面后,则需要代码实现内部相关的功能。

1、保存路径中“选择路径”

效果如下:

因为保存路径需要输入绝对路径,那么路径太长的时候,手动输入还是比较麻烦,直接点击按钮“选择路径”,会弹出窗口,然后自己选择本机中对应的路径文件夹,然后该路径同时显示在textbox中。

实现代码(参考[5]): 

        /// <summary>
        /// 对话框形式选择文件夹路径
        /// </summary>
        /// <returns>返回所选择的文件夹路径</returns>
        public string ChooseFolderPath()
        {
            FolderBrowserDialog fbd = new FolderBrowserDialog();
            fbd.RootFolder = System.Environment.SpecialFolder.Desktop;
            fbd.ShowNewFolderButton = true;
            fbd.Description = "请选择目录";
            if (fbd.ShowDialog() == DialogResult.OK)
            {
                return fbd.SelectedPath.ToString();
            }
            else
            {
                return "";
            }
        }
        //点击事件调用
        private void button3_Click(object sender, EventArgs e)
        {
            string path;
            if ((path = ChooseFolderPath()) == "")
            {
                //do nothing
            }
            else
            {
                textBox3.Text = path;
            }
        }

2、运行结束后清空控件中输入的数据

这个主要代码(参考[6]):

private void ClearAllTxt()
{
    foreach (Control ctr in splitContainer1.Panel1.Controls)
    {
        if (ctr is TextBox)
        {
            TextBox tb = ctr as TextBox;
            tb.Text = string.Empty;
            //ctr.Text = "";
        }
        if (ctr is ComboBox)
        {
            ComboBox cob = ctr as ComboBox;
            cob.SelectedIndex = -1;
        }
     }
}

3、各控件内容判断和处理

       各个textbox控件输入信息的判断和处理,包括空值的警告和报错,然后主要的是邮箱账号和邮箱类型的处理与判断,这给用户有很大的便捷。输入账户不带邮箱后缀,将会代码自动补全,另外检查选择的邮箱类型和输入账户邮箱类型是否一致。主要代码:

            if (textBox1.Text == "")
            {
                MessageBox.Show("邮箱账号不能为空,请输入需要批量下载附件的邮箱账号!", "错误信息", MessageBoxButtons.OK, MessageBoxIcon.Information);
                return;
            }
            string UserName = "";
            string emailtype = "";
            string typeEmail = textBox1.Text;//C#中字符串处理后需要再次赋值的,否则不改变
            int index = typeEmail.IndexOf("@");//检索用户名@后面的字符,获取输入的邮箱类型
            int len = typeEmail.Length;
            if(index == -1)//用户名未带邮箱类型
            {
                if (comboBox1.Text == "")
                {
                    MessageBox.Show("未输入邮箱类型,请手动输入或选择!", "错误信息", MessageBoxButtons.OK, MessageBoxIcon.Information);
                    return;
                }
                else
                {
                    UserName = textBox1.Text + "@" + comboBox1.Text;
                    emailtype = comboBox1.Text;
                }
            }
            else
            {
                typeEmail = typeEmail.Substring(index+1);//取后面的邮箱类型
                //typeEmail = typeEmail.Remove(0,index+1);//取后面的邮箱类型                            
                if (comboBox1.Text == "")
                {//以手动输入为准
                    UserName = textBox1.Text;
                    emailtype = typeEmail;
                }
                else
                {
                    if (typeEmail.Equals(comboBox1.Text))
                    {
                        emailtype = comboBox1.Text;
                        //正确一致
                    }
                    else
                    {
                        MessageBox.Show("选择和手动输入的邮箱类型不一致,请检查!", "错误信息", MessageBoxButtons.OK, MessageBoxIcon.Information);
                        return;
                    }
                }        
            }

这里面主要涉及了字符串的处理(截取,比较等),参考[7、8]。

4、C#调用外部多参数的exe

主要参考[9],代码为:

string str = Environment.CurrentDirectory;//当前路径的字符串
string exePath = str + @"\\GetEmailfile.exe";//这样写的弊端是需将执行的exe文件存放至界面开发的同目录下,命名也是固定的
//调用exe文件
string[] the_args = { UserName, PassWord, FilePath, StarDay, EndDay, emailtype };   // 被调exe接受的参数
StartProcess(exePath, the_args);

这里面存在一定的弊端:需将执行的exe文件存放至界面开发的同目录下,命名也是固定的,大家有什么好的解决方法也可在评论区告知。

五、问题或求教

5.1 超大附件无法下载问题

       因为邮件发送形式多样有附件和超大附件等,很多同学发送大文件也必须发送超大附件。特别是其他类型邮箱之间互发送超大附件还要跳转。

       在这样情况下,大批量的邮件中如果有同学通过邮箱发送了超大附件该怎么一起下载下来呢?我测试发现以上方法不能下载超大附件,只能下载邮件附件呀。有了解的朋友可以写在评论区呀。

5.2 C#调用exe代码中的弊端

     根据我写的代码可以发现调用exe需要与开发的GetEmailFileApp.exe放在一个目录下,即需将执行的exe文件存放至界面开发的同目录下,GetEmailfile.exe命名也必须固定的。大家有什么好的解决方法也可在评论区告知。

5.3 C#程序运行依赖.NET环境

     通常情况下,运行C#程序,需要系统安装.NET 框架,为此生成Release版本的exe后,交给别人使用,需要他人电脑上有.NET环境,这个查过一些资料似乎没有特别有效的解决方法,除了安装环境,另外参考[10](需要携带飞信文件夹)。

六、结语

       基于上述功能的实现,最终这个小程序的核心也就完成了,细枝末节看个人添加,譬如我就在右边加上了软件参数输入提示,Help内有版本等开发信息等。有兴趣的小伙伴也可以自己练练手,后面我也会将该资源进行上传,有需要的小伙伴可自己下载。

参考:

[1]python指定时间邮件附件下载:python 批量下载指定时间段的邮箱附件_沈帅杰的博客-CSDN博客

[2]python外部传参和打包:python 外部传参程序编写并打包exe及其调用方式_nanke_yh的博客-CSDN博客

[3]python实现邮件附件下载:[506]python实现邮件接收、附件下载_周小董-CSDN博客_python下载邮件附件

python批量下载邮件附件 - 臭咸鱼 - 博客园 (cnblogs.com)

python实现邮件接收、附件下载操作_u012209894的专栏-CSDN博客

[4]python异常解决:

Python中常出现TabError: inconsistent use of tabs and spaces in indentation错误解决方法_糖糖的唐的博客-CSDN博客Python3解决UnicodeDecodeError: 'utf-8' codec can't decode byte..问题 终极解决方案_技术笔记-CSDN博客

[5]C#窗口形式打开文件路径:C#winForm窗体,以窗口对话形式选择文件夹、文件路径_iceberg的专栏-CSDN博客

[6]控件数据清空:C#小程序执行后及时清空控件中的数据_nanke_yh的博客-CSDN博客

【C#】一次性清空textbox、combobox中所有的内容_李光 未来科技-CSDN博客_c#清空textbox内容

[7]C#字符串截取:C#字符串的截取函数用法总结_C#教程_脚本之家 (jb51.net)

[8]C#字符串操作:c#中字符串操作函数 - 张@天 - 博客园 (cnblogs.com)

[9]C#调用exe:C# 调用外部exe,且带参数_nanke_yh的博客-CSDN博客

[10]C#程序.NET环境问题:C#程序不用安装.NET环境运行(让C#程序脱离.net框架) - 郑文亮 - 博客园 (cnblogs.com)

以上是关于python实现电子邮件附件指定时间段,批量下载以及C#小程序集成实现的主要内容,如果未能解决你的问题,请参考以下文章

使用Python实现批量发送邮件

java如何实现批量发送邮件

python实现邮件接收附件下载操作

python实现邮件接收附件下载操作

Python批量发邮件--加附件/抄送

使用Python从邮件中下载附件和提取元数据