#!/bin/bash # Colors for output RED='\033[0;31m' GREEN='\033[0;32m' YELLOW='\033[1;33m' BLUE='\033[0;34m' NC='\033[0m' # No Color SERVICE_NAME="kspguti-schedule" INSTALL_DIR="/opt/kspguti-schedule" show_usage() { echo -e "${BLUE}Usage: $0 {start|stop|restart|status|logs|update|enable|disable}${NC}" echo "" echo "Commands:" echo " start - Start the service" echo " stop - Stop the service" echo " restart - Restart the service" echo " status - Show service status" echo " logs - Show service logs (use -f for follow)" echo " update - Update the application (pull, install, build, restart)" echo " enable - Enable service to start on boot" echo " disable - Disable service from starting on boot" exit 1 } check_root() { if [ "$1" != "status" ] && [ "$1" != "logs" ] && [ "$EUID" -ne 0 ]; then echo -e "${RED}This command requires root privileges (use sudo)${NC}" exit 1 fi } case "$1" in start) check_root "$1" echo -e "${YELLOW}Starting $SERVICE_NAME...${NC}" systemctl start "$SERVICE_NAME" systemctl status "$SERVICE_NAME" --no-pager ;; stop) check_root "$1" echo -e "${YELLOW}Stopping $SERVICE_NAME...${NC}" systemctl stop "$SERVICE_NAME" echo -e "${GREEN}Service stopped${NC}" ;; restart) check_root "$1" echo -e "${YELLOW}Restarting $SERVICE_NAME...${NC}" systemctl restart "$SERVICE_NAME" systemctl status "$SERVICE_NAME" --no-pager ;; status) systemctl status "$SERVICE_NAME" --no-pager ;; logs) if [ "$2" == "-f" ]; then journalctl -u "$SERVICE_NAME" -f else journalctl -u "$SERVICE_NAME" -n 50 --no-pager fi ;; update) check_root "$1" echo -e "${YELLOW}Updating $SERVICE_NAME...${NC}" if [ ! -d "$INSTALL_DIR" ]; then echo -e "${RED}Installation directory not found: $INSTALL_DIR${NC}" exit 1 fi # Get the directory where the script is located (project root) SCRIPT_DIR="$( cd "$( dirname "${BASH_SOURCE[0]}" )" && pwd )" PROJECT_DIR="$( cd "$SCRIPT_DIR/.." && pwd )" echo -e "${YELLOW}Project directory: $PROJECT_DIR${NC}" echo -e "${YELLOW}Installation directory: $INSTALL_DIR${NC}" # Stop service echo -e "${YELLOW}Stopping service...${NC}" systemctl stop "$SERVICE_NAME" # Pull latest changes in project root (if using git) cd "$PROJECT_DIR" if [ -d ".git" ]; then echo -e "${YELLOW}Pulling latest changes from git...${NC}" git pull else echo -e "${YELLOW}Not a git repository, skipping pull${NC}" fi # Copy updated files to installation directory echo -e "${YELLOW}Copying updated files to installation directory...${NC}" rsync -av --exclude='node_modules' \ --exclude='.next' \ --exclude='.git' \ --exclude='*.log' \ --exclude='.env.local' \ --exclude='.env.development' \ --exclude='.env.development.local' \ --exclude='.env.test' \ --exclude='.env.test.local' \ --exclude='*.md' \ "$PROJECT_DIR/" "$INSTALL_DIR/" # Handle .env file (preserve existing if present) if [ -f "$INSTALL_DIR/.env" ]; then echo -e "${GREEN}Preserving existing .env file${NC}" elif [ -f "$PROJECT_DIR/.env" ]; then echo -e "${YELLOW}Copying .env file...${NC}" cp "$PROJECT_DIR/.env" "$INSTALL_DIR/.env" fi # Change to installation directory for build cd "$INSTALL_DIR" # Install dependencies (with check) echo -e "${YELLOW}Checking dependencies...${NC}" # Check if node_modules exists and is up to date NEED_INSTALL=true LOCK_FILE="" if [ -f "package-lock.json" ]; then LOCK_FILE="package-lock.json" elif [ -f "pnpm-lock.yaml" ]; then LOCK_FILE="pnpm-lock.yaml" fi if [ -d "node_modules" ] && [ -n "$LOCK_FILE" ]; then # Check if package.json is newer than lock file if [ "package.json" -nt "$LOCK_FILE" ]; then echo -e "${YELLOW}package.json is newer than $LOCK_FILE, reinstalling...${NC}" NEED_INSTALL=true else # Check if all dependencies are installed by checking if node_modules/.bin exists and has entries if [ -d "node_modules/.bin" ] && [ "$(ls -A node_modules/.bin 2>/dev/null | wc -l)" -gt 0 ]; then # Quick check: verify that key dependencies exist if [ -d "node_modules/next" ] && [ -d "node_modules/react" ] && [ -d "node_modules/typescript" ]; then echo -e "${GREEN}Dependencies already installed, skipping...${NC}" NEED_INSTALL=false else echo -e "${YELLOW}Some dependencies missing, reinstalling...${NC}" NEED_INSTALL=true fi else echo -e "${YELLOW}node_modules appears incomplete, reinstalling...${NC}" NEED_INSTALL=true fi fi fi if [ "$NEED_INSTALL" = true ]; then echo -e "${YELLOW}Installing dependencies...${NC}" npm ci --legacy-peer-deps --production=false else echo -e "${GREEN}Dependencies are up to date, skipping installation${NC}" fi # Build echo -e "${YELLOW}Building application...${NC}" npm run build # Ensure public directory is accessible in standalone build echo -e "${YELLOW}Setting up static files...${NC}" if [ -d "$INSTALL_DIR/.next/standalone" ]; then # Copy public to standalone directory if it's not there if [ ! -d "$INSTALL_DIR/.next/standalone/public" ]; then cp -r "$INSTALL_DIR/public" "$INSTALL_DIR/.next/standalone/public" 2>/dev/null || true fi # Ensure .next/static is accessible from standalone if [ ! -d "$INSTALL_DIR/.next/standalone/.next" ]; then mkdir -p "$INSTALL_DIR/.next/standalone/.next" fi if [ ! -d "$INSTALL_DIR/.next/standalone/.next/static" ]; then cp -r "$INSTALL_DIR/.next/static" "$INSTALL_DIR/.next/standalone/.next/static" 2>/dev/null || true fi fi # Set ownership chown -R www-data:www-data "$INSTALL_DIR" # Reload systemd and restart systemctl daemon-reload echo -e "${YELLOW}Starting service...${NC}" systemctl start "$SERVICE_NAME" echo -e "${GREEN}Update completed!${NC}" systemctl status "$SERVICE_NAME" --no-pager ;; enable) check_root "$1" echo -e "${YELLOW}Enabling $SERVICE_NAME to start on boot...${NC}" systemctl enable "$SERVICE_NAME" echo -e "${GREEN}Service enabled${NC}" ;; disable) check_root "$1" echo -e "${YELLOW}Disabling $SERVICE_NAME from starting on boot...${NC}" systemctl disable "$SERVICE_NAME" echo -e "${GREEN}Service disabled${NC}" ;; *) show_usage ;; esac exit 0