blob: 22e989624980e2312662fb5ff7518f62c01af92b [file] [log] [blame]
Christian Heimes4fbc72b2008-03-22 00:47:35 +00001# Copyright (c) 2001-2006 Twisted Matrix Laboratories.
2#
3# Permission is hereby granted, free of charge, to any person obtaining
4# a copy of this software and associated documentation files (the
5# "Software"), to deal in the Software without restriction, including
6# without limitation the rights to use, copy, modify, merge, publish,
7# distribute, sublicense, and/or sell copies of the Software, and to
8# permit persons to whom the Software is furnished to do so, subject to
9# the following conditions:
10#
11# The above copyright notice and this permission notice shall be
12# included in all copies or substantial portions of the Software.
13#
14# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15# EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16# MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17# NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18# LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19# OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20# WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
21"""
22Tests for epoll wrapper.
23"""
Christian Heimes4fbc72b2008-03-22 00:47:35 +000024import errno
Victor Stinnerdaf45552013-08-28 00:53:59 +020025import os
Christian Heimes4fbc72b2008-03-22 00:47:35 +000026import select
Victor Stinnerdaf45552013-08-28 00:53:59 +020027import socket
28import time
Christian Heimes4fbc72b2008-03-22 00:47:35 +000029import unittest
30
Benjamin Petersonee8712c2008-05-20 21:35:26 +000031from test import support
Christian Heimes4fbc72b2008-03-22 00:47:35 +000032if not hasattr(select, "epoll"):
Benjamin Petersone549ead2009-03-28 21:42:05 +000033 raise unittest.SkipTest("test works only on Linux 2.6")
Christian Heimes4fbc72b2008-03-22 00:47:35 +000034
Christian Heimesfe337bf2008-03-23 21:54:12 +000035try:
36 select.epoll()
Andrew Svetlovf7a17b42012-12-25 16:47:37 +020037except OSError as e:
Christian Heimesfe337bf2008-03-23 21:54:12 +000038 if e.errno == errno.ENOSYS:
Benjamin Petersone549ead2009-03-28 21:42:05 +000039 raise unittest.SkipTest("kernel doesn't support epoll()")
Benjamin Peterson0e613502011-12-27 15:16:34 -060040 raise
Christian Heimesfe337bf2008-03-23 21:54:12 +000041
Christian Heimes4fbc72b2008-03-22 00:47:35 +000042class TestEPoll(unittest.TestCase):
43
44 def setUp(self):
45 self.serverSocket = socket.socket()
46 self.serverSocket.bind(('127.0.0.1', 0))
47 self.serverSocket.listen(1)
48 self.connections = [self.serverSocket]
49
Victor Stinner665486e2014-01-21 01:41:00 +010050 def test_timeout_rounding(self):
51 # epoll_wait() has a resolution of 1 millisecond, check if the timeout
52 # is correctly rounded to the upper bound
53 epoll = select.epoll()
54 self.addCleanup(epoll.close)
55 for timeout in (1e-2, 1e-3, 1e-4):
56 t0 = time.perf_counter()
57 epoll.poll(timeout)
58 dt = time.perf_counter() - t0
59 self.assertGreaterEqual(dt, timeout)
60
Christian Heimes4fbc72b2008-03-22 00:47:35 +000061
62 def tearDown(self):
63 for skt in self.connections:
64 skt.close()
65
66 def _connected_pair(self):
67 client = socket.socket()
68 client.setblocking(False)
69 try:
70 client.connect(('127.0.0.1', self.serverSocket.getsockname()[1]))
Andrew Svetlov0832af62012-12-18 23:10:48 +020071 except OSError as e:
Ezio Melottib3aedd42010-11-20 19:04:17 +000072 self.assertEqual(e.args[0], errno.EINPROGRESS)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000073 else:
74 raise AssertionError("Connect should have raised EINPROGRESS")
75 server, addr = self.serverSocket.accept()
76
77 self.connections.extend((client, server))
78 return client, server
79
80 def test_create(self):
81 try:
82 ep = select.epoll(16)
83 except OSError as e:
84 raise AssertionError(str(e))
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000085 self.assertTrue(ep.fileno() > 0, ep.fileno())
86 self.assertTrue(not ep.closed)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000087 ep.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000088 self.assertTrue(ep.closed)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000089 self.assertRaises(ValueError, ep.fileno)
Benjamin Peterson70069fc2011-12-27 16:52:20 -060090 if hasattr(select, "EPOLL_CLOEXEC"):
91 select.epoll(select.EPOLL_CLOEXEC).close()
92 self.assertRaises(OSError, select.epoll, flags=12356)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000093
94 def test_badcreate(self):
95 self.assertRaises(TypeError, select.epoll, 1, 2, 3)
96 self.assertRaises(TypeError, select.epoll, 'foo')
97 self.assertRaises(TypeError, select.epoll, None)
98 self.assertRaises(TypeError, select.epoll, ())
99 self.assertRaises(TypeError, select.epoll, ['foo'])
100 self.assertRaises(TypeError, select.epoll, {})
101
Antoine Pitrou09bb89b2012-12-15 21:14:21 +0100102 def test_context_manager(self):
103 with select.epoll(16) as ep:
104 self.assertGreater(ep.fileno(), 0)
105 self.assertFalse(ep.closed)
106 self.assertTrue(ep.closed)
107 self.assertRaises(ValueError, ep.fileno)
108
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000109 def test_add(self):
110 server, client = self._connected_pair()
111
112 ep = select.epoll(2)
113 try:
114 ep.register(server.fileno(), select.EPOLLIN | select.EPOLLOUT)
115 ep.register(client.fileno(), select.EPOLLIN | select.EPOLLOUT)
116 finally:
117 ep.close()
118
R. David Murray1855c212009-05-31 19:44:27 +0000119 # adding by object w/ fileno works, too.
120 ep = select.epoll(2)
121 try:
122 ep.register(server, select.EPOLLIN | select.EPOLLOUT)
123 ep.register(client, select.EPOLLIN | select.EPOLLOUT)
124 finally:
125 ep.close()
126
127 ep = select.epoll(2)
128 try:
129 # TypeError: argument must be an int, or have a fileno() method.
130 self.assertRaises(TypeError, ep.register, object(),
131 select.EPOLLIN | select.EPOLLOUT)
132 self.assertRaises(TypeError, ep.register, None,
133 select.EPOLLIN | select.EPOLLOUT)
134 # ValueError: file descriptor cannot be a negative integer (-1)
135 self.assertRaises(ValueError, ep.register, -1,
136 select.EPOLLIN | select.EPOLLOUT)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200137 # OSError: [Errno 9] Bad file descriptor
138 self.assertRaises(OSError, ep.register, 10000,
R. David Murray1855c212009-05-31 19:44:27 +0000139 select.EPOLLIN | select.EPOLLOUT)
140 # registering twice also raises an exception
141 ep.register(server, select.EPOLLIN | select.EPOLLOUT)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200142 self.assertRaises(OSError, ep.register, server,
R. David Murray1855c212009-05-31 19:44:27 +0000143 select.EPOLLIN | select.EPOLLOUT)
144 finally:
145 ep.close()
146
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000147 def test_fromfd(self):
148 server, client = self._connected_pair()
149
150 ep = select.epoll(2)
151 ep2 = select.epoll.fromfd(ep.fileno())
152
153 ep2.register(server.fileno(), select.EPOLLIN | select.EPOLLOUT)
154 ep2.register(client.fileno(), select.EPOLLIN | select.EPOLLOUT)
155
156 events = ep.poll(1, 4)
157 events2 = ep2.poll(0.9, 4)
158 self.assertEqual(len(events), 2)
159 self.assertEqual(len(events2), 2)
160
161 ep.close()
162 try:
163 ep2.poll(1, 4)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200164 except OSError as e:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000165 self.assertEqual(e.args[0], errno.EBADF, e)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000166 else:
167 self.fail("epoll on closed fd didn't raise EBADF")
168
169 def test_control_and_wait(self):
170 client, server = self._connected_pair()
171
172 ep = select.epoll(16)
173 ep.register(server.fileno(),
174 select.EPOLLIN | select.EPOLLOUT | select.EPOLLET)
175 ep.register(client.fileno(),
176 select.EPOLLIN | select.EPOLLOUT | select.EPOLLET)
177
178 now = time.time()
179 events = ep.poll(1, 4)
180 then = time.time()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000181 self.assertFalse(then - now > 0.1, then - now)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000182
183 events.sort()
184 expected = [(client.fileno(), select.EPOLLOUT),
185 (server.fileno(), select.EPOLLOUT)]
186 expected.sort()
187
Ezio Melottib3aedd42010-11-20 19:04:17 +0000188 self.assertEqual(events, expected)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000189 self.assertFalse(then - now > 0.01, then - now)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000190
191 now = time.time()
192 events = ep.poll(timeout=2.1, maxevents=4)
193 then = time.time()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000194 self.assertFalse(events)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000195
196 client.send(b"Hello!")
197 server.send(b"world!!!")
198
199 now = time.time()
200 events = ep.poll(1, 4)
201 then = time.time()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000202 self.assertFalse(then - now > 0.01)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000203
204 events.sort()
205 expected = [(client.fileno(), select.EPOLLIN | select.EPOLLOUT),
206 (server.fileno(), select.EPOLLIN | select.EPOLLOUT)]
207 expected.sort()
208
Ezio Melottib3aedd42010-11-20 19:04:17 +0000209 self.assertEqual(events, expected)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000210
211 ep.unregister(client.fileno())
212 ep.modify(server.fileno(), select.EPOLLOUT)
213 now = time.time()
214 events = ep.poll(1, 4)
215 then = time.time()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000216 self.assertFalse(then - now > 0.01)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000217
218 expected = [(server.fileno(), select.EPOLLOUT)]
Ezio Melottib3aedd42010-11-20 19:04:17 +0000219 self.assertEqual(events, expected)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000220
221 def test_errors(self):
222 self.assertRaises(ValueError, select.epoll, -2)
223 self.assertRaises(ValueError, select.epoll().register, -1,
224 select.EPOLLIN)
225
226 def test_unregister_closed(self):
227 server, client = self._connected_pair()
228 fd = server.fileno()
229 ep = select.epoll(16)
230 ep.register(server)
231
232 now = time.time()
233 events = ep.poll(1, 4)
234 then = time.time()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000235 self.assertFalse(then - now > 0.01)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000236
237 server.close()
238 ep.unregister(fd)
239
Victor Stinner13423c32013-08-22 00:19:50 +0200240 def test_close(self):
241 open_file = open(__file__, "rb")
242 self.addCleanup(open_file.close)
243 fd = open_file.fileno()
244 epoll = select.epoll()
245
246 # test fileno() method and closed attribute
247 self.assertIsInstance(epoll.fileno(), int)
248 self.assertFalse(epoll.closed)
249
250 # test close()
251 epoll.close()
252 self.assertTrue(epoll.closed)
253 self.assertRaises(ValueError, epoll.fileno)
254
255 # close() can be called more than once
256 epoll.close()
257
258 # operations must fail with ValueError("I/O operation on closed ...")
259 self.assertRaises(ValueError, epoll.modify, fd, select.EPOLLIN)
260 self.assertRaises(ValueError, epoll.poll, 1.0)
261 self.assertRaises(ValueError, epoll.register, fd, select.EPOLLIN)
262 self.assertRaises(ValueError, epoll.unregister, fd)
263
Victor Stinnerdaf45552013-08-28 00:53:59 +0200264 def test_fd_non_inheritable(self):
265 epoll = select.epoll()
266 self.addCleanup(epoll.close)
267 self.assertEqual(os.get_inheritable(epoll.fileno()), False)
268
Victor Stinner13423c32013-08-22 00:19:50 +0200269
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000270def test_main():
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000271 support.run_unittest(TestEPoll)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000272
273if __name__ == "__main__":
274 test_main()