File size: 5,084 Bytes
7e7541a
 
 
 
 
 
8bf3938
 
 
 
dc5e8f2
3c4954f
 
 
 
8bf3938
dc5e8f2
8bf3938
dc5e8f2
01e374d
dc5e8f2
01e374d
 
 
8bf3938
dc5e8f2
8bf3938
3c4954f
 
8bf3938
dc5e8f2
 
 
 
 
3c4954f
dc5e8f2
 
 
 
 
3c4954f
 
 
 
 
dc5e8f2
 
3c4954f
8bf3938
 
7e7541a
 
 
 
 
 
 
 
 
 
 
 
 
 
 
3c4954f
7e7541a
 
 
 
3c4954f
7e7541a
 
 
 
3c4954f
7e7541a
 
 
 
 
 
 
 
 
 
8bf3938
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
3c4954f
8bf3938
 
 
 
 
 
 
3c4954f
8bf3938
 
 
 
3c4954f
8bf3938
 
 
 
 
 
 
 
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
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
import subprocess
import sys

# Ensure compatible versions of httpx and httpcore are installed
subprocess.check_call([sys.executable, "-m", "pip", "install", "httpx==0.18.2", "httpcore==0.13.6"])

import gradio as gr
import requests
from bs4 import BeautifulSoup
import re
import yt_dlp
import logging

# Configure logging for debugging purposes
logging.basicConfig(level=logging.DEBUG, format='%(asctime)s - %(levelname)s - %(message)s')

# Function to search YouTube videos using yt-dlp for better reliability
def youtube_search(query, max_results=50):
    ydl_opts = {
        'quiet': False,  # Set to False to get more detailed output from yt-dlp
        'extract_flat': 'in_playlist',
        'logger': logging.getLogger(),  # Use the logging module to capture yt-dlp logs
        'simulate': True,
        'noplaylist': True,  # To avoid playlist entries
    }
    search_url = f"ytsearch{max_results}:{query}"

    logging.debug(f"Starting YouTube search for query: {query}")

    try:
        with yt_dlp.YoutubeDL(ydl_opts) as ydl:
            result = ydl.extract_info(search_url, download=False)
            gallery_items = []

            if 'entries' in result:
                logging.debug(f"Number of entries found: {len(result['entries'])}")
                for entry in result['entries']:
                    video_id = entry.get('id')
                    thumbnail_url = entry.get('thumbnail')
                    if video_id and thumbnail_url:
                        gallery_items.append((thumbnail_url, video_id))
                        logging.debug(f"Added video: ID={video_id}, Thumbnail={thumbnail_url}")
                    else:
                        logging.debug(f"Missing video ID or thumbnail for entry: {entry}")
            else:
                logging.warning("No entries found in search result.")
            return gallery_items
    except Exception as e:
        logging.error(f"Error during YouTube yt-dlp request: {e}")
        return []

# Function to display the video using the video URL
def show_video(video_url):
    # Regular expression to extract the YouTube video ID from the URL
    video_id = None
    patterns = [
        r"youtube\.com/watch\?v=([^&?\/]+)",
        r"youtube\.com/embed/([^&?\/]+)",
        r"youtube\.com/v/([^&?\/]+)",
        r"youtu\.be/([^&?\/]+)"
    ]

    for pattern in patterns:
        match = re.search(pattern, video_url)
        if match:
            video_id = match.group(1)
            logging.debug(f"Extracted video ID: {video_id}")
            break

    # If no video ID is found, return an error message
    if not video_id:
        logging.error("Invalid YouTube URL. Please enter a valid YouTube video link.")
        return "Invalid YouTube URL. Please enter a valid YouTube video link."

    # Create the embed URL
    embed_url = f"https://www.youtube.com/embed/{video_id}"
    logging.debug(f"Embed URL generated: {embed_url}")

    # Return an iframe with the video
    html_code = f'''
    <iframe width="560" height="315" src="{embed_url}" 
    frameborder="0" allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture" 
    allowfullscreen></iframe>
    '''
    return html_code

# Create the Gradio interface
with gr.Blocks() as demo:
    gr.Markdown("## YouTube Video Search, Selection, and Playback")

    with gr.Row():
        with gr.Column(scale=3):
            search_query_input = gr.Textbox(label="Search YouTube", placeholder="Enter your search query here")
            search_button = gr.Button("Search")
            search_output = gr.Gallery(label="Search Results", columns=5, height="1500px")
        
        with gr.Column(scale=2):
            selected_video_link = gr.Textbox(label="Selected Video Link", interactive=False)
            play_video_button = gr.Button("Play Video")
            video_output = gr.HTML(label="Video Player")

    # Define search button behavior
    def update_search_results(query):
        gallery_items = youtube_search(query)
        logging.debug(f"Number of gallery items returned: {len(gallery_items)}")
        return gallery_items

    # Update the selected video link field when a video is clicked in the gallery
    def on_video_select(evt: gr.SelectData):
        # Extract the video ID from the event value, which is a dictionary containing details of the selected item
        selected_video_id = evt.value["caption"]
        video_url = f"https://www.youtube.com/watch?v={selected_video_id}"
        logging.debug(f"Video selected: {video_url}")
        return video_url

    # Play the video when the Play Video button is clicked
    def play_video(video_url):
        logging.debug(f"Playing video with URL: {video_url}")
        return show_video(video_url)

    search_button.click(update_search_results, inputs=search_query_input, outputs=search_output)
    search_output.select(on_video_select, inputs=None, outputs=selected_video_link)
    play_video_button.click(play_video, inputs=selected_video_link, outputs=video_output)

# Launch the Gradio interface
demo.launch()