blob: f79bd8978c1b87885f965b2bbf23461921dc23bb [file] [log] [blame]
Christian Heimes4fbc72b2008-03-22 00:47:35 +00001"""
2Tests for kqueue wrapper.
3"""
Christian Heimes4fbc72b2008-03-22 00:47:35 +00004import errno
Victor Stinner125b2ba2011-06-29 17:20:02 +02005import os
Christian Heimes4fbc72b2008-03-22 00:47:35 +00006import select
Victor Stinner125b2ba2011-06-29 17:20:02 +02007import socket
Christian Heimes4fbc72b2008-03-22 00:47:35 +00008import sys
Victor Stinner125b2ba2011-06-29 17:20:02 +02009import time
Christian Heimes4fbc72b2008-03-22 00:47:35 +000010import unittest
11
Benjamin Petersonee8712c2008-05-20 21:35:26 +000012from test import support
Christian Heimes4fbc72b2008-03-22 00:47:35 +000013if not hasattr(select, "kqueue"):
Benjamin Petersone549ead2009-03-28 21:42:05 +000014 raise unittest.SkipTest("test works only on BSD")
Christian Heimes4fbc72b2008-03-22 00:47:35 +000015
16class TestKQueue(unittest.TestCase):
17 def test_create_queue(self):
18 kq = select.kqueue()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000019 self.assertTrue(kq.fileno() > 0, kq.fileno())
20 self.assertTrue(not kq.closed)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000021 kq.close()
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000022 self.assertTrue(kq.closed)
Christian Heimes4fbc72b2008-03-22 00:47:35 +000023 self.assertRaises(ValueError, kq.fileno)
24
25 def test_create_event(self):
Mark Dickinsona56c4672009-01-27 18:17:45 +000026 from operator import lt, le, gt, ge
Victor Stinner46b20912011-06-29 14:59:10 +020027
28 fd = os.open(os.devnull, os.O_WRONLY)
29 self.addCleanup(os.close, fd)
30
Christian Heimes4fbc72b2008-03-22 00:47:35 +000031 ev = select.kevent(fd)
32 other = select.kevent(1000)
33 self.assertEqual(ev.ident, fd)
34 self.assertEqual(ev.filter, select.KQ_FILTER_READ)
35 self.assertEqual(ev.flags, select.KQ_EV_ADD)
36 self.assertEqual(ev.fflags, 0)
37 self.assertEqual(ev.data, 0)
38 self.assertEqual(ev.udata, 0)
39 self.assertEqual(ev, ev)
40 self.assertNotEqual(ev, other)
Benjamin Petersonc9c0f202009-06-30 23:06:06 +000041 self.assertTrue(ev < other)
42 self.assertTrue(other >= ev)
Mark Dickinsona56c4672009-01-27 18:17:45 +000043 for op in lt, le, gt, ge:
44 self.assertRaises(TypeError, op, ev, None)
45 self.assertRaises(TypeError, op, ev, 1)
46 self.assertRaises(TypeError, op, ev, "ev")
Christian Heimes4fbc72b2008-03-22 00:47:35 +000047
48 ev = select.kevent(fd, select.KQ_FILTER_WRITE)
49 self.assertEqual(ev.ident, fd)
50 self.assertEqual(ev.filter, select.KQ_FILTER_WRITE)
51 self.assertEqual(ev.flags, select.KQ_EV_ADD)
52 self.assertEqual(ev.fflags, 0)
53 self.assertEqual(ev.data, 0)
54 self.assertEqual(ev.udata, 0)
55 self.assertEqual(ev, ev)
56 self.assertNotEqual(ev, other)
57
58 ev = select.kevent(fd, select.KQ_FILTER_WRITE, select.KQ_EV_ONESHOT)
59 self.assertEqual(ev.ident, fd)
60 self.assertEqual(ev.filter, select.KQ_FILTER_WRITE)
61 self.assertEqual(ev.flags, select.KQ_EV_ONESHOT)
62 self.assertEqual(ev.fflags, 0)
63 self.assertEqual(ev.data, 0)
64 self.assertEqual(ev.udata, 0)
65 self.assertEqual(ev, ev)
66 self.assertNotEqual(ev, other)
67
68 ev = select.kevent(1, 2, 3, 4, 5, 6)
69 self.assertEqual(ev.ident, 1)
70 self.assertEqual(ev.filter, 2)
71 self.assertEqual(ev.flags, 3)
72 self.assertEqual(ev.fflags, 4)
73 self.assertEqual(ev.data, 5)
74 self.assertEqual(ev.udata, 6)
75 self.assertEqual(ev, ev)
76 self.assertNotEqual(ev, other)
77
Charles-François Natali6ddd2652013-08-02 10:21:20 +020078 bignum = 0x7fff
79 ev = select.kevent(bignum, 1, 2, 3, bignum - 1, bignum)
Antoine Pitroud83f1e62009-11-04 21:10:38 +000080 self.assertEqual(ev.ident, bignum)
81 self.assertEqual(ev.filter, 1)
82 self.assertEqual(ev.flags, 2)
83 self.assertEqual(ev.fflags, 3)
Charles-François Natali6ddd2652013-08-02 10:21:20 +020084 self.assertEqual(ev.data, bignum - 1)
Antoine Pitroud83f1e62009-11-04 21:10:38 +000085 self.assertEqual(ev.udata, bignum)
86 self.assertEqual(ev, ev)
87 self.assertNotEqual(ev, other)
88
R David Murray90f7afd2014-10-12 12:39:46 -040089 # Issue 11973
90 bignum = 0xffff
91 ev = select.kevent(0, 1, bignum)
92 self.assertEqual(ev.ident, 0)
93 self.assertEqual(ev.filter, 1)
94 self.assertEqual(ev.flags, bignum)
95 self.assertEqual(ev.fflags, 0)
96 self.assertEqual(ev.data, 0)
97 self.assertEqual(ev.udata, 0)
98 self.assertEqual(ev, ev)
99 self.assertNotEqual(ev, other)
100
101 # Issue 11973
102 bignum = 0xffffffff
103 ev = select.kevent(0, 1, 2, bignum)
104 self.assertEqual(ev.ident, 0)
105 self.assertEqual(ev.filter, 1)
106 self.assertEqual(ev.flags, 2)
107 self.assertEqual(ev.fflags, bignum)
108 self.assertEqual(ev.data, 0)
109 self.assertEqual(ev.udata, 0)
110 self.assertEqual(ev, ev)
111 self.assertNotEqual(ev, other)
112
113
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000114 def test_queue_event(self):
115 serverSocket = socket.socket()
116 serverSocket.bind(('127.0.0.1', 0))
117 serverSocket.listen(1)
118 client = socket.socket()
119 client.setblocking(False)
120 try:
121 client.connect(('127.0.0.1', serverSocket.getsockname()[1]))
Andrew Svetlov0832af62012-12-18 23:10:48 +0200122 except OSError as e:
Ezio Melottib3aedd42010-11-20 19:04:17 +0000123 self.assertEqual(e.args[0], errno.EINPROGRESS)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000124 else:
125 #raise AssertionError("Connect should have raised EINPROGRESS")
126 pass # FreeBSD doesn't raise an exception here
127 server, addr = serverSocket.accept()
128
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000129 kq = select.kqueue()
130 kq2 = select.kqueue.fromfd(kq.fileno())
131
132 ev = select.kevent(server.fileno(),
133 select.KQ_FILTER_WRITE,
134 select.KQ_EV_ADD | select.KQ_EV_ENABLE)
135 kq.control([ev], 0)
136 ev = select.kevent(server.fileno(),
137 select.KQ_FILTER_READ,
138 select.KQ_EV_ADD | select.KQ_EV_ENABLE)
139 kq.control([ev], 0)
140 ev = select.kevent(client.fileno(),
141 select.KQ_FILTER_WRITE,
142 select.KQ_EV_ADD | select.KQ_EV_ENABLE)
143 kq2.control([ev], 0)
144 ev = select.kevent(client.fileno(),
145 select.KQ_FILTER_READ,
146 select.KQ_EV_ADD | select.KQ_EV_ENABLE)
147 kq2.control([ev], 0)
148
149 events = kq.control(None, 4, 1)
Charles-Francois Natali7a03e2d2013-03-31 20:36:57 +0200150 events = set((e.ident, e.filter) for e in events)
151 self.assertEqual(events, set([
152 (client.fileno(), select.KQ_FILTER_WRITE),
153 (server.fileno(), select.KQ_FILTER_WRITE)]))
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000154
155 client.send(b"Hello!")
156 server.send(b"world!!!")
157
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000158 # We may need to call it several times
Mark Dickinson3dfe55b2009-01-24 16:22:21 +0000159 for i in range(10):
160 events = kq.control(None, 4, 1)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000161 if len(events) == 4:
162 break
Mark Dickinson3dfe55b2009-01-24 16:22:21 +0000163 time.sleep(1.0)
164 else:
165 self.fail('timeout waiting for event notifications')
166
Charles-Francois Natali7a03e2d2013-03-31 20:36:57 +0200167 events = set((e.ident, e.filter) for e in events)
168 self.assertEqual(events, set([
169 (client.fileno(), select.KQ_FILTER_WRITE),
170 (client.fileno(), select.KQ_FILTER_READ),
171 (server.fileno(), select.KQ_FILTER_WRITE),
172 (server.fileno(), select.KQ_FILTER_READ)]))
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000173
174 # Remove completely client, and server read part
175 ev = select.kevent(client.fileno(),
176 select.KQ_FILTER_WRITE,
177 select.KQ_EV_DELETE)
178 kq.control([ev], 0)
179 ev = select.kevent(client.fileno(),
180 select.KQ_FILTER_READ,
181 select.KQ_EV_DELETE)
182 kq.control([ev], 0)
183 ev = select.kevent(server.fileno(),
184 select.KQ_FILTER_READ,
185 select.KQ_EV_DELETE)
186 kq.control([ev], 0, 0)
187
188 events = kq.control([], 4, 0.99)
Charles-Francois Natali7a03e2d2013-03-31 20:36:57 +0200189 events = set((e.ident, e.filter) for e in events)
190 self.assertEqual(events, set([
191 (server.fileno(), select.KQ_FILTER_WRITE)]))
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000192
193 client.close()
194 server.close()
195 serverSocket.close()
196
Alexandre Vassalottie52e3782009-07-17 09:18:18 +0000197 def testPair(self):
198 kq = select.kqueue()
199 a, b = socket.socketpair()
200
201 a.send(b'foo')
202 event1 = select.kevent(a, select.KQ_FILTER_READ, select.KQ_EV_ADD | select.KQ_EV_ENABLE)
203 event2 = select.kevent(b, select.KQ_FILTER_READ, select.KQ_EV_ADD | select.KQ_EV_ENABLE)
204 r = kq.control([event1, event2], 1, 1)
205 self.assertTrue(r)
206 self.assertFalse(r[0].flags & select.KQ_EV_ERROR)
Ezio Melottib3aedd42010-11-20 19:04:17 +0000207 self.assertEqual(b.recv(r[0].data), b'foo')
Alexandre Vassalottie52e3782009-07-17 09:18:18 +0000208
209 a.close()
210 b.close()
211 kq.close()
212
Victor Stinner13423c32013-08-22 00:19:50 +0200213 def test_close(self):
214 open_file = open(__file__, "rb")
215 self.addCleanup(open_file.close)
216 fd = open_file.fileno()
217 kqueue = select.kqueue()
218
219 # test fileno() method and closed attribute
220 self.assertIsInstance(kqueue.fileno(), int)
221 self.assertFalse(kqueue.closed)
222
223 # test close()
224 kqueue.close()
225 self.assertTrue(kqueue.closed)
226 self.assertRaises(ValueError, kqueue.fileno)
227
228 # close() can be called more than once
229 kqueue.close()
230
231 # operations must fail with ValueError("I/O operation on closed ...")
232 self.assertRaises(ValueError, kqueue.control, None, 4)
233
Victor Stinnerdaf45552013-08-28 00:53:59 +0200234 def test_fd_non_inheritable(self):
235 kqueue = select.kqueue()
236 self.addCleanup(kqueue.close)
237 self.assertEqual(os.get_inheritable(kqueue.fileno()), False)
238
Victor Stinner13423c32013-08-22 00:19:50 +0200239
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000240def test_main():
Benjamin Petersonee8712c2008-05-20 21:35:26 +0000241 support.run_unittest(TestKQueue)
Christian Heimes4fbc72b2008-03-22 00:47:35 +0000242
243if __name__ == "__main__":
244 test_main()