{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "62e47b28",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import torch.optim as optim\n",
    "\n",
    "from torch.utils.data import DataLoader, Dataset\n",
    "import torchvision\n",
    "import torchvision.transforms as transforms\n",
    "\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from sklearn.manifold import TSNE\n",
    "\n",
    "from tqdm import tqdm\n",
    "import random\n",
    "import os\n",
    "\n",
    "from modules.COMMSM import CommsMod\n",
    "\n",
    "from utils.HybridSpikeLoss import HybridSpikeLoss\n",
    "from utils.PoissonEncoder import PoissonEncoder\n",
    "from utils.TemporalContrastEncoder import TemporalContrastEncoder\n",
    "\n",
    "\n",
    "def extract_embeddings(model, data_loader, device='cuda'):\n",
    "    \"\"\"Extract embeddings with spike statistics\"\"\"\n",
    "    model.eval()\n",
    "    embeddings = []\n",
    "    labels = []\n",
    "    spike_stats = []\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        for data, target in tqdm(data_loader, desc='Extracting embeddings'):\n",
    "            data = data.to(device)\n",
    "            \n",
    "            embedding, _, spike_trains = model(data)\n",
    "            \n",
    "            # Store embeddings\n",
    "            embeddings.append(embedding.cpu().numpy())\n",
    "            labels.append(target.numpy())\n",
    "            \n",
    "            # Compute spike statistics\n",
    "            spike_rates = spike_trains.mean(0).cpu().numpy()\n",
    "            spike_stats.append({\n",
    "                'rates': spike_rates,\n",
    "                'sparsity': (spike_rates == 0).mean(),\n",
    "                'mean_rate': spike_rates.mean()\n",
    "            })\n",
    "    \n",
    "    embeddings = np.concatenate(embeddings, axis=0)\n",
    "    labels = np.concatenate(labels, axis=0)\n",
    "    \n",
    "    return embeddings, labels, spike_stats\n",
    "\n",
    "def print_model_performance(model, data_loader, hybrid_loss, device='cuda'):\n",
    "    \"\"\"Print model performance metrics\"\"\"\n",
    "    model.eval()\n",
    "    val_loss = 0.0\n",
    "    val_correct = 0\n",
    "    val_total = 0\n",
    "    \n",
    "    ce_loss = nn.CrossEntropyLoss()\n",
    "      \n",
    "    with torch.no_grad():\n",
    "        for data, target in data_loader:\n",
    "            data, target = data.to(device), target.to(device)\n",
    "                \n",
    "            embeddings, logits, _ = model(data)\n",
    "                \n",
    "            loss = ce_loss(logits, target)\n",
    "            val_loss += loss.item()\n",
    "                \n",
    "            _, predicted = logits.max(1)\n",
    "            val_total += target.size(0)\n",
    "            val_correct += predicted.eq(target).sum().item()\n",
    "    \n",
    "    accuracy = 100 * val_correct / val_total\n",
    "    avg_loss = val_total / len(data_loader)\n",
    "    \n",
    "    print(f'Average Loss: {avg_loss:.4f}, Accuracy: {accuracy:.2f}%')\n",
    "    \n",
    "    \n",
    "def analyze_encoding_quality(model, test_loader, device='cuda'):\n",
    "    \"\"\"Analyze how well the encoding preserves information\"\"\"\n",
    "    model.eval()\n",
    "    \n",
    "    # Get one batch\n",
    "    data, labels = next(iter(test_loader))\n",
    "    data = data.to(device)\n",
    "    \n",
    "    # Compare different encoding methods\n",
    "    fig, axes = plt.subplots(2, 3, figsize=(15, 10))\n",
    "    \n",
    "    # Original image\n",
    "    ax = axes[0, 0]\n",
    "    ax.imshow(data[0, 0].cpu().numpy(), cmap='gray')\n",
    "    ax.set_title('Original Image')\n",
    "    ax.axis('off')\n",
    "    \n",
    "    # Poisson encoding visualization\n",
    "    encoder_poisson = PoissonEncoder(num_steps=25)\n",
    "    spikes_poisson = encoder_poisson(data[0:1])\n",
    "    \n",
    "    ax = axes[0, 1]\n",
    "    spike_img = spikes_poisson[:, 0, 0].cpu().numpy().reshape(25, 28, 28).mean(0)\n",
    "    ax.imshow(spike_img, cmap='hot')\n",
    "    ax.set_title('Poisson Encoding (avg)')\n",
    "    ax.axis('off')\n",
    "    \n",
    "    # Temporal contrast encoding\n",
    "    encoder_temp = TemporalContrastEncoder(num_steps=25)\n",
    "    spikes_temp = encoder_temp(data[0:1])\n",
    "    \n",
    "    ax = axes[0, 2]\n",
    "    spike_img = spikes_temp[:, 0, 0].cpu().numpy().reshape(25, 28, 28).mean(0)\n",
    "    ax.imshow(spike_img, cmap='hot')\n",
    "    ax.set_title('Temporal Encoding (avg)')\n",
    "    ax.axis('off')\n",
    "    \n",
    "    # Spike raster for single pixel over time\n",
    "    ax = axes[1, 0]\n",
    "    pixel_spikes = spikes_poisson[:, 0, 0, 14, 14].cpu().numpy()\n",
    "    ax.eventplot([np.where(pixel_spikes)[0]], colors='black')\n",
    "    ax.set_title('Single Pixel Spike Train')\n",
    "    ax.set_xlabel('Time step')\n",
    "    ax.set_ylabel('Spike')\n",
    "    \n",
    "    # Reconstruction from spikes\n",
    "    ax = axes[1, 1]\n",
    "    reconstruction = spikes_poisson.mean(0)[0, 0].cpu().numpy()\n",
    "    ax.imshow(reconstruction, cmap='gray')\n",
    "    ax.set_title('Reconstruction from Spikes')\n",
    "    ax.axis('off')\n",
    "    \n",
    "    # Information preservation\n",
    "    ax = axes[1, 2]\n",
    "    original_flat = data[0, 0].cpu().numpy().flatten()\n",
    "    reconstructed_flat = reconstruction.flatten()\n",
    "    ax.scatter(original_flat, reconstructed_flat, alpha=0.5, s=1)\n",
    "    ax.plot([0, 1], [0, 1], 'r--')\n",
    "    ax.set_xlabel('Original Intensity')\n",
    "    ax.set_ylabel('Reconstructed Intensity')\n",
    "    ax.set_title('Information Preservation')\n",
    "    \n",
    "    plt.tight_layout()\n",
    "    plt.show()\n",
    "\n",
    "\n",
    "def visualize_learned_features(model, embeddings, labels, spike_stats, loss_module=None):\n",
    "    \"\"\"Visualize what the network has learned\"\"\"\n",
    "    \n",
    "    # 1. Embedding space visualization\n",
    "    fig, axes = plt.subplots(1, 2, figsize=(16, 7))\n",
    "    \n",
    "    # t-SNE\n",
    "    tsne = TSNE(n_components=2, random_state=42)\n",
    "    embeddings_2d = tsne.fit_transform(embeddings[:5000])\n",
    "    \n",
    "    ax = axes[0]\n",
    "    scatter = ax.scatter(embeddings_2d[:, 0], embeddings_2d[:, 1], \n",
    "                        c=labels[:5000], cmap='tab10', alpha=0.6, s=10)\n",
    "    ax.set_title('t-SNE of Spike-Based Embeddings')\n",
    "    ax.set_xlabel('t-SNE 1')\n",
    "    ax.set_ylabel('t-SNE 2')\n",
    "    \n",
    "    # Add class labels\n",
    "    class_names = ['T-shirt', 'Trouser', 'Pullover', 'Dress', 'Coat',\n",
    "                   'Sandal', 'Shirt', 'Sneaker', 'Bag', 'Ankle boot']\n",
    "    handles, _ = scatter.legend_elements()\n",
    "    ax.legend(handles, class_names, loc='center left', bbox_to_anchor=(1, 0.5))\n",
    "    \n",
    "    # 2. Spike statistics per class\n",
    "    ax = axes[1]\n",
    "    mean_rates_per_class = []\n",
    "    for i in range(10):\n",
    "        class_idx = labels[:5000] == i\n",
    "        class_rates = [s['mean_rate'] for s, l in zip(spike_stats[:5000], labels[:5000]) if l == i]\n",
    "        mean_rates_per_class.append(class_rates)\n",
    "    \n",
    "    ax.boxplot(mean_rates_per_class, labels=[name[:7] for name in class_names])\n",
    "    ax.set_xlabel('Class')\n",
    "    ax.set_ylabel('Mean Spike Rate')\n",
    "    ax.set_title('Spike Rate Distribution by Class')\n",
    "    ax.tick_params(axis='x', rotation=45)\n",
    "    \n",
    "    plt.tight_layout()\n",
    "    plt.show()\n",
    "\n",
    "def _load_pretrained_CommsMod(path: str, device):\n",
    "        \"\"\"Load pretrained CommsMod from file\"\"\"\n",
    "        model = CommsMod(\n",
    "        embedding_dim=128,\n",
    "        num_classes=10,\n",
    "        num_steps=25,\n",
    "        beta=0.80,\n",
    "        threshold=1.0,\n",
    "        encoder_type='poisson'\n",
    "    )\n",
    "        model.to(device)\n",
    "        model.load_state_dict(torch.load(path, map_location='cpu'))\n",
    "        return model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7a1562a4",
   "metadata": {},
   "outputs": [],
   "source": [
    "device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "\n",
    "model = _load_pretrained_CommsMod(\"fashion_mnist_improved_snn.pth\", device)\n",
    "\n",
    "print(f\"Using device: {device}\")\n",
    "hybrid_loss = HybridSpikeLoss(num_classes=10, embedding_dim=128).to(device)\n",
    "\n",
    "transform = transforms.Compose([\n",
    "        transforms.ToTensor(),\n",
    "    ])\n",
    "test_dataset = torchvision.datasets.FashionMNIST(\n",
    "        root='./data', train=False, transform=transform\n",
    "    )\n",
    "    \n",
    "test_loader = DataLoader(test_dataset, batch_size=128, shuffle=False, \n",
    "                           num_workers=2, pin_memory=True)\n",
    "\n",
    "print(\"\\nEvaluating model performance...\")\n",
    "print_model_performance(model, test_loader, device)\n",
    "\n",
    "print(\"\\nExtracting embeddings...\")\n",
    "embeddings, labels, spike_stats = extract_embeddings(model, test_loader, device)\n",
    "    \n",
    "# Visualize learned features\n",
    "print(\"\\nVisualizing learned features...\")\n",
    "visualize_learned_features(model, embeddings, labels, spike_stats, hybrid_loss)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "6722e57e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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AACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNJwAAAACL0HACAAAAsAgNpzheeeUVzWazaRs2bGj2715wwQVaSUlJq+xXZ0Xn4qqrrmrV8wbtB51fOs9///vf9/euALQL9LmUmpq62+0OPfRQ419LOfTQQ7UhQ4ZoHUmbajgtXrxYO/XUU7UePXpoXq9XKy4u1g4//HDtiSee2N+7Bh30vD/wwAPa+++/3+rv0x6hPsLuvqDI//Lz87XJkydrM2fO3N+712E8/fTTxrE98MAD9/eu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",
      "text/plain": [
       "<Figure size 600x800 with 10 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "from torchvision import datasets, transforms\n",
    "\n",
    "# Load FashionMNIST with PyTorch\n",
    "transform = transforms.Compose([transforms.ToTensor()])\n",
    "dataset = datasets.FashionMNIST(root='./data', train=True, download=True, transform=transform)\n",
    "\n",
    "# Class names for FashionMNIST\n",
    "class_names = [\n",
    "    \"T-shirt/top\", \"Trouser\", \"Pullover\", \"Dress\", \"Coat\",\n",
    "    \"Sandal\", \"Shirt\", \"Sneaker\", \"Bag\", \"Ankle boot\"\n",
    "]\n",
    "\n",
    "# Find the first instance of each class\n",
    "first_indices = []\n",
    "for class_index in range(len(class_names)):\n",
    "    # torch.where returns a tuple, take the first tensor and its first element\n",
    "    idx = torch.where(dataset.targets == class_index)[0][0].item()\n",
    "    first_indices.append(idx)\n",
    "\n",
    "# Prepare a 4x3 grid\n",
    "fig, axes = plt.subplots(2, 5, figsize=(6, 8))\n",
    "axes = axes.flatten()\n",
    "\n",
    "# Plot each class image\n",
    "for ax, class_index in zip(axes, range(len(first_indices))):\n",
    "    img, label = dataset[first_indices[class_index]]\n",
    "    # img is a tensor with shape [1, 28, 28], squeeze to [28,28]\n",
    "    ax.imshow(img.squeeze(), cmap='coolwarm')\n",
    "    ax.set_title(f\"{class_names[label]}\")\n",
    "    ax.axis('off')\n",
    "\n",
    "# Hide any unused subplots (2 remaining slots)\n",
    "for ax in axes[len(first_indices):]:\n",
    "    ax.axis('off')\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "544fa002",
   "metadata": {},
   "outputs": [],
   "source": [
    "#!/usr/bin/env python3\n",
    "\n",
    "import json\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import os\n",
    "from scipy import stats\n",
    "\n",
    "# Import the confidence plotting functions\n",
    "try:\n",
    "    from utils.confidence_plots import compute_confidence_intervals\n",
    "    print(\"✅ Successfully imported confidence plotting functions\")\n",
    "except ImportError as e:\n",
    "    print(f\"❌ Error importing confidence plotting functions: {e}\")\n",
    "    print(\"Make sure confidence_plots.py is in the same directory\")\n",
    "    exit(1)\n",
    "\n",
    "\n",
    "def load_single_epoch_file(filepath):\n",
    "    \"\"\"\n",
    "    Load a single JSON file containing all epoch data.\n",
    "    \"\"\"\n",
    "    try:\n",
    "        with open(filepath, 'r') as f:\n",
    "            data = json.load(f)\n",
    "        print(f\"✅ Successfully loaded {filepath}\")\n",
    "        return data\n",
    "    except Exception as e:\n",
    "        print(f\"❌ Error loading {filepath}: {e}\")\n",
    "        return None\n",
    "\n",
    "\n",
    "def plot_metric_with_confidence_intervals(metric_name, values, method='sem', output_dir=\"single_file_plots\", \n",
    "                                        convergence_threshold=90.0):\n",
    "    \"\"\"\n",
    "    Create a confidence interval plot for a single metric.\n",
    "    \n",
    "    Args:\n",
    "        metric_name: Name of the metric\n",
    "        values: List of values across epochs\n",
    "        method: CI method ('sem', 'normal', 'bootstrap')\n",
    "        output_dir: Directory to save plots\n",
    "        convergence_threshold: Threshold for convergence detection (for accuracy-like metrics)\n",
    "    \"\"\"\n",
    "    \n",
    "    if not values or len(values) < 3:\n",
    "        print(f\"⚠️  Skipping {metric_name}: insufficient data ({len(values) if values else 0} points)\")\n",
    "        return\n",
    "    \n",
    "    # Convert to numpy array and handle any non-numeric values\n",
    "    try:\n",
    "        clean_values = []\n",
    "        for v in values:\n",
    "            if isinstance(v, (int, float)) and not np.isnan(v):\n",
    "                clean_values.append(v)\n",
    "            else:\n",
    "                print(f\"⚠️  Skipping non-numeric value in {metric_name}: {v}\")\n",
    "        \n",
    "        if len(clean_values) < 3:\n",
    "            print(f\"⚠️  Insufficient clean data for {metric_name}\")\n",
    "            return\n",
    "            \n",
    "        values_array = np.array(clean_values)\n",
    "        epochs = range(1, len(values_array) + 1)\n",
    "        \n",
    "    except Exception as e:\n",
    "        print(f\"❌ Error processing {metric_name}: {e}\")\n",
    "        return\n",
    "    \n",
    "    # Compute confidence intervals\n",
    "    try:\n",
    "        ci_data = compute_confidence_intervals(values_array.tolist(), method=method, confidence_level=0.95)\n",
    "    except Exception as e:\n",
    "        print(f\"❌ Error computing CI for {metric_name}: {e}\")\n",
    "        return\n",
    "    \n",
    "    # Create the plot\n",
    "    fig, ax = plt.subplots(figsize=(12, 8))\n",
    "    \n",
    "    # Main line\n",
    "    ax.plot(epochs, ci_data['mean'], 'b-', linewidth=3, label=f'{metric_name}', alpha=0.9)\n",
    "    \n",
    "    # Confidence interval\n",
    "    ax.fill_between(epochs, ci_data['lower'], ci_data['upper'], \n",
    "                   alpha=0.3, color='darkturquoise', \n",
    "                   label=f'95% Confidence Interval ({method.upper()})')\n",
    "    \n",
    "    # Convergence detection and marking (for accuracy-like metrics)\n",
    "    convergence_epoch = None\n",
    "    if (convergence_threshold is not None and \n",
    "        metric_name.lower() in ['accuracy', 'avg_reward'] and \n",
    "        max(ci_data['mean']) >= convergence_threshold):\n",
    "        # Find first epoch where metric >= threshold\n",
    "        converged_indices = np.where(np.array(ci_data['mean']) >= convergence_threshold)[0]\n",
    "        if len(converged_indices) > 0:\n",
    "            convergence_epoch = converged_indices[0] + 1  # +1 for 1-indexing\n",
    "            convergence_value = ci_data['mean'][converged_indices[0]]\n",
    "            \n",
    "            # Add convergence line\n",
    "            ax.axvline(convergence_epoch, color='red', linestyle='--', linewidth=2,\n",
    "                      label=f'Convergence (epoch {convergence_epoch})')\n",
    "            \n",
    "            # Add convergence threshold line\n",
    "            threshold_label = f'Threshold ({convergence_threshold}%' if convergence_threshold >= 1 else f'Threshold ({convergence_threshold:.1f}'\n",
    "            ax.axhline(convergence_threshold, color='orange', linestyle=':', alpha=0.7,\n",
    "                      label=threshold_label + ')')\n",
    "            \n",
    "            # Add annotation\n",
    "            value_str = f'{convergence_value:.1f}%' if convergence_threshold >= 1 else f'{convergence_value:.3f}'\n",
    "            ax.annotate(f'Convergence\\n{value_str}', \n",
    "                       xy=(convergence_epoch, convergence_value),\n",
    "                       xytext=(convergence_epoch + 3, convergence_value + (0.05 * (max(ci_data['mean']) - min(ci_data['mean'])))),\n",
    "                       arrowprops=dict(arrowstyle='->', color='red', alpha=0.7),\n",
    "                       fontsize=10, ha='left',\n",
    "                       bbox=dict(boxstyle=\"round,pad=0.3\", facecolor='white', alpha=0.8))\n",
    "    \n",
    "    # Styling\n",
    "    ax.set_xlabel('Training Epoch', fontsize=14, fontweight='bold')\n",
    "    ax.set_ylabel(f'{metric_name.replace(\"_\", \" \").title()}', fontsize=14, fontweight='bold')\n",
    "    ax.set_title(f'{metric_name.replace(\"_\", \" \").title()}', \n",
    "                fontsize=16, fontweight='bold', pad=20)\n",
    "    ax.legend(loc='best', fontsize=12)\n",
    "    ax.grid(True, alpha=0.3)\n",
    "    \n",
    "    # Add statistics box\n",
    "    final_value = ci_data['mean'][-1]\n",
    "    final_ci_width = ci_data['upper'][-1] - ci_data['lower'][-1]\n",
    "    initial_value = ci_data['mean'][0]\n",
    "    total_change = final_value - initial_value\n",
    "    \n",
    "    stats_text = f\"\"\"Training Statistics:\n",
    "Initial: {initial_value:.3f}\n",
    "Final: {final_value:.3f} ± {final_ci_width/2:.3f}\n",
    "Total Change: {total_change:+.3f}\n",
    "Epochs: {len(epochs)}\"\"\"\n",
    "\n",
    "    if convergence_epoch and convergence_threshold is not None:\n",
    "        stats_text += f\"\"\"\n",
    "Converged: Epoch {convergence_epoch}\n",
    "Threshold: {convergence_threshold}\"\"\"\n",
    "    \n",
    "    stats_text += f\"\"\"\n",
    "Method: {method.upper()}\"\"\"\n",
    "    \n",
    "    ax.text(0.02, 0.98, stats_text, transform=ax.transAxes, \n",
    "           fontsize=10, verticalalignment='top',\n",
    "           bbox=dict(boxstyle=\"round,pad=0.5\", facecolor='lightyellow', alpha=0.8))\n",
    "    \n",
    "    # Save the plot\n",
    "    os.makedirs(output_dir, exist_ok=True)\n",
    "    save_path = os.path.join(output_dir, f\"{metric_name}_confidence_intervals.png\")\n",
    "    plt.tight_layout()\n",
    "    plt.savefig(save_path, dpi=300, bbox_inches='tight', facecolor='white')\n",
    "    plt.close()\n",
    "    \n",
    "    print(f\"📊 {metric_name}: Final value {final_value:.3f} ± {final_ci_width/2:.3f} | Saved to {save_path}\")\n",
    "\n",
    "\n",
    "def create_summary_grid_plot(data, output_dir=\"single_file_plots\", method='sem'):\n",
    "    \"\"\"\n",
    "    Create a grid plot showing multiple key metrics with confidence intervals.\n",
    "    \"\"\"\n",
    "    \n",
    "    # Key metrics to show in summary\n",
    "    key_metrics = ['accuracy', 'avg_reward', 'avg_loss', 'protocol_discriminability', \n",
    "                   'action_confidence', 'epsilon', 'mean_q', 'max_q']\n",
    "    \n",
    "    # Filter available metrics\n",
    "    available_metrics = []\n",
    "    for metric in key_metrics:\n",
    "        if metric in data and len(data[metric]) > 3:\n",
    "            available_metrics.append(metric)\n",
    "    \n",
    "    if len(available_metrics) == 0:\n",
    "        print(\"⚠️  No suitable metrics found for summary plot\")\n",
    "        return\n",
    "    \n",
    "    print(f\"📊 Creating summary grid with {len(available_metrics)} metrics: {available_metrics}\")\n",
    "    \n",
    "    # Determine grid size\n",
    "    n_metrics = len(available_metrics)\n",
    "    cols = 3\n",
    "    rows = (n_metrics + cols - 1) // cols\n",
    "    \n",
    "    fig, axes = plt.subplots(rows, cols, figsize=(18, 6*rows))\n",
    "    if n_metrics == 1:\n",
    "        axes = [axes]\n",
    "    elif rows == 1:\n",
    "        axes = axes.reshape(1, -1)\n",
    "    axes = axes.flatten()\n",
    "    \n",
    "    for idx, metric in enumerate(available_metrics):\n",
    "        ax = axes[idx]\n",
    "        \n",
    "        values = data[metric]\n",
    "        epochs = range(1, len(values) + 1)\n",
    "        \n",
    "        try:\n",
    "            # Compute confidence intervals\n",
    "            ci_data = compute_confidence_intervals(values, method=method)\n",
    "            \n",
    "            # Plot\n",
    "            ax.plot(epochs, ci_data['mean'], 'b-', linewidth=2, alpha=0.8)\n",
    "            ax.fill_between(epochs, ci_data['lower'], ci_data['upper'], \n",
    "                           alpha=0.3, color='darkturquoise')\n",
    "            \n",
    "            # Styling\n",
    "            ax.set_xlabel('Epoch', fontsize=11)\n",
    "            ax.set_ylabel(metric.replace('_', ' ').title(), fontsize=11)\n",
    "            ax.set_title(f'{metric.replace(\"_\", \" \").title()}', fontsize=12, fontweight='bold')\n",
    "            ax.grid(True, alpha=0.3)\n",
    "            \n",
    "            # Add final value annotation\n",
    "            final_val = ci_data['mean'][-1]\n",
    "            final_ci_width = ci_data['upper'][-1] - ci_data['lower'][-1]\n",
    "            ax.text(0.98, 0.02, f'{final_val:.3f}\\n±{final_ci_width/2:.3f}',\n",
    "                   transform=ax.transAxes, fontsize=9, ha='right', va='bottom',\n",
    "                   bbox=dict(boxstyle=\"round,pad=0.3\", facecolor='white', alpha=0.8))\n",
    "                   \n",
    "        except Exception as e:\n",
    "            ax.text(0.5, 0.5, f'Error plotting\\n{metric}:\\n{str(e)[:30]}...', \n",
    "                   ha='center', va='center', transform=ax.transAxes,\n",
    "                   fontsize=10, color='red')\n",
    "            ax.set_title(f'{metric.replace(\"_\", \" \").title()} (Error)', color='red')\n",
    "    \n",
    "    # Remove empty subplots\n",
    "    for idx in range(n_metrics, len(axes)):\n",
    "        fig.delaxes(axes[idx])\n",
    "    \n",
    "    plt.suptitle('Training Metrics Summary with 95% Confidence Intervals', \n",
    "                fontsize=16, fontweight='bold')\n",
    "    plt.tight_layout()\n",
    "    \n",
    "    # Save\n",
    "    os.makedirs(output_dir, exist_ok=True)\n",
    "    save_path = os.path.join(output_dir, f\"summary_all_metrics_{method}_ci.png\")\n",
    "    plt.savefig(save_path, dpi=300, bbox_inches='tight', facecolor='white')\n",
    "    plt.close()\n",
    "    \n",
    "    print(f\"📊 Summary grid plot saved to {save_path}\")\n",
    "\n",
    "\n",
    "def process_single_file(filepath, methods=['sem'], output_dir=\"single_file_plots\"):\n",
    "    \"\"\"\n",
    "    Process a single JSON file and create confidence interval plots for all metrics.\n",
    "    \n",
    "    Args:\n",
    "        filepath: Path to the JSON file\n",
    "        methods: List of CI methods to try\n",
    "        output_dir: Output directory for plots\n",
    "    \"\"\"\n",
    "    \n",
    "    print(f\"\\n{'='*60}\")\n",
    "    print(f\"PROCESSING SINGLE FILE: {filepath}\")\n",
    "    print(f\"{'='*60}\")\n",
    "    \n",
    "    # Load the data\n",
    "    data = load_single_epoch_file(filepath)\n",
    "    if data is None:\n",
    "        return\n",
    "    \n",
    "    print(f\"\\n📋 Available metrics in file:\")\n",
    "    for key, value in data.items():\n",
    "        if isinstance(value, list):\n",
    "            if len(value) > 0 and all(isinstance(v, (int, float)) for v in value):\n",
    "                print(f\"  - {key}: {len(value)} values, range [{min(value):.3f}, {max(value):.3f}]\")\n",
    "            else:\n",
    "                print(f\"  - {key}: {len(value)} values (empty or non-numeric)\")\n",
    "        else:\n",
    "            print(f\"  - {key}: {type(value)} (skipped)\")\n",
    "    \n",
    "    # Filter numeric list metrics\n",
    "    numeric_metrics = {}\n",
    "    for key, value in data.items():\n",
    "        if isinstance(value, list) and len(value) > 3:\n",
    "            # Check if all values are numeric and not NaN\n",
    "            try:\n",
    "                numeric_values = []\n",
    "                for v in value:\n",
    "                    if isinstance(v, (int, float)) and not np.isnan(v):\n",
    "                        numeric_values.append(v)\n",
    "                \n",
    "                if len(numeric_values) > 3:  # Need at least 4 points for CI\n",
    "                    numeric_metrics[key] = numeric_values\n",
    "            except Exception as e:\n",
    "                print(f\"  ⚠️  Skipping {key}: {e}\")\n",
    "    \n",
    "    print(f\"\\n📊 Will plot {len(numeric_metrics)} numeric metrics: {list(numeric_metrics.keys())}\")\n",
    "    \n",
    "    # Create individual plots for each method\n",
    "    for method in methods:\n",
    "        print(f\"\\n🎨 Creating plots with {method.upper()} method...\")\n",
    "        method_dir = os.path.join(output_dir, f\"{method}_method\")\n",
    "        \n",
    "        for metric_name, values in numeric_metrics.items():\n",
    "            # Use different convergence thresholds for different metrics\n",
    "            if metric_name in ['accuracy', 'avg_reward']:\n",
    "                convergence_threshold = 90.0 if metric_name == 'accuracy' else 0.9\n",
    "            else:\n",
    "                convergence_threshold = None  # No convergence detection for other metrics\n",
    "                \n",
    "            plot_metric_with_confidence_intervals(\n",
    "                metric_name, values, method=method, output_dir=method_dir,\n",
    "                convergence_threshold=convergence_threshold\n",
    "            )\n",
    "    \n",
    "    # Create summary grid plot\n",
    "    print(f\"\\n🎨 Creating summary grid plot...\")\n",
    "    create_summary_grid_plot(data, output_dir, method=methods[0])\n",
    "    \n",
    "    print(f\"\\n✅ PROCESSING COMPLETE!\")\n",
    "    print(f\"📁 All plots saved to: {output_dir}\")\n",
    "    print(f\"📊 Total metrics plotted: {len(numeric_metrics)}\")\n",
    "\n",
    "\n",
    "def main():\n",
    "    \"\"\"\n",
    "    Main function - processes the metrics_epoch_50.json file.\n",
    "    \"\"\"\n",
    "    \n",
    "    print(\"=\" * 60)\n",
    "    print(\"SINGLE FILE CONFIDENCE INTERVAL PLOTTING\")\n",
    "    print(\"=\" * 60)\n",
    "    \n",
    "    # Default file path\n",
    "    filepath = \"metrics\\metrics_epoch_50.json\"\n",
    "    \n",
    "    # Check if file exists\n",
    "    if not os.path.exists(filepath):\n",
    "        print(f\"❌ File not found: {filepath}\")\n",
    "        print(\"Please make sure 'metrics_epoch_50.json' is in the current directory\")\n",
    "        return\n",
    "    \n",
    "    # Process the file with multiple methods\n",
    "    methods = ['sem', 'normal']  # SEM works well for time series, normal as comparison\n",
    "    process_single_file(filepath, methods=methods, output_dir=\"epoch_50_plots\")\n",
    "    \n",
    "\n",
    "\n",
    "if __name__ == \"__main__\":\n",
    "    main()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "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.12.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
