Spaces:
Paused
Paused
Create app-backup.py
Browse files- app-backup.py +1139 -0
app-backup.py
ADDED
@@ -0,0 +1,1139 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
import random
|
3 |
+
import time
|
4 |
+
import html
|
5 |
+
import base64
|
6 |
+
import string
|
7 |
+
import json
|
8 |
+
import asyncio
|
9 |
+
import requests
|
10 |
+
import anthropic
|
11 |
+
import openai
|
12 |
+
import io
|
13 |
+
import logging
|
14 |
+
|
15 |
+
from http import HTTPStatus
|
16 |
+
from typing import Dict, List, Optional, Tuple
|
17 |
+
from functools import partial
|
18 |
+
|
19 |
+
import gradio as gr
|
20 |
+
import modelscope_studio.components.base as ms
|
21 |
+
import modelscope_studio.components.legacy as legacy
|
22 |
+
import modelscope_studio.components.antd as antd
|
23 |
+
|
24 |
+
# === [1] 로거 설정 ===
|
25 |
+
log_stream = io.StringIO()
|
26 |
+
handler = logging.StreamHandler(log_stream)
|
27 |
+
logger = logging.getLogger()
|
28 |
+
logger.setLevel(logging.DEBUG) # 원하는 레벨로 설정
|
29 |
+
logger.addHandler(handler)
|
30 |
+
|
31 |
+
def get_logs():
|
32 |
+
"""StringIO에 쌓인 로그를 문자열로 반환"""
|
33 |
+
return log_stream.getvalue()
|
34 |
+
|
35 |
+
import re
|
36 |
+
|
37 |
+
def get_deployment_update(code):
|
38 |
+
logger.debug(f"[get_deployment_update] 받은 code 길이: {len(code) if code else 0}")
|
39 |
+
if not code or len(code.strip()) < 10:
|
40 |
+
logger.info("[get_deployment_update] 코드가 너무 짧아 배포를 진행하지 않습니다.")
|
41 |
+
return gr.update(
|
42 |
+
value='<div style="color: orange;">⚠️ 배포할 코드가 없습니다. 먼저 게임을 생성하거나 코드를 입력해주세요.</div>',
|
43 |
+
visible=True
|
44 |
+
)
|
45 |
+
|
46 |
+
clean_code = remove_code_block(code)
|
47 |
+
logger.debug(f"[get_deployment_update] 코드 블록 제거 후 code 길이: {len(clean_code)}")
|
48 |
+
|
49 |
+
# 배포 중임을 알리는 메시지 표시
|
50 |
+
yield gr.update(
|
51 |
+
value='<div class="deploy-loading"><div class="loading-spinner"></div> <span class="loading-message">🚀 Vercel에 배포 중입니다... 잠시만 기다려주세요...</span></div>',
|
52 |
+
visible=True
|
53 |
+
)
|
54 |
+
|
55 |
+
# 실제 배포 함수 호출 (시간이 걸릴 수 있으므로 비동기적으로 처리되거나, Gradio의 yield를 통해 중간 업데이트)
|
56 |
+
# deploy_to_vercel이 동기 함수이므로, 이 부분에서 UI가 멈출 수 있습니다.
|
57 |
+
# 더 나은 사용자 경험을 위해서는 deploy_to_vercel을 비동기화하거나 별도 스레드에서 실행해야 하지만,
|
58 |
+
# 현재 구조에서는 일단 동기 호출로 진행합니다.
|
59 |
+
result = deploy_to_vercel(clean_code)
|
60 |
+
logger.info(f"[get_deployment_update] deploy_to_vercel 결과: {result}")
|
61 |
+
|
62 |
+
# 배포 URL 추출
|
63 |
+
match = re.search(r'https?://[\\w\\-]+\\.vercel\\.app', result)
|
64 |
+
if match:
|
65 |
+
url = match.group(0)
|
66 |
+
logger.info(f"[get_deployment_update] 배포 URL 추출 성공: {url}")
|
67 |
+
# HTML 앵커로 링크 표시
|
68 |
+
final_html = f"""
|
69 |
+
<div class="deploy-success">
|
70 |
+
<span class="success-icon">✅</span>
|
71 |
+
<span class="success-message">배포 성공! 앱이 준비되었습니다.</span>
|
72 |
+
</div>
|
73 |
+
<div class="url-box">
|
74 |
+
<a href="{url}" target="_blank">{url}</a>
|
75 |
+
<button class="copy-btn" onclick="navigator.clipboard.writeText('{url}')">복사</button>
|
76 |
+
</div>
|
77 |
+
"""
|
78 |
+
yield gr.update(
|
79 |
+
value=final_html,
|
80 |
+
visible=True
|
81 |
+
)
|
82 |
+
else:
|
83 |
+
logger.warning("[get_deployment_update] 배포 URL을 찾을 수 없습니다.")
|
84 |
+
error_message = html.escape(result)
|
85 |
+
# 오류 메시지에 Vercel 로그가 포함될 수 있으므로, 너무 길지 않게 자릅니다.
|
86 |
+
if len(error_message) > 500:
|
87 |
+
error_message = error_message[:500] + "..."
|
88 |
+
|
89 |
+
final_html = f"""
|
90 |
+
<div class="deploy-error">
|
91 |
+
<span class="error-icon">❌</span>
|
92 |
+
<span class="error-message">배포 URL을 찾을 수 없습니다.</span>
|
93 |
+
</div>
|
94 |
+
<div style="font-size: 0.9em; color: var(--text-secondary); margin-top: 5px;">
|
95 |
+
<strong>Vercel 응답:</strong><br>
|
96 |
+
<pre style="white-space: pre-wrap; word-break: break-all; max-height: 100px; overflow-y: auto; background-color: #f0f0f0; padding: 5px; border-radius: 4px;">{error_message}</pre>
|
97 |
+
</div>
|
98 |
+
"""
|
99 |
+
yield gr.update(
|
100 |
+
value=final_html,
|
101 |
+
visible=True
|
102 |
+
)
|
103 |
+
|
104 |
+
|
105 |
+
# ------------------------
|
106 |
+
# 1) DEMO_LIST 및 SystemPrompt
|
107 |
+
# ------------------------
|
108 |
+
|
109 |
+
|
110 |
+
DEMO_LIST = [
|
111 |
+
{"description": "블록이 위에서 떨어지는 클래식 테트리스 게임을 개발해주세요. 화살표 키로 조작하며, 가로줄이 채워지면 해당 줄이 제거되고 점수가 올라가는 메커니즘이 필요합니다. 난이도는 시간이 지날수록 블록이 빨라지도록 구현하고, 게임오버 조건과 점수 표시 기능을 포함해주세요."},
|
112 |
+
{"description": "두 명이 번갈아가며 플레이할 수 있는 체스 게임을 만들어주세요. 기본적인 체스 규칙(킹, 퀸, 룩, 비숍, 나이트, 폰의 이동 규칙)을 구현하고, 체크와 체크메이트 감지 기능이 필요합니다. 드래그 앤 드롭으로 말을 움직일 수 있게 하며, 이동 기록도 표시해주세요."},
|
113 |
+
{"description": "짝을 맞추는 메모리 카드 게임을 개발해주세요. 카드를 뒤집으면 그림이 나타나고, 같은 그림의 카드 두 장을 찾으면 점수를 얻는 방식입니다. 카드 뒤집기 애니메이션과 함께 시도 횟수를 기록하는 점수 시스템, 그리고 쉬움/보통/어려움 난이도 선택 기능(카드 수 변경)도 구현해주세요."},
|
114 |
+
{"description": "플레이어가 우주선을 조종하여 적 우주선을 파괴하는 슈팅 게임을 만들어주세요. 키보드 방향키로 움직이고 스페이스바로 발사하며, 다양한 적 웨이브가 공격해오는 구조입니다. 충돌 감지 시스템과 함께 파워업 아이템(방패, 다중 발사, 속도 증가 등)을 구현하고, 난이도가 점진적으로 증가하는 시스템을 추가해주세요."},
|
115 |
+
{"description": "3x3 또는 4x4 크기의 슬라이드 퍼즐 게임을 만들어주세요. 숫자나 이미지 조각을 섞은 후, 빈 칸을 이용해 조각들을 올바른 위치로 밀어 맞추는 게임입니다. 섞기 기능과 이동 횟수 카운터, 완성 시 축하 메시지를 표시하고, 난이도 설정(크기 변경)도 구현해주세요."},
|
116 |
+
{"description": "고전적인 뱀 게임을 구현해주세요. 플레이어는 방향키로 뱀을 조종하여 필드에 랜덤하게 생성되는 먹이를 먹으며, 먹이를 먹을 때마다 뱀의 길이가 늘어납니다. 자신의 몸에 부딪히거나 벽에 부딪히면 게임이 종료되며, 점수는 먹은 먹이의 수에 비례합니다. 시간이 지날수록 뱀의 이동 속도가 빨라지는 난이도 조절 기능도 추가해주세요."},
|
117 |
+
{"description": "화면 상단에 여러 줄의 벽돌이 배치된 브레이크아웃 게임을 만들어주세요. 플레이어는 화면 하단의 패들을 좌우로 움직여 공을 튕겨내어 벽돌을 깨야 합니다. 벽돌을 모두 깨면 스테이지 클리어, 공이 바닥에 떨어지면 생명이 감소합니다. 공의 속도는 시간이 지날수록 증가하며, 특수 벽돌(추가 생명, 패들 확장 등)도 구현해주세요."},
|
118 |
+
{"description": "길을 따라 이동하는 적들을 방어하는 타워 디펜스 게임을 개발해주세요. 플레이어는 맵의 특정 위치에 다양한 타워(기본 공격, 범위 공격, 감속 효과 등)를 설치하여 적을 물리쳐야 합니다. 웨이브 시스템으로 난이도가 점진적으로 증가하며, 적을 처치하면 자원을 얻어 타워를 업그레이드하거나 새 타워를 건설할 수 있는 경제 시스템을 구현해주세요."},
|
119 |
+
{"description": "캐릭터가 끝없이 달리며 장애물을 뛰어넘는 엔드리스 러너 게임을 만들어주세요. 스페이스바나 마우스 클릭으로 점프하여 다가오는 장애물(바위, 구덩이, 적 등)을 피해야 합니다. 거리에 따라 점수가 증가하며, 코인 등의 수집품을 모으는 요소와 파워업(일시적 무적, 자석 효과 등)도 추가해주세요. 시간이 지날수록 게임 속도가 빨라지는 난이도 시스템도 구현해주세요."},
|
120 |
+
{"description": "2D 플랫포머 게임을 개발해주세요. 플레이어는 방향키로 캐릭터를 조종하여 발판 위를 이동하고, 스페이스바로 점프하며 코인이나 보석 같은 아이템을 수집합니다. 적 캐릭터(간단한 AI로 움직임)와 함정(가시, 떨어지는 발판 등)을 피해 목표 지점까지 도달하는 레벨 기반 구조로 만들어주세요. 체력 시스템과 체크포인트 기능도 구현해주세요."},
|
121 |
+
{"description": "매번 새로운 미로를 자동 생성하는 미로 게임을 만들어주세요. 플레이어는 시작점에서 출발하여 방향키로 캐릭터를 조종해 출구를 찾아야 합니다. 미로 생성 알고리즘(예: 깊이 우선 탐색, 프림 알고리즘 등)을 활용하여 다양한 크기와 복잡도의 미로를 만들고, 타이머로 시간을 측정하며, 선택적으로 최단 경로를 보여주는 힌트 기능도 구현해주세요."},
|
122 |
+
{"description": "간단한 턴제 RPG 게임을 개발해주세요. 플레이어는 탑다운 뷰에서 타일 기반으로 이동하며, 몬스터와 마주치면 턴제 전투가 시작됩니다. 기본 공격, 특수 스킬, 아이템 사용 등의 전투 옵션과 함께 레벨업 시스템(경험치, 능력치 상승)을 구현해주세요. 또한 전투에서 승리하면 골드와 아이템을 획득할 수 있으며, 상점에서 장비를 구매하는 기능도 추가해주세요."},
|
123 |
+
{"description": "같은 색상/모양의 아이템 3개 이상을 일렬로 맞추는 매치-3 퍼즐 게임을 만들어주세요. 아이템을 스와이프하여 위치를 바꾸고, 매치되면 아이템이 사라지며 점수를 얻는 방식입니다. 특수 매치(4개 이상, T자 모양 등)는 특수 아이템을 생성하며, 연속 매치(콤보)는 추가 점수나 보너스 효과를 제공합니다. 목표 점수 또는 제한 시간/이동 횟수 모드를 구��해주세요."},
|
124 |
+
{"description": "플래피 버드 스타일의 게임을 개발해주세요. 플레이어는 스페이스바나 마우스 클릭으로 새를 점프시켜 위아래로 움직이는 파이프 사이를 통과해야 합니다. 파이프에 부딪히거나 화면 상단/하단에 닿으면 게임 오버이며, 통과한 파이프 쌍마다 점수가 1점씩 증가합니다. 파이프 간격은 랜덤하게 생성되며, 최고 점수를 로컬 스토리지에 저장하는 기능도 구현해주세요."},
|
125 |
+
{"description": "두 개의 유사한 이미지에서 차이점을 찾는 게임을 만들어주세요. 5-10개의 차이점이 있는 이미지 쌍을 준비하고, 플레이어가 차이점을 클릭하면 표시되도록 합니다. 제한 시간 내에 모든 차이점을 찾아야 하며, 오답 클릭 시 시간 패널티가 부과됩니다. 힌트 시스템(차이점 하나를 자동으로 표시)과 난이도 선택(쉬움: 차이점이 명확, 어려움: 미묘한 차이)도 구현해주세요."},
|
126 |
+
{"description": "화면 상단에서 단어가 떨어지는 타이핑 게임을 개발해주세요. 플레이어는 키보드로 해당 단어를 정확히 입력하여 단어가 바닥에 닿기 전에 제거해야 합니다. 정확히 입력한 단어는 사라지고 점수를 얻으며, 난이도에 따라 단어의 길이와 떨어지는 속도가 조절됩니다. 특수 단어(빨간색 등)는 보너스 점수나 시간 추가 등의 효과를 제공하며, 일정 시간/점수마다 난이도가 상승하는 시스템도 구현해주세요."},
|
127 |
+
{"description": "물리 엔진 기반의 미니 골프 게임을 만들어주세요. 플레이어는 마우스 드래그로 공을 치는 방향과 세기를 조절하여 홀에 공을 넣어야 합니다. 다양한 장애물(모래 함정, 물웅덩이, 경사로 등)이 있는 여러 개의 코스를 구현하고, 각 홀마다 타수를 기록하여 최종 점수를 계산합니다. 바람 방향/세기 같은 환경 요소와 함께 궤적 미리보기 기능도 추가해주세요."},
|
128 |
+
{"description": "플레이어가 낚시를 즐기는 시뮬레이션 게임을 개발해주세요. 마우스 클릭으로 낚싯줄을 던지고, 물고기가 물면 타이밍 맞추기 미니게임으로 물고기를 낚아야 합니다. 다양한 종류의 물고기(희귀도별 점수 차등)를 구현하고, 낚은 물고기에 따라 골드를 획득하여 더 좋은 낚싯대, 미끼 등을 구매할 수 있는 업그레이드 시스템을 추가해주세요. 시간대나 날씨에 따라 출현하는 물고기가 달라지는 기능도 구현해주세요."},
|
129 |
+
{"description": "1인용 또는 AI 대전 빙고 게임을 만들어주세요. 5x5 그리드에 1-25 숫자를 무작위로 배치하고, 번갈아가며 숫자를 선택하여 해당 칸을 마킹합니다. 가로, 세로, 대각선으로 5개의 연속된 마킹이 완성되면 빙고가 되며, 먼저 3빙고를 달성하는 쪽이 승리합니다. 컴퓨터 AI는 랜덤하게 또는 전략적으로(빙고에 가까운 라인 우선) 숫자를 선택하도록 구현하고, 타이머와 승/패 기록 시스템도 추가해주세요."},
|
130 |
+
{"description": "화면 하단에서 상단으로 노트가 올라오면 정확한 타이밍에 키를 눌러 점수를 얻는 리듬 게임을 개발해주세요. 4개의 레인(D, F, J, K 키)에 노트가 등장하며, 타이밍 정확도에 따라 Perfect, Good, Miss 등급이 표시됩니다. 배경 음악에 맞춰 노트가 생성되며, 연속 성공 시 콤보 시스템으로 추가 점수를 제공합니다. 난이도 선택(노트 속도와 밀도 조절)과 함께 최종 결과 화면(정확도, 콤보, 등급)도 구현해주세요."},
|
131 |
+
{"description": "탑다운 뷰의 2D 레이싱 게임을 만들어주세요. 플레이어는 방향키로 자동차를 조종하여 트랙을 따라 주행하며, 트랙 이탈 시 감속되는 메커니즘을 구현합니다. 여러 AI 경쟁자들과 경쟁하며 3바퀴를 가장 빨리 완주하는 게임 모드와 함께, 시간 제한 내에 체크포인트를 통과하는 타임 어택 모드도 구현해주세요. 다양한 차량 선택지(속도와 핸들링 특성 차등)와 부스트 아이템, 장애물 등도 추가해주세요."},
|
132 |
+
{"description": "다양한 카테고리의 퀴즈를 풀어나가는 게임을 개발해주세요. 주어진 질문에 4개의 보기 중 정답을 선택하는 방식으로, 정답 시 점수를 획득하고 오답 시 생명이 감소합니다. 30초 제한 시간 내에 답을 선택해야 하며, 난이도에 따라 질문의 복잡도와 제한 시간이 조절됩니다. 50:50 힌트(오답 2개 제거), 시간 추가 등의 도움 아이템과 함께 최종 결과 요약(정답률, 카테고리별 성적)도 구현해주세요."},
|
133 |
+
{"description": "움직이는 표적을 맞추는 사격 갤러리 게임을 만들어주세요. 마우스 클릭으로 발사하며, 다양한 속도와 패턴으로 움직이는 표적(오리, 병, 풍선 등)을 ���추면 점수를 획득합니다. 제한된 시간과 총알 수 안에 최대한 많은 점수를 얻는 것이 목표이며, 특수 표적(황금 표적 등)은 보너스 점수나 추가 시간/총알을 제공합니다. 연속 명중 시 점수 배율이 증가하는 콤보 시스템과 함께 다양한 난이도 레벨(표적 속도/수 증가)도 구현해주세요."},
|
134 |
+
{"description": "가상 주사위를 굴려 보드판을 돌아다니는 보드 게임을 개발해주세요. 플레이어는 차례대로 1-6 주사위를 굴려 말을 이동시키며, 도착한 칸에 따라 다양한 이벤트(앞으로/뒤로 이동, 한 턴 쉬기, 미니게임 등)가 발생합니다. 특수 아이템(추가 주사위, 이벤트 회피 등)을 수집하고 사용할 수 있으며, 먼저 결승점에 도달하거나 가장 많은 포인트를 모은 플레이어가 승리합니다. 1-4명의 로컬 멀티플레이어를 지원하며, AI 플레이어도 구현해주세요."},
|
135 |
+
{"description": "탑다운 뷰의 좀비 서바이벌 게임을 만들어주세요. WASD로 이동하고 마우스로 조준/발사하며, 끊임없이 몰려오는 좀비 웨이브를 최대한 오래 생존하는 것이 목표입니다. 다양한 무기(권총, 샷건, 기관총 등)와 제한된 탄약, 그리고 체력 회복 아이템과 폭탄 같은 특수 아이템을 맵에서 획득할 수 있습니다. 시간이 지날수록 좀비의 수와 속도가 증가하며, 특수 좀비(탱커, 러너 등)도 등장하는 난이도 시스템을 구현해주세요."},
|
136 |
+
{"description": "축구 페널티킥 게임을 개발해주세요. 공격 시에는 방향과 파워를 조절하여 슛을 날리고, 수비 시에는 골키퍼를 좌/중앙/우 중 한 방향으로 다이빙시켜 공을 막아야 합니다. 5번의 키커-골키퍼 대결 후 더 많은 골을 넣은 쪽이 승리하며, 동점일 경우 서든데스로 승부를 가립니다. 슛의 정확도와 파워에 따라 결과가 달라지며, 골키퍼 AI는 패턴 학습을 통해 플레이어의 경향성을 파악하도록 구현해주세요. 1인 플레이와 2인 로컬 대전 모드를 모두 지원해주세요."},
|
137 |
+
{"description": "클래식한 지뢰찾기 게임을 구현해주세요. NxN 크기의 그리드에 M개의 지뢰가 무작위로 배치되며, 플레이어는 좌클릭으로 칸을 열고 우클릭으로 지뢰 위치에 깃발을 표시합니다. 열린 칸에는 주변 8칸의 지뢰 수가 표시되며, 주변에 지뢰가 없는 칸을 열면 연쇄적으로 주변 칸들이 열립니다. 지뢰가 있는 칸을 열면 게임 오버, 지뢰가 아닌 모든 칸을 열면 승리입니다. 난이도 설정(쉬움: 9x9/10개, 중간: 16x16/40개, 어려움: 30x16/99개)과 함께 첫 클릭은 항상 안전하도록 구현해주세요."},
|
138 |
+
{"description": "두 플레이어가 번갈아가며 7x6 그리드에 색깔 디스크를 떨어뜨려 가로, 세로, 대각선으로 4개의 연속된 디스크를 만드는 Connect Four 게임을 개발해주세요. 플레이어는 열을 클릭하여 디스크를 해당 열의 가장 아래 빈 칸에 배치합니다. 4개의 연속된 디스크를 먼저 만드는 플레이어가 승리하며, 모든 칸이 차면 무승부입니다. 1인 플레이(AI 대전)과 2인 로컬 대전 모드를 구현하고, AI는 최소한 1단계 앞을 내다보는 논리로 작동하도록 해주세요."},
|
139 |
+
{"description": "글자 타일을 배치하여 단어를 만드는 스크래블 스타일의 단어 게임을 만들어주세요. 각 플레이어는 7개의 글자 타일을 받고, 이를 보드에 배치하여 가로나 세로로 단어를 형성합니다. 새 단어는 기존 단어와 반드시 연결되어야 하며, 각 타일에는 점수가 있어 단어의 총점이 계산됩니다. 특수 칸(2배 글자 점수, 3배 단어 점수 등)을 활용한 전략적 배치가 가능하며, 사전 검증 기능으로 유효한 단어만 허용합니다. 1-4인 로컬 멀티플레이어와 AI 대전을 지원해주세요."},
|
140 |
+
{"description": "2D 환경에서 진행되는 탱크 전투 게임을 개발해주세요. 플레이어는 WASD로 탱크를 조종하고, 마우스로 포탑을 조준하여 클릭으로 발사합니다. 파괴 가능한 지형(벽돌, 나무 등)과 파괴 불가능한 장애물(강철, 물 등)이 있는 맵에서 적 탱크들과 전투를 벌입니다. 다양한 무기(기본 포탄, 확산탄, 레이저 등)와 아이템(속도 증가, 방어력 강화, 추가 생명 등)을 구현하고, 스테이지별로 증가하는 적 AI 난이도와 보스 전투도 추가해주세요."},
|
141 |
+
{"description": "3개 이상의 같은 보석을 맞추어 제거하는 퍼즐 게임을 만들어주세요. 인접한 두 보석을 스왑하여 매치를 만들며, 매치된 보석이 사라지면 위의 보석들이 떨어지고 새 보석이 채워집니다. 4개 이상 매치 시 특수 보석(가로/세로 폭발, 주변 9칸 폭발 등)이 생성되며, 연쇄 매치가 발생하면 콤보 점수가 ���가됩니다. 제한 시간 또는 제한 이동 횟수 내에 목표 점수를 달성하는 레벨 기반 진행 구조와 함께, 특수 미션(특정 색상 N개 제거, 장애물 파괴 등)도 구현해주세요."},
|
142 |
+
{"description": "단일 타워가 끊임없이 몰려오는 적들을 격퇴하는 타워 디펜스 게임을 개발해주세요. 화면 중앙의 타워는 자동으로 가장 가까운 적을 향해 발사하며, 플레이어는 웨이브 사이에 획득한 자원으로 타워를 업그레이드(공격력, 공격 속도, 범위 등)할 수 있습니다. 시간이 지날수록 더 강력하고 다양한 적(빠른 적, 방어력 높은 적, 분열하는 적 등)이 등장하며, 타워의 체력이 0이 되면 게임 오버입니다. 특수 능력(범위 공격, 일시 정지, 즉시 회복 등)과 함께 생존한 웨이브 수에 따른 랭킹 시스템도 구현해주세요."},
|
143 |
+
{"description": "캐릭터가 끝없이 달리며 좀비와 장애물을 피하는 사이드 스크롤링 러너 게임을 만들어주세요. 스페이스바로 점프, S키로 슬라이딩하여 다양한 장애물(웅덩이, 장벽, 좀비 무리 등)을 피해야 합니다. 코인과 파워업(일시적 무적, 자석 효과, 속도 감소 등)을 수집하며, 특정 구간마다 미니 보스 좀비와의 간단한 전투도 포함됩니다. 거리에 따라 점수가 증가하고, 코인으로 캐릭터 업그레이드(더블 점프, 체력 증가 등)를 구매할 수 있는 시스템도 구현해주세요."},
|
144 |
+
{"description": "탑다운 뷰의 간단한 액션 RPG 게임을 개발해주세요. WASD로 이동하고, 마우스 클릭으로 기본 공격, 1-4 키로 특수 스킬을 사용합니다. 플레이어는 몬스터를 처치하며 경험치와 아이템을 획득하고, 레벨업 시 능력치(공격력, 체력, 속도 등)를 향상시킵니다. 다양한 무기와 방어구를 착용할 수 있으며, 스킬 트리 시스템으로 캐릭터를 특화시킬 수 있습니다. 여러 지역과 보스 몬스터, 간단한 퀘스트 시스템도 구현해주세요."},
|
145 |
+
]
|
146 |
+
|
147 |
+
|
148 |
+
SystemPrompt = """
|
149 |
+
# GameCraft 시스템 프롬프트
|
150 |
+
|
151 |
+
## 1. 기본 정보 및 역할
|
152 |
+
당신의 이름은 'GameCraft'입니다. 당신은 게임플레이 메커니즘, 인터랙티브 디자인, 성능 최적화에 뛰어난 웹 게임 개발 전문가입니다. HTML, JavaScript, CSS를 활용하여 간결하고 효율적인 웹 기반 게임을 제작하는 것이 당신의 임무입니다.
|
153 |
+
|
154 |
+
## 2. 핵심 기술 스택
|
155 |
+
- **프론트엔드**: HTML5, CSS3, JavaScript(ES6+)
|
156 |
+
- **렌더링 방식**: 브라우저에서 직접 렌더링 가능한 코드 생성
|
157 |
+
- **코드 스타일**: 바닐라 JavaScript 우선, 외부 라이브러리 최소화
|
158 |
+
|
159 |
+
## 3. 게임 유형별 특화 지침
|
160 |
+
### 3.1 아케이드/액션 게임
|
161 |
+
- 간결한 충돌 감지 시스템 구현
|
162 |
+
- 키보드/터치 입력 최적화
|
163 |
+
- 기본적인 점수 시스템
|
164 |
+
|
165 |
+
### 3.2 퍼즐 게임
|
166 |
+
- 명확한 게임 규칙 및 승리 조건
|
167 |
+
- 기본 난이도 구현
|
168 |
+
- 핵심 게임 메커니즘에 집중
|
169 |
+
|
170 |
+
### 3.3 카드/보드 게임
|
171 |
+
- 간소화된 턴 기반 시스템
|
172 |
+
- 기본 게임 규칙 자동화
|
173 |
+
- 핵심 게임 로직 중심
|
174 |
+
|
175 |
+
### 3.4 시뮬레이션 게임
|
176 |
+
- 효율적인 상태 관리
|
177 |
+
- 가장 중요한 상호작용 구현
|
178 |
+
- 핵심 요소만 포함
|
179 |
+
|
180 |
+
## 4. 이모지 활용 지침 🎮
|
181 |
+
- 게임 UI 요소에 이모지 활용 (예: 생명력 ❤️, 코인 💰, 시간 ⏱️)
|
182 |
+
- 이모지 사용은 핵심 요소에만 집중
|
183 |
+
|
184 |
+
## 5. 기술적 구현 가이드라인
|
185 |
+
### 5.1 코드 구조
|
186 |
+
- **간결성 중시**: 코드는 최대한 간결하게 작성하고, 주석은 최소화
|
187 |
+
- **모듈화**: 코드 기능별로 분리하되 불필요한 추상화 지양
|
188 |
+
- **최적화**: 게임 루프와 렌더링 최적화에 집중
|
189 |
+
- **코드 크기 제한**: 전체 코드는 200줄을 넘지 않도록 함
|
190 |
+
|
191 |
+
### 5.2 성능 최적화
|
192 |
+
- DOM 조작 최소화
|
193 |
+
- 불필요한 변수와 함수 제거
|
194 |
+
- 메모리 관리에 주의
|
195 |
+
|
196 |
+
### 5.3 반응형 디자인
|
197 |
+
- 기본적인 반응형 지원
|
198 |
+
- 핵심 기능에 집중한 심플한 UI
|
199 |
+
|
200 |
+
## 6. 외부 라이브러리
|
201 |
+
- 라이브러리 사용은 최소화하고, 필요한 경우에만 사용
|
202 |
+
- 라이브러리 사용 시 CDN으로 가져올 것
|
203 |
+
|
204 |
+
## 7. 접근성 및 포용성
|
205 |
+
- 핵심 접근성 기능에만 집중
|
206 |
+
|
207 |
+
## 8. 제약사항 및 유의사항
|
208 |
+
- 외부 API 호출 금지
|
209 |
+
- 코드 크기 최소화에 우선순위 (200줄 이내)
|
210 |
+
- 주석 최소화 - 필수적인 설명만 포함
|
211 |
+
- 불필요한 기능 구현 지양
|
212 |
+
|
213 |
+
## 9. 출력 형식
|
214 |
+
- HTML 코드 블록으로만 코드 반환
|
215 |
+
- 추가 설명 없이 즉시 실행 가능한 코드만 제공
|
216 |
+
- 모든 코드는 단일 HTML 파일에 인라인으로 포함
|
217 |
+
|
218 |
+
## 10. 코드 품질 기준
|
219 |
+
- 효율성과 간결함이 최우선
|
220 |
+
- 핵심 게임플레이 메커니즘에만 집중
|
221 |
+
- 복잡한 기능보다 작동하는 기본 기능 우선
|
222 |
+
- 불필요한 주석이나 장황한 코드 지양
|
223 |
+
- 단일 파일에 모든 코드 포함
|
224 |
+
- 코드 길이 제한: 완성된 게임 코드는 200줄 이내로 작성
|
225 |
+
|
226 |
+
## 11. 중요: 코드 생성 제한
|
227 |
+
- 게임 코드는 반드시 200줄 이내로 제한
|
228 |
+
- 불필요한 설명이나 주석 제외
|
229 |
+
- 핵심 기능만 구현하고 부가 기능은 생략
|
230 |
+
- 코드 크기가 커질 경우 기능을 간소화하거나 생략할 것
|
231 |
+
"""
|
232 |
+
|
233 |
+
|
234 |
+
# ------------------------
|
235 |
+
# 2) 공통 상수, 함수, 클래스
|
236 |
+
# ------------------------
|
237 |
+
|
238 |
+
class Role:
|
239 |
+
SYSTEM = "system"
|
240 |
+
USER = "user"
|
241 |
+
ASSISTANT = "assistant"
|
242 |
+
|
243 |
+
History = List[Tuple[str, str]]
|
244 |
+
Messages = List[Dict[str, str]]
|
245 |
+
|
246 |
+
IMAGE_CACHE = {}
|
247 |
+
|
248 |
+
def get_image_base64(image_path):
|
249 |
+
"""
|
250 |
+
이미지 파일을 base64로 읽어서 캐싱
|
251 |
+
"""
|
252 |
+
if image_path in IMAGE_CACHE:
|
253 |
+
return IMAGE_CACHE[image_path]
|
254 |
+
try:
|
255 |
+
with open(image_path, "rb") as image_file:
|
256 |
+
encoded_string = base64.b64encode(image_file.read()).decode()
|
257 |
+
IMAGE_CACHE[image_path] = encoded_string
|
258 |
+
return encoded_string
|
259 |
+
except:
|
260 |
+
return IMAGE_CACHE.get('default.png', '')
|
261 |
+
|
262 |
+
def history_to_messages(history: History, system: str) -> Messages:
|
263 |
+
messages = [{'role': Role.SYSTEM, 'content': system}]
|
264 |
+
for h in history:
|
265 |
+
messages.append({'role': Role.USER, 'content': h[0]})
|
266 |
+
messages.append({'role': Role.ASSISTANT, 'content': h[1]})
|
267 |
+
return messages
|
268 |
+
|
269 |
+
def messages_to_history(messages: Messages) -> History:
|
270 |
+
assert messages[0]['role'] == Role.SYSTEM
|
271 |
+
history = []
|
272 |
+
for q, r in zip(messages[1::2], messages[2::2]):
|
273 |
+
history.append([q['content'], r['content']])
|
274 |
+
return history
|
275 |
+
|
276 |
+
|
277 |
+
# ------------------------
|
278 |
+
# 3) API 연동 설정
|
279 |
+
# ------------------------
|
280 |
+
|
281 |
+
YOUR_ANTHROPIC_TOKEN = os.getenv('ANTHROPIC_API_KEY', '').strip()
|
282 |
+
YOUR_OPENAI_TOKEN = os.getenv('OPENAI_API_KEY', '').strip()
|
283 |
+
|
284 |
+
claude_client = anthropic.Anthropic(api_key=YOUR_ANTHROPIC_TOKEN)
|
285 |
+
openai_client = openai.OpenAI(api_key=YOUR_OPENAI_TOKEN)
|
286 |
+
|
287 |
+
async def try_claude_api(system_message, claude_messages, timeout=15):
|
288 |
+
"""
|
289 |
+
Claude API 호출 (스트리밍)
|
290 |
+
"""
|
291 |
+
try:
|
292 |
+
system_message_with_limit = system_message + "\n\n추가 중요 지침: 생성하는 코드는 절대로 200줄을 넘지 마세요. 코드 간결성이 최우선입니다. 주석을 최소화하고, 핵심 기능만 구현하세요."
|
293 |
+
|
294 |
+
start_time = time.time()
|
295 |
+
with claude_client.messages.stream(
|
296 |
+
model="claude-3-opus-20240229", # Sonnet 대신 Opus 사용 또는 최신 Sonnet 모델명 확인
|
297 |
+
max_tokens=4096, # Claude-3 모델의 최대 토큰 수에 맞게 조정
|
298 |
+
system=system_message_with_limit,
|
299 |
+
messages=claude_messages,
|
300 |
+
temperature=0.3,
|
301 |
+
) as stream:
|
302 |
+
collected_content = ""
|
303 |
+
for chunk in stream:
|
304 |
+
current_time = time.time()
|
305 |
+
if current_time - start_time > timeout:
|
306 |
+
raise TimeoutError("Claude API timeout")
|
307 |
+
if chunk.type == "content_block_delta":
|
308 |
+
collected_content += chunk.delta.text
|
309 |
+
yield collected_content
|
310 |
+
await asyncio.sleep(0) # CPU 양보
|
311 |
+
# stream.text_stream 사용 시 아래와 같이 처리 가능
|
312 |
+
# for text in stream.text_stream:
|
313 |
+
# collected_content += text
|
314 |
+
# yield collected_content
|
315 |
+
# await asyncio.sleep(0)
|
316 |
+
# 마지막으로 수집된 전체 콘텐츠 반환 (필요시)
|
317 |
+
# yield collected_content
|
318 |
+
except Exception as e:
|
319 |
+
logger.error(f"Claude API Error: {e}")
|
320 |
+
raise e
|
321 |
+
|
322 |
+
async def try_openai_api(openai_messages):
|
323 |
+
"""
|
324 |
+
OpenAI API 호출 (스트리밍) - 코드 길이 제한 강화
|
325 |
+
"""
|
326 |
+
try:
|
327 |
+
if openai_messages and openai_messages[0]["role"] == "system":
|
328 |
+
openai_messages[0]["content"] += "\n\n추가 중요 지침: 생성하는 코드는 절대로 200줄을 넘지 마세요. 코드 간결성이 최우선입니다. 주석은 최소화하고, 핵심 기능만 구현하세요."
|
329 |
+
|
330 |
+
stream = openai_client.chat.completions.create(
|
331 |
+
model="gpt-4o", # 또는 "gpt-4-turbo" 등 최신/적절한 모델
|
332 |
+
messages=openai_messages,
|
333 |
+
stream=True,
|
334 |
+
max_tokens=4096, # 모델 최대 토큰 수에 맞게 조정
|
335 |
+
temperature=0.2
|
336 |
+
)
|
337 |
+
collected_content = ""
|
338 |
+
for chunk in stream:
|
339 |
+
if chunk.choices[0].delta.content is not None:
|
340 |
+
collected_content += chunk.choices[0].delta.content
|
341 |
+
yield collected_content
|
342 |
+
await asyncio.sleep(0) # CPU 양보
|
343 |
+
except Exception as e:
|
344 |
+
logger.error(f"OpenAI API Error: {e}")
|
345 |
+
raise e
|
346 |
+
|
347 |
+
|
348 |
+
# ------------------------
|
349 |
+
# 4) 템플릿(하나로 통합)
|
350 |
+
# ------------------------
|
351 |
+
|
352 |
+
def load_json_data():
|
353 |
+
data_list = []
|
354 |
+
for item in DEMO_LIST:
|
355 |
+
data_list.append({
|
356 |
+
"name": f"[게임] {item['description'][:20]}...",
|
357 |
+
"prompt": item['description']
|
358 |
+
})
|
359 |
+
return data_list
|
360 |
+
|
361 |
+
def create_template_html(title, items):
|
362 |
+
"""
|
363 |
+
이미지 없이 템플릿 HTML 생성
|
364 |
+
"""
|
365 |
+
html_content = r"""
|
366 |
+
<style>
|
367 |
+
.prompt-grid {
|
368 |
+
display: grid;
|
369 |
+
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
|
370 |
+
gap: 16px;
|
371 |
+
padding: 12px;
|
372 |
+
}
|
373 |
+
.prompt-card {
|
374 |
+
background: white;
|
375 |
+
border: 1px solid #eee;
|
376 |
+
border-radius: 12px;
|
377 |
+
padding: 12px;
|
378 |
+
cursor: pointer;
|
379 |
+
box-shadow: 0 4px 8px rgba(0,0,0,0.05);
|
380 |
+
transition: all 0.3s ease;
|
381 |
+
}
|
382 |
+
.prompt-card:hover {
|
383 |
+
transform: translateY(-4px);
|
384 |
+
box-shadow: 0 6px 12px rgba(0,0,0,0.1);
|
385 |
+
}
|
386 |
+
.card-name {
|
387 |
+
font-weight: bold;
|
388 |
+
margin-bottom: 8px;
|
389 |
+
font-size: 13px;
|
390 |
+
color: #444;
|
391 |
+
}
|
392 |
+
.card-prompt {
|
393 |
+
font-size: 11px;
|
394 |
+
line-height: 1.4;
|
395 |
+
color: #666;
|
396 |
+
display: -webkit-box;
|
397 |
+
-webkit-line-clamp: 7;
|
398 |
+
-webkit-box-orient: vertical;
|
399 |
+
overflow: hidden;
|
400 |
+
height: 84px;
|
401 |
+
background-color: #f8f9fa;
|
402 |
+
padding: 8px;
|
403 |
+
border-radius: 6px;
|
404 |
+
}
|
405 |
+
</style>
|
406 |
+
<div class="prompt-grid">
|
407 |
+
"""
|
408 |
+
import html as html_lib
|
409 |
+
for item in items:
|
410 |
+
card_html = f"""
|
411 |
+
<div class="prompt-card" onclick="copyToInput(this)" data-prompt="{html_lib.escape(item.get('prompt', ''))}">
|
412 |
+
<div class="card-name">{html_lib.escape(item.get('name', ''))}</div>
|
413 |
+
<div class="card-prompt">{html_lib.escape(item.get('prompt', ''))}</div>
|
414 |
+
</div>
|
415 |
+
"""
|
416 |
+
html_content += card_html
|
417 |
+
html_content += r"""
|
418 |
+
</div>
|
419 |
+
<script>
|
420 |
+
function copyToInput(card) {
|
421 |
+
const prompt = card.dataset.prompt;
|
422 |
+
// Gradio 3.x / 4.x 에서 textarea 선택자 변경 가능성 고려
|
423 |
+
// 우선 .ant-input-textarea-large textarea 시도, 없으면 일반 textarea 시도
|
424 |
+
let textarea = document.querySelector('.ant-input-textarea-large textarea');
|
425 |
+
if (!textarea) {
|
426 |
+
// 일반적인 Gradio textarea 선택자 (버전에 따라 다를 수 있음)
|
427 |
+
const textareas = document.querySelectorAll('textarea');
|
428 |
+
// 여러 textarea 중 올바른 것을 특정하기 어려우므로, 좀 더 구체적인 ID나 클래스 필요
|
429 |
+
// 여기서는 첫 번째 textarea를 사용하거나, ID 기반으로 찾아야 함.
|
430 |
+
// 예: textarea = document.querySelector('#input_text_area_id textarea');
|
431 |
+
// 지금은 antd 컴포넌트 기반으로 유지
|
432 |
+
if (textareas.length > 0) {
|
433 |
+
// input_text 컴포넌트에 ID를 부여하고 해당 ID로 찾는 것이 더 안정적
|
434 |
+
// 예: input_text = antd.InputTextarea(..., elem_id="main_prompt_input")
|
435 |
+
// textarea = document.querySelector('#main_prompt_input textarea');
|
436 |
+
// 임시로 첫번째 textarea 사용 (주의: 다른 textarea가 있을 경우 문제 발생 가능)
|
437 |
+
// textarea = textareas[0];
|
438 |
+
}
|
439 |
+
}
|
440 |
+
|
441 |
+
if (textarea) {
|
442 |
+
textarea.value = prompt;
|
443 |
+
// Gradio 입력값 업데이트를 위해 'input' 이벤트 트리거
|
444 |
+
const event = new Event('input', { bubbles: true });
|
445 |
+
textarea.dispatchEvent(event);
|
446 |
+
|
447 |
+
// 템플릿 Drawer 닫기 (Drawer 닫기 버튼의 클래스나 ID로 정확히 선택)
|
448 |
+
const closeButton = document.querySelector('.session-drawer .ant-drawer-close, .session-drawer .close-btn'); // antd Drawer의 기본 닫기 버튼 또는 커스텀 버튼
|
449 |
+
if (closeButton) {
|
450 |
+
closeButton.click();
|
451 |
+
}
|
452 |
+
} else {
|
453 |
+
console.warn('Prompt textarea not found.');
|
454 |
+
}
|
455 |
+
}
|
456 |
+
</script>
|
457 |
+
"""
|
458 |
+
return gr.HTML(value=html_content)
|
459 |
+
|
460 |
+
def load_all_templates():
|
461 |
+
return create_template_html("🎮 모든 게임 템플릿", load_json_data())
|
462 |
+
|
463 |
+
|
464 |
+
# ------------------------
|
465 |
+
# 5) 배포/부스트/기타 유틸
|
466 |
+
# ------------------------
|
467 |
+
|
468 |
+
def remove_code_block(text):
|
469 |
+
if not text: return ""
|
470 |
+
# HTML 주석 제거 (코드 블록 제거 전에 하는 것이 좋을 수 있음)
|
471 |
+
text = re.sub(r'<!--.*?-->', '', text, flags=re.DOTALL)
|
472 |
+
|
473 |
+
# ```html ... ``` 패턴
|
474 |
+
pattern_html = r'```html\s*([\s\S]+?)\s*```'
|
475 |
+
match_html = re.search(pattern_html, text, re.DOTALL)
|
476 |
+
if match_html:
|
477 |
+
return match_html.group(1).strip()
|
478 |
+
|
479 |
+
# ``` ... ``` 일반 코드 블록 패턴
|
480 |
+
pattern_general = r'```(?:\w+)?\s*([\s\S]+?)\s*```'
|
481 |
+
match_general = re.search(pattern_general, text, re.DOTALL)
|
482 |
+
if match_general:
|
483 |
+
return match_general.group(1).strip()
|
484 |
+
|
485 |
+
# 코드 블록 마커가 없는 경우, 원본 텍스트가 순수 HTML 코드일 수 있다고 가정
|
486 |
+
# 또는 마커가 깨진 경우를 대비해 간단히 마커만 제거 시도
|
487 |
+
text_no_markers = re.sub(r'^```html\s*', '', text, flags=re.MULTILINE).strip()
|
488 |
+
text_no_markers = re.sub(r'\s*```$', '', text_no_markers, flags=re.MULTILINE).strip()
|
489 |
+
|
490 |
+
# 만약 마커 제거 후에도 변화가 없다면 원본 텍스트 반환 (이미 순수 코드일 가능성)
|
491 |
+
if text_no_markers == text.strip():
|
492 |
+
return text.strip()
|
493 |
+
return text_no_markers
|
494 |
+
|
495 |
+
|
496 |
+
def optimize_code(code: str) -> str:
|
497 |
+
if not code or len(code.strip()) == 0:
|
498 |
+
return code
|
499 |
+
|
500 |
+
lines = code.split('\n')
|
501 |
+
# 코드 길이 제한은 LLM에게 맡기는 것이 더 효과적일 수 있음
|
502 |
+
# if len(lines) <= 200:
|
503 |
+
# return code
|
504 |
+
|
505 |
+
# 주석 제거 (HTML, CSS, JS)
|
506 |
+
# HTML 주석
|
507 |
+
cleaned_code = re.sub(r'<!--[\s\S]*?-->', '', code, flags=re.MULTILINE)
|
508 |
+
# CSS 주석
|
509 |
+
cleaned_code = re.sub(r'/\*[\s\S]*?\*/', '', cleaned_code, flags=re.MULTILINE)
|
510 |
+
# JS 한 줄 주석 (주의: URL 등에서 //가 사용될 수 있으므로, 정교한 패턴 필요)
|
511 |
+
# cleaned_code = re.sub(r'(?<!:)//.*?$', '', cleaned_code, flags=re.MULTILINE) # Positive lookbehind for : to avoid http://
|
512 |
+
|
513 |
+
# JS 주석은 LLM이 생성 시 최소화하도록 유도하는 것이 더 안전
|
514 |
+
|
515 |
+
# 빈 줄 제거 (연속된 빈 줄은 하나로)
|
516 |
+
cleaned_lines = []
|
517 |
+
empty_line_count = 0
|
518 |
+
for line in cleaned_code.split('\n'):
|
519 |
+
stripped_line = line.strip()
|
520 |
+
if stripped_line == '':
|
521 |
+
empty_line_count += 1
|
522 |
+
if empty_line_count <= 1: # 연속된 빈 줄 중 첫 번째는 유지 (가독성 위해)
|
523 |
+
cleaned_lines.append('')
|
524 |
+
else:
|
525 |
+
empty_line_count = 0
|
526 |
+
cleaned_lines.append(line) # 원본 줄 유지 (들여쓰기 보존)
|
527 |
+
|
528 |
+
cleaned_code = '\n'.join(cleaned_lines)
|
529 |
+
|
530 |
+
# console.log 제거 (주의: 프로덕션 코드에서는 필요할 수 있음)
|
531 |
+
# cleaned_code = re.sub(r'console\.log\(.*?\);?', '', cleaned_code, flags=re.MULTILINE)
|
532 |
+
|
533 |
+
# 연속된 공백을 하나로 (주의: 문자열 내부 공백에 영향 줄 수 있음, HTML 태그 속성 등)
|
534 |
+
# 이 부분은 매우 신중해야 하며, 보통은 적용하지 않는 것이 안전
|
535 |
+
# cleaned_code = re.sub(r' {2,}', ' ', cleaned_code)
|
536 |
+
|
537 |
+
return cleaned_code.strip()
|
538 |
+
|
539 |
+
|
540 |
+
def send_to_sandbox(code):
|
541 |
+
if not code:
|
542 |
+
return '<div style="color:red;">미리보기를 위한 코드가 없습니다.</div>'
|
543 |
+
|
544 |
+
clean_code = remove_code_block(code) # 코드 블록 제거
|
545 |
+
# clean_code = optimize_code(clean_code) # 최적화는 선택 사항, LLM이 이미 최적화된 코드를 주도록 유도
|
546 |
+
|
547 |
+
# 이미 완전한 HTML 문서인지 확인
|
548 |
+
is_full_html = clean_code.strip().lower().startswith('<!doctype html>') or \
|
549 |
+
clean_code.strip().lower().startswith('<html')
|
550 |
+
|
551 |
+
if not is_full_html:
|
552 |
+
# 기본적인 HTML 구조로 감싸기
|
553 |
+
# CSS나 JS만 있는 경우를 고려하여 <style>이나 <script> 태그가 있다면 head나 body에 적절히 배치
|
554 |
+
# 여기서는 단순화를 위해 body에 모두 넣음
|
555 |
+
clean_code = f"""<!DOCTYPE html>
|
556 |
+
<html lang="ko">
|
557 |
+
<head>
|
558 |
+
<meta charset="UTF-AF">
|
559 |
+
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
560 |
+
<title>Game Preview</title>
|
561 |
+
<style>
|
562 |
+
body {{ margin: 0; overflow: hidden; background-color: #f0f0f0; }}
|
563 |
+
/* Add any default iframe styling here */
|
564 |
+
</style>
|
565 |
+
</head>
|
566 |
+
<body>
|
567 |
+
{clean_code}
|
568 |
+
</body>
|
569 |
+
</html>"""
|
570 |
+
|
571 |
+
# Data URI 생성
|
572 |
+
# b64encode는 bytes를 받으므로, 문자열을 utf-8로 인코딩
|
573 |
+
encoded_html = base64.b64encode(clean_code.encode('utf-8')).decode('utf-8')
|
574 |
+
data_uri = f"data:text/html;charset=utf-8;base64,{encoded_html}"
|
575 |
+
|
576 |
+
# iframe으로 반환
|
577 |
+
# sandbox 속성으로 보안 강화 (필요에 따라 조정)
|
578 |
+
return f'<iframe src="{data_uri}" width="100%" height="920px" style="border:none;" sandbox="allow-scripts allow-same-origin allow-popups allow-forms"></iframe>'
|
579 |
+
|
580 |
+
|
581 |
+
def boost_prompt(prompt: str) -> str:
|
582 |
+
if not prompt:
|
583 |
+
return ""
|
584 |
+
boost_system_prompt = """당신은 웹 게임 개발 프롬프트 전문가입니다.
|
585 |
+
주어진 프롬프트를 분석하여 더 명확하고 간결한 요구사항으로 변환하되,
|
586 |
+
원래 의도와 목적은 그대로 유지하면서 다음 관점들을 고려하여 증강하십시오:
|
587 |
+
|
588 |
+
1. 게임플레이 핵심 메커니즘 명확히 정의 (예: 테트리스 - 블록 낙하, 회전, 줄 제거)
|
589 |
+
2. 필수적인 상호작용 요소만 포함 (예: 키보드 화살표 키 조작)
|
590 |
+
3. 핵심 UI 요소 간략히 기술 (예: 점수판, 다음 블록 표시)
|
591 |
+
4. 코드 간결성 유지를 위한 우선순위 설정 (예: 그래픽 최소화, 사운드 제외)
|
592 |
+
5. 기본적인 게임 규칙과 승리/패배 조건 명시 (예: 화면 상단까지 블록 쌓이면 게임 오버)
|
593 |
+
|
594 |
+
다음 중요 지침을 반드시 준수하세요:
|
595 |
+
- 불필요한 세부 사항이나 부가 기능은 제외 (예: 멀티플레이어, 사용자 정의 테마)
|
596 |
+
- 생성될 코드가 HTML 단일 파일 기준 200줄을 넘지 않도록 기능을 제한 (매우 중요)
|
597 |
+
- 명확하고 간결한 언어로 요구사항 작성
|
598 |
+
- 최소한의 필수 게임 요소만 포함 (MVP - Minimum Viable Product)
|
599 |
+
- "HTML, CSS, JavaScript를 사용하여 단일 파일로 만들어주세요." 라는 문구를 명시적으로 포함.
|
600 |
+
- "코드는 200줄 이내로 매우 간결해야 합니다." 라는 문구를 명시적으로 포함.
|
601 |
+
"""
|
602 |
+
try:
|
603 |
+
# Claude 우선 시도
|
604 |
+
try:
|
605 |
+
response = claude_client.messages.create(
|
606 |
+
model="claude-3-opus-20240229", # 또는 claude-3-sonnet-20240229
|
607 |
+
max_tokens=1024, # 증강된 프롬프트는 길지 않으므로 토큰 수 줄임
|
608 |
+
temperature=0.2, # 일관된 결과 선호
|
609 |
+
messages=[{
|
610 |
+
"role": "user",
|
611 |
+
"content": f"다음 게임 프롬프트를 분석하고 증강하여, 200줄 이내의 단일 HTML 파일 게임 코드를 생성할 수 있도록 매우 간결하게 만들어주세요: {prompt}"
|
612 |
+
}],
|
613 |
+
system=boost_system_prompt
|
614 |
+
)
|
615 |
+
if hasattr(response, 'content') and len(response.content) > 0 and hasattr(response.content[0], 'text'):
|
616 |
+
return response.content[0].text.strip()
|
617 |
+
logger.warning("Claude API 응답 형식 오류 또는 내용 없음 (boost_prompt)")
|
618 |
+
except Exception as e_claude:
|
619 |
+
logger.warning(f"Claude API 호출 실패 (boost_prompt): {e_claude}. OpenAI로 재시도합니다.")
|
620 |
+
|
621 |
+
# OpenAI 재시도
|
622 |
+
completion = openai_client.chat.completions.create(
|
623 |
+
model="gpt-4o", # 또는 gpt-4-turbo
|
624 |
+
messages=[
|
625 |
+
{"role": "system", "content": boost_system_prompt},
|
626 |
+
{"role": "user", "content": f"다음 게임 프롬프트를 분석하고 증강하여, 200줄 이내의 단일 HTML 파일 게임 코드를 생성할 수 있도록 매우 간결하게 만들어주세요: {prompt}"}
|
627 |
+
],
|
628 |
+
max_tokens=1024,
|
629 |
+
temperature=0.2
|
630 |
+
)
|
631 |
+
if completion.choices and len(completion.choices) > 0 and completion.choices[0].message:
|
632 |
+
return completion.choices[0].message.content.strip()
|
633 |
+
logger.warning("OpenAI API 응답 형식 오류 또는 내용 없음 (boost_prompt)")
|
634 |
+
|
635 |
+
# 두 API 모두 실패 시 원본 프롬프트 반환
|
636 |
+
return prompt
|
637 |
+
except Exception as e_general:
|
638 |
+
logger.error(f"Boost prompt 중 일반 오류 발생: {e_general}")
|
639 |
+
return prompt # 예외 발생 시 원본 프롬프트 반환
|
640 |
+
|
641 |
+
def handle_boost(prompt: str):
|
642 |
+
if not prompt or prompt.strip() == "":
|
643 |
+
return gr.update(value="증강할 내용이 없습니다. 먼저 게임 설명을 입력해주세요."), gr.update(active_key="empty")
|
644 |
+
try:
|
645 |
+
boosted_prompt = boost_prompt(prompt)
|
646 |
+
return boosted_prompt, gr.update(active_key="empty") # 증강 후에는 미리보기 탭을 비움
|
647 |
+
except Exception as e:
|
648 |
+
logger.error(f"Handle_boost error: {str(e)}")
|
649 |
+
return gr.update(value=f"증강 중 오류 발생: {str(e)}"), gr.update(active_key="empty")
|
650 |
+
|
651 |
+
|
652 |
+
def history_render(history: History):
|
653 |
+
# history가 비어있으면 Drawer를 열지 않거나, 비어있다는 메시지를 Chatbot에 표시
|
654 |
+
if not history:
|
655 |
+
# Chatbot에 "히스토리가 없습니다" 메시지 표시하는 방법은 Gradio 버전에 따라 다를 수 있음
|
656 |
+
# 여기서는 Drawer는 열되, Chatbot은 비어있게 됨
|
657 |
+
return gr.update(open=True), [] # 빈 히스토리 전달
|
658 |
+
return gr.update(open=True), history
|
659 |
+
|
660 |
+
|
661 |
+
def execute_code(query: str): # query는 코드 문자열을 받음
|
662 |
+
if not query or query.strip() == '':
|
663 |
+
return None, gr.update(active_key="empty") # 코드가 없으면 아무것도 안 함
|
664 |
+
try:
|
665 |
+
# query는 이미 코드 문자열이므로 remove_code_block은 필요 없을 수 있으나,
|
666 |
+
# 사용자가 입력창에 ```html ... ``` 형식으로 넣을 수도 있으므로 안전하게 호출
|
667 |
+
clean_code = remove_code_block(query)
|
668 |
+
|
669 |
+
# send_to_sandbox 함수는 이미 내부적으로 HTML 구조를 완성시켜줌
|
670 |
+
return send_to_sandbox(clean_code), gr.update(active_key="render")
|
671 |
+
except Exception as e:
|
672 |
+
logger.error(f"Execute code error: {str(e)}")
|
673 |
+
error_html = f"<div style='color:red; padding:10px;'>코드 실행 중 오류 발생: {html.escape(str(e))}</div>"
|
674 |
+
return error_html, gr.update(active_key="render") # 오류도 HTML로 렌더링
|
675 |
+
|
676 |
+
|
677 |
+
# ------------------------
|
678 |
+
# 6) 데모 클래스
|
679 |
+
# ------------------------
|
680 |
+
|
681 |
+
class Demo:
|
682 |
+
def __init__(self):
|
683 |
+
pass
|
684 |
+
|
685 |
+
async def generation_code(self, query: Optional[str], _setting: Dict[str, str], _history: Optional[History]):
|
686 |
+
if not query or query.strip() == '':
|
687 |
+
# query = random.choice(DEMO_LIST)['description'] # 랜덤 프롬프트 대신 사용자 입력 유도
|
688 |
+
yield [
|
689 |
+
gr.update(value="게임 설명을 입력해주세요."), # code_output에 메시지
|
690 |
+
_history if _history else [],
|
691 |
+
gr.update(value='<div style="padding: 20px; text-align: center; color: grey;">생성할 게임 설명을 입력하고 "전송" 버튼을 클릭하세요.</div>'), # sandbox
|
692 |
+
gr.update(active_key="empty"), # state_tab
|
693 |
+
gr.update(open=False) # code_drawer (열지 않음)
|
694 |
+
]
|
695 |
+
return
|
696 |
+
|
697 |
+
|
698 |
+
if _history is None:
|
699 |
+
_history = []
|
700 |
+
|
701 |
+
# LLM에게 전달할 최종 프롬프트 구성
|
702 |
+
# 시스템 프롬프트에서 코드 길이 제한 등을 이미 강조하고 있으므로, query에는 게임 요청만 명확히 전달
|
703 |
+
# query_prefix = """
|
704 |
+
# 다음 게임을 제작해주세요.
|
705 |
+
# 중요 요구사항:
|
706 |
+
# 1. 모든 코드는 하나의 HTML 파일에 통합해주세요.
|
707 |
+
# 2. 코드는 극도로 간결해야 하며, 200줄을 넘지 않도록 해주세요. (매우 중요)
|
708 |
+
# 3. 불필요한 주석, 설명, 외부 라이브러리 사용은 피해주세요.
|
709 |
+
# 4. 핵심 게임 기능만 구현하고, 부가 기능은 생략해주세요.
|
710 |
+
# ---
|
711 |
+
# 게임 요청:
|
712 |
+
# """
|
713 |
+
# final_query = query_prefix + query
|
714 |
+
# SystemPrompt에 이미 상세 지침이 있으므로, 사용자 query는 그대로 사용
|
715 |
+
final_query = query
|
716 |
+
|
717 |
+
messages = history_to_messages(_history, _setting['system'])
|
718 |
+
system_message = messages[0]['content'] # SystemPrompt
|
719 |
+
|
720 |
+
# Claude API용 메시지 준비 (System role을 messages list 밖으로 빼는 방식)
|
721 |
+
claude_messages_for_api = [
|
722 |
+
msg for msg in messages[1:] if msg["content"].strip() != ''
|
723 |
+
]
|
724 |
+
claude_messages_for_api.append({'role': Role.USER, 'content': final_query})
|
725 |
+
|
726 |
+
# OpenAI API용 메시지 준비 (System role을 messages list 안에 포함)
|
727 |
+
openai_messages_for_api = [{"role": "system", "content": system_message}]
|
728 |
+
openai_messages_for_api.extend([
|
729 |
+
msg for msg in messages[1:] if msg["content"].strip() != ''
|
730 |
+
])
|
731 |
+
openai_messages_for_api.append({"role": "user", "content": final_query})
|
732 |
+
|
733 |
+
# 초기 UI 업데이트: 로딩 시작
|
734 |
+
yield [
|
735 |
+
gr.update(value="코드를 생성하고 있습니다... 🎮"), # code_output
|
736 |
+
_history,
|
737 |
+
gr.update(value=None), # sandbox (로딩 중에는 비움)
|
738 |
+
gr.update(active_key="loading"), # state_tab
|
739 |
+
gr.update(open=False) # code_drawer (로딩 중에는 닫음)
|
740 |
+
]
|
741 |
+
await asyncio.sleep(0.01) # UI 업데이트를 위한 짧은 지연
|
742 |
+
|
743 |
+
collected_content = None
|
744 |
+
error_occurred = False
|
745 |
+
error_message = ""
|
746 |
+
|
747 |
+
# API 호출 시도 (Claude 우선)
|
748 |
+
try:
|
749 |
+
logger.info("Claude API 호출 시도...")
|
750 |
+
async for content_chunk in try_claude_api(system_message, claude_messages_for_api):
|
751 |
+
collected_content = content_chunk
|
752 |
+
yield [
|
753 |
+
gr.update(value=collected_content + "\n\nClaude API로부터 스트리밍 중..."),
|
754 |
+
_history, None, gr.update(active_key="loading"), gr.update(open=False)
|
755 |
+
]
|
756 |
+
logger.info(f"Claude API 호출 성공. 내용 길이: {len(collected_content if collected_content else '')}")
|
757 |
+
except Exception as e_claude:
|
758 |
+
logger.warning(f"Claude API 실패: {e_claude}. OpenAI API로 재시도합니다.")
|
759 |
+
collected_content = None # Claude 실패 시 내용 초기화
|
760 |
+
# OpenAI 시도 전, 사용자에게 알림 (선택 사항)
|
761 |
+
# yield [
|
762 |
+
# gr.update(value=f"Claude API 호출에 실패했습니다. OpenAI API로 재시도합니다...\n에러: {str(e_claude)[:100]}..."),
|
763 |
+
# _history, None, gr.update(active_key="loading"), gr.update(open=False)
|
764 |
+
# ]
|
765 |
+
# await asyncio.sleep(1) # 메시지 확인 시간
|
766 |
+
|
767 |
+
try:
|
768 |
+
logger.info("OpenAI API 호출 시도...")
|
769 |
+
async for content_chunk in try_openai_api(openai_messages_for_api):
|
770 |
+
collected_content = content_chunk
|
771 |
+
yield [
|
772 |
+
gr.update(value=collected_content + "\n\nOpenAI API로부터 스트리밍 중..."),
|
773 |
+
_history, None, gr.update(active_key="loading"), gr.update(open=False)
|
774 |
+
]
|
775 |
+
logger.info(f"OpenAI API 호출 성공. 내용 길이: {len(collected_content if collected_content else '')}")
|
776 |
+
except Exception as e_openai:
|
777 |
+
logger.error(f"OpenAI API도 실패: {e_openai}")
|
778 |
+
error_occurred = True
|
779 |
+
error_message = f"Claude와 OpenAI API 모두 호출에 실패했습니다.\nClaude: {str(e_claude)}\nOpenAI: {str(e_openai)}"
|
780 |
+
collected_content = None
|
781 |
+
|
782 |
+
|
783 |
+
if error_occurred or not collected_content:
|
784 |
+
final_error_message = error_message if error_message else "API로부터 응답을 받지 못했습니다. 네트워크 연결 또는 API 키 설정을 확인해주세요."
|
785 |
+
yield [
|
786 |
+
gr.update(value=final_error_message),
|
787 |
+
_history,
|
788 |
+
gr.update(value=f'<div style="color:red; padding:10px;">{html.escape(final_error_message)}</div>'),
|
789 |
+
gr.update(active_key="empty"),
|
790 |
+
gr.update(open=True) # 오류 메시지 확인을 위해 코드 보기 창 열기
|
791 |
+
]
|
792 |
+
return
|
793 |
+
|
794 |
+
# API 호출 성공 후 처리
|
795 |
+
clean_code_from_llm = remove_code_block(collected_content) # LLM 응답에서 코드만 추출
|
796 |
+
|
797 |
+
# 코드 길이 경고 (LLM이 생성한 코드 기준)
|
798 |
+
# 이 부분은 SystemPrompt에서 이미 강력하게 요청하고 있으므로, LLM이 잘 지켰을 것으로 기대
|
799 |
+
code_lines = clean_code_from_llm.count('\n') + 1
|
800 |
+
if code_lines > 250: # SystemPrompt는 200줄 요청, 약간의 여유
|
801 |
+
warning_msg = f"""⚠️ **경고: 생성된 코드가 너무 깁니다 ({code_lines}줄).**
|
802 |
+
요청하신 게임의 복잡도 때문일 수 있습니다. SystemPrompt는 200줄 이내를 요청했습니다.
|
803 |
+
코드가 너무 길면 실행이 느리거나, 브라우저에서 오류가 발생할 수 있습니다.
|
804 |
+
더 간단한 게임을 요청하시거나, "코드" 버튼으로 직접 실행해보세요.
|
805 |
+
|
806 |
+
--- 코드 시작 (일부만 표시) ---
|
807 |
+
{html.escape(clean_code_from_llm[:1500])}
|
808 |
+
... (코드가 너무 길어 일부만 표시) ...
|
809 |
+
"""
|
810 |
+
# sandbox_content = send_to_sandbox(clean_code_from_llm) # 긴 코드도 일단 실행 시도
|
811 |
+
sandbox_content = f'<div style="padding:10px; background-color:#fff3cd; color:#664d03; border:1px solid #ffe69c; border-radius:5px;">{warning_msg.replace("--- 코드 시작 (일부만 표시) ---", "").replace("... (코드가 너무 길어 일부만 표시) ...","").replace("⚠️ **경고: 생성된 코드가 너무 깁니다", "⚠️ 경고: 생성된 코드가 너무 깁니다")} <br><br>그래도 미리보기를 시도합니다...</div>' + send_to_sandbox(clean_code_from_llm)
|
812 |
+
|
813 |
+
yield [
|
814 |
+
gr.update(value=warning_msg), # code_output에 경고와 코드 표시
|
815 |
+
_history, # 히스토리는 업데이트하지 않음 (긴 코드는 저장하지 않거나, 사용자 선택)
|
816 |
+
sandbox_content, # sandbox에는 경고와 함께 미리보기 시도
|
817 |
+
gr.update(active_key="render"), # state_tab
|
818 |
+
gr.update(open=True) # code_drawer 열어서 경고 확인
|
819 |
+
]
|
820 |
+
else:
|
821 |
+
# 정상적인 길이의 코드 처리
|
822 |
+
# 히스토리 업데이트
|
823 |
+
# final_query (사용자 입력)과 collected_content (LLM 응답)을 히스토리에 추가
|
824 |
+
_history.append((final_query, collected_content))
|
825 |
+
|
826 |
+
yield [
|
827 |
+
gr.update(value=collected_content), # code_output에 전체 LLM 응답 (코드블록 포함)
|
828 |
+
_history,
|
829 |
+
send_to_sandbox(clean_code_from_llm), # sandbox에는 순수 코드 미리보기
|
830 |
+
gr.update(active_key="render"), # state_tab
|
831 |
+
gr.update(open=False) # code_drawer (성공 시 자동으로 열지 않음, 사용자가 원할 때 열도록)
|
832 |
+
]
|
833 |
+
|
834 |
+
def clear_history(self):
|
835 |
+
return [], gr.update(value="히스토리가 초기화되었습니다."), None, gr.update(active_key="empty") # Chatbot, code_output, sandbox, state_tab 순서로 가정
|
836 |
+
|
837 |
+
|
838 |
+
####################################################
|
839 |
+
# 1) deploy_to_vercel 함수
|
840 |
+
####################################################
|
841 |
+
def deploy_to_vercel(code: str):
|
842 |
+
logger.debug(f"[VercelDeploy] 시작. 코드 길이: {len(code) if code else 0}")
|
843 |
+
try:
|
844 |
+
if not code or len(code.strip()) < 10: # HTML 기본 구조보다 짧으면 배포 의미 없음
|
845 |
+
logger.info("[VercelDeploy] 배포 불가: 코드가 너무 짧습니다.")
|
846 |
+
return "배포할 코드가 너무 짧습니다. 최소한의 HTML 구조를 포함해야 합니다."
|
847 |
+
|
848 |
+
# Vercel API 토큰 (환경 변수 또는 직접 설정)
|
849 |
+
# token = os.getenv("VERCEL_API_TOKEN")
|
850 |
+
token = "A8IFZmgW2cqA4yUNlLPnci0N" # 제공된 토큰 사용 (보안상 환경변수 사용 권장)
|
851 |
+
if not token:
|
852 |
+
logger.error("[VercelDeploy] Vercel API 토큰이 설정되지 않았습니다.")
|
853 |
+
return "Vercel API 토큰이 설정되지 않았습니다. 서버 환경 설정을 확인해주세요."
|
854 |
+
|
855 |
+
# 프로젝트 이름 랜덤 생성 (충돌 방지)
|
856 |
+
project_name = "gamecraft-" + ''.join(random.choice(string.ascii_lowercase + string.digits) for _ in range(8))
|
857 |
+
logger.info(f"[VercelDeploy] 생성된 프로젝트 이름: {project_name}")
|
858 |
+
|
859 |
+
deploy_url = "https://api.vercel.com/v13/deployments"
|
860 |
+
headers = {
|
861 |
+
"Authorization": f"Bearer {token}",
|
862 |
+
"Content-Type": "application/json"
|
863 |
+
}
|
864 |
+
|
865 |
+
# Vercel 배포를 위한 package.json (Vite 사용 예시, 정적 HTML이므로 간단하게)
|
866 |
+
# Vite를 사용하지 않고 순수 정적 파일만 배포할 경우, buildCommand 등이 필요 없을 수 있음
|
867 |
+
# Vercel은 index.html을 자동으로 인식함
|
868 |
+
package_json_content = {
|
869 |
+
"name": project_name,
|
870 |
+
"version": "1.0.0",
|
871 |
+
"private": True,
|
872 |
+
# "scripts": { # 순수 HTML만 있다면 빌드 스크립트 불필요
|
873 |
+
# "build": "echo 'No build needed, deploying static HTML.' && mkdir -p dist && cp index.html dist/"
|
874 |
+
# }
|
875 |
+
}
|
876 |
+
|
877 |
+
# 배포할 파일 목록
|
878 |
+
files_to_deploy = [
|
879 |
+
{"file": "index.html", "data": code},
|
880 |
+
# {"file": "package.json", "data": json.dumps(package_json_content, indent=2)} # 순수 HTML만 있다면 package.json 불필요
|
881 |
+
]
|
882 |
+
|
883 |
+
# 프로젝트 설정 (순수 HTML의 경우 framework: null 또는 생략 가능)
|
884 |
+
project_settings = {
|
885 |
+
# "buildCommand": "npm run build", # 빌드 명령 불필요
|
886 |
+
# "outputDirectory": "dist", # 빌드 결과 디렉토리 불필요
|
887 |
+
# "installCommand": "npm install", # 의존성 설치 불필요
|
888 |
+
"framework": None # 정적 사이트로 명시
|
889 |
+
}
|
890 |
+
|
891 |
+
# 배포 요청 데이터
|
892 |
+
deploy_data = {
|
893 |
+
"name": project_name, # 프로젝트 이름 (Vercel 대시보드에 표시될 이름)
|
894 |
+
"files": files_to_deploy,
|
895 |
+
"target": "production", # 배포 환경 (production 또는 staging)
|
896 |
+
"projectSettings": project_settings
|
897 |
+
}
|
898 |
+
|
899 |
+
logger.info("[VercelDeploy] Vercel API에 배포 요청 전송 중...")
|
900 |
+
# 타임아웃 설정 (예: 60초)
|
901 |
+
try:
|
902 |
+
deploy_response = requests.post(deploy_url, headers=headers, json=deploy_data, timeout=60)
|
903 |
+
deploy_response.raise_for_status() # 오류 발생 시 예외 발생
|
904 |
+
except requests.exceptions.RequestException as req_err:
|
905 |
+
logger.error(f"[VercelDeploy] Vercel API 요청 실패: {req_err}")
|
906 |
+
logger.error(f"[VercelDeploy] 응답 내용: {req_err.response.text if req_err.response else 'No response'}")
|
907 |
+
return f"Vercel API 요청 실패: {req_err}. 응답: {req_err.response.text if req_err.response else 'No response'}"
|
908 |
+
|
909 |
+
|
910 |
+
response_data = deploy_response.json()
|
911 |
+
logger.debug(f"[VercelDeploy] Vercel API 응답 상태 코드: {deploy_response.status_code}")
|
912 |
+
logger.debug(f"[VercelDeploy] Vercel API 응답 내용 (일부): {str(response_data)[:200]}...")
|
913 |
+
|
914 |
+
# 응답에서 배포 URL 확인 (Vercel 응답 구조에 따라 다를 수 있음)
|
915 |
+
# 일반적으로 response_data['url'] 또는 response_data['alias'][0] 등에 URL이 포함됨
|
916 |
+
# Vercel API v13에서는 생성된 URL이 응답에 바로 포함됨
|
917 |
+
deployment_final_url = response_data.get("url")
|
918 |
+
if not deployment_final_url:
|
919 |
+
# 때로는 project_name 기반으로 URL을 구성해야 할 수 있음
|
920 |
+
# Vercel은 보통 <project_name>.vercel.app 또는 <project_name>-<unique_hash>.vercel.app 형식
|
921 |
+
# 응답에 alias가 있다면 그것을 사용
|
922 |
+
if response_data.get("alias") and len(response_data["alias"]) > 0:
|
923 |
+
deployment_final_url = response_data["alias"][0]
|
924 |
+
else: # 최후의 수단으로 project_name 사용 (정확하지 않을 수 있음)
|
925 |
+
deployment_final_url = f"{project_name}.vercel.app"
|
926 |
+
|
927 |
+
# URL에 https 스키마 추가 (Vercel은 항상 https)
|
928 |
+
if not deployment_final_url.startswith("https://"):
|
929 |
+
deployment_final_url = "https://" + deployment_final_url
|
930 |
+
|
931 |
+
|
932 |
+
logger.info(f"[VercelDeploy] 배포 성공! 앱 URL: {deployment_final_url}")
|
933 |
+
|
934 |
+
# 배포 후 DNS 전파 등을 위해 약간의 시간 대기 (선택 사항)
|
935 |
+
# time.sleep(5) # 실제 서비스에서는 비동기 폴링 등으로 상태 확인 권장
|
936 |
+
|
937 |
+
return (
|
938 |
+
f"✅ **배포 완료!**\n"
|
939 |
+
f"앱이 다음 주소에서 실행 중입니다:\n"
|
940 |
+
f"[{deployment_final_url}]({deployment_final_url})" # 마크다운 링크 형식
|
941 |
+
)
|
942 |
+
|
943 |
+
except Exception as e:
|
944 |
+
logger.error(f"[VercelDeploy] 배포 중 예기치 않은 오류 발생: {e}", exc_info=True)
|
945 |
+
return f"배포 중 오류 발생: {str(e)}"
|
946 |
+
|
947 |
+
|
948 |
+
# ------------------------
|
949 |
+
# (3) handle_deploy_legacy - 이 함수는 get_deployment_update로 대체되었으므로 주석 처리 또는 삭제 가능
|
950 |
+
# ------------------------
|
951 |
+
# def handle_deploy_legacy(code):
|
952 |
+
# # ... (이전 코드) ...
|
953 |
+
|
954 |
+
|
955 |
+
# ------------------------
|
956 |
+
# 8) Gradio / Modelscope UI 빌드
|
957 |
+
# ------------------------
|
958 |
+
demo_instance = Demo()
|
959 |
+
theme = gr.themes.Soft(
|
960 |
+
primary_hue=gr.themes.colors.blue, # 직접 색상 객체 사용
|
961 |
+
secondary_hue=gr.themes.colors.purple,
|
962 |
+
neutral_hue=gr.themes.colors.slate,
|
963 |
+
spacing_size=gr.themes.sizes.spacing_md,
|
964 |
+
radius_size=gr.themes.sizes.radius_md,
|
965 |
+
text_size=gr.themes.sizes.text_md,
|
966 |
+
)
|
967 |
+
|
968 |
+
with gr.Blocks(css="app.css", theme=theme) as demo: # css 파일명 직접 전달
|
969 |
+
# header_html = gr.HTML(""" ... """) # 기존 HTML 사용 또는 Python에서 동적 생성
|
970 |
+
|
971 |
+
gr.HTML("""
|
972 |
+
<div class="app-header">
|
973 |
+
<h1>🎮 Vibe Game Craft</h1>
|
974 |
+
<p>설명을 입력하면 웹 기반 HTML5, JavaScript, CSS 게임을 생성합니다. 실시간 미리보기와 원클릭 배포 기능도 지원됩니다!</p>
|
975 |
+
</div>
|
976 |
+
""")
|
977 |
+
|
978 |
+
history = gr.State([])
|
979 |
+
setting = gr.State({"system": SystemPrompt})
|
980 |
+
# deploy_status = gr.State(...) # 현재 사용되지 않으므로 제거 또는 필요시 재활성화
|
981 |
+
|
982 |
+
with ms.Application() as app: # ModelScope Studio 컴포넌트 사용 시
|
983 |
+
with antd.ConfigProvider(): # Ant Design 컴포넌트 사용 시
|
984 |
+
|
985 |
+
with antd.Drawer(open=False, title="💻 생성된 코드 보기", placement="left", width="50%") as code_drawer:
|
986 |
+
code_output = legacy.Markdown(elem_classes="code-markdown-output", संतति_rendering=False) # संतति_rendering 옵션 확인
|
987 |
+
|
988 |
+
with antd.Drawer(open=False, title="📜 대화 히스토리", placement="left", width="60%") as history_drawer:
|
989 |
+
history_output = legacy.Chatbot(
|
990 |
+
show_label=False,
|
991 |
+
flushing=False, # 실시간 스트리밍 시 flushing 동작 방식 확인
|
992 |
+
height=700, # 높이 조정
|
993 |
+
elem_classes="history_chatbot",
|
994 |
+
show_copy_button=True
|
995 |
+
)
|
996 |
+
|
997 |
+
with antd.Drawer(
|
998 |
+
open=False,
|
999 |
+
title="🎲 게임 템플릿 선택",
|
1000 |
+
placement="right",
|
1001 |
+
width="60%", # 너비 조정
|
1002 |
+
elem_classes="session-drawer"
|
1003 |
+
) as session_drawer:
|
1004 |
+
with antd.Flex(vertical=True, gap="middle"):
|
1005 |
+
# gr.Markdown("### 사용 가능한 게임 템플릿") # 제목은 Drawer title로 충분
|
1006 |
+
session_history_html_output = gr.HTML(elem_classes="session-history-grid") # CSS 클래스명 변경 및 그리드 레이아웃용
|
1007 |
+
close_btn_template_drawer = antd.Button("닫기", type="default", elem_classes="close-btn-template-drawer")
|
1008 |
+
|
1009 |
+
|
1010 |
+
with antd.Row(gutter=[16, 16], align="top", elem_classes="main-layout-row"): # gutter 값 조정
|
1011 |
+
|
1012 |
+
# 왼쪽 Col: 게임 미리보기
|
1013 |
+
with antd.Col(xs=24, sm=24, md=14, lg=15, xl=16, elem_classes="preview-column"): # 반응형 컬럼 크기 조정
|
1014 |
+
with ms.Div(elem_classes="preview-panel panel"): # 클래스명 변경
|
1015 |
+
gr.HTML(r"""
|
1016 |
+
<div class="render_header">
|
1017 |
+
<span class="header_btn red"></span><span class="header_btn yellow"></span><span class="header_btn green"></span>
|
1018 |
+
<span class="header_title">Game Preview</span>
|
1019 |
+
</div>
|
1020 |
+
""")
|
1021 |
+
with antd.Tabs(active_key="empty", render_tab_bar="() => null", elem_classes="preview-tabs-content") as state_tab:
|
1022 |
+
with antd.Tabs.Item(key="empty"):
|
1023 |
+
empty = antd.Empty(description="게임 설명을 입력하고 '전송' 버튼을 누르면 여기에 미리보기가 나타납니다.", elem_classes="right_content empty-state")
|
1024 |
+
with antd.Tabs.Item(key="loading"):
|
1025 |
+
loading = antd.Spin(True, tip="게임 코드 생성 중... 잠시만 기다려주세요...", size="large", elem_classes="right_content loading-state")
|
1026 |
+
with antd.Tabs.Item(key="render"):
|
1027 |
+
sandbox = gr.HTML(elem_classes="html_content sandbox-iframe-container")
|
1028 |
+
|
1029 |
+
|
1030 |
+
# 오른쪽 Col: 입력 및 컨트롤
|
1031 |
+
with antd.Col(xs=24, sm=24, md=10, lg=9, xl=8, elem_classes="control-column"): # 반응형 컬럼 크기 조정
|
1032 |
+
with antd.Flex(vertical=True, gap="middle", elem_classes="control-panel-flex"):
|
1033 |
+
with antd.Flex(gap="small", elem_classes="setting-buttons-flex", justify="space-between"):
|
1034 |
+
codeBtn = antd.Button("코드", type="default", icon="💻", elem_classes="control-button code-btn") # 아이콘 추가 (antd 아이콘 문자열 또는 컴포넌트)
|
1035 |
+
historyBtn = antd.Button("히스토리", type="default", icon="📜", elem_classes="control-button history-btn")
|
1036 |
+
template_btn = antd.Button("템플릿", type="default", icon="🎲", elem_classes="control-button template-btn")
|
1037 |
+
|
1038 |
+
# 입력창을 위로 배치
|
1039 |
+
with antd.Flex(vertical=True, gap="small", elem_classes="input-area-flex"):
|
1040 |
+
input_text = antd.InputTextarea(
|
1041 |
+
size="large",
|
1042 |
+
allow_clear=True,
|
1043 |
+
placeholder="예시: 벽돌깨기 게임을 만들어주세요. 공을 튕겨 벽돌을 맞추고, 패들은 마우스로 조작합니다.", # 플레이스홀더 개선
|
1044 |
+
max_length=500, # 사용자 입력 길이 제한 (너무 길면 LLM 처리 어려움)
|
1045 |
+
auto_size={"minRows": 5, "maxRows": 10}, # 자동 높이 조절
|
1046 |
+
elem_id="main_prompt_input" # JS에서 선택하기 위한 ID
|
1047 |
+
)
|
1048 |
+
gr.HTML('<div class="help-text">💡 만들고 싶은 게임을 설명해주세요. 간결하고 명확할수록 좋아요!</div>')
|
1049 |
+
|
1050 |
+
with antd.Flex(gap="small", justify="space-between", wrap="wrap", elem_classes="action-buttons-flex"): # wrap 추가
|
1051 |
+
btn = antd.Button("🚀 전송", type="primary", size="large", elem_classes="action-button send-btn")
|
1052 |
+
boost_btn = antd.Button("✨ 증강", type="default", size="large", elem_classes="action-button boost-btn")
|
1053 |
+
execute_btn = antd.Button("🎮 코드 실행", type="default", size="large", elem_classes="action-button execute-btn")
|
1054 |
+
deploy_btn = antd.Button("☁️ 배포", type="default", size="large", elem_classes="action-button deploy-btn")
|
1055 |
+
clear_btn = antd.Button("🧹 클리어", type="default", size="large", elem_classes="action-button clear-btn")
|
1056 |
+
|
1057 |
+
# 배포 결과 표시 컨테이너
|
1058 |
+
deploy_result_container = gr.HTML(
|
1059 |
+
value='<div class="no-deploy">아직 배포된 게임이 없습니다. "배포" 버튼을 클릭하여 게임을 공유하세요!</div>', # 초기 메시지 개선
|
1060 |
+
visible=True, # 항상 보이도록 설정 (CSS로 내용 없을 때 스타일링)
|
1061 |
+
elem_id="deploy_result_container",
|
1062 |
+
elem_classes="deploy-output-container" # CSS 스타일링용 클래스
|
1063 |
+
)
|
1064 |
+
|
1065 |
+
|
1066 |
+
# Event Handlers
|
1067 |
+
codeBtn.click(lambda: gr.update(open=True), inputs=[], outputs=[code_drawer])
|
1068 |
+
# code_drawer.close(...) # antd Drawer는 자체 닫기 버튼 있음
|
1069 |
+
|
1070 |
+
historyBtn.click(history_render, inputs=[history], outputs=[history_drawer, history_output])
|
1071 |
+
# history_drawer.close(...)
|
1072 |
+
|
1073 |
+
template_btn.click(
|
1074 |
+
fn=lambda: (gr.update(open=True), load_all_templates()),
|
1075 |
+
outputs=[session_drawer, session_history_html_output], # session_history -> session_history_html_output
|
1076 |
+
queue=False # 빠른 UI 업데이트
|
1077 |
+
)
|
1078 |
+
close_btn_template_drawer.click(lambda: gr.update(open=False), outputs=[session_drawer])
|
1079 |
+
|
1080 |
+
|
1081 |
+
# 전송 버튼 (코드 생성)
|
1082 |
+
btn.click(
|
1083 |
+
demo_instance.generation_code,
|
1084 |
+
inputs=[input_text, setting, history],
|
1085 |
+
outputs=[code_output, history, sandbox, state_tab, code_drawer]
|
1086 |
+
)
|
1087 |
+
|
1088 |
+
# 클리어 버튼
|
1089 |
+
clear_btn.click(
|
1090 |
+
demo_instance.clear_history,
|
1091 |
+
inputs=[],
|
1092 |
+
outputs=[history, code_output, sandbox, state_tab] # 초기화할 출력들 명시
|
1093 |
+
).then(
|
1094 |
+
lambda: ( # 추가로 입력창과 배포 결과도 초기화
|
1095 |
+
gr.update(value=""), # input_text
|
1096 |
+
gr.update(value='<div class="no-deploy">아직 배포된 게임이 없습니다.</div>') # deploy_result_container
|
1097 |
+
),
|
1098 |
+
outputs=[input_text, deploy_result_container]
|
1099 |
+
)
|
1100 |
+
|
1101 |
+
|
1102 |
+
# 증강 버튼
|
1103 |
+
boost_btn.click(
|
1104 |
+
fn=handle_boost,
|
1105 |
+
inputs=[input_text],
|
1106 |
+
outputs=[input_text, state_tab] # state_tab 업데이트는 증강 결과에 따라 필요 없을 수 있음
|
1107 |
+
)
|
1108 |
+
|
1109 |
+
# 코드 실행 버튼 (입력창의 코드를 직접 실행)
|
1110 |
+
execute_btn.click(
|
1111 |
+
fn=execute_code,
|
1112 |
+
inputs=[input_text], # 입력창의 내용을 코드로 간주하고 실행
|
1113 |
+
outputs=[sandbox, state_tab]
|
1114 |
+
)
|
1115 |
+
|
1116 |
+
# 배포 버튼 (생성된 코드(code_output)를 배포)
|
1117 |
+
deploy_btn.click(
|
1118 |
+
fn=get_deployment_update, # 제너레이터 함수이므로 outputs를 적절히 처리해야 함
|
1119 |
+
inputs=[code_output], # code_output (Markdown 컴포넌트의 값)을 입력으로 사용
|
1120 |
+
outputs=[deploy_result_container]
|
1121 |
+
)
|
1122 |
+
|
1123 |
+
|
1124 |
+
# ------------------------
|
1125 |
+
# 9) 실행
|
1126 |
+
# ------------------------
|
1127 |
+
if __name__ == "__main__":
|
1128 |
+
try:
|
1129 |
+
# demo_instance = Demo() # 이미 위에서 생성됨
|
1130 |
+
demo.queue(default_concurrency_limit=10, max_batch_size=4).launch(
|
1131 |
+
ssr_mode=False,
|
1132 |
+
# share=True, # 필요시 외부 공유 활성화
|
1133 |
+
# server_name="0.0.0.0" # 모든 인터페이스에서 접속 허용
|
1134 |
+
# auth=("user", "password") # 필요시 인증 추가
|
1135 |
+
)
|
1136 |
+
except Exception as e:
|
1137 |
+
print(f"Gradio App ���행 중 오류 발생: {e}")
|
1138 |
+
# logger 사용 가능하면 logger.critical(f"Gradio App 실행 중 오류 발생: {e}", exc_info=True)
|
1139 |
+
raise
|