blob: 31ab3b64f5694f26b7d3d606bd63c2189c2df9d2 [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 Heimes9e7f1d22008-02-28 12:27:11 +00006import os
Guido van Rossum39f1b362001-07-10 11:52:38 +00007import select
Christian Heimes9e7f1d22008-02-28 12:27:11 +00008import signal
9import socket
Antoine Pitroub0a9c662012-04-09 00:47:24 +020010import select
11import errno
Christian Heimes9e7f1d22008-02-28 12:27:11 +000012import tempfile
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
Victor Stinner45df8202010-04-28 22:31:17 +000018try:
19 import threading
20except ImportError:
21 threading = None
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"
Benjamin Petersonee8712c2008-05-20 21:35:26 +000026HOST = test.support.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
Christian Heimes292d3512008-02-03 16:51:08 +000042def receive(sock, n, timeout=20):
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
Christian Heimes15ebc882008-02-04 18:48:49 +000049if HAVE_UNIX_SOCKETS:
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):
61 pid = os.fork()
62 if pid == 0:
Andrew Svetlov737fb892012-12-18 21:14:22 +020063 # Don't raise an exception; it would be caught by the test harness.
Christian Heimes70e7ea22008-02-28 20:02:27 +000064 os._exit(72)
65 yield None
66 pid2, status = os.waitpid(pid, 0)
Ezio Melottib3aedd42010-11-20 19:04:17 +000067 testcase.assertEqual(pid2, pid)
68 testcase.assertEqual(72 << 8, status)
Christian Heimes70e7ea22008-02-28 20:02:27 +000069
70
Victor Stinner5f22e722010-05-26 17:33:03 +000071@unittest.skipUnless(threading, 'Threading required for this test.')
Christian Heimes292d3512008-02-03 16:51:08 +000072class SocketServerTest(unittest.TestCase):
73 """Test all socket servers."""
Guido van Rossum39f1b362001-07-10 11:52:38 +000074
Christian Heimes292d3512008-02-03 16:51:08 +000075 def setUp(self):
Victor Stinner5702ae62011-07-14 14:53:24 +020076 signal_alarm(60) # Kill deadlocks after 60 seconds.
Christian Heimes292d3512008-02-03 16:51:08 +000077 self.port_seed = 0
78 self.test_files = []
Guido van Rossum39f1b362001-07-10 11:52:38 +000079
Christian Heimes292d3512008-02-03 16:51:08 +000080 def tearDown(self):
Christian Heimesfe337bf2008-03-23 21:54:12 +000081 signal_alarm(0) # Didn't deadlock.
Christian Heimes292d3512008-02-03 16:51:08 +000082 reap_children()
Thomas Wouters00ee7ba2006-08-21 19:07:27 +000083
Christian Heimes292d3512008-02-03 16:51:08 +000084 for fn in self.test_files:
85 try:
86 os.remove(fn)
Andrew Svetlovad28c7f2012-12-18 22:02:39 +020087 except OSError:
Christian Heimes292d3512008-02-03 16:51:08 +000088 pass
89 self.test_files[:] = []
Christian Heimes292d3512008-02-03 16:51:08 +000090
91 def pickaddr(self, proto):
92 if proto == socket.AF_INET:
Christian Heimes9e7f1d22008-02-28 12:27:11 +000093 return (HOST, 0)
Christian Heimes292d3512008-02-03 16:51:08 +000094 else:
Christian Heimes9e7f1d22008-02-28 12:27:11 +000095 # XXX: We need a way to tell AF_UNIX to pick its own name
96 # like AF_INET provides port==0.
97 dir = None
Christian Heimes9e7f1d22008-02-28 12:27:11 +000098 fn = tempfile.mktemp(prefix='unix_socket.', dir=dir)
Christian Heimes292d3512008-02-03 16:51:08 +000099 self.test_files.append(fn)
100 return fn
101
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000102 def make_server(self, addr, svrcls, hdlrbase):
103 class MyServer(svrcls):
104 def handle_error(self, request, client_address):
105 self.close_request(request)
106 self.server_close()
107 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")
115 server = MyServer(addr, MyHandler)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000116 self.assertEqual(server.server_address, server.socket.getsockname())
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000117 return server
118
Antoine Pitrouc00d4b42009-10-27 21:32:38 +0000119 @reap_threads
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000120 def run_server(self, svrcls, hdlrbase, testfunc):
121 server = self.make_server(self.pickaddr(svrcls.address_family),
122 svrcls, hdlrbase)
123 # We had the OS pick a port, so pull the real address out of
124 # the server.
125 addr = server.server_address
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000126 if verbose:
127 print("ADDR =", addr)
128 print("CLASS =", svrcls)
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000129
130 t = threading.Thread(
131 name='%s serving' % svrcls,
132 target=server.serve_forever,
133 # Short poll interval to make the test finish quickly.
134 # Time between requests is short enough that we won't wake
135 # up spuriously too many times.
136 kwargs={'poll_interval':0.01})
Benjamin Peterson72753702008-08-18 18:09:21 +0000137 t.daemon = True # In case this function raises.
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000138 t.start()
139 if verbose: print("server running")
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000140 for i in range(3):
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000141 if verbose: print("test client", i)
142 testfunc(svrcls.address_family, addr)
143 if verbose: print("waiting for server")
Christian Heimesdd15f6c2008-03-16 00:07:10 +0000144 server.shutdown()
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000145 t.join()
Victor Stinner109761b2011-01-03 14:30:39 +0000146 server.server_close()
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000147 if verbose: print("done")
Christian Heimes292d3512008-02-03 16:51:08 +0000148
149 def stream_examine(self, proto, addr):
150 s = socket.socket(proto, socket.SOCK_STREAM)
151 s.connect(addr)
152 s.sendall(TEST_STR)
153 buf = data = receive(s, 100)
Christian Heimesfdb6bb52008-02-03 17:32:13 +0000154 while data and b'\n' not in buf:
Christian Heimes292d3512008-02-03 16:51:08 +0000155 data = receive(s, 100)
156 buf += data
Ezio Melottib3aedd42010-11-20 19:04:17 +0000157 self.assertEqual(buf, TEST_STR)
Christian Heimes292d3512008-02-03 16:51:08 +0000158 s.close()
159
160 def dgram_examine(self, proto, addr):
161 s = socket.socket(proto, socket.SOCK_DGRAM)
162 s.sendto(TEST_STR, addr)
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
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000170 def test_TCPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000171 self.run_server(socketserver.TCPServer,
172 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000173 self.stream_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000174
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000175 def test_ThreadingTCPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000176 self.run_server(socketserver.ThreadingTCPServer,
177 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000178 self.stream_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000179
Serhiy Storchaka43767632013-11-03 21:31:38 +0200180 @requires_forking
181 def test_ForkingTCPServer(self):
182 with simple_subprocess(self):
183 self.run_server(socketserver.ForkingTCPServer,
Alexandre Vassalottice261952008-05-12 02:31:37 +0000184 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000185 self.stream_examine)
186
Serhiy Storchaka43767632013-11-03 21:31:38 +0200187 @requires_unix_sockets
188 def test_UnixStreamServer(self):
189 self.run_server(socketserver.UnixStreamServer,
190 socketserver.StreamRequestHandler,
191 self.stream_examine)
192
193 @requires_unix_sockets
194 def test_ThreadingUnixStreamServer(self):
195 self.run_server(socketserver.ThreadingUnixStreamServer,
196 socketserver.StreamRequestHandler,
197 self.stream_examine)
198
199 @requires_unix_sockets
200 @requires_forking
201 def test_ForkingUnixStreamServer(self):
202 with simple_subprocess(self):
203 self.run_server(ForkingUnixStreamServer,
Alexandre Vassalottice261952008-05-12 02:31:37 +0000204 socketserver.StreamRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000205 self.stream_examine)
206
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000207 def test_UDPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000208 self.run_server(socketserver.UDPServer,
209 socketserver.DatagramRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000210 self.dgram_examine)
211
212 def test_ThreadingUDPServer(self):
Alexandre Vassalottice261952008-05-12 02:31:37 +0000213 self.run_server(socketserver.ThreadingUDPServer,
214 socketserver.DatagramRequestHandler,
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000215 self.dgram_examine)
216
Serhiy Storchaka43767632013-11-03 21:31:38 +0200217 @requires_forking
218 def test_ForkingUDPServer(self):
219 with simple_subprocess(self):
220 self.run_server(socketserver.ForkingUDPServer,
221 socketserver.DatagramRequestHandler,
222 self.dgram_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000223
224 # Alas, on Linux (at least) recvfrom() doesn't return a meaningful
225 # client address so this cannot work:
226
Serhiy Storchaka43767632013-11-03 21:31:38 +0200227 # @requires_unix_sockets
228 # def test_UnixDatagramServer(self):
229 # self.run_server(socketserver.UnixDatagramServer,
230 # socketserver.DatagramRequestHandler,
231 # self.dgram_examine)
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000232 #
Serhiy Storchaka43767632013-11-03 21:31:38 +0200233 # @requires_unix_sockets
234 # def test_ThreadingUnixDatagramServer(self):
235 # self.run_server(socketserver.ThreadingUnixDatagramServer,
236 # socketserver.DatagramRequestHandler,
237 # self.dgram_examine)
Christian Heimes9e7f1d22008-02-28 12:27:11 +0000238 #
Serhiy Storchaka43767632013-11-03 21:31:38 +0200239 # @requires_unix_sockets
240 # @requires_forking
241 # def test_ForkingUnixDatagramServer(self):
242 # self.run_server(socketserver.ForkingUnixDatagramServer,
243 # socketserver.DatagramRequestHandler,
244 # self.dgram_examine)
Christian Heimes292d3512008-02-03 16:51:08 +0000245
Antoine Pitrou3bcba8e2010-04-25 22:01:43 +0000246 @reap_threads
247 def test_shutdown(self):
248 # Issue #2302: shutdown() should always succeed in making an
249 # other thread leave serve_forever().
250 class MyServer(socketserver.TCPServer):
251 pass
252
253 class MyHandler(socketserver.StreamRequestHandler):
254 pass
255
256 threads = []
257 for i in range(20):
258 s = MyServer((HOST, 0), MyHandler)
259 t = threading.Thread(
260 name='MyServer serving',
261 target=s.serve_forever,
262 kwargs={'poll_interval':0.01})
263 t.daemon = True # In case this function raises.
264 threads.append((t, s))
265 for t, s in threads:
266 t.start()
267 s.shutdown()
268 for t, s in threads:
269 t.join()
Victor Stinner109761b2011-01-03 14:30:39 +0000270 s.server_close()
Antoine Pitrou3bcba8e2010-04-25 22:01:43 +0000271
Charles-François Natali0f4f0482014-10-13 19:19:26 +0100272 def test_tcpserver_bind_leak(self):
273 # Issue #22435: the server socket wouldn't be closed if bind()/listen()
274 # failed.
275 # Create many servers for which bind() will fail, to see if this result
276 # in FD exhaustion.
277 for i in range(1024):
278 with self.assertRaises(OverflowError):
279 socketserver.TCPServer((HOST, -1),
280 socketserver.StreamRequestHandler)
281
Guido van Rossum39f1b362001-07-10 11:52:38 +0000282
Berker Peksag32653442015-02-03 11:55:09 +0200283class MiscTestCase(unittest.TestCase):
Tim Petersa9f6f222001-09-17 23:56:20 +0000284
Berker Peksag32653442015-02-03 11:55:09 +0200285 def test_all(self):
286 # objects defined in the module should be in __all__
287 expected = []
288 for name in dir(socketserver):
289 if not name.startswith('_'):
290 mod_object = getattr(socketserver, name)
291 if getattr(mod_object, '__module__', None) == 'socketserver':
292 expected.append(name)
293 self.assertCountEqual(socketserver.__all__, expected)
294
Guido van Rossum39f1b362001-07-10 11:52:38 +0000295
Tim Petersa9f6f222001-09-17 23:56:20 +0000296if __name__ == "__main__":
Berker Peksag32653442015-02-03 11:55:09 +0200297 unittest.main()