mirror of https://github.com/devine-dl/devine.git
Aria2c: Improve download progress and error handling
This commit is contained in:
parent
e8b07bf03a
commit
4e12b867f1
|
@ -17,28 +17,33 @@ from rich.text import Text
|
||||||
|
|
||||||
from devine.core.config import config
|
from devine.core.config import config
|
||||||
from devine.core.console import console
|
from devine.core.console import console
|
||||||
|
from devine.core.constants import DOWNLOAD_CANCELLED
|
||||||
from devine.core.utilities import get_binary_path, get_free_port
|
from devine.core.utilities import get_binary_path, get_free_port
|
||||||
|
|
||||||
|
|
||||||
def rpc(caller: Callable, secret: str, method: str, *params: Any) -> dict[str, Any]:
|
def rpc(caller: Callable, secret: str, method: str, params: Optional[list[Any]] = None) -> Any:
|
||||||
"""Make a call to Aria2's JSON-RPC API."""
|
"""Make a call to Aria2's JSON-RPC API."""
|
||||||
rpc_res = caller(
|
try:
|
||||||
json={
|
rpc_res = caller(
|
||||||
"jsonrpc": "2.0",
|
json={
|
||||||
"id": get_random_bytes(16).hex(),
|
"jsonrpc": "2.0",
|
||||||
"method": method,
|
"id": get_random_bytes(16).hex(),
|
||||||
"params": [f"token:{secret}", *params]
|
"method": method,
|
||||||
}
|
"params": [f"token:{secret}", *(params or [])]
|
||||||
).json()
|
}
|
||||||
if rpc_res.get("code"):
|
).json()
|
||||||
# wrap to console width - padding - '[Aria2c]: '
|
if rpc_res.get("code"):
|
||||||
error_pretty = "\n ".join(textwrap.wrap(
|
# wrap to console width - padding - '[Aria2c]: '
|
||||||
f"RPC Error: {rpc_res['message']} ({rpc_res['code']})".strip(),
|
error_pretty = "\n ".join(textwrap.wrap(
|
||||||
width=console.width - 20,
|
f"RPC Error: {rpc_res['message']} ({rpc_res['code']})".strip(),
|
||||||
initial_indent=""
|
width=console.width - 20,
|
||||||
))
|
initial_indent=""
|
||||||
console.log(Text.from_ansi("\n[Aria2c]: " + error_pretty))
|
))
|
||||||
return rpc_res["result"]
|
console.log(Text.from_ansi("\n[Aria2c]: " + error_pretty))
|
||||||
|
return rpc_res["result"]
|
||||||
|
except requests.exceptions.ConnectionError:
|
||||||
|
# absorb, process likely ended as it was calling RPC
|
||||||
|
return
|
||||||
|
|
||||||
|
|
||||||
def download(
|
def download(
|
||||||
|
@ -176,6 +181,8 @@ def download(
|
||||||
continue
|
continue
|
||||||
arguments.extend(["--header", f"{header}: {value}"])
|
arguments.extend(["--header", f"{header}: {value}"])
|
||||||
|
|
||||||
|
yield dict(total=len(urls))
|
||||||
|
|
||||||
try:
|
try:
|
||||||
p = subprocess.Popen(
|
p = subprocess.Popen(
|
||||||
[
|
[
|
||||||
|
@ -190,28 +197,50 @@ def download(
|
||||||
p.stdin.close()
|
p.stdin.close()
|
||||||
|
|
||||||
while p.poll() is None:
|
while p.poll() is None:
|
||||||
global_stats = rpc(
|
global_stats: dict[str, Any] = rpc(
|
||||||
caller=partial(rpc_session.post, url=rpc_uri),
|
caller=partial(rpc_session.post, url=rpc_uri),
|
||||||
secret=rpc_secret,
|
secret=rpc_secret,
|
||||||
method="aria2.getGlobalStat"
|
method="aria2.getGlobalStat"
|
||||||
)
|
)
|
||||||
if global_stats:
|
if global_stats:
|
||||||
active = int(global_stats["numActive"])
|
|
||||||
waiting = int(global_stats["numWaiting"])
|
|
||||||
stopped = int(global_stats["numStopped"])
|
|
||||||
total = active + waiting + stopped
|
|
||||||
yield dict(
|
yield dict(
|
||||||
total=total,
|
|
||||||
completed=stopped,
|
|
||||||
downloaded=f"{filesize.decimal(int(global_stats['downloadSpeed']))}/s"
|
downloaded=f"{filesize.decimal(int(global_stats['downloadSpeed']))}/s"
|
||||||
)
|
)
|
||||||
if total == stopped:
|
|
||||||
rpc(
|
stopped_downloads: list[dict[str, Any]] = rpc(
|
||||||
caller=partial(rpc_session.post, url=rpc_uri),
|
caller=partial(rpc_session.post, url=rpc_uri),
|
||||||
secret=rpc_secret,
|
secret=rpc_secret,
|
||||||
method="aria2.shutdown"
|
method="aria2.tellStopped",
|
||||||
|
params=[0, 999999]
|
||||||
|
)
|
||||||
|
for dl in stopped_downloads or []:
|
||||||
|
if dl["status"] == "complete":
|
||||||
|
yield dict(advance=1)
|
||||||
|
elif dl["status"] == "error":
|
||||||
|
used_uri = next(
|
||||||
|
uri["uri"]
|
||||||
|
for file in dl["files"]
|
||||||
|
if file["selected"] == "true"
|
||||||
|
for uri in file["uris"]
|
||||||
|
if uri["status"] == "used"
|
||||||
)
|
)
|
||||||
break
|
error = f"Download Error (#{dl['gid']}): {dl['errorMessage']} ({dl['errorCode']}), {used_uri}"
|
||||||
|
error_pretty = "\n ".join(textwrap.wrap(
|
||||||
|
error,
|
||||||
|
width=console.width - 20,
|
||||||
|
initial_indent=""
|
||||||
|
))
|
||||||
|
console.log(Text.from_ansi("\n[Aria2c]: " + error_pretty))
|
||||||
|
raise ValueError(error)
|
||||||
|
|
||||||
|
if len(stopped_downloads) == len(urls):
|
||||||
|
rpc(
|
||||||
|
caller=partial(rpc_session.post, url=rpc_uri),
|
||||||
|
secret=rpc_secret,
|
||||||
|
method="aria2.shutdown"
|
||||||
|
)
|
||||||
|
break
|
||||||
|
|
||||||
time.sleep(1)
|
time.sleep(1)
|
||||||
|
|
||||||
p.wait()
|
p.wait()
|
||||||
|
@ -227,6 +256,20 @@ def download(
|
||||||
# 0xC000013A is when it never got the chance to
|
# 0xC000013A is when it never got the chance to
|
||||||
raise KeyboardInterrupt()
|
raise KeyboardInterrupt()
|
||||||
raise
|
raise
|
||||||
|
except KeyboardInterrupt:
|
||||||
|
DOWNLOAD_CANCELLED.set() # skip pending track downloads
|
||||||
|
yield dict(downloaded="[yellow]CANCELLED")
|
||||||
|
raise
|
||||||
|
except Exception:
|
||||||
|
DOWNLOAD_CANCELLED.set() # skip pending track downloads
|
||||||
|
yield dict(downloaded="[red]FAILED")
|
||||||
|
raise
|
||||||
|
finally:
|
||||||
|
rpc(
|
||||||
|
caller=partial(rpc_session.post, url=rpc_uri),
|
||||||
|
secret=rpc_secret,
|
||||||
|
method="aria2.shutdown"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
def aria2c(
|
def aria2c(
|
||||||
|
|
Loading…
Reference in New Issue