mirror of
https://github.com/Grasscutters/mitmproxy.git
synced 2024-11-23 00:01:36 +00:00
add asyncio_utils
This commit is contained in:
parent
cfc180e211
commit
9b7dfb0fc9
65
mitmproxy/utils/asyncio_utils.py
Normal file
65
mitmproxy/utils/asyncio_utils.py
Normal file
@ -0,0 +1,65 @@
|
||||
import asyncio
|
||||
import sys
|
||||
import time
|
||||
from collections.abc import Coroutine
|
||||
from typing import Optional
|
||||
|
||||
from mitmproxy.utils import human
|
||||
|
||||
|
||||
def cancel_task(task: asyncio.Task, message: str) -> None:
|
||||
"""Like task.cancel(), but optionally with a message if the Python version supports it."""
|
||||
if sys.version_info >= (3, 9):
|
||||
task.cancel(message) # type: ignore
|
||||
else:
|
||||
task.cancel()
|
||||
|
||||
|
||||
def create_task(
|
||||
coro: Coroutine, *,
|
||||
name: str,
|
||||
client: Optional[tuple] = None,
|
||||
ignore_closed_loop: bool = True,
|
||||
) -> Optional[asyncio.Task]:
|
||||
"""
|
||||
Like asyncio.create_task, but also store some debug info on the task object.
|
||||
|
||||
If ignore_closed_loop is True, the task will be silently discarded if the event loop is closed.
|
||||
This is currently useful during shutdown where no new tasks can be spawned.
|
||||
Ideally we stop closing the event loop during shutdown and then remove this parameter.
|
||||
"""
|
||||
try:
|
||||
t = asyncio.create_task(coro, name=name)
|
||||
except RuntimeError:
|
||||
if ignore_closed_loop:
|
||||
coro.close()
|
||||
return None
|
||||
else:
|
||||
raise
|
||||
set_task_debug_info(t, name=name, client=client)
|
||||
return t
|
||||
|
||||
|
||||
def set_task_debug_info(
|
||||
task: asyncio.Task,
|
||||
*,
|
||||
name: str,
|
||||
client: Optional[tuple] = None,
|
||||
) -> None:
|
||||
"""Set debug info for an externally-spawned task."""
|
||||
task.created = time.time() # type: ignore
|
||||
task.set_name(name)
|
||||
if client:
|
||||
task.client = client # type: ignore
|
||||
|
||||
|
||||
def task_repr(task: asyncio.Task) -> str:
|
||||
"""Get a task representation with debug info."""
|
||||
name = task.get_name()
|
||||
age = getattr(task, "created", "")
|
||||
if age:
|
||||
age = f" (age: {time.time() - age:.0f}s)"
|
||||
client = getattr(task, "client", "")
|
||||
if client:
|
||||
client = f"{human.format_address(client)}: "
|
||||
return f"{client}{name}{age}"
|
46
test/mitmproxy/utils/test_asyncio_utils.py
Normal file
46
test/mitmproxy/utils/test_asyncio_utils.py
Normal file
@ -0,0 +1,46 @@
|
||||
import asyncio
|
||||
import sys
|
||||
|
||||
import pytest
|
||||
|
||||
from mitmproxy.utils import asyncio_utils
|
||||
|
||||
|
||||
async def ttask():
|
||||
asyncio_utils.set_task_debug_info(
|
||||
asyncio.current_task(),
|
||||
name="newname",
|
||||
)
|
||||
await asyncio.sleep(999)
|
||||
|
||||
|
||||
@pytest.mark.asyncio
|
||||
@pytest.mark.skipif(sys.version_info < (3, 8), reason="requires Python 3.8")
|
||||
async def test_simple():
|
||||
task = asyncio_utils.create_task(
|
||||
ttask(),
|
||||
name="ttask",
|
||||
client=("127.0.0.1", 42313)
|
||||
)
|
||||
assert asyncio_utils.task_repr(task) == "127.0.0.1:42313: ttask (age: 0s)"
|
||||
await asyncio.sleep(0)
|
||||
assert "newname" in asyncio_utils.task_repr(task)
|
||||
asyncio_utils.cancel_task(task, "bye")
|
||||
await asyncio.sleep(0)
|
||||
assert task.cancelled()
|
||||
|
||||
|
||||
@pytest.mark.skipif(sys.version_info < (3, 8), reason="requires Python 3.8")
|
||||
def test_closed_loop():
|
||||
# Crude test for line coverage.
|
||||
# This should eventually go, see the description in asyncio_utils.create_task for details.
|
||||
asyncio_utils.create_task(
|
||||
ttask(),
|
||||
name="ttask",
|
||||
)
|
||||
with pytest.raises(RuntimeError):
|
||||
asyncio_utils.create_task(
|
||||
ttask(),
|
||||
name="ttask",
|
||||
ignore_closed_loop=False,
|
||||
)
|
Loading…
Reference in New Issue
Block a user