Improve YouTube cookies handling and script

This commit is contained in:
vrubel 2025-12-17 17:53:34 +03:00
parent 1e7f3be3f3
commit fdaaddff98
3 changed files with 126 additions and 22 deletions

View file

@ -32,29 +32,48 @@ def _safe_filename(title: str) -> str:
return str(DOWNLOADS_DIR / f'{uuid.uuid4()}_{safe_title}.%(ext)s')
def _is_valid_cookies_file(cookies_path: Path) -> bool:
"""Проверяет, что файл cookies существует и содержит данные (не только заголовки)"""
if not cookies_path.exists():
return False
try:
with open(cookies_path, 'r', encoding='utf-8', errors='ignore') as f:
lines = [line.strip() for line in f.readlines() if line.strip() and not line.strip().startswith('#')]
# Проверяем, что есть хотя бы одна строка с данными cookie
return len(lines) > 0
except Exception as e:
logger.warning(f"Ошибка при проверке файла cookies: {e}")
return False
def download_youtube_video(url: str, max_retries: int = 3) -> Path:
"""Скачивает видео с YouTube - используем cookies для обхода блокировок"""
cookies_file = os.getenv('YOUTUBE_COOKIES_FILE', 'youtube_cookies.txt')
cookies_file_path = Path(cookies_file)
if not cookies_file_path.exists():
logger.info(f"YouTube: файл cookies не найден ({cookies_file_path}). Работаем без cookies. "
f"Для лучшей работы рекомендуется добавить cookies через скрипт get_youtube_cookies_local.sh")
cookies_valid = _is_valid_cookies_file(cookies_file_path)
if not cookies_valid:
logger.warning(f"YouTube: файл cookies не найден или невалиден ({cookies_file_path}). "
f"Работаем без cookies. Для лучшей работы рекомендуется обновить cookies через скрипт get_youtube_cookies.sh")
user_agent = 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36'
last_error = None
for attempt in range(max_retries):
try:
# Определяем, это Shorts или обычное видео
is_shorts = '/shorts/' in url
# Базовые настройки для получения информации
ydl_opts_info = {
'quiet': False,
'no_warnings': False,
'user_agent': user_agent,
'socket_timeout': 30,
'socket_timeout': 60, # Увеличиваем таймаут
'extractor_args': {
'youtube': {
'player_client': ['android', 'web'],
'player_client': ['android', 'web'] if not is_shorts else ['android', 'ios', 'web'],
'player_skip': ['webpage'],
},
},
@ -67,10 +86,12 @@ def download_youtube_video(url: str, max_retries: int = 3) -> Path:
},
}
# Если есть файл с cookies, используем его
if cookies_file_path.exists():
# Если есть валидный файл с cookies, используем его
if cookies_valid:
ydl_opts_info['cookiefile'] = str(cookies_file_path.absolute())
logger.info(f"YouTube: используем cookies из {cookies_file_path}")
logger.info(f"YouTube: используем cookies из {cookies_file_path.absolute()} (попытка {attempt + 1})")
else:
logger.info(f"YouTube: работаем без cookies (попытка {attempt + 1})")
with yt_dlp.YoutubeDL(ydl_opts_info) as ydl:
info = ydl.extract_info(url, download=False)
@ -84,10 +105,10 @@ def download_youtube_video(url: str, max_retries: int = 3) -> Path:
'quiet': False,
'no_warnings': False,
'user_agent': user_agent,
'socket_timeout': 30,
'socket_timeout': 60, # Увеличиваем таймаут
'extractor_args': {
'youtube': {
'player_client': ['android', 'web'],
'player_client': ['android', 'web'] if not is_shorts else ['android', 'ios', 'web'],
'player_skip': ['webpage'],
},
},
@ -100,11 +121,11 @@ def download_youtube_video(url: str, max_retries: int = 3) -> Path:
},
}
# Если есть файл с cookies, используем его для скачивания
if cookies_file_path.exists():
# Если есть валидный файл с cookies, используем его для скачивания
if cookies_valid:
ydl_opts_download['cookiefile'] = str(cookies_file_path.absolute())
logger.info(f"YouTube: начинаем скачивание (попытка {attempt + 1}/{max_retries})")
logger.info(f"YouTube: начинаем скачивание (попытка {attempt + 1}/{max_retries}, Shorts: {is_shorts})")
with yt_dlp.YoutubeDL(ydl_opts_download) as ydl:
ydl.download([url])
@ -118,7 +139,16 @@ def download_youtube_video(url: str, max_retries: int = 3) -> Path:
except Exception as e:
last_error = e
logger.warning(f"YouTube: попытка {attempt + 1}/{max_retries} не удалась: {e}")
error_str = str(e)
logger.warning(f"YouTube: попытка {attempt + 1}/{max_retries} не удалась: {error_str}")
# Если ошибка связана с cookies и они были использованы, попробуем без cookies на следующей попытке
if 'cookies' in error_str.lower() or 'bot' in error_str.lower() or 'sign in' in error_str.lower():
if cookies_valid and attempt == 0:
logger.warning("YouTube: ошибка с cookies, попробуем обновить cookies или работать без них")
# На следующей попытке попробуем без cookies
cookies_valid = False
if attempt < max_retries - 1:
import time
time.sleep((attempt + 1) * 2)
@ -176,8 +206,22 @@ def download_stream():
}
except Exception as e:
logger.error(f"Ошибка при скачивании: {e}")
return jsonify({'error': str(e)}), 500
error_str = str(e)
logger.error(f"Ошибка при скачивании: {error_str}")
# Улучшаем сообщение об ошибке, если проблема с cookies
if 'cookies' in error_str.lower() or 'bot' in error_str.lower() or 'sign in' in error_str.lower():
error_msg = (
f"{error_str}\n\n"
"💡 Совет: Cookies устарели или недействительны. "
"Обновите cookies, запустив скрипт:\n"
" ./youtube-downloader/get_youtube_cookies.sh\n"
"Затем перезапустите сервис."
)
else:
error_msg = error_str
return jsonify({'error': error_msg}), 500
if __name__ == '__main__':