# Initial tests are copied as is from "test_poll.py"
+import errno
import os
import random
import select
self.assertRaises(ValueError, devpoll.register, fd, select.POLLIN)
self.assertRaises(ValueError, devpoll.unregister, fd)
+ def test_close_error(self):
+ # gh-146205: close() should raise OSError if underlying fd is invalid
+ devpoll = select.devpoll()
+ fd = devpoll.fileno()
+ os.close(fd)
+ with self.assertRaises(OSError) as cm:
+ devpoll.close()
+ self.assertEqual(cm.exception.errno, errno.EBADF)
+
def test_fd_non_inheritable(self):
devpoll = select.devpoll()
self.addCleanup(devpoll.close)
self.assertRaises(ValueError, epoll.register, fd, select.EPOLLIN)
self.assertRaises(ValueError, epoll.unregister, fd)
+ def test_close_error(self):
+ # gh-146205: close() should raise OSError if underlying fd is invalid
+ epoll = select.epoll()
+ fd = epoll.fileno()
+ os.close(fd)
+ with self.assertRaises(OSError) as cm:
+ epoll.close()
+ self.assertEqual(cm.exception.errno, errno.EBADF)
+
def test_fd_non_inheritable(self):
epoll = select.epoll()
self.addCleanup(epoll.close)
# operations must fail with ValueError("I/O operation on closed ...")
self.assertRaises(ValueError, kqueue.control, None, 4)
+ def test_close_error(self):
+ # gh-146205: close() should raise OSError if underlying fd is invalid
+ kqueue = select.kqueue()
+ fd = kqueue.fileno()
+ os.close(fd)
+ with self.assertRaises(OSError) as cm:
+ kqueue.close()
+ self.assertEqual(cm.exception.errno, errno.EBADF)
+
def test_fd_non_inheritable(self):
kqueue = select.kqueue()
self.addCleanup(kqueue.close)
--- /dev/null
+Fixed a bug where :meth:`select.epoll.close`, :meth:`select.kqueue.close`,\r
+and :meth:`select.devpoll.close` silently ignored errors.
select_devpoll_close_impl(devpollObject *self)
/*[clinic end generated code: output=26b355bd6429f21b input=408fde21a377ccfb]*/
{
- errno = devpoll_internal_close(self);
- if (errno < 0) {
+ int err = devpoll_internal_close(self);
+ if (err != 0) {
+ errno = err;
PyErr_SetFromErrno(PyExc_OSError);
return NULL;
}
select_epoll_close_impl(pyEpoll_Object *self)
/*[clinic end generated code: output=ee2144c446a1a435 input=f626a769192e1dbe]*/
{
- errno = pyepoll_internal_close(self);
- if (errno < 0) {
+ int err = pyepoll_internal_close(self);
+ if (err != 0) {
+ errno = err;
PyErr_SetFromErrno(PyExc_OSError);
return NULL;
}
select_kqueue_close_impl(kqueue_queue_Object *self)
/*[clinic end generated code: output=d1c7df0b407a4bc1 input=6d763c858b17b690]*/
{
- errno = kqueue_queue_internal_close(self);
- if (errno < 0) {
+ int err = kqueue_queue_internal_close(self);
+ if (err != 0) {
+ errno = err;
PyErr_SetFromErrno(PyExc_OSError);
return NULL;
}