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
















REPO_ROOT="$(cd -- "$(dirname -- "${BASH_SOURCE[0]}")/.." && pwd)"

_is_truthy() {
  case "$(echo "${1:-}" | tr '[:upper:]' '[:lower:]')" in
    1|true|t|yes|y|on) return 0 ;;
    *) return 1 ;;
  esac
}

USE_DEFAULT_CKPT=0
if _is_truthy "${GRAPH_GRPO_USE_DEFAULT_CKPT:-}"; then
  USE_DEFAULT_CKPT=1
fi

SINGLE_CKPT=""
MODE="${1:-}"
if [[ "${MODE}" == "--batch" || "${MODE}" == "--all" ]]; then
  CKPT_ROOT="${2:-}"
  MAX_ORACLE_CALLS="${3:-10000}"
  SEED="${4:-0}"

  if [[ -n "${CKPT_ROOT}" && -f "${CKPT_ROOT}" ]]; then
    SINGLE_CKPT="${CKPT_ROOT}"
    CKPT_ROOT=""
  fi

  if [[ -z "${CKPT_ROOT}" && -z "${SINGLE_CKPT}" && "${USE_DEFAULT_CKPT}" -ne 1 ]]; then
    echo "Usage: $0 --batch /abs/path/to/ckpt_root_or_ckpt [max_oracle_calls] [seed]" >&2
    exit 2
  fi
else
  CKPT_PATH="${1:-}"
  ORACLE_NAME="${2:-}"
  MAX_ORACLE_CALLS="${3:-10000}"
  SEED="${4:-0}"

  if [[ -z "${CKPT_PATH}" || -z "${ORACLE_NAME}" ]]; then
    echo "Usage: $0 /abs/path/to/ckpt Ranolazine_MPO [max_oracle_calls] [seed]" >&2
    echo "   or: $0 --batch /abs/path/to/ckpt_root [max_oracle_calls] [seed]" >&2
    exit 2
  fi
fi

SEEDS=("${SEED}")
if [[ -n "${GRAPH_GRPO_SEEDS:-}" ]]; then
  seeds_raw="${GRAPH_GRPO_SEEDS//,/ }"
  read -r -a SEEDS <<< "${seeds_raw}"
  if [[ "${#SEEDS[@]}" -eq 0 ]]; then
    SEEDS=("${SEED}")
  fi
fi

: "${MOLOPT_REPO:=${REPO_ROOT}/../mol_opt}"
: "${MOLOPT_CONDA_ENV:=molopt}"
: "${RL_GRAPH_REPO:=${REPO_ROOT}}"
: "${RL_GRAPH_CONDA_ENV:=defog}"
: "${MOLOPT_RESULTS_DIR:=${MOLOPT_REPO}/main/graph_grpo/results}"

if [[ ! -d "${MOLOPT_REPO}" ]]; then
  echo "ERROR: MOLOPT_REPO not found: ${MOLOPT_REPO}" >&2
  exit 2
fi

export RL_GRAPH_REPO RL_GRAPH_CONDA_ENV
export PYTHONWARNINGS="ignore"

export KMP_DISABLE_SHM=1
export KMP_USE_SHM=0

RUNS_BASE_DIR="${MOLOPT_RESULTS_DIR}"

_na() { echo "NA"; }

_find_grpo_config() {
  local oracle_name="$1"
  local cfg_dir="${REPO_ROOT}/configs/grpo"
  local exact="${cfg_dir}/${oracle_name}.yaml"
  if [[ -f "${exact}" ]]; then
    echo "${exact}"
    return 0
  fi
  local lower="${cfg_dir}/${oracle_name,,}.yaml"
  if [[ -f "${lower}" ]]; then
    echo "${lower}"
    return 0
  fi
  local f
  for f in "${cfg_dir}"/*.yaml; do
    if [[ "${f,,}" == "${cfg_dir,,}/${oracle_name,,}.yaml" ]]; then
      echo "${f}"
      return 0
    fi
  done
  return 1
}

_read_grpo_yaml_key() {
  local cfg_path="$1"
  local key="$2"
  python - <<'PY' "${cfg_path}" "${key}"
import re
import sys

cfg_path = sys.argv[1]
key = sys.argv[2]

val = ""
in_grpo = False
try:
    with open(cfg_path, "r", encoding="utf-8") as f:
        for line in f:
            if re.match(r"^\s*#", line):
                continue
            if re.match(r"^\s*grpo:\s*$", line):
                in_grpo = True
                continue
            if in_grpo and re.match(r"^[^\s]", line):
                in_grpo = False
            if not in_grpo:
                continue
            m = re.match(rf"^\s+{re.escape(key)}\s*:\s*(.*)$", line)
            if not m:
                continue
            raw = m.group(1).split("#", 1)[0].strip()
            if raw in {"", "null", "Null", "NULL", "~"}:
                break
            if (raw.startswith("'") and raw.endswith("'")) or (raw.startswith('"') and raw.endswith('"')):
                raw = raw[1:-1]
            val = raw
            break
except Exception:
    pass

if val:
    print(val)
PY
}

_find_summary_yaml() {
  local base_dir="$1"
  local oracle_name="$2"
  local seed="$3"

  local run_dir="${base_dir}/${oracle_name}_${seed}"
  local summary_yaml="${run_dir}/final_summary_graph_grpo_${oracle_name}_${seed}.yaml"
  if [[ -f "${summary_yaml}" ]]; then
    echo "${summary_yaml}"
    return 0
  fi

  shopt -s nullglob nocaseglob
  local f
  if [[ -d "${run_dir}" ]]; then
    for f in "${run_dir}/final_summary_graph_grpo_"*"_${seed}.yaml"; do
      if [[ -f "${f}" ]]; then
        echo "${f}"
        shopt -u nullglob nocaseglob
        return 0
      fi
    done
  fi
  shopt -u nullglob nocaseglob
  return 1
}

_find_progress_csv() {
  local base_dir="$1"
  local oracle_name="$2"
  local seed="$3"

  local run_dir="${base_dir}/${oracle_name}_${seed}"
  if [[ ! -d "${run_dir}" ]]; then
    return 1
  fi

  shopt -s nullglob nocaseglob
  local latest=""
  local f
  for f in "${run_dir}/progress_graph_grpo_"*"_${seed}.csv"; do
    if [[ -f "${f}" ]]; then
      if [[ -z "${latest}" ]]; then
        latest="${f}"
      else
        if [[ "${f}" -nt "${latest}" ]]; then
          latest="${f}"
        fi
      fi
    fi
  done
  shopt -u nullglob nocaseglob

  if [[ -n "${latest}" ]]; then
    echo "${latest}"
    return 0
  fi
  return 1
}

_parse_progress_last_row() {
  local progress_csv="$1"
  local py_bin="${GRAPH_GRPO_SUMMARY_PYTHON:-python3}"
  if ! command -v "${py_bin}" >/dev/null 2>&1; then
    if command -v python >/dev/null 2>&1; then
      py_bin="python"
    else
      return 0
    fi
  fi
  "${py_bin}" - <<'PY' "${progress_csv}"
import csv
import sys

path = sys.argv[1]
try:
    with open(path, "r", encoding="utf-8") as f:
        reader = csv.reader(f)
        header = next(reader, None)
        if not header:
            raise RuntimeError("missing header")
        rows = [row for row in reader if row]
except Exception:
    rows = []
    header = None

if not header or not rows:
    print("")
    raise SystemExit

last = rows[-1]
def get_col(name: str):
    try:
        idx = header.index(name)
    except ValueError:
        return None
    if idx >= len(last):
        return None
    return last[idx]

avg_top1 = get_col("avg_top1")
avg_top10 = get_col("avg_top10")
auc_top10 = get_col("auc_top10")

if avg_top1 is None or avg_top10 is None or auc_top10 is None:
    print("")
    raise SystemExit

print(f\"{avg_top1},{avg_top10},{auc_top10}\")
PY
}

_append_batch_summary_row() {
  local out_csv="$1"
  local oracle_name="$2"
  local elapsed_sec="${3:-}"

  local summary_yaml
  summary_yaml="$(_find_summary_yaml "${RUNS_BASE_DIR}" "${oracle_name}" "${SEED}" || true)"

  local top1="" top10="" auc10=""
  local py_bin="${GRAPH_GRPO_SUMMARY_PYTHON:-python3}"
  if ! command -v "${py_bin}" >/dev/null 2>&1; then
    if command -v python >/dev/null 2>&1; then
      py_bin="python"
    else
      py_bin=""
    fi
  fi

  if [[ -n "${summary_yaml}" && -f "${summary_yaml}" ]]; then
    local parsed=""
    if [[ -n "${py_bin}" ]]; then
      parsed="$(
        SUMMARY_YAML="${summary_yaml}" \
          "${py_bin}" - <<'PY' 2>/dev/null || true
import base64
import math
import os
import re
import struct

path = os.environ.get("SUMMARY_YAML", "")
try:
    with open(path, "r", encoding="utf-8") as f:
        lines = f.read().splitlines()
except Exception:
    lines = []

key_re = re.compile(r"^[A-Za-z0-9_].*?:")

def is_top_level_key(line: str) -> bool:
    return bool(key_re.match(line))

def parse_float(val: str):
    try:
        x = float(val)
    except Exception:
        return None
    if not math.isfinite(x):
        return None
    return x

def find_scalar(key: str):
    pat = re.compile(rf"^{re.escape(key)}:\s*(.*)$")
    for line in lines:
        m = pat.match(line)
        if not m:
            continue
        tail = m.group(1).strip()
        if tail and not tail.startswith("!!"):
            return parse_float(tail)
        return None
    return None

def parse_numpy_scalar(key: str):
    pat = re.compile(rf"^{re.escape(key)}:\s*(.*)$")
    for i, line in enumerate(lines):
        m = pat.match(line)
        if not m:
            continue
        tail = m.group(1).strip()
        if tail and not tail.startswith("!!"):
            return parse_float(tail)
        in_binary = False
        chunks = []
        for l in lines[i + 1 :]:
            if is_top_level_key(l):
                break
            if "!!binary" in l:
                in_binary = True
                continue
            if not in_binary:
                continue
            if l.strip() == "" or l.lstrip().startswith("-"):
                continue
            if l.startswith(" "):
                chunks.append(l.strip())
        if not chunks:
            return None
        try:
            raw = base64.b64decode("".join(chunks))
        except Exception:
            return None
        if len(raw) == 8:
            val = struct.unpack("<d", raw)[0]
            if not math.isfinite(val) or abs(val) > 1e12:
                val = struct.unpack(">d", raw)[0]
        elif len(raw) == 4:
            val = struct.unpack("<f", raw)[0]
            if not math.isfinite(val) or abs(val) > 1e12:
                val = struct.unpack(">f", raw)[0]
        else:
            return None
        if not math.isfinite(val):
            return None
        return float(val)
    return None

def fmt(x):
    if x is None or not math.isfinite(x):
        return "NA"
    return str(x)

top1 = find_scalar("avg_top1")
top10 = find_scalar("avg_top10")
auc10 = parse_numpy_scalar("auc_top10")

print(",".join([fmt(top1), fmt(top10), fmt(auc10)]))
PY
      )"
    fi
    if [[ -n "${parsed}" ]]; then
      IFS=',' read -r top1 top10 auc10 <<< "${parsed}"
    else
      echo "⚠️  [mol_opt] Failed to parse summary metrics: ${summary_yaml}" >&2
    fi
  else
    echo "⚠️  [mol_opt] Summary not found: oracle=${oracle_name} seed=${SEED} base=${RUNS_BASE_DIR}" >&2
  fi

  local progress_csv=""
  progress_csv="$(_find_progress_csv "${RUNS_BASE_DIR}" "${oracle_name}" "${SEED}" || true)"
  if [[ -n "${progress_csv}" ]]; then
    local parsed_progress=""
    parsed_progress="$(_parse_progress_last_row "${progress_csv}" || true)"
    if [[ -n "${parsed_progress}" ]]; then
      local p_top1="" p_top10="" p_auc10=""
      IFS=',' read -r p_top1 p_top10 p_auc10 <<< "${parsed_progress}"
      if [[ -z "${top1}" || -z "${top10}" || -z "${auc10}" ]]; then
        top1="${p_top1}"
        top10="${p_top10}"
        auc10="${p_auc10}"
      else
        if [[ -n "${py_bin}" ]]; then
          "${py_bin}" - <<'PY' "${top1}" "${top10}" "${auc10}" "${p_top1}" "${p_top10}" "${p_auc10}" "${oracle_name}" "${SEED}" "${progress_csv}"
import math
import sys

def to_float(x):
    try:
        return float(x)
    except Exception:
        return None

top1, top10, auc10, p_top1, p_top10, p_auc10, oracle_name, seed, path = sys.argv[1:]
f1 = to_float(auc10)
f2 = to_float(p_auc10)
if f1 is None or f2 is None:
    raise SystemExit
if abs(f1 - f2) > 1e-6:
    print(f"⚠️  [mol_opt] auc10 mismatch vs progress CSV for {oracle_name} seed={seed}; using progress ({path})", file=sys.stderr)
PY
        fi
        if [[ -n "${p_top1}" && -n "${p_top10}" && -n "${p_auc10}" ]]; then
          local mismatch=0
          if [[ -n "${py_bin}" ]]; then
            mismatch="$("${py_bin}" - <<'PY' "${auc10}" "${p_auc10}"
import sys
try:
    a = float(sys.argv[1])
    b = float(sys.argv[2])
    print(int(abs(a - b) > 1e-6))
except Exception:
    print(0)
PY
)"
          fi
          if [[ "${mismatch}" == "1" ]]; then
            top1="${p_top1}"
            top10="${p_top10}"
            auc10="${p_auc10}"
          fi
        fi
      fi
    fi
  fi

  [[ -n "${top1}" ]] || top1="$(_na)"
  [[ -n "${top10}" ]] || top10="$(_na)"
  [[ -n "${auc10}" ]] || auc10="$(_na)"
  [[ -n "${elapsed_sec}" ]] || elapsed_sec="$(_na)"

  printf "%s,%s,%s,%s,%s,%s\n" "${oracle_name}" "${SEED}" "${top1}" "${top10}" "${auc10}" "${elapsed_sec}" >> "${out_csv}"
}

_resolve_ckpt() {
  local root="$1"
  local name="$2"

  if [[ -f "${root}/${name}.ckpt" ]]; then
    echo "${root}/${name}.ckpt"
    return 0
  fi
  if [[ -f "${root}/${name}" ]]; then
    echo "${root}/${name}"
    return 0
  fi

  if [[ -d "${root}/${name}" ]]; then
    local latest
    latest="$(ls -t "${root}/${name}"/*.ckpt 2>/dev/null | head -n 1 || true)"
    if [[ -n "${latest}" ]]; then
      echo "${latest}"
      return 0
    fi
  fi

  return 1
}

_run_one() {
  local ckpt_path="$1"
  local oracle_name="$2"
  local ckpt_label="(default)"
  if [[ -n "${ckpt_path}" ]]; then
    ckpt_label="${ckpt_path}"
    export GRAPH_GRPO_CKPT="${ckpt_path}"
  else
    unset GRAPH_GRPO_CKPT
  fi
  echo "🧪 [mol_opt] oracle=${oracle_name} ckpt=${ckpt_label} max_oracle_calls=${MAX_ORACLE_CALLS} seed=${SEED}" >&2



  local early_patience=""
  local early_eps="${GRAPH_GRPO_EARLY_STOP_EPS:-}"
  local early_min_calls="${GRAPH_GRPO_EARLY_STOP_MIN_CALLS:-}"
  local early_topk="${GRAPH_GRPO_EARLY_STOP_TOPK:-}"

  if [[ -z "${early_patience}" || -z "${early_eps}" || -z "${early_min_calls}" || -z "${early_topk}" ]]; then
    local cfg_path
    cfg_path="$(_find_grpo_config "${oracle_name}" || true)"
    if [[ -n "${cfg_path}" ]]; then
      [[ -z "${early_patience}" ]] && early_patience="$(_read_grpo_yaml_key "${cfg_path}" "early_stop_patience")"
      [[ -z "${early_eps}" ]] && early_eps="$(_read_grpo_yaml_key "${cfg_path}" "early_stop_eps")"
      [[ -z "${early_min_calls}" ]] && early_min_calls="$(_read_grpo_yaml_key "${cfg_path}" "early_stop_min_calls")"
      [[ -z "${early_topk}" ]] && early_topk="$(_read_grpo_yaml_key "${cfg_path}" "early_stop_topk")"
    fi
  fi

  if [[ -n "${early_patience}" && "${early_patience}" != "0" ]]; then
    [[ -n "${early_eps}" ]] || early_eps="0"


    [[ -n "${early_min_calls}" ]] || early_min_calls="0"
    [[ -n "${early_topk}" ]] || early_topk="10"
    local ckpt_for_fast="${ckpt_path}"
    if [[ "${USE_DEFAULT_CKPT}" -eq 1 && -z "${ckpt_for_fast}" ]]; then
      ckpt_for_fast="__default__"
    fi
    PYTHONUNBUFFERED=1 conda run --no-capture-output -n "${MOLOPT_CONDA_ENV}" \
      python -u "${REPO_ROOT}/scripts/run_mol_opt_graph_grpo_fast.py" \
      --ckpt "${ckpt_for_fast}" \
      --oracle "${oracle_name}" \
      --max-oracle-calls "${MAX_ORACLE_CALLS}" \
      --seed "${SEED}" \
      --output-dir "${RUNS_BASE_DIR}" \
      --early-stop \
      --early-stop-topk "${early_topk}" \
      --early-stop-patience "${early_patience}" \
      --early-stop-eps "${early_eps}" \
      --early-stop-min-calls "${early_min_calls}"
    return 0
  fi

  PYTHONUNBUFFERED=1 conda run --no-capture-output -n "${MOLOPT_CONDA_ENV}" \
    python -u "${MOLOPT_REPO}/run.py" \
    --method graph_grpo \
    --task simple \
    --oracles "${oracle_name}" \
    --max_oracle_calls "${MAX_ORACLE_CALLS}" \
    --seed "${SEED}" \
    --output_dir "${RUNS_BASE_DIR}"
}

_write_result_csv() {
  local summary_csv="$1"
  local result_csv="$2"
  local py_bin="${GRAPH_GRPO_SUMMARY_PYTHON:-python3}"
  if ! command -v "${py_bin}" >/dev/null 2>&1; then
    if command -v python >/dev/null 2>&1; then
      py_bin="python"
    else
      return 0
    fi
  fi

  "${py_bin}" - <<'PY' "${summary_csv}" "${result_csv}"
import csv
import math
import sys

summary_csv = sys.argv[1]
result_csv = sys.argv[2]

try:
    with open(summary_csv, "r", encoding="utf-8") as f:
        reader = csv.DictReader(f)
        rows = list(reader)
except Exception:
    sys.exit(0)

if not rows:
    sys.exit(0)

by_task = {}
by_seed = {}
for row in rows:
    task = row.get("task")
    seed = row.get("seed")
    if not task or not seed:
        continue
    try:
        val = float(row.get("auc10", ""))
    except Exception:
        continue
    if not math.isfinite(val):
        continue
    by_task.setdefault(task, []).append(val)
    by_seed.setdefault(seed, 0.0)
    by_seed[seed] += val

def mean_std(vals):
    if not vals:
        return None, None
    mean = sum(vals) / len(vals)
    var = sum((v - mean) ** 2 for v in vals) / len(vals)
    return mean, math.sqrt(var)

with open(result_csv, "w", encoding="utf-8", newline="") as f:
    writer = csv.writer(f)
    writer.writerow(["task", "auc10_mean", "auc10_std"])
    for task in sorted(by_task.keys()):
        mean, std = mean_std(by_task[task])
        if mean is None:
            continue
        writer.writerow([task, mean, std])

    total_vals = list(by_seed.values())
    mean, std = mean_std(total_vals)
    if mean is not None:
        writer.writerow(["total", mean, std])
PY
}

if [[ "${MODE}" == "--batch" || "${MODE}" == "--all" ]]; then
  ORACLES=(
    albuterol_similarity
    amlodipine_mpo
    celecoxib_rediscovery
    deco_hop
    drd2
    fexofenadine_mpo
    gsk3b
    isomers_c7h8n2o2
    isomers_c9h10n2o2pf2cl
    jnk3
    median1
    median2
    mestranol_similarity
    osimertinib_mpo
    perindopril_mpo
    qed
    ranolazine_mpo
    scaffold_hop
    sitagliptin_mpo
    thiothixene_rediscovery
    troglitazone_rediscovery
    valsartan_smarts
    zaleplon_mpo
  )

  ts="$(date +%Y%m%d_%H%M%S)"
  seed_tag=""
  if [[ "${#SEEDS[@]}" -gt 1 ]]; then
    seed_tag="$(printf "%s_" "${SEEDS[@]}")"
    seed_tag="${seed_tag%_}"
  fi

  MULTI_BASE_DIR=""
  COMBINED_CSV=""
  RESULT_CSV=""
  if [[ "${#SEEDS[@]}" -gt 1 ]]; then
    MULTI_BASE_DIR="${GRAPH_GRPO_BATCH_OUTPUT_DIR:-${MOLOPT_RESULTS_DIR}/batch_${ts}_seeds${seed_tag}_budget${MAX_ORACLE_CALLS}}"
    mkdir -p "${MULTI_BASE_DIR}"
    COMBINED_CSV="${MULTI_BASE_DIR}/batch_summary_seeds${seed_tag}_budget${MAX_ORACLE_CALLS}.csv"
    RESULT_CSV="${MULTI_BASE_DIR}/result.csv"
    echo "task,seed,top1,top10,auc10,elapsed_sec" > "${COMBINED_CSV}"
  fi

  for SEED in "${SEEDS[@]}"; do
    if [[ -n "${MULTI_BASE_DIR}" ]]; then
      RUNS_BASE_DIR="${MULTI_BASE_DIR}/seed${SEED}"
    else
      RUNS_BASE_DIR="${GRAPH_GRPO_BATCH_OUTPUT_DIR:-${MOLOPT_RESULTS_DIR}/batch_${ts}_seed${SEED}_budget${MAX_ORACLE_CALLS}}"
    fi
    mkdir -p "${RUNS_BASE_DIR}"

    {
      echo "command: $0 $*"
      echo "timestamp: ${ts}"
      echo "ckpt_root: ${CKPT_ROOT}"
      echo "single_ckpt: ${SINGLE_CKPT}"
      echo "max_oracle_calls: ${MAX_ORACLE_CALLS}"
      echo "seed: ${SEED}"
      echo "seeds: ${SEEDS[*]}"
      echo "GRAPH_GRPO_USE_DEFAULT_CKPT: ${GRAPH_GRPO_USE_DEFAULT_CKPT:-}"
      echo "GRAPH_GRPO_DISABLE_REFINE: ${GRAPH_GRPO_DISABLE_REFINE:-}"
      echo "GRAPH_GRPO_SCREEN_MODE: ${GRAPH_GRPO_SCREEN_MODE:-}"
      echo "GRAPH_GRPO_SCREEN_CSV: ${GRAPH_GRPO_SCREEN_CSV:-}"
      echo "GRAPH_GRPO_SCREEN_COLUMN: ${GRAPH_GRPO_SCREEN_COLUMN:-}"
      echo "GRAPH_GRPO_SCREEN_TOPK: ${GRAPH_GRPO_SCREEN_TOPK:-}"
      echo "GRAPH_GRPO_SCREEN_CACHE_DIR: ${GRAPH_GRPO_SCREEN_CACHE_DIR:-}"
      echo "GRAPH_GRPO_EARLY_STOP_EPS: ${GRAPH_GRPO_EARLY_STOP_EPS:-}"
      echo "GRAPH_GRPO_EARLY_STOP_MIN_CALLS: ${GRAPH_GRPO_EARLY_STOP_MIN_CALLS:-}"
    } > "${RUNS_BASE_DIR}/run_meta.txt"

    SUMMARY_CSV="${RUNS_BASE_DIR}/batch_summary_seed${SEED}_budget${MAX_ORACLE_CALLS}.csv"
    echo "task,seed,top1,top10,auc10,elapsed_sec" > "${SUMMARY_CSV}"

    missing=0
    for oracle in "${ORACLES[@]}"; do
      ckpt=""
      if [[ -n "${SINGLE_CKPT}" ]]; then
        ckpt="${SINGLE_CKPT}"
      elif [[ "${USE_DEFAULT_CKPT}" -ne 1 ]]; then
        ckpt="$(_resolve_ckpt "${CKPT_ROOT}" "${oracle}" || true)"
        if [[ -z "${ckpt}" ]]; then
          echo "⚠️  [mol_opt] SKIP (ckpt not found): oracle=${oracle} root=${CKPT_ROOT}" >&2
          missing=$((missing + 1))
          _append_batch_summary_row "${SUMMARY_CSV}" "${oracle}" "$(_na)"
          continue
        fi
      fi
      start_ts="$(date +%s)"
      _run_one "${ckpt}" "${oracle}"
      end_ts="$(date +%s)"
      elapsed_sec="$((end_ts - start_ts))"
      _append_batch_summary_row "${SUMMARY_CSV}" "${oracle}" "${elapsed_sec}"
    done
    echo "📦 [mol_opt] All outputs written under: ${RUNS_BASE_DIR}" >&2
    echo "📄 [mol_opt] Batch summary written to: ${SUMMARY_CSV}" >&2
    if [[ -n "${COMBINED_CSV}" ]]; then
      tail -n +2 "${SUMMARY_CSV}" >> "${COMBINED_CSV}"
    fi
    if [[ "${missing}" -gt 0 ]]; then
      echo "⚠️  [mol_opt] Done with ${missing} task(s) skipped due to missing ckpt(s)." >&2
    fi
  done

  if [[ -n "${COMBINED_CSV}" ]]; then
    echo "📄 [mol_opt] Multi-seed summary written to: ${COMBINED_CSV}" >&2
    if [[ -n "${RESULT_CSV}" ]]; then
      _write_result_csv "${COMBINED_CSV}" "${RESULT_CSV}"
      echo "📄 [mol_opt] Result summary written to: ${RESULT_CSV}" >&2
    fi
  fi
else
  _run_one "${CKPT_PATH}" "${ORACLE_NAME}"
fi
