aboutsummaryrefslogtreecommitdiffstats
path: root/youtube/ytdlp_proxy.py
blob: 4eb7a994114d194942f926cc66ad0df645b07499 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
#!/usr/bin/env python3
"""
Proxy for serving videos with specific audio using yt-dlp.

This module provides streaming functionality for unified formats
with specific audio languages.
"""
import logging
from flask import Response, request, stream_with_context
import urllib.request
import urllib.error
from youtube.ytdlp_service import find_best_unified_format

logger = logging.getLogger(__name__)


def stream_video_with_audio(video_id: str, audio_language: str = 'en', max_quality: int = 720):
    """
    Stream video with specific audio language.
    
    Args:
        video_id: YouTube video ID
        audio_language: Preferred audio language (default: 'en')
        max_quality: Maximum video height (default: 720)
        
    Returns:
        Flask Response with video stream, or 404 if not available
    """
    logger.info(f'Stream request: {video_id} | audio={audio_language} | quality={max_quality}p')
    
    # Find best unified format
    best_format = find_best_unified_format(video_id, audio_language, max_quality)
    
    if not best_format:
        logger.info(f'No suitable unified format found, returning 404 to trigger fallback')
        return Response('No suitable unified format available', status=404)
    
    url = best_format.get('url')
    if not url:
        logger.error('Format found but no URL available')
        return Response('Format URL not available', status=500)
    
    logger.debug(f'Streaming from: {url[:80]}...')
    
    # Stream the video
    try:
        req = urllib.request.Request(url)
        req.add_header('User-Agent', 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36')
        req.add_header('Accept', '*/*')
        
        # Add Range header if client requests it
        if 'Range' in request.headers:
            req.add_header('Range', request.headers['Range'])
            logger.debug(f'Range request: {request.headers["Range"]}')
        
        resp = urllib.request.urlopen(req, timeout=60)
        
        def generate():
            """Generator for streaming video chunks."""
            try:
                while True:
                    chunk = resp.read(65536)  # 64KB chunks
                    if not chunk:
                        break
                    yield chunk
            except Exception as e:
                logger.error(f'Stream error: {e}')
                raise
        
        # Build response headers
        response_headers = {
            'Content-Type': resp.headers.get('Content-Type', 'video/mp4'),
            'Access-Control-Allow-Origin': '*',
        }
        
        # Copy important headers
        for header in ['Content-Length', 'Content-Range', 'Accept-Ranges']:
            if header in resp.headers:
                response_headers[header] = resp.headers[header]
        
        status_code = resp.getcode()
        logger.info(f'Streaming started: {status_code}')
        
        return Response(
            stream_with_context(generate()),
            status=status_code,
            headers=response_headers,
            direct_passthrough=True
        )
        
    except urllib.error.HTTPError as e:
        logger.error(f'HTTP error streaming: {e.code} {e.reason}')
        return Response(f'Error: {e.code} {e.reason}', status=e.code)
    except urllib.error.URLError as e:
        logger.error(f'URL error streaming: {e.reason}')
        return Response(f'Network error: {e.reason}', status=502)
    except Exception as e:
        logger.error(f'Streaming error: {e}', exc_info=True)
        return Response(f'Error: {e}', status=500)