{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Multi-Seed Robustness Check\n",
    "\n",
    "This notebook runs all 6 models with 4 different seeds to check result stability.\n",
    "\n",
    "**Key Features:**\n",
    "- Checkpointing: Saves after each model/seed completion\n",
    "- Resume capability: Continues from last checkpoint if interrupted\n",
    "- Memory management: Aggressive cleanup after each experiment (10GB/11GB GPU usage)\n",
    "- Reuse existing: Can load seed=42 models from `checkpoints/` folder\n",
    "- Statistics: Computes mean ± std across all 4 seeds\n",
    "- Safe storage: Uses `checkpoints/multi_seed/` to avoid overwriting existing models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2026-01-25 12:47:04.460610: I tensorflow/core/util/port.cc:113] oneDNN custom operations are on. You may see slightly different numerical results due to floating-point round-off errors from different computation orders. To turn them off, set the environment variable `TF_ENABLE_ONEDNN_OPTS=0`.\n",
      "2026-01-25 12:47:04.490549: E external/local_xla/xla/stream_executor/cuda/cuda_dnn.cc:9261] Unable to register cuDNN factory: Attempting to register factory for plugin cuDNN when one has already been registered\n",
      "2026-01-25 12:47:04.490583: E external/local_xla/xla/stream_executor/cuda/cuda_fft.cc:607] Unable to register cuFFT factory: Attempting to register factory for plugin cuFFT when one has already been registered\n",
      "2026-01-25 12:47:04.491565: E external/local_xla/xla/stream_executor/cuda/cuda_blas.cc:1515] Unable to register cuBLAS factory: Attempting to register factory for plugin cuBLAS when one has already been registered\n",
      "2026-01-25 12:47:04.497106: I tensorflow/core/platform/cpu_feature_guard.cc:182] This TensorFlow binary is optimized to use available CPU instructions in performance-critical operations.\n",
      "To enable the following instructions: AVX2 AVX512F AVX512_VNNI AVX512_BF16 FMA, in other operations, rebuild TensorFlow with the appropriate compiler flags.\n",
      "2026-01-25 12:47:06.021404: W tensorflow/compiler/tf2tensorrt/utils/py_utils.cc:38] TF-TRT Warning: Could not find TensorRT\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✓ GPU memory limited to 30GB\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2026-01-25 12:47:09.202286: W tensorflow/core/common_runtime/gpu/gpu_device.cc:2348] TensorFlow was not built with CUDA kernel binaries compatible with compute capability 9.0. CUDA kernels will be jit-compiled from PTX, which could take 30 minutes or longer.\n"
     ]
    }
   ],
   "source": [
    "# CRITICAL: GPU configuration MUST happen BEFORE TensorFlow is imported!\n",
    "import os\n",
    "os.environ['TF_FORCE_GPU_ALLOW_GROWTH'] = 'true'\n",
    "os.environ['TF_DETERMINISTIC_OPS'] = '1'\n",
    "os.environ['TF_CUDNN_DETERMINISTIC'] = '1'\n",
    "\n",
    "# Now import TensorFlow\n",
    "import gc\n",
    "import json\n",
    "import time\n",
    "import pickle\n",
    "import shutil\n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "from datetime import datetime\n",
    "from pathlib import Path\n",
    "import tensorflow as tf\n",
    "\n",
    "# Configure GPU memory limit (30GB) - must be done before any TF operations\n",
    "gpus = tf.config.list_physical_devices('GPU')\n",
    "if gpus:\n",
    "    try:\n",
    "        # Option 1: Set hard memory limit (recommended for multi-experiment runs)\n",
    "        tf.config.set_logical_device_configuration(\n",
    "            gpus[0],\n",
    "            [tf.config.LogicalDeviceConfiguration(memory_limit=30*1024)]  # 30GB\n",
    "        )\n",
    "        print(f\"✓ GPU memory limited to 30GB\")\n",
    "    except RuntimeError as e:\n",
    "        print(f\"GPU already initialized: {e}\")\n",
    "        # Fallback: at least try memory growth\n",
    "        try:\n",
    "            tf.config.experimental.set_memory_growth(gpus[0], True)\n",
    "        except:\n",
    "            pass"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Configuration"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Running 4 seeds: [42, 123, 456, 2026]\n",
      "Total experiments: 4 × 6 = 24\n",
      "\n",
      "Checkpoint directory: checkpoints/multi_seed\n",
      "Results directory: multi_seed_results\n",
      "\n",
      "Reuse existing seed=42 models: True\n",
      "  Loading from: checkpoints\n",
      "\n",
      "Estimated time (new seeds): ~22.3 hours\n"
     ]
    }
   ],
   "source": [
    "# Experiment configuration\n",
    "SEEDS = [42, 123, 456, 2026]  # seeds total\n",
    "CHECKPOINT_DIR = \"checkpoints/multi_seed\"  # Dedicated folder (won't overwrite existing)\n",
    "RESULTS_DIR = \"multi_seed_results\"\n",
    "EXISTING_MODELS_DIR = \"checkpoints\"  # Original checkpoint folder\n",
    "\n",
    "# Model names for tracking\n",
    "MODEL_NAMES = [\n",
    "    \"Baseline_Transformer\",\n",
    "    \"Full_Rank_BBB\",\n",
    "    \"Low_Rank_Random_Init\",\n",
    "    \"Low_Rank_SVD_Init\",\n",
    "    \"Rank1_BBB\",\n",
    "    \"Deep_Ensemble\"\n",
    "]\n",
    "\n",
    "# Mapping from model names to existing checkpoint files (for seed=42 reuse)\n",
    "EXISTING_MODEL_MAPPING = {\n",
    "    \"Baseline_Transformer\": \"model_0.h5\",\n",
    "    \"Full_Rank_BBB\": \"model_1.h5\",\n",
    "    \"Low_Rank_Random_Init\": \"model_2.h5\",\n",
    "    \"Low_Rank_SVD_Init\": \"model_3.h5\",\n",
    "    \"Rank1_BBB\": \"model_4.h5\",\n",
    "    \"Deep_Ensemble\": \"model_5\"  # Directory for ensemble\n",
    "}\n",
    "\n",
    "# Option to reuse existing seed=42 models\n",
    "REUSE_SEED_42 = True  # Set to False to retrain everything\n",
    "\n",
    "# Create directories\n",
    "os.makedirs(CHECKPOINT_DIR, exist_ok=True)\n",
    "os.makedirs(RESULTS_DIR, exist_ok=True)\n",
    "\n",
    "print(f\"Running {len(SEEDS)} seeds: {SEEDS}\")\n",
    "print(f\"Total experiments: {len(SEEDS)} × {len(MODEL_NAMES)} = {len(SEEDS) * len(MODEL_NAMES)}\")\n",
    "print(f\"\\nCheckpoint directory: {CHECKPOINT_DIR}\")\n",
    "print(f\"Results directory: {RESULTS_DIR}\")\n",
    "print(f\"\\nReuse existing seed=42 models: {REUSE_SEED_42}\")\n",
    "if REUSE_SEED_42:\n",
    "    print(f\"  Loading from: {EXISTING_MODELS_DIR}\")\n",
    "print(f\"\\nEstimated time (new seeds): ~{(len(SEEDS) - (1 if REUSE_SEED_42 else 0)) * (25000 + 1800) / 3600:.1f} hours\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Import Required Modules"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "2026-01-25 12:47:09.996076: W tensorflow/core/common_runtime/gpu/gpu_device.cc:2348] TensorFlow was not built with CUDA kernel binaries compatible with compute capability 9.0. CUDA kernels will be jit-compiled from PTX, which could take 30 minutes or longer.\n",
      "2026-01-25 12:47:10.154859: W tensorflow/core/common_runtime/gpu/gpu_bfc_allocator.cc:47] Overriding orig_value setting because the TF_FORCE_GPU_ALLOW_GROWTH environment variable is set. Original config value was 0.\n",
      "2026-01-25 12:47:10.155969: I tensorflow/core/common_runtime/gpu/gpu_device.cc:1929] Created device /job:localhost/replica:0/task:0/device:GPU:0 with 30720 MB memory:  -> device: 0, name: NVIDIA H100 NVL, pci bus id: 0000:c3:00.0, compute capability: 9.0\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "All modules imported successfully!\n",
      "\n",
      "Configuration:\n",
      "  VOCAB_SIZE: 30522\n",
      "  MAX_LEN: 64\n",
      "  EPOCHS_BASELINE: 5\n",
      "  EPOCHS_BAYESIAN: 20\n",
      "  EPOCHS_ENSEMBLE: 5\n",
      "  N_ENSEMBLE_MEMBERS: 5\n",
      "  KL_WARMUP_EPOCHS: 5\n",
      "  KL_ZERO_EPOCHS: 2\n",
      "  N_MC_SAMPLES: 100\n"
     ]
    }
   ],
   "source": [
    "# Cell 1: Imports & Setup\n",
    "\n",
    "import os\n",
    "import gc\n",
    "import time\n",
    "import random\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "from functools import partial\n",
    "\n",
    "# Import our custom modules\n",
    "from modules.config_seed import (\n",
    "    set_seed, SEED,\n",
    "    DATA_DIR, BATCH_SIZE, MAX_LEN, VOCAB_SIZE,\n",
    "    MODEL_CONFIG,\n",
    "    EPOCHS_BASELINE, EPOCHS_BAYESIAN, EPOCHS_ENSEMBLE,\n",
    "    N_ENSEMBLE_MEMBERS, N_MC_SAMPLES, OOD_SLICE,\n",
    "    KL_WARMUP_EPOCHS,\n",
    ")\n",
    "\n",
    "# Define KL_ZERO_EPOCHS locally (matches training.py default)\n",
    "KL_ZERO_EPOCHS = 2\n",
    "\n",
    "\n",
    "from modules.model_builders import (\n",
    "    build_tiny_transformer_baseline,\n",
    "    build_tiny_transformer_fullrank_bbb,\n",
    "    build_tiny_transformer_lowrank_bbb,\n",
    "    build_tiny_transformer_rank1_bbb,\n",
    "    DeepEnsemble,\n",
    "    clean_summary,\n",
    ")\n",
    "\n",
    "from modules.data_loading import (\n",
    "    load_all_datasets,\n",
    "    prepare_evaluation_data,\n",
    "    print_all_balances,\n",
    ")\n",
    "\n",
    "from modules.training import (\n",
    "    train_baseline_model,\n",
    "    train_bayesian_model,\n",
    ")\n",
    "\n",
    "from modules.evaluation import (\n",
    "    evaluate_all_metrics_unified,\n",
    ")\n",
    "\n",
    "import tensorflow as tf\n",
    "import tensorflow_probability as tfp\n",
    "set_seed(SEED)\n",
    "tfd = tfp.distributions\n",
    "\n",
    "print(\"\\nAll modules imported successfully!\")\n",
    "print(f\"\\nConfiguration:\")\n",
    "print(f\"  VOCAB_SIZE: {VOCAB_SIZE}\")\n",
    "print(f\"  MAX_LEN: {MAX_LEN}\")\n",
    "print(f\"  EPOCHS_BASELINE: {EPOCHS_BASELINE}\")\n",
    "print(f\"  EPOCHS_BAYESIAN: {EPOCHS_BAYESIAN}\")\n",
    "print(f\"  EPOCHS_ENSEMBLE: {EPOCHS_ENSEMBLE}\")\n",
    "print(f\"  N_ENSEMBLE_MEMBERS: {N_ENSEMBLE_MEMBERS}\")\n",
    "print(f\"  KL_WARMUP_EPOCHS: {KL_WARMUP_EPOCHS}\")\n",
    "print(f\"  KL_ZERO_EPOCHS: {KL_ZERO_EPOCHS}\")\n",
    "print(f\"  N_MC_SAMPLES: {N_MC_SAMPLES}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Memory Management Utilities"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✓ Memory management utilities defined\n"
     ]
    }
   ],
   "source": [
    "def aggressive_memory_cleanup():\n",
    "    \"\"\"\n",
    "    Aggressive memory cleanup to prevent OOM errors.\n",
    "    Critical since each run uses 10GB/11GB available GPU memory.\n",
    "    \"\"\"\n",
    "    # Clear Keras session\n",
    "    tf.keras.backend.clear_session()\n",
    "    \n",
    "    # Force garbage collection\n",
    "    gc.collect()\n",
    "    \n",
    "    # Additional TensorFlow cleanup\n",
    "    try:\n",
    "        tf.compat.v1.reset_default_graph()\n",
    "    except:\n",
    "        pass\n",
    "    \n",
    "    print(\"🧹 Memory cleaned\")\n",
    "\n",
    "def get_gpu_memory_info():\n",
    "    \"\"\"Get current GPU memory usage.\"\"\"\n",
    "    try:\n",
    "        gpus = tf.config.list_physical_devices('GPU')\n",
    "        if gpus:\n",
    "            memory_info = tf.config.experimental.get_memory_info('GPU:0')\n",
    "            return memory_info\n",
    "    except:\n",
    "        pass\n",
    "    return None\n",
    "\n",
    "print(\"✓ Memory management utilities defined\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Checkpoint & Progress Tracking"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "📂 Resuming: 20 experiments already completed\n",
      "✓ Experiment tracker initialized\n"
     ]
    }
   ],
   "source": [
    "class ExperimentTracker:\n",
    "    \"\"\"Tracks experiment progress and manages checkpoints.\"\"\"\n",
    "    \n",
    "    def __init__(self, checkpoint_dir):\n",
    "        self.checkpoint_dir = Path(checkpoint_dir)\n",
    "        self.progress_file = self.checkpoint_dir / \"progress.json\"\n",
    "        self.results_file = self.checkpoint_dir / \"results.pkl\"\n",
    "        self.load_progress()\n",
    "    \n",
    "    def load_progress(self):\n",
    "        \"\"\"Load existing progress or initialize new tracking.\"\"\"\n",
    "        if self.progress_file.exists():\n",
    "            with open(self.progress_file, 'r') as f:\n",
    "                self.progress = json.load(f)\n",
    "            print(f\"\\n📂 Resuming: {len(self.progress['completed'])} experiments already completed\")\n",
    "        else:\n",
    "            self.progress = {\n",
    "                'completed': [],\n",
    "                'start_time': datetime.now().isoformat(),\n",
    "                'timings': {},\n",
    "                'reused_seed_42': []\n",
    "            }\n",
    "        \n",
    "        # Load results\n",
    "        if self.results_file.exists():\n",
    "            with open(self.results_file, 'rb') as f:\n",
    "                self.results = pickle.load(f)\n",
    "        else:\n",
    "            self.results = {}\n",
    "    \n",
    "    def is_completed(self, seed, model_name):\n",
    "        \"\"\"Check if experiment is already completed.\"\"\"\n",
    "        key = f\"{seed}_{model_name}\"\n",
    "        return key in self.progress['completed']\n",
    "    \n",
    "    def mark_reused(self, seed, model_name):\n",
    "        \"\"\"Mark that existing model was reused.\"\"\"\n",
    "        key = f\"{seed}_{model_name}\"\n",
    "        if key not in self.progress['reused_seed_42']:\n",
    "            self.progress['reused_seed_42'].append(key)\n",
    "    \n",
    "    def save_result(self, seed, model_name, metrics, training_time, reused=False):\n",
    "        \"\"\"Save experiment result and update checkpoint.\"\"\"\n",
    "        key = f\"{seed}_{model_name}\"\n",
    "        \n",
    "        # Store result\n",
    "        self.results[key] = {\n",
    "            'seed': seed,\n",
    "            'model': model_name,\n",
    "            'metrics': metrics,\n",
    "            'training_time': training_time,\n",
    "            'reused': reused,\n",
    "            'timestamp': datetime.now().isoformat()\n",
    "        }\n",
    "        \n",
    "        # Update progress\n",
    "        if key not in self.progress['completed']:\n",
    "            self.progress['completed'].append(key)\n",
    "        self.progress['timings'][key] = training_time\n",
    "        \n",
    "        # Save checkpoint\n",
    "        with open(self.progress_file, 'w') as f:\n",
    "            json.dump(self.progress, f, indent=2)\n",
    "        \n",
    "        with open(self.results_file, 'wb') as f:\n",
    "            pickle.dump(self.results, f)\n",
    "        \n",
    "        status = \"♻️  Reused\" if reused else \"✅ Trained\"\n",
    "        print(f\"{status}: {key} ({training_time:.1f}s)\")\n",
    "    \n",
    "    def get_summary(self):\n",
    "        \"\"\"Get summary of completed experiments.\"\"\"\n",
    "        total = len(SEEDS) * len(MODEL_NAMES)\n",
    "        completed = len(self.progress['completed'])\n",
    "        return {\n",
    "            'total': total,\n",
    "            'completed': completed,\n",
    "            'remaining': total - completed,\n",
    "            'progress_pct': (completed / total) * 100,\n",
    "            'reused': len(self.progress.get('reused_seed_42', []))\n",
    "        }\n",
    "\n",
    "tracker = ExperimentTracker(CHECKPOINT_DIR)\n",
    "print(\"✓ Experiment tracker initialized\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Load Data (Once)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Loading data from processed_data_agnews... (seed=42)\n",
      "Train (SST-2): (67349, 64)\n",
      "Dev (SST-2):   (872, 64)\n",
      "OOD (IMDb):    (7600, 64)\n",
      "Ready. KL Weight: 0.000001\n",
      "\n",
      "============================================================\n",
      "CLASS BALANCE\n",
      "============================================================\n",
      "--- Training Set (67349 samples) ---\n",
      "  Class 0 (Negative): 29780 (44.2%)\n",
      "  Class 1 (Positive): 37569 (55.8%)\n",
      "--- Validation Set (Dev) (872 samples) ---\n",
      "  Class 0 (Negative): 428 (49.1%)\n",
      "  Class 1 (Positive): 444 (50.9%)\n",
      "--- OOD Set (IMDB) (7600 samples) ---\n",
      "  Class 0 (Negative): 1900 (25.0%)\n",
      "  Class 1 (Positive): 1900 (25.0%)\n",
      "  Class 2 (Negative): 1900 (25.0%)\n",
      "  Class 3 (Negative): 1900 (25.0%)\n",
      "\n",
      "Evaluation data prepared:\n",
      "  ID Test: 872 samples\n",
      "  OOD Test: 7600 samples\n"
     ]
    }
   ],
   "source": [
    "# Cell 3: Data Loading & Preprocessing\n",
    "\n",
    "# Load all datasets\n",
    "data = load_all_datasets(data_dir=DATA_DIR, batch_size=BATCH_SIZE)\n",
    "\n",
    "# Extract datasets and arrays\n",
    "train_dataset = data['train_dataset']\n",
    "val_dataset = data['val_dataset']\n",
    "ood_dataset = data['ood_dataset']\n",
    "\n",
    "train_labels = data['train_labels']\n",
    "dev_labels = data['dev_labels']\n",
    "ood_labels = data['ood_labels']\n",
    "\n",
    "vocab_size = data['vocab_size']\n",
    "kl_weight = data['kl_weight']\n",
    "\n",
    "# Print class balance\n",
    "print(\"\\n\" + \"=\" * 60)\n",
    "print(\"CLASS BALANCE\")\n",
    "print(\"=\" * 60)\n",
    "print_all_balances(train_labels, dev_labels, ood_labels)\n",
    "\n",
    "# Prepare evaluation data dictionaries\n",
    "X_test_dict, y_test_arr, X_ood_dict, y_ood_arr = prepare_evaluation_data(\n",
    "    data['dev_ids'], data['dev_mask'], data['dev_labels'],\n",
    "    data['ood_ids'], data['ood_mask'], data['ood_labels'],\n",
    "    ood_slice=OOD_SLICE\n",
    ")\n",
    "\n",
    "print(f\"\\nEvaluation data prepared:\")\n",
    "print(f\"  ID Test: {len(y_test_arr)} samples\")\n",
    "print(f\"  OOD Test: {len(y_ood_arr)} samples\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Model Training & Evaluation Functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✓ Training and evaluation functions defined\n",
      "✓ Now using DISK-BASED baseline caching (no more RAM hoarding!)\n"
     ]
    }
   ],
   "source": [
    "def load_existing_model(model_name, seed=42):\n",
    "    \"\"\"\n",
    "    Load existing model from checkpoints folder.\n",
    "    Only works for seed=42.\n",
    "    \n",
    "    Returns:\n",
    "        model or None if not found/not seed 42\n",
    "    \"\"\"\n",
    "    if seed != 42 or not REUSE_SEED_42:\n",
    "        return None\n",
    "    \n",
    "    if model_name not in EXISTING_MODEL_MAPPING:\n",
    "        return None\n",
    "    \n",
    "    model_path = Path(EXISTING_MODELS_DIR) / EXISTING_MODEL_MAPPING[model_name]\n",
    "    \n",
    "    if not model_path.exists():\n",
    "        print(f\"  ⚠️  Existing model not found: {model_path}\")\n",
    "        return None\n",
    "    \n",
    "    try:\n",
    "        if model_name == \"Deep_Ensemble\":\n",
    "            # Load ensemble - returns DeepEnsemble object\n",
    "            ensemble_models = []\n",
    "            for i in range(N_ENSEMBLE_MEMBERS):\n",
    "                member_path = model_path / f\"member_{i}.h5\"\n",
    "                if member_path.exists():\n",
    "                    model = tf.keras.models.load_model(member_path, compile=False)\n",
    "                    ensemble_models.append(model)\n",
    "                else:\n",
    "                    print(f\"  ⚠️  Ensemble member {i} not found\")\n",
    "                    return None\n",
    "            print(f\"  ♻️  Loaded {len(ensemble_models)} ensemble members from {model_path}\")\n",
    "            # Create DeepEnsemble wrapper with loaded models\n",
    "            ensemble = DeepEnsemble.__new__(DeepEnsemble)\n",
    "            ensemble.members = ensemble_models\n",
    "            ensemble.n_members = len(ensemble_models)\n",
    "            return ensemble\n",
    "        else:\n",
    "            # Load single model\n",
    "            model = tf.keras.models.load_model(model_path, compile=False)\n",
    "            print(f\"  ♻️  Loaded existing model from {model_path}\")\n",
    "            return model\n",
    "    except Exception as e:\n",
    "        print(f\"  ⚠️  Failed to load model: {e}\")\n",
    "        import traceback\n",
    "        traceback.print_exc()\n",
    "        return None\n",
    "\n",
    "\n",
    "def save_baseline_to_disk(model, seed, temp_dir=\"temp_baseline_cache\"):\n",
    "    \"\"\"\n",
    "    Save baseline model to disk instead of keeping in RAM.\n",
    "    \n",
    "    Parameters:\n",
    "    -----------\n",
    "    model : tf.keras.Model\n",
    "        Baseline model to save\n",
    "    seed : int\n",
    "        Random seed (used for filename)\n",
    "    temp_dir : str\n",
    "        Temporary directory for cached baselines\n",
    "    \n",
    "    Returns:\n",
    "        Path to saved model\n",
    "    \"\"\"\n",
    "    os.makedirs(temp_dir, exist_ok=True)\n",
    "    cache_path = Path(temp_dir) / f\"baseline_seed_{seed}.h5\"\n",
    "    model.save(cache_path)\n",
    "    print(f\"  💾 Saved baseline to disk: {cache_path}\")\n",
    "    return cache_path\n",
    "\n",
    "\n",
    "def load_baseline_from_disk(seed, temp_dir=\"temp_baseline_cache\"):\n",
    "    \"\"\"\n",
    "    Load baseline model from disk cache.\n",
    "    \n",
    "    Parameters:\n",
    "    -----------\n",
    "    seed : int\n",
    "        Random seed (used for filename)\n",
    "    temp_dir : str\n",
    "        Temporary directory for cached baselines\n",
    "    \n",
    "    Returns:\n",
    "        Loaded model or None if not found\n",
    "    \"\"\"\n",
    "    cache_path = Path(temp_dir) / f\"baseline_seed_{seed}.h5\"\n",
    "    if cache_path.exists():\n",
    "        model = tf.keras.models.load_model(cache_path, compile=False)\n",
    "        print(f\"  ♻️  Loaded baseline from disk cache: {cache_path}\")\n",
    "        return model\n",
    "    return None\n",
    "\n",
    "\n",
    "def clear_baseline_cache(seed=None, temp_dir=\"temp_baseline_cache\"):\n",
    "    \"\"\"\n",
    "    Clear baseline model cache from disk.\n",
    "    \n",
    "    Parameters:\n",
    "    -----------\n",
    "    seed : int or None\n",
    "        Specific seed to clear, or None to clear all\n",
    "    temp_dir : str\n",
    "        Temporary directory for cached baselines\n",
    "    \"\"\"\n",
    "    cache_dir = Path(temp_dir)\n",
    "    if not cache_dir.exists():\n",
    "        return\n",
    "    \n",
    "    if seed is not None:\n",
    "        # Clear specific seed\n",
    "        cache_path = cache_dir / f\"baseline_seed_{seed}.h5\"\n",
    "        if cache_path.exists():\n",
    "            cache_path.unlink()\n",
    "            print(f\"  🗑️  Cleared baseline cache for seed {seed}\")\n",
    "    else:\n",
    "        # Clear all\n",
    "        import shutil\n",
    "        shutil.rmtree(cache_dir, ignore_errors=True)\n",
    "        print(f\"  🗑️  Cleared all baseline caches\")\n",
    "\n",
    "\n",
    "def train_single_model(seed, model_name, train_dataset, val_dataset, ood_dataset, kl_weight):\n",
    "    \"\"\"\n",
    "    Train a single model from scratch using correct module functions.\n",
    "    NO MORE IN-MEMORY CACHE - uses disk-based caching instead.\n",
    "    \n",
    "    Parameters:\n",
    "    -----------\n",
    "    seed : int\n",
    "        Random seed\n",
    "    model_name : str\n",
    "        Name of the model to train\n",
    "    train_dataset : tf.data.Dataset\n",
    "        Training dataset\n",
    "    val_dataset : tf.data.Dataset\n",
    "        Validation dataset\n",
    "    ood_dataset : tf.data.Dataset\n",
    "        OOD evaluation dataset\n",
    "    kl_weight : float\n",
    "        KL weight for Bayesian models\n",
    "    \n",
    "    Returns:\n",
    "        tuple: (model, training_time)\n",
    "    \"\"\"\n",
    "    # Clear memory and set seed\n",
    "    aggressive_memory_cleanup()\n",
    "    set_seed(seed)\n",
    "    \n",
    "    # Build and train based on model type\n",
    "    if model_name == \"Baseline_Transformer\":\n",
    "        # Create builder function\n",
    "        def baseline_builder():\n",
    "            return build_tiny_transformer_baseline(\n",
    "                vocab_size=VOCAB_SIZE,\n",
    "                max_length=MAX_LEN,\n",
    "                **MODEL_CONFIG\n",
    "            )\n",
    "        \n",
    "        model, history, training_time = train_baseline_model(\n",
    "            builder=baseline_builder,\n",
    "            train_dataset=train_dataset,\n",
    "            val_dataset=val_dataset,\n",
    "            ood_dataset=ood_dataset,\n",
    "            epochs=EPOCHS_BASELINE\n",
    "        )\n",
    "        \n",
    "        # Save to DISK instead of keeping in RAM\n",
    "        save_baseline_to_disk(model, seed)\n",
    "        \n",
    "        return model, training_time\n",
    "        \n",
    "    elif model_name == \"Full_Rank_BBB\":\n",
    "        def fullrank_builder():\n",
    "            return build_tiny_transformer_fullrank_bbb(\n",
    "                vocab_size=VOCAB_SIZE,\n",
    "                max_length=MAX_LEN,\n",
    "                **MODEL_CONFIG,\n",
    "                kl_scale=kl_weight\n",
    "            )\n",
    "        \n",
    "        model, history, training_time = train_bayesian_model(\n",
    "            builder=fullrank_builder,\n",
    "            kl_weight=kl_weight,\n",
    "            train_dataset=train_dataset,\n",
    "            val_dataset=val_dataset,\n",
    "            ood_dataset=ood_dataset,\n",
    "            epochs=EPOCHS_BAYESIAN,\n",
    "            name=\"Full-Rank BBB\",\n",
    "            use_kl_annealing=True,\n",
    "            kl_warmup_epochs=KL_WARMUP_EPOCHS,\n",
    "            kl_zero_epochs=KL_ZERO_EPOCHS\n",
    "        )\n",
    "        return model, training_time\n",
    "        \n",
    "    elif model_name == \"Low_Rank_Random_Init\":\n",
    "        def lowrank_random_builder():\n",
    "            return build_tiny_transformer_lowrank_bbb(\n",
    "                vocab_size=VOCAB_SIZE,\n",
    "                max_length=MAX_LEN,\n",
    "                **MODEL_CONFIG,\n",
    "                rank=16,  # Explicit to match Refactored notebook\n",
    "                rank_emb=10,  # Explicit to match Refactored notebook\n",
    "                kl_scale=kl_weight,\n",
    "                init_from_deterministic=None  # Random init\n",
    "            )\n",
    "        \n",
    "        model, history, training_time = train_bayesian_model(\n",
    "            builder=lowrank_random_builder,\n",
    "            kl_weight=kl_weight,\n",
    "            train_dataset=train_dataset,\n",
    "            val_dataset=val_dataset,\n",
    "            ood_dataset=ood_dataset,\n",
    "            epochs=EPOCHS_BAYESIAN,\n",
    "            name=\"Low-Rank Random Init\",\n",
    "            use_kl_annealing=True,\n",
    "            kl_warmup_epochs=KL_WARMUP_EPOCHS,\n",
    "            kl_zero_epochs=KL_ZERO_EPOCHS\n",
    "        )\n",
    "        return model, training_time\n",
    "        \n",
    "    elif model_name == \"Low_Rank_SVD_Init\":\n",
    "        # Try to load baseline from DISK cache\n",
    "        baseline_model = load_baseline_from_disk(seed)\n",
    "        \n",
    "        if baseline_model is None:\n",
    "            # Fallback: train new baseline\n",
    "            print(\"  ⚠️  No cached baseline found on disk, training from scratch...\")\n",
    "            print(\"     (This shouldn't happen if Baseline_Transformer runs before SVD init)\")\n",
    "            def baseline_for_svd():\n",
    "                return build_tiny_transformer_baseline(\n",
    "                    vocab_size=VOCAB_SIZE,\n",
    "                    max_length=MAX_LEN,\n",
    "                    **MODEL_CONFIG\n",
    "                )\n",
    "            \n",
    "            baseline_model, _, _ = train_baseline_model(\n",
    "                builder=baseline_for_svd,\n",
    "                train_dataset=train_dataset,\n",
    "                val_dataset=val_dataset,\n",
    "                ood_dataset=ood_dataset,\n",
    "                epochs=EPOCHS_BASELINE\n",
    "            )\n",
    "            \n",
    "            # Save to disk\n",
    "            save_baseline_to_disk(baseline_model, seed)\n",
    "        \n",
    "        # Build low-rank with SVD init from baseline\n",
    "        def lowrank_svd_builder():\n",
    "            return build_tiny_transformer_lowrank_bbb(\n",
    "                vocab_size=VOCAB_SIZE,\n",
    "                max_length=MAX_LEN,\n",
    "                **MODEL_CONFIG,\n",
    "                rank=16,  # Explicit to match Refactored notebook\n",
    "                rank_emb=10,  # Explicit to match Refactored notebook\n",
    "                kl_scale=kl_weight,\n",
    "                init_from_deterministic=baseline_model  # Uses loaded baseline!\n",
    "            )\n",
    "        \n",
    "        model, history, training_time = train_bayesian_model(\n",
    "            builder=lowrank_svd_builder,\n",
    "            kl_weight=kl_weight,\n",
    "            train_dataset=train_dataset,\n",
    "            val_dataset=val_dataset,\n",
    "            ood_dataset=ood_dataset,\n",
    "            epochs=EPOCHS_BAYESIAN,\n",
    "            name=\"Low-Rank SVD Init\",\n",
    "            use_kl_annealing=True,\n",
    "            kl_warmup_epochs=KL_WARMUP_EPOCHS,\n",
    "            kl_zero_epochs=KL_ZERO_EPOCHS\n",
    "        )\n",
    "        \n",
    "        # CRITICAL: Delete baseline from RAM immediately after use\n",
    "        del baseline_model\n",
    "        aggressive_memory_cleanup()\n",
    "        \n",
    "        return model, training_time\n",
    "        \n",
    "    elif model_name == \"Rank1_BBB\":\n",
    "        def rank1_builder():\n",
    "            return build_tiny_transformer_rank1_bbb(\n",
    "                vocab_size=VOCAB_SIZE,\n",
    "                max_length=MAX_LEN,\n",
    "                **MODEL_CONFIG,\n",
    "                kl_scale=kl_weight\n",
    "            )\n",
    "        \n",
    "        model, history, training_time = train_bayesian_model(\n",
    "            builder=rank1_builder,\n",
    "            kl_weight=kl_weight,\n",
    "            train_dataset=train_dataset,\n",
    "            val_dataset=val_dataset,\n",
    "            ood_dataset=ood_dataset,\n",
    "            epochs=EPOCHS_BAYESIAN,\n",
    "            name=\"Rank-1 BBB\",\n",
    "            use_kl_annealing=True,\n",
    "            kl_warmup_epochs=KL_WARMUP_EPOCHS,\n",
    "            kl_zero_epochs=KL_ZERO_EPOCHS\n",
    "        )\n",
    "        return model, training_time\n",
    "        \n",
    "    elif model_name == \"Deep_Ensemble\":\n",
    "        # Create builder function for ensemble members\n",
    "        def ensemble_member_builder():\n",
    "            return build_tiny_transformer_baseline(\n",
    "                vocab_size=VOCAB_SIZE,\n",
    "                max_length=MAX_LEN,\n",
    "                **MODEL_CONFIG\n",
    "            )\n",
    "        \n",
    "        # Create and train deep ensemble\n",
    "        ensemble = DeepEnsemble(\n",
    "            builder_fn=ensemble_member_builder,\n",
    "            n_members=N_ENSEMBLE_MEMBERS\n",
    "        )\n",
    "        \n",
    "        start_time = time.time()\n",
    "        ensemble.train(\n",
    "            train_dataset=train_dataset,\n",
    "            val_dataset=val_dataset,\n",
    "            epochs= 20, #EPOCHS_ENSEMBLE,\n",
    "            verbose=1\n",
    "        )\n",
    "        training_time = time.time() - start_time\n",
    "        \n",
    "        return ensemble, training_time\n",
    "    \n",
    "    raise ValueError(f\"Unknown model: {model_name}\")\n",
    "\n",
    "\n",
    "def run_single_experiment(seed, model_name, train_dataset, val_dataset, ood_dataset, \n",
    "                          X_test_dict, y_test_arr, X_ood_dict, y_ood_arr, kl_weight):\n",
    "    \"\"\"\n",
    "    Run complete experiment: train or load model, then evaluate.\n",
    "    REMOVED baseline_cache parameter - now uses disk-based caching.\n",
    "    \n",
    "    Parameters:\n",
    "    -----------\n",
    "    seed : int\n",
    "        Random seed for this experiment\n",
    "    model_name : str\n",
    "        Name of the model to train/evaluate\n",
    "    train_dataset, val_dataset, ood_dataset : tf.data.Dataset\n",
    "        Training, validation, and OOD datasets\n",
    "    X_test_dict, y_test_arr, X_ood_dict, y_ood_arr : dict/array\n",
    "        Evaluation data\n",
    "    kl_weight : float\n",
    "        KL weight for Bayesian models\n",
    "    \n",
    "    Returns:\n",
    "        tuple: (metrics, training_time, reused)\n",
    "    \"\"\"\n",
    "    print(f\"\\n{'='*80}\")\n",
    "    print(f\"Experiment: {model_name} | Seed: {seed}\")\n",
    "    print(f\"{'='*80}\\n\")\n",
    "    \n",
    "    # Try to load existing model (only for seed=42)\n",
    "    model = load_existing_model(model_name, seed)\n",
    "    reused = (model is not None)\n",
    "    \n",
    "    if not reused:\n",
    "        # Train from scratch (uses disk-based baseline caching internally)\n",
    "        print(f\"Training {model_name} from scratch...\")\n",
    "        model, training_time = train_single_model(\n",
    "            seed=seed,\n",
    "            model_name=model_name,\n",
    "            train_dataset=train_dataset,\n",
    "            val_dataset=val_dataset,\n",
    "            ood_dataset=ood_dataset,\n",
    "            kl_weight=kl_weight\n",
    "        )\n",
    "        print(f\"\\n⏱️  Training completed in {training_time:.1f}s ({training_time/3600:.2f}h)\")\n",
    "    else:\n",
    "        # Using existing model\n",
    "        print(f\"Using existing model (no training needed)\")\n",
    "        training_time = 0.0\n",
    "        tracker.mark_reused(seed, model_name)\n",
    "    \n",
    "    # Evaluate using unified evaluation function\n",
    "    print(f\"\\nEvaluating {model_name}...\")\n",
    "    eval_start = time.time()\n",
    "    \n",
    "    # Use the proper evaluation function from evaluation.py\n",
    "    metrics = evaluate_all_metrics_unified(\n",
    "        model=model,\n",
    "        X_test=X_test_dict,\n",
    "        y_test=y_test_arr,\n",
    "        X_ood=X_ood_dict,\n",
    "        y_ood=y_ood_arr,\n",
    "        model_name=model_name,\n",
    "        n_samples=N_MC_SAMPLES\n",
    "    )\n",
    "    \n",
    "    eval_time = time.time() - eval_start\n",
    "    print(f\"✓ Evaluation completed in {eval_time:.1f}s\")\n",
    "    \n",
    "    # Clean up model from memory\n",
    "    del model\n",
    "    aggressive_memory_cleanup()\n",
    "    \n",
    "    return metrics, training_time, reused\n",
    "\n",
    "print(\"✓ Training and evaluation functions defined\")\n",
    "print(\"✓ Now using DISK-BASED baseline caching (no more RAM hoarding!)\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Main Experiment Loop\n",
    "\n",
    "Run all experiments with checkpoint/resume capability."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✓ Baseline cache initialized\n",
      "  This cache will store trained baseline models per seed\n",
      "  Low_Rank_SVD_Init will reuse the cached baseline instead of retraining\n"
     ]
    }
   ],
   "source": [
    "# Initialize baseline cache\n",
    "# This cache stores the trained baseline model for each seed to avoid retraining for SVD init\n",
    "\n",
    "baseline_cache = {}  # {seed: baseline_model}\n",
    "\n",
    "print(\"✓ Baseline cache initialized\")\n",
    "print(\"  This cache will store trained baseline models per seed\")\n",
    "print(\"  Low_Rank_SVD_Init will reuse the cached baseline instead of retraining\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "================================================================================\n",
      "Multi-Seed Robustness Check\n",
      "================================================================================\n",
      "Seeds: [42, 123, 456, 2026]\n",
      "Models: 6\n",
      "Total experiments: 24\n",
      "Already completed: 20\n",
      "Remaining: 4\n",
      "================================================================================\n",
      "\n",
      "\n",
      "################################################################################\n",
      "# SEED: 42\n",
      "################################################################################\n",
      "\n",
      "⏭️  [21/24] Skipping Baseline_Transformer (seed=42) - already completed\n",
      "⏭️  [22/24] Skipping Full_Rank_BBB (seed=42) - already completed\n",
      "⏭️  [23/24] Skipping Low_Rank_Random_Init (seed=42) - already completed\n",
      "⏭️  [24/24] Skipping Low_Rank_SVD_Init (seed=42) - already completed\n",
      "⏭️  [25/24] Skipping Rank1_BBB (seed=42) - already completed\n",
      "\n",
      "🚀 [26/24] Starting: Deep_Ensemble (seed=42)\n",
      "\n",
      "================================================================================\n",
      "Experiment: Deep_Ensemble | Seed: 42\n",
      "================================================================================\n",
      "\n",
      "  ⚠️  Failed to load model: No model config found in the file at <tensorflow.python.platform.gfile.GFile object at 0x14bb34908b80>.\n",
      "Training Deep_Ensemble from scratch...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Traceback (most recent call last):\n",
      "  File \"/tmp/ipykernel_85410/3290469380.py\", line 28, in load_existing_model\n",
      "    model = tf.keras.models.load_model(member_path, compile=False)\n",
      "  File \"/home/math/mame.toure@MAIL.MCGILL.CA/.conda/envs/bnn-gpu115/lib/python3.10/site-packages/keras/src/saving/saving_api.py\", line 262, in load_model\n",
      "    return legacy_sm_saving_lib.load_model(\n",
      "  File \"/home/math/mame.toure@MAIL.MCGILL.CA/.conda/envs/bnn-gpu115/lib/python3.10/site-packages/keras/src/utils/traceback_utils.py\", line 70, in error_handler\n",
      "    raise e.with_traceback(filtered_tb) from None\n",
      "  File \"/home/math/mame.toure@MAIL.MCGILL.CA/.conda/envs/bnn-gpu115/lib/python3.10/site-packages/keras/src/saving/legacy/hdf5_format.py\", line 197, in load_model_from_hdf5\n",
      "    raise ValueError(\n",
      "ValueError: No model config found in the file at <tensorflow.python.platform.gfile.GFile object at 0x14bb34908b80>.\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "🧹 Memory cleaned\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 1/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 91s 81ms/step - loss: 0.4011 - acc: 0.8236 - val_loss: 0.4738 - val_acc: 0.7837 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 65s 61ms/step - loss: 0.1770 - acc: 0.9332 - val_loss: 0.5397 - val_acc: 0.7885 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0981 - acc: 0.9651 - val_loss: 0.6225 - val_acc: 0.8041 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0578 - acc: 0.9806 - val_loss: 0.6836 - val_acc: 0.8005 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 65s 62ms/step - loss: 0.0395 - acc: 0.9865 - val_loss: 0.8233 - val_acc: 0.8029 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0285 - acc: 0.9907\n",
      "Epoch 6: ReduceLROnPlateau reducing learning rate to 7.96013482613489e-05.\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0285 - acc: 0.9907 - val_loss: 1.0013 - val_acc: 0.7993 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0212 - acc: 0.9928 - val_loss: 1.0232 - val_acc: 0.7897 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - 62s 59ms/step - loss: 0.0166 - acc: 0.9944 - val_loss: 1.0537 - val_acc: 0.8065 - lr: 1.3161e-04\n",
      "Epoch 9/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0111 - acc: 0.9959 - val_loss: 1.3977 - val_acc: 0.8005 - lr: 1.1650e-04\n",
      "Epoch 10/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0087 - acc: 0.9969 - val_loss: 1.3488 - val_acc: 0.8077 - lr: 1.0101e-04\n",
      "Epoch 11/20\n",
      "1052/1052 [==============================] - 64s 60ms/step - loss: 0.0051 - acc: 0.9981 - val_loss: 1.6334 - val_acc: 0.7885 - lr: 8.5528e-05\n",
      "Epoch 12/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0030 - acc: 0.9988 - val_loss: 1.7304 - val_acc: 0.7969 - lr: 7.0421e-05\n",
      "Epoch 13/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0020 - acc: 0.9992\n",
      "Epoch 13: ReduceLROnPlateau reducing learning rate to 2.8034057322656736e-05.\n",
      "1052/1052 [==============================] - 61s 58ms/step - loss: 0.0020 - acc: 0.9992 - val_loss: 1.8481 - val_acc: 0.8005 - lr: 5.6068e-05\n",
      "Epoch 14/20\n",
      "1052/1052 [==============================] - 61s 58ms/step - loss: 0.0018 - acc: 0.9993 - val_loss: 1.8827 - val_acc: 0.8017 - lr: 4.2821e-05\n",
      "Epoch 15/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 9.1490e-04 - acc: 0.9996Restoring model weights from the end of the best epoch: 10.\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 9.1490e-04 - acc: 0.9996 - val_loss: 2.0927 - val_acc: 0.8077 - lr: 3.1007e-05\n",
      "Epoch 15: early stopping\n",
      "Member 1 Final Val Accuracy: 0.8077\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 2/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 90s 80ms/step - loss: 0.4022 - acc: 0.8249 - val_loss: 0.4923 - val_acc: 0.7909 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 65s 61ms/step - loss: 0.1828 - acc: 0.9303 - val_loss: 0.5341 - val_acc: 0.7957 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.1047 - acc: 0.9613 - val_loss: 0.6896 - val_acc: 0.7921 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0578 - acc: 0.9807 - val_loss: 0.9064 - val_acc: 0.8065 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0406 - acc: 0.9865 - val_loss: 0.9445 - val_acc: 0.7921 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - 62s 59ms/step - loss: 0.0299 - acc: 0.9899 - val_loss: 0.7957 - val_acc: 0.7921 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0210 - acc: 0.9927\n",
      "Epoch 7: ReduceLROnPlateau reducing learning rate to 7.29791063349694e-05.\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0210 - acc: 0.9927 - val_loss: 1.1293 - val_acc: 0.7885 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0150 - acc: 0.9951 - val_loss: 1.2488 - val_acc: 0.8053 - lr: 1.3161e-04\n",
      "Epoch 9/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0130 - acc: 0.9956Restoring model weights from the end of the best epoch: 4.\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0130 - acc: 0.9956 - val_loss: 1.1970 - val_acc: 0.7933 - lr: 1.1650e-04\n",
      "Epoch 9: early stopping\n",
      "Member 2 Final Val Accuracy: 0.7933\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 3/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 89s 79ms/step - loss: 0.4086 - acc: 0.8250 - val_loss: 0.5599 - val_acc: 0.7993 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.1833 - acc: 0.9309 - val_loss: 0.5036 - val_acc: 0.7993 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.1033 - acc: 0.9635 - val_loss: 0.5687 - val_acc: 0.7981 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0597 - acc: 0.9795 - val_loss: 0.6915 - val_acc: 0.8089 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0397 - acc: 0.9868 - val_loss: 0.7688 - val_acc: 0.7957 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - 65s 61ms/step - loss: 0.0314 - acc: 0.9895 - val_loss: 1.0520 - val_acc: 0.8029 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0223 - acc: 0.9924\n",
      "Epoch 7: ReduceLROnPlateau reducing learning rate to 7.29791063349694e-05.\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0223 - acc: 0.9924 - val_loss: 0.9839 - val_acc: 0.8029 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0170 - acc: 0.9945 - val_loss: 1.0670 - val_acc: 0.8041 - lr: 1.3161e-04\n",
      "Epoch 9/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0126 - acc: 0.9960Restoring model weights from the end of the best epoch: 4.\n",
      "1052/1052 [==============================] - 63s 59ms/step - loss: 0.0126 - acc: 0.9960 - val_loss: 1.0812 - val_acc: 0.8053 - lr: 1.1650e-04\n",
      "Epoch 9: early stopping\n",
      "Member 3 Final Val Accuracy: 0.8053\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 4/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 90s 80ms/step - loss: 0.4296 - acc: 0.8182 - val_loss: 0.4626 - val_acc: 0.7969 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 85s 81ms/step - loss: 0.1783 - acc: 0.9324 - val_loss: 0.5165 - val_acc: 0.8029 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.1051 - acc: 0.9611 - val_loss: 0.6701 - val_acc: 0.8065 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0647 - acc: 0.9772 - val_loss: 0.7954 - val_acc: 0.8017 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0447 - acc: 0.9850 - val_loss: 0.8840 - val_acc: 0.7728 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0331 - acc: 0.9895\n",
      "Epoch 6: ReduceLROnPlateau reducing learning rate to 7.96013482613489e-05.\n",
      "1052/1052 [==============================] - 89s 84ms/step - loss: 0.0331 - acc: 0.9895 - val_loss: 1.0392 - val_acc: 0.7849 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - 86s 82ms/step - loss: 0.0251 - acc: 0.9918 - val_loss: 1.0021 - val_acc: 0.7981 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0178 - acc: 0.9945Restoring model weights from the end of the best epoch: 3.\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0178 - acc: 0.9945 - val_loss: 1.1492 - val_acc: 0.7776 - lr: 1.3161e-04\n",
      "Epoch 8: early stopping\n",
      "Member 4 Final Val Accuracy: 0.7776\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 5/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 115s 104ms/step - loss: 0.4076 - acc: 0.8238 - val_loss: 0.4710 - val_acc: 0.7969 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 89s 84ms/step - loss: 0.1859 - acc: 0.9292 - val_loss: 0.5164 - val_acc: 0.8089 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.1021 - acc: 0.9624 - val_loss: 0.6361 - val_acc: 0.8017 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0618 - acc: 0.9781 - val_loss: 0.7041 - val_acc: 0.8065 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0408 - acc: 0.9869\n",
      "Epoch 5: ReduceLROnPlateau reducing learning rate to 8.550701750209555e-05.\n",
      "1052/1052 [==============================] - 87s 82ms/step - loss: 0.0408 - acc: 0.9869 - val_loss: 0.8394 - val_acc: 0.7909 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - 89s 84ms/step - loss: 0.0298 - acc: 0.9903 - val_loss: 1.0649 - val_acc: 0.7933 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0225 - acc: 0.9923Restoring model weights from the end of the best epoch: 2.\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0225 - acc: 0.9923 - val_loss: 1.1803 - val_acc: 0.7885 - lr: 1.4596e-04\n",
      "Epoch 7: early stopping\n",
      "Member 5 Final Val Accuracy: 0.7885\n",
      "\n",
      "============================================================\n",
      "Deep Ensemble Training Complete!\n",
      "============================================================\n",
      "\n",
      "⏱️  Training completed in 3524.0s (0.98h)\n",
      "\n",
      "Evaluating Deep_Ensemble...\n",
      "\n",
      "================================================================================\n",
      "Evaluating: Deep_Ensemble\n",
      "================================================================================\n",
      "Computing in-domain predictions (ensemble)...\n",
      "Computing OOD predictions (ensemble)...\n",
      "Computing metrics...\n",
      "✓ Evaluation completed in 27.7s\n",
      "🧹 Memory cleaned\n",
      "✅ Trained: 42_Deep_Ensemble (3524.0s)\n",
      "\n",
      "📊 Progress: 21/24 (87.5%)\n",
      "   Remaining: 3 experiments\n",
      "\n",
      "################################################################################\n",
      "# SEED: 123\n",
      "################################################################################\n",
      "\n",
      "⏭️  [27/24] Skipping Baseline_Transformer (seed=123) - already completed\n",
      "⏭️  [28/24] Skipping Full_Rank_BBB (seed=123) - already completed\n",
      "⏭️  [29/24] Skipping Low_Rank_Random_Init (seed=123) - already completed\n",
      "⏭️  [30/24] Skipping Low_Rank_SVD_Init (seed=123) - already completed\n",
      "⏭️  [31/24] Skipping Rank1_BBB (seed=123) - already completed\n",
      "\n",
      "🚀 [32/24] Starting: Deep_Ensemble (seed=123)\n",
      "\n",
      "================================================================================\n",
      "Experiment: Deep_Ensemble | Seed: 123\n",
      "================================================================================\n",
      "\n",
      "Training Deep_Ensemble from scratch...\n",
      "🧹 Memory cleaned\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 1/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 117s 105ms/step - loss: 0.3877 - acc: 0.8302 - val_loss: 0.4782 - val_acc: 0.7897 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 91s 86ms/step - loss: 0.1829 - acc: 0.9307 - val_loss: 0.6264 - val_acc: 0.7849 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.1031 - acc: 0.9620 - val_loss: 0.7595 - val_acc: 0.7752 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0599 - acc: 0.9791\n",
      "Epoch 4: ReduceLROnPlateau reducing learning rate to 9.055068221641704e-05.\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0599 - acc: 0.9791 - val_loss: 0.8208 - val_acc: 0.7885 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 89s 85ms/step - loss: 0.0402 - acc: 0.9862 - val_loss: 0.8652 - val_acc: 0.7981 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0280 - acc: 0.9910 - val_loss: 0.9326 - val_acc: 0.7885 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0225 - acc: 0.9924 - val_loss: 1.0967 - val_acc: 0.8029 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0165 - acc: 0.9945 - val_loss: 1.1811 - val_acc: 0.7800 - lr: 1.3161e-04\n",
      "Epoch 9/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0110 - acc: 0.9963 - val_loss: 1.0830 - val_acc: 0.7885 - lr: 1.1650e-04\n",
      "Epoch 10/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0088 - acc: 0.9972\n",
      "Epoch 10: ReduceLROnPlateau reducing learning rate to 5.0507394917076454e-05.\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0088 - acc: 0.9972 - val_loss: 1.2667 - val_acc: 0.7921 - lr: 1.0101e-04\n",
      "Epoch 11/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0046 - acc: 0.9983 - val_loss: 1.7210 - val_acc: 0.7981 - lr: 8.5528e-05\n",
      "Epoch 12/20\n",
      "1052/1052 [==============================] - 83s 79ms/step - loss: 0.0046 - acc: 0.9984 - val_loss: 1.5841 - val_acc: 0.8065 - lr: 7.0421e-05\n",
      "Epoch 13/20\n",
      "1052/1052 [==============================] - 68s 64ms/step - loss: 0.0020 - acc: 0.9991 - val_loss: 1.6372 - val_acc: 0.7957 - lr: 5.6068e-05\n",
      "Epoch 14/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0010 - acc: 0.9995 - val_loss: 1.9201 - val_acc: 0.7909 - lr: 4.2821e-05\n",
      "Epoch 15/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 9.2464e-04 - acc: 0.9997\n",
      "Epoch 15: ReduceLROnPlateau reducing learning rate to 1.5503437680308707e-05.\n",
      "1052/1052 [==============================] - 62s 59ms/step - loss: 9.2464e-04 - acc: 0.9997 - val_loss: 1.7714 - val_acc: 0.7969 - lr: 3.1007e-05\n",
      "Epoch 16/20\n",
      "1052/1052 [==============================] - 61s 58ms/step - loss: 4.1588e-04 - acc: 0.9998 - val_loss: 1.8533 - val_acc: 0.7933 - lr: 2.0916e-05\n",
      "Epoch 17/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 3.6552e-04 - acc: 0.9998Restoring model weights from the end of the best epoch: 12.\n",
      "1052/1052 [==============================] - 61s 58ms/step - loss: 3.6552e-04 - acc: 0.9998 - val_loss: 2.0540 - val_acc: 0.7885 - lr: 1.2797e-05\n",
      "Epoch 17: early stopping\n",
      "Member 1 Final Val Accuracy: 0.7885\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 2/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 91s 81ms/step - loss: 0.4196 - acc: 0.8191 - val_loss: 0.5906 - val_acc: 0.7945 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.1829 - acc: 0.9305 - val_loss: 0.6156 - val_acc: 0.7800 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 65s 62ms/step - loss: 0.1027 - acc: 0.9629 - val_loss: 0.6277 - val_acc: 0.7644 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0607 - acc: 0.9795 - val_loss: 0.6889 - val_acc: 0.8137 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0415 - acc: 0.9863 - val_loss: 0.7818 - val_acc: 0.7909 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0294 - acc: 0.9909 - val_loss: 1.1151 - val_acc: 0.7728 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0245 - acc: 0.9913\n",
      "Epoch 7: ReduceLROnPlateau reducing learning rate to 7.29791063349694e-05.\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0245 - acc: 0.9913 - val_loss: 1.1201 - val_acc: 0.7897 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - 62s 59ms/step - loss: 0.0182 - acc: 0.9941 - val_loss: 0.9935 - val_acc: 0.7921 - lr: 1.3161e-04\n",
      "Epoch 9/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0118 - acc: 0.9959Restoring model weights from the end of the best epoch: 4.\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0118 - acc: 0.9959 - val_loss: 0.9507 - val_acc: 0.7945 - lr: 1.1650e-04\n",
      "Epoch 9: early stopping\n",
      "Member 2 Final Val Accuracy: 0.7945\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 3/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 92s 82ms/step - loss: 0.4241 - acc: 0.8196 - val_loss: 0.5079 - val_acc: 0.7873 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 66s 63ms/step - loss: 0.1782 - acc: 0.9327 - val_loss: 0.6467 - val_acc: 0.7800 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 62s 59ms/step - loss: 0.1021 - acc: 0.9633 - val_loss: 0.6417 - val_acc: 0.8005 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 61s 58ms/step - loss: 0.0623 - acc: 0.9785 - val_loss: 0.8278 - val_acc: 0.7873 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0426 - acc: 0.9858 - val_loss: 0.8353 - val_acc: 0.7933 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0294 - acc: 0.9903\n",
      "Epoch 6: ReduceLROnPlateau reducing learning rate to 7.96013482613489e-05.\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0294 - acc: 0.9903 - val_loss: 0.8518 - val_acc: 0.7909 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0232 - acc: 0.9926 - val_loss: 0.8163 - val_acc: 0.7945 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0176 - acc: 0.9942Restoring model weights from the end of the best epoch: 3.\n",
      "1052/1052 [==============================] - 63s 59ms/step - loss: 0.0176 - acc: 0.9942 - val_loss: 1.2522 - val_acc: 0.7885 - lr: 1.3161e-04\n",
      "Epoch 8: early stopping\n",
      "Member 3 Final Val Accuracy: 0.7885\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 4/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 90s 80ms/step - loss: 0.4188 - acc: 0.8191 - val_loss: 0.4687 - val_acc: 0.8101 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 65s 62ms/step - loss: 0.1814 - acc: 0.9318 - val_loss: 0.4822 - val_acc: 0.8041 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.1034 - acc: 0.9629 - val_loss: 0.6232 - val_acc: 0.7692 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0624 - acc: 0.9788\n",
      "Epoch 4: ReduceLROnPlateau reducing learning rate to 9.055068221641704e-05.\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0624 - acc: 0.9788 - val_loss: 0.7517 - val_acc: 0.8017 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0423 - acc: 0.9860 - val_loss: 0.8163 - val_acc: 0.7704 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0315 - acc: 0.9896Restoring model weights from the end of the best epoch: 1.\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0315 - acc: 0.9896 - val_loss: 0.9541 - val_acc: 0.7909 - lr: 1.5920e-04\n",
      "Epoch 6: early stopping\n",
      "Member 4 Final Val Accuracy: 0.7909\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 5/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 89s 80ms/step - loss: 0.4185 - acc: 0.8189 - val_loss: 0.4826 - val_acc: 0.7812 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.1778 - acc: 0.9322 - val_loss: 0.5381 - val_acc: 0.8149 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 65s 61ms/step - loss: 0.0990 - acc: 0.9652 - val_loss: 0.6968 - val_acc: 0.7837 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0629 - acc: 0.9788 - val_loss: 0.6514 - val_acc: 0.7885 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0406 - acc: 0.9871\n",
      "Epoch 5: ReduceLROnPlateau reducing learning rate to 8.550701750209555e-05.\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0406 - acc: 0.9871 - val_loss: 0.9151 - val_acc: 0.8041 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0291 - acc: 0.9906 - val_loss: 0.8818 - val_acc: 0.8113 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0222 - acc: 0.9928 - val_loss: 1.0158 - val_acc: 0.8173 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - 64s 60ms/step - loss: 0.0169 - acc: 0.9947 - val_loss: 1.0621 - val_acc: 0.8017 - lr: 1.3161e-04\n",
      "Epoch 9/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0127 - acc: 0.9957 - val_loss: 1.5166 - val_acc: 0.7873 - lr: 1.1650e-04\n",
      "Epoch 10/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0086 - acc: 0.9973\n",
      "Epoch 10: ReduceLROnPlateau reducing learning rate to 5.0507394917076454e-05.\n",
      "1052/1052 [==============================] - 61s 58ms/step - loss: 0.0086 - acc: 0.9973 - val_loss: 1.2494 - val_acc: 0.7957 - lr: 1.0101e-04\n",
      "Epoch 11/20\n",
      "1052/1052 [==============================] - 61s 58ms/step - loss: 0.0044 - acc: 0.9984 - val_loss: 1.7834 - val_acc: 0.8125 - lr: 8.5528e-05\n",
      "Epoch 12/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0036 - acc: 0.9988Restoring model weights from the end of the best epoch: 7.\n",
      "1052/1052 [==============================] - 62s 59ms/step - loss: 0.0036 - acc: 0.9988 - val_loss: 1.7694 - val_acc: 0.8029 - lr: 7.0421e-05\n",
      "Epoch 12: early stopping\n",
      "Member 5 Final Val Accuracy: 0.8029\n",
      "\n",
      "============================================================\n",
      "Deep Ensemble Training Complete!\n",
      "============================================================\n",
      "\n",
      "⏱️  Training completed in 3734.3s (1.04h)\n",
      "\n",
      "Evaluating Deep_Ensemble...\n",
      "\n",
      "================================================================================\n",
      "Evaluating: Deep_Ensemble\n",
      "================================================================================\n",
      "Computing in-domain predictions (ensemble)...\n",
      "Computing OOD predictions (ensemble)...\n",
      "Computing metrics...\n",
      "✓ Evaluation completed in 15.1s\n",
      "🧹 Memory cleaned\n",
      "✅ Trained: 123_Deep_Ensemble (3734.3s)\n",
      "\n",
      "📊 Progress: 22/24 (91.7%)\n",
      "   Remaining: 2 experiments\n",
      "\n",
      "################################################################################\n",
      "# SEED: 456\n",
      "################################################################################\n",
      "\n",
      "⏭️  [33/24] Skipping Baseline_Transformer (seed=456) - already completed\n",
      "⏭️  [34/24] Skipping Full_Rank_BBB (seed=456) - already completed\n",
      "⏭️  [35/24] Skipping Low_Rank_Random_Init (seed=456) - already completed\n",
      "⏭️  [36/24] Skipping Low_Rank_SVD_Init (seed=456) - already completed\n",
      "⏭️  [37/24] Skipping Rank1_BBB (seed=456) - already completed\n",
      "\n",
      "🚀 [38/24] Starting: Deep_Ensemble (seed=456)\n",
      "\n",
      "================================================================================\n",
      "Experiment: Deep_Ensemble | Seed: 456\n",
      "================================================================================\n",
      "\n",
      "Training Deep_Ensemble from scratch...\n",
      "🧹 Memory cleaned\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 1/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 90s 80ms/step - loss: 0.4144 - acc: 0.8217 - val_loss: 0.5006 - val_acc: 0.7969 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.1807 - acc: 0.9314 - val_loss: 0.5795 - val_acc: 0.7933 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 65s 61ms/step - loss: 0.0961 - acc: 0.9653 - val_loss: 0.6683 - val_acc: 0.8029 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 63s 59ms/step - loss: 0.0598 - acc: 0.9787 - val_loss: 0.7808 - val_acc: 0.7921 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 62s 59ms/step - loss: 0.0384 - acc: 0.9878 - val_loss: 0.9285 - val_acc: 0.8041 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0299 - acc: 0.9901 - val_loss: 0.9323 - val_acc: 0.7957 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0226 - acc: 0.9925 - val_loss: 0.9551 - val_acc: 0.7861 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - 62s 59ms/step - loss: 0.0161 - acc: 0.9944 - val_loss: 1.0446 - val_acc: 0.8053 - lr: 1.3161e-04\n",
      "Epoch 9/20\n",
      "1052/1052 [==============================] - 61s 58ms/step - loss: 0.0108 - acc: 0.9965 - val_loss: 1.2727 - val_acc: 0.8053 - lr: 1.1650e-04\n",
      "Epoch 10/20\n",
      "1052/1052 [==============================] - 61s 58ms/step - loss: 0.0083 - acc: 0.9973 - val_loss: 1.3583 - val_acc: 0.7957 - lr: 1.0101e-04\n",
      "Epoch 11/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0051 - acc: 0.9982\n",
      "Epoch 11: ReduceLROnPlateau reducing learning rate to 4.276379695511423e-05.\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0051 - acc: 0.9982 - val_loss: 1.1948 - val_acc: 0.8053 - lr: 8.5528e-05\n",
      "Epoch 12/20\n",
      "1052/1052 [==============================] - 62s 59ms/step - loss: 0.0034 - acc: 0.9990 - val_loss: 1.5113 - val_acc: 0.7957 - lr: 7.0421e-05\n",
      "Epoch 13/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0013 - acc: 0.9996Restoring model weights from the end of the best epoch: 8.\n",
      "1052/1052 [==============================] - 62s 59ms/step - loss: 0.0013 - acc: 0.9996 - val_loss: 1.8296 - val_acc: 0.7981 - lr: 5.6068e-05\n",
      "Epoch 13: early stopping\n",
      "Member 1 Final Val Accuracy: 0.7981\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 2/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 91s 80ms/step - loss: 0.4197 - acc: 0.8211 - val_loss: 0.5455 - val_acc: 0.7716 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.1807 - acc: 0.9310 - val_loss: 0.5580 - val_acc: 0.8041 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.1059 - acc: 0.9617 - val_loss: 0.6995 - val_acc: 0.7945 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 64s 61ms/step - loss: 0.0638 - acc: 0.9785 - val_loss: 0.8529 - val_acc: 0.8065 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 64s 60ms/step - loss: 0.0424 - acc: 0.9862 - val_loss: 0.7086 - val_acc: 0.8029 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - 63s 60ms/step - loss: 0.0315 - acc: 0.9893 - val_loss: 0.7530 - val_acc: 0.7981 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0238 - acc: 0.9923\n",
      "Epoch 7: ReduceLROnPlateau reducing learning rate to 7.29791063349694e-05.\n",
      "1052/1052 [==============================] - 61s 58ms/step - loss: 0.0238 - acc: 0.9923 - val_loss: 0.8284 - val_acc: 0.7981 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - 61s 58ms/step - loss: 0.0167 - acc: 0.9944 - val_loss: 0.9633 - val_acc: 0.7825 - lr: 1.3161e-04\n",
      "Epoch 9/20\n",
      "1052/1052 [==============================] - 61s 58ms/step - loss: 0.0136 - acc: 0.9955 - val_loss: 1.1780 - val_acc: 0.8113 - lr: 1.1650e-04\n",
      "Epoch 10/20\n",
      "1052/1052 [==============================] - 62s 59ms/step - loss: 0.0094 - acc: 0.9971 - val_loss: 1.0501 - val_acc: 0.7837 - lr: 1.0101e-04\n",
      "Epoch 11/20\n",
      "1052/1052 [==============================] - 71s 67ms/step - loss: 0.0051 - acc: 0.9984 - val_loss: 1.3881 - val_acc: 0.8125 - lr: 8.5528e-05\n",
      "Epoch 12/20\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0033 - acc: 0.9990 - val_loss: 1.3782 - val_acc: 0.7993 - lr: 7.0421e-05\n",
      "Epoch 13/20\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0028 - acc: 0.9990 - val_loss: 1.5570 - val_acc: 0.8005 - lr: 5.6068e-05\n",
      "Epoch 14/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0017 - acc: 0.9994\n",
      "Epoch 14: ReduceLROnPlateau reducing learning rate to 2.141060758731328e-05.\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0017 - acc: 0.9994 - val_loss: 1.5389 - val_acc: 0.8089 - lr: 4.2821e-05\n",
      "Epoch 15/20\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 6.1852e-04 - acc: 0.9997 - val_loss: 1.4949 - val_acc: 0.8065 - lr: 3.1007e-05\n",
      "Epoch 16/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 4.2754e-04 - acc: 0.9998Restoring model weights from the end of the best epoch: 11.\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 4.2754e-04 - acc: 0.9998 - val_loss: 1.7664 - val_acc: 0.8041 - lr: 2.0916e-05\n",
      "Epoch 16: early stopping\n",
      "Member 2 Final Val Accuracy: 0.8041\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 3/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 116s 105ms/step - loss: 0.4061 - acc: 0.8246 - val_loss: 0.4783 - val_acc: 0.7933 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 90s 85ms/step - loss: 0.1825 - acc: 0.9301 - val_loss: 0.5687 - val_acc: 0.8113 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.1018 - acc: 0.9633 - val_loss: 0.6809 - val_acc: 0.7849 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0636 - acc: 0.9784 - val_loss: 0.6992 - val_acc: 0.7933 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0434 - acc: 0.9855\n",
      "Epoch 5: ReduceLROnPlateau reducing learning rate to 8.550701750209555e-05.\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0434 - acc: 0.9855 - val_loss: 0.8466 - val_acc: 0.7873 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0329 - acc: 0.9891 - val_loss: 0.9894 - val_acc: 0.7909 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0233 - acc: 0.9921Restoring model weights from the end of the best epoch: 2.\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0233 - acc: 0.9921 - val_loss: 0.9929 - val_acc: 0.7933 - lr: 1.4596e-04\n",
      "Epoch 7: early stopping\n",
      "Member 3 Final Val Accuracy: 0.7933\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 4/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 116s 105ms/step - loss: 0.4159 - acc: 0.8223 - val_loss: 0.4739 - val_acc: 0.7969 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 89s 85ms/step - loss: 0.1796 - acc: 0.9313 - val_loss: 0.5711 - val_acc: 0.7861 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.1003 - acc: 0.9638 - val_loss: 0.6413 - val_acc: 0.7752 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0626 - acc: 0.9779\n",
      "Epoch 4: ReduceLROnPlateau reducing learning rate to 9.055068221641704e-05.\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0626 - acc: 0.9779 - val_loss: 0.6724 - val_acc: 0.7885 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0414 - acc: 0.9861 - val_loss: 0.8007 - val_acc: 0.8065 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0329 - acc: 0.9894 - val_loss: 1.0491 - val_acc: 0.7764 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0231 - acc: 0.9926 - val_loss: 0.8826 - val_acc: 0.7740 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0176 - acc: 0.9942\n",
      "Epoch 8: ReduceLROnPlateau reducing learning rate to 6.580336776096374e-05.\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0176 - acc: 0.9942 - val_loss: 1.2494 - val_acc: 0.7812 - lr: 1.3161e-04\n",
      "Epoch 9/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0117 - acc: 0.9959 - val_loss: 1.2269 - val_acc: 0.7788 - lr: 1.1650e-04\n",
      "Epoch 10/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0105 - acc: 0.9965Restoring model weights from the end of the best epoch: 5.\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0105 - acc: 0.9965 - val_loss: 1.3504 - val_acc: 0.7800 - lr: 1.0101e-04\n",
      "Epoch 10: early stopping\n",
      "Member 4 Final Val Accuracy: 0.7800\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 5/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 116s 105ms/step - loss: 0.4027 - acc: 0.8235 - val_loss: 0.4879 - val_acc: 0.8161 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 89s 85ms/step - loss: 0.1775 - acc: 0.9330 - val_loss: 0.4972 - val_acc: 0.8137 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0962 - acc: 0.9648 - val_loss: 0.6292 - val_acc: 0.8005 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0594 - acc: 0.9798\n",
      "Epoch 4: ReduceLROnPlateau reducing learning rate to 9.055068221641704e-05.\n",
      "1052/1052 [==============================] - 72s 68ms/step - loss: 0.0594 - acc: 0.9798 - val_loss: 0.6567 - val_acc: 0.8041 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 85s 81ms/step - loss: 0.0420 - acc: 0.9865 - val_loss: 0.8736 - val_acc: 0.8005 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0295 - acc: 0.9906Restoring model weights from the end of the best epoch: 1.\n",
      "1052/1052 [==============================] - 87s 82ms/step - loss: 0.0295 - acc: 0.9906 - val_loss: 0.8182 - val_acc: 0.8053 - lr: 1.5920e-04\n",
      "Epoch 6: early stopping\n",
      "Member 5 Final Val Accuracy: 0.8053\n",
      "\n",
      "============================================================\n",
      "Deep Ensemble Training Complete!\n",
      "============================================================\n",
      "\n",
      "⏱️  Training completed in 4101.8s (1.14h)\n",
      "\n",
      "Evaluating Deep_Ensemble...\n",
      "\n",
      "================================================================================\n",
      "Evaluating: Deep_Ensemble\n",
      "================================================================================\n",
      "Computing in-domain predictions (ensemble)...\n",
      "Computing OOD predictions (ensemble)...\n",
      "Computing metrics...\n",
      "✓ Evaluation completed in 28.0s\n",
      "🧹 Memory cleaned\n",
      "✅ Trained: 456_Deep_Ensemble (4101.8s)\n",
      "\n",
      "📊 Progress: 23/24 (95.8%)\n",
      "   Remaining: 1 experiments\n",
      "\n",
      "################################################################################\n",
      "# SEED: 2026\n",
      "################################################################################\n",
      "\n",
      "⏭️  [39/24] Skipping Baseline_Transformer (seed=2026) - already completed\n",
      "⏭️  [40/24] Skipping Full_Rank_BBB (seed=2026) - already completed\n",
      "⏭️  [41/24] Skipping Low_Rank_Random_Init (seed=2026) - already completed\n",
      "⏭️  [42/24] Skipping Low_Rank_SVD_Init (seed=2026) - already completed\n",
      "⏭️  [43/24] Skipping Rank1_BBB (seed=2026) - already completed\n",
      "\n",
      "🚀 [44/24] Starting: Deep_Ensemble (seed=2026)\n",
      "\n",
      "================================================================================\n",
      "Experiment: Deep_Ensemble | Seed: 2026\n",
      "================================================================================\n",
      "\n",
      "Training Deep_Ensemble from scratch...\n",
      "🧹 Memory cleaned\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 1/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 115s 103ms/step - loss: 0.4166 - acc: 0.8211 - val_loss: 0.5486 - val_acc: 0.7885 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 89s 85ms/step - loss: 0.1819 - acc: 0.9311 - val_loss: 0.5176 - val_acc: 0.7885 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 89s 84ms/step - loss: 0.1026 - acc: 0.9630 - val_loss: 0.6481 - val_acc: 0.7897 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 90s 86ms/step - loss: 0.0635 - acc: 0.9781 - val_loss: 0.8470 - val_acc: 0.7993 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 89s 84ms/step - loss: 0.0429 - acc: 0.9862 - val_loss: 0.8884 - val_acc: 0.7861 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0324 - acc: 0.9897 - val_loss: 1.0208 - val_acc: 0.8029 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0252 - acc: 0.9923 - val_loss: 0.7939 - val_acc: 0.8149 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0179 - acc: 0.9937 - val_loss: 1.1970 - val_acc: 0.8077 - lr: 1.3161e-04\n",
      "Epoch 9/20\n",
      "1052/1052 [==============================] - 87s 82ms/step - loss: 0.0125 - acc: 0.9964 - val_loss: 1.0082 - val_acc: 0.8101 - lr: 1.1650e-04\n",
      "Epoch 10/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0085 - acc: 0.9972\n",
      "Epoch 10: ReduceLROnPlateau reducing learning rate to 5.0507394917076454e-05.\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0085 - acc: 0.9972 - val_loss: 1.1774 - val_acc: 0.8041 - lr: 1.0101e-04\n",
      "Epoch 11/20\n",
      "1052/1052 [==============================] - 87s 82ms/step - loss: 0.0064 - acc: 0.9977 - val_loss: 1.3255 - val_acc: 0.8113 - lr: 8.5528e-05\n",
      "Epoch 12/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0035 - acc: 0.9989Restoring model weights from the end of the best epoch: 7.\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0035 - acc: 0.9989 - val_loss: 1.4079 - val_acc: 0.8113 - lr: 7.0421e-05\n",
      "Epoch 12: early stopping\n",
      "Member 1 Final Val Accuracy: 0.8113\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 2/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 115s 104ms/step - loss: 0.4033 - acc: 0.8229 - val_loss: 0.5031 - val_acc: 0.7945 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 90s 86ms/step - loss: 0.1771 - acc: 0.9325 - val_loss: 0.5140 - val_acc: 0.8005 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0981 - acc: 0.9640 - val_loss: 0.6533 - val_acc: 0.7945 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0578 - acc: 0.9806 - val_loss: 0.8987 - val_acc: 0.7921 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0398 - acc: 0.9873\n",
      "Epoch 5: ReduceLROnPlateau reducing learning rate to 8.550701750209555e-05.\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0398 - acc: 0.9873 - val_loss: 0.7594 - val_acc: 0.7969 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0292 - acc: 0.9905 - val_loss: 0.7687 - val_acc: 0.8077 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0234 - acc: 0.9922 - val_loss: 0.9489 - val_acc: 0.8041 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - 87s 82ms/step - loss: 0.0163 - acc: 0.9945 - val_loss: 0.9884 - val_acc: 0.8137 - lr: 1.3161e-04\n",
      "Epoch 9/20\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0113 - acc: 0.9965 - val_loss: 1.0977 - val_acc: 0.7969 - lr: 1.1650e-04\n",
      "Epoch 10/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0082 - acc: 0.9968 - val_loss: 1.4239 - val_acc: 0.8077 - lr: 1.0101e-04\n",
      "Epoch 11/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0062 - acc: 0.9976\n",
      "Epoch 11: ReduceLROnPlateau reducing learning rate to 4.276379695511423e-05.\n",
      "1052/1052 [==============================] - 86s 81ms/step - loss: 0.0062 - acc: 0.9976 - val_loss: 1.2634 - val_acc: 0.8017 - lr: 8.5528e-05\n",
      "Epoch 12/20\n",
      "1052/1052 [==============================] - 71s 67ms/step - loss: 0.0037 - acc: 0.9987 - val_loss: 1.3572 - val_acc: 0.8041 - lr: 7.0421e-05\n",
      "Epoch 13/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0016 - acc: 0.9993Restoring model weights from the end of the best epoch: 8.\n",
      "1052/1052 [==============================] - 86s 82ms/step - loss: 0.0016 - acc: 0.9993 - val_loss: 1.6189 - val_acc: 0.8101 - lr: 5.6068e-05\n",
      "Epoch 13: early stopping\n",
      "Member 2 Final Val Accuracy: 0.8101\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 3/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 116s 105ms/step - loss: 0.4054 - acc: 0.8268 - val_loss: 0.4791 - val_acc: 0.7957 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 91s 86ms/step - loss: 0.1783 - acc: 0.9320 - val_loss: 0.5016 - val_acc: 0.8053 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 89s 84ms/step - loss: 0.0980 - acc: 0.9644 - val_loss: 0.5415 - val_acc: 0.8161 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0616 - acc: 0.9786 - val_loss: 0.6845 - val_acc: 0.7812 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0419 - acc: 0.9859 - val_loss: 0.8445 - val_acc: 0.7897 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0290 - acc: 0.9905\n",
      "Epoch 6: ReduceLROnPlateau reducing learning rate to 7.96013482613489e-05.\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0290 - acc: 0.9905 - val_loss: 1.0042 - val_acc: 0.7981 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0226 - acc: 0.9924 - val_loss: 1.1428 - val_acc: 0.7716 - lr: 1.4596e-04\n",
      "Epoch 8/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0162 - acc: 0.9944Restoring model weights from the end of the best epoch: 3.\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0162 - acc: 0.9944 - val_loss: 1.0375 - val_acc: 0.7849 - lr: 1.3161e-04\n",
      "Epoch 8: early stopping\n",
      "Member 3 Final Val Accuracy: 0.7849\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 4/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 115s 104ms/step - loss: 0.4144 - acc: 0.8212 - val_loss: 0.4327 - val_acc: 0.8149 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.1806 - acc: 0.9312 - val_loss: 0.4894 - val_acc: 0.8017 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.1004 - acc: 0.9634 - val_loss: 0.6141 - val_acc: 0.7957 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0619 - acc: 0.9784\n",
      "Epoch 4: ReduceLROnPlateau reducing learning rate to 9.055068221641704e-05.\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0619 - acc: 0.9784 - val_loss: 0.8415 - val_acc: 0.7969 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - 88s 83ms/step - loss: 0.0420 - acc: 0.9860 - val_loss: 0.8576 - val_acc: 0.8017 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0293 - acc: 0.9907Restoring model weights from the end of the best epoch: 1.\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0293 - acc: 0.9907 - val_loss: 1.0043 - val_acc: 0.7897 - lr: 1.5920e-04\n",
      "Epoch 6: early stopping\n",
      "Member 4 Final Val Accuracy: 0.7897\n",
      "\n",
      "============================================================\n",
      "Training Ensemble Member 5/5\n",
      "============================================================\n",
      "Epoch 1/20\n",
      "1052/1052 [==============================] - 114s 103ms/step - loss: 0.4095 - acc: 0.8230 - val_loss: 0.5583 - val_acc: 0.7620 - lr: 1.9878e-04\n",
      "Epoch 2/20\n",
      "1052/1052 [==============================] - 89s 85ms/step - loss: 0.1773 - acc: 0.9325 - val_loss: 0.5658 - val_acc: 0.7969 - lr: 1.9516e-04\n",
      "Epoch 3/20\n",
      "1052/1052 [==============================] - 89s 85ms/step - loss: 0.0992 - acc: 0.9626 - val_loss: 0.5984 - val_acc: 0.7957 - lr: 1.8922e-04\n",
      "Epoch 4/20\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0598 - acc: 0.9795 - val_loss: 0.7757 - val_acc: 0.7969 - lr: 1.8110e-04\n",
      "Epoch 5/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0411 - acc: 0.9865\n",
      "Epoch 5: ReduceLROnPlateau reducing learning rate to 8.550701750209555e-05.\n",
      "1052/1052 [==============================] - 87s 82ms/step - loss: 0.0411 - acc: 0.9865 - val_loss: 0.8743 - val_acc: 0.7825 - lr: 1.7101e-04\n",
      "Epoch 6/20\n",
      "1052/1052 [==============================] - 87s 83ms/step - loss: 0.0286 - acc: 0.9907 - val_loss: 1.0880 - val_acc: 0.7752 - lr: 1.5920e-04\n",
      "Epoch 7/20\n",
      "1052/1052 [==============================] - ETA: 0s - loss: 0.0245 - acc: 0.9920Restoring model weights from the end of the best epoch: 2.\n",
      "1052/1052 [==============================] - 88s 84ms/step - loss: 0.0245 - acc: 0.9920 - val_loss: 1.1855 - val_acc: 0.7921 - lr: 1.4596e-04\n",
      "Epoch 7: early stopping\n",
      "Member 5 Final Val Accuracy: 0.7921\n",
      "\n",
      "============================================================\n",
      "Deep Ensemble Training Complete!\n",
      "============================================================\n",
      "\n",
      "⏱️  Training completed in 4173.5s (1.16h)\n",
      "\n",
      "Evaluating Deep_Ensemble...\n",
      "\n",
      "================================================================================\n",
      "Evaluating: Deep_Ensemble\n",
      "================================================================================\n",
      "Computing in-domain predictions (ensemble)...\n",
      "Computing OOD predictions (ensemble)...\n",
      "Computing metrics...\n",
      "✓ Evaluation completed in 27.7s\n",
      "🧹 Memory cleaned\n",
      "✅ Trained: 2026_Deep_Ensemble (4173.5s)\n",
      "\n",
      "📊 Progress: 24/24 (100.0%)\n",
      "   Remaining: 0 experiments\n",
      "\n",
      "================================================================================\n",
      "✅ ALL EXPERIMENTS COMPLETED!\n",
      "================================================================================\n",
      "Total experiments: 24\n",
      "Reused seed=42: 0\n",
      "Newly trained: 24\n",
      "\n",
      "Results saved in: checkpoints/multi_seed\n"
     ]
    }
   ],
   "source": [
    "# Print initial status\n",
    "summary = tracker.get_summary()\n",
    "print(f\"\\n{'='*80}\")\n",
    "print(f\"Multi-Seed Robustness Check\")\n",
    "print(f\"{'='*80}\")\n",
    "print(f\"Seeds: {SEEDS}\")\n",
    "print(f\"Models: {len(MODEL_NAMES)}\")\n",
    "print(f\"Total experiments: {summary['total']}\")\n",
    "print(f\"Already completed: {summary['completed']}\")\n",
    "print(f\"Remaining: {summary['remaining']}\")\n",
    "if summary['reused'] > 0:\n",
    "    print(f\"Reused from seed=42: {summary['reused']}\")\n",
    "print(f\"{'='*80}\\n\")\n",
    "\n",
    "experiment_num = summary['completed']\n",
    "total_experiments = summary['total']\n",
    "\n",
    "# Run experiments\n",
    "for seed in SEEDS:\n",
    "    print(f\"\\n{'#'*80}\")\n",
    "    print(f\"# SEED: {seed}\")\n",
    "    print(f\"{'#'*80}\\n\")\n",
    "    \n",
    "    # Set seed at beginning of each seed iteration\n",
    "    set_seed(seed)\n",
    "    \n",
    "    for model_name in MODEL_NAMES:\n",
    "        # Skip if already completed\n",
    "        if tracker.is_completed(seed, model_name):\n",
    "            print(f\"⏭️  [{experiment_num + 1}/{total_experiments}] Skipping {model_name} (seed={seed}) - already completed\")\n",
    "            experiment_num += 1\n",
    "            continue\n",
    "        \n",
    "        experiment_num += 1\n",
    "        print(f\"\\n🚀 [{experiment_num}/{total_experiments}] Starting: {model_name} (seed={seed})\")\n",
    "        \n",
    "        try:\n",
    "            # Run experiment (no baseline_cache - uses disk-based caching now)\n",
    "            metrics, training_time, reused = run_single_experiment(\n",
    "                seed=seed,\n",
    "                model_name=model_name,\n",
    "                train_dataset=train_dataset,\n",
    "                val_dataset=val_dataset,\n",
    "                ood_dataset=ood_dataset,\n",
    "                X_test_dict=X_test_dict,\n",
    "                y_test_arr=y_test_arr,\n",
    "                X_ood_dict=X_ood_dict,\n",
    "                y_ood_arr=y_ood_arr,\n",
    "                kl_weight=kl_weight\n",
    "            )\n",
    "            tracker.save_result(seed, model_name, metrics, training_time, reused=reused)\n",
    "            \n",
    "            # Print updated progress\n",
    "            summary = tracker.get_summary()\n",
    "            print(f\"\\n📊 Progress: {summary['completed']}/{summary['total']} ({summary['progress_pct']:.1f}%)\")\n",
    "            print(f\"   Remaining: {summary['remaining']} experiments\")\n",
    "            \n",
    "        except Exception as e:\n",
    "            print(f\"\\n❌ ERROR in {model_name} (seed={seed}):\")\n",
    "            print(f\"   {e}\")\n",
    "            print(f\"\\n💾 Progress saved. You can resume later by running this cell again.\")\n",
    "            import traceback\n",
    "            traceback.print_exc()\n",
    "            raise\n",
    "    \n",
    "    # Clear baseline cache after each seed completes to free memory\n",
    "    clear_baseline_cache(seed)\n",
    "\n",
    "print(f\"\\n{'='*80}\")\n",
    "print(\"✅ ALL EXPERIMENTS COMPLETED!\")\n",
    "print(f\"{'='*80}\")\n",
    "summary = tracker.get_summary()\n",
    "print(f\"Total experiments: {summary['total']}\")\n",
    "print(f\"Reused seed=42: {summary['reused']}\")\n",
    "print(f\"Newly trained: {summary['total'] - summary['reused']}\")\n",
    "print(f\"\\nResults saved in: {CHECKPOINT_DIR}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Results Aggregation\n",
    "\n",
    "Compute mean ± std across all  seeds."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "================================================================================\n",
      "Aggregating Results Across Seeds\n",
      "================================================================================\n",
      "\n",
      "\n",
      "Baseline_Transformer:\n",
      "----------------------------------------------------------------------\n",
      "  AUPR_Error_MI                      : 0.1875 ± 0.0061\n",
      "                                       [min: 0.1812, max: 0.1972]\n",
      "  AUPR_Error_STD                     : 0.1875 ± 0.0061\n",
      "                                       [min: 0.1812, max: 0.1972]\n",
      "  AUPR_In_MI                         : 0.1029 ± 0.0000\n",
      "                                       [min: 0.1029, max: 0.1029]\n",
      "  AUPR_In_STD                        : 0.1029 ± 0.0000\n",
      "                                       [min: 0.1029, max: 0.1029]\n",
      "  AUPR_Out_MI                        : 0.8971 ± 0.0000\n",
      "                                       [min: 0.8971, max: 0.8971]\n",
      "  AUPR_Out_STD                       : 0.8971 ± 0.0000\n",
      "                                       [min: 0.8971, max: 0.8971]\n",
      "  AUPR_Success_MI                    : 0.8125 ± 0.0061\n",
      "                                       [min: 0.8028, max: 0.8188]\n",
      "  AUPR_Success_STD                   : 0.8125 ± 0.0061\n",
      "                                       [min: 0.8028, max: 0.8188]\n",
      "  AUROC_Classification               : 0.8918 ± 0.0040\n",
      "                                       [min: 0.8885, max: 0.8987]\n",
      "  AUROC_OOD_MI                       : 0.5000 ± 0.0000\n",
      "                                       [min: 0.5000, max: 0.5000]\n",
      "  AUROC_OOD_STD                      : 0.5000 ± 0.0000\n",
      "                                       [min: 0.5000, max: 0.5000]\n",
      "  Accuracy                           : 0.8128 ± 0.0062\n",
      "                                       [min: 0.8030, max: 0.8190]\n",
      "  Brier_Score                        : 0.1545 ± 0.0019\n",
      "                                       [min: 0.1529, max: 0.1577]\n",
      "  ECE                                : 0.1323 ± 0.0076\n",
      "                                       [min: 0.1271, max: 0.1454]\n",
      "  Entropy_Ratio                      : 1.0144 ± 0.0027\n",
      "                                       [min: 1.0106, max: 1.0180]\n",
      "  MI_Ratio                           : 0.0000 ± 0.0000\n",
      "                                       [min: 0.0000, max: 0.0000]\n",
      "  Mean_Entropy_In                    : 0.5971 ± 0.0011\n",
      "                                       [min: 0.5962, max: 0.5989]\n",
      "  Mean_Entropy_Out                   : 0.6057 ± 0.0009\n",
      "                                       [min: 0.6049, max: 0.6072]\n",
      "  Mean_MI_In                         : 0.0000 ± 0.0000\n",
      "                                       [min: 0.0000, max: 0.0000]\n",
      "  Mean_MI_Out                        : 0.0000 ± 0.0000\n",
      "                                       [min: 0.0000, max: 0.0000]\n",
      "  Mean_STD_In                        : 0.0000 ± 0.0000\n",
      "                                       [min: 0.0000, max: 0.0000]\n",
      "  Mean_STD_Out                       : 0.0000 ± 0.0000\n",
      "                                       [min: 0.0000, max: 0.0000]\n",
      "  NLL                                : 0.4906 ± 0.0040\n",
      "                                       [min: 0.4871, max: 0.4973]\n",
      "  STD_Ratio                          : 0.0000 ± 0.0000\n",
      "                                       [min: 0.0000, max: 0.0000]\n",
      "\n",
      "Full_Rank_BBB:\n",
      "----------------------------------------------------------------------\n",
      "  AUPR_Error_MI                      : 0.4177 ± 0.0319\n",
      "                                       [min: 0.3900, max: 0.4717]\n",
      "  AUPR_Error_STD                     : 0.4180 ± 0.0339\n",
      "                                       [min: 0.3884, max: 0.4750]\n",
      "  AUPR_In_MI                         : 0.2224 ± 0.0282\n",
      "                                       [min: 0.1862, max: 0.2618]\n",
      "  AUPR_In_STD                        : 0.2226 ± 0.0284\n",
      "                                       [min: 0.1860, max: 0.2621]\n",
      "  AUPR_Out_MI                        : 0.9212 ± 0.0058\n",
      "                                       [min: 0.9112, max: 0.9255]\n",
      "  AUPR_Out_STD                       : 0.9213 ± 0.0058\n",
      "                                       [min: 0.9114, max: 0.9258]\n",
      "  AUPR_Success_MI                    : 0.8883 ± 0.0067\n",
      "                                       [min: 0.8813, max: 0.8980]\n",
      "  AUPR_Success_STD                   : 0.8884 ± 0.0066\n",
      "                                       [min: 0.8814, max: 0.8982]\n",
      "  AUROC_Classification               : 0.8367 ± 0.0118\n",
      "                                       [min: 0.8208, max: 0.8482]\n",
      "  AUROC_OOD_MI                       : 0.6223 ± 0.0200\n",
      "                                       [min: 0.5961, max: 0.6461]\n",
      "  AUROC_OOD_STD                      : 0.6226 ± 0.0202\n",
      "                                       [min: 0.5969, max: 0.6467]\n",
      "  Accuracy                           : 0.7525 ± 0.0118\n",
      "                                       [min: 0.7350, max: 0.7660]\n",
      "  Brier_Score                        : 0.1830 ± 0.0050\n",
      "                                       [min: 0.1779, max: 0.1902]\n",
      "  ECE                                : 0.1189 ± 0.0055\n",
      "                                       [min: 0.1108, max: 0.1263]\n",
      "  Entropy_Ratio                      : 1.0268 ± 0.0052\n",
      "                                       [min: 1.0208, max: 1.0335]\n",
      "  MI_Ratio                           : 1.3108 ± 0.0661\n",
      "                                       [min: 1.2384, max: 1.4056]\n",
      "  Mean_Entropy_In                    : 0.6397 ± 0.0052\n",
      "                                       [min: 0.6347, max: 0.6475]\n",
      "  Mean_Entropy_Out                   : 0.6568 ± 0.0047\n",
      "                                       [min: 0.6492, max: 0.6610]\n",
      "  Mean_MI_In                         : 0.0248 ± 0.0027\n",
      "                                       [min: 0.0218, max: 0.0289]\n",
      "  Mean_MI_Out                        : 0.0325 ± 0.0031\n",
      "                                       [min: 0.0275, max: 0.0357]\n",
      "  Mean_STD_In                        : 0.0949 ± 0.0080\n",
      "                                       [min: 0.0864, max: 0.1068]\n",
      "  Mean_STD_Out                       : 0.1180 ± 0.0076\n",
      "                                       [min: 0.1057, max: 0.1256]\n",
      "  NLL                                : 0.5525 ± 0.0107\n",
      "                                       [min: 0.5414, max: 0.5679]\n",
      "  STD_Ratio                          : 1.2462 ± 0.0544\n",
      "                                       [min: 1.1761, max: 1.3252]\n",
      "\n",
      "Low_Rank_Random_Init:\n",
      "----------------------------------------------------------------------\n",
      "  AUPR_Error_MI                      : 0.3526 ± 0.0208\n",
      "                                       [min: 0.3377, max: 0.3885]\n",
      "  AUPR_Error_STD                     : 0.3545 ± 0.0193\n",
      "                                       [min: 0.3398, max: 0.3876]\n",
      "  AUPR_In_MI                         : 0.3021 ± 0.0271\n",
      "                                       [min: 0.2701, max: 0.3452]\n",
      "  AUPR_In_STD                        : 0.3026 ± 0.0268\n",
      "                                       [min: 0.2715, max: 0.3454]\n",
      "  AUPR_Out_MI                        : 0.9191 ± 0.0122\n",
      "                                       [min: 0.8995, max: 0.9317]\n",
      "  AUPR_Out_STD                       : 0.9197 ± 0.0118\n",
      "                                       [min: 0.9006, max: 0.9317]\n",
      "  AUPR_Success_MI                    : 0.9178 ± 0.0050\n",
      "                                       [min: 0.9117, max: 0.9250]\n",
      "  AUPR_Success_STD                   : 0.9180 ± 0.0051\n",
      "                                       [min: 0.9118, max: 0.9252]\n",
      "  AUROC_Classification               : 0.8841 ± 0.0023\n",
      "                                       [min: 0.8802, max: 0.8862]\n",
      "  AUROC_OOD_MI                       : 0.6407 ± 0.0287\n",
      "                                       [min: 0.5918, max: 0.6639]\n",
      "  AUROC_OOD_STD                      : 0.6418 ± 0.0277\n",
      "                                       [min: 0.5947, max: 0.6651]\n",
      "  Accuracy                           : 0.8068 ± 0.0033\n",
      "                                       [min: 0.8030, max: 0.8120]\n",
      "  Brier_Score                        : 0.1705 ± 0.0029\n",
      "                                       [min: 0.1685, max: 0.1755]\n",
      "  ECE                                : 0.1733 ± 0.0114\n",
      "                                       [min: 0.1628, max: 0.1920]\n",
      "  Entropy_Ratio                      : 1.0273 ± 0.0027\n",
      "                                       [min: 1.0237, max: 1.0312]\n",
      "  MI_Ratio                           : 1.3544 ± 0.1024\n",
      "                                       [min: 1.1794, max: 1.4353]\n",
      "  Mean_Entropy_In                    : 0.6423 ± 0.0042\n",
      "                                       [min: 0.6366, max: 0.6483]\n",
      "  Mean_Entropy_Out                   : 0.6599 ± 0.0049\n",
      "                                       [min: 0.6538, max: 0.6659]\n",
      "  Mean_MI_In                         : 0.0047 ± 0.0002\n",
      "                                       [min: 0.0044, max: 0.0049]\n",
      "  Mean_MI_Out                        : 0.0063 ± 0.0004\n",
      "                                       [min: 0.0057, max: 0.0069]\n",
      "  Mean_STD_In                        : 0.0410 ± 0.0018\n",
      "                                       [min: 0.0380, max: 0.0424]\n",
      "  Mean_STD_Out                       : 0.0519 ± 0.0025\n",
      "                                       [min: 0.0488, max: 0.0552]\n",
      "  NLL                                : 0.5272 ± 0.0063\n",
      "                                       [min: 0.5228, max: 0.5381]\n",
      "  STD_Ratio                          : 1.2650 ± 0.0596\n",
      "                                       [min: 1.1685, max: 1.3247]\n",
      "\n",
      "Low_Rank_SVD_Init:\n",
      "----------------------------------------------------------------------\n",
      "  AUPR_Error_MI                      : 0.3829 ± 0.0119\n",
      "                                       [min: 0.3627, max: 0.3925]\n",
      "  AUPR_Error_STD                     : 0.3848 ± 0.0115\n",
      "                                       [min: 0.3650, max: 0.3927]\n",
      "  AUPR_In_MI                         : 0.2730 ± 0.0124\n",
      "                                       [min: 0.2588, max: 0.2928]\n",
      "  AUPR_In_STD                        : 0.2738 ± 0.0124\n",
      "                                       [min: 0.2594, max: 0.2936]\n",
      "  AUPR_Out_MI                        : 0.9031 ± 0.0100\n",
      "                                       [min: 0.8897, max: 0.9179]\n",
      "  AUPR_Out_STD                       : 0.9045 ± 0.0097\n",
      "                                       [min: 0.8914, max: 0.9187]\n",
      "  AUPR_Success_MI                    : 0.9233 ± 0.0037\n",
      "                                       [min: 0.9191, max: 0.9284]\n",
      "  AUPR_Success_STD                   : 0.9235 ± 0.0037\n",
      "                                       [min: 0.9193, max: 0.9285]\n",
      "  AUROC_Classification               : 0.8857 ± 0.0036\n",
      "                                       [min: 0.8803, max: 0.8892]\n",
      "  AUROC_OOD_MI                       : 0.6162 ± 0.0118\n",
      "                                       [min: 0.6008, max: 0.6332]\n",
      "  AUROC_OOD_STD                      : 0.6189 ± 0.0111\n",
      "                                       [min: 0.6045, max: 0.6346]\n",
      "  Accuracy                           : 0.8008 ± 0.0058\n",
      "                                       [min: 0.7920, max: 0.8060]\n",
      "  Brier_Score                        : 0.1660 ± 0.0010\n",
      "                                       [min: 0.1650, max: 0.1673]\n",
      "  ECE                                : 0.1583 ± 0.0068\n",
      "                                       [min: 0.1515, max: 0.1662]\n",
      "  Entropy_Ratio                      : 1.0304 ± 0.0012\n",
      "                                       [min: 1.0285, max: 1.0317]\n",
      "  MI_Ratio                           : 1.2237 ± 0.1047\n",
      "                                       [min: 1.0654, max: 1.3598]\n",
      "  Mean_Entropy_In                    : 0.6334 ± 0.0043\n",
      "                                       [min: 0.6284, max: 0.6403]\n",
      "  Mean_Entropy_Out                   : 0.6527 ± 0.0038\n",
      "                                       [min: 0.6479, max: 0.6585]\n",
      "  Mean_MI_In                         : 0.0030 ± 0.0004\n",
      "                                       [min: 0.0024, max: 0.0034]\n",
      "  Mean_MI_Out                        : 0.0036 ± 0.0005\n",
      "                                       [min: 0.0032, max: 0.0043]\n",
      "  Mean_STD_In                        : 0.0300 ± 0.0010\n",
      "                                       [min: 0.0285, max: 0.0309]\n",
      "  Mean_STD_Out                       : 0.0374 ± 0.0017\n",
      "                                       [min: 0.0353, max: 0.0395]\n",
      "  NLL                                : 0.5168 ± 0.0023\n",
      "                                       [min: 0.5145, max: 0.5201]\n",
      "  STD_Ratio                          : 1.2478 ± 0.0378\n",
      "                                       [min: 1.1856, max: 1.2794]\n",
      "\n",
      "Rank1_BBB:\n",
      "----------------------------------------------------------------------\n",
      "  AUPR_Error_MI                      : 0.4268 ± 0.0419\n",
      "                                       [min: 0.3751, max: 0.4910]\n",
      "  AUPR_Error_STD                     : 0.4273 ± 0.0418\n",
      "                                       [min: 0.3768, max: 0.4921]\n",
      "  AUPR_In_MI                         : 0.2729 ± 0.0277\n",
      "                                       [min: 0.2376, max: 0.3063]\n",
      "  AUPR_In_STD                        : 0.2733 ± 0.0276\n",
      "                                       [min: 0.2379, max: 0.3066]\n",
      "  AUPR_Out_MI                        : 0.9043 ± 0.0054\n",
      "                                       [min: 0.8958, max: 0.9101]\n",
      "  AUPR_Out_STD                       : 0.9053 ± 0.0053\n",
      "                                       [min: 0.8969, max: 0.9112]\n",
      "  AUPR_Success_MI                    : 0.9314 ± 0.0046\n",
      "                                       [min: 0.9246, max: 0.9370]\n",
      "  AUPR_Success_STD                   : 0.9315 ± 0.0046\n",
      "                                       [min: 0.9247, max: 0.9370]\n",
      "  AUROC_Classification               : 0.8895 ± 0.0034\n",
      "                                       [min: 0.8838, max: 0.8922]\n",
      "  AUROC_OOD_MI                       : 0.6131 ± 0.0158\n",
      "                                       [min: 0.5966, max: 0.6380]\n",
      "  AUROC_OOD_STD                      : 0.6149 ± 0.0158\n",
      "                                       [min: 0.5980, max: 0.6395]\n",
      "  Accuracy                           : 0.7950 ± 0.0131\n",
      "                                       [min: 0.7740, max: 0.8070]\n",
      "  Brier_Score                        : 0.1623 ± 0.0026\n",
      "                                       [min: 0.1591, max: 0.1657]\n",
      "  ECE                                : 0.1488 ± 0.0116\n",
      "                                       [min: 0.1377, max: 0.1683]\n",
      "  Entropy_Ratio                      : 1.0267 ± 0.0031\n",
      "                                       [min: 1.0219, max: 1.0297]\n",
      "  MI_Ratio                           : 1.1105 ± 0.0883\n",
      "                                       [min: 1.0003, max: 1.2349]\n",
      "  Mean_Entropy_In                    : 0.6199 ± 0.0026\n",
      "                                       [min: 0.6159, max: 0.6231]\n",
      "  Mean_Entropy_Out                   : 0.6365 ± 0.0031\n",
      "                                       [min: 0.6335, max: 0.6413]\n",
      "  Mean_MI_In                         : 0.0000 ± 0.0000\n",
      "                                       [min: 0.0000, max: 0.0000]\n",
      "  Mean_MI_Out                        : 0.0000 ± 0.0000\n",
      "                                       [min: 0.0000, max: 0.0000]\n",
      "  Mean_STD_In                        : 0.0023 ± 0.0000\n",
      "                                       [min: 0.0022, max: 0.0023]\n",
      "  Mean_STD_Out                       : 0.0028 ± 0.0001\n",
      "                                       [min: 0.0027, max: 0.0029]\n",
      "  NLL                                : 0.5077 ± 0.0052\n",
      "                                       [min: 0.5012, max: 0.5144]\n",
      "  STD_Ratio                          : 1.2344 ± 0.0436\n",
      "                                       [min: 1.1714, max: 1.2814]\n",
      "\n",
      "Deep_Ensemble:\n",
      "----------------------------------------------------------------------\n",
      "  AUPR_Error_MI                      : 0.3430 ± 0.0284\n",
      "                                       [min: 0.3129, max: 0.3879]\n",
      "  AUPR_Error_STD                     : 0.3699 ± 0.0349\n",
      "                                       [min: 0.3409, max: 0.4295]\n",
      "  AUPR_In_MI                         : 0.2672 ± 0.0162\n",
      "                                       [min: 0.2395, max: 0.2799]\n",
      "  AUPR_In_STD                        : 0.2816 ± 0.0196\n",
      "                                       [min: 0.2487, max: 0.2984]\n",
      "  AUPR_Out_MI                        : 0.9302 ± 0.0022\n",
      "                                       [min: 0.9276, max: 0.9335]\n",
      "  AUPR_Out_STD                       : 0.9317 ± 0.0019\n",
      "                                       [min: 0.9290, max: 0.9341]\n",
      "  AUPR_Success_MI                    : 0.9339 ± 0.0039\n",
      "                                       [min: 0.9288, max: 0.9395]\n",
      "  AUPR_Success_STD                   : 0.9385 ± 0.0035\n",
      "                                       [min: 0.9345, max: 0.9421]\n",
      "  AUROC_Classification               : 0.9056 ± 0.0024\n",
      "                                       [min: 0.9037, max: 0.9097]\n",
      "  AUROC_OOD_MI                       : 0.6577 ± 0.0100\n",
      "                                       [min: 0.6493, max: 0.6739]\n",
      "  AUROC_OOD_STD                      : 0.6631 ± 0.0085\n",
      "                                       [min: 0.6551, max: 0.6771]\n",
      "  Accuracy                           : 0.8250 ± 0.0066\n",
      "                                       [min: 0.8140, max: 0.8300]\n",
      "  Brier_Score                        : 0.1284 ± 0.0032\n",
      "                                       [min: 0.1233, max: 0.1320]\n",
      "  ECE                                : 0.0659 ± 0.0111\n",
      "                                       [min: 0.0472, max: 0.0757]\n",
      "  Entropy_Ratio                      : 1.5141 ± 0.0339\n",
      "                                       [min: 1.4566, max: 1.5429]\n",
      "  MI_Ratio                           : 1.5531 ± 0.0516\n",
      "                                       [min: 1.4671, max: 1.6038]\n",
      "  Mean_Entropy_In                    : 0.2636 ± 0.0187\n",
      "                                       [min: 0.2453, max: 0.2920]\n",
      "  Mean_Entropy_Out                   : 0.3986 ± 0.0217\n",
      "                                       [min: 0.3760, max: 0.4253]\n",
      "  Mean_MI_In                         : 0.0920 ± 0.0045\n",
      "                                       [min: 0.0870, max: 0.0993]\n",
      "  Mean_MI_Out                        : 0.1430 ± 0.0112\n",
      "                                       [min: 0.1276, max: 0.1592]\n",
      "  Mean_STD_In                        : 0.1200 ± 0.0044\n",
      "                                       [min: 0.1153, max: 0.1259]\n",
      "  Mean_STD_Out                       : 0.1872 ± 0.0085\n",
      "                                       [min: 0.1811, max: 0.2018]\n",
      "  NLL                                : 0.4342 ± 0.0231\n",
      "                                       [min: 0.3967, max: 0.4534]\n",
      "  STD_Ratio                          : 1.5606 ± 0.0429\n",
      "                                       [min: 1.4892, max: 1.6028]\n",
      "\n",
      "✅ Aggregated results saved to: multi_seed_results/aggregated_results.json\n"
     ]
    }
   ],
   "source": [
    "print(\"\\n\" + \"=\"*80)\n",
    "print(\"Aggregating Results Across Seeds\")\n",
    "print(\"=\"*80 + \"\\n\")\n",
    "\n",
    "# Organize results by model\n",
    "results_by_model = {model: [] for model in MODEL_NAMES}\n",
    "\n",
    "for key, result in tracker.results.items():\n",
    "    model_name = result['model']\n",
    "    results_by_model[model_name].append(result)\n",
    "\n",
    "# Compute statistics\n",
    "def compute_stats(values):\n",
    "    \"\"\"Compute mean, std, min, max for a list of values.\"\"\"\n",
    "    arr = np.array(values)\n",
    "    return {\n",
    "        'mean': float(np.mean(arr)),\n",
    "        'std': float(np.std(arr)),\n",
    "        'min': float(np.min(arr)),\n",
    "        'max': float(np.max(arr)),\n",
    "        'values': [float(v) for v in values]\n",
    "    }\n",
    "\n",
    "aggregated_results = {}\n",
    "\n",
    "for model_name, results in results_by_model.items():\n",
    "    if not results:\n",
    "        print(f\"⚠️  No results for {model_name}\")\n",
    "        continue\n",
    "    \n",
    "    print(f\"\\n{model_name}:\")\n",
    "    print(\"-\" * 70)\n",
    "    \n",
    "    # Get metric names from first result\n",
    "    metric_names = results[0]['metrics'].keys()\n",
    "    aggregated_results[model_name] = {}\n",
    "    \n",
    "    for metric_name in sorted(metric_names):\n",
    "        # Collect values across seeds\n",
    "        values = [r['metrics'][metric_name] for r in results]\n",
    "        stats = compute_stats(values)\n",
    "        aggregated_results[model_name][metric_name] = stats\n",
    "        \n",
    "        print(f\"  {metric_name:35s}: {stats['mean']:.4f} ± {stats['std']:.4f}\")\n",
    "        print(f\"  {'':35s}  [min: {stats['min']:.4f}, max: {stats['max']:.4f}]\")\n",
    "\n",
    "# Save aggregated results\n",
    "aggregated_file = Path(RESULTS_DIR) / \"aggregated_results.json\"\n",
    "with open(aggregated_file, 'w') as f:\n",
    "    json.dump(aggregated_results, f, indent=2)\n",
    "\n",
    "print(f\"\\n✅ Aggregated results saved to: {aggregated_file}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Summary Table"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "   Models: ['Baseline_Transformer', 'Full_Rank_BBB', 'Low_Rank_Random_Init', 'Low_Rank_SVD_Init', 'Rank1_BBB', 'Deep_Ensemble']\n"
     ]
    }
   ],
   "source": [
    "# Load pre-computed 3-seed results (excluding seed 789)\n",
    "import json\n",
    "\n",
    "with open('multi_seed_results/aggregated_results.json', 'r') as f:\n",
    "    aggregated_results = json.load(f)\n",
    "\n",
    "print(f\"   Models: {list(aggregated_results.keys())}\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "✅ Summary saved to: multi_seed_results/summary_4_seeds.csv\n",
      "\n",
      "================================================================================\n",
      "Summary: Mean ± Std (4 seeds)\n",
      "================================================================================\n",
      "\n",
      "               Model  Accuracy_mean  Accuracy_std  NLL_mean  NLL_std  ECE_mean  ECE_std  AUROC_OOD_STD_mean  AUROC_OOD_STD_std  AUROC_OOD_MI_mean  AUROC_OOD_MI_std  AUPR_In_MI_mean  AUPR_In_MI_std  AUPR_Out_MI_mean  AUPR_Out_MI_std  AUPR_Error_MI_mean  AUPR_Error_MI_std\n",
      "Baseline_Transformer        0.81275      0.006180  0.490577 0.003973  0.132337 0.007563            0.500000           0.000000           0.500000          0.000000         0.102927        0.000000          0.897073         0.000000            0.187500           0.006149\n",
      "       Full_Rank_BBB        0.75250      0.011800  0.552464 0.010743  0.118850 0.005524            0.622637           0.020166           0.622279          0.019984         0.222439        0.028240          0.921157         0.005810            0.417692           0.031902\n",
      "Low_Rank_Random_Init        0.80675      0.003269  0.527173 0.006315  0.173343 0.011413            0.641818           0.027740           0.640739          0.028711         0.302078        0.027119          0.919117         0.012230            0.352602           0.020841\n",
      "   Low_Rank_SVD_Init        0.80075      0.005804  0.516760 0.002279  0.158312 0.006803            0.618942           0.011057           0.616197          0.011834         0.272990        0.012371          0.903111         0.009980            0.382924           0.011858\n",
      "           Rank1_BBB        0.79500      0.013096  0.507728 0.005241  0.148819 0.011604            0.614914           0.015752           0.613073          0.015827         0.272868        0.027665          0.904329         0.005410            0.426752           0.041881\n",
      "       Deep_Ensemble        0.82500      0.006557  0.434229 0.023082  0.065852 0.011141            0.663051           0.008539           0.657712          0.009997         0.267245        0.016177          0.930193         0.002174            0.342975           0.028420\n"
     ]
    }
   ],
   "source": [
    "\n",
    "# Create summary DataFrame\n",
    "summary_data = []\n",
    "\n",
    "for model_name in MODEL_NAMES:\n",
    "    if model_name not in aggregated_results:\n",
    "        continue\n",
    "    \n",
    "    row = {'Model': model_name}\n",
    "    for metric_name, stats in aggregated_results[model_name].items():\n",
    "        row[f\"{metric_name}_mean\"] = stats['mean']\n",
    "        row[f\"{metric_name}_std\"] = stats['std']\n",
    "    summary_data.append(row)\n",
    "\n",
    "summary_df = pd.DataFrame(summary_data)\n",
    "\n",
    "# Save to CSV\n",
    "summary_csv = Path(RESULTS_DIR) / f\"summary_{len(SEEDS)}_seeds.csv\"\n",
    "summary_df.to_csv(summary_csv, index=False)\n",
    "print(f\"\\n✅ Summary saved to: {summary_csv}\")\n",
    "\n",
    "# Display key metrics (using correct metric names from evaluation.py)\n",
    "print(\"\\n\" + \"=\"*80)\n",
    "print(f\"Summary: Mean ± Std ({len(SEEDS)} seeds)\")\n",
    "print(\"=\"*80 + \"\\n\")\n",
    "\n",
    "# Key metrics from evaluation.py\n",
    "key_metrics = ['Accuracy', 'NLL', 'ECE', 'AUROC_OOD_STD', 'AUROC_OOD_MI', \"AUPR_In_MI\",\"AUPR_Out_MI\",\"AUPR_Error_MI\" ]\n",
    "display_cols = ['Model']\n",
    "\n",
    "for m in key_metrics:\n",
    "    mean_col = f\"{m}_mean\"\n",
    "    std_col = f\"{m}_std\"\n",
    "    if mean_col in summary_df.columns:\n",
    "        display_cols.append(mean_col)\n",
    "    if std_col in summary_df.columns:\n",
    "        display_cols.append(std_col)\n",
    "\n",
    "display_cols = [c for c in display_cols if c in summary_df.columns]\n",
    "\n",
    "if len(display_cols) > 1:\n",
    "    print(summary_df[display_cols].to_string(index=False))\n",
    "else:\n",
    "    print(summary_df.to_string(index=False))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Stability Analysis (Coefficient of Variation)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "================================================================================\n",
      "Stability Analysis (CV = std/mean × 100%)\n",
      "================================================================================\n",
      "\n",
      "\n",
      "Baseline_Transformer:\n",
      "----------------------------------------------------------------------\n",
      "  AUPR_Error_MI                      : CV =   3.28% (Stable)\n",
      "  AUPR_Error_STD                     : CV =   3.28% (Stable)\n",
      "  AUPR_In_MI                         : CV =   0.00% (Very Stable)\n",
      "  AUPR_In_STD                        : CV =   0.00% (Very Stable)\n",
      "  AUPR_Out_MI                        : CV =   0.00% (Very Stable)\n",
      "  AUPR_Out_STD                       : CV =   0.00% (Very Stable)\n",
      "  AUPR_Success_MI                    : CV =   0.76% (Very Stable)\n",
      "  AUPR_Success_STD                   : CV =   0.76% (Very Stable)\n",
      "  AUROC_Classification               : CV =   0.45% (Very Stable)\n",
      "  AUROC_OOD_MI                       : CV =   0.00% (Very Stable)\n",
      "  AUROC_OOD_STD                      : CV =   0.00% (Very Stable)\n",
      "  Accuracy                           : CV =   0.76% (Very Stable)\n",
      "  Brier_Score                        : CV =   1.22% (Stable)\n",
      "  ECE                                : CV =   5.71% (Moderate)\n",
      "  Entropy_Ratio                      : CV =   0.26% (Very Stable)\n",
      "  MI_Ratio                           : CV =   0.00% (Very Stable)\n",
      "  Mean_Entropy_In                    : CV =   0.18% (Very Stable)\n",
      "  Mean_Entropy_Out                   : CV =   0.15% (Very Stable)\n",
      "  Mean_MI_In                         : CV =   0.00% (Very Stable)\n",
      "  Mean_MI_Out                        : CV =   0.00% (Very Stable)\n",
      "  Mean_STD_In                        : CV =   0.00% (Very Stable)\n",
      "  Mean_STD_Out                       : CV =   0.00% (Very Stable)\n",
      "  NLL                                : CV =   0.81% (Very Stable)\n",
      "  STD_Ratio                          : CV =   0.00% (Very Stable)\n",
      "\n",
      "Full_Rank_BBB:\n",
      "----------------------------------------------------------------------\n",
      "  AUPR_Error_MI                      : CV =   7.64% (Moderate)\n",
      "  AUPR_Error_STD                     : CV =   8.11% (Moderate)\n",
      "  AUPR_In_MI                         : CV =  12.70% (Variable)\n",
      "  AUPR_In_STD                        : CV =  12.75% (Variable)\n",
      "  AUPR_Out_MI                        : CV =   0.63% (Very Stable)\n",
      "  AUPR_Out_STD                       : CV =   0.63% (Very Stable)\n",
      "  AUPR_Success_MI                    : CV =   0.75% (Very Stable)\n",
      "  AUPR_Success_STD                   : CV =   0.75% (Very Stable)\n",
      "  AUROC_Classification               : CV =   1.41% (Stable)\n",
      "  AUROC_OOD_MI                       : CV =   3.21% (Stable)\n",
      "  AUROC_OOD_STD                      : CV =   3.24% (Stable)\n",
      "  Accuracy                           : CV =   1.57% (Stable)\n",
      "  Brier_Score                        : CV =   2.73% (Stable)\n",
      "  ECE                                : CV =   4.65% (Stable)\n",
      "  Entropy_Ratio                      : CV =   0.50% (Very Stable)\n",
      "  MI_Ratio                           : CV =   5.04% (Moderate)\n",
      "  Mean_Entropy_In                    : CV =   0.81% (Very Stable)\n",
      "  Mean_Entropy_Out                   : CV =   0.71% (Very Stable)\n",
      "  Mean_MI_In                         : CV =  10.92% (Variable)\n",
      "  Mean_MI_Out                        : CV =   9.59% (Moderate)\n",
      "  Mean_STD_In                        : CV =   8.43% (Moderate)\n",
      "  Mean_STD_Out                       : CV =   6.43% (Moderate)\n",
      "  NLL                                : CV =   1.94% (Stable)\n",
      "  STD_Ratio                          : CV =   4.37% (Stable)\n",
      "\n",
      "Low_Rank_Random_Init:\n",
      "----------------------------------------------------------------------\n",
      "  AUPR_Error_MI                      : CV =   5.91% (Moderate)\n",
      "  AUPR_Error_STD                     : CV =   5.43% (Moderate)\n",
      "  AUPR_In_MI                         : CV =   8.98% (Moderate)\n",
      "  AUPR_In_STD                        : CV =   8.85% (Moderate)\n",
      "  AUPR_Out_MI                        : CV =   1.33% (Stable)\n",
      "  AUPR_Out_STD                       : CV =   1.28% (Stable)\n",
      "  AUPR_Success_MI                    : CV =   0.55% (Very Stable)\n",
      "  AUPR_Success_STD                   : CV =   0.56% (Very Stable)\n",
      "  AUROC_Classification               : CV =   0.26% (Very Stable)\n",
      "  AUROC_OOD_MI                       : CV =   4.48% (Stable)\n",
      "  AUROC_OOD_STD                      : CV =   4.32% (Stable)\n",
      "  Accuracy                           : CV =   0.41% (Very Stable)\n",
      "  Brier_Score                        : CV =   1.71% (Stable)\n",
      "  ECE                                : CV =   6.58% (Moderate)\n",
      "  Entropy_Ratio                      : CV =   0.26% (Very Stable)\n",
      "  MI_Ratio                           : CV =   7.56% (Moderate)\n",
      "  Mean_Entropy_In                    : CV =   0.65% (Very Stable)\n",
      "  Mean_Entropy_Out                   : CV =   0.74% (Very Stable)\n",
      "  Mean_MI_In                         : CV =   4.64% (Stable)\n",
      "  Mean_MI_Out                        : CV =   6.73% (Moderate)\n",
      "  Mean_STD_In                        : CV =   4.33% (Stable)\n",
      "  Mean_STD_Out                       : CV =   4.79% (Stable)\n",
      "  NLL                                : CV =   1.20% (Stable)\n",
      "  STD_Ratio                          : CV =   4.71% (Stable)\n",
      "\n",
      "Low_Rank_SVD_Init:\n",
      "----------------------------------------------------------------------\n",
      "  AUPR_Error_MI                      : CV =   3.10% (Stable)\n",
      "  AUPR_Error_STD                     : CV =   2.98% (Stable)\n",
      "  AUPR_In_MI                         : CV =   4.53% (Stable)\n",
      "  AUPR_In_STD                        : CV =   4.54% (Stable)\n",
      "  AUPR_Out_MI                        : CV =   1.11% (Stable)\n",
      "  AUPR_Out_STD                       : CV =   1.07% (Stable)\n",
      "  AUPR_Success_MI                    : CV =   0.41% (Very Stable)\n",
      "  AUPR_Success_STD                   : CV =   0.40% (Very Stable)\n",
      "  AUROC_Classification               : CV =   0.40% (Very Stable)\n",
      "  AUROC_OOD_MI                       : CV =   1.92% (Stable)\n",
      "  AUROC_OOD_STD                      : CV =   1.79% (Stable)\n",
      "  Accuracy                           : CV =   0.72% (Very Stable)\n",
      "  Brier_Score                        : CV =   0.59% (Very Stable)\n",
      "  ECE                                : CV =   4.30% (Stable)\n",
      "  Entropy_Ratio                      : CV =   0.11% (Very Stable)\n",
      "  MI_Ratio                           : CV =   8.56% (Moderate)\n",
      "  Mean_Entropy_In                    : CV =   0.68% (Very Stable)\n",
      "  Mean_Entropy_Out                   : CV =   0.58% (Very Stable)\n",
      "  Mean_MI_In                         : CV =  13.25% (Variable)\n",
      "  Mean_MI_Out                        : CV =  12.52% (Variable)\n",
      "  Mean_STD_In                        : CV =   3.30% (Stable)\n",
      "  Mean_STD_Out                       : CV =   4.43% (Stable)\n",
      "  NLL                                : CV =   0.44% (Very Stable)\n",
      "  STD_Ratio                          : CV =   3.03% (Stable)\n",
      "\n",
      "Rank1_BBB:\n",
      "----------------------------------------------------------------------\n",
      "  AUPR_Error_MI                      : CV =   9.81% (Moderate)\n",
      "  AUPR_Error_STD                     : CV =   9.79% (Moderate)\n",
      "  AUPR_In_MI                         : CV =  10.14% (Variable)\n",
      "  AUPR_In_STD                        : CV =  10.10% (Variable)\n",
      "  AUPR_Out_MI                        : CV =   0.60% (Very Stable)\n",
      "  AUPR_Out_STD                       : CV =   0.59% (Very Stable)\n",
      "  AUPR_Success_MI                    : CV =   0.50% (Very Stable)\n",
      "  AUPR_Success_STD                   : CV =   0.49% (Very Stable)\n",
      "  AUROC_Classification               : CV =   0.38% (Very Stable)\n",
      "  AUROC_OOD_MI                       : CV =   2.58% (Stable)\n",
      "  AUROC_OOD_STD                      : CV =   2.56% (Stable)\n",
      "  Accuracy                           : CV =   1.65% (Stable)\n",
      "  Brier_Score                        : CV =   1.61% (Stable)\n",
      "  ECE                                : CV =   7.80% (Moderate)\n",
      "  Entropy_Ratio                      : CV =   0.31% (Very Stable)\n",
      "  MI_Ratio                           : CV =   7.95% (Moderate)\n",
      "  Mean_Entropy_In                    : CV =   0.42% (Very Stable)\n",
      "  Mean_Entropy_Out                   : CV =   0.48% (Very Stable)\n",
      "  Mean_MI_In                         : CV =   9.53% (Moderate)\n",
      "  Mean_MI_Out                        : CV =   7.09% (Moderate)\n",
      "  Mean_STD_In                        : CV =   1.06% (Stable)\n",
      "  Mean_STD_Out                       : CV =   2.53% (Stable)\n",
      "  NLL                                : CV =   1.03% (Stable)\n",
      "  STD_Ratio                          : CV =   3.53% (Stable)\n",
      "\n",
      "Deep_Ensemble:\n",
      "----------------------------------------------------------------------\n",
      "  AUPR_Error_MI                      : CV =   8.29% (Moderate)\n",
      "  AUPR_Error_STD                     : CV =   9.42% (Moderate)\n",
      "  AUPR_In_MI                         : CV =   6.05% (Moderate)\n",
      "  AUPR_In_STD                        : CV =   6.96% (Moderate)\n",
      "  AUPR_Out_MI                        : CV =   0.23% (Very Stable)\n",
      "  AUPR_Out_STD                       : CV =   0.20% (Very Stable)\n",
      "  AUPR_Success_MI                    : CV =   0.42% (Very Stable)\n",
      "  AUPR_Success_STD                   : CV =   0.38% (Very Stable)\n",
      "  AUROC_Classification               : CV =   0.26% (Very Stable)\n",
      "  AUROC_OOD_MI                       : CV =   1.52% (Stable)\n",
      "  AUROC_OOD_STD                      : CV =   1.29% (Stable)\n",
      "  Accuracy                           : CV =   0.79% (Very Stable)\n",
      "  Brier_Score                        : CV =   2.48% (Stable)\n",
      "  ECE                                : CV =  16.92% (Variable)\n",
      "  Entropy_Ratio                      : CV =   2.24% (Stable)\n",
      "  MI_Ratio                           : CV =   3.32% (Stable)\n",
      "  Mean_Entropy_In                    : CV =   7.09% (Moderate)\n",
      "  Mean_Entropy_Out                   : CV =   5.43% (Moderate)\n",
      "  Mean_MI_In                         : CV =   4.90% (Stable)\n",
      "  Mean_MI_Out                        : CV =   7.84% (Moderate)\n",
      "  Mean_STD_In                        : CV =   3.67% (Stable)\n",
      "  Mean_STD_Out                       : CV =   4.54% (Stable)\n",
      "  NLL                                : CV =   5.32% (Moderate)\n",
      "  STD_Ratio                          : CV =   2.75% (Stable)\n",
      "\n",
      "✅ Stability analysis saved to: multi_seed_results/stability_analysis.csv\n"
     ]
    }
   ],
   "source": [
    "print(\"\\n\" + \"=\"*80)\n",
    "print(\"Stability Analysis (CV = std/mean × 100%)\")\n",
    "print(\"=\"*80 + \"\\n\")\n",
    "\n",
    "stability_data = []\n",
    "\n",
    "for model_name in MODEL_NAMES:\n",
    "    if model_name not in aggregated_results:\n",
    "        continue\n",
    "    \n",
    "    print(f\"\\n{model_name}:\")\n",
    "    print(\"-\" * 70)\n",
    "    \n",
    "    row = {'Model': model_name}\n",
    "    \n",
    "    for metric_name, stats in aggregated_results[model_name].items():\n",
    "        cv = (stats['std'] / abs(stats['mean']) * 100) if stats['mean'] != 0 else 0\n",
    "        row[f\"{metric_name}_CV\"] = cv\n",
    "        \n",
    "        # Stability interpretation\n",
    "        if cv < 1:\n",
    "            stability = \"Very Stable\"\n",
    "        elif cv < 5:\n",
    "            stability = \"Stable\"\n",
    "        elif cv < 10:\n",
    "            stability = \"Moderate\"\n",
    "        else:\n",
    "            stability = \"Variable\"\n",
    "        \n",
    "        print(f\"  {metric_name:35s}: CV = {cv:6.2f}% ({stability})\")\n",
    "    \n",
    "    stability_data.append(row)\n",
    "\n",
    "# Save stability analysis\n",
    "stability_df = pd.DataFrame(stability_data)\n",
    "stability_csv = Path(RESULTS_DIR) / \"stability_analysis.csv\"\n",
    "stability_df.to_csv(stability_csv, index=False)\n",
    "print(f\"\\n✅ Stability analysis saved to: {stability_csv}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Training Time Analysis"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "================================================================================\n",
      "Training Time Analysis\n",
      "================================================================================\n",
      "\n",
      "Baseline_Transformer          : 0.13 ± 0.02 hours\n",
      "                                (462 ± 86 seconds)\n",
      "Full_Rank_BBB                 : 0.38 ± 0.07 hours\n",
      "                                (1385 ± 244 seconds)\n",
      "Low_Rank_Random_Init          : 0.14 ± 0.01 hours\n",
      "                                (493 ± 49 seconds)\n",
      "Low_Rank_SVD_Init             : 0.13 ± 0.01 hours\n",
      "                                (477 ± 49 seconds)\n",
      "Rank1_BBB                     : 0.21 ± 0.05 hours\n",
      "                                (751 ± 188 seconds)\n",
      "Deep_Ensemble                 : 1.08 ± 0.07 hours\n",
      "                                (3883 ± 266 seconds)\n",
      "\n",
      "Total training time (new models): 8.28 hours (0.34 days)\n"
     ]
    }
   ],
   "source": [
    "print(\"\\n\" + \"=\"*80)\n",
    "print(\"Training Time Analysis\")\n",
    "print(\"=\"*80 + \"\\n\")\n",
    "\n",
    "time_by_model = {model: [] for model in MODEL_NAMES}\n",
    "\n",
    "for key, result in tracker.results.items():\n",
    "    if not result.get('reused', False):  # Only count actual training time\n",
    "        model_name = result['model']\n",
    "        time_by_model[model_name].append(result['training_time'])\n",
    "\n",
    "for model_name in MODEL_NAMES:\n",
    "    times = time_by_model[model_name]\n",
    "    if times:\n",
    "        stats = compute_stats(times)\n",
    "        print(f\"{model_name:30s}: {stats['mean']/3600:.2f} ± {stats['std']/3600:.2f} hours\")\n",
    "        print(f\"{'':30s}  ({stats['mean']:.0f} ± {stats['std']:.0f} seconds)\")\n",
    "    else:\n",
    "        print(f\"{model_name:30s}: (no new training - reused existing)\")\n",
    "\n",
    "# Total time excluding reused models\n",
    "total_new_time = sum([t for k, t in tracker.progress['timings'].items() if t > 0])\n",
    "print(f\"\\nTotal training time (new models): {total_new_time/3600:.2f} hours ({total_new_time/86400:.2f} days)\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Visualization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_85410/506408213.py:44: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df_plot, x='Model', y='Value', ax=ax, palette='Set2')\n",
      "/tmp/ipykernel_85410/506408213.py:44: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df_plot, x='Model', y='Value', ax=ax, palette='Set2')\n",
      "/tmp/ipykernel_85410/506408213.py:44: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df_plot, x='Model', y='Value', ax=ax, palette='Set2')\n",
      "/tmp/ipykernel_85410/506408213.py:44: FutureWarning: \n",
      "\n",
      "Passing `palette` without assigning `hue` is deprecated and will be removed in v0.14.0. Assign the `x` variable to `hue` and set `legend=False` for the same effect.\n",
      "\n",
      "  sns.boxplot(data=df_plot, x='Model', y='Value', ax=ax, palette='Set2')\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "✅ Visualization saved to: multi_seed_results/results_across_4_seeds.png\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAB8YAAAHpCAYAAADwLVaJAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjYsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvq6yFwwAAAAlwSFlzAAAPYQAAD2EBqD+naQABAABJREFUeJzs3X1ck/X+P/DXNhkIG7cDVDBFRVDuxEzDKLI8amSmcrQ7Qv1aWh7Ljnos7VhRp+icr1qaYXdoopSnwr5ZctSytFDCTFQkRcU7RMUNGIzbwbbfH/zYYW7cDMY24PV8PHrIrrvP+1qD69r1/nzeH4FOp9OBiIiIiIiIiIiIiIiIiIiohxLaOgAiIiIiIiIiIiIiIiIiIqKuxMQ4ERERERERERERERERERH1aEyMExERERERERERERERERFRj8bEOBERERERERERERERERER9WhMjBMRERERERERERERERERUY/GxDgREREREREREREREREREfVoTIwTEREREREREREREREREVGPxsQ4ERERERERERERERERERH1aEyMExERERERERERERERERFRj8bEOBERERERERERERERERER9Wh9bB0AEXXMK6+8gn//+9/618uWLcOCBQtsGBERERHZg/feew8bN27Uv/7oo48QExOjf/3SSy/h66+/BgC89tpreOyxxwyWLV68GM8991yLxw8KCtL/vH//fvj7+1v6FIiIiHqtW6/jt5JKpTh69KjBssOHD+OLL77A8ePHUVJSAolEAj8/P9x7772YOXMmBgwYAAC47777UFRU1OKxExIS8PLLL1vmRIiIiG7RnufZV69exf33369/nZ+fb7D+ySefxJEjRwAASUlJmDlzJgDD77lN+vTpAw8PD4waNQpPPfUURo0aZTKuo0ePIi0tDceOHUNJSQmcnJwwePBg3H///XjyySchkUhM7qdUKrF9+3YcOHAAly9fRl1dHfr164fg4GBMmzYN999/PwQCQZvvS2FhIT799FMcOnQIN27cAAD069cP48ePx7x58zBw4ECT+1VWVmL79u344YcfcOnSJdTW1sLLywujR4/G448/jjvuuMNg+1vfW6FQCLFYDHd3dwwePBj33nsvZs2a1eL5mnL+/Hls2rQJ2dnZKCsrg4uLCzw8PDB8+HCMGzcO8fHxAAyfI7Rl//79KCoqQkJC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kcH6ZcuWYcGCBe2KmddxIiKyFXu7jndXvJYTEZGt8FreebyOExGRrbTnOs7EuAlNvQnCwsIgEolsHA0REfUGGo0Gubm5dtUzPSMjA0lJSUhMTERERAS2bt2K+fPnY8+ePfDy8jK5j0QiwZ49e/Svb+0dnpmZafD6559/xssvv4zJkycbLH/++ecxe/Zs/WsXF5d2x83rOBERWZs9Xse7M17LiYjI2ngttxxex4mIyNrMuY4zMW5C00N8kUjEizcREVmVPZUZ27JlC2bPno24uDgAQGJiIg4cOID09PQWR28LBAJ4e3u3eMxb1+3fvx/jxo3DwIEDDZa7uLi0epzW2NN7SEREvQuvQZbB7+RERGQrvJZ3Hq/jRERkK+25jjMxTkREREbUajXy8vKwcOFC/TKhUIjx48cjJyenxf2qq6sxYcIEaLVajBw5EkuXLkVgYKDJbRUKBQ4ePIi3337baN3HH3+MTZs2oX///pg6dSrmzp2LPn3Mu21h6TYiIiIiIiIiIiIiasLEOBERERkpKyuDRqMxKpnu5eWFCxcumNwnICAAb731FoKCgqBSqbB582Y8+uij2L17N/r162e0/ddffw0XFxdMmjTJYPmTTz6JkSNHws3NDTk5OVi3bh3kcrnR/OVtYdk2IiKylqaybURERERERERkv5gYJyIiIouIjIxEZGSkwevY2Fjs2LEDL7zwgtH26enpeOihh+Do6GiwfN68efqfg4OD4eDggFdffRXLli2DWCxudzws20ZERERERERERERETdqehZyIiIh6HQ8PD4hEIpSUlBgsLykpgUwma9cxHBwcMGLECFy5csVo3dGjR3Hx4kXMmjWrzeNERESgoaEBV69ebV/wRERERERERERERES3YGKciIiIjIjFYoSEhCArK0u/TKvVIisry2BUeGs0Gg3Onj0Lb29vo3VfffUVQkJCEBwc3OZxTp8+DaFQaFTWnYiIiIiIiIiIiIiovVhKnYiIiEyaN28eXnzxRYSGhiI8PBxbt25FTU0NZs6cCQBYsWIFfH19sWzZMgDAxo0bMWrUKAwaNAgVFRVISUnBtWvXjEaFV1ZWYs+ePXjxxReN2szJycGJEydw5513wsXFBTk5OUhKSsK0adPg5ubW9SdNRERERERERERERD0SE+NERERkUmxsLEpLS7FhwwbI5XKMGDECn3zyib6U+vXr1yEU/rf4TEVFBVavXg25XA43NzeEhIRgx44dGDZsmMFxd+/eDZ1Oh6lTpxq1KRaLkZGRgY0bN0KtVsPf3x9z5841mHeciIiIiIiIiIiIiMhcAp1Op7N1EPZGo9Hg+PHjGDVqFEQika3DISKiXoDXHsvhe0lERNbGa49l8f0kIiJr47XHcvheEhGRtZlz7eEc40RERERERERERERERERE1KMxMU5ERERERERERERERERERD0aE+NERERERERERERERERERNSj9bF1AERERNaiUCiQkZGB7OxsAMC4ceMQGxsLmUxm48iIeg+FQoHMzEwoFArIZDJER0fzd5CIiIjsHu9hiIiIiIj+q7veHzMxTkREvUJWVhb+9a9/IT8/HzqdDgDw008/4f/+7//wt7/9DVFRUTaOkKjny8rKQmpqKrRarX7Z3r17kZCQwN9BIiIislu8hyEiIiIi+q/ufH/MUupERNTjKRQKfPjhhwZJcQDQ6XQ4c+YMPvroIygUChtGSNTzKRQKoxtmANBqtUhNTeXvIBEREdkl3sMQEREREf1Xd78/5ohxIiKyKwqFAiqVyuz9SktLUVNTY3Ldzz//jD/++APV1dUm1+fl5eGjjz7CPffc0+72+vbtC09PT7PjlEql3aKkDFF7FBcXt/h7dat9+/ahvLy8xfXp6emYNGlSq8dwdnaGr6+vWTESERERmePW+xtL3MO0hPc2RERERGQv2vucr7s/42NinIiI7IZCocDy5cugVtdb9LgFBQW4dOkS6urqTK6vrq7GN998g1OnTlm0XVPEYgesWbOWyXHq9ioqKrB06VKDKgytKSgoQElJSavrDx482OoxhEIhkpOT4erqalasRERERO1h6v7GEvcwLeG9DRERERHZA3Oe83X3Z3xMjBMRkd1QqVRQq+vx+JQq+Hhq296hmfJKAerUApPrsn7XIaMKUJSaPqbMC5h4pw5Rt1e1uz1HsQ5ukvYlBJvcLBXisz0uUKlUTIxTt+fq6op169aZNWL84MGDqKqqQl5eHkJCQuDi4qJfHxMT067epHxwTERERF3F1P1N0z1MS2JiYhASEoLk5GQsWrQIfn5+7W6P9zZE1Ja0tDSkpKRALpcjODgYq1evRnh4eIvbV1RU4J133sH3338PpVIJPz8/rFq1CjExMQCA++67D0VFRUb7Pf7443j11VcB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7969OnTokCSpXbt2CgoKcm5AAADAyu7EeOfOnZWWllZu1nhaWpo6duzosMAAAIDzxcbGVrh0uiStXLnykm3nzJlTYfk999yje+65p8JjoaGhevfdd+0LEgAAAHbLz8/XpEmTVFZWdkXtFy9erMzMTOXk5FRaJzMzU5s3b5YkGQwGLV682LqqEADUNzt27NATTzyh/fv3W3+2uri4qGPHjnr22WcVHh7u5AgBAIDdifGHHnpIjz76qA4fPqybb75Z0vn9pD744AP2FwcAAAAAAKgDmjZtqvnz59s9Y/xiGzdutCa+K9KnTx+bGeMkxQHUV/v379fo0aPVoUMHvfjii+rQoYO1/M0339SYMWP07rvvMrEMAAAnszsx3rdvX7366qt6/fXX9dFHH6lx48YKDg5WUlISe38CAAAAAADUEVe7tHlMTIy2b99ebo9x6fwM8ZiYGPYYB9AgLFq0SLfccosWLVpks6d4SEiIBg4cqEceeUSLFi3Syy+/bFe/q1at0rJly5Sdna3g4GDNmDGj0pnna9as0bRp02zK3NzctHPnTuvrypZ1/+c//6n77rtP0vnP/y9sm3FBfHy8xo0bZ1fsAADURnYnxiXpz3/+s/785z9Lkk6dOqUPPvhAzz//vHbv3q09e/Y4Mj4AAAAAAADUQiaTSaNGjVJycrJNctxgMGj06NEkxQE0GNu2bdOSJUtskuIXuLi4aPz48XYnltPS0pSYmKiEhARFREQoKSlJcXFx2rBhg/z8/Cps06RJE23YsMHm3BfbunWrzesvvvhCTzzxhO644w6b8scee0wjRoywvvb09LQrdgAAaqsrSoxL0rfffquUlBRt3LhRzZs31+23366ZM2c6MjYAAAAAAADUYj179lSnTp20detWmc1mmUwm9erVi6Q4gAalsLDwkj/3/P39VVhYaFefK1as0IgRIxQTEyNJSkhI0KZNm5Samlppkt3FxUX+/v6XjONin376qW666Sa1bt3aptzT0/OS/VSFxWK5qvYAAFSVPWOOXYnx7OxsrV27VikpKTp16pTuuusuFRcX69VXX2V/FAAAAAAAgAbIZDJpyJAhzg4DAJymZcuW2rFjh6699toKj//www9q2bJllfsrLi7W7t27NX78eGuZwWBQZGSkMjIyKm13+vRp3XbbbSotLVWXLl00adIkderUqcK6ZrNZmzdv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KgAAAAAANFCrVq1S3759FRYWpuHDh2vHjh2XrP/hhx/qzjvvVFhYmO6++25t3ry5XJ3MzEw98MAD+tOf/qRu3bopJiZGx44dq65LAAAAAACgSkiMAwAAAADQAKWlpSkxMVEPP/yw1q5dq+DgYMXFxSknJ6fC+v/9738VHx+vYcOGad26derXr58efvhh7du3z1rn119/1d/+9je1b99eK1eu1HvvvaeHHnpIjRs3rqnLAgAAAACgQiTGAQAAAABogFasWKERI0YoJiZGHTt2VEJCgtzd3ZWamlph/eTkZPXu3Vv33XefOnTooIkTJ6pLly566623rHVeeukl3XrrrZoyZYq6dOmi6667Tv369ZOfn19NXRYAAAAAABVij3EAAAAAABqY4uJi7d69W+PHj7eWGQwGRUZGKiMjo8I227dv15gxY2zKevXqpU8++USSVFpaqk2bNum+++5TXFycfvzxR7Vq1Urjx49XVFSU3TFaLBa72wAA6o7S0lLrV2f/zHf2+QEAQM0gMQ4AAAAAQANz4sQJWSyWcjO5/fz8dODAgQrbmM1mmUymcvXNZrMkKScnR6dPn9aSJUs0ceJETZ48WVu2bNEjjzyi5ORk3XjjjXbFuHPnTrvqAwDqlqysLEnS3r17lZeX59xgAABAg0BiHAAAAAAAXLULM//69etnnVkeEhKi//73v1q9erXdifGwsDAZjUZHhwkAqCUOHTokSQoKClLbtm2dGovFYuGBLAAAGgAS4wAAAAAANDA+Pj4yGo3KycmxKc/JySk3K/wCk8lknR1eUX0fHx81atRIHTp0sKnToUMHff/993bHaDQaSYwDQD1mMBisX/l5DwAAaoLB2QEAAAAAAICa5ebmptDQUKWnp1vLSktLlZ6eru7du1fYplu3bvr6669tyr766it169bN2mdYWJgOHjxoU+fQoUMKDAx07AUAAAAAAGAnEuMAAAAAADRAY8eO1bvvvqu1a9cqMzNTs2bNUlFRkYYOHSpJmjJliubNm2etP2rUKG3ZskXLly9XZmamFi1apF27dik2NtZaJy4uTh9++KHeffdd/fLLL3rrrbf0+eef6957763x6wMAAAAA4GIspQ4AAAAAQAMUHR2t3NxcLVy4UNnZ2QoJCdHSpUutS6MfP37cusytJF1//fWaO3euFixYoPnz56tt27Z69dVX1blz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xnLHYRwHKsdYfmmM5bhSjOOOxVgOVIxx/NIawjjOjPFqUFJSoqKiIjVt2lR79+7Vvffeq7/85S+Kjo7W4sWLrfU+++wz3X333Ro8eLAGDhyoTz75RJKUnZ2tCRMmaNiwYbr77rv10ksvVXiekSNHWts8/vjjmjlzpkaPHq077rhDjzzyiIqLi63xzJ07V8OGDdPgwYM1YcIEnTx5sprvguNs27ZNgwcPtr7et2+f+vbt68SInG/Lli36y1/+orvvvluxsbHav3+/JCk+Pl7vv/++JGnVqlXq2rWr9enKUaNG6dtvv3VazNWtId6TgIAAtW/fXkePHlV6erruueceDRkyRAMGDNC///1va71L/Xy42P79+zVw4EBt3rxZknTDDTeoRYsWNXY91aF///7661//qmXLll3yZ+GpU6f05JNPWn/uzpgxw3qPRo4cqdmzZysmJka333675syZo7KyMmdeFqoZ47jjMZbbaohjVlU0tPvCOH55jOO4UozljsU4Xl5DG7OqoiHeE8byy2Msx5VgHHc8xnJbDXHMupyGeE8Yxy+vvo7jJMYdaOLEiRo8eLBuueUWGQwG3XXXXQoMDFRSUpLWrl2rNWvWaOPGjdq+fbuk809ePP300/rPf/6j9957TzfeeKOk8//R/v73vyslJUVr167Vrl279OGHH172/Hv27NHrr7+utLQ0mc1mbdy4UZK0bNkyXXPNNUpJSdF//vMfde7cWQsWLKiu2+AQF+7l4MGD69XSLY6Qk5OjyZMna86cOXr//fc1YsQIPfbYYyorK1PPnj311VdfSZK++uorde3aVd9++62Kioq0d+9edevWzbnBV5OGek8yMzOVl5enm266SV26dNG//vUvrVu3TqtWrdLixYv122+/WetW9vPhgm3btumxxx7T888/rz59+tT0pVSriIgI7d+//5I/C59//nn16NFDKSkpeu+991RaWqrk5GRrH/v379fq1av13nvv6ZtvvtEHH3zgpKtBdWIcdyzG8oo11DHrchrifWEcrxrGcdiDsdxxGMcr1xDHrMtpqPeEsbxqGMtRVYzjjsVYXrGGOmZdSkO9J4zjVVMfx3GWUnegC0sMnDt3TjNnztTcuXN1//33KyEhQT/99JNcXFz022+/ac+ePerWrZt69uypZ599VnfccYd69eqlkJAQnT59Wunp6TKbzdZ+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uGeix6sGa7QVmViRSeno8spHl3seerxqsD5bkYkVmZSOHq98NanHOTFeipSUFEnSpUuXNHbsWPfP93M6nfL1rdgYT58+raysLH322Wdl2q8y5lKZ494vMTFRwcHBunbtmqKiotS7d2+FhoZW6nfWdGRiRSalu5dPUVGRZs2apT179ugnP/mJgoKC3OvX1atXFRkZqaioKD3++OOSpOnTp2vatGmSpPfee0/r16/Xhg0bquswirl3TMCjossrd9x7WJ+tyMSKTEpHjwNW9Hjljns/1mgrMrEik9LR5UBx9Hjljns/1mcrMrEik9LR43UHJ8YfweDBgzVy5EgdOXJErVu3Vnx8vBYuXKhbt26poKBA4eHhWrZsmby9vZWcnKzU1FQFBgbq3Llz8vPz0/r169WqVStlZWUpPj5e+fn5MsZo8ODBGjNmjBYtWqRr164pJiZGL730koYOHarVq1crIyNDkjRixAjNnTtXkvTiiy+qQ4cOysjIkL+/v8aOHev+vjNnzqhhw4ZavXq13nnnHX399dd64okntHHjRj322GNyOBxav3690tPT5XA49PTTT2vlypVq1KiR4uPj5eXlJbvdruvXr2vfvn1Vkm3z5s3Vpk0bXb58Wbdu3VJiYqIKCgrkcDg0bdo0Pffcc5LuXoljs9mUnZ2tq1evql27dnr77bdls9mKjXf+/HktWLBAixcv1qBBg6rkGCoamViRSekcDodu376thg0bWrbl5eUpICBAfn5+lm3GGOXl5alRo0ZVMc1H1qFDB/3iF7/QgQMHdOPGDf385z/XuHHjVFRUpNWrVystLU1+fn7y9fXVRx99JH9/fx08eFCbN29WQUGBvL29tWjRIvXu3VtHjhzRqlWrFBYWphMnTkiS1q5dq+3btyszM1P16tXTpk2b1Lx5c0l385s1a5bsdruaNGmit956S0899ZRljnv37tWHH34op9OpgIAALV++XM8880yV5oTS0eWVg/XZikysyKR09Dg9jgejxysPa7QVmViRSenocrocpaPHKw/rsxWZWJFJ6ejxOtDjBg908eJFExYW5v45IiLCLF261BQVFRljjLlz547Jy8szxhjjdDrNzJkzzSeffGKMMWb37t2me/fuxm63G2OMWbt2rVm+fLkxxphVq1aZLVu2uMf973//a4wxJj093URHR7vfX7NmjVm4cKFx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",
      "text/plain": [
       "<Figure size 2000x500 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "\n",
    "sns.set_style(\"whitegrid\")\n",
    "plt.rcParams['figure.dpi'] = 100\n",
    "\n",
    "# Select key metrics (using correct names from evaluation.py)\n",
    "key_metrics = ['Accuracy', 'NLL', 'ECE', 'AUROC_OOD_STD']\n",
    "\n",
    "# Check which metrics are available\n",
    "available_metrics = []\n",
    "if aggregated_results:\n",
    "    first_model = list(aggregated_results.keys())[0]\n",
    "    available_metrics = [m for m in key_metrics if m in aggregated_results[first_model]]\n",
    "\n",
    "if not available_metrics:\n",
    "    print(\"⚠️  No metrics available for visualization\")\n",
    "    print(f\"Available metrics in results: {list(aggregated_results[first_model].keys()) if aggregated_results else 'None'}\")\n",
    "else:\n",
    "    n_metrics = len(available_metrics)\n",
    "    fig, axes = plt.subplots(1, n_metrics, figsize=(5*n_metrics, 5))\n",
    "    if n_metrics == 1:\n",
    "        axes = [axes]\n",
    "    \n",
    "    for idx, metric in enumerate(available_metrics):\n",
    "        ax = axes[idx]\n",
    "        \n",
    "        # Prepare data\n",
    "        plot_data = []\n",
    "        for model_name in MODEL_NAMES:\n",
    "            if model_name in aggregated_results and metric in aggregated_results[model_name]:\n",
    "                stats = aggregated_results[model_name][metric]\n",
    "                for seed_idx, value in enumerate(stats['values']):\n",
    "                    plot_data.append({\n",
    "                        'Model': model_name.replace('_', '\\n'),\n",
    "                        'Value': value,\n",
    "                        'Seed': SEEDS[seed_idx] if seed_idx < len(SEEDS) else seed_idx\n",
    "                    })\n",
    "        \n",
    "        if plot_data:\n",
    "            df_plot = pd.DataFrame(plot_data)\n",
    "            \n",
    "            # Box plot with points\n",
    "            sns.boxplot(data=df_plot, x='Model', y='Value', ax=ax, palette='Set2')\n",
    "            sns.stripplot(data=df_plot, x='Model', y='Value', ax=ax,\n",
    "                         color='black', alpha=0.6, size=6, jitter=True)\n",
    "            \n",
    "            ax.set_title(metric.replace('_', ' '), fontsize=12, fontweight='bold')\n",
    "            ax.set_xlabel('')\n",
    "            ax.set_ylabel(metric.replace('_', ' '), fontsize=10)\n",
    "            ax.tick_params(axis='x', rotation=0, labelsize=8)\n",
    "    \n",
    "    plt.tight_layout()\n",
    "    plot_file = Path(RESULTS_DIR) / f\"results_across_{len(SEEDS)}_seeds.png\"\n",
    "    plt.savefig(plot_file, dpi=300, bbox_inches='tight')\n",
    "    print(f\"\\n✅ Visualization saved to: {plot_file}\")\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Export for LaTeX"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "================================================================================\n",
      "LaTeX Table\n",
      "================================================================================\n",
      "\n",
      "Baseline Transformer & $0.813 \\pm 0.006$ & $0.491 \\pm 0.004$ & $0.132 \\pm 0.008$ & $0.500 \\pm 0.000$ \\\\\n",
      "Full Rank BBB & $0.752 \\pm 0.012$ & $0.552 \\pm 0.011$ & $0.119 \\pm 0.006$ & $0.623 \\pm 0.020$ \\\\\n",
      "Low Rank Random Init & $0.807 \\pm 0.003$ & $0.527 \\pm 0.006$ & $0.173 \\pm 0.011$ & $0.642 \\pm 0.028$ \\\\\n",
      "Low Rank SVD Init & $0.801 \\pm 0.006$ & $0.517 \\pm 0.002$ & $0.158 \\pm 0.007$ & $0.619 \\pm 0.011$ \\\\\n",
      "Rank1 BBB & $0.795 \\pm 0.013$ & $0.508 \\pm 0.005$ & $0.149 \\pm 0.012$ & $0.615 \\pm 0.016$ \\\\\n",
      "Deep Ensemble & $0.825 \\pm 0.007$ & $0.434 \\pm 0.023$ & $0.066 \\pm 0.011$ & $0.663 \\pm 0.009$ \\\\\n",
      "\n",
      "✅ LaTeX table saved to: multi_seed_results/latex_table.txt\n"
     ]
    },
    {
     "ename": "",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31mThe Kernel crashed while executing code in the current cell or a previous cell. \n",
      "\u001b[1;31mPlease review the code in the cell(s) to identify a possible cause of the failure. \n",
      "\u001b[1;31mClick <a href='https://aka.ms/vscodeJupyterKernelCrash'>here</a> for more info. \n",
      "\u001b[1;31mView Jupyter <a href='command:jupyter.viewOutput'>log</a> for further details."
     ]
    }
   ],
   "source": [
    "print(\"\\n\" + \"=\"*80)\n",
    "print(\"LaTeX Table\")\n",
    "print(\"=\"*80 + \"\\n\")\n",
    "\n",
    "latex_rows = []\n",
    "# Use correct metric names from evaluation.py\n",
    "metrics_to_show = ['Accuracy', 'NLL', 'ECE', 'AUROC_OOD_STD']\n",
    "\n",
    "for model_name in MODEL_NAMES:\n",
    "    if model_name not in aggregated_results:\n",
    "        continue\n",
    "    \n",
    "    model_latex = model_name.replace('_', ' ')\n",
    "    row_parts = [model_latex]\n",
    "    \n",
    "    for metric in metrics_to_show:\n",
    "        if metric in aggregated_results[model_name]:\n",
    "            stats = aggregated_results[model_name][metric]\n",
    "            formatted = f\"${stats['mean']:.3f} \\\\pm {stats['std']:.3f}$\"\n",
    "            row_parts.append(formatted)\n",
    "        else:\n",
    "            row_parts.append(\"-\")\n",
    "    \n",
    "    latex_row = \" & \".join(row_parts) + \" \\\\\\\\\"\n",
    "    latex_rows.append(latex_row)\n",
    "    print(latex_row)\n",
    "\n",
    "# Save\n",
    "latex_file = Path(RESULTS_DIR) / \"latex_table.txt\"\n",
    "with open(latex_file, 'w') as f:\n",
    "    f.write(\"\\\\begin{table}[h]\\n\")\n",
    "    f.write(\"\\\\centering\\n\")\n",
    "    f.write(\"\\\\begin{tabular}{lcccc}\\n\")\n",
    "    f.write(\"\\\\toprule\\n\")\n",
    "    f.write(\"Model & Accuracy & NLL & ECE & OOD AUROC \\\\\\\\\\n\")\n",
    "    f.write(\"\\\\midrule\\n\")\n",
    "    f.write(\"\\n\".join(latex_rows) + \"\\n\")\n",
    "    f.write(\"\\\\bottomrule\\n\")\n",
    "    f.write(\"\\\\end{tabular}\\n\")\n",
    "    f.write(f\"\\\\caption{{Model comparison across {len(SEEDS)} seeds (mean $\\\\pm$ std)}}\\n\")\n",
    "    f.write(\"\\\\label{tab:robustness_check}\\n\")\n",
    "    f.write(\"\\\\end{table}\\n\")\n",
    "\n",
    "print(f\"\\n✅ LaTeX table saved to: {latex_file}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Final Summary\n",
    "\n",
    "**Completed!**\n",
    "\n",
    "This notebook has:\n",
    "1. ✅ Run all 6 models with 4 different seeds\n",
    "2. ✅ Saved checkpoints after each experiment (resume-capable)\n",
    "3. ✅ Managed memory aggressively (10GB/11GB GPU)\n",
    "4. ✅ Reused existing seed=42 models when available\n",
    "5. ✅ Computed mean ± std for all metrics\n",
    "6. ✅ Analyzed stability (coefficient of variation)\n",
    "7. ✅ Created visualizations\n",
    "8. ✅ Generated LaTeX tables\n",
    "\n",
    "**Results saved in:**\n",
    "- Checkpoints: `checkpoints/multi_seed/`\n",
    "- Results: `multi_seed_results/`"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "BNN GPU H100",
   "language": "python",
   "name": "bnn-gpu115-h100"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.18"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
