Shutdown HLS & DASH dl pool, pass exceptions to dl

This results in a noticeably faster speed cancelling segmented track downloads on CTRL+C and Errors. It's also reducing code duplication as the dl code will now handle the exception and cleanup for them.

This also simplifies the STOPPING/STOPPED and FAILING/FAILED status messages by quite a bit.
This commit is contained in:
rlaphoenix 2023-03-01 10:45:04 +00:00
parent fbe78308eb
commit 9f48aab80c
2 changed files with 72 additions and 98 deletions

View File

@ -4,10 +4,8 @@ import base64
import logging import logging
import math import math
import re import re
import shutil
import sys import sys
import time import time
import traceback
from concurrent import futures from concurrent import futures
from concurrent.futures import ThreadPoolExecutor from concurrent.futures import ThreadPoolExecutor
from copy import copy from copy import copy
@ -450,7 +448,8 @@ class DASH:
def download_segment(filename: str, segment: tuple[str, Optional[str]]) -> int: def download_segment(filename: str, segment: tuple[str, Optional[str]]) -> int:
if stop_event.is_set(): if stop_event.is_set():
return 0 # the track already started downloading, but another failed or was stopped
raise KeyboardInterrupt()
segment_save_path = (save_dir / filename).with_suffix(".mp4") segment_save_path = (save_dir / filename).with_suffix(".mp4")
@ -507,8 +506,6 @@ class DASH:
progress(total=len(segments)) progress(total=len(segments))
finished_threads = 0 finished_threads = 0
has_stopped = False
has_failed = False
download_sizes = [] download_sizes = []
last_speed_refresh = time.time() last_speed_refresh = time.time()
@ -522,28 +519,27 @@ class DASH:
for i, segment in enumerate(segments) for i, segment in enumerate(segments)
)): )):
finished_threads += 1 finished_threads += 1
try: try:
download_size = download.result() download_size = download.result()
except KeyboardInterrupt: except KeyboardInterrupt:
stop_event.set() stop_event.set() # skip pending track downloads
progress(downloaded="[yellow]STOPPING")
pool.shutdown(wait=True, cancel_futures=True)
progress(downloaded="[yellow]STOPPED")
# tell dl that it was cancelled
# the pool is already shut down, so exiting loop is fine
raise
except Exception as e: except Exception as e:
stop_event.set() stop_event.set() # skip pending track downloads
if has_stopped:
# we don't care because we were stopping anyway
continue
if not has_failed:
has_failed = True
progress(downloaded="[red]FAILING") progress(downloaded="[red]FAILING")
traceback.print_exception(e) pool.shutdown(wait=True, cancel_futures=True)
log.error(f"Segment Download worker threw an unhandled exception: {e!r}") progress(downloaded="[red]FAILED")
continue # tell dl that it failed
# the pool is already shut down, so exiting loop is fine
if stop_event.is_set(): raise e
if not has_stopped: else:
has_stopped = True # it successfully downloaded, and it was not cancelled
progress(downloaded="[orange]STOPPING")
continue
progress(advance=1) progress(advance=1)
now = time.time() now = time.time()
@ -559,22 +555,13 @@ class DASH:
last_speed_refresh = now last_speed_refresh = now
download_sizes.clear() download_sizes.clear()
try:
if has_stopped:
progress(downloaded="[yellow]STOPPED")
return
if has_failed:
progress(downloaded="[red]FAILED")
return
with open(save_path, "wb") as f: with open(save_path, "wb") as f:
for segment_file in sorted(save_dir.iterdir()): for segment_file in sorted(save_dir.iterdir()):
f.write(segment_file.read_bytes()) f.write(segment_file.read_bytes())
segment_file.unlink() segment_file.unlink()
track.path = save_path track.path = save_path
finally: save_dir.rmdir()
shutil.rmtree(save_dir)
@staticmethod @staticmethod
def get_language(*options: Any) -> Optional[Language]: def get_language(*options: Any) -> Optional[Language]:

View File

@ -2,10 +2,8 @@ from __future__ import annotations
import logging import logging
import re import re
import shutil
import sys import sys
import time import time
import traceback
from concurrent import futures from concurrent import futures
from concurrent.futures import ThreadPoolExecutor from concurrent.futures import ThreadPoolExecutor
from functools import partial from functools import partial
@ -217,7 +215,8 @@ class HLS:
def download_segment(filename: str, segment: m3u8.Segment, init_data: Queue, segment_key: Queue) -> int: def download_segment(filename: str, segment: m3u8.Segment, init_data: Queue, segment_key: Queue) -> int:
if stop_event.is_set(): if stop_event.is_set():
return 0 # the track already started downloading, but another failed or was stopped
raise KeyboardInterrupt()
segment_save_path = (save_dir / filename).with_suffix(".mp4") segment_save_path = (save_dir / filename).with_suffix(".mp4")
@ -345,8 +344,6 @@ class HLS:
progress(total=len(master.segments)) progress(total=len(master.segments))
finished_threads = 0 finished_threads = 0
has_stopped = False
has_failed = False
download_sizes = [] download_sizes = []
last_speed_refresh = time.time() last_speed_refresh = time.time()
@ -362,28 +359,27 @@ class HLS:
for i, segment in enumerate(master.segments) for i, segment in enumerate(master.segments)
)): )):
finished_threads += 1 finished_threads += 1
try: try:
download_size = download.result() download_size = download.result()
except KeyboardInterrupt: except KeyboardInterrupt:
stop_event.set() stop_event.set() # skip pending track downloads
progress(downloaded="[yellow]STOPPING")
pool.shutdown(wait=True, cancel_futures=True)
progress(downloaded="[yellow]STOPPED")
# tell dl that it was cancelled
# the pool is already shut down, so exiting loop is fine
raise
except Exception as e: except Exception as e:
stop_event.set() stop_event.set() # skip pending track downloads
if has_stopped:
# we don't care because we were stopping anyway
continue
if not has_failed:
has_failed = True
progress(downloaded="[red]FAILING") progress(downloaded="[red]FAILING")
traceback.print_exception(e) pool.shutdown(wait=True, cancel_futures=True)
log.error(f"Segment Download worker threw an unhandled exception: {e!r}") progress(downloaded="[red]FAILED")
continue # tell dl that it failed
# the pool is already shut down, so exiting loop is fine
if stop_event.is_set(): raise e
if not has_stopped: else:
has_stopped = True # it successfully downloaded, and it was not cancelled
progress(downloaded="[orange]STOPPING")
continue
progress(advance=1) progress(advance=1)
now = time.time() now = time.time()
@ -399,22 +395,13 @@ class HLS:
last_speed_refresh = now last_speed_refresh = now
download_sizes.clear() download_sizes.clear()
try:
if has_stopped:
progress(downloaded="[yellow]STOPPED")
return
if has_failed:
progress(downloaded="[red]FAILED")
return
with open(save_path, "wb") as f: with open(save_path, "wb") as f:
for segment_file in sorted(save_dir.iterdir()): for segment_file in sorted(save_dir.iterdir()):
f.write(segment_file.read_bytes()) f.write(segment_file.read_bytes())
segment_file.unlink() segment_file.unlink()
track.path = save_path track.path = save_path
finally: save_dir.rmdir()
shutil.rmtree(save_dir)
@staticmethod @staticmethod
def get_drm( def get_drm(