blob: 53ce1d55ff9c4d0426967b27f1574339307e61aa [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
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()
Andrew Svetlovf7a17b42012-12-25 16:47:37 +020036except OSError as e:
Christian Heimesfe337bf2008-03-23 21:54:12 +000037 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))
Charles-François Natali6e204602014-07-23 19:28:13 +010046 self.serverSocket.listen()
Christian Heimes4fbc72b2008-03-22 00:47:35 +000047 self.connections = [self.serverSocket]
48
Christian Heimes4fbc72b2008-03-22 00:47:35 +000049 def tearDown(self):
50 for skt in self.connections:
51 skt.close()
52
53 def _connected_pair(self):
54 client = socket.socket()
55 client.setblocking(False)
56 try:
57 client.connect(('127.0.0.1', self.serverSocket.getsockname()[1]))
Andrew Svetlov0832af62012-12-18 23:10:48 +020058 except OSError as e:
Ezio Melottib3aedd42010-11-20 19:04:17 +000059 self.assertEqual(e.args[0], errno.EINPROGRESS)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000060 else:
61 raise AssertionError("Connect should have raised EINPROGRESS")
62 server, addr = self.serverSocket.accept()
63
64 self.connections.extend((client, server))
65 return client, server
66
67 def test_create(self):
68 try:
69 ep = select.epoll(16)
70 except OSError as e:
71 raise AssertionError(str(e))
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000072 self.assertTrue(ep.fileno() > 0, ep.fileno())
73 self.assertTrue(not ep.closed)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000074 ep.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000075 self.assertTrue(ep.closed)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000076 self.assertRaises(ValueError, ep.fileno)
Tal Einat0cdf5f42018-06-30 15:43:23 +030077
Benjamin Peterson70069fc2011-12-27 16:52:20 -060078 if hasattr(select, "EPOLL_CLOEXEC"):
Tal Einat0cdf5f42018-06-30 15:43:23 +030079 select.epoll(-1, select.EPOLL_CLOEXEC).close()
Berker Peksage2197d12016-09-26 23:30:41 +030080 select.epoll(flags=select.EPOLL_CLOEXEC).close()
81 select.epoll(flags=0).close()
Christian Heimes4fbc72b2008-03-22 00:47:35 +000082
83 def test_badcreate(self):
84 self.assertRaises(TypeError, select.epoll, 1, 2, 3)
85 self.assertRaises(TypeError, select.epoll, 'foo')
86 self.assertRaises(TypeError, select.epoll, None)
87 self.assertRaises(TypeError, select.epoll, ())
88 self.assertRaises(TypeError, select.epoll, ['foo'])
89 self.assertRaises(TypeError, select.epoll, {})
90
Tal Einat0cdf5f42018-06-30 15:43:23 +030091 self.assertRaises(ValueError, select.epoll, 0)
92 self.assertRaises(ValueError, select.epoll, -2)
93 self.assertRaises(ValueError, select.epoll, sizehint=-2)
94
95 if hasattr(select, "EPOLL_CLOEXEC"):
96 self.assertRaises(OSError, select.epoll, flags=12356)
97
Antoine Pitrou09bb89b2012-12-15 21:14:21 +010098 def test_context_manager(self):
99 with select.epoll(16) as ep:
100 self.assertGreater(ep.fileno(), 0)
101 self.assertFalse(ep.closed)
102 self.assertTrue(ep.closed)
103 self.assertRaises(ValueError, ep.fileno)
104
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000105 def test_add(self):
106 server, client = self._connected_pair()
107
108 ep = select.epoll(2)
109 try:
110 ep.register(server.fileno(), select.EPOLLIN | select.EPOLLOUT)
111 ep.register(client.fileno(), select.EPOLLIN | select.EPOLLOUT)
112 finally:
113 ep.close()
114
R. David Murray1855c212009-05-31 19:44:27 +0000115 # adding by object w/ fileno works, too.
116 ep = select.epoll(2)
117 try:
118 ep.register(server, select.EPOLLIN | select.EPOLLOUT)
119 ep.register(client, select.EPOLLIN | select.EPOLLOUT)
120 finally:
121 ep.close()
122
123 ep = select.epoll(2)
124 try:
125 # TypeError: argument must be an int, or have a fileno() method.
126 self.assertRaises(TypeError, ep.register, object(),
Tal Einat0cdf5f42018-06-30 15:43:23 +0300127 select.EPOLLIN | select.EPOLLOUT)
R. David Murray1855c212009-05-31 19:44:27 +0000128 self.assertRaises(TypeError, ep.register, None,
Tal Einat0cdf5f42018-06-30 15:43:23 +0300129 select.EPOLLIN | select.EPOLLOUT)
R. David Murray1855c212009-05-31 19:44:27 +0000130 # ValueError: file descriptor cannot be a negative integer (-1)
131 self.assertRaises(ValueError, ep.register, -1,
Tal Einat0cdf5f42018-06-30 15:43:23 +0300132 select.EPOLLIN | select.EPOLLOUT)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200133 # OSError: [Errno 9] Bad file descriptor
134 self.assertRaises(OSError, ep.register, 10000,
Tal Einat0cdf5f42018-06-30 15:43:23 +0300135 select.EPOLLIN | select.EPOLLOUT)
R. David Murray1855c212009-05-31 19:44:27 +0000136 # registering twice also raises an exception
137 ep.register(server, select.EPOLLIN | select.EPOLLOUT)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200138 self.assertRaises(OSError, ep.register, server,
Tal Einat0cdf5f42018-06-30 15:43:23 +0300139 select.EPOLLIN | select.EPOLLOUT)
R. David Murray1855c212009-05-31 19:44:27 +0000140 finally:
141 ep.close()
142
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000143 def test_fromfd(self):
144 server, client = self._connected_pair()
145
Serhiy Storchaka5b10b982019-03-05 10:06:26 +0200146 with select.epoll(2) as ep:
147 ep2 = select.epoll.fromfd(ep.fileno())
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000148
Serhiy Storchaka5b10b982019-03-05 10:06:26 +0200149 ep2.register(server.fileno(), select.EPOLLIN | select.EPOLLOUT)
150 ep2.register(client.fileno(), select.EPOLLIN | select.EPOLLOUT)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000151
Serhiy Storchaka5b10b982019-03-05 10:06:26 +0200152 events = ep.poll(1, 4)
153 events2 = ep2.poll(0.9, 4)
154 self.assertEqual(len(events), 2)
155 self.assertEqual(len(events2), 2)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000156
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000157 try:
158 ep2.poll(1, 4)
Andrew Svetlovf7a17b42012-12-25 16:47:37 +0200159 except OSError as e:
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000160 self.assertEqual(e.args[0], errno.EBADF, e)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000161 else:
162 self.fail("epoll on closed fd didn't raise EBADF")
163
164 def test_control_and_wait(self):
165 client, server = self._connected_pair()
166
167 ep = select.epoll(16)
168 ep.register(server.fileno(),
Tal Einat0cdf5f42018-06-30 15:43:23 +0300169 select.EPOLLIN | select.EPOLLOUT | select.EPOLLET)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000170 ep.register(client.fileno(),
Tal Einat0cdf5f42018-06-30 15:43:23 +0300171 select.EPOLLIN | select.EPOLLOUT | select.EPOLLET)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000172
Victor Stinnerb60ac7a2014-03-25 12:50:50 +0100173 now = time.monotonic()
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000174 events = ep.poll(1, 4)
Victor Stinnerb60ac7a2014-03-25 12:50:50 +0100175 then = time.monotonic()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +0000176 self.assertFalse(then - now > 0.1, then - now)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000177
178 events.sort()
179 expected = [(client.fileno(), select.EPOLLOUT),
180 (server.fileno(), select.EPOLLOUT)]
181 expected.sort()
182
Ezio Melottib3aedd42010-11-20 19:04:17 +0000183 self.assertEqual(events, expected)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000184
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000185 events = ep.poll(timeout=2.1, maxevents=4)
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
Victor Stinnerb60ac7a2014-03-25 12:50:50 +0100191 now = time.monotonic()
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000192 events = ep.poll(1, 4)
Victor Stinnerb60ac7a2014-03-25 12:50:50 +0100193 then = time.monotonic()
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)
Victor Stinnerb60ac7a2014-03-25 12:50:50 +0100205 now = time.monotonic()
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000206 events = ep.poll(1, 4)
Victor Stinnerb60ac7a2014-03-25 12:50:50 +0100207 then = time.monotonic()
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
Victor Stinnerb60ac7a2014-03-25 12:50:50 +0100224 now = time.monotonic()
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000225 events = ep.poll(1, 4)
Victor Stinnerb60ac7a2014-03-25 12:50:50 +0100226 then = time.monotonic()
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
Victor Stinner13423c32013-08-22 00:19:50 +0200232 def test_close(self):
233 open_file = open(__file__, "rb")
234 self.addCleanup(open_file.close)
235 fd = open_file.fileno()
236 epoll = select.epoll()
237
238 # test fileno() method and closed attribute
239 self.assertIsInstance(epoll.fileno(), int)
240 self.assertFalse(epoll.closed)
241
242 # test close()
243 epoll.close()
244 self.assertTrue(epoll.closed)
245 self.assertRaises(ValueError, epoll.fileno)
246
247 # close() can be called more than once
248 epoll.close()
249
250 # operations must fail with ValueError("I/O operation on closed ...")
251 self.assertRaises(ValueError, epoll.modify, fd, select.EPOLLIN)
252 self.assertRaises(ValueError, epoll.poll, 1.0)
253 self.assertRaises(ValueError, epoll.register, fd, select.EPOLLIN)
254 self.assertRaises(ValueError, epoll.unregister, fd)
255
Victor Stinnerdaf45552013-08-28 00:53:59 +0200256 def test_fd_non_inheritable(self):
257 epoll = select.epoll()
258 self.addCleanup(epoll.close)
259 self.assertEqual(os.get_inheritable(epoll.fileno()), False)
260
Victor Stinner13423c32013-08-22 00:19:50 +0200261
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000262if __name__ == "__main__":
Zachary Ware38c707e2015-04-13 15:00:43 -0500263 unittest.main()