commit be223814812bad96000da37ab7690704214b2f5a Author: Isis Lovecruft isis@torproject.org Date: Tue May 20 18:20:42 2014 +0000
Add utility unittest.TestCase mixin class for testing SMTP.
* ADD a new class, `test_email_server.SMTPTestsMixin`, which has the following methods useful for testing SMTP within `t.t.unittest.TestCase` subclasses:
- SMTPTestsMixin._buildEmail(fromAddr=None, toAddr=None, subject=None, body=None) Creates email text (including headers) for use in an SMTP DATA segment. Includes the SMTP DATA EOM command ('.') at the end. If no keyword arguments are given, the defaults are fairly sane.
- SMTPTestsMixin._buildSMTP(commands) Format a list of SMTP protocol commands into a string, using the proper protocol delimiter.
- SMTPTestsMixin._test(commands, expected) Send the commands to the TestCase's protocol (this must be the `proto` attribute of your `TestCase`), and then check that the expected output matches what was received from `TestCase.transport` (another attribute you must assign). --- lib/bridgedb/test/test_email_server.py | 113 ++++++++++++++++++++++++++++++++ lib/bridgedb/test/util.py | 10 +++ 2 files changed, 123 insertions(+)
diff --git a/lib/bridgedb/test/test_email_server.py b/lib/bridgedb/test/test_email_server.py index 23c65b8..30d27f1 100644 --- a/lib/bridgedb/test/test_email_server.py +++ b/lib/bridgedb/test/test_email_server.py @@ -26,9 +26,12 @@ from bridgedb.persistent import Conf from bridgedb.schedule import Unscheduled from bridgedb.test.test_HTTPServer import DummyBridge from bridgedb.test.util import fileCheckDecorator +from bridgedb.test.util import TestCaseMixin
from twisted.python import log from twisted.internet import defer +from twisted.internet import reactor +from twisted.test import proto_helpers from twisted.trial import unittest
@@ -416,6 +419,116 @@ class MailDeliveryTest(unittest.TestCase): self.delivery.validateTo, user)
+class SMTPTestCaseMixin(TestCaseMixin): + """Utility methods for use within any subclasses of + :api:`twisted.trial.unittest.TestCase` which test SMTP. + + To use me, subclass :api:`twisted.trial.unittest.TestCase` and mix me into + the middle of your class inheritance declarations, like so:: + + class ExampleSMTPTests(SMTPTestCaseMixin, unittest.TestCase): + pass + + and then make certain that your ``TestCase`` subclass has its ``proto`` + and attribute assigned properly:: + + class ExampleSMTPTests(SMTPTestCaseMixin, unittest.TestCase): + def setUp(self): + factory = twisted.mail.smtp.SMTPFactory() + self.proto = self.factory.buildProtocol(('127.0.0.1', 0)) + + + :ivar proto: A :api:`Protocol <twisted.internet.protocol.Protocol>` + associated with a + :api:`ServerFactory <twisted.internet.protocol.ServerFactory>`. + :ivar transport: Anything that implements + :api:`ITransport <twisted.internet.interfaces.ITransport>`. The default + is a :api:`twisted.test.proto_helpers.StringTransportWithDisconection`. + :ivar str smtpFromAddr: The default email address for the server. + """ + + proto = None + transport = proto_helpers.StringTransportWithDisconnection() + smtpFromAddr = None + + def tearDown(self): + """Cleanup method after each ``test_*`` method runs; removes timed out + connections on the reactor and clears the :ivar:`transport`. + """ + self.transport.clear() # Clear bytes from the transport. + + for delay in reactor.getDelayedCalls(): + try: + delay.cancel() + except (AlreadyCalled, AlreadyCancelled): + pass + + def _buildEmail(self, fromAddr=None, toAddr=None, subject=None, body=None): + """Creates email text (including headers) for use in an SMTP DATA + segment. Includes the SMTP DATA EOM command ('.') at the end. If no + keyword arguments are given, the defaults are fairly sane. + + Suitable for testing a :class:`bridgedb.email.server.MailFactory`. + + :param str fromAddr: An email address for the 'From:' header. + :param str toAddr: An email address for the 'To:' header. + :param str subject: The contents of the 'Subject:' header. + :param str body: The contents of the email body. + :rtype: str + :returns: The email text. + """ + fromAddr = fromAddr if fromAddr else 'testing@localhost' + toAddr = toAddr if toAddr else self.smtpFromAddr + subject = subject if subject else 'testing testing one two three' + body = body if body else 'get bridges' + + contents = ['From: %s' % fromAddr, + 'To: %s' % toAddr, + 'Subject: %s' % subject, + '\r\n %s' % body, + '.'] # SMTP DATA EOM command + emailText = self.proto.delimiter.join(contents) + return emailText + + def _buildSMTP(self, commands): + """Format a list of SMTP protocol commands into a string, using the proper + protocol delimiter. + + :param list commands: A list of raw SMTP-protocol command lines. + :rtype: str + :returns: The string for sending those **commands**, suitable for + giving to a :api:`twisted.internet.Protocol.dataReceived` method. + """ + data = self.proto.delimiter.join(commands) + self.proto.delimiter + return data + + def _test(self, commands, expected, noisy=False): + """Send the SMTP **commands** to the ``dataReceived`` method of your + TestCase's protocol (this must be the `proto` attribute of your + `TestCase`, i.e. this uses ``TestCase.proto.dataReceived``). Next, + check that the substring which is **expected** to be within the + server's output matches what was received from :ivar`transport`. + + :param list commands: A sequence of raw SMTP command lines. This will + automatically be passed to :meth:`_buildSMTP`. + :param str expected: A substring which should occur in the "server" + output (taken from the :ivar:`transport`). + :param bool noisy: If ``True``, print the conversation between the + "client" and the "server" in a nicely formatted manner. + """ + data = self._buildSMTP(commands) + self.proto.dataReceived(data) + recv = self.transport.value() + + if noisy: + client = data.replace('\r\n', '\r\n ') + server = recv.replace('\r\n', '\r\n\t\t ') + print('\n CLIENT --------->', '\n %s' % client) + print('\t\t', '<--------- SERVER', '\n\t\t %s' % server) + + self.assertSubstring(expected, recv) + + class EmailServerServiceTests(unittest.TestCase): def setUp(self): self.config = _createConfig() diff --git a/lib/bridgedb/test/util.py b/lib/bridgedb/test/util.py index a012597..ca68d1d 100644 --- a/lib/bridgedb/test/util.py +++ b/lib/bridgedb/test/util.py @@ -14,11 +14,14 @@ from __future__ import print_function from __future__ import unicode_literals
+import abc import doctest import os
from functools import wraps
+from twisted.trial import unittest +
def fileCheckDecorator(func): """Method decorator for a t.t.unittest.TestCase test_* method. @@ -62,5 +65,12 @@ def fileCheckDecorator(func): % (str(description), dst, src)) return wrapper
+ +class TestCaseMixin: + """Subclasses of me can be used as mix-in classes with ``TestCase``s.""" + __metaclass__ = abc.ABCMeta +TestCaseMixin.register(unittest.TestCase) + + if __name__ == "__main__": doctest.run_docstring_examples(fileCheckDecorator, None)
tor-commits@lists.torproject.org