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. |
"""Utilities for with-statement contexts. See PEP 343."""
import abc
import sys
import _collections_abc
from collections import deque
from functools import wraps
from types import MethodType
__all__ = ["asynccontextmanager", "contextmanager", "closing", "nullcontext",
"AbstractContextManager", "AbstractAsyncContextManager",
"AsyncExitStack", "ContextDecorator", "ExitStack",
"redirect_stdout", "redirect_stderr", "suppress"]
class AbstractContextManager(abc.ABC):
"""An abstract base class for context managers."""
def __enter__(self):
"""Return `self` upon entering the runtime context."""
return self
@abc.abstractmethod
def __exit__(self, exc_type, exc_value, traceback):
"""Raise any exception triggered within the runtime context."""
return None
@classmethod
def __subclasshook__(cls, C):
if cls is AbstractContextManager:
return _collections_abc._check_methods(C, "__enter__", "__exit__")
return NotImplemented
class AbstractAsyncContextManager(abc.ABC):
"""An abstract base class for asynchronous context managers."""
async def __aenter__(self):
"""Return `self` upon entering the runtime context."""
return self
@abc.abstractmethod
async def __aexit__(self, exc_type, exc_value, traceback):
"""Raise any exception triggered within the runtime context."""
return None
@classmethod
def __subclasshook__(cls, C):
if cls is AbstractAsyncContextManager:
return _collections_abc._check_methods(C, "__aenter__",
"__aexit__")
return NotImplemented
class ContextDecorator(object):
"A base class or mixin that enables context managers to work as decorators."
def _recreate_cm(self):
"""Return a recreated instance of self.
Allows an otherwise one-shot context manager like
_GeneratorContextManager to support use as
a decorator via implicit recreation.
This is a private interface just for _GeneratorContextManager.
See issue #11647 for details.
"""
return self
def __call__(self, func):
@wraps(func)
def inner(*args, **kwds):
with self._recreate_cm():
return func(*args, **kwds)
return inner
class _GeneratorContextManagerBase:
"""Shared functionality for @contextmanager and @asynccontextmanager."""
def __init__(self, func, args, kwds):
self.gen = func(*args, **kwds)
self.func, self.args, self.kwds = func, args, kwds
# Issue 19330: ensure context manager instances have good docstrings
doc = getattr(func, "__doc__", None)
if doc is None:
doc = type(self).__doc__
self.__doc__ = doc
# Unfortunately, this still doesn't provide good help output when
# inspecting the created context manager instances, since pydoc
# currently bypasses the instance docstring and shows the docstring
# for the class instead.
# See http://bugs.python.org/issue19404 for more details.
class _GeneratorContextManager(_GeneratorContextManagerBase,
AbstractContextManager,
ContextDecorator):
"""Helper for @contextmanager decorator."""
def _recreate_cm(self):
# _GCM instances are one-shot context managers, so the
# CM must be recreated each time a decorated function is
# called
return self.__class__(self.func, self.args, self.kwds)
def __enter__(self):
# do not keep args and kwds alive unnecessarily
# they are only needed for recreation, which is not possible anymore
del self.args, self.kwds, self.func
try:
return next(self.gen)
except StopIteration:
raise RuntimeError("generator didn't yield") from None
def __exit__(self, type, value, traceback):
if type is None:
try:
next(self.gen)
except StopIteration:
return False
else:
raise RuntimeError("generator didn't stop")
else:
if value is None:
# Need to force instantiation so we can reliably
# tell if we get the same exception back
value = type()
try:
self.gen.throw(type, value, traceback)
except StopIteration as exc:
# Suppress StopIteration *unless* it's the same exception that
# was passed to throw(). This prevents a StopIteration
# raised inside the "with" statement from being suppressed.
return exc is not value
except RuntimeError as exc:
# Don't re-raise the passed in exception. (issue27122)
if exc is value:
return False
# Likewise, avoid suppressing if a StopIteration exception
# was passed to throw() and later wrapped into a RuntimeError
# (see PEP 479).
if type is StopIteration and exc.__cause__ is value:
return False
raise
except:
# only re-raise if it's *not* the exception that was
# passed to throw(), because __exit__() must not raise
# an exception unless __exit__() itself failed. But throw()
# has to raise the exception to signal propagation, so this
# fixes the impedance mismatch between the throw() protocol
# and the __exit__() protocol.
#
# This cannot use 'except BaseException as exc' (as in the
# async implementation) to maintain compatibility with
# Python 2, where old-style class exceptions are not caught
# by 'except BaseException'.
if sys.exc_info()[1] is value:
return False
raise
raise RuntimeError("generator didn't stop after throw()")
class _AsyncGeneratorContextManager(_GeneratorContextManagerBase,
AbstractAsyncContextManager):
"""Helper for @asynccontextmanager."""
async def __aenter__(self):
try:
return await self.gen.__anext__()
except StopAsyncIteration:
raise RuntimeError("generator didn't yield") from None
async def __aexit__(self, typ, value, traceback):
if typ is None:
try:
await self.gen.__anext__()
except StopAsyncIteration:
return
else:
raise RuntimeError("generator didn't stop")
else:
if value is None:
value = typ()
# See _GeneratorContextManager.__exit__ for comments on subtleties
# in this implementation
try:
await self.gen.athrow(typ, value, traceback)
raise RuntimeError("generator didn't stop after athrow()")
except StopAsyncIteration as exc:
return exc is not value
except RuntimeError as exc:
if exc is value:
return False
# Avoid suppressing if a StopIteration exception
# was passed to throw() and later wrapped into a RuntimeError
# (see PEP 479 for sync generators; async generators also
# have this behavior). But do this only if the exception wrapped
# by the RuntimeError is actully Stop(Async)Iteration (see
# issue29692).
if isinstance(value, (StopIteration, StopAsyncIteration)):
if exc.__cause__ is value:
return False
raise
except BaseException as exc:
if exc is not value:
raise
def contextmanager(func):
"""@contextmanager decorator.
Typical usage:
@contextmanager
def some_generator(<arguments>):
<setup>
try:
yield <value>
finally:
<cleanup>
This makes this:
with some_generator(<arguments>) as <variable>:
<body>
equivalent to this:
<setup>
try:
<variable> = <value>
<body>
finally:
<cleanup>
"""
@wraps(func)
def helper(*args, **kwds):
return _GeneratorContextManager(func, args, kwds)
return helper
def asynccontextmanager(func):
"""@asynccontextmanager decorator.
Typical usage:
@asynccontextmanager
async def some_async_generator(<arguments>):
<setup>
try:
yield <value>
finally:
<cleanup>
This makes this:
async with some_async_generator(<arguments>) as <variable>:
<body>
equivalent to this:
<setup>
try:
<variable> = <value>
<body>
finally:
<cleanup>
"""
@wraps(func)
def helper(*args, **kwds):
return _AsyncGeneratorContextManager(func, args, kwds)
return helper
class closing(AbstractContextManager):
"""Context to automatically close something at the end of a block.
Code like this:
with closing(<module>.open(<arguments>)) as f:
<block>
is equivalent to this:
f = <module>.open(<arguments>)
try:
<block>
finally:
f.close()
"""
def __init__(self, thing):
self.thing = thing
def __enter__(self):
return self.thing
def __exit__(self, *exc_info):
self.thing.close()
class _RedirectStream(AbstractContextManager):
_stream = None
def __init__(self, new_target):
self._new_target = new_target
# We use a list of old targets to make this CM re-entrant
self._old_targets = []
def __enter__(self):
self._old_targets.append(getattr(sys, self._stream))
setattr(sys, self._stream, self._new_target)
return self._new_target
def __exit__(self, exctype, excinst, exctb):
setattr(sys, self._stream, self._old_targets.pop())
class redirect_stdout(_RedirectStream):
"""Context manager for temporarily redirecting stdout to another file.
# How to send help() to stderr
with redirect_stdout(sys.stderr):
help(dir)
# How to write help() to a file
with open('help.txt', 'w') as f:
with redirect_stdout(f):
help(pow)
"""
_stream = "stdout"
class redirect_stderr(_RedirectStream):
"""Context manager for temporarily redirecting stderr to another file."""
_stream = "stderr"
class suppress(AbstractContextManager):
"""Context manager to suppress specified exceptions
After the exception is suppressed, execution proceeds with the next
statement following the with statement.
with suppress(FileNotFoundError):
os.remove(somefile)
# Execution still resumes here if the file was already removed
"""
def __init__(self, *exceptions):
self._exceptions = exceptions
def __enter__(self):
pass
def __exit__(self, exctype, excinst, exctb):
# Unlike isinstance and issubclass, CPython exception handling
# currently only looks at the concrete type hierarchy (ignoring
# the instance and subclass checking hooks). While Guido considers
# that a bug rather than a feature, it's a fairly hard one to fix
# due to various internal implementation details. suppress provides
# the simpler issubclass based semantics, rather than trying to
# exactly reproduce the limitations of the CPython interpreter.
#
# See http://bugs.python.org/issue12029 for more details
return exctype is not None and issubclass(exctype, self._exceptions)
class _BaseExitStack:
"""A base class for ExitStack and AsyncExitStack."""
@staticmethod
def _create_exit_wrapper(cm, cm_exit):
return MethodType(cm_exit, cm)
@staticmethod
def _create_cb_wrapper(callback, /, *args, **kwds):
def _exit_wrapper(exc_type, exc, tb):
callback(*args, **kwds)
return _exit_wrapper
def __init__(self):
self._exit_callbacks = deque()
def pop_all(self):
"""Preserve the context stack by transferring it to a new instance."""
new_stack = type(self)()
new_stack._exit_callbacks = self._exit_callbacks
self._exit_callbacks = deque()
return new_stack
def push(self, exit):
"""Registers a callback with the standard __exit__ method signature.
Can suppress exceptions the same way __exit__ method can.
Also accepts any object with an __exit__ method (registering a call
to the method instead of the object itself).
"""
# We use an unbound method rather than a bound method to follow
# the standard lookup behaviour for special methods.
_cb_type = type(exit)
try:
exit_method = _cb_type.__exit__
except AttributeError:
# Not a context manager, so assume it's a callable.
self._push_exit_callback(exit)
else:
self._push_cm_exit(exit, exit_method)
return exit # Allow use as a decorator.
def enter_context(self, cm):
"""Enters the supplied context manager.
If successful, also pushes its __exit__ method as a callback and
returns the result of the __enter__ method.
"""
# We look up the special methods on the type to match the with
# statement.
_cm_type = type(cm)
_exit = _cm_type.__exit__
result = _cm_type.__enter__(cm)
self._push_cm_exit(cm, _exit)
return result
def callback(*args, **kwds):
"""Registers an arbitrary callback and arguments.
Cannot suppress exceptions.
"""
if len(args) >= 2:
self, callback, *args = args
elif not args:
raise TypeError("descriptor 'callback' of '_BaseExitStack' object "
"needs an argument")
elif 'callback' in kwds:
callback = kwds.pop('callback')
self, *args = args
import warnings
warnings.warn("Passing 'callback' as keyword argument is deprecated",
DeprecationWarning, stacklevel=2)
else:
raise TypeError('callback expected at least 1 positional argument, '
'got %d' % (len(args)-1))
_exit_wrapper = self._create_cb_wrapper(callback, *args, **kwds)
# We changed the signature, so using @wraps is not appropriate, but
# setting __wrapped__ may still help with introspection.
_exit_wrapper.__wrapped__ = callback
self._push_exit_callback(_exit_wrapper)
return callback # Allow use as a decorator
callback.__text_signature__ = '($self, callback, /, *args, **kwds)'
def _push_cm_exit(self, cm, cm_exit):
"""Helper to correctly register callbacks to __exit__ methods."""
_exit_wrapper = self._create_exit_wrapper(cm, cm_exit)
self._push_exit_callback(_exit_wrapper, True)
def _push_exit_callback(self, callback, is_sync=True):
self._exit_callbacks.append((is_sync, callback))
# Inspired by discussions on http://bugs.python.org/issue13585
class ExitStack(_BaseExitStack, AbstractContextManager):
"""Context manager for dynamic management of a stack of exit callbacks.
For example:
with ExitStack() as stack:
files = [stack.enter_context(open(fname)) for fname in filenames]
# All opened files will automatically be closed at the end of
# the with statement, even if attempts to open files later
# in the list raise an exception.
"""
def __enter__(self):
return self
def __exit__(self, *exc_details):
received_exc = exc_details[0] is not None
# We manipulate the exception state so it behaves as though
# we were actually nesting multiple with statements
frame_exc = sys.exc_info()[1]
def _fix_exception_context(new_exc, old_exc):
# Context may not be correct, so find the end of the chain
while 1:
exc_context = new_exc.__context__
if exc_context is old_exc:
# Context is already set correctly (see issue 20317)
return
if exc_context is None or exc_context is frame_exc:
break
new_exc = exc_context
# Change the end of the chain to point to the exception
# we expect it to reference
new_exc.__context__ = old_exc
# Callbacks are invoked in LIFO order to match the behaviour of
# nested context managers
suppressed_exc = False
pending_raise = False
while self._exit_callbacks:
is_sync, cb = self._exit_callbacks.pop()
assert is_sync
try:
if cb(*exc_details):
suppressed_exc = True
pending_raise = False
exc_details = (None, None, None)
except:
new_exc_details = sys.exc_info()
# simulate the stack of exceptions by setting the context
_fix_exception_context(new_exc_details[1], exc_details[1])
pending_raise = True
exc_details = new_exc_details
if pending_raise:
try:
# bare "raise exc_details[1]" replaces our carefully
# set-up context
fixed_ctx = exc_details[1].__context__
raise exc_details[1]
except BaseException:
exc_details[1].__context__ = fixed_ctx
raise
return received_exc and suppressed_exc
def close(self):
"""Immediately unwind the context stack."""
self.__exit__(None, None, None)
# Inspired by discussions on https://bugs.python.org/issue29302
class AsyncExitStack(_BaseExitStack, AbstractAsyncContextManager):
"""Async context manager for dynamic management of a stack of exit
callbacks.
For example:
async with AsyncExitStack() as stack:
connections = [await stack.enter_async_context(get_connection())
for i in range(5)]
# All opened connections will automatically be released at the
# end of the async with statement, even if attempts to open a
# connection later in the list raise an exception.
"""
@staticmethod
def _create_async_exit_wrapper(cm, cm_exit):
return MethodType(cm_exit, cm)
@staticmethod
def _create_async_cb_wrapper(callback, /, *args, **kwds):
async def _exit_wrapper(exc_type, exc, tb):
await callback(*args, **kwds)
return _exit_wrapper
async def enter_async_context(self, cm):
"""Enters the supplied async context manager.
If successful, also pushes its __aexit__ method as a callback and
returns the result of the __aenter__ method.
"""
_cm_type = type(cm)
_exit = _cm_type.__aexit__
result = await _cm_type.__aenter__(cm)
self._push_async_cm_exit(cm, _exit)
return result
def push_async_exit(self, exit):
"""Registers a coroutine function with the standard __aexit__ method
signature.
Can suppress exceptions the same way __aexit__ method can.
Also accepts any object with an __aexit__ method (registering a call
to the method instead of the object itself).
"""
_cb_type = type(exit)
try:
exit_method = _cb_type.__aexit__
except AttributeError:
# Not an async context manager, so assume it's a coroutine function
self._push_exit_callback(exit, False)
else:
self._push_async_cm_exit(exit, exit_method)
return exit # Allow use as a decorator
def push_async_callback(*args, **kwds):
"""Registers an arbitrary coroutine function and arguments.
Cannot suppress exceptions.
"""
if len(args) >= 2:
self, callback, *args = args
elif not args:
raise TypeError("descriptor 'push_async_callback' of "
"'AsyncExitStack' object needs an argument")
elif 'callback' in kwds:
callback = kwds.pop('callback')
self, *args = args
import warnings
warnings.warn("Passing 'callback' as keyword argument is deprecated",
DeprecationWarning, stacklevel=2)
else:
raise TypeError('push_async_callback expected at least 1 '
'positional argument, got %d' % (len(args)-1))
_exit_wrapper = self._create_async_cb_wrapper(callback, *args, **kwds)
# We changed the signature, so using @wraps is not appropriate, but
# setting __wrapped__ may still help with introspection.
_exit_wrapper.__wrapped__ = callback
self._push_exit_callback(_exit_wrapper, False)
return callback # Allow use as a decorator
push_async_callback.__text_signature__ = '($self, callback, /, *args, **kwds)'
async def aclose(self):
"""Immediately unwind the context stack."""
await self.__aexit__(None, None, None)
def _push_async_cm_exit(self, cm, cm_exit):
"""Helper to correctly register coroutine function to __aexit__
method."""
_exit_wrapper = self._create_async_exit_wrapper(cm, cm_exit)
self._push_exit_callback(_exit_wrapper, False)
async def __aenter__(self):
return self
async def __aexit__(self, *exc_details):
received_exc = exc_details[0] is not None
# We manipulate the exception state so it behaves as though
# we were actually nesting multiple with statements
frame_exc = sys.exc_info()[1]
def _fix_exception_context(new_exc, old_exc):
# Context may not be correct, so find the end of the chain
while 1:
exc_context = new_exc.__context__
if exc_context is old_exc:
# Context is already set correctly (see issue 20317)
return
if exc_context is None or exc_context is frame_exc:
break
new_exc = exc_context
# Change the end of the chain to point to the exception
# we expect it to reference
new_exc.__context__ = old_exc
# Callbacks are invoked in LIFO order to match the behaviour of
# nested context managers
suppressed_exc = False
pending_raise = False
while self._exit_callbacks:
is_sync, cb = self._exit_callbacks.pop()
try:
if is_sync:
cb_suppress = cb(*exc_details)
else:
cb_suppress = await cb(*exc_details)
if cb_suppress:
suppressed_exc = True
pending_raise = False
exc_details = (None, None, None)
except:
new_exc_details = sys.exc_info()
# simulate the stack of exceptions by setting the context
_fix_exception_context(new_exc_details[1], exc_details[1])
pending_raise = True
exc_details = new_exc_details
if pending_raise:
try:
# bare "raise exc_details[1]" replaces our carefully
# set-up context
fixed_ctx = exc_details[1].__context__
raise exc_details[1]
except BaseException:
exc_details[1].__context__ = fixed_ctx
raise
return received_exc and suppressed_exc
class nullcontext(AbstractContextManager):
"""Context manager that does no additional processing.
Used as a stand-in for a normal context manager, when a particular
block of code is only sometimes used with a normal context manager:
cm = optional_cm if condition else nullcontext()
with cm:
# Perform operation, using optional_cm if condition is True
"""
def __init__(self, enter_result=None):
self.enter_result = enter_result
def __enter__(self):
return self.enter_result
def __exit__(self, *excinfo):
pass
SILENT KILLER Tool