blob: 59d8e5dcb76ff2bbe1271d0a6c65f65d9ae492d6 [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
Christian Heimes292d3512008-02-03 16:51:08 +00006import imp
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
Antoine Pitroub0a9c662012-04-09 00:47:24 +020011import select
12import errno
Christian Heimes9e7f1d22008-02-28 12:27:11 +000013import tempfile
Christian Heimes292d3512008-02-03 16:51:08 +000014import unittest
Alexandre Vassalottice261952008-05-12 02:31:37 +000015import socketserver
Christian Heimes292d3512008-02-03 16:51:08 +000016
Benjamin Petersonee8712c2008-05-20 21:35:26 +000017import test.support
Antoine Pitrouc00d4b42009-10-27 21:32:38 +000018from test.support import reap_children, reap_threads, verbose
Victor Stinner45df8202010-04-28 22:31:17 +000019try:
20 import threading
21except ImportError:
22 threading = None
Christian Heimes292d3512008-02-03 16:51:08 +000023
Benjamin Petersonee8712c2008-05-20 21:35:26 +000024test.support.requires("network")
Guido van Rossum39f1b362001-07-10 11:52:38 +000025
Christian Heimes292d3512008-02-03 16:51:08 +000026TEST_STR = b"hello world\n"
Benjamin Petersonee8712c2008-05-20 21:35:26 +000027HOST = test.support.HOST
Christian Heimes292d3512008-02-03 16:51:08 +000028
29HAVE_UNIX_SOCKETS = hasattr(socket, "AF_UNIX")
Serhiy Storchaka79080682013-11-03 21:31:18 +020030requires_unix_sockets = unittest.skipUnless(HAVE_UNIX_SOCKETS,
31 'requires Unix sockets')
Christian Heimes292d3512008-02-03 16:51:08 +000032HAVE_FORKING = hasattr(os, "fork") and os.name != "os2"
Serhiy Storchaka79080682013-11-03 21:31:18 +020033requires_forking = unittest.skipUnless(HAVE_FORKING, 'requires forking')
Christian Heimes292d3512008-02-03 16:51:08 +000034
Christian Heimesdd15f6c2008-03-16 00:07:10 +000035def signal_alarm(n):
36 """Call signal.alarm when it exists (i.e. not on Windows)."""
37 if hasattr(signal, 'alarm'):
38 signal.alarm(n)
Guido van Rossum39f1b362001-07-10 11:52:38 +000039
Antoine Pitrouf18d6f32012-04-09 01:16:47 +020040# Remember real select() to avoid interferences with mocking
41_real_select = select.select
42
Christian Heimes292d3512008-02-03 16:51:08 +000043def receive(sock, n, timeout=20):
Antoine Pitrouf18d6f32012-04-09 01:16:47 +020044 r, w, x = _real_select([sock], [], [], timeout)
Christian Heimes292d3512008-02-03 16:51:08 +000045 if sock in r:
46 return sock.recv(n)
47 else:
48 raise RuntimeError("timed out on %r" % (sock,))
49
Christian Heimes15ebc882008-02-04 18:48:49 +000050if HAVE_UNIX_SOCKETS:
Alexandre Vassalottice261952008-05-12 02:31:37 +000051 class ForkingUnixStreamServer(socketserver.ForkingMixIn,
52 socketserver.UnixStreamServer):
Christian Heimes15ebc882008-02-04 18:48:49 +000053 pass
Christian Heimes292d3512008-02-03 16:51:08 +000054
Alexandre Vassalottice261952008-05-12 02:31:37 +000055 class ForkingUnixDatagramServer(socketserver.ForkingMixIn,
56 socketserver.UnixDatagramServer):
Christian Heimes15ebc882008-02-04 18:48:49 +000057 pass
Christian Heimes292d3512008-02-03 16:51:08 +000058
59
Christian Heimes70e7ea22008-02-28 20:02:27 +000060@contextlib.contextmanager
61def simple_subprocess(testcase):
62 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)
66 yield None
67 pid2, status = os.waitpid(pid, 0)
Ezio Melottib3aedd42010-11-20 19:04:17 +000068 testcase.assertEqual(pid2, pid)
69 testcase.assertEqual(72 << 8, status)
Christian Heimes70e7ea22008-02-28 20:02:27 +000070
71
Victor Stinner5f22e722010-05-26 17:33:03 +000072@unittest.skipUnless(threading, 'Threading required for this test.')
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)
88 except os.error:
89 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
99 if os.name == 'os2':
100 dir = '\socket'
101 fn = tempfile.mktemp(prefix='unix_socket.', dir=dir)
Christian Heimes292d3512008-02-03 16:51:08 +0000102 if os.name == 'os2':
103 # AF_UNIX socket names on OS/2 require a specific prefix
104 # which can't include a drive letter and must also use
105 # backslashes as directory separators
106 if fn[1] == ':':
107 fn = fn[2:]
108 if fn[0] in (os.sep, os.altsep):
109 fn = fn[1:]
Christian Heimes292d3512008-02-03 16:51:08 +0000110 if os.sep == '/':
111 fn = fn.replace(os.sep, os.altsep)
112 else:
113 fn = fn.replace(os.altsep, os.sep)
114 self.test_files.append(fn)
115 return fn
116
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000117 def make_server(self, addr, svrcls, hdlrbase):
118 class MyServer(svrcls):
119 def handle_error(self, request, client_address):
120 self.close_request(request)
121 self.server_close()
122 raise
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000123
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000124 class MyHandler(hdlrbase):
125 def handle(self):
126 line = self.rfile.readline()
127 self.wfile.write(line)
128
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000129 if verbose: print("creating server")
130 server = MyServer(addr, MyHandler)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000131 self.assertEqual(server.server_address, server.socket.getsockname())
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000132 return server
133
Antoine Pitrouc00d4b42009-10-27 21:32:38 +0000134 @reap_threads
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000135 def run_server(self, svrcls, hdlrbase, testfunc):
136 server = self.make_server(self.pickaddr(svrcls.address_family),
137 svrcls, hdlrbase)
138 # We had the OS pick a port, so pull the real address out of
139 # the server.
140 addr = server.server_address
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000141 if verbose:
142 print("ADDR =", addr)
143 print("CLASS =", svrcls)
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000144
145 t = threading.Thread(
146 name='%s serving' % svrcls,
147 target=server.serve_forever,
148 # Short poll interval to make the test finish quickly.
149 # Time between requests is short enough that we won't wake
150 # up spuriously too many times.
151 kwargs={'poll_interval':0.01})
Benjamin Peterson72753702008-08-18 18:09:21 +0000152 t.daemon = True # In case this function raises.
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000153 t.start()
154 if verbose: print("server running")
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000155 for i in range(3):
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000156 if verbose: print("test client", i)
157 testfunc(svrcls.address_family, addr)
158 if verbose: print("waiting for server")
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000159 server.shutdown()
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000160 t.join()
Victor Stinner109761b2011-01-03 14:30:39 +0000161 server.server_close()
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000162 if verbose: print("done")
Christian Heimes292d3512008-02-03 16:51:08 +0000163
164 def stream_examine(self, proto, addr):
165 s = socket.socket(proto, socket.SOCK_STREAM)
166 s.connect(addr)
167 s.sendall(TEST_STR)
168 buf = data = receive(s, 100)
Christian Heimesfdb6bb52008-02-03 17:32:13 +0000169 while data and b'\n' not in buf:
Christian Heimes292d3512008-02-03 16:51:08 +0000170 data = receive(s, 100)
171 buf += data
Ezio Melottib3aedd42010-11-20 19:04:17 +0000172 self.assertEqual(buf, TEST_STR)
Christian Heimes292d3512008-02-03 16:51:08 +0000173 s.close()
174
175 def dgram_examine(self, proto, addr):
176 s = socket.socket(proto, socket.SOCK_DGRAM)
177 s.sendto(TEST_STR, addr)
178 buf = data = receive(s, 100)
Christian Heimesfdb6bb52008-02-03 17:32:13 +0000179 while data and b'\n' not in buf:
Christian Heimes292d3512008-02-03 16:51:08 +0000180 data = receive(s, 100)
181 buf += data
Ezio Melottib3aedd42010-11-20 19:04:17 +0000182 self.assertEqual(buf, TEST_STR)
Christian Heimes292d3512008-02-03 16:51:08 +0000183 s.close()
184
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000185 def test_TCPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000186 self.run_server(socketserver.TCPServer,
187 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000188 self.stream_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000189
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000190 def test_ThreadingTCPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000191 self.run_server(socketserver.ThreadingTCPServer,
192 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000193 self.stream_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000194
Serhiy Storchaka79080682013-11-03 21:31:18 +0200195 @requires_forking
196 def test_ForkingTCPServer(self):
197 with simple_subprocess(self):
198 self.run_server(socketserver.ForkingTCPServer,
Alexandre Vassalottice261952008-05-12 02:31:37 +0000199 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000200 self.stream_examine)
201
Serhiy Storchaka79080682013-11-03 21:31:18 +0200202 @requires_unix_sockets
203 def test_UnixStreamServer(self):
204 self.run_server(socketserver.UnixStreamServer,
205 socketserver.StreamRequestHandler,
206 self.stream_examine)
207
208 @requires_unix_sockets
209 def test_ThreadingUnixStreamServer(self):
210 self.run_server(socketserver.ThreadingUnixStreamServer,
211 socketserver.StreamRequestHandler,
212 self.stream_examine)
213
214 @requires_unix_sockets
215 @requires_forking
216 def test_ForkingUnixStreamServer(self):
217 with simple_subprocess(self):
218 self.run_server(ForkingUnixStreamServer,
Alexandre Vassalottice261952008-05-12 02:31:37 +0000219 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000220 self.stream_examine)
221
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000222 def test_UDPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000223 self.run_server(socketserver.UDPServer,
224 socketserver.DatagramRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000225 self.dgram_examine)
226
227 def test_ThreadingUDPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000228 self.run_server(socketserver.ThreadingUDPServer,
229 socketserver.DatagramRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000230 self.dgram_examine)
231
Serhiy Storchaka79080682013-11-03 21:31:18 +0200232 @requires_forking
233 def test_ForkingUDPServer(self):
234 with simple_subprocess(self):
235 self.run_server(socketserver.ForkingUDPServer,
236 socketserver.DatagramRequestHandler,
237 self.dgram_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000238
Antoine Pitroub0a9c662012-04-09 00:47:24 +0200239 @contextlib.contextmanager
240 def mocked_select_module(self):
241 """Mocks the select.select() call to raise EINTR for first call"""
242 old_select = select.select
243
244 class MockSelect:
245 def __init__(self):
246 self.called = 0
247
248 def __call__(self, *args):
249 self.called += 1
250 if self.called == 1:
251 # raise the exception on first call
252 raise OSError(errno.EINTR, os.strerror(errno.EINTR))
253 else:
254 # Return real select value for consecutive calls
255 return old_select(*args)
256
257 select.select = MockSelect()
258 try:
259 yield select.select
260 finally:
261 select.select = old_select
262
263 def test_InterruptServerSelectCall(self):
264 with self.mocked_select_module() as mock_select:
265 pid = self.run_server(socketserver.TCPServer,
266 socketserver.StreamRequestHandler,
267 self.stream_examine)
268 # Make sure select was called again:
269 self.assertGreater(mock_select.called, 1)
270
Christian Heimes292d3512008-02-03 16:51:08 +0000271 # Alas, on Linux (at least) recvfrom() doesn't return a meaningful
272 # client address so this cannot work:
273
Serhiy Storchaka79080682013-11-03 21:31:18 +0200274 # @requires_unix_sockets
275 # def test_UnixDatagramServer(self):
276 # self.run_server(socketserver.UnixDatagramServer,
277 # socketserver.DatagramRequestHandler,
278 # self.dgram_examine)
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000279 #
Serhiy Storchaka79080682013-11-03 21:31:18 +0200280 # @requires_unix_sockets
281 # def test_ThreadingUnixDatagramServer(self):
282 # self.run_server(socketserver.ThreadingUnixDatagramServer,
283 # socketserver.DatagramRequestHandler,
284 # self.dgram_examine)
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000285 #
Serhiy Storchaka79080682013-11-03 21:31:18 +0200286 # @requires_unix_sockets
287 # @requires_forking
288 # def test_ForkingUnixDatagramServer(self):
289 # self.run_server(socketserver.ForkingUnixDatagramServer,
290 # socketserver.DatagramRequestHandler,
291 # self.dgram_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000292
Antoine Pitrou3bcba8e2010-04-25 22:01:43 +0000293 @reap_threads
294 def test_shutdown(self):
295 # Issue #2302: shutdown() should always succeed in making an
296 # other thread leave serve_forever().
297 class MyServer(socketserver.TCPServer):
298 pass
299
300 class MyHandler(socketserver.StreamRequestHandler):
301 pass
302
303 threads = []
304 for i in range(20):
305 s = MyServer((HOST, 0), MyHandler)
306 t = threading.Thread(
307 name='MyServer serving',
308 target=s.serve_forever,
309 kwargs={'poll_interval':0.01})
310 t.daemon = True # In case this function raises.
311 threads.append((t, s))
312 for t, s in threads:
313 t.start()
314 s.shutdown()
315 for t, s in threads:
316 t.join()
Victor Stinner109761b2011-01-03 14:30:39 +0000317 s.server_close()
Antoine Pitrou3bcba8e2010-04-25 22:01:43 +0000318
Guido van Rossum39f1b362001-07-10 11:52:38 +0000319
Tim Petersa9f6f222001-09-17 23:56:20 +0000320def test_main():
Tim Petersa9f6f222001-09-17 23:56:20 +0000321 if imp.lock_held():
Christian Heimes292d3512008-02-03 16:51:08 +0000322 # If the import lock is held, the threads will hang
Benjamin Petersone549ead2009-03-28 21:42:05 +0000323 raise unittest.SkipTest("can't run when import lock is held")
Tim Petersa9f6f222001-09-17 23:56:20 +0000324
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000325 test.support.run_unittest(SocketServerTest)
Guido van Rossum39f1b362001-07-10 11:52:38 +0000326
Tim Petersa9f6f222001-09-17 23:56:20 +0000327if __name__ == "__main__":
328 test_main()