{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "1125f020",
   "metadata": {},
   "source": [
    "## My manifold Learning for Celeba"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "dc0fe3d0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Found existing best model, loading...\n",
      "\n",
      "==================================================\n",
      "EVALUATING SOTA REGRESSION\n",
      "==================================================\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 312/312 [00:11<00:00, 26.24it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--------------------------------------------------\n",
      "CelebA Manifold Results (5% Sparsity):\n",
      "  MSE ([0,1]):  0.003767  (Target: ~0.0050)\n",
      "  PSNR:         24.68 dB  (Target: ~23 dB)\n",
      "  Inference:    0.00000 sec (Batch)\n",
      "--------------------------------------------------\n",
      "SUCCESS: Achieved SOTA Regression Level.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torch.utils.data import DataLoader, Dataset\n",
    "from torchvision import datasets, transforms\n",
    "import torchvision.utils as vutils\n",
    "import numpy as np\n",
    "from tqdm import tqdm\n",
    "import os\n",
    "import copy\n",
    "\n",
    "# ============================================================================\n",
    "# 1. CONFIGURATION (v3)\n",
    "# ============================================================================\n",
    "CONFIG = {\n",
    "    'sparsity': 0.05,\n",
    "    'img_size': 64,       \n",
    "    'batch_size': 64,\n",
    "    'epochs': 60,\n",
    "    'lr': 2e-4,           \n",
    "    'latent_dim': 1024,\n",
    "    'attr_dim': 40,\n",
    "    'patience': 8,         # UPDATED: Set to 8 as requested\n",
    "    'consistency_w': 1.0,  \n",
    "    'attr_weight': 0.1, \n",
    "    'attr_dropout': 0.5,   # CRITICAL: Probability of zeroing attributes during train\n",
    "    'device': 'cuda' if torch.cuda.is_available() else 'cpu'\n",
    "}\n",
    "\n",
    "# ============================================================================\n",
    "# 2. DATASET\n",
    "# ============================================================================\n",
    "class EnhancedCelebADataset(Dataset):\n",
    "    def __init__(self, root, split='train', sparsity=0.05, img_size=64):\n",
    "        self.transform = transforms.Compose([\n",
    "            transforms.CenterCrop(178),\n",
    "            transforms.Resize(img_size),\n",
    "            transforms.ToTensor(),\n",
    "            transforms.Normalize((0.5, 0.5, 0.5), (0.5, 0.5, 0.5))\n",
    "        ])\n",
    "        try:\n",
    "            self.data = datasets.CelebA(root=root, split=split, download=False, transform=self.transform)\n",
    "        except:\n",
    "            self.data = datasets.CelebA(root=root, split=split, download=True, transform=self.transform)\n",
    "        self.sparsity = sparsity\n",
    "\n",
    "    def __len__(self): return len(self.data)\n",
    "\n",
    "    def __getitem__(self, idx):\n",
    "        full_image, attributes = self.data[idx]\n",
    "        mask = (torch.rand_like(full_image) < self.sparsity).float()\n",
    "        sparse_image = full_image * mask\n",
    "        attributes = attributes.float()\n",
    "        x = {'sparse_image': sparse_image, 'mask': mask}\n",
    "        s = {'full_image': full_image, 'attributes': attributes} \n",
    "        return x, s\n",
    "\n",
    "# ============================================================================\n",
    "# 3. COMPONENTS: AdaIN & ResBlocks\n",
    "# ============================================================================\n",
    "class AdaIN(nn.Module):\n",
    "    \"\"\"Adaptive Instance Normalization - The StyleGAN way to inject attributes\"\"\"\n",
    "    def __init__(self, channels, embed_dim):\n",
    "        super().__init__()\n",
    "        self.channels = channels\n",
    "        # Learn to predict scale (gamma) and shift (beta) from attributes\n",
    "        self.linear = nn.Linear(embed_dim, channels * 2)\n",
    "\n",
    "    def forward(self, x, attr_embed):\n",
    "        # x: [B, C, H, W]\n",
    "        # attr_embed: [B, embed_dim]\n",
    "        \n",
    "        # Calculate style parameters\n",
    "        style = self.linear(attr_embed) # [B, 2*C]\n",
    "        style = style.view(-1, 2, self.channels, 1, 1)\n",
    "        gamma, beta = style[:, 0], style[:, 1]\n",
    "        \n",
    "        # Normalize x\n",
    "        mean = x.mean(dim=[2, 3], keepdim=True)\n",
    "        std = x.std(dim=[2, 3], keepdim=True) + 1e-8\n",
    "        norm_x = (x - mean) / std\n",
    "        \n",
    "        # Modulate\n",
    "        return gamma * norm_x + beta\n",
    "\n",
    "class AdaINResBlock(nn.Module):\n",
    "    def __init__(self, c, embed_dim):\n",
    "        super().__init__()\n",
    "        self.conv1 = nn.Conv2d(c, c, 3, 1, 1, bias=False)\n",
    "        self.conv2 = nn.Conv2d(c, c, 3, 1, 1, bias=False)\n",
    "        self.adain1 = AdaIN(c, embed_dim)\n",
    "        self.adain2 = AdaIN(c, embed_dim)\n",
    "        self.act = nn.ReLU()\n",
    "        \n",
    "    def forward(self, x, attr_embed):\n",
    "        residual = x\n",
    "        \n",
    "        out = self.conv1(x)\n",
    "        out = self.act(self.adain1(out, attr_embed)) \n",
    "        \n",
    "        out = self.conv2(out)\n",
    "        out = self.adain2(out, attr_embed) \n",
    "        \n",
    "        return self.act(out + residual)\n",
    "\n",
    "# ============================================================================\n",
    "# 4. ENCODER (Robust Spatial Focus)\n",
    "# ============================================================================\n",
    "class RobustEncoder(nn.Module):\n",
    "    def __init__(self, input_channels=6, latent_channels=512, attr_dim=40, embed_dim=256):\n",
    "        super().__init__()\n",
    "        # Standard convolutions\n",
    "        self.entry = nn.Sequential(nn.Conv2d(input_channels, 64, 4, 2, 1), nn.ReLU())\n",
    "        self.layer1 = nn.Sequential(nn.Conv2d(64, 128, 4, 2, 1), nn.BatchNorm2d(128), nn.ReLU())\n",
    "        self.layer2 = nn.Sequential(nn.Conv2d(128, 256, 4, 2, 1), nn.BatchNorm2d(256), nn.ReLU())\n",
    "        \n",
    "        # We use AdaIN blocks here to condition encoder on attributes\n",
    "        self.res1 = AdaINResBlock(256, embed_dim)\n",
    "        self.res2 = AdaINResBlock(256, embed_dim)\n",
    "        \n",
    "        self.exit = nn.Conv2d(256, latent_channels, 4, 2, 1) \n",
    "\n",
    "    def forward(self, x, mask, attr_embed):\n",
    "        # Concatenate mask (essential for robustness)\n",
    "        if mask is not None: x = torch.cat([x, mask], dim=1)\n",
    "        \n",
    "        h = self.entry(x)\n",
    "        h = self.layer1(h)\n",
    "        h = self.layer2(h)\n",
    "        \n",
    "        # Inject attributes via AdaIN (Gating) rather than Concatenation\n",
    "        h = self.res1(h, attr_embed)\n",
    "        h = self.res2(h, attr_embed)\n",
    "        \n",
    "        return self.exit(h)\n",
    "\n",
    "class RobustDecoder(nn.Module):\n",
    "    def __init__(self, latent_channels=512, embed_dim=256):\n",
    "        super().__init__()\n",
    "        \n",
    "        self.layer1 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(latent_channels, 256, 4, 2, 1),\n",
    "            nn.BatchNorm2d(256), nn.ReLU()\n",
    "        )\n",
    "        \n",
    "        # Decoder also uses AdaIN to \"paint\" the attributes onto the features\n",
    "        self.res1 = AdaINResBlock(256, embed_dim)\n",
    "        self.res2 = AdaINResBlock(256, embed_dim)\n",
    "        \n",
    "        self.layer2 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(256, 128, 4, 2, 1),\n",
    "            nn.BatchNorm2d(128), nn.ReLU()\n",
    "        )\n",
    "        self.layer3 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(128, 64, 4, 2, 1),\n",
    "            nn.BatchNorm2d(64), nn.ReLU()\n",
    "        )\n",
    "        self.exit = nn.Sequential(\n",
    "            nn.ConvTranspose2d(64, 32, 4, 2, 1),\n",
    "            nn.BatchNorm2d(32), nn.ReLU(),\n",
    "            nn.Conv2d(32, 3, 3, 1, 1),\n",
    "            nn.Tanh()\n",
    "        )\n",
    "\n",
    "    def forward(self, z, attr_embed):\n",
    "        h = self.layer1(z)\n",
    "        h = self.res1(h, attr_embed)\n",
    "        h = self.res2(h, attr_embed)\n",
    "        h = self.layer2(h)\n",
    "        h = self.layer3(h)\n",
    "        return self.exit(h)\n",
    "\n",
    "# ============================================================================\n",
    "# 5. MAIN MODEL\n",
    "# ============================================================================\n",
    "class AttributeEmbedding(nn.Module):\n",
    "    def __init__(self, attr_dim=40, embed_dim=256):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.Linear(attr_dim, 128), nn.ReLU(),\n",
    "            nn.Linear(128, embed_dim), nn.ReLU(),\n",
    "            nn.Linear(embed_dim, embed_dim)\n",
    "        )\n",
    "    def forward(self, x): return self.net(x)\n",
    "\n",
    "class AttributePredictor(nn.Module):\n",
    "    def __init__(self, latent_channels, attr_dim):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.AdaptiveAvgPool2d(1), nn.Flatten(),\n",
    "            nn.Linear(latent_channels, 128), nn.ReLU(),\n",
    "            nn.Linear(128, attr_dim), nn.Sigmoid()\n",
    "        )\n",
    "    def forward(self, x): return self.net(x)\n",
    "\n",
    "class SOTARobustModel(nn.Module):\n",
    "    def __init__(self):\n",
    "        super().__init__()\n",
    "        embed_dim = 256\n",
    "        self.attr_embedding = AttributeEmbedding(CONFIG['attr_dim'], embed_dim)\n",
    "        self.encoder_x = RobustEncoder(6, CONFIG['latent_dim'], CONFIG['attr_dim'], embed_dim)\n",
    "        self.encoder_s = RobustEncoder(3, CONFIG['latent_dim'], CONFIG['attr_dim'], embed_dim)\n",
    "        self.decoder = RobustDecoder(CONFIG['latent_dim'], embed_dim)\n",
    "        self.attr_predictor = AttributePredictor(CONFIG['latent_dim'], CONFIG['attr_dim'])\n",
    "        \n",
    "    def forward(self, x, s, training_dropout=False):\n",
    "        # 1. Get Attributes\n",
    "        attr = s['attributes']\n",
    "        attr_embed = self.attr_embedding(attr)\n",
    "        \n",
    "        # 2. ATTRIBUTE DROPOUT (The Robustness Fix)\n",
    "        # If we are training and hit the probability, we ZERO out the attribute embedding.\n",
    "        # This forces the network to rely on the image pixels in 'x'.\n",
    "        if training_dropout and np.random.random() < CONFIG['attr_dropout']:\n",
    "            used_embed = torch.zeros_like(attr_embed)\n",
    "        else:\n",
    "            used_embed = attr_embed\n",
    "            \n",
    "        # 3. Encode & Decode\n",
    "        z_x = self.encoder_x(x['sparse_image'], x['mask'], used_embed)\n",
    "        z_s = self.encoder_s(s['full_image'], None, used_embed)\n",
    "        \n",
    "        recon_x = self.decoder(z_x, used_embed)\n",
    "        recon_s = self.decoder(z_s, used_embed)\n",
    "        \n",
    "        # 4. Predict Attributes (Always use z to ensure latent holds info)\n",
    "        pred_attr_x = self.attr_predictor(z_x)\n",
    "        \n",
    "        return z_x, z_s, recon_x, recon_s, pred_attr_x\n",
    "\n",
    "# ============================================================================\n",
    "# 6. TRAINING (FIXED: Best Model Save + Patience=8 + Correct Name)\n",
    "# ============================================================================\n",
    "def train_v3(model, train_loader, val_loader, epochs, device):\n",
    "    l1_loss = nn.L1Loss()\n",
    "    bce_loss = nn.BCELoss()\n",
    "    \n",
    "    optimizer = torch.optim.Adam(model.parameters(), lr=CONFIG['lr'])\n",
    "    # Reduce LR if validation loss plateaus (helps converge before stopping)\n",
    "    scheduler = torch.optim.lr_scheduler.ReduceLROnPlateau(optimizer, 'min', patience=3, factor=0.5)\n",
    "    \n",
    "    # --- EARLY STOPPING VARS ---\n",
    "    best_val_loss = float('inf')\n",
    "    epochs_no_improve = 0\n",
    "    best_model_wts = copy.deepcopy(model.state_dict())\n",
    "    \n",
    "    print(\"=\"*60)\n",
    "    print(f\"STARTING V3: Robust Semantic Manifold (Patience={CONFIG['patience']})\")\n",
    "    print(f\"Attribute Dropout: {CONFIG['attr_dropout']}\")\n",
    "    print(\"=\"*60)\n",
    "    \n",
    "    for epoch in range(epochs):\n",
    "        model.train()\n",
    "        loop = tqdm(train_loader, desc=f\"Ep {epoch+1}/{epochs}\")\n",
    "        \n",
    "        train_loss_accum = 0\n",
    "        \n",
    "        for x, s in loop:\n",
    "            # Move data\n",
    "            sparse = x['sparse_image'].to(device)\n",
    "            mask = x['mask'].to(device)\n",
    "            full = s['full_image'].to(device)\n",
    "            attr = s['attributes'].to(device)\n",
    "            \n",
    "            x_in = {'sparse_image': sparse, 'mask': mask}\n",
    "            s_in = {'full_image': full, 'attributes': attr}\n",
    "            \n",
    "            optimizer.zero_grad()\n",
    "            \n",
    "            # Forward with Attribute Dropout (Robustness)\n",
    "            # This enables the random zeroing of attributes\n",
    "            z_x, z_s, recon_x, recon_s, pred_attr_x = model(x_in, s_in, training_dropout=True)\n",
    "            \n",
    "            # Loss Calculation\n",
    "            loss_pix = l1_loss(recon_x, full) + l1_loss(recon_s, full)\n",
    "            loss_cons = F.mse_loss(z_x, z_s)\n",
    "            loss_attr = bce_loss(pred_attr_x, attr) \n",
    "            \n",
    "            loss = loss_pix + (CONFIG['consistency_w'] * loss_cons) + (CONFIG['attr_weight'] * loss_attr)\n",
    "            \n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            \n",
    "            train_loss_accum += loss.item()\n",
    "            loop.set_postfix(L1=loss_pix.item(), Attr=loss_attr.item())\n",
    "\n",
    "        # --- VALIDATION PHASE ---\n",
    "        model.eval()\n",
    "        val_loss = 0\n",
    "        with torch.no_grad():\n",
    "            for x, s in val_loader:\n",
    "                sparse = x['sparse_image'].to(device)\n",
    "                mask = x['mask'].to(device)\n",
    "                full = s['full_image'].to(device)\n",
    "                attr = s['attributes'].to(device)\n",
    "                \n",
    "                x_in = {'sparse_image': sparse, 'mask': mask}\n",
    "                s_in = {'full_image': full, 'attributes': attr}\n",
    "                \n",
    "                # In validation, we use training_dropout=False to use ALL info available\n",
    "                # This gives us the best possible metric for saving the model\n",
    "                _, _, recon, _, _ = model(x_in, s_in, training_dropout=False)\n",
    "                \n",
    "                # Metric: L1 Reconstruction Loss\n",
    "                val_loss += l1_loss(recon, full).item()\n",
    "\n",
    "        avg_val_loss = val_loss / len(val_loader)\n",
    "        scheduler.step(avg_val_loss)\n",
    "        \n",
    "        current_lr = optimizer.param_groups[0]['lr']\n",
    "        print(f\"  >> Val L1: {avg_val_loss:.5f} | Best: {best_val_loss:.5f} | LR: {current_lr:.2e}\")\n",
    "        \n",
    "        # --- EARLY STOPPING LOGIC ---\n",
    "        if avg_val_loss < best_val_loss:\n",
    "            best_val_loss = avg_val_loss\n",
    "            best_model_wts = copy.deepcopy(model.state_dict())\n",
    "            epochs_no_improve = 0\n",
    "            \n",
    "            # Save strictly the best model\n",
    "            torch.save(model.state_dict(), 'sota_v3_best.pth')\n",
    "            print(\"  >> New Best Model Saved!\")\n",
    "        else:\n",
    "            epochs_no_improve += 1\n",
    "            print(f\"  >> No Improvement. Patience: {epochs_no_improve}/{CONFIG['patience']}\")\n",
    "            \n",
    "        if epochs_no_improve >= CONFIG['patience']:\n",
    "            print(\"\\n\" + \"=\"*40)\n",
    "            print(f\"EARLY STOPPING TRIGGERED AT EPOCH {epoch+1}\")\n",
    "            print(\"=\"*40)\n",
    "            break\n",
    "\n",
    "    # Load best weights before returning\n",
    "    print(\"Loading best model weights...\")\n",
    "    model.load_state_dict(best_model_wts)\n",
    "    return model\n",
    "\n",
    "# ============================================================================\n",
    "# 5. SCIENTIFIC EVALUATION\n",
    "# ============================================================================\n",
    "\n",
    "# ============================================================================\n",
    "# 5. SCIENTIFIC EVALUATION\n",
    "# ============================================================================\n",
    "def evaluate_manifold(model, loader, device):\n",
    "    print(\"\\n\" + \"=\"*50)\n",
    "    print(\"EVALUATING SOTA REGRESSION\")\n",
    "    print(\"=\"*50)\n",
    "    \n",
    "    model.eval()\n",
    "    mses = []\n",
    "    psnrs = []\n",
    "    times = []\n",
    "    \n",
    "    l1_loss = nn.L1Loss()\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        for i, (x, s) in enumerate(tqdm(loader)):\n",
    "            # Unpack dictionaries\n",
    "            sparse = x['sparse_image'].to(device)\n",
    "            mask = x['mask'].to(device)\n",
    "            full = s['full_image'].to(device)\n",
    "            attr = s['attributes'].to(device)\n",
    "            \n",
    "            x_in = {'sparse_image': sparse, 'mask': mask}\n",
    "            s_in = {'full_image': full, 'attributes': attr}\n",
    "            \n",
    "            # Get reconstruction (no dropout during evaluation)\n",
    "            _, _, recon_x, _, _ = model(x_in, s_in, training_dropout=False)\n",
    "            \n",
    "            # Unnormalize to [0, 1]\n",
    "            recon_01 = torch.clamp((recon_x + 1) / 2, 0, 1)\n",
    "            full_01 = torch.clamp((full + 1) / 2, 0, 1)\n",
    "            \n",
    "            # Batch Metrics\n",
    "            batch_mse = F.mse_loss(recon_01, full_01, reduction='none').mean(dim=[1,2,3])\n",
    "            \n",
    "            for m in batch_mse:\n",
    "                mses.append(m.item())\n",
    "                if m.item() > 0:\n",
    "                    psnrs.append(-10 * np.log10(m.item()))\n",
    "    \n",
    "    avg_mse = np.mean(mses)\n",
    "    avg_psnr = np.mean(psnrs)\n",
    "    avg_time = np.mean(times) if times else 0\n",
    "    \n",
    "    print(\"-\" * 50)\n",
    "    print(f\"CelebA Manifold Results (5% Sparsity):\")\n",
    "    print(f\"  MSE ([0,1]):  {avg_mse:.6f}  (Target: ~0.0050)\")\n",
    "    print(f\"  PSNR:         {avg_psnr:.2f} dB  (Target: ~23 dB)\")\n",
    "    print(f\"  Inference:    {avg_time:.5f} sec (Batch)\")\n",
    "    print(\"-\" * 50)\n",
    "    \n",
    "    if avg_mse <= 0.0055:\n",
    "        print(\"SUCCESS: Achieved SOTA Regression Level.\")\n",
    "    else:\n",
    "        print(\"NOTE: Train longer to reach target MSE.\")\n",
    "    \n",
    "    return avg_mse, avg_psnr\n",
    " \n",
    "\n",
    "# ============================================================================\n",
    "# MAIN\n",
    "# ============================================================================\n",
    "if __name__ == \"__main__\":\n",
    "    device = CONFIG['device']\n",
    "    \n",
    "    # 1. Dataset\n",
    "    train_ds = EnhancedCelebADataset('./data', split='train', sparsity=CONFIG['sparsity'])\n",
    "    val_ds = EnhancedCelebADataset('./data', split='valid', sparsity=CONFIG['sparsity'])\n",
    "    test_ds = EnhancedCelebADataset('./data', split='test', sparsity=CONFIG['sparsity'])\n",
    "    \n",
    "    train_loader = DataLoader(train_ds, batch_size=CONFIG['batch_size'], shuffle=True, num_workers=4)\n",
    "    val_loader = DataLoader(val_ds, batch_size=CONFIG['batch_size'], shuffle=False)\n",
    "    test_loader = DataLoader(test_ds, batch_size=CONFIG['batch_size'], shuffle=False)\n",
    "    \n",
    "    # 2. Model\n",
    "    model = SOTARobustModel().to(device)\n",
    "    \n",
    "    # 3. Train (Correctly calls train_v3)\n",
    "    if os.path.exists('sota_v3_best.pth'):\n",
    "        print(\"Found existing best model, loading...\")\n",
    "        model.load_state_dict(torch.load('sota_v3_best.pth'))\n",
    "        # Optionally continue training or skip\n",
    "    else:\n",
    "        model = train_v3(model, train_loader, val_loader, CONFIG['epochs'], device)\n",
    "    \n",
    "    evaluate_manifold(model, test_loader, device)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ac7c9a05",
   "metadata": {},
   "source": [
    "## Best Benchmark"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "4e9c333a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Loading saved model...\n",
      "\n",
      "==================================================\n",
      "EVALUATING SOTA REGRESSION\n",
      "==================================================\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 312/312 [00:11<00:00, 28.11it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--------------------------------------------------\n",
      "ResUNet Results (5% Sparsity):\n",
      "  MSE ([0,1]):  0.003199  (Target: ~0.0050)\n",
      "  PSNR:         25.44 dB  (Target: ~23 dB)\n",
      "  Inference:    0.00086 sec (Batch)\n",
      "--------------------------------------------------\n",
      "SUCCESS: Achieved SOTA Regression Level.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torch.utils.data import DataLoader, Dataset\n",
    "from torchvision import datasets, transforms\n",
    "import torchvision.utils as vutils\n",
    "import numpy as np\n",
    "from tqdm import tqdm\n",
    "import os\n",
    "import time\n",
    "\n",
    "# ============================================================================\n",
    "# 1. CONFIGURATION (SOTA SETTINGS)\n",
    "# ============================================================================\n",
    "CONFIG = {\n",
    "    'sparsity': 0.05,        # 5% visible pixels\n",
    "    'img_size': 64,          # CelebA standard size\n",
    "    'batch_size': 64,\n",
    "    'epochs': 50,            # Needs longer training to reach 0.0050\n",
    "    'lr': 2e-4,              # Lower learning rate for precision\n",
    "    'device': 'cuda' if torch.cuda.is_available() else 'cpu'\n",
    "}\n",
    "\n",
    "# ============================================================================\n",
    "# 2. DATASET\n",
    "# ============================================================================\n",
    "class SparseCelebA(Dataset):\n",
    "    def __init__(self, root, split='train', sparsity=0.05, img_size=64):\n",
    "        self.transform = transforms.Compose([\n",
    "            transforms.CenterCrop(178),\n",
    "            transforms.Resize(img_size),\n",
    "            transforms.ToTensor(),\n",
    "            transforms.Normalize((0.5, 0.5, 0.5), (0.5, 0.5, 0.5))\n",
    "        ])\n",
    "        self.data = datasets.CelebA(root=root, split=split, download=False, transform=self.transform)\n",
    "        self.sparsity = sparsity\n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self.data)\n",
    "\n",
    "    def __getitem__(self, idx):\n",
    "        img, _ = self.data[idx]\n",
    "        mask = (torch.rand_like(img) < self.sparsity).float()\n",
    "        sparse_img = img * mask\n",
    "        return sparse_img, mask, img\n",
    "\n",
    "# ============================================================================\n",
    "# 3. SOTA ARCHITECTURE: DEEP RES-UNET\n",
    "# ============================================================================\n",
    "# This is stronger than a standard U-Net because it uses Residual Blocks\n",
    "# in the bottleneck, allowing it to learn \"identity\" mappings easily.\n",
    "\n",
    "class ResBlock(nn.Module):\n",
    "    def __init__(self, c):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.BatchNorm2d(c), nn.ReLU(), nn.Conv2d(c, c, 3, 1, 1, bias=False),\n",
    "            nn.BatchNorm2d(c), nn.ReLU(), nn.Conv2d(c, c, 3, 1, 1, bias=False)\n",
    "        )\n",
    "    def forward(self, x): return x + self.net(x)\n",
    "\n",
    "class ResUNet(nn.Module):\n",
    "    def __init__(self, in_ch=6, out_ch=3):\n",
    "        super().__init__()\n",
    "        \n",
    "        # Encoder (Downsampling)\n",
    "        self.enc1 = nn.Sequential(nn.Conv2d(in_ch, 64, 4, 2, 1), nn.ReLU()) # 64 -> 32\n",
    "        self.enc2 = nn.Sequential(nn.Conv2d(64, 128, 4, 2, 1), nn.BatchNorm2d(128), nn.ReLU()) # 32 -> 16\n",
    "        self.enc3 = nn.Sequential(nn.Conv2d(128, 256, 4, 2, 1), nn.BatchNorm2d(256), nn.ReLU()) # 16 -> 8\n",
    "        self.enc4 = nn.Sequential(nn.Conv2d(256, 512, 4, 2, 1), nn.BatchNorm2d(512), nn.ReLU()) # 8 -> 4\n",
    "        \n",
    "        # Bottleneck (Deep Residual Processing)\n",
    "        # This is the \"Brain\" of the regression model\n",
    "        self.bottleneck = nn.Sequential(\n",
    "            ResBlock(512), ResBlock(512), ResBlock(512), ResBlock(512)\n",
    "        )\n",
    "        \n",
    "        # Decoder (Upsampling with Skip Connections)\n",
    "        self.dec4 = nn.Sequential(nn.ConvTranspose2d(512, 256, 4, 2, 1), nn.BatchNorm2d(256), nn.ReLU())\n",
    "        self.dec3 = nn.Sequential(nn.ConvTranspose2d(512, 128, 4, 2, 1), nn.BatchNorm2d(128), nn.ReLU()) # 256+256 inputs\n",
    "        self.dec2 = nn.Sequential(nn.ConvTranspose2d(256, 64, 4, 2, 1), nn.BatchNorm2d(64), nn.ReLU()) # 128+128 inputs\n",
    "        self.dec1 = nn.Sequential(nn.ConvTranspose2d(128, 64, 4, 2, 1), nn.BatchNorm2d(64), nn.ReLU()) # 64+64 inputs\n",
    "        \n",
    "        self.final = nn.Sequential(nn.Conv2d(64, out_ch, 3, 1, 1), nn.Tanh())\n",
    "\n",
    "    def forward(self, x, mask):\n",
    "        # Concatenate Input + Mask\n",
    "        inp = torch.cat([x, mask], dim=1)\n",
    "        \n",
    "        # Encode\n",
    "        e1 = self.enc1(inp)\n",
    "        e2 = self.enc2(e1)\n",
    "        e3 = self.enc3(e2)\n",
    "        e4 = self.enc4(e3)\n",
    "        \n",
    "        # Bottleneck\n",
    "        b = self.bottleneck(e4)\n",
    "        \n",
    "        # Decode with Skips\n",
    "        d4 = self.dec4(b)\n",
    "        d3 = self.dec3(torch.cat([d4, e3], 1))\n",
    "        d2 = self.dec2(torch.cat([d3, e2], 1))\n",
    "        d1 = self.dec1(torch.cat([d2, e1], 1))\n",
    "        \n",
    "        return self.final(d1)\n",
    "\n",
    "# ============================================================================\n",
    "# 4. TRAINING LOOP (Optimized for L1 Loss)\n",
    "# ============================================================================\n",
    "def train_sota(model, loader, val_loader, epochs, device):\n",
    "    # L1 Loss is crucial for SOTA Regression. \n",
    "    # MSE Loss often causes \"blur\" which paradoxically results in worse metrics.\n",
    "    criterion = nn.L1Loss() \n",
    "    optimizer = torch.optim.Adam(model.parameters(), lr=CONFIG['lr'])\n",
    "    scheduler = torch.optim.lr_scheduler.ReduceLROnPlateau(optimizer, 'min', patience=3, factor=0.5)\n",
    "    \n",
    "    print(f\"Training ResUNet on {device}...\")\n",
    "    \n",
    "    for epoch in range(epochs):\n",
    "        model.train()\n",
    "        loop = tqdm(loader, desc=f\"Epoch {epoch+1}/{epochs}\")\n",
    "        train_loss = 0\n",
    "        \n",
    "        for sparse, mask, full in loop:\n",
    "            sparse, mask, full = sparse.to(device), mask.to(device), full.to(device)\n",
    "            \n",
    "            optimizer.zero_grad()\n",
    "            recon = model(sparse, mask)\n",
    "            \n",
    "            # Reconstruction Loss\n",
    "            loss = criterion(recon, full)\n",
    "            \n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            \n",
    "            train_loss += loss.item()\n",
    "            loop.set_postfix(L1_loss=loss.item())\n",
    "            \n",
    "        # Validation\n",
    "        model.eval()\n",
    "        val_mse = 0\n",
    "        with torch.no_grad():\n",
    "            for sparse, mask, full in val_loader:\n",
    "                sparse, mask, full = sparse.to(device), mask.to(device), full.to(device)\n",
    "                recon = model(sparse, mask)\n",
    "                # We calculate MSE for validation monitoring (since that's our target metric)\n",
    "                val_mse += F.mse_loss(recon, full).item()\n",
    "        \n",
    "        avg_val_mse = val_mse / len(val_loader)\n",
    "        scheduler.step(avg_val_mse)\n",
    "        print(f\"  >> Val MSE: {avg_val_mse:.6f} | LR: {optimizer.param_groups[0]['lr']:.2e}\")\n",
    "        \n",
    "    return model\n",
    "\n",
    "# ============================================================================\n",
    "# 5. SCIENTIFIC EVALUATION\n",
    "# ============================================================================\n",
    "def evaluate_sota(model, loader, device):\n",
    "    print(\"\\n\" + \"=\"*50)\n",
    "    print(\"EVALUATING SOTA REGRESSION\")\n",
    "    print(\"=\"*50)\n",
    "    \n",
    "    model.eval()\n",
    "    mses = []\n",
    "    psnrs = []\n",
    "    times = []\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        for i, (sparse, mask, full) in enumerate(tqdm(loader)):\n",
    "            sparse, mask, full = sparse.to(device), mask.to(device), full.to(device)\n",
    "            \n",
    "            # Timing\n",
    "            start = time.time()\n",
    "            recon = model(sparse, mask)\n",
    "            if i > 0: # Skip first batch for warm-up\n",
    "                times.append(time.time() - start)\n",
    "            \n",
    "            # Unnormalize to [0, 1]\n",
    "            recon_01 = torch.clamp((recon + 1) / 2, 0, 1)\n",
    "            full_01 = torch.clamp((full + 1) / 2, 0, 1)\n",
    "            \n",
    "            # Batch Metrics\n",
    "            batch_mse = F.mse_loss(recon_01, full_01, reduction='none').mean(dim=[1,2,3])\n",
    "            \n",
    "            for m in batch_mse:\n",
    "                mses.append(m.item())\n",
    "                if m.item() > 0:\n",
    "                    psnrs.append(-10 * np.log10(m.item()))\n",
    "    \n",
    "    avg_mse = np.mean(mses)\n",
    "    avg_psnr = np.mean(psnrs)\n",
    "    avg_time = np.mean(times) if times else 0\n",
    "    \n",
    "    print(\"-\" * 50)\n",
    "    print(f\"ResUNet Results (5% Sparsity):\")\n",
    "    print(f\"  MSE ([0,1]):  {avg_mse:.6f}  (Target: ~0.0050)\")\n",
    "    print(f\"  PSNR:         {avg_psnr:.2f} dB  (Target: ~23 dB)\")\n",
    "    print(f\"  Inference:    {avg_time:.5f} sec (Batch)\")\n",
    "    print(\"-\" * 50)\n",
    "    \n",
    "    if avg_mse <= 0.0055:\n",
    "        print(\"SUCCESS: Achieved SOTA Regression Level.\")\n",
    "    else:\n",
    "        print(\"NOTE: Train longer (>30 epochs) to reach 0.0050.\")\n",
    "\n",
    "# ============================================================================\n",
    "# MAIN\n",
    "# ============================================================================\n",
    "if __name__ == \"__main__\":\n",
    "    device = CONFIG['device']\n",
    "    \n",
    "    # Data\n",
    "    train_ds = SparseCelebA('./data', split='train', sparsity=CONFIG['sparsity'])\n",
    "    val_ds = SparseCelebA('./data', split='valid', sparsity=CONFIG['sparsity'])\n",
    "    test_ds = SparseCelebA('./data', split='test', sparsity=CONFIG['sparsity'])\n",
    "    \n",
    "    train_loader = DataLoader(train_ds, batch_size=CONFIG['batch_size'], shuffle=True, num_workers=4)\n",
    "    val_loader = DataLoader(val_ds, batch_size=CONFIG['batch_size'], shuffle=False)\n",
    "    test_loader = DataLoader(test_ds, batch_size=CONFIG['batch_size'], shuffle=False)\n",
    "    \n",
    "    # Model\n",
    "    model = ResUNet().to(device)\n",
    "    \n",
    "    # Train\n",
    "    if os.path.exists('sota_resunet.pth'):\n",
    "        print(\"Loading saved model...\")\n",
    "        model.load_state_dict(torch.load('sota_resunet.pth'))\n",
    "    else:\n",
    "        model = train_sota(model, train_loader, val_loader, CONFIG['epochs'], device)\n",
    "        torch.save(model.state_dict(), 'sota_resunet.pth')\n",
    "    \n",
    "    # Eval\n",
    "    evaluate_sota(model, test_loader, device)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f3781bfe",
   "metadata": {},
   "source": [
    "## Robust Verfication"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1820108b",
   "metadata": {},
   "source": [
    "## Robust Verification"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "0b48e730",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Running on cuda\n",
      "Loading Dataset...\n",
      "Initializing Models...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/to247392/anaconda3/envs/manitorch/lib/python3.11/site-packages/torch/cuda/__init__.py:789: UserWarning: Can't initialize NVML\n",
      "  warnings.warn(\"Can't initialize NVML\")\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Loaded V3-ResUNet weights\n",
      "Loaded sota_best_manifold weights\n",
      "Weights check complete.\n",
      "\n",
      "Evaluating V3-ResUNet...\n",
      "  [gaussian | σ=0.100] MSE: 0.002368\n",
      "  [gaussian | σ=0.200] MSE: 0.004909\n",
      "  [gaussian | σ=0.300] MSE: 0.009111\n",
      "  [gaussian | σ=0.400] MSE: 0.014833\n",
      "  [gaussian | σ=0.500] MSE: 0.021988\n",
      "  [gaussian | σ=0.600] MSE: 0.029877\n",
      "  [gaussian | σ=0.700] MSE: 0.037604\n",
      "  [gaussian | σ=0.800] MSE: 0.046209\n",
      "  [gaussian | σ=0.900] MSE: 0.053610\n",
      "  [salt_pepper | σ=0.010] MSE: 0.002897\n",
      "  [salt_pepper | σ=0.020] MSE: 0.004054\n",
      "  [salt_pepper | σ=0.030] MSE: 0.005193\n",
      "  [salt_pepper | σ=0.040] MSE: 0.006516\n",
      "  [salt_pepper | σ=0.050] MSE: 0.007749\n",
      "  [salt_pepper | σ=0.060] MSE: 0.009081\n",
      "  [salt_pepper | σ=0.070] MSE: 0.010450\n",
      "  [salt_pepper | σ=0.080] MSE: 0.011793\n",
      "  [salt_pepper | σ=0.090] MSE: 0.013249\n",
      "  [uniform | σ=0.100] MSE: 0.001986\n",
      "  [uniform | σ=0.200] MSE: 0.002642\n",
      "  [uniform | σ=0.300] MSE: 0.004189\n",
      "  [uniform | σ=0.400] MSE: 0.006166\n",
      "  [uniform | σ=0.500] MSE: 0.008775\n",
      "  [uniform | σ=0.600] MSE: 0.012197\n",
      "  [uniform | σ=0.700] MSE: 0.016198\n",
      "  [uniform | σ=0.800] MSE: 0.020595\n",
      "  [uniform | σ=0.900] MSE: 0.025372\n",
      "\n",
      "Evaluating V4-Robust...\n",
      "  [gaussian | σ=0.100] MSE: 0.003131\n",
      "  [gaussian | σ=0.200] MSE: 0.005015\n",
      "  [gaussian | σ=0.300] MSE: 0.008496\n",
      "  [gaussian | σ=0.400] MSE: 0.013181\n",
      "  [gaussian | σ=0.500] MSE: 0.018891\n",
      "  [gaussian | σ=0.600] MSE: 0.025000\n",
      "  [gaussian | σ=0.700] MSE: 0.030766\n",
      "  [gaussian | σ=0.800] MSE: 0.036754\n",
      "  [gaussian | σ=0.900] MSE: 0.043045\n",
      "  [salt_pepper | σ=0.010] MSE: 0.003389\n",
      "  [salt_pepper | σ=0.020] MSE: 0.004166\n",
      "  [salt_pepper | σ=0.030] MSE: 0.005074\n",
      "  [salt_pepper | σ=0.040] MSE: 0.006166\n",
      "  [salt_pepper | σ=0.050] MSE: 0.007023\n",
      "  [salt_pepper | σ=0.060] MSE: 0.008209\n",
      "  [salt_pepper | σ=0.070] MSE: 0.009422\n",
      "  [salt_pepper | σ=0.080] MSE: 0.010485\n",
      "  [salt_pepper | σ=0.090] MSE: 0.011622\n",
      "  [uniform | σ=0.100] MSE: 0.002790\n",
      "  [uniform | σ=0.200] MSE: 0.003392\n",
      "  [uniform | σ=0.300] MSE: 0.004459\n",
      "  [uniform | σ=0.400] MSE: 0.006115\n",
      "  [uniform | σ=0.500] MSE: 0.008336\n",
      "  [uniform | σ=0.600] MSE: 0.011050\n",
      "  [uniform | σ=0.700] MSE: 0.014321\n",
      "  [uniform | σ=0.800] MSE: 0.017854\n",
      "  [uniform | σ=0.900] MSE: 0.021625\n",
      "Plot saved to noise_comparison.png\n",
      "Visual reconstruction saved.\n",
      "Done.\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAABv4AAAHqCAYAAADMEzkrAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjcsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvTLEjVAAAAAlwSFlzAAAPYQAAD2EBqD+naQABAABJREFUeJzs3XdYFNcaBvB3dylLR6lKEcWCBSuK2LBgsAdjj4k1JjGxkmg0sZfYDcZY4k0siaIGe4sGscQoKopYEjV2VKSqgCBtd+4fyMSVIigwsPv+nofnuXvmzMz3AVe+zDlzjkwQBAFEREREREREREREREREVK7JpQ6AiIiIiIiIiIiIiIiIiN4eB/6IiIiIiIiIiIiIiIiItAAH/oiIiIiIiIiIiIiIiIi0AAf+iIiIiIiIiIiIiIiIiLQAB/6IiIiIiIiIiIiIiIiItAAH/oiIiIiIiIiIiIiIiIi0AAf+iIiIiIiIiIiIiIiIiLQAB/6IiIiIiIiIiIiIiIiItAAH/oiIiIiIiIiIiIiIiIi0AAf+iIi0yLFjxyCTyXDs2DGpQyEiIiLSOTKZDDNmzJA6DCIiIiJJrV+/HjKZDHfv3pU6FCKdxIE/ItJw584djBo1CjVr1oSxsTGMjY1Rp04dfP7557h06VK+502cOBEymQz9+vXL83jOgNS2bdvyPD5q1CjIZDKNtoyMDCxbtgyNGjWCubk5LC0tUbduXXz88ce4du2aRt/Lly+jd+/eqFKlCpRKJRwcHNCxY0csX75co5+Liwu6deuWZwxPnz6FUqmETCbD1atX8+wzZMgQyGQy1K9fH4Ig5Douk8kwatSoPM99NQ6ZTIbRo0fnOva67xURERGVvsLWGi/r27cvZDIZvvrqqzyP5/c3P6dOeN3X+vXrCxV7Tv2S82Vubo4GDRpgyZIlSE9PF/vNmDGjwPtFR0cDAO7evavRrlAo4OzsjJ49eyIiIkLj3i/3k8vlqFy5Mt55551ck5QKyrlTp075xqivrw8XFxeMGTMGT58+LdL3423rOSIiIip9ObVAfHx8nsfr1auHtm3bip9frlu2b99eqOsNGTIEpqamGv3atm2bb63y6jOqyMhIfPrpp3BxcYGhoSFsbW3h5+eHkydP5rp/Tj34cl1la2uL3r175/lsKqeOMTc3x/Pnz3Mdv3HjhnitxYsX5/k9ellO3yVLluQ6ljN4d+7cuddeh4jKFj2pAyCismPfvn3o168f9PT0MHDgQDRo0AByuRzXrl3Djh07sGrVKty5cwdVqlTROE8QBGzevBkuLi7Yu3cvkpOTYWZm9tbx9OrVC7///jsGDBiAESNGIDMzE9euXcO+ffvQokULuLm5AQBOnTqFdu3awdnZGSNGjIC9vT3u37+P06dPY9myZXkOruUlKCgIMpkM9vb22LRpE+bMmZNv38uXL2PHjh3o1avXW+X4v//9D5MnT0blypXf6jo52rRpg+fPn8PAwKBYrkdERERvVmskJSVh7969cHFxwebNmzF//vxck5zyExAQgGfPnomfDxw4gM2bN+O7776DtbW12N6iRYtC52BoaIiffvoJQPZkp+3bt+PLL79EWFgYtmzZotF31apVuR52AYClpaXG5wEDBqBLly5QqVS4evUqVq1ahd9//x2nT59Gw4YNxX4dO3bEoEGDIAgC7ty5g5UrV6J9+/bYv38/OnfuLPZr2LAhvvjii1z3zatOyokxJSUFISEhWL58OcLDw/HXX38V+ntSXPXcy54/fw49Pf5nNhERUVk0a9YsvPfee4WuyV7l6OiIefPm5Wp/uVY5efIkunTpAgD46KOPUKdOHURHR2P9+vVo3bp1vrXjmDFj0LRpU2RmZuLSpUtYvXo1jh07hitXrsDe3l6jr56eHlJTU7F371707dtX49imTZugVCqRlpZWpNwWLVqEkSNHwtjYuEjn5efDDz9E//79YWhoWCzXI6Ki4X+REBEA4NatW+jfvz+qVKmCkJAQVKpUSeP4ggULsHLlSsjluV8UPnbsGB48eIAjR47A19cXO3bswODBg98qnrCwMOzbtw9z587F119/rXHshx9+0JjRPXfuXFhYWCAsLCzXA6nY2NhC33Pjxo3o0qULqlSpgsDAwHwH/oyMjODk5PTWBWPdunVx/fp1zJ8/H99///0bXeNVcrkcSqWyWK5FRERE2d6k1ti+fTtUKhXWrl2L9u3b488//4S3t3eh7ufn56fxOTo6Gps3b4afnx9cXFzeIIPsB0QffPCB+Pmzzz6Dp6cntm7diqVLl2o8sOrdu7fGAGN+GjdurHHNli1bokePHli1ahV+/PFHsb1mzZoa/Xr27In69esjICBAY+DPwcFBo19BXo7xk08+Qf/+/bF161acPXsWzZo1e+35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iEhyHPgjIiIiIiIqh86fP48nT57ke9zIyAgtW7YsxYiIiIiIiIhIahz4IyIiIiIiIiIiIiIiItICcqkDICIiIiIiIiIiIiIiIqK3pyd1AGWRWq1GVFQUzMzMIJPJpA6HiIiISpEgCEhOTkblypUhl3OO1NtiXUVERKS7WFcVL9ZVREREuqsodRUH/vIQFRUFJycnqcMgIiIiCd2/fx+Ojo5Sh1Husa4iIiIi1lXFg3UVERERFaau4sBfHszMzABkfwPNzc2L9dpqtRpxcXGwsbHR+tluupKrruQJ6E6uupInoDu5Mk/tU5K5JiUlwcnJSawH6O2wrioeupKrruQJ6E6uzFP76EquupInwLqqPGFdVTx0JVddyRPQnVyZp/bRlVx1JU+g7NRVHPjLQ85yCebm5iVSSKWlpcHc3Fwnfsl1IVddyRPQnVx1JU9Ad3JlntqnNHLl8knFg3VV8dCVXHUlT0B3cmWe2kdXctWVPAHWVeUJ66rioSu56kqegO7kyjy1j67kqit5AmWnrtLu7zIRERERERERERERERGRjuDAHxEREREREREREREREZEW4MAfERERERERERERERERkRbgHn9vQaVSITMzs0jnqNVqZGZmIi0tTSfWs9XmXPX19aFQKKQOg4iISCuwriqYtufKuoqIiKj4sK4qmLbnyrqKiIg48PcGBEFAdHQ0nj59+kbnqtVqJCcna/3m1rqQq6WlJWxtbaUOg4iIqNxiXVU4upAr6yoiIqK3w7qqcHQhV9ZVRES6jQN/byCniLK1tYWxsXGRigRBEJCVlQU9PT2tLS5yaHOugiAgNTUVsbGxEARBK2eIERERlQbWVYWjzbmyriIiIioerKsKR5tzZV1FREQAB/6KTKVSiUWUlZVVkc/X5uLiVdqeq5GREQAgJiYGlpaW0gZDRERUDrGuKjxtz5V1FRER0dthXVV42p4r6yoiIuK0jyLKWSPd2NhY4kioLMj5PVCpVBJHQkREVP6wrqKXsa4iIiJ6c6yr6GWsq4iIdBsH/t6QNs4IoqLj7wEREdHb499TAvh7QEREVBz495QA/h4QEek6DvwRERERERERERERERERaQEO/BERERERERERERERERFpAQ78SUSlFhB2Lxa//x2JsHuxUKmFEr1f9+7d0alTpzyPnThxAjKZDMePH0enTp1QuXJlGBoawsnJ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7GCj5E3rRERE/QELfwZWeqnlVLrLEGQ4D4b7+QpY1TWiWilH7mBbiEubUouuH47/NyGIm1REREREHSht1cpTH/OIiIiITFVn10vONhYI9XDQFvr8VXZwYptOIiKifo2FPwNr/T4+IZHg3BC7dueFuDtw0URERER0FZ19zzHfh0xERER0dZ1dL718azjCvVQGjoaIiIj0ib0kDSzUw7H5TqgOjksAONlYINTDsTfDIiIiIhpwuK4iIiIi0g+uq4iIiIwXC38GJpNK8PS0EABos5hq+frpaSGQSfm0HxEREdHVcF1FREREpB9cVxERERkvFv56wdQAN7x1eyRUNhY64yobC7x1eySmBrj1UWREREREAwvXVURERET6wXUVERGRceI7/nrJ1AA3TPZ3RXxOMTLzi+HnqsJYTxXvnCIiIiLqIq6riIiIiPSD6yoiIiLjw8JfL5JJJQjzdISnUkClcoSUiygiIiKibuG6ioiIiEg/uK4iIiIyLmz1SURERERERERERERERGQEWPgjIiIiIiIiIiIiIiIiMgIs/BEREREREREREREREREZARb+iIiIiIiIiIiIiIiIiIwAC39ERERERERERERERERERoCFPyIiIiIiIiIiIiIiIiIj0C8Kfx9++CG8vb2hVCoRERGBw4cPX3X+999/j8DAQCiVSowcORK//fabznEhBJ577jm4uLjAwsICU6dORXp6uiFTICIiIiIiIiIiIiIiIupTfV7427RpE5YuXYrnn38eCQkJGD16NKKjo1FcXNzu/AMHDuDuu+/Gww8/jGPHjmHOnDmYM2cOkpKStHPeeOMNvP/++1i7di0OHToEKysrREdHo66urrfSIiIiIiIiIiIiIiIiIupVfV74e+edd7Bw4UI89NBDGD58ONauXQtLS0usX7++3fnvvfcepk+fjqeeegpBQUF46aWXEBoaig8++ABA89N+q1evxvLlyzF79myMGjUKX375JfLz8xETE9OLmRERERERERERERERERH1HrO+/OYNDQ2Ij4/HsmXLtGNSqRRTp05FXFxcu+fExcVh6dKlOmPR0dHaol5WVhYKCwsxdepU7XE7OztEREQgLi4Od911V5tr1tfXo76+Xvt1RUUFAECj0UCj0XQ7v/ZoNBoIIfR+3f7IVHI1lTwB08nVVPIETCdX5ml8DJmrKfz+ERERERER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AG1+RUVF9X7g3dCVXL28vNrN9fnnn+/9wLuoK3k+++yzYujQoUKpVAp7e3sRGRkpNm7c2AdRd09Xcr3SQFpMdSXPJUuWaOc6OTmJW265RSQkJPRB1F3X1c/zwIEDIiIiQigUCuHr6yteeeUV0dTU1MtRd09Xc01NTRUAxPbt23s50p7pSp6NjY3ihRdeEH5+fkKpVAoPDw/xj3/8Y8AUw7qS6549e0RQUJBQKBRiyJAh4v777xd5eXl9EDX1VFf/W/7uu+/EsGHDhLm5uQgODhZbt27VOf7555/3y/WHPvPsaD0JQOzevbuXMuqYPnOtra0Vt912m3B1dRXm5ubCxcVFzJo1Sxw+fLi30umQvv/sXqm/FP6E0G+uNTU14qabbhKOjo5CLpcLLy8vsXDhQu0Pzn3JEJ/pZ599pv1ZYfTo0SImJsbQaVyTIfL85JNPhIWFhbh48aKhw+8SfedaUFAgHnzwQeHq6iqUSqUICAgQb7/9ttBoNL2RTof0ned7770n3N3dhVwuF56enmL58uV9fmMJdQ/3q5pxv2rg7VeZyl6VENyvEoL7Vdyv6v8Gwn6VRIh+/OwkEREREREREREREREREXUK3/FHREREREREREREREREZARY+CMiIiIiIiIiIiIiIiIyAiz8ERERERERERERERERERkBFv6IiIiIiIiIiIiIiIiIjAALf0RERERERERERERERERGgIU/IiIiIiIiIiIiIiIiIiPAwh8RERERERERERERERGREWDhj4iIiIiIiIiIiIiIiMgIsPBHREREREREREREREREZARY+COifsnb2xurV6/u6zAMZs+ePZBIJLh48eJV58XGxiIoKAhqtbrT177rrrvw9ttv9zBCIiIiMhZcVzXjuoqIiIh6iuuqZlxXEfVvLPwRkd48+OCDkEgkeO2113TGY2JiIJFIunStI0eO4JFHHtFneDoGykLt6aefxvLlyyGTyTp9zvLly/HKK6+gvLzcgJERERGRIXFdpX9cVxEREZkmrqv0j+sqov6NhT8i0iulUonXX38dFy5c6NF1HB0dYWlpqaeoBqa//voLmZmZuOOOO7p03ogRI+Dn54evv/7aQJERERFRb+C6Sn+4riIiIjJtXFfpD9dVRP0fC39EpFdTp06Fs7MzVq1addV5P/74I4KDg6FQKODt7d3mUf/WdzgJIfDC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",
      "text/plain": [
       "<Figure size 1800x500 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAABJIAAAEwCAYAAADsAVtdAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjcsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvTLEjVAAAAAlwSFlzAAAPYQAAD2EBqD+naQAA5q5JREFUeJzsnXe8HVW5/t+Z2f30c9IpCb1LFRGpiqAgRUBEQKqigKJcy+9argTbRcFyEVEQAUWDgNIUkY5YKEpREGkhAUJJPXX3PTO/P7g5l+dZw977pJCgz/fzyR/v1DVr1qy1ZnLe7/biOI5NCCGEEEIIIYQQQogW+Gu6AEIIIYQQQgghhBDijYE+JAkhhBBCCCGEEEKIttCHJCGEEEIIIYQQQgjRFvqQJIQQQgghhBBCCCHaQh+ShBBCCCGEEEIIIURb6EOSEEIIIYQQQgghhGgLfUgSQgghhBBCCCGEEG2hD0lCCCGEEEIIIYQQoi30IUkIIYQQQgghhBBCtIU+JImVxvM8mz179pouRlOOP/546+zsXNPFEEKsQq666irr7++3sbGxNV2UVc5jjz1mqVTKHn300TVdFCGEEEL8G7PXXnvZ1ltvvaaLIdYy9CHpdWLevHn2sY99zDbddFMrFApWKBRsyy23tNNOO83+/ve/r+nirVb22msv8zyv5b+V/RhVKpVs9uzZdtddd62Scgvxr8pll11mnufZX//61zVdFDNbsWc3DEM788wz7eMf//jr/pF4aGjITj75ZJs8ebJ1dHTY3nvvbQ8++GDL/aIosssuu8wOOuggW2+99ayjo8O23npr++pXv2qVSgW23XLLLe2AAw6wL33pS6vrMoR4Q3HQQQdZoVCw0dHR19zm6KOPtkwmY0uXLrUzzjjDdthhB+vv77dCoWBbbLGFzZ49u+0Pz7Nnz4Y5SjqdtlmzZtnpp59uQ0NDq+iqXJq9sC1ZssSZLy3vz3O5nL3wwgsTOl4r5syZY9/97ndXaF8hxCtMtO96NXPnzrVcLjehOdvxxx8PfVc2m7VNN93UvvSlLzlzjbWdr3/963bdddet6WKI1yC1pgvw78BvfvMbe//732+pVMqOPvpo23bbbc33fXv88cftmmuusR/84Ac2b948mzlz5pou6mrhC1/4gn3oQx8aj//yl7/YeeedZ5///Odtiy22GF/+pje9aaXOUyqV7KyzzjKzVyZOQog3Bivy7P7617+2J554wk4++eTVWDKXKIrsgAMOsL/97W/2mc98xiZNmmQXXHCB7bXXXvbAAw/YJpts8pr7lkolO+GEE2yXXXaxj370ozZlyhS755577Mwzz7Tbb7/d7rjjDvM8b3z7j370o7b//vvb3LlzbaONNno9Lk+ItZajjz7afv3rX9u1115rxx57rLO+VCrZ9ddfb+9617tsYGDA/vKXv9juu+9uJ5xwguVyOXvooYfs7LPPtttuu83uvvtu8/32/i/1Bz/4gXV2dlqxWLTbb7/dvve979mDDz5of/zjH1f1Ja4U1WrVzj77bPve9763yo45Z84ce/TRR+2Tn/zkKjumEP9uTLTvejVnnHGGpVIpq1arEzpnNpu1iy++2MzMhoeH7frrr7evfOUrNnfuXPv5z3++4hfzOvP1r3/dDj/8cDvkkEPWdFFEAvqQtJqZO3euHXnkkTZz5ky7/fbbbfr06bD+G9/4hl1wwQUtJzTFYtE6OjpWZ1FXG+985zshzuVydt5559k73/nOpi+Nb+RrFkKsXi699FJ729veZuuss87ret5f/vKX9uc//9muvvpqO/zww83M7IgjjrBNN93UzjzzTJszZ85r7pvJZOxPf/qT7brrruPLPvzhD9usWbPGPybts88+4+v22Wcf6+vrs5/85Cf25S9/efVdlBBvAA466CDr6uqyOXPmJL6MXX/99VYsFu3oo482M0v80LPRRhvZpz/9abv//vttl112aeu8hx9+uE2aNMnMzD7ykY/YkUceaVdeeaXdf//9tvPOO6/EFa1atttuO/vRj35kn/vc52zGjBlrujhCiP9lon3Xcm6++Wa7+eab7bOf/ax99atfndA5U6mUHXPMMePxqaeearvuuqtdccUV9u1vf9umTp26YhcjxKtQattq5pvf/KYVi0W79NJLnY9IZq886Keffrqtt95648uW+3zmzp1r+++/v3V1dY13LsVi0T71qU/ZeuutZ9ls1jbbbDM799xzLY7j8f3nz59vnufZZZdd5pyP/yR6+Z9uP/3003b88cdbb2+v9fT02AknnGClUgn2rVardsYZZ9jkyZOtq6vLDjroIFuwYMFK1hCW47HHHrOjjjrK+vr6bLfddjOzV/5CIemD0/HHH2+zZs0av+bJkyebmdlZZ531mulyL7zwgh1yyCHW2dlpkydPtk9/+tMWhuEquQYh3sgs73daPSPL+5dzzz3XvvOd79jMmTMtn8/bnnvu6fh8VuWz+2oqlYr97ne/g48uyxkcHLSPfOQjts4661gQBKs8hfaXv/ylTZ061Q499NDxZZMnT7YjjjjCrr/++qb/a5jJZOAj0nLe+973mpnZP//5T1ieTqdtr732suuvv36lyizEvwL5fN4OPfRQu/32223RokXO+jlz5ozPTV6L5f3OyqSm7b777mb2yn8Uvpr77rvP3vWud1lPT48VCgXbc8897U9/+hNsMzo6ap/85Cdt1qxZls1mbcqUKfbOd76zrdTYVnz+85+3MAzt7LPPbmv7n/3sZ7bjjjtaPp+3/v5+O/LII+35558fX7/XXnvZjTfeaM8+++x4/7m8/oQQ7bMifVe9XrdPfOIT9olPfGKV/EWy53m22267WRzH9swzz8C6Cy64wLbaaivLZrM2Y8YMO+20016zj3zggQds1113tXw+bxtssIH98Ic/hPXLU23nz58Py++66y7zPA8UBk899ZQddthhNm3aNMvlcrbuuuvakUceacPDw+NlLhaL9pOf/GS8Dzr++ONXtirEKkQfklYzv/nNb2zjjTe2t7zlLRPar9Fo2H777WdTpkyxc8891w477DCL49gOOugg+853vmPvete77Nvf/rZtttlm9pnPfMb+4z/+Y6XKecQRR9jo6Kj993//tx1xxBF22WWXjaeaLOdDH/qQffe737V9993Xzj77bEun03bAAQes1HmZ973vfVYqlezrX/+6ffjDH257v8mTJ9sPfvADM3vlpezyyy+3yy+/HF72wjC0/fbbzwYGBuzcc8+1Pffc0771rW/ZRRddtEqvQYg3KhN5Rn7605/aeeedZ6eddpp97nOfs0cffdTe/va328KFCyd0znaeXeaBBx6wWq1mO+ywAyyP49je+9732kUXXWR77LGHfe9737MTTzzRzF6ZyB188MG2+eabm9krk7QlS5a09S+KovFzPPTQQ7bDDjs4f0W68847W6lUsieffHJC129m9vLLL5uZjf/Vw6vZcccd7dFHH7WRkZEJH1eIfzWOPvpoazQadtVVV8HyZcuW2c0332zvfe97LZ/Pjy9vNBq2ZMkSe/HFF+2WW26xL37xi9bV1bVSf0m0/AWpr69vfNkdd9xhe+yxh42MjNiZZ55pX//6121oaMje/va32/333z++3Uc/+lH7wQ9+YIcddphdcMEF9ulPf9ry+bzzEXlF2GCDDezYY4+1H/3oR/biiy823fZrX/uaHXvssbbJJpvYt7/9bfvkJz9pt99+u+2xxx7jL5Bf+MIXbLvttrNJkyaN98vyJQmxYky07/rud79rg4OD9sUvfnGVlSGp75o9e7addtppNmPGDPvWt75lhx12mF144YW27777Wr1eh/0HBwdt//33tx133NG++c1v2rrrrmunnHKKXXLJJRMuS61Ws/3228/uvfde+/jHP27f//737eSTT7ZnnnlmvA+6/PLLLZvN2u677z7eB33kIx9Z4esXq4FYrDaGh4djM4sPOeQQZ93g4GC8ePHi8X+lUml83XHHHRebWfyf//mfsM91110Xm1n81a9+FZYffvjhsed58dNPPx3HcRzPmzcvNrP40ksvdc5rZvGZZ545Hp955pmxmcUnnngibPfe9743HhgYGI8ffvjh2MziU089FbY76qijnGO24uqrr47NLL7zzjudcnzgAx9wtt9zzz3jPffc01l+3HHHxTNnzhyPFy9e/JplWV6nX/7yl2H59ttvH++4445tl12IfwUuvfTS2Mziv/zlL+PL2n1Glvcv+Xw+XrBgwfjy++67Lzaz+Iwzzhhftiqe3SQuvvji2MziRx55BJbfddddsZnFp5xyCiz/3Oc+F5tZ/Pe//3182Z133hmbWVv/5s2bN75fR0eH01/GcRzfeOONsZnFv/vd79q6hlezzz77xN3d3fHg4KCzbs6cObGZxffdd9+EjyvEvxqNRiOePn16/Na3vhWW//CHP4zNLL755pth+T333APP8mabbQZzj2Ysn5c88cQT8eLFi+P58+fHl1xySZzP5+PJkyfHxWIxjuM4jqIo3mSTTeL99tsvjqJofP9SqRRvsMEG8Tvf+c7xZT09PfFpp53W9Lx77rlnvNVWWyWuS+orX92fz507N06lUvHpp5/+msebP39+HARB/LWvfQ2O/cgjj8SpVAqWH3DAAdBXCyFWjIn0XS+99FLc1dUVX3jhhXEcJ8/ZmnHcccfFHR0d4++YTz/9dHzuuefGnufFW2+99Xg/tWjRojiTycT77rtvHIbh+P7nn39+bGbxJZdcMr5szz33jM0s/ta3vjW+rFqtxtttt108ZcqUuFarQVlfPW+K4/+bcy3vfx966KHYzOKrr7666bV0dHTExx13XFvXLV5/9BdJq5Hl/4Oc9ItCe+21l02ePHn83/e//31nm1NOOQXi3/72txYEgZ1++umw/FOf+pTFcWw33XTTCpf1ox/9KMS77767LV26dPwafvvb35qZOede1QJGLseqJuk6+U88hfh3pt1n5JBDDgE/0c4772xvectbxvuK1cnyXzV59f+qmZndeeedZmb2iU98ApYvj1/9P4Hbbrut3XrrrW39mzZt2vh+5XLZstmsU6ZcLje+fiJ8/etft9tuu83OPvts6+3tddYvv8YlS5ZM6LhC/CsSBIEdeeSRds8990DqxJw5c2zq1Kn2jne8A7bfcsst7dZbb7XrrrvOPvvZz1pHR0fbv9q2nM0228wmT55ss2bNshNPPNE23nhju+mmm6xQKJiZ2cMPP2xPPfWUHXXUUbZ06dLxv2QsFov2jne8w+6+++7xv2rs7e21++67r+VfDK0oG264oX3wgx+0iy66yF566aXEba655hqLosiOOOII+MvLadOm2SabbDLejwohVh0T6bv+3//7f7bhhhvCDxVNlGKxOP6OufHGG9unP/1pe9vb3mbXX3/9+I963HbbbVar1eyTn/wk/JX1hz/8Yevu7rYbb7wRjplKpeAvgjKZjH3kIx+xRYsW2QMPPDCh8vX09JjZKx4oVqmINw6Sba9Gurq6zMwSJy0XXnihjY6O2sKFC0GGtpxUKmXrrrsuLHv22WdtxowZ48ddzvJfPnv22WdXuKzrr78+xMtfXgYHB627u9ueffZZ833fydPdbLPNVvicSWywwQar9HivJpfLjbtYltPX12eDg4Or7ZxCvJGYyDOS9Otkm266qfNn26uT+FVuODOzF1980YIgsI033hiWT5061fr6+uCDWF9fX6JjqRX5fD7Rg7T8J3Vf/afprbjyyivti1/8op100knOfxwsZ/k1vvrX3IT4d+boo4+273znOzZnzhz7/Oc/bwsWLLA//OEPdvrpp1sQBLBtd3f3+HN+8MEH25w5c+zggw+2Bx980Lbddlur1Wq2bNky2Gfy5MlwnF/96lfW3d1tixcvtvPOO8/mzZsHz/lTTz1lZmbHHXfca5Z5eHjY+vr67Jvf/KYdd9xxtt5669mOO+5o+++/vx177LG24YYbTqgOmvUHX/ziF+3yyy+3s88+2/7nf/7HWf/UU09ZHMev+QuT6XR6QmURQrRHO33Xvffea5dffrndfvvtTX+IqVwuj7uElvPq//TK5XL261//2szMFixYYN/85jdt0aJF0Hctf2/kd7lMJmMbbrih8145Y8YM50eQNt10UzN7JW2u3R8wMHvlfe8//uM/7Nvf/rb9/Oc/t913390OOuggO+aYY8Y/Mom1H31IWo309PTY9OnTHQGtmY07k1hGtpxsNtv2T9MyrzXBaCaV5snXcvhFbXWT9BLmeV5iOSYqyX6taxRCvMKqfkZW1bPLLP953MHBQfjgHgSBxXFsURQ51xKGIeT7J71AvhavfrGcPn164v/0L1/W7q8l3XrrrXbsscfaAQcc4MgqX83yj3hJ/iQh/h3ZcccdbfPNN7crrrjCPv/5z9sVV1xhcRw7v3iUxKGHHmof/OAH7Re/+IVtu+229uc//9n23ntv2GbevHkgld5jjz3Gn78DDzzQttlmGzv66KPtgQceMN/3x//a6JxzzrHtttsu8bzL/zL9iCOOsN13392uvfZau+WWW+ycc86xb3zjG3bNNdfYu9/9bjN75QXwtf6ycfn/3C//C8gkNtxwQzvmmGPsoosusv/8z/901kdRZJ7n2U033ZTY5yf9Fb0QYuVpp+/67Gc/a7vvvrttsMEG4++Iy/8i+aWXXrLnnnvO1l9/fbvyyivthBNOgOO/er4VBAH8Z9l+++1nm2++uX3kIx+xG264YbVd40TeQb/1rW/Z8ccfb9dff73dcsstdvrpp9t///d/27333uv8MYVYO9GHpNXMAQccYBdffPEq+ZnYmTNn2m233Wajo6PwV0mPP/74+Hqz//trIjbur8xfLM2cOdOiKLK5c+fCl+snnnhihY/ZLvyXBMvh69H/2Avx+rH8f+FfzZNPPgkvYKvr2V0uzJ43b55ts80248s32mgji6LInnzySdtqq63Gl7/00ks2MjICf/GY9AL5Wrz6xXK77bazP/zhDxZFEXzsv++++6xQKIz/71wz7rvvPnvve99rO+20k1111VWWSr32UDxv3jzzfb+t4wrx78LRRx9t//Vf/2V///vfbc6cObbJJpvYm9/85pb7VatVi6Jo/H/yl6e4vppX/68+09nZaWeeeaadcMIJdtVVV9mRRx45/pfar/7rp2ZMnz7dTj31VDv11FNt0aJFtsMOO9jXvva18Q9JM2fOtDvuuMPK5bLzn2vL51zL53uvxRe/+EX72c9+Zt/4xjecdRtttJHFcWwbbLBBy35F8yohVi2t+q7nnnvOnn322cQMjYMOOsh6enpsaGjI9ttvP6fvasb06dPtjDPOsLPOOsvuvfde22WXXcb7kSeeeAL+KrJWq9m8efOc/uzFF1+0YrEIf5W0/AdGls+RJvoOus0229g222xjX/ziF+3Pf/6zve1tb7Mf/vCH9tWvftXM1Aet7ciRtJr57Gc/a4VCwU488cTEXzOayF/87L///haGoZ1//vmw/Dvf+Y55njc+Cenu7rZJkybZ3XffDdtdcMEFK3AFr7D82Oeddx4sfz1+wWOjjTayxx9/3BYvXjy+7G9/+5vzs7rLfQUr87O+Qoj2uO666+yFF14Yj++//3677777xvsKs9X37O64446WyWTsr3/9Kyx/z3veY2av9Imv5lvf+paZGfzK5Io6kg4//HBbuHChXXPNNePLlixZYldffbUdeOCB4E+aO3eu8xPh//znP+2AAw6wWbNm2W9+85uWqXAPPPCAbbXVVvpTbyFexfL/wf/Sl75kDz/8sPPXSENDQ84vDpmZXXzxxWZmttNOO5nZ/6W4vvpfs7/2WX7uddddd/wjzY477mgbbbSRnXvuuYkqg+X9XxiGTirKlClTbMaMGZAuu//++1u9XrcLL7wQto2iyH7wgx9YJpNxXFDMRhttZMccc4xdeOGF478KuZxDDz3UgiCws846y5mDxnE87qAzM+vo6HDKLIRYcVr1XRdddJFde+218O/jH/+4mZmde+659vOf/9zMXvkwxH1XKz7+8Y9boVCws88+28zM9tlnH8tkMnbeeedBX/DjH//YhoeHnV/mbjQa0C/VajW78MILbfLkybbjjjuamY1/WH/1O2gYhs6v/46MjFij0YBl22yzjfm+D/1hR0eH3uvWYvQXSauZTTbZxObMmWMf+MAHbLPNNrOjjz7att12W4vj2ObNm2dz5swx3/fb+hO+Aw880Pbee2/7whe+YPPnz7dtt93WbrnlFrv++uvtk5/8JPiLPvShD9nZZ59tH/rQh2ynnXayu+++e4V+lno52223nX3gAx+wCy64wIaHh23XXXe122+/3Z5++ukVPma7nHjiifbtb3/b9ttvPzvppJNs0aJF9sMf/tC22mor+EnsfD5vW265pV155ZW26aabWn9/v2299da29dZbr/YyCvHvxsYbb2y77babnXLKKVatVu273/2uDQwM2Gc/+9nxbVbXs5vL5Wzfffe12267zb785S+PL998883t1FNPtQsuuMBGRkZsr732svvuu89++tOf2mGHHWZ77LHH+LYr6kg6/PDDbZdddrETTjjBHnvsMZs0aZJdcMEFFoahnXXWWbDt8pe95X+ePjo6avvtt58NDg7aZz7zGUdkudFGG9lb3/rW8bher9vvf/97O/XUUydcTiH+ldlggw1s1113teuvv97MzHkZu+uuu+z000+3ww8/3DbZZBOr1Wr2hz/8wa655hrbaaedEt2U7ZJOp+0Tn/iEfeYzn7Hf/e539q53vcsuvvhie/e7321bbbWVnXDCCbbOOuvYCy+8YHfeead1d3fbr3/9axsdHbV1113XDj/8cNt2222ts7PTbrvtNvvLX/4y/rHb7JW53r777mtnnHGG3X///bbrrrtaqVSyG264wf70pz/ZV7/6Vcdll8QXvvAFu/zyy+2JJ56Av9DcaKON7Ktf/ap97nOfs/nz59shhxxiXV1dNm/ePLv22mvt5JNPtk9/+tNm9spHsiuvvNL+4z/+w9785jdbZ2enHXjggStcd0L8u9Oq79p3332dfZZ/SNlzzz3HP4KvCAMDA3bCCSfYBRdcYP/85z9tiy22sM997nN21lln2bve9S476KCD7IknnrALLrjA3vzmNzv95IwZM+wb3/iGzZ8/3zbddFO78sor7eGHH7aLLrpo3K221VZb2S677GKf+9znbNmyZdbf32+/+MUvnI9Gd9xxh33sYx+z973vfbbppptao9Gwyy+/3IIgsMMOO2x8ux133NFuu+02+/a3v20zZsywDTbYYFwPI9YCXvffifs35emnn45POeWUeOONN45zuVycz+fjzTffPP7oRz8aP/zww7Dt8p9tTGJ0dDQ+44wz4hkzZsTpdDreZJNN4nPOOQd+cjaOX/nZ2ZNOOinu6emJu7q64iOOOCJetGiR87Oxy3/edvHixbB/0s83lsvl+PTTT48HBgbijo6O+MADD4yff/75Cf1sdxzH8dVXXw0/AdmsHMv52c9+Fm+44YZxJpOJt9tuu/jmm292fkI8juP4z3/+c7zjjjvGmUwGyvVadbr8vEL8O5H0U7LtPiPz5s2L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",
      "text/plain": [
       "<Figure size 1200x300 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torch.utils.data import DataLoader, Dataset\n",
    "from torchvision import datasets, transforms\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from tqdm import tqdm\n",
    "import os\n",
    "import json\n",
    "import time\n",
    "import copy\n",
    "\n",
    "# ============================================================================\n",
    "# 1. CONFIGURATION\n",
    "# ============================================================================\n",
    "CONFIG = {\n",
    "    'sparsity': 0.15,\n",
    "    'img_size': 64,\n",
    "    'batch_size': 64,\n",
    "    'device': 'cuda' if torch.cuda.is_available() else 'cpu'\n",
    "}\n",
    "\n",
    "# ============================================================================\n",
    "# 2. UNIFIED DATASET\n",
    "# ============================================================================\n",
    "class UnifiedCelebADataset(Dataset):\n",
    "    def __init__(self, root, split='test', sparsity=0.05, img_size=64):\n",
    "        self.transform = transforms.Compose([\n",
    "            transforms.CenterCrop(178),\n",
    "            transforms.Resize(img_size),\n",
    "            transforms.ToTensor(),\n",
    "            transforms.Normalize((0.5, 0.5, 0.5), (0.5, 0.5, 0.5))\n",
    "        ])\n",
    "        # Using download=True for convenience, switch to False if already downloaded\n",
    "        try:\n",
    "            self.data = datasets.CelebA(root=root, split=split, download=False, transform=self.transform)\n",
    "        except:\n",
    "            self.data = datasets.CelebA(root=root, split=split, download=True, transform=self.transform)\n",
    "        self.sparsity = sparsity\n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self.data)\n",
    "\n",
    "    def __getitem__(self, idx):\n",
    "        full_image, attributes = self.data[idx]\n",
    "        # Create mask\n",
    "        mask = (torch.rand_like(full_image) < self.sparsity).float()\n",
    "        sparse_image = full_image * mask\n",
    "        \n",
    "        # Ensure attributes are float tensors\n",
    "        attributes = attributes.float()\n",
    "        \n",
    "        # Return format compatible with all models (now including attributes)\n",
    "        return sparse_image, mask, full_image, attributes\n",
    "\n",
    "# ============================================================================\n",
    "# 3. SHARED BLOCKS\n",
    "# ============================================================================\n",
    "class ResBlock(nn.Module):\n",
    "    def __init__(self, c):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.BatchNorm2d(c), nn.ReLU(), nn.Conv2d(c, c, 3, 1, 1, bias=False),\n",
    "            nn.BatchNorm2d(c), nn.ReLU(), nn.Conv2d(c, c, 3, 1, 1, bias=False)\n",
    "        )\n",
    "    def forward(self, x): return x + self.net(x)\n",
    "\n",
    "# ============================================================================\n",
    "# 4. MODEL VERSION 3: RES-UNET (Deep Regression Benchmark)\n",
    "# ============================================================================\n",
    "class ResUNet(nn.Module):\n",
    "    def __init__(self, in_ch=6, out_ch=3):\n",
    "        super().__init__()\n",
    "        self.enc1 = nn.Sequential(nn.Conv2d(in_ch, 64, 4, 2, 1), nn.ReLU()) \n",
    "        self.enc2 = nn.Sequential(nn.Conv2d(64, 128, 4, 2, 1), nn.BatchNorm2d(128), nn.ReLU()) \n",
    "        self.enc3 = nn.Sequential(nn.Conv2d(128, 256, 4, 2, 1), nn.BatchNorm2d(256), nn.ReLU()) \n",
    "        self.enc4 = nn.Sequential(nn.Conv2d(256, 512, 4, 2, 1), nn.BatchNorm2d(512), nn.ReLU()) \n",
    "        \n",
    "        self.bottleneck = nn.Sequential(\n",
    "            ResBlock(512), ResBlock(512), ResBlock(512), ResBlock(512)\n",
    "        )\n",
    "        \n",
    "        self.dec4 = nn.Sequential(nn.ConvTranspose2d(512, 256, 4, 2, 1), nn.BatchNorm2d(256), nn.ReLU())\n",
    "        self.dec3 = nn.Sequential(nn.ConvTranspose2d(512, 128, 4, 2, 1), nn.BatchNorm2d(128), nn.ReLU()) \n",
    "        self.dec2 = nn.Sequential(nn.ConvTranspose2d(256, 64, 4, 2, 1), nn.BatchNorm2d(64), nn.ReLU()) \n",
    "        self.dec1 = nn.Sequential(nn.ConvTranspose2d(128, 64, 4, 2, 1), nn.BatchNorm2d(64), nn.ReLU()) \n",
    "        \n",
    "        self.final = nn.Sequential(nn.Conv2d(64, out_ch, 3, 1, 1), nn.Tanh())\n",
    "\n",
    "    def forward(self, x, mask):\n",
    "        inp = torch.cat([x, mask], dim=1)\n",
    "        e1 = self.enc1(inp)\n",
    "        e2 = self.enc2(e1)\n",
    "        e3 = self.enc3(e2)\n",
    "        e4 = self.enc4(e3)\n",
    "        b = self.bottleneck(e4)\n",
    "        d4 = self.dec4(b)\n",
    "        d3 = self.dec3(torch.cat([d4, e3], 1))\n",
    "        d2 = self.dec2(torch.cat([d3, e2], 1))\n",
    "        d1 = self.dec1(torch.cat([d2, e1], 1))\n",
    "        return self.final(d1)\n",
    "    \n",
    "    def infer(self, sparse, mask, attributes=None):\n",
    "        \"\"\"Standardized inference method\"\"\"\n",
    "        return self.forward(sparse, mask)\n",
    "\n",
    "# ============================================================================\n",
    "# 5. MODEL VERSION 4: ROBUST SEMANTIC MODEL (AdaIN-based, matches sota_v3_best.pth)\n",
    "# ============================================================================\n",
    "class AdaIN(nn.Module):\n",
    "    \"\"\"Adaptive Instance Normalization - The StyleGAN way to inject attributes\"\"\"\n",
    "    def __init__(self, channels, embed_dim):\n",
    "        super().__init__()\n",
    "        self.channels = channels\n",
    "        # Learn to predict scale (gamma) and shift (beta) from attributes\n",
    "        self.linear = nn.Linear(embed_dim, channels * 2)\n",
    "\n",
    "    def forward(self, x, attr_embed):\n",
    "        # x: [B, C, H, W]\n",
    "        # attr_embed: [B, embed_dim]\n",
    "        \n",
    "        # Calculate style parameters\n",
    "        style = self.linear(attr_embed) # [B, 2*C]\n",
    "        style = style.view(-1, 2, self.channels, 1, 1)\n",
    "        gamma, beta = style[:, 0], style[:, 1]\n",
    "        \n",
    "        # Normalize x\n",
    "        mean = x.mean(dim=[2, 3], keepdim=True)\n",
    "        std = x.std(dim=[2, 3], keepdim=True) + 1e-8\n",
    "        norm_x = (x - mean) / std\n",
    "        \n",
    "        # Modulate\n",
    "        return gamma * norm_x + beta\n",
    "\n",
    "class AdaINResBlock(nn.Module):\n",
    "    def __init__(self, c, embed_dim):\n",
    "        super().__init__()\n",
    "        self.conv1 = nn.Conv2d(c, c, 3, 1, 1, bias=False)\n",
    "        self.conv2 = nn.Conv2d(c, c, 3, 1, 1, bias=False)\n",
    "        self.adain1 = AdaIN(c, embed_dim)\n",
    "        self.adain2 = AdaIN(c, embed_dim)\n",
    "        self.act = nn.ReLU()\n",
    "        \n",
    "    def forward(self, x, attr_embed):\n",
    "        residual = x\n",
    "        \n",
    "        out = self.conv1(x)\n",
    "        out = self.act(self.adain1(out, attr_embed)) \n",
    "        \n",
    "        out = self.conv2(out)\n",
    "        out = self.adain2(out, attr_embed) \n",
    "        \n",
    "        return self.act(out + residual)\n",
    "\n",
    "class RobustEncoder(nn.Module):\n",
    "    def __init__(self, input_channels=6, latent_channels=1024, attr_dim=40, embed_dim=256):\n",
    "        super().__init__()\n",
    "        # Standard convolutions\n",
    "        self.entry = nn.Sequential(nn.Conv2d(input_channels, 64, 4, 2, 1), nn.ReLU())\n",
    "        self.layer1 = nn.Sequential(nn.Conv2d(64, 128, 4, 2, 1), nn.BatchNorm2d(128), nn.ReLU())\n",
    "        self.layer2 = nn.Sequential(nn.Conv2d(128, 256, 4, 2, 1), nn.BatchNorm2d(256), nn.ReLU())\n",
    "        \n",
    "        # We use AdaIN blocks here to condition encoder on attributes\n",
    "        self.res1 = AdaINResBlock(256, embed_dim)\n",
    "        self.res2 = AdaINResBlock(256, embed_dim)\n",
    "        \n",
    "        self.exit = nn.Conv2d(256, latent_channels, 4, 2, 1) \n",
    "\n",
    "    def forward(self, x, mask, attr_embed):\n",
    "        # Concatenate mask (essential for robustness)\n",
    "        if mask is not None: x = torch.cat([x, mask], dim=1)\n",
    "        \n",
    "        h = self.entry(x)\n",
    "        h = self.layer1(h)\n",
    "        h = self.layer2(h)\n",
    "        \n",
    "        # Inject attributes via AdaIN (Gating) rather than Concatenation\n",
    "        h = self.res1(h, attr_embed)\n",
    "        h = self.res2(h, attr_embed)\n",
    "        \n",
    "        return self.exit(h)\n",
    "\n",
    "class RobustDecoder(nn.Module):\n",
    "    def __init__(self, latent_channels=1024, embed_dim=256):\n",
    "        super().__init__()\n",
    "        \n",
    "        self.layer1 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(latent_channels, 256, 4, 2, 1),\n",
    "            nn.BatchNorm2d(256), nn.ReLU()\n",
    "        )\n",
    "        \n",
    "        # Decoder also uses AdaIN to \"paint\" the attributes onto the features\n",
    "        self.res1 = AdaINResBlock(256, embed_dim)\n",
    "        self.res2 = AdaINResBlock(256, embed_dim)\n",
    "        \n",
    "        self.layer2 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(256, 128, 4, 2, 1),\n",
    "            nn.BatchNorm2d(128), nn.ReLU()\n",
    "        )\n",
    "        self.layer3 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(128, 64, 4, 2, 1),\n",
    "            nn.BatchNorm2d(64), nn.ReLU()\n",
    "        )\n",
    "        self.exit = nn.Sequential(\n",
    "            nn.ConvTranspose2d(64, 32, 4, 2, 1),\n",
    "            nn.BatchNorm2d(32), nn.ReLU(),\n",
    "            nn.Conv2d(32, 3, 3, 1, 1),\n",
    "            nn.Tanh()\n",
    "        )\n",
    "\n",
    "    def forward(self, z, attr_embed):\n",
    "        h = self.layer1(z)\n",
    "        h = self.res1(h, attr_embed)\n",
    "        h = self.res2(h, attr_embed)\n",
    "        h = self.layer2(h)\n",
    "        h = self.layer3(h)\n",
    "        return self.exit(h)\n",
    "\n",
    "class AttributeEmbedding(nn.Module):\n",
    "    def __init__(self, attr_dim=40, embed_dim=256):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.Linear(attr_dim, 128), nn.ReLU(),\n",
    "            nn.Linear(128, embed_dim), nn.ReLU(),\n",
    "            nn.Linear(embed_dim, embed_dim)\n",
    "        )\n",
    "    def forward(self, x): return self.net(x)\n",
    "\n",
    "class AttributePredictor(nn.Module):\n",
    "    def __init__(self, latent_channels, attr_dim):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.AdaptiveAvgPool2d(1), nn.Flatten(),\n",
    "            nn.Linear(latent_channels, 128), nn.ReLU(),\n",
    "            nn.Linear(128, attr_dim), nn.Sigmoid()\n",
    "        )\n",
    "    def forward(self, x): return self.net(x)\n",
    "\n",
    "class SOTARobustModel(nn.Module):\n",
    "    def __init__(self, latent_dim=1024, attr_dim=40):\n",
    "        super().__init__()\n",
    "        embed_dim = 256\n",
    "        self.attr_embedding = AttributeEmbedding(attr_dim, embed_dim)\n",
    "        self.encoder_x = RobustEncoder(6, latent_dim, attr_dim, embed_dim)\n",
    "        self.encoder_s = RobustEncoder(3, latent_dim, attr_dim, embed_dim)\n",
    "        self.decoder = RobustDecoder(latent_dim, embed_dim)\n",
    "        self.attr_predictor = AttributePredictor(latent_dim, attr_dim)\n",
    "        \n",
    "    def forward(self, x, s, training_dropout=False):\n",
    "        # 1. Get Attributes\n",
    "        attr = s['attributes']\n",
    "        attr_embed = self.attr_embedding(attr)\n",
    "        \n",
    "        # 2. ATTRIBUTE DROPOUT (The Robustness Fix)\n",
    "        # If we are training and hit the probability, we ZERO out the attribute embedding.\n",
    "        # For inference, we can optionally test with zeros to measure robustness\n",
    "        if training_dropout and np.random.random() < 0.5:  # Using 0.5 as default\n",
    "            used_embed = torch.zeros_like(attr_embed)\n",
    "        else:\n",
    "            used_embed = attr_embed\n",
    "            \n",
    "        # 3. Encode & Decode\n",
    "        z_x = self.encoder_x(x['sparse_image'], x['mask'], used_embed)\n",
    "        z_s = self.encoder_s(s['full_image'], None, used_embed)\n",
    "        \n",
    "        recon_x = self.decoder(z_x, used_embed)\n",
    "        recon_s = self.decoder(z_s, used_embed)\n",
    "        \n",
    "        # 4. Predict Attributes (Always use z to ensure latent holds info)\n",
    "        pred_attr_x = self.attr_predictor(z_x)\n",
    "        \n",
    "        return z_x, z_s, recon_x, recon_s, pred_attr_x\n",
    "    \n",
    "    def infer(self, sparse, mask, attributes=None):\n",
    "        \"\"\"Standardized inference method\"\"\"\n",
    "        self.eval()\n",
    "        with torch.no_grad():\n",
    "            # If attributes provided, use them; otherwise predict or use zeros\n",
    "            if attributes is None:\n",
    "                # Create dummy attributes (all zeros) to test robustness\n",
    "                attr_embed = torch.zeros(sparse.size(0), 256, device=sparse.device)\n",
    "            else:\n",
    "                attr_embed = self.attr_embedding(attributes)\n",
    "            \n",
    "            z_x = self.encoder_x(sparse, mask, attr_embed)\n",
    "            recon = self.decoder(z_x, attr_embed)\n",
    "            return recon\n",
    "\n",
    "# ============================================================================\n",
    "# 6. NOISE ROBUSTNESS LOGIC\n",
    "# ============================================================================\n",
    "\n",
    "def add_noise_to_sparse(sparse_image, mask, noise_type='gaussian', noise_level=0.1):\n",
    "    noisy_sparse = sparse_image.clone()\n",
    "    \n",
    "    if noise_type == 'gaussian':\n",
    "        noise = torch.randn_like(sparse_image) * noise_level\n",
    "        noisy_sparse = sparse_image + noise * mask\n",
    "        \n",
    "    elif noise_type == 'salt_pepper':\n",
    "        flip_mask = (torch.rand_like(sparse_image) < noise_level) * mask\n",
    "        num_flips = int(flip_mask.sum().item())\n",
    "        if num_flips > 0:\n",
    "            random_values = torch.randint(0, 2, (num_flips,), device=sparse_image.device).float() * 2 - 1\n",
    "            noisy_sparse[flip_mask > 0] = random_values\n",
    "        \n",
    "    elif noise_type == 'uniform':\n",
    "        noise = (torch.rand_like(sparse_image) * 2 - 1) * noise_level\n",
    "        noisy_sparse = sparse_image + noise * mask\n",
    "    \n",
    "    # Clamp to valid range\n",
    "    noisy_sparse = torch.clamp(noisy_sparse, -1, 1)\n",
    "    return noisy_sparse\n",
    "\n",
    "def evaluate_models(models_dict, test_loader, device, noise_config, max_samples=1000):\n",
    "    \"\"\"\n",
    "    Evaluates a dictionary of models: {'Name': model_instance}\n",
    "    noise_config: dict mapping noise_type -> list of noise levels\n",
    "    \"\"\"\n",
    "    results = {name: {} for name in models_dict.keys()}\n",
    "    \n",
    "    for name, model in models_dict.items():\n",
    "        print(f\"\\nEvaluating {name}...\")\n",
    "        model.eval()\n",
    "        results[name] = {nt: {} for nt in noise_config.keys()}\n",
    "        \n",
    "        for noise_type, noise_levels in noise_config.items():\n",
    "            for level in noise_levels:\n",
    "                mse_list = []\n",
    "                sample_count = 0\n",
    "                \n",
    "                with torch.no_grad():\n",
    "                    # UPDATED: Unpack attributes\n",
    "                    for sparse, mask, full, attributes in test_loader:\n",
    "                        if sample_count >= max_samples: break\n",
    "                        \n",
    "                        sparse = sparse.to(device)\n",
    "                        mask = mask.to(device)\n",
    "                        full = full.to(device)\n",
    "                        attributes = attributes.to(device)\n",
    "                        \n",
    "                        noisy_sparse = add_noise_to_sparse(sparse, mask, noise_type, level)\n",
    "                        \n",
    "                        # UPDATED: Unified Inference Call with attributes\n",
    "                        recon = model.infer(noisy_sparse, mask, attributes=attributes)\n",
    "                        \n",
    "                        # Metrics\n",
    "                        recon_01 = torch.clamp((recon + 1) / 2, 0, 1)\n",
    "                        full_01 = torch.clamp((full + 1) / 2, 0, 1)\n",
    "                        mse_list.extend(F.mse_loss(recon_01, full_01, reduction='none').mean(dim=[1,2,3]).tolist())\n",
    "                        sample_count += sparse.size(0)\n",
    "                \n",
    "                mean_mse = np.mean(mse_list)\n",
    "                results[name][noise_type][level] = mean_mse\n",
    "                print(f\"  [{noise_type} | σ={level:.3f}] MSE: {mean_mse:.6f}\")\n",
    "                \n",
    "    return results\n",
    "\n",
    "def visualize_comparison(results, save_path='noise_comparison.png'):\n",
    "    noise_types = list(results[list(results.keys())[0]].keys())\n",
    "    model_names = list(results.keys())\n",
    "    colors = ['#2E86AB', '#A23B72', '#F18F01', '#28a745', '#9a2eab', '#FF5733']\n",
    "    \n",
    "    fig, axes = plt.subplots(1, len(noise_types), figsize=(6*len(noise_types), 5))\n",
    "    if len(noise_types) == 1: axes = [axes]\n",
    "    \n",
    "    for idx, nt in enumerate(noise_types):\n",
    "        ax = axes[idx]\n",
    "        noise_levels = sorted(results[model_names[0]][nt].keys())\n",
    "        \n",
    "        for i, name in enumerate(model_names):\n",
    "            mses = [results[name][nt][nl] for nl in noise_levels]\n",
    "            ax.plot(noise_levels, mses, 'o-', linewidth=2, label=name, color=colors[i % len(colors)])\n",
    "            \n",
    "        ax.set_title(f\"{nt.upper()} Noise\")\n",
    "        ax.set_xlabel(\"Noise Level (σ)\")\n",
    "        ax.set_ylabel(\"MSE\")\n",
    "        ax.legend()\n",
    "        ax.grid(True, alpha=0.3)\n",
    "        \n",
    "    plt.tight_layout()\n",
    "    plt.savefig(save_path)\n",
    "    print(f\"Plot saved to {save_path}\")\n",
    "\n",
    "def visual_reconstruction(models_dict, dataset, device, noise_level=0.2):\n",
    "    \"\"\"Generates visual grid: Top=GT, Then rows for each model\"\"\"\n",
    "    idx = np.random.randint(len(dataset))\n",
    "    sparse, mask, full, attributes = dataset[idx] # Unpack 4\n",
    "    sparse = sparse.unsqueeze(0).to(device)\n",
    "    mask = mask.unsqueeze(0).to(device)\n",
    "    full = full.unsqueeze(0).to(device)\n",
    "    attributes = attributes.unsqueeze(0).to(device)\n",
    "    \n",
    "    noisy_sparse = add_noise_to_sparse(sparse, mask, 'gaussian', noise_level)\n",
    "    \n",
    "    imgs = []\n",
    "    titles = []\n",
    "    \n",
    "    # 1. Ground Truth\n",
    "    imgs.append(full)\n",
    "    titles.append(\"Ground Truth\")\n",
    "    \n",
    "    # 2. Noisy Input (Visualization)\n",
    "    imgs.append(noisy_sparse)\n",
    "    titles.append(f\"Input (σ={noise_level})\")\n",
    "    \n",
    "    # 3. Models\n",
    "    for name, model in models_dict.items():\n",
    "        model.eval()\n",
    "        with torch.no_grad():\n",
    "            recon = model.infer(noisy_sparse, mask, attributes=attributes)\n",
    "            imgs.append(recon)\n",
    "            titles.append(name)\n",
    "            \n",
    "    # Plotting\n",
    "    fig, axes = plt.subplots(1, len(imgs), figsize=(3*len(imgs), 3))\n",
    "    for i, ax in enumerate(axes):\n",
    "        img_np = (imgs[i].squeeze().cpu().permute(1,2,0).numpy() + 1) / 2\n",
    "        ax.imshow(np.clip(img_np, 0, 1))\n",
    "        ax.set_title(titles[i])\n",
    "        ax.axis('off')\n",
    "        \n",
    "    plt.tight_layout()\n",
    "    plt.savefig(\"visual_reconstruction.png\")\n",
    "    print(\"Visual reconstruction saved.\")\n",
    "\n",
    "# ============================================================================\n",
    "# 7. MAIN EXECUTION\n",
    "# ============================================================================\n",
    "if __name__ == \"__main__\":\n",
    "    device = CONFIG['device']\n",
    "    print(f\"Running on {device}\")\n",
    "    \n",
    "    # 1. Load Data\n",
    "    print(\"Loading Dataset...\")\n",
    "    test_ds = UnifiedCelebADataset('./data', split='test', sparsity=CONFIG['sparsity'])\n",
    "    test_loader = DataLoader(test_ds, batch_size=CONFIG['batch_size'], shuffle=False)\n",
    "    \n",
    "    # 2. Initialize Models\n",
    "    print(\"Initializing Models...\")\n",
    "    models = {\n",
    "        'V3-ResUNet': ResUNet().to(device),\n",
    "        'V4-Robust': SOTARobustModel(latent_dim=1024, attr_dim=40).to(device),\n",
    "    }\n",
    "    \n",
    "    # 3. Load Weights\n",
    "    try:\n",
    "        if os.path.exists('sota_resunet.pth'):\n",
    "            models['V3-ResUNet'].load_state_dict(torch.load('sota_resunet.pth'))\n",
    "            print(\"Loaded V3-ResUNet weights\")\n",
    "        if os.path.exists('sota_v3_best.pth'):\n",
    "            models['V4-Robust'].load_state_dict(torch.load('sota_v3_best.pth'))\n",
    "            print(\"Loaded sota_best_manifold weights\")\n",
    "        print(\"Weights check complete.\")\n",
    "    except Exception as e:\n",
    "        print(f\"Weight loading warning: {e}. Using random weights for demonstration.\")\n",
    "\n",
    "    # 4. Define Noise Configuration (Different ranges for different noise types)\n",
    "    noise_config = {\n",
    "        'gaussian': [0.1, 0.2, 0.3, 0.4, 0.5, 0.6, 0.7, 0.8, 0.9],\n",
    "        'salt_pepper': [0.01, 0.02, 0.03, 0.04, 0.05, 0.06, 0.07, 0.08, 0.09],\n",
    "        'uniform': [0.1, 0.2, 0.3, 0.4, 0.5, 0.6, 0.7, 0.8, 0.9]\n",
    "    }\n",
    "    \n",
    "    # 5. Run Evaluation\n",
    "    results = evaluate_models(models, test_loader, device, noise_config, max_samples=200)\n",
    "    \n",
    "    # 6. Visualize\n",
    "    visualize_comparison(results)\n",
    "    visual_reconstruction(models, test_ds, device)\n",
    "    \n",
    "    # 7. Save JSON\n",
    "    with open('robustness_results.json', 'w') as f:\n",
    "        # Convert numpy floats to native float for JSON serialization\n",
    "        clean_results = {}\n",
    "        for m in results:\n",
    "            clean_results[m] = {}\n",
    "            for nt in results[m]:\n",
    "                clean_results[m][nt] = {str(k): float(v) for k, v in results[m][nt].items()}\n",
    "        json.dump(clean_results, f, indent=2)\n",
    "    print(\"Done.\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c41255ef",
   "metadata": {},
   "source": [
    "## Four Metrics"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "5193468d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "================================================================================\n",
      "HOMEOMORPHISM VERIFICATION - CELEBA MANIFOLD MODEL\n",
      "Theorem 1: Topological Unification\n",
      "Proposition 1: Metric-Theoretical Bridge\n",
      "Four-Metric Verification: β₀, Trust, W₂, Alignment, Continuity\n",
      "================================================================================\n",
      "\n",
      "Loading trained CelebA model...\n",
      "Loaded: sota_v3_best.pth\n",
      "Model loaded on cuda\n",
      "\n",
      "\n",
      "================================================================================\n",
      "TESTING SPARSITY LEVEL: ρ = 0.05\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "HOMEOMORPHISM VERIFICATION - CELEBA MANIFOLD MODEL\n",
      "================================================================================\n",
      "Latent dimension: 1024 (spatial)\n",
      "Sparsity levels: [0.05]\n",
      "κ (neighbors): 5\n",
      "Distance metric: cosine\n",
      "Max samples per split: 5000\n",
      "================================================================================\n",
      "\n",
      "THEORETICAL JUSTIFICATION:\n",
      "  Theorem 1: Topological Unification requires:\n",
      "    (i) Local homeomorphism (each encoder)\n",
      "    (ii) Global connectivity (non-trivial overlap)\n",
      "\n",
      "  Proposition 1: Violations manifest as:\n",
      "    (i) Collapse (c₁→0) ⟹ Low Trust\n",
      "    (ii) Misalignment ⟹ High W₂\n",
      "    (iii) Disjointness (μ→0) ⟹ High Alignment Error\n",
      "    (iv) Distortion (c₂→∞) ⟹ Low Continuity\n",
      "================================================================================\n",
      "\n",
      "VERIFICATION METRICS:\n",
      "  0. β₀ = 1 - Topological connectivity check\n",
      "  1. Trust Score (τ_t ≥ 0.80) - Local preservation (c₁ > 0)\n",
      "  2. Sliced W₂ (τ_w ≤ 0.30) - Global distribution alignment (μ > 0)\n",
      "  3. Alignment Error (τ_a ≤ 0.15/0.10) - Coordinate matching\n",
      "  4. Continuity (τ_c ≥ 0.80) - Manifold compactness (SPARSE only)\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "EVALUATING AT ρ = 0.05 (5% visible pixels)\n",
      "================================================================================\n",
      "\n",
      "Encoding TRAINING samples...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                               \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding VALIDATION samples...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                             \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Encoded shapes:\n",
      "  Train Sparse: (5000, 16384)\n",
      "  Train Full:   (5000, 16384)\n",
      "  Val Sparse:   (5000, 16384)\n",
      "  Val Full:     (5000, 16384)\n",
      "\n",
      "[0/4] Computing β₀ (connectivity check)...\n",
      "  β₀ = 1\n",
      "  ✓ Single connected manifold (Theorem 1(ii) satisfied)\n",
      "\n",
      "[1/4] Computing TRUST SCORE (τ_t ≥ 0.80)...\n",
      "  Trust Score: 0.8770\n",
      "  ✓ Local preservation verified (c₁ > 0)\n",
      "\n",
      "[2/4] Computing SLICED WASSERSTEIN-2 (τ_w ≤ 0.30)...\n",
      "  Sliced W₂ (Sparse↔Full): 0.0058\n",
      "  ✓ Global alignment verified (distributions overlap)\n",
      "\n",
      "[3/4] Computing ALIGNMENT ERROR (τ_a ≤ 0.1)...\n",
      "  Distance metric: cosine\n",
      "  Alignment Error (avg):  0.0429\n",
      "  Global Alignment Error: 0.0433\n",
      "  ✓ Sparse and full map to THE SAME POINTS (perfect merger)\n",
      "  Per-bin alignment errors:\n",
      "    Bin 0: 0.0452 ✓\n",
      "    Bin 1: 0.0424 ✓\n",
      "    Bin 2: 0.0410 ✓\n",
      "\n",
      "[4/4] Computing ATTRIBUTE-BASED CONTINUITY (τ_c ≥ 0.80)...\n",
      "  Measuring on SPARSE encodings only\n",
      "  Continuity (Sparse): 0.8877\n",
      "  ✓ Manifold groups form coherent, compact clusters (c₂ < ∞)\n",
      "  Per-group continuity (first 5):\n",
      "    Group 0: 0.8959 ✓\n",
      "    Group 1: 0.8863 ✓\n",
      "    Group 2: 0.8910 ✓\n",
      "    Group 3: 0.8833 ✓\n",
      "    Group 4: 0.8838 ✓\n",
      "\n",
      "================================================================================\n",
      "SUMMARY FOR ρ = 0.05:\n",
      "  β₀:              1\n",
      "  Trust Score:     0.8770 ✓\n",
      "  Sliced W₂:       0.0058 ✓\n",
      "  Alignment Error: 0.0429 ✓\n",
      "  Continuity:      0.8877 ✓\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "TOPOLOGICAL UNIFICATION VERIFICATION (THEOREM 1)\n",
      "================================================================================\n",
      "ρ        β₀     Trust      W₂         Align      Cont      \n",
      "--------------------------------------------------------------------------------\n",
      "0.05     1      0.8770     0.0058     0.0429     0.8877    \n",
      "================================================================================\n",
      "\n",
      "TARGET THRESHOLDS:\n",
      "  β₀ = 1             (single connected manifold)\n",
      "  Trust ≥ 0.80       (local preservation, c₁ > 0)\n",
      "  W₂ ≤ 0.30          (global distribution alignment, μ > 0)\n",
      "  Align ≤ 0.1          (coordinate matching, topological merger)\n",
      "  Cont ≥ 0.80        (manifold compactness, c₂ < ∞, SPARSE only)\n",
      "--------------------------------------------------------------------------------\n",
      "\n",
      "ρ = 0.05: ✓ PASS\n",
      "  β₀:        ✓ (connectivity)\n",
      "  Trust:     ✓ (local preservation)\n",
      "  W₂:        ✓ (global alignment)\n",
      "  Align:     ✓ (coordinate matching)\n",
      "  Cont:      ✓ (manifold compactness)\n",
      "\n",
      "================================================================================\n",
      "VERIFICATION RESULT: ✓ THEOREM 1 SATISFIED (COMPLETE)\n",
      "  - β₀ = 1: Single connected manifold\n",
      "  - Trust ≥ 0.80: Local homeomorphism (Condition i, c₁ > 0)\n",
      "  - W₂ ≤ 0.30: Sparse↔Full distributions co-located (Condition ii, μ > 0)\n",
      "  - Align ≤ 0.1: Sparse↔Full map to SAME points (perfect merger)\n",
      "  - Cont ≥ 0.80: Manifold groups form compact clusters (c₂ < ∞)\n",
      "\n",
      "CONCLUSION: Sparse and Full encodings form ONE UNIVERSAL MANIFOLD\n",
      "            with TOPOLOGICALLY MERGED representations and\n",
      "            COMPACT, SEMANTICALLY COHERENT attribute clusters.\n",
      "================================================================================\n",
      "\n",
      "\n",
      "================================================================================\n",
      "TESTING SPARSITY LEVEL: ρ = 0.08\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "HOMEOMORPHISM VERIFICATION - CELEBA MANIFOLD MODEL\n",
      "================================================================================\n",
      "Latent dimension: 1024 (spatial)\n",
      "Sparsity levels: [0.08]\n",
      "κ (neighbors): 5\n",
      "Distance metric: cosine\n",
      "Max samples per split: 5000\n",
      "================================================================================\n",
      "\n",
      "THEORETICAL JUSTIFICATION:\n",
      "  Theorem 1: Topological Unification requires:\n",
      "    (i) Local homeomorphism (each encoder)\n",
      "    (ii) Global connectivity (non-trivial overlap)\n",
      "\n",
      "  Proposition 1: Violations manifest as:\n",
      "    (i) Collapse (c₁→0) ⟹ Low Trust\n",
      "    (ii) Misalignment ⟹ High W₂\n",
      "    (iii) Disjointness (μ→0) ⟹ High Alignment Error\n",
      "    (iv) Distortion (c₂→∞) ⟹ Low Continuity\n",
      "================================================================================\n",
      "\n",
      "VERIFICATION METRICS:\n",
      "  0. β₀ = 1 - Topological connectivity check\n",
      "  1. Trust Score (τ_t ≥ 0.80) - Local preservation (c₁ > 0)\n",
      "  2. Sliced W₂ (τ_w ≤ 0.30) - Global distribution alignment (μ > 0)\n",
      "  3. Alignment Error (τ_a ≤ 0.15/0.10) - Coordinate matching\n",
      "  4. Continuity (τ_c ≥ 0.80) - Manifold compactness (SPARSE only)\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "EVALUATING AT ρ = 0.08 (8% visible pixels)\n",
      "================================================================================\n",
      "\n",
      "Encoding TRAINING samples...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                               \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding VALIDATION samples...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                             \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Encoded shapes:\n",
      "  Train Sparse: (5000, 16384)\n",
      "  Train Full:   (5000, 16384)\n",
      "  Val Sparse:   (5000, 16384)\n",
      "  Val Full:     (5000, 16384)\n",
      "\n",
      "[0/4] Computing β₀ (connectivity check)...\n",
      "  β₀ = 1\n",
      "  ✓ Single connected manifold (Theorem 1(ii) satisfied)\n",
      "\n",
      "[1/4] Computing TRUST SCORE (τ_t ≥ 0.80)...\n",
      "  Trust Score: 0.8732\n",
      "  ✓ Local preservation verified (c₁ > 0)\n",
      "\n",
      "[2/4] Computing SLICED WASSERSTEIN-2 (τ_w ≤ 0.30)...\n",
      "  Sliced W₂ (Sparse↔Full): 0.0080\n",
      "  ✓ Global alignment verified (distributions overlap)\n",
      "\n",
      "[3/4] Computing ALIGNMENT ERROR (τ_a ≤ 0.1)...\n",
      "  Distance metric: cosine\n",
      "  Alignment Error (avg):  0.0414\n",
      "  Global Alignment Error: 0.0422\n",
      "  ✓ Sparse and full map to THE SAME POINTS (perfect merger)\n",
      "  Per-bin alignment errors:\n",
      "    Bin 0: 0.0443 ✓\n",
      "    Bin 1: 0.0412 ✓\n",
      "    Bin 2: 0.0387 ✓\n",
      "\n",
      "[4/4] Computing ATTRIBUTE-BASED CONTINUITY (τ_c ≥ 0.80)...\n",
      "  Measuring on SPARSE encodings only\n",
      "  Continuity (Sparse): 0.8893\n",
      "  ✓ Manifold groups form coherent, compact clusters (c₂ < ∞)\n",
      "  Per-group continuity (first 5):\n",
      "    Group 0: 0.8902 ✓\n",
      "    Group 1: 0.8908 ✓\n",
      "    Group 2: 0.9032 ✓\n",
      "    Group 3: 0.8838 ✓\n",
      "    Group 4: 0.8927 ✓\n",
      "\n",
      "================================================================================\n",
      "SUMMARY FOR ρ = 0.08:\n",
      "  β₀:              1\n",
      "  Trust Score:     0.8732 ✓\n",
      "  Sliced W₂:       0.0080 ✓\n",
      "  Alignment Error: 0.0414 ✓\n",
      "  Continuity:      0.8893 ✓\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "TOPOLOGICAL UNIFICATION VERIFICATION (THEOREM 1)\n",
      "================================================================================\n",
      "ρ        β₀     Trust      W₂         Align      Cont      \n",
      "--------------------------------------------------------------------------------\n",
      "0.08     1      0.8732     0.0080     0.0414     0.8893    \n",
      "================================================================================\n",
      "\n",
      "TARGET THRESHOLDS:\n",
      "  β₀ = 1             (single connected manifold)\n",
      "  Trust ≥ 0.80       (local preservation, c₁ > 0)\n",
      "  W₂ ≤ 0.30          (global distribution alignment, μ > 0)\n",
      "  Align ≤ 0.1          (coordinate matching, topological merger)\n",
      "  Cont ≥ 0.80        (manifold compactness, c₂ < ∞, SPARSE only)\n",
      "--------------------------------------------------------------------------------\n",
      "\n",
      "ρ = 0.08: ✓ PASS\n",
      "  β₀:        ✓ (connectivity)\n",
      "  Trust:     ✓ (local preservation)\n",
      "  W₂:        ✓ (global alignment)\n",
      "  Align:     ✓ (coordinate matching)\n",
      "  Cont:      ✓ (manifold compactness)\n",
      "\n",
      "================================================================================\n",
      "VERIFICATION RESULT: ✓ THEOREM 1 SATISFIED (COMPLETE)\n",
      "  - β₀ = 1: Single connected manifold\n",
      "  - Trust ≥ 0.80: Local homeomorphism (Condition i, c₁ > 0)\n",
      "  - W₂ ≤ 0.30: Sparse↔Full distributions co-located (Condition ii, μ > 0)\n",
      "  - Align ≤ 0.1: Sparse↔Full map to SAME points (perfect merger)\n",
      "  - Cont ≥ 0.80: Manifold groups form compact clusters (c₂ < ∞)\n",
      "\n",
      "CONCLUSION: Sparse and Full encodings form ONE UNIVERSAL MANIFOLD\n",
      "            with TOPOLOGICALLY MERGED representations and\n",
      "            COMPACT, SEMANTICALLY COHERENT attribute clusters.\n",
      "================================================================================\n",
      "\n",
      "\n",
      "================================================================================\n",
      "TESTING SPARSITY LEVEL: ρ = 0.10\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "HOMEOMORPHISM VERIFICATION - CELEBA MANIFOLD MODEL\n",
      "================================================================================\n",
      "Latent dimension: 1024 (spatial)\n",
      "Sparsity levels: [0.1]\n",
      "κ (neighbors): 5\n",
      "Distance metric: cosine\n",
      "Max samples per split: 5000\n",
      "================================================================================\n",
      "\n",
      "THEORETICAL JUSTIFICATION:\n",
      "  Theorem 1: Topological Unification requires:\n",
      "    (i) Local homeomorphism (each encoder)\n",
      "    (ii) Global connectivity (non-trivial overlap)\n",
      "\n",
      "  Proposition 1: Violations manifest as:\n",
      "    (i) Collapse (c₁→0) ⟹ Low Trust\n",
      "    (ii) Misalignment ⟹ High W₂\n",
      "    (iii) Disjointness (μ→0) ⟹ High Alignment Error\n",
      "    (iv) Distortion (c₂→∞) ⟹ Low Continuity\n",
      "================================================================================\n",
      "\n",
      "VERIFICATION METRICS:\n",
      "  0. β₀ = 1 - Topological connectivity check\n",
      "  1. Trust Score (τ_t ≥ 0.80) - Local preservation (c₁ > 0)\n",
      "  2. Sliced W₂ (τ_w ≤ 0.30) - Global distribution alignment (μ > 0)\n",
      "  3. Alignment Error (τ_a ≤ 0.15/0.10) - Coordinate matching\n",
      "  4. Continuity (τ_c ≥ 0.80) - Manifold compactness (SPARSE only)\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "EVALUATING AT ρ = 0.10 (10% visible pixels)\n",
      "================================================================================\n",
      "\n",
      "Encoding TRAINING samples...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                               \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding VALIDATION samples...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                             \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Encoded shapes:\n",
      "  Train Sparse: (5000, 16384)\n",
      "  Train Full:   (5000, 16384)\n",
      "  Val Sparse:   (5000, 16384)\n",
      "  Val Full:     (5000, 16384)\n",
      "\n",
      "[0/4] Computing β₀ (connectivity check)...\n",
      "  β₀ = 1\n",
      "  ✓ Single connected manifold (Theorem 1(ii) satisfied)\n",
      "\n",
      "[1/4] Computing TRUST SCORE (τ_t ≥ 0.80)...\n",
      "  Trust Score: 0.8711\n",
      "  ✓ Local preservation verified (c₁ > 0)\n",
      "\n",
      "[2/4] Computing SLICED WASSERSTEIN-2 (τ_w ≤ 0.30)...\n",
      "  Sliced W₂ (Sparse↔Full): 0.0109\n",
      "  ✓ Global alignment verified (distributions overlap)\n",
      "\n",
      "[3/4] Computing ALIGNMENT ERROR (τ_a ≤ 0.1)...\n",
      "  Distance metric: cosine\n",
      "  Alignment Error (avg):  0.0429\n",
      "  Global Alignment Error: 0.0442\n",
      "  ✓ Sparse and full map to THE SAME POINTS (perfect merger)\n",
      "  Per-bin alignment errors:\n",
      "    Bin 0: 0.0466 ✓\n",
      "    Bin 1: 0.0430 ✓\n",
      "    Bin 2: 0.0390 ✓\n",
      "\n",
      "[4/4] Computing ATTRIBUTE-BASED CONTINUITY (τ_c ≥ 0.80)...\n",
      "  Measuring on SPARSE encodings only\n",
      "  Continuity (Sparse): 0.8851\n",
      "  ✓ Manifold groups form coherent, compact clusters (c₂ < ∞)\n",
      "  Per-group continuity (first 5):\n",
      "    Group 0: 0.8952 ✓\n",
      "    Group 1: 0.8894 ✓\n",
      "    Group 2: 0.8838 ✓\n",
      "    Group 3: 0.8754 ✓\n",
      "    Group 4: 0.8849 ✓\n",
      "\n",
      "================================================================================\n",
      "SUMMARY FOR ρ = 0.10:\n",
      "  β₀:              1\n",
      "  Trust Score:     0.8711 ✓\n",
      "  Sliced W₂:       0.0109 ✓\n",
      "  Alignment Error: 0.0429 ✓\n",
      "  Continuity:      0.8851 ✓\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "TOPOLOGICAL UNIFICATION VERIFICATION (THEOREM 1)\n",
      "================================================================================\n",
      "ρ        β₀     Trust      W₂         Align      Cont      \n",
      "--------------------------------------------------------------------------------\n",
      "0.10     1      0.8711     0.0109     0.0429     0.8851    \n",
      "================================================================================\n",
      "\n",
      "TARGET THRESHOLDS:\n",
      "  β₀ = 1             (single connected manifold)\n",
      "  Trust ≥ 0.80       (local preservation, c₁ > 0)\n",
      "  W₂ ≤ 0.30          (global distribution alignment, μ > 0)\n",
      "  Align ≤ 0.1          (coordinate matching, topological merger)\n",
      "  Cont ≥ 0.80        (manifold compactness, c₂ < ∞, SPARSE only)\n",
      "--------------------------------------------------------------------------------\n",
      "\n",
      "ρ = 0.10: ✓ PASS\n",
      "  β₀:        ✓ (connectivity)\n",
      "  Trust:     ✓ (local preservation)\n",
      "  W₂:        ✓ (global alignment)\n",
      "  Align:     ✓ (coordinate matching)\n",
      "  Cont:      ✓ (manifold compactness)\n",
      "\n",
      "================================================================================\n",
      "VERIFICATION RESULT: ✓ THEOREM 1 SATISFIED (COMPLETE)\n",
      "  - β₀ = 1: Single connected manifold\n",
      "  - Trust ≥ 0.80: Local homeomorphism (Condition i, c₁ > 0)\n",
      "  - W₂ ≤ 0.30: Sparse↔Full distributions co-located (Condition ii, μ > 0)\n",
      "  - Align ≤ 0.1: Sparse↔Full map to SAME points (perfect merger)\n",
      "  - Cont ≥ 0.80: Manifold groups form compact clusters (c₂ < ∞)\n",
      "\n",
      "CONCLUSION: Sparse and Full encodings form ONE UNIVERSAL MANIFOLD\n",
      "            with TOPOLOGICALLY MERGED representations and\n",
      "            COMPACT, SEMANTICALLY COHERENT attribute clusters.\n",
      "================================================================================\n",
      "\n",
      "\n",
      "================================================================================\n",
      "TESTING SPARSITY LEVEL: ρ = 0.12\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "HOMEOMORPHISM VERIFICATION - CELEBA MANIFOLD MODEL\n",
      "================================================================================\n",
      "Latent dimension: 1024 (spatial)\n",
      "Sparsity levels: [0.12]\n",
      "κ (neighbors): 5\n",
      "Distance metric: cosine\n",
      "Max samples per split: 5000\n",
      "================================================================================\n",
      "\n",
      "THEORETICAL JUSTIFICATION:\n",
      "  Theorem 1: Topological Unification requires:\n",
      "    (i) Local homeomorphism (each encoder)\n",
      "    (ii) Global connectivity (non-trivial overlap)\n",
      "\n",
      "  Proposition 1: Violations manifest as:\n",
      "    (i) Collapse (c₁→0) ⟹ Low Trust\n",
      "    (ii) Misalignment ⟹ High W₂\n",
      "    (iii) Disjointness (μ→0) ⟹ High Alignment Error\n",
      "    (iv) Distortion (c₂→∞) ⟹ Low Continuity\n",
      "================================================================================\n",
      "\n",
      "VERIFICATION METRICS:\n",
      "  0. β₀ = 1 - Topological connectivity check\n",
      "  1. Trust Score (τ_t ≥ 0.80) - Local preservation (c₁ > 0)\n",
      "  2. Sliced W₂ (τ_w ≤ 0.30) - Global distribution alignment (μ > 0)\n",
      "  3. Alignment Error (τ_a ≤ 0.15/0.10) - Coordinate matching\n",
      "  4. Continuity (τ_c ≥ 0.80) - Manifold compactness (SPARSE only)\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "EVALUATING AT ρ = 0.12 (12% visible pixels)\n",
      "================================================================================\n",
      "\n",
      "Encoding TRAINING samples...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                               \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding VALIDATION samples...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                             \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Encoded shapes:\n",
      "  Train Sparse: (5000, 16384)\n",
      "  Train Full:   (5000, 16384)\n",
      "  Val Sparse:   (5000, 16384)\n",
      "  Val Full:     (5000, 16384)\n",
      "\n",
      "[0/4] Computing β₀ (connectivity check)...\n",
      "  β₀ = 1\n",
      "  ✓ Single connected manifold (Theorem 1(ii) satisfied)\n",
      "\n",
      "[1/4] Computing TRUST SCORE (τ_t ≥ 0.80)...\n",
      "  Trust Score: 0.8685\n",
      "  ✓ Local preservation verified (c₁ > 0)\n",
      "\n",
      "[2/4] Computing SLICED WASSERSTEIN-2 (τ_w ≤ 0.30)...\n",
      "  Sliced W₂ (Sparse↔Full): 0.0116\n",
      "  ✓ Global alignment verified (distributions overlap)\n",
      "\n",
      "[3/4] Computing ALIGNMENT ERROR (τ_a ≤ 0.1)...\n",
      "  Distance metric: cosine\n",
      "  Alignment Error (avg):  0.0462\n",
      "  Global Alignment Error: 0.0476\n",
      "  ✓ Sparse and full map to THE SAME POINTS (perfect merger)\n",
      "  Per-bin alignment errors:\n",
      "    Bin 0: 0.0504 ✓\n",
      "    Bin 1: 0.0462 ✓\n",
      "    Bin 2: 0.0419 ✓\n",
      "\n",
      "[4/4] Computing ATTRIBUTE-BASED CONTINUITY (τ_c ≥ 0.80)...\n",
      "  Measuring on SPARSE encodings only\n",
      "  Continuity (Sparse): 0.8855\n",
      "  ✓ Manifold groups form coherent, compact clusters (c₂ < ∞)\n",
      "  Per-group continuity (first 5):\n",
      "    Group 0: 0.8869 ✓\n",
      "    Group 1: 0.8873 ✓\n",
      "    Group 2: 0.8736 ✓\n",
      "    Group 3: 0.8884 ✓\n",
      "    Group 4: 0.8822 ✓\n",
      "\n",
      "================================================================================\n",
      "SUMMARY FOR ρ = 0.12:\n",
      "  β₀:              1\n",
      "  Trust Score:     0.8685 ✓\n",
      "  Sliced W₂:       0.0116 ✓\n",
      "  Alignment Error: 0.0462 ✓\n",
      "  Continuity:      0.8855 ✓\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "TOPOLOGICAL UNIFICATION VERIFICATION (THEOREM 1)\n",
      "================================================================================\n",
      "ρ        β₀     Trust      W₂         Align      Cont      \n",
      "--------------------------------------------------------------------------------\n",
      "0.12     1      0.8685     0.0116     0.0462     0.8855    \n",
      "================================================================================\n",
      "\n",
      "TARGET THRESHOLDS:\n",
      "  β₀ = 1             (single connected manifold)\n",
      "  Trust ≥ 0.80       (local preservation, c₁ > 0)\n",
      "  W₂ ≤ 0.30          (global distribution alignment, μ > 0)\n",
      "  Align ≤ 0.1          (coordinate matching, topological merger)\n",
      "  Cont ≥ 0.80        (manifold compactness, c₂ < ∞, SPARSE only)\n",
      "--------------------------------------------------------------------------------\n",
      "\n",
      "ρ = 0.12: ✓ PASS\n",
      "  β₀:        ✓ (connectivity)\n",
      "  Trust:     ✓ (local preservation)\n",
      "  W₂:        ✓ (global alignment)\n",
      "  Align:     ✓ (coordinate matching)\n",
      "  Cont:      ✓ (manifold compactness)\n",
      "\n",
      "================================================================================\n",
      "VERIFICATION RESULT: ✓ THEOREM 1 SATISFIED (COMPLETE)\n",
      "  - β₀ = 1: Single connected manifold\n",
      "  - Trust ≥ 0.80: Local homeomorphism (Condition i, c₁ > 0)\n",
      "  - W₂ ≤ 0.30: Sparse↔Full distributions co-located (Condition ii, μ > 0)\n",
      "  - Align ≤ 0.1: Sparse↔Full map to SAME points (perfect merger)\n",
      "  - Cont ≥ 0.80: Manifold groups form compact clusters (c₂ < ∞)\n",
      "\n",
      "CONCLUSION: Sparse and Full encodings form ONE UNIVERSAL MANIFOLD\n",
      "            with TOPOLOGICALLY MERGED representations and\n",
      "            COMPACT, SEMANTICALLY COHERENT attribute clusters.\n",
      "================================================================================\n",
      "\n",
      "\n",
      "================================================================================\n",
      "TESTING SPARSITY LEVEL: ρ = 0.15\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "HOMEOMORPHISM VERIFICATION - CELEBA MANIFOLD MODEL\n",
      "================================================================================\n",
      "Latent dimension: 1024 (spatial)\n",
      "Sparsity levels: [0.15]\n",
      "κ (neighbors): 5\n",
      "Distance metric: cosine\n",
      "Max samples per split: 5000\n",
      "================================================================================\n",
      "\n",
      "THEORETICAL JUSTIFICATION:\n",
      "  Theorem 1: Topological Unification requires:\n",
      "    (i) Local homeomorphism (each encoder)\n",
      "    (ii) Global connectivity (non-trivial overlap)\n",
      "\n",
      "  Proposition 1: Violations manifest as:\n",
      "    (i) Collapse (c₁→0) ⟹ Low Trust\n",
      "    (ii) Misalignment ⟹ High W₂\n",
      "    (iii) Disjointness (μ→0) ⟹ High Alignment Error\n",
      "    (iv) Distortion (c₂→∞) ⟹ Low Continuity\n",
      "================================================================================\n",
      "\n",
      "VERIFICATION METRICS:\n",
      "  0. β₀ = 1 - Topological connectivity check\n",
      "  1. Trust Score (τ_t ≥ 0.80) - Local preservation (c₁ > 0)\n",
      "  2. Sliced W₂ (τ_w ≤ 0.30) - Global distribution alignment (μ > 0)\n",
      "  3. Alignment Error (τ_a ≤ 0.15/0.10) - Coordinate matching\n",
      "  4. Continuity (τ_c ≥ 0.80) - Manifold compactness (SPARSE only)\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "EVALUATING AT ρ = 0.15 (15% visible pixels)\n",
      "================================================================================\n",
      "\n",
      "Encoding TRAINING samples...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                               \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding VALIDATION samples...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                             \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Encoded shapes:\n",
      "  Train Sparse: (5000, 16384)\n",
      "  Train Full:   (5000, 16384)\n",
      "  Val Sparse:   (5000, 16384)\n",
      "  Val Full:     (5000, 16384)\n",
      "\n",
      "[0/4] Computing β₀ (connectivity check)...\n",
      "  β₀ = 1\n",
      "  ✓ Single connected manifold (Theorem 1(ii) satisfied)\n",
      "\n",
      "[1/4] Computing TRUST SCORE (τ_t ≥ 0.80)...\n",
      "  Trust Score: 0.8657\n",
      "  ✓ Local preservation verified (c₁ > 0)\n",
      "\n",
      "[2/4] Computing SLICED WASSERSTEIN-2 (τ_w ≤ 0.30)...\n",
      "  Sliced W₂ (Sparse↔Full): 0.0135\n",
      "  ✓ Global alignment verified (distributions overlap)\n",
      "\n",
      "[3/4] Computing ALIGNMENT ERROR (τ_a ≤ 0.1)...\n",
      "  Distance metric: cosine\n",
      "  Alignment Error (avg):  0.0546\n",
      "  Global Alignment Error: 0.0552\n",
      "  ✓ Sparse and full map to THE SAME POINTS (perfect merger)\n",
      "  Per-bin alignment errors:\n",
      "    Bin 0: 0.0586 ✓\n",
      "    Bin 1: 0.0535 ✓\n",
      "    Bin 2: 0.0516 ✓\n",
      "\n",
      "[4/4] Computing ATTRIBUTE-BASED CONTINUITY (τ_c ≥ 0.80)...\n",
      "  Measuring on SPARSE encodings only\n",
      "  Continuity (Sparse): 0.8793\n",
      "  ✓ Manifold groups form coherent, compact clusters (c₂ < ∞)\n",
      "  Per-group continuity (first 5):\n",
      "    Group 0: 0.8763 ✓\n",
      "    Group 1: 0.8975 ✓\n",
      "    Group 2: 0.8629 ✓\n",
      "    Group 3: 0.8802 ✓\n",
      "    Group 4: 0.8857 ✓\n",
      "\n",
      "================================================================================\n",
      "SUMMARY FOR ρ = 0.15:\n",
      "  β₀:              1\n",
      "  Trust Score:     0.8657 ✓\n",
      "  Sliced W₂:       0.0135 ✓\n",
      "  Alignment Error: 0.0546 ✓\n",
      "  Continuity:      0.8793 ✓\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "TOPOLOGICAL UNIFICATION VERIFICATION (THEOREM 1)\n",
      "================================================================================\n",
      "ρ        β₀     Trust      W₂         Align      Cont      \n",
      "--------------------------------------------------------------------------------\n",
      "0.15     1      0.8657     0.0135     0.0546     0.8793    \n",
      "================================================================================\n",
      "\n",
      "TARGET THRESHOLDS:\n",
      "  β₀ = 1             (single connected manifold)\n",
      "  Trust ≥ 0.80       (local preservation, c₁ > 0)\n",
      "  W₂ ≤ 0.30          (global distribution alignment, μ > 0)\n",
      "  Align ≤ 0.1          (coordinate matching, topological merger)\n",
      "  Cont ≥ 0.80        (manifold compactness, c₂ < ∞, SPARSE only)\n",
      "--------------------------------------------------------------------------------\n",
      "\n",
      "ρ = 0.15: ✓ PASS\n",
      "  β₀:        ✓ (connectivity)\n",
      "  Trust:     ✓ (local preservation)\n",
      "  W₂:        ✓ (global alignment)\n",
      "  Align:     ✓ (coordinate matching)\n",
      "  Cont:      ✓ (manifold compactness)\n",
      "\n",
      "================================================================================\n",
      "VERIFICATION RESULT: ✓ THEOREM 1 SATISFIED (COMPLETE)\n",
      "  - β₀ = 1: Single connected manifold\n",
      "  - Trust ≥ 0.80: Local homeomorphism (Condition i, c₁ > 0)\n",
      "  - W₂ ≤ 0.30: Sparse↔Full distributions co-located (Condition ii, μ > 0)\n",
      "  - Align ≤ 0.1: Sparse↔Full map to SAME points (perfect merger)\n",
      "  - Cont ≥ 0.80: Manifold groups form compact clusters (c₂ < ∞)\n",
      "\n",
      "CONCLUSION: Sparse and Full encodings form ONE UNIVERSAL MANIFOLD\n",
      "            with TOPOLOGICALLY MERGED representations and\n",
      "            COMPACT, SEMANTICALLY COHERENT attribute clusters.\n",
      "================================================================================\n",
      "\n",
      "\n",
      "================================================================================\n",
      "LaTeX Table (CelebA Verification):\n",
      "================================================================================\n",
      "\\begin{table}[!htpb]\n",
      "\\centering\n",
      "\\caption{Topological Unification Verification - CelebA ($\\kappa=5$, Cosine distance)}\n",
      "\\label{tab:celeba_unification}\n",
      "\\begin{tabular}{lccccc}\n",
      "\\hline\n",
      "\\textbf{Metric} & $\\rho=0.05$ & $\\rho=0.08$ & $\\rho=0.10$ & $\\rho=0.12$ & $\\rho=0.15$ \\\\\n",
      "\\hline\n",
      "$\\beta_0$ (connectivity) & 1 & 1 & 1 & 1 & 1 \\\\\n",
      "Trust $\\tau_t$ ($\\geq 0.80$) & 0.877 & 0.873 & 0.871 & 0.868 & 0.866 \\\\\n",
      "Sliced $W_2$ $\\tau_w$ ($\\leq 0.30$) & 0.006 & 0.008 & 0.011 & 0.012 & 0.013 \\\\\n",
      "Alignment $\\tau_a$ ($\\leq 0.10$) & 0.043 & 0.041 & 0.043 & 0.046 & 0.055 \\\\\n",
      "Continuity $\\tau_c$ ($\\geq 0.80$) & 0.888 & 0.889 & 0.885 & 0.886 & 0.879 \\\\\n",
      "\\hline\n",
      "\\end{tabular}\n",
      "\\end{table}\n",
      "================================================================================\n",
      "\n",
      "✓ Results saved to 'celeba_verification_results_cosine.json'\n",
      "\n",
      "================================================================================\n",
      "VERIFICATION COMPLETE!\n",
      "================================================================================\n"
     ]
    }
   ],
   "source": [
    "# ============================================================================\n",
    "# HOMEOMORPHISM VERIFICATION FOR CELEBA MANIFOLD MODEL\n",
    "# Four Metrics: β₀, Trust, Sliced Wasserstein, Alignment, In-Class Continuity\n",
    "# ============================================================================\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torch.utils.data import DataLoader\n",
    "from torchvision import datasets, transforms\n",
    "from sklearn.neighbors import NearestNeighbors\n",
    "from sklearn.metrics.pairwise import cosine_distances\n",
    "import numpy as np\n",
    "from tqdm import tqdm\n",
    "import json\n",
    "from scipy.stats import wasserstein_distance\n",
    "from scipy.sparse import csr_matrix\n",
    "from scipy.sparse.csgraph import connected_components\n",
    "import sys\n",
    "import os\n",
    "\n",
    "# Import from your training code\n",
    "# Note: These should be imported from your training file\n",
    "# For now, we'll define CONFIG here as a workaround\n",
    "CONFIG = {\n",
    "    'sparsity': 0.05,\n",
    "    'img_size': 64,       \n",
    "    'batch_size': 64,\n",
    "    'epochs': 60,\n",
    "    'lr': 2e-4,           \n",
    "    'latent_dim': 1024,\n",
    "    'attr_dim': 40,\n",
    "    'patience': 8,\n",
    "    'consistency_w': 1.0,  \n",
    "    'attr_weight': 0.1, \n",
    "    'attr_dropout': 0.5,\n",
    "    'device': 'cuda' if torch.cuda.is_available() else 'cpu'\n",
    "}\n",
    "\n",
    "# ============================================================================\n",
    "# MODEL COMPONENTS (Imported from training code)\n",
    "# ============================================================================\n",
    "import torch.nn as nn\n",
    "from torchvision import datasets, transforms\n",
    "\n",
    "class EnhancedCelebADataset(torch.utils.data.Dataset):\n",
    "    def __init__(self, root, split='train', sparsity=0.05, img_size=64):\n",
    "        self.transform = transforms.Compose([\n",
    "            transforms.CenterCrop(178),\n",
    "            transforms.Resize(img_size),\n",
    "            transforms.ToTensor(),\n",
    "            transforms.Normalize((0.5, 0.5, 0.5), (0.5, 0.5, 0.5))\n",
    "        ])\n",
    "        try:\n",
    "            self.data = datasets.CelebA(root=root, split=split, download=False, transform=self.transform)\n",
    "        except:\n",
    "            self.data = datasets.CelebA(root=root, split=split, download=True, transform=self.transform)\n",
    "        self.sparsity = sparsity\n",
    "\n",
    "    def __len__(self): return len(self.data)\n",
    "\n",
    "    def __getitem__(self, idx):\n",
    "        full_image, attributes = self.data[idx]\n",
    "        mask = (torch.rand_like(full_image) < self.sparsity).float()\n",
    "        sparse_image = full_image * mask\n",
    "        attributes = attributes.float()\n",
    "        x = {'sparse_image': sparse_image, 'mask': mask}\n",
    "        s = {'full_image': full_image, 'attributes': attributes} \n",
    "        return x, s\n",
    "\n",
    "class AdaIN(nn.Module):\n",
    "    def __init__(self, channels, embed_dim):\n",
    "        super().__init__()\n",
    "        self.channels = channels\n",
    "        self.linear = nn.Linear(embed_dim, channels * 2)\n",
    "\n",
    "    def forward(self, x, attr_embed):\n",
    "        style = self.linear(attr_embed)\n",
    "        style = style.view(-1, 2, self.channels, 1, 1)\n",
    "        gamma, beta = style[:, 0], style[:, 1]\n",
    "        mean = x.mean(dim=[2, 3], keepdim=True)\n",
    "        std = x.std(dim=[2, 3], keepdim=True) + 1e-8\n",
    "        norm_x = (x - mean) / std\n",
    "        return gamma * norm_x + beta\n",
    "\n",
    "class AdaINResBlock(nn.Module):\n",
    "    def __init__(self, c, embed_dim):\n",
    "        super().__init__()\n",
    "        self.conv1 = nn.Conv2d(c, c, 3, 1, 1, bias=False)\n",
    "        self.conv2 = nn.Conv2d(c, c, 3, 1, 1, bias=False)\n",
    "        self.adain1 = AdaIN(c, embed_dim)\n",
    "        self.adain2 = AdaIN(c, embed_dim)\n",
    "        self.act = nn.ReLU()\n",
    "        \n",
    "    def forward(self, x, attr_embed):\n",
    "        residual = x\n",
    "        out = self.conv1(x)\n",
    "        out = self.act(self.adain1(out, attr_embed)) \n",
    "        out = self.conv2(out)\n",
    "        out = self.adain2(out, attr_embed) \n",
    "        return self.act(out + residual)\n",
    "\n",
    "class RobustEncoder(nn.Module):\n",
    "    def __init__(self, input_channels=6, latent_channels=512, attr_dim=40, embed_dim=256):\n",
    "        super().__init__()\n",
    "        self.entry = nn.Sequential(nn.Conv2d(input_channels, 64, 4, 2, 1), nn.ReLU())\n",
    "        self.layer1 = nn.Sequential(nn.Conv2d(64, 128, 4, 2, 1), nn.BatchNorm2d(128), nn.ReLU())\n",
    "        self.layer2 = nn.Sequential(nn.Conv2d(128, 256, 4, 2, 1), nn.BatchNorm2d(256), nn.ReLU())\n",
    "        self.res1 = AdaINResBlock(256, embed_dim)\n",
    "        self.res2 = AdaINResBlock(256, embed_dim)\n",
    "        self.exit = nn.Conv2d(256, latent_channels, 4, 2, 1) \n",
    "\n",
    "    def forward(self, x, mask, attr_embed):\n",
    "        if mask is not None: x = torch.cat([x, mask], dim=1)\n",
    "        h = self.entry(x)\n",
    "        h = self.layer1(h)\n",
    "        h = self.layer2(h)\n",
    "        h = self.res1(h, attr_embed)\n",
    "        h = self.res2(h, attr_embed)\n",
    "        return self.exit(h)\n",
    "\n",
    "class RobustDecoder(nn.Module):\n",
    "    def __init__(self, latent_channels=512, embed_dim=256):\n",
    "        super().__init__()\n",
    "        self.layer1 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(latent_channels, 256, 4, 2, 1),\n",
    "            nn.BatchNorm2d(256), nn.ReLU()\n",
    "        )\n",
    "        self.res1 = AdaINResBlock(256, embed_dim)\n",
    "        self.res2 = AdaINResBlock(256, embed_dim)\n",
    "        self.layer2 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(256, 128, 4, 2, 1),\n",
    "            nn.BatchNorm2d(128), nn.ReLU()\n",
    "        )\n",
    "        self.layer3 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(128, 64, 4, 2, 1),\n",
    "            nn.BatchNorm2d(64), nn.ReLU()\n",
    "        )\n",
    "        self.exit = nn.Sequential(\n",
    "            nn.ConvTranspose2d(64, 32, 4, 2, 1),\n",
    "            nn.BatchNorm2d(32), nn.ReLU(),\n",
    "            nn.Conv2d(32, 3, 3, 1, 1),\n",
    "            nn.Tanh()\n",
    "        )\n",
    "\n",
    "    def forward(self, z, attr_embed):\n",
    "        h = self.layer1(z)\n",
    "        h = self.res1(h, attr_embed)\n",
    "        h = self.res2(h, attr_embed)\n",
    "        h = self.layer2(h)\n",
    "        h = self.layer3(h)\n",
    "        return self.exit(h)\n",
    "\n",
    "class AttributeEmbedding(nn.Module):\n",
    "    def __init__(self, attr_dim=40, embed_dim=256):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.Linear(attr_dim, 128), nn.ReLU(),\n",
    "            nn.Linear(128, embed_dim), nn.ReLU(),\n",
    "            nn.Linear(embed_dim, embed_dim)\n",
    "        )\n",
    "    def forward(self, x): return self.net(x)\n",
    "\n",
    "class AttributePredictor(nn.Module):\n",
    "    def __init__(self, latent_channels, attr_dim):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.AdaptiveAvgPool2d(1), nn.Flatten(),\n",
    "            nn.Linear(latent_channels, 128), nn.ReLU(),\n",
    "            nn.Linear(128, attr_dim), nn.Sigmoid()\n",
    "        )\n",
    "    def forward(self, x): return self.net(x)\n",
    "\n",
    "class SOTARobustModel(nn.Module):\n",
    "    def __init__(self):\n",
    "        super().__init__()\n",
    "        embed_dim = 256\n",
    "        self.attr_embedding = AttributeEmbedding(CONFIG['attr_dim'], embed_dim)\n",
    "        self.encoder_x = RobustEncoder(6, CONFIG['latent_dim'], CONFIG['attr_dim'], embed_dim)\n",
    "        self.encoder_s = RobustEncoder(3, CONFIG['latent_dim'], CONFIG['attr_dim'], embed_dim)\n",
    "        self.decoder = RobustDecoder(CONFIG['latent_dim'], embed_dim)\n",
    "        self.attr_predictor = AttributePredictor(CONFIG['latent_dim'], CONFIG['attr_dim'])\n",
    "        \n",
    "    def forward(self, x, s, training_dropout=False):\n",
    "        attr = s['attributes']\n",
    "        attr_embed = self.attr_embedding(attr)\n",
    "        \n",
    "        if training_dropout and np.random.random() < CONFIG['attr_dropout']:\n",
    "            used_embed = torch.zeros_like(attr_embed)\n",
    "        else:\n",
    "            used_embed = attr_embed\n",
    "            \n",
    "        z_x = self.encoder_x(x['sparse_image'], x['mask'], used_embed)\n",
    "        z_s = self.encoder_s(s['full_image'], None, used_embed)\n",
    "        \n",
    "        recon_x = self.decoder(z_x, used_embed)\n",
    "        recon_s = self.decoder(z_s, used_embed)\n",
    "        \n",
    "        pred_attr_x = self.attr_predictor(z_x)\n",
    "        \n",
    "        return z_x, z_s, recon_x, recon_s, pred_attr_x\n",
    "\n",
    "# ============================================================================\n",
    "# VERIFICATION METRICS\n",
    "# ============================================================================\n",
    "\n",
    "\n",
    "def compute_betti_0(latent_codes, kappa=10):\n",
    "    \"\"\"\n",
    "    Compute β₀ (0-th Betti number) = number of connected components\n",
    "    \n",
    "    Theoretical Link: Detects violations of Topological Disjointness (μ → 0)\n",
    "    \n",
    "    β₀ = 1: Single connected manifold (unification verified)\n",
    "    β₀ > 1: Disjoint components (unification failed)\n",
    "    \n",
    "    Args:\n",
    "        latent_codes: (N, D) array of latent codes\n",
    "        kappa: number of nearest neighbors for graph construction\n",
    "    \n",
    "    Returns:\n",
    "        beta_0: number of connected components\n",
    "        labels: component assignment for each point\n",
    "    \"\"\"\n",
    "    N = len(latent_codes)\n",
    "    \n",
    "    # Build k-NN graph\n",
    "    nn = NearestNeighbors(n_neighbors=kappa+1, metric='euclidean', \n",
    "                         algorithm='auto', n_jobs=-1)\n",
    "    nn.fit(latent_codes)\n",
    "    distances, neighbors = nn.kneighbors(latent_codes)\n",
    "    \n",
    "    # Create adjacency matrix (undirected graph)\n",
    "    row_indices = []\n",
    "    col_indices = []\n",
    "    \n",
    "    for i in range(N):\n",
    "        for j in neighbors[i, 1:]:  # Skip self\n",
    "            row_indices.append(i)\n",
    "            col_indices.append(j)\n",
    "            row_indices.append(j)\n",
    "            col_indices.append(i)\n",
    "    \n",
    "    # Create sparse adjacency matrix\n",
    "    data = np.ones(len(row_indices))\n",
    "    adjacency = csr_matrix((data, (row_indices, col_indices)), shape=(N, N))\n",
    "    \n",
    "    # Count connected components\n",
    "    n_components, labels = connected_components(\n",
    "        csgraph=adjacency,\n",
    "        directed=False,\n",
    "        return_labels=True\n",
    "    )\n",
    "    \n",
    "    return n_components, labels\n",
    "\n",
    "\n",
    "def compute_trust_score(latent_codes, attributes, kappa=5, attribute_threshold=0.5):\n",
    "    \"\"\"\n",
    "    METRIC 1: Trust Score (τ_t ≥ 0.80)\n",
    "    \n",
    "    For CelebA: Measures if k-nearest neighbors share similar attributes\n",
    "    Uses average attribute similarity instead of exact label matching\n",
    "    \n",
    "    Theoretical Link: Empirical proxy for local preservation and injectivity \n",
    "    (c₁ > 0) from Theorem 2 and Proposition 1(i).\n",
    "    \n",
    "    Args:\n",
    "        latent_codes: (N, D) array of latent representations\n",
    "        attributes: (N, A) array of binary attributes (40 attributes for CelebA)\n",
    "        kappa: number of nearest neighbors\n",
    "        attribute_threshold: minimum similarity to count as \"same\"\n",
    "    \n",
    "    Returns:\n",
    "        trust_score: average attribute similarity with neighbors\n",
    "    \"\"\"\n",
    "    N = len(latent_codes)\n",
    "    \n",
    "    # Build k-NN graph\n",
    "    nn = NearestNeighbors(n_neighbors=kappa+1, metric='euclidean', \n",
    "                         algorithm='auto', n_jobs=-1)\n",
    "    nn.fit(latent_codes)\n",
    "    _, neighbors = nn.kneighbors(latent_codes)\n",
    "    neighbors = neighbors[:, 1:]  # Exclude self\n",
    "    \n",
    "    # Compute trust: average attribute similarity with neighbors\n",
    "    trust_scores = []\n",
    "    for i in range(N):\n",
    "        neighbor_attrs = attributes[neighbors[i]]\n",
    "        # Compute Jaccard similarity or Hamming distance\n",
    "        similarities = np.mean(neighbor_attrs == attributes[i], axis=1)\n",
    "        trust_scores.append(np.mean(similarities))\n",
    "    \n",
    "    return np.mean(trust_scores)\n",
    "\n",
    "\n",
    "def compute_sliced_wasserstein(X, Y, n_projections=100):\n",
    "    \"\"\"\n",
    "    METRIC 2: Sliced Wasserstein-2 Discrepancy (τ_w ≤ 0.30)\n",
    "    \n",
    "    Theoretical Link: Proxy for global geometric alignment from Proposition 1(ii).\n",
    "    \n",
    "    Measures: Transport cost between distributions P_z^[sparse] and P_z^[full]\n",
    "    \n",
    "    Args:\n",
    "        X: (N1, D) samples from distribution 1 (sparse)\n",
    "        Y: (N2, D) samples from distribution 2 (full)\n",
    "        n_projections: number of random 1D projections\n",
    "    \n",
    "    Returns:\n",
    "        sliced_w2: Sliced Wasserstein-2 distance\n",
    "    \"\"\"\n",
    "    D = X.shape[1]\n",
    "    distances = []\n",
    "    \n",
    "    for _ in range(n_projections):\n",
    "        # Random direction on unit sphere\n",
    "        theta = np.random.randn(D)\n",
    "        theta = theta / np.linalg.norm(theta)\n",
    "        \n",
    "        # Project both distributions\n",
    "        X_proj = X @ theta\n",
    "        Y_proj = Y @ theta\n",
    "        \n",
    "        # Compute 1D Wasserstein distance\n",
    "        w1d = wasserstein_distance(X_proj, Y_proj)\n",
    "        distances.append(w1d ** 2)\n",
    "    \n",
    "    return np.sqrt(np.mean(distances))\n",
    "\n",
    "\n",
    "def compute_alignment_error(latent_codes_sparse, latent_codes_full, \n",
    "                           attributes_sparse, attributes_full,\n",
    "                           distance_metric='euclidean', n_bins=5):\n",
    "    \"\"\"\n",
    "    METRIC 3: Alignment Error (τ_a ≤ 0.15 for Euclidean, ≤ 0.10 for Cosine)\n",
    "    \n",
    "    For CelebA: Group by attribute similarity bins instead of discrete classes\n",
    "    \n",
    "    Theoretical Link: Direct measure of coordinate system matching. Confirms that\n",
    "    paired encodings (sparse and full of the SAME sample) map to the SAME point.\n",
    "    \n",
    "    Args:\n",
    "        latent_codes_sparse: (N, D) from sparse encoder (encoder_x)\n",
    "        latent_codes_full: (N, D) from full encoder (encoder_s) - SAME samples\n",
    "        attributes_sparse: (N, A) attributes for sparse data\n",
    "        attributes_full: (N, A) attributes for full data (should match)\n",
    "        distance_metric: 'euclidean' or 'cosine'\n",
    "        n_bins: number of attribute bins for grouping\n",
    "    \n",
    "    Returns:\n",
    "        avg_error: average alignment error\n",
    "        per_bin_error: dict mapping bin_id -> alignment error\n",
    "        global_error: alignment error across all samples\n",
    "    \"\"\"\n",
    "    N = len(latent_codes_sparse)\n",
    "    \n",
    "    # Compute global error first\n",
    "    if distance_metric == 'euclidean':\n",
    "        global_error = float(np.mean(np.linalg.norm(\n",
    "            latent_codes_sparse - latent_codes_full, axis=1\n",
    "        )))\n",
    "    elif distance_metric == 'cosine':\n",
    "        global_error = float(np.mean(np.diag(cosine_distances(\n",
    "            latent_codes_sparse, latent_codes_full\n",
    "        ))))\n",
    "    else:\n",
    "        raise ValueError(f\"Unknown distance metric: {distance_metric}\")\n",
    "    \n",
    "    # Group samples by attribute similarity\n",
    "    # Use number of positive attributes as a simple binning strategy\n",
    "    attr_counts = np.sum(attributes_sparse > 0.5, axis=1)\n",
    "    bins = np.linspace(0, attributes_sparse.shape[1], n_bins + 1)\n",
    "    bin_indices = np.digitize(attr_counts, bins) - 1\n",
    "    bin_indices = np.clip(bin_indices, 0, n_bins - 1)\n",
    "    \n",
    "    per_bin_error = {}\n",
    "    \n",
    "    for bin_id in range(n_bins):\n",
    "        mask = (bin_indices == bin_id)\n",
    "        \n",
    "        if np.sum(mask) == 0:\n",
    "            continue\n",
    "        \n",
    "        sparse_bin = latent_codes_sparse[mask]\n",
    "        full_bin = latent_codes_full[mask]\n",
    "        \n",
    "        # Compute distance between paired encodings\n",
    "        if distance_metric == 'euclidean':\n",
    "            distances = np.linalg.norm(sparse_bin - full_bin, axis=1)\n",
    "        else:  # cosine\n",
    "            distances = np.diag(cosine_distances(sparse_bin, full_bin))\n",
    "        \n",
    "        per_bin_error[int(bin_id)] = float(np.mean(distances))\n",
    "    \n",
    "    # Average across all bins\n",
    "    avg_error = np.mean(list(per_bin_error.values())) if per_bin_error else global_error\n",
    "    \n",
    "    return avg_error, per_bin_error, global_error\n",
    "\n",
    "\n",
    "def compute_attribute_based_continuity(latent_codes, attributes, kappa=5, n_groups=10):\n",
    "    \"\"\"\n",
    "    METRIC 4: Attribute-Based Continuity (τ_c ≥ 0.80)\n",
    "    \n",
    "    For CelebA: Measures continuity within attribute-based groups\n",
    "    Groups samples by similar attribute patterns\n",
    "    \n",
    "    Theoretical Link: Ensures bounded stretching (c₂ < ∞) within the manifold\n",
    "    \n",
    "    Args:\n",
    "        latent_codes: (N, D) latent representations from ONE encoder only\n",
    "        attributes: (N, A) binary attributes\n",
    "        kappa: number of nearest neighbors\n",
    "        n_groups: number of attribute-based groups\n",
    "    \n",
    "    Returns:\n",
    "        continuity_score: average compactness score\n",
    "        per_group_continuity: dict mapping group_id to continuity score\n",
    "    \"\"\"\n",
    "    N = len(latent_codes)\n",
    "    \n",
    "    # Group samples by attribute patterns using k-means on attributes\n",
    "    from sklearn.cluster import KMeans\n",
    "    \n",
    "    kmeans = KMeans(n_clusters=n_groups, random_state=42, n_init=10)\n",
    "    group_labels = kmeans.fit_predict(attributes)\n",
    "    \n",
    "    per_group_continuity = {}\n",
    "    \n",
    "    for g in range(n_groups):\n",
    "        mask = (group_labels == g)\n",
    "        group_latents = latent_codes[mask]\n",
    "        \n",
    "        if len(group_latents) < kappa + 1:\n",
    "            continue\n",
    "        \n",
    "        # Within this group, build k-NN\n",
    "        nn = NearestNeighbors(n_neighbors=kappa+1, metric='euclidean', \n",
    "                             algorithm='auto', n_jobs=-1)\n",
    "        nn.fit(group_latents)\n",
    "        distances, _ = nn.kneighbors(group_latents)\n",
    "        distances = distances[:, 1:]  # Exclude self\n",
    "        \n",
    "        # Measure compactness via coefficient of variation (CV)\n",
    "        mean_dist = np.mean(distances)\n",
    "        std_dist = np.std(distances)\n",
    "        \n",
    "        # Continuity score: lower CV = more compact\n",
    "        cv = std_dist / (mean_dist + 1e-8)\n",
    "        continuity = 1.0 / (1.0 + cv)\n",
    "        \n",
    "        per_group_continuity[int(g)] = float(continuity)\n",
    "    \n",
    "    avg_continuity = np.mean(list(per_group_continuity.values()))\n",
    "    return avg_continuity, per_group_continuity\n",
    "\n",
    "\n",
    "def verify_celeba_homeomorphism(model, dataset_train, dataset_val, \n",
    "                                sparsity_levels=[0.05],\n",
    "                                kappa=5, device='cpu', n_projections=100,\n",
    "                                distance_metric='euclidean',\n",
    "                                max_samples=5000):\n",
    "    \"\"\"\n",
    "    Verify Topological Unification (Theorem 1) for CelebA Manifold Model\n",
    "    \n",
    "    THEORETICAL FRAMEWORK:\n",
    "    \n",
    "    Theorem 1 (Topological Unification):\n",
    "        Requires (i) Local homeomorphism and (ii) Global connectivity\n",
    "    \n",
    "    Theorem 2 (Bi-Lipschitz Sufficiency):\n",
    "        (c₁, c₂)-bi-Lipschitz encoder ⟹ homeomorphism\n",
    "    \n",
    "    Proposition 1 (Metric-Theoretical Bridge):\n",
    "        Violations manifest as:\n",
    "        (i) Manifold Collapse (c₁→0) ⟹ Low Trust Score\n",
    "        (ii) Geometric Misalignment ⟹ High Wasserstein Distance\n",
    "        (iii) Topological Disjointness (μ→0) ⟹ High Alignment Error\n",
    "        (iv) Metric Distortion (c₂→∞) ⟹ Low Continuity\n",
    "    \n",
    "    VERIFICATION PROTOCOL (FOUR METRICS):\n",
    "        ✓ Metric 0: β₀ = 1 - Topological connectivity\n",
    "        ✓ Metric 1: Trust Score (τ_t ≥ 0.80) - Local preservation (c₁ > 0)\n",
    "        ✓ Metric 2: Sliced W₂ (τ_w ≤ 0.30) - Global alignment (μ > 0)\n",
    "        ✓ Metric 3: Alignment Error (τ_a ≤ 0.15/0.10) - Coordinate matching\n",
    "        ✓ Metric 4: Continuity (τ_c ≥ 0.80) - Manifold compactness (c₂ < ∞)\n",
    "    \n",
    "    Args:\n",
    "        model: Trained SOTARobustModel\n",
    "        dataset_train: EnhancedCelebADataset for training\n",
    "        dataset_val: EnhancedCelebADataset for validation\n",
    "        sparsity_levels: List of sparsity ratios to test\n",
    "        kappa: Number of nearest neighbors\n",
    "        device: Computing device\n",
    "        n_projections: Random projections for Sliced Wasserstein\n",
    "        distance_metric: 'euclidean' or 'cosine' for alignment error\n",
    "        max_samples: Maximum samples to use (for computational efficiency)\n",
    "    \n",
    "    Returns:\n",
    "        results: Dictionary with all metrics per sparsity level\n",
    "    \"\"\"\n",
    "    print(\"\\n\" + \"=\"*80)\n",
    "    print(\"HOMEOMORPHISM VERIFICATION - CELEBA MANIFOLD MODEL\")\n",
    "    print(\"=\"*80)\n",
    "    print(f\"Latent dimension: {model.encoder_x.exit.out_channels} (spatial)\")\n",
    "    print(f\"Sparsity levels: {sparsity_levels}\")\n",
    "    print(f\"κ (neighbors): {kappa}\")\n",
    "    print(f\"Distance metric: {distance_metric}\")\n",
    "    print(f\"Max samples per split: {max_samples}\")\n",
    "    print(\"=\"*80)\n",
    "    \n",
    "    print(\"\\nTHEORETICAL JUSTIFICATION:\")\n",
    "    print(\"  Theorem 1: Topological Unification requires:\")\n",
    "    print(\"    (i) Local homeomorphism (each encoder)\")\n",
    "    print(\"    (ii) Global connectivity (non-trivial overlap)\")\n",
    "    print()\n",
    "    print(\"  Proposition 1: Violations manifest as:\")\n",
    "    print(\"    (i) Collapse (c₁→0) ⟹ Low Trust\")\n",
    "    print(\"    (ii) Misalignment ⟹ High W₂\")\n",
    "    print(\"    (iii) Disjointness (μ→0) ⟹ High Alignment Error\")\n",
    "    print(\"    (iv) Distortion (c₂→∞) ⟹ Low Continuity\")\n",
    "    print(\"=\"*80)\n",
    "    \n",
    "    print(\"\\nVERIFICATION METRICS:\")\n",
    "    print(\"  0. β₀ = 1 - Topological connectivity check\")\n",
    "    print(\"  1. Trust Score (τ_t ≥ 0.80) - Local preservation (c₁ > 0)\")\n",
    "    print(\"  2. Sliced W₂ (τ_w ≤ 0.30) - Global distribution alignment (μ > 0)\")\n",
    "    print(\"  3. Alignment Error (τ_a ≤ 0.15/0.10) - Coordinate matching\")\n",
    "    print(\"  4. Continuity (τ_c ≥ 0.80) - Manifold compactness (SPARSE only)\")\n",
    "    print(\"=\"*80)\n",
    "    \n",
    "    model.eval()\n",
    "    model.to(device)\n",
    "    \n",
    "    results = {}\n",
    "    \n",
    "    for sparsity in sparsity_levels:\n",
    "        print(f\"\\n{'='*80}\")\n",
    "        print(f\"EVALUATING AT ρ = {sparsity:.2f} ({sparsity:.0%} visible pixels)\")\n",
    "        print(f\"{'='*80}\")\n",
    "        \n",
    "        # ====================================================================\n",
    "        # ENCODE TRAINING DATA\n",
    "        # ====================================================================\n",
    "        print(f\"\\nEncoding TRAINING samples...\")\n",
    "        \n",
    "        # Limit samples for computational efficiency\n",
    "        train_indices = np.random.choice(len(dataset_train), \n",
    "                                        min(max_samples, len(dataset_train)), \n",
    "                                        replace=False)\n",
    "        \n",
    "        latent_codes_train_sparse = []\n",
    "        latent_codes_train_full = []\n",
    "        attributes_train = []\n",
    "        \n",
    "        train_loader = DataLoader(\n",
    "            torch.utils.data.Subset(dataset_train, train_indices),\n",
    "            batch_size=64, shuffle=False, num_workers=0\n",
    "        )\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            for x, s in tqdm(train_loader, desc=\"Train Encoding\", leave=False):\n",
    "                sparse_image = x['sparse_image'].to(device)\n",
    "                mask = x['mask'].to(device)\n",
    "                full_image = s['full_image'].to(device)\n",
    "                attr = s['attributes'].to(device)\n",
    "                \n",
    "                x_in = {'sparse_image': sparse_image, 'mask': mask}\n",
    "                s_in = {'full_image': full_image, 'attributes': attr}\n",
    "                \n",
    "                # Get embeddings (no dropout during evaluation)\n",
    "                attr_embed = model.attr_embedding(attr)\n",
    "                \n",
    "                # Encode sparse and full\n",
    "                z_sparse = model.encoder_x(sparse_image, mask, attr_embed)\n",
    "                z_full = model.encoder_s(full_image, None, attr_embed)\n",
    "                \n",
    "                # Flatten spatial dimensions\n",
    "                z_sparse_flat = z_sparse.view(z_sparse.size(0), -1)\n",
    "                z_full_flat = z_full.view(z_full.size(0), -1)\n",
    "                \n",
    "                latent_codes_train_sparse.append(z_sparse_flat.cpu().numpy())\n",
    "                latent_codes_train_full.append(z_full_flat.cpu().numpy())\n",
    "                attributes_train.append(attr.cpu().numpy())\n",
    "        \n",
    "        latent_codes_train_sparse = np.vstack(latent_codes_train_sparse)\n",
    "        latent_codes_train_full = np.vstack(latent_codes_train_full)\n",
    "        attributes_train = np.vstack(attributes_train)\n",
    "        \n",
    "        # ====================================================================\n",
    "        # ENCODE VALIDATION DATA\n",
    "        # ====================================================================\n",
    "        print(f\"Encoding VALIDATION samples...\")\n",
    "        \n",
    "        val_indices = np.random.choice(len(dataset_val), \n",
    "                                      min(max_samples, len(dataset_val)), \n",
    "                                      replace=False)\n",
    "        \n",
    "        latent_codes_val_sparse = []\n",
    "        latent_codes_val_full = []\n",
    "        attributes_val = []\n",
    "        \n",
    "        val_loader = DataLoader(\n",
    "            torch.utils.data.Subset(dataset_val, val_indices),\n",
    "            batch_size=64, shuffle=False, num_workers=0\n",
    "        )\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            for x, s in tqdm(val_loader, desc=\"Val Encoding\", leave=False):\n",
    "                sparse_image = x['sparse_image'].to(device)\n",
    "                mask = x['mask'].to(device)\n",
    "                full_image = s['full_image'].to(device)\n",
    "                attr = s['attributes'].to(device)\n",
    "                \n",
    "                x_in = {'sparse_image': sparse_image, 'mask': mask}\n",
    "                s_in = {'full_image': full_image, 'attributes': attr}\n",
    "                \n",
    "                attr_embed = model.attr_embedding(attr)\n",
    "                \n",
    "                z_sparse = model.encoder_x(sparse_image, mask, attr_embed)\n",
    "                z_full = model.encoder_s(full_image, None, attr_embed)\n",
    "                \n",
    "                z_sparse_flat = z_sparse.view(z_sparse.size(0), -1)\n",
    "                z_full_flat = z_full.view(z_full.size(0), -1)\n",
    "                \n",
    "                latent_codes_val_sparse.append(z_sparse_flat.cpu().numpy())\n",
    "                latent_codes_val_full.append(z_full_flat.cpu().numpy())\n",
    "                attributes_val.append(attr.cpu().numpy())\n",
    "        \n",
    "        latent_codes_val_sparse = np.vstack(latent_codes_val_sparse)\n",
    "        latent_codes_val_full = np.vstack(latent_codes_val_full)\n",
    "        attributes_val = np.vstack(attributes_val)\n",
    "        \n",
    "        print(f\"\\nEncoded shapes:\")\n",
    "        print(f\"  Train Sparse: {latent_codes_train_sparse.shape}\")\n",
    "        print(f\"  Train Full:   {latent_codes_train_full.shape}\")\n",
    "        print(f\"  Val Sparse:   {latent_codes_val_sparse.shape}\")\n",
    "        print(f\"  Val Full:     {latent_codes_val_full.shape}\")\n",
    "        \n",
    "        N_train = len(latent_codes_train_sparse)\n",
    "        N_val = len(latent_codes_val_sparse)\n",
    "        \n",
    "        # ====================================================================\n",
    "        # METRIC 0: β₀ (Topological Connectivity Check)\n",
    "        # ====================================================================\n",
    "        print(f\"\\n[0/4] Computing β₀ (connectivity check)...\")\n",
    "        \n",
    "        # Combine ALL data: train_sparse, train_full, val_sparse, val_full\n",
    "        latent_codes_all = np.vstack([\n",
    "            latent_codes_train_sparse,\n",
    "            latent_codes_train_full,\n",
    "            latent_codes_val_sparse,\n",
    "            latent_codes_val_full\n",
    "        ])\n",
    "        \n",
    "        beta_0, component_labels = compute_betti_0(latent_codes_all, kappa=10)\n",
    "        \n",
    "        print(f\"  β₀ = {beta_0}\")\n",
    "        if beta_0 == 1:\n",
    "            print(f\"  ✓ Single connected manifold (Theorem 1(ii) satisfied)\")\n",
    "        else:\n",
    "            print(f\"  ✗ {beta_0} disconnected components (μ→0 violation)\")\n",
    "            unique_components, counts = np.unique(component_labels, return_counts=True)\n",
    "            print(f\"  Component sizes: {dict(zip(unique_components, counts))}\")\n",
    "        \n",
    "        # ====================================================================\n",
    "        # METRIC 1: TRUST SCORE (Local Preservation)\n",
    "        # ====================================================================\n",
    "        print(f\"\\n[1/4] Computing TRUST SCORE (τ_t ≥ 0.80)...\")\n",
    "        \n",
    "        # Compute trust on SPARSE encodings (combined train+val)\n",
    "        latent_codes_sparse_all = np.vstack([latent_codes_train_sparse, \n",
    "                                             latent_codes_val_sparse])\n",
    "        attributes_all = np.vstack([attributes_train, attributes_val])\n",
    "        \n",
    "        trust_score = compute_trust_score(latent_codes_sparse_all, \n",
    "                                         attributes_all, kappa=kappa)\n",
    "        \n",
    "        print(f\"  Trust Score: {trust_score:.4f}\")\n",
    "        if trust_score >= 0.80:\n",
    "            print(f\"  ✓ Local preservation verified (c₁ > 0)\")\n",
    "        else:\n",
    "            print(f\"  ✗ Manifold collapse detected (c₁→0 violation)\")\n",
    "        \n",
    "        # ====================================================================\n",
    "        # METRIC 2: SLICED WASSERSTEIN (Global Alignment)\n",
    "        # ====================================================================\n",
    "        print(f\"\\n[2/4] Computing SLICED WASSERSTEIN-2 (τ_w ≤ 0.30)...\")\n",
    "        \n",
    "        # Compare SPARSE vs FULL encodings (using training data)\n",
    "        max_w2_samples = min(2000, N_train)\n",
    "        if N_train > max_w2_samples:\n",
    "            idx = np.random.choice(N_train, max_w2_samples, replace=False)\n",
    "            sparse_sample = latent_codes_train_sparse[idx]\n",
    "            full_sample = latent_codes_train_full[idx]\n",
    "        else:\n",
    "            sparse_sample = latent_codes_train_sparse\n",
    "            full_sample = latent_codes_train_full\n",
    "        \n",
    "        w2_distance = compute_sliced_wasserstein(sparse_sample, full_sample, \n",
    "                                                 n_projections=n_projections)\n",
    "        \n",
    "        print(f\"  Sliced W₂ (Sparse↔Full): {w2_distance:.4f}\")\n",
    "        if w2_distance <= 0.30:\n",
    "            print(f\"  ✓ Global alignment verified (distributions overlap)\")\n",
    "        else:\n",
    "            print(f\"  ✗ Geometric misalignment detected\")\n",
    "        \n",
    "        # ====================================================================\n",
    "        # METRIC 3: ALIGNMENT ERROR (Coordinate Matching/Merger)\n",
    "        # ====================================================================\n",
    "        threshold_align = 0.15 if distance_metric == 'euclidean' else 0.10\n",
    "        print(f\"\\n[3/4] Computing ALIGNMENT ERROR (τ_a ≤ {threshold_align})...\")\n",
    "        print(f\"  Distance metric: {distance_metric}\")\n",
    "        \n",
    "        # Compute on COMBINED train+val for robust statistics\n",
    "        latent_codes_sparse_all = np.vstack([latent_codes_train_sparse, \n",
    "                                             latent_codes_val_sparse])\n",
    "        latent_codes_full_all = np.vstack([latent_codes_train_full, \n",
    "                                           latent_codes_val_full])\n",
    "        attributes_all = np.vstack([attributes_train, attributes_val])\n",
    "        \n",
    "        alignment_error, per_bin_error, global_error = compute_alignment_error(\n",
    "            latent_codes_sparse_all,\n",
    "            latent_codes_full_all,\n",
    "            attributes_all,\n",
    "            attributes_all,  # Same attributes (same samples, different encodings)\n",
    "            distance_metric=distance_metric\n",
    "        )\n",
    "        \n",
    "        print(f\"  Alignment Error (avg):  {alignment_error:.4f}\")\n",
    "        print(f\"  Global Alignment Error: {global_error:.4f}\")\n",
    "        \n",
    "        if alignment_error <= threshold_align:\n",
    "            print(f\"  ✓ Sparse and full map to THE SAME POINTS (perfect merger)\")\n",
    "        else:\n",
    "            print(f\"  ✗ Sparse and full use DIFFERENT coordinate systems\")\n",
    "        \n",
    "        print(f\"  Per-bin alignment errors:\")\n",
    "        for bin_id in sorted(per_bin_error.keys()):\n",
    "            status = \"✓\" if per_bin_error[bin_id] <= threshold_align else \"✗\"\n",
    "            print(f\"    Bin {bin_id}: {per_bin_error[bin_id]:.4f} {status}\")\n",
    "        \n",
    "        # ====================================================================\n",
    "        # METRIC 4: CONTINUITY (Manifold Compactness - SPARSE ONLY)\n",
    "        # ====================================================================\n",
    "        print(f\"\\n[4/4] Computing ATTRIBUTE-BASED CONTINUITY (τ_c ≥ 0.80)...\")\n",
    "        print(f\"  Measuring on SPARSE encodings only\")\n",
    "        \n",
    "        continuity_score, per_group_continuity = compute_attribute_based_continuity(\n",
    "            latent_codes_sparse_all,\n",
    "            attributes_all,\n",
    "            kappa=kappa,\n",
    "            n_groups=10\n",
    "        )\n",
    "        \n",
    "        print(f\"  Continuity (Sparse): {continuity_score:.4f}\")\n",
    "        if continuity_score >= 0.80:\n",
    "            print(f\"  ✓ Manifold groups form coherent, compact clusters (c₂ < ∞)\")\n",
    "        else:\n",
    "            print(f\"  ⚠ Groups may be fragmented (metric distortion, c₂→∞)\")\n",
    "        \n",
    "        print(f\"  Per-group continuity (first 5):\")\n",
    "        for group_id in sorted(list(per_group_continuity.keys())[:5]):\n",
    "            status = \"✓\" if per_group_continuity[group_id] >= 0.80 else \"✗\"\n",
    "            print(f\"    Group {group_id}: {per_group_continuity[group_id]:.4f} {status}\")\n",
    "        \n",
    "        # ====================================================================\n",
    "        # STORE RESULTS\n",
    "        # ====================================================================\n",
    "        results[sparsity] = {\n",
    "            # Topology\n",
    "            'beta_0': int(beta_0),\n",
    "            \n",
    "            # Metric 1: Trust (local preservation)\n",
    "            'trust_score': float(trust_score),\n",
    "            \n",
    "            # Metric 2: Wasserstein (global alignment)\n",
    "            'sliced_w2': float(w2_distance),\n",
    "            \n",
    "            # Metric 3: Alignment Error (coordinate matching)\n",
    "            'alignment_error': float(alignment_error),\n",
    "            'global_alignment_error': float(global_error),\n",
    "            'per_bin_alignment': per_bin_error,\n",
    "            \n",
    "            # Metric 4: Continuity (manifold compactness - SPARSE)\n",
    "            'continuity': float(continuity_score),\n",
    "            'per_group_continuity': per_group_continuity,\n",
    "            \n",
    "            # Metadata\n",
    "            'kappa': kappa,\n",
    "            'distance_metric': distance_metric,\n",
    "            'n_train': N_train,\n",
    "            'n_val': N_val,\n",
    "            'n_projections': n_projections\n",
    "        }\n",
    "        \n",
    "        print(f\"\\n{'='*80}\")\n",
    "        print(f\"SUMMARY FOR ρ = {sparsity:.2f}:\")\n",
    "        print(f\"  β₀:              {beta_0}\")\n",
    "        print(f\"  Trust Score:     {trust_score:.4f} {'✓' if trust_score >= 0.80 else '✗'}\")\n",
    "        print(f\"  Sliced W₂:       {w2_distance:.4f} {'✓' if w2_distance <= 0.30 else '✗'}\")\n",
    "        print(f\"  Alignment Error: {alignment_error:.4f} {'✓' if alignment_error <= threshold_align else '✗'}\")\n",
    "        print(f\"  Continuity:      {continuity_score:.4f} {'✓' if continuity_score >= 0.80 else '✗'}\")\n",
    "        print(f\"{'='*80}\")\n",
    "    \n",
    "    # ========================================================================\n",
    "    # FINAL VERIFICATION SUMMARY\n",
    "    # ========================================================================\n",
    "    threshold_align = 0.15 if distance_metric == 'euclidean' else 0.10\n",
    "    \n",
    "    print(\"\\n\" + \"=\"*80)\n",
    "    print(\"TOPOLOGICAL UNIFICATION VERIFICATION (THEOREM 1)\")\n",
    "    print(\"=\"*80)\n",
    "    print(f\"{'ρ':<8} {'β₀':<6} {'Trust':<10} {'W₂':<10} {'Align':<10} {'Cont':<10}\")\n",
    "    print(\"-\"*80)\n",
    "    for sparsity in sparsity_levels:\n",
    "        r = results[sparsity]\n",
    "        print(f\"{sparsity:<8.2f} {r['beta_0']:<6} {r['trust_score']:<10.4f} \"\n",
    "              f\"{r['sliced_w2']:<10.4f} {r['alignment_error']:<10.4f} \"\n",
    "              f\"{r['continuity']:<10.4f}\")\n",
    "    print(\"=\"*80)\n",
    "    \n",
    "    print(\"\\nTARGET THRESHOLDS:\")\n",
    "    print(\"  β₀ = 1             (single connected manifold)\")\n",
    "    print(\"  Trust ≥ 0.80       (local preservation, c₁ > 0)\")\n",
    "    print(\"  W₂ ≤ 0.30          (global distribution alignment, μ > 0)\")\n",
    "    print(f\"  Align ≤ {threshold_align}          (coordinate matching, topological merger)\")\n",
    "    print(\"  Cont ≥ 0.80        (manifold compactness, c₂ < ∞, SPARSE only)\")\n",
    "    print(\"-\"*80)\n",
    "    \n",
    "    # Check verification status\n",
    "    all_pass = True\n",
    "    \n",
    "    for sparsity in sparsity_levels:\n",
    "        r = results[sparsity]\n",
    "        \n",
    "        beta_pass = (r['beta_0'] == 1)\n",
    "        trust_pass = (r['trust_score'] >= 0.80)\n",
    "        w2_pass = (r['sliced_w2'] <= 0.30)\n",
    "        align_pass = (r['alignment_error'] <= threshold_align)\n",
    "        cont_pass = (r['continuity'] >= 0.80)\n",
    "        \n",
    "        status = \"✓ PASS\" if (beta_pass and trust_pass and w2_pass and \n",
    "                             align_pass and cont_pass) else \"✗ FAIL\"\n",
    "        \n",
    "        print(f\"\\nρ = {sparsity:.2f}: {status}\")\n",
    "        print(f\"  β₀:        {'✓' if beta_pass else '✗'} (connectivity)\")\n",
    "        print(f\"  Trust:     {'✓' if trust_pass else '✗'} (local preservation)\")\n",
    "        print(f\"  W₂:        {'✓' if w2_pass else '✗'} (global alignment)\")\n",
    "        print(f\"  Align:     {'✓' if align_pass else '✗'} (coordinate matching)\")\n",
    "        print(f\"  Cont:      {'✓' if cont_pass else '✗'} (manifold compactness)\")\n",
    "        \n",
    "        if not (beta_pass and trust_pass and w2_pass and align_pass and cont_pass):\n",
    "            all_pass = False\n",
    "    \n",
    "    print(\"\\n\" + \"=\"*80)\n",
    "    if all_pass:\n",
    "        print(\"VERIFICATION RESULT: ✓ THEOREM 1 SATISFIED (COMPLETE)\")\n",
    "        print(\"  - β₀ = 1: Single connected manifold\")\n",
    "        print(\"  - Trust ≥ 0.80: Local homeomorphism (Condition i, c₁ > 0)\")\n",
    "        print(\"  - W₂ ≤ 0.30: Sparse↔Full distributions co-located (Condition ii, μ > 0)\")\n",
    "        print(f\"  - Align ≤ {threshold_align}: Sparse↔Full map to SAME points (perfect merger)\")\n",
    "        print(\"  - Cont ≥ 0.80: Manifold groups form compact clusters (c₂ < ∞)\")\n",
    "        print()\n",
    "        print(\"CONCLUSION: Sparse and Full encodings form ONE UNIVERSAL MANIFOLD\")\n",
    "        print(\"            with TOPOLOGICALLY MERGED representations and\")\n",
    "        print(\"            COMPACT, SEMANTICALLY COHERENT attribute clusters.\")\n",
    "    else:\n",
    "        print(\"VERIFICATION RESULT: ✗ THEOREM 1 VIOLATED\")\n",
    "        print(\"  Check individual metrics above for failure modes\")\n",
    "    print(\"=\"*80 + \"\\n\")\n",
    "    \n",
    "    return results\n",
    "\n",
    "\n",
    "def print_latex_table_celeba(results, kappa=5, distance_metric='euclidean'):\n",
    "    \"\"\"\n",
    "    Generate LaTeX table for CelebA verification results\n",
    "    \"\"\"\n",
    "    sparsity_levels = sorted(results.keys())\n",
    "    threshold_align = 0.15 if distance_metric == 'euclidean' else 0.10\n",
    "    \n",
    "    print(\"\\n\" + \"=\"*80)\n",
    "    print(\"LaTeX Table (CelebA Verification):\")\n",
    "    print(\"=\"*80)\n",
    "    print(r\"\\begin{table}[!htpb]\")\n",
    "    print(r\"\\centering\")\n",
    "    print(r\"\\caption{Topological Unification Verification - CelebA ($\\kappa=\" + str(kappa) + \n",
    "          r\"$, \" + distance_metric.capitalize() + r\" distance)}\")\n",
    "    print(r\"\\label{tab:celeba_unification}\")\n",
    "    \n",
    "    # Determine number of columns\n",
    "    n_cols = len(sparsity_levels)\n",
    "    col_spec = \"l\" + \"c\" * n_cols\n",
    "    print(r\"\\begin{tabular}{\" + col_spec + \"}\")\n",
    "    print(r\"\\hline\")\n",
    "    \n",
    "    # Header\n",
    "    header = r\"\\textbf{Metric}\"\n",
    "    for rho in sparsity_levels:\n",
    "        header += f\" & $\\\\rho={rho:.2f}$\"\n",
    "    header += r\" \\\\\"\n",
    "    print(header)\n",
    "    print(r\"\\hline\")\n",
    "    \n",
    "    # β₀\n",
    "    beta_row = r\"$\\beta_0$ (connectivity)\"\n",
    "    for rho in sparsity_levels:\n",
    "        val = results[rho]['beta_0']\n",
    "        beta_row += f\" & {val}\"\n",
    "    beta_row += r\" \\\\\"\n",
    "    print(beta_row)\n",
    "    \n",
    "    # Trust\n",
    "    trust_row = r\"Trust $\\tau_t$ ($\\geq 0.80$)\"\n",
    "    for rho in sparsity_levels:\n",
    "        val = results[rho]['trust_score']\n",
    "        trust_row += f\" & {val:.3f}\"\n",
    "    trust_row += r\" \\\\\"\n",
    "    print(trust_row)\n",
    "    \n",
    "    # Sliced W₂\n",
    "    w2_row = r\"Sliced $W_2$ $\\tau_w$ ($\\leq 0.30$)\"\n",
    "    for rho in sparsity_levels:\n",
    "        val = results[rho]['sliced_w2']\n",
    "        w2_row += f\" & {val:.3f}\"\n",
    "    w2_row += r\" \\\\\"\n",
    "    print(w2_row)\n",
    "    \n",
    "    # Alignment Error\n",
    "    align_row = f\"Alignment $\\\\tau_a$ ($\\\\leq {threshold_align:.2f}$)\"\n",
    "    for rho in sparsity_levels:\n",
    "        val = results[rho]['alignment_error']\n",
    "        align_row += f\" & {val:.3f}\"\n",
    "    align_row += r\" \\\\\"\n",
    "    print(align_row)\n",
    "    \n",
    "    # Continuity\n",
    "    cont_row = r\"Continuity $\\tau_c$ ($\\geq 0.80$)\"\n",
    "    for rho in sparsity_levels:\n",
    "        val = results[rho]['continuity']\n",
    "        cont_row += f\" & {val:.3f}\"\n",
    "    cont_row += r\" \\\\\"\n",
    "    print(cont_row)\n",
    "    \n",
    "    print(r\"\\hline\")\n",
    "    print(r\"\\end{tabular}\")\n",
    "    print(r\"\\end{table}\")\n",
    "    print(\"=\"*80 + \"\\n\")\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# MAIN EXECUTION\n",
    "# ============================================================================\n",
    "\n",
    "if __name__ == '__main__':\n",
    "    print(\"=\"*80)\n",
    "    print(\"HOMEOMORPHISM VERIFICATION - CELEBA MANIFOLD MODEL\")\n",
    "    print(\"Theorem 1: Topological Unification\")\n",
    "    print(\"Proposition 1: Metric-Theoretical Bridge\")\n",
    "    print(\"Four-Metric Verification: β₀, Trust, W₂, Alignment, Continuity\")\n",
    "    print(\"=\"*80 + \"\\n\")\n",
    "    \n",
    "    # Load trained model\n",
    "    print(\"Loading trained CelebA model...\")\n",
    "    model = SOTARobustModel()\n",
    "    \n",
    "    if os.path.exists('sota_v3_best.pth'):\n",
    "        model.load_state_dict(torch.load('sota_v3_best.pth', map_location='cpu'))\n",
    "        print(\"Loaded: sota_v3_best.pth\")\n",
    "    elif os.path.exists('sota_best_manifold.pth'):\n",
    "        model.load_state_dict(torch.load('sota_best_manifold.pth', map_location='cpu'))\n",
    "        print(\"Loaded: sota_best_manifold.pth\")\n",
    "    else:\n",
    "        print(\"ERROR: No trained model found!\")\n",
    "        print(\"Please ensure 'sota_v3_best.pth' or 'sota_best_manifold.pth' exists.\")\n",
    "        sys.exit(1)\n",
    "    \n",
    "    device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "    model.to(device)\n",
    "    print(f\"Model loaded on {device}\\n\")\n",
    "    \n",
    "    # Run verification for multiple sparsity levels\n",
    "    sparsity_levels = [0.05, 0.08, 0.10, 0.12, 0.15]\n",
    "    \n",
    "    # Choose distance metric: 'euclidean' or 'cosine'\n",
    "    DISTANCE_METRIC = 'cosine'\n",
    "    \n",
    "    # We need to recreate datasets for each sparsity level\n",
    "    # So we'll call the verification function directly with dataset creation inside\n",
    "    results = {}\n",
    "    \n",
    "    for sparsity in sparsity_levels:\n",
    "        print(f\"\\n{'='*80}\")\n",
    "        print(f\"TESTING SPARSITY LEVEL: ρ = {sparsity:.2f}\")\n",
    "        print(f\"{'='*80}\")\n",
    "        \n",
    "        # Create datasets with this sparsity level\n",
    "        train_ds = EnhancedCelebADataset('./data', split='train', sparsity=sparsity)\n",
    "        val_ds = EnhancedCelebADataset('./data', split='valid', sparsity=sparsity)\n",
    "        \n",
    "        # Run verification for this sparsity level\n",
    "        sparsity_results = verify_celeba_homeomorphism(\n",
    "            model=model,\n",
    "            dataset_train=train_ds,\n",
    "            dataset_val=val_ds,\n",
    "            sparsity_levels=[sparsity],  # Single sparsity per run\n",
    "            kappa=5,\n",
    "            device=device,\n",
    "            n_projections=100,\n",
    "            distance_metric=DISTANCE_METRIC,\n",
    "            max_samples=5000\n",
    "        )\n",
    "        \n",
    "        # Store results\n",
    "        results.update(sparsity_results)\n",
    "    \n",
    "    # Generate LaTeX table\n",
    "    print_latex_table_celeba(results, kappa=5, distance_metric=DISTANCE_METRIC)\n",
    "    \n",
    "    # Save results\n",
    "    output_file = f'celeba_verification_results_{DISTANCE_METRIC}.json'\n",
    "    with open(output_file, 'w') as f:\n",
    "        json.dump(results, f, indent=2)\n",
    "    print(f\"✓ Results saved to '{output_file}'\")\n",
    "    \n",
    "    print(\"\\n\" + \"=\"*80)\n",
    "    print(\"VERIFICATION COMPLETE!\")\n",
    "    print(\"=\"*80)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "9c9b0d63",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "================================================================================\n",
      "CELEBA SPARSE RECOVERY EVALUATION\n",
      "================================================================================\n",
      "\n",
      "Loading models...\n",
      "✓ Loaded: sota_v3_best.pth\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/to247392/anaconda3/envs/manitorch/lib/python3.11/site-packages/torch/cuda/__init__.py:789: UserWarning: Can't initialize NVML\n",
      "  warnings.warn(\"Can't initialize NVML\")\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Models loaded: ['V4-Robust']\n",
      "Device: cuda\n",
      "\n",
      "\n",
      "================================================================================\n",
      "EVALUATING: V4-Robust\n",
      "================================================================================\n",
      "================================================================================\n",
      "CELEBA SPARSE RECOVERY PERFORMANCE EVALUATION\n",
      "================================================================================\n",
      "Model: SOTARobustModel\n",
      "Sparsity levels: [0.05, 0.08, 0.1, 0.12, 0.15]\n",
      "Device: cuda\n",
      "================================================================================\n",
      "\n",
      "\n",
      "============================================================\n",
      "Evaluating at ρ = 0.05 (5% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                               \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results for ρ = 0.05:\n",
      "  MSE:  0.003762 ± 0.001804\n",
      "  PSNR: 24.68 ± 1.94 dB\n",
      "  Samples: 19962\n",
      "  ✓ Excellent quality (MSE ≤ 0.005)\n",
      "\n",
      "============================================================\n",
      "Evaluating at ρ = 0.08 (8% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                               \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results for ρ = 0.08:\n",
      "  MSE:  0.002789 ± 0.001332\n",
      "  PSNR: 25.97 ± 1.90 dB\n",
      "  Samples: 19962\n",
      "  ✓ Excellent quality (MSE ≤ 0.005)\n",
      "\n",
      "============================================================\n",
      "Evaluating at ρ = 0.10 (10% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                               \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results for ρ = 0.10:\n",
      "  MSE:  0.002500 ± 0.001228\n",
      "  PSNR: 26.44 ± 1.87 dB\n",
      "  Samples: 19962\n",
      "  ✓ Excellent quality (MSE ≤ 0.005)\n",
      "\n",
      "============================================================\n",
      "Evaluating at ρ = 0.12 (12% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                               \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results for ρ = 0.12:\n",
      "  MSE:  0.002378 ± 0.001443\n",
      "  PSNR: 26.68 ± 1.87 dB\n",
      "  Samples: 19962\n",
      "  ✓ Excellent quality (MSE ≤ 0.005)\n",
      "\n",
      "============================================================\n",
      "Evaluating at ρ = 0.15 (15% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                               "
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results for ρ = 0.15:\n",
      "  MSE:  0.002427 ± 0.002422\n",
      "  PSNR: 26.70 ± 1.91 dB\n",
      "  Samples: 19962\n",
      "  ✓ Excellent quality (MSE ≤ 0.005)\n",
      "\n",
      "================================================================================\n",
      "LaTeX Table - CelebA Sparse Recovery Performance:\n",
      "================================================================================\n",
      "\\begin{table}[!htpb]\n",
      "\\centering\n",
      "\\caption{CelebA sparse recovery performance across sparsity levels}\n",
      "\\label{tab:celeba_performance}\n",
      "\\begin{tabular}{lccccc}\n",
      "\\hline\n",
      "\\textbf{Method} & $\\boldsymbol{\\rho=0.05}$ & $\\boldsymbol{\\rho=0.08}$ & $\\boldsymbol{\\rho=0.10}$ & $\\boldsymbol{\\rho=0.12}$ & $\\boldsymbol{\\rho=0.15}$ \\\\\n",
      "\\hline\n",
      "V4-Robust (MSE) & 0.0038 & 0.0028 & 0.0025 & 0.0024 & 0.0024 \\\\\n",
      "V4-Robust (PSNR dB) & 24.68 & 25.97 & 26.44 & 26.68 & 26.70 \\\\\n",
      "\\hline\n",
      "\\end{tabular}\n",
      "\\end{table}\n",
      "================================================================================\n",
      "\n",
      "✓ Results saved to 'celeba_performance_results.json'\n",
      "\n",
      "================================================================================\n",
      "EVALUATION COMPLETE!\n",
      "================================================================================\n",
      "\n",
      "V4-Robust Summary:\n",
      "Sparsity     MSE          PSNR (dB)   \n",
      "----------------------------------------\n",
      "ρ=0.05       0.003762     24.68       \n",
      "ρ=0.08       0.002789     25.97       \n",
      "ρ=0.10       0.002500     26.44       \n",
      "ρ=0.12       0.002378     26.68       \n",
      "ρ=0.15       0.002427     26.70       \n",
      "\n",
      "================================================================================\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r"
     ]
    }
   ],
   "source": [
    "# ============================================================================\n",
    "# CELEBA SPARSE RECOVERY PERFORMANCE EVALUATION\n",
    "# Tests reconstruction quality across multiple sparsity levels\n",
    "# Generates LaTeX table with MSE and PSNR metrics\n",
    "# ============================================================================\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torch.utils.data import DataLoader\n",
    "from torchvision import datasets, transforms\n",
    "import numpy as np\n",
    "from tqdm import tqdm\n",
    "import json\n",
    "import os\n",
    "\n",
    "# ============================================================================\n",
    "# CONFIGURATION\n",
    "# ============================================================================\n",
    "CONFIG = {\n",
    "    'img_size': 64,\n",
    "    'batch_size': 64,\n",
    "    'latent_dim': 1024,\n",
    "    'attr_dim': 40,\n",
    "    'attr_dropout': 0.5,\n",
    "    'device': 'cuda' if torch.cuda.is_available() else 'cpu'\n",
    "}\n",
    "\n",
    "# ============================================================================\n",
    "# DATASET\n",
    "# ============================================================================\n",
    "class EnhancedCelebADataset(torch.utils.data.Dataset):\n",
    "    def __init__(self, root, split='test', sparsity=0.05, img_size=64):\n",
    "        self.transform = transforms.Compose([\n",
    "            transforms.CenterCrop(178),\n",
    "            transforms.Resize(img_size),\n",
    "            transforms.ToTensor(),\n",
    "            transforms.Normalize((0.5, 0.5, 0.5), (0.5, 0.5, 0.5))\n",
    "        ])\n",
    "        try:\n",
    "            self.data = datasets.CelebA(root=root, split=split, download=False, transform=self.transform)\n",
    "        except:\n",
    "            self.data = datasets.CelebA(root=root, split=split, download=True, transform=self.transform)\n",
    "        self.sparsity = sparsity\n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self.data)\n",
    "\n",
    "    def __getitem__(self, idx):\n",
    "        full_image, attributes = self.data[idx]\n",
    "        mask = (torch.rand_like(full_image) < self.sparsity).float()\n",
    "        sparse_image = full_image * mask\n",
    "        attributes = attributes.float()\n",
    "        x = {'sparse_image': sparse_image, 'mask': mask}\n",
    "        s = {'full_image': full_image, 'attributes': attributes}\n",
    "        return x, s\n",
    "\n",
    "# ============================================================================\n",
    "# MODEL COMPONENTS\n",
    "# ============================================================================\n",
    "class AdaIN(nn.Module):\n",
    "    def __init__(self, channels, embed_dim):\n",
    "        super().__init__()\n",
    "        self.channels = channels\n",
    "        self.linear = nn.Linear(embed_dim, channels * 2)\n",
    "\n",
    "    def forward(self, x, attr_embed):\n",
    "        style = self.linear(attr_embed)\n",
    "        style = style.view(-1, 2, self.channels, 1, 1)\n",
    "        gamma, beta = style[:, 0], style[:, 1]\n",
    "        mean = x.mean(dim=[2, 3], keepdim=True)\n",
    "        std = x.std(dim=[2, 3], keepdim=True) + 1e-8\n",
    "        norm_x = (x - mean) / std\n",
    "        return gamma * norm_x + beta\n",
    "\n",
    "class AdaINResBlock(nn.Module):\n",
    "    def __init__(self, c, embed_dim):\n",
    "        super().__init__()\n",
    "        self.conv1 = nn.Conv2d(c, c, 3, 1, 1, bias=False)\n",
    "        self.conv2 = nn.Conv2d(c, c, 3, 1, 1, bias=False)\n",
    "        self.adain1 = AdaIN(c, embed_dim)\n",
    "        self.adain2 = AdaIN(c, embed_dim)\n",
    "        self.act = nn.ReLU()\n",
    "        \n",
    "    def forward(self, x, attr_embed):\n",
    "        residual = x\n",
    "        out = self.conv1(x)\n",
    "        out = self.act(self.adain1(out, attr_embed))\n",
    "        out = self.conv2(out)\n",
    "        out = self.adain2(out, attr_embed)\n",
    "        return self.act(out + residual)\n",
    "\n",
    "class RobustEncoder(nn.Module):\n",
    "    def __init__(self, input_channels=6, latent_channels=512, attr_dim=40, embed_dim=256):\n",
    "        super().__init__()\n",
    "        self.entry = nn.Sequential(nn.Conv2d(input_channels, 64, 4, 2, 1), nn.ReLU())\n",
    "        self.layer1 = nn.Sequential(nn.Conv2d(64, 128, 4, 2, 1), nn.BatchNorm2d(128), nn.ReLU())\n",
    "        self.layer2 = nn.Sequential(nn.Conv2d(128, 256, 4, 2, 1), nn.BatchNorm2d(256), nn.ReLU())\n",
    "        self.res1 = AdaINResBlock(256, embed_dim)\n",
    "        self.res2 = AdaINResBlock(256, embed_dim)\n",
    "        self.exit = nn.Conv2d(256, latent_channels, 4, 2, 1)\n",
    "\n",
    "    def forward(self, x, mask, attr_embed):\n",
    "        if mask is not None:\n",
    "            x = torch.cat([x, mask], dim=1)\n",
    "        h = self.entry(x)\n",
    "        h = self.layer1(h)\n",
    "        h = self.layer2(h)\n",
    "        h = self.res1(h, attr_embed)\n",
    "        h = self.res2(h, attr_embed)\n",
    "        return self.exit(h)\n",
    "\n",
    "class RobustDecoder(nn.Module):\n",
    "    def __init__(self, latent_channels=512, embed_dim=256):\n",
    "        super().__init__()\n",
    "        self.layer1 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(latent_channels, 256, 4, 2, 1),\n",
    "            nn.BatchNorm2d(256), nn.ReLU()\n",
    "        )\n",
    "        self.res1 = AdaINResBlock(256, embed_dim)\n",
    "        self.res2 = AdaINResBlock(256, embed_dim)\n",
    "        self.layer2 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(256, 128, 4, 2, 1),\n",
    "            nn.BatchNorm2d(128), nn.ReLU()\n",
    "        )\n",
    "        self.layer3 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(128, 64, 4, 2, 1),\n",
    "            nn.BatchNorm2d(64), nn.ReLU()\n",
    "        )\n",
    "        self.exit = nn.Sequential(\n",
    "            nn.ConvTranspose2d(64, 32, 4, 2, 1),\n",
    "            nn.BatchNorm2d(32), nn.ReLU(),\n",
    "            nn.Conv2d(32, 3, 3, 1, 1),\n",
    "            nn.Tanh()\n",
    "        )\n",
    "\n",
    "    def forward(self, z, attr_embed):\n",
    "        h = self.layer1(z)\n",
    "        h = self.res1(h, attr_embed)\n",
    "        h = self.res2(h, attr_embed)\n",
    "        h = self.layer2(h)\n",
    "        h = self.layer3(h)\n",
    "        return self.exit(h)\n",
    "\n",
    "class AttributeEmbedding(nn.Module):\n",
    "    def __init__(self, attr_dim=40, embed_dim=256):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.Linear(attr_dim, 128), nn.ReLU(),\n",
    "            nn.Linear(128, embed_dim), nn.ReLU(),\n",
    "            nn.Linear(embed_dim, embed_dim)\n",
    "        )\n",
    "    def forward(self, x):\n",
    "        return self.net(x)\n",
    "\n",
    "class AttributePredictor(nn.Module):\n",
    "    def __init__(self, latent_channels, attr_dim):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.AdaptiveAvgPool2d(1), nn.Flatten(),\n",
    "            nn.Linear(latent_channels, 128), nn.ReLU(),\n",
    "            nn.Linear(128, attr_dim), nn.Sigmoid()\n",
    "        )\n",
    "    def forward(self, x):\n",
    "        return self.net(x)\n",
    "\n",
    "class SOTARobustModel(nn.Module):\n",
    "    def __init__(self):\n",
    "        super().__init__()\n",
    "        embed_dim = 256\n",
    "        self.attr_embedding = AttributeEmbedding(CONFIG['attr_dim'], embed_dim)\n",
    "        self.encoder_x = RobustEncoder(6, CONFIG['latent_dim'], CONFIG['attr_dim'], embed_dim)\n",
    "        self.encoder_s = RobustEncoder(3, CONFIG['latent_dim'], CONFIG['attr_dim'], embed_dim)\n",
    "        self.decoder = RobustDecoder(CONFIG['latent_dim'], embed_dim)\n",
    "        self.attr_predictor = AttributePredictor(CONFIG['latent_dim'], CONFIG['attr_dim'])\n",
    "        \n",
    "    def forward(self, x, s, training_dropout=False):\n",
    "        attr = s['attributes']\n",
    "        attr_embed = self.attr_embedding(attr)\n",
    "        \n",
    "        if training_dropout and np.random.random() < CONFIG['attr_dropout']:\n",
    "            used_embed = torch.zeros_like(attr_embed)\n",
    "        else:\n",
    "            used_embed = attr_embed\n",
    "            \n",
    "        z_x = self.encoder_x(x['sparse_image'], x['mask'], used_embed)\n",
    "        z_s = self.encoder_s(s['full_image'], None, used_embed)\n",
    "        \n",
    "        recon_x = self.decoder(z_x, used_embed)\n",
    "        recon_s = self.decoder(z_s, used_embed)\n",
    "        \n",
    "        pred_attr_x = self.attr_predictor(z_x)\n",
    "        \n",
    "        return z_x, z_s, recon_x, recon_s, pred_attr_x\n",
    "\n",
    "# ============================================================================\n",
    "# EVALUATION FUNCTION\n",
    "# ============================================================================\n",
    "def evaluate_reconstruction(model, dataloader, device, sparsity_level):\n",
    "    \"\"\"\n",
    "    Evaluate reconstruction quality on test set\n",
    "    \n",
    "    Returns:\n",
    "        dict with MSE and PSNR metrics\n",
    "    \"\"\"\n",
    "    model.eval()\n",
    "    mses = []\n",
    "    psnrs = []\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        for x, s in tqdm(dataloader, desc=f\"Eval ρ={sparsity_level:.2f}\", leave=False):\n",
    "            # Move to device\n",
    "            sparse = x['sparse_image'].to(device)\n",
    "            mask = x['mask'].to(device)\n",
    "            full = s['full_image'].to(device)\n",
    "            attr = s['attributes'].to(device)\n",
    "            \n",
    "            x_in = {'sparse_image': sparse, 'mask': mask}\n",
    "            s_in = {'full_image': full, 'attributes': attr}\n",
    "            \n",
    "            # Forward pass (no dropout)\n",
    "            _, _, recon_x, _, _ = model(x_in, s_in, training_dropout=False)\n",
    "            \n",
    "            # Unnormalize to [0, 1]\n",
    "            recon_01 = torch.clamp((recon_x + 1) / 2, 0, 1)\n",
    "            full_01 = torch.clamp((full + 1) / 2, 0, 1)\n",
    "            \n",
    "            # Compute MSE per sample\n",
    "            batch_mse = F.mse_loss(recon_01, full_01, reduction='none').mean(dim=[1, 2, 3])\n",
    "            \n",
    "            for mse_val in batch_mse:\n",
    "                mse_item = mse_val.item()\n",
    "                mses.append(mse_item)\n",
    "                if mse_item > 0:\n",
    "                    psnrs.append(-10 * np.log10(mse_item))\n",
    "    \n",
    "    avg_mse = np.mean(mses)\n",
    "    avg_psnr = np.mean(psnrs)\n",
    "    std_mse = np.std(mses)\n",
    "    std_psnr = np.std(psnrs)\n",
    "    \n",
    "    return {\n",
    "        'mse': avg_mse,\n",
    "        'psnr': avg_psnr,\n",
    "        'mse_std': std_mse,\n",
    "        'psnr_std': std_psnr,\n",
    "        'n_samples': len(mses)\n",
    "    }\n",
    "\n",
    "# ============================================================================\n",
    "# MULTI-SPARSITY EVALUATION\n",
    "# ============================================================================\n",
    "def evaluate_multi_sparsity(model, data_root, sparsity_levels, device, batch_size=64):\n",
    "    \"\"\"\n",
    "    Evaluate model across multiple sparsity levels\n",
    "    \n",
    "    Args:\n",
    "        model: Trained model\n",
    "        data_root: Path to CelebA data\n",
    "        sparsity_levels: List of sparsity levels to test\n",
    "        device: torch device\n",
    "        batch_size: Batch size for evaluation\n",
    "    \n",
    "    Returns:\n",
    "        dict: Results for each sparsity level\n",
    "    \"\"\"\n",
    "    print(\"=\"*80)\n",
    "    print(\"CELEBA SPARSE RECOVERY PERFORMANCE EVALUATION\")\n",
    "    print(\"=\"*80)\n",
    "    print(f\"Model: {model.__class__.__name__}\")\n",
    "    print(f\"Sparsity levels: {sparsity_levels}\")\n",
    "    print(f\"Device: {device}\")\n",
    "    print(\"=\"*80 + \"\\n\")\n",
    "    \n",
    "    results = {}\n",
    "    \n",
    "    for sparsity in sparsity_levels:\n",
    "        print(f\"\\n{'='*60}\")\n",
    "        print(f\"Evaluating at ρ = {sparsity:.2f} ({sparsity:.0%} visible pixels)\")\n",
    "        print(f\"{'='*60}\")\n",
    "        \n",
    "        # Create dataset for this sparsity level\n",
    "        test_dataset = EnhancedCelebADataset(data_root, split='test', sparsity=sparsity)\n",
    "        test_loader = DataLoader(test_dataset, batch_size=batch_size, \n",
    "                                shuffle=False, num_workers=4)\n",
    "        \n",
    "        print(f\"Test samples: {len(test_dataset)}\")\n",
    "        \n",
    "        # Evaluate\n",
    "        metrics = evaluate_reconstruction(model, test_loader, device, sparsity)\n",
    "        \n",
    "        # Store results\n",
    "        results[sparsity] = metrics\n",
    "        \n",
    "        # Print results\n",
    "        print(f\"\\nResults for ρ = {sparsity:.2f}:\")\n",
    "        print(f\"  MSE:  {metrics['mse']:.6f} ± {metrics['mse_std']:.6f}\")\n",
    "        print(f\"  PSNR: {metrics['psnr']:.2f} ± {metrics['psnr_std']:.2f} dB\")\n",
    "        print(f\"  Samples: {metrics['n_samples']}\")\n",
    "        \n",
    "        # Quality assessment\n",
    "        if metrics['mse'] <= 0.005:\n",
    "            print(f\"  ✓ Excellent quality (MSE ≤ 0.005)\")\n",
    "        elif metrics['mse'] <= 0.010:\n",
    "            print(f\"  ✓ Good quality (MSE ≤ 0.010)\")\n",
    "        elif metrics['mse'] <= 0.020:\n",
    "            print(f\"  ⚠ Acceptable quality (MSE ≤ 0.020)\")\n",
    "        else:\n",
    "            print(f\"  ✗ Poor quality (MSE > 0.020)\")\n",
    "    \n",
    "    return results\n",
    "\n",
    "# ============================================================================\n",
    "# LATEX TABLE GENERATION\n",
    "# ============================================================================\n",
    "def print_performance_table(results, model_name=\"SOTARobustModel\"):\n",
    "    \"\"\"\n",
    "    Generate LaTeX table for reconstruction performance\n",
    "    \n",
    "    Args:\n",
    "        results: Dict of results per sparsity level\n",
    "        model_name: Name of the model for caption\n",
    "    \"\"\"\n",
    "    sparsity_levels = sorted(results.keys())\n",
    "    \n",
    "    print(\"\\n\" + \"=\"*80)\n",
    "    print(\"LaTeX Table - CelebA Sparse Recovery Performance:\")\n",
    "    print(\"=\"*80)\n",
    "    \n",
    "    print(r\"\\begin{table}[!htpb]\")\n",
    "    print(r\"\\centering\")\n",
    "    print(r\"\\caption{CelebA sparse recovery performance across sparsity levels}\")\n",
    "    print(r\"\\label{tab:celeba_performance}\")\n",
    "    \n",
    "    # Determine number of columns\n",
    "    n_cols = len(sparsity_levels)\n",
    "    col_spec = \"l\" + \"c\" * n_cols\n",
    "    print(r\"\\begin{tabular}{\" + col_spec + \"}\")\n",
    "    print(r\"\\hline\")\n",
    "    \n",
    "    # Header row\n",
    "    header = r\"\\textbf{Method}\"\n",
    "    for rho in sparsity_levels:\n",
    "        header += f\" & $\\\\boldsymbol{{\\\\rho={rho:.2f}}}$\"\n",
    "    header += r\" \\\\\"\n",
    "    print(header)\n",
    "    print(r\"\\hline\")\n",
    "    \n",
    "    # MSE row\n",
    "    mse_row = f\"{model_name} (MSE)\"\n",
    "    for rho in sparsity_levels:\n",
    "        mse_val = results[rho]['mse']\n",
    "        mse_row += f\" & {mse_val:.4f}\"\n",
    "    mse_row += r\" \\\\\"\n",
    "    print(mse_row)\n",
    "    \n",
    "    # PSNR row\n",
    "    psnr_row = f\"{model_name} (PSNR dB)\"\n",
    "    for rho in sparsity_levels:\n",
    "        psnr_val = results[rho]['psnr']\n",
    "        psnr_row += f\" & {psnr_val:.2f}\"\n",
    "    psnr_row += r\" \\\\\"\n",
    "    print(psnr_row)\n",
    "    \n",
    "    print(r\"\\hline\")\n",
    "    print(r\"\\end{tabular}\")\n",
    "    print(r\"\\end{table}\")\n",
    "    print(\"=\"*80 + \"\\n\")\n",
    "\n",
    "def print_performance_table_combined(all_results):\n",
    "    \"\"\"\n",
    "    Generate LaTeX table with multiple models\n",
    "    \n",
    "    Args:\n",
    "        all_results: Dict of {model_name: {sparsity: metrics}}\n",
    "    \"\"\"\n",
    "    # Get all sparsity levels (assuming all models tested on same levels)\n",
    "    first_model = list(all_results.keys())[0]\n",
    "    sparsity_levels = sorted(all_results[first_model].keys())\n",
    "    \n",
    "    print(\"\\n\" + \"=\"*80)\n",
    "    print(\"LaTeX Table - Multi-Model Comparison:\")\n",
    "    print(\"=\"*80)\n",
    "    \n",
    "    print(r\"\\begin{table}[!htpb]\")\n",
    "    print(r\"\\centering\")\n",
    "    print(r\"\\caption{CelebA sparse recovery performance across sparsity levels}\")\n",
    "    print(r\"\\label{tab:celeba_performance_comparison}\")\n",
    "    \n",
    "    n_cols = len(sparsity_levels)\n",
    "    col_spec = \"l\" + \"c\" * n_cols\n",
    "    print(r\"\\begin{tabular}{\" + col_spec + \"}\")\n",
    "    print(r\"\\hline\")\n",
    "    \n",
    "    # Header\n",
    "    header = r\"\\textbf{Method}\"\n",
    "    for rho in sparsity_levels:\n",
    "        header += f\" & $\\\\boldsymbol{{\\\\rho={rho:.2f}}}$\"\n",
    "    header += r\" \\\\\"\n",
    "    print(header)\n",
    "    print(r\"\\hline\")\n",
    "    \n",
    "    # For each model, print MSE and PSNR rows\n",
    "    for model_name in all_results.keys():\n",
    "        results = all_results[model_name]\n",
    "        \n",
    "        # MSE row\n",
    "        mse_row = f\"{model_name} (MSE)\"\n",
    "        for rho in sparsity_levels:\n",
    "            mse_val = results[rho]['mse']\n",
    "            mse_row += f\" & {mse_val:.4f}\"\n",
    "        mse_row += r\" \\\\\"\n",
    "        print(mse_row)\n",
    "        \n",
    "        # PSNR row\n",
    "        psnr_row = f\"{model_name} (PSNR dB)\"\n",
    "        for rho in sparsity_levels:\n",
    "            psnr_val = results[rho]['psnr']\n",
    "            psnr_row += f\" & {psnr_val:.2f}\"\n",
    "        psnr_row += r\" \\\\\"\n",
    "        print(psnr_row)\n",
    "        \n",
    "        print(r\"\\hline\")\n",
    "    \n",
    "    print(r\"\\end{tabular}\")\n",
    "    print(r\"\\end{table}\")\n",
    "    print(\"=\"*80 + \"\\n\")\n",
    "\n",
    "# ============================================================================\n",
    "# MAIN EXECUTION\n",
    "# ============================================================================\n",
    "if __name__ == '__main__':\n",
    "    device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "    \n",
    "    print(\"=\"*80)\n",
    "    print(\"CELEBA SPARSE RECOVERY EVALUATION\")\n",
    "    print(\"=\"*80 + \"\\n\")\n",
    "    \n",
    "    # Define sparsity levels to test\n",
    "    sparsity_levels = [0.05, 0.08, 0.10, 0.12, 0.15]\n",
    "    \n",
    "    # Initialize models dictionary\n",
    "    models = {}\n",
    "    \n",
    "    # Load V4-Robust (your main model)\n",
    "    print(\"Loading models...\")\n",
    "    model = SOTARobustModel()\n",
    "    \n",
    "    if os.path.exists('sota_v3_best.pth'):\n",
    "        model.load_state_dict(torch.load('sota_v3_best.pth', map_location='cpu'))\n",
    "        print(\"✓ Loaded: sota_v3_best.pth\")\n",
    "        models['V4-Robust'] = model.to(device)\n",
    "    else:\n",
    "        print(\"✗ ERROR: sota_v3_best.pth not found!\")\n",
    "        print(\"Please ensure the trained model exists.\")\n",
    "        exit(1)\n",
    "    \n",
    "    # Optional: Load additional models for comparison\n",
    "    # Uncomment if you have other models to compare\n",
    "    \"\"\"\n",
    "    if os.path.exists('sota_resunet.pth'):\n",
    "        model_v3 = SOTARobustModel()  # Or appropriate model class\n",
    "        model_v3.load_state_dict(torch.load('sota_resunet.pth', map_location='cpu'))\n",
    "        print(\"✓ Loaded: sota_resunet.pth\")\n",
    "        models['V3-ResUNet'] = model_v3.to(device)\n",
    "    \"\"\"\n",
    "    \n",
    "    print(f\"\\nModels loaded: {list(models.keys())}\")\n",
    "    print(f\"Device: {device}\\n\")\n",
    "    \n",
    "    # Evaluate each model\n",
    "    all_results = {}\n",
    "    \n",
    "    for model_name, model_obj in models.items():\n",
    "        print(f\"\\n{'='*80}\")\n",
    "        print(f\"EVALUATING: {model_name}\")\n",
    "        print(f\"{'='*80}\")\n",
    "        \n",
    "        results = evaluate_multi_sparsity(\n",
    "            model=model_obj,\n",
    "            data_root='./data',\n",
    "            sparsity_levels=sparsity_levels,\n",
    "            device=device,\n",
    "            batch_size=64\n",
    "        )\n",
    "        \n",
    "        all_results[model_name] = results\n",
    "    \n",
    "    # Generate LaTeX tables\n",
    "    if len(models) == 1:\n",
    "        # Single model table\n",
    "        model_name = list(models.keys())[0]\n",
    "        print_performance_table(all_results[model_name], model_name)\n",
    "    else:\n",
    "        # Multi-model comparison table\n",
    "        print_performance_table_combined(all_results)\n",
    "    \n",
    "    # Save results to JSON\n",
    "    output_file = 'celeba_performance_results.json'\n",
    "    with open(output_file, 'w') as f:\n",
    "        json.dump(all_results, f, indent=2)\n",
    "    print(f\"✓ Results saved to '{output_file}'\")\n",
    "    \n",
    "    # Print summary\n",
    "    print(\"\\n\" + \"=\"*80)\n",
    "    print(\"EVALUATION COMPLETE!\")\n",
    "    print(\"=\"*80)\n",
    "    \n",
    "    for model_name in all_results.keys():\n",
    "        print(f\"\\n{model_name} Summary:\")\n",
    "        results = all_results[model_name]\n",
    "        print(f\"{'Sparsity':<12} {'MSE':<12} {'PSNR (dB)':<12}\")\n",
    "        print(\"-\"*40)\n",
    "        for rho in sorted(results.keys()):\n",
    "            mse = results[rho]['mse']\n",
    "            psnr = results[rho]['psnr']\n",
    "            print(f\"ρ={rho:<10.2f} {mse:<12.6f} {psnr:<12.2f}\")\n",
    "    \n",
    "    print(\"\\n\" + \"=\"*80)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "4447a32a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "================================================================================\n",
      "CELEBA SPARSE RECOVERY WITH GAUSSIAN NOISE EVALUATION\n",
      "Noise Level: σ = 0.6\n",
      "================================================================================\n",
      "\n",
      "Loading models...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/to247392/anaconda3/envs/manitorch/lib/python3.11/site-packages/torch/cuda/__init__.py:789: UserWarning: Can't initialize NVML\n",
      "  warnings.warn(\"Can't initialize NVML\")\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "✓ Loaded: sota_resunet.pth (V3-ResUNet)\n",
      "✓ Loaded: sota_v3_best.pth (V4-Robust)\n",
      "\n",
      "Models loaded: ['V3-ResUNet', 'V4-Robust']\n",
      "Device: cuda\n",
      "Gaussian Noise Level: σ = 0.6\n",
      "\n",
      "\n",
      "================================================================================\n",
      "EVALUATING: V3-ResUNet\n",
      "Gaussian Noise Level: σ = 0.6\n",
      "================================================================================\n",
      "\n",
      "============================================================\n",
      "Sparsity: ρ = 0.05 (5% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                       \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results:\n",
      "  MSE:  0.036007 ± 0.004558\n",
      "  PSNR: 14.47 ± 0.56 dB\n",
      "  ✗ Poor quality (MSE > 0.030)\n",
      "\n",
      "============================================================\n",
      "Sparsity: ρ = 0.08 (8% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                       \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results:\n",
      "  MSE:  0.034188 ± 0.003723\n",
      "  PSNR: 14.69 ± 0.48 dB\n",
      "  ✗ Poor quality (MSE > 0.030)\n",
      "\n",
      "============================================================\n",
      "Sparsity: ρ = 0.10 (10% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                       \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results:\n",
      "  MSE:  0.032958 ± 0.003339\n",
      "  PSNR: 14.84 ± 0.44 dB\n",
      "  ✗ Poor quality (MSE > 0.030)\n",
      "\n",
      "============================================================\n",
      "Sparsity: ρ = 0.12 (12% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                       \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results:\n",
      "  MSE:  0.031681 ± 0.003056\n",
      "  PSNR: 15.01 ± 0.42 dB\n",
      "  ✗ Poor quality (MSE > 0.030)\n",
      "\n",
      "============================================================\n",
      "Sparsity: ρ = 0.15 (15% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                       \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results:\n",
      "  MSE:  0.029747 ± 0.002669\n",
      "  PSNR: 15.28 ± 0.39 dB\n",
      "  ⚠ Degraded quality (MSE ≤ 0.030)\n",
      "\n",
      "================================================================================\n",
      "EVALUATING: V4-Robust\n",
      "Gaussian Noise Level: σ = 0.6\n",
      "================================================================================\n",
      "\n",
      "============================================================\n",
      "Sparsity: ρ = 0.05 (5% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                       \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results:\n",
      "  MSE:  0.035238 ± 0.004294\n",
      "  PSNR: 14.56 ± 0.53 dB\n",
      "  ✗ Poor quality (MSE > 0.030)\n",
      "\n",
      "============================================================\n",
      "Sparsity: ρ = 0.08 (8% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                       \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results:\n",
      "  MSE:  0.030503 ± 0.003250\n",
      "  PSNR: 15.18 ± 0.47 dB\n",
      "  ✗ Poor quality (MSE > 0.030)\n",
      "\n",
      "============================================================\n",
      "Sparsity: ρ = 0.10 (10% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                       \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results:\n",
      "  MSE:  0.028065 ± 0.002827\n",
      "  PSNR: 15.54 ± 0.44 dB\n",
      "  ⚠ Degraded quality (MSE ≤ 0.030)\n",
      "\n",
      "============================================================\n",
      "Sparsity: ρ = 0.12 (12% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                       \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results:\n",
      "  MSE:  0.026233 ± 0.002592\n",
      "  PSNR: 15.83 ± 0.43 dB\n",
      "  ⚠ Degraded quality (MSE ≤ 0.030)\n",
      "\n",
      "============================================================\n",
      "Sparsity: ρ = 0.15 (15% visible pixels)\n",
      "============================================================\n",
      "Test samples: 19962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                       "
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Results:\n",
      "  MSE:  0.024711 ± 0.002520\n",
      "  PSNR: 16.09 ± 0.45 dB\n",
      "  ⚠ Degraded quality (MSE ≤ 0.030)\n",
      "\n",
      "================================================================================\n",
      "LaTeX Table - Noisy Reconstruction (σ=0.6):\n",
      "================================================================================\n",
      "\\begin{table}[!htpb]\n",
      "\\centering\n",
      "\\caption{CelebA sparse recovery with Gaussian noise ($\\sigma=0.6$)}\n",
      "\\label{tab:celeba_noisy_performance}\n",
      "\\begin{tabular}{lccccc}\n",
      "\\hline\n",
      "\\textbf{Method} & $\\boldsymbol{\\rho=0.05}$ & $\\boldsymbol{\\rho=0.08}$ & $\\boldsymbol{\\rho=0.10}$ & $\\boldsymbol{\\rho=0.12}$ & $\\boldsymbol{\\rho=0.15}$ \\\\\n",
      "\\hline\n",
      "V3-ResUNet (MSE) & 0.0360 & 0.0342 & 0.0330 & 0.0317 & 0.0297 \\\\\n",
      "V3-ResUNet (PSNR dB) & 14.47 & 14.69 & 14.84 & 15.01 & 15.28 \\\\\n",
      "\\hline\n",
      "V4-Robust (MSE) & 0.0352 & 0.0305 & 0.0281 & 0.0262 & 0.0247 \\\\\n",
      "V4-Robust (PSNR dB) & 14.56 & 15.18 & 15.54 & 15.83 & 16.09 \\\\\n",
      "\\hline\n",
      "\\end{tabular}\n",
      "\\end{table}\n",
      "================================================================================\n",
      "\n",
      "✓ Results saved to 'celeba_noisy_performance_sigma0.6.json'\n",
      "\n",
      "================================================================================\n",
      "EVALUATION COMPLETE!\n",
      "Gaussian Noise: σ = 0.6\n",
      "================================================================================\n",
      "\n",
      "V3-ResUNet Summary:\n",
      "Sparsity     MSE          PSNR (dB)   \n",
      "----------------------------------------\n",
      "ρ=0.05       0.036007     14.47       \n",
      "ρ=0.08       0.034188     14.69       \n",
      "ρ=0.10       0.032958     14.84       \n",
      "ρ=0.12       0.031681     15.01       \n",
      "ρ=0.15       0.029747     15.28       \n",
      "\n",
      "V4-Robust Summary:\n",
      "Sparsity     MSE          PSNR (dB)   \n",
      "----------------------------------------\n",
      "ρ=0.05       0.035238     14.56       \n",
      "ρ=0.08       0.030503     15.18       \n",
      "ρ=0.10       0.028065     15.54       \n",
      "ρ=0.12       0.026233     15.83       \n",
      "ρ=0.15       0.024711     16.09       \n",
      "\n",
      "================================================================================\n",
      "ROBUSTNESS IMPROVEMENT (V4 vs V3):\n",
      "================================================================================\n",
      "Sparsity     V3 MSE       V4 MSE       Improvement \n",
      "--------------------------------------------------\n",
      "ρ=0.05       0.036007     0.035238           2.13%\n",
      "ρ=0.08       0.034188     0.030503          10.78%\n",
      "ρ=0.10       0.032958     0.028065          14.85%\n",
      "ρ=0.12       0.031681     0.026233          17.20%\n",
      "ρ=0.15       0.029747     0.024711          16.93%\n",
      "\n",
      "================================================================================\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r"
     ]
    }
   ],
   "source": [
    "# ============================================================================\n",
    "# CELEBA SPARSE RECOVERY WITH NOISE ROBUSTNESS EVALUATION\n",
    "# Tests reconstruction quality across multiple sparsity levels with Gaussian noise\n",
    "# Compares ResUNet (V3) vs Robust Model (V4) under noise σ=0.6\n",
    "# Generates LaTeX table with MSE and PSNR metrics\n",
    "# ============================================================================\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torch.utils.data import DataLoader\n",
    "from torchvision import datasets, transforms\n",
    "import numpy as np\n",
    "from tqdm import tqdm\n",
    "import json\n",
    "import os\n",
    "\n",
    "# ============================================================================\n",
    "# CONFIGURATION\n",
    "# ============================================================================\n",
    "CONFIG = {\n",
    "    'img_size': 64,\n",
    "    'batch_size': 64,\n",
    "    'latent_dim': 1024,\n",
    "    'attr_dim': 40,\n",
    "    'attr_dropout': 0.5,\n",
    "    'noise_level': 0.6,  # Gaussian noise level\n",
    "    'device': 'cuda' if torch.cuda.is_available() else 'cpu'\n",
    "}\n",
    "\n",
    "# ============================================================================\n",
    "# DATASET\n",
    "# ============================================================================\n",
    "class EnhancedCelebADataset(torch.utils.data.Dataset):\n",
    "    def __init__(self, root, split='test', sparsity=0.05, img_size=64):\n",
    "        self.transform = transforms.Compose([\n",
    "            transforms.CenterCrop(178),\n",
    "            transforms.Resize(img_size),\n",
    "            transforms.ToTensor(),\n",
    "            transforms.Normalize((0.5, 0.5, 0.5), (0.5, 0.5, 0.5))\n",
    "        ])\n",
    "        try:\n",
    "            self.data = datasets.CelebA(root=root, split=split, download=False, transform=self.transform)\n",
    "        except:\n",
    "            self.data = datasets.CelebA(root=root, split=split, download=True, transform=self.transform)\n",
    "        self.sparsity = sparsity\n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self.data)\n",
    "\n",
    "    def __getitem__(self, idx):\n",
    "        full_image, attributes = self.data[idx]\n",
    "        mask = (torch.rand_like(full_image) < self.sparsity).float()\n",
    "        sparse_image = full_image * mask\n",
    "        attributes = attributes.float()\n",
    "        return sparse_image, mask, full_image, attributes\n",
    "\n",
    "# ============================================================================\n",
    "# NOISE INJECTION\n",
    "# ============================================================================\n",
    "def add_gaussian_noise(sparse_image, mask, noise_level=0.6):\n",
    "    \"\"\"Add Gaussian noise to sparse pixels only\"\"\"\n",
    "    noisy_sparse = sparse_image.clone()\n",
    "    noise = torch.randn_like(sparse_image) * noise_level\n",
    "    noisy_sparse = sparse_image + noise * mask\n",
    "    noisy_sparse = torch.clamp(noisy_sparse, -1, 1)\n",
    "    return noisy_sparse\n",
    "\n",
    "# ============================================================================\n",
    "# MODEL COMPONENTS (V3 - RESUNET)\n",
    "# ============================================================================\n",
    "class ResBlock(nn.Module):\n",
    "    def __init__(self, c):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.BatchNorm2d(c), nn.ReLU(), nn.Conv2d(c, c, 3, 1, 1, bias=False),\n",
    "            nn.BatchNorm2d(c), nn.ReLU(), nn.Conv2d(c, c, 3, 1, 1, bias=False)\n",
    "        )\n",
    "    def forward(self, x): return x + self.net(x)\n",
    "\n",
    "class ResUNet(nn.Module):\n",
    "    def __init__(self, in_ch=6, out_ch=3):\n",
    "        super().__init__()\n",
    "        self.enc1 = nn.Sequential(nn.Conv2d(in_ch, 64, 4, 2, 1), nn.ReLU()) \n",
    "        self.enc2 = nn.Sequential(nn.Conv2d(64, 128, 4, 2, 1), nn.BatchNorm2d(128), nn.ReLU()) \n",
    "        self.enc3 = nn.Sequential(nn.Conv2d(128, 256, 4, 2, 1), nn.BatchNorm2d(256), nn.ReLU()) \n",
    "        self.enc4 = nn.Sequential(nn.Conv2d(256, 512, 4, 2, 1), nn.BatchNorm2d(512), nn.ReLU()) \n",
    "        \n",
    "        self.bottleneck = nn.Sequential(\n",
    "            ResBlock(512), ResBlock(512), ResBlock(512), ResBlock(512)\n",
    "        )\n",
    "        \n",
    "        self.dec4 = nn.Sequential(nn.ConvTranspose2d(512, 256, 4, 2, 1), nn.BatchNorm2d(256), nn.ReLU())\n",
    "        self.dec3 = nn.Sequential(nn.ConvTranspose2d(512, 128, 4, 2, 1), nn.BatchNorm2d(128), nn.ReLU()) \n",
    "        self.dec2 = nn.Sequential(nn.ConvTranspose2d(256, 64, 4, 2, 1), nn.BatchNorm2d(64), nn.ReLU()) \n",
    "        self.dec1 = nn.Sequential(nn.ConvTranspose2d(128, 64, 4, 2, 1), nn.BatchNorm2d(64), nn.ReLU()) \n",
    "        \n",
    "        self.final = nn.Sequential(nn.Conv2d(64, out_ch, 3, 1, 1), nn.Tanh())\n",
    "\n",
    "    def forward(self, x, mask):\n",
    "        inp = torch.cat([x, mask], dim=1)\n",
    "        e1 = self.enc1(inp)\n",
    "        e2 = self.enc2(e1)\n",
    "        e3 = self.enc3(e2)\n",
    "        e4 = self.enc4(e3)\n",
    "        b = self.bottleneck(e4)\n",
    "        d4 = self.dec4(b)\n",
    "        d3 = self.dec3(torch.cat([d4, e3], 1))\n",
    "        d2 = self.dec2(torch.cat([d3, e2], 1))\n",
    "        d1 = self.dec1(torch.cat([d2, e1], 1))\n",
    "        return self.final(d1)\n",
    "    \n",
    "    def infer(self, sparse, mask, attributes=None):\n",
    "        \"\"\"Standardized inference method\"\"\"\n",
    "        return self.forward(sparse, mask)\n",
    "\n",
    "# ============================================================================\n",
    "# MODEL COMPONENTS (V4 - ROBUST MODEL)\n",
    "# ============================================================================\n",
    "class AdaIN(nn.Module):\n",
    "    def __init__(self, channels, embed_dim):\n",
    "        super().__init__()\n",
    "        self.channels = channels\n",
    "        self.linear = nn.Linear(embed_dim, channels * 2)\n",
    "\n",
    "    def forward(self, x, attr_embed):\n",
    "        style = self.linear(attr_embed)\n",
    "        style = style.view(-1, 2, self.channels, 1, 1)\n",
    "        gamma, beta = style[:, 0], style[:, 1]\n",
    "        mean = x.mean(dim=[2, 3], keepdim=True)\n",
    "        std = x.std(dim=[2, 3], keepdim=True) + 1e-8\n",
    "        norm_x = (x - mean) / std\n",
    "        return gamma * norm_x + beta\n",
    "\n",
    "class AdaINResBlock(nn.Module):\n",
    "    def __init__(self, c, embed_dim):\n",
    "        super().__init__()\n",
    "        self.conv1 = nn.Conv2d(c, c, 3, 1, 1, bias=False)\n",
    "        self.conv2 = nn.Conv2d(c, c, 3, 1, 1, bias=False)\n",
    "        self.adain1 = AdaIN(c, embed_dim)\n",
    "        self.adain2 = AdaIN(c, embed_dim)\n",
    "        self.act = nn.ReLU()\n",
    "        \n",
    "    def forward(self, x, attr_embed):\n",
    "        residual = x\n",
    "        out = self.conv1(x)\n",
    "        out = self.act(self.adain1(out, attr_embed))\n",
    "        out = self.conv2(out)\n",
    "        out = self.adain2(out, attr_embed)\n",
    "        return self.act(out + residual)\n",
    "\n",
    "class RobustEncoder(nn.Module):\n",
    "    def __init__(self, input_channels=6, latent_channels=1024, attr_dim=40, embed_dim=256):\n",
    "        super().__init__()\n",
    "        self.entry = nn.Sequential(nn.Conv2d(input_channels, 64, 4, 2, 1), nn.ReLU())\n",
    "        self.layer1 = nn.Sequential(nn.Conv2d(64, 128, 4, 2, 1), nn.BatchNorm2d(128), nn.ReLU())\n",
    "        self.layer2 = nn.Sequential(nn.Conv2d(128, 256, 4, 2, 1), nn.BatchNorm2d(256), nn.ReLU())\n",
    "        self.res1 = AdaINResBlock(256, embed_dim)\n",
    "        self.res2 = AdaINResBlock(256, embed_dim)\n",
    "        self.exit = nn.Conv2d(256, latent_channels, 4, 2, 1)\n",
    "\n",
    "    def forward(self, x, mask, attr_embed):\n",
    "        if mask is not None:\n",
    "            x = torch.cat([x, mask], dim=1)\n",
    "        h = self.entry(x)\n",
    "        h = self.layer1(h)\n",
    "        h = self.layer2(h)\n",
    "        h = self.res1(h, attr_embed)\n",
    "        h = self.res2(h, attr_embed)\n",
    "        return self.exit(h)\n",
    "\n",
    "class RobustDecoder(nn.Module):\n",
    "    def __init__(self, latent_channels=1024, embed_dim=256):\n",
    "        super().__init__()\n",
    "        self.layer1 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(latent_channels, 256, 4, 2, 1),\n",
    "            nn.BatchNorm2d(256), nn.ReLU()\n",
    "        )\n",
    "        self.res1 = AdaINResBlock(256, embed_dim)\n",
    "        self.res2 = AdaINResBlock(256, embed_dim)\n",
    "        self.layer2 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(256, 128, 4, 2, 1),\n",
    "            nn.BatchNorm2d(128), nn.ReLU()\n",
    "        )\n",
    "        self.layer3 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(128, 64, 4, 2, 1),\n",
    "            nn.BatchNorm2d(64), nn.ReLU()\n",
    "        )\n",
    "        self.exit = nn.Sequential(\n",
    "            nn.ConvTranspose2d(64, 32, 4, 2, 1),\n",
    "            nn.BatchNorm2d(32), nn.ReLU(),\n",
    "            nn.Conv2d(32, 3, 3, 1, 1),\n",
    "            nn.Tanh()\n",
    "        )\n",
    "\n",
    "    def forward(self, z, attr_embed):\n",
    "        h = self.layer1(z)\n",
    "        h = self.res1(h, attr_embed)\n",
    "        h = self.res2(h, attr_embed)\n",
    "        h = self.layer2(h)\n",
    "        h = self.layer3(h)\n",
    "        return self.exit(h)\n",
    "\n",
    "class AttributeEmbedding(nn.Module):\n",
    "    def __init__(self, attr_dim=40, embed_dim=256):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.Linear(attr_dim, 128), nn.ReLU(),\n",
    "            nn.Linear(128, embed_dim), nn.ReLU(),\n",
    "            nn.Linear(embed_dim, embed_dim)\n",
    "        )\n",
    "    def forward(self, x):\n",
    "        return self.net(x)\n",
    "\n",
    "class AttributePredictor(nn.Module):\n",
    "    def __init__(self, latent_channels, attr_dim):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.AdaptiveAvgPool2d(1), nn.Flatten(),\n",
    "            nn.Linear(latent_channels, 128), nn.ReLU(),\n",
    "            nn.Linear(128, attr_dim), nn.Sigmoid()\n",
    "        )\n",
    "    def forward(self, x):\n",
    "        return self.net(x)\n",
    "\n",
    "class SOTARobustModel(nn.Module):\n",
    "    def __init__(self, latent_dim=1024, attr_dim=40):\n",
    "        super().__init__()\n",
    "        embed_dim = 256\n",
    "        self.attr_embedding = AttributeEmbedding(attr_dim, embed_dim)\n",
    "        self.encoder_x = RobustEncoder(6, latent_dim, attr_dim, embed_dim)\n",
    "        self.encoder_s = RobustEncoder(3, latent_dim, attr_dim, embed_dim)\n",
    "        self.decoder = RobustDecoder(latent_dim, embed_dim)\n",
    "        self.attr_predictor = AttributePredictor(latent_dim, attr_dim)\n",
    "        \n",
    "    def forward(self, x, s, training_dropout=False):\n",
    "        attr = s['attributes']\n",
    "        attr_embed = self.attr_embedding(attr)\n",
    "        \n",
    "        if training_dropout and np.random.random() < CONFIG['attr_dropout']:\n",
    "            used_embed = torch.zeros_like(attr_embed)\n",
    "        else:\n",
    "            used_embed = attr_embed\n",
    "            \n",
    "        z_x = self.encoder_x(x['sparse_image'], x['mask'], used_embed)\n",
    "        z_s = self.encoder_s(s['full_image'], None, used_embed)\n",
    "        \n",
    "        recon_x = self.decoder(z_x, used_embed)\n",
    "        recon_s = self.decoder(z_s, used_embed)\n",
    "        \n",
    "        pred_attr_x = self.attr_predictor(z_x)\n",
    "        \n",
    "        return z_x, z_s, recon_x, recon_s, pred_attr_x\n",
    "    \n",
    "    def infer(self, sparse, mask, attributes=None):\n",
    "        \"\"\"Standardized inference method\"\"\"\n",
    "        self.eval()\n",
    "        with torch.no_grad():\n",
    "            if attributes is None:\n",
    "                attr_embed = torch.zeros(sparse.size(0), 256, device=sparse.device)\n",
    "            else:\n",
    "                attr_embed = self.attr_embedding(attributes)\n",
    "            \n",
    "            z_x = self.encoder_x(sparse, mask, attr_embed)\n",
    "            recon = self.decoder(z_x, attr_embed)\n",
    "            return recon\n",
    "\n",
    "# ============================================================================\n",
    "# EVALUATION FUNCTION WITH NOISE\n",
    "# ============================================================================\n",
    "def evaluate_reconstruction_with_noise(model, dataloader, device, sparsity_level, noise_level):\n",
    "    \"\"\"\n",
    "    Evaluate reconstruction quality on test set with Gaussian noise\n",
    "    \n",
    "    Args:\n",
    "        model: Model to evaluate\n",
    "        dataloader: DataLoader for test set\n",
    "        device: torch device\n",
    "        sparsity_level: Sparsity level (ρ)\n",
    "        noise_level: Gaussian noise level (σ)\n",
    "    \n",
    "    Returns:\n",
    "        dict with MSE and PSNR metrics\n",
    "    \"\"\"\n",
    "    model.eval()\n",
    "    mses = []\n",
    "    psnrs = []\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        for sparse, mask, full, attributes in tqdm(dataloader, \n",
    "                                                    desc=f\"Eval ρ={sparsity_level:.2f}, σ={noise_level:.2f}\", \n",
    "                                                    leave=False):\n",
    "            # Move to device\n",
    "            sparse = sparse.to(device)\n",
    "            mask = mask.to(device)\n",
    "            full = full.to(device)\n",
    "            attributes = attributes.to(device)\n",
    "            \n",
    "            # Add Gaussian noise\n",
    "            noisy_sparse = add_gaussian_noise(sparse, mask, noise_level)\n",
    "            \n",
    "            # Forward pass\n",
    "            recon = model.infer(noisy_sparse, mask, attributes=attributes)\n",
    "            \n",
    "            # Unnormalize to [0, 1]\n",
    "            recon_01 = torch.clamp((recon + 1) / 2, 0, 1)\n",
    "            full_01 = torch.clamp((full + 1) / 2, 0, 1)\n",
    "            \n",
    "            # Compute MSE per sample\n",
    "            batch_mse = F.mse_loss(recon_01, full_01, reduction='none').mean(dim=[1, 2, 3])\n",
    "            \n",
    "            for mse_val in batch_mse:\n",
    "                mse_item = mse_val.item()\n",
    "                mses.append(mse_item)\n",
    "                if mse_item > 0:\n",
    "                    psnrs.append(-10 * np.log10(mse_item))\n",
    "    \n",
    "    avg_mse = np.mean(mses)\n",
    "    avg_psnr = np.mean(psnrs)\n",
    "    std_mse = np.std(mses)\n",
    "    std_psnr = np.std(psnrs)\n",
    "    \n",
    "    return {\n",
    "        'mse': avg_mse,\n",
    "        'psnr': avg_psnr,\n",
    "        'mse_std': std_mse,\n",
    "        'psnr_std': std_psnr,\n",
    "        'n_samples': len(mses)\n",
    "    }\n",
    "\n",
    "# ============================================================================\n",
    "# MULTI-SPARSITY EVALUATION\n",
    "# ============================================================================\n",
    "def evaluate_multi_sparsity_with_noise(model, model_name, data_root, sparsity_levels, \n",
    "                                       noise_level, device, batch_size=64):\n",
    "    \"\"\"\n",
    "    Evaluate model across multiple sparsity levels with fixed noise level\n",
    "    \n",
    "    Args:\n",
    "        model: Trained model\n",
    "        model_name: Name of the model\n",
    "        data_root: Path to CelebA data\n",
    "        sparsity_levels: List of sparsity levels to test\n",
    "        noise_level: Fixed Gaussian noise level\n",
    "        device: torch device\n",
    "        batch_size: Batch size for evaluation\n",
    "    \n",
    "    Returns:\n",
    "        dict: Results for each sparsity level\n",
    "    \"\"\"\n",
    "    print(f\"\\n{'='*80}\")\n",
    "    print(f\"EVALUATING: {model_name}\")\n",
    "    print(f\"Gaussian Noise Level: σ = {noise_level}\")\n",
    "    print(f\"{'='*80}\")\n",
    "    \n",
    "    results = {}\n",
    "    \n",
    "    for sparsity in sparsity_levels:\n",
    "        print(f\"\\n{'='*60}\")\n",
    "        print(f\"Sparsity: ρ = {sparsity:.2f} ({sparsity:.0%} visible pixels)\")\n",
    "        print(f\"{'='*60}\")\n",
    "        \n",
    "        # Create dataset for this sparsity level\n",
    "        test_dataset = EnhancedCelebADataset(data_root, split='test', sparsity=sparsity)\n",
    "        test_loader = DataLoader(test_dataset, batch_size=batch_size, \n",
    "                                shuffle=False, num_workers=4)\n",
    "        \n",
    "        print(f\"Test samples: {len(test_dataset)}\")\n",
    "        \n",
    "        # Evaluate\n",
    "        metrics = evaluate_reconstruction_with_noise(model, test_loader, device, \n",
    "                                                     sparsity, noise_level)\n",
    "        \n",
    "        # Store results\n",
    "        results[sparsity] = metrics\n",
    "        \n",
    "        # Print results\n",
    "        print(f\"\\nResults:\")\n",
    "        print(f\"  MSE:  {metrics['mse']:.6f} ± {metrics['mse_std']:.6f}\")\n",
    "        print(f\"  PSNR: {metrics['psnr']:.2f} ± {metrics['psnr_std']:.2f} dB\")\n",
    "        \n",
    "        # Quality assessment\n",
    "        if metrics['mse'] <= 0.010:\n",
    "            print(f\"  ✓ Good quality (MSE ≤ 0.010)\")\n",
    "        elif metrics['mse'] <= 0.020:\n",
    "            print(f\"  ⚠ Acceptable quality (MSE ≤ 0.020)\")\n",
    "        elif metrics['mse'] <= 0.030:\n",
    "            print(f\"  ⚠ Degraded quality (MSE ≤ 0.030)\")\n",
    "        else:\n",
    "            print(f\"  ✗ Poor quality (MSE > 0.030)\")\n",
    "    \n",
    "    return results\n",
    "\n",
    "# ============================================================================\n",
    "# LATEX TABLE GENERATION\n",
    "# ============================================================================\n",
    "def print_noisy_performance_table(all_results, noise_level):\n",
    "    \"\"\"\n",
    "    Generate LaTeX table for reconstruction performance with noise\n",
    "    \n",
    "    Args:\n",
    "        all_results: Dict of {model_name: {sparsity: metrics}}\n",
    "        noise_level: The noise level used\n",
    "    \"\"\"\n",
    "    # Get all sparsity levels\n",
    "    first_model = list(all_results.keys())[0]\n",
    "    sparsity_levels = sorted(all_results[first_model].keys())\n",
    "    \n",
    "    print(\"\\n\" + \"=\"*80)\n",
    "    print(f\"LaTeX Table - Noisy Reconstruction (σ={noise_level}):\")\n",
    "    print(\"=\"*80)\n",
    "    \n",
    "    print(r\"\\begin{table}[!htpb]\")\n",
    "    print(r\"\\centering\")\n",
    "    print(f\"\\\\caption{{CelebA sparse recovery with Gaussian noise ($\\\\sigma={noise_level}$)}}\")\n",
    "    print(r\"\\label{tab:celeba_noisy_performance}\")\n",
    "    \n",
    "    n_cols = len(sparsity_levels)\n",
    "    col_spec = \"l\" + \"c\" * n_cols\n",
    "    print(r\"\\begin{tabular}{\" + col_spec + \"}\")\n",
    "    print(r\"\\hline\")\n",
    "    \n",
    "    # Header\n",
    "    header = r\"\\textbf{Method}\"\n",
    "    for rho in sparsity_levels:\n",
    "        header += f\" & $\\\\boldsymbol{{\\\\rho={rho:.2f}}}$\"\n",
    "    header += r\" \\\\\"\n",
    "    print(header)\n",
    "    print(r\"\\hline\")\n",
    "    \n",
    "    # For each model, print MSE and PSNR rows\n",
    "    for model_name in all_results.keys():\n",
    "        results = all_results[model_name]\n",
    "        \n",
    "        # MSE row\n",
    "        mse_row = f\"{model_name} (MSE)\"\n",
    "        for rho in sparsity_levels:\n",
    "            mse_val = results[rho]['mse']\n",
    "            mse_row += f\" & {mse_val:.4f}\"\n",
    "        mse_row += r\" \\\\\"\n",
    "        print(mse_row)\n",
    "        \n",
    "        # PSNR row\n",
    "        psnr_row = f\"{model_name} (PSNR dB)\"\n",
    "        for rho in sparsity_levels:\n",
    "            psnr_val = results[rho]['psnr']\n",
    "            psnr_row += f\" & {psnr_val:.2f}\"\n",
    "        psnr_row += r\" \\\\\"\n",
    "        print(psnr_row)\n",
    "        \n",
    "        print(r\"\\hline\")\n",
    "    \n",
    "    print(r\"\\end{tabular}\")\n",
    "    print(r\"\\end{table}\")\n",
    "    print(\"=\"*80 + \"\\n\")\n",
    "\n",
    "# ============================================================================\n",
    "# MAIN EXECUTION\n",
    "# ============================================================================\n",
    "if __name__ == '__main__':\n",
    "    device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "    \n",
    "    print(\"=\"*80)\n",
    "    print(\"CELEBA SPARSE RECOVERY WITH GAUSSIAN NOISE EVALUATION\")\n",
    "    print(f\"Noise Level: σ = {CONFIG['noise_level']}\")\n",
    "    print(\"=\"*80 + \"\\n\")\n",
    "    \n",
    "    # Define sparsity levels to test\n",
    "    sparsity_levels = [0.05, 0.08, 0.10, 0.12, 0.15]\n",
    "    \n",
    "    # Initialize models dictionary\n",
    "    models = {}\n",
    "    \n",
    "    # Load V3-ResUNet\n",
    "    print(\"Loading models...\")\n",
    "    if os.path.exists('sota_resunet.pth'):\n",
    "        model_v3 = ResUNet().to(device)\n",
    "        model_v3.load_state_dict(torch.load('sota_resunet.pth', map_location=device))\n",
    "        print(\"✓ Loaded: sota_resunet.pth (V3-ResUNet)\")\n",
    "        models['V3-ResUNet'] = model_v3\n",
    "    else:\n",
    "        print(\"⚠ WARNING: sota_resunet.pth not found! Skipping V3-ResUNet.\")\n",
    "    \n",
    "    # Load V4-Robust\n",
    "    if os.path.exists('sota_v3_best.pth'):\n",
    "        model_v4 = SOTARobustModel(latent_dim=1024, attr_dim=40).to(device)\n",
    "        model_v4.load_state_dict(torch.load('sota_v3_best.pth', map_location=device))\n",
    "        print(\"✓ Loaded: sota_v3_best.pth (V4-Robust)\")\n",
    "        models['V4-Robust'] = model_v4\n",
    "    else:\n",
    "        print(\"⚠ WARNING: sota_v3_best.pth not found! Skipping V4-Robust.\")\n",
    "    \n",
    "    if len(models) == 0:\n",
    "        print(\"\\n✗ ERROR: No models found! Please ensure model weights exist.\")\n",
    "        exit(1)\n",
    "    \n",
    "    print(f\"\\nModels loaded: {list(models.keys())}\")\n",
    "    print(f\"Device: {device}\")\n",
    "    print(f\"Gaussian Noise Level: σ = {CONFIG['noise_level']}\\n\")\n",
    "    \n",
    "    # Evaluate each model\n",
    "    all_results = {}\n",
    "    \n",
    "    for model_name, model_obj in models.items():\n",
    "        results = evaluate_multi_sparsity_with_noise(\n",
    "            model=model_obj,\n",
    "            model_name=model_name,\n",
    "            data_root='./data',\n",
    "            sparsity_levels=sparsity_levels,\n",
    "            noise_level=CONFIG['noise_level'],\n",
    "            device=device,\n",
    "            batch_size=64\n",
    "        )\n",
    "        \n",
    "        all_results[model_name] = results\n",
    "    \n",
    "    # Generate LaTeX table\n",
    "    print_noisy_performance_table(all_results, CONFIG['noise_level'])\n",
    "    \n",
    "    # Save results to JSON\n",
    "    output_file = f'celeba_noisy_performance_sigma{CONFIG[\"noise_level\"]}.json'\n",
    "    with open(output_file, 'w') as f:\n",
    "        # Convert for JSON serialization\n",
    "        clean_results = {}\n",
    "        for m in all_results:\n",
    "            clean_results[m] = {}\n",
    "            for rho in all_results[m]:\n",
    "                clean_results[m][str(rho)] = {\n",
    "                    k: float(v) if isinstance(v, (np.floating, float)) else int(v)\n",
    "                    for k, v in all_results[m][rho].items()\n",
    "                }\n",
    "        json.dump(clean_results, f, indent=2)\n",
    "    print(f\"✓ Results saved to '{output_file}'\")\n",
    "    \n",
    "    # Print summary\n",
    "    print(\"\\n\" + \"=\"*80)\n",
    "    print(\"EVALUATION COMPLETE!\")\n",
    "    print(f\"Gaussian Noise: σ = {CONFIG['noise_level']}\")\n",
    "    print(\"=\"*80)\n",
    "    \n",
    "    for model_name in all_results.keys():\n",
    "        print(f\"\\n{model_name} Summary:\")\n",
    "        results = all_results[model_name]\n",
    "        print(f\"{'Sparsity':<12} {'MSE':<12} {'PSNR (dB)':<12}\")\n",
    "        print(\"-\"*40)\n",
    "        for rho in sorted(results.keys()):\n",
    "            mse = results[rho]['mse']\n",
    "            psnr = results[rho]['psnr']\n",
    "            print(f\"ρ={rho:<10.2f} {mse:<12.6f} {psnr:<12.2f}\")\n",
    "    \n",
    "    # Calculate improvement\n",
    "    if len(models) == 2:\n",
    "        print(\"\\n\" + \"=\"*80)\n",
    "        print(\"ROBUSTNESS IMPROVEMENT (V4 vs V3):\")\n",
    "        print(\"=\"*80)\n",
    "        v3_name = 'V3-ResUNet'\n",
    "        v4_name = 'V4-Robust'\n",
    "        \n",
    "        if v3_name in all_results and v4_name in all_results:\n",
    "            print(f\"{'Sparsity':<12} {'V3 MSE':<12} {'V4 MSE':<12} {'Improvement':<12}\")\n",
    "            print(\"-\"*50)\n",
    "            for rho in sorted(sparsity_levels):\n",
    "                v3_mse = all_results[v3_name][rho]['mse']\n",
    "                v4_mse = all_results[v4_name][rho]['mse']\n",
    "                improvement = ((v3_mse - v4_mse) / v3_mse) * 100\n",
    "                print(f\"ρ={rho:<10.2f} {v3_mse:<12.6f} {v4_mse:<12.6f} {improvement:>10.2f}%\")\n",
    "    \n",
    "    print(\"\\n\" + \"=\"*80)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "190879b8",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "manitorch",
   "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.11.14"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
