#!/usr/bin/env bash
set -euo pipefail

# Run whispersplat_refiner.py
SCENES=(kitchen) 
# SCENES=(bicycle bonsai counter drjohnson flowers garden kitchen playroom room stump train truck treehill) 
BASE_DIR="leakages/0.51lr0.01" # Base directory containing scene folders, each with hidden, target, and clean images
# For example: base_dir/bicycle/bicycle.png, base_dir/bicycle/bicycle_target.png, base_dir/bicycle/bicycle_clean.png
SCRIPT="whispersplat_refiner.py"  
PY=python3

OUT_ROOT="whispersplat_refiner/0.51lr0.01" # Output directory for results, metrics and logs
mkdir -p "$OUT_ROOT"
SUMMARY="$OUT_ROOT/metrics_summary.tsv"
echo -e "scene\tbaseline_psnr\tbaseline_ssim\tbaseline_lpips\tstage1_psnr\tstage1_ssim\tstage1_lpips\tafter_predictor_psnr\tafter_predictor_ssim\tafter_predictor_lpips\tafter_refine_psnr\tafter_refine_ssim\tafter_refine_lpips\tfinal_psnr\tfinal_ssim\tfinal_lpips" > "$SUMMARY"

for scene in "${SCENES[@]}"; do
  echo "=== Processing scene: ${scene} ==="
  SCENE_DIR="${BASE_DIR}/${scene}"
  HIDDEN="${SCENE_DIR}/${scene}.png"
  TARGET="${SCENE_DIR}/${scene}_target.png"
  CLEAN="${SCENE_DIR}/${scene}_clean.png"

  if [[ ! -f "$HIDDEN" || ! -f "$TARGET" || ! -f "$CLEAN" ]]; then
    echo "Skipping ${scene}: missing one of ${HIDDEN}, ${TARGET}, or ${CLEAN}" >&2
    continue
  fi

  OUT_DIR="${OUT_ROOT}/${scene}"
  mkdir -p "$OUT_DIR"
  LOG="${OUT_DIR}/run.log"

  echo "Running whispersplat_refiner.py for ${scene}..."
  "$PY" "$SCRIPT" \
    --hidden "${HIDDEN}" \
    --target "${TARGET}" \
    --clean "${CLEAN}" \
    --out_dir "${OUT_DIR}" \
    --patch_sizes 64,32,16 \
    --alpha_iters 300 \
    --alpha_lr 1e-2 \
    --lambda_reg_alpha 1e-3 \
    --lambda_tv 0.04 \
    --predictor_steps1 1000 \
    --predictor_lr1 1e-3 \
    --predictor_steps2 500 \
    --predictor_lr2 5e-4 \
    --weight_decay 1e-4 \
    --lambda_perc 0.1 \
    --blend_gamma 1.0 \
    --alt_rounds 1 \
    --hidden_channels 32 \
    --device cuda \
    > "${LOG}" 2>&1 || { echo "whispersplat_refiner.py failed for ${scene}, see ${LOG}"; continue; }

  # Parse baseline metrics
  BASELINE_LINE=$(grep -E "SNR before any correction:" "${LOG}" | tail -1 || true)
  baseline_psnr="nan"; baseline_ssim="nan"; baseline_lpips="nan"
  if [[ -n "$BASELINE_LINE" ]]; then
    if [[ "$BASELINE_LINE" =~ ([0-9]+\.[0-9]+)\ dB ]]; then
      baseline_psnr="${BASH_REMATCH[1]}"
    fi
    if [[ "$BASELINE_LINE" =~ SSIM:\ ([0-9]+\.[0-9]+) ]]; then
      baseline_ssim="${BASH_REMATCH[1]}"
    fi
    if [[ "$BASELINE_LINE" =~ LPIPS:\ ([0-9]+\.[0-9]+) ]]; then
      baseline_lpips="${BASH_REMATCH[1]}"
    fi
  fi

  # Stage1 (take the last "After scale" line)
  STAGE1_LINE=$(grep -E "After scale .* PSNR:" "${LOG}" | tail -1 || true)
  stage1_psnr="nan"; stage1_ssim="nan"; stage1_lpips="nan"
  if [[ -n "$STAGE1_LINE" ]]; then
    if [[ "$STAGE1_LINE" =~ PSNR:\ ([0-9]+\.[0-9]+) ]]; then
      stage1_psnr="${BASH_REMATCH[1]}"
    fi
    if [[ "$STAGE1_LINE" =~ SSIM:\ ([0-9]+\.[0-9]+) ]]; then
      stage1_ssim="${BASH_REMATCH[1]}"
    fi
    if [[ "$STAGE1_LINE" =~ LPIPS:\ ([0-9]+\.[0-9]+) ]]; then
      stage1_lpips="${BASH_REMATCH[1]}"
    fi
  fi

  # After predictor round
  PRED_LINE=$(grep -E "After predictor \(round" "${LOG}" | tail -1 || true)
  after_predictor_psnr="nan"; after_predictor_ssim="nan"; after_predictor_lpips="nan"
  if [[ -n "$PRED_LINE" ]]; then
    if [[ "$PRED_LINE" =~ PSNR:\ ([0-9]+\.[0-9]+) ]]; then
      after_predictor_psnr="${BASH_REMATCH[1]}"
    fi
    if [[ "$PRED_LINE" =~ SSIM:\ ([0-9]+\.[0-9]+) ]]; then
      after_predictor_ssim="${BASH_REMATCH[1]}"
    fi
    if [[ "$PRED_LINE" =~ LPIPS:\ ([0-9]+\.[0-9]+) ]]; then
      after_predictor_lpips="${BASH_REMATCH[1]}"
    fi
  fi

  # After refine (last "After refine scale" line)
  REFINE_LINE=$(grep -E "After refine scale .* PSNR:" "${LOG}" | tail -1 || true)
  after_refine_psnr="nan"; after_refine_ssim="nan"; after_refine_lpips="nan"
  if [[ -n "$REFINE_LINE" ]]; then
    if [[ "$REFINE_LINE" =~ PSNR:\ ([0-9]+\.[0-9]+) ]]; then
      after_refine_psnr="${BASH_REMATCH[1]}"
    fi
    if [[ "$REFINE_LINE" =~ SSIM:\ ([0-9]+\.[0-9]+) ]]; then
      after_refine_ssim="${BASH_REMATCH[1]}"
    fi
    if [[ "$REFINE_LINE" =~ LPIPS:\ ([0-9]+\.[0-9]+) ]]; then
      after_refine_lpips="${BASH_REMATCH[1]}"
    fi
  fi

  # Final
  FINAL_LINE=$(grep -E "Final PSNR:" "${LOG}" | tail -1 || true)
  final_psnr="nan"; final_ssim="nan"; final_lpips="nan"
  if [[ -n "$FINAL_LINE" ]]; then
    if [[ "$FINAL_LINE" =~ PSNR:\ ([0-9]+\.[0-9]+) ]]; then
      final_psnr="${BASH_REMATCH[1]}"
    fi
    if [[ "$FINAL_LINE" =~ SSIM:\ ([0-9]+\.[0-9]+) ]]; then
      final_ssim="${BASH_REMATCH[1]}"
    fi
    if [[ "$FINAL_LINE" =~ LPIPS:\ ([0-9]+\.[0-9]+) ]]; then
      final_lpips="${BASH_REMATCH[1]}"
    fi
  fi

  echo -e "${scene}\t${baseline_psnr}\t${baseline_ssim}\t${baseline_lpips}\t${stage1_psnr}\t${stage1_ssim}\t${stage1_lpips}\t${after_predictor_psnr}\t${after_predictor_ssim}\t${after_predictor_lpips}\t${after_refine_psnr}\t${after_refine_ssim}\t${after_refine_lpips}\t${final_psnr}\t${final_ssim}\t${final_lpips}" >> "$SUMMARY"

  echo "Scene ${scene} metrics:"
  echo "  baseline: PSNR=${baseline_psnr}, SSIM=${baseline_ssim}, LPIPS=${baseline_lpips}"
  echo "  stage1: PSNR=${stage1_psnr}, SSIM=${stage1_ssim}, LPIPS=${stage1_lpips}"
  echo "  after predictor: PSNR=${after_predictor_psnr}, SSIM=${after_predictor_ssim}, LPIPS=${after_predictor_lpips}"
  echo "  after refine: PSNR=${after_refine_psnr}, SSIM=${after_refine_ssim}, LPIPS=${after_refine_lpips}"
  echo "  final: PSNR=${final_psnr}, SSIM=${final_ssim}, LPIPS=${final_lpips}"

done