blob: 954e0331352fb54e5af42b194f1b803bc1b0d154 [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
Hai Shie80697d2020-05-28 06:10:27 +080017from test.support import reap_children, verbose
Hai Shi79bb2c92020-08-06 19:51:29 +080018from test.support import os_helper
Serhiy Storchaka16994912020-04-25 10:06:29 +030019from test.support import socket_helper
Hai Shie80697d2020-05-28 06:10:27 +080020from test.support import threading_helper
Antoine Pitroua6a4dc82017-09-07 18:56:24 +020021
Christian Heimes292d3512008-02-03 16:51:08 +000022
Benjamin Petersonee8712c2008-05-20 21:35:26 +000023test.support.requires("network")
Guido van Rossum39f1b362001-07-10 11:52:38 +000024
Christian Heimes292d3512008-02-03 16:51:08 +000025TEST_STR = b"hello world\n"
Serhiy Storchaka16994912020-04-25 10:06:29 +030026HOST = socket_helper.HOST
Christian Heimes292d3512008-02-03 16:51:08 +000027
28HAVE_UNIX_SOCKETS = hasattr(socket, "AF_UNIX")
Serhiy Storchaka43767632013-11-03 21:31:38 +020029requires_unix_sockets = unittest.skipUnless(HAVE_UNIX_SOCKETS,
30 'requires Unix sockets')
Jesus Ceaf1af7052012-10-05 02:48:46 +020031HAVE_FORKING = hasattr(os, "fork")
Serhiy Storchaka43767632013-11-03 21:31:38 +020032requires_forking = unittest.skipUnless(HAVE_FORKING, 'requires forking')
Christian Heimes292d3512008-02-03 16:51:08 +000033
Christian Heimesdd15f6c2008-03-16 00:07:10 +000034def signal_alarm(n):
35 """Call signal.alarm when it exists (i.e. not on Windows)."""
36 if hasattr(signal, 'alarm'):
37 signal.alarm(n)
Guido van Rossum39f1b362001-07-10 11:52:38 +000038
Antoine Pitrouf18d6f32012-04-09 01:16:47 +020039# Remember real select() to avoid interferences with mocking
40_real_select = select.select
41
Victor Stinner0d63bac2019-12-11 11:30:03 +010042def receive(sock, n, timeout=test.support.SHORT_TIMEOUT):
Antoine Pitrouf18d6f32012-04-09 01:16:47 +020043 r, w, x = _real_select([sock], [], [], timeout)
Christian Heimes292d3512008-02-03 16:51:08 +000044 if sock in r:
45 return sock.recv(n)
46 else:
47 raise RuntimeError("timed out on %r" % (sock,))
48
Gregory P. Smith ext:(%20%5BGoogle%20Inc.%5D)a9459692016-06-03 05:26:14 +000049if HAVE_UNIX_SOCKETS and HAVE_FORKING:
Alexandre Vassalottice261952008-05-12 02:31:37 +000050 class ForkingUnixStreamServer(socketserver.ForkingMixIn,
51 socketserver.UnixStreamServer):
Christian Heimes15ebc882008-02-04 18:48:49 +000052 pass
Christian Heimes292d3512008-02-03 16:51:08 +000053
Alexandre Vassalottice261952008-05-12 02:31:37 +000054 class ForkingUnixDatagramServer(socketserver.ForkingMixIn,
55 socketserver.UnixDatagramServer):
Christian Heimes15ebc882008-02-04 18:48:49 +000056 pass
Christian Heimes292d3512008-02-03 16:51:08 +000057
58
Christian Heimes70e7ea22008-02-28 20:02:27 +000059@contextlib.contextmanager
60def simple_subprocess(testcase):
Martin Panterd9108d12016-02-21 08:49:56 +000061 """Tests that a custom child process is not waited on (Issue 1540386)"""
Christian Heimes70e7ea22008-02-28 20:02:27 +000062 pid = os.fork()
63 if pid == 0:
Andrew Svetlov737fb892012-12-18 21:14:22 +020064 # Don't raise an exception; it would be caught by the test harness.
Christian Heimes70e7ea22008-02-28 20:02:27 +000065 os._exit(72)
xdegaye9001d1f2017-11-18 18:10:53 +010066 try:
67 yield None
68 except:
69 raise
70 finally:
Victor Stinner278c1e12020-03-31 20:08:12 +020071 test.support.wait_process(pid, exitcode=72)
Christian Heimes70e7ea22008-02-28 20:02:27 +000072
73
Christian Heimes292d3512008-02-03 16:51:08 +000074class SocketServerTest(unittest.TestCase):
75 """Test all socket servers."""
Guido van Rossum39f1b362001-07-10 11:52:38 +000076
Christian Heimes292d3512008-02-03 16:51:08 +000077 def setUp(self):
Victor Stinner5702ae62011-07-14 14:53:24 +020078 signal_alarm(60) # Kill deadlocks after 60 seconds.
Christian Heimes292d3512008-02-03 16:51:08 +000079 self.port_seed = 0
80 self.test_files = []
Guido van Rossum39f1b362001-07-10 11:52:38 +000081
Christian Heimes292d3512008-02-03 16:51:08 +000082 def tearDown(self):
Christian Heimesfe337bf2008-03-23 21:54:12 +000083 signal_alarm(0) # Didn't deadlock.
Christian Heimes292d3512008-02-03 16:51:08 +000084 reap_children()
Thomas Wouters00ee7ba2006-08-21 19:07:27 +000085
Christian Heimes292d3512008-02-03 16:51:08 +000086 for fn in self.test_files:
87 try:
88 os.remove(fn)
Andrew Svetlovad28c7f2012-12-18 22:02:39 +020089 except OSError:
Christian Heimes292d3512008-02-03 16:51:08 +000090 pass
91 self.test_files[:] = []
Christian Heimes292d3512008-02-03 16:51:08 +000092
93 def pickaddr(self, proto):
94 if proto == socket.AF_INET:
Christian Heimes9e7f1d22008-02-28 12:27:11 +000095 return (HOST, 0)
Christian Heimes292d3512008-02-03 16:51:08 +000096 else:
Christian Heimes9e7f1d22008-02-28 12:27:11 +000097 # XXX: We need a way to tell AF_UNIX to pick its own name
98 # like AF_INET provides port==0.
99 dir = None
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000100 fn = tempfile.mktemp(prefix='unix_socket.', dir=dir)
Christian Heimes292d3512008-02-03 16:51:08 +0000101 self.test_files.append(fn)
102 return fn
103
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000104 def make_server(self, addr, svrcls, hdlrbase):
105 class MyServer(svrcls):
106 def handle_error(self, request, client_address):
107 self.close_request(request)
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000108 raise
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000109
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000110 class MyHandler(hdlrbase):
111 def handle(self):
112 line = self.rfile.readline()
113 self.wfile.write(line)
114
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000115 if verbose: print("creating server")
xdegaye9001d1f2017-11-18 18:10:53 +0100116 try:
117 server = MyServer(addr, MyHandler)
118 except PermissionError as e:
119 # Issue 29184: cannot bind() a Unix socket on Android.
120 self.skipTest('Cannot create server (%s, %s): %s' %
121 (svrcls, addr, e))
Ezio Melottib3aedd42010-11-20 19:04:17 +0000122 self.assertEqual(server.server_address, server.socket.getsockname())
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000123 return server
124
Hai Shie80697d2020-05-28 06:10:27 +0800125 @threading_helper.reap_threads
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000126 def run_server(self, svrcls, hdlrbase, testfunc):
127 server = self.make_server(self.pickaddr(svrcls.address_family),
128 svrcls, hdlrbase)
129 # We had the OS pick a port, so pull the real address out of
130 # the server.
131 addr = server.server_address
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000132 if verbose:
133 print("ADDR =", addr)
134 print("CLASS =", svrcls)
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000135
136 t = threading.Thread(
137 name='%s serving' % svrcls,
138 target=server.serve_forever,
139 # Short poll interval to make the test finish quickly.
140 # Time between requests is short enough that we won't wake
141 # up spuriously too many times.
142 kwargs={'poll_interval':0.01})
Benjamin Peterson72753702008-08-18 18:09:21 +0000143 t.daemon = True # In case this function raises.
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000144 t.start()
145 if verbose: print("server running")
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000146 for i in range(3):
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000147 if verbose: print("test client", i)
148 testfunc(svrcls.address_family, addr)
149 if verbose: print("waiting for server")
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000150 server.shutdown()
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000151 t.join()
Victor Stinner109761b2011-01-03 14:30:39 +0000152 server.server_close()
Robert Collins1ee92832015-07-29 12:52:40 +1200153 self.assertEqual(-1, server.socket.fileno())
Victor Stinneraa8ec342017-08-10 15:28:16 +0200154 if HAVE_FORKING and isinstance(server, socketserver.ForkingMixIn):
155 # bpo-31151: Check that ForkingMixIn.server_close() waits until
156 # all children completed
157 self.assertFalse(server.active_children)
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000158 if verbose: print("done")
Christian Heimes292d3512008-02-03 16:51:08 +0000159
160 def stream_examine(self, proto, addr):
Serhiy Storchaka5b10b982019-03-05 10:06:26 +0200161 with socket.socket(proto, socket.SOCK_STREAM) as s:
162 s.connect(addr)
163 s.sendall(TEST_STR)
164 buf = data = receive(s, 100)
165 while data and b'\n' not in buf:
166 data = receive(s, 100)
167 buf += data
168 self.assertEqual(buf, TEST_STR)
Christian Heimes292d3512008-02-03 16:51:08 +0000169
170 def dgram_examine(self, proto, addr):
Serhiy Storchaka5b10b982019-03-05 10:06:26 +0200171 with socket.socket(proto, socket.SOCK_DGRAM) as s:
172 if HAVE_UNIX_SOCKETS and proto == socket.AF_UNIX:
173 s.bind(self.pickaddr(proto))
174 s.sendto(TEST_STR, addr)
175 buf = data = receive(s, 100)
176 while data and b'\n' not in buf:
177 data = receive(s, 100)
178 buf += data
179 self.assertEqual(buf, TEST_STR)
Christian Heimes292d3512008-02-03 16:51:08 +0000180
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000181 def test_TCPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000182 self.run_server(socketserver.TCPServer,
183 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000184 self.stream_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000185
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000186 def test_ThreadingTCPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000187 self.run_server(socketserver.ThreadingTCPServer,
188 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000189 self.stream_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000190
Serhiy Storchaka43767632013-11-03 21:31:38 +0200191 @requires_forking
192 def test_ForkingTCPServer(self):
193 with simple_subprocess(self):
194 self.run_server(socketserver.ForkingTCPServer,
Alexandre Vassalottice261952008-05-12 02:31:37 +0000195 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000196 self.stream_examine)
197
Serhiy Storchaka43767632013-11-03 21:31:38 +0200198 @requires_unix_sockets
199 def test_UnixStreamServer(self):
200 self.run_server(socketserver.UnixStreamServer,
201 socketserver.StreamRequestHandler,
202 self.stream_examine)
203
204 @requires_unix_sockets
205 def test_ThreadingUnixStreamServer(self):
206 self.run_server(socketserver.ThreadingUnixStreamServer,
207 socketserver.StreamRequestHandler,
208 self.stream_examine)
209
210 @requires_unix_sockets
211 @requires_forking
212 def test_ForkingUnixStreamServer(self):
213 with simple_subprocess(self):
214 self.run_server(ForkingUnixStreamServer,
Alexandre Vassalottice261952008-05-12 02:31:37 +0000215 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000216 self.stream_examine)
217
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000218 def test_UDPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000219 self.run_server(socketserver.UDPServer,
220 socketserver.DatagramRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000221 self.dgram_examine)
222
223 def test_ThreadingUDPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000224 self.run_server(socketserver.ThreadingUDPServer,
225 socketserver.DatagramRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000226 self.dgram_examine)
227
Serhiy Storchaka43767632013-11-03 21:31:38 +0200228 @requires_forking
229 def test_ForkingUDPServer(self):
230 with simple_subprocess(self):
231 self.run_server(socketserver.ForkingUDPServer,
232 socketserver.DatagramRequestHandler,
233 self.dgram_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000234
Martin Panter1827eff2016-02-24 05:12:59 +0000235 @requires_unix_sockets
236 def test_UnixDatagramServer(self):
237 self.run_server(socketserver.UnixDatagramServer,
238 socketserver.DatagramRequestHandler,
239 self.dgram_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000240
Martin Panter1827eff2016-02-24 05:12:59 +0000241 @requires_unix_sockets
242 def test_ThreadingUnixDatagramServer(self):
243 self.run_server(socketserver.ThreadingUnixDatagramServer,
244 socketserver.DatagramRequestHandler,
245 self.dgram_examine)
246
247 @requires_unix_sockets
248 @requires_forking
249 def test_ForkingUnixDatagramServer(self):
250 self.run_server(ForkingUnixDatagramServer,
251 socketserver.DatagramRequestHandler,
252 self.dgram_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000253
Hai Shie80697d2020-05-28 06:10:27 +0800254 @threading_helper.reap_threads
Antoine Pitrou3bcba8e2010-04-25 22:01:43 +0000255 def test_shutdown(self):
256 # Issue #2302: shutdown() should always succeed in making an
257 # other thread leave serve_forever().
258 class MyServer(socketserver.TCPServer):
259 pass
260
261 class MyHandler(socketserver.StreamRequestHandler):
262 pass
263
264 threads = []
265 for i in range(20):
266 s = MyServer((HOST, 0), MyHandler)
267 t = threading.Thread(
268 name='MyServer serving',
269 target=s.serve_forever,
270 kwargs={'poll_interval':0.01})
271 t.daemon = True # In case this function raises.
272 threads.append((t, s))
273 for t, s in threads:
274 t.start()
275 s.shutdown()
276 for t, s in threads:
277 t.join()
Victor Stinner109761b2011-01-03 14:30:39 +0000278 s.server_close()
Antoine Pitrou3bcba8e2010-04-25 22:01:43 +0000279
Jason R. Coombsb5711c92020-12-31 15:19:30 -0500280 def test_close_immediately(self):
281 class MyServer(socketserver.ThreadingMixIn, socketserver.TCPServer):
282 pass
283
284 server = MyServer((HOST, 0), lambda: None)
285 server.server_close()
286
Charles-François Natali0f4f0482014-10-13 19:19:26 +0100287 def test_tcpserver_bind_leak(self):
288 # Issue #22435: the server socket wouldn't be closed if bind()/listen()
289 # failed.
290 # Create many servers for which bind() will fail, to see if this result
291 # in FD exhaustion.
292 for i in range(1024):
293 with self.assertRaises(OverflowError):
294 socketserver.TCPServer((HOST, -1),
295 socketserver.StreamRequestHandler)
296
Martin Panter0cab9c12016-04-13 00:36:52 +0000297 def test_context_manager(self):
298 with socketserver.TCPServer((HOST, 0),
299 socketserver.StreamRequestHandler) as server:
300 pass
301 self.assertEqual(-1, server.socket.fileno())
302
Guido van Rossum39f1b362001-07-10 11:52:38 +0000303
Martin Panterd9108d12016-02-21 08:49:56 +0000304class ErrorHandlerTest(unittest.TestCase):
305 """Test that the servers pass normal exceptions from the handler to
306 handle_error(), and that exiting exceptions like SystemExit and
307 KeyboardInterrupt are not passed."""
308
309 def tearDown(self):
Hai Shi79bb2c92020-08-06 19:51:29 +0800310 os_helper.unlink(os_helper.TESTFN)
Martin Panterd9108d12016-02-21 08:49:56 +0000311
312 def test_sync_handled(self):
313 BaseErrorTestServer(ValueError)
314 self.check_result(handled=True)
315
316 def test_sync_not_handled(self):
317 with self.assertRaises(SystemExit):
318 BaseErrorTestServer(SystemExit)
319 self.check_result(handled=False)
320
Martin Panterd9108d12016-02-21 08:49:56 +0000321 def test_threading_handled(self):
322 ThreadingErrorTestServer(ValueError)
323 self.check_result(handled=True)
324
Martin Panterd9108d12016-02-21 08:49:56 +0000325 def test_threading_not_handled(self):
326 ThreadingErrorTestServer(SystemExit)
327 self.check_result(handled=False)
328
329 @requires_forking
330 def test_forking_handled(self):
331 ForkingErrorTestServer(ValueError)
332 self.check_result(handled=True)
333
334 @requires_forking
335 def test_forking_not_handled(self):
336 ForkingErrorTestServer(SystemExit)
337 self.check_result(handled=False)
338
339 def check_result(self, handled):
Hai Shi79bb2c92020-08-06 19:51:29 +0800340 with open(os_helper.TESTFN) as log:
Martin Panterd9108d12016-02-21 08:49:56 +0000341 expected = 'Handler called\n' + 'Error handled\n' * handled
342 self.assertEqual(log.read(), expected)
343
344
345class BaseErrorTestServer(socketserver.TCPServer):
346 def __init__(self, exception):
347 self.exception = exception
348 super().__init__((HOST, 0), BadHandler)
349 with socket.create_connection(self.server_address):
350 pass
351 try:
352 self.handle_request()
353 finally:
354 self.server_close()
355 self.wait_done()
356
357 def handle_error(self, request, client_address):
Hai Shi79bb2c92020-08-06 19:51:29 +0800358 with open(os_helper.TESTFN, 'a') as log:
Martin Panterd9108d12016-02-21 08:49:56 +0000359 log.write('Error handled\n')
360
361 def wait_done(self):
362 pass
363
364
365class BadHandler(socketserver.BaseRequestHandler):
366 def handle(self):
Hai Shi79bb2c92020-08-06 19:51:29 +0800367 with open(os_helper.TESTFN, 'a') as log:
Martin Panterd9108d12016-02-21 08:49:56 +0000368 log.write('Handler called\n')
369 raise self.server.exception('Test error')
370
371
372class ThreadingErrorTestServer(socketserver.ThreadingMixIn,
373 BaseErrorTestServer):
374 def __init__(self, *pos, **kw):
375 self.done = threading.Event()
376 super().__init__(*pos, **kw)
377
378 def shutdown_request(self, *pos, **kw):
379 super().shutdown_request(*pos, **kw)
380 self.done.set()
381
382 def wait_done(self):
383 self.done.wait()
384
385
Gregory P. Smith ext:(%20%5BGoogle%20Inc.%5D)a9459692016-06-03 05:26:14 +0000386if HAVE_FORKING:
387 class ForkingErrorTestServer(socketserver.ForkingMixIn, BaseErrorTestServer):
Victor Stinneraa8ec342017-08-10 15:28:16 +0200388 pass
Martin Panterd9108d12016-02-21 08:49:56 +0000389
390
Martin Panter34eeed42016-06-29 10:12:22 +0000391class SocketWriterTest(unittest.TestCase):
392 def test_basics(self):
393 class Handler(socketserver.StreamRequestHandler):
394 def handle(self):
395 self.server.wfile = self.wfile
396 self.server.wfile_fileno = self.wfile.fileno()
397 self.server.request_fileno = self.request.fileno()
398
399 server = socketserver.TCPServer((HOST, 0), Handler)
400 self.addCleanup(server.server_close)
401 s = socket.socket(
402 server.address_family, socket.SOCK_STREAM, socket.IPPROTO_TCP)
403 with s:
404 s.connect(server.server_address)
405 server.handle_request()
406 self.assertIsInstance(server.wfile, io.BufferedIOBase)
407 self.assertEqual(server.wfile_fileno, server.request_fileno)
408
Martin Panter34eeed42016-06-29 10:12:22 +0000409 def test_write(self):
410 # Test that wfile.write() sends data immediately, and that it does
411 # not truncate sends when interrupted by a Unix signal
412 pthread_kill = test.support.get_attribute(signal, 'pthread_kill')
413
414 class Handler(socketserver.StreamRequestHandler):
415 def handle(self):
416 self.server.sent1 = self.wfile.write(b'write data\n')
417 # Should be sent immediately, without requiring flush()
418 self.server.received = self.rfile.readline()
Serhiy Storchaka5f1a5182016-09-11 14:41:02 +0300419 big_chunk = b'\0' * test.support.SOCK_MAX_SIZE
Martin Panter34eeed42016-06-29 10:12:22 +0000420 self.server.sent2 = self.wfile.write(big_chunk)
421
422 server = socketserver.TCPServer((HOST, 0), Handler)
423 self.addCleanup(server.server_close)
424 interrupted = threading.Event()
425
426 def signal_handler(signum, frame):
427 interrupted.set()
428
429 original = signal.signal(signal.SIGUSR1, signal_handler)
430 self.addCleanup(signal.signal, signal.SIGUSR1, original)
431 response1 = None
432 received2 = None
433 main_thread = threading.get_ident()
434
435 def run_client():
436 s = socket.socket(server.address_family, socket.SOCK_STREAM,
437 socket.IPPROTO_TCP)
438 with s, s.makefile('rb') as reader:
439 s.connect(server.server_address)
440 nonlocal response1
441 response1 = reader.readline()
442 s.sendall(b'client response\n')
443
444 reader.read(100)
445 # The main thread should now be blocking in a send() syscall.
446 # But in theory, it could get interrupted by other signals,
447 # and then retried. So keep sending the signal in a loop, in
448 # case an earlier signal happens to be delivered at an
449 # inconvenient moment.
450 while True:
451 pthread_kill(main_thread, signal.SIGUSR1)
452 if interrupted.wait(timeout=float(1)):
453 break
454 nonlocal received2
455 received2 = len(reader.read())
456
457 background = threading.Thread(target=run_client)
458 background.start()
459 server.handle_request()
460 background.join()
461 self.assertEqual(server.sent1, len(response1))
462 self.assertEqual(response1, b'write data\n')
463 self.assertEqual(server.received, b'client response\n')
464 self.assertEqual(server.sent2, test.support.SOCK_MAX_SIZE)
465 self.assertEqual(received2, test.support.SOCK_MAX_SIZE - 100)
466
467
Berker Peksag32653442015-02-03 11:55:09 +0200468class MiscTestCase(unittest.TestCase):
Tim Petersa9f6f222001-09-17 23:56:20 +0000469
Berker Peksag32653442015-02-03 11:55:09 +0200470 def test_all(self):
471 # objects defined in the module should be in __all__
472 expected = []
473 for name in dir(socketserver):
474 if not name.startswith('_'):
475 mod_object = getattr(socketserver, name)
476 if getattr(mod_object, '__module__', None) == 'socketserver':
477 expected.append(name)
478 self.assertCountEqual(socketserver.__all__, expected)
479
Martin Panter4bf42742016-02-19 02:16:42 +0000480 def test_shutdown_request_called_if_verify_request_false(self):
481 # Issue #26309: BaseServer should call shutdown_request even if
482 # verify_request is False
483
484 class MyServer(socketserver.TCPServer):
485 def verify_request(self, request, client_address):
486 return False
487
488 shutdown_called = 0
489 def shutdown_request(self, request):
490 self.shutdown_called += 1
491 socketserver.TCPServer.shutdown_request(self, request)
492
493 server = MyServer((HOST, 0), socketserver.StreamRequestHandler)
494 s = socket.socket(server.address_family, socket.SOCK_STREAM)
495 s.connect(server.server_address)
496 s.close()
497 server.handle_request()
498 self.assertEqual(server.shutdown_called, 1)
499 server.server_close()
500
Jason R. Coombsb5711c92020-12-31 15:19:30 -0500501 def test_threads_reaped(self):
502 """
503 In #37193, users reported a memory leak
504 due to the saving of every request thread. Ensure that
505 not all threads are kept forever.
506 """
507 class MyServer(socketserver.ThreadingMixIn, socketserver.TCPServer):
508 pass
509
510 server = MyServer((HOST, 0), socketserver.StreamRequestHandler)
511 for n in range(10):
512 with socket.create_connection(server.server_address):
513 server.handle_request()
514 self.assertLess(len(server._threads), 10)
515 server.server_close()
516
Guido van Rossum39f1b362001-07-10 11:52:38 +0000517
Tim Petersa9f6f222001-09-17 23:56:20 +0000518if __name__ == "__main__":
Berker Peksag32653442015-02-03 11:55:09 +0200519 unittest.main()