blob: dd51ac99d73a0e3ad682497b2a7a67197e7b2c07 [file] [log] [blame]
Christian Heimesd3956292008-11-05 19:48:27 +00001"""Test script for poplib module."""
2
3# Modified by Giampaolo Rodola' to give poplib.POP3 and poplib.POP3_SSL
4# a real test suite
5
Guido van Rossumd8faa362007-04-27 19:54:29 +00006import poplib
Christian Heimesd3956292008-11-05 19:48:27 +00007import asyncore
8import asynchat
9import socket
10import os
Guido van Rossumd8faa362007-04-27 19:54:29 +000011import time
Antoine Pitroud3f8ab82010-04-24 21:26:44 +000012import errno
Guido van Rossumd8faa362007-04-27 19:54:29 +000013
14from unittest import TestCase
Christian Heimesd3956292008-11-05 19:48:27 +000015from test import support as test_support
Victor Stinner45df8202010-04-28 22:31:17 +000016threading = test_support.import_module('threading')
Guido van Rossumd8faa362007-04-27 19:54:29 +000017
Christian Heimesd3956292008-11-05 19:48:27 +000018HOST = test_support.HOST
19PORT = 0
Guido van Rossumd8faa362007-04-27 19:54:29 +000020
Antoine Pitrou8618d742012-11-23 20:13:48 +010021SUPPORTS_SSL = False
22if hasattr(poplib, 'POP3_SSL'):
23 import ssl
24
25 SUPPORTS_SSL = True
26 CERTFILE = os.path.join(os.path.dirname(__file__) or os.curdir, "keycert.pem")
27
Christian Heimesd3956292008-11-05 19:48:27 +000028# the dummy data returned by server when LIST and RETR commands are issued
29LIST_RESP = b'1 1\r\n2 2\r\n3 3\r\n4 4\r\n5 5\r\n.\r\n'
30RETR_RESP = b"""From: postmaster@python.org\
31\r\nContent-Type: text/plain\r\n\
32MIME-Version: 1.0\r\n\
33Subject: Dummy\r\n\
34\r\n\
35line1\r\n\
36line2\r\n\
37line3\r\n\
38.\r\n"""
Guido van Rossumd8faa362007-04-27 19:54:29 +000039
Christian Heimesd3956292008-11-05 19:48:27 +000040
41class DummyPOP3Handler(asynchat.async_chat):
42
Antoine Pitrou25cee192012-11-23 20:07:39 +010043 CAPAS = {'UIDL': [], 'IMPLEMENTATION': ['python-testlib-pop-server']}
44
Christian Heimesd3956292008-11-05 19:48:27 +000045 def __init__(self, conn):
46 asynchat.async_chat.__init__(self, conn)
47 self.set_terminator(b"\r\n")
48 self.in_buffer = []
Mark Dickinsonea1158f2009-08-06 16:06:25 +000049 self.push('+OK dummy pop3 server ready. <timestamp>')
Antoine Pitrou8618d742012-11-23 20:13:48 +010050 self.tls_active = False
51 self.tls_starting = False
Christian Heimesd3956292008-11-05 19:48:27 +000052
53 def collect_incoming_data(self, data):
54 self.in_buffer.append(data)
55
56 def found_terminator(self):
57 line = b''.join(self.in_buffer)
58 line = str(line, 'ISO-8859-1')
59 self.in_buffer = []
60 cmd = line.split(' ')[0].lower()
61 space = line.find(' ')
62 if space != -1:
63 arg = line[space + 1:]
64 else:
65 arg = ""
66 if hasattr(self, 'cmd_' + cmd):
67 method = getattr(self, 'cmd_' + cmd)
68 method(arg)
69 else:
70 self.push('-ERR unrecognized POP3 command "%s".' %cmd)
71
72 def handle_error(self):
73 raise
74
75 def push(self, data):
76 asynchat.async_chat.push(self, data.encode("ISO-8859-1") + b'\r\n')
77
78 def cmd_echo(self, arg):
79 # sends back the received string (used by the test suite)
80 self.push(arg)
81
82 def cmd_user(self, arg):
83 if arg != "guido":
84 self.push("-ERR no such user")
85 self.push('+OK password required')
86
87 def cmd_pass(self, arg):
88 if arg != "python":
89 self.push("-ERR wrong password")
90 self.push('+OK 10 messages')
91
92 def cmd_stat(self, arg):
93 self.push('+OK 10 100')
94
95 def cmd_list(self, arg):
96 if arg:
Georg Brandl7e27abb2013-10-27 07:23:53 +010097 self.push('+OK %s %s' % (arg, arg))
Christian Heimesd3956292008-11-05 19:48:27 +000098 else:
99 self.push('+OK')
100 asynchat.async_chat.push(self, LIST_RESP)
101
102 cmd_uidl = cmd_list
103
104 def cmd_retr(self, arg):
105 self.push('+OK %s bytes' %len(RETR_RESP))
106 asynchat.async_chat.push(self, RETR_RESP)
107
108 cmd_top = cmd_retr
109
110 def cmd_dele(self, arg):
111 self.push('+OK message marked for deletion.')
112
113 def cmd_noop(self, arg):
114 self.push('+OK done nothing.')
115
116 def cmd_rpop(self, arg):
117 self.push('+OK done nothing.')
118
Mark Dickinsonea1158f2009-08-06 16:06:25 +0000119 def cmd_apop(self, arg):
120 self.push('+OK done nothing.')
121
Giampaolo Rodolà95bcb932011-02-25 22:28:24 +0000122 def cmd_quit(self, arg):
123 self.push('+OK closing.')
124 self.close_when_done()
125
Antoine Pitrou8618d742012-11-23 20:13:48 +0100126 def _get_capas(self):
127 _capas = dict(self.CAPAS)
128 if not self.tls_active and SUPPORTS_SSL:
129 _capas['STLS'] = []
130 return _capas
131
Antoine Pitrou25cee192012-11-23 20:07:39 +0100132 def cmd_capa(self, arg):
133 self.push('+OK Capability list follows')
Antoine Pitrou8618d742012-11-23 20:13:48 +0100134 if self._get_capas():
135 for cap, params in self._get_capas().items():
Antoine Pitrou25cee192012-11-23 20:07:39 +0100136 _ln = [cap]
137 if params:
138 _ln.extend(params)
139 self.push(' '.join(_ln))
140 self.push('.')
141
Antoine Pitrou8618d742012-11-23 20:13:48 +0100142 if SUPPORTS_SSL:
143
144 def cmd_stls(self, arg):
145 if self.tls_active is False:
146 self.push('+OK Begin TLS negotiation')
147 tls_sock = ssl.wrap_socket(self.socket, certfile=CERTFILE,
148 server_side=True,
149 do_handshake_on_connect=False,
150 suppress_ragged_eofs=False)
151 self.del_channel()
152 self.set_socket(tls_sock)
153 self.tls_active = True
154 self.tls_starting = True
155 self.in_buffer = []
156 self._do_tls_handshake()
157 else:
158 self.push('-ERR Command not permitted when TLS active')
159
160 def _do_tls_handshake(self):
161 try:
162 self.socket.do_handshake()
163 except ssl.SSLError as err:
164 if err.args[0] in (ssl.SSL_ERROR_WANT_READ,
165 ssl.SSL_ERROR_WANT_WRITE):
166 return
167 elif err.args[0] == ssl.SSL_ERROR_EOF:
168 return self.handle_close()
169 raise
Andrew Svetlov0832af62012-12-18 23:10:48 +0200170 except OSError as err:
Antoine Pitrou8618d742012-11-23 20:13:48 +0100171 if err.args[0] == errno.ECONNABORTED:
172 return self.handle_close()
173 else:
174 self.tls_active = True
175 self.tls_starting = False
176
177 def handle_read(self):
178 if self.tls_starting:
179 self._do_tls_handshake()
180 else:
181 try:
182 asynchat.async_chat.handle_read(self)
183 except ssl.SSLEOFError:
184 self.handle_close()
Christian Heimesd3956292008-11-05 19:48:27 +0000185
186class DummyPOP3Server(asyncore.dispatcher, threading.Thread):
187
188 handler = DummyPOP3Handler
189
190 def __init__(self, address, af=socket.AF_INET):
191 threading.Thread.__init__(self)
192 asyncore.dispatcher.__init__(self)
193 self.create_socket(af, socket.SOCK_STREAM)
194 self.bind(address)
195 self.listen(5)
196 self.active = False
197 self.active_lock = threading.Lock()
198 self.host, self.port = self.socket.getsockname()[:2]
Giampaolo Rodolà42382fe2010-08-17 16:09:53 +0000199 self.handler_instance = None
Christian Heimesd3956292008-11-05 19:48:27 +0000200
201 def start(self):
202 assert not self.active
203 self.__flag = threading.Event()
204 threading.Thread.start(self)
205 self.__flag.wait()
206
207 def run(self):
208 self.active = True
209 self.__flag.set()
210 while self.active and asyncore.socket_map:
211 self.active_lock.acquire()
212 asyncore.loop(timeout=0.1, count=1)
213 self.active_lock.release()
214 asyncore.close_all(ignore_all=True)
215
216 def stop(self):
217 assert self.active
218 self.active = False
219 self.join()
220
Giampaolo Rodolà977c7072010-10-04 21:08:36 +0000221 def handle_accepted(self, conn, addr):
Giampaolo Rodolà42382fe2010-08-17 16:09:53 +0000222 self.handler_instance = self.handler(conn)
Christian Heimesd3956292008-11-05 19:48:27 +0000223
224 def handle_connect(self):
225 self.close()
226 handle_read = handle_connect
227
228 def writable(self):
229 return 0
230
231 def handle_error(self):
232 raise
233
234
235class TestPOP3Class(TestCase):
236 def assertOK(self, resp):
237 self.assertTrue(resp.startswith(b"+OK"))
238
239 def setUp(self):
240 self.server = DummyPOP3Server((HOST, PORT))
241 self.server.start()
Giampaolo Rodolà95bcb932011-02-25 22:28:24 +0000242 self.client = poplib.POP3(self.server.host, self.server.port, timeout=3)
Christian Heimesd3956292008-11-05 19:48:27 +0000243
244 def tearDown(self):
Giampaolo Rodolà95bcb932011-02-25 22:28:24 +0000245 self.client.close()
Christian Heimesd3956292008-11-05 19:48:27 +0000246 self.server.stop()
247
248 def test_getwelcome(self):
Mark Dickinsonea1158f2009-08-06 16:06:25 +0000249 self.assertEqual(self.client.getwelcome(),
250 b'+OK dummy pop3 server ready. <timestamp>')
Christian Heimesd3956292008-11-05 19:48:27 +0000251
252 def test_exceptions(self):
253 self.assertRaises(poplib.error_proto, self.client._shortcmd, 'echo -err')
254
255 def test_user(self):
256 self.assertOK(self.client.user('guido'))
257 self.assertRaises(poplib.error_proto, self.client.user, 'invalid')
258
259 def test_pass_(self):
260 self.assertOK(self.client.pass_('python'))
261 self.assertRaises(poplib.error_proto, self.client.user, 'invalid')
262
263 def test_stat(self):
264 self.assertEqual(self.client.stat(), (10, 100))
265
266 def test_list(self):
267 self.assertEqual(self.client.list()[1:],
268 ([b'1 1', b'2 2', b'3 3', b'4 4', b'5 5'],
269 25))
270 self.assertTrue(self.client.list('1').endswith(b"OK 1 1"))
271
272 def test_retr(self):
273 expected = (b'+OK 116 bytes',
274 [b'From: postmaster@python.org', b'Content-Type: text/plain',
275 b'MIME-Version: 1.0', b'Subject: Dummy',
276 b'', b'line1', b'line2', b'line3'],
277 113)
278 foo = self.client.retr('foo')
279 self.assertEqual(foo, expected)
280
Georg Brandl7e27abb2013-10-27 07:23:53 +0100281 def test_too_long_lines(self):
282 self.assertRaises(poplib.error_proto, self.client._shortcmd,
283 'echo +%s' % ((poplib._MAXLINE + 10) * 'a'))
284
Christian Heimesd3956292008-11-05 19:48:27 +0000285 def test_dele(self):
286 self.assertOK(self.client.dele('foo'))
287
288 def test_noop(self):
289 self.assertOK(self.client.noop())
290
291 def test_rpop(self):
292 self.assertOK(self.client.rpop('foo'))
293
Mark Dickinsonea1158f2009-08-06 16:06:25 +0000294 def test_apop(self):
295 self.assertOK(self.client.apop('foo', 'dummypassword'))
296
Christian Heimesd3956292008-11-05 19:48:27 +0000297 def test_top(self):
298 expected = (b'+OK 116 bytes',
299 [b'From: postmaster@python.org', b'Content-Type: text/plain',
300 b'MIME-Version: 1.0', b'Subject: Dummy', b'',
301 b'line1', b'line2', b'line3'],
302 113)
303 self.assertEqual(self.client.top(1, 1), expected)
304
305 def test_uidl(self):
306 self.client.uidl()
307 self.client.uidl('foo')
308
Antoine Pitrou25cee192012-11-23 20:07:39 +0100309 def test_capa(self):
310 capa = self.client.capa()
311 self.assertTrue('IMPLEMENTATION' in capa.keys())
312
Giampaolo Rodolà95bcb932011-02-25 22:28:24 +0000313 def test_quit(self):
314 resp = self.client.quit()
315 self.assertTrue(resp)
316 self.assertIsNone(self.client.sock)
317 self.assertIsNone(self.client.file)
318
Antoine Pitrou8618d742012-11-23 20:13:48 +0100319 if SUPPORTS_SSL:
Christian Heimesd3956292008-11-05 19:48:27 +0000320
Antoine Pitrou8618d742012-11-23 20:13:48 +0100321 def test_stls_capa(self):
322 capa = self.client.capa()
323 self.assertTrue('STLS' in capa.keys())
Christian Heimesd3956292008-11-05 19:48:27 +0000324
Antoine Pitrou8618d742012-11-23 20:13:48 +0100325 def test_stls(self):
326 expected = b'+OK Begin TLS negotiation'
327 resp = self.client.stls()
328 self.assertEqual(resp, expected)
329
330 def test_stls_context(self):
331 expected = b'+OK Begin TLS negotiation'
332 ctx = ssl.SSLContext(ssl.PROTOCOL_TLSv1)
333 resp = self.client.stls(context=ctx)
334 self.assertEqual(resp, expected)
335
336
337if SUPPORTS_SSL:
Christian Heimesd3956292008-11-05 19:48:27 +0000338
339 class DummyPOP3_SSLHandler(DummyPOP3Handler):
340
341 def __init__(self, conn):
342 asynchat.async_chat.__init__(self, conn)
343 ssl_socket = ssl.wrap_socket(self.socket, certfile=CERTFILE,
Antoine Pitroud3f8ab82010-04-24 21:26:44 +0000344 server_side=True,
345 do_handshake_on_connect=False)
Christian Heimesd3956292008-11-05 19:48:27 +0000346 self.del_channel()
347 self.set_socket(ssl_socket)
Antoine Pitroud3f8ab82010-04-24 21:26:44 +0000348 # Must try handshake before calling push()
Antoine Pitrou8618d742012-11-23 20:13:48 +0100349 self.tls_active = True
350 self.tls_starting = True
351 self._do_tls_handshake()
Christian Heimesd3956292008-11-05 19:48:27 +0000352 self.set_terminator(b"\r\n")
353 self.in_buffer = []
Mark Dickinsonea1158f2009-08-06 16:06:25 +0000354 self.push('+OK dummy pop3 server ready. <timestamp>')
Christian Heimesd3956292008-11-05 19:48:27 +0000355
Giampaolo Rodolà95bcb932011-02-25 22:28:24 +0000356
Christian Heimesd3956292008-11-05 19:48:27 +0000357 class TestPOP3_SSLClass(TestPOP3Class):
358 # repeat previous tests by using poplib.POP3_SSL
359
360 def setUp(self):
361 self.server = DummyPOP3Server((HOST, PORT))
362 self.server.handler = DummyPOP3_SSLHandler
363 self.server.start()
364 self.client = poplib.POP3_SSL(self.server.host, self.server.port)
365
366 def test__all__(self):
Benjamin Peterson577473f2010-01-19 00:09:57 +0000367 self.assertIn('POP3_SSL', poplib.__all__)
Christian Heimesd3956292008-11-05 19:48:27 +0000368
Giampaolo Rodolà42382fe2010-08-17 16:09:53 +0000369 def test_context(self):
370 ctx = ssl.SSLContext(ssl.PROTOCOL_TLSv1)
371 self.assertRaises(ValueError, poplib.POP3_SSL, self.server.host,
372 self.server.port, keyfile=CERTFILE, context=ctx)
373 self.assertRaises(ValueError, poplib.POP3_SSL, self.server.host,
374 self.server.port, certfile=CERTFILE, context=ctx)
375 self.assertRaises(ValueError, poplib.POP3_SSL, self.server.host,
376 self.server.port, keyfile=CERTFILE,
377 certfile=CERTFILE, context=ctx)
378
379 self.client.quit()
380 self.client = poplib.POP3_SSL(self.server.host, self.server.port,
381 context=ctx)
382 self.assertIsInstance(self.client.sock, ssl.SSLSocket)
383 self.assertIs(self.client.sock.context, ctx)
384 self.assertTrue(self.client.noop().startswith(b'+OK'))
385
Antoine Pitrou8618d742012-11-23 20:13:48 +0100386 def test_stls(self):
387 self.assertRaises(poplib.error_proto, self.client.stls)
388
389 test_stls_context = test_stls
390
391 def test_stls_capa(self):
392 capa = self.client.capa()
393 self.assertFalse('STLS' in capa.keys())
394
395
396 class TestPOP3_TLSClass(TestPOP3Class):
397 # repeat previous tests by using poplib.POP3.stls()
398
399 def setUp(self):
400 self.server = DummyPOP3Server((HOST, PORT))
401 self.server.start()
402 self.client = poplib.POP3(self.server.host, self.server.port, timeout=3)
403 self.client.stls()
404
405 def tearDown(self):
406 if self.client.file is not None and self.client.sock is not None:
Georg Brandlb89b5df2013-10-27 07:46:09 +0100407 try:
408 self.client.quit()
409 except poplib.error_proto:
410 # happens in the test_too_long_lines case; the overlong
411 # response will be treated as response to QUIT and raise
412 # this exception
413 pass
Antoine Pitrou8618d742012-11-23 20:13:48 +0100414 self.server.stop()
415
416 def test_stls(self):
417 self.assertRaises(poplib.error_proto, self.client.stls)
418
419 test_stls_context = test_stls
420
421 def test_stls_capa(self):
422 capa = self.client.capa()
423 self.assertFalse(b'STLS' in capa.keys())
424
Christian Heimesd3956292008-11-05 19:48:27 +0000425
426class TestTimeouts(TestCase):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000427
428 def setUp(self):
429 self.evt = threading.Event()
Christian Heimes5e696852008-04-09 08:37:03 +0000430 self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
Charles-François Natali83ef2542011-12-14 19:28:56 +0100431 self.sock.settimeout(60) # Safety net. Look issue 11812
Christian Heimesd3956292008-11-05 19:48:27 +0000432 self.port = test_support.bind_port(self.sock)
Charles-François Natali83ef2542011-12-14 19:28:56 +0100433 self.thread = threading.Thread(target=self.server, args=(self.evt,self.sock))
434 self.thread.setDaemon(True)
435 self.thread.start()
436 self.evt.wait()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000437
438 def tearDown(self):
Charles-François Natali83ef2542011-12-14 19:28:56 +0100439 self.thread.join()
440 del self.thread # Clear out any dangling Thread objects.
Guido van Rossumd8faa362007-04-27 19:54:29 +0000441
Christian Heimesd3956292008-11-05 19:48:27 +0000442 def server(self, evt, serv):
443 serv.listen(5)
Charles-François Natali83ef2542011-12-14 19:28:56 +0100444 evt.set()
Christian Heimesd3956292008-11-05 19:48:27 +0000445 try:
446 conn, addr = serv.accept()
Christian Heimesd3956292008-11-05 19:48:27 +0000447 conn.send(b"+ Hola mundo\n")
448 conn.close()
Charles-François Natali83ef2542011-12-14 19:28:56 +0100449 except socket.timeout:
450 pass
Christian Heimesd3956292008-11-05 19:48:27 +0000451 finally:
452 serv.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000453
454 def testTimeoutDefault(self):
Georg Brandlf78e02b2008-06-10 17:40:04 +0000455 self.assertTrue(socket.getdefaulttimeout() is None)
456 socket.setdefaulttimeout(30)
457 try:
Charles-François Natali83ef2542011-12-14 19:28:56 +0100458 pop = poplib.POP3(HOST, self.port)
Georg Brandlf78e02b2008-06-10 17:40:04 +0000459 finally:
460 socket.setdefaulttimeout(None)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000461 self.assertEqual(pop.sock.gettimeout(), 30)
462 pop.sock.close()
463
464 def testTimeoutNone(self):
Georg Brandlf78e02b2008-06-10 17:40:04 +0000465 self.assertTrue(socket.getdefaulttimeout() is None)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000466 socket.setdefaulttimeout(30)
467 try:
Christian Heimes5e696852008-04-09 08:37:03 +0000468 pop = poplib.POP3(HOST, self.port, timeout=None)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000469 finally:
Georg Brandlf78e02b2008-06-10 17:40:04 +0000470 socket.setdefaulttimeout(None)
471 self.assertTrue(pop.sock.gettimeout() is None)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000472 pop.sock.close()
473
Georg Brandlf78e02b2008-06-10 17:40:04 +0000474 def testTimeoutValue(self):
Charles-François Natali83ef2542011-12-14 19:28:56 +0100475 pop = poplib.POP3(HOST, self.port, timeout=30)
Georg Brandlf78e02b2008-06-10 17:40:04 +0000476 self.assertEqual(pop.sock.gettimeout(), 30)
477 pop.sock.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000478
479
Christian Heimesd3956292008-11-05 19:48:27 +0000480def test_main():
481 tests = [TestPOP3Class, TestTimeouts]
482 if SUPPORTS_SSL:
483 tests.append(TestPOP3_SSLClass)
Antoine Pitrou8618d742012-11-23 20:13:48 +0100484 tests.append(TestPOP3_TLSClass)
Christian Heimesd3956292008-11-05 19:48:27 +0000485 thread_info = test_support.threading_setup()
486 try:
487 test_support.run_unittest(*tests)
488 finally:
489 test_support.threading_cleanup(*thread_info)
490
Guido van Rossumd8faa362007-04-27 19:54:29 +0000491
492if __name__ == '__main__':
493 test_main()