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

Last change on this file since 17296 was 16428, checked in by Henrik Bettermann, 4 years ago

Fix test.

  • Property svn:keywords set to Id
File size: 16.0 KB
Line 
1# -*- coding: utf-8 -*-
2## $Id: test_smtp.py 16428 2021-03-23 13:15:55Z 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
58
59def external_mail_test(func):
60    if not EXTERNAL_MAIL_TESTS:
61        myself = __file__
62        if myself.endswith('.pyc'):
63            myself = myself[:-1]
64        print "WARNING: external mail tests are skipped!"
65        print "WARNING: edit %s to enable them." % myself
66        return
67    return func
68
69
70class HelperTests(unittest.TestCase):
71
72    def test_encode_header_item(self):
73        # encoding header items works with strings and unicodes, as
74        # long as we pass in utf-8 encoded stuff.
75        result1 = encode_header_item(u'Plain Name'.encode('utf-8'))
76        result2 = encode_header_item(u'Name with umläut'.encode('utf-8'))
77        result3 = encode_header_item(u'Plain Name')
78        result4 = encode_header_item(u'Name with umläut')
79        result5 = encode_header_item(None)
80        self.assertEqual(result1, u'Plain Name')
81        self.assertEqual(result2, u'=?iso-8859-1?q?Name_with_uml=E4ut?=')
82        self.assertEqual(result3, u'Plain Name')
83        self.assertEqual(result4, u'=?iso-8859-1?q?Name_with_uml=E4ut?=')
84        self.assertTrue(result5 is None)
85        return
86
87    def test_encode_address(self):
88        # we can correctly encode address parts
89        result1 = encode_address('foo@bar.baz')
90        result2 = encode_address(u'foo@bar.baz', 'The Foo')
91        result3 = encode_address('foo@bar.baz', u'The Foo')
92        result4 = encode_address('foo@bar.baz', u'With Umläut')
93        self.assertEqual(result1, 'foo@bar.baz')
94        self.assertEqual(result2, 'The Foo <foo@bar.baz>')
95        self.assertEqual(result3, 'The Foo <foo@bar.baz>')
96        self.assertEqual(result4,
97                         '=?iso-8859-1?q?With_Uml=E4ut?= <foo@bar.baz>')
98        return
99
100    def test_encode_body(self):
101        result1 = encode_body(u'Simple Text')
102        self.assertEqual(
103            str(result1).split('\n')[1:],
104            ['MIME-Version: 1.0',
105             'Content-Type: text/plain; charset="us-ascii"',
106             'Content-Transfer-Encoding: 7bit',
107             '',
108             'Simple Text'])
109        return
110
111    def test_encode_body_latin1_string(self):
112        # utf-8 encoded byte streams are transferred correctly when
113        # they contain chars unrepresentable in ascii but available in latin1
114        text = u'Simple text with Ümläut'.encode('utf-8')
115        result1 = encode_body(text)
116        result1 = str(result1).split('\n')[1:]
117        self.assertEqual(
118            result1,
119            ['MIME-Version: 1.0',
120             'Content-Type: text/plain; charset="iso-8859-1"',
121             'Content-Transfer-Encoding: quoted-printable',
122             '',
123             'Simple text with =DCml=E4ut'])
124        return
125
126    def test_encode_body_latin1_unicode(self):
127        # unicode strings are transferred correctly when they contain
128        # chars unrepresentable in ascii but available in latin1
129        text = u'Simple text with Ümläut'
130        result1 = encode_body(text)
131        result1 = str(result1).split('\n')[1:]
132        self.assertEqual(
133            result1,
134            ['MIME-Version: 1.0',
135             'Content-Type: text/plain; charset="iso-8859-1"',
136             'Content-Transfer-Encoding: quoted-printable',
137             '',
138             'Simple text with =DCml=E4ut'])
139        return
140
141    def test_encode_body_utf8_string(self):
142        # utf-8 encoded byte streams are transferred correctly when
143        # they contain chars unrepresentable in ascii but available in latin1
144        text = u"Simple text with ü and capital pi: " + unichr(0x3a0)
145        text = text.encode('utf-8')  # turn unicode into byte stream
146        result1 = encode_body(text)
147        result1 = str(result1).split('\n')[1:]
148        self.assertEqual(
149            result1,
150            ['MIME-Version: 1.0',
151             'Content-Type: text/plain; charset="utf-8"',
152             'Content-Transfer-Encoding: base64',
153             '',
154             'U2ltcGxlIHRleHQgd2l0aCDDvCBhbmQgY2FwaXRhbCBwaTogzqA=', ''])
155        self.assertEqual(
156            base64.b64decode(result1[-2]).decode('utf-8'),
157            u"Simple text with ü and capital pi: " + unichr(0x3a0))
158        return
159
160    def test_encode_body_utf8_unicode(self):
161        # utf-8 encoded byte streams are transferred correctly when
162        # they contain chars unrepresentable in latin1
163        text = u"Simple text with ü and capital pi: " + unichr(0x3a0)
164        result1 = encode_body(text)
165        result1 = str(result1).split('\n')[1:]
166        self.assertEqual(
167            result1,
168            ['MIME-Version: 1.0',
169             'Content-Type: text/plain; charset="utf-8"',
170             'Content-Transfer-Encoding: base64',
171             '',
172             'U2ltcGxlIHRleHQgd2l0aCDDvCBhbmQgY2FwaXRhbCBwaTogzqA=', ''])
173        self.assertEqual(
174            base64.b64decode(result1[-2]).decode('utf-8'),
175            u"Simple text with ü and capital pi: " + unichr(0x3a0))
176        return
177
178
179class FunctionalMailerTests(FunctionalTestCase):
180
181    layer = FunctionalLayer
182
183    def setUp(self):
184        super(FunctionalMailerTests, self).setUp()
185        self.setup_logging()
186        return
187
188    def tearDown(self):
189        super(FunctionalMailerTests, self).tearDown()
190        self.teardown_logging()
191        return
192
193    def setup_logging(self):
194        # setup a log-handler that catches all fake mailer output
195        self.stream = StringIO()
196        handler = logging.StreamHandler(self.stream)
197        logger = logging.getLogger('test.smtp')
198        logger.addHandler(handler)
199        logger.setLevel(logging.INFO)
200        return
201
202    def get_fake_smtp_output(self):
203        # get output generated by fake mailer
204        self.stream.flush()
205        self.stream.seek(0)
206        return self.stream.read()
207
208    def teardown_logging(self):
209        # remove the log handler for fake mailer output
210        logger = logging.getLogger('test.smtp')
211        handlers = [x for x in logger.handlers]
212        for handler in handlers:
213            logger.removeHandler(handler)
214        return
215
216    def test_get_fake_mailer(self):
217        # we can get the fake mailer if we want
218        mailer = getUtility(IMailDelivery, name='No email service')
219        self.assertTrue(isinstance(mailer, FakeSMTPDelivery))
220        return
221
222    def test_get_default_mailer(self):
223        # we can get a default mailer if we want
224        mailer = getUtility(IMailService)
225        self.assertTrue(isinstance(mailer(), FakeSMTPDelivery))
226        return
227
228    def test_send_mail(self):
229        # we can really send mail.
230        mail_id = send_mail(
231            u'A sender', u'sender@example.com',
232            u'A recipient', u'recpt@example.com',
233            u'A subject',
234            u'This is a test mail.')
235        self.assertEqual(mail_id, 'fake-message-id@example.com')
236        self.assertEqual(
237            self.get_fake_smtp_output().split('\n'),
238            [u'Sending email from no-reply@waeup.org to '
239             u'recpt@example.com, sender@example.com:',
240             u'Message:',
241             u'msg: MIME-Version: 1.0',
242             u'msg: Content-Type: text/plain; charset="us-ascii"',
243             u'msg: Content-Transfer-Encoding: 7bit',
244             u'msg: From: A sender <no-reply@waeup.org>',
245             u'msg: To: A recipient <recpt@example.com>',
246             # u'msg: Cc: A sender <sender@example.com>',
247             u'msg: Reply-To: A sender <sender@example.com>',
248             u'msg: Subject: A subject',
249             u'msg: ',
250             u'msg: This is a test mail.',
251             u'']
252            )
253        return
254
255    def test_send_mail_with_cc(self):
256        # with cc=True the message will be CCed to the from address
257        mail_id = send_mail(
258            u'A sender', u'sender@example.com',
259            u'A recipient', u'recpt@example.com',
260            u'A subject',
261            u'This is a test mail.',None,
262            (('recpt1@example.com', 'A cc recipient'),),
263            (('recpt2@example.com', 'A bcc recipient'),
264             ('recpt3@example.com', 'A 2nd bcc recipient')),
265            )
266        self.assertEqual(mail_id, 'fake-message-id@example.com')
267        self.assertEqual(
268            self.get_fake_smtp_output().split('\n'),
269            [u'Sending email from no-reply@waeup.org to '
270             u'recpt@example.com, recpt1@example.com, recpt2@example.com, '
271             u'recpt3@example.com, sender@example.com:',
272             u'Message:',
273             u'msg: MIME-Version: 1.0',
274             u'msg: Content-Type: text/plain; charset="us-ascii"',
275             u'msg: Content-Transfer-Encoding: 7bit',
276             u'msg: From: A sender <no-reply@waeup.org>',
277             u'msg: To: A recipient <recpt@example.com>',
278             u'msg: Cc: A cc recipient <recpt1@example.com>',
279             u'msg: Bcc: A bcc recipient <recpt2@example.com>',
280             u'msg: Bcc: A 2nd bcc recipient <recpt3@example.com>',
281             u'msg: Reply-To: A sender <sender@example.com>',
282             u'msg: Subject: A subject',
283             u'msg: ',
284             u'msg: This is a test mail.',
285             u'']
286
287
288            )
289        return
290
291    def test_send_mail_utf8_strings(self):
292        # we can send mail with utf-8 encoded strings as input
293        mail_id = send_mail(
294            u'A sender', u'sender@example.com',
295            u'A recipient', u'recpt@example.com',
296            u'A subject',
297            u'This is a test mail with ümläut.'.encode('utf-8'))
298        self.assertEqual(mail_id, 'fake-message-id@example.com')
299        self.assertEqual(
300            self.get_fake_smtp_output().split('\n'),
301            [u'Sending email from no-reply@waeup.org '
302             u'to recpt@example.com, sender@example.com:',
303             u'Message:', u'msg: MIME-Version: 1.0',
304             u'msg: Content-Type: text/plain; charset="iso-8859-1"',
305             u'msg: Content-Transfer-Encoding: quoted-printable',
306             u'msg: From: A sender <no-reply@waeup.org>',
307             u'msg: To: A recipient <recpt@example.com>',
308             # u'msg: Cc: A sender <sender@example.com>',
309             u'msg: Reply-To: A sender <sender@example.com>',
310             u'msg: Subject: A subject',
311             u'msg: ',
312             u'msg: This is a test mail with =FCml=E4ut.',
313             u'']
314            )
315        return
316
317    def test_send_mail_utf8_unicode(self):
318        # we can send mail with utf-8 encoded unicode as input
319        mail_id = send_mail(
320            u'A sender', u'sender@example.com',
321            u'A recipient', u'recpt@example.com',
322            u'A subject',
323            u'This is a test mail with ümläut.')
324        self.assertEqual(mail_id, 'fake-message-id@example.com')
325        self.assertEqual(
326            self.get_fake_smtp_output().split('\n'),
327            [u'Sending email from no-reply@waeup.org '
328             u'to recpt@example.com, sender@example.com:',
329             u'Message:', u'msg: MIME-Version: 1.0',
330             u'msg: Content-Type: text/plain; charset="iso-8859-1"',
331             u'msg: Content-Transfer-Encoding: quoted-printable',
332             u'msg: From: A sender <no-reply@waeup.org>',
333             u'msg: To: A recipient <recpt@example.com>',
334             # u'msg: Cc: A sender <sender@example.com>',
335             u'msg: Reply-To: A sender <sender@example.com>',
336             u'msg: Subject: A subject',
337             u'msg: ',
338             u'msg: This is a test mail with =FCml=E4ut.',
339             u'']
340            )
341        return
342
343
344class ExternalMailerTests(FunctionalTestCase):
345
346    layer = FunctionalLayer
347
348    def setUp(self):
349        super(ExternalMailerTests, self).setUp()
350        # Setup a sample site for each test
351        app = University()
352        self.dc_root = tempfile.mkdtemp()
353        app['datacenter'].setStoragePath(self.dc_root)
354
355        # Prepopulate the ZODB...
356        self.getRootFolder()['app'] = app
357        self.app = self.getRootFolder()['app']
358        return
359
360    def tearDown(self):
361        super(ExternalMailerTests, self).tearDown()
362        shutil.rmtree(self.dc_root)
363        return
364
365    def test_config_default_mailer(self):
366        # The default mailer set in config is 'no mailer'.
367        self.assertEqual(
368            getattr(self.app.get('configuration', None), 'smtp_mailer'),
369            u'No email service')
370        return
371
372    @external_mail_test
373    def test_send_direct_mail(self):
374        # send mail using direct mail delivery
375        self.app['configuration'].smtp_mailer = EXTERNAL_DIRECT_DELIVERY
376        setSite(self.app)
377        result = send_mail(
378            'test program', 'no-reply@waeup.org',
379            'test mail receiver', EXTERNAL_MAIL_RECEIVER,
380            'Test Mail from WAeUP Kofa',
381            'Hi from test mailer!\n\nRegards,\nTest Programme\n'
382            )
383        import transaction
384        transaction.commit()  # The mail is really sent when transactions is
385                              # committed
386        self.assertTrue('@' in result)
387        return
388
389    @external_mail_test
390    def test_send_direct_mails(self):
391        # send several mails using direct mail delivery
392        self.app['configuration'].smtp_mailer = EXTERNAL_DIRECT_DELIVERY
393        setSite(self.app)
394        result = send_mail(
395            'test program', 'no-reply@waeup.org',
396            'test mail receiver',
397            '%s, %s' % (EXTERNAL_MAIL_RECEIVER, EXTERNAL_MAIL_RECEIVER),
398            'Test Mail from WAeUP Kofa',
399            'Hi from test mailer!\n\nRegards,\nTest Programme\n'
400            )
401        import transaction
402        transaction.commit()  # The mail is really sent when transactions is
403                              # committed
404        self.assertTrue('@' in result)
405        return
406
407    @external_mail_test
408    def test_send_queued_mail(self):
409        # send mail using queued mail delivery
410        self.app['configuration'].smtp_mailer = EXTERNAL_QUEUED_DELIVERY
411        setSite(self.app)
412        result = send_mail(
413            'test program', 'no-reply@waeup.org',
414            'test mail receiver', EXTERNAL_MAIL_RECEIVER,
415            'Test Mail from WAeUP Kofa',
416            'Hi from test mailer!\n\nRegards,\nTest Programme\n'
417            )
418        import transaction
419        transaction.commit()  # The mail is really sent when transactions is
420                              # committed
421        self.assertTrue('@' in result)
422        return
Note: See TracBrowser for help on using the repository browser.