# splits.py
from typing import List, Dict, Tuple
import numpy as np
from sklearn.model_selection import KFold

OUTER_SPLITS = 5
INNER_SPLITS = 5
SEED_OUTER = 2
SEED_INNER = 2


def make_nested_split_plan(n_samples: int) -> List[Dict[str, List[Tuple[np.ndarray, np.ndarray]]]]:
    """
    Precompute the full split plan:
      - outer_splits: list of (train_idx, test_idx) for OUTER_SPLITS folds
      - inner_splits_per_outer: for each outer fold, a list of (in_train_idx, in_val_idx)
        where these are RELATIVE indices within the outer TRAIN set.
    Returns a list of dicts, one per outer fold:
      [
        {
          "outer": (train_idx, test_idx),
          "inner": [(in_tr_rel, in_va_rel), ...],  # length INNER_SPLITS
        },
        ...
      ]
    """
    outer_kf = KFold(n_splits=OUTER_SPLITS, shuffle=True, random_state=SEED_OUTER)
    plan: List[Dict[str, List[Tuple[np.ndarray, np.ndarray]]]] = []

    for tr_idx, te_idx in outer_kf.split(np.arange(n_samples)):
        # Inner KFold is over the OUTER TRAIN set (relative indices)
        inner_kf = KFold(n_splits=INNER_SPLITS, shuffle=True, random_state=SEED_INNER)
        inner_rel_indices = list(inner_kf.split(np.arange(len(tr_idx))))
        plan.append({
            "outer": (tr_idx, te_idx),
            "inner": inner_rel_indices,
        })
    return plan
