import logging import shutil import tempfile import subprocess from pathlib import Path from moviepy.editor import VideoFileClip import gradio as gr import requests from urllib.parse import urlparse import http.server import socketserver import threading import atexit logging.basicConfig(level=logging.INFO) PORT = 8000 temp_dir = Path(tempfile.mkdtemp()) # Create a temporary directory using mkdtemp() instead of TemporaryDirectory() class Handler(http.server.SimpleHTTPRequestHandler): def __init__(self, *args, **kwargs): super().__init__(*args, directory=str(temp_dir), **kwargs) def start_server(): with socketserver.TCPServer(("", PORT), Handler) as httpd: print(f"Serving at port {PORT}") httpd.serve_forever() t = threading.Thread(target=start_server) t.start() # Cleanup function to remove the temporary directory when the script is exited @atexit.register def cleanup(): shutil.rmtree(temp_dir) logging.basicConfig(level=logging.INFO) def download_file(url, destination): """Downloads a file from a url to a destination.""" response = requests.get(url) response.raise_for_status() with open(destination, 'wb') as f: f.write(response.content) def get_input_path(video_file, video_url, temp_dir): """Returns the path to the video file, downloading it if necessary.""" if video_file is not None: return Path(video_file.name) elif video_url: url_path = urlparse(video_url).path file_name = Path(url_path).name destination = temp_dir / file_name download_file(video_url, destination) return destination else: raise ValueError("No input was provided.") def get_output_path(input_path, temp_dir, res): """Returns the path to the output file, creating it if necessary.""" output_path = temp_dir / (Path(input_path).stem + f"_{res}.m3u8") return output_path def get_aspect_ratio(input_path, aspect_ratio): """Returns the aspect ratio of the video, calculating it if necessary.""" if aspect_ratio is not None: return aspect_ratio video = VideoFileClip(str(input_path)) return f"{video.size[0]}:{video.size[1]}" def create_master_playlist(output_paths, temp_dir): """Creates a master playlist .m3u8 file that includes all other .m3u8 files.""" master_playlist_path = temp_dir / "master_playlist.m3u8" with open(master_playlist_path, 'w') as f: f.write("#EXTM3U\n") for path in output_paths: f.write(f"#EXT-X-STREAM-INF:BANDWIDTH={1000*1000},RESOLUTION={path.stem.split('_')[-1]}\n") f.write(f"{path.name}\n") return master_playlist_path import logging import shutil import tempfile import subprocess from pathlib import Path from moviepy.editor import VideoFileClip import gradio as gr import requests from urllib.parse import urlparse import http.server import socketserver import threading import atexit logging.basicConfig(level=logging.INFO) PORT = 8000 temp_dir = Path(tempfile.mkdtemp()) # Create a temporary directory using mkdtemp() instead of TemporaryDirectory() class Handler(http.server.SimpleHTTPRequestHandler): def __init__(self, *args, **kwargs): super().__init__(*args, directory=str(temp_dir), **kwargs) def start_server(): with socketserver.TCPServer(("", PORT), Handler) as httpd: print(f"Serving at port {PORT}") httpd.serve_forever() t = threading.Thread(target=start_server) t.start() # Cleanup function to remove the temporary directory when the script is exited @atexit.register def cleanup(): shutil.rmtree(temp_dir) logging.basicConfig(level=logging.INFO) # ... rest of your code ... def convert_video(video_file, quality, aspect_ratio, video_url): # Use the already-created temp_dir instead of creating a new one # temp_dir = Path(tempfile.mkdtemp()) input_path = get_input_path(video_file, video_url, temp_dir) aspect_ratio = get_aspect_ratio(input_path, aspect_ratio) video = VideoFileClip(str(input_path)) original_height = video.size[1] output_paths = [] # Define output_paths as an empty list for res in standard_resolutions: # Skip if resolution is higher than original if res > original_height: continue scale = "-1:" + str(res) # we scale the height to res and keep aspect ratio output_path = get_output_path(input_path, temp_dir, str(res) + 'p') # pass the resolution to create a unique output file ffmpeg_command = [ "ffmpeg", "-i", str(input_path), "-c:v", "libx264", "-crf", str(quality), "-vf", f"scale={scale}:force_original_aspect_ratio=decrease,pad=ceil(iw/2)*2:ceil(ih/2)*2", "-hls_time", "10", "-hls_playlist_type", "vod", "-hls_segment_filename", str(temp_dir / f"{output_path.stem}_%03d.ts"), str(output_path) ] logging.info("Running ffmpeg command: " + ' '.join(ffmpeg_command)) subprocess.run(ffmpeg_command, check=True) output_paths.append(output_path) # Append each completed output file to the output_paths list master_playlist_path = create_master_playlist(output_paths, temp_dir) output_paths.append(master_playlist_path) # Convert Path objects to URLs before returning return [{'name': path.stem, 'url': f"http://localhost:{PORT}/{path.name}"} for path in output_paths] video_file = gr.File(label="Video File") quality = gr.Dropdown( choices=["18", "23", "27", "28", "32"], label="Quality", default="27") aspect_ratio = gr.Textbox(default="16:9", lines=1, label="Aspect ratio (width:height)") standard_resolutions = [4320, 2160, 1440, 1080, 720, 480, 360, 240, 144] # 8K, 4K, 2K, Full HD, HD, SD in pixels video_url = gr.Textbox(label="Or enter video URL") def format_output(output): html = "" for file in output: html += f'

{file["name"]}

' return html interface = gr.Interface( fn=convert_video, inputs=[video_file, quality, aspect_ratio, video_url], outputs=gr.outputs.HTML(label="Download Links", postprocess=format_output), title="Video Converter", description="A simple video converter app", allow_flagging=False, ) interface.launch()