# import streamlit as st # import yt_dlp # import os # from pathlib import Path # import random # import time # # 페이지 설정 # st.set_page_config(page_title="Simple YouTube Downloader", page_icon="📺") # st.title("Simple YouTube Downloader 📺") # # 스타일 설정 # st.markdown(""" # # """, unsafe_allow_html=True) # # 다운로드 폴더 생성 # output_dir = Path("downloads") # output_dir.mkdir(exist_ok=True) # def download_video(url, progress_bar, status_text): # try: # user_agents = [ # 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36', # 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.2.1 Safari/605.1.15', # 'Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:122.0) Gecko/20100101 Firefox/122.0' # ] # ydl_opts = { # 'format': 'best', # 'outtmpl': str(output_dir / '%(title)s.%(ext)s'), # 'quiet': True, # 'no_warnings': True, # 'cookiefile': 'youtube.com_cookies.txt', # 'user_agent': random.choice(user_agents), # 'referer': 'https://www.youtube.com/', # 'http_chunk_size': random.randint(10000000, 15000000), # 'retries': 3, # 'sleep_interval': 3, # 'max_sleep_interval': 8, # 'headers': { # 'Accept-Language': 'en-US,en;q=0.9', # 'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8', # }, # } # def progress_hook(d): # if d['status'] == 'downloading': # try: # progress = d['downloaded_bytes'] / d['total_bytes'] # progress_bar.progress(progress) # status_text.text(f"다운로드 중: {progress:.1%}") # except: # status_text.text("다운로드 중...") # elif d['status'] == 'finished': # progress_bar.progress(1.0) # status_text.text("처리 완료!") # ydl_opts['progress_hooks'] = [progress_hook] # for attempt in range(3): # try: # with yt_dlp.YoutubeDL(ydl_opts) as ydl: # info = ydl.extract_info(url, download=True) # filename = ydl.prepare_filename(info) # return filename # except yt_dlp.utils.ExtractorError as e: # if "Sign in to confirm you're not a bot" in str(e): # status_text.text(f"재시도 중... ({attempt + 1}/3)") # time.sleep(random.uniform(3, 5)) # continue # raise e # except Exception as e: # if attempt < 2: # status_text.text(f"재시도 중... ({attempt + 1}/3)") # time.sleep(2) # continue # raise e # except Exception as e: # st.error(f"오류 발생: {str(e)}") # return None # # 쿠키 파일 체크 # if not os.path.exists('youtube.com_cookies.txt'): # st.warning("⚠️ 'youtube.com_cookies.txt' 파일이 필요합니다. YouTube에 로그인된 쿠키 파일을 업로드해주세요.") # uploaded_file = st.file_uploader("쿠키 파일 업로드", type=['txt']) # if uploaded_file is not None: # with open('youtube.com_cookies.txt', 'wb') as f: # f.write(uploaded_file.getvalue()) # st.success("✅ 쿠키 파일이 업로드되었습니다!") # st.rerun() # # 메인 UI # if 'video_url' not in st.session_state: # st.session_state.video_url = "" # video_url = st.text_input("YouTube URL 입력:", # value=st.session_state.video_url, # placeholder="https://www.youtube.com/watch?v=...") # # 버튼을 같은 줄에 배치 # col1, col2 = st.columns(2) # with col1: # download_button = st.button("다운로드", type="primary", key="download_btn", use_container_width=True) # with col2: # reset_button = st.button("초기화", key="reset_btn", use_container_width=True) # # 다운로드 로직 # if download_button: # if video_url: # progress_bar = st.progress(0) # status_text = st.empty() # downloaded_file = download_video(video_url, progress_bar, status_text) # if downloaded_file and os.path.exists(downloaded_file): # with open(downloaded_file, 'rb') as file: # st.download_button( # label="⬇️ 파일 다운로드", # data=file, # file_name=os.path.basename(downloaded_file), # mime="video/mp4", # use_container_width=True # ) # else: # st.warning("⚠️ YouTube URL을 입력해주세요.") # # 초기화 로직 # if reset_button: # st.session_state.video_url = "" # st.rerun() # import streamlit as st # import yt_dlp # import os # from pathlib import Path # import random # import time # # 페이지 설정 # st.set_page_config(page_title="Simple YouTube Downloader", page_icon="📺") # st.title("Simple YouTube Downloader 📺") # # 스타일 설정 # st.markdown(""" # # """, unsafe_allow_html=True) # # 다운로드 폴더 생성 # output_dir = Path("downloads") # output_dir.mkdir(exist_ok=True) # def download_video(url, progress_bar, status_text): # try: # user_agents = [ # 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36', # 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.2.1 Safari/605.1.15', # 'Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:122.0) Gecko/20100101 Firefox/122.0' # ] # ydl_opts = { # 'format': 'best', # 'outtmpl': str(output_dir / '%(title)s.%(ext)s'), # 'quiet': True, # 'no_warnings': True, # 'cookiefile': 'youtube.com_cookies.txt', # 'user_agent': random.choice(user_agents), # 'referer': 'https://www.youtube.com/', # 'http_chunk_size': random.randint(10000000, 15000000), # 'retries': 5, # 재시도 횟수를 5로 늘림 # 'sleep_interval': 3, # 'max_sleep_interval': 8, # 'headers': { # 'Accept-Language': 'en-US,en;q=0.9', # 'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8', # }, # } # def progress_hook(d): # if d['status'] == 'downloading': # try: # progress = d['downloaded_bytes'] / d['total_bytes'] # progress_bar.progress(progress) # status_text.text(f"다운로드 중: {progress:.1%}") # except: # status_text.text("다운로드 중...") # elif d['status'] == 'finished': # progress_bar.progress(1.0) # status_text.text("처리 완료!") # ydl_opts['progress_hooks'] = [progress_hook] # for attempt in range(5): # 재시도 횟수를 5로 설정 # try: # with yt_dlp.YoutubeDL(ydl_opts) as ydl: # info = ydl.extract_info(url, download=True) # filename = ydl.prepare_filename(info) # return filename # except yt_dlp.utils.ExtractorError as e: # if "Sign in to confirm you're not a bot" in str(e): # status_text.text(f"재시도 중... ({attempt + 1}/5)") # time.sleep(random.uniform(3, 5)) # continue # status_text.text(f"오류 발생: {str(e)}") # st.error(f"다운로드 실패: {str(e)}") # 실패 메시지 표시 # st.info("문제가 지속될 경우 URL이 올바른지 확인하고 다시 시도하세요.") # 추가 메모 # return None # except Exception as e: # if attempt < 4: # status_text.text(f"재시도 중... ({attempt + 1}/5)") # time.sleep(2) # continue # status_text.text(f"오류 발생: {str(e)}") # st.error(f"다운로드 실패: {str(e)}") # 실패 메시지 표시 # st.info("문제가 지속될 경우 URL이 올바른지 확인하고 다시 시도하세요.") # 추가 메모 # return None # except Exception as e: # st.error(f"오류 발생: {str(e)}") # return None # # 쿠키 파일 체크 # if not os.path.exists('youtube.com_cookies.txt'): # st.warning("⚠️ 'youtube.com_cookies.txt' 파일이 필요합니다. YouTube에 로그인된 쿠키 파일을 업로드해주세요.") # uploaded_file = st.file_uploader("쿠키 파일 업로드", type=['txt']) # if uploaded_file is not None: # with open('youtube.com_cookies.txt', 'wb') as f: # f.write(uploaded_file.getvalue()) # st.success("✅ 쿠키 파일이 업로드되었습니다!") # st.rerun() # # 메인 UI # if 'video_url' not in st.session_state: # st.session_state.video_url = "" # video_url = st.text_input("YouTube URL 입력:", # value=st.session_state.video_url, # placeholder="https://www.youtube.com/watch?v=...") # # 버튼을 같은 줄에 배치 # col1, col2 = st.columns(2) # with col1: # download_button = st.button("다운로드", type="primary", key="download_btn", use_container_width=True) # with col2: # reset_button = st.button("초기화", key="reset_btn", use_container_width=True) # # 다운로드 로직 # if download_button: # if video_url: # progress_bar = st.progress(0) # status_text = st.empty() # downloaded_file = download_video(video_url, progress_bar, status_text) # if downloaded_file and os.path.exists(downloaded_file): # with open(downloaded_file, 'rb') as file: # st.download_button( # label="⬇️ 파일 다운로드", # data=file, # file_name=os.path.basename(downloaded_file), # mime="video/mp4", # use_container_width=True # ) # else: # st.warning("⚠️ YouTube URL을 입력해주세요.") # # 초기화 로직 # if reset_button: # st.session_state.video_url = "" # st.rerun() import streamlit as st import yt_dlp import os from pathlib import Path import random import time import logging # 로깅 설정 logging.basicConfig(level=logging.ERROR, filename='error.log') # 페이지 설정 st.set_page_config(page_title="Simple YouTube Downloader", page_icon="📺") st.title("Simple YouTube Downloader 📺") # 스타일 설정 st.markdown(""" """, unsafe_allow_html=True) # 다운로드 폴더 생성 output_dir = Path("downloads") output_dir.mkdir(exist_ok=True) def download_video(url, progress_bar, status_text): try: user_agents = [ 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/120.0.0.0 Safari/537.36', 'Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/605.1.15 (KHTML, like Gecko) Version/17.2.1 Safari/605.1.15', 'Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:122.0) Gecko/20100101 Firefox/122.0' ] ydl_opts = { 'format': 'best', 'outtmpl': str(output_dir / '%(title)s.%(ext)s'), 'quiet': True, 'no_warnings': True, 'cookiefile': 'youtube.com_cookies.txt', 'user_agent': random.choice(user_agents), 'referer': 'https://www.youtube.com/', 'http_chunk_size': random.randint(100000000, 150000000), 'retries': 5, # 재시도 횟수 'sleep_interval': 3, 'max_sleep_interval': 10, 'headers': { 'Accept-Language': 'en-US,en;q=0.9', 'Accept': 'text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8', }, } def progress_hook(d): if d['status'] == 'downloading': try: progress = d['downloaded_bytes'] / d['total_bytes'] progress_bar.progress(progress) status_text.text(f"다운로드 중: {progress:.1%}") except Exception as e: logging.error(f"Progress Hook Error: {e}") status_text.text("다운로드 중...") elif d['status'] == 'finished': progress_bar.progress(1.0) status_text.text("처리 완료!") ydl_opts['progress_hooks'] = [progress_hook] for attempt in range(5): # 재시도 횟수를 5로 설정 try: with yt_dlp.YoutubeDL(ydl_opts) as ydl: info = ydl.extract_info(url, download=True) filename = ydl.prepare_filename(info) return filename except yt_dlp.utils.ExtractorError as e: if "Sign in to confirm you're not a bot" in str(e): status_text.text(f"재시도 중... ({attempt + 1}/5)") time.sleep(random.uniform(3, 5)) continue logging.error(f"Extractor Error: {str(e)}") st.info("문제가 지속될 경우 URL이 올바른지 확인하고 다시 시도하세요.") return None except Exception as e: logging.error(f"Download Error: {str(e)}") if attempt < 4: status_text.text(f"재시도 중... ({attempt + 1}/5)") time.sleep(2) continue st.info("문제가 지속될 경우 다른 URL을 시도해주세요.") return None except Exception as e: logging.error(f"General Error: {str(e)}") st.info("문제가 지속될 경우 다른 URL을 시도해주세요.") return None # 쿠키 파일 체크 if not os.path.exists('youtube.com_cookies.txt'): st.warning("⚠️ 'youtube.com_cookies.txt' 파일이 필요합니다. YouTube에 로그인된 쿠키 파일을 업로드해주세요.") uploaded_file = st.file_uploader("쿠키 파일 업로드", type=['txt']) if uploaded_file is not None: with open('youtube.com_cookies.txt', 'wb') as f: f.write(uploaded_file.getvalue()) st.success("✅ 쿠키 파일이 업로드되었습니다!") st.rerun() # 메인 UI if 'video_url' not in st.session_state: st.session_state.video_url = "" video_url = st.text_input("YouTube URL 입력:", value=st.session_state.video_url, placeholder="https://www.youtube.com/watch?v=...") # 버튼을 같은 줄에 배치 col1, col2 = st.columns(2) with col1: download_button = st.button("다운로드", type="primary", key="download_btn", use_container_width=True) with col2: reset_button = st.button("초기화", key="reset_btn", use_container_width=True) # 다운로드 로직 if download_button: if video_url: progress_bar = st.progress(0) status_text = st.empty() downloaded_file = download_video(video_url, progress_bar, status_text) if downloaded_file and os.path.exists(downloaded_file): with open(downloaded_file, 'rb') as file: st.download_button( label="⬇️ 파일 다운로드", data=file, file_name=os.path.basename(downloaded_file), mime="video/mp4", use_container_width=True ) else: st.warning("⚠️ YouTube URL을 입력해주세요.") # 초기화 로직 if reset_button: st.session_state.video_url = "" st.rerun() # st.experimental_rerun()을 st.rerun()으로 변경했습니다.