tubesync/tubesync/sync/hooks.py
2025-01-30 20:51:22 -05:00

104 lines
3.4 KiB
Python

import os
import yt_dlp
from common.logger import log
from django.conf import settings
class ProgressHookStatus:
valid = frozenset((
'downloading',
'finished',
'error',
))
def __init__(self):
self.download_progress = 0
class PPHookStatus:
valid = frozenset((
'started',
'processing',
'finished',
))
def __init__(self, *args, status=None, postprocessor=None, info_dict={}, **kwargs):
self.info = info_dict
self.name = postprocessor
self.status = status
def yt_dlp_progress_hook(event):
hook = progress_hook.get('status', None)
filename = os.path.basename(event['filename'])
if hook is None:
log.error('yt_dlp_progress_hook: failed to get hook status object')
return None
if event['status'] not in ProgressHookStatus.valid:
log.warn(f'[youtube-dl] unknown event: {str(event)}')
return None
if event.get('downloaded_bytes') is None or event.get('total_bytes') is None:
return None
if event['status'] == 'error':
log.error(f'[youtube-dl] error occured downloading: {filename}')
elif event['status'] == 'downloading':
downloaded_bytes = event.get('downloaded_bytes', 0)
total_bytes = event.get('total_bytes', 0)
eta = event.get('_eta_str', '?').strip()
percent_done = event.get('_percent_str', '?').strip()
speed = event.get('_speed_str', '?').strip()
total = event.get('_total_bytes_str', '?').strip()
if downloaded_bytes > 0 and total_bytes > 0:
p = round((event['downloaded_bytes'] / event['total_bytes']) * 100)
if (p % 5 == 0) and p > hook.download_progress:
hook.download_progress = p
log.info(f'[youtube-dl] downloading: {filename} - {percent_done} '
f'of {total} at {speed}, {eta} remaining')
else:
# No progress to monitor, just spam every 10 download messages instead
hook.download_progress += 1
if hook.download_progress % 10 == 0:
log.info(f'[youtube-dl] downloading: {filename} - {percent_done} '
f'of {total} at {speed}, {eta} remaining')
elif event['status'] == 'finished':
total_size_str = event.get('_total_bytes_str', '?').strip()
elapsed_str = event.get('_elapsed_str', '?').strip()
log.info(f'[youtube-dl] finished downloading: {filename} - '
f'{total_size_str} in {elapsed_str}')
def yt_dlp_postprocessor_hook(event):
if event['status'] not in PPHookStatus.valid:
log.warn(f'[youtube-dl] unknown event: {str(event)}')
return None
postprocessor_hook['status'] = PPHookStatus(*event)
if 'started' == event['status']:
if 'formats' in event['info_dict']:
del event['info_dict']['formats']
log.debug(repr(event['info_dict']))
key = 'Unknown'
name = key
if 'id' in event['info_dict']:
key = event['info_dict']['id']
title = None
if 'title' in event['info_dict']:
title = event['info_dict']['title']
name = f'{key}: {title}'
log.info(f'[{event["postprocessor"]}] {event["status"]} for: {name}')
progress_hook = {
'status': ProgressHookStatus(),
'function': yt_dlp_progress_hook,
}
postprocessor_hook = {
'status': PPHookStatus(),
'function': yt_dlp_postprocessor_hook,
}