Current Path: > > lib64 > python3.8
Operation : Linux premium107.web-hosting.com 4.18.0-553.44.1.lve.el8.x86_64 #1 SMP Thu Mar 13 14:29:12 UTC 2025 x86_64 Software : Apache Server IP : 198.54.126.246 | Your IP: 216.73.216.181 Domains : 1034 Domain(s) Permission : [ 0755 ]
| Name | Type | Size | Last Modified | Actions |
|---|---|---|---|---|
| __pycache__ | Directory | - | - | |
| asyncio | Directory | - | - | |
| collections | Directory | - | - | |
| concurrent | Directory | - | - | |
| config-3.8-x86_64-linux-gnu | Directory | - | - | |
| ctypes | Directory | - | - | |
| curses | Directory | - | - | |
| dbm | Directory | - | - | |
| distutils | Directory | - | - | |
| Directory | - | - | ||
| encodings | Directory | - | - | |
| ensurepip | Directory | - | - | |
| html | Directory | - | - | |
| http | Directory | - | - | |
| importlib | Directory | - | - | |
| json | Directory | - | - | |
| lib-dynload | Directory | - | - | |
| lib2to3 | Directory | - | - | |
| logging | Directory | - | - | |
| multiprocessing | Directory | - | - | |
| pydoc_data | Directory | - | - | |
| site-packages | Directory | - | - | |
| sqlite3 | Directory | - | - | |
| tkinter | Directory | - | - | |
| turtledemo | Directory | - | - | |
| unittest | Directory | - | - | |
| urllib | Directory | - | - | |
| venv | Directory | - | - | |
| wsgiref | Directory | - | - | |
| xml | Directory | - | - | |
| xmlrpc | Directory | - | - | |
| LICENSE.txt | File | 13937 bytes | June 06 2023 13:32:21. | |
| __future__.py | File | 5147 bytes | June 06 2023 13:32:21. | |
| __phello__.foo.py | File | 64 bytes | June 06 2023 13:32:21. | |
| _bootlocale.py | File | 1801 bytes | June 06 2023 13:32:21. | |
| _collections_abc.py | File | 26100 bytes | June 06 2023 13:32:21. | |
| _compat_pickle.py | File | 8749 bytes | June 06 2023 13:32:21. | |
| _compression.py | File | 5340 bytes | June 06 2023 13:32:21. | |
| _dummy_thread.py | File | 6027 bytes | June 06 2023 13:32:21. | |
| _markupbase.py | File | 14598 bytes | June 06 2023 13:32:21. | |
| _osx_support.py | File | 21774 bytes | June 06 2023 13:32:21. | |
| _py_abc.py | File | 6189 bytes | June 06 2023 13:32:21. | |
| _pydecimal.py | File | 228666 bytes | June 06 2023 13:32:21. | |
| _pyio.py | File | 93177 bytes | June 06 2023 13:32:21. | |
| _sitebuiltins.py | File | 3115 bytes | June 06 2023 13:32:21. | |
| _strptime.py | File | 25268 bytes | June 06 2023 13:32:21. | |
| _sysconfigdata__linux_x86_64-linux-gnu.py | File | 38510 bytes | October 17 2023 18:12:19. | |
| _sysconfigdata_d_linux_x86_64-linux-gnu.py | File | 38237 bytes | October 17 2023 18:03:44. | |
| _threading_local.py | File | 7220 bytes | June 06 2023 13:32:21. | |
| _weakrefset.py | File | 5735 bytes | June 06 2023 13:32:21. | |
| abc.py | File | 4489 bytes | June 06 2023 13:32:21. | |
| aifc.py | File | 32814 bytes | June 06 2023 13:32:21. | |
| antigravity.py | File | 477 bytes | June 06 2023 13:32:21. | |
| argparse.py | File | 96015 bytes | June 06 2023 13:32:21. | |
| ast.py | File | 19234 bytes | June 06 2023 13:32:21. | |
| asynchat.py | File | 11328 bytes | June 06 2023 13:32:21. | |
| asyncore.py | File | 20094 bytes | June 06 2023 13:32:21. | |
| base64.py | File | 20382 bytes | June 06 2023 13:32:21. | |
| bdb.py | File | 32056 bytes | June 06 2023 13:32:21. | |
| binhex.py | File | 13954 bytes | June 06 2023 13:32:21. | |
| bisect.py | File | 2214 bytes | June 06 2023 13:32:21. | |
| bz2.py | File | 12558 bytes | June 06 2023 13:32:21. | |
| cProfile.py | File | 7010 bytes | June 06 2023 13:32:21. | |
| calendar.py | File | 24832 bytes | June 06 2023 13:32:21. | |
| cgi.py | File | 33934 bytes | October 17 2023 18:02:14. | |
| cgitb.py | File | 12096 bytes | June 06 2023 13:32:21. | |
| chunk.py | File | 5435 bytes | June 06 2023 13:32:21. | |
| cmd.py | File | 14860 bytes | June 06 2023 13:32:21. | |
| code.py | File | 10622 bytes | June 06 2023 13:32:21. | |
| codecs.py | File | 36667 bytes | June 06 2023 13:32:21. | |
| codeop.py | File | 6330 bytes | June 06 2023 13:32:21. | |
| colorsys.py | File | 4064 bytes | June 06 2023 13:32:21. | |
| compileall.py | File | 13678 bytes | June 06 2023 13:32:21. | |
| configparser.py | File | 54374 bytes | June 06 2023 13:32:21. | |
| contextlib.py | File | 24995 bytes | June 06 2023 13:32:21. | |
| contextvars.py | File | 129 bytes | June 06 2023 13:32:21. | |
| copy.py | File | 8661 bytes | June 06 2023 13:32:21. | |
| copyreg.py | File | 7135 bytes | June 06 2023 13:32:21. | |
| crypt.py | File | 3610 bytes | June 06 2023 13:32:21. | |
| csv.py | File | 16144 bytes | June 06 2023 13:32:21. | |
| dataclasses.py | File | 49973 bytes | June 06 2023 13:32:21. | |
| datetime.py | File | 88287 bytes | June 06 2023 13:32:21. | |
| decimal.py | File | 320 bytes | June 06 2023 13:32:21. | |
| difflib.py | File | 84058 bytes | June 06 2023 13:32:21. | |
| dis.py | File | 20570 bytes | June 06 2023 13:32:21. | |
| doctest.py | File | 104543 bytes | June 06 2023 13:32:21. | |
| dummy_threading.py | File | 2815 bytes | June 06 2023 13:32:21. | |
| enum.py | File | 38136 bytes | June 06 2023 13:32:21. | |
| filecmp.py | File | 9830 bytes | June 06 2023 13:32:21. | |
| fileinput.py | File | 14709 bytes | June 06 2023 13:32:21. | |
| fnmatch.py | File | 4079 bytes | June 06 2023 13:32:21. | |
| formatter.py | File | 15143 bytes | June 06 2023 13:32:21. | |
| fractions.py | File | 24329 bytes | June 06 2023 13:32:21. | |
| ftplib.py | File | 35129 bytes | June 06 2023 13:32:21. | |
| functools.py | File | 37406 bytes | June 06 2023 13:32:21. | |
| genericpath.py | File | 4975 bytes | June 06 2023 13:32:21. | |
| getopt.py | File | 7489 bytes | June 06 2023 13:32:21. | |
| getpass.py | File | 5994 bytes | June 06 2023 13:32:21. | |
| gettext.py | File | 27138 bytes | June 06 2023 13:32:21. | |
| glob.py | File | 5697 bytes | June 06 2023 13:32:21. | |
| gzip.py | File | 21413 bytes | June 06 2023 13:32:21. | |
| hashlib.py | File | 8333 bytes | October 17 2023 18:02:14. | |
| heapq.py | File | 22877 bytes | June 06 2023 13:32:21. | |
| hmac.py | File | 7850 bytes | October 17 2023 18:02:14. | |
| imaplib.py | File | 53606 bytes | June 06 2023 13:32:21. | |
| imghdr.py | File | 3808 bytes | June 06 2023 13:32:21. | |
| imp.py | File | 10536 bytes | June 06 2023 13:32:21. | |
| inspect.py | File | 118550 bytes | June 06 2023 13:32:21. | |
| io.py | File | 3541 bytes | June 06 2023 13:32:21. | |
| ipaddress.py | File | 71644 bytes | June 06 2023 13:32:21. | |
| keyword.py | File | 945 bytes | June 06 2023 13:32:21. | |
| linecache.py | File | 5330 bytes | June 06 2023 13:32:21. | |
| locale.py | File | 78191 bytes | June 06 2023 13:32:21. | |
| lzma.py | File | 12983 bytes | June 06 2023 13:32:21. | |
| mailbox.py | File | 78661 bytes | June 06 2023 13:32:21. | |
| mailcap.py | File | 9067 bytes | June 06 2023 13:32:21. | |
| mimetypes.py | File | 21664 bytes | June 06 2023 13:32:21. | |
| modulefinder.py | File | 24430 bytes | June 06 2023 13:32:21. | |
| netrc.py | File | 5566 bytes | June 06 2023 13:32:21. | |
| nntplib.py | File | 43261 bytes | June 06 2023 13:32:21. | |
| ntpath.py | File | 27734 bytes | June 06 2023 13:32:21. | |
| nturl2path.py | File | 2887 bytes | June 06 2023 13:32:21. | |
| numbers.py | File | 10244 bytes | June 06 2023 13:32:21. | |
| opcode.py | File | 5808 bytes | June 06 2023 13:32:21. | |
| operator.py | File | 10711 bytes | June 06 2023 13:32:21. | |
| optparse.py | File | 60369 bytes | June 06 2023 13:32:21. | |
| os.py | File | 38995 bytes | June 06 2023 13:32:21. | |
| pathlib.py | File | 52610 bytes | June 06 2023 13:32:21. | |
| pdb.py | File | 62738 bytes | June 06 2023 13:32:21. | |
| pickle.py | File | 64467 bytes | June 06 2023 13:32:21. | |
| pickletools.py | File | 93486 bytes | June 06 2023 13:32:21. | |
| pipes.py | File | 8916 bytes | June 06 2023 13:32:21. | |
| pkgutil.py | File | 21500 bytes | June 06 2023 13:32:21. | |
| platform.py | File | 40425 bytes | June 06 2023 13:32:21. | |
| plistlib.py | File | 32220 bytes | June 06 2023 13:32:21. | |
| poplib.py | File | 15077 bytes | June 06 2023 13:32:21. | |
| posixpath.py | File | 15627 bytes | June 06 2023 13:32:21. | |
| pprint.py | File | 21484 bytes | June 06 2023 13:32:21. | |
| profile.py | File | 23546 bytes | June 06 2023 13:32:21. | |
| pstats.py | File | 27345 bytes | June 06 2023 13:32:21. | |
| pty.py | File | 4807 bytes | June 06 2023 13:32:21. | |
| py_compile.py | File | 8203 bytes | October 17 2023 18:02:14. | |
| pyclbr.py | File | 15255 bytes | June 06 2023 13:32:21. | |
| pydoc.py | File | 106700 bytes | October 17 2023 18:12:57. | |
| queue.py | File | 11356 bytes | June 06 2023 13:32:21. | |
| quopri.py | File | 7252 bytes | June 06 2023 13:32:21. | |
| random.py | File | 28802 bytes | June 06 2023 13:32:21. | |
| re.py | File | 15861 bytes | June 06 2023 13:32:21. | |
| reprlib.py | File | 5267 bytes | June 06 2023 13:32:21. | |
| rlcompleter.py | File | 7097 bytes | June 06 2023 13:32:21. | |
| runpy.py | File | 12052 bytes | June 06 2023 13:32:21. | |
| sched.py | File | 6442 bytes | June 06 2023 13:32:21. | |
| secrets.py | File | 2038 bytes | June 06 2023 13:32:21. | |
| selectors.py | File | 18561 bytes | June 06 2023 13:32:21. | |
| shelve.py | File | 8527 bytes | June 06 2023 13:32:21. | |
| shlex.py | File | 13325 bytes | June 06 2023 13:32:21. | |
| shutil.py | File | 51761 bytes | June 06 2023 13:32:21. | |
| signal.py | File | 2273 bytes | June 06 2023 13:32:21. | |
| site.py | File | 21838 bytes | October 17 2023 18:02:14. | |
| smtpd.py | File | 34709 bytes | June 06 2023 13:32:21. | |
| smtplib.py | File | 45001 bytes | June 06 2023 13:32:21. | |
| sndhdr.py | File | 7099 bytes | June 06 2023 13:32:21. | |
| socket.py | File | 35243 bytes | June 06 2023 13:32:21. | |
| socketserver.py | File | 27296 bytes | June 06 2023 13:32:21. | |
| sre_compile.py | File | 26695 bytes | June 06 2023 13:32:21. | |
| sre_constants.py | File | 7154 bytes | June 06 2023 13:32:21. | |
| sre_parse.py | File | 40230 bytes | June 06 2023 13:32:21. | |
| ssl.py | File | 50760 bytes | June 06 2023 13:32:21. | |
| stat.py | File | 5485 bytes | June 06 2023 13:32:21. | |
| statistics.py | File | 39690 bytes | June 06 2023 13:32:21. | |
| string.py | File | 10535 bytes | June 06 2023 13:32:21. | |
| stringprep.py | File | 12917 bytes | June 06 2023 13:32:21. | |
| struct.py | File | 257 bytes | June 06 2023 13:32:21. | |
| subprocess.py | File | 78250 bytes | June 06 2023 13:32:21. | |
| sunau.py | File | 18375 bytes | June 06 2023 13:32:21. | |
| symbol.py | File | 2109 bytes | October 17 2023 18:04:15. | |
| symtable.py | File | 8021 bytes | June 06 2023 13:32:21. | |
| sysconfig.py | File | 24896 bytes | October 17 2023 18:12:55. | |
| tabnanny.py | File | 11406 bytes | June 06 2023 13:32:21. | |
| tarfile.py | File | 106473 bytes | October 17 2023 18:02:14. | |
| telnetlib.py | File | 23254 bytes | June 06 2023 13:32:21. | |
| tempfile.py | File | 27537 bytes | June 06 2023 13:32:21. | |
| textwrap.py | File | 19407 bytes | June 06 2023 13:32:21. | |
| this.py | File | 1003 bytes | June 06 2023 13:32:21. | |
| threading.py | File | 50820 bytes | June 06 2023 13:32:21. | |
| timeit.py | File | 13480 bytes | June 06 2023 13:32:21. | |
| token.py | File | 2368 bytes | June 06 2023 13:32:21. | |
| tokenize.py | File | 25841 bytes | June 06 2023 13:32:21. | |
| trace.py | File | 29870 bytes | June 06 2023 13:32:21. | |
| traceback.py | File | 23611 bytes | June 06 2023 13:32:21. | |
| tracemalloc.py | File | 17076 bytes | June 06 2023 13:32:21. | |
| tty.py | File | 879 bytes | June 06 2023 13:32:21. | |
| turtle.py | File | 143716 bytes | June 06 2023 13:32:21. | |
| types.py | File | 9713 bytes | June 06 2023 13:32:21. | |
| typing.py | File | 68962 bytes | June 06 2023 13:32:21. | |
| uu.py | File | 7277 bytes | October 17 2023 18:12:57. | |
| uuid.py | File | 30515 bytes | October 17 2023 18:02:14. | |
| warnings.py | File | 19688 bytes | June 06 2023 13:32:21. | |
| wave.py | File | 18230 bytes | June 06 2023 13:32:21. | |
| weakref.py | File | 21387 bytes | June 06 2023 13:32:21. | |
| webbrowser.py | File | 24083 bytes | June 06 2023 13:32:21. | |
| xdrlib.py | File | 5913 bytes | June 06 2023 13:32:21. | |
| zipapp.py | File | 7535 bytes | June 06 2023 13:32:21. | |
| zipfile.py | File | 87728 bytes | June 06 2023 13:32:21. | |
| zipimport.py | File | 30765 bytes | June 06 2023 13:32:21. |
"""Selectors module.
This module allows high-level and efficient I/O multiplexing, built upon the
`select` module primitives.
"""
from abc import ABCMeta, abstractmethod
from collections import namedtuple
from collections.abc import Mapping
import math
import select
import sys
# generic events, that must be mapped to implementation-specific ones
EVENT_READ = (1 << 0)
EVENT_WRITE = (1 << 1)
def _fileobj_to_fd(fileobj):
"""Return a file descriptor from a file object.
Parameters:
fileobj -- file object or file descriptor
Returns:
corresponding file descriptor
Raises:
ValueError if the object is invalid
"""
if isinstance(fileobj, int):
fd = fileobj
else:
try:
fd = int(fileobj.fileno())
except (AttributeError, TypeError, ValueError):
raise ValueError("Invalid file object: "
"{!r}".format(fileobj)) from None
if fd < 0:
raise ValueError("Invalid file descriptor: {}".format(fd))
return fd
SelectorKey = namedtuple('SelectorKey', ['fileobj', 'fd', 'events', 'data'])
SelectorKey.__doc__ = """SelectorKey(fileobj, fd, events, data)
Object used to associate a file object to its backing
file descriptor, selected event mask, and attached data.
"""
if sys.version_info >= (3, 5):
SelectorKey.fileobj.__doc__ = 'File object registered.'
SelectorKey.fd.__doc__ = 'Underlying file descriptor.'
SelectorKey.events.__doc__ = 'Events that must be waited for on this file object.'
SelectorKey.data.__doc__ = ('''Optional opaque data associated to this file object.
For example, this could be used to store a per-client session ID.''')
class _SelectorMapping(Mapping):
"""Mapping of file objects to selector keys."""
def __init__(self, selector):
self._selector = selector
def __len__(self):
return len(self._selector._fd_to_key)
def __getitem__(self, fileobj):
try:
fd = self._selector._fileobj_lookup(fileobj)
return self._selector._fd_to_key[fd]
except KeyError:
raise KeyError("{!r} is not registered".format(fileobj)) from None
def __iter__(self):
return iter(self._selector._fd_to_key)
class BaseSelector(metaclass=ABCMeta):
"""Selector abstract base class.
A selector supports registering file objects to be monitored for specific
I/O events.
A file object is a file descriptor or any object with a `fileno()` method.
An arbitrary object can be attached to the file object, which can be used
for example to store context information, a callback, etc.
A selector can use various implementations (select(), poll(), epoll()...)
depending on the platform. The default `Selector` class uses the most
efficient implementation on the current platform.
"""
@abstractmethod
def register(self, fileobj, events, data=None):
"""Register a file object.
Parameters:
fileobj -- file object or file descriptor
events -- events to monitor (bitwise mask of EVENT_READ|EVENT_WRITE)
data -- attached data
Returns:
SelectorKey instance
Raises:
ValueError if events is invalid
KeyError if fileobj is already registered
OSError if fileobj is closed or otherwise is unacceptable to
the underlying system call (if a system call is made)
Note:
OSError may or may not be raised
"""
raise NotImplementedError
@abstractmethod
def unregister(self, fileobj):
"""Unregister a file object.
Parameters:
fileobj -- file object or file descriptor
Returns:
SelectorKey instance
Raises:
KeyError if fileobj is not registered
Note:
If fileobj is registered but has since been closed this does
*not* raise OSError (even if the wrapped syscall does)
"""
raise NotImplementedError
def modify(self, fileobj, events, data=None):
"""Change a registered file object monitored events or attached data.
Parameters:
fileobj -- file object or file descriptor
events -- events to monitor (bitwise mask of EVENT_READ|EVENT_WRITE)
data -- attached data
Returns:
SelectorKey instance
Raises:
Anything that unregister() or register() raises
"""
self.unregister(fileobj)
return self.register(fileobj, events, data)
@abstractmethod
def select(self, timeout=None):
"""Perform the actual selection, until some monitored file objects are
ready or a timeout expires.
Parameters:
timeout -- if timeout > 0, this specifies the maximum wait time, in
seconds
if timeout <= 0, the select() call won't block, and will
report the currently ready file objects
if timeout is None, select() will block until a monitored
file object becomes ready
Returns:
list of (key, events) for ready file objects
`events` is a bitwise mask of EVENT_READ|EVENT_WRITE
"""
raise NotImplementedError
def close(self):
"""Close the selector.
This must be called to make sure that any underlying resource is freed.
"""
pass
def get_key(self, fileobj):
"""Return the key associated to a registered file object.
Returns:
SelectorKey for this file object
"""
mapping = self.get_map()
if mapping is None:
raise RuntimeError('Selector is closed')
try:
return mapping[fileobj]
except KeyError:
raise KeyError("{!r} is not registered".format(fileobj)) from None
@abstractmethod
def get_map(self):
"""Return a mapping of file objects to selector keys."""
raise NotImplementedError
def __enter__(self):
return self
def __exit__(self, *args):
self.close()
class _BaseSelectorImpl(BaseSelector):
"""Base selector implementation."""
def __init__(self):
# this maps file descriptors to keys
self._fd_to_key = {}
# read-only mapping returned by get_map()
self._map = _SelectorMapping(self)
def _fileobj_lookup(self, fileobj):
"""Return a file descriptor from a file object.
This wraps _fileobj_to_fd() to do an exhaustive search in case
the object is invalid but we still have it in our map. This
is used by unregister() so we can unregister an object that
was previously registered even if it is closed. It is also
used by _SelectorMapping.
"""
try:
return _fileobj_to_fd(fileobj)
except ValueError:
# Do an exhaustive search.
for key in self._fd_to_key.values():
if key.fileobj is fileobj:
return key.fd
# Raise ValueError after all.
raise
def register(self, fileobj, events, data=None):
if (not events) or (events & ~(EVENT_READ | EVENT_WRITE)):
raise ValueError("Invalid events: {!r}".format(events))
key = SelectorKey(fileobj, self._fileobj_lookup(fileobj), events, data)
if key.fd in self._fd_to_key:
raise KeyError("{!r} (FD {}) is already registered"
.format(fileobj, key.fd))
self._fd_to_key[key.fd] = key
return key
def unregister(self, fileobj):
try:
key = self._fd_to_key.pop(self._fileobj_lookup(fileobj))
except KeyError:
raise KeyError("{!r} is not registered".format(fileobj)) from None
return key
def modify(self, fileobj, events, data=None):
try:
key = self._fd_to_key[self._fileobj_lookup(fileobj)]
except KeyError:
raise KeyError("{!r} is not registered".format(fileobj)) from None
if events != key.events:
self.unregister(fileobj)
key = self.register(fileobj, events, data)
elif data != key.data:
# Use a shortcut to update the data.
key = key._replace(data=data)
self._fd_to_key[key.fd] = key
return key
def close(self):
self._fd_to_key.clear()
self._map = None
def get_map(self):
return self._map
def _key_from_fd(self, fd):
"""Return the key associated to a given file descriptor.
Parameters:
fd -- file descriptor
Returns:
corresponding key, or None if not found
"""
try:
return self._fd_to_key[fd]
except KeyError:
return None
class SelectSelector(_BaseSelectorImpl):
"""Select-based selector."""
def __init__(self):
super().__init__()
self._readers = set()
self._writers = set()
def register(self, fileobj, events, data=None):
key = super().register(fileobj, events, data)
if events & EVENT_READ:
self._readers.add(key.fd)
if events & EVENT_WRITE:
self._writers.add(key.fd)
return key
def unregister(self, fileobj):
key = super().unregister(fileobj)
self._readers.discard(key.fd)
self._writers.discard(key.fd)
return key
if sys.platform == 'win32':
def _select(self, r, w, _, timeout=None):
r, w, x = select.select(r, w, w, timeout)
return r, w + x, []
else:
_select = select.select
def select(self, timeout=None):
timeout = None if timeout is None else max(timeout, 0)
ready = []
try:
r, w, _ = self._select(self._readers, self._writers, [], timeout)
except InterruptedError:
return ready
r = set(r)
w = set(w)
for fd in r | w:
events = 0
if fd in r:
events |= EVENT_READ
if fd in w:
events |= EVENT_WRITE
key = self._key_from_fd(fd)
if key:
ready.append((key, events & key.events))
return ready
class _PollLikeSelector(_BaseSelectorImpl):
"""Base class shared between poll, epoll and devpoll selectors."""
_selector_cls = None
_EVENT_READ = None
_EVENT_WRITE = None
def __init__(self):
super().__init__()
self._selector = self._selector_cls()
def register(self, fileobj, events, data=None):
key = super().register(fileobj, events, data)
poller_events = 0
if events & EVENT_READ:
poller_events |= self._EVENT_READ
if events & EVENT_WRITE:
poller_events |= self._EVENT_WRITE
try:
self._selector.register(key.fd, poller_events)
except:
super().unregister(fileobj)
raise
return key
def unregister(self, fileobj):
key = super().unregister(fileobj)
try:
self._selector.unregister(key.fd)
except OSError:
# This can happen if the FD was closed since it
# was registered.
pass
return key
def modify(self, fileobj, events, data=None):
try:
key = self._fd_to_key[self._fileobj_lookup(fileobj)]
except KeyError:
raise KeyError(f"{fileobj!r} is not registered") from None
changed = False
if events != key.events:
selector_events = 0
if events & EVENT_READ:
selector_events |= self._EVENT_READ
if events & EVENT_WRITE:
selector_events |= self._EVENT_WRITE
try:
self._selector.modify(key.fd, selector_events)
except:
super().unregister(fileobj)
raise
changed = True
if data != key.data:
changed = True
if changed:
key = key._replace(events=events, data=data)
self._fd_to_key[key.fd] = key
return key
def select(self, timeout=None):
# This is shared between poll() and epoll().
# epoll() has a different signature and handling of timeout parameter.
if timeout is None:
timeout = None
elif timeout <= 0:
timeout = 0
else:
# poll() has a resolution of 1 millisecond, round away from
# zero to wait *at least* timeout seconds.
timeout = math.ceil(timeout * 1e3)
ready = []
try:
fd_event_list = self._selector.poll(timeout)
except InterruptedError:
return ready
for fd, event in fd_event_list:
events = 0
if event & ~self._EVENT_READ:
events |= EVENT_WRITE
if event & ~self._EVENT_WRITE:
events |= EVENT_READ
key = self._key_from_fd(fd)
if key:
ready.append((key, events & key.events))
return ready
if hasattr(select, 'poll'):
class PollSelector(_PollLikeSelector):
"""Poll-based selector."""
_selector_cls = select.poll
_EVENT_READ = select.POLLIN
_EVENT_WRITE = select.POLLOUT
if hasattr(select, 'epoll'):
class EpollSelector(_PollLikeSelector):
"""Epoll-based selector."""
_selector_cls = select.epoll
_EVENT_READ = select.EPOLLIN
_EVENT_WRITE = select.EPOLLOUT
def fileno(self):
return self._selector.fileno()
def select(self, timeout=None):
if timeout is None:
timeout = -1
elif timeout <= 0:
timeout = 0
else:
# epoll_wait() has a resolution of 1 millisecond, round away
# from zero to wait *at least* timeout seconds.
timeout = math.ceil(timeout * 1e3) * 1e-3
# epoll_wait() expects `maxevents` to be greater than zero;
# we want to make sure that `select()` can be called when no
# FD is registered.
max_ev = max(len(self._fd_to_key), 1)
ready = []
try:
fd_event_list = self._selector.poll(timeout, max_ev)
except InterruptedError:
return ready
for fd, event in fd_event_list:
events = 0
if event & ~select.EPOLLIN:
events |= EVENT_WRITE
if event & ~select.EPOLLOUT:
events |= EVENT_READ
key = self._key_from_fd(fd)
if key:
ready.append((key, events & key.events))
return ready
def close(self):
self._selector.close()
super().close()
if hasattr(select, 'devpoll'):
class DevpollSelector(_PollLikeSelector):
"""Solaris /dev/poll selector."""
_selector_cls = select.devpoll
_EVENT_READ = select.POLLIN
_EVENT_WRITE = select.POLLOUT
def fileno(self):
return self._selector.fileno()
def close(self):
self._selector.close()
super().close()
if hasattr(select, 'kqueue'):
class KqueueSelector(_BaseSelectorImpl):
"""Kqueue-based selector."""
def __init__(self):
super().__init__()
self._selector = select.kqueue()
def fileno(self):
return self._selector.fileno()
def register(self, fileobj, events, data=None):
key = super().register(fileobj, events, data)
try:
if events & EVENT_READ:
kev = select.kevent(key.fd, select.KQ_FILTER_READ,
select.KQ_EV_ADD)
self._selector.control([kev], 0, 0)
if events & EVENT_WRITE:
kev = select.kevent(key.fd, select.KQ_FILTER_WRITE,
select.KQ_EV_ADD)
self._selector.control([kev], 0, 0)
except:
super().unregister(fileobj)
raise
return key
def unregister(self, fileobj):
key = super().unregister(fileobj)
if key.events & EVENT_READ:
kev = select.kevent(key.fd, select.KQ_FILTER_READ,
select.KQ_EV_DELETE)
try:
self._selector.control([kev], 0, 0)
except OSError:
# This can happen if the FD was closed since it
# was registered.
pass
if key.events & EVENT_WRITE:
kev = select.kevent(key.fd, select.KQ_FILTER_WRITE,
select.KQ_EV_DELETE)
try:
self._selector.control([kev], 0, 0)
except OSError:
# See comment above.
pass
return key
def select(self, timeout=None):
timeout = None if timeout is None else max(timeout, 0)
max_ev = len(self._fd_to_key)
ready = []
try:
kev_list = self._selector.control(None, max_ev, timeout)
except InterruptedError:
return ready
for kev in kev_list:
fd = kev.ident
flag = kev.filter
events = 0
if flag == select.KQ_FILTER_READ:
events |= EVENT_READ
if flag == select.KQ_FILTER_WRITE:
events |= EVENT_WRITE
key = self._key_from_fd(fd)
if key:
ready.append((key, events & key.events))
return ready
def close(self):
self._selector.close()
super().close()
# Choose the best implementation, roughly:
# epoll|kqueue|devpoll > poll > select.
# select() also can't accept a FD > FD_SETSIZE (usually around 1024)
if 'KqueueSelector' in globals():
DefaultSelector = KqueueSelector
elif 'EpollSelector' in globals():
DefaultSelector = EpollSelector
elif 'DevpollSelector' in globals():
DefaultSelector = DevpollSelector
elif 'PollSelector' in globals():
DefaultSelector = PollSelector
else:
DefaultSelector = SelectSelector
SILENT KILLER Tool