blob: ad545d3a3b15c0945c0c7dae575bc060aeade7b7 [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 socket
25import errno
26import time
27import select
Christian Heimes4fbc72b2008-03-22 00:47:35 +000028import unittest
29
Benjamin Petersonee8712c2008-05-20 21:35:26 +000030from test import support
Christian Heimes4fbc72b2008-03-22 00:47:35 +000031if not hasattr(select, "epoll"):
Benjamin Petersone549ead2009-03-28 21:42:05 +000032 raise unittest.SkipTest("test works only on Linux 2.6")
Christian Heimes4fbc72b2008-03-22 00:47:35 +000033
Christian Heimesfe337bf2008-03-23 21:54:12 +000034try:
35 select.epoll()
36except IOError as e:
37 if e.errno == errno.ENOSYS:
Benjamin Petersone549ead2009-03-28 21:42:05 +000038 raise unittest.SkipTest("kernel doesn't support epoll()")
Benjamin Peterson0e613502011-12-27 15:16:34 -060039 raise
Christian Heimesfe337bf2008-03-23 21:54:12 +000040
Christian Heimes4fbc72b2008-03-22 00:47:35 +000041class TestEPoll(unittest.TestCase):
42
43 def setUp(self):
44 self.serverSocket = socket.socket()
45 self.serverSocket.bind(('127.0.0.1', 0))
46 self.serverSocket.listen(1)
47 self.connections = [self.serverSocket]
48
Victor Stinner665486e2014-01-21 01:41:00 +010049 def test_timeout_rounding(self):
50 # epoll_wait() has a resolution of 1 millisecond, check if the timeout
51 # is correctly rounded to the upper bound
52 epoll = select.epoll()
53 self.addCleanup(epoll.close)
54 for timeout in (1e-2, 1e-3, 1e-4):
55 t0 = time.perf_counter()
56 epoll.poll(timeout)
57 dt = time.perf_counter() - t0
58 self.assertGreaterEqual(dt, timeout)
59
Christian Heimes4fbc72b2008-03-22 00:47:35 +000060
61 def tearDown(self):
62 for skt in self.connections:
63 skt.close()
64
65 def _connected_pair(self):
66 client = socket.socket()
67 client.setblocking(False)
68 try:
69 client.connect(('127.0.0.1', self.serverSocket.getsockname()[1]))
70 except socket.error as e:
Ezio Melottib3aedd42010-11-20 19:04:17 +000071 self.assertEqual(e.args[0], errno.EINPROGRESS)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000072 else:
73 raise AssertionError("Connect should have raised EINPROGRESS")
74 server, addr = self.serverSocket.accept()
75
76 self.connections.extend((client, server))
77 return client, server
78
79 def test_create(self):
80 try:
81 ep = select.epoll(16)
82 except OSError as e:
83 raise AssertionError(str(e))
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000084 self.assertTrue(ep.fileno() > 0, ep.fileno())
85 self.assertTrue(not ep.closed)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000086 ep.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000087 self.assertTrue(ep.closed)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000088 self.assertRaises(ValueError, ep.fileno)
Benjamin Peterson70069fc2011-12-27 16:52:20 -060089 if hasattr(select, "EPOLL_CLOEXEC"):
90 select.epoll(select.EPOLL_CLOEXEC).close()
91 self.assertRaises(OSError, select.epoll, flags=12356)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000092
93 def test_badcreate(self):
94 self.assertRaises(TypeError, select.epoll, 1, 2, 3)
95 self.assertRaises(TypeError, select.epoll, 'foo')
96 self.assertRaises(TypeError, select.epoll, None)
97 self.assertRaises(TypeError, select.epoll, ())
98 self.assertRaises(TypeError, select.epoll, ['foo'])
99 self.assertRaises(TypeError, select.epoll, {})
100
101 def test_add(self):
102 server, client = self._connected_pair()
103
104 ep = select.epoll(2)
105 try:
106 ep.register(server.fileno(), select.EPOLLIN | select.EPOLLOUT)
107 ep.register(client.fileno(), select.EPOLLIN | select.EPOLLOUT)
108 finally:
109 ep.close()
110
R. David Murray1855c212009-05-31 19:44:27 +0000111 # adding by object w/ fileno works, too.
112 ep = select.epoll(2)
113 try:
114 ep.register(server, select.EPOLLIN | select.EPOLLOUT)
115 ep.register(client, select.EPOLLIN | select.EPOLLOUT)
116 finally:
117 ep.close()
118
119 ep = select.epoll(2)
120 try:
121 # TypeError: argument must be an int, or have a fileno() method.
122 self.assertRaises(TypeError, ep.register, object(),
123 select.EPOLLIN | select.EPOLLOUT)
124 self.assertRaises(TypeError, ep.register, None,
125 select.EPOLLIN | select.EPOLLOUT)
126 # ValueError: file descriptor cannot be a negative integer (-1)
127 self.assertRaises(ValueError, ep.register, -1,
128 select.EPOLLIN | select.EPOLLOUT)
129 # IOError: [Errno 9] Bad file descriptor
130 self.assertRaises(IOError, ep.register, 10000,
131 select.EPOLLIN | select.EPOLLOUT)
132 # registering twice also raises an exception
133 ep.register(server, select.EPOLLIN | select.EPOLLOUT)
134 self.assertRaises(IOError, ep.register, server,
135 select.EPOLLIN | select.EPOLLOUT)
136 finally:
137 ep.close()
138
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000139 def test_fromfd(self):
140 server, client = self._connected_pair()
141
142 ep = select.epoll(2)
143 ep2 = select.epoll.fromfd(ep.fileno())
144
145 ep2.register(server.fileno(), select.EPOLLIN | select.EPOLLOUT)
146 ep2.register(client.fileno(), select.EPOLLIN | select.EPOLLOUT)
147
148 events = ep.poll(1, 4)
149 events2 = ep2.poll(0.9, 4)
150 self.assertEqual(len(events), 2)
151 self.assertEqual(len(events2), 2)
152
153 ep.close()
154 try:
155 ep2.poll(1, 4)
156 except IOError as e:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000157 self.assertEqual(e.args[0], errno.EBADF, e)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000158 else:
159 self.fail("epoll on closed fd didn't raise EBADF")
160
161 def test_control_and_wait(self):
162 client, server = self._connected_pair()
163
164 ep = select.epoll(16)
165 ep.register(server.fileno(),
166 select.EPOLLIN | select.EPOLLOUT | select.EPOLLET)
167 ep.register(client.fileno(),
168 select.EPOLLIN | select.EPOLLOUT | select.EPOLLET)
169
170 now = time.time()
171 events = ep.poll(1, 4)
172 then = time.time()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000173 self.assertFalse(then - now > 0.1, then - now)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000174
175 events.sort()
176 expected = [(client.fileno(), select.EPOLLOUT),
177 (server.fileno(), select.EPOLLOUT)]
178 expected.sort()
179
Ezio Melottib3aedd42010-11-20 19:04:17 +0000180 self.assertEqual(events, expected)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000181 self.assertFalse(then - now > 0.01, then - now)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000182
183 now = time.time()
184 events = ep.poll(timeout=2.1, maxevents=4)
185 then = time.time()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000186 self.assertFalse(events)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000187
188 client.send(b"Hello!")
189 server.send(b"world!!!")
190
191 now = time.time()
192 events = ep.poll(1, 4)
193 then = time.time()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000194 self.assertFalse(then - now > 0.01)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000195
196 events.sort()
197 expected = [(client.fileno(), select.EPOLLIN | select.EPOLLOUT),
198 (server.fileno(), select.EPOLLIN | select.EPOLLOUT)]
199 expected.sort()
200
Ezio Melottib3aedd42010-11-20 19:04:17 +0000201 self.assertEqual(events, expected)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000202
203 ep.unregister(client.fileno())
204 ep.modify(server.fileno(), select.EPOLLOUT)
205 now = time.time()
206 events = ep.poll(1, 4)
207 then = time.time()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000208 self.assertFalse(then - now > 0.01)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000209
210 expected = [(server.fileno(), select.EPOLLOUT)]
Ezio Melottib3aedd42010-11-20 19:04:17 +0000211 self.assertEqual(events, expected)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000212
213 def test_errors(self):
214 self.assertRaises(ValueError, select.epoll, -2)
215 self.assertRaises(ValueError, select.epoll().register, -1,
216 select.EPOLLIN)
217
218 def test_unregister_closed(self):
219 server, client = self._connected_pair()
220 fd = server.fileno()
221 ep = select.epoll(16)
222 ep.register(server)
223
224 now = time.time()
225 events = ep.poll(1, 4)
226 then = time.time()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000227 self.assertFalse(then - now > 0.01)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000228
229 server.close()
230 ep.unregister(fd)
231
232def test_main():
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000233 support.run_unittest(TestEPoll)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000234
235if __name__ == "__main__":
236 test_main()