| import gradio as gr |
| import replicate |
| from PIL import Image |
| from google import genai |
| from google.genai import types |
| import io |
| import base64 |
| import tempfile |
| import os |
| import uuid |
| import requests |
| from io import BytesIO |
| import time |
| import json |
| import datetime |
|
|
| |
| REPLICATE_API_TOKEN = os.getenv("REPLICATE_API_TOKEN") |
| GEMINI_API_KEY = os.getenv("GEMINI_API_KEY") |
| PASSWORD = os.getenv("APP_PASSWORD") |
|
|
| |
| def validate_api_keys(): |
| """API ํค๋ค์ด ์ฌ๋ฐ๋ฅด๊ฒ ์ค์ ๋์๋์ง ํ์ธ""" |
| missing_keys = [] |
| |
| if not REPLICATE_API_TOKEN: |
| missing_keys.append("REPLICATE_API_TOKEN") |
| |
| if not GEMINI_API_KEY: |
| missing_keys.append("GEMINI_API_KEY") |
| |
| if missing_keys: |
| error_msg = f"๋ค์ ํ๊ฒฝ๋ณ์๊ฐ ์ค์ ๋์ง ์์์ต๋๋ค: {', '.join(missing_keys)}" |
| raise ValueError(error_msg) |
| |
| return True |
|
|
| def log_message(message, log_list=None): |
| """๋ก๊ทธ ๋ฉ์์ง๋ฅผ ํ์์คํฌํ์ ํจ๊ป ๊ธฐ๋ก""" |
| timestamp = datetime.datetime.now().strftime("%H:%M:%S") |
| formatted_message = f"[{timestamp}] {message}" |
| print(formatted_message) |
| if log_list is not None: |
| log_list.append(formatted_message) |
| return formatted_message |
|
|
| def translate_to_english(korean_prompt, log_list): |
| """ํ๊ตญ์ด ํ๋กฌํํธ๋ฅผ FLUX ์ด๋ฏธ์ง ์์ฑ์ ์ต์ ํ๋ ์์ด๋ก ๋ฒ์ญ""" |
| log_message("=== ๋ฒ์ญ ํ๋ก์ธ์ค ์์ ===", log_list) |
| log_message(f"์
๋ ฅ๋ ํ๊ตญ์ด ํ๋กฌํํธ: '{korean_prompt}'", log_list) |
| |
| try: |
| log_message("Gemini API ํด๋ผ์ด์ธํธ ์ด๊ธฐํ ์์...", log_list) |
| client = genai.Client(api_key=GEMINI_API_KEY) |
| log_message("Gemini API ํด๋ผ์ด์ธํธ ์ด๊ธฐํ ์๋ฃ", log_list) |
| |
| system_instruction = """You are a professional prompt translator for FLUX image generation models. |
| Translate Korean image editing prompts to optimized English prompts that work best with FLUX models. |
| Keep the translation: |
| - Clear and specific |
| - Descriptive but concise |
| - Focused on visual elements |
| - Suitable for AI image generation |
| |
| Only return the translated English prompt, nothing else.""" |
| |
| log_message("๋ฒ์ญ ์์ฒญ ์ค๋น ์๋ฃ", log_list) |
| log_message("Gemini API ํธ์ถ ์์...", log_list) |
| |
| start_time = time.time() |
| response = client.models.generate_content( |
| model="gemini-2.0-flash", |
| config=types.GenerateContentConfig( |
| system_instruction=system_instruction, |
| temperature=0.3 |
| ), |
| contents=[f"Translate this Korean prompt to English for FLUX image editing: {korean_prompt}"] |
| ) |
| end_time = time.time() |
| |
| log_message(f"Gemini API ์๋ต ์์ ์๋ฃ (์์์๊ฐ: {end_time - start_time:.2f}์ด)", log_list) |
| |
| translated_text = response.text.strip() |
| log_message(f"๋ฒ์ญ ๊ฒฐ๊ณผ: '{translated_text}'", log_list) |
| log_message("=== ๋ฒ์ญ ํ๋ก์ธ์ค ์๋ฃ ===", log_list) |
| |
| return translated_text |
| |
| except Exception as e: |
| error_msg = f"๋ฒ์ญ ์ค ์ค๋ฅ ๋ฐ์: {str(e)}" |
| log_message(error_msg, log_list) |
| log_message("์๋ณธ ํ๊ตญ์ด ํ๋กฌํํธ๋ฅผ ๊ทธ๋๋ก ์ฌ์ฉํฉ๋๋ค", log_list) |
| log_message("=== ๋ฒ์ญ ํ๋ก์ธ์ค ์คํจ ===", log_list) |
| return korean_prompt |
|
|
| def upscale_image(image_path, output_format, english_prompt, log_list): |
| """Clarity Upscaler๋ฅผ ์ฌ์ฉํ์ฌ ์ด๋ฏธ์ง ์
์ค์ผ์ผ๋ง (๋ชจ๋ ๋งค๊ฐ๋ณ์ ๊ณ ์ )""" |
| log_message("=== ํ์ง ๊ฐ์ (์
์ค์ผ์ผ๋ง) ํ๋ก์ธ์ค ์์ ===", log_list) |
| log_message(f"์
๋ ฅ ์ด๋ฏธ์ง ๊ฒฝ๋ก: {image_path}", log_list) |
| log_message(f"์ถ๋ ฅ ํฌ๋งท: {output_format}", log_list) |
| log_message(f"์ฌ์ฉํ ํ๋กฌํํธ: '{english_prompt}'", log_list) |
| |
| try: |
| |
| if not os.path.exists(image_path): |
| error_msg = f"์
๋ ฅ ์ด๋ฏธ์ง ํ์ผ์ด ์กด์ฌํ์ง ์์ต๋๋ค: {image_path}" |
| log_message(error_msg, log_list) |
| return None |
| |
| |
| file_size = os.path.getsize(image_path) |
| log_message(f"์
๋ ฅ ํ์ผ ํฌ๊ธฐ: {file_size} bytes ({file_size/1024/1024:.2f} MB)", log_list) |
| |
| log_message("Replicate API ํด๋ผ์ด์ธํธ ์ด๊ธฐํ ์์...", log_list) |
| client = replicate.Client(api_token=REPLICATE_API_TOKEN) |
| log_message("Replicate API ํด๋ผ์ด์ธํธ ์ด๊ธฐํ ์๋ฃ", log_list) |
| |
| log_message("์ด๋ฏธ์ง ํ์ผ ์ฝ๊ธฐ ์์...", log_list) |
| with open(image_path, "rb") as file: |
| log_message("์ด๋ฏธ์ง ํ์ผ ์ฝ๊ธฐ ์๋ฃ", log_list) |
| |
| |
| input_data = { |
| "image": file, |
| "scale_factor": 2, |
| "resemblance": 0.8, |
| "creativity": 0.2, |
| "output_format": output_format.lower(), |
| "prompt": english_prompt, |
| "negative_prompt": "(worst quality, low quality, normal quality:2)" |
| } |
| |
| log_message("์
์ค์ผ์ผ ํ๋ผ๋ฏธํฐ ์ค์ :", log_list) |
| log_message(f" - scale_factor: 2", log_list) |
| log_message(f" - resemblance: 0.8", log_list) |
| log_message(f" - creativity: 0.2", log_list) |
| log_message(f" - output_format: {output_format.lower()}", log_list) |
| log_message(f" - negative_prompt: (worst quality, low quality, normal quality:2)", log_list) |
| |
| log_message("Clarity Upscaler API ํธ์ถ ์์...", log_list) |
| log_message("๋ชจ๋ธ: philz1337x/clarity-upscaler", log_list) |
| |
| start_time = time.time() |
| try: |
| output = client.run( |
| "philz1337x/clarity-upscaler:dfad41707589d68ecdccd1dfa600d55a208f9310748e44bfe35b4a6291453d5e", |
| input=input_data |
| ) |
| end_time = time.time() |
| log_message(f"Clarity Upscaler API ์๋ต ์์ ์๋ฃ (์์์๊ฐ: {end_time - start_time:.2f}์ด)", log_list) |
| |
| except Exception as api_error: |
| error_msg = f"Clarity Upscaler API ํธ์ถ ์คํจ: {str(api_error)}" |
| log_message(error_msg, log_list) |
| log_message("=== ํ์ง ๊ฐ์ ํ๋ก์ธ์ค ์คํจ ===", log_list) |
| return None |
| |
| |
| log_message(f"API ์๋ต ํ์
: {type(output)}", log_list) |
| log_message(f"API ์๋ต ๋ด์ฉ: {output}", log_list) |
| |
| |
| if output and isinstance(output, list) and len(output) > 0: |
| result_url = output[0] |
| log_message(f"๊ฒฐ๊ณผ ์ด๋ฏธ์ง URL: {result_url}", log_list) |
| log_message("๊ฒฐ๊ณผ ์ด๋ฏธ์ง ๋ค์ด๋ก๋ ์์...", log_list) |
| |
| try: |
| download_start = time.time() |
| response = requests.get(result_url, timeout=30) |
| download_end = time.time() |
| |
| log_message(f"๋ค์ด๋ก๋ ์๋ต ์ํ: {response.status_code}", log_list) |
| log_message(f"๋ค์ด๋ก๋ ์์์๊ฐ: {download_end - download_start:.2f}์ด", log_list) |
| log_message(f"๋ค์ด๋ก๋๋ ๋ฐ์ดํฐ ํฌ๊ธฐ: {len(response.content)} bytes ({len(response.content)/1024/1024:.2f} MB)", log_list) |
| |
| if response.status_code == 200: |
| log_message("์ด๋ฏธ์ง ๋ฐ์ดํฐ๋ฅผ PIL Image๋ก ๋ณํ ์ค...", log_list) |
| result_image = Image.open(BytesIO(response.content)) |
| log_message(f"๋ณํ๋ ์ด๋ฏธ์ง ํฌ๊ธฐ: {result_image.size}", log_list) |
| log_message(f"๋ณํ๋ ์ด๋ฏธ์ง ๋ชจ๋: {result_image.mode}", log_list) |
| |
| |
| ext = output_format.lower() |
| upscaled_filename = f"upscaled_temp_{uuid.uuid4()}.{ext}" |
| log_message(f"์
์ค์ผ์ผ๋ ์ด๋ฏธ์ง ์ ์ฅ ํ์ผ๋ช
: {upscaled_filename}", log_list) |
| |
| |
| if ext == 'jpg' and result_image.mode == 'RGBA': |
| log_message("RGBA ๋ชจ๋๋ฅผ RGB๋ก ๋ณํ ์ค (JPG ์ ์ฅ์ ์ํด)...", log_list) |
| result_image = result_image.convert('RGB') |
| |
| |
| save_start = time.time() |
| if ext == 'jpg': |
| result_image.save(upscaled_filename, format='JPEG', quality=95) |
| else: |
| result_image.save(upscaled_filename, format='PNG') |
| save_end = time.time() |
| |
| log_message(f"์ด๋ฏธ์ง ์ ์ฅ ์๋ฃ (์์์๊ฐ: {save_end - save_start:.2f}์ด)", log_list) |
| |
| |
| saved_size = os.path.getsize(upscaled_filename) |
| log_message(f"์ ์ฅ๋ ํ์ผ ํฌ๊ธฐ: {saved_size} bytes ({saved_size/1024/1024:.2f} MB)", log_list) |
| |
| log_message("=== ํ์ง ๊ฐ์ ํ๋ก์ธ์ค ์๋ฃ ===", log_list) |
| return upscaled_filename |
| else: |
| error_msg = f"์ด๋ฏธ์ง ๋ค์ด๋ก๋ ์คํจ - HTTP ์ํ: {response.status_code}" |
| log_message(error_msg, log_list) |
| log_message("=== ํ์ง ๊ฐ์ ํ๋ก์ธ์ค ์คํจ ===", log_list) |
| return None |
| |
| except requests.exceptions.Timeout: |
| log_message("์ด๋ฏธ์ง ๋ค์ด๋ก๋ ํ์์์ (30์ด)", log_list) |
| log_message("=== ํ์ง ๊ฐ์ ํ๋ก์ธ์ค ์คํจ ===", log_list) |
| return None |
| except Exception as download_error: |
| error_msg = f"์ด๋ฏธ์ง ๋ค์ด๋ก๋ ์ค ์ค๋ฅ: {str(download_error)}" |
| log_message(error_msg, log_list) |
| log_message("=== ํ์ง ๊ฐ์ ํ๋ก์ธ์ค ์คํจ ===", log_list) |
| return None |
| else: |
| log_message("API ์๋ต์ด ๋น์ด์๊ฑฐ๋ ์์ ํ์๊ณผ ๋ค๋ฆ
๋๋ค", log_list) |
| if output: |
| log_message(f"์ค์ ์๋ต: {json.dumps(output, indent=2) if isinstance(output, dict) else str(output)}", log_list) |
| log_message("=== ํ์ง ๊ฐ์ ํ๋ก์ธ์ค ์คํจ ===", log_list) |
| return None |
| |
| except Exception as e: |
| error_msg = f"์
์ค์ผ์ผ๋ง ํ๋ก์ธ์ค ์ค ์์์น ๋ชปํ ์ค๋ฅ: {str(e)}" |
| log_message(error_msg, log_list) |
| log_message("=== ํ์ง ๊ฐ์ ํ๋ก์ธ์ค ์คํจ ===", log_list) |
| return None |
|
|
| def edit_image(input_image, password, korean_prompt, output_format, aspect_ratio, upscale_option, current_images, current_downloads): |
| log_list = [] |
| log_message("=" * 60, log_list) |
| log_message("์๋ก์ด ์ด๋ฏธ์ง ํธ์ง ์์
์์", log_list) |
| log_message("=" * 60, log_list) |
| |
| |
| try: |
| validate_api_keys() |
| log_message("API ํค ๊ฒ์ฆ ์๋ฃ", log_list) |
| except ValueError as e: |
| error_msg = str(e) |
| log_message(f"API ํค ๊ฒ์ฆ ์คํจ: {error_msg}", log_list) |
| return None, None, "\n".join(log_list) + f"\n์ค๋ฅ: {error_msg}", current_images, current_downloads |
| |
| |
| log_message("=== ์
๋ ฅ ๊ฒ์ฆ ๋จ๊ณ ===", log_list) |
| |
| if not password: |
| error_msg = "๋น๋ฐ๋ฒํธ๋ฅผ ์
๋ ฅํ์ธ์." |
| log_message(f"๊ฒ์ฆ ์คํจ: {error_msg}", log_list) |
| return None, None, "\n".join(log_list) + f"\n์ค๋ฅ: {error_msg}", current_images, current_downloads |
| |
| if password != PASSWORD: |
| error_msg = "๋น๋ฐ๋ฒํธ๊ฐ ์ฌ๋ฐ๋ฅด์ง ์์ต๋๋ค." |
| log_message(f"๊ฒ์ฆ ์คํจ: {error_msg}", log_list) |
| return None, None, "\n".join(log_list) + f"\n์ค๋ฅ: {error_msg}", current_images, current_downloads |
| |
| log_message("๋น๋ฐ๋ฒํธ ๊ฒ์ฆ ํต๊ณผ", log_list) |
| |
| if input_image is None: |
| error_msg = "์ด๋ฏธ์ง๋ฅผ ์
๋ก๋ํ์ธ์." |
| log_message(f"๊ฒ์ฆ ์คํจ: {error_msg}", log_list) |
| return None, None, "\n".join(log_list) + f"\n์ค๋ฅ: {error_msg}", current_images, current_downloads |
| |
| log_message(f"์
๋ ฅ ์ด๋ฏธ์ง ๊ฒฝ๋ก: {input_image}", log_list) |
| |
| if not korean_prompt.strip(): |
| error_msg = "ํธ์ง ์ง์์ฌํญ์ ์
๋ ฅํ์ธ์." |
| log_message(f"๊ฒ์ฆ ์คํจ: {error_msg}", log_list) |
| return None, None, "\n".join(log_list) + f"\n์ค๋ฅ: {error_msg}", current_images, current_downloads |
| |
| log_message(f"ํธ์ง ์ง์์ฌํญ: '{korean_prompt.strip()}'", log_list) |
| log_message(f"์ถ๋ ฅ ํฌ๋งท: {output_format}", log_list) |
| log_message(f"ํ๋ฉด ๋น์จ: {aspect_ratio}", log_list) |
| log_message(f"ํ์ง ๊ฐ์ ์ต์
: {upscale_option}", log_list) |
| log_message("๋ชจ๋ ์
๋ ฅ ๊ฒ์ฆ ํต๊ณผ", log_list) |
| |
| try: |
| |
| log_message("=== Replicate ์ค์ ๋จ๊ณ ===", log_list) |
| log_message("Replicate ํด๋ผ์ด์ธํธ ์ด๊ธฐํ ์์...", log_list) |
| client = replicate.Client(api_token=REPLICATE_API_TOKEN) |
| log_message("Replicate ํด๋ผ์ด์ธํธ ์ด๊ธฐํ ์๋ฃ", log_list) |
| |
| |
| english_prompt = translate_to_english(korean_prompt, log_list) |
| |
| |
| log_message("=== ์
๋ ฅ ์ด๋ฏธ์ง ๋ถ์ ===", log_list) |
| try: |
| with Image.open(input_image) as img: |
| log_message(f"์ด๋ฏธ์ง ํฌ๊ธฐ: {img.size}", log_list) |
| log_message(f"์ด๋ฏธ์ง ๋ชจ๋: {img.mode}", log_list) |
| log_message(f"์ด๋ฏธ์ง ํฌ๋งท: {img.format}", log_list) |
| except Exception as img_error: |
| log_message(f"์ด๋ฏธ์ง ์ ๋ณด ์ฝ๊ธฐ ์ค๋ฅ: {str(img_error)}", log_list) |
| |
| |
| file_size = os.path.getsize(input_image) |
| log_message(f"์
๋ ฅ ํ์ผ ํฌ๊ธฐ: {file_size} bytes ({file_size/1024/1024:.2f} MB)", log_list) |
| |
| |
| log_message("=== FLUX ์ด๋ฏธ์ง ํธ์ง ๋จ๊ณ ===", log_list) |
| log_message("์ด๋ฏธ์ง ํ์ผ ์ฝ๊ธฐ ์์...", log_list) |
| |
| with open(input_image, "rb") as file: |
| input_file = file |
| log_message("ํ์ผ ์ฝ๊ธฐ ์๋ฃ", log_list) |
| |
| |
| input_data = { |
| "prompt": english_prompt, |
| "input_image": input_file, |
| "output_format": output_format.lower(), |
| "aspect_ratio": aspect_ratio |
| } |
| |
| |
| flux_start_time = time.time() |
| try: |
| output = client.run( |
| "black-forest-labs/flux-kontext-pro", |
| input=input_data |
| ) |
| flux_end_time = time.time() |
| log_message(f"FLUX API ์๋ต ์์ ์๋ฃ (์์์๊ฐ: {flux_end_time - flux_start_time:.2f}์ด)", log_list) |
| |
| except Exception as flux_error: |
| error_msg = f"FLUX API ํธ์ถ ์คํจ: {str(flux_error)}" |
| log_message(error_msg, log_list) |
| return None, None, "\n".join(log_list) + f"\n์ค๋ฅ: {error_msg}", current_images, current_downloads |
| |
| log_message(f"FLUX API ์๋ต ํ์
: {type(output)}", log_list) |
| |
| |
| log_message("=== ๊ฒฐ๊ณผ ์ด๋ฏธ์ง ์ฒ๋ฆฌ ๋จ๊ณ ===", log_list) |
| temp_filename = f"temp_output_{uuid.uuid4()}.{output_format.lower()}" |
| log_message(f"์์ ํ์ผ๋ช
: {temp_filename}", log_list) |
| |
| try: |
| log_message("๊ฒฐ๊ณผ ์ด๋ฏธ์ง ์ ์ฅ ์์...", log_list) |
| with open(temp_filename, "wb") as file: |
| file.write(output.read()) |
| |
| |
| saved_size = os.path.getsize(temp_filename) |
| log_message(f"FLUX ๊ฒฐ๊ณผ ์ด๋ฏธ์ง ์ ์ฅ ์๋ฃ", log_list) |
| log_message(f"์ ์ฅ๋ ํ์ผ ํฌ๊ธฐ: {saved_size} bytes ({saved_size/1024/1024:.2f} MB)", log_list) |
| |
| except Exception as save_error: |
| error_msg = f"๊ฒฐ๊ณผ ์ด๋ฏธ์ง ์ ์ฅ ์คํจ: {str(save_error)}" |
| log_message(error_msg, log_list) |
| return None, None, "\n".join(log_list) + f"\n์ค๋ฅ: {error_msg}", current_images, current_downloads |
| |
| |
| try: |
| log_message("๊ฒฐ๊ณผ ์ด๋ฏธ์ง๋ฅผ PIL Image๋ก ๋ก๋ ์ค...", log_list) |
| result_image = Image.open(temp_filename) |
| log_message(f"๋ก๋๋ ์ด๋ฏธ์ง ํฌ๊ธฐ: {result_image.size}", log_list) |
| log_message(f"๋ก๋๋ ์ด๋ฏธ์ง ๋ชจ๋: {result_image.mode}", log_list) |
| |
| except Exception as load_error: |
| error_msg = f"๊ฒฐ๊ณผ ์ด๋ฏธ์ง ๋ก๋ ์คํจ: {str(load_error)}" |
| log_message(error_msg, log_list) |
| return None, None, "\n".join(log_list) + f"\n์ค๋ฅ: {error_msg}", current_images, current_downloads |
| |
| |
| final_image = result_image |
| if upscale_option == "์ ์ฉ": |
| upscaled_path = upscale_image(temp_filename, output_format, english_prompt, log_list) |
| |
| if upscaled_path: |
| try: |
| log_message("์
์ค์ผ์ผ๋ ์ด๋ฏธ์ง ๋ก๋ ์ค...", log_list) |
| final_image = Image.open(upscaled_path) |
| log_message(f"์ต์ข
์ด๋ฏธ์ง ํฌ๊ธฐ: {final_image.size}", log_list) |
| log_message("ํ์ง ๊ฐ์ ์ด ์ฑ๊ณต์ ์ผ๋ก ์ ์ฉ๋์์ต๋๋ค", log_list) |
| |
| |
| try: |
| os.remove(upscaled_path) |
| log_message("์
์ค์ผ์ผ ์์ ํ์ผ ์ ๋ฆฌ ์๋ฃ", log_list) |
| except: |
| pass |
| |
| except Exception as upscale_load_error: |
| log_message(f"์
์ค์ผ์ผ๋ ์ด๋ฏธ์ง ๋ก๋ ์คํจ: {str(upscale_load_error)}", log_list) |
| log_message("์๋ณธ FLUX ๊ฒฐ๊ณผ ์ด๋ฏธ์ง๋ฅผ ์ฌ์ฉํฉ๋๋ค", log_list) |
| else: |
| log_message("ํ์ง ๊ฐ์ ์คํจ - ์๋ณธ FLUX ๊ฒฐ๊ณผ ์ด๋ฏธ์ง๋ฅผ ์ฌ์ฉํฉ๋๋ค", log_list) |
| else: |
| log_message("ํ์ง ๊ฐ์ ์ต์
์ด ์ ํ๋์ง ์์ - ์๋ณธ FLUX ๊ฒฐ๊ณผ ์ฌ์ฉ", log_list) |
| |
| |
| log_message("=== ๊ฒฐ๊ณผ ์ ์ฅ ๋จ๊ณ ===", log_list) |
| |
| if current_images is None: |
| current_images = [] |
| if current_downloads is None: |
| current_downloads = [] |
| |
| |
| gallery_temp_file = tempfile.NamedTemporaryFile(delete=False, suffix='.png') |
| final_image.save(gallery_temp_file.name, 'png') |
| current_images.append(gallery_temp_file.name) |
| log_message(f"๊ฐค๋ฌ๋ฆฌ์ฉ ์ด๋ฏธ์ง ์ ์ฅ: {gallery_temp_file.name}", log_list) |
| |
| |
| download_filename = f"edited_image_{len(current_downloads) + 1}.{output_format.lower()}" |
| final_image.save(download_filename) |
| current_downloads.append(download_filename) |
| log_message(f"๋ค์ด๋ก๋์ฉ ์ด๋ฏธ์ง ์ ์ฅ: {download_filename}", log_list) |
| |
| |
| try: |
| os.remove(temp_filename) |
| log_message("FLUX ์์ ํ์ผ ์ ๋ฆฌ ์๋ฃ", log_list) |
| except: |
| pass |
| |
| log_message("=" * 60, log_list) |
| log_message("์ด๋ฏธ์ง ํธ์ง ์์
์ฑ๊ณต์ ์ผ๋ก ์๋ฃ!", log_list) |
| log_message("=" * 60, log_list) |
| |
| return final_image, current_downloads, "\n".join(log_list), current_images, current_downloads |
| |
| except Exception as e: |
| error_msg = f"์์์น ๋ชปํ ์ค๋ฅ ๋ฐ์: {str(e)}" |
| log_message(error_msg, log_list) |
| log_message("=" * 60, log_list) |
| log_message("์ด๋ฏธ์ง ํธ์ง ์์
์คํจ", log_list) |
| log_message("=" * 60, log_list) |
| return None, current_downloads, "\n".join(log_list), current_images, current_downloads |
|
|
| def save_output_as_input(output_image, current_images, current_downloads): |
| """์ถ๋ ฅ ์ด๋ฏธ์ง๋ฅผ ์
๋ ฅ์ผ๋ก ์ ์ฅ""" |
| log_list = [] |
| log_message("=== ์ถ๋ ฅ ์ด๋ฏธ์ง๋ฅผ ์
๋ ฅ์ผ๋ก ์ ์ฅ ์์ ===", log_list) |
| |
| if output_image is None: |
| log_message("์ ์ฅํ ์ถ๋ ฅ ์ด๋ฏธ์ง๊ฐ ์์ต๋๋ค", log_list) |
| return None, current_images, current_downloads |
| |
| try: |
| log_message(f"์ถ๋ ฅ ์ด๋ฏธ์ง ํ์
: {type(output_image)}", log_list) |
| |
| |
| if hasattr(output_image, 'shape'): |
| log_message("numpy array๋ฅผ PIL Image๋ก ๋ณํ ์ค...", log_list) |
| output_image = Image.fromarray(output_image) |
| |
| log_message(f"๋ณํ๋ ์ด๋ฏธ์ง ํฌ๊ธฐ: {output_image.size}", log_list) |
| log_message(f"๋ณํ๋ ์ด๋ฏธ์ง ๋ชจ๋: {output_image.mode}", log_list) |
| |
| |
| temp_file = tempfile.NamedTemporaryFile(delete=False, suffix='.png') |
| output_image.save(temp_file.name, 'png') |
| |
| file_size = os.path.getsize(temp_file.name) |
| log_message(f"์
๋ ฅ์ฉ ์ด๋ฏธ์ง ์ ์ฅ ์๋ฃ: {temp_file.name}", log_list) |
| log_message(f"์ ์ฅ๋ ํ์ผ ํฌ๊ธฐ: {file_size} bytes ({file_size/1024/1024:.2f} MB)", log_list) |
| log_message("=== ์ ์ฅ ํ๋ก์ธ์ค ์๋ฃ ===", log_list) |
| |
| print("\n".join(log_list)) |
| return temp_file.name, current_images, current_downloads |
| |
| except Exception as e: |
| error_msg = f"์ด๋ฏธ์ง ์ ์ฅ ์ค ์ค๋ฅ: {str(e)}" |
| log_message(error_msg, log_list) |
| log_message("=== ์ ์ฅ ํ๋ก์ธ์ค ์คํจ ===", log_list) |
| print("\n".join(log_list)) |
| return None, current_images, current_downloads |
|
|
| |
| try: |
| validate_api_keys() |
| print("โ
๋ชจ๋ API ํค๊ฐ ์ฌ๋ฐ๋ฅด๊ฒ ์ค์ ๋์์ต๋๋ค.") |
| except ValueError as e: |
| print(f"โ {e}") |
|
|
| |
| with gr.Blocks(title="์ด๋ฏธ์ง ํธ์ง๊ธฐ (๋ณด์ ๊ฐํ ๋ฒ์ )") as demo: |
| |
| saved_images = gr.State([]) |
| saved_downloads = gr.State([]) |
| |
| gr.Markdown("# ๐จ AI ์ด๋ฏธ์ง ํธ์ง๊ธฐ (๋ณด์ ๊ฐํ ๋ฒ์ )") |
| |
| with gr.Row(): |
| with gr.Column(): |
| input_image = gr.Image(type="filepath", label="๐ค ์
๋ ฅ ์ด๋ฏธ์ง") |
| password = gr.Textbox( |
| label="๐ ๋น๋ฐ๋ฒํธ", |
| type="password", |
| placeholder="๋น๋ฐ๋ฒํธ๋ฅผ ์
๋ ฅํ์ธ์" |
| ) |
| korean_prompt = gr.Textbox( |
| label="โ๏ธ ํธ์ง ์ง์์ฌํญ (ํ๊ตญ์ด)", |
| placeholder="์: ์ด ์ฌ๋์ ๋งํ ์บ๋ฆญํฐ๋ก ๋ฐ๊ฟ์ค", |
| lines=3 |
| ) |
| |
| with gr.Row(): |
| output_format = gr.Radio( |
| choices=["jpg", "png"], |
| value="jpg", |
| label="๐ ์ถ๋ ฅ ํฌ๋งท" |
| ) |
| aspect_ratio = gr.Dropdown( |
| choices=["match_input_image", "1:1", "3:2", "2:3"], |
| value="match_input_image", |
| label="๐ ํ๋ฉด ๋น์จ" |
| ) |
| |
| upscale_option = gr.Radio( |
| choices=["์์", "์ ์ฉ"], |
| value="์์", |
| label="๐ ํ์ง ๊ฐ์ (2๋ฐฐ ํ๋)" |
| ) |
| |
| edit_btn = gr.Button("๐ ์ด๋ฏธ์ง ํธ์ง", variant="primary", size="lg") |
| |
| with gr.Column(): |
| output_image = gr.Image(label="โจ ํธ์ง๋ ์ด๋ฏธ์ง") |
| log_output = gr.Textbox( |
| label="๐ ์์ธ ๋ก๊ทธ", |
| lines=15, |
| max_lines=20, |
| show_copy_button=True |
| ) |
| download_files = gr.File(label="๐พ ๋ค์ด๋ก๋", file_count="multiple") |
| |
| save_btn = gr.Button("๐ ์ถ๋ ฅ ์ด๋ฏธ์ง๋ฅผ ์
๋ ฅ์ผ๋ก ์ ์ฅ", variant="secondary") |
| |
| |
| with gr.Row(): |
| output_gallery = gr.Gallery( |
| label="๐ธ ํธ์ง ๊ธฐ๋ก", |
| show_label=True, |
| elem_id="gallery", |
| columns=3, |
| rows=2, |
| height="auto" |
| ) |
| |
| |
| edit_btn.click( |
| fn=edit_image, |
| inputs=[input_image, password, korean_prompt, output_format, aspect_ratio, upscale_option, saved_images, saved_downloads], |
| outputs=[output_image, download_files, log_output, saved_images, saved_downloads] |
| ) |
| |
| save_btn.click( |
| fn=save_output_as_input, |
| inputs=[output_image, saved_images, saved_downloads], |
| outputs=[input_image, saved_images, saved_downloads] |
| ) |
| |
| |
| saved_images.change( |
| fn=lambda images: images if images else [], |
| inputs=[saved_images], |
| outputs=[output_gallery] |
| ) |
|
|
| if __name__ == "__main__": |
| print("๐ ์ด๋ฏธ์ง ํธ์ง ์ ํ๋ฆฌ์ผ์ด์
์์ ์ค...") |
|
|
| |
| try: |
| validate_api_keys() |
| print("โ
๋ชจ๋ API ํค๊ฐ ์ค์ ๋์ด ์ ํ๋ฆฌ์ผ์ด์
์ ์์ํฉ๋๋ค.") |
| except ValueError as e: |
| print(f"โ ๏ธ ๊ฒฝ๊ณ : {e}") |
| print("์ผ๋ถ ๊ธฐ๋ฅ์ด ์ ํ๋ ์ ์์ต๋๋ค.") |
| |
| demo.launch( |
| server_name="0.0.0.0", |
| server_port=7860, |
| share=False, |
| debug=True |
| ) |