прерывистый smtp сломанный канал (ошибка 32)

У меня есть скрипт python, который периодически в течение дня отправляет несколько наборов электронных писем (на разные адреса с разным содержимым), используя smtplib. Довольно часто (скажем, примерно в 1 случае из 5 при отправке пакетов из более чем одного электронного письма одновременно) я получаю IOError (ошибка: сломанная труба). Я пытаюсь перезагрузить и выйти из 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

Любые мысли о том, как продолжить отладку этого? Сервер stmp не обслуживается мной, хотя должен поддерживаться в хорошем состоянии (используется для организации ~10 тыс. человек). Первоначально я подключался и отключался от smtpserver после отправки каждого электронного письма, но это приводило к большему количеству ошибок, чем этот метод.

Также было бы безопаснее использовать /usr/sbin/sendmail, а не smtplib?


person dr jimbob    schedule 05.11.2010    source источник


Ответы (1)


Also would it be safer to use /usr/sbin/sendmail rather than smtplib?

С точки зрения обработки очереди сообщений и очереди для повторных попыток. Да, sendmail или другой локальный MTA может справиться с этим за вас.

Any thoughts on how to proceed debugging this?

Захват пакетов. Используйте wireshark для захвата smtp-трафика, посмотрите, что происходит. У вас есть много широкой обработки исключений, которая не обязательно показывает вам точную ошибку.

person MattH    schedule 05.11.2010