selectors34.py 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637
  1. # pylint: skip-file
  2. # vendored from https://github.com/berkerpeksag/selectors34
  3. # at commit ff61b82168d2cc9c4922ae08e2a8bf94aab61ea2 (unreleased, ~1.2)
  4. #
  5. # Original author: Charles-Francois Natali (c.f.natali[at]gmail.com)
  6. # Maintainer: Berker Peksag (berker.peksag[at]gmail.com)
  7. # Also see https://pypi.python.org/pypi/selectors34
  8. """Selectors module.
  9. This module allows high-level and efficient I/O multiplexing, built upon the
  10. `select` module primitives.
  11. The following code adapted from trollius.selectors.
  12. """
  13. from __future__ import absolute_import
  14. from abc import ABCMeta, abstractmethod
  15. from collections import namedtuple, Mapping
  16. from errno import EINTR
  17. import math
  18. import select
  19. import sys
  20. from kafka.vendor import six
  21. def _wrap_error(exc, mapping, key):
  22. if key not in mapping:
  23. return
  24. new_err_cls = mapping[key]
  25. new_err = new_err_cls(*exc.args)
  26. # raise a new exception with the original traceback
  27. if hasattr(exc, '__traceback__'):
  28. traceback = exc.__traceback__
  29. else:
  30. traceback = sys.exc_info()[2]
  31. six.reraise(new_err_cls, new_err, traceback)
  32. # generic events, that must be mapped to implementation-specific ones
  33. EVENT_READ = (1 << 0)
  34. EVENT_WRITE = (1 << 1)
  35. def _fileobj_to_fd(fileobj):
  36. """Return a file descriptor from a file object.
  37. Parameters:
  38. fileobj -- file object or file descriptor
  39. Returns:
  40. corresponding file descriptor
  41. Raises:
  42. ValueError if the object is invalid
  43. """
  44. if isinstance(fileobj, six.integer_types):
  45. fd = fileobj
  46. else:
  47. try:
  48. fd = int(fileobj.fileno())
  49. except (AttributeError, TypeError, ValueError):
  50. raise ValueError("Invalid file object: "
  51. "{0!r}".format(fileobj))
  52. if fd < 0:
  53. raise ValueError("Invalid file descriptor: {0}".format(fd))
  54. return fd
  55. SelectorKey = namedtuple('SelectorKey', ['fileobj', 'fd', 'events', 'data'])
  56. """Object used to associate a file object to its backing file descriptor,
  57. selected event mask and attached data."""
  58. class _SelectorMapping(Mapping):
  59. """Mapping of file objects to selector keys."""
  60. def __init__(self, selector):
  61. self._selector = selector
  62. def __len__(self):
  63. return len(self._selector._fd_to_key)
  64. def __getitem__(self, fileobj):
  65. try:
  66. fd = self._selector._fileobj_lookup(fileobj)
  67. return self._selector._fd_to_key[fd]
  68. except KeyError:
  69. raise KeyError("{0!r} is not registered".format(fileobj))
  70. def __iter__(self):
  71. return iter(self._selector._fd_to_key)
  72. # Using six.add_metaclass() decorator instead of six.with_metaclass() because
  73. # the latter leaks temporary_class to garbage with gc disabled
  74. @six.add_metaclass(ABCMeta)
  75. class BaseSelector(object):
  76. """Selector abstract base class.
  77. A selector supports registering file objects to be monitored for specific
  78. I/O events.
  79. A file object is a file descriptor or any object with a `fileno()` method.
  80. An arbitrary object can be attached to the file object, which can be used
  81. for example to store context information, a callback, etc.
  82. A selector can use various implementations (select(), poll(), epoll()...)
  83. depending on the platform. The default `Selector` class uses the most
  84. efficient implementation on the current platform.
  85. """
  86. @abstractmethod
  87. def register(self, fileobj, events, data=None):
  88. """Register a file object.
  89. Parameters:
  90. fileobj -- file object or file descriptor
  91. events -- events to monitor (bitwise mask of EVENT_READ|EVENT_WRITE)
  92. data -- attached data
  93. Returns:
  94. SelectorKey instance
  95. Raises:
  96. ValueError if events is invalid
  97. KeyError if fileobj is already registered
  98. OSError if fileobj is closed or otherwise is unacceptable to
  99. the underlying system call (if a system call is made)
  100. Note:
  101. OSError may or may not be raised
  102. """
  103. raise NotImplementedError
  104. @abstractmethod
  105. def unregister(self, fileobj):
  106. """Unregister a file object.
  107. Parameters:
  108. fileobj -- file object or file descriptor
  109. Returns:
  110. SelectorKey instance
  111. Raises:
  112. KeyError if fileobj is not registered
  113. Note:
  114. If fileobj is registered but has since been closed this does
  115. *not* raise OSError (even if the wrapped syscall does)
  116. """
  117. raise NotImplementedError
  118. def modify(self, fileobj, events, data=None):
  119. """Change a registered file object monitored events or attached data.
  120. Parameters:
  121. fileobj -- file object or file descriptor
  122. events -- events to monitor (bitwise mask of EVENT_READ|EVENT_WRITE)
  123. data -- attached data
  124. Returns:
  125. SelectorKey instance
  126. Raises:
  127. Anything that unregister() or register() raises
  128. """
  129. self.unregister(fileobj)
  130. return self.register(fileobj, events, data)
  131. @abstractmethod
  132. def select(self, timeout=None):
  133. """Perform the actual selection, until some monitored file objects are
  134. ready or a timeout expires.
  135. Parameters:
  136. timeout -- if timeout > 0, this specifies the maximum wait time, in
  137. seconds
  138. if timeout <= 0, the select() call won't block, and will
  139. report the currently ready file objects
  140. if timeout is None, select() will block until a monitored
  141. file object becomes ready
  142. Returns:
  143. list of (key, events) for ready file objects
  144. `events` is a bitwise mask of EVENT_READ|EVENT_WRITE
  145. """
  146. raise NotImplementedError
  147. def close(self):
  148. """Close the selector.
  149. This must be called to make sure that any underlying resource is freed.
  150. """
  151. pass
  152. def get_key(self, fileobj):
  153. """Return the key associated to a registered file object.
  154. Returns:
  155. SelectorKey for this file object
  156. """
  157. mapping = self.get_map()
  158. if mapping is None:
  159. raise RuntimeError('Selector is closed')
  160. try:
  161. return mapping[fileobj]
  162. except KeyError:
  163. raise KeyError("{0!r} is not registered".format(fileobj))
  164. @abstractmethod
  165. def get_map(self):
  166. """Return a mapping of file objects to selector keys."""
  167. raise NotImplementedError
  168. def __enter__(self):
  169. return self
  170. def __exit__(self, *args):
  171. self.close()
  172. class _BaseSelectorImpl(BaseSelector):
  173. """Base selector implementation."""
  174. def __init__(self):
  175. # this maps file descriptors to keys
  176. self._fd_to_key = {}
  177. # read-only mapping returned by get_map()
  178. self._map = _SelectorMapping(self)
  179. def _fileobj_lookup(self, fileobj):
  180. """Return a file descriptor from a file object.
  181. This wraps _fileobj_to_fd() to do an exhaustive search in case
  182. the object is invalid but we still have it in our map. This
  183. is used by unregister() so we can unregister an object that
  184. was previously registered even if it is closed. It is also
  185. used by _SelectorMapping.
  186. """
  187. try:
  188. return _fileobj_to_fd(fileobj)
  189. except ValueError:
  190. # Do an exhaustive search.
  191. for key in self._fd_to_key.values():
  192. if key.fileobj is fileobj:
  193. return key.fd
  194. # Raise ValueError after all.
  195. raise
  196. def register(self, fileobj, events, data=None):
  197. if (not events) or (events & ~(EVENT_READ | EVENT_WRITE)):
  198. raise ValueError("Invalid events: {0!r}".format(events))
  199. key = SelectorKey(fileobj, self._fileobj_lookup(fileobj), events, data)
  200. if key.fd in self._fd_to_key:
  201. raise KeyError("{0!r} (FD {1}) is already registered"
  202. .format(fileobj, key.fd))
  203. self._fd_to_key[key.fd] = key
  204. return key
  205. def unregister(self, fileobj):
  206. try:
  207. key = self._fd_to_key.pop(self._fileobj_lookup(fileobj))
  208. except KeyError:
  209. raise KeyError("{0!r} is not registered".format(fileobj))
  210. return key
  211. def modify(self, fileobj, events, data=None):
  212. # TODO: Subclasses can probably optimize this even further.
  213. try:
  214. key = self._fd_to_key[self._fileobj_lookup(fileobj)]
  215. except KeyError:
  216. raise KeyError("{0!r} is not registered".format(fileobj))
  217. if events != key.events:
  218. self.unregister(fileobj)
  219. key = self.register(fileobj, events, data)
  220. elif data != key.data:
  221. # Use a shortcut to update the data.
  222. key = key._replace(data=data)
  223. self._fd_to_key[key.fd] = key
  224. return key
  225. def close(self):
  226. self._fd_to_key.clear()
  227. self._map = None
  228. def get_map(self):
  229. return self._map
  230. def _key_from_fd(self, fd):
  231. """Return the key associated to a given file descriptor.
  232. Parameters:
  233. fd -- file descriptor
  234. Returns:
  235. corresponding key, or None if not found
  236. """
  237. try:
  238. return self._fd_to_key[fd]
  239. except KeyError:
  240. return None
  241. class SelectSelector(_BaseSelectorImpl):
  242. """Select-based selector."""
  243. def __init__(self):
  244. super(SelectSelector, self).__init__()
  245. self._readers = set()
  246. self._writers = set()
  247. def register(self, fileobj, events, data=None):
  248. key = super(SelectSelector, self).register(fileobj, events, data)
  249. if events & EVENT_READ:
  250. self._readers.add(key.fd)
  251. if events & EVENT_WRITE:
  252. self._writers.add(key.fd)
  253. return key
  254. def unregister(self, fileobj):
  255. key = super(SelectSelector, self).unregister(fileobj)
  256. self._readers.discard(key.fd)
  257. self._writers.discard(key.fd)
  258. return key
  259. if sys.platform == 'win32':
  260. def _select(self, r, w, _, timeout=None):
  261. r, w, x = select.select(r, w, w, timeout)
  262. return r, w + x, []
  263. else:
  264. _select = staticmethod(select.select)
  265. def select(self, timeout=None):
  266. timeout = None if timeout is None else max(timeout, 0)
  267. ready = []
  268. try:
  269. r, w, _ = self._select(self._readers, self._writers, [], timeout)
  270. except select.error as exc:
  271. if exc.args[0] == EINTR:
  272. return ready
  273. else:
  274. raise
  275. r = set(r)
  276. w = set(w)
  277. for fd in r | w:
  278. events = 0
  279. if fd in r:
  280. events |= EVENT_READ
  281. if fd in w:
  282. events |= EVENT_WRITE
  283. key = self._key_from_fd(fd)
  284. if key:
  285. ready.append((key, events & key.events))
  286. return ready
  287. if hasattr(select, 'poll'):
  288. class PollSelector(_BaseSelectorImpl):
  289. """Poll-based selector."""
  290. def __init__(self):
  291. super(PollSelector, self).__init__()
  292. self._poll = select.poll()
  293. def register(self, fileobj, events, data=None):
  294. key = super(PollSelector, self).register(fileobj, events, data)
  295. poll_events = 0
  296. if events & EVENT_READ:
  297. poll_events |= select.POLLIN
  298. if events & EVENT_WRITE:
  299. poll_events |= select.POLLOUT
  300. self._poll.register(key.fd, poll_events)
  301. return key
  302. def unregister(self, fileobj):
  303. key = super(PollSelector, self).unregister(fileobj)
  304. self._poll.unregister(key.fd)
  305. return key
  306. def select(self, timeout=None):
  307. if timeout is None:
  308. timeout = None
  309. elif timeout <= 0:
  310. timeout = 0
  311. else:
  312. # poll() has a resolution of 1 millisecond, round away from
  313. # zero to wait *at least* timeout seconds.
  314. timeout = int(math.ceil(timeout * 1e3))
  315. ready = []
  316. try:
  317. fd_event_list = self._poll.poll(timeout)
  318. except select.error as exc:
  319. if exc.args[0] == EINTR:
  320. return ready
  321. else:
  322. raise
  323. for fd, event in fd_event_list:
  324. events = 0
  325. if event & ~select.POLLIN:
  326. events |= EVENT_WRITE
  327. if event & ~select.POLLOUT:
  328. events |= EVENT_READ
  329. key = self._key_from_fd(fd)
  330. if key:
  331. ready.append((key, events & key.events))
  332. return ready
  333. if hasattr(select, 'epoll'):
  334. class EpollSelector(_BaseSelectorImpl):
  335. """Epoll-based selector."""
  336. def __init__(self):
  337. super(EpollSelector, self).__init__()
  338. self._epoll = select.epoll()
  339. def fileno(self):
  340. return self._epoll.fileno()
  341. def register(self, fileobj, events, data=None):
  342. key = super(EpollSelector, self).register(fileobj, events, data)
  343. epoll_events = 0
  344. if events & EVENT_READ:
  345. epoll_events |= select.EPOLLIN
  346. if events & EVENT_WRITE:
  347. epoll_events |= select.EPOLLOUT
  348. self._epoll.register(key.fd, epoll_events)
  349. return key
  350. def unregister(self, fileobj):
  351. key = super(EpollSelector, self).unregister(fileobj)
  352. try:
  353. self._epoll.unregister(key.fd)
  354. except IOError:
  355. # This can happen if the FD was closed since it
  356. # was registered.
  357. pass
  358. return key
  359. def select(self, timeout=None):
  360. if timeout is None:
  361. timeout = -1
  362. elif timeout <= 0:
  363. timeout = 0
  364. else:
  365. # epoll_wait() has a resolution of 1 millisecond, round away
  366. # from zero to wait *at least* timeout seconds.
  367. timeout = math.ceil(timeout * 1e3) * 1e-3
  368. # epoll_wait() expects `maxevents` to be greater than zero;
  369. # we want to make sure that `select()` can be called when no
  370. # FD is registered.
  371. max_ev = max(len(self._fd_to_key), 1)
  372. ready = []
  373. try:
  374. fd_event_list = self._epoll.poll(timeout, max_ev)
  375. except IOError as exc:
  376. if exc.errno == EINTR:
  377. return ready
  378. else:
  379. raise
  380. for fd, event in fd_event_list:
  381. events = 0
  382. if event & ~select.EPOLLIN:
  383. events |= EVENT_WRITE
  384. if event & ~select.EPOLLOUT:
  385. events |= EVENT_READ
  386. key = self._key_from_fd(fd)
  387. if key:
  388. ready.append((key, events & key.events))
  389. return ready
  390. def close(self):
  391. self._epoll.close()
  392. super(EpollSelector, self).close()
  393. if hasattr(select, 'devpoll'):
  394. class DevpollSelector(_BaseSelectorImpl):
  395. """Solaris /dev/poll selector."""
  396. def __init__(self):
  397. super(DevpollSelector, self).__init__()
  398. self._devpoll = select.devpoll()
  399. def fileno(self):
  400. return self._devpoll.fileno()
  401. def register(self, fileobj, events, data=None):
  402. key = super(DevpollSelector, self).register(fileobj, events, data)
  403. poll_events = 0
  404. if events & EVENT_READ:
  405. poll_events |= select.POLLIN
  406. if events & EVENT_WRITE:
  407. poll_events |= select.POLLOUT
  408. self._devpoll.register(key.fd, poll_events)
  409. return key
  410. def unregister(self, fileobj):
  411. key = super(DevpollSelector, self).unregister(fileobj)
  412. self._devpoll.unregister(key.fd)
  413. return key
  414. def select(self, timeout=None):
  415. if timeout is None:
  416. timeout = None
  417. elif timeout <= 0:
  418. timeout = 0
  419. else:
  420. # devpoll() has a resolution of 1 millisecond, round away from
  421. # zero to wait *at least* timeout seconds.
  422. timeout = math.ceil(timeout * 1e3)
  423. ready = []
  424. try:
  425. fd_event_list = self._devpoll.poll(timeout)
  426. except OSError as exc:
  427. if exc.errno == EINTR:
  428. return ready
  429. else:
  430. raise
  431. for fd, event in fd_event_list:
  432. events = 0
  433. if event & ~select.POLLIN:
  434. events |= EVENT_WRITE
  435. if event & ~select.POLLOUT:
  436. events |= EVENT_READ
  437. key = self._key_from_fd(fd)
  438. if key:
  439. ready.append((key, events & key.events))
  440. return ready
  441. def close(self):
  442. self._devpoll.close()
  443. super(DevpollSelector, self).close()
  444. if hasattr(select, 'kqueue'):
  445. class KqueueSelector(_BaseSelectorImpl):
  446. """Kqueue-based selector."""
  447. def __init__(self):
  448. super(KqueueSelector, self).__init__()
  449. self._kqueue = select.kqueue()
  450. def fileno(self):
  451. return self._kqueue.fileno()
  452. def register(self, fileobj, events, data=None):
  453. key = super(KqueueSelector, self).register(fileobj, events, data)
  454. if events & EVENT_READ:
  455. kev = select.kevent(key.fd, select.KQ_FILTER_READ,
  456. select.KQ_EV_ADD)
  457. self._kqueue.control([kev], 0, 0)
  458. if events & EVENT_WRITE:
  459. kev = select.kevent(key.fd, select.KQ_FILTER_WRITE,
  460. select.KQ_EV_ADD)
  461. self._kqueue.control([kev], 0, 0)
  462. return key
  463. def unregister(self, fileobj):
  464. key = super(KqueueSelector, self).unregister(fileobj)
  465. if key.events & EVENT_READ:
  466. kev = select.kevent(key.fd, select.KQ_FILTER_READ,
  467. select.KQ_EV_DELETE)
  468. try:
  469. self._kqueue.control([kev], 0, 0)
  470. except OSError:
  471. # This can happen if the FD was closed since it
  472. # was registered.
  473. pass
  474. if key.events & EVENT_WRITE:
  475. kev = select.kevent(key.fd, select.KQ_FILTER_WRITE,
  476. select.KQ_EV_DELETE)
  477. try:
  478. self._kqueue.control([kev], 0, 0)
  479. except OSError:
  480. # See comment above.
  481. pass
  482. return key
  483. def select(self, timeout=None):
  484. timeout = None if timeout is None else max(timeout, 0)
  485. max_ev = len(self._fd_to_key)
  486. ready = []
  487. try:
  488. kev_list = self._kqueue.control(None, max_ev, timeout)
  489. except OSError as exc:
  490. if exc.errno == EINTR:
  491. return ready
  492. else:
  493. raise
  494. for kev in kev_list:
  495. fd = kev.ident
  496. flag = kev.filter
  497. events = 0
  498. if flag == select.KQ_FILTER_READ:
  499. events |= EVENT_READ
  500. if flag == select.KQ_FILTER_WRITE:
  501. events |= EVENT_WRITE
  502. key = self._key_from_fd(fd)
  503. if key:
  504. ready.append((key, events & key.events))
  505. return ready
  506. def close(self):
  507. self._kqueue.close()
  508. super(KqueueSelector, self).close()
  509. # Choose the best implementation, roughly:
  510. # epoll|kqueue|devpoll > poll > select.
  511. # select() also can't accept a FD > FD_SETSIZE (usually around 1024)
  512. if 'KqueueSelector' in globals():
  513. DefaultSelector = KqueueSelector
  514. elif 'EpollSelector' in globals():
  515. DefaultSelector = EpollSelector
  516. elif 'DevpollSelector' in globals():
  517. DefaultSelector = DevpollSelector
  518. elif 'PollSelector' in globals():
  519. DefaultSelector = PollSelector
  520. else:
  521. DefaultSelector = SelectSelector