blob: 6584ba5ba864e209418a3036f5e81c032ce9d0e7 [file] [log] [blame]
Christian Heimes292d3512008-02-03 16:51:08 +00001"""
Alexandre Vassalottice261952008-05-12 02:31:37 +00002Test suite for socketserver.
Christian Heimes292d3512008-02-03 16:51:08 +00003"""
Guido van Rossum39f1b362001-07-10 11:52:38 +00004
Christian Heimes70e7ea22008-02-28 20:02:27 +00005import contextlib
Martin Panter34eeed42016-06-29 10:12:22 +00006import io
Christian Heimes9e7f1d22008-02-28 12:27:11 +00007import os
Guido van Rossum39f1b362001-07-10 11:52:38 +00008import select
Christian Heimes9e7f1d22008-02-28 12:27:11 +00009import signal
10import socket
11import tempfile
Antoine Pitroua6a4dc82017-09-07 18:56:24 +020012import threading
Christian Heimes292d3512008-02-03 16:51:08 +000013import unittest
Alexandre Vassalottice261952008-05-12 02:31:37 +000014import socketserver
Christian Heimes292d3512008-02-03 16:51:08 +000015
Benjamin Petersonee8712c2008-05-20 21:35:26 +000016import test.support
Antoine Pitrouc00d4b42009-10-27 21:32:38 +000017from test.support import reap_children, reap_threads, verbose
Antoine Pitroua6a4dc82017-09-07 18:56:24 +020018
Christian Heimes292d3512008-02-03 16:51:08 +000019
Benjamin Petersonee8712c2008-05-20 21:35:26 +000020test.support.requires("network")
Guido van Rossum39f1b362001-07-10 11:52:38 +000021
Christian Heimes292d3512008-02-03 16:51:08 +000022TEST_STR = b"hello world\n"
Benjamin Petersonee8712c2008-05-20 21:35:26 +000023HOST = test.support.HOST
Christian Heimes292d3512008-02-03 16:51:08 +000024
25HAVE_UNIX_SOCKETS = hasattr(socket, "AF_UNIX")
Serhiy Storchaka43767632013-11-03 21:31:38 +020026requires_unix_sockets = unittest.skipUnless(HAVE_UNIX_SOCKETS,
27 'requires Unix sockets')
Jesus Ceaf1af7052012-10-05 02:48:46 +020028HAVE_FORKING = hasattr(os, "fork")
Serhiy Storchaka43767632013-11-03 21:31:38 +020029requires_forking = unittest.skipUnless(HAVE_FORKING, 'requires forking')
Christian Heimes292d3512008-02-03 16:51:08 +000030
Christian Heimesdd15f6c2008-03-16 00:07:10 +000031def signal_alarm(n):
32 """Call signal.alarm when it exists (i.e. not on Windows)."""
33 if hasattr(signal, 'alarm'):
34 signal.alarm(n)
Guido van Rossum39f1b362001-07-10 11:52:38 +000035
Antoine Pitrouf18d6f32012-04-09 01:16:47 +020036# Remember real select() to avoid interferences with mocking
37_real_select = select.select
38
Christian Heimes292d3512008-02-03 16:51:08 +000039def receive(sock, n, timeout=20):
Antoine Pitrouf18d6f32012-04-09 01:16:47 +020040 r, w, x = _real_select([sock], [], [], timeout)
Christian Heimes292d3512008-02-03 16:51:08 +000041 if sock in r:
42 return sock.recv(n)
43 else:
44 raise RuntimeError("timed out on %r" % (sock,))
45
Gregory P. Smith ext:(%20%5BGoogle%20Inc.%5D)a9459692016-06-03 05:26:14 +000046if HAVE_UNIX_SOCKETS and HAVE_FORKING:
Alexandre Vassalottice261952008-05-12 02:31:37 +000047 class ForkingUnixStreamServer(socketserver.ForkingMixIn,
48 socketserver.UnixStreamServer):
Christian Heimes15ebc882008-02-04 18:48:49 +000049 pass
Christian Heimes292d3512008-02-03 16:51:08 +000050
Alexandre Vassalottice261952008-05-12 02:31:37 +000051 class ForkingUnixDatagramServer(socketserver.ForkingMixIn,
52 socketserver.UnixDatagramServer):
Christian Heimes15ebc882008-02-04 18:48:49 +000053 pass
Christian Heimes292d3512008-02-03 16:51:08 +000054
55
Christian Heimes70e7ea22008-02-28 20:02:27 +000056@contextlib.contextmanager
57def simple_subprocess(testcase):
Martin Panterd9108d12016-02-21 08:49:56 +000058 """Tests that a custom child process is not waited on (Issue 1540386)"""
Christian Heimes70e7ea22008-02-28 20:02:27 +000059 pid = os.fork()
60 if pid == 0:
Andrew Svetlov737fb892012-12-18 21:14:22 +020061 # Don't raise an exception; it would be caught by the test harness.
Christian Heimes70e7ea22008-02-28 20:02:27 +000062 os._exit(72)
xdegaye9001d1f2017-11-18 18:10:53 +010063 try:
64 yield None
65 except:
66 raise
67 finally:
68 pid2, status = os.waitpid(pid, 0)
69 testcase.assertEqual(pid2, pid)
70 testcase.assertEqual(72 << 8, status)
Christian Heimes70e7ea22008-02-28 20:02:27 +000071
72
Christian Heimes292d3512008-02-03 16:51:08 +000073class SocketServerTest(unittest.TestCase):
74 """Test all socket servers."""
Guido van Rossum39f1b362001-07-10 11:52:38 +000075
Christian Heimes292d3512008-02-03 16:51:08 +000076 def setUp(self):
Victor Stinner5702ae62011-07-14 14:53:24 +020077 signal_alarm(60) # Kill deadlocks after 60 seconds.
Christian Heimes292d3512008-02-03 16:51:08 +000078 self.port_seed = 0
79 self.test_files = []
Guido van Rossum39f1b362001-07-10 11:52:38 +000080
Christian Heimes292d3512008-02-03 16:51:08 +000081 def tearDown(self):
Christian Heimesfe337bf2008-03-23 21:54:12 +000082 signal_alarm(0) # Didn't deadlock.
Christian Heimes292d3512008-02-03 16:51:08 +000083 reap_children()
Thomas Wouters00ee7ba2006-08-21 19:07:27 +000084
Christian Heimes292d3512008-02-03 16:51:08 +000085 for fn in self.test_files:
86 try:
87 os.remove(fn)
Andrew Svetlovad28c7f2012-12-18 22:02:39 +020088 except OSError:
Christian Heimes292d3512008-02-03 16:51:08 +000089 pass
90 self.test_files[:] = []
Christian Heimes292d3512008-02-03 16:51:08 +000091
92 def pickaddr(self, proto):
93 if proto == socket.AF_INET:
Christian Heimes9e7f1d22008-02-28 12:27:11 +000094 return (HOST, 0)
Christian Heimes292d3512008-02-03 16:51:08 +000095 else:
Christian Heimes9e7f1d22008-02-28 12:27:11 +000096 # XXX: We need a way to tell AF_UNIX to pick its own name
97 # like AF_INET provides port==0.
98 dir = None
Christian Heimes9e7f1d22008-02-28 12:27:11 +000099 fn = tempfile.mktemp(prefix='unix_socket.', dir=dir)
Christian Heimes292d3512008-02-03 16:51:08 +0000100 self.test_files.append(fn)
101 return fn
102
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000103 def make_server(self, addr, svrcls, hdlrbase):
104 class MyServer(svrcls):
105 def handle_error(self, request, client_address):
106 self.close_request(request)
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000107 raise
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000108
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000109 class MyHandler(hdlrbase):
110 def handle(self):
111 line = self.rfile.readline()
112 self.wfile.write(line)
113
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000114 if verbose: print("creating server")
xdegaye9001d1f2017-11-18 18:10:53 +0100115 try:
116 server = MyServer(addr, MyHandler)
117 except PermissionError as e:
118 # Issue 29184: cannot bind() a Unix socket on Android.
119 self.skipTest('Cannot create server (%s, %s): %s' %
120 (svrcls, addr, e))
Ezio Melottib3aedd42010-11-20 19:04:17 +0000121 self.assertEqual(server.server_address, server.socket.getsockname())
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000122 return server
123
Antoine Pitrouc00d4b42009-10-27 21:32:38 +0000124 @reap_threads
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000125 def run_server(self, svrcls, hdlrbase, testfunc):
126 server = self.make_server(self.pickaddr(svrcls.address_family),
127 svrcls, hdlrbase)
128 # We had the OS pick a port, so pull the real address out of
129 # the server.
130 addr = server.server_address
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000131 if verbose:
132 print("ADDR =", addr)
133 print("CLASS =", svrcls)
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000134
135 t = threading.Thread(
136 name='%s serving' % svrcls,
137 target=server.serve_forever,
138 # Short poll interval to make the test finish quickly.
139 # Time between requests is short enough that we won't wake
140 # up spuriously too many times.
141 kwargs={'poll_interval':0.01})
Benjamin Peterson72753702008-08-18 18:09:21 +0000142 t.daemon = True # In case this function raises.
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000143 t.start()
144 if verbose: print("server running")
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000145 for i in range(3):
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000146 if verbose: print("test client", i)
147 testfunc(svrcls.address_family, addr)
148 if verbose: print("waiting for server")
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000149 server.shutdown()
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000150 t.join()
Victor Stinner109761b2011-01-03 14:30:39 +0000151 server.server_close()
Robert Collins1ee92832015-07-29 12:52:40 +1200152 self.assertEqual(-1, server.socket.fileno())
Victor Stinneraa8ec342017-08-10 15:28:16 +0200153 if HAVE_FORKING and isinstance(server, socketserver.ForkingMixIn):
154 # bpo-31151: Check that ForkingMixIn.server_close() waits until
155 # all children completed
156 self.assertFalse(server.active_children)
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000157 if verbose: print("done")
Christian Heimes292d3512008-02-03 16:51:08 +0000158
159 def stream_examine(self, proto, addr):
160 s = socket.socket(proto, socket.SOCK_STREAM)
161 s.connect(addr)
162 s.sendall(TEST_STR)
163 buf = data = receive(s, 100)
Christian Heimesfdb6bb52008-02-03 17:32:13 +0000164 while data and b'\n' not in buf:
Christian Heimes292d3512008-02-03 16:51:08 +0000165 data = receive(s, 100)
166 buf += data
Ezio Melottib3aedd42010-11-20 19:04:17 +0000167 self.assertEqual(buf, TEST_STR)
Christian Heimes292d3512008-02-03 16:51:08 +0000168 s.close()
169
170 def dgram_examine(self, proto, addr):
171 s = socket.socket(proto, socket.SOCK_DGRAM)
Martin Panter1827eff2016-02-24 05:12:59 +0000172 if HAVE_UNIX_SOCKETS and proto == socket.AF_UNIX:
173 s.bind(self.pickaddr(proto))
Christian Heimes292d3512008-02-03 16:51:08 +0000174 s.sendto(TEST_STR, addr)
175 buf = data = receive(s, 100)
Christian Heimesfdb6bb52008-02-03 17:32:13 +0000176 while data and b'\n' not in buf:
Christian Heimes292d3512008-02-03 16:51:08 +0000177 data = receive(s, 100)
178 buf += data
Ezio Melottib3aedd42010-11-20 19:04:17 +0000179 self.assertEqual(buf, TEST_STR)
Christian Heimes292d3512008-02-03 16:51:08 +0000180 s.close()
181
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000182 def test_TCPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000183 self.run_server(socketserver.TCPServer,
184 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000185 self.stream_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000186
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000187 def test_ThreadingTCPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000188 self.run_server(socketserver.ThreadingTCPServer,
189 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000190 self.stream_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000191
Serhiy Storchaka43767632013-11-03 21:31:38 +0200192 @requires_forking
193 def test_ForkingTCPServer(self):
194 with simple_subprocess(self):
195 self.run_server(socketserver.ForkingTCPServer,
Alexandre Vassalottice261952008-05-12 02:31:37 +0000196 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000197 self.stream_examine)
198
Serhiy Storchaka43767632013-11-03 21:31:38 +0200199 @requires_unix_sockets
200 def test_UnixStreamServer(self):
201 self.run_server(socketserver.UnixStreamServer,
202 socketserver.StreamRequestHandler,
203 self.stream_examine)
204
205 @requires_unix_sockets
206 def test_ThreadingUnixStreamServer(self):
207 self.run_server(socketserver.ThreadingUnixStreamServer,
208 socketserver.StreamRequestHandler,
209 self.stream_examine)
210
211 @requires_unix_sockets
212 @requires_forking
213 def test_ForkingUnixStreamServer(self):
214 with simple_subprocess(self):
215 self.run_server(ForkingUnixStreamServer,
Alexandre Vassalottice261952008-05-12 02:31:37 +0000216 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000217 self.stream_examine)
218
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000219 def test_UDPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000220 self.run_server(socketserver.UDPServer,
221 socketserver.DatagramRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000222 self.dgram_examine)
223
224 def test_ThreadingUDPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000225 self.run_server(socketserver.ThreadingUDPServer,
226 socketserver.DatagramRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000227 self.dgram_examine)
228
Serhiy Storchaka43767632013-11-03 21:31:38 +0200229 @requires_forking
230 def test_ForkingUDPServer(self):
231 with simple_subprocess(self):
232 self.run_server(socketserver.ForkingUDPServer,
233 socketserver.DatagramRequestHandler,
234 self.dgram_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000235
Martin Panter1827eff2016-02-24 05:12:59 +0000236 @requires_unix_sockets
237 def test_UnixDatagramServer(self):
238 self.run_server(socketserver.UnixDatagramServer,
239 socketserver.DatagramRequestHandler,
240 self.dgram_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000241
Martin Panter1827eff2016-02-24 05:12:59 +0000242 @requires_unix_sockets
243 def test_ThreadingUnixDatagramServer(self):
244 self.run_server(socketserver.ThreadingUnixDatagramServer,
245 socketserver.DatagramRequestHandler,
246 self.dgram_examine)
247
248 @requires_unix_sockets
249 @requires_forking
250 def test_ForkingUnixDatagramServer(self):
251 self.run_server(ForkingUnixDatagramServer,
252 socketserver.DatagramRequestHandler,
253 self.dgram_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000254
Antoine Pitrou3bcba8e2010-04-25 22:01:43 +0000255 @reap_threads
256 def test_shutdown(self):
257 # Issue #2302: shutdown() should always succeed in making an
258 # other thread leave serve_forever().
259 class MyServer(socketserver.TCPServer):
260 pass
261
262 class MyHandler(socketserver.StreamRequestHandler):
263 pass
264
265 threads = []
266 for i in range(20):
267 s = MyServer((HOST, 0), MyHandler)
268 t = threading.Thread(
269 name='MyServer serving',
270 target=s.serve_forever,
271 kwargs={'poll_interval':0.01})
272 t.daemon = True # In case this function raises.
273 threads.append((t, s))
274 for t, s in threads:
275 t.start()
276 s.shutdown()
277 for t, s in threads:
278 t.join()
Victor Stinner109761b2011-01-03 14:30:39 +0000279 s.server_close()
Antoine Pitrou3bcba8e2010-04-25 22:01:43 +0000280
Charles-François Natali0f4f0482014-10-13 19:19:26 +0100281 def test_tcpserver_bind_leak(self):
282 # Issue #22435: the server socket wouldn't be closed if bind()/listen()
283 # failed.
284 # Create many servers for which bind() will fail, to see if this result
285 # in FD exhaustion.
286 for i in range(1024):
287 with self.assertRaises(OverflowError):
288 socketserver.TCPServer((HOST, -1),
289 socketserver.StreamRequestHandler)
290
Martin Panter0cab9c12016-04-13 00:36:52 +0000291 def test_context_manager(self):
292 with socketserver.TCPServer((HOST, 0),
293 socketserver.StreamRequestHandler) as server:
294 pass
295 self.assertEqual(-1, server.socket.fileno())
296
Guido van Rossum39f1b362001-07-10 11:52:38 +0000297
Martin Panterd9108d12016-02-21 08:49:56 +0000298class ErrorHandlerTest(unittest.TestCase):
299 """Test that the servers pass normal exceptions from the handler to
300 handle_error(), and that exiting exceptions like SystemExit and
301 KeyboardInterrupt are not passed."""
302
303 def tearDown(self):
304 test.support.unlink(test.support.TESTFN)
305
306 def test_sync_handled(self):
307 BaseErrorTestServer(ValueError)
308 self.check_result(handled=True)
309
310 def test_sync_not_handled(self):
311 with self.assertRaises(SystemExit):
312 BaseErrorTestServer(SystemExit)
313 self.check_result(handled=False)
314
Martin Panterd9108d12016-02-21 08:49:56 +0000315 def test_threading_handled(self):
316 ThreadingErrorTestServer(ValueError)
317 self.check_result(handled=True)
318
Martin Panterd9108d12016-02-21 08:49:56 +0000319 def test_threading_not_handled(self):
320 ThreadingErrorTestServer(SystemExit)
321 self.check_result(handled=False)
322
323 @requires_forking
324 def test_forking_handled(self):
325 ForkingErrorTestServer(ValueError)
326 self.check_result(handled=True)
327
328 @requires_forking
329 def test_forking_not_handled(self):
330 ForkingErrorTestServer(SystemExit)
331 self.check_result(handled=False)
332
333 def check_result(self, handled):
334 with open(test.support.TESTFN) as log:
335 expected = 'Handler called\n' + 'Error handled\n' * handled
336 self.assertEqual(log.read(), expected)
337
338
339class BaseErrorTestServer(socketserver.TCPServer):
340 def __init__(self, exception):
341 self.exception = exception
342 super().__init__((HOST, 0), BadHandler)
343 with socket.create_connection(self.server_address):
344 pass
345 try:
346 self.handle_request()
347 finally:
348 self.server_close()
349 self.wait_done()
350
351 def handle_error(self, request, client_address):
352 with open(test.support.TESTFN, 'a') as log:
353 log.write('Error handled\n')
354
355 def wait_done(self):
356 pass
357
358
359class BadHandler(socketserver.BaseRequestHandler):
360 def handle(self):
361 with open(test.support.TESTFN, 'a') as log:
362 log.write('Handler called\n')
363 raise self.server.exception('Test error')
364
365
366class ThreadingErrorTestServer(socketserver.ThreadingMixIn,
367 BaseErrorTestServer):
368 def __init__(self, *pos, **kw):
369 self.done = threading.Event()
370 super().__init__(*pos, **kw)
371
372 def shutdown_request(self, *pos, **kw):
373 super().shutdown_request(*pos, **kw)
374 self.done.set()
375
376 def wait_done(self):
377 self.done.wait()
378
379
Gregory P. Smith ext:(%20%5BGoogle%20Inc.%5D)a9459692016-06-03 05:26:14 +0000380if HAVE_FORKING:
381 class ForkingErrorTestServer(socketserver.ForkingMixIn, BaseErrorTestServer):
Victor Stinneraa8ec342017-08-10 15:28:16 +0200382 pass
Martin Panterd9108d12016-02-21 08:49:56 +0000383
384
Martin Panter34eeed42016-06-29 10:12:22 +0000385class SocketWriterTest(unittest.TestCase):
386 def test_basics(self):
387 class Handler(socketserver.StreamRequestHandler):
388 def handle(self):
389 self.server.wfile = self.wfile
390 self.server.wfile_fileno = self.wfile.fileno()
391 self.server.request_fileno = self.request.fileno()
392
393 server = socketserver.TCPServer((HOST, 0), Handler)
394 self.addCleanup(server.server_close)
395 s = socket.socket(
396 server.address_family, socket.SOCK_STREAM, socket.IPPROTO_TCP)
397 with s:
398 s.connect(server.server_address)
399 server.handle_request()
400 self.assertIsInstance(server.wfile, io.BufferedIOBase)
401 self.assertEqual(server.wfile_fileno, server.request_fileno)
402
Martin Panter34eeed42016-06-29 10:12:22 +0000403 def test_write(self):
404 # Test that wfile.write() sends data immediately, and that it does
405 # not truncate sends when interrupted by a Unix signal
406 pthread_kill = test.support.get_attribute(signal, 'pthread_kill')
407
408 class Handler(socketserver.StreamRequestHandler):
409 def handle(self):
410 self.server.sent1 = self.wfile.write(b'write data\n')
411 # Should be sent immediately, without requiring flush()
412 self.server.received = self.rfile.readline()
Serhiy Storchaka5f1a5182016-09-11 14:41:02 +0300413 big_chunk = b'\0' * test.support.SOCK_MAX_SIZE
Martin Panter34eeed42016-06-29 10:12:22 +0000414 self.server.sent2 = self.wfile.write(big_chunk)
415
416 server = socketserver.TCPServer((HOST, 0), Handler)
417 self.addCleanup(server.server_close)
418 interrupted = threading.Event()
419
420 def signal_handler(signum, frame):
421 interrupted.set()
422
423 original = signal.signal(signal.SIGUSR1, signal_handler)
424 self.addCleanup(signal.signal, signal.SIGUSR1, original)
425 response1 = None
426 received2 = None
427 main_thread = threading.get_ident()
428
429 def run_client():
430 s = socket.socket(server.address_family, socket.SOCK_STREAM,
431 socket.IPPROTO_TCP)
432 with s, s.makefile('rb') as reader:
433 s.connect(server.server_address)
434 nonlocal response1
435 response1 = reader.readline()
436 s.sendall(b'client response\n')
437
438 reader.read(100)
439 # The main thread should now be blocking in a send() syscall.
440 # But in theory, it could get interrupted by other signals,
441 # and then retried. So keep sending the signal in a loop, in
442 # case an earlier signal happens to be delivered at an
443 # inconvenient moment.
444 while True:
445 pthread_kill(main_thread, signal.SIGUSR1)
446 if interrupted.wait(timeout=float(1)):
447 break
448 nonlocal received2
449 received2 = len(reader.read())
450
451 background = threading.Thread(target=run_client)
452 background.start()
453 server.handle_request()
454 background.join()
455 self.assertEqual(server.sent1, len(response1))
456 self.assertEqual(response1, b'write data\n')
457 self.assertEqual(server.received, b'client response\n')
458 self.assertEqual(server.sent2, test.support.SOCK_MAX_SIZE)
459 self.assertEqual(received2, test.support.SOCK_MAX_SIZE - 100)
460
461
Berker Peksag32653442015-02-03 11:55:09 +0200462class MiscTestCase(unittest.TestCase):
Tim Petersa9f6f222001-09-17 23:56:20 +0000463
Berker Peksag32653442015-02-03 11:55:09 +0200464 def test_all(self):
465 # objects defined in the module should be in __all__
466 expected = []
467 for name in dir(socketserver):
468 if not name.startswith('_'):
469 mod_object = getattr(socketserver, name)
470 if getattr(mod_object, '__module__', None) == 'socketserver':
471 expected.append(name)
472 self.assertCountEqual(socketserver.__all__, expected)
473
Martin Panter4bf42742016-02-19 02:16:42 +0000474 def test_shutdown_request_called_if_verify_request_false(self):
475 # Issue #26309: BaseServer should call shutdown_request even if
476 # verify_request is False
477
478 class MyServer(socketserver.TCPServer):
479 def verify_request(self, request, client_address):
480 return False
481
482 shutdown_called = 0
483 def shutdown_request(self, request):
484 self.shutdown_called += 1
485 socketserver.TCPServer.shutdown_request(self, request)
486
487 server = MyServer((HOST, 0), socketserver.StreamRequestHandler)
488 s = socket.socket(server.address_family, socket.SOCK_STREAM)
489 s.connect(server.server_address)
490 s.close()
491 server.handle_request()
492 self.assertEqual(server.shutdown_called, 1)
493 server.server_close()
494
Guido van Rossum39f1b362001-07-10 11:52:38 +0000495
Tim Petersa9f6f222001-09-17 23:56:20 +0000496if __name__ == "__main__":
Berker Peksag32653442015-02-03 11:55:09 +0200497 unittest.main()