import os
import time
from concurrent.futures import ThreadPoolExecutor, as_completed
import subprocess, sys
def download_segment(CACHE_FOLDER, url: str, order: int, name_prefix: str = ""):
print(f"Downloading {name_prefix} - {order}")
video_download_command = (f".\\aria2c -o '{CACHE_FOLDER}/{name_prefix}-{order}.mp4'"
f" -x 16 -s 16 '{url}' -c -l aria2c_video.log --log-level warn")
result = subprocess.run(['powershell', '-Command', video_download_command], text=True)
return result
def download_segment_m3u8(CACHE_FOLDER, url: str, order: int, name_prefix: str = "", max_retries: int = 35):
print(f"Downloading {name_prefix} - {order}")
print(f"Downloading from {url}")
# Initial download attempt with 32 workers
# video_download_command = (f".\\ffmpeg -i '{url}' -c copy -n '{CACHE_FOLDER}/{name_prefix}-{order}.mp4'"
# f" -hide_banner -loglevel error -stats")
# video_download_command = (f".\\HLSDownloader -u '{url}' -o '{CACHE_FOLDER}/{name_prefix}-{order}.mp4'"
# f" -w 32 -workers 32 ")
# video_download_command = (f".\\m3u8dl-windows-amd64.exe -i '{url}' -o '{CACHE_FOLDER}/{name_prefix}-{order}.mp4'"
# f" -retry 1 -t '{CACHE_FOLDER}' -thread 32")
# video_download_command = (
# f".\\vsd.exe save '{url}' -o '{CACHE_FOLDER}/{name_prefix}-{order}.mp4' "
# f"-d {CACHE_FOLDER} --retry-count 15 -t 4")
output_path = f"{CACHE_FOLDER}/{name_prefix}-{order}"
save_dir = os.path.dirname(output_path)
save_name = os.path.basename(output_path)
if 'mp3' in url:
video_download_command = (
f".\\ffmpeg -i '{url}' -c:v copy -c:a copy -n '{CACHE_FOLDER}/{name_prefix}-{order}.mp4' "
f"-hide_banner -loglevel error -stats"
)
else:
video_download_command = (
f".\\N_m3u8DL-RE.exe '{url}' --tmp-dir './{CACHE_FOLDER}' "
f"--save-dir '{save_dir}' --save-name '{save_name}' -M format=mp4 "
f"--check-segments-count false --download-retry-count 15 --thread-count 64"
)
result = subprocess.run(['powershell', '-Command', video_download_command], text=True)
return result
def download_segments_in_parallel(fallback_flag, CACHE_FOLDER, lesson_video_data, name_prefix):
has_error = False
if fallback_flag:
# Create a ThreadPoolExecutor to manage parallel downloads
with ThreadPoolExecutor(max_workers=3) as executor:
# Dictionary to hold future results
future_to_order = {}
for order, segment in enumerate(lesson_video_data['data']['live_timeline']):
replay_url = segment['replay_url']
# Determine which function to use based on the presence of 'm3u8' in the replay_url
if 'm3u8' in replay_url:
future = executor.submit(download_segment_m3u8, CACHE_FOLDER, replay_url, order, name_prefix,
max_retries=10)
else:
future = executor.submit(download_segment, CACHE_FOLDER, replay_url, order, name_prefix)
# Store the future and order for tracking
future_to_order[future] = order
# Iterate over the completed futures
for future in as_completed(future_to_order):
order = future_to_order[future]
try:
future.result() # Get the result (will raise exception if there was one)
print(f"Successfully downloaded {name_prefix} - {order}")
except Exception as e:
print(e)
print(f"Failed to download {name_prefix} - {order}", file=sys.stderr)
has_error = True
else:
# Create a ThreadPoolExecutor to manage parallel downloads
with ThreadPoolExecutor(max_workers=3) as executor:
# Dictionary to hold future results
future_to_order = {}
for order, segment in enumerate(lesson_video_data['data']['live']):
url = segment['url']
# Determine which function to use based on the presence of 'm3u8' in the URL
if 'm3u8' in url:
future = executor.submit(download_segment_m3u8, CACHE_FOLDER, url, order, name_prefix,
max_retries=10)
else:
future = executor.submit(download_segment, CACHE_FOLDER, url, order, name_prefix)
# Store the future and order for tracking
future_to_order[future] = order
# Iterate over the completed futures
for future in as_completed(future_to_order):
order = future_to_order[future]
try:
future.result() # Get the result (will raise exception if there was one)
print(f"Successfully downloaded {name_prefix} - {order}")
except Exception as e:
print(e)
print(f"Failed to download {name_prefix} - {order}", file=sys.stderr)
has_error = True
return has_error
def concatenate_segments(CACHE_FOLDER, DOWNLOAD_FOLDER, name_prefix, num_segments):
# Create the concat file with segment paths
with open(f"{CACHE_FOLDER}/concat.txt", "w", encoding='utf-8') as f:
for i in range(num_segments):
video_file_mp4 = f"../{CACHE_FOLDER}/{name_prefix}-{i}.mp4"
video_file_ts = f"../{CACHE_FOLDER}/{name_prefix}-{i}.ts"
if os.path.exists(os.path.join(CACHE_FOLDER, f"{name_prefix}-{i}.mp4")): # Check if the file exists
f.write(f"file '{video_file_mp4}'\n")
if os.path.exists(os.path.join(CACHE_FOLDER, f"{name_prefix}-{i}.ts")): # Check if the file exists
f.write(f"file '{video_file_ts}'\n")
# First video concatenation command using CUDA acceleration
video_concatenating_command = (
f"ffmpeg -f concat -safe 0 -hwaccel cuda -hwaccel_output_format cuda "
f"-i '{CACHE_FOLDER}/concat.txt' "
f"-c:v hevc_nvenc -cq 28 -surfaces 64 -bufsize 12800k -r 7.5 -rc-lookahead 63 "
f"-c:a aac -ac 1 -rematrix_maxval 1.0 -b:a 64k '{DOWNLOAD_FOLDER}/{name_prefix}.mp4' -n "
f"-hide_banner -loglevel error -stats"
)
# Run the first command
result = subprocess.run(['powershell', '-Command', video_concatenating_command], text=True)
# If the first command fails, try the fallback
if result.returncode != 0:
print(f"First attempt failed. Attempting fallback with software encoding.")
# Fallback video concatenation command using cuvid acceleration
video_concatenating_command_fallback = (
f"ffmpeg -f concat -safe 0 "
f"-i '{CACHE_FOLDER}/concat.txt' "
f"-c:v hevc_nvenc -cq 28 -surfaces 64 -bufsize 12800k -r 7.5 -rc-lookahead 63 "
f"-c:a copy '{DOWNLOAD_FOLDER}/{name_prefix}.mp4' -y "
f"-hide_banner -loglevel error -stats -err_detect ignore_err -fflags +discardcorrupt"
)
# Run the fallback command
fallback_result = subprocess.run(['powershell', '-Command', video_concatenating_command_fallback], text=True)
# Check if the fallback also fails
if fallback_result.returncode != 0:
print(f"Both attempts failed to concatenate video segments.")
else:
print(f"Successfully concatenated video segments.")
else:
print(f"Successfully concatenated video segments using CUDA acceleration.")
return result