Skip to content

Commit 703fc54

Browse files
Abort other fetches when resolution fails
1 parent 7ad2963 commit 703fc54

File tree

2 files changed

+36
-4
lines changed

2 files changed

+36
-4
lines changed

micropip/_compat_in_pyodide.py

+25-3
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
from asyncio import CancelledError
12
from pathlib import Path
23
from urllib.parse import urlparse
34

@@ -7,7 +8,7 @@
78

89
try:
910
import pyodide_js
10-
from js import Object
11+
from js import AbortController, Object
1112
from pyodide_js import loadedPackages, loadPackage
1213
from pyodide_js._api import ( # type: ignore[import]
1314
loadBinaryFile,
@@ -21,6 +22,27 @@
2122
raise
2223
# Otherwise, this is pytest test collection so let it go.
2324

25+
if IN_BROWSER:
26+
27+
async def _pyfetch(urls: str, **kwargs):
28+
if "signal" in kwargs:
29+
return await pyfetch(urls, **kwargs)
30+
31+
controller = AbortController.new()
32+
kwargs["signal"] = controller.signal
33+
34+
async def fetch_with_abort():
35+
try:
36+
return await pyfetch(urls, **kwargs)
37+
except CancelledError:
38+
controller.abort()
39+
raise
40+
41+
return await fetch_with_abort()
42+
43+
else:
44+
_pyfetch = pyfetch
45+
2446

2547
async def fetch_bytes(url: str, kwargs: dict[str, str]) -> bytes:
2648
parsed_url = urlparse(url)
@@ -29,13 +51,13 @@ async def fetch_bytes(url: str, kwargs: dict[str, str]) -> bytes:
2951
if parsed_url.scheme == "file":
3052
return (await loadBinaryFile(parsed_url.path)).to_bytes()
3153

32-
return await (await pyfetch(url, **kwargs)).bytes()
54+
return await (await _pyfetch(url, **kwargs)).bytes()
3355

3456

3557
async def fetch_string_and_headers(
3658
url: str, kwargs: dict[str, str]
3759
) -> tuple[str, dict[str, str]]:
38-
response = await pyfetch(url, **kwargs)
60+
response = await _pyfetch(url, **kwargs)
3961

4062
content = await response.string()
4163
# TODO: replace with response.headers when pyodide>= 0.24 is released

micropip/transaction.py

+11-1
Original file line numberDiff line numberDiff line change
@@ -52,7 +52,17 @@ async def gather_requirements(
5252
for requirement in requirements:
5353
requirement_promises.append(self.add_requirement(requirement))
5454

55-
await asyncio.gather(*requirement_promises)
55+
futures: list[asyncio.Future] = []
56+
try:
57+
for coro in requirement_promises:
58+
futures.append(asyncio.ensure_future(coro))
59+
await asyncio.gather(*futures)
60+
except ValueError:
61+
if not self.keep_going:
62+
for future in futures:
63+
if not future.done():
64+
future.cancel()
65+
raise
5666

5767
async def add_requirement(self, req: str | Requirement) -> None:
5868
if isinstance(req, Requirement):

0 commit comments

Comments
 (0)