#!/bin/bash

# MLVU Evaluation Script for RWKV-Qwen Hybrid Model
# Usage: ./eval_mlvu_rwkv_qwen.sh <model_path> <video_dir> <output_dir> <num_chunks>

# --- CONFIGURATION: Replace these paths with your own setup ---
# Navigate to evaluation script directory
cd /path/to/your/eval/eval_mlvu

# --- DEFAULT PATHS (override with command-line arguments) ---
MODEL_PATH=${1:-"/path/to/your/model/checkpoint"}
VIDEO_DIR=${2:-"/path/to/your/videos"}
OUTPUT_DIR=${3:-"${MODEL_PATH}/eval_output_mlvu"}
NUM_CHUNKS=${4:-1}

# --- AUTO-DETECT AVAILABLE GPUS ---
gpu_list=$(nvidia-smi --query-gpu=index --format=csv,noheader | tr '\n' ',' | sed 's/,$//')
read -a GPULIST <<< ${gpu_list//,/ }

# Use number of available GPUs if num_chunks is default (1)
if [ $NUM_CHUNKS -eq 1 ]; then
    NUM_CHUNKS=${#GPULIST[@]}
fi

echo "=== MLVU Evaluation with RWKV-Qwen Hybrid Model ==="
echo "Model Path: $MODEL_PATH"
echo "Video Directory: $VIDEO_DIR"
echo "Output Directory: $OUTPUT_DIR"
echo "Number of Chunks: $NUM_CHUNKS"
echo "Available GPUs: ${GPULIST[@]}"

# Create output directory
mkdir -p $OUTPUT_DIR

# Define MLVU JSON annotation files (replace with your paths)
MLVU_JSON_FILES=(
    "/path/to/your/annotations/1_plotQA.json"
    "/path/to/your/annotations/2_needle.json"
    "/path/to/your/annotations/3_ego.json"
    "/path/to/your/annotations/4_count.json"
    "/path/to/your/annotations/5_order.json"
    "/path/to/your/annotations/6_anomaly_reco.json"
    "/path/to/your/annotations/7_topic_reasoning.json"
)

echo ""
echo "=== Starting MLVU Evaluation for All Tasks ==="

# Launch evaluation for each chunk in parallel
for IDX in $(seq 0 $((NUM_CHUNKS-1))); do
    if [ $IDX -lt ${#GPULIST[@]} ]; then
        CUDA_VISIBLE_DEVICES=${GPULIST[$IDX]} python model_mlvu_qa_rwkv_qwen.py \
            --model-path $MODEL_PATH \
            --video_dir $VIDEO_DIR \
            --json_files_list "${MLVU_JSON_FILES[@]}" \
            --output_dir $OUTPUT_DIR \
            --output_name pred \
            --num-chunks $NUM_CHUNKS \
            --chunk-idx $IDX  &
    else
        echo "Warning: Not enough GPUs for chunk $IDX"
    fi
done

# Wait for all background processes to finish
wait

echo "=== All MLVU Evaluations Complete ==="
echo "Results saved to: $OUTPUT_DIR"

# Combine all prediction files
all_predictions_file="${OUTPUT_DIR}/all_predictions.jsonl"
> "$all_predictions_file"

# Concatenate all prediction files
for pred_file in "${OUTPUT_DIR}"/*.jsonl; do
    if [ -f "$pred_file" ]; then
        cat "$pred_file" >> "$all_predictions_file"
    fi
done

echo "Combined all predictions to: $all_predictions_file"

# Calculate scores
echo ""
echo "=== Calculating Scores ==="

python calculate_score.py \
    --output_path $OUTPUT_DIR \
    --score_path $OUTPUT_DIR/score.json

echo "=== Final Results ==="
echo "Overall accuracy report saved to: $OUTPUT_DIR/score.json"

# Display overall accuracy if score file exists
if [ -f "$OUTPUT_DIR/score.json" ]; then
    ACCURACY=$(python3 -c "
import sys, json;
try:
    data = json.load(open('$OUTPUT_DIR/score.json'));
    print(f'{data[\"scores\"][\"overall\"][\"accuracy\"]:.4f}')
except Exception as e:
    print('Error reading score:', e)
")
    echo "Overall MLVU accuracy: $ACCURACY"
fi

echo ""
echo "=== Task Summary ==="
# Extract task names from the JSON files and display individual scores
for json_file in "${MLVU_JSON_FILES[@]}"; do
    if [ -f "$json_file" ]; then
        task_name=$(basename "$json_file" .json)
        # Check if this task name is in the score file
        if python3 -c "import sys, json; data=json.load(open('$OUTPUT_DIR/score.json')); print('$task_name' in data.get('scores', {}))" 2>/dev/null | grep -q "True"; then
            accuracy=$(python3 -c "import sys, json; data=json.load(open('$OUTPUT_DIR/score.json')); print(f'{data[\"scores\"][\"$task_name\"][\"accuracy\"]:.4f}')")
            total=$(python3 -c "import sys, json; data=json.load(open('$OUTPUT_DIR/score.json')); print(data['scores']['$task_name']['total'])")
            echo "  $task_name: $accuracy ($total questions)"
        fi
    fi
done