source: main/waeup.kofa/trunk/src/waeup/kofa/tests/test_smtp.py @ 10854

Last change on this file since 10854 was 10854, checked in by Henrik Bettermann, 11 years ago

Send cc of all mails to sender.

  • Property svn:keywords set to Id
File size: 14.1 KB
Line 
1# -*- coding: utf-8 -*-
2## $Id: test_smtp.py 10854 2013-12-19 09:43:34Z henrik $
3##
4## Copyright (C) 2012 Uli Fouquet & Henrik Bettermann
5## This program is free software; you can redistribute it and/or modify
6## it under the terms of the GNU General Public License as published by
7## the Free Software Foundation; either version 2 of the License, or
8## (at your option) any later version.
9##
10## This program is distributed in the hope that it will be useful,
11## but WITHOUT ANY WARRANTY; without even the implied warranty of
12## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13## GNU General Public License for more details.
14##
15## You should have received a copy of the GNU General Public License
16## along with this program; if not, write to the Free Software
17## Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
18##
19# Tests for email-related components
20import base64
21import logging
22import tempfile
23import shutil
24import unittest
25from StringIO import StringIO
26from zope.component import getUtility
27from zope.component.hooks import setSite
28from zope.sendmail.interfaces import IMailDelivery
29from waeup.kofa.app import University
30from waeup.kofa.interfaces import IMailService
31from waeup.kofa.smtp import (
32    encode_header_item, encode_address, encode_body, FakeSMTPDelivery,
33    send_mail)
34from waeup.kofa.testing import FunctionalLayer, FunctionalTestCase
35
36#
37# SMTP test config
38#
39
40# Also run tests that send mail to external servers?
41#   If you enable this, please make sure the external smtp hosts and receivers
42#   do exist really and are not bothered by being spammed by a test programme.
43EXTERNAL_MAIL_TESTS = False
44
45# Maybe existing receiver of externally sent mail. If this mail account
46# exists really, it might receive mail from the tests!
47EXTERNAL_MAIL_RECEIVER = 'no-reply@waeup.org'
48
49# Names of mail deliveries to use when external mail tests are enabled.
50#   See local mail.zcml for names of available deliveries.
51EXTERNAL_DIRECT_DELIVERY = 'Direct SMTP on localhost'
52EXTERNAL_QUEUED_DELIVERY = 'Queued SMTP on localhost'
53
54#
55# end of SMTP test config
56#
57
58def external_mail_test(func):
59    if not EXTERNAL_MAIL_TESTS:
60        myself = __file__
61        if myself.endswith('.pyc'):
62            myself = myself[:-2]
63        print "WARNING: external mail tests are skipped!"
64        print "WARNING: edit %s to enable them." % myself
65        return
66    return func
67
68class HelperTests(unittest.TestCase):
69
70    def test_encode_header_item(self):
71        # encoding header items works with strings and unicodes, as
72        # long as we pass in utf-8 encoded stuff.
73        result1 = encode_header_item(u'Plain Name'.encode('utf-8'))
74        result2 = encode_header_item(u'Name with umläut'.encode('utf-8'))
75        result3 = encode_header_item(u'Plain Name')
76        result4 = encode_header_item(u'Name with umläut')
77        result5 = encode_header_item(None)
78        self.assertEqual(result1, u'Plain Name')
79        self.assertEqual(result2, u'=?iso-8859-1?q?Name_with_uml=E4ut?=')
80        self.assertEqual(result3, u'Plain Name')
81        self.assertEqual(result4, u'=?iso-8859-1?q?Name_with_uml=E4ut?=')
82        self.assertTrue(result5 is None)
83        return
84
85    def test_encode_address(self):
86        # we can correctly encode address parts
87        result1 = encode_address('foo@bar.baz')
88        result2 = encode_address(u'foo@bar.baz', 'The Foo' )
89        result3 = encode_address('foo@bar.baz', u'The Foo')
90        result4 = encode_address('foo@bar.baz', u'With Umläut')
91        self.assertEqual(result1, 'foo@bar.baz')
92        self.assertEqual(result2, 'The Foo <foo@bar.baz>')
93        self.assertEqual(result3, 'The Foo <foo@bar.baz>')
94        self.assertEqual(result4,
95                         '=?iso-8859-1?q?With_Uml=E4ut?= <foo@bar.baz>')
96        return
97
98    def test_encode_body(self):
99        result1 = encode_body(u'Simple Text')
100        self.assertEqual(
101            str(result1).split('\n')[1:],
102            ['MIME-Version: 1.0',
103             'Content-Type: text/plain; charset="us-ascii"',
104             'Content-Transfer-Encoding: 7bit',
105             '',
106             'Simple Text'])
107        return
108
109    def test_encode_body_latin1_string(self):
110        # utf-8 encoded byte streams are transferred correctly when
111        # they contain chars unrepresentable in ascii but available in latin1
112        text = u'Simple text with Ümläut'.encode('utf-8')
113        result1 = encode_body(text)
114        result1 = str(result1).split('\n')[1:]
115        self.assertEqual(
116            result1,
117            ['MIME-Version: 1.0',
118             'Content-Type: text/plain; charset="iso-8859-1"',
119             'Content-Transfer-Encoding: quoted-printable',
120             '',
121             'Simple text with =DCml=E4ut'])
122        return
123
124    def test_encode_body_latin1_unicode(self):
125        # unicode strings are transferred correctly when they contain
126        # chars unrepresentable in ascii but available in latin1
127        text = u'Simple text with Ümläut'
128        result1 = encode_body(text)
129        result1 = str(result1).split('\n')[1:]
130        self.assertEqual(
131            result1,
132            ['MIME-Version: 1.0',
133             'Content-Type: text/plain; charset="iso-8859-1"',
134             'Content-Transfer-Encoding: quoted-printable',
135             '',
136             'Simple text with =DCml=E4ut'])
137        return
138
139    def test_encode_body_utf8_string(self):
140        # utf-8 encoded byte streams are transferred correctly when
141        # they contain chars unrepresentable in ascii but available in latin1
142        text = u"Simple text with ü and capital pi: " + unichr(0x3a0)
143        text = text.encode('utf-8') # turn unicode into byte stream
144        result1 = encode_body(text)
145        result1 = str(result1).split('\n')[1:]
146        self.assertEqual(
147            result1,
148            ['MIME-Version: 1.0',
149             'Content-Type: text/plain; charset="utf-8"',
150             'Content-Transfer-Encoding: base64',
151             '',
152             'U2ltcGxlIHRleHQgd2l0aCDDvCBhbmQgY2FwaXRhbCBwaTogzqA=', ''])
153        self.assertEqual(
154            base64.b64decode(result1[-2]).decode('utf-8'),
155            u"Simple text with ü and capital pi: " + unichr(0x3a0))
156        return
157
158    def test_encode_body_utf8_unicode(self):
159        # utf-8 encoded byte streams are transferred correctly when
160        # they contain chars unrepresentable in latin1
161        text = u"Simple text with ü and capital pi: " + unichr(0x3a0)
162        result1 = encode_body(text)
163        result1 = str(result1).split('\n')[1:]
164        self.assertEqual(
165            result1,
166            ['MIME-Version: 1.0',
167             'Content-Type: text/plain; charset="utf-8"',
168             'Content-Transfer-Encoding: base64',
169             '',
170             'U2ltcGxlIHRleHQgd2l0aCDDvCBhbmQgY2FwaXRhbCBwaTogzqA=', ''])
171        self.assertEqual(
172            base64.b64decode(result1[-2]).decode('utf-8'),
173            u"Simple text with ü and capital pi: " + unichr(0x3a0))
174        return
175
176class FunctionalMailerTests(FunctionalTestCase):
177
178    layer = FunctionalLayer
179
180    def setUp(self):
181        super(FunctionalMailerTests, self).setUp()
182        self.setup_logging()
183        return
184
185    def tearDown(self):
186        super(FunctionalMailerTests, self).tearDown()
187        self.teardown_logging()
188        return
189
190    def setup_logging(self):
191        # setup a log-handler that catches all fake mailer output
192        self.stream = StringIO()
193        handler = logging.StreamHandler(self.stream)
194        logger = logging.getLogger('test.smtp')
195        logger.addHandler(handler)
196        logger.setLevel(logging.INFO)
197        return
198
199    def get_fake_smtp_output(self):
200        # get output generated by fake mailer
201        self.stream.flush()
202        self.stream.seek(0)
203        return self.stream.read()
204
205    def teardown_logging(self):
206        # remove the log handler for fake mailer output
207        logger = logging.getLogger('test.smtp')
208        handlers = [x for x in logger.handlers]
209        for handler in handlers:
210            logger.removeHandler(handler)
211        return
212
213    def test_get_fake_mailer(self):
214        # we can get the fake mailer if we want
215        mailer = getUtility(IMailDelivery, name='No email service')
216        self.assertTrue(isinstance(mailer, FakeSMTPDelivery))
217        return
218
219    def test_get_default_mailer(self):
220        # we can get a default mailer if we want
221        mailer = getUtility(IMailService)
222        self.assertTrue(isinstance(mailer(), FakeSMTPDelivery))
223        return
224
225    def test_send_mail(self):
226        # we can really send mail.
227        mail_id = send_mail(
228            u'A sender', u'sender@example.com',
229            u'A recipient', u'recpt@example.com',
230            u'A subject',
231            u'This is a test mail.')
232        self.assertEqual(mail_id, 'fake-message-id@example.com')
233        self.assertEqual(
234            self.get_fake_smtp_output().split('\n'),
235            [u'Sending email from sender@example.com to recpt@example.com:',
236             u'Message:',
237             u'msg: MIME-Version: 1.0',
238             u'msg: Content-Type: text/plain; charset="us-ascii"',
239             u'msg: Content-Transfer-Encoding: 7bit',
240             u'msg: From: A sender <sender@example.com>',
241             u'msg: Cc: A sender <sender@example.com>',
242             u'msg: To: A recipient <recpt@example.com>',
243             u'msg: Subject: A subject',
244             u'msg: ',
245             u'msg: This is a test mail.',
246             u'']
247            )
248        return
249
250    def test_send_mail_utf8_strings(self):
251        # we can send mail with utf-8 encoded strings as input
252        mail_id = send_mail(
253            u'A sender', u'sender@example.com',
254            u'A recipient', u'recpt@example.com',
255            u'A subject',
256            u'This is a test mail with ümläut.'.encode('utf-8'))
257        self.assertEqual(mail_id, 'fake-message-id@example.com')
258        self.assertEqual(
259            self.get_fake_smtp_output().split('\n'),
260            [u'Sending email from sender@example.com to recpt@example.com:',
261             u'Message:', u'msg: MIME-Version: 1.0',
262             u'msg: Content-Type: text/plain; charset="iso-8859-1"',
263             u'msg: Content-Transfer-Encoding: quoted-printable',
264             u'msg: From: A sender <sender@example.com>',
265             u'msg: Cc: A sender <sender@example.com>',
266             u'msg: To: A recipient <recpt@example.com>',
267             u'msg: Subject: A subject',
268             u'msg: ',
269             u'msg: This is a test mail with =FCml=E4ut.',
270             u'']
271            )
272        return
273
274    def test_send_mail_utf8_unicode(self):
275        # we can send mail with utf-8 encoded unicode as input
276        mail_id = send_mail(
277            u'A sender', u'sender@example.com',
278            u'A recipient', u'recpt@example.com',
279            u'A subject',
280            u'This is a test mail with ümläut.')
281        self.assertEqual(mail_id, 'fake-message-id@example.com')
282        self.assertEqual(
283            self.get_fake_smtp_output().split('\n'),
284            [u'Sending email from sender@example.com to recpt@example.com:',
285             u'Message:', u'msg: MIME-Version: 1.0',
286             u'msg: Content-Type: text/plain; charset="iso-8859-1"',
287             u'msg: Content-Transfer-Encoding: quoted-printable',
288             u'msg: From: A sender <sender@example.com>',
289             u'msg: Cc: A sender <sender@example.com>',
290             u'msg: To: A recipient <recpt@example.com>',
291             u'msg: Subject: A subject',
292             u'msg: ',
293             u'msg: This is a test mail with =FCml=E4ut.',
294             u'']
295            )
296        return
297
298
299class ExternalMailerTests(FunctionalTestCase):
300
301    layer = FunctionalLayer
302
303    def setUp(self):
304        super(ExternalMailerTests, self).setUp()
305        # Setup a sample site for each test
306        app = University()
307        self.dc_root = tempfile.mkdtemp()
308        app['datacenter'].setStoragePath(self.dc_root)
309
310        # Prepopulate the ZODB...
311        self.getRootFolder()['app'] = app
312        self.app = self.getRootFolder()['app']
313        return
314
315    def tearDown(self):
316        super(ExternalMailerTests, self).tearDown()
317        shutil.rmtree(self.dc_root)
318        return
319
320    def test_config_default_mailer(self):
321        # The default mailer set in config is 'no mailer'.
322        self.assertEqual(
323            getattr(self.app.get('configuration', None), 'smtp_mailer'),
324            u'No email service')
325        return
326
327    @external_mail_test
328    def test_send_direct_mail(self):
329        # send mail using direct mail delivery
330        self.app['configuration'].smtp_mailer = EXTERNAL_DIRECT_DELIVERY
331        setSite(self.app)
332        result = send_mail(
333            'test program', 'no-reply@waeup.org',
334            'test mail receiver', EXTERNAL_MAIL_RECEIVER,
335            'Test Mail from WAeUP Kofa',
336            'Hi from test mailer!\n\nRegards,\nTest Programme\n'
337            )
338        import transaction
339        transaction.commit() # The mail is really sent when transactions is
340                             # committed
341        self.assertTrue('@' in result)
342        return
343
344    @external_mail_test
345    def test_send_direct_mails(self):
346        # send several mails using direct mail delivery
347        self.app['configuration'].smtp_mailer = EXTERNAL_DIRECT_DELIVERY
348        setSite(self.app)
349        result = send_mail(
350            'test program', 'no-reply@waeup.org',
351            'test mail receiver',
352            '%s, %s' % (EXTERNAL_MAIL_RECEIVER, EXTERNAL_MAIL_RECEIVER),
353            'Test Mail from WAeUP Kofa',
354            'Hi from test mailer!\n\nRegards,\nTest Programme\n'
355            )
356        import transaction
357        transaction.commit() # The mail is really sent when transactions is
358                             # committed
359        self.assertTrue('@' in result)
360        return
361
362    @external_mail_test
363    def test_send_queued_mail(self):
364        # send mail using queued mail delivery
365        self.app['configuration'].smtp_mailer = EXTERNAL_QUEUED_DELIVERY
366        setSite(self.app)
367        result = send_mail(
368            'test program', 'no-reply@waeup.org',
369            'test mail receiver', EXTERNAL_MAIL_RECEIVER,
370            'Test Mail from WAeUP Kofa',
371            'Hi from test mailer!\n\nRegards,\nTest Programme\n'
372            )
373        import transaction
374        transaction.commit() # The mail is really sent when transactions is
375                             # committed
376        self.assertTrue('@' in result)
377        return
Note: See TracBrowser for help on using the repository browser.