间歇性SMTP断管(错误号32)

1 投票
1 回答
1862 浏览
提问于 2025-04-16 06:36

我有一个Python脚本,它会定期发送几组电子邮件(发送到不同的地址,内容也不一样)。但是,有时候(大约每发送五次邮件就会遇到一次),我会遇到一个IOError(错误号:broken pipe)。我尝试重置和退出SMTP服务器,然后再连接到服务器并重新发送邮件,但如果第一次发送失败,第二次也总是失败,错误信息是一样的。这个SMTP服务器是学校维护的,应该是可靠的(只要在内部网络上,就可以发送不需要登录的邮件)。

虽然下面的代码有点乱(没有遵循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

关于如何调试这个问题,有什么想法吗?这个SMTP服务器不是我维护的,但应该维护得很好(用于大约1万人的组织)。我最开始在发送每封邮件后就连接和断开SMTP服务器,但那样产生的错误比现在的方法还要多。

另外,使用/usr/sbin/sendmail会比smtplib更安全吗?

1 个回答

1

那么使用/usr/sbin/sendmail会比smtplib更安全吗?

从处理消息队列和重试的角度来看,使用sendmail或者其他本地邮件传输代理(MTA)确实可以帮你处理这些问题。

关于如何进行调试,你有什么想法吗?

可以使用数据包捕获工具。用wireshark来捕获SMTP流量,看看发生了什么。你现在的异常处理比较宽泛,可能并没有显示出具体的错误。

撰写回答