python - 间歇性 smtp 管道损坏 (errno 32)

标签 python smtp smtplib

我有一个 python 脚本,它全天定期使用 smtplib 发送几组电子邮件(发送到具有不同内容的不同地址)。有时(例如,当同时发送多批电子邮件时,大约五分之一),我会收到 IOError(errno:管道损坏)。我尝试重置并退出 SMTP 服务器,然后再次连接到服务器并尝试重新发送,但如果第一次失败,则总是会失败(有相同的异常(exception))。 SMTP 服务器由学院维护,应该是可靠的(只要您位于 Intranet 上,就允许免登录电子邮件)。

忽略下面代码的丑陋(缺乏 DRY),任何人都可以建议一种更可靠的连接方式吗?

我创建一个名为 EmailSet 的类,它将发送一批包含成员函数 send_emails 的几封电子邮件:

class EmailSet(object):
    ...
    def send_emails(self):
        try: # Connect to server
            server = smtplib.SMTP( smtp_server_name_str, 25)
            server.set_debuglevel(self.verbose)
            server.ehlo()
            for email in self.email_set:
                try: # send 1st mail
                    logging.debug("Sending email to %r" % email.recipients)        
                    response_dict = server.sendmail(email.fromaddr, email.recipients, email.msg_str())
                    logging.info("Sent email to %r" % email.recipients)        
                except Exception as inst:
                    logging.error('RD: %r' % response_dict)
                    logging.error("Email Sending Failed")
                    logging.error("%r %s" % ( type(inst), inst ) )
                    try: # send second mail
                        logging.info("Second Attempt to send to %r" % email.recipients)
                        try:
                            server.rset() 
                            server.quit()
                        except:
                            pass
                        time.sleep(60) # wait 60s
                        server = smtplib.SMTP( smtp_server_name_str, 25)
                        server.set_debuglevel(self.verbose)
                        server.ehlo()
                        response_dict = server.sendmail(email.fromaddr, email.recipients, email.msg_str())
                        logging.info("Sent email to %r (2nd Attempt)" % email.recipients)        
                    except Exception as inst:
                        try:
                            logging.error('RD: %r' % response_dict)
                        except:
                            pass
                        logging.error("Second Attempt Email Sending Failed")
        except:
            logging.debug("Can't connect to server")
        finally:
            logging.debug("Reseting and Quitting Server")
            server.rset()
            server.quit()
            logging.debug("Successfully Quit Server")
        return True

关于如何继续调试这个有什么想法吗? stmp 服务器不是由我维护的,但应该维护良好(用于约 10k 人的组织)。我最初在发送每封电子邮件后连接并断开与 smtp 服务器的连接,但这产生的错误比此方法更多。

此外,使用/usr/sbin/sendmail 而不是 smtplib 会更安全吗?

最佳答案

另外,使用/usr/sbin/sendmail 而不是 smtplib 更安全吗?

从处理消息队列和重试队列运行程序的角度来看。是的,sendmail 或其他本地 MTA 可以为您处理此问题。

关于如何继续调试这个问题有什么想法吗?

数据包捕获。使用wireshark捕获smtp流量,看看发生了什么。您有很多广泛的异常处理,但不一定会向您显示确切的错误。

关于python - 间歇性 smtp 管道损坏 (errno 32),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4106141/

相关文章:

python - 如何创建特定类型但为空的列表

python - 如何在我的树莓派上打开终端并自动运行其中的程序?

python - 加速 "closest"字符串匹配算法

ruby-on-rails - 使用 Amazon SES 对 Rails 应用程序进行 DKIM 签名

ruby-on-rails - 尝试通过 SMTP 和 Rails 3.1.3 使用 Amazon SES 时出现 EOFError 错误

ssl - smtplib.SMTP starttls 失败并出现 tlsv1 警报解码错误

python - 如何使用 matplotlib 渲染 latex 矩阵

c# - GMAIL SMTP : A call to SSPI failed exception - The function requested is not supported

python - 为什么从列表发送电子邮件后“TO”字段为空?

python - 如何将 smtp 调试信息记录到文件中?