处理 python3 邮件附件中文名乱码的问题

最近接手了一个 py 机器人的开发,发现不论传入的附件路径是变量、字符串、先编码后解码的字符串都解决不了名称乱码。

打开源码一看,是这个逻辑:

def send_smtp_mail(server=None,port=25,psw=None,sender=None,receivers=None,cc=None,bcc=None,subject=None,body=None,attachments=None,ssl='no'):
    '''
SMTP发送邮件
    server:smtp服务器  port:端口号   psw:登陆密码  sender:发送方  receivers:接收者
    cc:抄送  bcc:密抄  subject:标题  body:邮件正文  attachments:附件路径

'''
    __logger.debug('smtp Send mail:[' + str(server) + '][' + str(port) + ']')
    re = []
    try:
        msgRoot = MIMEMultipart()
        #msgRoot['Subject'] = subject   构造标题
        msgRoot['Subject'] = Header(subject, 'utf-8').encode()
        msgRoot['Cc'] = "".join(str(cc))
        msgRoot['Bcc'] = "".join(str(bcc))
        msgRoot['From'] = formataddr(["", sender])
        msgRoot['To'] = formataddr(["", receivers])
        msgRoot.attach(MIMEText(body, 'plain', 'utf-8'))

        if attachments != None:
            for attachment in attachments.split(','):
                rst= os.path.exists(attachment)
                if rst :
                        excelFile = open(attachment, 'rb').read()
                        fileName = os.path.basename(os.path.realpath(attachment))
                        att = MIMEApplication(excelFile)
                        att.add_header('Content-Disposition', 'attachment', fileName=('gbk', '', fileName))
                        msgRoot.attach(att)
                else:
                    __logger.debug(u'Attachment path does not exist')


        if receivers != None and receivers != '':
            re = receivers.split(',')
        if cc != None and cc != '':
            re = re + str(cc).split(',')
        if bcc !=None and bcc != '':
            re = re + str(bcc).split(',')

        smtp = smtplib.SMTP()
        if ssl=='yes':
            smtp = smtplib.SMTP_SSL()
        smtp.connect(server,port)
        psw = encrypt.decrypt(psw)
        smtp.login(sender, psw)
        smtp.sendmail(sender,re, msgRoot.as_string())
        smtp.quit() 
    except Exception as e:
        raise e
    finally:
        __logger.echo_msg(u"end execute[sendMail]")

发现这里对附件的处理只是获取真正的附件名,并传入 Content-Disposition 中。在 segmentfault 查到,要给 fileName 进行两次 utf-8 编码、然后用 email.header 库的 make_header() 函数处理一次附件名、才传给 Content-Disposition 中。 修改之后的源码可以正常识别中文名字的附件了,贴一下修改后的源代码:

def send_smtp_mail1(server=None,port=25,psw=None,sender=None,receivers=None,cc=None,bcc=None,subject=None,body=None,attachments=None,ssl='no'):
    '''
SMTP发送邮件
    server:smtp服务器  port:端口号   psw:登陆密码  sender:发送方  receivers:接收者
    cc:抄送  bcc:密抄  subject:标题  body:邮件正文  attachments:附件路径

'''
    __logger.debug('smtp Send mail:[' + str(server) + '][' + str(port) + ']')
    re = []
    try:
        msgRoot = MIMEMultipart()
        #msgRoot['Subject'] = subject   构造标题
        msgRoot['Subject'] = Header(subject, 'utf-8').encode()
        msgRoot['Cc'] = "".join(str(cc))
        msgRoot['Bcc'] = "".join(str(bcc))
        msgRoot['From'] = formataddr(["", sender])
        msgRoot['To'] = formataddr(["", receivers])
        msgRoot.attach(MIMEText(body, 'plain', 'utf-8'))

        if attachments != None:
            for attachment in attachments.split(','):
                rst= os.path.exists(attachment)
                if rst :
                        excelFile = open(attachment, 'rb').read()
                        fileName = os.path.basename(os.path.realpath(attachment))
                        att = MIMEApplication(excelFile)
                        att.add_header('Content-Disposition', 'attachment', fileName = "%s" % make_header([(fileName, 'UTF-8')]).encode('UTF-8') ) 
                        msgRoot.attach(att)
                else:
                    __logger.debug(u'Attachment path does not exist')


        if receivers != None and receivers != '':
            re = receivers.split(',')
        if cc != None and cc != '':
            re = re + str(cc).split(',')
        if bcc !=None and bcc != '':
            re = re + str(bcc).split(',')

        smtp = smtplib.SMTP()
        if ssl=='yes':
            smtp = smtplib.SMTP_SSL()
        smtp.connect(server,port)
        psw = encrypt.decrypt(psw)
        smtp.login(sender, psw)
        smtp.sendmail(sender,re, msgRoot.as_string())
        smtp.quit() 
    except Exception as e:
        raise e
    finally:
        __logger.echo_msg(u"end execute[sendMail]")

所以不要迷信什么产品的内置函数,,业务开发真的有很多细节要考虑周全啊。原始答案在: python3发邮件,附件名称为中文时出错 - SegmentFault 思否

(完)

  • 6
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: 你可以使用 Python 的 poplib 模块来接收邮件附件。首先,你需要连接到邮件服务器,然后登录到你的邮箱,接着可以使用 retr 命令获取邮件的原始文本,最后通过解析邮件内容来获取附件。 下面是一个简单的示例代码: ``` import poplib from email import parser # 连接到邮件服务器 server = poplib.POP3("pop.example.com") # 登录到邮箱 server.user("username") server.pass_("password") # 获取邮件列表 num_messages = len(server.list()[1]) # 获取邮件的原始文本 message_text = server.retr(num_messages)[1] # 将原始文本解析为邮件对象 message = parser.Parser().parsestr("\n".join(message_text)) # 获取附件 for part in message.walk(): if part.get_content_maintype() == "multipart": continue if part.get("Content-Disposition") is None: continue filename = part.get_filename() if not filename: continue # 保存附件到磁盘 with open(filename, "wb") as fp: fp.write(part.get_payload(decode=True)) # 退出邮件服务器 server.quit() ``` 希望这个示例代码能帮助你更好的理解如何使用 poplib 模块接收邮件附件。 ### 回答2: 使用Python可以使用poplib库来接收邮件附件。 poplib是Python标准库中的一个模块,它提供了一个POP3协议客户端,可以与POP3服务器进行通信,用于接收邮件。 首先,我们需要导入poplib库来使用它的功能: ```python import poplib ``` 接下来,我们需要连接到POP3服务器: ```python pop_server = poplib.POP3('pop.example.com') pop_server.user('username') pop_server.pass_('password') ``` 这里pop.example.com为POP3服务器的地址,'username'和'password'分别为账户的用户名和密码,需要替换为实际的值。 连接到服务器后,我们可以使用list()方法来获取邮件的列表: ```python email_list = pop_server.list() ``` email_list是一个包含所有邮件的列表,每个元素是一个包含邮件编号和大小的元组。 接下来,我们可以使用retr()方法来获取特定邮件的内容: ```python email_index = 1 # 获取第一封邮件的内容 response, email_lines, size = pop_server.retr(email_index) ``` email_lines是一个包含邮件内容的列表,每个元素是邮件的一行。 接收邮件附件需要先解析邮件内容,利用email库来处理邮件的各个部分。可以使用以下代码来获取附件: ```python import email email_message = email.message_from_bytes(b'\n'.join(email_lines)) for part in email_message.walk(): if part.get_content_type().startswith('application/'): # 判断附件的类型 attachment_data = part.get_payload(decode=True) file_name = part.get_filename() with open(file_name, 'wb') as f: f.write(attachment_data) ``` 这里使用email库的message_from_bytes()方法将邮件内容转换为邮件对象。然后,我们使用walk()方法遍历邮件的每个部分,判断附件的类型,并使用get_payload()方法获取附件的内容。 最后,将附件保存到本地文件中。 以上就是使用Python通过POP3接收邮件附件的方法。 ### 回答3: 使用Python接收邮件附件,可以通过POP3协议来实现。POP3是一种用于接收邮件的协议,它允许我们从邮件服务器上下载并获取邮件。 首先,我们需要导入相应的模块,如poplib和email。poplib模块提供了POP3协议的接口,email模块则能够解析邮件内容。 接下来,我们需要建立与邮件服务器的连接,使用poplib的POP3方法进行连接。我们需要提供邮件服务器的地址、端口号、用户名和密码来进行身份验证。 连接成功后,我们可以使用list()方法获取邮件的总数量和大小。然后,使用retr()方法打开邮件并获取邮件内容。在获取邮件内容的同时,我们可以使用email模块的相关方法来解析邮件内容,并获取附件的信息。 在解析邮件内容时,我们可以使用email模块的Message和MIMEMultipart类来处理邮件,其中MIMEMultipart类用于解析多部分的邮件(包括附件),而Message类则用于解析其他类型的邮件。 通过遍历邮件的各个部分,我们可以找到附件的相关信息,并将其保存到本地磁盘中。 最后,我们需要调用quit()方法来关闭与邮件服务器的连接。 总结以上步骤,我们可以通过Python的POP3库和email库来实现接收邮件附件的功能。通过连接到邮件服务器、解析邮件内容和保存附件,我们可以轻松地获取邮件附件

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值