-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathrun.sh
More file actions
executable file
·394 lines (339 loc) · 11.9 KB
/
run.sh
File metadata and controls
executable file
·394 lines (339 loc) · 11.9 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
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
#!/bin/bash
# RouteX - Complete Startup Script
# Starts backend (FastAPI + Celery), Redis, and frontend servers
set -e
# Colors for output
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
BLUE='\033[0;34m'
NC='\033[0m' # No Color
# Configuration
BACKEND_PORT=8000
FRONTEND_PORT=3000
REDIS_PORT=6379
DOCKER_REDIS_CONTAINER="routex-redis"
# Platform detection
OS_NAME="$(uname -s)"
IS_WINDOWS=false
case "$OS_NAME" in
MINGW*|MSYS*|CYGWIN*)
IS_WINDOWS=true
;;
esac
# PID tracking
BACKEND_PID=""
FRONTEND_PID=""
REDIS_PID=""
CELERY_WORKER_PID=""
CELERY_BEAT_PID=""
REDIS_CONTAINER_STARTED=false
command_exists() {
command -v "$1" >/dev/null 2>&1
}
stop_pid() {
local pid="$1"
[ -z "$pid" ] && return 0
if kill -0 "$pid" 2>/dev/null; then
kill "$pid" 2>/dev/null || true
fi
}
port_in_use() {
local port="$1"
if command_exists lsof; then
lsof -Pi :"$port" -sTCP:LISTEN -t >/dev/null 2>&1
return $?
fi
if command_exists netstat; then
# netstat format differs by platform; this pattern matches both.
netstat -an 2>/dev/null | grep -E "[:.]${port}[[:space:]]" | grep -Ei "LISTEN|LISTENING" >/dev/null 2>&1
return $?
fi
echo -e "${YELLOW}⚠️ Could not check if port $port is in use (no lsof/netstat found)${NC}"
return 1
}
redis_ping() {
if command_exists redis-cli; then
redis-cli -p "$REDIS_PORT" ping >/dev/null 2>&1
return $?
fi
if [ "$REDIS_CONTAINER_STARTED" = true ] && command_exists docker; then
docker exec "$DOCKER_REDIS_CONTAINER" redis-cli -p 6379 ping >/dev/null 2>&1
return $?
fi
return 1
}
# Get project root directory
PROJECT_ROOT="$(cd "$(dirname "$0")" && pwd)"
echo -e "${BLUE}🚢 Starting RouteX - HPCL Coastal Tanker Fleet Optimizer${NC}"
echo "=========================================================="
# Cleanup function
cleanup() {
echo ""
echo -e "${YELLOW}🛑 Stopping all services...${NC}"
# Kill processes by PID
stop_pid "$CELERY_BEAT_PID"
stop_pid "$CELERY_WORKER_PID"
stop_pid "$FRONTEND_PID"
stop_pid "$BACKEND_PID"
stop_pid "$REDIS_PID"
if [ "$REDIS_CONTAINER_STARTED" = true ] && command_exists docker; then
docker rm -f "$DOCKER_REDIS_CONTAINER" >/dev/null 2>&1 || true
fi
# Kill any remaining processes
if command_exists pkill; then
pkill -f "celery.*worker" 2>/dev/null || true
pkill -f "celery.*beat" 2>/dev/null || true
pkill -f "uvicorn.*backend.app.main:app" 2>/dev/null || true
pkill -f "next dev" 2>/dev/null || true
pkill -f "redis-server.*$REDIS_PORT" 2>/dev/null || true
fi
if [ "$IS_WINDOWS" = true ] && command_exists taskkill; then
taskkill //F //IM redis-server.exe >/dev/null 2>&1 || true
taskkill //F //IM celery.exe >/dev/null 2>&1 || true
taskkill //F //IM node.exe >/dev/null 2>&1 || true
taskkill //F //IM python.exe >/dev/null 2>&1 || true
fi
echo -e "${GREEN}✅ All services stopped${NC}"
exit 0
}
# Trap signals
trap cleanup INT TERM EXIT
# Check system requirements
echo ""
echo -e "${BLUE}🔍 Checking system requirements...${NC}"
# Check Python
if ! command_exists python3; then
echo -e "${RED}❌ Python 3 is required but not found${NC}"
echo " Please install Python 3.9 or higher"
exit 1
fi
PYTHON_VERSION=$(python3 --version)
echo -e "${GREEN}✅ ${PYTHON_VERSION}${NC}"
# Check Node.js
if ! command_exists node; then
echo -e "${RED}❌ Node.js is required but not found${NC}"
echo " Please install Node.js 18 or higher"
exit 1
fi
NODE_VERSION=$(node --version)
echo -e "${GREEN}✅ Node.js ${NODE_VERSION}${NC}"
# Check Redis
if ! command_exists redis-server; then
if [ "$IS_WINDOWS" = true ]; then
if command_exists docker; then
echo -e "${YELLOW}⚠️ Redis server not found. Will use Docker container redis:7-alpine${NC}"
REDIS_CONTAINER_STARTED=true
else
echo -e "${RED}❌ Redis not found on Windows and Docker is unavailable${NC}"
echo " Install Docker Desktop (recommended) or Redis for Windows, then re-run."
exit 1
fi
else
echo -e "${YELLOW}⚠️ Redis not found - installing via Homebrew...${NC}"
if command_exists brew; then
brew install redis
else
echo -e "${RED}❌ Homebrew not found. Please install Redis manually:${NC}"
echo " brew install redis"
exit 1
fi
fi
fi
echo -e "${GREEN}✅ Redis available${NC}"
# Backend Setup
echo ""
echo -e "${BLUE}📦 Setting up Backend...${NC}"
cd "$PROJECT_ROOT"
# Create virtual environment if it doesn't exist
if [ ! -d "venv" ]; then
echo "🔧 Creating virtual environment..."
python3 -m venv venv
fi
# Activate virtual environment
if [ "$IS_WINDOWS" = true ] && [ -f "venv/Scripts/activate" ]; then
# Git Bash on Windows uses the Scripts activation path.
source venv/Scripts/activate
else
source venv/bin/activate
fi
# Install backend dependencies
if [ ! -f "venv/.dependencies_installed" ] || [ backend/requirements.txt -nt venv/.dependencies_installed ]; then
echo "📦 Installing backend dependencies..."
pip install --upgrade pip --quiet
pip install -r backend/requirements.txt --quiet
touch venv/.dependencies_installed
echo -e "${GREEN}✅ Backend dependencies installed${NC}"
else
echo -e "${GREEN}✅ Backend dependencies already installed${NC}"
fi
# Frontend Setup
echo ""
echo -e "${BLUE}📦 Setting up Frontend...${NC}"
cd "$PROJECT_ROOT/frontend"
# Install frontend dependencies
if [ ! -d "node_modules" ] || [ ! -f "node_modules/.deps_installed" ] || [ package-lock.json -nt node_modules/.deps_installed ]; then
echo "📦 Installing frontend dependencies..."
npm install --silent
touch node_modules/.deps_installed
echo -e "${GREEN}✅ Frontend dependencies installed${NC}"
else
echo -e "${GREEN}✅ Frontend dependencies already installed${NC}"
fi
cd "$PROJECT_ROOT"
# Check if ports are available
echo ""
echo -e "${BLUE}🔍 Checking ports...${NC}"
check_port() {
local port=$1
local service=$2
if port_in_use "$port"; then
echo -e "${RED}❌ Port $port is already in use (required for $service)${NC}"
if command_exists lsof; then
echo " Kill the process using: lsof -ti:$port | xargs kill -9"
else
echo " Kill the process and re-run the script"
fi
return 1
else
echo -e "${GREEN}✅ Port $port is available ($service)${NC}"
return 0
fi
}
check_port $BACKEND_PORT "Backend API" || exit 1
check_port $FRONTEND_PORT "Frontend" || exit 1
check_port $REDIS_PORT "Redis" || exit 1
# Start services
echo ""
echo -e "${BLUE}🚀 Starting services...${NC}"
echo ""
# 1. Start Redis
echo -e "${BLUE}▶️ Starting Redis on port $REDIS_PORT...${NC}"
if [ "$REDIS_CONTAINER_STARTED" = true ]; then
docker rm -f "$DOCKER_REDIS_CONTAINER" >/dev/null 2>&1 || true
docker run -d --name "$DOCKER_REDIS_CONTAINER" -p "$REDIS_PORT:6379" redis:7-alpine >/dev/null
else
redis-server --port "$REDIS_PORT" --logfile "$PROJECT_ROOT/redis.log" > /dev/null 2>&1 &
REDIS_PID=$!
fi
sleep 2
# Verify Redis is running
if ! redis_ping; then
echo -e "${RED}❌ Redis failed to start${NC}"
exit 1
fi
if [ "$REDIS_CONTAINER_STARTED" = true ]; then
REDIS_PID=$(docker ps --filter "name=$DOCKER_REDIS_CONTAINER" --format "{{.ID}}")
echo -e "${GREEN}✅ Redis started in Docker (Container: $REDIS_PID)${NC}"
else
echo -e "${GREEN}✅ Redis started (PID: $REDIS_PID)${NC}"
fi
# 2. Start Backend API
echo -e "${BLUE}▶️ Starting Backend API on port $BACKEND_PORT...${NC}"
cd "$PROJECT_ROOT"
python -m uvicorn backend.app.main:app --host 0.0.0.0 --port $BACKEND_PORT --reload > backend.log 2>&1 &
BACKEND_PID=$!
sleep 3
# Verify backend is running
if ! kill -0 $BACKEND_PID 2>/dev/null; then
echo -e "${RED}❌ Backend failed to start. Check backend.log${NC}"
cat backend.log
exit 1
fi
echo -e "${GREEN}✅ Backend API started (PID: $BACKEND_PID)${NC}"
# 3. Start Celery Worker (optional - for background tasks)
echo -e "${BLUE}▶️ Starting Celery Worker...${NC}"
cd "$PROJECT_ROOT/backend"
celery -A app.core.celery_app worker --loglevel=info --logfile="$PROJECT_ROOT/celery_worker.log" > /dev/null 2>&1 &
CELERY_WORKER_PID=$!
sleep 3
# Verify Celery worker is running
if kill -0 $CELERY_WORKER_PID 2>/dev/null; then
echo -e "${GREEN}✅ Celery Worker started (PID: $CELERY_WORKER_PID)${NC}"
# 4. Start Celery Beat scheduler
echo -e "${BLUE}▶️ Starting Celery Beat scheduler...${NC}"
celery -A app.core.celery_app beat --loglevel=info --logfile="$PROJECT_ROOT/celery_beat.log" > /dev/null 2>&1 &
CELERY_BEAT_PID=$!
sleep 2
# Verify Celery beat is running
if kill -0 $CELERY_BEAT_PID 2>/dev/null; then
echo -e "${GREEN}✅ Celery Beat started (PID: $CELERY_BEAT_PID)${NC}"
else
echo -e "${YELLOW}⚠️ Celery Beat failed to start (non-critical, check celery_beat.log)${NC}"
CELERY_BEAT_PID=""
fi
else
echo -e "${YELLOW}⚠️ Celery Worker failed to start (non-critical, background tasks disabled)${NC}"
echo -e "${YELLOW} This is normal - the API will work without Celery${NC}"
CELERY_WORKER_PID=""
CELERY_BEAT_PID=""
fi
# 5. Start Frontend
echo -e "${BLUE}▶️ Starting Frontend on port $FRONTEND_PORT...${NC}"
cd "$PROJECT_ROOT/frontend"
npm run dev > "$PROJECT_ROOT/frontend.log" 2>&1 &
FRONTEND_PID=$!
sleep 5
# Verify frontend is running
if ! kill -0 $FRONTEND_PID 2>/dev/null; then
echo -e "${RED}❌ Frontend failed to start. Check frontend.log${NC}"
cat "$PROJECT_ROOT/frontend.log"
exit 1
fi
echo -e "${GREEN}✅ Frontend started (PID: $FRONTEND_PID)${NC}"
cd "$PROJECT_ROOT"
# Display success message
echo ""
echo "=========================================================="
echo -e "${GREEN}✅ RouteX is fully operational!${NC}"
echo "=========================================================="
echo ""
echo -e "${BLUE}📊 Service URLs:${NC}"
echo " 🌐 Frontend: http://localhost:$FRONTEND_PORT"
echo " 📚 Backend API: http://localhost:$BACKEND_PORT"
echo " 📖 API Docs: http://localhost:$BACKEND_PORT/docs"
echo " 📖 ReDoc: http://localhost:$BACKEND_PORT/redoc"
echo " 💊 Health Check: http://localhost:$BACKEND_PORT/health"
echo ""
echo -e "${BLUE}📝 Log Files:${NC}"
echo " Backend: $PROJECT_ROOT/backend.log"
echo " Frontend: $PROJECT_ROOT/frontend.log"
echo " Redis: $PROJECT_ROOT/redis.log"
echo " Celery: $PROJECT_ROOT/celery_worker.log"
echo " Beat: $PROJECT_ROOT/celery_beat.log"
echo ""
echo -e "${BLUE}🔧 Running Services:${NC}"
echo " Redis: PID $REDIS_PID"
echo " Backend API: PID $BACKEND_PID"
if [ ! -z "$CELERY_WORKER_PID" ]; then
echo " Celery Worker: PID $CELERY_WORKER_PID"
fi
if [ ! -z "$CELERY_BEAT_PID" ]; then
echo " Celery Beat: PID $CELERY_BEAT_PID"
fi
echo " Frontend: PID $FRONTEND_PID"
echo ""
echo -e "${YELLOW}Press Ctrl+C to stop all services${NC}"
echo "=========================================================="
# Monitor processes and restart if they crash
while true; do
# Check if any critical process died
if ! kill -0 $BACKEND_PID 2>/dev/null; then
echo -e "${RED}❌ Backend crashed! Check backend.log${NC}"
cleanup
fi
if ! kill -0 $FRONTEND_PID 2>/dev/null; then
echo -e "${RED}❌ Frontend crashed! Check frontend.log${NC}"
cleanup
fi
if ! redis_ping; then
echo -e "${RED}❌ Redis crashed! Check redis.log${NC}"
cleanup
fi
# Optional: Check Celery worker if it's running
if [ ! -z "$CELERY_WORKER_PID" ] && ! kill -0 $CELERY_WORKER_PID 2>/dev/null; then
echo -e "${YELLOW}⚠️ Celery worker stopped (non-critical)${NC}"
CELERY_WORKER_PID=""
fi
sleep 5
done