1
+ from asyncio import CancelledError
1
2
from pathlib import Path
2
3
from urllib .parse import urlparse
3
4
7
8
8
9
try :
9
10
import pyodide_js
10
- from js import Object
11
+ from js import AbortController , Object
11
12
from pyodide_js import loadedPackages , loadPackage
12
13
from pyodide_js ._api import ( # type: ignore[import]
13
14
loadBinaryFile ,
21
22
raise
22
23
# Otherwise, this is pytest test collection so let it go.
23
24
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
+
24
46
25
47
async def fetch_bytes (url : str , kwargs : dict [str , str ]) -> bytes :
26
48
parsed_url = urlparse (url )
@@ -29,13 +51,13 @@ async def fetch_bytes(url: str, kwargs: dict[str, str]) -> bytes:
29
51
if parsed_url .scheme == "file" :
30
52
return (await loadBinaryFile (parsed_url .path )).to_bytes ()
31
53
32
- return await (await pyfetch (url , ** kwargs )).bytes ()
54
+ return await (await _pyfetch (url , ** kwargs )).bytes ()
33
55
34
56
35
57
async def fetch_string_and_headers (
36
58
url : str , kwargs : dict [str , str ]
37
59
) -> tuple [str , dict [str , str ]]:
38
- response = await pyfetch (url , ** kwargs )
60
+ response = await _pyfetch (url , ** kwargs )
39
61
40
62
content = await response .string ()
41
63
# TODO: replace with response.headers when pyodide>= 0.24 is released
0 commit comments