#!/bin/bash
# Complete pipeline script: Start server, run experiments, and view results

# Safety: if this script is accidentally invoked via `sh`/`dash` (ignoring the shebang),
# re-exec it with bash so bash-specific syntax (e.g., [[ ... ]], arrays) won't break later.
if [ -z "${BASH_VERSION:-}" ]; then
    exec /bin/bash "$0" "$@"
fi

set -e

SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
# Determine project root: if script is in multi-turn/, go up one level; otherwise assume we're at root
if [[ "$SCRIPT_DIR" == */multi-turn ]]; then
    PROJECT_ROOT="$(cd "$SCRIPT_DIR/.." && pwd)"
    MULTI_TURN_DIR="$SCRIPT_DIR"
else
    PROJECT_ROOT="$SCRIPT_DIR"
    MULTI_TURN_DIR="$PROJECT_ROOT/multi-turn"
fi

# Change to project root for running the script
cd "$PROJECT_ROOT"

# Colors for output
RED='\033[0;31m'
GREEN='\033[0;32m'
YELLOW='\033[1;33m'
BLUE='\033[0;34m'
NC='\033[0m' # No Color
# Parse command line arguments

NUM_BEHAVIORS="10"  # Empty means process all behaviors
PLAN_GENERATOR="crescendo" # xteaming, coa, crescendo, fitd, actorattack
UPDATER=""  # Empty means use same as PLAN_GENERATOR
OUTPUT_NAME="testrun"
START_SERVER=true
STOP_SERVER=false
RUN_EXPERIMENT=true
VIEW_RESULTS=true
MAX_WORKERS=20
FORCE_RERUN=false
SERVER_NODE="localhost"
REWRITE_SERVER_NODE=""  # Node for rewrite model server
ATTACKER_MODEL="Qwen/Qwen2.5-32B-Instruct"  # Attacker model name (empty means read from config file)
TARGET_MODEL=""  # Target model name (empty means read from config file)
# If set, overrides target provider in config (e.g., local for local OpenAI-compatible server)
TARGET_PROVIDER=""
JUDGE_MODEL=""  # Judge model name (empty means read from config file)
JUDGE_TYPE=""  # Judge type (empty means read from config file; options: cka, xteaming, pair)
REWRITE_MODEL=""  # Rewrite model name (empty means read from config file)  # Rewrite direction for mislead defense: increase|decrease (default: increase)
ENABLE_MISLEAD=false  # Default to enabled (mislead defense / rewrite)
ENABLE_PROACT=false  # Default to disabled (ProAct proactive defense)
ENABLE_GUARD=false   # Default to disabled (Guard defense / LlamaGuard replace-if-unsafe)

# Guard server settings (OpenAI-compatible vLLM server, see deploy/deploy_guard*.sh)
GUARD_SERVER_NODE="localhost"
GUARD_SERVER_PORT="30001"
# Derived (resolved) URL from node+port; will be recomputed after CLI arg parsing too.
GUARD_SERVER_URL="http://${GUARD_SERVER_NODE}:${GUARD_SERVER_PORT}/v1"
GUARD_MODEL="guard"
GUARD_API_MODE="chat"  # "chat" or "completions", default: "chat"
GUARD_MULTI_TURN=false  # true for multi-turn context, false for single-turn (last user/assistant pair only), default: false

usage() {
    echo "Usage: $0 [OPTIONS]"
    echo ""
    echo "Options:"
    echo "  --num-behaviors N       Number of behaviors to process (default: 1; omit/empty to process all)"
    echo "  --plan-generator NAME   Plan generator to use (default: xteaming)"
    echo "                          Options: xteaming, actorattack, coa, crescendo, fitd"
    echo "  --updater NAME          Updater to use (default: same as --plan-generator)"
    echo "                          Options: xteaming, actorattack, coa, crescendo, fitd"
    echo "  --output-name NAME      Output name for results (default: TestRun)"
    echo "  --no-server             Don't start SGLang server (assume it's already running)"
    echo "  --stop-server           Stop server after experiments complete"
    echo "  --no-experiment         Don't run experiments (only start server)"
    echo "  --no-view-results       Don't view results summary (default: view results)"
    echo "  --max-workers N         Max parallel workers (default: 10)"
    echo "  --all-generators        Run all plan generators (overrides --plan-generator)"
    echo "  --force                 Force re-run (deletes existing output dir for this run)"
    echo "  --no-force              Don't force re-run (default: skip if output dir exists)"
    echo "  --server-node NODE      Node where SGLang server is running (default: localhost)"
    echo "                          If set to localhost, deploys local server"
    echo "                          If set to other node, uses remote server"
    echo "                          If set to 'none' or empty, skips server deployment"
    echo "                          Examples: localhost, nid001040, 192.168.1.100"
    echo "  --rewrite-server-node NODE  Node where rewrite model server is running (default: localhost)"
    echo "                          If set to localhost, deploys local server"
    echo "                          If set to other node, uses remote server"
    echo "                          If set to 'none' or empty, uses OpenAI API instead"
    echo "  --attacker-model MODEL  Attacker model name for SGLang server (default: read from config file)"
    echo "                          If set, uses this model instead of reading from config"
    echo "                          Examples: Qwen/Qwen2.5-32B-Instruct, Qwen/Qwen3-4B-Instruct-2507"
    echo "  --target-model MODEL    Target model name (default: read from config file)"
    echo "                          If set, uses this model instead of reading from config"
    echo "                          Examples: gpt-4o, gpt-4o-2024-11-20"
    echo "  --target-provider NAME  Target provider (default: auto-detect from --target-model, else read from config)"
    echo "                          Examples: gemini, openai, aws, local (alias: sglang)"
    echo "  --judge-model MODEL     Judge model name (default: read from config file)"
    echo "                          If set, uses this model instead of reading from config"
    echo "                          Examples: gpt-4o, gpt-4o-2024-11-20"
    echo "  --judge-type TYPE       Judge type (default: read from config file)"
    echo "                          Options: cka, xteaming, pair"
    echo "                          Examples: --judge-type xteaming, --judge-type cka"
    echo "  --add-reframing-note    Add reframing note to judge evaluation (default: disabled)"
    echo "                          When enabled, adds note about response reframing before evaluation"
    echo "  --rewrite-model MODEL   Rewrite model name for defense (default: read from config file)"
    echo "                          If set, uses this model instead of reading from config"
    echo "                          Examples: gpt-4.1-mini, increase (for server mode)"
    echo "  --rewrite-direction DIR Rewrite direction for mislead defense (default: increase)"
    echo "                          Options: increase, decrease"
    echo "  --disable-mislead       Disable mislead defense mechanism (default: enabled)"
    echo "  --enable-proact         Enable ProAct proactive defense (default: disabled)"
    echo "  --disable-proact        Disable ProAct proactive defense (default: disabled)"
    echo "  --enable-guard          Enable Guard defense (default: disabled)"
    echo "  --disable-guard         Disable Guard defense (default: disabled)"
    echo "  --guard-server-node NODE   Node where guard server is running (default: localhost)"
    echo "  --guard-server-port PORT   Guard server port (default: 30001)"
    echo "  --guard-model NAME         Served guard model name (default: guard)"
    echo "  --disable-determinism   Disable determinism (seed) via ENABLE_DETERMINISM=false (default: enabled)"
    echo "  --early-stop-budget N   Updater early-stop budget (sets env EARLY_STOP_BUDGET; code default: 0)"
    echo "  -h, --help              Show this help message"
    exit 1
}

# Parse arguments
ALL_GENERATORS=false
while [[ $# -gt 0 ]]; do
    case $1 in
        --num-behaviors)
            NUM_BEHAVIORS="$2"
            shift 2
            ;;
        --plan-generator)
            PLAN_GENERATOR="$2"
            shift 2
            ;;
        --updater)
            UPDATER="$2"
            shift 2
            ;;
        --output-name)
            OUTPUT_NAME="$2"
            shift 2
            ;;
        --no-server)
            START_SERVER=false
            shift
            ;;
        --stop-server)
            STOP_SERVER=true
            shift
            ;;
        --no-experiment)
            RUN_EXPERIMENT=false
            shift
            ;;
        --view-results)
            VIEW_RESULTS=true
            shift
            ;;
        --no-view-results)
            VIEW_RESULTS=false
            shift
            ;;
        --max-workers)
            MAX_WORKERS="$2"
            shift 2
            ;;
        --all-generators)
            ALL_GENERATORS=true
            shift
            ;;
        --force)
            FORCE_RERUN=true
            shift
            ;;
        --no-force)
            FORCE_RERUN=false
            shift
            ;;
        --server-node)
            SERVER_NODE="$2"
            shift 2
            ;;
        --rewrite-server-node)
            REWRITE_SERVER_NODE="$2"
            shift 2
            ;;
        --attacker-model)
            ATTACKER_MODEL="$2"
            shift 2
            ;;
        --target-model)
            TARGET_MODEL="$2"
            shift 2
            ;;
        --target-provider)
            TARGET_PROVIDER="$2"
            shift 2
            ;;
        --judge-model)
            JUDGE_MODEL="$2"
            shift 2
            ;;
        --judge-type)
            JUDGE_TYPE="$2"
            shift 2
            ;;
        --add-reframing-note)
            ADD_REFRAMING_NOTE=true
            shift
            ;;
        --rewrite-model)
            REWRITE_MODEL="$2"
            shift 2
            ;;
        --rewrite-direction)
            REWRITE_DIRECTION="$2"
            shift 2
            ;;
        --enable-mislead)
            ENABLE_MISLEAD=true
            shift
            ;;
        --disable-mislead)
            ENABLE_MISLEAD=false
            shift
            ;;
        --enable-proact)
            ENABLE_PROACT=true
            shift
            ;;
        --disable-proact)
            ENABLE_PROACT=false
            shift
            ;;
        --enable-guard)
            ENABLE_GUARD=true
            shift
            ;;
        --disable-guard)
            ENABLE_GUARD=false
            shift
            ;;
        --guard-server-node)
            GUARD_SERVER_NODE="$2"
            shift 2
            ;;
        --guard-server-port)
            GUARD_SERVER_PORT="$2"
            shift 2
            ;;
        --guard-model)
            GUARD_MODEL="$2"
            shift 2
            ;;
        --disable-determinism)
            ENABLE_DETERMINISM=false
            shift
            ;;
        --early-stop-budget)
            EARLY_STOP_BUDGET="$2"
            shift 2
            ;;
        -h|--help)
            usage
            ;;
        *)
            echo "Unknown option: $1"
            usage
            ;;
    esac
done

# Validate rewrite direction early (used by defense code via env REWRITE_DIRECTION)
if [ -n "$REWRITE_DIRECTION" ]; then
    if [ "$REWRITE_DIRECTION" != "increase" ] && [ "$REWRITE_DIRECTION" != "decrease" ]; then
        echo -e "${RED}❌ Error: invalid --rewrite-direction: $REWRITE_DIRECTION${NC}"
        echo "  Allowed values: increase | decrease"
        exit 1
    fi
fi

# Finalize derived guard URL after CLI parsing (so --guard-server-node/--guard-server-port overrides take effect)
GUARD_SERVER_URL="http://${GUARD_SERVER_NODE}:${GUARD_SERVER_PORT}/v1"

# Set server URL based on node
# Logic for SERVER_NODE (same as REWRITE_SERVER_NODE):
# - If SERVER_NODE is "localhost" or "127.0.0.1", deploy local server
# - If SERVER_NODE is set to other node, use remote server
# - If SERVER_NODE is empty/"none" or not set, skip server deployment
if [ -n "$SERVER_NODE" ] && [ "$SERVER_NODE" != "none" ] && [ "$SERVER_NODE" != "" ]; then
    SERVER_URL="http://${SERVER_NODE}:30000"
    # If server is on remote node, skip starting it locally
    if [ "$SERVER_NODE" != "localhost" ] && [ "$SERVER_NODE" != "127.0.0.1" ]; then
        if [ "$START_SERVER" = true ]; then
            echo -e "${YELLOW}⚠️  Remote server node specified ($SERVER_NODE)${NC}"
            echo "  Skipping local server start. Will only check if server is running."
            echo -e "${YELLOW}  Note: Make sure multi-turn/config/updater/<updater>-config.yaml has correct base_url(s):${NC}"
            echo "    target_model.base_url: \"${SERVER_URL}/v1\""
            echo "    attacker_model.base_url: \"${SERVER_URL}/v1\" (if using SGLang for attacker)"
            START_SERVER=false
        fi
    fi
else
    # SERVER_NODE is empty/"none" or not set, skip server deployment
    SERVER_URL=""
    if [ "$START_SERVER" = true ]; then
        echo -e "${YELLOW}⚠️  SERVER_NODE is not set or set to 'none'${NC}"
        echo "  Skipping SGLang server deployment"
        START_SERVER=false
    fi
fi

# Rewrite server (OpenAI-compatible), base_url should include /v1
# Single-direction mode: only one rewrite server is used (default: increase on port 8000).
REWRITE_SERVER_URL="http://${REWRITE_SERVER_NODE}:8000/v1"
REWRITE_SERVER_URL_INCREASE="${REWRITE_SERVER_URL}"
# Default: use the same rewrite server for both directions unless user overrides in config/env elsewhere.
REWRITE_SERVER_URL_DECREASE="${REWRITE_SERVER_URL}"

# Set default rewrite model configuration (can be overridden by config file or env vars)
# Logic: 
# - If REWRITE_SERVER_NODE is "localhost" or "127.0.0.1", deploy local server (server mode)
# - If REWRITE_SERVER_NODE is set to other node, use remote server (server mode)
# - If REWRITE_SERVER_NODE is empty/"none" or not set, use OpenAI API (openai mode)
if [ -n "$REWRITE_SERVER_NODE" ] && [ "$REWRITE_SERVER_NODE" != "none" ] && [ "$REWRITE_SERVER_NODE" != "" ]; then
    # REWRITE_SERVER_NODE is set (localhost or remote node), use server mode
    export REWRITE_MODEL_TYPE="${REWRITE_MODEL_TYPE:-server}"
else
    # REWRITE_SERVER_NODE is empty/"none" or not set, use OpenAI API
    export REWRITE_MODEL_TYPE="${REWRITE_MODEL_TYPE:-openai}"
fi
export REWRITE_SERVER_URL_INCREASE="${REWRITE_SERVER_URL_INCREASE}"
export REWRITE_SERVER_URL_DECREASE="${REWRITE_SERVER_URL_DECREASE}"
export REWRITE_SERVER_URL="${REWRITE_SERVER_URL}"
# Default rewrite direction for mislead defense (consumed by python via env).
export REWRITE_DIRECTION="${REWRITE_DIRECTION}"
# Served model name on the rewrite server (must match vLLM --served-model-name)
# Default: align model name with direction ("increase" by default), matching deploy/deploy_rewriter.sh defaults.
export REWRITE_SERVER_MODEL="${REWRITE_SERVER_MODEL:-$REWRITE_DIRECTION}"

echo -e "${BLUE}========================================${NC}"
echo -e "${BLUE}Multi-turn Jailbreak Pipeline${NC}"
echo -e "${BLUE}========================================${NC}"
echo ""

# Check conda
if ! command -v conda &> /dev/null; then
    echo -e "${RED}❌ Error: conda is not installed or not in PATH${NC}"
    exit 1
fi

# Initialize conda
eval "$(conda shell.bash hook)"

# Check environment variables
echo -e "${YELLOW}Step 1: Checking environment variables...${NC}"
MISSING=0
if [ -z "$OPENAI_API_KEY" ]; then
    echo -e "${RED}⚠️  OPENAI_API_KEY is not set${NC}"
    MISSING=1
fi
if [ -z "$HF_TOKEN" ]; then
    echo -e "${RED}⚠️  HF_TOKEN is not set${NC}"
    MISSING=1
fi
if [ -z "$HF_HOME" ]; then
    echo -e "${YELLOW}⚠️  HF_HOME is not set, setting to \$SCRATCH/LLMs${NC}"
    export HF_HOME=${SCRATCH}/LLMs
fi

if [ $MISSING -eq 1 ]; then
    echo ""
    echo -e "${RED}Please set the required environment variables:${NC}"
    echo "  export OPENAI_API_KEY=your_key"
    echo "  export HF_TOKEN=your_token"
    echo "  export HF_HOME=\$SCRATCH/LLMs"
    exit 1
fi

echo -e "${GREEN}✓ All environment variables are set${NC}"
echo ""

# Step 2: Start SGLang server
SERVER_PID=""
if [ "$START_SERVER" = true ]; then
    echo -e "${YELLOW}Step 2: Starting SGLang server...${NC}"
    echo "  Server node: $SERVER_NODE"
    if [ -n "$SERVER_URL" ] && [ "$SERVER_URL" != "" ]; then
        echo "  Server URL: $SERVER_URL"
    fi
    echo ""
    
    # Determine attacker model: use command line argument if set, otherwise read from config file
    # If --updater was specified, use it; otherwise use PLAN_GENERATOR
    CURRENT_UPDATER="${UPDATER:-$PLAN_GENERATOR}"
    
    # Only read from config file if ATTACKER_MODEL is not already set
    if [ -z "$ATTACKER_MODEL" ] || [ "$ATTACKER_MODEL" = "" ]; then
        UPDATER_CONFIG_FILE="$MULTI_TURN_DIR/config/updater/${CURRENT_UPDATER}-config.yaml"
        if [ -f "$UPDATER_CONFIG_FILE" ]; then
            # Try to extract attacker model from YAML (look for attacker_config.model)
            # This is a simple extraction - assumes the model is on a line like "model: <value>"
            # under attacker_config section
            ATTACKER_MODEL=$(python3 -c "
import yaml
import sys
try:
    with open('$UPDATER_CONFIG_FILE', 'r') as f:
        cfg = yaml.safe_load(f)
    # Check various possible paths for attacker model
    if 'attacker_config' in cfg and 'model' in cfg['attacker_config']:
        model = cfg['attacker_config']['model']
        if model and model != 'local':
            print(model)
    # Also check for updater-specific sections (e.g., xteaming.attacker_config)
    for key in cfg:
        if isinstance(cfg[key], dict) and 'attacker_config' in cfg[key]:
            if 'model' in cfg[key]['attacker_config']:
                model = cfg[key]['attacker_config']['model']
                if model and model != 'local':
                    print(model)
                    break
except Exception:
    pass
" 2>/dev/null | head -1)
        fi
    fi
    
    # Build deploy arguments
    DEPLOY_ARGS=(
        --server-node "$SERVER_NODE"
        --server-port 30000
    )
    
    # Add model if we found one
    if [ -n "$ATTACKER_MODEL" ] && [ "$ATTACKER_MODEL" != "" ]; then
        DEPLOY_ARGS+=(--model "$ATTACKER_MODEL")
        echo "  Attacker model: $ATTACKER_MODEL"
    else
        echo "  Attacker model: (default from deploy/deploy_attacker.sh)"
    fi
    echo ""
    
    # Use deploy_attacker_background.sh to start server
    # It handles all the checking and waiting logic
    bash "$PROJECT_ROOT/deploy/deploy_attacker_background.sh" "${DEPLOY_ARGS[@]}"
    
    # Get the PID if on localhost (for potential cleanup later)
    if [ "$SERVER_NODE" = "localhost" ] || [ "$SERVER_NODE" = "127.0.0.1" ]; then
        SERVER_PID=$(pgrep -f "sglang.launch_server.*30000" | head -1 || echo "")
    fi
    
    echo ""
else
    echo -e "${YELLOW}Step 2: Skipping server start (--no-server)${NC}"
    echo ""
fi

# Step 2.5: Start rewrite model server (if defense is enabled and using server type)
if [ "$ENABLE_MISLEAD" = true ]; then
    if [ "$REWRITE_MODEL_TYPE" != "server" ]; then
        echo -e "${YELLOW}Step 2.5: Skipping rewrite server (using $REWRITE_MODEL_TYPE type)${NC}"
        echo ""
    elif [ "$REWRITE_SERVER_NODE" != "localhost" ] && [ "$REWRITE_SERVER_NODE" != "127.0.0.1" ]; then
        echo -e "${YELLOW}Step 2.5: Checking remote rewrite server...${NC}"
        echo "  Server node: $REWRITE_SERVER_NODE"
        echo "  Rewrite server URL: $REWRITE_SERVER_URL_INCREASE"
        echo ""
        
        if curl -s "$REWRITE_SERVER_URL_INCREASE/health" > /dev/null 2>&1; then
            echo -e "${GREEN}✓ Remote rewrite server is running and accessible${NC}"
        else
            echo -e "${RED}❌ Remote rewrite server is not accessible at $REWRITE_SERVER_URL_INCREASE${NC}"
            echo "  Please start the server manually or check the connection"
            exit 1
        fi
        echo ""
    else
        echo -e "${YELLOW}Step 2.5: Starting rewrite model server...${NC}"
        echo "  Server node: $REWRITE_SERVER_NODE"
        echo "  Rewrite server URL: $REWRITE_SERVER_URL_INCREASE"
        echo "  Model: \${SCRATCH}/increase_rewriter/dpo (default from deploy/deploy_rewriter.sh)"
        echo "  Served model name: increase (configured in deploy/deploy_rewriter.sh)"
        echo ""
        
        # Check if server is already running
        if curl -s "$REWRITE_SERVER_URL_INCREASE/health" > /dev/null 2>&1; then
            echo -e "${GREEN}✓ Rewrite server is already running${NC}"
        else
            # Stop any existing server processes
            pkill -f "vllm serve .*--port 8000" 2>/dev/null || true
            pkill -f "uvicorn.*:8000" 2>/dev/null || true
            sleep 2
            
            # Start server using deploy_rewriter_background.sh
            # Configuration is already set in deploy_rewriter.sh defaults
            echo "Starting rewrite server using vLLM..."
            echo "  Model: \${SCRATCH}/increase_rewriter/dpo (default from deploy_rewriter.sh)"
            echo "  Served model name: ${REWRITE_SERVER_MODEL} (override; default: increase)"
            echo "  Server port: 8000"
            echo "  Server node: $REWRITE_SERVER_NODE"
            mkdir -p "$MULTI_TURN_DIR/logs"
            
            # Only override server-node if it's not localhost (default)
            REWRITE_ARGS=()
            if [ "$REWRITE_SERVER_NODE" != "localhost" ] && [ "$REWRITE_SERVER_NODE" != "127.0.0.1" ]; then
                REWRITE_ARGS+=(--server-node "$REWRITE_SERVER_NODE")
            fi

            # Ensure served model name matches what python will request (env REWRITE_SERVER_MODEL),
            # so `model=<name>` in OpenAI-compatible requests lines up with vLLM.
            if [ -n "$REWRITE_SERVER_MODEL" ]; then
                REWRITE_ARGS+=(--served-model-name "$REWRITE_SERVER_MODEL")
            fi

            # Best-effort: if a matching model directory exists, pass it explicitly.
            # This keeps decrease working out-of-the-box and enables switching if both models exist.
            if [ -n "${SCRATCH:-}" ]; then
                CANDIDATE_MODEL_PATH="${SCRATCH}/${REWRITE_SERVER_MODEL}_rewriter/dpo"
                if [ -d "$CANDIDATE_MODEL_PATH" ]; then
                    REWRITE_ARGS+=(--model "$CANDIDATE_MODEL_PATH")
                fi
            fi

            # Let deploy script do the waiting. Show progress in terminal and also save to log.
            bash "$PROJECT_ROOT/deploy/deploy_rewriter_background.sh" "${REWRITE_ARGS[@]}" 2>&1 | tee "$MULTI_TURN_DIR/logs/rewrite_server_deploy.log"
            DEPLOY_EXIT_CODE=${PIPESTATUS[0]}
            if [ $DEPLOY_EXIT_CODE -ne 0 ]; then
                echo -e "${RED}❌ Failed to start rewrite server. Check logs:${NC}"
                echo "  tail -60 $MULTI_TURN_DIR/logs/rewrite_server_deploy.log"
                exit 1
            fi

            # Final verification
            if curl -s "$REWRITE_SERVER_URL_INCREASE/health" > /dev/null 2>&1; then
                echo -e "${GREEN}✓ Rewrite server verification successful${NC}"
            else
                echo -e "${RED}❌ Rewrite server verification failed. Check logs:${NC}"
                echo "  tail -60 $MULTI_TURN_DIR/logs/rewrite_server_deploy.log"
                exit 1
            fi
        fi
        echo ""
    fi
fi

# Step 2.6: Start guard server (if enabled)
if [ "$ENABLE_GUARD" = true ]; then
    # Build guard server URL (used by python via env var GUARD_SERVER_URL)
    GUARD_SERVER_URL="http://${GUARD_SERVER_NODE}:${GUARD_SERVER_PORT}/v1"

    if [ "$GUARD_SERVER_NODE" != "localhost" ] && [ "$GUARD_SERVER_NODE" != "127.0.0.1" ]; then
        echo -e "${YELLOW}Step 2.6: Checking remote guard server...${NC}"
        echo "  Guard server node: $GUARD_SERVER_NODE"
        echo "  Guard server URL:  $GUARD_SERVER_URL"
        echo "  Guard model:       $GUARD_MODEL"
        echo ""
        if curl -s "http://${GUARD_SERVER_NODE}:${GUARD_SERVER_PORT}/health" > /dev/null 2>&1; then
            echo -e "${GREEN}✓ Remote guard server is running and accessible${NC}"
        else
            echo -e "${RED}❌ Remote guard server is not accessible at $GUARD_SERVER_URL${NC}"
            echo "  Please start the guard server manually or check the connection"
            exit 1
        fi
        echo ""
    else
        echo -e "${YELLOW}Step 2.6: Starting guard server (vLLM)...${NC}"
        echo "  Guard server URL:  $GUARD_SERVER_URL"
        echo "  Guard model:       $GUARD_MODEL"
        echo ""
        mkdir -p "$MULTI_TURN_DIR/logs"

        if curl -s "http://localhost:${GUARD_SERVER_PORT}/health" > /dev/null 2>&1; then
            echo -e "${GREEN}✓ Guard server is already running${NC}"
        else
            # Start server using deploy_guard_background.sh (handles waiting + logging)
            bash "$PROJECT_ROOT/deploy/deploy_guard_background.sh" \
                --server-port "$GUARD_SERVER_PORT" \
                --served-model-name "$GUARD_MODEL" 2>&1 | tee "$MULTI_TURN_DIR/logs/guard_server_deploy.log"
            DEPLOY_EXIT_CODE=${PIPESTATUS[0]}
            if [ $DEPLOY_EXIT_CODE -ne 0 ]; then
                echo -e "${RED}❌ Failed to start guard server. Check logs:${NC}"
                echo "  tail -60 $MULTI_TURN_DIR/logs/guard_server_deploy.log"
                exit 1
            fi

            if curl -s "http://localhost:${GUARD_SERVER_PORT}/health" > /dev/null 2>&1; then
                echo -e "${GREEN}✓ Guard server verification successful${NC}"
            else
                echo -e "${RED}❌ Guard server verification failed. Check logs:${NC}"
                echo "  tail -60 $MULTI_TURN_DIR/logs/guard_server_deploy.log"
                exit 1
            fi
        fi
        echo ""
    fi
fi

# Step 3: Run experiments
if [ "$RUN_EXPERIMENT" = true ]; then
    echo -e "${YELLOW}Step 3: Running experiments...${NC}"

    # Output root directory (must be set before invoking main.py; otherwise empty expands to CWD)
    OUTPUT_DIR="$MULTI_TURN_DIR/outputs"
    
    # Save original OUTPUT_NAME to check if it was user-specified
    ORIGINAL_OUTPUT_NAME="$OUTPUT_NAME"
    
    # Final check: Verify server is still running (if we started it or if using sglang)
    if [ -n "$SERVER_URL" ] && [ "$SERVER_URL" != "" ]; then
        if [ "$START_SERVER" = true ] || [ -n "$SERVER_PID" ] || ([ -n "$SERVER_NODE" ] && [ "$SERVER_NODE" != "localhost" ] && [ "$SERVER_NODE" != "127.0.0.1" ]); then
            echo "Verifying server is still available on $SERVER_NODE..."
            if ! curl -s "$SERVER_URL/health" > /dev/null 2>&1; then
                echo -e "${RED}❌ Error: Server is not responding!${NC}"
                echo "Please check server status:"
                echo "  curl $SERVER_URL/health"
                echo "  tail -f $MULTI_TURN_DIR/logs/sglang_server.log"
                exit 1
            fi
            echo -e "${GREEN}✓ Server is available${NC}"
            echo ""
        fi
    fi
    
    # Activate conda environment
    if [ -n "$SCRATCH" ] && [ -d "$SCRATCH/miniconda3/envs/multi-turn-new" ]; then
        conda activate "$SCRATCH/miniconda3/envs/multi-turn-new"
    else
        conda activate multi-turn-new
    fi
    
    # Determine which generators to run
    if [ "$ALL_GENERATORS" = true ]; then
        GENERATORS=("xteaming" "actorattack" "coa" "crescendo" "fitd")
    else
        GENERATORS=("$PLAN_GENERATOR")
    fi
    
    # Array to store output names generated during experiment runs
    RUN_OUTPUT_NAMES=()
    
    # Run experiments
    for gen in "${GENERATORS[@]}"; do
        # Determine updater for this run:
        # - If --updater was specified, use it (even with --all-generators)
        # - If --all-generators and no --updater, use same as generator
        # - If single generator and no --updater, use same as generator
        if [ -n "$UPDATER" ]; then
            # Updater was explicitly specified, use it
            CURRENT_UPDATER="$UPDATER"
        else
            # No updater specified, use same as generator
            CURRENT_UPDATER="$gen"
        fi
        
        # Generate output name for this generator if not user-specified
        if [ -z "$ORIGINAL_OUTPUT_NAME" ]; then
            # Auto-generate: {plan_generator}-{updater}[-mislead][-proact]
            OUTPUT_NAME="${gen}-${CURRENT_UPDATER}"
            if [ "$ENABLE_MISLEAD" = true ]; then
                OUTPUT_NAME="${OUTPUT_NAME}-mislead"
            fi
            if [ "$ENABLE_PROACT" = true ]; then
                OUTPUT_NAME="${OUTPUT_NAME}-proact"
            fi
            if [ "$ENABLE_GUARD" = true ]; then
                OUTPUT_NAME="${OUTPUT_NAME}-guard"
            fi
        else
            # User specified, use it directly
            OUTPUT_NAME="$ORIGINAL_OUTPUT_NAME"
        fi
        
        echo ""
        echo -e "${BLUE}Running experiment with plan generator: $gen, updater: $CURRENT_UPDATER${NC}"
        if [ -n "$NUM_BEHAVIORS" ]; then
            echo "  Behaviors: $NUM_BEHAVIORS"
            NUM_BEHAVIORS_ARG="--number_of_behaviors $NUM_BEHAVIORS"
        else
            echo "  Behaviors: ALL (processing all behaviors in dataset)"
            NUM_BEHAVIORS_ARG=""
        fi
        echo "  Output name: $OUTPUT_NAME"
        echo "  Max workers: $MAX_WORKERS"
        # Show model settings if specified
        if [ -n "$TARGET_MODEL" ] && [ "$TARGET_MODEL" != "" ]; then
            echo "  Target model: $TARGET_MODEL (from command line)"
        fi
        if [ -n "$JUDGE_MODEL" ] && [ "$JUDGE_MODEL" != "" ]; then
            echo "  Judge model: $JUDGE_MODEL (from command line)"
        fi
        if [ -n "$JUDGE_TYPE" ] && [ "$JUDGE_TYPE" != "" ]; then
            echo "  Judge type: $JUDGE_TYPE (from command line)"
        fi
        if [ "$ADD_REFRAMING_NOTE" = true ]; then
            echo "  Add reframing note: ENABLED (from command line)"
        fi
        if [ -n "$REWRITE_MODEL" ] && [ "$REWRITE_MODEL" != "" ]; then
            echo "  Rewrite model: $REWRITE_MODEL (from command line)"
        fi
        echo ""

        # Export updater early-stop budget as an environment variable (no YAML edits).
        # Empty means "do not override" (updaters fall back to config/default=0).
        if [ -n "$EARLY_STOP_BUDGET" ]; then
            export EARLY_STOP_BUDGET="$EARLY_STOP_BUDGET"
            echo "  Early-stop budget: $EARLY_STOP_BUDGET (via env EARLY_STOP_BUDGET)"
        else
            unset EARLY_STOP_BUDGET 2>/dev/null || true
        fi

        # Let main.py handle checking and skipping existing behaviors
        # It will automatically skip already processed behavior files
        RUN_OUTPUT_DIR="$OUTPUT_DIR/$OUTPUT_NAME"
        if [ -d "$RUN_OUTPUT_DIR" ] && [ "$FORCE_RERUN" = false ]; then
            EXISTING_COUNT=$(find "$RUN_OUTPUT_DIR" -name "behavior_*.json" 2>/dev/null | wc -l)
            if [ "$EXISTING_COUNT" -gt 0 ]; then
                echo -e "${YELLOW}↪ Output directory exists with $EXISTING_COUNT behavior files${NC}"
                echo -e "${YELLOW}  main.py will automatically skip existing files and process remaining behaviors.${NC}"
            fi
        fi
        
        FORCE_FLAG=""
        if [ "$FORCE_RERUN" = true ]; then
            FORCE_FLAG="--force"
        fi
        
        MISLEAD_FLAG=""
        if [ "$ENABLE_MISLEAD" = true ]; then
            MISLEAD_FLAG="--enable-defense"
            echo "  Mislead defense: ENABLED"
            # Set rewrite server env for single-direction (increase) rewrite server
            export REWRITE_SERVER_URL_INCREASE="$REWRITE_SERVER_URL_INCREASE"
            export REWRITE_SERVER_URL_DECREASE="$REWRITE_SERVER_URL_DECREASE"
            export REWRITE_SERVER_URL="$REWRITE_SERVER_URL"
            export REWRITE_DIRECTION="$REWRITE_DIRECTION"
            # Use REWRITE_MODEL if specified, otherwise use default REWRITE_SERVER_MODEL
            if [ -n "$REWRITE_MODEL" ] && [ "$REWRITE_MODEL" != "" ]; then
                export REWRITE_SERVER_MODEL="$REWRITE_MODEL"
            fi
        else
            echo "  Mislead defense: DISABLED"
        fi

        PROACT_FLAG=""
        if [ "$ENABLE_PROACT" = true ]; then
            PROACT_FLAG="--enable-proact"
            echo "  ProAct defense: ENABLED"
        else
            echo "  ProAct defense: DISABLED"
        fi

        GUARD_FLAG=""
        if [ "$ENABLE_GUARD" = true ]; then
            GUARD_FLAG="--enable-guard"
            echo "  Guard defense: ENABLED"
            export GUARD_SERVER_URL="$GUARD_SERVER_URL"
            export GUARD_MODEL="$GUARD_MODEL"
            export GUARD_API_MODE="${GUARD_API_MODE:-chat}"
            export GUARD_MULTI_TURN="${GUARD_MULTI_TURN:-false}"
        else
            echo "  Guard defense: DISABLED"
        fi
        
        DETERMINISM_FLAG=""
        if [ "$ENABLE_DETERMINISM" = false ]; then
            DETERMINISM_FLAG="--disable-determinism"
            echo "  Determinism: DISABLED (via --disable-determinism)"
        else
            echo "  Determinism: ENABLED"
        fi
        
        # Build command arguments array to handle empty flags properly
        CMD_ARGS=(
            --plan_generator "$gen"
            --updater "$CURRENT_UPDATER"
            --output_name "$OUTPUT_NAME"
            --max_workers "$MAX_WORKERS"
            --output_dir "$OUTPUT_DIR"
        )
        
        # Add optional arguments only if they are set
        if [ -n "$NUM_BEHAVIORS_ARG" ]; then
            CMD_ARGS+=(--number_of_behaviors "$NUM_BEHAVIORS")
        fi
        if [ -n "$FORCE_FLAG" ]; then
            CMD_ARGS+=(--force)
        fi
        if [ -n "$MISLEAD_FLAG" ]; then
            CMD_ARGS+=("$MISLEAD_FLAG")
        fi
        if [ -n "$PROACT_FLAG" ]; then
            CMD_ARGS+=("$PROACT_FLAG")
        fi
        if [ -n "$GUARD_FLAG" ]; then
            CMD_ARGS+=("$GUARD_FLAG")
        fi
        if [ -n "$DETERMINISM_FLAG" ]; then
            CMD_ARGS+=(--disable-determinism)
        fi
        
        # Update config file with model parameters if specified
        UPDATER_CONFIG_FILE="$MULTI_TURN_DIR/config/updater/${CURRENT_UPDATER}-config.yaml"
        CONFIG_BACKUP=""
        if [ -f "$UPDATER_CONFIG_FILE" ] && ([ -n "$TARGET_MODEL" ] || [ -n "$TARGET_PROVIDER" ] || [ -n "$JUDGE_MODEL" ] || [ -n "$JUDGE_TYPE" ] || [ "$ADD_REFRAMING_NOTE" = true ] || [ -n "$REWRITE_MODEL" ]); then
            # Change to multi-turn directory for config file operations (relative paths in Python code)
            cd "$MULTI_TURN_DIR"
            # If target is SGLang local/remote, pass a best-effort base_url to the config updater code.
            # This matches the OpenAI-compatible server started by deploy/deploy_attacker*.sh.
            if [ -n "$SERVER_URL" ] && [ "$SERVER_URL" != "" ]; then
                export TARGET_BASE_URL="${SERVER_URL}/v1"
            else
                # Fallback (common local default in configs)
                export TARGET_BASE_URL="http://127.0.0.1:30000/v1"
            fi

            # Best-effort: use the same OpenAI-compatible server for local judge too.
            if [ -n "$SERVER_URL" ] && [ "$SERVER_URL" != "" ]; then
                export JUDGE_BASE_URL="${SERVER_URL}/v1"
            else
                export JUDGE_BASE_URL="http://127.0.0.1:30000/v1"
            fi

            # Update config file with specified models (use relative path since we're in multi-turn dir)
            CONFIG_FILE_REL="config/updater/${CURRENT_UPDATER}-config.yaml"
            CONFIG_BACKUP="${CONFIG_FILE_REL}.backup.$$"
            
            # Create backup of original config (use relative path since we're in multi-turn dir)
            cp "$CONFIG_FILE_REL" "$CONFIG_BACKUP"
            python3 -c "
import yaml
import sys
import os

config_file = '$CONFIG_FILE_REL'
target_model = '$TARGET_MODEL' if '$TARGET_MODEL' else None
target_provider = '$TARGET_PROVIDER' if '$TARGET_PROVIDER' else None
target_base_url = os.getenv('TARGET_BASE_URL') or None
judge_base_url = os.getenv('JUDGE_BASE_URL') or None
judge_model = '$JUDGE_MODEL' if '$JUDGE_MODEL' else None
judge_type = '$JUDGE_TYPE' if '$JUDGE_TYPE' else None
add_reframing_note = '$ADD_REFRAMING_NOTE' == 'true'
rewrite_model = '$REWRITE_MODEL' if '$REWRITE_MODEL' else None

try:
    with open(config_file, 'r') as f:
        cfg = yaml.safe_load(f)
    
    # Update target_model
    if target_model and 'target_model' in cfg:
        cfg['target_model']['model'] = target_model
        # Auto-detect provider based on model name unless explicitly provided.
        target_model_lower = target_model.lower()
        if target_provider:
            cfg['target_model']['provider'] = target_provider
            print(f'Updated target_model to: {target_model} (provider: {target_provider})')
        elif target_model_lower == 'local':
            # Local model served by SGLang (OpenAI-compatible). Keep model name as 'local'.
            # We use provider='local' as an alias of provider='sglang' in this repo.
            cfg['target_model']['provider'] = 'local'
            print(f'Updated target_model to: {target_model} (provider: local)')
        elif target_model_lower.startswith('gemini'):
            cfg['target_model']['provider'] = 'gemini'
            print(f'Updated target_model to: {target_model} (provider: gemini)')
        elif target_model_lower.startswith('gpt'):
            cfg['target_model']['provider'] = 'openai'
            print(f'Updated target_model to: {target_model} (provider: openai)')
        else:
            print(f'Updated target_model to: {target_model}')

        # If target provider is local/sglang, ensure base_url exists so UnifiedLLMClient can connect.
        if cfg.get('target_model', {}).get('provider') in ('local', 'sglang'):
            if target_base_url:
                cfg['target_model']['base_url'] = target_base_url
            else:
                cfg['target_model']['base_url'] = cfg['target_model'].get('base_url') or 'http://127.0.0.1:30000/v1'
            print(f\"Ensured target_model.base_url: {cfg['target_model']['base_url']}\")
    
    # Update judge_model
    if judge_model and 'judge_model' in cfg:
        cfg['judge_model']['model'] = judge_model
        # Auto-detect provider based on model name
        judge_model_lower = judge_model.lower()
        if judge_model_lower == 'local':
            cfg['judge_model']['provider'] = 'local'
            if judge_base_url:
                cfg['judge_model']['base_url'] = judge_base_url
            else:
                cfg['judge_model']['base_url'] = cfg['judge_model'].get('base_url') or 'http://127.0.0.1:30000/v1'
            print(f\"Updated judge_model to: {judge_model} (provider: local, base_url: {cfg['judge_model']['base_url']})\")
        elif judge_model_lower.startswith('gemini'):
            cfg['judge_model']['provider'] = 'gemini'
            print(f'Updated judge_model to: {judge_model} (provider: gemini)')
        elif judge_model_lower.startswith('claude'):
            # Claude models typically use AWS provider in this codebase
            # But judge framework uses Anthropic directly, so we might not need to set provider here
            print(f'Updated judge_model to: {judge_model}')
        else:
            # Default to openai for GPT models
            if 'provider' not in cfg['judge_model'] or cfg['judge_model'].get('provider') == 'gemini':
                cfg['judge_model']['provider'] = 'openai'
            print(f'Updated judge_model to: {judge_model}')
    
    # Update judge_type
    if judge_type and 'judge_model' in cfg:
        if not isinstance(cfg['judge_model'], dict):
            cfg['judge_model'] = {}
        cfg['judge_model']['judge_type'] = judge_type
        print(f'Updated judge_type to: {judge_type}')
    
    # Update add_reframing_note
    if add_reframing_note and 'judge_model' in cfg:
        if not isinstance(cfg['judge_model'], dict):
            cfg['judge_model'] = {}
        cfg['judge_model']['add_reframing_note'] = True
        print(f'Updated add_reframing_note to: True')
    
    # Update rewrite_model (in rewrite_model section or defense config)
    if rewrite_model:
        if 'rewrite_model' in cfg:
            if isinstance(cfg['rewrite_model'], dict):
                cfg['rewrite_model']['model'] = rewrite_model
            else:
                cfg['rewrite_model'] = {'model': rewrite_model, 'type': cfg.get('rewrite_model', {}).get('type', 'openai')}
        else:
            cfg['rewrite_model'] = {'model': rewrite_model}
        print(f'Updated rewrite_model to: {rewrite_model}')
    
    # Write back to file
    with open(config_file, 'w') as f:
        yaml.dump(cfg, f, default_flow_style=False, sort_keys=False, allow_unicode=True)
    
except Exception as e:
    print(f'Error updating config file: {e}', file=sys.stderr)
    sys.exit(1)
" || {
                echo -e "${RED}❌ Error updating config file${NC}"
                if [ -f "$CONFIG_BACKUP" ]; then
                    mv "$CONFIG_BACKUP" "$CONFIG_FILE_REL"
                fi
                cd "$PROJECT_ROOT"
                exit 1
            }
            # Return to project root after config update
            cd "$PROJECT_ROOT"
        fi
        
        # Change to multi-turn directory to run main.py (it expects relative paths)
        # Ensure repo-root modules (e.g., `defense/`, `unified_judge_framework.py`) are importable.
        (cd "$MULTI_TURN_DIR" && PYTHONPATH="$PROJECT_ROOT:${PYTHONPATH:-}" python main.py "${CMD_ARGS[@]}")
        MAIN_EXIT_CODE=$?
        
        # Restore original config if backup exists (need to be in multi-turn dir)
        if [ -n "$CONFIG_BACKUP" ] && [ -f "$MULTI_TURN_DIR/$CONFIG_BACKUP" ]; then
            (cd "$MULTI_TURN_DIR" && mv "$CONFIG_BACKUP" "config/updater/${CURRENT_UPDATER}-config.yaml")
        fi
        
        if [ $MAIN_EXIT_CODE -eq 0 ]; then
            echo -e "${GREEN}✓ Completed: plan_generator=$gen, updater=$CURRENT_UPDATER${NC}"
            # Save the output name for later use in viewing results
            RUN_OUTPUT_NAMES+=("$OUTPUT_NAME")
        else
            echo -e "${RED}✗ Failed: plan_generator=$gen, updater=$CURRENT_UPDATER${NC}"
        fi
    done
    
    echo ""
    echo -e "${GREEN}✓ All experiments completed${NC}"
    echo ""
fi

# Step 4: View results
if [ "$VIEW_RESULTS" = true ] && [ "$RUN_EXPERIMENT" = true ]; then
    echo -e "${YELLOW}Step 4: Viewing results...${NC}"
    
    OUTPUT_DIR="$MULTI_TURN_DIR/outputs"
    
    # Use the output names generated during experiment runs
    if [ ${#RUN_OUTPUT_NAMES[@]} -gt 0 ]; then
        for OUTPUT_NAME in "${RUN_OUTPUT_NAMES[@]}"; do
            RESULT_DIR="$OUTPUT_DIR/$OUTPUT_NAME"
            if [ -d "$RESULT_DIR" ]; then
                echo ""
                echo -e "${BLUE}Results for: $OUTPUT_NAME${NC}"
                echo "  Directory: $RESULT_DIR"
                
                # Count result files
                FILE_COUNT=$(find "$RESULT_DIR" -name "behavior_*.json" | wc -l)
                echo "  Files: $FILE_COUNT"
                
                # Show first few files
                echo "  Sample files:"
                find "$RESULT_DIR" -name "behavior_*.json" | head -3 | while read file; do
                    echo "    - $(basename $file)"
                done
            else
                echo -e "${RED}  No results found for $OUTPUT_NAME${NC}"
            fi
        done
    else
        echo -e "${YELLOW}  No output directories from this run to display${NC}"
    fi
    
    echo ""
    echo "To view detailed results:"
    if [ ${#RUN_OUTPUT_NAMES[@]} -gt 0 ]; then
        for OUTPUT_NAME in "${RUN_OUTPUT_NAMES[@]}"; do
            echo "  ls -lh $OUTPUT_DIR/$OUTPUT_NAME/"
        done
    fi
    echo ""
    
    # Run analysis.py if it exists and we have results
    if [ -f "$MULTI_TURN_DIR/analysis.py" ]; then
        echo -e "${YELLOW}Running analysis...${NC}"
        # Activate conda environment for analysis
        if [ -n "$SCRATCH" ] && [ -d "$SCRATCH/miniconda3/envs/multi-turn-new" ]; then
            conda activate "$SCRATCH/miniconda3/envs/multi-turn-new" 2>/dev/null || true
        else
            conda activate multi-turn-new 2>/dev/null || true
        fi
        
        # Use the output names generated during experiment runs
        ANALYSIS_DIRS=()
        if [ ${#RUN_OUTPUT_NAMES[@]} -gt 0 ]; then
            for OUTPUT_NAME in "${RUN_OUTPUT_NAMES[@]}"; do
                # Only add if directory exists
                if [ -d "$OUTPUT_DIR/$OUTPUT_NAME" ]; then
                    ANALYSIS_DIRS+=("$OUTPUT_NAME")
                fi
            done
        fi
        
        # Analyze each directory from this run
        if [ ${#ANALYSIS_DIRS[@]} -gt 0 ]; then
            for ANALYSIS_DIR in "${ANALYSIS_DIRS[@]}"; do
                echo "  Analyzing: $ANALYSIS_DIR"
                if [ -n "$NUM_BEHAVIORS" ]; then
                    (cd "$MULTI_TURN_DIR" && python analysis.py --pattern "$ANALYSIS_DIR" --expected-count "$NUM_BEHAVIORS" --no-plot) 2>&1 || {
                        echo -e "${YELLOW}  Analysis completed with warnings (this is normal for small test runs)${NC}"
                    }
                else
                    (cd "$MULTI_TURN_DIR" && python analysis.py --pattern "$ANALYSIS_DIR" --no-plot) 2>&1 || {
                        echo -e "${YELLOW}  Analysis completed with warnings (this is normal for small test runs)${NC}"
                    }
                fi
            done
        else
            echo "  No result directories found for analysis"
        fi
        echo ""
    fi
fi

# Step 5: Stop servers if requested
if [ "$STOP_SERVER" = true ]; then
    if [ -n "$SERVER_PID" ]; then
        echo -e "${YELLOW}Step 5: Stopping SGLang server...${NC}"
        kill $SERVER_PID 2>/dev/null || pkill -f "sglang.launch_server.*30000" || true
        sleep 2
        echo -e "${GREEN}✓ SGLang server stopped${NC}"
    fi
    # Stop rewrite server (vLLM)
    if [ "$ENABLE_MISLEAD" = true ] && [ "$REWRITE_MODEL_TYPE" = "server" ]; then
        echo -e "${YELLOW}Step 5: Stopping rewrite server...${NC}"
        pkill -f "vllm serve .*--port 8000" 2>/dev/null || true
        pkill -f "uvicorn.*:8000" 2>/dev/null || true
        sleep 2
        echo -e "${GREEN}✓ Rewrite server stopped${NC}"
    fi
    echo ""
fi

# Summary
echo -e "${BLUE}========================================${NC}"
echo -e "${GREEN}Pipeline Complete!${NC}"
echo -e "${BLUE}========================================${NC}"
echo ""

if [ "$RUN_EXPERIMENT" = true ]; then
    OUTPUT_DIR="$MULTI_TURN_DIR/outputs"
    echo "Results are saved in:"
    # Use the output names generated during experiment runs
    if [ ${#RUN_OUTPUT_NAMES[@]} -gt 0 ]; then
        for OUTPUT_NAME in "${RUN_OUTPUT_NAMES[@]}"; do
            echo "  $OUTPUT_DIR/$OUTPUT_NAME/"
        done
    else
        echo "  (No experiments were completed)"
    fi
    echo ""
elif [ -n "$SERVER_NODE" ] && [ "$SERVER_NODE" != "localhost" ] && [ "$SERVER_NODE" != "127.0.0.1" ] && [ "$SERVER_NODE" != "none" ] && [ "$SERVER_NODE" != "" ]; then
    # Server is on remote node, just check if it's available
    if [ -n "$SERVER_URL" ] && [ "$SERVER_URL" != "" ]; then
        echo -e "${YELLOW}Step 2: Checking remote SGLang server...${NC}"
        echo "  Server node: $SERVER_NODE"
        echo "  Server URL: $SERVER_URL"
        echo ""
        
        if curl -s "$SERVER_URL/health" > /dev/null 2>&1; then
            echo -e "${GREEN}✓ Remote SGLang server is running and accessible${NC}"
            HEALTH_RESPONSE=$(curl -s "$SERVER_URL/health" 2>&1)
            if [ $? -eq 0 ]; then
                echo -e "${GREEN}✓ Server health check passed${NC}"
            fi
        else
            echo -e "${RED}❌ Remote SGLang server is not accessible at $SERVER_URL${NC}"
            echo ""
            echo "Please ensure:"
            echo "  1. Server is running on $SERVER_NODE"
            echo "  2. Network connectivity to $SERVER_NODE"
            echo "  3. Firewall allows connections on port 30000"
            echo ""
            echo "To check server status on remote node:"
            echo "  ssh $SERVER_NODE 'curl http://localhost:30000/health'"
            exit 1
        fi
        echo ""
    fi
fi

if [ "$START_SERVER" = true ] && [ "$STOP_SERVER" = false ]; then
    echo "SGLang server is still running."
    echo "To check status: curl $SERVER_URL/health"
    echo "To view logs: tail -f $MULTI_TURN_DIR/logs/sglang_server.log"
    echo ""
fi

