#!/usr/bin/env bash
# queue_experiments.sh
set -euo pipefail

# --------------------------------------------------
# Requirements
# --------------------------------------------------
command -v jq >/dev/null 2>&1 || { echo "jq is required"; exit 1; }
if command -v emulator-train >/dev/null 2>&1; then
  EMULATOR_TRAIN="emulator-train"
elif [[ -x ".venv/bin/emulator-train" ]]; then
  EMULATOR_TRAIN=".venv/bin/emulator-train"
else
  echo "emulator-train not found. Did you run: uv sync ?"
  exit 1
fi

# --------------------------------------------------
# Paths (repo-root relative)
# --------------------------------------------------
REPO_ROOT="$(cd "$(dirname "${BASH_SOURCE[0]}")/.." && pwd)"
cd "${REPO_ROOT}"

TRAIN_BASE="${REPO_ROOT}/configs/train.json"
EVAL_CFG="${REPO_ROOT}/configs/eval.json"
SUMMARY_CFG_DIR="${REPO_ROOT}/configs/summary"
DIST_CFG_DIR="${REPO_ROOT}/configs/distance"

TS="$(date +"%Y%m%d_%H%M%S")"
SWEEP_DIR="${REPO_ROOT}/configs/sweeps/${TS}"
LOG_DIR="${REPO_ROOT}/logs/${TS}"
mkdir -p "${SWEEP_DIR}" "${LOG_DIR}"

BACKUP_DIR="${SWEEP_DIR}/_base_configs"
mkdir -p "${BACKUP_DIR}"
cp -f "${SUMMARY_CFG_DIR}/projection_config.json" "${BACKUP_DIR}/projection_config.json"
cp -f "${SUMMARY_CFG_DIR}/mlp_config.json" "${BACKUP_DIR}/mlp_config.json"
cp -f "${SUMMARY_CFG_DIR}/linear_config.json" "${BACKUP_DIR}/linear_config.json"
cp -f "${SUMMARY_CFG_DIR}/identity_config.json" "${BACKUP_DIR}/identity_config.json"
cp -f "${DIST_CFG_DIR}/sinkhorn_config.json" "${BACKUP_DIR}/sinkhorn_config.json"
cp -f "${DIST_CFG_DIR}/wgan_config.json" "${BACKUP_DIR}/wgan_config.json"

restore_configs() {
  cp -f "${BACKUP_DIR}/projection_config.json" "${SUMMARY_CFG_DIR}/projection_config.json"
  cp -f "${BACKUP_DIR}/mlp_config.json" "${SUMMARY_CFG_DIR}/mlp_config.json"
  cp -f "${BACKUP_DIR}/linear_config.json" "${SUMMARY_CFG_DIR}/linear_config.json"
  cp -f "${BACKUP_DIR}/identity_config.json" "${SUMMARY_CFG_DIR}/identity_config.json"
  cp -f "${BACKUP_DIR}/sinkhorn_config.json" "${DIST_CFG_DIR}/sinkhorn_config.json"
  cp -f "${BACKUP_DIR}/wgan_config.json" "${DIST_CFG_DIR}/wgan_config.json"
}
trap restore_configs EXIT

# --------------------------------------------------
# Optional W&B glue
# --------------------------------------------------
export WANDB_PROJECT="${WANDB_PROJECT:-chaos-emulator}"
export WANDB_ENTITY="${WANDB_ENTITY:-}"
export WANDB_RUN_GROUP="${WANDB_RUN_GROUP:-lorenz63}"

# --------------------------------------------------
# Sweep definitions
# --------------------------------------------------
SEEDS=(0 1 2)

NOISE_LEVELS=(0.9 0.7 0.5 0.3 0.1 0.0)

MAX_PARALLEL="${MAX_PARALLEL:-1}"
START_SEED="${START_SEED:-}"
SKIP_EXISTING="${SKIP_EXISTING:-0}"

if [[ "${MAX_PARALLEL}" != "1" ]]; then
  echo "MAX_PARALLEL>1 is not supported because configs are shared. Run with MAX_PARALLEL=1."
  exit 1
fi

# --------------------------------------------------
# Helper: build config
# --------------------------------------------------
make_cfg() {
  local in="$1"
  local out="$2"
  local seed="$3"
  local noise="$4"

  jq "
    .seed = ${seed}
  | .noise_level = ${noise}
  " "${in}" > "${out}"
}

# --------------------------------------------------
# Runner
# --------------------------------------------------
run_single() {
  local cfg="$1"
  local run_name="$2"
  local log_file="$3"

  if [[ "${SKIP_EXISTING}" == "1" && -s "${log_file}" ]]; then
    echo ">>> SKIP ${run_name} (log exists)"
    return 0
  fi

  echo ">>> ${run_name}"
  echo "Config: ${cfg}"

  if [[ -n "${WANDB_TAGS:-}" ]]; then
    export WANDB_TAGS="${WANDB_TAGS},${run_name}"
  else
    export WANDB_TAGS="${run_name}"
  fi

  "${EMULATOR_TRAIN}" \
    --train-config "${cfg}" \
    --eval-config "${EVAL_CFG}" \
    --run-eval \
    > "${log_file}" 2>&1
}

# --------------------------------------------------
# Sweep orchestration
# --------------------------------------------------
pids=()

for seed in "${SEEDS[@]}"; do
  if [[ -n "${START_SEED}" && "${seed}" -lt "${START_SEED}" ]]; then
    echo "Skipping seed ${seed} (START_SEED=${START_SEED})"
    continue
  fi
  for noise in "${NOISE_LEVELS[@]}"; do
    run_id="s${seed}_n${noise}"
    cfg="${SWEEP_DIR}/train_${run_id}.json"
    log="${LOG_DIR}/${run_id}.log"

    make_cfg "${TRAIN_BASE}" "${cfg}" \
      "${seed}" "${noise}"

    run_single "${cfg}" "${run_id}" "${log}"
  done
done

echo "All experiments finished."
echo "Logs:    ${LOG_DIR}"
echo "Configs: ${SWEEP_DIR}"
