{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "565ceb4d",
   "metadata": {},
   "source": [
    "## My algorithm in Manifold"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "cd275fc7",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "======================================================================\n",
      "HYBRID U-NET + RESNET MANIFOLD LEARNING\n",
      "Combining U-Net compression + ResNet gradient flow\n",
      "======================================================================\n",
      "\n",
      "Using device: cuda\n",
      "\n",
      "Dataset splits:\n",
      "  Train: 54000 samples\n",
      "  Val:   3000 samples\n",
      "  Test:  3000 samples\n",
      "\n",
      "Dataset size: 54000, Sparsity: 10% visible\n",
      "Dataset size: 3000, Sparsity: 10% visible\n",
      "Dataset size: 3000, Sparsity: 10% visible\n",
      "\n",
      "Model architecture:\n",
      "  Total parameters: 3,239,297\n",
      "  Latent dimension: 3136 (64×7×7 spatial)\n",
      "  ResNet blocks per resolution: 2\n",
      "\n",
      "======================================================================\n",
      "HYBRID U-NET + RESNET MANIFOLD LEARNING - TRAINING\n",
      "Latent dimension: 3136 (64×7×7 spatial)\n",
      "ResNet blocks per resolution: 2\n",
      "Max epochs: 100, Patience: 7\n",
      "======================================================================\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": [
      "Epoch 1/100, Batch 0, Loss: 1.6690 (Recon_x: 0.7565, Recon_s: 0.7794, Consistency: 1.3316)\n",
      "Epoch 1/100, Batch 100, Loss: 0.1279 (Recon_x: 0.1095, Recon_s: 0.0126, Consistency: 0.0571)\n",
      "Epoch 1/100, Batch 200, Loss: 0.1115 (Recon_x: 0.0985, Recon_s: 0.0089, Consistency: 0.0408)\n",
      "Epoch 1/100, Batch 300, Loss: 0.1045 (Recon_x: 0.0928, Recon_s: 0.0078, Consistency: 0.0396)\n",
      "Epoch 1/100, Batch 400, Loss: 0.0997 (Recon_x: 0.0879, Recon_s: 0.0081, Consistency: 0.0366)\n",
      "\n",
      "======================================================================\n",
      "Epoch 1/100 Summary:\n",
      "  TRAIN - Total: 0.1351, Recon_x: 0.1079, Recon_s: 0.0203, Consistency: 0.0700\n",
      "  VAL   - Total: 0.1027, Recon_x: 0.0901, Recon_s: 0.0087, Consistency: 0.0386\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.1027)\n",
      "======================================================================\n",
      "\n",
      "Epoch 2/100, Batch 0, Loss: 0.0950 (Recon_x: 0.0851, Recon_s: 0.0061, Consistency: 0.0377)\n",
      "Epoch 2/100, Batch 100, Loss: 0.1034 (Recon_x: 0.0946, Recon_s: 0.0054, Consistency: 0.0339)\n",
      "Epoch 2/100, Batch 200, Loss: 0.0943 (Recon_x: 0.0858, Recon_s: 0.0060, Consistency: 0.0255)\n",
      "Epoch 2/100, Batch 300, Loss: 0.0866 (Recon_x: 0.0802, Recon_s: 0.0040, Consistency: 0.0247)\n",
      "Epoch 2/100, Batch 400, Loss: 0.0810 (Recon_x: 0.0744, Recon_s: 0.0040, Consistency: 0.0250)\n",
      "\n",
      "======================================================================\n",
      "Epoch 2/100 Summary:\n",
      "  TRAIN - Total: 0.0903, Recon_x: 0.0822, Recon_s: 0.0054, Consistency: 0.0272\n",
      "  VAL   - Total: 0.0960, Recon_x: 0.0809, Recon_s: 0.0126, Consistency: 0.0260\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0960)\n",
      "======================================================================\n",
      "\n",
      "Epoch 3/100, Batch 0, Loss: 0.0812 (Recon_x: 0.0756, Recon_s: 0.0031, Consistency: 0.0250)\n",
      "Epoch 3/100, Batch 100, Loss: 0.0856 (Recon_x: 0.0793, Recon_s: 0.0040, Consistency: 0.0233)\n",
      "Epoch 3/100, Batch 200, Loss: 0.0805 (Recon_x: 0.0746, Recon_s: 0.0037, Consistency: 0.0222)\n",
      "Epoch 3/100, Batch 300, Loss: 0.0815 (Recon_x: 0.0761, Recon_s: 0.0033, Consistency: 0.0215)\n",
      "Epoch 3/100, Batch 400, Loss: 0.0858 (Recon_x: 0.0803, Recon_s: 0.0034, Consistency: 0.0215)\n",
      "\n",
      "======================================================================\n",
      "Epoch 3/100 Summary:\n",
      "  TRAIN - Total: 0.0826, Recon_x: 0.0766, Recon_s: 0.0036, Consistency: 0.0236\n",
      "  VAL   - Total: 0.0909, Recon_x: 0.0793, Recon_s: 0.0098, Consistency: 0.0187\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0909)\n",
      "======================================================================\n",
      "\n",
      "Epoch 4/100, Batch 0, Loss: 0.0777 (Recon_x: 0.0728, Recon_s: 0.0030, Consistency: 0.0185)\n",
      "Epoch 4/100, Batch 100, Loss: 0.0786 (Recon_x: 0.0734, Recon_s: 0.0033, Consistency: 0.0181)\n",
      "Epoch 4/100, Batch 200, Loss: 0.0729 (Recon_x: 0.0687, Recon_s: 0.0024, Consistency: 0.0178)\n",
      "Epoch 4/100, Batch 300, Loss: 0.0799 (Recon_x: 0.0759, Recon_s: 0.0023, Consistency: 0.0170)\n",
      "Epoch 4/100, Batch 400, Loss: 0.0806 (Recon_x: 0.0757, Recon_s: 0.0031, Consistency: 0.0180)\n",
      "\n",
      "======================================================================\n",
      "Epoch 4/100 Summary:\n",
      "  TRAIN - Total: 0.0772, Recon_x: 0.0728, Recon_s: 0.0027, Consistency: 0.0178\n",
      "  VAL   - Total: 0.0867, Recon_x: 0.0752, Recon_s: 0.0099, Consistency: 0.0169\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0867)\n",
      "======================================================================\n",
      "\n",
      "Epoch 5/100, Batch 0, Loss: 0.0786 (Recon_x: 0.0751, Recon_s: 0.0018, Consistency: 0.0164)\n",
      "Epoch 5/100, Batch 100, Loss: 0.0761 (Recon_x: 0.0721, Recon_s: 0.0024, Consistency: 0.0159)\n",
      "Epoch 5/100, Batch 200, Loss: 0.0740 (Recon_x: 0.0707, Recon_s: 0.0018, Consistency: 0.0142)\n",
      "Epoch 5/100, Batch 300, Loss: 0.0671 (Recon_x: 0.0638, Recon_s: 0.0019, Consistency: 0.0137)\n",
      "Epoch 5/100, Batch 400, Loss: 0.0765 (Recon_x: 0.0730, Recon_s: 0.0020, Consistency: 0.0148)\n",
      "\n",
      "======================================================================\n",
      "Epoch 5/100 Summary:\n",
      "  TRAIN - Total: 0.0743, Recon_x: 0.0707, Recon_s: 0.0020, Consistency: 0.0155\n",
      "  VAL   - Total: 0.0840, Recon_x: 0.0740, Recon_s: 0.0085, Consistency: 0.0147\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0840)\n",
      "======================================================================\n",
      "\n",
      "Epoch 6/100, Batch 0, Loss: 0.0811 (Recon_x: 0.0782, Recon_s: 0.0015, Consistency: 0.0143)\n",
      "Epoch 6/100, Batch 100, Loss: 0.0730 (Recon_x: 0.0701, Recon_s: 0.0017, Consistency: 0.0121)\n",
      "Epoch 6/100, Batch 200, Loss: 0.0699 (Recon_x: 0.0669, Recon_s: 0.0016, Consistency: 0.0144)\n",
      "Epoch 6/100, Batch 300, Loss: 0.0705 (Recon_x: 0.0675, Recon_s: 0.0017, Consistency: 0.0133)\n",
      "Epoch 6/100, Batch 400, Loss: 0.0719 (Recon_x: 0.0692, Recon_s: 0.0013, Consistency: 0.0130)\n",
      "\n",
      "======================================================================\n",
      "Epoch 6/100 Summary:\n",
      "  TRAIN - Total: 0.0720, Recon_x: 0.0690, Recon_s: 0.0017, Consistency: 0.0130\n",
      "  VAL   - Total: 0.0829, Recon_x: 0.0727, Recon_s: 0.0089, Consistency: 0.0125\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0829)\n",
      "======================================================================\n",
      "\n",
      "Epoch 7/100, Batch 0, Loss: 0.0723 (Recon_x: 0.0697, Recon_s: 0.0014, Consistency: 0.0117)\n",
      "Epoch 7/100, Batch 100, Loss: 0.0790 (Recon_x: 0.0742, Recon_s: 0.0023, Consistency: 0.0255)\n",
      "Epoch 7/100, Batch 200, Loss: 0.0703 (Recon_x: 0.0675, Recon_s: 0.0016, Consistency: 0.0123)\n",
      "Epoch 7/100, Batch 300, Loss: 0.0650 (Recon_x: 0.0627, Recon_s: 0.0011, Consistency: 0.0113)\n",
      "Epoch 7/100, Batch 400, Loss: 0.0658 (Recon_x: 0.0631, Recon_s: 0.0016, Consistency: 0.0105)\n",
      "\n",
      "======================================================================\n",
      "Epoch 7/100 Summary:\n",
      "  TRAIN - Total: 0.0706, Recon_x: 0.0678, Recon_s: 0.0015, Consistency: 0.0128\n",
      "  VAL   - Total: 0.1821, Recon_x: 0.1691, Recon_s: 0.0105, Consistency: 0.0242\n",
      "  No improvement for 1 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 8/100, Batch 0, Loss: 0.0711 (Recon_x: 0.0685, Recon_s: 0.0014, Consistency: 0.0119)\n",
      "Epoch 8/100, Batch 100, Loss: 0.0669 (Recon_x: 0.0646, Recon_s: 0.0013, Consistency: 0.0104)\n",
      "Epoch 8/100, Batch 200, Loss: 0.0701 (Recon_x: 0.0675, Recon_s: 0.0016, Consistency: 0.0108)\n",
      "Epoch 8/100, Batch 300, Loss: 0.0704 (Recon_x: 0.0681, Recon_s: 0.0013, Consistency: 0.0103)\n",
      "Epoch 8/100, Batch 400, Loss: 0.0692 (Recon_x: 0.0665, Recon_s: 0.0018, Consistency: 0.0092)\n",
      "\n",
      "======================================================================\n",
      "Epoch 8/100 Summary:\n",
      "  TRAIN - Total: 0.0693, Recon_x: 0.0669, Recon_s: 0.0013, Consistency: 0.0104\n",
      "  VAL   - Total: 0.0783, Recon_x: 0.0671, Recon_s: 0.0101, Consistency: 0.0115\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0783)\n",
      "======================================================================\n",
      "\n",
      "Epoch 9/100, Batch 0, Loss: 0.0721 (Recon_x: 0.0695, Recon_s: 0.0015, Consistency: 0.0112)\n",
      "Epoch 9/100, Batch 100, Loss: 0.0681 (Recon_x: 0.0660, Recon_s: 0.0012, Consistency: 0.0094)\n",
      "Epoch 9/100, Batch 200, Loss: 0.0653 (Recon_x: 0.0631, Recon_s: 0.0012, Consistency: 0.0101)\n",
      "Epoch 9/100, Batch 300, Loss: 0.0708 (Recon_x: 0.0681, Recon_s: 0.0016, Consistency: 0.0099)\n",
      "Epoch 9/100, Batch 400, Loss: 0.0720 (Recon_x: 0.0699, Recon_s: 0.0012, Consistency: 0.0091)\n",
      "\n",
      "======================================================================\n",
      "Epoch 9/100 Summary:\n",
      "  TRAIN - Total: 0.0679, Recon_x: 0.0658, Recon_s: 0.0012, Consistency: 0.0095\n",
      "  VAL   - Total: 0.0769, Recon_x: 0.0668, Recon_s: 0.0091, Consistency: 0.0097\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0769)\n",
      "======================================================================\n",
      "\n",
      "Epoch 10/100, Batch 0, Loss: 0.0663 (Recon_x: 0.0642, Recon_s: 0.0011, Consistency: 0.0088)\n",
      "Epoch 10/100, Batch 100, Loss: 0.0696 (Recon_x: 0.0678, Recon_s: 0.0009, Consistency: 0.0092)\n",
      "Epoch 10/100, Batch 200, Loss: 0.0693 (Recon_x: 0.0674, Recon_s: 0.0010, Consistency: 0.0088)\n",
      "Epoch 10/100, Batch 300, Loss: 0.0691 (Recon_x: 0.0660, Recon_s: 0.0017, Consistency: 0.0137)\n",
      "Epoch 10/100, Batch 400, Loss: 0.0691 (Recon_x: 0.0670, Recon_s: 0.0010, Consistency: 0.0109)\n",
      "\n",
      "======================================================================\n",
      "Epoch 10/100 Summary:\n",
      "  TRAIN - Total: 0.0682, Recon_x: 0.0656, Recon_s: 0.0015, Consistency: 0.0116\n",
      "  VAL   - Total: 0.0786, Recon_x: 0.0679, Recon_s: 0.0097, Consistency: 0.0103\n",
      "  No improvement for 1 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 11/100, Batch 0, Loss: 0.0687 (Recon_x: 0.0661, Recon_s: 0.0015, Consistency: 0.0102)\n",
      "Epoch 11/100, Batch 100, Loss: 0.0708 (Recon_x: 0.0688, Recon_s: 0.0011, Consistency: 0.0088)\n",
      "Epoch 11/100, Batch 200, Loss: 0.0592 (Recon_x: 0.0575, Recon_s: 0.0009, Consistency: 0.0086)\n",
      "Epoch 11/100, Batch 300, Loss: 0.0732 (Recon_x: 0.0712, Recon_s: 0.0011, Consistency: 0.0085)\n",
      "Epoch 11/100, Batch 400, Loss: 0.0658 (Recon_x: 0.0643, Recon_s: 0.0007, Consistency: 0.0079)\n",
      "\n",
      "======================================================================\n",
      "Epoch 11/100 Summary:\n",
      "  TRAIN - Total: 0.0667, Recon_x: 0.0649, Recon_s: 0.0010, Consistency: 0.0087\n",
      "  VAL   - Total: 0.0774, Recon_x: 0.0657, Recon_s: 0.0109, Consistency: 0.0084\n",
      "  No improvement for 2 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 12/100, Batch 0, Loss: 0.0592 (Recon_x: 0.0578, Recon_s: 0.0006, Consistency: 0.0076)\n",
      "Epoch 12/100, Batch 100, Loss: 0.0680 (Recon_x: 0.0662, Recon_s: 0.0009, Consistency: 0.0086)\n",
      "Epoch 12/100, Batch 200, Loss: 0.0652 (Recon_x: 0.0637, Recon_s: 0.0008, Consistency: 0.0074)\n",
      "Epoch 12/100, Batch 300, Loss: 0.0648 (Recon_x: 0.0631, Recon_s: 0.0009, Consistency: 0.0076)\n",
      "Epoch 12/100, Batch 400, Loss: 0.0600 (Recon_x: 0.0585, Recon_s: 0.0009, Consistency: 0.0067)\n",
      "\n",
      "======================================================================\n",
      "Epoch 12/100 Summary:\n",
      "  TRAIN - Total: 0.0658, Recon_x: 0.0641, Recon_s: 0.0009, Consistency: 0.0076\n",
      "  VAL   - Total: 0.0944, Recon_x: 0.0825, Recon_s: 0.0110, Consistency: 0.0092\n",
      "  No improvement for 3 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 13/100, Batch 0, Loss: 0.0631 (Recon_x: 0.0606, Recon_s: 0.0017, Consistency: 0.0075)\n",
      "Epoch 13/100, Batch 100, Loss: 0.0655 (Recon_x: 0.0638, Recon_s: 0.0009, Consistency: 0.0079)\n",
      "Epoch 13/100, Batch 200, Loss: 0.0643 (Recon_x: 0.0626, Recon_s: 0.0010, Consistency: 0.0069)\n",
      "Epoch 13/100, Batch 300, Loss: 0.0679 (Recon_x: 0.0665, Recon_s: 0.0008, Consistency: 0.0068)\n",
      "Epoch 13/100, Batch 400, Loss: 0.0670 (Recon_x: 0.0655, Recon_s: 0.0008, Consistency: 0.0065)\n",
      "  → Learning rate reduced: 1.00e-03 → 5.00e-04\n",
      "\n",
      "======================================================================\n",
      "Epoch 13/100 Summary:\n",
      "  TRAIN - Total: 0.0652, Recon_x: 0.0636, Recon_s: 0.0009, Consistency: 0.0077\n",
      "  VAL   - Total: 0.0784, Recon_x: 0.0669, Recon_s: 0.0107, Consistency: 0.0083\n",
      "  No improvement for 4 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 14/100, Batch 0, Loss: 0.0623 (Recon_x: 0.0606, Recon_s: 0.0009, Consistency: 0.0072)\n",
      "Epoch 14/100, Batch 100, Loss: 0.0612 (Recon_x: 0.0602, Recon_s: 0.0005, Consistency: 0.0050)\n",
      "Epoch 14/100, Batch 200, Loss: 0.0574 (Recon_x: 0.0564, Recon_s: 0.0005, Consistency: 0.0051)\n",
      "Epoch 14/100, Batch 300, Loss: 0.0654 (Recon_x: 0.0641, Recon_s: 0.0009, Consistency: 0.0047)\n",
      "Epoch 14/100, Batch 400, Loss: 0.0620 (Recon_x: 0.0611, Recon_s: 0.0004, Consistency: 0.0047)\n",
      "\n",
      "======================================================================\n",
      "Epoch 14/100 Summary:\n",
      "  TRAIN - Total: 0.0610, Recon_x: 0.0600, Recon_s: 0.0005, Consistency: 0.0052\n",
      "  VAL   - Total: 0.0752, Recon_x: 0.0598, Recon_s: 0.0149, Consistency: 0.0051\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0752)\n",
      "======================================================================\n",
      "\n",
      "Epoch 15/100, Batch 0, Loss: 0.0594 (Recon_x: 0.0584, Recon_s: 0.0005, Consistency: 0.0049)\n",
      "Epoch 15/100, Batch 100, Loss: 0.0573 (Recon_x: 0.0564, Recon_s: 0.0004, Consistency: 0.0047)\n",
      "Epoch 15/100, Batch 200, Loss: 0.0563 (Recon_x: 0.0554, Recon_s: 0.0004, Consistency: 0.0047)\n",
      "Epoch 15/100, Batch 300, Loss: 0.0661 (Recon_x: 0.0652, Recon_s: 0.0005, Consistency: 0.0047)\n",
      "Epoch 15/100, Batch 400, Loss: 0.0625 (Recon_x: 0.0615, Recon_s: 0.0006, Consistency: 0.0047)\n",
      "\n",
      "======================================================================\n",
      "Epoch 15/100 Summary:\n",
      "  TRAIN - Total: 0.0605, Recon_x: 0.0595, Recon_s: 0.0005, Consistency: 0.0048\n",
      "  VAL   - Total: 0.0737, Recon_x: 0.0615, Recon_s: 0.0117, Consistency: 0.0055\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0737)\n",
      "======================================================================\n",
      "\n",
      "Epoch 16/100, Batch 0, Loss: 0.0594 (Recon_x: 0.0585, Recon_s: 0.0004, Consistency: 0.0048)\n",
      "Epoch 16/100, Batch 100, Loss: 0.0687 (Recon_x: 0.0675, Recon_s: 0.0007, Consistency: 0.0052)\n",
      "Epoch 16/100, Batch 200, Loss: 0.0639 (Recon_x: 0.0621, Recon_s: 0.0014, Consistency: 0.0046)\n",
      "Epoch 16/100, Batch 300, Loss: 0.0609 (Recon_x: 0.0601, Recon_s: 0.0003, Consistency: 0.0046)\n",
      "Epoch 16/100, Batch 400, Loss: 0.0603 (Recon_x: 0.0595, Recon_s: 0.0003, Consistency: 0.0048)\n",
      "\n",
      "======================================================================\n",
      "Epoch 16/100 Summary:\n",
      "  TRAIN - Total: 0.0605, Recon_x: 0.0596, Recon_s: 0.0005, Consistency: 0.0046\n",
      "  VAL   - Total: 0.0749, Recon_x: 0.0623, Recon_s: 0.0122, Consistency: 0.0047\n",
      "  No improvement for 1 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 17/100, Batch 0, Loss: 0.0623 (Recon_x: 0.0614, Recon_s: 0.0005, Consistency: 0.0044)\n",
      "Epoch 17/100, Batch 100, Loss: 0.0549 (Recon_x: 0.0540, Recon_s: 0.0004, Consistency: 0.0047)\n",
      "Epoch 17/100, Batch 200, Loss: 0.0640 (Recon_x: 0.0631, Recon_s: 0.0005, Consistency: 0.0048)\n",
      "Epoch 17/100, Batch 300, Loss: 0.0598 (Recon_x: 0.0589, Recon_s: 0.0005, Consistency: 0.0044)\n",
      "Epoch 17/100, Batch 400, Loss: 0.0613 (Recon_x: 0.0603, Recon_s: 0.0005, Consistency: 0.0045)\n",
      "\n",
      "======================================================================\n",
      "Epoch 17/100 Summary:\n",
      "  TRAIN - Total: 0.0597, Recon_x: 0.0588, Recon_s: 0.0005, Consistency: 0.0045\n",
      "  VAL   - Total: 0.0722, Recon_x: 0.0602, Recon_s: 0.0115, Consistency: 0.0062\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0722)\n",
      "======================================================================\n",
      "\n",
      "Epoch 18/100, Batch 0, Loss: 0.0590 (Recon_x: 0.0579, Recon_s: 0.0006, Consistency: 0.0045)\n",
      "Epoch 18/100, Batch 100, Loss: 0.0644 (Recon_x: 0.0635, Recon_s: 0.0005, Consistency: 0.0043)\n",
      "Epoch 18/100, Batch 200, Loss: 0.0566 (Recon_x: 0.0558, Recon_s: 0.0003, Consistency: 0.0044)\n",
      "Epoch 18/100, Batch 300, Loss: 0.0584 (Recon_x: 0.0575, Recon_s: 0.0005, Consistency: 0.0042)\n",
      "Epoch 18/100, Batch 400, Loss: 0.0571 (Recon_x: 0.0563, Recon_s: 0.0004, Consistency: 0.0044)\n",
      "\n",
      "======================================================================\n",
      "Epoch 18/100 Summary:\n",
      "  TRAIN - Total: 0.0599, Recon_x: 0.0590, Recon_s: 0.0004, Consistency: 0.0044\n",
      "  VAL   - Total: 0.0715, Recon_x: 0.0611, Recon_s: 0.0100, Consistency: 0.0047\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0715)\n",
      "======================================================================\n",
      "\n",
      "Epoch 19/100, Batch 0, Loss: 0.0630 (Recon_x: 0.0619, Recon_s: 0.0006, Consistency: 0.0041)\n",
      "Epoch 19/100, Batch 100, Loss: 0.0627 (Recon_x: 0.0617, Recon_s: 0.0005, Consistency: 0.0048)\n",
      "Epoch 19/100, Batch 200, Loss: 0.0610 (Recon_x: 0.0601, Recon_s: 0.0005, Consistency: 0.0044)\n",
      "Epoch 19/100, Batch 300, Loss: 0.0562 (Recon_x: 0.0553, Recon_s: 0.0005, Consistency: 0.0045)\n",
      "Epoch 19/100, Batch 400, Loss: 0.0613 (Recon_x: 0.0605, Recon_s: 0.0003, Consistency: 0.0042)\n",
      "\n",
      "======================================================================\n",
      "Epoch 19/100 Summary:\n",
      "  TRAIN - Total: 0.0597, Recon_x: 0.0588, Recon_s: 0.0005, Consistency: 0.0043\n",
      "  VAL   - Total: 0.0991, Recon_x: 0.0858, Recon_s: 0.0127, Consistency: 0.0059\n",
      "  No improvement for 1 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 20/100, Batch 0, Loss: 0.0583 (Recon_x: 0.0575, Recon_s: 0.0004, Consistency: 0.0042)\n",
      "Epoch 20/100, Batch 100, Loss: 0.0597 (Recon_x: 0.0588, Recon_s: 0.0004, Consistency: 0.0042)\n",
      "Epoch 20/100, Batch 200, Loss: 0.0621 (Recon_x: 0.0606, Recon_s: 0.0006, Consistency: 0.0096)\n",
      "Epoch 20/100, Batch 300, Loss: 0.0582 (Recon_x: 0.0573, Recon_s: 0.0004, Consistency: 0.0044)\n",
      "Epoch 20/100, Batch 400, Loss: 0.0580 (Recon_x: 0.0572, Recon_s: 0.0003, Consistency: 0.0043)\n",
      "\n",
      "======================================================================\n",
      "Epoch 20/100 Summary:\n",
      "  TRAIN - Total: 0.0594, Recon_x: 0.0585, Recon_s: 0.0005, Consistency: 0.0045\n",
      "  VAL   - Total: 0.0738, Recon_x: 0.0611, Recon_s: 0.0122, Consistency: 0.0045\n",
      "  No improvement for 2 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 21/100, Batch 0, Loss: 0.0593 (Recon_x: 0.0582, Recon_s: 0.0007, Consistency: 0.0041)\n",
      "Epoch 21/100, Batch 100, Loss: 0.0563 (Recon_x: 0.0554, Recon_s: 0.0005, Consistency: 0.0041)\n",
      "Epoch 21/100, Batch 200, Loss: 0.0586 (Recon_x: 0.0578, Recon_s: 0.0004, Consistency: 0.0040)\n",
      "Epoch 21/100, Batch 300, Loss: 0.0575 (Recon_x: 0.0567, Recon_s: 0.0004, Consistency: 0.0041)\n",
      "Epoch 21/100, Batch 400, Loss: 0.0631 (Recon_x: 0.0623, Recon_s: 0.0003, Consistency: 0.0039)\n",
      "\n",
      "======================================================================\n",
      "Epoch 21/100 Summary:\n",
      "  TRAIN - Total: 0.0591, Recon_x: 0.0583, Recon_s: 0.0004, Consistency: 0.0041\n",
      "  VAL   - Total: 0.0771, Recon_x: 0.0623, Recon_s: 0.0143, Consistency: 0.0042\n",
      "  No improvement for 3 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 22/100, Batch 0, Loss: 0.0570 (Recon_x: 0.0562, Recon_s: 0.0004, Consistency: 0.0041)\n",
      "Epoch 22/100, Batch 100, Loss: 0.0574 (Recon_x: 0.0566, Recon_s: 0.0004, Consistency: 0.0040)\n",
      "Epoch 22/100, Batch 200, Loss: 0.0599 (Recon_x: 0.0590, Recon_s: 0.0005, Consistency: 0.0040)\n",
      "Epoch 22/100, Batch 300, Loss: 0.0590 (Recon_x: 0.0582, Recon_s: 0.0004, Consistency: 0.0040)\n",
      "Epoch 22/100, Batch 400, Loss: 0.0581 (Recon_x: 0.0575, Recon_s: 0.0003, Consistency: 0.0038)\n",
      "  → Learning rate reduced: 5.00e-04 → 2.50e-04\n",
      "\n",
      "======================================================================\n",
      "Epoch 22/100 Summary:\n",
      "  TRAIN - Total: 0.0591, Recon_x: 0.0582, Recon_s: 0.0004, Consistency: 0.0040\n",
      "  VAL   - Total: 0.0728, Recon_x: 0.0606, Recon_s: 0.0119, Consistency: 0.0038\n",
      "  No improvement for 4 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 23/100, Batch 0, Loss: 0.0601 (Recon_x: 0.0594, Recon_s: 0.0003, Consistency: 0.0037)\n",
      "Epoch 23/100, Batch 100, Loss: 0.0553 (Recon_x: 0.0547, Recon_s: 0.0003, Consistency: 0.0032)\n",
      "Epoch 23/100, Batch 200, Loss: 0.0551 (Recon_x: 0.0545, Recon_s: 0.0003, Consistency: 0.0033)\n",
      "Epoch 23/100, Batch 300, Loss: 0.0554 (Recon_x: 0.0549, Recon_s: 0.0002, Consistency: 0.0030)\n",
      "Epoch 23/100, Batch 400, Loss: 0.0544 (Recon_x: 0.0539, Recon_s: 0.0002, Consistency: 0.0030)\n",
      "\n",
      "======================================================================\n",
      "Epoch 23/100 Summary:\n",
      "  TRAIN - Total: 0.0568, Recon_x: 0.0562, Recon_s: 0.0003, Consistency: 0.0032\n",
      "  VAL   - Total: 0.0716, Recon_x: 0.0576, Recon_s: 0.0136, Consistency: 0.0035\n",
      "  No improvement for 5 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 24/100, Batch 0, Loss: 0.0605 (Recon_x: 0.0599, Recon_s: 0.0003, Consistency: 0.0030)\n",
      "Epoch 24/100, Batch 100, Loss: 0.0545 (Recon_x: 0.0540, Recon_s: 0.0002, Consistency: 0.0029)\n",
      "Epoch 24/100, Batch 200, Loss: 0.0544 (Recon_x: 0.0538, Recon_s: 0.0003, Consistency: 0.0030)\n",
      "Epoch 24/100, Batch 300, Loss: 0.0559 (Recon_x: 0.0554, Recon_s: 0.0002, Consistency: 0.0030)\n",
      "Epoch 24/100, Batch 400, Loss: 0.0570 (Recon_x: 0.0565, Recon_s: 0.0002, Consistency: 0.0029)\n",
      "\n",
      "======================================================================\n",
      "Epoch 24/100 Summary:\n",
      "  TRAIN - Total: 0.0562, Recon_x: 0.0556, Recon_s: 0.0002, Consistency: 0.0030\n",
      "  VAL   - Total: 0.0703, Recon_x: 0.0573, Recon_s: 0.0125, Consistency: 0.0052\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0703)\n",
      "======================================================================\n",
      "\n",
      "Epoch 25/100, Batch 0, Loss: 0.0598 (Recon_x: 0.0593, Recon_s: 0.0003, Consistency: 0.0029)\n",
      "Epoch 25/100, Batch 100, Loss: 0.0573 (Recon_x: 0.0567, Recon_s: 0.0003, Consistency: 0.0028)\n",
      "Epoch 25/100, Batch 200, Loss: 0.0593 (Recon_x: 0.0588, Recon_s: 0.0002, Consistency: 0.0029)\n",
      "Epoch 25/100, Batch 300, Loss: 0.0527 (Recon_x: 0.0522, Recon_s: 0.0002, Consistency: 0.0028)\n",
      "Epoch 25/100, Batch 400, Loss: 0.0555 (Recon_x: 0.0549, Recon_s: 0.0003, Consistency: 0.0029)\n",
      "\n",
      "======================================================================\n",
      "Epoch 25/100 Summary:\n",
      "  TRAIN - Total: 0.0562, Recon_x: 0.0557, Recon_s: 0.0002, Consistency: 0.0029\n",
      "  VAL   - Total: 0.0717, Recon_x: 0.0579, Recon_s: 0.0132, Consistency: 0.0052\n",
      "  No improvement for 1 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 26/100, Batch 0, Loss: 0.0582 (Recon_x: 0.0576, Recon_s: 0.0003, Consistency: 0.0029)\n",
      "Epoch 26/100, Batch 100, Loss: 0.0544 (Recon_x: 0.0538, Recon_s: 0.0003, Consistency: 0.0028)\n",
      "Epoch 26/100, Batch 200, Loss: 0.0562 (Recon_x: 0.0557, Recon_s: 0.0002, Consistency: 0.0027)\n",
      "Epoch 26/100, Batch 300, Loss: 0.0574 (Recon_x: 0.0569, Recon_s: 0.0003, Consistency: 0.0027)\n",
      "Epoch 26/100, Batch 400, Loss: 0.0605 (Recon_x: 0.0599, Recon_s: 0.0003, Consistency: 0.0029)\n",
      "\n",
      "======================================================================\n",
      "Epoch 26/100 Summary:\n",
      "  TRAIN - Total: 0.0558, Recon_x: 0.0553, Recon_s: 0.0002, Consistency: 0.0028\n",
      "  VAL   - Total: 0.0715, Recon_x: 0.0587, Recon_s: 0.0125, Consistency: 0.0029\n",
      "  No improvement for 2 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 27/100, Batch 0, Loss: 0.0587 (Recon_x: 0.0581, Recon_s: 0.0003, Consistency: 0.0028)\n",
      "Epoch 27/100, Batch 100, Loss: 0.0618 (Recon_x: 0.0613, Recon_s: 0.0002, Consistency: 0.0028)\n",
      "Epoch 27/100, Batch 200, Loss: 0.0528 (Recon_x: 0.0523, Recon_s: 0.0002, Consistency: 0.0028)\n",
      "Epoch 27/100, Batch 300, Loss: 0.0540 (Recon_x: 0.0535, Recon_s: 0.0002, Consistency: 0.0028)\n",
      "Epoch 27/100, Batch 400, Loss: 0.0553 (Recon_x: 0.0548, Recon_s: 0.0002, Consistency: 0.0028)\n",
      "\n",
      "======================================================================\n",
      "Epoch 27/100 Summary:\n",
      "  TRAIN - Total: 0.0559, Recon_x: 0.0554, Recon_s: 0.0003, Consistency: 0.0028\n",
      "  VAL   - Total: 0.0717, Recon_x: 0.0569, Recon_s: 0.0145, Consistency: 0.0029\n",
      "  No improvement for 3 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 28/100, Batch 0, Loss: 0.0552 (Recon_x: 0.0547, Recon_s: 0.0002, Consistency: 0.0028)\n",
      "Epoch 28/100, Batch 100, Loss: 0.0536 (Recon_x: 0.0531, Recon_s: 0.0002, Consistency: 0.0026)\n",
      "Epoch 28/100, Batch 200, Loss: 0.0553 (Recon_x: 0.0548, Recon_s: 0.0002, Consistency: 0.0027)\n",
      "Epoch 28/100, Batch 300, Loss: 0.0637 (Recon_x: 0.0632, Recon_s: 0.0002, Consistency: 0.0027)\n",
      "Epoch 28/100, Batch 400, Loss: 0.0530 (Recon_x: 0.0525, Recon_s: 0.0002, Consistency: 0.0026)\n",
      "  → Learning rate reduced: 2.50e-04 → 1.25e-04\n",
      "\n",
      "======================================================================\n",
      "Epoch 28/100 Summary:\n",
      "  TRAIN - Total: 0.0558, Recon_x: 0.0553, Recon_s: 0.0003, Consistency: 0.0027\n",
      "  VAL   - Total: 0.0706, Recon_x: 0.0567, Recon_s: 0.0136, Consistency: 0.0027\n",
      "  No improvement for 4 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 29/100, Batch 0, Loss: 0.0493 (Recon_x: 0.0488, Recon_s: 0.0002, Consistency: 0.0027)\n",
      "Epoch 29/100, Batch 100, Loss: 0.0525 (Recon_x: 0.0521, Recon_s: 0.0002, Consistency: 0.0025)\n",
      "Epoch 29/100, Batch 200, Loss: 0.0525 (Recon_x: 0.0520, Recon_s: 0.0002, Consistency: 0.0024)\n",
      "Epoch 29/100, Batch 300, Loss: 0.0568 (Recon_x: 0.0563, Recon_s: 0.0002, Consistency: 0.0024)\n",
      "Epoch 29/100, Batch 400, Loss: 0.0574 (Recon_x: 0.0570, Recon_s: 0.0002, Consistency: 0.0024)\n",
      "\n",
      "======================================================================\n",
      "Epoch 29/100 Summary:\n",
      "  TRAIN - Total: 0.0543, Recon_x: 0.0539, Recon_s: 0.0002, Consistency: 0.0025\n",
      "  VAL   - Total: 0.0700, Recon_x: 0.0564, Recon_s: 0.0133, Consistency: 0.0025\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0700)\n",
      "======================================================================\n",
      "\n",
      "Epoch 30/100, Batch 0, Loss: 0.0511 (Recon_x: 0.0508, Recon_s: 0.0002, Consistency: 0.0024)\n",
      "Epoch 30/100, Batch 100, Loss: 0.0476 (Recon_x: 0.0472, Recon_s: 0.0002, Consistency: 0.0022)\n",
      "Epoch 30/100, Batch 200, Loss: 0.0537 (Recon_x: 0.0534, Recon_s: 0.0001, Consistency: 0.0022)\n",
      "Epoch 30/100, Batch 300, Loss: 0.0530 (Recon_x: 0.0526, Recon_s: 0.0002, Consistency: 0.0022)\n",
      "Epoch 30/100, Batch 400, Loss: 0.0520 (Recon_x: 0.0516, Recon_s: 0.0002, Consistency: 0.0022)\n",
      "\n",
      "======================================================================\n",
      "Epoch 30/100 Summary:\n",
      "  TRAIN - Total: 0.0540, Recon_x: 0.0536, Recon_s: 0.0002, Consistency: 0.0022\n",
      "  VAL   - Total: 0.0697, Recon_x: 0.0565, Recon_s: 0.0129, Consistency: 0.0022\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0697)\n",
      "======================================================================\n",
      "\n",
      "Epoch 31/100, Batch 0, Loss: 0.0508 (Recon_x: 0.0504, Recon_s: 0.0001, Consistency: 0.0021)\n",
      "Epoch 31/100, Batch 100, Loss: 0.0553 (Recon_x: 0.0549, Recon_s: 0.0002, Consistency: 0.0020)\n",
      "Epoch 31/100, Batch 200, Loss: 0.0548 (Recon_x: 0.0545, Recon_s: 0.0002, Consistency: 0.0021)\n",
      "Epoch 31/100, Batch 300, Loss: 0.0555 (Recon_x: 0.0552, Recon_s: 0.0001, Consistency: 0.0021)\n",
      "Epoch 31/100, Batch 400, Loss: 0.0579 (Recon_x: 0.0575, Recon_s: 0.0002, Consistency: 0.0021)\n",
      "\n",
      "======================================================================\n",
      "Epoch 31/100 Summary:\n",
      "  TRAIN - Total: 0.0539, Recon_x: 0.0535, Recon_s: 0.0002, Consistency: 0.0021\n",
      "  VAL   - Total: 0.0682, Recon_x: 0.0555, Recon_s: 0.0124, Consistency: 0.0021\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0682)\n",
      "======================================================================\n",
      "\n",
      "Epoch 32/100, Batch 0, Loss: 0.0615 (Recon_x: 0.0610, Recon_s: 0.0003, Consistency: 0.0022)\n",
      "Epoch 32/100, Batch 100, Loss: 0.0563 (Recon_x: 0.0559, Recon_s: 0.0001, Consistency: 0.0021)\n",
      "Epoch 32/100, Batch 200, Loss: 0.0542 (Recon_x: 0.0537, Recon_s: 0.0002, Consistency: 0.0021)\n",
      "Epoch 32/100, Batch 300, Loss: 0.0527 (Recon_x: 0.0524, Recon_s: 0.0001, Consistency: 0.0020)\n",
      "Epoch 32/100, Batch 400, Loss: 0.0534 (Recon_x: 0.0530, Recon_s: 0.0002, Consistency: 0.0020)\n",
      "\n",
      "======================================================================\n",
      "Epoch 32/100 Summary:\n",
      "  TRAIN - Total: 0.0538, Recon_x: 0.0534, Recon_s: 0.0002, Consistency: 0.0021\n",
      "  VAL   - Total: 0.0695, Recon_x: 0.0567, Recon_s: 0.0125, Consistency: 0.0021\n",
      "  No improvement for 1 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 33/100, Batch 0, Loss: 0.0502 (Recon_x: 0.0498, Recon_s: 0.0002, Consistency: 0.0020)\n",
      "Epoch 33/100, Batch 100, Loss: 0.0531 (Recon_x: 0.0527, Recon_s: 0.0001, Consistency: 0.0020)\n",
      "Epoch 33/100, Batch 200, Loss: 0.0574 (Recon_x: 0.0570, Recon_s: 0.0002, Consistency: 0.0020)\n",
      "Epoch 33/100, Batch 300, Loss: 0.0542 (Recon_x: 0.0539, Recon_s: 0.0001, Consistency: 0.0020)\n",
      "Epoch 33/100, Batch 400, Loss: 0.0537 (Recon_x: 0.0533, Recon_s: 0.0001, Consistency: 0.0021)\n",
      "\n",
      "======================================================================\n",
      "Epoch 33/100 Summary:\n",
      "  TRAIN - Total: 0.0537, Recon_x: 0.0533, Recon_s: 0.0002, Consistency: 0.0020\n",
      "  VAL   - Total: 0.0697, Recon_x: 0.0551, Recon_s: 0.0144, Consistency: 0.0021\n",
      "  No improvement for 2 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 34/100, Batch 0, Loss: 0.0530 (Recon_x: 0.0526, Recon_s: 0.0001, Consistency: 0.0021)\n",
      "Epoch 34/100, Batch 100, Loss: 0.0527 (Recon_x: 0.0523, Recon_s: 0.0001, Consistency: 0.0019)\n",
      "Epoch 34/100, Batch 200, Loss: 0.0541 (Recon_x: 0.0538, Recon_s: 0.0002, Consistency: 0.0019)\n",
      "Epoch 34/100, Batch 300, Loss: 0.0549 (Recon_x: 0.0545, Recon_s: 0.0002, Consistency: 0.0020)\n",
      "Epoch 34/100, Batch 400, Loss: 0.0563 (Recon_x: 0.0559, Recon_s: 0.0002, Consistency: 0.0020)\n",
      "\n",
      "======================================================================\n",
      "Epoch 34/100 Summary:\n",
      "  TRAIN - Total: 0.0535, Recon_x: 0.0531, Recon_s: 0.0002, Consistency: 0.0020\n",
      "  VAL   - Total: 0.0712, Recon_x: 0.0565, Recon_s: 0.0145, Consistency: 0.0021\n",
      "  No improvement for 3 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 35/100, Batch 0, Loss: 0.0549 (Recon_x: 0.0546, Recon_s: 0.0001, Consistency: 0.0020)\n",
      "Epoch 35/100, Batch 100, Loss: 0.0513 (Recon_x: 0.0509, Recon_s: 0.0002, Consistency: 0.0020)\n",
      "Epoch 35/100, Batch 200, Loss: 0.0583 (Recon_x: 0.0580, Recon_s: 0.0001, Consistency: 0.0020)\n",
      "Epoch 35/100, Batch 300, Loss: 0.0551 (Recon_x: 0.0547, Recon_s: 0.0002, Consistency: 0.0019)\n",
      "Epoch 35/100, Batch 400, Loss: 0.0534 (Recon_x: 0.0531, Recon_s: 0.0001, Consistency: 0.0020)\n",
      "  → Learning rate reduced: 1.25e-04 → 6.25e-05\n",
      "\n",
      "======================================================================\n",
      "Epoch 35/100 Summary:\n",
      "  TRAIN - Total: 0.0535, Recon_x: 0.0531, Recon_s: 0.0001, Consistency: 0.0020\n",
      "  VAL   - Total: 0.0693, Recon_x: 0.0561, Recon_s: 0.0130, Consistency: 0.0020\n",
      "  No improvement for 4 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 36/100, Batch 0, Loss: 0.0498 (Recon_x: 0.0494, Recon_s: 0.0001, Consistency: 0.0019)\n",
      "Epoch 36/100, Batch 100, Loss: 0.0509 (Recon_x: 0.0506, Recon_s: 0.0001, Consistency: 0.0019)\n",
      "Epoch 36/100, Batch 200, Loss: 0.0551 (Recon_x: 0.0547, Recon_s: 0.0002, Consistency: 0.0019)\n",
      "Epoch 36/100, Batch 300, Loss: 0.0533 (Recon_x: 0.0530, Recon_s: 0.0001, Consistency: 0.0018)\n",
      "Epoch 36/100, Batch 400, Loss: 0.0537 (Recon_x: 0.0534, Recon_s: 0.0001, Consistency: 0.0018)\n",
      "\n",
      "======================================================================\n",
      "Epoch 36/100 Summary:\n",
      "  TRAIN - Total: 0.0529, Recon_x: 0.0526, Recon_s: 0.0001, Consistency: 0.0019\n",
      "  VAL   - Total: 0.0677, Recon_x: 0.0554, Recon_s: 0.0122, Consistency: 0.0018\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0677)\n",
      "======================================================================\n",
      "\n",
      "Epoch 37/100, Batch 0, Loss: 0.0557 (Recon_x: 0.0554, Recon_s: 0.0001, Consistency: 0.0018)\n",
      "Epoch 37/100, Batch 100, Loss: 0.0530 (Recon_x: 0.0527, Recon_s: 0.0001, Consistency: 0.0018)\n",
      "Epoch 37/100, Batch 200, Loss: 0.0550 (Recon_x: 0.0547, Recon_s: 0.0001, Consistency: 0.0018)\n",
      "Epoch 37/100, Batch 300, Loss: 0.0510 (Recon_x: 0.0507, Recon_s: 0.0001, Consistency: 0.0017)\n",
      "Epoch 37/100, Batch 400, Loss: 0.0499 (Recon_x: 0.0496, Recon_s: 0.0001, Consistency: 0.0017)\n",
      "\n",
      "======================================================================\n",
      "Epoch 37/100 Summary:\n",
      "  TRAIN - Total: 0.0526, Recon_x: 0.0523, Recon_s: 0.0001, Consistency: 0.0017\n",
      "  VAL   - Total: 0.0681, Recon_x: 0.0551, Recon_s: 0.0128, Consistency: 0.0017\n",
      "  No improvement for 1 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 38/100, Batch 0, Loss: 0.0526 (Recon_x: 0.0524, Recon_s: 0.0001, Consistency: 0.0017)\n",
      "Epoch 38/100, Batch 100, Loss: 0.0540 (Recon_x: 0.0537, Recon_s: 0.0002, Consistency: 0.0018)\n",
      "Epoch 38/100, Batch 200, Loss: 0.0539 (Recon_x: 0.0536, Recon_s: 0.0001, Consistency: 0.0017)\n",
      "Epoch 38/100, Batch 300, Loss: 0.0545 (Recon_x: 0.0541, Recon_s: 0.0002, Consistency: 0.0017)\n",
      "Epoch 38/100, Batch 400, Loss: 0.0489 (Recon_x: 0.0486, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "\n",
      "======================================================================\n",
      "Epoch 38/100 Summary:\n",
      "  TRAIN - Total: 0.0524, Recon_x: 0.0521, Recon_s: 0.0001, Consistency: 0.0017\n",
      "  VAL   - Total: 0.0682, Recon_x: 0.0548, Recon_s: 0.0132, Consistency: 0.0017\n",
      "  No improvement for 2 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 39/100, Batch 0, Loss: 0.0530 (Recon_x: 0.0526, Recon_s: 0.0002, Consistency: 0.0017)\n",
      "Epoch 39/100, Batch 100, Loss: 0.0614 (Recon_x: 0.0611, Recon_s: 0.0001, Consistency: 0.0017)\n",
      "Epoch 39/100, Batch 200, Loss: 0.0520 (Recon_x: 0.0518, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "Epoch 39/100, Batch 300, Loss: 0.0516 (Recon_x: 0.0513, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "Epoch 39/100, Batch 400, Loss: 0.0533 (Recon_x: 0.0530, Recon_s: 0.0001, Consistency: 0.0017)\n",
      "\n",
      "======================================================================\n",
      "Epoch 39/100 Summary:\n",
      "  TRAIN - Total: 0.0524, Recon_x: 0.0521, Recon_s: 0.0001, Consistency: 0.0017\n",
      "  VAL   - Total: 0.0676, Recon_x: 0.0547, Recon_s: 0.0127, Consistency: 0.0017\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0676)\n",
      "======================================================================\n",
      "\n",
      "Epoch 40/100, Batch 0, Loss: 0.0582 (Recon_x: 0.0579, Recon_s: 0.0001, Consistency: 0.0017)\n",
      "Epoch 40/100, Batch 100, Loss: 0.0496 (Recon_x: 0.0493, Recon_s: 0.0001, Consistency: 0.0017)\n",
      "Epoch 40/100, Batch 200, Loss: 0.0512 (Recon_x: 0.0510, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "Epoch 40/100, Batch 300, Loss: 0.0548 (Recon_x: 0.0545, Recon_s: 0.0001, Consistency: 0.0017)\n",
      "Epoch 40/100, Batch 400, Loss: 0.0526 (Recon_x: 0.0523, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "\n",
      "======================================================================\n",
      "Epoch 40/100 Summary:\n",
      "  TRAIN - Total: 0.0524, Recon_x: 0.0521, Recon_s: 0.0001, Consistency: 0.0016\n",
      "  VAL   - Total: 0.0676, Recon_x: 0.0545, Recon_s: 0.0129, Consistency: 0.0016\n",
      "  No improvement for 1 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 41/100, Batch 0, Loss: 0.0505 (Recon_x: 0.0502, Recon_s: 0.0001, Consistency: 0.0017)\n",
      "Epoch 41/100, Batch 100, Loss: 0.0494 (Recon_x: 0.0492, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "Epoch 41/100, Batch 200, Loss: 0.0483 (Recon_x: 0.0481, Recon_s: 0.0001, Consistency: 0.0017)\n",
      "Epoch 41/100, Batch 300, Loss: 0.0530 (Recon_x: 0.0527, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "Epoch 41/100, Batch 400, Loss: 0.0508 (Recon_x: 0.0506, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "\n",
      "======================================================================\n",
      "Epoch 41/100 Summary:\n",
      "  TRAIN - Total: 0.0522, Recon_x: 0.0519, Recon_s: 0.0001, Consistency: 0.0016\n",
      "  VAL   - Total: 0.0665, Recon_x: 0.0539, Recon_s: 0.0125, Consistency: 0.0016\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0665)\n",
      "======================================================================\n",
      "\n",
      "Epoch 42/100, Batch 0, Loss: 0.0560 (Recon_x: 0.0558, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 42/100, Batch 100, Loss: 0.0530 (Recon_x: 0.0528, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "Epoch 42/100, Batch 200, Loss: 0.0544 (Recon_x: 0.0541, Recon_s: 0.0002, Consistency: 0.0016)\n",
      "Epoch 42/100, Batch 300, Loss: 0.0491 (Recon_x: 0.0488, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "Epoch 42/100, Batch 400, Loss: 0.0520 (Recon_x: 0.0518, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "\n",
      "======================================================================\n",
      "Epoch 42/100 Summary:\n",
      "  TRAIN - Total: 0.0522, Recon_x: 0.0520, Recon_s: 0.0001, Consistency: 0.0016\n",
      "  VAL   - Total: 0.0667, Recon_x: 0.0546, Recon_s: 0.0119, Consistency: 0.0015\n",
      "  No improvement for 1 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 43/100, Batch 0, Loss: 0.0520 (Recon_x: 0.0517, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 43/100, Batch 100, Loss: 0.0504 (Recon_x: 0.0502, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "Epoch 43/100, Batch 200, Loss: 0.0531 (Recon_x: 0.0528, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 43/100, Batch 300, Loss: 0.0524 (Recon_x: 0.0522, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 43/100, Batch 400, Loss: 0.0537 (Recon_x: 0.0535, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "\n",
      "======================================================================\n",
      "Epoch 43/100 Summary:\n",
      "  TRAIN - Total: 0.0522, Recon_x: 0.0520, Recon_s: 0.0001, Consistency: 0.0016\n",
      "  VAL   - Total: 0.0663, Recon_x: 0.0539, Recon_s: 0.0122, Consistency: 0.0016\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0663)\n",
      "======================================================================\n",
      "\n",
      "Epoch 44/100, Batch 0, Loss: 0.0473 (Recon_x: 0.0470, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "Epoch 44/100, Batch 100, Loss: 0.0573 (Recon_x: 0.0570, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "Epoch 44/100, Batch 200, Loss: 0.0532 (Recon_x: 0.0530, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "Epoch 44/100, Batch 300, Loss: 0.0501 (Recon_x: 0.0498, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 44/100, Batch 400, Loss: 0.0523 (Recon_x: 0.0520, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "\n",
      "======================================================================\n",
      "Epoch 44/100 Summary:\n",
      "  TRAIN - Total: 0.0521, Recon_x: 0.0518, Recon_s: 0.0001, Consistency: 0.0015\n",
      "  VAL   - Total: 0.0672, Recon_x: 0.0542, Recon_s: 0.0129, Consistency: 0.0016\n",
      "  No improvement for 1 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 45/100, Batch 0, Loss: 0.0482 (Recon_x: 0.0480, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 45/100, Batch 100, Loss: 0.0531 (Recon_x: 0.0528, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "Epoch 45/100, Batch 200, Loss: 0.0528 (Recon_x: 0.0526, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 45/100, Batch 300, Loss: 0.0532 (Recon_x: 0.0529, Recon_s: 0.0001, Consistency: 0.0016)\n",
      "Epoch 45/100, Batch 400, Loss: 0.0519 (Recon_x: 0.0516, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "\n",
      "======================================================================\n",
      "Epoch 45/100 Summary:\n",
      "  TRAIN - Total: 0.0520, Recon_x: 0.0517, Recon_s: 0.0001, Consistency: 0.0015\n",
      "  VAL   - Total: 0.0656, Recon_x: 0.0532, Recon_s: 0.0123, Consistency: 0.0016\n",
      "  ✓ NEW BEST MODEL! (Val loss: 0.0656)\n",
      "======================================================================\n",
      "\n",
      "Epoch 46/100, Batch 0, Loss: 0.0496 (Recon_x: 0.0493, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 46/100, Batch 100, Loss: 0.0577 (Recon_x: 0.0575, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 46/100, Batch 200, Loss: 0.0530 (Recon_x: 0.0528, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 46/100, Batch 300, Loss: 0.0504 (Recon_x: 0.0501, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 46/100, Batch 400, Loss: 0.0547 (Recon_x: 0.0544, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "\n",
      "======================================================================\n",
      "Epoch 46/100 Summary:\n",
      "  TRAIN - Total: 0.0521, Recon_x: 0.0518, Recon_s: 0.0001, Consistency: 0.0015\n",
      "  VAL   - Total: 0.0670, Recon_x: 0.0548, Recon_s: 0.0121, Consistency: 0.0015\n",
      "  No improvement for 1 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 47/100, Batch 0, Loss: 0.0516 (Recon_x: 0.0514, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 47/100, Batch 100, Loss: 0.0520 (Recon_x: 0.0517, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "Epoch 47/100, Batch 200, Loss: 0.0513 (Recon_x: 0.0511, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "Epoch 47/100, Batch 300, Loss: 0.0557 (Recon_x: 0.0555, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 47/100, Batch 400, Loss: 0.0572 (Recon_x: 0.0569, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "\n",
      "======================================================================\n",
      "Epoch 47/100 Summary:\n",
      "  TRAIN - Total: 0.0521, Recon_x: 0.0518, Recon_s: 0.0001, Consistency: 0.0015\n",
      "  VAL   - Total: 0.0668, Recon_x: 0.0545, Recon_s: 0.0122, Consistency: 0.0015\n",
      "  No improvement for 2 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 48/100, Batch 0, Loss: 0.0496 (Recon_x: 0.0494, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "Epoch 48/100, Batch 100, Loss: 0.0484 (Recon_x: 0.0481, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "Epoch 48/100, Batch 200, Loss: 0.0542 (Recon_x: 0.0540, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 48/100, Batch 300, Loss: 0.0479 (Recon_x: 0.0477, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "Epoch 48/100, Batch 400, Loss: 0.0549 (Recon_x: 0.0547, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "\n",
      "======================================================================\n",
      "Epoch 48/100 Summary:\n",
      "  TRAIN - Total: 0.0520, Recon_x: 0.0517, Recon_s: 0.0001, Consistency: 0.0014\n",
      "  VAL   - Total: 0.0660, Recon_x: 0.0540, Recon_s: 0.0119, Consistency: 0.0015\n",
      "  No improvement for 3 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 49/100, Batch 0, Loss: 0.0518 (Recon_x: 0.0516, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "Epoch 49/100, Batch 100, Loss: 0.0550 (Recon_x: 0.0547, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 49/100, Batch 200, Loss: 0.0490 (Recon_x: 0.0488, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "Epoch 49/100, Batch 300, Loss: 0.0532 (Recon_x: 0.0529, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 49/100, Batch 400, Loss: 0.0580 (Recon_x: 0.0578, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "  → Learning rate reduced: 6.25e-05 → 3.13e-05\n",
      "\n",
      "======================================================================\n",
      "Epoch 49/100 Summary:\n",
      "  TRAIN - Total: 0.0520, Recon_x: 0.0517, Recon_s: 0.0001, Consistency: 0.0014\n",
      "  VAL   - Total: 0.0663, Recon_x: 0.0539, Recon_s: 0.0122, Consistency: 0.0015\n",
      "  No improvement for 4 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 50/100, Batch 0, Loss: 0.0490 (Recon_x: 0.0488, Recon_s: 0.0001, Consistency: 0.0015)\n",
      "Epoch 50/100, Batch 100, Loss: 0.0524 (Recon_x: 0.0522, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "Epoch 50/100, Batch 200, Loss: 0.0531 (Recon_x: 0.0528, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "Epoch 50/100, Batch 300, Loss: 0.0551 (Recon_x: 0.0549, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "Epoch 50/100, Batch 400, Loss: 0.0557 (Recon_x: 0.0554, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "\n",
      "======================================================================\n",
      "Epoch 50/100 Summary:\n",
      "  TRAIN - Total: 0.0516, Recon_x: 0.0514, Recon_s: 0.0001, Consistency: 0.0014\n",
      "  VAL   - Total: 0.0663, Recon_x: 0.0543, Recon_s: 0.0119, Consistency: 0.0014\n",
      "  No improvement for 5 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 51/100, Batch 0, Loss: 0.0552 (Recon_x: 0.0549, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "Epoch 51/100, Batch 100, Loss: 0.0523 (Recon_x: 0.0521, Recon_s: 0.0001, Consistency: 0.0013)\n",
      "Epoch 51/100, Batch 200, Loss: 0.0505 (Recon_x: 0.0503, Recon_s: 0.0001, Consistency: 0.0013)\n",
      "Epoch 51/100, Batch 300, Loss: 0.0514 (Recon_x: 0.0512, Recon_s: 0.0001, Consistency: 0.0013)\n",
      "Epoch 51/100, Batch 400, Loss: 0.0504 (Recon_x: 0.0502, Recon_s: 0.0001, Consistency: 0.0013)\n",
      "\n",
      "======================================================================\n",
      "Epoch 51/100 Summary:\n",
      "  TRAIN - Total: 0.0516, Recon_x: 0.0514, Recon_s: 0.0001, Consistency: 0.0013\n",
      "  VAL   - Total: 0.0657, Recon_x: 0.0531, Recon_s: 0.0125, Consistency: 0.0013\n",
      "  No improvement for 6 epoch(s)\n",
      "======================================================================\n",
      "\n",
      "Epoch 52/100, Batch 0, Loss: 0.0477 (Recon_x: 0.0474, Recon_s: 0.0001, Consistency: 0.0013)\n",
      "Epoch 52/100, Batch 100, Loss: 0.0527 (Recon_x: 0.0524, Recon_s: 0.0001, Consistency: 0.0013)\n",
      "Epoch 52/100, Batch 200, Loss: 0.0541 (Recon_x: 0.0539, Recon_s: 0.0001, Consistency: 0.0014)\n",
      "Epoch 52/100, Batch 300, Loss: 0.0491 (Recon_x: 0.0489, Recon_s: 0.0001, Consistency: 0.0013)\n",
      "Epoch 52/100, Batch 400, Loss: 0.0528 (Recon_x: 0.0525, Recon_s: 0.0001, Consistency: 0.0013)\n",
      "\n",
      "======================================================================\n",
      "Epoch 52/100 Summary:\n",
      "  TRAIN - Total: 0.0512, Recon_x: 0.0509, Recon_s: 0.0001, Consistency: 0.0013\n",
      "  VAL   - Total: 0.0660, Recon_x: 0.0534, Recon_s: 0.0125, Consistency: 0.0013\n",
      "  No improvement for 7 epoch(s)\n",
      "\n",
      "======================================================================\n",
      "EARLY STOPPING at epoch 52\n",
      "Best model from epoch 45 with val loss 0.0656\n",
      "======================================================================\n",
      "\n",
      "✓ Restored best model from epoch 45\n",
      "\n",
      "Training Complete!\n",
      "\n",
      "======================================================================\n",
      "FAIR TESTING - SPARSE INPUT ONLY (VISUAL)\n",
      "======================================================================\n",
      "\n",
      "Sample 0 (Digit 2):\n",
      "  Reconstruction MSE: 0.037976\n",
      "\n",
      "Sample 1 (Digit 2):\n",
      "  Reconstruction MSE: 0.038425\n",
      "\n",
      "Sample 2 (Digit 8):\n",
      "  Reconstruction MSE: 0.116548\n",
      "\n",
      "Sample 3 (Digit 4):\n",
      "  Reconstruction MSE: 0.041888\n",
      "\n",
      "Sample 4 (Digit 9):\n",
      "  Reconstruction MSE: 0.040457\n",
      "\n",
      "======================================================================\n",
      "AVERAGE RECONSTRUCTION MSE: 0.055059\n",
      "======================================================================\n",
      "\n",
      "Results saved to 'hybrid_reconstruction_result.png'\n"
     ]
    },
    {
     "data": {
      "image/png": "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q9uck0XlDz/8YCSZtWvXek13/1hd2fC4rVu3zoSEhJjo6GiTlJTkeN1WsqPx/v77782ZM2e8ph0/ftwUK1bM3HHHHT7xkyZNMmFhYT7z4MbRrl07U6xYMXPy5Emf144cOZLr+eRWG3gjFJWTJ082ISEhPt/PEydOeAqaiRMnZlpU/vHHH57CK6P1JyUlGZfLZb744gtjjDFpaWmmYsWKZtasWZ6YsWPHmtq1a3sVcflNbhWVR44cMf7+/uZ//ud/HM0PZzL6nX722WeNJLNkyZI8yixnZKW9yKl2NT8UlZ06dTJ33nmnz/To6GjTokULExERYaZOner12v79+42fn5/p1q2bz2emevXqpkaNGubixYs+y7z6N1GSGTRokM2tsmf8+PFGkvnuu+8803799Vfj7+9vRo0aZTl/27ZtTalSpUxiYqJn2ptvvmkkmVWrVmU6b0pKiqlVq5apWrWq1/TKlSub+vXre53oSExMNOHh4aZTp06eaQcOHDAFChTw2idpaWmmSZMmpmzZsiYlJcVruYMHDzbR0dFey80u1/XlrxMmTNC5c+c0b948lSpVyuf1gIAADR06VOXKlfNMc9//t3v3brVr104FCxbU/fffL0k6d+6cRowYoXLlyikoKEhVq1bVpEmTZIzxzJ/ZNedXX2bqvr9v165d6tevn6KiohQZGamHHnpISUlJXvNevHhRw4cPV7FixVSwYEF16tRJBw4cuMY95J3H9u3b1bt3bxUqVEh33nmnpMuXz6Z36dWV99Ps3btXxYoVkyQlJCRkeEntwYMHde+99yo8PFzFihXTU089pdTUVMv8PvzwQwUGBuquu+6yvU133323nn/+ee3bt08LFizw2dYrnT9/XkOHDlXRokU9+/bgwYM+23D1PZUVKlTQL7/8oi+++MKzzVfuq927d2v37t2WudatW9fnntMiRYqoSZMm+vXXX33iW7VqpXPnzmnNmjU29gSuR7t371aNGjUUFRXl81rx4sW9/ne5XBo8eLAWLlyoqlWrKjg4WHXr1tWXX37pFbdv3z4NHDhQVatWVUhIiIoUKaLu3bv7XILp/px/8cUXGjhwoIoXL66yZctKks6cOaNhw4apQoUKCgoKUvHixdWqVSv98MMPXsv49ttvdc899ygyMlKhoaFq2rSp4/sL3fl88803evLJJ1WsWDGFhYWpS5cuPvfIVahQQR06dNDq1atVu3ZtBQcHq3r16nr//fe94jK6tzqr3/H0fPjhh2rYsKHPd7pw4cIqWLCgrW0uV66c/Pwy//m9cOGCjDEqVKiQpMufg6ioKM9vx8GDB/XKK6/otddes1yW26RJk+RyubRv3z6f10aNGqXAwECdPHlSUvr3VP7v//6v6tatq4IFCyoiIkK33HKLXnvtNc/r6d1T6bZ582Y1btxYISEhqlixombNmmUr5x07dig2NlaFCxdWcHCw6tWrp+XLl/vEFS9eXLfeeqs++ugjW8tFzmrSpIkk+fxG2n0/T506peHDh3vaorJly6pPnz46fvy4J+bo0aPq37+/SpQooeDgYNWqVUvvvPOO13KuvOR+zpw5iomJUVBQkOrXr69NmzZ5xR4+fFgPPfSQypYtq6CgIJUqVUqdO3e21V5k1q5mdH9yRu3UggUL1KBBA4WGhqpQoUK66667tHr1assc3Ptt2LBhnn5spUqVNH78eKWlpfns3379+ikyMlJRUVHq27evTp065ZNLei5cuKCVK1dmeIlrcHCwunbt6nPZ6uLFi1WoUCG1adPGZ57du3erfv36CgwM9Hnt6t9Eu5KSkrRjxw6vz0xGli1bpvr166t+/fqeadWqVVOLFi20dOnSTOc9ffq01qxZowceeEARERGe6X369FF4eLjl/P7+/ipXrpzP/j99+rSKFy/u9RmJiIhQeHi4QkJCPNM++ugjJScna+DAgZ5pLpdLTzzxhA4cOKCNGzd6LbdVq1bat2+ftmzZkmleTlzXReUnn3yiSpUqqWHDhlmaLyUlRW3atFHx4sU1adIkdevWTcYYderUSVOmTNE999yjyZMnq2rVqnr66af15JNPXlOePXr00JkzZzRu3Dj16NFDb7/9thISErxiHnnkEU2dOlWtW7fWK6+8ogIFCqh9+/bXtN6rde/eXUlJSXr55Zf16KOP2p6vWLFimjlzpiSpS5cumj9/vubPn6+uXbt6YlJTU9WmTRsVKVJEkyZNUtOmTfXqq69qzpw5lsvfsGGDatasqQIFCmRpex588EFJ8jS2GenXr5+mT5+udu3aafz48QoJCbG1b6dOnaqyZcuqWrVqnm0ePXq05/UWLVqoRYsWWcr5SocPH1bRokV9plevXl0hISE39ENA/u6io6O1efNm/fzzz7biv/jiCw0bNkwPPPCAxowZoxMnTuiee+7xmn/Tpk3asGGDevXqpWnTpmnAgAFat26dmjVr5nMQS5IGDhyo7du364UXXtDIkSMlSQMGDNDMmTPVrVs3zZgxQ0899ZRCQkK8Dn6sX79ed911l06fPq24uDi9/PLLOnXqlO6++2599913jvfJkCFDtHXrVsXFxemJJ57Qxx9/7HWPtdvvv/+unj17qm3btho3bpwCAgLUvXt3RwdhrL7jV0tOTtamTZt02223ZXldWVWoUCHFxMTo5Zdf1p49e7Rw4UJt2bJFDRo0kCQ988wzatu2bZYOxvXo0UMulyvdTs7SpUvVunVrTxF7tTVr1ui+++5ToUKFNH78eL3yyitq1qyZrXbq5MmTateunerWrasJEyaobNmyeuKJJzR37txM5/vll190++2369dff9XIkSP16quvKiwsTPfee68++OADn/i6detqw4YNlvkg57kLsSs/T3bfz7Nnz6pJkyaaPn26Wrdurddee00DBgzQjh07PAfbz58/r2bNmmn+/Pm6//77NXHiREVGRqpfv35eBzrcFi1apIkTJ+rxxx/Xiy++qL1796pr165KTk72xHTr1k0ffPCBHnroIc2YMUNDhw7VmTNn9Mcff0iy116k165mRUJCgh588EEVKFBAY8aMUUJCgsqVK6f169db5pCUlKSmTZtqwYIF6tOnj6ZNm6Y77rhDo0aN8urHGmPUuXNnzZ8/Xw888IBefPFFHThwQH379rWV4+bNm3Xp0qVM28HevXvru+++8zqosGjRIsXGxqbb14uOjta6detsn0y5cOGCjh8/7vN36dIlT8x3332nm2++Wa+//nqmy0pLS9O2bdtUr149n9caNGig3bt3Z/oMj59++kkpKSk+8wcGBqp27dr68ccffeY5d+6cjh8/rt27d2vKlClasWKFT3+yWbNmWrlypaZPn669e/dqx44dGjRokBITE/WPf/zDE/fjjz8qLCxMN998s0/u7tevVLduXUnKmT5mtp/7zCWJiYlGkrn33nt9Xjt58qQ5duyY5+/KyyP79u1rJJmRI0d6zfPhhx8aSebFF1/0mh4bG2tcLpfZtWuXMSbzywN01eWhcXFxRpJ5+OGHveK6dOliihQp4vl/y5YtRpIZOHCgV1zv3r2z5fJXdx733XefT3zTpk3TvYyib9++Jjo62vO/1eWv+v/3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9PPPP6d7+j4j69ev19ixY706WOlxF5xXvz/Tp0+3tZ6wsLAMt9nukCLS5bNIS5Ys0YwZM7yempue7du368KFC2rcuLGtZeP64z6qeDX3/dVXX4a0ceNGr/si9+/fr48++kitW7f2HIH39/f3Web06dNtHw1PTU31uUy2ePHiKl26tOe7WbduXcXExGjSpEnp3npw9RAgOeHQoUNeT4o8ffq0/vWvf6l27dqey9Tcxe+V7fS5c+fSPaOW2Xf8agUKFFC9evXS7XzktJ07d+r111/Xa6+95ulUlihRwtOJlqRff/3V1qV63bp1k7+/vxYvXqx3331XHTp08Lk89Wru+6jc/Pz8dOutt0pK/7KvK6WkpGj27Nme/y9duqTZs2erWLFiXvcKX6l48eJq1qyZZs+enW6HMr3P2ubNm9WoUaNMc0HuaNasmRo0aKCpU6fqwoULWXo/u3Xr5rkl4GruNq5du3Y6fPiw5xJA6fLnbPr06QoPD8/0wG16kpKSPJcLusXExKhgwYJen++stBdXLicxMVHbtm3zTPvzzz99tu/ee++Vn5+fxowZ43N1yZVte0Y59OjRQxs3btSqVat8Xjt16pSnEGnXrp1SUlI8T/WXLrf/dvtFdevWVWBgoK128JFHHlFcXJzXwc70rFu3Lt3pGf0m2pGVIUViY2O1adMmr2367bfftH79enXv3t0rdseOHZ4nAkuXr5xs2bKlFixY4PWMj/nz5+vs2bNe87svQb5ScnKy/vWvfykkJMRTFFaqVEl+fn5asmSJ13t/4MABffXVV14HEzt37qwCBQp49XONMZo1a5bKlCnj05d038eeE33M6/pMZeXKlbVo0SLdd999qlq1qu6//37VqlVLxhjt2bNHixYtkp+fX6an5906duyo5s2ba/To0dq7d69q1aql1atX66OPPtKwYcO87nd85JFH9Morr+iRRx5RvXr19OWXX3qOujpRu3Zt3XfffZoxY4YSExPVuHFjrVu3zvbZtGvx8MMPa/LkyWrTpo369++vo0ePatasWapRo4bnQUKSPB/2JUuWqEqVKipcuLBq1qyZ6SWcdnXu3Fljx47VF198odatW/u8vmLFCu3YsUMpKSk6cuSI1q9frzVr1ig6OlrLly/33Iycnrp166pbt26aOnWqTpw4odtvv11ffPGF5/2yOupYt25dzZw5Uy+++KIqVaqk4sWL6+6775Ykz+OfrR7WM3XqVM2YMUONGjVSaGio17ia0uVhWq7s0K1Zs0ahoaFq1apVpsvF9WvIkCFKSkpSly5dVK1aNV26dEkbNmzQkiVLVKFCBT300ENe8TVr1lSbNm00dOhQBQUFeX48rhyaqEOHDpo/f74iIyNVvXp1bdy4UWvXrvVcqmTlzJkzKlu2rGJjY1WrVi2Fh4dr7dq12rRpk6dD4Ofnp7feektt27ZVjRo19NBDD6lMmTI6ePCgPvvsM0VERHjd/5ETqlSpov79+2vTpk0qUaKE5s6dqyNHjmjevHmemNatW6t8+fLq37+/nn76afn7+2vu3LkqVqyYV2dAyvw7np7OnTtr9OjROn36tNc9iImJiZ5Omfug3Ouvv66oqChFRUV5DY/y5ZdfegreY8eO6dy5c3rxxRclSXfddVe6w4QMHz5cPXv29LoqJzY2Vp07d9Zzzz0nSfr444/1ySefWO7D4sWLq3nz5po8ebLOnDmjnj17Ws7zyCOP6K+//tLdd9+tsmXLat++fZo+fbpq165tecS7dOnSGj9+vPbu3asqVapoyZIl2rJli+bMmZPpUFJvvPGG7rzzTt1yyy169NFHddNNN+nIkSPauHGjDhw4oK1bt3pijx49qm3btmnQoEGW24Lc8fTTT6t79+56++23NWDAANvv59NPP61ly5ape/fuevjhh1W3bl399ddfWr58uWbNmqVatWrpscce0+zZs9WvXz9t3rxZFSpU0LJly/TNN99o6tSptseMddu5c6datGihHj16qHr16goICNAHH3ygI0eOqFevXp64rLYXktSrVy89++yz6tKli4YOHaqkpCTNnDlTVapU8TpYWKlSJY0ePdrzkJquXbsqKChImzZtUunSpTVu3LhMc3j66ae1fPlydejQQf369VPdunV17tw5/fTTT1q2bJn27t2rokWLqmPHjrrjjjs0cuRI7d271zPWb3r33qcnODhYrVu31tq1azVmzJhMY6Ojoy0flCNdblcrVqyojh07KiYmRufOndPatWv18ccfq379+j5XUuzcudOnLyVdPtDm7jt99913at68ueLi4ixzGDhwoN588021b99eTz31lAoUKKDJkyerRIkSGjFihFfszTffrKZNm3qNx/vSSy+pcePGatq0qR577DEdOHBAr776qlq3bu113/njjz+u06dP66677lKZMmV0+PBhLVy4UDt27NCrr77qOXNcrFgxPfzww3rrrbfUokULde3aVWfOnNGMGTN0/vx5jRo1yrPMsmXLatiwYZo4caKSk5NVv359ffjhh/rqq6+0cOFCn0u/16xZo/Lly9u+pDtLsv15snlg165d5oknnjCVKlUywcHBJiQkxFSrVs0MGDDAbNmyxSu2b9++JiwsLN3lnDlzxgwfPtyULl3aFChQwFSuXNlMnDjR61HOxlx+fH7//v1NZGSkKViwoOnRo4c5evRohkOKHDt2zGv+9B5rf/78eTN06FBTpEgRExYWZjp27Gj279+frUOKXJ2H24IFC8xNN91kAgMDTe3atc2qVavSfQT2hg0bTN26dU1gYKBXXhnt04we65+eW2+91fTv399rmns/uf8CAwNNyZIlTatWrcxrr73meWS41TrPnTtnBg0aZAoXLmzCw8PNvffea3777Tcjybzyyis+67vyfTl8+LBp3769KViwoJHk9Rhvu0OKuIdcyejv6mEHGjZsaB544AHL5eL6tWLFCvPwww+batWqmfDwcBMYGGgqVapkhgwZ4jMcgiQzaNAgs2DBAlO5cmUTFBRk6tSp4/UdN+byI/EfeughU7RoURMeHm7atGljduzYYaKjo03fvn09cRkN1XPx4kXz9NNPm1q1apmCBQuasLAwU6tWLTNjxgyf/H/88UfTtWtXU6RIERMUFGSio6NNjx49zLp16zLd7syGFLk6n/SGp3A/rn7VqlXm1ltvNUFBQaZatWrm3Xff9VnX5s2bTcOGDU1gYKApX768mTx5cpa/4+k5cuSICQgIMPPnz09329L7u7qdcLdT6f2l197/+9//NuHh4ebQoUM+r40bN86ULl3alCpVyowfPz7T3K/05ptvGkmmYMGC5vz58z6vX/0bsGzZMtO6dWtTvHhxzz59/PHHzZ9//umJyWhIkRo1apjvv//eNGrUyAQHB5vo6Gjz+uuve60vo+G6du/ebfr06WNKlixpChQoYMqUKWM6dOhgli1b5hU3c+ZMExoamu7vAnJOZkN/paammpiYGBMTE+MZZsPu+3nixAkzePBgU6ZMGRMYGGjKli1r+vbta44fP+6JOXLkiKfNCwwMNLfccovP5ye9Nsftyu/b8ePHzaBBg0y1atVMWFiYiYyMNA0bNjRLly71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M2ea0aNHm6CgIHPPPfd4xf373/82bdq0MfHx8WbOnDlm6tSppkmTJkaSmT17tlfstGnTjCTTvn17M3PmTDNlyhRTq1YtI8m89957GW5H6dKlTVhYWLr98tTUVFOyZEnz/PPPZ2Hv2HddF5W7du0yYWFh5uabbzaHDh3yeT05Odm89tpr5o8//sh0OWfPns2pFK+ZVVF5tXfffdfWD3NWtzkni8offvjBSDJr1671mu7+sbqy4XFbt26dCQkJMdHR0SYpKcnxuq1kR+P9/fffmzNnznhNO378uClWrJi54447fOInTZpkwsLCfObBjaNdu3amWLFi5uTJkz6vHTlyJNfzya028EYoKidPnmxCQkJ8vp8nTpwwp0+fNsYYM3HixEyLyurVq5tatWqZ5ORkz7TRo0cbl8tlfv31V2OMMb/++quRZPbt22eMMebcuXMmJCTErFy50jNP//79TceOHW3nnhdyq6j8/vvvjSSzbt06R/PDmYx+p5999lkjySxZsiSPMssZWWkvcqpdzQ9FZadOncydd97pMz06Otq0aNHCREREmKlTp3q9tn//fuPn52e6devm85mpXr26qVGjRroHsq7+TZRkBg0aZHOr7Bk/fryRZL777jvPtF9//dX4+/ubUaNGWc7ftm1bU6pUKa+Dfm+++aaRZHnQLyUlxdSqVctUrVrVa3rlypVN/fr1vQ6UJSYmmvDwcNOpU6d0l/Xss8+aqlWrmvvvvz/DfvngwYNz7ADcdX3564QJE3Tu3DnNmzdPpUqV8nk9ICBAQ4cOVbly5TzT3Pf/7d69W+3atVPBggV1//33S5LOnTunESNGqFy5cgoKClLVqlU1adIkGWM882d2zfnVl5m67+/btWuX+vXrp6ioKEVGRuqhhx5SUlKS17wXL17U8OHDVaxYMRUsWFCdOnXSgQMHrnEPeeexfft29e7dW4UKFdKdd94pKeP7Zvr166cKFSp4trlYsWKSpISEhAwvqT148KDuvfdehYeHq1ixYnrqqaeUmppqmd+HH36owMBA3XXXXba36e6779bzzz+vffv2acGCBT7beqXz589r6NChKlq0qGffHjx40Gcbrr4fqEKFCvrll1/0xRdfeLb5yn21e/du7d692zLXunXr+lyGUKRIETVp0kS//vqrT3yrVq107tw5rVmzxsaewPVo9+7dqlGjhqKionxeK168uNf/LpdLgwcP1sKFC1W1alUFBwerbt26+vLLL73i9u3bp4EDB6pq1aoKCQlRkSJF1L17d59LMN2f8y+++EIDBw5U8eLFVbZsWUnSmTNnNGzYMFWoUEFBQUEqXry4WrVqpR9++MFrGd9++63uueceRUZGKjQ0VE2bNnV8f6E7n2+++UZPPvmkihUrprCwMHXp0kXHjh3zinXfn7d69WrVrl1bwcHBql69ut5//32vuIzurc7qdzw9H374oRo2bOjznS5cuLAKFixoub3bt2/X9u3b9dhjjykgIMAzfeDAgTLGaNmyZZIut1uSVKhQIUlSaGioQkJCPL8dP/zwgxYuXKjJkydbrtNt8ODBCg8P9/n9kaT77rtPJUuW9LTZ6f02TJ8+XTVq1FBoaKgKFSqkevXqeV2altl96VbvWUbsftbq1q2rwoUL66OPPrK1XOSsJk2aSJLPb+SOHTsUGxurwoULKzg4WPXq1dPy5ct95j916pSGDx/uaYvKli2rPn366Pjx456Yo0ePqn///ipRooSCg4NVq1YtvfPOO17LufKS+zlz5igmJkZBQUGqX7++Nm3a5BV7+PBhPfTQQypbtqyCgoJUqlQpde7c2VZ7kVm7emV/6koZtVMLFixQgwYNPN+zu+66S6tXr7bMwb3fhg0b5unHVqpUSePHj1daWprP/u3Xr58iIyMVFRWlvn376tSpUz65pOfChQtauXJlhpe4BgcHq2vXrj6XrS5evFiFChVSmzZtfObZvXu36tevr8DAQJ/Xrv5NtCspKUk7duzw+sxkZNmyZapfv77q16/vmVatWjW1aNFCS5cuzXTe06dPa82aNXrggQcUERHhmd6nTx+Fh4dbzu/v769y5cr57P/Tp0+rePHiXp+RiIgIhYeHKyQkxGc5v//+u6ZMmaLJkyd7/bZcrVWrVtq3b5+2bNmSaV5OXNdF5SeffKJKlSqpYcOGWZovJSVFbdq0UfHixTVp0iR169ZNxhh16tRJU6ZM0T333KPJkyeratWqevrpp/Xkk09eU549evTQmTNnNG7cOPXo0UNvv/22EhISvGIeeeQRTZ06Va1bt9Yrr7yiAgUKqH379te03qt1795dSUlJevnll/Xoo4/anq9YsWKaOXOmJKlLly6aP3++5s+fr65du3piUlNT1aZNGxUpUkSTJk1S06ZN9eqrr2rOnDmWy9+wYYNq1qypAgUKZGl7HnzwQUnyNLYZ6devn6ZPn6527dpp/PjxCgkJsbVvp06dqrJly6patWqebR49erTn9RYtWqhFixZZyvlKhw8fVtGiRX2mV69eXSEhITf0Q0D+7qKjo7V582b9/PPPtuK/+OILDRs2TA888IDGjBmjEydO6J577vGaf9OmTdqwYYN69eqladOmacCAAVq3bp2aNWuWbhExcOBAbd++XS+88IJGjhwpSRowYIBmzpypbt26acaMGXrqqacUEhLidfBj/fr1uuuuu3T69GnFxcXp5Zdf1qlTp3T33Xfru+++c7xPhgwZoq1btyouLk5PPPGEPv74Y697rN1+//139ezZU23bttW4ceMUEBCg7t27OzoIY/Udv1pycrI2bdqk2267Lcvrcvvxxx8lSfXq1fOaXrp0aZUtW9bzepUqVRQZGan4+Hjt27dPEydO1OnTpz3rHjp0qAYPHqxKlSrZXnfPnj117tw5n3tykpKS9PHHHys2Nlb+/v7pzvvmm29q6NChql69uqZOnaqEhATVrl1b3377reV6nb5nWf2s3XbbbbSb+YS7EHMfFJEu3092++2369dff9XIkSP16quvKiwsTPfee68++OADT9zZs2fVpEkTTZ8+Xa1bt9Zrr72mAQMGaMeOHZ6D7efPn1ezZs00f/583X///Zo4caIiIyPVr18/vfbaaz75LFq0SBMnTtTjjz+uF198UXv37lXXrl2VnJzsienWrZs++OADPfTQQ5oxY4aGDh2qM2fO6I8//pBkr71Ir13NioSEBD344IMqUKCAxowZo4SEBJUrV07r16+3zCEpKUlNmzbVggUL1KdPH02bNk133HGHRo0a5dWPNcaoc+fOmj9/vh544AG9+OKLOnDggPr27Wsrx82bN+vSpUuZtoO9e/fWd99953VQYdGiRYqNjU23rxcdHa1169bZPply4cIFHT9+3Ofv0qVLnpjvvvtON998s15//fVMl5WWlqZt27b5tMmS1KBBA+3evTvTZ3j89NNPSklJ8Zk/MDBQtWvX9rTpVzp37pyOHz+u3bt3a8qUKVqxYoVPf7JZs2ZauXKlpk+frr1792rHjh0aNGiQEhMT9Y9//MNnmcOGDVPz5s3Vrl27TLe3bt26kpQzbWW2n/vMJYmJiUaSuffee31eO3nypDl27Jjn78rLI/v27WskmZEjR3rN8+GHHxpJ5sUXX/SaHhsba1wul9m1a5cxJvPLA3TV5aFxcXFGknn44Ye94rp06WKKFCni+X/Lli1Gkhk4cKBXXO/evbPl8ld3Hvfdd59PfNOmTdO9jKJv374mOjra87/V5a/6//dUXKlOnTqmbt26ljmXLVvWdOvWzWd6Zpe/ukVGRpo6dep4/ndvq9vmzZuNJDNs2DCv+fr16+ezPe71XXnJWmaXmURHR3vto6z48ssvjcvlyvC69ipVqpi2bds6Wjbyv9WrVxt/f3/j7+9vGjVqZJ555hmzatUqc+nSJZ9YSUaS+f777z3T9u3bZ4KDg02XLl0809K7DHzjxo1GkvnXv/7lmeb+nN95550mJSXFKz4yMjLTy4rS0tJM5cqVTZs2bbwunUlKSjIVK1Y0rVq1ynS707v81Z1Py5YtvZY5fPhw4+/vb06dOuWZFh0d7XM/SWJioilVqlSm7cDV67L7Hb/arl27jCQzffr0TOMyu/zV/Vp6t2XUr1/f3H777Z7/Fy1aZEJCQowk4+/vbyZNmmSMMWbhwoWmRIkSXpda2ZGWlmbKlCnj094uXbrUSDJffvmlZ9rVvw2dO3c2NWrUyHT56e1fu+/Z1Ze/OvmsPfbYYyYkJCTTHJG93O/52rVrzbFjx8z+/fvNsmXLTLFixUxQUJDZv3+/J7ZFixbmlltuMRcuXPBMS0tLM40bNzaVK1f2THvhhReMJPP+++/7rM/9WZg6daqRZBYsWOB57dKlS6ZRo0YmPDzccym6u80pUqSI+euvvzyxH330kZFkPv74Y2PM5X7j1W1TejJqLzJrV6/uT7ld3U79/vvvxs/Pz3Tp0sWkpqamu92Z5TB27FgTFhZmdu7c6TV95MiRxt/f39PmuPu7EyZM8MSkpKR47u2zuvz1rbfeMpLMTz/95POa+3L3lJQUU7JkSTN27FhjjDHbt283kswXX3yRbt/uf/7nf4wkExgYaJo3b26ef/5589VXX/nsB2P++5uY3t/ixYs9ce42xaoP7e7fXt2HNcaYN954w0gyO3bsyHB+d7/7yvbTrXv37qZkyZI+0x9//HFPzn5+fiY2Ntbr82nM5ct+W7Ro4bV9RYsWNRs2bPBZ3ieffGICAgLML7/8Yoyxvi0tMDDQPPHEExm+7tR1e6by9OnTkpTu042aNWumYsWKef7eeOMNn5gnnnjC6/9PP/1U/v7+Gjp0qNf0ESNGyBijFStWOM716qdRNWnSRCdOnPBsw6effipJPuseNmyY43XaySO7pbeddp6cdeLECa+jmVkRHh6e6RGklStXSrp89PBKQ4YMcbS+K+3du9fR8CNHjx5V7969VbFixQwf7VyoUCFbl2zg+tSqVStt3LhRnTp10tatWzVhwgS1adNGZcqUSfdSsEaNGnmOLkpS+fLl1blzZ61atcpzueKVl8MkJyfrxIkTqlSpkqKionwuX5WkRx991OesVFRUlL799lsdOnQo3by3bNmi33//Xb1799aJEyc8R4fPnTunFi1a6Msvv/S5zMquxx57zOsynyZNmig1NVX79u3ziitdurS6dOni+T8iIkJ9+vTRjz/+mHNPtPv/Tpw4IUmO2yvpv5e1BgUF+bwWHBzseV26fEnqwYMHtXHjRh08eFAjRoxQUlKSnn32Wb300ksKDw9XQkKCbrrpJt16661eZ3vS43K51L17d3366adeTyVcsmSJypQp47ktIj1RUVE6cOCAzyWDdjh5z5x81goVKqTz58+ne2YeOatly5YqVqyYypUrp9jYWIWFhWn58uWeS0D/+usvrV+/3nPllvv9PHHihNq0aaPff//d87TY9957T7Vq1fL6zLi524hPP/1UJUuW1H333ed5rUCBAp4nbn7xxRde8/Xs2dPre+u+PNfdRwkJCVFgYKA+//xznTx50vF+SK9dtevDDz9UWlqaXnjhBfn5eXfP7QyV9u6776pJkyae/oP7r2XLlkpNTfXcMvHpp58qICDAqx/s7+9vu19kpx309/dXjx49tHjxYkmXn0harlw5z36/2sMPP6yVK1eqWbNm+vrrrzV27Fg1adJElStX1oYNG3ziO3furDVr1vj8NW/e3BPTrFkzGWMsRz+wapOvjHEyf3rzDhs2TGvWrNE777yjtm3bKjU11essq3T5loeqVauqb9++evfddzV37lyVKlVKXbt21a5duzxxly5d0vDhwzVgwABVr1490211y6k+ZsYX3eZz7ntX0hsfZ/bs2Tpz5oyOHDmiBx54wOf1gIAAT0Pntm/fPpUuXdrnnpibb77Z87pT5cuX9/rf/UU8efKkIiIitG/fPvn5+SkmJsYrrmrVqo7XmZ6KFStm6/KuFBwc7Lnv0q1QoUK2G2dzxX2rWXH27NlMr7d379urtz0rl4xlp3PnzqlDhw46c+aMvv766wzH9zTGOB5vE9eH+vXr6/3339elS5e0detWffDBB5oyZYpiY2O1ZcsWrx+HypUr+8xfpUoVJSUl6dixYypZsqTOnz+vcePGad68eTp48KDXdyoxMdFn/vTagwkTJqhv374qV66c6tatq3bt2qlPnz666aabJF2+jFFSppdJJSYmOiq6Mmsnr1SpUiWf70aVKlUkXT7QU7JkySyvO6uctlfSf4v/9MZUvHDhgs+9MoUKFdLtt9/u+X/cuHEqXry4HnroIc2dO1ezZs3SwoULtXfvXvXs2VPbt2/PtH3r2bOnpk6dquXLl6t37946e/asPv30Uz3++OOZtjnPPvus1q5dqwYNGqhSpUpq3bq1evfu7TWsSkacvGdOPmvu94W2M/e98cYbqlKlihITEzV37lx9+eWXXp3sXbt2yRij559/Xs8//3y6yzh69KjKlCmj3bt3q1u3bpmub9++fapcubJP8ZVRn82qfQkKCtL48eM1YsQIlShRQrfffrs6dOigPn36ZKlNuZZ+1u7du+Xn52e7MLja77//rm3btvn0xdyOHj0q6fK+KVWqlE//I6t9Tqt2sHfv3po2bZq2bt2qRYsWqVevXpl+N9u0aaM2bdooKSlJmzdv1pIlSzRr1ix16NBBO3bs8OrrlS1bNsN7OrPKqk2+MsbJ/OnNW61aNVWrVk3S5XsvW7durY4dO+rbb7/17KPu3bsrICDAa7iRzp07q3Llyho9erSWLFkiSZoyZYqOHz/uc1tdZnKqj3ndFpWRkZEqVapUuvckue+xzOgsUlBQkE9DZFdGb0JmD6TJ6KjVtXRMnEjvg+1yudLNw84Ddq7k9MicdPmhNU6ODB44cECJiYl5ViBm1aVLl9S1a1dt27ZNq1atUs2aNTOMPXnyZLqFBG48gYGBngcEVKlSRQ899JDeffddxcXFZWk5Q4YM0bx58zRs2DA1atRIkZGRcrlc6tWrV7pnD9NrD3r06KEmTZrogw8+0OrVqzVx4kSNHz9e77//vtq2betZzsSJE1W7du1088joQImV7GwnnbTTdhQpUkSSb6GbFe6Hyv35559eD5FzT2vQoEGG8+7du1evvvqqVq9eLT8/Py1evFiPP/647r77bknSO++8o//93//VP//5zwyXcfvtt6tC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y9W5c2ft2rVLERERkqSjR49qzJgxWrp0qQICsvfnfPXq1VmeJ702NDcEBwerb9++mjx5soYMGeL4ahU45/6dvnDhgv7v//5Pb7/9tr7++mv9/PPPCg4Ozuv0sk129AluhBw+/vhj/fbbb3rsscd8XgsODtZnn32mw4cPq2TJkl6vLVy4UMHBwbpw4YLX9HfffVc9e/ZU7dq19Y9//EOFChXSnj179OWXX+rNN99U7969veJbtWqlPn36+Ky7SpUqjrbn7Nmzat68uRITE/Xcc8+pQIECmjJlipo2baotW7aoSJEimc6/aNEiLVmyRLfddptKly6daew///lPlSxZUnXq1NGqVasyjY2Pj9eYMWMUGxurfv36KTk5WT///HO6B2zS2yd16tTx+b9evXp69dVX9a9//SvTdTtirmO7du0yYWFh5uabbzaHDh3yeT05Odm89tpr5o8//sh0OWfPns2pFK+ZJBMXF2c7/t133zWSzGeffZZpXFa3+dixYxnm0rdvXxMWFpal5V3phx9+MJLM2rVrvabPmzfPSDKbNm3ymWfdunUmJCTEREdHm6SkJMfrtlKjRg3TtGnTa1rGwoULjSTzySefeE1/4YUXjCTzww8/eE0fPHiwiY6ONmlpade0XuRf7dq1M8WKFTMnT570ee3IkSO5nk9utYF79uwxkszEiRNzZX12ZPU7PnnyZBMSEmLOnDnjNf3EiRPm9OnTxhhjJk6caCSZPXv2pLuM6tWrm1r/j707jY+iSv+//+3sG4QACVsgaNgEFBABQZAgArKDBFBcAHFBtsEVHEaTAIoIIoqCO86A+EcZQXGUfZsRRhDFDVDIsKtssiasybkfcHf/6HQnVSmyEPy8X688SPVVdU5VV58+V9epOg0amHPnznmWjRkzxrhcLrNlyxZjjDFbtmwxksyuXbuMMcZkZGSY8PBws2jRIs86gwYNMl27drVd9+KQkJBgOnfubBm3cuVKW99dufn666+NJLN8+XJH68OZ3L6nR40aZSSZuXPnFlPNCkd+2ovCalcLol/ij7t9njlzpmVst27dTMuWLX2WJyQkmLZt25rSpUubqVOner22Z88eExAQYHr16uVzztStW9fUq1fPnDlzxmebOb8TJZmhQ4fa3Ct7Jk6caCSZ9evXe5Zt2bLFBAYGmqeeespy/X379pmzZ88aY4zp3LmzSUhIyDXW/b2QV7/aGGPWrVtnXC6XmTJlimX5+TkmkydPNpGRkT7fYQWhRA9/feGFF5SRkaGZM2eqUqVKPq8HBQVpxIgRqlq1qmeZ+/6/9PR0derUSaVKldJdd90lScrIyNBjjz2mqlWrKjQ0VLVr19bkyZN14f26IK8x5zkvY7vv79u+fbsGDBigMmXKKDo6WgMHDlRmZqbXumfOnNEjjzyi2NhYlSpVSt26ddPevXsv8Qh512Pz5s3q16+fYmJi1LJlS0m53zczYMAAz+X7nTt3KjY2VpKUlpaW65Daffv2qUePHoqKilJsbKwef/xxZWVlWdZvwYIFCgkJ0c0332x7n2655RY9/fTT2rVrl2bPnu2zrxc7deqURowYofLly3uO7b59+3z2Ief9QNWrV9dPP/2k1atXe/b54mOVnp6u9PR0y7oeP35cklShQgWv5e5zNjw83Gt5u3bttGvXrjyvdKBkS09PV7169VSmTBmf1+Li4rz+d7lcGjZsmN5//33Vrl1bYWFhaty4sdasWeMVt2vXLg0ZMkS1a9dWeHi4ypUrp969e/sMwXSf56tXr9aQIUMUFxen+Ph4SdKJEyc0cuRIVa9eXaGhoYqLi1O7du30zTffeG3jq6++0m233abo6GhFRESodevWju8vdNfnyy+/1KOPPqrY2FhFRkaqZ8+eOnjwoFes+/68JUuWqGHDhgoLC1PdunX18ccfe8Xldm91fj/j/ixYsEDNmjXzuY+8bNmyKlWqlOX+bt68WZs3b9aDDz7odXVxyJAhMsZo3rx5ki60W5IUExMjSYqIiFB4eLjnu+Obb77R+++/rylTpliW6TZs2DBFRUX5fP9I0p133qmKFSt62mx/3w3Tpk1TvXr1FBERoZiYGN1www1eQ9Pyui/d6j3Ljd1zrXHjxipbtqw++eQTW9tF4WrVqpUk+XxHbt26VcnJySpbtqzCwsJ0ww036NNPP/VZ/+jRo3rkkUc8bVF8fLzuvfdeHTp0yBNz4MABDRo0SBUqVFBYWJgaNGjgM4Lg4iH3b775phITExUaGqomTZpow4YNXrG///67Bg4cqPj4eIWGhqpSpUrq3r27rfYir3b14v7UxXJrp2bPnq2mTZt6Pmc333yzZ+SAVZt19OhRjRw50tOPrVGjhiZOnKjs7Gyf4ztgwABFR0erTJky6t+/v44ePepTF39Onz6tRYsW5TrENSwsTLfffrvPsNUPPvhAMTEx6tChg8866enpatKkiUJCQnxey/mdaFdmZqa2bt3qdc7kZt68eWrSpImaNGniWVanTh21bdtWH374oeX6lStXVnBwsK165TU09mJTp05VxYoV9Ze//EXGGFv3rp46dcrnKnBO7dq1U0ZGhpYuXWqrHvlRopPKzz77TDVq1FCzZs3ytd758+fVoUMHxcXFafLkyerVq5eMMerWrZteeukl3XbbbZoyZYpq166tJ554Qo8++ugl1bNPnz46ceKEJkyYoD59+ui9997zDCV1u//++zV16lS1b99ezz//vIKDg9W5c+dLKjen3r17KzMzU88995weeOAB2+vFxsZ6hmT17NlTs2bN0qxZs3T77bd7YrKystShQweVK1dOkydPVuvWrfXiiy/qzTfftNz+2rVrVb9+fdsfSLd77rlHkvUwrQEDBmjatGnq1KmTJk6cqPDwcFvHdurUqYqPj1edOnU8+zxmzBjP623btlXbtm0tt3PzzTcrICBAf/nLX/Tf//5Xe/fu1eeff65nn31WPXr0UJ06dbziGzduLElX9ENA/uwSEhK0ceNG/fjjj7biV69erZEjR+ruu+/W2LFjdfjwYd12221e62/YsEFr167VHXfcoVdeeUWDBw/W8uXLlZSU5DeJGDJkiDZv3qxnnnnGM2R88ODBmjFjhnr16qXp06fr8ccfV3h4uLZs2eJZb8WKFbr55pt1/PhxpaSk6LnnntPRo0d1yy23aP369Y6PyfDhw/Xdd98pJSVFDz/8sBYuXOh1j7Xbtm3b1LdvX3Xs2FETJkxQUFCQevfu7egL0uozntO5c+e0YcMGXX/99fkuy+3bb7+VJN1www1eyytXrqz4+HjP67Vq1VJ0dLRSU1O1a9cuTZo0ScePH/eUPWLECA0bNkw1atSwXXbfvn2VkZGhf/3rX17LMzMztXDhQiUnJyswMNDvum+99ZZGjBihunXraurUqUpLS1PDhg311VdfWZbr9D3L77l2/fXX025eJtyJmPtHEUn66aefdOONN2rLli0aPXq0XnzxRUVGRqpHjx6aP3++J+7kyZNq1aqVpk2bpvbt2+vll1/W4MGDtXXrVs+P7adOnVJSUpJmzZqlu+66S5MmTVJ0dLQGDBigl19+2ac+c+bM0aRJk/TQQw9p/Pjx2rlzp26//XadO3fOE9OrVy/Nnz9fAwcO1PTp0zVixAidOHFCu3fvlmSvvfDXruZHWlqa7rnnHgUHB2vs2LFKS0tT1apVPffQ5VWHzMxMtW7dWrNnz9a9996rV155RTfddJOeeuopr36sMUbdu3fXrFmzdPfdd2v8+PHau3ev+vfvb6uOGzdu1NmzZ/NsB/v166f169d7/agwZ84cJScn++3rJSQkaPny5bYvppw+fVqHDh3y+Tt79qwnZv369brmmmv06quv5rmt7Oxsff/99z5tsiQ1bdpU6enpRfIMj5yWL1+uJk2a6JVXXvFccKpUqVKu+/Pee+8pMjJS4eHhqlu3rk9S71a3bl2Fh4cXTltZ4Nc+i8ixY8eMJNOjRw+f144cOWIOHjzo+bt4eGT//v2NJDN69GivdRYsWGAkmfHjx3stT05ONi6Xy2zfvt0Yk/fwAOW4jJ2SkmIkmfvuu88rrmfPnqZcuXKe/zdt2mQkmSFDhnjF9evXr0CGv7rrceedd/rEt27d2u8wiv79+3tdvrca/irJjB071mt5o0aNTOPGjS3rHB8fb3r16uWzPK/hr27R0dGmUaNGnv/d++q2ceNGI8mMHDnSa70BAwb47I+7vIuHrOU1zCQhISHPIQ4Xe/vtt02ZMmWMJM9f//79vYa/XSwkJMQ8/PDDtraNkmfJkiUmMDDQBAYGmubNm5snn3zSLF682DN85mLu8+Xrr7/2LNu1a5cJCwszPXv29CzzNwx83bp1RpL5xz/+4VnmPs9btmxpzp8/7xUfHR2d5xCa7OxsU7NmTdOhQwev4dmZmZnmqquuMu3atctzv/0Nf3XX59Zbb/Xa5iOPPGICAwPN0aNHPcsSEhKMJPPPf/7Ts+zYsWOmUqVKebYDOcuy+xnPafv27UaSmTZtWp5xeQ1/db/m77aMJk2amBtvvNHz/5w5c0x4eLiRZAIDA83kyZONMReG1FeoUMEcO3bMVr3dsrOzTZUqVXza2w8//NBIMmvWrPEsy/nd0L17d1OvXr08t+/v+Np9z3IOf3Vyrj344IMmPDw8zzqiYLnf82XLlpmDBw+aPXv2mHnz5pnY2FgTGhpq9uzZ44lt27atufbaa83p06c9y7Kzs02LFi1MzZo1Pcvct4Z8/PHHPuW5z4WpU6caSWb27Nme186ePWuaN29uoqKiPEPR3W1OuXLlzB9//OGJ/eSTT4wks3DhQmPMhX5jzrbJn9zai7za1Zz9Kbec7dS2bdtMQECA6dmzp8nKyvK733nVYdy4cSYyMtL88ssvXstHjx5tAgMDPW2Ou7/7wgsveGLOnz9vWrVqZWv469tvv20kmR9++MHnNfdw9/Pnz5uKFSuacePGGWOM2bx5s5FkVq9e7bdv98477xhJJiQkxLRp08Y8/fTT5t///rfPcTDm/74T/f198MEHnjh3m2LVh3b3b3P2YY0x5rXXXjOSzNatW/PcxsWshr/mLNdf/f744w/PeRsVFWUmTZpk5s6da2677TYjybz++ute8S1atDBTp041n3zyiZkxY4apX7++kWSmT5/ut+xatWqZjh072t4nu0rslUr3kEJ/U1kkJSUpNjbW8/faa6/5xDz88MNe/3/++ecKDAzUiBEjvJY/9thjMsboiy++cFzXnE+jatWqlQ4fPuzZh88//1ySfMoeOXKk4zLt1KOg+dtPO0/OOnz4sNevmfkRFRWV5y9IixYtknTh18OLFcSDUHbu3Gl7+pEqVaqoadOmmjp1qubPn69HH31U77//fq6/ZMbExNgasoGSqV27dlq3bp26deum7777Ti+88II6dOigKlWq+B0K1rx5c88VbEmqVq2aunfvrsWLF3uGK148jPrcuXM6fPiwatSooTJlyvgMX5WkBx54wOeqVJkyZfTVV1/p119/9VvvTZs2adu2berXr58OHz7s+XU4IyNDbdu21Zo1a3yGWdn14IMPeg0Fa9WqlbKysrRr1y6vuMqVK6tnz56e/0uXLq17771X3377baE/Ndn9tGan7ZX0f8Na/T3MJiwszPO6dGFI6r59+7Ru3Trt27dPjz32mDIzMzVq1Cg9++yzioqKUlpamq6++mpdd911Xld7/HG5XOrdu7c+//xzr6FUc+fOVZUqVTy3RfhTpkwZ7d2712fIoB1O3jMn51pMTIxOnTrl98o8Ctett96q2NhYVa1aVcnJyYqMjNSnn37qGQL6xx9/aMWKFZ6RW+738/Dhw+rQoYO2bdvmefjIP//5TzVo0MDrnHFztxGff/65KlasqDvvvNPzWnBwsEaMGKGTJ09q9erVXuv17dvX63PrHp7r7qOEh4crJCREq1at0pEjRxwfB3/tql0LFixQdna2nnnmGc/D+9zsPHzqo48+UqtWrTz9B/ffrbfeqqysLM8tE59//rmCgoK8+sGBgYG2+0V22sHAwED16dNHH3zwgaQLD+ipWrWq57jndN9992nRokVKSkrSf/7zH40bN06tWrVSzZo1tXbtWp/47t27a+nSpT5/bdq08cQkJSXJGGP5dFWrNvnimKLibp8PHz6st99+W48//rj69Omjf/3rX6pbt67Gjx/vFf/ll1/qL3/5i7p166bBgwdr48aNql+/vv7617/6rXth9TFL7NNf3feu+Btj/MYbb+jEiRPav3+/7r77bp/Xg4KCPA2d265du1S5cmWfe2LcT1DN2bHJj2rVqnn97/4gHjlyRKVLl9auXbsUEBCgxMREr7jatWs7LtOfq666qkC3d7GwsDDPfZduMTExthtnc9F9q/lx8uTJPMfbu49tzn3Pz5CxS/Xll1+qS5cu+u9//+sZXtGjRw+VLl1aaWlpuu+++1S3bl2vdYwxPMHwCtekSRN9/PHHOnv2rL777jvNnz9fL730kpKTk7Vp0yavc6JmzZo+69eqVUuZmZk6ePCgKlasqFOnTmnChAmaOXOm9u3b5/WZOnbsmM/6/tqDF154Qf3791fVqlXVuHFjderUSffee6+uvvpqSReGMUrKc5jUsWPHHCVdebWTF6tRo4bPZ8P9xL+dO3f6PG2wMDhtr6T/S/79zal4+vRpn3usY2JidOONN3r+nzBhguLi4jRw4EC9++67ev311/X+++9r586d6tu3rzZv3pxn+9a3b19NnTpVn376qfr166eTJ0/q888/10MPPZRnmzNq1CgtW7ZMTZs2VY0aNdS+fXv169fPa1qV3Dh5z5yca+73hbaz6L322muqVauWjh07pnfffVdr1qzx6qRv375dxhg9/fTTevrpp/1u48CBA6pSpYrS09PVq1evPMvbtWuXatas6ZN85dZns2pfQkNDNXHiRD322GOqUKGCbrzxRnXp0kX33ntvvtqUS+lnpaenKyAgwKc/YNe2bdv0/fff+/TF3A4cOCDpwrGpVKmSz0WZ/PY5rdrBfv366ZVXXtF3332nOXPm6I477sjzs9mhQwd16NBBmZmZ2rhxo+bOnavXX39dXbp00datW736evHx8bne05lfVm3yxTFFxV1ecHCwkpOTPcsDAgLUt29fpaSkaPfu3T7ntVtISIiGDRvmSTBz/mBYWH3MEptURkdHq1KlSn7vSXLfY5nbVaTQ0FCfhsiu3N6EvB5Ik9uvVpfSMXHC34fC5XL5rYedB+xczOkvc5JUrlw5R78M7t27V8eOHSvSBNGJN954QxUqVPAZr9+tWzelpqZq7dq1Pl8iR48eVfny5YuymigmISEhngcE1KpVSwMHDtRHH32klJSUfG1n+PDhmjlzpkaOHKnmzZsrOjpaLpdLd9xxh9+rh/7agz59+qhVq1aaP3++lixZokmTJmnixIn6+OOP1bFjR892Jk2apIYNG/qth7/RI3YUZDvppJ22w/1Y+Uu5kuF+QNdvv/3m9RA597KmTZvmuu7OnTv14osvasmSJQoICNAHH3yghx56SLfccosk6e9//7v+3//7f/rb3/6W6zZuvPFGVa9eXR9++KH69eunhQsX6tSpU+rbt2+e9b7mmmv0888/67PPPtOiRYv0z3/+U9OnT9czzzzj84yAguDkXDty5IjngUYoWk2bNvX60bRly5bq16+ffv75Z0VFRXnez8cff9zvg1qkwv2x1077MnLkSHXt2lULFizQ4sWL9fTTT2vChAlasWKFz9QMucmtn+XPpbZHOWVnZ6tdu3Z68skn/b7udLqNnC5uB3NeoLlYs2bNlJiYqJEjR2rHjh0+04LkJiIiQq1atVKrVq1Uvnx5paWl6YsvvrB9z2d+lS1bVqGhofrtt998XnMvs5ompDDqFBYWpjJlyvicu+7k+siRI7kmlZI83y9//PGHz2tHjhzx+2P1pSqxSaUkde7cWW+//bbWr1+f5xexHQkJCVq2bJlOnDjhdbVy69atntel//t1K+dTsi7lSmZCQoKys7OVnp7u9UvRzz//7HibdsXExPgdoppzfwrzl986depox44d+V5v1qxZkpTrF5T0f8d2x44dXh+g7du32yqjIPZ7//79fr883A8IyDlp+b59+3T27FmveUbx5+DulOX8cnNftbnYL7/8ooiICM+v0vPmzVP//v314osvemJOnz5t+4l+bpUqVdKQIUM0ZMgQHThwQNdff72effZZdezY0TOaonTp0gX2K3F+ua94XPzZ/OWXXyT931P1Lm6nL37Crr92Oj+f8WrVqik8PNxRe+XmTpC+/vprr++tX3/9VXv37vU775vb448/rm7dunl+df7111+9OjuVK1e2NeF8nz599PLLL+v48eOaO3euqlev7nU1NDeRkZHq27ev+vbtq7Nnz+r222/Xs88+q6eeeirPuQjtvGc5OTnXduzYQbt5GQgMDNSECRPUpk0bvfrqqxo9erRntENwcLDl+5mYmGj5ELOEhAR9//33ys7O9rpIkLPPll+JiYl67LHH9Nhjj2nbtm1q2LChXnzxRc9T5p30CWJiYvy2wznbo8TERGVnZ2vz5s25/pCSVx0SExN18uRJy+PrfijOyZMnvX6YsdvndD9ccMeOHbr22mvzjL3zzjs1fvx4XXPNNXnuU25y+04sSAEBAbr22mv19ddf+7z21Vdf6eqrr7b1ZO+CrlPDhg21YcMGnT171uupuO7bU3K7Iu3m7tvnjDt//rz27Nmjbt26FXCtS/jTX5988klFRETovvvu0/79+31ez88v3J06dVJWVpbPU5VeeukluVwudezYUdKFL7jy5cv7PM5/+vTpDvbgAve2X3nlFa/lU6dOdbxNuxITE7V161avR/d/9913Pk+FioiIkOSbTBeE5s2b68cff/Q79CA3K1as0Lhx43TVVVd5poTxx51w5nx/pk2bZqucyMjIXPfZ7pQitWrV0v79+7Vq1Sqv5e57DXL+Arpx40ZJUosWLWzVESXPypUr/bZP7vurcw5DWrdundd9kXv27NEnn3yi9u3be37FDAwM9NnmtGnTbP8anpWV5TNMNi4uTpUrV/Z8Nhs3bqzExERNnjzZ760HOacAKQy//vqr172Dx48f1z/+8Q81bNjQM0zNnZBc3E5nZGT4TDcg5f0Zzyk4OFg33HCD386HXfXq1VOdOnX05ptver03M2bMkMvl8hrqdLGVK1fq888/1wsvvOBZVqFCBU8nWpK2bNlia6he3759debMGf3973/XokWL1KdPH8t13PdRuYWEhKhu3boyxng9QdMfO+9ZTk7OtW+++YZ28zKRlJTkeY7A6dOnFRcXp6SkJL3xxht+E4SL389evXp5bgnIyd3GderUSb///rvmzp3ree38+fOaNm2aoqKi1Lp163zVNzMz02cqhsTERJUqVcqrb5Kf9uLi7Rw7dkzff/+9Z9lvv/3ms389evRQQECAxo4d6zO65OK2Pbc69OnTR+vWrdPixYt9Xjt69KjnB+xOnTrp/Pnznqf6Sxfaf7v9osaNGyskJMRWO3j//fcrJSXF68dOf5YvX+53eW7fiXbkZ0qR5ORkbdiwwWuffv75Z61YsUK9e/f2it26davnicCFqW/fvsrKyvL63jp9+rTef/991a1b1/ODor+28MSJE5o6darKly/v9TwG6cK0VqdPny6UtrJEX6msWbOm5syZozvvvFO1a9fWXXfdpQYNGsgYox07dmjOnDkKCAjI8/K8W9euXdWmTRuNGTNGO3fuVIMGDbRkyRJ98sknGjlypNf9jvfff7+ef/553X///brhhhu0Zs0az6+uTjRs2FB33nmnpk+frmPHjqlFixZavny57atpl+K+++7TlClT1KFDBw0aNEgHDhzQ66+/rnr16nkeJCTJ84jiuXPnqlatWipbtqzq16+v+vXrX3IdunfvrnHjxmn16tVq3769z+tffPGFtm7dqvPnz2v//v1asWKFli5dqoSEBH366ad5/kLeuHFj9erVS1OnTtXhw4d14403avXq1Z73y+pXx8aNG2vGjBkaP368atSoobi4OM9QM/d0IlYP6xk2bJhmzpyprl27avjw4UpISNDq1av1wQcfqF27dj5T4ixdulTVqlWzPdwGJc/w4cOVmZmpnj17qk6dOjp79qzWrl3ruWo0cOBAr/j69eurQ4cOGjFihEJDQz0/klw87LBLly6aNWuWoqOjVbduXa1bt07Lli3zDFWycuLECcXHxys5OVkNGjRQVFSUli1bpg0bNng6BAEBAXr77bfVsWNH1atXTwMHDlSVKlW0b98+rVy5UqVLl9bChQsL6Cj5V6tWLQ0aNEgbNmxQhQoV9O6772r//v2aOXOmJ6Z9+/aqVq2aBg0apCeeeEKBgYF69913FRsb69MZyOsz7k/37t01ZswYHT9+XKVLl/YsP3bsmKdT5v5R7tVXX1WZMmVUpkwZr+lRJk2apG7duql9+/a644479OOPP+rVV1/V/fff7/dKW1ZWlkaOHKknnnjCa7hTcnKynnzyScXGxmrXrl364Ycf9P7771sew+uvv141atTQmDFjdObMGcuhr9KFY1qxYkXddNNNqlChgrZs2aJXX31VnTt3tvwV3857llN+z7WNGzfqjz/+UPfu3S33BUXjiSeeUO/evfXee+9p8ODBeu2119SyZUtde+21euCBB3T11Vdr//79Wrdunfbu3avvvvvOs968efPUu3dv3XfffWrcuLH++OMPffrpp3r99dfVoEEDPfjgg3rjjTc0YMAAbdy4UdWrV9e8efP05ZdfaurUqfm+svTLL7+obdu26tOnj+rWraugoCDNnz9f+/fv1x133OGJy297IUl33HGHRo0apZ49e2rEiBHKzMzUjBkzVKtWLa8fC92fSfdDam6//XaFhoZqw4YNqly5siZMmJBnHZ544gl9+umn6tKliwYMGKDGjRsrIyNDP/zwg+bNm6edO3eqfPny6tq1q2666SaNHj1aO3fu9Mwb6+/ee3/CwsLUvn17LVu2TGPHjs0zNiEhwfJBOdKFdvWqq65S165dlZiYqIyMDC1btkwLFy5UkyZN1LVrV6/4X375xWuOcrcKFSqoXbt2ki5MKdKmTRulpKRY1mHIkCF666231LlzZz3++OMKDg7WlClTVKFCBT322GNesddcc41at27tdaHg+++/9zxkb/v27Tp27JjnYToNGjTwqv+sWbO0a9cuzwPF1qxZ44m95557PFfZH3roIb399tsaOnSofvnlF1WrVs2z7sVt32uvvaYFCxaoa9euqlatmn777Te9++672r17t2bNmuUz9+fSpUsVERHhOU4FqsCfJ1sMtm/fbh5++GFTo0YNExYWZsLDw02dOnXM4MGDzaZNm7xi+/fvbyIjI/1u58SJE+aRRx4xlStXNsHBwaZmzZpm0qRJXo9yNubCI80HDRpkoqOjTalSpUyfPn3MgQMHcp1S5ODBg17r+3vs+qlTp8yIESNMuXLlTGRkpOnatavZs2dPgU4pkrMebrNnzzZXX321CQkJMQ0bNjSLFy/2+wjstWvXmsaNG5uQkBCveuV2THN7rL8/1113nRk0aJDXMvdxcv+FhISYihUrmnbt2pmXX37Z88hwqzIzMjLM0KFDTdmyZU1UVJTp0aOH+fnnn40k8/zzz/uUd/H78vvvv5vOnTubUqVKGUlej/HOz5QiW7duNcnJyaZq1aomODjYJCQkmMcff9xkZGR4xWVlZZlKlSqZv/3tb7a2i5Lpiy++MPfdd5+pU6eOiYqKMiEhIaZGjRpm+PDhZv/+/V6xkszQoUPN7NmzTc2aNU1oaKhp1KiR12fcmAuPxB84cKApX768iYqKMh06dDBbt241CQkJpn///p643KbqOXPmjHniiSdMgwYNTKlSpUxkZKRp0KCB30eSf/vtt+b222835cqVM6GhoSYhIcH06dPHLF++PM/9zmtKkZz1yTnFhDH/97j6xYsX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88036/jx40pJSdFzzz2no0eP6pZbbtH69esdH5Phw4fru+++U0pKih5++GEtXLjQ6x5rt23btqlv377q2LGjJkyYoKCgIPXu3VtLly7Nd5lWn/Gczp07pw0bNuj666/Pd1n5FRMTo8TERD333HPasWOH3n//fW3atElNmzaVJD355JPq2LFjvn6M69Onj1wul99Ozocffqj27dt7kticli5dqjvvvFMxMTGaOHGinn/+eSUlJdlqp44cOaJOnTqpcePGeuGFFxQfH6+HH35Y7777bp7r/fTTT7rxxhu1ZcsWjR49Wi+++KIiIyPVo0cPzZ8/3ye+cePGWrt2rWV9UPjcidjF55Pd9/PkyZNq1aqVpk2bpvbt2+vll1/W4MGDtXXrVs+P7adOnVJSUpJmzZqlu+66S5MmTVJ0dLQGDBjg9UOH25w5czRp0iQ99NBDGj9+vHbu3Knbb79d586d88T06tVL8+fP18CBAzV9+nSNGDFCJ06c0O7duyXZay/8tav5kZaWpnvuuUfBwcEaO3as0tLSVLVqVa1YscKyDpmZmWrdurVmz56te++9V6+88opuuukmPfXUU179WGOMunfvrlmzZunuu+/W+PHjtXfvXvXv399WHTdu3KizZ8/m2Q7269dP69ev9/pRYc6cOUpOTvbb10tISNDy5cttX0w5ffq0Dh065PN39uxZT8z69et1zTXX6NVXX81zW9nZ2fr+++91ww03+LzWtGlTpaen5/kMjx9++EHnz5/3WT8kJEQNGzbUt99+67U8KytLw4cP1/33369rr73Wzu7m6cyZMwoPD/dZHhERIenC+3WxQu1jFvi1zyJy7NgxI8n06NHD57UjR46YgwcPev4uHh7Zv39/I8mMHj3aa50FCxYYSWb8+PFey5OTk43L5TLbt283xuQ9PEA5hoempKQYSea+++7ziuvZs6cpV66c5/9NmzYZSWbIkCFecf369SuQ4a/uetx5550+8a1bt/Y7jKJ///5el9qthr/q/7+n4mKNGjUyjRs3tqxzfHy86dWrl8/yvIa/ukVHR5tGjRp5/nfvq9vGjRuNJDNy5Eiv9QYMGOCzP+7yLh6yltcwk4SEhFyHI+T02WefmaCgIPPTTz8ZY6yHDIeEhJiHH37Y1rZR8ixZssQEBgaawMBA07x5c/Pkk0+axYsXe4a6XEySkWS+/vpr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uou53xfazqL32muvaenSpZo3b546deqkQ4cOeXWy8/N+pqenW85tumvXLtWsWdMn+cqtz2bVvoSGhmrixIn64osvVKFCBd1888164YUX8v1D1aX0s9LT0xUQEGD7x+ectm3bpkWLFvkcX/e9j+7ju2vXLlWqVMnnokx++5xW7WC/fv20efNmfffdd5ozZ47uuOOOPD+bHTp00OLFi3X06FGtWbNGQ4cO1a5du9SlSxefz3p8fLxuvfVWn78KFSrkax+k//tB9syZMz6vnT592ivGyfoXr/u3v/1NZcuWLZALG24dO3bUtGnTtGbNGl1//fWqXbu2/vWvf3nmffd38a2w+pi5Py3kMhcdHa1KlSr5vSfJfY9lbl9WoaGhth9skFNub0JeD6TJ7VerouyYSP4/FC6Xy2897Dxg52JOf5mTpHLlyjn6ZXDv3r06duxYkSaI+XXs2DGNHz9eQ4YM0fHjxz1Xp0+ePCljjHbu3KmIiAifG9GPHj2q8uXLF0eVUcRCQkI8DwioVauWBg4cqI8++sjT0bJr+PDhmjlzpkaOHKnmzZsrOjpaLpdLd9xxh9+rh/7agz59+qhVq1aaP3++lixZokmTJmnixIn6+OOP1bFjR892Jk2apIYNG/qth78vMDsKsp100k7bUa5cOUm+iW5ROX78uMaMGaPJkycrMjJSH3zwgZKTk9WjRw9JUnJyst5//32vX+tzqly5slq1aqUPP/xQf/3rXz1XQSdOnJhn2XFxcdq0aZMWL16sL774Ql988YVmzpype++91+fhKAXBfa49/vjjfh/sIfn+OOh+X2g7i17Tpk0995T16NFDLVu2VL9+/fTzzz8rKirK0ftZkOy0LyNHjlTXrl21YMECLV68WE8//bQmTJigFStWqFGjRrbKya2f5c+ltkc5ZWdnq127drlObu/+IedSXdwO5rxAc7FmzZopMTFRI0eO1I4dO9SvXz9b24+IiFCrVq3UqlUrlS9fXmlpafriiy9s3/OZX2XLllVoaKh+++03n9fcyypXrpzr+u4Hhea2vnvdbdu26c0339TUqVO9RgOdPn1a586d086dO1W6dGmVLVs23/swbNgwDRw4UN9//73nXs533nlHkv/3vbD6mCU2qZSkzp076+2339b69es9Dy9wKiEhQcuWLdOJEye8rnxt3brV87r0f79u5XxK1qVcyUxISFB2drbS09O9fin6+eefHW/TrpiYGL9DVHPuT2H+8lunTh3t2LEj3+vNmjVLknL9gpL+79ju2LHD62rP9u3bbZVxqft95MgRnTx5Ui+88IJeeOEFn9evuuoqde/eXQsWLPAs27dvn86ePev5xRV/Hu5OWc4vp5xPb5OkX375RREREZ4RAvPmzVP//v314osvemJOnz5t+4l+bpUqVdKQIUM0ZMgQHThwQNdff72effZZdezY0TOaonTp0rk++a+wua94XPzZ/OWXXyTJ84TFi9vpi5+w66+dzs9nvFq1agoPD3fUXhWEsWPH6qqrrvI8sfzXX3/16uxWrlzZ1hP9+vbtqyFDhujnn3/W3LlzFRERoa5du1quFxISoq5du6pr167Kzs7WkCFD9MYbb+jpp5/OMyH49ddflZGR4XW1Mud7ltPVV18t6cKQRrvn2o4dOzxX6FF8AgMDNWHCBLVp00avvvqqRo8ena/3MzEx0fIhZgkJCfr++++VnZ3tdZEgZ58tvxITE/XYY4/pscce07Zt29SwYUO9+OKLnpFvTvoEMTExftvhnO1RYmKisrOztXnz5lx/tMurDomJiTp58qTl8XU/FOfkyZNePwLa7XPWqVNH0oXPm9WDZu68806NHz9e11xzTZ77lJvcvhMLUkBAgK699lrPcNGLffXVV7r66qvzHBFXv359BQUF6euvv1afPn08y8+ePatNmzZ5lu3bt89z+0LO292kC/3Bv/zlL3k+ETYvkZGRat68uef/ZcuWKTw8XDfddJNXXGH2MUvs8FfpwtPvIiIidN9992n//v0+r+fnF+5OnTopKyvL5ylRL730klwulzp27CjpQmeqfPnyPo/znz59uoM9uMC97VdeecVrudMTKz8SExO1detWr0f3f/fddz5PhXI/RSq/HVQ7mjdvrh9//NHv0IHcrFixQuPGjfPqYPnjTjhzvj/Tpk2zVU5kZGSu+2xnSpG4uDjNnz/f569NmzYKCwvT/Pnz9dRTT3mt477HqEWLFrbqiJJn5cqVftsn9/3VOYchrVu3zuu+yD179uiTTz5R+/btPb/ABwYG+mxz2rRptn8Nz8rK8hm6GhcXp8qVK3s+m40bN1ZiYqImT57s99aDnFOAFIZff/3V60mRx48f1z/+8Q81bNhQFStWlCRP8ntxO52RkeH3ilpen/GcgoODdcMNN/jtfBS2X375Ra+++qpefvllT6eyQoUKnk60JG3ZssVzDPLSq1cvBQYG6oMPPtBHH32kLl26+AxPzcl9H5VbQECArrvuOkn+h31d7Pz583rjjTc8/589e1ZvvPGGYmNjve4VvlhcXJySkpL0xhtv+O1Q+jvXNm7c6NWpQvFJSkpS06ZNNXXqVJ0+fTpf72evXr303Xff+X3Cr7uN69Spk37//XfPMG7pwnk2bdo0RUVFqXXr1vmqb2Zmpmeoo1tiYqJKlSrldX7np724eDvHjh3T999/71n222+/+exfjx49FBAQoLFjx/qMLrm4bc+tDn369NG6deu0ePFin9eOHj2q8+fPS7pw7M6fP+95qr90of232y9q3LixQkJCbLWD999/v1JSUrx+7PRn+fLlfpfn9p1oR36mFElOTtaGDRu89unnn3/WihUr1Lt3b6/YrVu3ep4ILF0YOXnrrbdq9uzZXs/4mDVrlk6ePOlZv379+n77g/Xq1VO1atU0f/58DRo0KN/76c/atWv18ccfa9CgQYqOjvZ6rTD7mCX6SmXNmjU1Z84c3Xnnnapdu7buuusuNWjQQMYY7dixQ3PmzFFAQECel+fdunbtqjZt2mjMmDHauXOnGjRooCVLluiTTz7RyJEjve53vP/++/X888/r/vvv1w033KA1a9Z4fnV1omHDhrrzzjs1ffp0HTt2TC1atNDy5cttX027FPfdd5+mTJmiDh06aNCgQTpw4IBef/111atXzzNUU7owpKNu3bqaO3euatWqpbJly6p+/fqW9z3Y0b17d40bN06rV69W+/btfV7/4osvtHXrVp0/f1779+/XihUrtHTpUiUkJOjTTz/13EjtT+PGjdWrVy9NnTpVhw8f1o033qjVq1d73i+rXx0bN26sGTNmaPz48apRo4bi4uJ0yy23SJJnOpG87uOKiIjwDE272IIFC7R+/Xq/ry1dulTVqlWzPdwGJc/w4cOVmZmpnj17qk6dOjp79qzWrl2ruXPnqnr16ho4cKBXfP369dWhQweNGDFCoaGhnh9JLr5Pt0uXLpo1a5aio6NVt25drVu3TsuWLfMMVbJy4sQJxcfHKzk5WQ0aNFBUVJSWLVumDRs2eDoEAQEBevvtt9WxY0fVq1dPAwcOVJUqVbRv3z6tXLlSpUuX1sKFCwvoKPlXq1YtDRo0SBs2bFCFChX07rvvav/+/Zo5c6Ynpn379qpWrZoGDRqkJ554QoGBgXr33XcVGxvr1RmQ8v6M+9O9e3eNGTNGx48f97oH8dixY55OmftHuVdffVVlypRRmTJlvKZHWbNmjSfhPXjwoDIyMjR+/HhJ0s033+x3mpBHHnlEffv29RqVk5ycrO7du+uvf/2rJGnhwoX67LPPLI9hXFyc2rRpoylTpujEiRPq27ev5Tr333+//vjjD91yyy2Kj4/Xrl27NG3aNDVs2NDyF+/KlStr4sSJ2rlzp2rVqqW5c+dq06ZNevPNN/OcSuq1115Ty5Ytde211+qBBx7Q1Vdfrf3792vdunXau3evvvvuO0/sgQMH9P3332vo0KGW+4Ki8cQTT6h379567733NHjwYNvv5xNPPKF58+apd+/euu+++9S4cWP98ccf+vTTT/X666+rQYMGevDBB/XGG29owIAB2rhxo6pXr6558+bpyy+/1NSpU23PGev2yy+/qG3bturTp4/q1q2roKAgzZ8/X/v379cdd9zhictveyFJd9xxh0aNGqWePXtqxIgRyszM1IwZM1SrVi2vHwtr1KihMWPGeB5Sc/vttys0NFQbNmxQ5cqVNWHChDzr8MQTT+jTTz9Vly5dNGDAADVu3FgZGRn64YcfNG/ePO3cuVPly5dX165dddNNN2n06NHauXOnZ65fu/fDh4WFqX379lq2bJnGjh2bZ2xCQoLlg3KkC+3qVVddpa5duyoxMVEZGRlatmyZFi5cqCZNmviMpPjll1/8PjelQoUKateunaQLU4q0adNGKSkplnUYMmSI3nrrLXXu3FmPP/64goODNWXKFFWoUEGPPfaYV+w111yj1q1be83H++yzz6pFixZq3bq1HnzwQe3du1cvvvii2rdvr9tuu03ShWH5/vp87gtIOV/7/vvvPQ/u2759u+d2Kklq0KCB55js2rVLffr0Ubdu3VSxYkX99NNPev3113Xdddfpueee8ymvUPuYBf482WKwfft28/DDD5saNWqYsLAwEx4eburUqWMGDx5sNm3a5BWb11QOJ06cMI888oipXLmyCQ4ONjVr1jSTJk3yepSzMRcenz9o0CATHR1tSpUqZfr06WMOHDiQ65QiBw8e9Frf32PtT506ZUaMGGHKlStnIiMjTdeuXc2ePXsKdEqRnPVwmz17trn66qtN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AAAAgGMklQAAAAAAx0gqAQAAAACOkVQCAAAAABxzGWNMcVcCAAAAAFAycaUSAAAAAOAYSSUAAAAAwDGSSgAAAACAYySVAAAAAADHSCoBAAAAAI6RVAIAAAAAHCOpBAAAAAA4RlIJAAAAAHCMpBIAAAAA4BhJJQAAAADAMZJKAAAAAIBjJJUAAAAAAMdIKgEAAAAAjpFUAgAAAAAcI6m8zLlcLqWmphZ3NfI0YMAARUVFXdI2srOzVb9+fT377LOXtJ3U1FS5XC5H67733ntyuVzauXPnJdXhUtx444168skni618ANb27NmjsLAwffnll8VdlUKTlJSkpKSkfK3jrw2tXr26unTpYrnuqlWr5HK5tGrVqvxVVNLrr7+uatWq6cyZM/leF8AFO3fulMvl0nvvvWcr/sMPP1TZsmV18uTJwq0YCsS5c+dUtWpVTZ8+vdDKuCKSyh07dmjYsGGqVauWIiIiFBERobp162ro0KH6/vvvi7t6hSopKUkul8vy71IT08zMTKWmpjr6wrfjgw8+0J49ezRs2DDPMncHxf0XFhamypUrq0OHDnrllVd04sSJQqnLxaZPn267gc3L9u3blZycrJiYGEVERKhly5ZauXKlT9yoUaP02muv6ffff7/kMnF5++GHH5ScnKyEhASFhYWpSpUqateunaZNm1bcVSs07k7L5MmTi7sqHk4+42PHjlWzZs100003eZb9/PPPeuSRR9SiRQuFhYVZ/kD16aef6vrrr1dYWJiqVaumlJQUnT9/3itm8+bNatWqlUqVKqUbbrhB69at89nOlClTVK9ePZ91/0wGDBigs2fP6o033ijuqvzp5PyeDgoKUpUqVTRgwADt27evuKtX4AqqT1DS65CVlaWUlBQNHz7c66JC9erV5XK5dOutt/pd76233vKcK19//bXXa//5z3/UsWNHValSxdMudu3aVXPmzPGKy6uvO3jwYMf7tG/fPvXp00dlypRR6dKl1b17d/3vf/+zvf7atWvVsmVLRUREqGLFihoxYoRlwv3ss8/K5XKpfv36jrfp/kHO399///tfT1xwcLAeffRRPfvsszp9+rTt/coXU8ItXLjQREREmNKlS5uHH37YvP766+bNN980jz76qKlevbpxuVxm586dxV1Nx5AZJY8AAD1+SURBVCSZlJSUXF9fsmSJmTVrludvxIgRRpL561//6rX8u+++u6R6HDx4MNe69O/f30RGRl7S9hs0aGAefPBBr2UzZ840kszYsWPNrFmzzLvvvmuee+450759e+NyuUxCQoLPfp07d86cOnXKUR3Onz9vTp06ZbKzsz3L6tWrZ1q3bu1oe267d+825cuXNxUqVDDPPvusmTp1qmnQoIEJCgoyq1ev9orNysoyFStWNE8//fQllYnL25dffmlCQkJMjRo1zLhx48xbb71lnnnmGdO+fXuTmJhY3NUrNDt27DCSzKRJk4q7Kh75/YwfOHDABAcHmzlz5ngtnzlzpgkICDD169c3DRs2NJLMjh07/G7j888/Ny6Xy7Rp08a8+eabZvjw4SYgIMAMHjzYE3P+/HlTu3Zt07x5czNjxgzTsWNHExsba44dO+aJ2b9/v4mOjjaLFy/O1z7bcebMGXPmzJl8reOvDU1ISDCdO3e2XHflypVGklm5cmV+q2qMMebJJ580CQkJXmWj8OX8nn7rrbfMoEGDTGBgoElMTHT8fXy5Kog+weVaB3f7PHPmTMvY+fPnG5fLZfbu3eu1PCEhwYSFhZmAgADz22+/+azXunVrExYWZiSZDRs2eJZ/+OGHxuVymUaNGpmJEyeaN9980zz11FPmpptuMklJSV7bkGTatWvn1cd1/3311VeO9v3EiROmZs2aJi4uzkycONFMmTLFVK1a1cTHx5tDhw5Zrv/tt9+asLAw06hRIzNjxgwzZswYExoaam677bZc19mzZ4+JiIgwkZGRpl69eo636W47R4wY4XM8Dh486BV75MgRExISYt555x2bRyZ/SnRSuX37dhMZGWmuueYa8+uvv/q8fu7cOfPyyy+b3bt357mdkydPFlYVL5lVUpnTRx99ZOuLOb/7XJhJ5TfffGMkmWXLlnktd39ZXdzwuC1fvtyEh4ebhIQEk5mZ6bhsKwXReA8ZMsQEBQWZrVu3epZlZGSYqlWrmuuvv94nftiwYXSOrnCdOnUysbGx5siRIz6v7d+/v8jrU1Rt4JWQVE6ZMsWEh4ebEydOeC0/fPiwOX78uDHGmEmTJuWZVNatW9c0aNDAnDt3zrNszJgxxuVymS1bthhjjNmyZYuRZHbt2mWMudBmhIeHm0WLFnnWGTRokOnatavtuheHokoqv/76ayPJLF++3NH6cCa37+lRo0YZSWbu3LnFVLPCkZ/2orDa1cshqezWrZtp2bKlz/KEhATTtm1bU7p0aTN16lSv1/bs2WMCAgJMr169fM6ZunXrmnr16vn9ISvnd6IkM3ToUJt7Zc/EiRONJLN+/XrPsi1btpjAwEDz1FNPWa7fsWNHU6lSJa8f/d566y0jKdcf/fr27WtuueUW07p1a79Jpd1tutvOjz76yNa+dunSxbRq1cpWbH6V6OGvL7zwgjIyMjRz5kxVqlTJ5/WgoCCNGDFCVatW9Sxz3/+Xnp6uTp06qVSpUrrrrrskSRkZGXrsscdUtWpVhYaGqnbt2po8ebKMMZ718xpznnOYqfv+vu3bt2vAgAEqU6aMoqOjNXDgQGVmZnqte+bMGT3yyCOKjY1VqVKl1K1bN+3du/cSj5B3PTZv3qx+/fopJiZGLVu2lJT7fTMDBgxQ9erVPfscGxsrSUpLS8t1SO2+ffvUo0cPRUVFKTY2Vo8//riysrIs67dgwQKFhITo5ptvtr1Pt9xyi55++mnt2rVLs2fP9tnXi506dUojRoxQ+fLlPcd23759PvuQ836g6tWr66efftLq1as9+3zxsUpPT1d6erplXf/973+rUaNGql27tmdZRESEunXrpm+++Ubbtm3zim/Xrp127dqlTZs22T4eKFnS09NVr149lSlTxue1uLg4r/9dLpeGDRum999/X7Vr11ZYWJgaN26sNWvWeMXt2rVLQ4YMUe3atRUeHq5y5cqpd+/ePkMw3ef56tWrNWTIEMXFxSk+Pl6SdOLECY0cOVLVq1dXaGio4uLi1K5dO33zzTde2/jqq6902223KTo6WhEREWrdurXj+wvd9fnyyy/16KOPKjY2VpGRkerZs6cOHjzoFeu+P2/JkiVq2LChwsLCVLduXX388cdecbndW53fz7g/CxYsULNmzXzuIy9btqxKlSplub+bN2/W5s2b9eCDDyooKMizfMiQITLGaN68eZIutFuSFBMTI+lCmxEeHu757vjmm2/0/vvva8qUKZZlug0bNkxRUVE+3z+SdOedd6pixYqeNtvfd8O0adNUr149RUREKCYmRjfccIPX0LS87ku3es9yY/dca9y4scqWLatPPvnE1nZRuFq1aiVJPt+RW7duVXJyssqWLauwsDDdcMMN+vTTT33WP3r0qB555BFPWxQfH697771Xhw4d8sQcOHBAgwYNUoUKFRQWFqYGDRro73//u9d2Lh5y/+abbyoxMVGhoaFq0qSJNmzY4BX7+++/a+DAgYqPj1doaKgqVaqk7t2722ov8mpXL+5PXSy3dmr27Nlq2rSp53N28803a8mSJZZ1cB+3kSNHevqxNWrU0MSJE5Wdne1zfAcMGKDo6GiVKVNG/fv319GjR33q4s/p06e1aNGiXIe4hoWF6fbbb/cZtvrBBx8oJiZGHTp08FknPT1dTZo0UUhIiM9rOb8T7crMzNTWrVu9zpnczJs3T02aNFGTJk08y+rUqaO2bdvqww8/zHPd48ePa+nSpbr77rtVunRpz/J7771XUVFRftdfs2aN5s2bp6lTpxbYNqUL3+FWt0K0a9dO//nPf/THH3/kGedEiU4qP/vsM9WoUUPNmjXL13rnz59Xhw4dFBcXp8mTJ6tXr14yxqhbt2566aWXdNttt2nKlCmqXbu2nnjiCT366KOXVM8+ffroxIkTmjBhgvr06aP33ntPaWlpXjH333+/pk6dqvbt2+v5559XcHCwOnfufEnl5tS7d29lZmbqueee0wMPPGB7vdjYWM2YMUOS1LNnT82aNUuzZs3S7bff7onJyspShw4dVK5cOU2ePFmtW7fWiy++qDfffNNy+2vXrlX9+vUVHBycr/255557JMnT2OZmwIABmjZtmjp16qSJEycqPDzc1rGdOnWq4uPjVadOHc8+jxkzxvN627Zt1bZtW8vtnDlzRuHh4T7LIyIiJEkbN270Wt64cWNJuqIfAvJnl5CQoI0bN+rHH3+0Fb969WqNHDlSd999t8aOHavDhw/rtttu81p/w4YNWrt2re644w698sorGjx4sJYvX66kpCS/ScSQIUO0efNmPfPMMxo9erQkafDgwZoxY4Z69eql6dOn6/HHH1d4eLi2bNniWW/FihW6+eabdfz4caWkpOi5557T0aNHdcstt2j9+vWOj8nw4cP13XffKSUlRQ8//LAWLlzodY+127Zt29S3b1917NhREyZMUFBQkHr37q2lS5fmu0yrz3hO586d04YNG3T99dfnuyy3b7/9VpJ0ww03eC2vXLmy4uPjPa/XqlVL0dHRSk1N1a5duzRp0iQdP37cU/aIESM0bNgw1ahRw3bZffv2VUZGhv71r395Lc/MzNTChQuVnJyswMBAv+u+9dZbGjFihOrWraupU6cqLS1NDRs21Fdf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GxC5+Xf8RoGBgXLfffcVeFc9E82bN5emTZvKggULPN6befPmicPhkD59+nh93JYtW2TdunUydepU989q1KjhnkSLiPzrX/9yvwY3069fP8nMzJT3339f1q9fbzRBcV1H5RIUFCQxMTGilPK4g6Y3Ju/ZjWw+a//85z/ZbpYSHTt2lLZt28rMmTPl6tWrUr16denYsaO8++67XguEvO/n008/Lbt37/Z6jbBrG/f444/LyZMnZeXKle7fZWdny6xZsyQsLEweeeSRWxrv5cuX3ac6ukRHR0uFChU85ia3sr3I28758+flu+++c//sxx9/zPf8evfuLX5+fpKUlJTv7JK82/aCxhAbGyvbt2+XDRs25Ptdenq6e6mJxx9/XLKzs9139Re5vv03nRe1adNGgoKCjLaDzz//vCQkJHjs7PTmiy++8Przgv4mmriVJUX69OkjO3bs8HhO+/fvl82bN0vfvn09svv27XPfEVjk+pmTnTt3lmXLlnnc42Pp0qVy6dIl9+NbtGgha9asyfevefPmEhUVJWvWrJG4uLhbalPE+7Zw9+7dsnbtWunSpUu+MzN37twpDofD4yCHr5TpI5WNGzeW5cuXy4ABA+Suu+6SX//619KqVStRSsnhw4dl+fLl4ufnd9PD8y49evSQTp06yfjx4+XIkSPSqlUr2bhxo3z66acycuRIj+sdn3/+eXnrrbfk+eefl/vuu0+2bdvm3utqo3Xr1jJgwACZO3eunD9/Xtq1aydffPGF8dG0whgyZIjMmDFDunbtKnFxcXL69GmZP3++NG/e3H0jIZHrp3TExMTIypUrpUmTJlKlShVp0aKF9roHE7169ZKJEyfKl19+KV26dMn3++TkZNm3b59kZ2fLqVOnZPPmzZKSkiL16tWTtWvX3nQPeZs2beTpp5+WmTNnSlpamjz44IPy5Zdfut8v3V7HNm3ayLx58+T3v/+9NGrUSKpXr+4+1cy1nIjuOq7U1FSJjY2Vnj17SmRkpPzf//2fzJ8/X1q2bClvvvlmvnxKSopERUUZn26DsueVV16Ry5cvy5NPPilNmzaVrKws+frrr91HjQYPHuyRb9GihXTt2lVGjBghwcHB7p0keU877N69uyxdulTCw8MlJiZGtm/fLps2bXKfqqRz8eJFqVOnjvTp00datWolYWFhsmnTJtmxY4d7QuDn5ycL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",
      "text/plain": [
       "<Figure size 1000x1500 with 15 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "======================================================================\n",
      "MULTI-SPARSITY EVALUATION\n",
      "Testing at sparsity levels: [0.1, 0.15, 0.2, 0.25]\n",
      "======================================================================\n",
      "\n",
      "\n",
      "──────────────────────────────────────────────────────────────────────\n",
      "Evaluating at ρ = 0.10 (10% visible pixels)\n",
      "──────────────────────────────────────────────────────────────────────\n",
      "Dataset size: 3000, Sparsity: 10% visible\n",
      "\n",
      "Results for ρ = 0.10:\n",
      "  Mean MSE:   0.053065\n",
      "  Median MSE: 0.046525\n",
      "\n",
      "──────────────────────────────────────────────────────────────────────\n",
      "Evaluating at ρ = 0.15 (15% visible pixels)\n",
      "──────────────────────────────────────────────────────────────────────\n",
      "Dataset size: 3000, Sparsity: 15% visible\n",
      "\n",
      "Results for ρ = 0.15:\n",
      "  Mean MSE:   0.033833\n",
      "  Median MSE: 0.030242\n",
      "\n",
      "──────────────────────────────────────────────────────────────────────\n",
      "Evaluating at ρ = 0.20 (20% visible pixels)\n",
      "──────────────────────────────────────────────────────────────────────\n",
      "Dataset size: 3000, Sparsity: 20% visible\n",
      "\n",
      "Results for ρ = 0.20:\n",
      "  Mean MSE:   0.024975\n",
      "  Median MSE: 0.022386\n",
      "\n",
      "──────────────────────────────────────────────────────────────────────\n",
      "Evaluating at ρ = 0.25 (25% visible pixels)\n",
      "──────────────────────────────────────────────────────────────────────\n",
      "Dataset size: 3000, Sparsity: 25% visible\n",
      "\n",
      "Results for ρ = 0.25:\n",
      "  Mean MSE:   0.020382\n",
      "  Median MSE: 0.018410\n",
      "\n",
      "======================================================================\n",
      "SUMMARY TABLE\n",
      "======================================================================\n",
      "Sparsity ρ      Mean MSE        Median MSE      Std MSE        \n",
      "----------------------------------------------------------------------\n",
      "0.10            0.053065        0.046525        0.031698       \n",
      "0.15            0.033833        0.030242        0.018940       \n",
      "0.20            0.024975        0.022386        0.013053       \n",
      "0.25            0.020382        0.018410        0.010399       \n",
      "======================================================================\n",
      "\n",
      "✓ Best model saved to 'best_hybrid_model.pth'\n",
      "✓ Results saved to 'hybrid_multi_sparsity_results.json'\n",
      "✓ Training history saved to 'hybrid_training_history.json'\n",
      "\n",
      "======================================================================\n",
      "COMPLETE! Hybrid U-Net + ResNet model trained successfully\n",
      "======================================================================\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torchvision import datasets, transforms\n",
    "from torch.utils.data import DataLoader, Dataset, Subset\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import json\n",
    "\n",
    "# ============================================================================\n",
    "# 1. DATASET WITH CONFIGURABLE SPARSITY (UNCHANGED)\n",
    "# ============================================================================\n",
    "\n",
    "class EnhancedMNISTDataset(Dataset):\n",
    "    def __init__(self, mnist_dataset, sparsity_level=0.15):\n",
    "        \"\"\"\n",
    "        Args:\n",
    "            sparsity_level: Fraction of pixels to KEEP (0.15 = 15% visible)\n",
    "        \"\"\"\n",
    "        self.mnist_dataset = mnist_dataset\n",
    "        self.sparsity_level = sparsity_level\n",
    "        print(f\"Dataset size: {len(mnist_dataset)}, Sparsity: {sparsity_level:.0%} visible\")\n",
    "    \n",
    "    def __len__(self):\n",
    "        return len(self.mnist_dataset)\n",
    "    \n",
    "    def __getitem__(self, idx):\n",
    "        full_image, digit_label = self.mnist_dataset[idx]\n",
    "        \n",
    "        # Create sparse measurement - KEEP sparsity_level fraction of pixels\n",
    "        mask = (torch.rand_like(full_image) < self.sparsity_level).float()\n",
    "        sparse_image = full_image * mask\n",
    "        \n",
    "        x = {'sparse_image': sparse_image, 'mask': mask}\n",
    "        s = {\n",
    "            'full_image': full_image,\n",
    "            'digit_label': digit_label\n",
    "        }\n",
    "        \n",
    "        return x, s\n",
    "\n",
    "# ============================================================================\n",
    "# 2. RESIDUAL BLOCKS (ResNet Component)\n",
    "# ============================================================================\n",
    "\n",
    "class ResidualBlock(nn.Module):\n",
    "    \"\"\"\n",
    "    Residual block with pre-activation (modern ResNet design)\n",
    "    Maintains spatial resolution\n",
    "    \"\"\"\n",
    "    def __init__(self, channels):\n",
    "        super().__init__()\n",
    "        self.bn1 = nn.BatchNorm2d(channels)\n",
    "        self.conv1 = nn.Conv2d(channels, channels, 3, padding=1, bias=False)\n",
    "        self.bn2 = nn.BatchNorm2d(channels)\n",
    "        self.conv2 = nn.Conv2d(channels, channels, 3, padding=1, bias=False)\n",
    "    \n",
    "    def forward(self, x):\n",
    "        residual = x\n",
    "        \n",
    "        # Pre-activation design\n",
    "        out = F.relu(self.bn1(x))\n",
    "        out = self.conv1(out)\n",
    "        out = F.relu(self.bn2(out))\n",
    "        out = self.conv2(out)\n",
    "        \n",
    "        # Add residual connection\n",
    "        out = out + residual\n",
    "        return out\n",
    "\n",
    "# ============================================================================\n",
    "# 3. HYBRID U-NET + RESNET ENCODERS\n",
    "# ============================================================================\n",
    "\n",
    "class HybridSparseEncoder(nn.Module):\n",
    "    \"\"\"\n",
    "    Hybrid encoder: U-Net structure + ResNet blocks\n",
    "    \n",
    "    Architecture:\n",
    "    - Initial conv to expand channels\n",
    "    - ResBlock at each resolution for feature refinement\n",
    "    - Downsample between resolutions (U-Net style)\n",
    "    - Final bottleneck: 64×7×7\n",
    "    \"\"\"\n",
    "    def __init__(self, latent_channels=64, num_res_blocks=2):\n",
    "        super().__init__()\n",
    "        \n",
    "        # Input: 2 channels (sparse_image + mask), 28×28\n",
    "        self.input_conv = nn.Sequential(\n",
    "            nn.Conv2d(2, 64, 3, padding=1, bias=False),\n",
    "            nn.BatchNorm2d(64),\n",
    "            nn.ReLU()\n",
    "        )\n",
    "        \n",
    "        # Resolution 1: 28×28, 64 channels\n",
    "        self.res_blocks_1 = nn.Sequential(\n",
    "            *[ResidualBlock(64) for _ in range(num_res_blocks)]\n",
    "        )\n",
    "        \n",
    "        # Downsample 1: 28×28 → 14×14\n",
    "        self.down1 = nn.Sequential(\n",
    "            nn.Conv2d(64, 128, 3, stride=2, padding=1, bias=False),\n",
    "            nn.BatchNorm2d(128),\n",
    "            nn.ReLU()\n",
    "        )\n",
    "        \n",
    "        # Resolution 2: 14×14, 128 channels\n",
    "        self.res_blocks_2 = nn.Sequential(\n",
    "            *[ResidualBlock(128) for _ in range(num_res_blocks)]\n",
    "        )\n",
    "        \n",
    "        # Downsample 2: 14×14 → 7×7\n",
    "        self.down2 = nn.Sequential(\n",
    "            nn.Conv2d(128, latent_channels, 3, stride=2, padding=1, bias=False),\n",
    "            nn.BatchNorm2d(latent_channels),\n",
    "            nn.ReLU()\n",
    "        )\n",
    "        \n",
    "        # Resolution 3: 7×7, latent_channels (bottleneck)\n",
    "        self.res_blocks_bottleneck = nn.Sequential(\n",
    "            *[ResidualBlock(latent_channels) for _ in range(num_res_blocks)]\n",
    "        )\n",
    "        \n",
    "        self.latent_channels = latent_channels\n",
    "        self.num_res_blocks = num_res_blocks\n",
    "    \n",
    "    def forward(self, sparse_image, mask):\n",
    "        x = torch.cat([sparse_image, mask], dim=1)  # (B, 2, 28, 28)\n",
    "        \n",
    "        x = self.input_conv(x)      # (B, 64, 28, 28)\n",
    "        x = self.res_blocks_1(x)    # (B, 64, 28, 28) - ResNet refinement\n",
    "        \n",
    "        x = self.down1(x)            # (B, 128, 14, 14) - U-Net downsample\n",
    "        x = self.res_blocks_2(x)    # (B, 128, 14, 14) - ResNet refinement\n",
    "        \n",
    "        x = self.down2(x)            # (B, 64, 7, 7) - U-Net downsample\n",
    "        z = self.res_blocks_bottleneck(x)  # (B, 64, 7, 7) - ResNet refinement\n",
    "        \n",
    "        return z\n",
    "\n",
    "\n",
    "class HybridFullImageEncoder(nn.Module):\n",
    "    \"\"\"\n",
    "    Hybrid encoder for full images (same structure as sparse encoder)\n",
    "    \"\"\"\n",
    "    def __init__(self, latent_channels=64, num_res_blocks=2):\n",
    "        super().__init__()\n",
    "        \n",
    "        # Input: 1 channel (full_image), 28×28\n",
    "        self.input_conv = nn.Sequential(\n",
    "            nn.Conv2d(1, 64, 3, padding=1, bias=False),\n",
    "            nn.BatchNorm2d(64),\n",
    "            nn.ReLU()\n",
    "        )\n",
    "        \n",
    "        # Resolution 1: 28×28, 64 channels\n",
    "        self.res_blocks_1 = nn.Sequential(\n",
    "            *[ResidualBlock(64) for _ in range(num_res_blocks)]\n",
    "        )\n",
    "        \n",
    "        # Downsample 1: 28×28 → 14×14\n",
    "        self.down1 = nn.Sequential(\n",
    "            nn.Conv2d(64, 128, 3, stride=2, padding=1, bias=False),\n",
    "            nn.BatchNorm2d(128),\n",
    "            nn.ReLU()\n",
    "        )\n",
    "        \n",
    "        # Resolution 2: 14×14, 128 channels\n",
    "        self.res_blocks_2 = nn.Sequential(\n",
    "            *[ResidualBlock(128) for _ in range(num_res_blocks)]\n",
    "        )\n",
    "        \n",
    "        # Downsample 2: 14×14 → 7×7\n",
    "        self.down2 = nn.Sequential(\n",
    "            nn.Conv2d(128, latent_channels, 3, stride=2, padding=1, bias=False),\n",
    "            nn.BatchNorm2d(latent_channels),\n",
    "            nn.ReLU()\n",
    "        )\n",
    "        \n",
    "        # Resolution 3: 7×7, latent_channels (bottleneck)\n",
    "        self.res_blocks_bottleneck = nn.Sequential(\n",
    "            *[ResidualBlock(latent_channels) for _ in range(num_res_blocks)]\n",
    "        )\n",
    "        \n",
    "        self.latent_channels = latent_channels\n",
    "        self.num_res_blocks = num_res_blocks\n",
    "    \n",
    "    def forward(self, full_image):\n",
    "        x = self.input_conv(full_image)  # (B, 64, 28, 28)\n",
    "        x = self.res_blocks_1(x)         # (B, 64, 28, 28)\n",
    "        \n",
    "        x = self.down1(x)                # (B, 128, 14, 14)\n",
    "        x = self.res_blocks_2(x)         # (B, 128, 14, 14)\n",
    "        \n",
    "        x = self.down2(x)                # (B, 64, 7, 7)\n",
    "        z = self.res_blocks_bottleneck(x)  # (B, 64, 7, 7)\n",
    "        \n",
    "        return z\n",
    "\n",
    "# ============================================================================\n",
    "# 4. HYBRID DECODER (U-Net structure + ResNet blocks)\n",
    "# ============================================================================\n",
    "\n",
    "class HybridDecoder(nn.Module):\n",
    "    \"\"\"\n",
    "    Hybrid decoder: U-Net upsampling + ResNet refinement blocks\n",
    "    \"\"\"\n",
    "    def __init__(self, latent_channels=64, num_res_blocks=2):\n",
    "        super().__init__()\n",
    "        \n",
    "        # Input: latent_channels, 7×7\n",
    "        \n",
    "        # Resolution 1: 7×7, latent_channels (bottleneck)\n",
    "        self.res_blocks_bottleneck = nn.Sequential(\n",
    "            *[ResidualBlock(latent_channels) for _ in range(num_res_blocks)]\n",
    "        )\n",
    "        \n",
    "        # Upsample 1: 7×7 → 14×14\n",
    "        self.up1 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(latent_channels, 128, 4, stride=2, padding=1, bias=False),\n",
    "            nn.BatchNorm2d(128),\n",
    "            nn.ReLU()\n",
    "        )\n",
    "        \n",
    "        # Resolution 2: 14×14, 128 channels\n",
    "        self.res_blocks_2 = nn.Sequential(\n",
    "            *[ResidualBlock(128) for _ in range(num_res_blocks)]\n",
    "        )\n",
    "        \n",
    "        # Upsample 2: 14×14 → 28×28\n",
    "        self.up2 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(128, 64, 4, stride=2, padding=1, bias=False),\n",
    "            nn.BatchNorm2d(64),\n",
    "            nn.ReLU()\n",
    "        )\n",
    "        \n",
    "        # Resolution 3: 28×28, 64 channels\n",
    "        self.res_blocks_3 = nn.Sequential(\n",
    "            *[ResidualBlock(64) for _ in range(num_res_blocks)]\n",
    "        )\n",
    "        \n",
    "        # Output convolution\n",
    "        self.output_conv = nn.Sequential(\n",
    "            nn.Conv2d(64, 32, 3, padding=1),\n",
    "            nn.ReLU(),\n",
    "            nn.Conv2d(32, 1, 3, padding=1),\n",
    "            nn.Tanh()\n",
    "        )\n",
    "    \n",
    "    def forward(self, z):\n",
    "        x = self.res_blocks_bottleneck(z)  # (B, 64, 7, 7)\n",
    "        \n",
    "        x = self.up1(x)                    # (B, 128, 14, 14)\n",
    "        x = self.res_blocks_2(x)          # (B, 128, 14, 14)\n",
    "        \n",
    "        x = self.up2(x)                    # (B, 64, 28, 28)\n",
    "        x = self.res_blocks_3(x)          # (B, 64, 28, 28)\n",
    "        \n",
    "        img = self.output_conv(x)          # (B, 1, 28, 28)\n",
    "        return img\n",
    "\n",
    "# ============================================================================\n",
    "# 5. COMPLETE HYBRID MODEL\n",
    "# ============================================================================\n",
    "\n",
    "class HybridUNetResNetModel(nn.Module):\n",
    "    \"\"\"\n",
    "    Hybrid U-Net + ResNet Manifold Learning Model\n",
    "    \n",
    "    Combines:\n",
    "    - U-Net's hierarchical compression (28→14→7)\n",
    "    - ResNet's residual blocks for gradient flow and refinement\n",
    "    \n",
    "    Benefits:\n",
    "    - Better gradient flow than pure U-Net\n",
    "    - More expressive features at each resolution\n",
    "    - Stronger reconstruction quality\n",
    "    - Still maintains compact 64×7×7 manifold\n",
    "    \"\"\"\n",
    "    def __init__(self, latent_channels=64, num_res_blocks=2):\n",
    "        super().__init__()\n",
    "        self.encoder_x = HybridSparseEncoder(latent_channels, num_res_blocks)\n",
    "        self.encoder_s = HybridFullImageEncoder(latent_channels, num_res_blocks)\n",
    "        self.decoder = HybridDecoder(latent_channels, num_res_blocks)\n",
    "        \n",
    "        self.latent_channels = latent_channels\n",
    "        self.num_res_blocks = num_res_blocks\n",
    "        self.latent_dim = latent_channels * 7 * 7\n",
    "        \n",
    "    def forward(self, x, s):\n",
    "        \"\"\"Forward pass for both modalities\"\"\"\n",
    "        sparse_image = x['sparse_image']\n",
    "        mask = x['mask']\n",
    "        full_image = s['full_image']\n",
    "        \n",
    "        # Encode both modalities to spatial latent\n",
    "        z_x = self.encoder_x(sparse_image, mask)  # (B, 64, 7, 7)\n",
    "        z_s = self.encoder_s(full_image)          # (B, 64, 7, 7)\n",
    "        \n",
    "        # Decode both\n",
    "        img_recon_x = self.decoder(z_x)           # (B, 1, 28, 28)\n",
    "        img_recon_s = self.decoder(z_s)           # (B, 1, 28, 28)\n",
    "        \n",
    "        return z_x, z_s, img_recon_x, img_recon_s\n",
    "\n",
    "# ============================================================================\n",
    "# 6. TRAINING WITH VALIDATION AND EARLY STOPPING\n",
    "# ============================================================================\n",
    "\n",
    "def train_model_with_validation(model, train_loader, val_loader, \n",
    "                                max_epochs=50, patience=7, device='cpu'):\n",
    "    \"\"\"\n",
    "    Train with validation-based early stopping\n",
    "    \"\"\"\n",
    "    print(\"=\"*70)\n",
    "    print(\"HYBRID U-NET + RESNET MANIFOLD LEARNING - TRAINING\")\n",
    "    print(f\"Latent dimension: {model.latent_dim} ({model.latent_channels}×7×7 spatial)\")\n",
    "    print(f\"ResNet blocks per resolution: {model.num_res_blocks}\")\n",
    "    print(f\"Max epochs: {max_epochs}, Patience: {patience}\")\n",
    "    print(\"=\"*70)\n",
    "    \n",
    "    model.to(device)\n",
    "    optimizer = torch.optim.Adam(model.parameters(), lr=1e-3, weight_decay=1e-5)\n",
    "    scheduler = torch.optim.lr_scheduler.ReduceLROnPlateau(\n",
    "        optimizer, mode='min', factor=0.5, patience=3\n",
    "    )\n",
    "    \n",
    "    best_val_loss = float('inf')\n",
    "    best_epoch = 0\n",
    "    epochs_without_improvement = 0\n",
    "    best_model_state = None\n",
    "    \n",
    "    train_losses = []\n",
    "    val_losses = []\n",
    "    \n",
    "    for epoch in range(max_epochs):\n",
    "        # ==================== TRAINING ====================\n",
    "        model.train()\n",
    "        total_loss = 0\n",
    "        total_recon_x = 0\n",
    "        total_recon_s = 0\n",
    "        total_consistency = 0\n",
    "        \n",
    "        for batch_idx, (x, s) in enumerate(train_loader):\n",
    "            sparse_image = x['sparse_image'].to(device)\n",
    "            mask = x['mask'].to(device)\n",
    "            full_image = s['full_image'].to(device)\n",
    "            \n",
    "            optimizer.zero_grad()\n",
    "            \n",
    "            # Encode both modalities\n",
    "            z_x = model.encoder_x(sparse_image, mask)\n",
    "            z_s = model.encoder_s(full_image)\n",
    "            \n",
    "            # Decode both\n",
    "            img_recon_x = model.decoder(z_x)\n",
    "            img_recon_s = model.decoder(z_s)\n",
    "            \n",
    "            # PRIMARY GOAL: Both should reconstruct the full image\n",
    "            loss_recon_x = F.mse_loss(img_recon_x, full_image)\n",
    "            loss_recon_s = F.mse_loss(img_recon_s, full_image)\n",
    "            \n",
    "            # SECONDARY: Spatial consistency (in feature space)\n",
    "            loss_consistency = F.mse_loss(z_x, z_s)\n",
    "            \n",
    "            # Combined loss: prioritize reconstruction, weak consistency\n",
    "            loss = loss_recon_x + loss_recon_s + 0.1 * loss_consistency\n",
    "            \n",
    "            loss.backward()\n",
    "            \n",
    "            # Gradient clipping\n",
    "            torch.nn.utils.clip_grad_norm_(model.parameters(), max_norm=1.0)\n",
    "            \n",
    "            optimizer.step()\n",
    "            \n",
    "            total_loss += loss.item()\n",
    "            total_recon_x += loss_recon_x.item()\n",
    "            total_recon_s += loss_recon_s.item()\n",
    "            total_consistency += loss_consistency.item()\n",
    "            \n",
    "            if batch_idx % 100 == 0:\n",
    "                print(f'Epoch {epoch+1}/{max_epochs}, Batch {batch_idx}, '\n",
    "                      f'Loss: {loss.item():.4f} '\n",
    "                      f'(Recon_x: {loss_recon_x.item():.4f}, '\n",
    "                      f'Recon_s: {loss_recon_s.item():.4f}, '\n",
    "                      f'Consistency: {loss_consistency.item():.4f})')\n",
    "        \n",
    "        avg_train_loss = total_loss / len(train_loader)\n",
    "        avg_train_recon_x = total_recon_x / len(train_loader)\n",
    "        avg_train_recon_s = total_recon_s / len(train_loader)\n",
    "        avg_train_consistency = total_consistency / len(train_loader)\n",
    "        train_losses.append(avg_train_loss)\n",
    "        \n",
    "        # ==================== VALIDATION ====================\n",
    "        model.eval()\n",
    "        val_loss = 0\n",
    "        val_recon_x = 0\n",
    "        val_recon_s = 0\n",
    "        val_consistency = 0\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            for x, s in val_loader:\n",
    "                sparse_image = x['sparse_image'].to(device)\n",
    "                mask = x['mask'].to(device)\n",
    "                full_image = s['full_image'].to(device)\n",
    "                \n",
    "                z_x = model.encoder_x(sparse_image, mask)\n",
    "                z_s = model.encoder_s(full_image)\n",
    "                \n",
    "                img_recon_x = model.decoder(z_x)\n",
    "                img_recon_s = model.decoder(z_s)\n",
    "                \n",
    "                loss_recon_x = F.mse_loss(img_recon_x, full_image)\n",
    "                loss_recon_s = F.mse_loss(img_recon_s, full_image)\n",
    "                loss_consistency = F.mse_loss(z_x, z_s)\n",
    "                \n",
    "                batch_loss = loss_recon_x + loss_recon_s + 0.1 * loss_consistency\n",
    "                val_loss += batch_loss.item()\n",
    "                val_recon_x += loss_recon_x.item()\n",
    "                val_recon_s += loss_recon_s.item()\n",
    "                val_consistency += loss_consistency.item()\n",
    "        \n",
    "        avg_val_loss = val_loss / len(val_loader)\n",
    "        avg_val_recon_x = val_recon_x / len(val_loader)\n",
    "        avg_val_recon_s = val_recon_s / len(val_loader)\n",
    "        avg_val_consistency = val_consistency / len(val_loader)\n",
    "        val_losses.append(avg_val_loss)\n",
    "        \n",
    "        # Learning rate scheduling\n",
    "        old_lr = optimizer.param_groups[0]['lr']\n",
    "        scheduler.step(avg_val_loss)\n",
    "        new_lr = optimizer.param_groups[0]['lr']\n",
    "        if new_lr != old_lr:\n",
    "            print(f'  → Learning rate reduced: {old_lr:.2e} → {new_lr:.2e}')\n",
    "        \n",
    "        # Print epoch summary\n",
    "        print(f'\\n{\"=\"*70}')\n",
    "        print(f'Epoch {epoch+1}/{max_epochs} Summary:')\n",
    "        print(f'  TRAIN - Total: {avg_train_loss:.4f}, Recon_x: {avg_train_recon_x:.4f}, '\n",
    "              f'Recon_s: {avg_train_recon_s:.4f}, Consistency: {avg_train_consistency:.4f}')\n",
    "        print(f'  VAL   - Total: {avg_val_loss:.4f}, Recon_x: {avg_val_recon_x:.4f}, '\n",
    "              f'Recon_s: {avg_val_recon_s:.4f}, Consistency: {avg_val_consistency:.4f}')\n",
    "        \n",
    "        # ==================== EARLY STOPPING CHECK ====================\n",
    "        if avg_val_loss < best_val_loss:\n",
    "            best_val_loss = avg_val_loss\n",
    "            best_epoch = epoch + 1\n",
    "            epochs_without_improvement = 0\n",
    "            best_model_state = {\n",
    "                'epoch': epoch + 1,\n",
    "                'encoder_x': model.encoder_x.state_dict(),\n",
    "                'encoder_s': model.encoder_s.state_dict(),\n",
    "                'decoder': model.decoder.state_dict(),\n",
    "                'val_loss': avg_val_loss,\n",
    "                'train_loss': avg_train_loss,\n",
    "            }\n",
    "            print(f'  ✓ NEW BEST MODEL! (Val loss: {best_val_loss:.4f})')\n",
    "        else:\n",
    "            epochs_without_improvement += 1\n",
    "            print(f'  No improvement for {epochs_without_improvement} epoch(s)')\n",
    "            \n",
    "            if epochs_without_improvement >= patience:\n",
    "                print(f'\\n{\"=\"*70}')\n",
    "                print(f'EARLY STOPPING at epoch {epoch+1}')\n",
    "                print(f'Best model from epoch {best_epoch} with val loss {best_val_loss:.4f}')\n",
    "                print(f'{\"=\"*70}\\n')\n",
    "                break\n",
    "        \n",
    "        print(f'{\"=\"*70}\\n')\n",
    "    \n",
    "    # Restore best model\n",
    "    if best_model_state is not None:\n",
    "        model.encoder_x.load_state_dict(best_model_state['encoder_x'])\n",
    "        model.encoder_s.load_state_dict(best_model_state['encoder_s'])\n",
    "        model.decoder.load_state_dict(best_model_state['decoder'])\n",
    "        print(f\"✓ Restored best model from epoch {best_epoch}\\n\")\n",
    "    \n",
    "    print(\"Training Complete!\\n\")\n",
    "    return model, best_model_state, train_losses, val_losses\n",
    "\n",
    "# ============================================================================\n",
    "# 7. TESTING AND EVALUATION (Same as U-Net)\n",
    "# ============================================================================\n",
    "\n",
    "def test_reconstruction(model, test_dataset, device='cpu', num_samples=5):\n",
    "    \"\"\"FAIR TEST: Given ONLY sparse image, reconstruct full image\"\"\"\n",
    "    print(\"=\"*70)\n",
    "    print(\"FAIR TESTING - SPARSE INPUT ONLY (VISUAL)\")\n",
    "    print(\"=\"*70)\n",
    "    \n",
    "    model.eval()\n",
    "    model.to(device)\n",
    "    \n",
    "    fig, axes = plt.subplots(num_samples, 3, figsize=(10, 3*num_samples))\n",
    "    if num_samples == 1:\n",
    "        axes = axes.reshape(1, -1)\n",
    "    \n",
    "    total_mse = 0\n",
    "    \n",
    "    for i in range(num_samples):\n",
    "        x, s = test_dataset[i * (len(test_dataset) // num_samples)]\n",
    "        \n",
    "        sparse_image = x['sparse_image'].unsqueeze(0).to(device)\n",
    "        mask = x['mask'].unsqueeze(0).to(device)\n",
    "        full_image = s['full_image'].unsqueeze(0).to(device)\n",
    "        digit_label = s['digit_label']\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            z_x = model.encoder_x(sparse_image, mask)\n",
    "            reconstructed = model.decoder(z_x)\n",
    "            mse = F.mse_loss(reconstructed, full_image).item()\n",
    "            total_mse += mse\n",
    "            \n",
    "            print(f\"\\nSample {i} (Digit {digit_label}):\")\n",
    "            print(f\"  Reconstruction MSE: {mse:.6f}\")\n",
    "        \n",
    "        def denorm(img):\n",
    "            return (img + 1) / 2\n",
    "        \n",
    "        axes[i, 0].imshow(denorm(full_image.squeeze().cpu()), cmap='gray', vmin=0, vmax=1)\n",
    "        axes[i, 0].set_title(f'Ground Truth (Digit: {digit_label})')\n",
    "        axes[i, 0].axis('off')\n",
    "        \n",
    "        axes[i, 1].imshow(denorm(sparse_image.squeeze().cpu()), cmap='gray', vmin=0, vmax=1)\n",
    "        axes[i, 1].set_title(f'Sparse Input ({mask.mean().item():.0%} visible)')\n",
    "        axes[i, 1].axis('off')\n",
    "        \n",
    "        axes[i, 2].imshow(denorm(reconstructed.squeeze().cpu()), cmap='gray', vmin=0, vmax=1)\n",
    "        axes[i, 2].set_title(f'Reconstructed (MSE: {mse:.4f})')\n",
    "        axes[i, 2].axis('off')\n",
    "    \n",
    "    avg_mse = total_mse / num_samples\n",
    "    print(f\"\\n{'='*70}\")\n",
    "    print(f\"AVERAGE RECONSTRUCTION MSE: {avg_mse:.6f}\")\n",
    "    print(f\"{'='*70}\")\n",
    "    \n",
    "    plt.tight_layout()\n",
    "    plt.savefig('hybrid_reconstruction_result.png', dpi=150, bbox_inches='tight')\n",
    "    print(\"\\nResults saved to 'hybrid_reconstruction_result.png'\")\n",
    "    plt.show()\n",
    "\n",
    "def evaluate_multiple_sparsity_levels(model, mnist_test, sparsity_levels, \n",
    "                                     device='cpu', num_samples=1000):\n",
    "    \"\"\"Evaluate reconstruction at multiple sparsity levels\"\"\"\n",
    "    print(\"\\n\" + \"=\"*70)\n",
    "    print(\"MULTI-SPARSITY EVALUATION\")\n",
    "    print(f\"Testing at sparsity levels: {sparsity_levels}\")\n",
    "    print(\"=\"*70 + \"\\n\")\n",
    "    \n",
    "    model.eval()\n",
    "    model.to(device)\n",
    "    \n",
    "    results = {}\n",
    "    \n",
    "    for sparsity in sparsity_levels:\n",
    "        print(f\"\\n{'─'*70}\")\n",
    "        print(f\"Evaluating at ρ = {sparsity:.2f} ({sparsity:.0%} visible pixels)\")\n",
    "        print(f\"{'─'*70}\")\n",
    "        \n",
    "        test_dataset = EnhancedMNISTDataset(mnist_test, sparsity_level=sparsity)\n",
    "        \n",
    "        mse_list = []\n",
    "        \n",
    "        for i in range(min(num_samples, len(test_dataset))):\n",
    "            x, s = test_dataset[i]\n",
    "            \n",
    "            sparse_image = x['sparse_image'].unsqueeze(0).to(device)\n",
    "            mask = x['mask'].unsqueeze(0).to(device)\n",
    "            full_image = s['full_image'].unsqueeze(0).to(device)\n",
    "            \n",
    "            with torch.no_grad():\n",
    "                z_x = model.encoder_x(sparse_image, mask)\n",
    "                reconstructed = model.decoder(z_x)\n",
    "                mse = F.mse_loss(reconstructed, full_image).item()\n",
    "                mse_list.append(mse)\n",
    "        \n",
    "        mse_array = np.array(mse_list)\n",
    "        \n",
    "        results[sparsity] = {\n",
    "            'mean': mse_array.mean(),\n",
    "            'median': np.median(mse_array),\n",
    "            'std': mse_array.std(),\n",
    "            'min': mse_array.min(),\n",
    "            'max': mse_array.max(),\n",
    "            'all_mse': mse_array\n",
    "        }\n",
    "        \n",
    "        print(f\"\\nResults for ρ = {sparsity:.2f}:\")\n",
    "        print(f\"  Mean MSE:   {results[sparsity]['mean']:.6f}\")\n",
    "        print(f\"  Median MSE: {results[sparsity]['median']:.6f}\")\n",
    "    \n",
    "    # Summary table\n",
    "    print(f\"\\n{'='*70}\")\n",
    "    print(\"SUMMARY TABLE\")\n",
    "    print(f\"{'='*70}\")\n",
    "    print(f\"{'Sparsity ρ':<15} {'Mean MSE':<15} {'Median MSE':<15} {'Std MSE':<15}\")\n",
    "    print(f\"{'-'*70}\")\n",
    "    for sparsity in sparsity_levels:\n",
    "        print(f\"{sparsity:<15.2f} \"\n",
    "              f\"{results[sparsity]['mean']:<15.6f} \"\n",
    "              f\"{results[sparsity]['median']:<15.6f} \"\n",
    "              f\"{results[sparsity]['std']:<15.6f}\")\n",
    "    print(f\"{'='*70}\\n\")\n",
    "    \n",
    "    return results\n",
    "\n",
    "# ============================================================================\n",
    "# 8. MAIN EXECUTION\n",
    "# ============================================================================\n",
    "\n",
    "if __name__ == '__main__':\n",
    "    print(\"\\n\" + \"=\"*70)\n",
    "    print(\"HYBRID U-NET + RESNET MANIFOLD LEARNING\")\n",
    "    print(\"Combining U-Net compression + ResNet gradient flow\")\n",
    "    print(\"=\"*70 + \"\\n\")\n",
    "    \n",
    "    device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "    print(f\"Using device: {device}\\n\")\n",
    "    \n",
    "    # ==================== LOAD MNIST ====================\n",
    "    transform = transforms.Compose([\n",
    "        transforms.ToTensor(),\n",
    "        transforms.Normalize((0.5,), (0.5,))\n",
    "    ])\n",
    "    \n",
    "    mnist_train_full = datasets.MNIST('./data', train=True, download=True, \n",
    "                                      transform=transform)\n",
    "    mnist_test_full = datasets.MNIST('./data', train=False, transform=transform)\n",
    "    \n",
    "    # ==================== SPLIT: 90% TRAIN, 5% VAL, 5% TEST ====================\n",
    "    train_size = int(0.90 * len(mnist_train_full))\n",
    "    val_size = int(0.05 * len(mnist_train_full))\n",
    "    \n",
    "    indices = list(range(len(mnist_train_full)))\n",
    "    np.random.seed(42)\n",
    "    np.random.shuffle(indices)\n",
    "    \n",
    "    train_indices = indices[:train_size]\n",
    "    val_indices = indices[train_size:train_size+val_size]\n",
    "    test_indices = indices[train_size+val_size:]\n",
    "    \n",
    "    mnist_train = Subset(mnist_train_full, train_indices)\n",
    "    mnist_val = Subset(mnist_train_full, val_indices)\n",
    "    mnist_test = Subset(mnist_train_full, test_indices)\n",
    "    \n",
    "    print(f\"Dataset splits:\")\n",
    "    print(f\"  Train: {len(mnist_train)} samples\")\n",
    "    print(f\"  Val:   {len(mnist_val)} samples\")\n",
    "    print(f\"  Test:  {len(mnist_test)} samples\\n\")\n",
    "    \n",
    "    # ==================== CREATE DATASETS ====================\n",
    "    TRAIN_SPARSITY = 0.10\n",
    "    \n",
    "    train_dataset = EnhancedMNISTDataset(mnist_train, sparsity_level=TRAIN_SPARSITY)\n",
    "    val_dataset = EnhancedMNISTDataset(mnist_val, sparsity_level=TRAIN_SPARSITY)\n",
    "    test_dataset = EnhancedMNISTDataset(mnist_test, sparsity_level=TRAIN_SPARSITY)\n",
    "    \n",
    "    train_loader = DataLoader(train_dataset, batch_size=128, shuffle=True, num_workers=0)\n",
    "    val_loader = DataLoader(val_dataset, batch_size=128, shuffle=False, num_workers=0)\n",
    "    \n",
    "    # ==================== INITIALIZE MODEL ====================\n",
    "    model = HybridUNetResNetModel(latent_channels=64, num_res_blocks=2)\n",
    "    total_params = sum(p.numel() for p in model.parameters())\n",
    "    print(f\"\\nModel architecture:\")\n",
    "    print(f\"  Total parameters: {total_params:,}\")\n",
    "    print(f\"  Latent dimension: {model.latent_dim} ({model.latent_channels}×7×7 spatial)\")\n",
    "    print(f\"  ResNet blocks per resolution: {model.num_res_blocks}\\n\")\n",
    "    \n",
    "    # ==================== TRAINING ====================\n",
    "    model, best_state, train_losses, val_losses = train_model_with_validation(\n",
    "        model, train_loader, val_loader, \n",
    "        max_epochs=100, patience=7, device=device\n",
    "    )\n",
    "    \n",
    "    # ==================== TESTING ====================\n",
    "    test_reconstruction(model, test_dataset, device=device, num_samples=5)\n",
    "    \n",
    "    # ==================== MULTI-SPARSITY EVALUATION ====================\n",
    "    sparsity_levels = [0.10, 0.15, 0.20, 0.25]\n",
    "    multi_results = evaluate_multiple_sparsity_levels(\n",
    "        model, mnist_test, sparsity_levels, \n",
    "        device=device, num_samples=len(mnist_test)\n",
    "    )\n",
    "    \n",
    "    # ==================== SAVE EVERYTHING ====================\n",
    "    torch.save(best_state, 'best_hybrid_model.pth')\n",
    "    print(\"✓ Best model saved to 'best_hybrid_model.pth'\")\n",
    "    \n",
    "    results_summary = {\n",
    "        str(k): {key: float(val) if key != 'all_mse' else val.tolist() \n",
    "                 for key, val in v.items()}\n",
    "        for k, v in multi_results.items()\n",
    "    }\n",
    "    \n",
    "    with open('hybrid_multi_sparsity_results.json', 'w') as f:\n",
    "        json.dump(results_summary, f, indent=2)\n",
    "    print(\"✓ Results saved to 'hybrid_multi_sparsity_results.json'\")\n",
    "    \n",
    "    history = {\n",
    "        'train_losses': train_losses,\n",
    "        'val_losses': val_losses,\n",
    "        'best_epoch': best_state['epoch'],\n",
    "        'best_val_loss': best_state['val_loss']\n",
    "    }\n",
    "    \n",
    "    with open('hybrid_training_history.json', 'w') as f:\n",
    "        json.dump(history, f, indent=2)\n",
    "    print(\"✓ Training history saved to 'hybrid_training_history.json'\")\n",
    "    \n",
    "    print(\"\\n\" + \"=\"*70)\n",
    "    print(\"COMPLETE! Hybrid U-Net + ResNet model trained successfully\")\n",
    "    print(\"=\"*70)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fe2a8348",
   "metadata": {},
   "source": [
    "## Four Metrics"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "3bd98816",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "================================================================================\n",
      "HOMEOMORPHISM VERIFICATION - THEORETICAL FRAMEWORK\n",
      "CORRECTED: Continuity on SPARSE encodings only\n",
      "Theorem 1: Topological Unification\n",
      "Proposition 1: Metric-Theoretical Bridge\n",
      "Four-Metric Verification: Trust, W₂, Alignment, Continuity\n",
      "================================================================================\n",
      "\n",
      "Loading trained U-Net model...\n",
      "Model loaded on cuda\n",
      "\n",
      "\n",
      "================================================================================\n",
      "HOMEOMORPHISM VERIFICATION - THEORETICAL FRAMEWORK\n",
      "CORRECTED: Continuity on SPARSE encodings only\n",
      "================================================================================\n",
      "Latent representation: 64×7×7 = 3136-d\n",
      "Sparsity levels: [0.1, 0.15, 0.2, 0.25]\n",
      "κ (neighbors): 5\n",
      "Distance metric: cosine\n",
      "Train size: 60000\n",
      "Test size: 10000\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 In-Class 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. In-Class Continuity (τ_c ≥ 0.80) - Manifold compactness (SPARSE only)\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "EVALUATING AT ρ = 0.10 (10% visible pixels)\n",
      "================================================================================\n",
      "Dataset size: 60000, Sparsity: 10% visible\n",
      "Dataset size: 10000, Sparsity: 10% visible\n",
      "\n",
      "Encoding 60000 TRAINING samples (SPARSE)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                  \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding 60000 TRAINING samples (FULL)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding 10000 TEST samples (SPARSE)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                              \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding 10000 TEST samples (FULL)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                             \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Encoded shapes:\n",
      "  Train Sparse: (60000, 3136)\n",
      "  Train Full:   (60000, 3136)\n",
      "  Test Sparse:  (10000, 3136)\n",
      "  Test Full:    (10000, 3136)\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.7995\n",
      "  ✗ Manifold collapse detected (c₁→0 violation)\n",
      "\n",
      "[2/4] Computing SLICED WASSERSTEIN-2 (τ_w ≤ 0.30)...\n",
      "  Sliced W₂ (Sparse↔Full): 0.0144\n",
      "  ✓ Global alignment verified (distributions overlap)\n",
      "\n",
      "[3/4] Computing IN-CLASS ALIGNMENT ERROR (τ_a ≤ 0.1)...\n",
      "  Distance metric: cosine\n",
      "  In-Class Alignment Error (avg): 0.0274\n",
      "  Global Alignment Error:         0.0273\n",
      "  ✓ Sparse and full map to THE SAME POINTS (perfect merger)\n",
      "  Per-class alignment errors:\n",
      "    Digit 0: 0.0299 ✓\n",
      "    Digit 1: 0.0186 ✓\n",
      "    Digit 2: 0.0313 ✓\n",
      "    Digit 3: 0.0311 ✓\n",
      "    Digit 4: 0.0275 ✓\n",
      "    Digit 5: 0.0281 ✓\n",
      "    Digit 6: 0.0274 ✓\n",
      "    Digit 7: 0.0273 ✓\n",
      "    Digit 8: 0.0275 ✓\n",
      "    Digit 9: 0.0250 ✓\n",
      "\n",
      "[4/4] Computing IN-CLASS CONTINUITY (τ_c ≥ 0.80)...\n",
      "  CORRECTED: Measuring on SPARSE encodings only\n",
      "  In-Class Continuity (Sparse): 0.9058\n",
      "  ✓ Manifold classes form coherent, compact clusters (c₂ < ∞)\n",
      "  Per-class continuity:\n",
      "    Digit 0: 0.9241 ✓\n",
      "    Digit 1: 0.8940 ✓\n",
      "    Digit 2: 0.9160 ✓\n",
      "    Digit 3: 0.9039 ✓\n",
      "    Digit 4: 0.9073 ✓\n",
      "    Digit 5: 0.9080 ✓\n",
      "    Digit 6: 0.9037 ✓\n",
      "    Digit 7: 0.8916 ✓\n",
      "    Digit 8: 0.9085 ✓\n",
      "    Digit 9: 0.9009 ✓\n",
      "\n",
      "================================================================================\n",
      "SUMMARY FOR ρ = 0.10:\n",
      "  β₀:                1\n",
      "  Trust Score:       0.7995 ✗\n",
      "  Sliced W₂:         0.0144 ✓\n",
      "  Alignment Error:   0.0274 ✓\n",
      "  In-Class Cont:     0.9058 ✓\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "EVALUATING AT ρ = 0.15 (15% visible pixels)\n",
      "================================================================================\n",
      "Dataset size: 60000, Sparsity: 15% visible\n",
      "Dataset size: 10000, Sparsity: 15% visible\n",
      "\n",
      "Encoding 60000 TRAINING samples (SPARSE)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                 \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding 60000 TRAINING samples (FULL)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                               \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding 10000 TEST samples (SPARSE)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                              \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding 10000 TEST samples (FULL)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                            \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Encoded shapes:\n",
      "  Train Sparse: (60000, 3136)\n",
      "  Train Full:   (60000, 3136)\n",
      "  Test Sparse:  (10000, 3136)\n",
      "  Test Full:    (10000, 3136)\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.9018\n",
      "  ✓ Local preservation verified (c₁ > 0)\n",
      "\n",
      "[2/4] Computing SLICED WASSERSTEIN-2 (τ_w ≤ 0.30)...\n",
      "  Sliced W₂ (Sparse↔Full): 0.0190\n",
      "  ✓ Global alignment verified (distributions overlap)\n",
      "\n",
      "[3/4] Computing IN-CLASS ALIGNMENT ERROR (τ_a ≤ 0.1)...\n",
      "  Distance metric: cosine\n",
      "  In-Class Alignment Error (avg): 0.0300\n",
      "  Global Alignment Error:         0.0299\n",
      "  ✓ Sparse and full map to THE SAME POINTS (perfect merger)\n",
      "  Per-class alignment errors:\n",
      "    Digit 0: 0.0330 ✓\n",
      "    Digit 1: 0.0202 ✓\n",
      "    Digit 2: 0.0340 ✓\n",
      "    Digit 3: 0.0341 ✓\n",
      "    Digit 4: 0.0302 ✓\n",
      "    Digit 5: 0.0304 ✓\n",
      "    Digit 6: 0.0301 ✓\n",
      "    Digit 7: 0.0299 ✓\n",
      "    Digit 8: 0.0303 ✓\n",
      "    Digit 9: 0.0277 ✓\n",
      "\n",
      "[4/4] Computing IN-CLASS CONTINUITY (τ_c ≥ 0.80)...\n",
      "  CORRECTED: Measuring on SPARSE encodings only\n",
      "  In-Class Continuity (Sparse): 0.8989\n",
      "  ✓ Manifold classes form coherent, compact clusters (c₂ < ∞)\n",
      "  Per-class continuity:\n",
      "    Digit 0: 0.9150 ✓\n",
      "    Digit 1: 0.8762 ✓\n",
      "    Digit 2: 0.9129 ✓\n",
      "    Digit 3: 0.9007 ✓\n",
      "    Digit 4: 0.9035 ✓\n",
      "    Digit 5: 0.9023 ✓\n",
      "    Digit 6: 0.8980 ✓\n",
      "    Digit 7: 0.8824 ✓\n",
      "    Digit 8: 0.9051 ✓\n",
      "    Digit 9: 0.8928 ✓\n",
      "\n",
      "================================================================================\n",
      "SUMMARY FOR ρ = 0.15:\n",
      "  β₀:                1\n",
      "  Trust Score:       0.9018 ✓\n",
      "  Sliced W₂:         0.0190 ✓\n",
      "  Alignment Error:   0.0300 ✓\n",
      "  In-Class Cont:     0.8989 ✓\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "EVALUATING AT ρ = 0.20 (20% visible pixels)\n",
      "================================================================================\n",
      "Dataset size: 60000, Sparsity: 20% visible\n",
      "Dataset size: 10000, Sparsity: 20% visible\n",
      "\n",
      "Encoding 60000 TRAINING samples (SPARSE)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                 \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding 60000 TRAINING samples (FULL)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                               \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding 10000 TEST samples (SPARSE)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                              \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding 10000 TEST samples (FULL)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                            \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Encoded shapes:\n",
      "  Train Sparse: (60000, 3136)\n",
      "  Train Full:   (60000, 3136)\n",
      "  Test Sparse:  (10000, 3136)\n",
      "  Test Full:    (10000, 3136)\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.9371\n",
      "  ✓ Local preservation verified (c₁ > 0)\n",
      "\n",
      "[2/4] Computing SLICED WASSERSTEIN-2 (τ_w ≤ 0.30)...\n",
      "  Sliced W₂ (Sparse↔Full): 0.0237\n",
      "  ✓ Global alignment verified (distributions overlap)\n",
      "\n",
      "[3/4] Computing IN-CLASS ALIGNMENT ERROR (τ_a ≤ 0.1)...\n",
      "  Distance metric: cosine\n",
      "  In-Class Alignment Error (avg): 0.0337\n",
      "  Global Alignment Error:         0.0336\n",
      "  ✓ Sparse and full map to THE SAME POINTS (perfect merger)\n",
      "  Per-class alignment errors:\n",
      "    Digit 0: 0.0373 ✓\n",
      "    Digit 1: 0.0227 ✓\n",
      "    Digit 2: 0.0379 ✓\n",
      "    Digit 3: 0.0383 ✓\n",
      "    Digit 4: 0.0340 ✓\n",
      "    Digit 5: 0.0337 ✓\n",
      "    Digit 6: 0.0340 ✓\n",
      "    Digit 7: 0.0337 ✓\n",
      "    Digit 8: 0.0344 ✓\n",
      "    Digit 9: 0.0314 ✓\n",
      "\n",
      "[4/4] Computing IN-CLASS CONTINUITY (τ_c ≥ 0.80)...\n",
      "  CORRECTED: Measuring on SPARSE encodings only\n",
      "  In-Class Continuity (Sparse): 0.8896\n",
      "  ✓ Manifold classes form coherent, compact clusters (c₂ < ∞)\n",
      "  Per-class continuity:\n",
      "    Digit 0: 0.9064 ✓\n",
      "    Digit 1: 0.8609 ✓\n",
      "    Digit 2: 0.9073 ✓\n",
      "    Digit 3: 0.8927 ✓\n",
      "    Digit 4: 0.8951 ✓\n",
      "    Digit 5: 0.8955 ✓\n",
      "    Digit 6: 0.8890 ✓\n",
      "    Digit 7: 0.8700 ✓\n",
      "    Digit 8: 0.8973 ✓\n",
      "    Digit 9: 0.8815 ✓\n",
      "\n",
      "================================================================================\n",
      "SUMMARY FOR ρ = 0.20:\n",
      "  β₀:                1\n",
      "  Trust Score:       0.9371 ✓\n",
      "  Sliced W₂:         0.0237 ✓\n",
      "  Alignment Error:   0.0337 ✓\n",
      "  In-Class Cont:     0.8896 ✓\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "EVALUATING AT ρ = 0.25 (25% visible pixels)\n",
      "================================================================================\n",
      "Dataset size: 60000, Sparsity: 25% visible\n",
      "Dataset size: 10000, Sparsity: 25% visible\n",
      "\n",
      "Encoding 60000 TRAINING samples (SPARSE)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                 \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding 60000 TRAINING samples (FULL)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                                \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding 10000 TEST samples (SPARSE)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                              \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Encoding 10000 TEST samples (FULL)...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                             \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Encoded shapes:\n",
      "  Train Sparse: (60000, 3136)\n",
      "  Train Full:   (60000, 3136)\n",
      "  Test Sparse:  (10000, 3136)\n",
      "  Test Full:    (10000, 3136)\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.9525\n",
      "  ✓ Local preservation verified (c₁ > 0)\n",
      "\n",
      "[2/4] Computing SLICED WASSERSTEIN-2 (τ_w ≤ 0.30)...\n",
      "  Sliced W₂ (Sparse↔Full): 0.0271\n",
      "  ✓ Global alignment verified (distributions overlap)\n",
      "\n",
      "[3/4] Computing IN-CLASS ALIGNMENT ERROR (τ_a ≤ 0.1)...\n",
      "  Distance metric: cosine\n",
      "  In-Class Alignment Error (avg): 0.0378\n",
      "  Global Alignment Error:         0.0377\n",
      "  ✓ Sparse and full map to THE SAME POINTS (perfect merger)\n",
      "  Per-class alignment errors:\n",
      "    Digit 0: 0.0420 ✓\n",
      "    Digit 1: 0.0256 ✓\n",
      "    Digit 2: 0.0422 ✓\n",
      "    Digit 3: 0.0429 ✓\n",
      "    Digit 4: 0.0379 ✓\n",
      "    Digit 5: 0.0375 ✓\n",
      "    Digit 6: 0.0383 ✓\n",
      "    Digit 7: 0.0377 ✓\n",
      "    Digit 8: 0.0387 ✓\n",
      "    Digit 9: 0.0357 ✓\n",
      "\n",
      "[4/4] Computing IN-CLASS CONTINUITY (τ_c ≥ 0.80)...\n",
      "  CORRECTED: Measuring on SPARSE encodings only\n",
      "  In-Class Continuity (Sparse): 0.8817\n",
      "  ✓ Manifold classes form coherent, compact clusters (c₂ < ∞)\n",
      "  Per-class continuity:\n",
      "    Digit 0: 0.8987 ✓\n",
      "    Digit 1: 0.8486 ✓\n",
      "    Digit 2: 0.9022 ✓\n",
      "    Digit 3: 0.8870 ✓\n",
      "    Digit 4: 0.8886 ✓\n",
      "    Digit 5: 0.8878 ✓\n",
      "    Digit 6: 0.8801 ✓\n",
      "    Digit 7: 0.8606 ✓\n",
      "    Digit 8: 0.8912 ✓\n",
      "    Digit 9: 0.8717 ✓\n",
      "\n",
      "================================================================================\n",
      "SUMMARY FOR ρ = 0.25:\n",
      "  β₀:                1\n",
      "  Trust Score:       0.9525 ✓\n",
      "  Sliced W₂:         0.0271 ✓\n",
      "  Alignment Error:   0.0378 ✓\n",
      "  In-Class Cont:     0.8817 ✓\n",
      "================================================================================\n",
      "\n",
      "================================================================================\n",
      "TOPOLOGICAL UNIFICATION VERIFICATION (THEOREM 1)\n",
      "================================================================================\n",
      "ρ        β₀     Trust      W₂         Align      Cont      \n",
      "--------------------------------------------------------------------------------\n",
      "0.10     1      0.7995     0.0144     0.0274     0.9058    \n",
      "0.15     1      0.9018     0.0190     0.0300     0.8989    \n",
      "0.20     1      0.9371     0.0237     0.0337     0.8896    \n",
      "0.25     1      0.9525     0.0271     0.0378     0.8817    \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: ✗ FAIL\n",
      "  β₀:        ✓ (connectivity)\n",
      "  Trust:     ✗ (local preservation)\n",
      "  W₂:        ✓ (global alignment)\n",
      "  Align:     ✓ (coordinate matching)\n",
      "  Cont:      ✓ (manifold compactness)\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",
      "ρ = 0.20: ✓ PASS\n",
      "  β₀:        ✓ (connectivity)\n",
      "  Trust:     ✓ (local preservation)\n",
      "  W₂:        ✓ (global alignment)\n",
      "  Align:     ✓ (coordinate matching)\n",
      "  Cont:      ✓ (manifold compactness)\n",
      "\n",
      "ρ = 0.25: ✓ 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 VIOLATED\n",
      "  Check individual metrics above for failure modes\n",
      "================================================================================\n",
      "\n",
      "\n",
      "================================================================================\n",
      "LaTeX Table (Theoretical Framework - 4 Metrics):\n",
      "CORRECTED: Continuity measured on SPARSE encodings only\n",
      "================================================================================\n",
      "\\begin{table}[!htpb]\n",
      "\\centering\n",
      "\\caption{Topological Unification Verification ($\\kappa=5$, Cosine distance)}\n",
      "\\label{tab:unification}\n",
      "\\begin{tabular}{lcccc}\n",
      "\\hline\n",
      "\\textbf{Metric} & $\\rho=0.10$ & $\\rho=0.15$ & $\\rho=0.20$ & $\\rho=0.25$ \\\\\n",
      "\\hline\n",
      "Trust $\\tau_t$ ($\\geq 0.80$) & 0.800 & 0.902 & 0.937 & 0.953 \\\\\n",
      "Sliced $W_2$ $\\tau_w$ ($\\leq 0.30$) & 0.014 & 0.019 & 0.024 & 0.027 \\\\\n",
      "Alignment $\\tau_a$ ($\\leq 0.10$) & 0.027 & 0.030 & 0.034 & 0.038 \\\\\n",
      "Continuity $\\tau_c$ ($\\geq 0.80$) & 0.906 & 0.899 & 0.890 & 0.882 \\\\\n",
      "\\hline\n",
      "\\end{tabular}\n",
      "\\end{table}\n",
      "================================================================================\n",
      "\n",
      "✓ Results saved to 'theoretical_verification_results_cosine_corrected.json'\n",
      "\n",
      "================================================================================\n",
      "VERIFICATION COMPLETE!\n",
      "================================================================================\n"
     ]
    }
   ],
   "source": [
    "# ============================================================================\n",
    "# HOMEOMORPHISM VERIFICATION FOR U-NET MANIFOLD MODEL\n",
    "# CORRECTED: Continuity measured on SPARSE encodings only (manifold structure)\n",
    "# Four Metrics: Trust, Sliced Wasserstein, Alignment, In-Class Continuity\n",
    "# ============================================================================\n",
    "\n",
    "import torch\n",
    "import torch.nn.functional as F\n",
    "from torch.utils.data import DataLoader\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",
    "\n",
    "# Import from training code (assumes same notebook/file)\n",
    "# from training_code import EnhancedMNISTDataset, HybridUNetResNetModel\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",
    "    from Proposition 1(iii).\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, labels, kappa=5):\n",
    "    \"\"\"\n",
    "    METRIC 1: Trust Score (τ_t ≥ 0.80)\n",
    "    \n",
    "    Theoretical Link: Empirical proxy for local preservation and injectivity \n",
    "    (c₁ > 0) from Theorem 2 and Proposition 1(i).\n",
    "    \n",
    "    Measures: Do k-nearest neighbors in latent space share the same semantic label?\n",
    "    \n",
    "    Interpretation:\n",
    "    - High Trust (≥0.80): Local topology preserved, encoder is injective\n",
    "    - Low Trust (<0.80): Manifold collapse (distinct semantics merge)\n",
    "    \n",
    "    Args:\n",
    "        latent_codes: (N, D) array of latent representations\n",
    "        labels: (N,) array of semantic labels (digit classes)\n",
    "        kappa: number of nearest neighbors\n",
    "    \n",
    "    Returns:\n",
    "        trust_score: fraction of neighbors with matching labels\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: fraction of neighbors with same label\n",
    "    trust_scores = []\n",
    "    for i in range(N):\n",
    "        neighbor_labels = labels[neighbors[i]]\n",
    "        same_label_count = np.sum(neighbor_labels == labels[i])\n",
    "        trust_scores.append(same_label_count / kappa)\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^[i] and P_z^[j]\n",
    "    \n",
    "    Interpretation:\n",
    "    - Low W₂ (≤0.30): Distributions overlap substantially (alignment verified)\n",
    "    - High W₂ (>0.30): Distributions are globally distant (misaligned)\n",
    "    \n",
    "    Args:\n",
    "        X: (N1, D) samples from distribution 1\n",
    "        Y: (N2, D) samples from distribution 2\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_in_class_alignment_error(latent_codes_sparse, latent_codes_full, \n",
    "                                     labels_sparse, labels_full, \n",
    "                                     distance_metric='euclidean'):\n",
    "    \"\"\"\n",
    "    METRIC 3: In-Class Alignment Error (τ_a ≤ 0.15 for Euclidean, ≤ 0.10 for Cosine)\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",
    "    verifying topological merger.\n",
    "    \n",
    "    Measures: Distance between z_sparse(x_i) and z_full(x_i) within each class\n",
    "    \n",
    "    This is the STRONGEST test of unification:\n",
    "    - Wasserstein checks if DISTRIBUTIONS overlap\n",
    "    - Alignment checks if INDIVIDUAL POINTS match\n",
    "    \n",
    "    Interpretation:\n",
    "    - Low Error: Sparse and full use IDENTICAL coordinate systems (perfect merger)\n",
    "    - High Error: Sparse and full use DIFFERENT coordinate systems (not unified)\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",
    "        labels_sparse: (N,) class labels for sparse data\n",
    "        labels_full: (N,) class labels for full data (should match)\n",
    "        distance_metric: 'euclidean' or 'cosine'\n",
    "    \n",
    "    Returns:\n",
    "        avg_error: average alignment error per class\n",
    "        per_class_error: dict mapping class_id -> alignment error\n",
    "        global_error: alignment error across all samples (not class-specific)\n",
    "    \"\"\"\n",
    "    unique_classes = np.intersect1d(np.unique(labels_sparse), \n",
    "                                    np.unique(labels_full))\n",
    "    \n",
    "    per_class_error = {}\n",
    "    \n",
    "    for c in unique_classes:\n",
    "        mask = (labels_sparse == c)\n",
    "        \n",
    "        sparse_class = latent_codes_sparse[mask]\n",
    "        full_class = latent_codes_full[mask]\n",
    "        \n",
    "        if len(sparse_class) == 0:\n",
    "            continue\n",
    "        \n",
    "        # Compute distance between paired encodings\n",
    "        if distance_metric == 'euclidean':\n",
    "            # L2 distance: ||z_sparse - z_full||\n",
    "            distances = np.linalg.norm(sparse_class - full_class, axis=1)\n",
    "        elif distance_metric == 'cosine':\n",
    "            # Cosine distance: 1 - cos(z_sparse, z_full)\n",
    "            # Note: sklearn's cosine_distances already returns 1 - cosine_similarity\n",
    "            distances = np.diag(cosine_distances(sparse_class, full_class))\n",
    "        else:\n",
    "            raise ValueError(f\"Unknown distance metric: {distance_metric}\")\n",
    "        \n",
    "        per_class_error[int(c)] = float(np.mean(distances))\n",
    "    \n",
    "    # Average across all classes\n",
    "    avg_error = np.mean(list(per_class_error.values()))\n",
    "    \n",
    "    # Also compute global error (all samples, not class-specific)\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",
    "    else:\n",
    "        global_error = float(np.mean(np.diag(cosine_distances(\n",
    "            latent_codes_sparse, latent_codes_full\n",
    "        ))))\n",
    "    \n",
    "    return avg_error, per_class_error, global_error\n",
    "\n",
    "\n",
    "def compute_in_class_continuity(latent_codes, labels, kappa=5):\n",
    "    \"\"\"\n",
    "    METRIC 4: In-Class Continuity (τ_c ≥ 0.80)\n",
    "    \n",
    "    CORRECTED: Measures continuity of the MANIFOLD structure from a SINGLE encoder\n",
    "    (sparse OR full, not both combined).\n",
    "    \n",
    "    Theoretical Link: Ensures bounded stretching (c₂ < ∞) within the manifold\n",
    "    defined by one input condition.\n",
    "    \n",
    "    Measures: Within each class, uniformity of local neighborhood structure\n",
    "    \n",
    "    Interpretation:\n",
    "    - High Continuity (≥0.80): Classes form coherent, compact clusters\n",
    "    - Low Continuity (<0.80): Classes are fragmented/scattered (metric distortion)\n",
    "    \n",
    "    Args:\n",
    "        latent_codes: (N, D) latent representations from ONE encoder only\n",
    "        labels: (N,) semantic labels\n",
    "        kappa: number of nearest neighbors\n",
    "    \n",
    "    Returns:\n",
    "        continuity_score: average in-class compactness score\n",
    "        per_class_continuity: dict mapping class_id to continuity score\n",
    "    \"\"\"\n",
    "    unique_classes = np.unique(labels)\n",
    "    per_class_continuity = {}\n",
    "    \n",
    "    for c in unique_classes:\n",
    "        mask = (labels == c)\n",
    "        class_latents = latent_codes[mask]\n",
    "        \n",
    "        if len(class_latents) < kappa + 1:\n",
    "            continue\n",
    "        \n",
    "        # Within this class, build k-NN\n",
    "        nn = NearestNeighbors(n_neighbors=kappa+1, metric='euclidean', \n",
    "                             algorithm='auto', n_jobs=-1)\n",
    "        nn.fit(class_latents)\n",
    "        distances, _ = nn.kneighbors(class_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",
    "        # Transform to [0, 1] where 1 is best\n",
    "        cv = std_dist / (mean_dist + 1e-8)\n",
    "        continuity = 1.0 / (1.0 + cv)\n",
    "        \n",
    "        per_class_continuity[int(c)] = float(continuity)\n",
    "    \n",
    "    avg_continuity = np.mean(list(per_class_continuity.values()))\n",
    "    return avg_continuity, per_class_continuity\n",
    "\n",
    "\n",
    "def verify_homeomorphism_theoretical(model, mnist_train, mnist_test, \n",
    "                                    sparsity_levels=[0.10, 0.15, 0.20, 0.25],\n",
    "                                    kappa=5, device='cpu', n_projections=100,\n",
    "                                    distance_metric='euclidean'):\n",
    "    \"\"\"\n",
    "    Verify Topological Unification (Theorem 1) via Discrete Metrics (Proposition 1)\n",
    "    \n",
    "    CORRECTED: Continuity measured on SPARSE encodings only (manifold structure)\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 In-Class 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: In-Class Continuity (τ_c ≥ 0.80) - Manifold compactness (c₂ < ∞)\n",
    "    \n",
    "    Args:\n",
    "        model: Trained HybridUNetResNetModel\n",
    "        mnist_train: MNIST training dataset\n",
    "        mnist_test: MNIST test dataset\n",
    "        sparsity_levels: List of sparsity ratios to test\n",
    "        kappa: Number of nearest neighbors (κ=5)\n",
    "        device: Computing device\n",
    "        n_projections: Random projections for Sliced Wasserstein\n",
    "        distance_metric: 'euclidean' or 'cosine' for alignment error\n",
    "    \n",
    "    Returns:\n",
    "        results: Dictionary with all metrics per sparsity level\n",
    "    \"\"\"\n",
    "    print(\"\\n\" + \"=\"*80)\n",
    "    print(\"HOMEOMORPHISM VERIFICATION - THEORETICAL FRAMEWORK\")\n",
    "    print(\"CORRECTED: Continuity on SPARSE encodings only\")\n",
    "    print(\"=\"*80)\n",
    "    print(f\"Latent representation: {model.latent_channels}×7×7 = {model.latent_dim}-d\")\n",
    "    print(f\"Sparsity levels: {sparsity_levels}\")\n",
    "    print(f\"κ (neighbors): {kappa}\")\n",
    "    print(f\"Distance metric: {distance_metric}\")\n",
    "    print(f\"Train size: {len(mnist_train)}\")\n",
    "    print(f\"Test size: {len(mnist_test)}\")\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 In-Class 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. In-Class 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",
    "        # Create datasets\n",
    "        train_dataset = EnhancedMNISTDataset(mnist_train, sparsity_level=sparsity)\n",
    "        test_dataset = EnhancedMNISTDataset(mnist_test, sparsity_level=sparsity)\n",
    "        \n",
    "        # ====================================================================\n",
    "        # ENCODE TRAINING DATA (SPARSE)\n",
    "        # ====================================================================\n",
    "        print(f\"\\nEncoding {len(train_dataset)} TRAINING samples (SPARSE)...\")\n",
    "        latent_codes_train_sparse = []\n",
    "        labels_train = []\n",
    "        \n",
    "        train_loader = DataLoader(train_dataset, batch_size=128, \n",
    "                                 shuffle=False, num_workers=0)\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            for x, s in tqdm(train_loader, desc=\"Train (Sparse)\", leave=False):\n",
    "                sparse_image = x['sparse_image'].to(device)\n",
    "                mask = x['mask'].to(device)\n",
    "                \n",
    "                z_spatial = model.encoder_x(sparse_image, mask)\n",
    "                z_flat = z_spatial.view(z_spatial.size(0), -1)\n",
    "                \n",
    "                latent_codes_train_sparse.append(z_flat.cpu().numpy())\n",
    "                \n",
    "                if isinstance(s['digit_label'], torch.Tensor):\n",
    "                    labels_train.extend(s['digit_label'].cpu().numpy())\n",
    "                else:\n",
    "                    labels_train.extend(s['digit_label'])\n",
    "        \n",
    "        latent_codes_train_sparse = np.vstack(latent_codes_train_sparse)\n",
    "        labels_train = np.array(labels_train)\n",
    "        \n",
    "        # ====================================================================\n",
    "        # ENCODE TRAINING DATA (FULL) - Same samples for comparison\n",
    "        # ====================================================================\n",
    "        print(f\"Encoding {len(train_dataset)} TRAINING samples (FULL)...\")\n",
    "        latent_codes_train_full = []\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            for x, s in tqdm(train_loader, desc=\"Train (Full)\", leave=False):\n",
    "                full_image = s['full_image'].to(device)\n",
    "                \n",
    "                z_spatial = model.encoder_s(full_image)\n",
    "                z_flat = z_spatial.view(z_spatial.size(0), -1)\n",
    "                \n",
    "                latent_codes_train_full.append(z_flat.cpu().numpy())\n",
    "        \n",
    "        latent_codes_train_full = np.vstack(latent_codes_train_full)\n",
    "        \n",
    "        # ====================================================================\n",
    "        # ENCODE TEST DATA (SPARSE)\n",
    "        # ====================================================================\n",
    "        print(f\"Encoding {len(test_dataset)} TEST samples (SPARSE)...\")\n",
    "        latent_codes_test_sparse = []\n",
    "        labels_test = []\n",
    "        \n",
    "        test_loader = DataLoader(test_dataset, batch_size=128, \n",
    "                                shuffle=False, num_workers=0)\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            for x, s in tqdm(test_loader, desc=\"Test (Sparse)\", leave=False):\n",
    "                sparse_image = x['sparse_image'].to(device)\n",
    "                mask = x['mask'].to(device)\n",
    "                \n",
    "                z_spatial = model.encoder_x(sparse_image, mask)\n",
    "                z_flat = z_spatial.view(z_spatial.size(0), -1)\n",
    "                \n",
    "                latent_codes_test_sparse.append(z_flat.cpu().numpy())\n",
    "                \n",
    "                if isinstance(s['digit_label'], torch.Tensor):\n",
    "                    labels_test.extend(s['digit_label'].cpu().numpy())\n",
    "                else:\n",
    "                    labels_test.extend(s['digit_label'])\n",
    "        \n",
    "        latent_codes_test_sparse = np.vstack(latent_codes_test_sparse)\n",
    "        labels_test = np.array(labels_test)\n",
    "        \n",
    "        # ====================================================================\n",
    "        # ENCODE TEST DATA (FULL)\n",
    "        # ====================================================================\n",
    "        print(f\"Encoding {len(test_dataset)} TEST samples (FULL)...\")\n",
    "        latent_codes_test_full = []\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            for x, s in tqdm(test_loader, desc=\"Test (Full)\", leave=False):\n",
    "                full_image = s['full_image'].to(device)\n",
    "                \n",
    "                z_spatial = model.encoder_s(full_image)\n",
    "                z_flat = z_spatial.view(z_spatial.size(0), -1)\n",
    "                \n",
    "                latent_codes_test_full.append(z_flat.cpu().numpy())\n",
    "        \n",
    "        latent_codes_test_full = np.vstack(latent_codes_test_full)\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\"  Test Sparse:  {latent_codes_test_sparse.shape}\")\n",
    "        print(f\"  Test Full:    {latent_codes_test_full.shape}\")\n",
    "        \n",
    "        N_train = len(latent_codes_train_sparse)\n",
    "        N_test = len(latent_codes_test_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, test_sparse, test_full\n",
    "        latent_codes_all = np.vstack([\n",
    "            latent_codes_train_sparse,\n",
    "            latent_codes_train_full,\n",
    "            latent_codes_test_sparse,\n",
    "            latent_codes_test_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+test)\n",
    "        latent_codes_sparse_all = np.vstack([latent_codes_train_sparse, \n",
    "                                             latent_codes_test_sparse])\n",
    "        labels_all = np.concatenate([labels_train, labels_test])\n",
    "        \n",
    "        trust_score = compute_trust_score(latent_codes_sparse_all, labels_all, \n",
    "                                         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_samples = 5000\n",
    "        if N_train > max_samples:\n",
    "            idx = np.random.choice(N_train, max_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: IN-CLASS 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 IN-CLASS ALIGNMENT ERROR (τ_a ≤ {threshold_align})...\")\n",
    "        print(f\"  Distance metric: {distance_metric}\")\n",
    "        \n",
    "        # Compute on COMBINED train+test for robust statistics\n",
    "        latent_codes_sparse_all = np.vstack([latent_codes_train_sparse, \n",
    "                                             latent_codes_test_sparse])\n",
    "        latent_codes_full_all = np.vstack([latent_codes_train_full, \n",
    "                                           latent_codes_test_full])\n",
    "        labels_all = np.concatenate([labels_train, labels_test])\n",
    "        \n",
    "        alignment_error, per_class_error, global_error = compute_in_class_alignment_error(\n",
    "            latent_codes_sparse_all,\n",
    "            latent_codes_full_all,\n",
    "            labels_all,\n",
    "            labels_all,  # Same labels (same samples, different encodings)\n",
    "            distance_metric=distance_metric\n",
    "        )\n",
    "        \n",
    "        print(f\"  In-Class 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-class alignment errors:\")\n",
    "        for class_id in sorted(per_class_error.keys()):\n",
    "            status = \"✓\" if per_class_error[class_id] <= threshold_align else \"✗\"\n",
    "            print(f\"    Digit {class_id}: {per_class_error[class_id]:.4f} {status}\")\n",
    "        \n",
    "        # ====================================================================\n",
    "        # METRIC 4: IN-CLASS CONTINUITY (Manifold Compactness - SPARSE ONLY)\n",
    "        # ====================================================================\n",
    "        print(f\"\\n[4/4] Computing IN-CLASS CONTINUITY (τ_c ≥ 0.80)...\")\n",
    "        print(f\"  CORRECTED: Measuring on SPARSE encodings only\")\n",
    "        \n",
    "        # CORRECTED: Compute on SPARSE encodings ONLY (not combined)\n",
    "        continuity_score, per_class_continuity = compute_in_class_continuity(\n",
    "            latent_codes_sparse_all,  # ONLY sparse, not combined!\n",
    "            labels_all,\n",
    "            kappa=kappa\n",
    "        )\n",
    "        \n",
    "        print(f\"  In-Class Continuity (Sparse): {continuity_score:.4f}\")\n",
    "        if continuity_score >= 0.80:\n",
    "            print(f\"  ✓ Manifold classes form coherent, compact clusters (c₂ < ∞)\")\n",
    "        else:\n",
    "            print(f\"  ⚠ Classes may be fragmented (metric distortion, c₂→∞)\")\n",
    "        \n",
    "        print(f\"  Per-class continuity:\")\n",
    "        for class_id in sorted(per_class_continuity.keys()):\n",
    "            status = \"✓\" if per_class_continuity[class_id] >= 0.80 else \"✗\"\n",
    "            print(f\"    Digit {class_id}: {per_class_continuity[class_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_class_alignment': per_class_error,\n",
    "            \n",
    "            # Metric 4: In-Class Continuity (manifold compactness - SPARSE)\n",
    "            'in_class_continuity': float(continuity_score),\n",
    "            'per_class_continuity': per_class_continuity,\n",
    "            \n",
    "            # Metadata\n",
    "            'kappa': kappa,\n",
    "            'distance_metric': distance_metric,\n",
    "            'n_train': N_train,\n",
    "            'n_test': N_test,\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\"  In-Class Cont:     {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['in_class_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['in_class_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 classes 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 class clusters.\")\n",
    "        print()\n",
    "        print(\"This manifold is suitable for:\")\n",
    "        print(\"  • Zero-Shot Learning (unseen samples → correct clusters)\")\n",
    "        print(\"  • Linear Classification (compact clusters → separable)\")\n",
    "        print(\"  • Transfer Learning (sparsity-invariant representations)\")\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_theoretical(results, kappa=5, distance_metric='euclidean'):\n",
    "    \"\"\"\n",
    "    Generate LaTeX table aligned with theoretical framework (4 metrics)\n",
    "    CORRECTED: Continuity on SPARSE only\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 (Theoretical Framework - 4 Metrics):\")\n",
    "    print(\"CORRECTED: Continuity measured on SPARSE encodings only\")\n",
    "    print(\"=\"*80)\n",
    "    print(r\"\\begin{table}[!htpb]\")\n",
    "    print(r\"\\centering\")\n",
    "    print(r\"\\caption{Topological Unification Verification ($\\kappa=\" + str(kappa) + \n",
    "          r\"$, \" + distance_metric.capitalize() + r\" distance)}\")\n",
    "    print(r\"\\label{tab:unification}\")\n",
    "    print(r\"\\begin{tabular}{lcccc}\")\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",
    "    # 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",
    "    # In-Class Continuity (CORRECTED)\n",
    "    cont_row = r\"Continuity $\\tau_c$ ($\\geq 0.80$)\"\n",
    "    for rho in sparsity_levels:\n",
    "        val = results[rho]['in_class_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 (Same Notebook as Training)\n",
    "# ============================================================================\n",
    "\n",
    "if __name__ == '__main__':\n",
    "    from torchvision import datasets, transforms\n",
    "    \n",
    "    print(\"=\"*80)\n",
    "    print(\"HOMEOMORPHISM VERIFICATION - THEORETICAL FRAMEWORK\")\n",
    "    print(\"CORRECTED: Continuity on SPARSE encodings only\")\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 U-Net model...\")\n",
    "    model = HybridUNetResNetModel(latent_channels=64)\n",
    "    \n",
    "    checkpoint = torch.load('best_hybrid_model.pth', map_location='cpu')\n",
    "    model.encoder_x.load_state_dict(checkpoint['encoder_x'])\n",
    "    model.encoder_s.load_state_dict(checkpoint['encoder_s'])\n",
    "    model.decoder.load_state_dict(checkpoint['decoder'])\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",
    "    # Load MNIST\n",
    "    transform = transforms.Compose([\n",
    "        transforms.ToTensor(),\n",
    "        transforms.Normalize((0.5,), (0.5,))\n",
    "    ])\n",
    "    \n",
    "    mnist_train = datasets.MNIST('./data', train=True, download=True, \n",
    "                                transform=transform)\n",
    "    mnist_test = datasets.MNIST('./data', train=False, transform=transform)\n",
    "    \n",
    "    # Run verification\n",
    "    sparsity_levels = [0.10, 0.15, 0.20, 0.25]\n",
    "    \n",
    "    # Choose distance metric: 'euclidean' or 'cosine'\n",
    "    DISTANCE_METRIC = 'cosine'  # Change to 'euclidean' if desired\n",
    "    \n",
    "    results = verify_homeomorphism_theoretical(\n",
    "        model=model,\n",
    "        mnist_train=mnist_train,\n",
    "        mnist_test=mnist_test,\n",
    "        sparsity_levels=sparsity_levels,\n",
    "        kappa=5,\n",
    "        device=device,\n",
    "        n_projections=100,\n",
    "        distance_metric=DISTANCE_METRIC\n",
    "    )\n",
    "    \n",
    "    # Generate LaTeX table\n",
    "    print_latex_table_theoretical(results, kappa=5, distance_metric=DISTANCE_METRIC)\n",
    "    \n",
    "    # Save results\n",
    "    output_file = f'theoretical_verification_results_{DISTANCE_METRIC}_corrected.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": null,
   "id": "94267220",
   "metadata": {},
   "outputs": [],
   "source": [
    "\"\"\"\n",
    "Balanced Universal Manifold with MSE Priority\n",
    "==============================================\n",
    "\n",
    "Key Changes:\n",
    "1. MSE-FIRST training: Learn manifold structure before classification\n",
    "2. Balanced loss: MSE gets higher weight than accuracy\n",
    "3. Topological verification at sparsity levels [0.10, 0.15, 0.20, 0.25]\n",
    "4. Reports both reconstruction quality AND topological metrics\n",
    "\"\"\"\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torchvision import datasets, transforms\n",
    "from torch.utils.data import DataLoader, Dataset, Subset\n",
    "import numpy as np\n",
    "import json\n",
    "import time\n",
    "from tqdm import tqdm\n",
    "from sklearn.neighbors import NearestNeighbors\n",
    "from sklearn.metrics.pairwise import cosine_distances\n",
    "from scipy.stats import wasserstein_distance\n",
    "from scipy.sparse import csr_matrix\n",
    "from scipy.sparse.csgraph import connected_components\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# 1. MTAN ATTENTION MODULE\n",
    "# ============================================================================\n",
    "\n",
    "class TaskAttention(nn.Module):\n",
    "    \"\"\"Task-specific attention - learns what features matter for each task\"\"\"\n",
    "    def __init__(self, in_channels):\n",
    "        super().__init__()\n",
    "        self.attention = nn.Sequential(\n",
    "            nn.Conv2d(in_channels, in_channels // 4, kernel_size=1),\n",
    "            nn.BatchNorm2d(in_channels // 4),\n",
    "            nn.ReLU(inplace=True),\n",
    "            nn.Conv2d(in_channels // 4, in_channels, kernel_size=1),\n",
    "            nn.Sigmoid()\n",
    "        )\n",
    "    \n",
    "    def forward(self, x):\n",
    "        att_mask = self.attention(x)\n",
    "        return x * att_mask, att_mask\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# 2. ENHANCED SPARSE ENCODER\n",
    "# ============================================================================\n",
    "\n",
    "class EnhancedSparseEncoder(nn.Module):\n",
    "    \"\"\"Sparse encoder with MTAN-style multi-scale attention\"\"\"\n",
    "    def __init__(self, latent_dim=512, base_channels=64):\n",
    "        super().__init__()\n",
    "        \n",
    "        # Initial convolution\n",
    "        self.init_conv = nn.Sequential(\n",
    "            nn.Conv2d(2, base_channels, 3, padding=1),  # sparse_image + mask\n",
    "            nn.BatchNorm2d(base_channels),\n",
    "            nn.ReLU(inplace=True)\n",
    "        )\n",
    "        \n",
    "        # Encoder stages with downsampling\n",
    "        self.enc1 = self._make_block(base_channels, base_channels, stride=1)      # 28x28\n",
    "        self.enc2 = self._make_block(base_channels, base_channels * 2, stride=2)  # 14x14\n",
    "        self.enc3 = self._make_block(base_channels * 2, base_channels * 4, stride=2)  # 7x7\n",
    "        self.enc4 = self._make_block(base_channels * 4, base_channels * 8, stride=2)  # 3x3\n",
    "        \n",
    "        # Task attention at each scale\n",
    "        self.attn1 = TaskAttention(base_channels)\n",
    "        self.attn2 = TaskAttention(base_channels * 2)\n",
    "        self.attn3 = TaskAttention(base_channels * 4)\n",
    "        self.attn4 = TaskAttention(base_channels * 8)\n",
    "        \n",
    "        # Adaptive pooling + FC to latent space\n",
    "        self.pool = nn.AdaptiveAvgPool2d((4, 4))\n",
    "        self.fc = nn.Sequential(\n",
    "            nn.Flatten(),\n",
    "            nn.Linear(base_channels * 8 * 4 * 4, 1024),\n",
    "            nn.ReLU(inplace=True),\n",
    "            nn.Dropout(0.3),\n",
    "            nn.Linear(1024, latent_dim)\n",
    "        )\n",
    "    \n",
    "    def _make_block(self, in_ch, out_ch, stride=1):\n",
    "        return nn.Sequential(\n",
    "            nn.Conv2d(in_ch, out_ch, 3, stride=stride, padding=1),\n",
    "            nn.BatchNorm2d(out_ch),\n",
    "            nn.ReLU(inplace=True),\n",
    "            nn.Conv2d(out_ch, out_ch, 3, stride=1, padding=1),\n",
    "            nn.BatchNorm2d(out_ch),\n",
    "            nn.ReLU(inplace=True)\n",
    "        )\n",
    "    \n",
    "    def forward(self, sparse_image, mask):\n",
    "        x = torch.cat([sparse_image, mask], dim=1)\n",
    "        \n",
    "        # Multi-scale encoding with attention\n",
    "        x = self.init_conv(x)\n",
    "        \n",
    "        f1 = self.enc1(x)\n",
    "        f1_att, _ = self.attn1(f1)\n",
    "        \n",
    "        f2 = self.enc2(f1_att)\n",
    "        f2_att, _ = self.attn2(f2)\n",
    "        \n",
    "        f3 = self.enc3(f2_att)\n",
    "        f3_att, _ = self.attn3(f3)\n",
    "        \n",
    "        f4 = self.enc4(f3_att)\n",
    "        f4_att, _ = self.attn4(f4)\n",
    "        \n",
    "        # To latent space\n",
    "        pooled = self.pool(f4_att)\n",
    "        z = self.fc(pooled)\n",
    "        \n",
    "        # Return both latent code AND multi-scale features for decoder\n",
    "        features = {\n",
    "            'f1': f1_att,\n",
    "            'f2': f2_att,\n",
    "            'f3': f3_att,\n",
    "            'f4': f4_att\n",
    "        }\n",
    "        \n",
    "        return z, features\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# 3. FULL IMAGE ENCODER\n",
    "# ============================================================================\n",
    "\n",
    "class EnhancedFullImageEncoder(nn.Module):\n",
    "    \"\"\"Full image encoder with attention - for consistency learning\"\"\"\n",
    "    def __init__(self, latent_dim=512, base_channels=64):\n",
    "        super().__init__()\n",
    "        \n",
    "        self.init_conv = nn.Sequential(\n",
    "            nn.Conv2d(1, base_channels, 3, padding=1),\n",
    "            nn.BatchNorm2d(base_channels),\n",
    "            nn.ReLU(inplace=True)\n",
    "        )\n",
    "        \n",
    "        self.enc1 = self._make_block(base_channels, base_channels, stride=1)\n",
    "        self.enc2 = self._make_block(base_channels, base_channels * 2, stride=2)\n",
    "        self.enc3 = self._make_block(base_channels * 2, base_channels * 4, stride=2)\n",
    "        self.enc4 = self._make_block(base_channels * 4, base_channels * 8, stride=2)\n",
    "        \n",
    "        # Attention modules\n",
    "        self.attn1 = TaskAttention(base_channels)\n",
    "        self.attn2 = TaskAttention(base_channels * 2)\n",
    "        self.attn3 = TaskAttention(base_channels * 4)\n",
    "        self.attn4 = TaskAttention(base_channels * 8)\n",
    "        \n",
    "        self.pool = nn.AdaptiveAvgPool2d((4, 4))\n",
    "        self.fc = nn.Sequential(\n",
    "            nn.Flatten(),\n",
    "            nn.Linear(base_channels * 8 * 4 * 4, 1024),\n",
    "            nn.ReLU(inplace=True),\n",
    "            nn.Dropout(0.3),\n",
    "            nn.Linear(1024, latent_dim)\n",
    "        )\n",
    "    \n",
    "    def _make_block(self, in_ch, out_ch, stride=1):\n",
    "        return nn.Sequential(\n",
    "            nn.Conv2d(in_ch, out_ch, 3, stride=stride, padding=1),\n",
    "            nn.BatchNorm2d(out_ch),\n",
    "            nn.ReLU(inplace=True),\n",
    "            nn.Conv2d(out_ch, out_ch, 3, stride=1, padding=1),\n",
    "            nn.BatchNorm2d(out_ch),\n",
    "            nn.ReLU(inplace=True)\n",
    "        )\n",
    "    \n",
    "    def forward(self, full_image):\n",
    "        x = self.init_conv(full_image)\n",
    "        \n",
    "        f1 = self.enc1(x)\n",
    "        f1_att, _ = self.attn1(f1)\n",
    "        \n",
    "        f2 = self.enc2(f1_att)\n",
    "        f2_att, _ = self.attn2(f2)\n",
    "        \n",
    "        f3 = self.enc3(f2_att)\n",
    "        f3_att, _ = self.attn3(f3)\n",
    "        \n",
    "        f4 = self.enc4(f3_att)\n",
    "        f4_att, _ = self.attn4(f4)\n",
    "        \n",
    "        pooled = self.pool(f4_att)\n",
    "        z = self.fc(pooled)\n",
    "        \n",
    "        return z\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# 4. DECODER WITH SKIP CONNECTIONS\n",
    "# ============================================================================\n",
    "\n",
    "class AttentionDecoder(nn.Module):\n",
    "    \"\"\"Decoder with skip connections from encoder features\"\"\"\n",
    "    def __init__(self, latent_dim=512, base_channels=64):\n",
    "        super().__init__()\n",
    "        \n",
    "        # Expand latent to spatial\n",
    "        self.fc = nn.Sequential(\n",
    "            nn.Linear(latent_dim, 2048),\n",
    "            nn.ReLU(inplace=True),\n",
    "            nn.Dropout(0.2),\n",
    "            nn.Linear(2048, base_channels * 8 * 4 * 4),\n",
    "            nn.ReLU(inplace=True)\n",
    "        )\n",
    "        \n",
    "        # Decoder with upsampling\n",
    "        self.dec1 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(base_channels * 8, base_channels * 4, 4, stride=2, padding=1),\n",
    "            nn.BatchNorm2d(base_channels * 4),\n",
    "            nn.ReLU(inplace=True)\n",
    "        )\n",
    "        \n",
    "        self.dec2 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(base_channels * 4, base_channels * 2, 4, stride=2, padding=1),\n",
    "            nn.BatchNorm2d(base_channels * 2),\n",
    "            nn.ReLU(inplace=True)\n",
    "        )\n",
    "        \n",
    "        self.dec3 = nn.Sequential(\n",
    "            nn.ConvTranspose2d(base_channels * 2, base_channels, 4, stride=2, padding=1),\n",
    "            nn.BatchNorm2d(base_channels),\n",
    "            nn.ReLU(inplace=True)\n",
    "        )\n",
    "        \n",
    "        # Refinement layers\n",
    "        self.refine = nn.Sequential(\n",
    "            nn.Conv2d(base_channels, 32, 3, padding=1),\n",
    "            nn.BatchNorm2d(32),\n",
    "            nn.ReLU(inplace=True),\n",
    "            nn.Conv2d(32, 1, 3, padding=1),\n",
    "            nn.Tanh()\n",
    "        )\n",
    "        \n",
    "        self.crop = lambda x: x[:, :, 2:30, 2:30]  # 32->28\n",
    "    \n",
    "    def forward(self, z, encoder_features=None):\n",
    "        x = self.fc(z).view(-1, 512, 4, 4)\n",
    "        \n",
    "        # Upsample with optional skip connections\n",
    "        x = self.dec1(x)\n",
    "        if encoder_features and 'f3' in encoder_features:\n",
    "            if x.shape[2:] == encoder_features['f3'].shape[2:]:\n",
    "                x = x + encoder_features['f3']\n",
    "        \n",
    "        x = self.dec2(x)\n",
    "        if encoder_features and 'f2' in encoder_features:\n",
    "            if x.shape[2:] == encoder_features['f2'].shape[2:]:\n",
    "                x = x + encoder_features['f2']\n",
    "        \n",
    "        x = self.dec3(x)\n",
    "        if encoder_features and 'f1' in encoder_features:\n",
    "            if encoder_features['f1'].shape[2] == 28:\n",
    "                f1_up = F.interpolate(encoder_features['f1'], size=(32, 32), mode='bilinear')\n",
    "                x = x + f1_up\n",
    "        \n",
    "        x = self.refine(x)\n",
    "        x = self.crop(x)\n",
    "        \n",
    "        return x\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# 5. CLASSIFIER (Learned Later)\n",
    "# ============================================================================\n",
    "\n",
    "class Classifier(nn.Module):\n",
    "    \"\"\"Simple classifier from latent space - trained AFTER manifold learning\"\"\"\n",
    "    def __init__(self, latent_dim=512, num_classes=10):\n",
    "        super().__init__()\n",
    "        self.classifier = nn.Sequential(\n",
    "            nn.Linear(latent_dim, 256),\n",
    "            nn.ReLU(inplace=True),\n",
    "            nn.Dropout(0.4),\n",
    "            nn.Linear(256, num_classes)\n",
    "        )\n",
    "    \n",
    "    def forward(self, z):\n",
    "        return self.classifier(z)\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# 6. COMPLETE BALANCED UNIVERSAL MODEL\n",
    "# ============================================================================\n",
    "\n",
    "class BalancedUniversalManifold(nn.Module):\n",
    "    \"\"\"\n",
    "    Balanced Universal Manifold Learning\n",
    "    \n",
    "    Priority: MSE (manifold structure) > Accuracy (classification)\n",
    "    \"\"\"\n",
    "    def __init__(self, latent_dim=512, base_channels=64):\n",
    "        super().__init__()\n",
    "        \n",
    "        self.encoder_x = EnhancedSparseEncoder(latent_dim, base_channels)\n",
    "        self.encoder_s = EnhancedFullImageEncoder(latent_dim, base_channels)\n",
    "        self.decoder = AttentionDecoder(latent_dim, base_channels)\n",
    "        self.classifier = Classifier(latent_dim, num_classes=10)\n",
    "        \n",
    "        self.latent_dim = latent_dim\n",
    "    \n",
    "    def forward(self, sparse_image, mask, full_image=None, return_features=False):\n",
    "        # Encode sparse (always available)\n",
    "        z_x, encoder_features = self.encoder_x(sparse_image, mask)\n",
    "        \n",
    "        # Decode with skip connections\n",
    "        recon_x = self.decoder(z_x, encoder_features)\n",
    "        \n",
    "        # Classify\n",
    "        logits = self.classifier(z_x)\n",
    "        \n",
    "        if full_image is not None:\n",
    "            # Training: also encode full image\n",
    "            z_s = self.encoder_s(full_image)\n",
    "            recon_s = self.decoder(z_s)\n",
    "            \n",
    "            if return_features:\n",
    "                return recon_x, recon_s, logits, z_x, z_s\n",
    "            return recon_x, recon_s, logits, z_x, z_s\n",
    "        \n",
    "        # Testing: only sparse path\n",
    "        return recon_x, logits\n",
    "    \n",
    "    def count_parameters(self):\n",
    "        return sum(p.numel() for p in self.parameters() if p.requires_grad)\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# 7. DATASET\n",
    "# ============================================================================\n",
    "\n",
    "class UniversalDataset(Dataset):\n",
    "    def __init__(self, mnist_dataset, fashion_dataset, mask_ratio=0.85):\n",
    "        self.mnist_dataset = mnist_dataset\n",
    "        self.fashion_dataset = fashion_dataset\n",
    "        self.total_mnist = len(mnist_dataset)\n",
    "        self.total_fashion = len(fashion_dataset)\n",
    "        self.mask_ratio = mask_ratio\n",
    "    \n",
    "    def __len__(self):\n",
    "        return self.total_mnist + self.total_fashion\n",
    "    \n",
    "    def __getitem__(self, idx):\n",
    "        if idx < self.total_mnist:\n",
    "            image, class_label = self.mnist_dataset[idx]\n",
    "            domain = 0\n",
    "            full_image = image\n",
    "        else:\n",
    "            image, class_label = self.fashion_dataset[idx - self.total_mnist]\n",
    "            domain = 1\n",
    "            full_image = image\n",
    "        \n",
    "        mask = (torch.rand_like(full_image) > self.mask_ratio).float()\n",
    "        sparse_image = full_image * mask\n",
    "        \n",
    "        return {\n",
    "            'sparse_image': sparse_image,\n",
    "            'mask': mask,\n",
    "            'full_image': full_image,\n",
    "            'label': class_label,\n",
    "            'domain': domain\n",
    "        }\n",
    "\n",
    "\n",
    "def make_balanced_subset(dataset, n_per_domain=500, seed=42):\n",
    "    \"\"\"Create balanced test set\"\"\"\n",
    "    rng = np.random.RandomState(seed)\n",
    "    mnist_idx = []\n",
    "    fashion_idx = []\n",
    "    \n",
    "    for i in range(len(dataset)):\n",
    "        if dataset[i]['domain'] == 0:\n",
    "            mnist_idx.append(i)\n",
    "        else:\n",
    "            fashion_idx.append(i)\n",
    "    \n",
    "    sel_mnist = rng.choice(mnist_idx, n_per_domain, replace=False).tolist()\n",
    "    sel_fashion = rng.choice(fashion_idx, n_per_domain, replace=False).tolist()\n",
    "    \n",
    "    selected = sel_mnist + sel_fashion\n",
    "    rng.shuffle(selected)\n",
    "    \n",
    "    return Subset(dataset, selected)\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# 8. TRAINING (MSE-FIRST, then Classification)\n",
    "# ============================================================================\n",
    "\n",
    "def train_model(model, train_loader, val_loader, num_epochs=50, device='cpu',\n",
    "                mse_weight=2.0, class_weight=0.3, consist_weight=0.1):\n",
    "    \"\"\"\n",
    "    Training with BALANCED loss (MSE priority):\n",
    "    - MSE gets higher weight (manifold structure first)\n",
    "    - Classification gets lower weight (learned after manifold)\n",
    "    - Consistency loss maintains alignment\n",
    "    \n",
    "    Args:\n",
    "        mse_weight: Weight for reconstruction (default 2.0)\n",
    "        class_weight: Weight for classification (default 0.3)\n",
    "        consist_weight: Weight for consistency (default 0.1)\n",
    "    \"\"\"\n",
    "    print(f\"\\n{'='*70}\")\n",
    "    print(\"BALANCED UNIVERSAL MANIFOLD (MSE-FIRST)\")\n",
    "    print(f\"{'='*70}\")\n",
    "    print(f\"Parameters: {model.count_parameters():,}\")\n",
    "    print(f\"Device: {device}\")\n",
    "    print(f\"Epochs: {num_epochs}\")\n",
    "    print(f\"\\nLoss Weights:\")\n",
    "    print(f\"  MSE (Reconstruction): {mse_weight:.1f}x\")\n",
    "    print(f\"  Classification:       {class_weight:.1f}x\")\n",
    "    print(f\"  Consistency:          {consist_weight:.1f}x\")\n",
    "    print(f\"{'='*70}\\n\")\n",
    "    \n",
    "    model.to(device)\n",
    "    optimizer = torch.optim.Adam(model.parameters(), lr=1e-3, weight_decay=1e-5)\n",
    "    scheduler = torch.optim.lr_scheduler.ReduceLROnPlateau(\n",
    "        optimizer, mode='min', factor=0.5, patience=5\n",
    "    )\n",
    "    \n",
    "    best_val_mse = float('inf')\n",
    "    best_state = None\n",
    "    \n",
    "    for epoch in range(num_epochs):\n",
    "        # ========== TRAINING ==========\n",
    "        model.train()\n",
    "        train_loss = 0\n",
    "        train_mse_total = 0\n",
    "        train_class_loss = 0\n",
    "        train_consist_loss = 0\n",
    "        train_correct = 0\n",
    "        train_total = 0\n",
    "        \n",
    "        for batch in train_loader:\n",
    "            sparse = batch['sparse_image'].to(device)\n",
    "            mask = batch['mask'].to(device)\n",
    "            full = batch['full_image'].to(device)\n",
    "            labels = batch['label'].to(device)\n",
    "            \n",
    "            optimizer.zero_grad()\n",
    "            \n",
    "            # Forward pass\n",
    "            recon_x, recon_s, logits, z_x, z_s = model(sparse, mask, full, return_features=True)\n",
    "            \n",
    "            # Loss components\n",
    "            mse_x = F.mse_loss(recon_x, full)\n",
    "            mse_s = F.mse_loss(recon_s, full)\n",
    "            mse_total = (mse_x + mse_s) / 2\n",
    "            \n",
    "            loss_class = F.cross_entropy(logits, labels)\n",
    "            \n",
    "            # Consistency loss\n",
    "            z_x_norm = F.normalize(z_x, p=2, dim=1)\n",
    "            z_s_norm = F.normalize(z_s, p=2, dim=1)\n",
    "            loss_consistency = 1.0 - (z_x_norm * z_s_norm).sum(dim=1).mean()\n",
    "            \n",
    "            # BALANCED total loss (MSE priority)\n",
    "            loss = mse_weight * mse_total + class_weight * loss_class + consist_weight * loss_consistency\n",
    "            \n",
    "            loss.backward()\n",
    "            torch.nn.utils.clip_grad_norm_(model.parameters(), max_norm=1.0)\n",
    "            optimizer.step()\n",
    "            \n",
    "            # Stats\n",
    "            train_loss += loss.item()\n",
    "            train_mse_total += mse_total.item()\n",
    "            train_class_loss += loss_class.item()\n",
    "            train_consist_loss += loss_consistency.item()\n",
    "            \n",
    "            _, pred = logits.max(1)\n",
    "            train_total += labels.size(0)\n",
    "            train_correct += pred.eq(labels).sum().item()\n",
    "        \n",
    "        n_batches = len(train_loader)\n",
    "        train_acc = 100. * train_correct / train_total\n",
    "        \n",
    "        # ========== VALIDATION ==========\n",
    "        model.eval()\n",
    "        val_loss = 0\n",
    "        val_mse_total = 0\n",
    "        val_correct = 0\n",
    "        val_total = 0\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            for batch in val_loader:\n",
    "                sparse = batch['sparse_image'].to(device)\n",
    "                mask = batch['mask'].to(device)\n",
    "                full = batch['full_image'].to(device)\n",
    "                labels = batch['label'].to(device)\n",
    "                \n",
    "                recon_x, recon_s, logits, z_x, z_s = model(sparse, mask, full, return_features=True)\n",
    "                \n",
    "                mse_x = F.mse_loss(recon_x, full)\n",
    "                mse_s = F.mse_loss(recon_s, full)\n",
    "                mse_total = (mse_x + mse_s) / 2\n",
    "                \n",
    "                loss_class = F.cross_entropy(logits, labels)\n",
    "                z_x_norm = F.normalize(z_x, p=2, dim=1)\n",
    "                z_s_norm = F.normalize(z_s, p=2, dim=1)\n",
    "                loss_consistency = 1.0 - (z_x_norm * z_s_norm).sum(dim=1).mean()\n",
    "                \n",
    "                loss = mse_weight * mse_total + class_weight * loss_class + consist_weight * loss_consistency\n",
    "                val_loss += loss.item()\n",
    "                val_mse_total += mse_total.item()\n",
    "                \n",
    "                _, pred = logits.max(1)\n",
    "                val_total += labels.size(0)\n",
    "                val_correct += pred.eq(labels).sum().item()\n",
    "        \n",
    "        avg_val_loss = val_loss / len(val_loader)\n",
    "        avg_val_mse = val_mse_total / len(val_loader)\n",
    "        val_acc = 100. * val_correct / val_total\n",
    "        \n",
    "        scheduler.step(avg_val_mse)  # Schedule based on MSE (manifold quality)\n",
    "        \n",
    "        # Print\n",
    "        print(f'Epoch {epoch+1}/{num_epochs}:')\n",
    "        print(f'  Train - MSE: {train_mse_total/n_batches:.6f}, Acc: {train_acc:.2f}%, Loss: {train_loss/n_batches:.4f}')\n",
    "        print(f'    └─ Class: {train_class_loss/n_batches:.4f}, Consist: {train_consist_loss/n_batches:.4f}')\n",
    "        print(f'  Val   - MSE: {avg_val_mse:.6f}, Acc: {val_acc:.2f}%, Loss: {avg_val_loss:.4f}')\n",
    "        \n",
    "        if avg_val_mse < best_val_mse:\n",
    "            best_val_mse = avg_val_mse\n",
    "            best_state = model.state_dict()\n",
    "            print(f'  ✓ NEW BEST MSE!')\n",
    "        print()\n",
    "    \n",
    "    if best_state:\n",
    "        model.load_state_dict(best_state)\n",
    "    \n",
    "    return model\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# 9. EVALUATION\n",
    "# ============================================================================\n",
    "\n",
    "def evaluate_by_domain(model, test_loader, device='cpu'):\n",
    "    \"\"\"Evaluate separately on each domain\"\"\"\n",
    "    print(f\"\\n{'='*70}\")\n",
    "    print(\"EVALUATION BY DOMAIN\")\n",
    "    print(f\"{'='*70}\\n\")\n",
    "    \n",
    "    model.eval()\n",
    "    model.to(device)\n",
    "    \n",
    "    mnist_mse = []\n",
    "    mnist_correct = 0\n",
    "    mnist_total = 0\n",
    "    \n",
    "    fashion_mse = []\n",
    "    fashion_correct = 0\n",
    "    fashion_total = 0\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        for batch in test_loader:\n",
    "            sparse = batch['sparse_image'].to(device)\n",
    "            mask = batch['mask'].to(device)\n",
    "            full = batch['full_image'].to(device)\n",
    "            labels = batch['label'].to(device)\n",
    "            domains = batch['domain']\n",
    "            \n",
    "            # Testing mode: only sparse input\n",
    "            recon, logits = model(sparse, mask)\n",
    "            \n",
    "            # MSE per sample\n",
    "            mse_per_sample = F.mse_loss(recon, full, reduction='none').view(recon.size(0), -1).mean(dim=1)\n",
    "            \n",
    "            _, pred = logits.max(1)\n",
    "            \n",
    "            for i in range(len(domains)):\n",
    "                if domains[i] == 0:\n",
    "                    mnist_mse.append(mse_per_sample[i].item())\n",
    "                    mnist_total += 1\n",
    "                    if pred[i] == labels[i]:\n",
    "                        mnist_correct += 1\n",
    "                else:\n",
    "                    fashion_mse.append(mse_per_sample[i].item())\n",
    "                    fashion_total += 1\n",
    "                    if pred[i] == labels[i]:\n",
    "                        fashion_correct += 1\n",
    "    \n",
    "    results = {\n",
    "        'mnist': {\n",
    "            'mse': np.mean(mnist_mse),\n",
    "            'accuracy': 100. * mnist_correct / mnist_total,\n",
    "            'samples': mnist_total\n",
    "        },\n",
    "        'fashion': {\n",
    "            'mse': np.mean(fashion_mse),\n",
    "            'accuracy': 100. * fashion_correct / fashion_total,\n",
    "            'samples': fashion_total\n",
    "        }\n",
    "    }\n",
    "    \n",
    "    print(f\"MNIST ({results['mnist']['samples']} samples):\")\n",
    "    print(f\"  MSE:      {results['mnist']['mse']:.6f}\")\n",
    "    print(f\"  Accuracy: {results['mnist']['accuracy']:.2f}%\")\n",
    "    print()\n",
    "    print(f\"Fashion-MNIST ({results['fashion']['samples']} samples):\")\n",
    "    print(f\"  MSE:      {results['fashion']['mse']:.6f}\")\n",
    "    print(f\"  Accuracy: {results['fashion']['accuracy']:.2f}%\")\n",
    "    print()\n",
    "    \n",
    "    return results\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# 10. TOPOLOGICAL VERIFICATION FUNCTIONS\n",
    "# ============================================================================\n",
    "\n",
    "def compute_betti_0(latent_codes, kappa=10):\n",
    "    \"\"\"Compute β₀ (0-th Betti number) = number of connected components\"\"\"\n",
    "    N = len(latent_codes)\n",
    "    \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",
    "    row_indices = []\n",
    "    col_indices = []\n",
    "    \n",
    "    for i in range(N):\n",
    "        for j in neighbors[i, 1:]:\n",
    "            row_indices.append(i)\n",
    "            col_indices.append(j)\n",
    "            row_indices.append(j)\n",
    "            col_indices.append(i)\n",
    "    \n",
    "    data = np.ones(len(row_indices))\n",
    "    adjacency = csr_matrix((data, (row_indices, col_indices)), shape=(N, N))\n",
    "    \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, labels, kappa=5):\n",
    "    \"\"\"Trust Score: Do k-nearest neighbors share the same semantic label?\"\"\"\n",
    "    N = len(latent_codes)\n",
    "    \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:]\n",
    "    \n",
    "    trust_scores = []\n",
    "    for i in range(N):\n",
    "        neighbor_labels = labels[neighbors[i]]\n",
    "        same_label_count = np.sum(neighbor_labels == labels[i])\n",
    "        trust_scores.append(same_label_count / kappa)\n",
    "    \n",
    "    return np.mean(trust_scores)\n",
    "\n",
    "\n",
    "def compute_sliced_wasserstein(X, Y, n_projections=100):\n",
    "    \"\"\"Sliced Wasserstein-2 Discrepancy\"\"\"\n",
    "    D = X.shape[1]\n",
    "    distances = []\n",
    "    \n",
    "    for _ in range(n_projections):\n",
    "        theta = np.random.randn(D)\n",
    "        theta = theta / np.linalg.norm(theta)\n",
    "        \n",
    "        X_proj = X @ theta\n",
    "        Y_proj = Y @ theta\n",
    "        \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_in_class_alignment_error(latent_codes_sparse, latent_codes_full, \n",
    "                                     labels_sparse, labels_full, \n",
    "                                     distance_metric='cosine'):\n",
    "    \"\"\"In-Class Alignment Error: Distance between paired encodings\"\"\"\n",
    "    unique_classes = np.intersect1d(np.unique(labels_sparse), \n",
    "                                    np.unique(labels_full))\n",
    "    \n",
    "    per_class_error = {}\n",
    "    \n",
    "    for c in unique_classes:\n",
    "        mask = (labels_sparse == c)\n",
    "        \n",
    "        sparse_class = latent_codes_sparse[mask]\n",
    "        full_class = latent_codes_full[mask]\n",
    "        \n",
    "        if len(sparse_class) == 0:\n",
    "            continue\n",
    "        \n",
    "        if distance_metric == 'euclidean':\n",
    "            distances = np.linalg.norm(sparse_class - full_class, axis=1)\n",
    "        elif distance_metric == 'cosine':\n",
    "            distances = np.diag(cosine_distances(sparse_class, full_class))\n",
    "        else:\n",
    "            raise ValueError(f\"Unknown distance metric: {distance_metric}\")\n",
    "        \n",
    "        per_class_error[int(c)] = float(np.mean(distances))\n",
    "    \n",
    "    avg_error = np.mean(list(per_class_error.values()))\n",
    "    \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",
    "    else:\n",
    "        global_error = float(np.mean(np.diag(cosine_distances(\n",
    "            latent_codes_sparse, latent_codes_full\n",
    "        ))))\n",
    "    \n",
    "    return avg_error, per_class_error, global_error\n",
    "\n",
    "\n",
    "def compute_in_class_continuity(latent_codes, labels, kappa=5):\n",
    "    \"\"\"In-Class Continuity: Uniformity of local neighborhood structure\"\"\"\n",
    "    unique_classes = np.unique(labels)\n",
    "    per_class_continuity = {}\n",
    "    \n",
    "    for c in unique_classes:\n",
    "        mask = (labels == c)\n",
    "        class_latents = latent_codes[mask]\n",
    "        \n",
    "        if len(class_latents) < kappa + 1:\n",
    "            continue\n",
    "        \n",
    "        nn = NearestNeighbors(n_neighbors=kappa+1, metric='euclidean', \n",
    "                             algorithm='auto', n_jobs=-1)\n",
    "        nn.fit(class_latents)\n",
    "        distances, _ = nn.kneighbors(class_latents)\n",
    "        distances = distances[:, 1:]\n",
    "        \n",
    "        mean_dist = np.mean(distances)\n",
    "        std_dist = np.std(distances)\n",
    "        \n",
    "        cv = std_dist / (mean_dist + 1e-8)\n",
    "        continuity = 1.0 / (1.0 + cv)\n",
    "        \n",
    "        per_class_continuity[int(c)] = float(continuity)\n",
    "    \n",
    "    avg_continuity = np.mean(list(per_class_continuity.values()))\n",
    "    return avg_continuity, per_class_continuity\n",
    "\n",
    "\n",
    "# ============================================================================\n",
    "# 11. VERIFICATION AT MULTIPLE SPARSITY LEVELS\n",
    "# ============================================================================\n",
    "\n",
    "def verify_topological_unification(model, mnist_train, mnist_test, fashion_train, fashion_test,\n",
    "                                   sparsity_levels=[0.10, 0.15, 0.20, 0.25],\n",
    "                                   kappa=5, device='cpu', n_projections=100):\n",
    "    \"\"\"\n",
    "    Verify topological unification at multiple sparsity levels\n",
    "    Reports metrics for joint MNIST+Fashion-MNIST dataset\n",
    "    \"\"\"\n",
    "    print(f\"\\n{'='*80}\")\n",
    "    print(\"TOPOLOGICAL UNIFICATION VERIFICATION\")\n",
    "    print(\"Joint Dataset: MNIST + Fashion-MNIST\")\n",
    "    print(f\"{'='*80}\")\n",
    "    print(f\"Sparsity levels: {sparsity_levels}\")\n",
    "    print(f\"κ (neighbors): {kappa}\")\n",
    "    print(f\"Distance metric: cosine\")\n",
    "    print(f\"{'='*80}\\n\")\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} ({(1-sparsity):.0%} visible pixels)\")\n",
    "        print(f\"{'='*80}\")\n",
    "        \n",
    "        # Create joint datasets\n",
    "        train_dataset = UniversalDataset(mnist_train, fashion_train, mask_ratio=sparsity)\n",
    "        test_dataset = UniversalDataset(mnist_test, fashion_test, mask_ratio=sparsity)\n",
    "        \n",
    "        # Make balanced test subset\n",
    "        test_dataset_balanced = make_balanced_subset(test_dataset, n_per_domain=500, seed=42)\n",
    "        \n",
    "        # Encode training data\n",
    "        print(f\"\\nEncoding training data...\")\n",
    "        train_loader = DataLoader(train_dataset, batch_size=128, shuffle=False, num_workers=4)\n",
    "        \n",
    "        latent_codes_train_sparse = []\n",
    "        latent_codes_train_full = []\n",
    "        labels_train = []\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            for batch in tqdm(train_loader, desc=\"Train\", leave=False):\n",
    "                sparse = batch['sparse_image'].to(device)\n",
    "                mask = batch['mask'].to(device)\n",
    "                full = batch['full_image'].to(device)\n",
    "                labels = batch['label']\n",
    "                \n",
    "                # Sparse encoding\n",
    "                z_x, _ = model.encoder_x(sparse, mask)\n",
    "                latent_codes_train_sparse.append(z_x.cpu().numpy())\n",
    "                \n",
    "                # Full encoding\n",
    "                z_s = model.encoder_s(full)\n",
    "                latent_codes_train_full.append(z_s.cpu().numpy())\n",
    "                \n",
    "                labels_train.extend(labels.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",
    "        labels_train = np.array(labels_train)\n",
    "        \n",
    "        # Encode test data\n",
    "        print(f\"Encoding test data...\")\n",
    "        test_loader = DataLoader(test_dataset_balanced, batch_size=128, shuffle=False, num_workers=4)\n",
    "        \n",
    "        latent_codes_test_sparse = []\n",
    "        latent_codes_test_full = []\n",
    "        labels_test = []\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            for batch in tqdm(test_loader, desc=\"Test\", leave=False):\n",
    "                sparse = batch['sparse_image'].to(device)\n",
    "                mask = batch['mask'].to(device)\n",
    "                full = batch['full_image'].to(device)\n",
    "                labels = batch['label']\n",
    "                \n",
    "                z_x, _ = model.encoder_x(sparse, mask)\n",
    "                latent_codes_test_sparse.append(z_x.cpu().numpy())\n",
    "                \n",
    "                z_s = model.encoder_s(full)\n",
    "                latent_codes_test_full.append(z_s.cpu().numpy())\n",
    "                \n",
    "                labels_test.extend(labels.cpu().numpy())\n",
    "        \n",
    "        latent_codes_test_sparse = np.vstack(latent_codes_test_sparse)\n",
    "        latent_codes_test_full = np.vstack(latent_codes_test_full)\n",
    "        labels_test = np.array(labels_test)\n",
    "        \n",
    "        print(f\"\\nEncoded shapes:\")\n",
    "        print(f\"  Train: {latent_codes_train_sparse.shape}\")\n",
    "        print(f\"  Test:  {latent_codes_test_sparse.shape}\")\n",
    "        \n",
    "        # Combine for metrics\n",
    "        latent_codes_sparse_all = np.vstack([latent_codes_train_sparse, latent_codes_test_sparse])\n",
    "        latent_codes_full_all = np.vstack([latent_codes_train_full, latent_codes_test_full])\n",
    "        labels_all = np.concatenate([labels_train, labels_test])\n",
    "        \n",
    "        # METRIC 0: β₀\n",
    "        print(f\"\\n[0/4] Computing β₀...\")\n",
    "        latent_codes_all = np.vstack([latent_codes_sparse_all, latent_codes_full_all])\n",
    "        beta_0, _ = compute_betti_0(latent_codes_all, kappa=10)\n",
    "        print(f\"  β₀ = {beta_0} {'✓' if beta_0 == 1 else '✗'}\")\n",
    "        \n",
    "        # METRIC 1: Trust Score\n",
    "        print(f\"\\n[1/4] Computing Trust Score...\")\n",
    "        trust_score = compute_trust_score(latent_codes_sparse_all, labels_all, kappa=kappa)\n",
    "        print(f\"  Trust Score: {trust_score:.4f} {'✓' if trust_score >= 0.80 else '✗'}\")\n",
    "        \n",
    "        # METRIC 2: Sliced Wasserstein\n",
    "        print(f\"\\n[2/4] Computing Sliced Wasserstein...\")\n",
    "        max_samples = 5000\n",
    "        if len(latent_codes_train_sparse) > max_samples:\n",
    "            idx = np.random.choice(len(latent_codes_train_sparse), max_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_projections=n_projections)\n",
    "        print(f\"  Sliced W₂: {w2_distance:.4f} {'✓' if w2_distance <= 0.30 else '✗'}\")\n",
    "        \n",
    "        # METRIC 3: Alignment Error\n",
    "        print(f\"\\n[3/4] Computing Alignment Error...\")\n",
    "        alignment_error, per_class_error, global_error = compute_in_class_alignment_error(\n",
    "            latent_codes_sparse_all, latent_codes_full_all, labels_all, labels_all, \n",
    "            distance_metric='cosine'\n",
    "        )\n",
    "        print(f\"  Alignment Error: {alignment_error:.4f} {'✓' if alignment_error <= 0.10 else '✗'}\")\n",
    "        \n",
    "        # METRIC 4: In-Class Continuity\n",
    "        print(f\"\\n[4/4] Computing In-Class Continuity...\")\n",
    "        continuity_score, per_class_continuity = compute_in_class_continuity(\n",
    "            latent_codes_sparse_all, labels_all, kappa=kappa\n",
    "        )\n",
    "        print(f\"  Continuity: {continuity_score:.4f} {'✓' if continuity_score >= 0.80 else '✗'}\")\n",
    "        \n",
    "        # Store results\n",
    "        results[sparsity] = {\n",
    "            'beta_0': int(beta_0),\n",
    "            'trust_score': float(trust_score),\n",
    "            'sliced_w2': float(w2_distance),\n",
    "            'alignment_error': float(alignment_error),\n",
    "            'in_class_continuity': float(continuity_score)\n",
    "        }\n",
    "        \n",
    "        print(f\"\\n{'='*80}\")\n",
    "        print(f\"SUMMARY FOR ρ = {sparsity:.2f}:\")\n",
    "        print(f\"  β₀:        {beta_0}\")\n",
    "        print(f\"  Trust:     {trust_score:.3f}\")\n",
    "        print(f\"  W₂:        {w2_distance:.3f}\")\n",
    "        print(f\"  Align:     {alignment_error:.3f}\")\n",
    "        print(f\"  Cont:      {continuity_score:.3f}\")\n",
    "        print(f\"{'='*80}\")\n",
    "    \n",
    "    return results\n",
    "\n",
    "\n",
    "def print_latex_table(results):\n",
    "    \"\"\"Generate LaTeX table for topological metrics\"\"\"\n",
    "    sparsity_levels = sorted(results.keys())\n",
    "    \n",
    "    print(\"\\n\" + \"=\"*80)\n",
    "    print(\"LaTeX Table:\")\n",
    "    print(\"=\"*80)\n",
    "    print(r\"\\begin{table}[!htpb]\")\n",
    "    print(r\"\\centering\")\n",
    "    print(r\"\\caption{Topological Unification Verification ($\\kappa=5$, Cosine distance)}\")\n",
    "    print(r\"\\label{tab:unification}\")\n",
    "    print(r\"\\begin{tabular}{lcccc}\")\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",
    "    # 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 = r\"Alignment $\\tau_a$ ($\\leq 0.10$)\"\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",
    "    # In-Class Continuity\n",
    "    cont_row = r\"Continuity $\\tau_c$ ($\\geq 0.80$)\"\n",
    "    for rho in sparsity_levels:\n",
    "        val = results[rho]['in_class_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",
    "# 12. MAIN\n",
    "# ============================================================================\n",
    "\n",
    "if __name__ == '__main__':\n",
    "    print(\"\\n\" + \"=\"*70)\n",
    "    print(\"BALANCED UNIVERSAL MANIFOLD + TOPOLOGICAL VERIFICATION\")\n",
    "    print(\"=\"*70)\n",
    "    print(\"\\nKey Features:\")\n",
    "    print(\"  ✓ MSE-FIRST: Manifold structure prioritized\")\n",
    "    print(\"  ✓ Balanced loss: MSE (2.0x) > Classification (0.3x)\")\n",
    "    print(\"  ✓ Topological metrics at ρ = [0.10, 0.15, 0.20, 0.25]\")\n",
    "    print(\"  ✓ Joint dataset: MNIST + Fashion-MNIST\")\n",
    "    print(\"=\"*70)\n",
    "    \n",
    "    device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "    print(f\"\\nDevice: {device}\")\n",
    "    \n",
    "    # Load datasets\n",
    "    transform = transforms.Compose([\n",
    "        transforms.ToTensor(),\n",
    "        transforms.Normalize((0.5,), (0.5,))\n",
    "    ])\n",
    "    \n",
    "    print(\"\\nLoading datasets...\")\n",
    "    mnist_train = datasets.MNIST('./data', train=True, download=True, transform=transform)\n",
    "    mnist_test = datasets.MNIST('./data', train=False, download=True, transform=transform)\n",
    "    fashion_train = datasets.FashionMNIST('./data', train=True, download=True, transform=transform)\n",
    "    fashion_test = datasets.FashionMNIST('./data', train=False, download=True, transform=transform)\n",
    "    \n",
    "    # Split train/val\n",
    "    mnist_train_size = int(0.9 * len(mnist_train))\n",
    "    mnist_val_size = len(mnist_train) - mnist_train_size\n",
    "    mnist_train_sub, mnist_val_sub = torch.utils.data.random_split(\n",
    "        mnist_train, [mnist_train_size, mnist_val_size],\n",
    "        generator=torch.Generator().manual_seed(42)\n",
    "    )\n",
    "    \n",
    "    fashion_train_size = int(0.9 * len(fashion_train))\n",
    "    fashion_val_size = len(fashion_train) - fashion_train_size\n",
    "    fashion_train_sub, fashion_val_sub = torch.utils.data.random_split(\n",
    "        fashion_train, [fashion_train_size, fashion_val_size],\n",
    "        generator=torch.Generator().manual_seed(42)\n",
    "    )\n",
    "    \n",
    "    # Create universal datasets (starting at ρ=0.10 = 90% masked)\n",
    "    SPARSITY = 0.90  # Start with 90% masked for training\n",
    "    train_dataset = UniversalDataset(mnist_train_sub, fashion_train_sub, SPARSITY)\n",
    "    val_dataset = UniversalDataset(mnist_val_sub, fashion_val_sub, SPARSITY)\n",
    "    test_dataset_full = UniversalDataset(mnist_test, fashion_test, SPARSITY)\n",
    "    test_dataset = make_balanced_subset(test_dataset_full, n_per_domain=500, seed=42)\n",
    "    \n",
    "    print(f\"\\nDatasets:\")\n",
    "    print(f\"  Train: {len(train_dataset)}\")\n",
    "    print(f\"  Val:   {len(val_dataset)}\")\n",
    "    print(f\"  Test:  {len(test_dataset)} (500+500 balanced)\")\n",
    "    print(f\"  Training sparsity: {SPARSITY:.2f} ({(1-SPARSITY):.0%} visible)\")\n",
    "    \n",
    "    # DataLoaders\n",
    "    train_loader = DataLoader(train_dataset, batch_size=128, shuffle=True, num_workers=4)\n",
    "    val_loader = DataLoader(val_dataset, batch_size=128, shuffle=False, num_workers=4)\n",
    "    test_loader = DataLoader(test_dataset, batch_size=128, shuffle=False, num_workers=4)\n",
    "    \n",
    "    # Create model\n",
    "    model = BalancedUniversalManifold(latent_dim=512, base_channels=64)\n",
    "    print(f\"\\nModel: {model.count_parameters():,} parameters\")\n",
    "    \n",
    "    # Train (MSE-first, with balanced weights)\n",
    "    start_time = time.time()\n",
    "    model = train_model(\n",
    "        model, train_loader, val_loader, \n",
    "        num_epochs=50, device=device,\n",
    "        mse_weight=2.0,      # MSE priority\n",
    "        class_weight=0.3,    # Lower classification weight\n",
    "        consist_weight=0.1   # Consistency for alignment\n",
    "    )\n",
    "    training_time = time.time() - start_time\n",
    "    \n",
    "    # Evaluate on test set\n",
    "    results_test = evaluate_by_domain(model, test_loader, device=device)\n",
    "    \n",
    "    # Topological verification at multiple sparsity levels\n",
    "    print(\"\\n\" + \"=\"*70)\n",
    "    print(\"STARTING TOPOLOGICAL VERIFICATION\")\n",
    "    print(\"=\"*70)\n",
    "    \n",
    "    topo_results = verify_topological_unification(\n",
    "        model=model,\n",
    "        mnist_train=mnist_train_sub,\n",
    "        mnist_test=mnist_test,\n",
    "        fashion_train=fashion_train_sub,\n",
    "        fashion_test=fashion_test,\n",
    "        sparsity_levels=[0.10, 0.15, 0.20, 0.25],  # ρ values (visible pixels)\n",
    "        kappa=5,\n",
    "        device=device,\n",
    "        n_projections=100\n",
    "    )\n",
    "    \n",
    "    # Generate LaTeX table\n",
    "    print_latex_table(topo_results)\n",
    "    \n",
    "    # Save results\n",
    "    final_results = {\n",
    "        'model': 'Balanced_Universal_Manifold_MSE_First',\n",
    "        'training_sparsity': SPARSITY,\n",
    "        'parameters': model.count_parameters(),\n",
    "        'training_time_seconds': training_time,\n",
    "        'loss_weights': {\n",
    "            'mse': 2.0,\n",
    "            'classification': 0.3,\n",
    "            'consistency': 0.1\n",
    "        },\n",
    "        'test_results': {\n",
    "            'mnist': results_test['mnist'],\n",
    "            'fashion': results_test['fashion']\n",
    "        },\n",
    "        'topological_verification': topo_results\n",
    "    }\n",
    "    \n",
    "    with open('balanced_universal_results.json', 'w') as f:\n",
    "        json.dump(final_results, f, indent=2)\n",
    "    \n",
    "    # Final summary\n",
    "    print(\"\\n\" + \"=\"*70)\n",
    "    print(\"FINAL RESULTS\")\n",
    "    print(\"=\"*70)\n",
    "    print(f\"\\nModel Parameters: {model.count_parameters():,}\")\n",
    "    print(f\"Training Time: {training_time:.1f}s ({training_time/60:.1f} min)\")\n",
    "    print(f\"\\nTest Performance (ρ={SPARSITY:.2f}):\")\n",
    "    print(f\"  MNIST:         MSE={results_test['mnist']['mse']:.6f}, Acc={results_test['mnist']['accuracy']:.2f}%\")\n",
    "    print(f\"  Fashion-MNIST: MSE={results_test['fashion']['mse']:.6f}, Acc={results_test['fashion']['accuracy']:.2f}%\")\n",
    "    \n",
    "    print(f\"\\nTopological Unification (Joint Dataset):\")\n",
    "    for rho in [0.10, 0.15, 0.20, 0.25]:\n",
    "        r = topo_results[rho]\n",
    "        print(f\"  ρ={rho:.2f}: Trust={r['trust_score']:.3f}, W₂={r['sliced_w2']:.3f}, \"\n",
    "              f\"Align={r['alignment_error']:.3f}, Cont={r['in_class_continuity']:.3f}\")\n",
    "    \n",
    "    print(\"=\"*70)\n",
    "    \n",
    "    # Save model\n",
    "    torch.save(model.state_dict(), 'balanced_universal_manifold.pth')\n",
    "    print(f\"\\n✓ Model saved to: balanced_universal_manifold.pth\")\n",
    "    print(f\"✓ Results saved to: balanced_universal_results.json\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cb6500b0",
   "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
}
