diff options
author | shadchin <shadchin@yandex-team.ru> | 2022-02-10 16:44:30 +0300 |
---|---|---|
committer | Daniil Cherednik <dcherednik@yandex-team.ru> | 2022-02-10 16:44:30 +0300 |
commit | 2598ef1d0aee359b4b6d5fdd1758916d5907d04f (patch) | |
tree | 012bb94d777798f1f56ac1cec429509766d05181 /contrib/python/prompt-toolkit/py3/prompt_toolkit/eventloop/utils.py | |
parent | 6751af0b0c1b952fede40b19b71da8025b5d8bcf (diff) | |
download | ydb-2598ef1d0aee359b4b6d5fdd1758916d5907d04f.tar.gz |
Restoring authorship annotation for <shadchin@yandex-team.ru>. Commit 1 of 2.
Diffstat (limited to 'contrib/python/prompt-toolkit/py3/prompt_toolkit/eventloop/utils.py')
-rw-r--r-- | contrib/python/prompt-toolkit/py3/prompt_toolkit/eventloop/utils.py | 236 |
1 files changed, 118 insertions, 118 deletions
diff --git a/contrib/python/prompt-toolkit/py3/prompt_toolkit/eventloop/utils.py b/contrib/python/prompt-toolkit/py3/prompt_toolkit/eventloop/utils.py index 2e5a05e838..f9809e42b2 100644 --- a/contrib/python/prompt-toolkit/py3/prompt_toolkit/eventloop/utils.py +++ b/contrib/python/prompt-toolkit/py3/prompt_toolkit/eventloop/utils.py @@ -1,118 +1,118 @@ -import asyncio -import sys -import time -from types import TracebackType -from typing import Any, Awaitable, Callable, Dict, Optional, TypeVar, cast - -try: - import contextvars -except ImportError: - from . import dummy_contextvars as contextvars # type: ignore - -__all__ = [ - "run_in_executor_with_context", - "call_soon_threadsafe", - "get_traceback_from_context", - "get_event_loop", -] - -_T = TypeVar("_T") - - -def run_in_executor_with_context( - func: Callable[..., _T], - *args: Any, - loop: Optional[asyncio.AbstractEventLoop] = None, -) -> Awaitable[_T]: - """ - Run a function in an executor, but make sure it uses the same contextvars. - This is required so that the function will see the right application. - - See also: https://bugs.python.org/issue34014 - """ - loop = loop or get_event_loop() - ctx: contextvars.Context = contextvars.copy_context() - - return loop.run_in_executor(None, ctx.run, func, *args) - - -def call_soon_threadsafe( - func: Callable[[], None], - max_postpone_time: Optional[float] = None, - loop: Optional[asyncio.AbstractEventLoop] = None, -) -> None: - """ - Wrapper around asyncio's `call_soon_threadsafe`. - - This takes a `max_postpone_time` which can be used to tune the urgency of - the method. - - Asyncio runs tasks in first-in-first-out. However, this is not what we - want for the render function of the prompt_toolkit UI. Rendering is - expensive, but since the UI is invalidated very often, in some situations - we render the UI too often, so much that the rendering CPU usage slows down - the rest of the processing of the application. (Pymux is an example where - we have to balance the CPU time spend on rendering the UI, and parsing - process output.) - However, we want to set a deadline value, for when the rendering should - happen. (The UI should stay responsive). - """ - loop2 = loop or get_event_loop() - - # If no `max_postpone_time` has been given, schedule right now. - if max_postpone_time is None: - loop2.call_soon_threadsafe(func) - return - - max_postpone_until = time.time() + max_postpone_time - - def schedule() -> None: - # When there are no other tasks scheduled in the event loop. Run it - # now. - # Notice: uvloop doesn't have this _ready attribute. In that case, - # always call immediately. - if not getattr(loop2, "_ready", []): - func() - return - - # If the timeout expired, run this now. - if time.time() > max_postpone_until: - func() - return - - # Schedule again for later. - loop2.call_soon_threadsafe(schedule) - - loop2.call_soon_threadsafe(schedule) - - -def get_traceback_from_context(context: Dict[str, Any]) -> Optional[TracebackType]: - """ - Get the traceback object from the context. - """ - exception = context.get("exception") - if exception: - if hasattr(exception, "__traceback__"): - return cast(TracebackType, exception.__traceback__) - else: - # call_exception_handler() is usually called indirectly - # from an except block. If it's not the case, the traceback - # is undefined... - return sys.exc_info()[2] - - return None - - -def get_event_loop() -> asyncio.AbstractEventLoop: - """Backward compatible way to get the event loop""" - # Python 3.6 doesn't have get_running_loop - # Python 3.10 deprecated get_event_loop - if sys.version_info >= (3, 7): - getloop = asyncio.get_running_loop - else: - getloop = asyncio.get_event_loop - - try: - return getloop() - except RuntimeError: - return asyncio.get_event_loop_policy().get_event_loop() +import asyncio +import sys +import time +from types import TracebackType +from typing import Any, Awaitable, Callable, Dict, Optional, TypeVar, cast + +try: + import contextvars +except ImportError: + from . import dummy_contextvars as contextvars # type: ignore + +__all__ = [ + "run_in_executor_with_context", + "call_soon_threadsafe", + "get_traceback_from_context", + "get_event_loop", +] + +_T = TypeVar("_T") + + +def run_in_executor_with_context( + func: Callable[..., _T], + *args: Any, + loop: Optional[asyncio.AbstractEventLoop] = None, +) -> Awaitable[_T]: + """ + Run a function in an executor, but make sure it uses the same contextvars. + This is required so that the function will see the right application. + + See also: https://bugs.python.org/issue34014 + """ + loop = loop or get_event_loop() + ctx: contextvars.Context = contextvars.copy_context() + + return loop.run_in_executor(None, ctx.run, func, *args) + + +def call_soon_threadsafe( + func: Callable[[], None], + max_postpone_time: Optional[float] = None, + loop: Optional[asyncio.AbstractEventLoop] = None, +) -> None: + """ + Wrapper around asyncio's `call_soon_threadsafe`. + + This takes a `max_postpone_time` which can be used to tune the urgency of + the method. + + Asyncio runs tasks in first-in-first-out. However, this is not what we + want for the render function of the prompt_toolkit UI. Rendering is + expensive, but since the UI is invalidated very often, in some situations + we render the UI too often, so much that the rendering CPU usage slows down + the rest of the processing of the application. (Pymux is an example where + we have to balance the CPU time spend on rendering the UI, and parsing + process output.) + However, we want to set a deadline value, for when the rendering should + happen. (The UI should stay responsive). + """ + loop2 = loop or get_event_loop() + + # If no `max_postpone_time` has been given, schedule right now. + if max_postpone_time is None: + loop2.call_soon_threadsafe(func) + return + + max_postpone_until = time.time() + max_postpone_time + + def schedule() -> None: + # When there are no other tasks scheduled in the event loop. Run it + # now. + # Notice: uvloop doesn't have this _ready attribute. In that case, + # always call immediately. + if not getattr(loop2, "_ready", []): + func() + return + + # If the timeout expired, run this now. + if time.time() > max_postpone_until: + func() + return + + # Schedule again for later. + loop2.call_soon_threadsafe(schedule) + + loop2.call_soon_threadsafe(schedule) + + +def get_traceback_from_context(context: Dict[str, Any]) -> Optional[TracebackType]: + """ + Get the traceback object from the context. + """ + exception = context.get("exception") + if exception: + if hasattr(exception, "__traceback__"): + return cast(TracebackType, exception.__traceback__) + else: + # call_exception_handler() is usually called indirectly + # from an except block. If it's not the case, the traceback + # is undefined... + return sys.exc_info()[2] + + return None + + +def get_event_loop() -> asyncio.AbstractEventLoop: + """Backward compatible way to get the event loop""" + # Python 3.6 doesn't have get_running_loop + # Python 3.10 deprecated get_event_loop + if sys.version_info >= (3, 7): + getloop = asyncio.get_running_loop + else: + getloop = asyncio.get_event_loop + + try: + return getloop() + except RuntimeError: + return asyncio.get_event_loop_policy().get_event_loop() |