blob: 0e76c33d1bf12f948791d68394a2f56580412549 [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")
30HAVE_FORKING = hasattr(os, "fork") and os.name != "os2"
31
Christian Heimesdd15f6c2008-03-16 00:07:10 +000032def signal_alarm(n):
33 """Call signal.alarm when it exists (i.e. not on Windows)."""
34 if hasattr(signal, 'alarm'):
35 signal.alarm(n)
Guido van Rossum39f1b362001-07-10 11:52:38 +000036
Christian Heimes292d3512008-02-03 16:51:08 +000037def receive(sock, n, timeout=20):
38 r, w, x = select.select([sock], [], [], timeout)
39 if sock in r:
40 return sock.recv(n)
41 else:
42 raise RuntimeError("timed out on %r" % (sock,))
43
Christian Heimes15ebc882008-02-04 18:48:49 +000044if HAVE_UNIX_SOCKETS:
Alexandre Vassalottice261952008-05-12 02:31:37 +000045 class ForkingUnixStreamServer(socketserver.ForkingMixIn,
46 socketserver.UnixStreamServer):
Christian Heimes15ebc882008-02-04 18:48:49 +000047 pass
Christian Heimes292d3512008-02-03 16:51:08 +000048
Alexandre Vassalottice261952008-05-12 02:31:37 +000049 class ForkingUnixDatagramServer(socketserver.ForkingMixIn,
50 socketserver.UnixDatagramServer):
Christian Heimes15ebc882008-02-04 18:48:49 +000051 pass
Christian Heimes292d3512008-02-03 16:51:08 +000052
53
Christian Heimes70e7ea22008-02-28 20:02:27 +000054@contextlib.contextmanager
55def simple_subprocess(testcase):
56 pid = os.fork()
57 if pid == 0:
58 # Don't throw an exception; it would be caught by the test harness.
59 os._exit(72)
60 yield None
61 pid2, status = os.waitpid(pid, 0)
Ezio Melottib3aedd42010-11-20 19:04:17 +000062 testcase.assertEqual(pid2, pid)
63 testcase.assertEqual(72 << 8, status)
Christian Heimes70e7ea22008-02-28 20:02:27 +000064
65
Victor Stinner5f22e722010-05-26 17:33:03 +000066@unittest.skipUnless(threading, 'Threading required for this test.')
Christian Heimes292d3512008-02-03 16:51:08 +000067class SocketServerTest(unittest.TestCase):
68 """Test all socket servers."""
Guido van Rossum39f1b362001-07-10 11:52:38 +000069
Christian Heimes292d3512008-02-03 16:51:08 +000070 def setUp(self):
Victor Stinner5702ae62011-07-14 14:53:24 +020071 signal_alarm(60) # Kill deadlocks after 60 seconds.
Christian Heimes292d3512008-02-03 16:51:08 +000072 self.port_seed = 0
73 self.test_files = []
Guido van Rossum39f1b362001-07-10 11:52:38 +000074
Christian Heimes292d3512008-02-03 16:51:08 +000075 def tearDown(self):
Christian Heimesfe337bf2008-03-23 21:54:12 +000076 signal_alarm(0) # Didn't deadlock.
Christian Heimes292d3512008-02-03 16:51:08 +000077 reap_children()
Thomas Wouters00ee7ba2006-08-21 19:07:27 +000078
Christian Heimes292d3512008-02-03 16:51:08 +000079 for fn in self.test_files:
80 try:
81 os.remove(fn)
82 except os.error:
83 pass
84 self.test_files[:] = []
Christian Heimes292d3512008-02-03 16:51:08 +000085
86 def pickaddr(self, proto):
87 if proto == socket.AF_INET:
Christian Heimes9e7f1d22008-02-28 12:27:11 +000088 return (HOST, 0)
Christian Heimes292d3512008-02-03 16:51:08 +000089 else:
Christian Heimes9e7f1d22008-02-28 12:27:11 +000090 # XXX: We need a way to tell AF_UNIX to pick its own name
91 # like AF_INET provides port==0.
92 dir = None
93 if os.name == 'os2':
94 dir = '\socket'
95 fn = tempfile.mktemp(prefix='unix_socket.', dir=dir)
Christian Heimes292d3512008-02-03 16:51:08 +000096 if os.name == 'os2':
97 # AF_UNIX socket names on OS/2 require a specific prefix
98 # which can't include a drive letter and must also use
99 # backslashes as directory separators
100 if fn[1] == ':':
101 fn = fn[2:]
102 if fn[0] in (os.sep, os.altsep):
103 fn = fn[1:]
Christian Heimes292d3512008-02-03 16:51:08 +0000104 if os.sep == '/':
105 fn = fn.replace(os.sep, os.altsep)
106 else:
107 fn = fn.replace(os.altsep, os.sep)
108 self.test_files.append(fn)
109 return fn
110
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000111 def make_server(self, addr, svrcls, hdlrbase):
112 class MyServer(svrcls):
113 def handle_error(self, request, client_address):
114 self.close_request(request)
115 self.server_close()
116 raise
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000117
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000118 class MyHandler(hdlrbase):
119 def handle(self):
120 line = self.rfile.readline()
121 self.wfile.write(line)
122
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000123 if verbose: print("creating server")
124 server = MyServer(addr, MyHandler)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000125 self.assertEqual(server.server_address, server.socket.getsockname())
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000126 return server
127
Antoine Pitrouc00d4b42009-10-27 21:32:38 +0000128 @reap_threads
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000129 def run_server(self, svrcls, hdlrbase, testfunc):
130 server = self.make_server(self.pickaddr(svrcls.address_family),
131 svrcls, hdlrbase)
132 # We had the OS pick a port, so pull the real address out of
133 # the server.
134 addr = server.server_address
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000135 if verbose:
136 print("ADDR =", addr)
137 print("CLASS =", svrcls)
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000138
139 t = threading.Thread(
140 name='%s serving' % svrcls,
141 target=server.serve_forever,
142 # Short poll interval to make the test finish quickly.
143 # Time between requests is short enough that we won't wake
144 # up spuriously too many times.
145 kwargs={'poll_interval':0.01})
Benjamin Peterson72753702008-08-18 18:09:21 +0000146 t.daemon = True # In case this function raises.
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000147 t.start()
148 if verbose: print("server running")
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000149 for i in range(3):
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000150 if verbose: print("test client", i)
151 testfunc(svrcls.address_family, addr)
152 if verbose: print("waiting for server")
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000153 server.shutdown()
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000154 t.join()
Victor Stinner109761b2011-01-03 14:30:39 +0000155 server.server_close()
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000156 if verbose: print("done")
Christian Heimes292d3512008-02-03 16:51:08 +0000157
158 def stream_examine(self, proto, addr):
159 s = socket.socket(proto, socket.SOCK_STREAM)
160 s.connect(addr)
161 s.sendall(TEST_STR)
162 buf = data = receive(s, 100)
Christian Heimesfdb6bb52008-02-03 17:32:13 +0000163 while data and b'\n' not in buf:
Christian Heimes292d3512008-02-03 16:51:08 +0000164 data = receive(s, 100)
165 buf += data
Ezio Melottib3aedd42010-11-20 19:04:17 +0000166 self.assertEqual(buf, TEST_STR)
Christian Heimes292d3512008-02-03 16:51:08 +0000167 s.close()
168
169 def dgram_examine(self, proto, addr):
170 s = socket.socket(proto, socket.SOCK_DGRAM)
171 s.sendto(TEST_STR, addr)
172 buf = data = receive(s, 100)
Christian Heimesfdb6bb52008-02-03 17:32:13 +0000173 while data and b'\n' not in buf:
Christian Heimes292d3512008-02-03 16:51:08 +0000174 data = receive(s, 100)
175 buf += data
Ezio Melottib3aedd42010-11-20 19:04:17 +0000176 self.assertEqual(buf, TEST_STR)
Christian Heimes292d3512008-02-03 16:51:08 +0000177 s.close()
178
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000179 def test_TCPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000180 self.run_server(socketserver.TCPServer,
181 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000182 self.stream_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000183
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000184 def test_ThreadingTCPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000185 self.run_server(socketserver.ThreadingTCPServer,
186 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000187 self.stream_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000188
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000189 if HAVE_FORKING:
Christian Heimes70e7ea22008-02-28 20:02:27 +0000190 def test_ForkingTCPServer(self):
191 with simple_subprocess(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000192 self.run_server(socketserver.ForkingTCPServer,
193 socketserver.StreamRequestHandler,
Christian Heimes70e7ea22008-02-28 20:02:27 +0000194 self.stream_examine)
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000195
196 if HAVE_UNIX_SOCKETS:
197 def test_UnixStreamServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000198 self.run_server(socketserver.UnixStreamServer,
199 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000200 self.stream_examine)
201
202 def test_ThreadingUnixStreamServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000203 self.run_server(socketserver.ThreadingUnixStreamServer,
204 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000205 self.stream_examine)
206
Christian Heimes292d3512008-02-03 16:51:08 +0000207 if HAVE_FORKING:
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000208 def test_ForkingUnixStreamServer(self):
Christian Heimes70e7ea22008-02-28 20:02:27 +0000209 with simple_subprocess(self):
210 self.run_server(ForkingUnixStreamServer,
Alexandre Vassalottice261952008-05-12 02:31:37 +0000211 socketserver.StreamRequestHandler,
Christian Heimes70e7ea22008-02-28 20:02:27 +0000212 self.stream_examine)
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000213
214 def test_UDPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000215 self.run_server(socketserver.UDPServer,
216 socketserver.DatagramRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000217 self.dgram_examine)
218
219 def test_ThreadingUDPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000220 self.run_server(socketserver.ThreadingUDPServer,
221 socketserver.DatagramRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000222 self.dgram_examine)
223
224 if HAVE_FORKING:
225 def test_ForkingUDPServer(self):
Christian Heimes70e7ea22008-02-28 20:02:27 +0000226 with simple_subprocess(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000227 self.run_server(socketserver.ForkingUDPServer,
228 socketserver.DatagramRequestHandler,
Christian Heimes70e7ea22008-02-28 20:02:27 +0000229 self.dgram_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000230
Antoine Pitroub0a9c662012-04-09 00:47:24 +0200231 @contextlib.contextmanager
232 def mocked_select_module(self):
233 """Mocks the select.select() call to raise EINTR for first call"""
234 old_select = select.select
235
236 class MockSelect:
237 def __init__(self):
238 self.called = 0
239
240 def __call__(self, *args):
241 self.called += 1
242 if self.called == 1:
243 # raise the exception on first call
244 raise OSError(errno.EINTR, os.strerror(errno.EINTR))
245 else:
246 # Return real select value for consecutive calls
247 return old_select(*args)
248
249 select.select = MockSelect()
250 try:
251 yield select.select
252 finally:
253 select.select = old_select
254
255 def test_InterruptServerSelectCall(self):
256 with self.mocked_select_module() as mock_select:
257 pid = self.run_server(socketserver.TCPServer,
258 socketserver.StreamRequestHandler,
259 self.stream_examine)
260 # Make sure select was called again:
261 self.assertGreater(mock_select.called, 1)
262
Christian Heimes292d3512008-02-03 16:51:08 +0000263 # Alas, on Linux (at least) recvfrom() doesn't return a meaningful
264 # client address so this cannot work:
265
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000266 # if HAVE_UNIX_SOCKETS:
267 # def test_UnixDatagramServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000268 # self.run_server(socketserver.UnixDatagramServer,
269 # socketserver.DatagramRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000270 # self.dgram_examine)
271 #
272 # def test_ThreadingUnixDatagramServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000273 # self.run_server(socketserver.ThreadingUnixDatagramServer,
274 # socketserver.DatagramRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000275 # self.dgram_examine)
276 #
Christian Heimes292d3512008-02-03 16:51:08 +0000277 # if HAVE_FORKING:
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000278 # def test_ForkingUnixDatagramServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000279 # self.run_server(socketserver.ForkingUnixDatagramServer,
280 # socketserver.DatagramRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000281 # self.dgram_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000282
Antoine Pitrou3bcba8e2010-04-25 22:01:43 +0000283 @reap_threads
284 def test_shutdown(self):
285 # Issue #2302: shutdown() should always succeed in making an
286 # other thread leave serve_forever().
287 class MyServer(socketserver.TCPServer):
288 pass
289
290 class MyHandler(socketserver.StreamRequestHandler):
291 pass
292
293 threads = []
294 for i in range(20):
295 s = MyServer((HOST, 0), MyHandler)
296 t = threading.Thread(
297 name='MyServer serving',
298 target=s.serve_forever,
299 kwargs={'poll_interval':0.01})
300 t.daemon = True # In case this function raises.
301 threads.append((t, s))
302 for t, s in threads:
303 t.start()
304 s.shutdown()
305 for t, s in threads:
306 t.join()
Victor Stinner109761b2011-01-03 14:30:39 +0000307 s.server_close()
Antoine Pitrou3bcba8e2010-04-25 22:01:43 +0000308
Guido van Rossum39f1b362001-07-10 11:52:38 +0000309
Tim Petersa9f6f222001-09-17 23:56:20 +0000310def test_main():
Tim Petersa9f6f222001-09-17 23:56:20 +0000311 if imp.lock_held():
Christian Heimes292d3512008-02-03 16:51:08 +0000312 # If the import lock is held, the threads will hang
Benjamin Petersone549ead2009-03-28 21:42:05 +0000313 raise unittest.SkipTest("can't run when import lock is held")
Tim Petersa9f6f222001-09-17 23:56:20 +0000314
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000315 test.support.run_unittest(SocketServerTest)
Guido van Rossum39f1b362001-07-10 11:52:38 +0000316
Tim Petersa9f6f222001-09-17 23:56:20 +0000317if __name__ == "__main__":
318 test_main()