forked from AutoForgeAI/autoforge
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathstart_ui.py
More file actions
300 lines (234 loc) · 8.34 KB
/
start_ui.py
File metadata and controls
300 lines (234 loc) · 8.34 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
#!/usr/bin/env python3
"""
AutoCoder UI Launcher
=====================
Automated launcher that handles all setup:
1. Creates/activates Python virtual environment
2. Installs Python dependencies
3. Checks for Node.js
4. Installs npm dependencies
5. Builds React frontend (if needed)
6. Starts FastAPI server
7. Opens browser to the UI
Usage:
python start_ui.py [--dev]
Options:
--dev Run in development mode with Vite hot reload
"""
import os
import shutil
import socket
import subprocess
import sys
import time
import webbrowser
from pathlib import Path
ROOT = Path(__file__).parent.absolute()
VENV_DIR = ROOT / "venv"
UI_DIR = ROOT / "ui"
def print_step(step: int, total: int, message: str) -> None:
"""Print a formatted step message."""
print(f"\n[{step}/{total}] {message}")
print("-" * 50)
def find_available_port(start: int = 8888, max_attempts: int = 10) -> int:
"""Find an available port starting from the given port."""
for port in range(start, start + max_attempts):
try:
with socket.socket(socket.AF_INET, socket.SOCK_STREAM) as s:
s.bind(("127.0.0.1", port))
return port
except OSError:
continue
raise RuntimeError(f"No available ports found in range {start}-{start + max_attempts}")
def get_venv_python() -> Path:
"""Get the path to the virtual environment Python executable."""
if sys.platform == "win32":
return VENV_DIR / "Scripts" / "python.exe"
return VENV_DIR / "bin" / "python"
def run_command(cmd: list, cwd: Path | None = None, check: bool = True) -> bool:
"""Run a command and return success status."""
try:
subprocess.run(cmd, cwd=str(cwd) if cwd else None, check=check)
return True
except subprocess.CalledProcessError:
return False
except FileNotFoundError:
return False
def setup_python_venv() -> bool:
"""Create Python virtual environment if it doesn't exist."""
if VENV_DIR.exists() and get_venv_python().exists():
print(" Virtual environment already exists")
return True
print(" Creating virtual environment...")
return run_command([sys.executable, "-m", "venv", str(VENV_DIR)])
def install_python_deps() -> bool:
"""Install Python dependencies."""
venv_python = get_venv_python()
requirements = ROOT / "requirements.txt"
if not requirements.exists():
print(" ERROR: requirements.txt not found")
return False
print(" Installing Python dependencies...")
return run_command([
str(venv_python), "-m", "pip", "install",
"-q", "--upgrade", "pip"
]) and run_command([
str(venv_python), "-m", "pip", "install",
"-q", "-r", str(requirements)
])
def check_node() -> bool:
"""Check if Node.js is installed."""
node = shutil.which("node")
npm = shutil.which("npm")
if not node:
print(" ERROR: Node.js not found")
print(" Please install Node.js from https://nodejs.org")
return False
if not npm:
print(" ERROR: npm not found")
print(" Please install Node.js from https://nodejs.org")
return False
# Get version
try:
result = subprocess.run(
["node", "--version"],
capture_output=True,
text=True
)
print(f" Node.js version: {result.stdout.strip()}")
except Exception:
pass
return True
def install_npm_deps() -> bool:
"""Install npm dependencies if node_modules doesn't exist."""
node_modules = UI_DIR / "node_modules"
if node_modules.exists():
print(" npm dependencies already installed")
return True
print(" Installing npm dependencies (this may take a few minutes)...")
npm_cmd = "npm.cmd" if sys.platform == "win32" else "npm"
return run_command([npm_cmd, "install"], cwd=UI_DIR)
def build_frontend() -> bool:
"""Build the React frontend if dist doesn't exist."""
dist_dir = UI_DIR / "dist"
if dist_dir.exists():
print(" Frontend already built")
return True
print(" Building React frontend...")
npm_cmd = "npm.cmd" if sys.platform == "win32" else "npm"
return run_command([npm_cmd, "run", "build"], cwd=UI_DIR)
def start_dev_server(port: int) -> tuple:
"""Start both Vite and FastAPI in development mode."""
venv_python = get_venv_python()
print("\n Starting development servers...")
print(f" - FastAPI backend: http://127.0.0.1:{port}")
print(" - Vite frontend: http://127.0.0.1:5173")
# Start FastAPI
backend = subprocess.Popen([
str(venv_python), "-m", "uvicorn",
"server.main:app",
"--host", "127.0.0.1",
"--port", str(port),
"--reload"
], cwd=str(ROOT))
# Start Vite with API port env var for proxy configuration
npm_cmd = "npm.cmd" if sys.platform == "win32" else "npm"
vite_env = os.environ.copy()
vite_env["VITE_API_PORT"] = str(port)
frontend = subprocess.Popen([
npm_cmd, "run", "dev"
], cwd=str(UI_DIR), env=vite_env)
return backend, frontend
def start_production_server(port: int):
"""Start FastAPI server in production mode."""
venv_python = get_venv_python()
print(f"\n Starting server at http://127.0.0.1:{port}")
return subprocess.Popen([
str(venv_python), "-m", "uvicorn",
"server.main:app",
"--host", "127.0.0.1",
"--port", str(port)
], cwd=str(ROOT))
def main() -> None:
"""Main entry point."""
dev_mode = "--dev" in sys.argv
print("=" * 50)
print(" AutoCoder UI Setup")
print("=" * 50)
total_steps = 6 if not dev_mode else 5
# Step 1: Python venv
print_step(1, total_steps, "Setting up Python environment")
if not setup_python_venv():
print("ERROR: Failed to create virtual environment")
sys.exit(1)
# Step 2: Python dependencies
print_step(2, total_steps, "Installing Python dependencies")
if not install_python_deps():
print("ERROR: Failed to install Python dependencies")
sys.exit(1)
# Load environment variables now that dotenv is installed
try:
from dotenv import load_dotenv
load_dotenv(ROOT / ".env")
except ImportError:
pass # dotenv is optional for basic functionality
# Step 3: Check Node.js
print_step(3, total_steps, "Checking Node.js")
if not check_node():
sys.exit(1)
# Step 4: npm dependencies
print_step(4, total_steps, "Installing npm dependencies")
if not install_npm_deps():
print("ERROR: Failed to install npm dependencies")
sys.exit(1)
# Step 5: Build frontend (production only)
if not dev_mode:
print_step(5, total_steps, "Building frontend")
if not build_frontend():
print("ERROR: Failed to build frontend")
sys.exit(1)
# Step 6: Start server
step = 5 if dev_mode else 6
print_step(step, total_steps, "Starting server")
port = find_available_port()
try:
if dev_mode:
backend, frontend = start_dev_server(port)
# Open browser to Vite dev server
time.sleep(3)
webbrowser.open("http://127.0.0.1:5173")
print("\n" + "=" * 50)
print(" Development mode active")
print(" Press Ctrl+C to stop")
print("=" * 50)
try:
# Wait for either process to exit
while backend.poll() is None and frontend.poll() is None:
time.sleep(1)
except KeyboardInterrupt:
print("\n\nShutting down...")
finally:
backend.terminate()
frontend.terminate()
backend.wait()
frontend.wait()
else:
server = start_production_server(port)
# Open browser
time.sleep(2)
webbrowser.open(f"http://127.0.0.1:{port}")
print("\n" + "=" * 50)
print(f" Server running at http://127.0.0.1:{port}")
print(" Press Ctrl+C to stop")
print("=" * 50)
try:
server.wait()
except KeyboardInterrupt:
print("\n\nShutting down...")
server.terminate()
server.wait()
except Exception as e:
print(f"\nERROR: {e}")
sys.exit(1)
if __name__ == "__main__":
main()