2025-09-26 09:09:06 +05:30
|
|
|
#!/usr/bin/env bash
|
2025-09-26 10:57:51 +05:30
|
|
|
set -e # Exit on error
|
2025-09-26 09:09:06 +05:30
|
|
|
|
2025-12-27 09:25:20 +05:30
|
|
|
###############################################################################
|
|
|
|
|
### ARGUMENT PARSING
|
|
|
|
|
###############################################################################
|
|
|
|
|
|
|
|
|
|
DEV_MODE=false
|
|
|
|
|
|
|
|
|
|
show_help() {
|
|
|
|
|
echo "Usage: $0 [OPTIONS]"
|
|
|
|
|
echo ""
|
|
|
|
|
echo "Options:"
|
|
|
|
|
echo " --dev Enable development mode with auto-reload for API changes"
|
|
|
|
|
echo " --help Show this help message"
|
|
|
|
|
echo ""
|
|
|
|
|
echo "Examples:"
|
|
|
|
|
echo " $0 # Start in production mode"
|
|
|
|
|
echo " $0 --dev # Start in development mode with auto-reload"
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
while [[ $# -gt 0 ]]; do
|
|
|
|
|
case $1 in
|
|
|
|
|
--dev)
|
|
|
|
|
DEV_MODE=true
|
|
|
|
|
shift
|
|
|
|
|
;;
|
|
|
|
|
--help|-h)
|
|
|
|
|
show_help
|
|
|
|
|
exit 0
|
|
|
|
|
;;
|
|
|
|
|
*)
|
|
|
|
|
echo "Unknown option: $1"
|
|
|
|
|
show_help
|
|
|
|
|
exit 1
|
|
|
|
|
;;
|
|
|
|
|
esac
|
|
|
|
|
done
|
|
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
###############################################################################
|
|
|
|
|
### CONFIGURATION
|
|
|
|
|
###############################################################################
|
|
|
|
|
|
|
|
|
|
# Determine BASE_DIR as parent of the scripts directory
|
|
|
|
|
BASE_DIR="$(cd "$(dirname "$(dirname "${BASH_SOURCE[0]}")")" && pwd)"
|
|
|
|
|
|
|
|
|
|
ENV_FILE="$BASE_DIR/api/.env"
|
|
|
|
|
RUN_DIR="$BASE_DIR/run" # Where we keep *.pid
|
|
|
|
|
BASE_LOG_DIR="$BASE_DIR/logs" # Base logs directory
|
|
|
|
|
|
2025-09-27 22:53:11 +05:30
|
|
|
TIMESTAMP=$(date +"%Y%m%d_%H%M%S")
|
2025-10-09 17:54:31 +05:30
|
|
|
LOG_DIR="$BASE_LOG_DIR/$TIMESTAMP" # Timestamped log directory
|
|
|
|
|
LATEST_LINK="$BASE_LOG_DIR/latest" # Symlink to latest logs
|
2025-12-27 09:25:20 +05:30
|
|
|
VENV_PATH="$BASE_DIR/venv"
|
2025-10-09 17:54:31 +05:30
|
|
|
|
2025-09-26 10:57:51 +05:30
|
|
|
ARQ_WORKERS=${ARQ_WORKERS:-1}
|
2026-02-05 13:10:33 +05:30
|
|
|
LOG_TO_FILE=${LOG_TO_FILE:-true} # Set to false in Docker to use stdout
|
|
|
|
|
WAIT_FOR_PROCESSES=${WAIT_FOR_PROCESSES:-false} # Set to true in Docker to keep container alive
|
2025-09-26 10:57:51 +05:30
|
|
|
|
|
|
|
|
# Log startup
|
2025-10-09 17:54:31 +05:30
|
|
|
cd "$BASE_DIR"
|
2025-12-27 09:25:20 +05:30
|
|
|
if $DEV_MODE; then
|
|
|
|
|
echo "Starting Dograh Services (DEV MODE) at $(date) in BASE_DIR: ${BASE_DIR}"
|
|
|
|
|
echo "Auto-reload enabled for api/ directory changes"
|
|
|
|
|
else
|
|
|
|
|
echo "Starting Dograh Services at $(date) in BASE_DIR: ${BASE_DIR}"
|
|
|
|
|
fi
|
2025-09-26 09:09:06 +05:30
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
###############################################################################
|
|
|
|
|
### 1) Load environment variables
|
|
|
|
|
###############################################################################
|
2025-09-26 09:09:06 +05:30
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
# Load environment from a file if it exists
|
|
|
|
|
if [[ -f "$ENV_FILE" ]]; then
|
|
|
|
|
set -a && . "$ENV_FILE" && set +a
|
2025-09-26 09:09:06 +05:30
|
|
|
fi
|
|
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
FASTAPI_PORT=${FASTAPI_PORT:-8000}
|
2026-02-19 18:30:04 +05:30
|
|
|
CPU_CORES=$(nproc 2>/dev/null || sysctl -n hw.ncpu 2>/dev/null || echo 1)
|
|
|
|
|
FASTAPI_WORKERS=${FASTAPI_WORKERS:-$CPU_CORES}
|
2025-09-26 09:09:06 +05:30
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
###############################################################################
|
|
|
|
|
### 2) Define services
|
|
|
|
|
###############################################################################
|
|
|
|
|
|
|
|
|
|
# Map "service name" → "command to run"
|
|
|
|
|
# Using arrays for bash 3.2 compatibility
|
|
|
|
|
SERVICE_NAMES=(
|
|
|
|
|
"ari_manager"
|
|
|
|
|
"campaign_orchestrator"
|
|
|
|
|
"uvicorn"
|
|
|
|
|
)
|
|
|
|
|
|
2025-12-27 09:25:20 +05:30
|
|
|
# Build uvicorn command based on mode
|
|
|
|
|
if $DEV_MODE; then
|
|
|
|
|
# Dev mode: single worker with auto-reload (--reload is incompatible with --workers > 1)
|
|
|
|
|
UVICORN_CMD="uvicorn api.app:app --host 0.0.0.0 --port $FASTAPI_PORT --reload --reload-dir api"
|
|
|
|
|
else
|
|
|
|
|
# Production mode: multiple workers, no reload
|
|
|
|
|
UVICORN_CMD="uvicorn api.app:app --host 0.0.0.0 --port $FASTAPI_PORT --workers $FASTAPI_WORKERS"
|
|
|
|
|
fi
|
|
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
SERVICE_COMMANDS=(
|
|
|
|
|
"python -m api.services.telephony.ari_manager"
|
|
|
|
|
"python -m api.services.campaign.campaign_orchestrator"
|
2025-12-27 09:25:20 +05:30
|
|
|
"$UVICORN_CMD"
|
2025-09-26 09:09:06 +05:30
|
|
|
)
|
|
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
# Add ARQ workers dynamically
|
2025-09-26 09:09:06 +05:30
|
|
|
for ((i=1; i<=ARQ_WORKERS; i++)); do
|
2025-10-09 17:54:31 +05:30
|
|
|
SERVICE_NAMES+=("arq$i")
|
|
|
|
|
SERVICE_COMMANDS+=("python -m arq api.tasks.arq.WorkerSettings --custom-log-dict api.tasks.arq.LOG_CONFIG")
|
2025-09-26 09:09:06 +05:30
|
|
|
done
|
|
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
###############################################################################
|
|
|
|
|
### 3) Activate virtual environment
|
|
|
|
|
###############################################################################
|
|
|
|
|
|
|
|
|
|
if [[ -d "$VENV_PATH" && -f "$VENV_PATH/bin/activate" ]]; then
|
|
|
|
|
source "$VENV_PATH/bin/activate"
|
|
|
|
|
echo "Virtual environment activated: $VENV_PATH"
|
|
|
|
|
else
|
|
|
|
|
echo "Warning: Virtual environment not found at $VENV_PATH"
|
|
|
|
|
echo "Continuing without virtual environment activation..."
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
###############################################################################
|
|
|
|
|
### 4) Stop old services
|
|
|
|
|
###############################################################################
|
2025-09-26 09:09:06 +05:30
|
|
|
|
|
|
|
|
mkdir -p "$RUN_DIR"
|
2025-12-01 11:32:28 +05:30
|
|
|
|
|
|
|
|
# Function to get all descendant PIDs of a process (children, grandchildren, etc.)
|
|
|
|
|
get_descendants() {
|
|
|
|
|
local parent_pid=$1
|
|
|
|
|
local descendants=""
|
|
|
|
|
local children
|
|
|
|
|
|
|
|
|
|
# Get direct children
|
|
|
|
|
children=$(pgrep -P "$parent_pid" 2>/dev/null || true)
|
|
|
|
|
|
|
|
|
|
for child in $children; do
|
|
|
|
|
# Recursively get descendants of each child
|
|
|
|
|
descendants="$descendants $child $(get_descendants "$child")"
|
|
|
|
|
done
|
|
|
|
|
|
|
|
|
|
echo "$descendants"
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
# Function to kill a process and all its descendants
|
|
|
|
|
kill_process_tree() {
|
|
|
|
|
local pid=$1
|
|
|
|
|
local signal=$2
|
|
|
|
|
local descendants
|
|
|
|
|
|
|
|
|
|
descendants=$(get_descendants "$pid")
|
|
|
|
|
|
|
|
|
|
# Kill children first (bottom-up), then parent
|
|
|
|
|
for desc_pid in $descendants; do
|
|
|
|
|
if kill -0 "$desc_pid" 2>/dev/null; then
|
|
|
|
|
kill "$signal" "$desc_pid" 2>/dev/null || true
|
|
|
|
|
fi
|
|
|
|
|
done
|
|
|
|
|
|
|
|
|
|
# Kill the parent
|
|
|
|
|
if kill -0 "$pid" 2>/dev/null; then
|
|
|
|
|
kill "$signal" "$pid" 2>/dev/null || true
|
|
|
|
|
fi
|
|
|
|
|
}
|
|
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
for name in "${SERVICE_NAMES[@]}"; do
|
2025-09-26 09:09:06 +05:30
|
|
|
pidfile="$RUN_DIR/$name.pid"
|
2025-10-09 17:54:31 +05:30
|
|
|
|
2025-09-26 09:09:06 +05:30
|
|
|
if [[ -f $pidfile ]]; then
|
|
|
|
|
oldpid=$(<"$pidfile")
|
2025-10-09 17:54:31 +05:30
|
|
|
|
|
|
|
|
if kill -0 "$oldpid" 2>/dev/null; then
|
2025-12-01 11:32:28 +05:30
|
|
|
echo "Stopping $name (PID $oldpid and all descendants)…"
|
2025-10-09 17:54:31 +05:30
|
|
|
|
2025-12-01 11:32:28 +05:30
|
|
|
# Kill the entire process tree (parent + all descendants)
|
|
|
|
|
kill_process_tree "$oldpid" "-TERM"
|
2025-09-26 09:09:06 +05:30
|
|
|
sleep 4
|
2025-10-09 17:54:31 +05:30
|
|
|
|
2025-12-01 11:32:28 +05:30
|
|
|
# Check if parent or any descendants are still alive
|
|
|
|
|
still_alive=false
|
2025-10-09 17:54:31 +05:30
|
|
|
if kill -0 "$oldpid" 2>/dev/null; then
|
2025-12-01 11:32:28 +05:30
|
|
|
still_alive=true
|
|
|
|
|
else
|
|
|
|
|
for desc_pid in $(get_descendants "$oldpid"); do
|
|
|
|
|
if kill -0 "$desc_pid" 2>/dev/null; then
|
|
|
|
|
still_alive=true
|
|
|
|
|
break
|
|
|
|
|
fi
|
|
|
|
|
done
|
|
|
|
|
fi
|
|
|
|
|
|
|
|
|
|
if $still_alive; then
|
2025-09-26 09:09:06 +05:30
|
|
|
echo "⚠️ $name did not exit cleanly, forcing stop..."
|
2025-12-01 11:32:28 +05:30
|
|
|
kill_process_tree "$oldpid" "-KILL"
|
2025-09-26 09:09:06 +05:30
|
|
|
sleep 1
|
|
|
|
|
fi
|
|
|
|
|
fi
|
2025-10-09 17:54:31 +05:30
|
|
|
|
2025-09-26 09:09:06 +05:30
|
|
|
rm -f "$pidfile"
|
|
|
|
|
else
|
|
|
|
|
echo "No PID file for $name, skipping stop."
|
|
|
|
|
fi
|
|
|
|
|
done
|
|
|
|
|
|
|
|
|
|
# Clean up any port tracking files for uvicorn
|
|
|
|
|
rm -f "$RUN_DIR/uvicorn.port" "$RUN_DIR/uvicorn_new.port" "$RUN_DIR/uvicorn_old.pid"
|
|
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
###############################################################################
|
|
|
|
|
### 5) Run migrations
|
|
|
|
|
###############################################################################
|
|
|
|
|
|
|
|
|
|
alembic -c "$BASE_DIR/api/alembic.ini" upgrade head
|
2025-09-26 09:09:06 +05:30
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
###############################################################################
|
|
|
|
|
### 6) Prepare logs
|
|
|
|
|
###############################################################################
|
2025-09-26 09:09:06 +05:30
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
mkdir -p "$BASE_LOG_DIR" "$LOG_DIR"
|
|
|
|
|
|
|
|
|
|
# Remove old symlink and create a new one
|
2025-09-27 22:53:11 +05:30
|
|
|
if [[ -L "$LATEST_LINK" ]]; then
|
|
|
|
|
rm "$LATEST_LINK"
|
|
|
|
|
fi
|
|
|
|
|
ln -s "$TIMESTAMP" "$LATEST_LINK"
|
|
|
|
|
|
|
|
|
|
echo "Log directory: $LOG_DIR"
|
|
|
|
|
echo "Latest symlink: $LATEST_LINK -> $TIMESTAMP"
|
|
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
###############################################################################
|
|
|
|
|
### 7) Start services
|
|
|
|
|
###############################################################################
|
|
|
|
|
|
|
|
|
|
for i in "${!SERVICE_NAMES[@]}"; do
|
|
|
|
|
name="${SERVICE_NAMES[$i]}"
|
|
|
|
|
cmd="${SERVICE_COMMANDS[$i]}"
|
2025-09-26 10:57:51 +05:30
|
|
|
echo "→ Starting $name"
|
2025-10-09 17:54:31 +05:30
|
|
|
|
|
|
|
|
(
|
|
|
|
|
cd "$BASE_DIR"
|
2026-02-05 13:10:33 +05:30
|
|
|
if [[ "$LOG_TO_FILE" == "true" ]]; then
|
|
|
|
|
export LOG_FILE_PATH="$LOG_DIR/$name.log"
|
|
|
|
|
exec $cmd >>"$LOG_DIR/$name.log" 2>&1
|
|
|
|
|
else
|
|
|
|
|
# Log to stdout/stderr for Docker
|
|
|
|
|
exec $cmd
|
|
|
|
|
fi
|
2025-10-09 17:54:31 +05:30
|
|
|
) &
|
|
|
|
|
|
2025-09-26 09:09:06 +05:30
|
|
|
pid=$!
|
|
|
|
|
echo $pid >"$RUN_DIR/$name.pid"
|
2025-10-09 17:54:31 +05:30
|
|
|
echo " Started with PID $pid"
|
|
|
|
|
|
2025-09-26 09:09:06 +05:30
|
|
|
if [[ "$name" == "uvicorn" ]]; then
|
|
|
|
|
echo "$FASTAPI_PORT" >"$RUN_DIR/uvicorn.port"
|
|
|
|
|
fi
|
|
|
|
|
done
|
|
|
|
|
|
2025-10-09 17:54:31 +05:30
|
|
|
###############################################################################
|
|
|
|
|
### 8) Summary
|
|
|
|
|
###############################################################################
|
|
|
|
|
|
2025-09-26 09:09:06 +05:30
|
|
|
echo
|
|
|
|
|
echo "──────────────────────────────────────────────────"
|
2025-12-27 09:25:20 +05:30
|
|
|
if $DEV_MODE; then
|
|
|
|
|
echo "Mode: DEVELOPMENT (auto-reload enabled)"
|
|
|
|
|
else
|
|
|
|
|
echo "Mode: PRODUCTION"
|
|
|
|
|
fi
|
|
|
|
|
echo ""
|
2025-10-09 17:54:31 +05:30
|
|
|
for name in "${SERVICE_NAMES[@]}"; do
|
2025-09-26 09:09:06 +05:30
|
|
|
pid=$(<"$RUN_DIR/$name.pid")
|
|
|
|
|
echo "✓ $name (PID $pid) → $LOG_DIR/$name.log"
|
|
|
|
|
done
|
2025-12-27 09:25:20 +05:30
|
|
|
echo ""
|
2025-09-26 09:09:06 +05:30
|
|
|
echo " Rotation: ${LOG_ROTATION_SIZE:-100 MB}"
|
|
|
|
|
echo " Retention: ${LOG_RETENTION:-7 days}"
|
|
|
|
|
echo " Compression: ${LOG_COMPRESSION:-gz}"
|
|
|
|
|
echo "Logs: tail -f $LOG_DIR/*.log"
|
|
|
|
|
echo "Rotated logs: ls $LOG_DIR/*.log.*"
|
2025-12-27 09:25:20 +05:30
|
|
|
echo "To stop: ./scripts/stop_services.sh"
|
2025-09-26 09:09:06 +05:30
|
|
|
echo "──────────────────────────────────────────────────"
|
2026-02-05 13:10:33 +05:30
|
|
|
|
|
|
|
|
# In Docker mode, wait for all background processes to keep container alive
|
|
|
|
|
if [[ "$WAIT_FOR_PROCESSES" == "true" ]]; then
|
|
|
|
|
wait
|
|
|
|
|
fi
|