blob: 935848bc5f29a16446450d136ce4885825bc1dc4 [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:
97 self.push('+OK %s %s' %(arg, arg))
98 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
281 def test_dele(self):
282 self.assertOK(self.client.dele('foo'))
283
284 def test_noop(self):
285 self.assertOK(self.client.noop())
286
287 def test_rpop(self):
288 self.assertOK(self.client.rpop('foo'))
289
Mark Dickinsonea1158f2009-08-06 16:06:25 +0000290 def test_apop(self):
291 self.assertOK(self.client.apop('foo', 'dummypassword'))
292
Christian Heimesd3956292008-11-05 19:48:27 +0000293 def test_top(self):
294 expected = (b'+OK 116 bytes',
295 [b'From: postmaster@python.org', b'Content-Type: text/plain',
296 b'MIME-Version: 1.0', b'Subject: Dummy', b'',
297 b'line1', b'line2', b'line3'],
298 113)
299 self.assertEqual(self.client.top(1, 1), expected)
300
301 def test_uidl(self):
302 self.client.uidl()
303 self.client.uidl('foo')
304
Antoine Pitrou25cee192012-11-23 20:07:39 +0100305 def test_capa(self):
306 capa = self.client.capa()
307 self.assertTrue('IMPLEMENTATION' in capa.keys())
308
Giampaolo Rodolà95bcb932011-02-25 22:28:24 +0000309 def test_quit(self):
310 resp = self.client.quit()
311 self.assertTrue(resp)
312 self.assertIsNone(self.client.sock)
313 self.assertIsNone(self.client.file)
314
Antoine Pitrou8618d742012-11-23 20:13:48 +0100315 if SUPPORTS_SSL:
Christian Heimesd3956292008-11-05 19:48:27 +0000316
Antoine Pitrou8618d742012-11-23 20:13:48 +0100317 def test_stls_capa(self):
318 capa = self.client.capa()
319 self.assertTrue('STLS' in capa.keys())
Christian Heimesd3956292008-11-05 19:48:27 +0000320
Antoine Pitrou8618d742012-11-23 20:13:48 +0100321 def test_stls(self):
322 expected = b'+OK Begin TLS negotiation'
323 resp = self.client.stls()
324 self.assertEqual(resp, expected)
325
326 def test_stls_context(self):
327 expected = b'+OK Begin TLS negotiation'
328 ctx = ssl.SSLContext(ssl.PROTOCOL_TLSv1)
329 resp = self.client.stls(context=ctx)
330 self.assertEqual(resp, expected)
331
332
333if SUPPORTS_SSL:
Christian Heimesd3956292008-11-05 19:48:27 +0000334
335 class DummyPOP3_SSLHandler(DummyPOP3Handler):
336
337 def __init__(self, conn):
338 asynchat.async_chat.__init__(self, conn)
339 ssl_socket = ssl.wrap_socket(self.socket, certfile=CERTFILE,
Antoine Pitroud3f8ab82010-04-24 21:26:44 +0000340 server_side=True,
341 do_handshake_on_connect=False)
Christian Heimesd3956292008-11-05 19:48:27 +0000342 self.del_channel()
343 self.set_socket(ssl_socket)
Antoine Pitroud3f8ab82010-04-24 21:26:44 +0000344 # Must try handshake before calling push()
Antoine Pitrou8618d742012-11-23 20:13:48 +0100345 self.tls_active = True
346 self.tls_starting = True
347 self._do_tls_handshake()
Christian Heimesd3956292008-11-05 19:48:27 +0000348 self.set_terminator(b"\r\n")
349 self.in_buffer = []
Mark Dickinsonea1158f2009-08-06 16:06:25 +0000350 self.push('+OK dummy pop3 server ready. <timestamp>')
Christian Heimesd3956292008-11-05 19:48:27 +0000351
Giampaolo Rodolà95bcb932011-02-25 22:28:24 +0000352
Christian Heimesd3956292008-11-05 19:48:27 +0000353 class TestPOP3_SSLClass(TestPOP3Class):
354 # repeat previous tests by using poplib.POP3_SSL
355
356 def setUp(self):
357 self.server = DummyPOP3Server((HOST, PORT))
358 self.server.handler = DummyPOP3_SSLHandler
359 self.server.start()
360 self.client = poplib.POP3_SSL(self.server.host, self.server.port)
361
362 def test__all__(self):
Benjamin Peterson577473f2010-01-19 00:09:57 +0000363 self.assertIn('POP3_SSL', poplib.__all__)
Christian Heimesd3956292008-11-05 19:48:27 +0000364
Giampaolo Rodolà42382fe2010-08-17 16:09:53 +0000365 def test_context(self):
366 ctx = ssl.SSLContext(ssl.PROTOCOL_TLSv1)
367 self.assertRaises(ValueError, poplib.POP3_SSL, self.server.host,
368 self.server.port, keyfile=CERTFILE, context=ctx)
369 self.assertRaises(ValueError, poplib.POP3_SSL, self.server.host,
370 self.server.port, certfile=CERTFILE, context=ctx)
371 self.assertRaises(ValueError, poplib.POP3_SSL, self.server.host,
372 self.server.port, keyfile=CERTFILE,
373 certfile=CERTFILE, context=ctx)
374
375 self.client.quit()
376 self.client = poplib.POP3_SSL(self.server.host, self.server.port,
377 context=ctx)
378 self.assertIsInstance(self.client.sock, ssl.SSLSocket)
379 self.assertIs(self.client.sock.context, ctx)
380 self.assertTrue(self.client.noop().startswith(b'+OK'))
381
Antoine Pitrou8618d742012-11-23 20:13:48 +0100382 def test_stls(self):
383 self.assertRaises(poplib.error_proto, self.client.stls)
384
385 test_stls_context = test_stls
386
387 def test_stls_capa(self):
388 capa = self.client.capa()
389 self.assertFalse('STLS' in capa.keys())
390
391
392 class TestPOP3_TLSClass(TestPOP3Class):
393 # repeat previous tests by using poplib.POP3.stls()
394
395 def setUp(self):
396 self.server = DummyPOP3Server((HOST, PORT))
397 self.server.start()
398 self.client = poplib.POP3(self.server.host, self.server.port, timeout=3)
399 self.client.stls()
400
401 def tearDown(self):
402 if self.client.file is not None and self.client.sock is not None:
403 self.client.quit()
404 self.server.stop()
405
406 def test_stls(self):
407 self.assertRaises(poplib.error_proto, self.client.stls)
408
409 test_stls_context = test_stls
410
411 def test_stls_capa(self):
412 capa = self.client.capa()
413 self.assertFalse(b'STLS' in capa.keys())
414
Christian Heimesd3956292008-11-05 19:48:27 +0000415
416class TestTimeouts(TestCase):
Guido van Rossumd8faa362007-04-27 19:54:29 +0000417
418 def setUp(self):
419 self.evt = threading.Event()
Christian Heimes5e696852008-04-09 08:37:03 +0000420 self.sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
Charles-François Natali83ef2542011-12-14 19:28:56 +0100421 self.sock.settimeout(60) # Safety net. Look issue 11812
Christian Heimesd3956292008-11-05 19:48:27 +0000422 self.port = test_support.bind_port(self.sock)
Charles-François Natali83ef2542011-12-14 19:28:56 +0100423 self.thread = threading.Thread(target=self.server, args=(self.evt,self.sock))
424 self.thread.setDaemon(True)
425 self.thread.start()
426 self.evt.wait()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000427
428 def tearDown(self):
Charles-François Natali83ef2542011-12-14 19:28:56 +0100429 self.thread.join()
430 del self.thread # Clear out any dangling Thread objects.
Guido van Rossumd8faa362007-04-27 19:54:29 +0000431
Christian Heimesd3956292008-11-05 19:48:27 +0000432 def server(self, evt, serv):
433 serv.listen(5)
Charles-François Natali83ef2542011-12-14 19:28:56 +0100434 evt.set()
Christian Heimesd3956292008-11-05 19:48:27 +0000435 try:
436 conn, addr = serv.accept()
Christian Heimesd3956292008-11-05 19:48:27 +0000437 conn.send(b"+ Hola mundo\n")
438 conn.close()
Charles-François Natali83ef2542011-12-14 19:28:56 +0100439 except socket.timeout:
440 pass
Christian Heimesd3956292008-11-05 19:48:27 +0000441 finally:
442 serv.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000443
444 def testTimeoutDefault(self):
Georg Brandlf78e02b2008-06-10 17:40:04 +0000445 self.assertTrue(socket.getdefaulttimeout() is None)
446 socket.setdefaulttimeout(30)
447 try:
Charles-François Natali83ef2542011-12-14 19:28:56 +0100448 pop = poplib.POP3(HOST, self.port)
Georg Brandlf78e02b2008-06-10 17:40:04 +0000449 finally:
450 socket.setdefaulttimeout(None)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000451 self.assertEqual(pop.sock.gettimeout(), 30)
452 pop.sock.close()
453
454 def testTimeoutNone(self):
Georg Brandlf78e02b2008-06-10 17:40:04 +0000455 self.assertTrue(socket.getdefaulttimeout() is None)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000456 socket.setdefaulttimeout(30)
457 try:
Christian Heimes5e696852008-04-09 08:37:03 +0000458 pop = poplib.POP3(HOST, self.port, timeout=None)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000459 finally:
Georg Brandlf78e02b2008-06-10 17:40:04 +0000460 socket.setdefaulttimeout(None)
461 self.assertTrue(pop.sock.gettimeout() is None)
Guido van Rossumd8faa362007-04-27 19:54:29 +0000462 pop.sock.close()
463
Georg Brandlf78e02b2008-06-10 17:40:04 +0000464 def testTimeoutValue(self):
Charles-François Natali83ef2542011-12-14 19:28:56 +0100465 pop = poplib.POP3(HOST, self.port, timeout=30)
Georg Brandlf78e02b2008-06-10 17:40:04 +0000466 self.assertEqual(pop.sock.gettimeout(), 30)
467 pop.sock.close()
Guido van Rossumd8faa362007-04-27 19:54:29 +0000468
469
Christian Heimesd3956292008-11-05 19:48:27 +0000470def test_main():
471 tests = [TestPOP3Class, TestTimeouts]
472 if SUPPORTS_SSL:
473 tests.append(TestPOP3_SSLClass)
Antoine Pitrou8618d742012-11-23 20:13:48 +0100474 tests.append(TestPOP3_TLSClass)
Christian Heimesd3956292008-11-05 19:48:27 +0000475 thread_info = test_support.threading_setup()
476 try:
477 test_support.run_unittest(*tests)
478 finally:
479 test_support.threading_cleanup(*thread_info)
480
Guido van Rossumd8faa362007-04-27 19:54:29 +0000481
482if __name__ == '__main__':
483 test_main()