From af7631369c96d3da54abb4e1cab44ea61151c481 Mon Sep 17 00:00:00 2001 From: Folker Bernitt Date: Tue, 3 Nov 2015 17:12:15 +0100 Subject: Add new MailSender based on OutgoingMail - Issue #499 - No longer needs local smtp port --- service/pixelated/adapter/services/mail_sender.py | 26 ++++++++++++++++++- service/pixelated/bitmask_libraries/session.py | 2 ++ .../test/unit/adapter/services/test_mail_sender.py | 29 ++++++++++++++++++++-- 3 files changed, 54 insertions(+), 3 deletions(-) (limited to 'service') diff --git a/service/pixelated/adapter/services/mail_sender.py b/service/pixelated/adapter/services/mail_sender.py index 9c3ea3bb..1befd1cc 100644 --- a/service/pixelated/adapter/services/mail_sender.py +++ b/service/pixelated/adapter/services/mail_sender.py @@ -15,10 +15,11 @@ # along with Pixelated. If not, see . from StringIO import StringIO from email.utils import parseaddr +from leap.mail.outgoing.service import OutgoingMail from twisted.internet.defer import Deferred, fail from twisted.mail.smtp import SMTPSenderFactory -from twisted.internet import reactor +from twisted.internet import reactor, defer from pixelated.support.functional import flatten @@ -27,6 +28,29 @@ class SMTPDownException(Exception): Exception.__init__(self, "Couldn't send mail now, try again later.") +class MailSender(object): + + def __init__(self, account_email_address, keymanager, cert_path, remote_smtp_host, remote_smtp_port): + self._from = account_email_address + self._keymanager = keymanager + self._cert_path = cert_path + self._remote_smtp_host = remote_smtp_host + self._remote_smtp_port = remote_smtp_port + + def sendmail(self, mail): + recipients = flatten([mail.to, mail.cc, mail.bcc]) + outgoing_mail = self._create_outgoing_mail() + deferreds = [] + + for recipient in recipients: + deferreds.append(outgoing_mail.send_message(mail.to_smtp_format(), recipient)) + + return defer.gatherResults(deferreds) + + def _create_outgoing_mail(self): + return OutgoingMail(self._from, self._keymanager, self._cert_path, self._cert_path, self._remote_smtp_host, self._remote_smtp_port) + + class LocalSmtpMailSender(object): def __init__(self, account_email_address, smtp): diff --git a/service/pixelated/bitmask_libraries/session.py b/service/pixelated/bitmask_libraries/session.py index da62b084..e13e5863 100644 --- a/service/pixelated/bitmask_libraries/session.py +++ b/service/pixelated/bitmask_libraries/session.py @@ -151,6 +151,8 @@ class LeapSessionFactory(object): smtp = LeapSmtp(self._provider, auth, nicknym.keymanager) + # TODO: Create the new mail sender based on what we have in available LeapSmtp, e.g. the certs + return LeapSession(self._provider, auth, mail_store, soledad, nicknym, smtp) def _lookup_session(self, key): diff --git a/service/test/unit/adapter/services/test_mail_sender.py b/service/test/unit/adapter/services/test_mail_sender.py index 7806c92a..ca68bdc6 100644 --- a/service/test/unit/adapter/services/test_mail_sender.py +++ b/service/test/unit/adapter/services/test_mail_sender.py @@ -13,16 +13,41 @@ # # You should have received a copy of the GNU Affero General Public License # along with Pixelated. If not, see . +from leap.mail.outgoing.service import OutgoingMail from twisted.trial import unittest from mockito import mock, when, verify, any, unstub -from pixelated.adapter.services.mail_sender import LocalSmtpMailSender, SMTPDownException +from pixelated.adapter.services.mail_sender import LocalSmtpMailSender, SMTPDownException, MailSender from pixelated.adapter.model.mail import InputMail +from pixelated.support.functional import flatten from test.support.test_helper import mail_dict -from twisted.internet import reactor +from twisted.internet import reactor, defer from twisted.internet.defer import Deferred +class MailSenderTest(unittest.TestCase): + + def setUp(self): + self._cert_path = u'/some/cert/path' + self._keymanager_mock = mock() + self._remote_smtp_host = 'some.host.test' + self._remote_smtp_port = 1234 + + def tearDown(self): + unstub() + + def test_iterates_over_recipients(self): + sender = MailSender('someone@somedomain.tld', self._keymanager_mock, self._cert_path, self._remote_smtp_host, self._remote_smtp_port) + input_mail = InputMail.from_dict(mail_dict()) + + when(OutgoingMail).send_message(any(), any()).thenAnswer(lambda: defer.succeed(None)) + + sender.sendmail(input_mail) + + for recipient in flatten([input_mail.to, input_mail.cc, input_mail.bcc]): + verify(OutgoingMail).send_message(any(), recipient) + + class LocalSmtpMailSenderTest(unittest.TestCase): def setUp(self): self.smtp = mock() -- cgit v1.2.3