Spaces:
Sleeping
Sleeping
File size: 6,938 Bytes
7e7541a 18bdf4a 7e7541a 8bf3938 dc5e8f2 3c4954f 52f71c0 18bdf4a 3c4954f 8bf3938 18bdf4a dc5e8f2 58cc2af 52f71c0 dc5e8f2 01e374d 58cc2af 18bdf4a 8bf3938 52f71c0 3c4954f 8bf3938 18bdf4a dc5e8f2 3c4954f dc5e8f2 58cc2af 6e08d8f 58cc2af 6e08d8f 58cc2af 6e08d8f 3c4954f 6e08d8f 3c4954f 37f1681 dc5e8f2 37f1681 8bf3938 7e7541a 3c4954f 7e7541a 3c4954f 7e7541a 3c4954f 7e7541a 8bf3938 58cc2af 37f1681 8bf3938 37f1681 58cc2af 6e08d8f 8bf3938 3c4954f 8bf3938 3c4954f 8bf3938 37f1681 8bf3938 52f71c0 |
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 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 |
import subprocess
import sys
import random
import time
# 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
import re
import yt_dlp
import logging
import os
from fake_useragent import UserAgent
# Configure logging for debugging purposes
logging.basicConfig(level=logging.DEBUG, format='%(asctime)s - %(levelname)s - %(message)s')
# Set up User-Agent rotation
ua = UserAgent()
# Define a list of proxies (example placeholders, replace with your working proxies)
PROXIES = [
"http://proxy1.example.com:8080",
"http://proxy2.example.com:8080",
"http://proxy3.example.com:8080",
]
# Function to search YouTube videos using yt-dlp for better reliability
def youtube_search(query, max_results=10):
cookies_file = "cookies.txt" # You need to provide this file with cookies exported from YouTube
ydl_opts = {
'quiet': False, # Set to False to get more detailed output from yt-dlp
'logger': logging.getLogger(), # Use the logging module to capture yt-dlp logs
'simulate': True,
'noplaylist': True, # Avoid extracting playlists
'format': 'best',
'proxy': random.choice(PROXIES), # Randomly choose a proxy for each request
'http_headers': {
'User-Agent': ua.random # Rotate user agents to make requests appear less like a bot
},
}
if os.path.exists(cookies_file):
ydl_opts['cookiefile'] = cookies_file
logging.debug("Using cookies for YouTube authentication.")
search_url = f"ytsearch{max_results}:{query}"
logging.debug(f"Starting YouTube search for query: {query}")
try:
# Introduce a random delay to avoid rate-limiting issues
time.sleep(random.uniform(2, 5))
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')
# Fallback to YouTube static thumbnails if missing
thumbnail_url = entry.get('thumbnail') if entry.get('thumbnail') else f"https://img.youtube.com/vi/{video_id}/hqdefault.jpg"
video_title = entry.get('title', "Unknown Title")
video_description = entry.get('description', "No description available.")
if video_id:
gallery_items.append({
"thumbnail": thumbnail_url,
"video_id": video_id,
"title": video_title,
"description": video_description
})
logging.debug(f"Added video: ID={video_id}, Thumbnail={thumbnail_url}, Title={video_title}")
else:
logging.debug(f"Missing video ID for entry: {entry}")
else:
logging.warning("No entries found in search result.")
return gallery_items, ""
except Exception as e:
error_message = f"Error during YouTube yt-dlp request: {e}"
logging.error(error_message)
return [], error_message
# 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=2, height="1000px", elem_id="gallery")
error_output = gr.Textbox(label="Error Message", interactive=False, visible=False)
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, error_message = youtube_search(query)
if error_message:
return [], error_message, gr.update(visible=True)
# Prepare gallery items
gallery_items_display = [(item["thumbnail"], f"{item['title']}\n{item['description']}", item["video_id"]) for item in gallery_items]
return gallery_items_display, "", gr.update(visible=False)
# 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, error_output, error_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()
|