|
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 |
|
) |