{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "f2658ad7-d068-4eb6-9845-bf469b9eb821",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "from torchvision import datasets, transforms\n",
    "import random\n",
    "from torch.utils.data import DataLoader\n",
    "from PIL import Image\n",
    "import time\n",
    "import math\n",
    "from utils import *\n",
    "from transformers import BertTokenizer\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "d85ac23f-239a-49ac-ab21-7908535b10bc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Using device: mps\n"
     ]
    }
   ],
   "source": [
    "# Check if CUDA or MPS is running\n",
    "if torch.cuda.is_available():\n",
    "    device = 'cuda'\n",
    "elif torch.backends.mps.is_available():\n",
    "    device = 'mps'\n",
    "else:\n",
    "    device = \"cpu\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "d22168b0-6eca-45b2-98af-aefebb577efa",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Optimized Perlin noise generator for batch processing\n",
    "def generate_perlin_noise_batch(batch_size, size=28):\n",
    "    \"\"\"Generate a batch of Perlin noise images efficiently.\"\"\"\n",
    "    # Create random gradients for the batch\n",
    "    grid_size = 10  # Controls noise granularity\n",
    "    \n",
    "    # Generate grid points\n",
    "    lin_points = np.linspace(0, 1, grid_size, endpoint=False)\n",
    "    grid_x, grid_y = np.meshgrid(lin_points, lin_points)\n",
    "    grid_points = np.stack([grid_x, grid_y], axis=-1).reshape(-1, 2)\n",
    "    \n",
    "    # Initialize noise batch\n",
    "    noise_batch = np.zeros((batch_size, size, size))\n",
    "    \n",
    "    # Generate batch of random gradients at grid points\n",
    "    for b in range(batch_size):\n",
    "        # Random angles for gradients\n",
    "        angles = 2 * np.pi * np.random.rand(grid_size, grid_size)\n",
    "        gradients = np.stack([np.cos(angles), np.sin(angles)], axis=-1)\n",
    "        \n",
    "        # Calculate noise values\n",
    "        x_indices = np.linspace(0, grid_size-1, size, endpoint=False)\n",
    "        y_indices = np.linspace(0, grid_size-1, size, endpoint=False)\n",
    "        \n",
    "        # Get integer and fractional parts\n",
    "        x0, y0 = np.meshgrid(np.floor(x_indices).astype(int), np.floor(y_indices).astype(int))\n",
    "        x1, y1 = (x0 + 1) % grid_size, (y0 + 1) % grid_size\n",
    "        x_frac, y_frac = np.meshgrid(x_indices - np.floor(x_indices), y_indices - np.floor(y_indices))\n",
    "        \n",
    "        # Cubic interpolation function\n",
    "        def fade(t): return t * t * t * (t * (t * 6 - 15) + 10)\n",
    "        \n",
    "        # Apply fade function to fractions\n",
    "        u, v = fade(x_frac), fade(y_frac)\n",
    "        \n",
    "        # Get gradients and dot products\n",
    "        g00 = gradients[y0, x0]\n",
    "        g10 = gradients[y0, x1]\n",
    "        g01 = gradients[y1, x0]\n",
    "        g11 = gradients[y1, x1]\n",
    "        \n",
    "        # Position vectors\n",
    "        p00 = np.stack([x_frac, y_frac], axis=-1)\n",
    "        p10 = np.stack([x_frac - 1, y_frac], axis=-1)\n",
    "        p01 = np.stack([x_frac, y_frac - 1], axis=-1)\n",
    "        p11 = np.stack([x_frac - 1, y_frac - 1], axis=-1)\n",
    "        \n",
    "        # Dot products\n",
    "        n00 = np.sum(g00 * p00, axis=-1)\n",
    "        n10 = np.sum(g10 * p10, axis=-1)\n",
    "        n01 = np.sum(g01 * p01, axis=-1)\n",
    "        n11 = np.sum(g11 * p11, axis=-1)\n",
    "        \n",
    "        # Interpolation\n",
    "        n0 = n00 * (1 - u) + u * n10\n",
    "        n1 = n01 * (1 - u) + u * n11\n",
    "        noise = n0 * (1 - v) + v * n1\n",
    "        \n",
    "        # Normalize to [0, 1]\n",
    "        noise = (noise - noise.min()) / (noise.max() - noise.min() + 1e-8)\n",
    "        noise_batch[b] = noise\n",
    "    \n",
    "    return noise_batch"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "92cd8810-82df-49bc-9e17-2622fd9633b9",
   "metadata": {},
   "outputs": [],
   "source": [
    "class EfficientPairedMNIST:\n",
    "    def __init__(self, is_train=True, batch_size=1000):\n",
    "        self.is_train = is_train\n",
    "        self.batch_size = batch_size\n",
    "        \n",
    "        # Load dataset without transformation to get raw data\n",
    "        dataset = datasets.MNIST('./data', train=is_train, download=True, transform=transforms.ToTensor())\n",
    "        self.loader = DataLoader(dataset, batch_size=batch_size, shuffle=False)\n",
    "        \n",
    "        # Calculate target pairs\n",
    "        if is_train:\n",
    "            self.total_pairs = 2**18  # ~256K pairs\n",
    "        else:\n",
    "            self.total_pairs = 128\n",
    "        \n",
    "        self.pairs_per_class = np.ceil(self.total_pairs/10).astype(int)\n",
    "        \n",
    "    def create_paired_dataset(self):\n",
    "        start_time = time.time()\n",
    "        \n",
    "        # Pre-process to collect images by digit\n",
    "        print(\"Organizing images by digit...\")\n",
    "        data_by_digit = [[] for _ in range(10)]\n",
    "        \n",
    "        for images, labels in self.loader:\n",
    "            # Process batch by batch\n",
    "            for img, lbl in zip(images, labels):\n",
    "                data_by_digit[lbl.item()].append(img)\n",
    "        \n",
    "        print(f\"Organized {sum(len(d) for d in data_by_digit)} images in {time.time() - start_time:.2f} seconds\")\n",
    "        \n",
    "        # Create pairs using batched operations\n",
    "        print(\"Creating paired dataset...\")\n",
    "        paired_data = []\n",
    "        \n",
    "        for digit in range(10):\n",
    "            digit_images = data_by_digit[digit]\n",
    "            available_pairs = len(digit_images) * (len(digit_images) - 1) // 2\n",
    "            pairs_for_this_digit = min(available_pairs, self.pairs_per_class)\n",
    "            \n",
    "            print(f\"Digit {digit}: {len(digit_images)} images → {pairs_for_this_digit} pairs\")\n",
    "            \n",
    "            # Generate random pairs\n",
    "            all_indices = range(len(digit_images))\n",
    "            pair_indices = set()\n",
    "            \n",
    "            # Efficient pair generation\n",
    "            while len(pair_indices) < pairs_for_this_digit:\n",
    "                # Generate a batch of potential pairs\n",
    "                batch_size = min(1000, pairs_for_this_digit - len(pair_indices))\n",
    "                i_indices = random.sample(all_indices, batch_size)\n",
    "                j_indices = random.sample(all_indices, batch_size)\n",
    "                \n",
    "                # Add valid pairs (where i != j)\n",
    "                for i, j in zip(i_indices, j_indices):\n",
    "                    if i != j and (i, j) not in pair_indices and (j, i) not in pair_indices:\n",
    "                        pair_indices.add((i, j))\n",
    "                        \n",
    "                        if len(pair_indices) >= pairs_for_this_digit:\n",
    "                            break\n",
    "            \n",
    "            # Process the created pairs\n",
    "            for i, j in pair_indices:\n",
    "                img1 = digit_images[i]\n",
    "                img2 = digit_images[j]\n",
    "                \n",
    "                # Store the original images for later transformation\n",
    "                paired_data.append((img1, img2))\n",
    "        \n",
    "        print(f\"Created {len(paired_data)} pairs in {time.time() - start_time:.2f} seconds\")\n",
    "        \n",
    "        # Apply transformations batch by batch\n",
    "        print(\"Applying transformations in batches...\")\n",
    "        paired_tensor = torch.zeros(2, len(paired_data), 784)\n",
    "        \n",
    "        # Process in batches\n",
    "        for batch_start in range(0, len(paired_data), self.batch_size):\n",
    "            batch_time = time.time()\n",
    "            batch_end = min(batch_start + self.batch_size, len(paired_data))\n",
    "            current_batch_size = batch_end - batch_start\n",
    "            batch = paired_data[batch_start:batch_end]\n",
    "            \n",
    "            # Extract images for each view\n",
    "            view1_images = [p[0] for p in batch]\n",
    "            view2_images = [p[1] for p in batch]\n",
    "            \n",
    "            # Process View 1: Rotation and Scaling (done on CPU since it's more PIL-based)\n",
    "            view1_processed = []\n",
    "            for img in view1_images:\n",
    "                # Convert to PIL\n",
    "                pil_img = transforms.ToPILImage()(img)\n",
    "                \n",
    "                # Random rotation and scaling\n",
    "                angle = random.uniform(0, 90)\n",
    "                scale = random.uniform(0.5, 1.5)\n",
    "                \n",
    "                # Resize\n",
    "                new_size = max(1, int(28 * scale))\n",
    "                resized_img = pil_img.resize((new_size, new_size), Image.BILINEAR)\n",
    "                \n",
    "                # Create new image and paste\n",
    "                new_img = Image.new(\"L\", (28, 28), 0)\n",
    "                paste_x = max(0, (28 - resized_img.width) // 2)\n",
    "                paste_y = max(0, (28 - resized_img.height) // 2)\n",
    "                new_img.paste(resized_img, (paste_x, paste_y))\n",
    "                \n",
    "                # Rotate\n",
    "                rotated_img = new_img.rotate(angle, resample=Image.BILINEAR)\n",
    "                \n",
    "                # Convert back to tensor\n",
    "                tensor_img = transforms.ToTensor()(rotated_img)\n",
    "                view1_processed.append(tensor_img.view(784))\n",
    "            \n",
    "            # Process View 2: Add Perlin noise (leveraging batch operations)\n",
    "            # Convert images to numpy arrays\n",
    "            view2_np = torch.stack(view2_images).squeeze().numpy()\n",
    "            \n",
    "            # Generate batch of noise patterns efficiently\n",
    "            noise_patterns = generate_perlin_noise_batch(current_batch_size)\n",
    "            \n",
    "            # Randomly scale each noise pattern\n",
    "            noise_factors = np.random.uniform(0, 1, size=(current_batch_size, 1, 1))\n",
    "            scaled_noise = noise_patterns * noise_factors\n",
    "            \n",
    "            # Add noise to images\n",
    "            noisy_images = view2_np + scaled_noise\n",
    "            noisy_images = np.clip(noisy_images, 0, 1)\n",
    "            \n",
    "            # Convert back to tensors\n",
    "            view2_processed = torch.from_numpy(noisy_images).float().reshape(current_batch_size, 784)\n",
    "            \n",
    "            # Store in final tensor\n",
    "            paired_tensor[0, batch_start:batch_end] = torch.stack(view1_processed)\n",
    "            paired_tensor[1, batch_start:batch_end] = view2_processed\n",
    "            \n",
    "            print(f\"  Processed batch {batch_start//self.batch_size + 1}/{math.ceil(len(paired_data)/self.batch_size)} in {time.time() - batch_time:.2f} seconds\")\n",
    "        \n",
    "        print(f\"Applied all transformations in {time.time() - start_time:.2f} seconds\")\n",
    "        \n",
    "        return paired_tensor"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "729e9cf7-b614-49f6-97cd-6bc4933fe4ce",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Create and shuffle datasets\n",
    "def create_and_shuffle_dataset(is_train=True):\n",
    "    print(f\"Creating {'training' if is_train else 'test'} dataset...\")\n",
    "    start_time = time.time()\n",
    "    \n",
    "    paired_mnist = EfficientPairedMNIST(is_train)\n",
    "    paired_tensor = paired_mnist.create_paired_dataset()\n",
    "    \n",
    "    # Shuffle\n",
    "    print(\"Shuffling dataset...\")\n",
    "    num_samples = paired_tensor.shape[1]\n",
    "    permutation = torch.randperm(num_samples)\n",
    "    shuffled_tensor = paired_tensor[:, permutation, :]\n",
    "    \n",
    "    print(f\"Dataset created and shuffled in {time.time() - start_time:.2f} seconds\")\n",
    "    print(f\"Shape: {shuffled_tensor.shape}\")\n",
    "    \n",
    "    return shuffled_tensor"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 119,
   "id": "69a845d9-1597-4f67-9772-b2035a8584a4",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Creating training dataset...\n",
      "Organizing images by digit...\n",
      "Organized 60000 images in 0.98 seconds\n",
      "Creating paired dataset...\n",
      "Digit 0: 5923 images → 26215 pairs\n",
      "Digit 1: 6742 images → 26215 pairs\n",
      "Digit 2: 5958 images → 26215 pairs\n",
      "Digit 3: 6131 images → 26215 pairs\n",
      "Digit 4: 5842 images → 26215 pairs\n",
      "Digit 5: 5421 images → 26215 pairs\n",
      "Digit 6: 5918 images → 26215 pairs\n",
      "Digit 7: 6265 images → 26215 pairs\n",
      "Digit 8: 5851 images → 26215 pairs\n",
      "Digit 9: 5949 images → 26215 pairs\n",
      "Created 262150 pairs in 1.35 seconds\n",
      "Applying transformations in batches...\n",
      "  Processed batch 1/263 in 0.20 seconds\n",
      "  Processed batch 2/263 in 0.20 seconds\n",
      "  Processed batch 3/263 in 0.20 seconds\n",
      "  Processed batch 4/263 in 0.20 seconds\n",
      "  Processed batch 5/263 in 0.20 seconds\n",
      "  Processed batch 6/263 in 0.20 seconds\n",
      "  Processed batch 7/263 in 0.19 seconds\n",
      "  Processed batch 8/263 in 0.19 seconds\n",
      "  Processed batch 9/263 in 0.19 seconds\n",
      "  Processed batch 10/263 in 0.19 seconds\n",
      "  Processed batch 11/263 in 0.19 seconds\n",
      "  Processed batch 12/263 in 0.20 seconds\n",
      "  Processed batch 13/263 in 0.20 seconds\n",
      "  Processed batch 14/263 in 0.19 seconds\n",
      "  Processed batch 15/263 in 0.19 seconds\n",
      "  Processed batch 16/263 in 0.19 seconds\n",
      "  Processed batch 17/263 in 0.19 seconds\n",
      "  Processed batch 18/263 in 0.20 seconds\n",
      "  Processed batch 19/263 in 0.20 seconds\n",
      "  Processed batch 20/263 in 0.19 seconds\n",
      "  Processed batch 21/263 in 0.19 seconds\n",
      "  Processed batch 22/263 in 0.20 seconds\n",
      "  Processed batch 23/263 in 0.20 seconds\n",
      "  Processed batch 24/263 in 0.19 seconds\n",
      "  Processed batch 25/263 in 0.19 seconds\n",
      "  Processed batch 26/263 in 0.19 seconds\n",
      "  Processed batch 27/263 in 0.19 seconds\n",
      "  Processed batch 28/263 in 0.19 seconds\n",
      "  Processed batch 29/263 in 0.20 seconds\n",
      "  Processed batch 30/263 in 0.20 seconds\n",
      "  Processed batch 31/263 in 0.19 seconds\n",
      "  Processed batch 32/263 in 0.19 seconds\n",
      "  Processed batch 33/263 in 0.20 seconds\n",
      "  Processed batch 34/263 in 0.20 seconds\n",
      "  Processed batch 35/263 in 0.19 seconds\n",
      "  Processed batch 36/263 in 0.19 seconds\n",
      "  Processed batch 37/263 in 0.19 seconds\n",
      "  Processed batch 38/263 in 0.20 seconds\n",
      "  Processed batch 39/263 in 0.19 seconds\n",
      "  Processed batch 40/263 in 0.20 seconds\n",
      "  Processed batch 41/263 in 0.19 seconds\n",
      "  Processed batch 42/263 in 0.20 seconds\n",
      "  Processed batch 43/263 in 0.20 seconds\n",
      "  Processed batch 44/263 in 0.19 seconds\n",
      "  Processed batch 45/263 in 0.20 seconds\n",
      "  Processed batch 46/263 in 0.20 seconds\n",
      "  Processed batch 47/263 in 0.20 seconds\n",
      "  Processed batch 48/263 in 0.19 seconds\n",
      "  Processed batch 49/263 in 0.20 seconds\n",
      "  Processed batch 50/263 in 0.20 seconds\n",
      "  Processed batch 51/263 in 0.20 seconds\n",
      "  Processed batch 52/263 in 0.20 seconds\n",
      "  Processed batch 53/263 in 0.19 seconds\n",
      "  Processed batch 54/263 in 0.24 seconds\n",
      "  Processed batch 55/263 in 0.20 seconds\n",
      "  Processed batch 56/263 in 0.20 seconds\n",
      "  Processed batch 57/263 in 0.20 seconds\n",
      "  Processed batch 58/263 in 0.19 seconds\n",
      "  Processed batch 59/263 in 0.20 seconds\n",
      "  Processed batch 60/263 in 0.20 seconds\n",
      "  Processed batch 61/263 in 0.20 seconds\n",
      "  Processed batch 62/263 in 0.20 seconds\n",
      "  Processed batch 63/263 in 0.19 seconds\n",
      "  Processed batch 64/263 in 0.19 seconds\n",
      "  Processed batch 65/263 in 0.20 seconds\n",
      "  Processed batch 66/263 in 0.20 seconds\n",
      "  Processed batch 67/263 in 0.20 seconds\n",
      "  Processed batch 68/263 in 0.19 seconds\n",
      "  Processed batch 69/263 in 0.19 seconds\n",
      "  Processed batch 70/263 in 0.20 seconds\n",
      "  Processed batch 71/263 in 0.20 seconds\n",
      "  Processed batch 72/263 in 0.20 seconds\n",
      "  Processed batch 73/263 in 0.20 seconds\n",
      "  Processed batch 74/263 in 0.19 seconds\n",
      "  Processed batch 75/263 in 0.19 seconds\n",
      "  Processed batch 76/263 in 0.19 seconds\n",
      "  Processed batch 77/263 in 0.20 seconds\n",
      "  Processed batch 78/263 in 0.19 seconds\n",
      "  Processed batch 79/263 in 0.20 seconds\n",
      "  Processed batch 80/263 in 0.20 seconds\n",
      "  Processed batch 81/263 in 0.20 seconds\n",
      "  Processed batch 82/263 in 0.20 seconds\n",
      "  Processed batch 83/263 in 0.20 seconds\n",
      "  Processed batch 84/263 in 0.20 seconds\n",
      "  Processed batch 85/263 in 0.20 seconds\n",
      "  Processed batch 86/263 in 0.20 seconds\n",
      "  Processed batch 87/263 in 0.19 seconds\n",
      "  Processed batch 88/263 in 0.20 seconds\n",
      "  Processed batch 89/263 in 0.20 seconds\n",
      "  Processed batch 90/263 in 0.20 seconds\n",
      "  Processed batch 91/263 in 0.20 seconds\n",
      "  Processed batch 92/263 in 0.20 seconds\n",
      "  Processed batch 93/263 in 0.19 seconds\n",
      "  Processed batch 94/263 in 0.20 seconds\n",
      "  Processed batch 95/263 in 0.20 seconds\n",
      "  Processed batch 96/263 in 0.20 seconds\n",
      "  Processed batch 97/263 in 0.20 seconds\n",
      "  Processed batch 98/263 in 0.20 seconds\n",
      "  Processed batch 99/263 in 0.20 seconds\n",
      "  Processed batch 100/263 in 0.20 seconds\n",
      "  Processed batch 101/263 in 0.20 seconds\n",
      "  Processed batch 102/263 in 0.20 seconds\n",
      "  Processed batch 103/263 in 0.19 seconds\n",
      "  Processed batch 104/263 in 0.20 seconds\n",
      "  Processed batch 105/263 in 0.20 seconds\n",
      "  Processed batch 106/263 in 0.20 seconds\n",
      "  Processed batch 107/263 in 0.20 seconds\n",
      "  Processed batch 108/263 in 0.20 seconds\n",
      "  Processed batch 109/263 in 0.20 seconds\n",
      "  Processed batch 110/263 in 0.20 seconds\n",
      "  Processed batch 111/263 in 0.19 seconds\n",
      "  Processed batch 112/263 in 0.20 seconds\n",
      "  Processed batch 113/263 in 0.19 seconds\n",
      "  Processed batch 114/263 in 0.19 seconds\n",
      "  Processed batch 115/263 in 0.20 seconds\n",
      "  Processed batch 116/263 in 0.20 seconds\n",
      "  Processed batch 117/263 in 0.19 seconds\n",
      "  Processed batch 118/263 in 0.20 seconds\n",
      "  Processed batch 119/263 in 0.20 seconds\n",
      "  Processed batch 120/263 in 0.23 seconds\n",
      "  Processed batch 121/263 in 0.20 seconds\n",
      "  Processed batch 122/263 in 0.19 seconds\n",
      "  Processed batch 123/263 in 0.20 seconds\n",
      "  Processed batch 124/263 in 0.19 seconds\n",
      "  Processed batch 125/263 in 0.19 seconds\n",
      "  Processed batch 126/263 in 0.19 seconds\n",
      "  Processed batch 127/263 in 0.19 seconds\n",
      "  Processed batch 128/263 in 0.19 seconds\n",
      "  Processed batch 129/263 in 0.20 seconds\n",
      "  Processed batch 130/263 in 0.20 seconds\n",
      "  Processed batch 131/263 in 0.19 seconds\n",
      "  Processed batch 132/263 in 0.20 seconds\n",
      "  Processed batch 133/263 in 0.19 seconds\n",
      "  Processed batch 134/263 in 0.19 seconds\n",
      "  Processed batch 135/263 in 0.20 seconds\n",
      "  Processed batch 136/263 in 0.19 seconds\n",
      "  Processed batch 137/263 in 0.20 seconds\n",
      "  Processed batch 138/263 in 0.19 seconds\n",
      "  Processed batch 139/263 in 0.20 seconds\n",
      "  Processed batch 140/263 in 0.19 seconds\n",
      "  Processed batch 141/263 in 0.19 seconds\n",
      "  Processed batch 142/263 in 0.19 seconds\n",
      "  Processed batch 143/263 in 0.20 seconds\n",
      "  Processed batch 144/263 in 0.19 seconds\n",
      "  Processed batch 145/263 in 0.20 seconds\n",
      "  Processed batch 146/263 in 0.20 seconds\n",
      "  Processed batch 147/263 in 0.20 seconds\n",
      "  Processed batch 148/263 in 0.19 seconds\n",
      "  Processed batch 149/263 in 0.20 seconds\n",
      "  Processed batch 150/263 in 0.19 seconds\n",
      "  Processed batch 151/263 in 0.19 seconds\n",
      "  Processed batch 152/263 in 0.20 seconds\n",
      "  Processed batch 153/263 in 0.20 seconds\n",
      "  Processed batch 154/263 in 0.20 seconds\n",
      "  Processed batch 155/263 in 0.20 seconds\n",
      "  Processed batch 156/263 in 0.19 seconds\n",
      "  Processed batch 157/263 in 0.19 seconds\n",
      "  Processed batch 158/263 in 0.20 seconds\n",
      "  Processed batch 159/263 in 0.20 seconds\n",
      "  Processed batch 160/263 in 0.20 seconds\n",
      "  Processed batch 161/263 in 0.19 seconds\n",
      "  Processed batch 162/263 in 0.20 seconds\n",
      "  Processed batch 163/263 in 0.19 seconds\n",
      "  Processed batch 164/263 in 0.20 seconds\n",
      "  Processed batch 165/263 in 0.20 seconds\n",
      "  Processed batch 166/263 in 0.19 seconds\n",
      "  Processed batch 167/263 in 0.20 seconds\n",
      "  Processed batch 168/263 in 0.19 seconds\n",
      "  Processed batch 169/263 in 0.19 seconds\n",
      "  Processed batch 170/263 in 0.19 seconds\n",
      "  Processed batch 171/263 in 0.19 seconds\n",
      "  Processed batch 172/263 in 0.19 seconds\n",
      "  Processed batch 173/263 in 0.19 seconds\n",
      "  Processed batch 174/263 in 0.19 seconds\n",
      "  Processed batch 175/263 in 0.20 seconds\n",
      "  Processed batch 176/263 in 0.19 seconds\n",
      "  Processed batch 177/263 in 0.20 seconds\n",
      "  Processed batch 178/263 in 0.20 seconds\n",
      "  Processed batch 179/263 in 0.19 seconds\n",
      "  Processed batch 180/263 in 0.19 seconds\n",
      "  Processed batch 181/263 in 0.19 seconds\n",
      "  Processed batch 182/263 in 0.19 seconds\n",
      "  Processed batch 183/263 in 0.20 seconds\n",
      "  Processed batch 184/263 in 0.20 seconds\n",
      "  Processed batch 185/263 in 0.19 seconds\n",
      "  Processed batch 186/263 in 0.20 seconds\n",
      "  Processed batch 187/263 in 0.20 seconds\n",
      "  Processed batch 188/263 in 0.19 seconds\n",
      "  Processed batch 189/263 in 0.20 seconds\n",
      "  Processed batch 190/263 in 0.19 seconds\n",
      "  Processed batch 191/263 in 0.20 seconds\n",
      "  Processed batch 192/263 in 0.20 seconds\n",
      "  Processed batch 193/263 in 0.19 seconds\n",
      "  Processed batch 194/263 in 0.19 seconds\n",
      "  Processed batch 195/263 in 0.19 seconds\n",
      "  Processed batch 196/263 in 0.19 seconds\n",
      "  Processed batch 197/263 in 0.20 seconds\n",
      "  Processed batch 198/263 in 0.20 seconds\n",
      "  Processed batch 199/263 in 0.19 seconds\n",
      "  Processed batch 200/263 in 0.20 seconds\n",
      "  Processed batch 201/263 in 0.20 seconds\n",
      "  Processed batch 202/263 in 0.20 seconds\n",
      "  Processed batch 203/263 in 0.24 seconds\n",
      "  Processed batch 204/263 in 0.19 seconds\n",
      "  Processed batch 205/263 in 0.20 seconds\n",
      "  Processed batch 206/263 in 0.20 seconds\n",
      "  Processed batch 207/263 in 0.19 seconds\n",
      "  Processed batch 208/263 in 0.19 seconds\n",
      "  Processed batch 209/263 in 0.19 seconds\n",
      "  Processed batch 210/263 in 0.20 seconds\n",
      "  Processed batch 211/263 in 0.19 seconds\n",
      "  Processed batch 212/263 in 0.20 seconds\n",
      "  Processed batch 213/263 in 0.19 seconds\n",
      "  Processed batch 214/263 in 0.19 seconds\n",
      "  Processed batch 215/263 in 0.20 seconds\n",
      "  Processed batch 216/263 in 0.20 seconds\n",
      "  Processed batch 217/263 in 0.20 seconds\n",
      "  Processed batch 218/263 in 0.19 seconds\n",
      "  Processed batch 219/263 in 0.20 seconds\n",
      "  Processed batch 220/263 in 0.20 seconds\n",
      "  Processed batch 221/263 in 0.20 seconds\n",
      "  Processed batch 222/263 in 0.20 seconds\n",
      "  Processed batch 223/263 in 0.20 seconds\n",
      "  Processed batch 224/263 in 0.20 seconds\n",
      "  Processed batch 225/263 in 0.20 seconds\n",
      "  Processed batch 226/263 in 0.20 seconds\n",
      "  Processed batch 227/263 in 0.20 seconds\n",
      "  Processed batch 228/263 in 0.20 seconds\n",
      "  Processed batch 229/263 in 0.19 seconds\n",
      "  Processed batch 230/263 in 0.20 seconds\n",
      "  Processed batch 231/263 in 0.20 seconds\n",
      "  Processed batch 232/263 in 0.20 seconds\n",
      "  Processed batch 233/263 in 0.20 seconds\n",
      "  Processed batch 234/263 in 0.20 seconds\n",
      "  Processed batch 235/263 in 0.20 seconds\n",
      "  Processed batch 236/263 in 0.20 seconds\n",
      "  Processed batch 237/263 in 0.20 seconds\n",
      "  Processed batch 238/263 in 0.20 seconds\n",
      "  Processed batch 239/263 in 0.20 seconds\n",
      "  Processed batch 240/263 in 0.19 seconds\n",
      "  Processed batch 241/263 in 0.20 seconds\n",
      "  Processed batch 242/263 in 0.20 seconds\n",
      "  Processed batch 243/263 in 0.31 seconds\n",
      "  Processed batch 244/263 in 0.19 seconds\n",
      "  Processed batch 245/263 in 0.20 seconds\n",
      "  Processed batch 246/263 in 0.20 seconds\n",
      "  Processed batch 247/263 in 0.19 seconds\n",
      "  Processed batch 248/263 in 0.20 seconds\n",
      "  Processed batch 249/263 in 0.20 seconds\n",
      "  Processed batch 250/263 in 0.19 seconds\n",
      "  Processed batch 251/263 in 0.19 seconds\n",
      "  Processed batch 252/263 in 0.20 seconds\n",
      "  Processed batch 253/263 in 0.19 seconds\n",
      "  Processed batch 254/263 in 0.19 seconds\n",
      "  Processed batch 255/263 in 0.19 seconds\n",
      "  Processed batch 256/263 in 0.19 seconds\n",
      "  Processed batch 257/263 in 0.19 seconds\n",
      "  Processed batch 258/263 in 0.19 seconds\n",
      "  Processed batch 259/263 in 0.19 seconds\n",
      "  Processed batch 260/263 in 0.19 seconds\n",
      "  Processed batch 261/263 in 0.19 seconds\n",
      "  Processed batch 262/263 in 0.19 seconds\n",
      "  Processed batch 263/263 in 0.03 seconds\n",
      "Applied all transformations in 52.96 seconds\n",
      "Shuffling dataset...\n",
      "Dataset created and shuffled in 53.26 seconds\n",
      "Shape: torch.Size([2, 262150, 784])\n",
      "Creating test dataset...\n",
      "Organizing images by digit...\n",
      "Organized 10000 images in 0.16 seconds\n",
      "Creating paired dataset...\n",
      "Digit 0: 980 images → 13 pairs\n",
      "Digit 1: 1135 images → 13 pairs\n",
      "Digit 2: 1032 images → 13 pairs\n",
      "Digit 3: 1010 images → 13 pairs\n",
      "Digit 4: 982 images → 13 pairs\n",
      "Digit 5: 892 images → 13 pairs\n",
      "Digit 6: 958 images → 13 pairs\n",
      "Digit 7: 1028 images → 13 pairs\n",
      "Digit 8: 974 images → 13 pairs\n",
      "Digit 9: 1009 images → 13 pairs\n",
      "Created 130 pairs in 0.16 seconds\n",
      "Applying transformations in batches...\n",
      "  Processed batch 1/1 in 0.03 seconds\n",
      "Applied all transformations in 0.19 seconds\n",
      "Shuffling dataset...\n",
      "Dataset created and shuffled in 0.20 seconds\n",
      "Shape: torch.Size([2, 130, 784])\n",
      "\n",
      "Paired train dataset shape: torch.Size([2, 262150, 784])\n",
      "Paired test dataset shape: torch.Size([2, 130, 784])\n",
      "\n",
      "Paired train dataset shape: torch.Size([2, 262144, 784])\n",
      "Paired test dataset shape: torch.Size([2, 128, 784])\n"
     ]
    }
   ],
   "source": [
    "# Create datasets\n",
    "train_tensor = create_and_shuffle_dataset(is_train=True)\n",
    "test_tensor = create_and_shuffle_dataset(is_train=False)\n",
    "# Print shapes\n",
    "print(f\"\\nPaired train dataset shape: {train_tensor.shape}\")\n",
    "print(f\"Paired test dataset shape: {test_tensor.shape}\")\n",
    "\n",
    "# Slice up till the correct number:\n",
    "paired_train_tensor = train_tensor[:,:2**18,:]\n",
    "paired_test_tensor = test_tensor[:,:128,:]\n",
    "\n",
    "# Print shapes\n",
    "print(f\"\\nPaired train dataset shape: {paired_train_tensor.shape}\")\n",
    "print(f\"Paired test dataset shape: {paired_test_tensor.shape}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 121,
   "id": "ab82bbfd-86a3-4374-a2b3-8fc2f17ce9b7",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Datasets saved as 'paired_augmented_mnist_train.pt' and 'paired_augmented_mnist_test.pt'\n"
     ]
    }
   ],
   "source": [
    "# Move to device and save as .pt files\n",
    "paired_train_tensor = paired_train_tensor.to(device)\n",
    "paired_test_tensor = paired_test_tensor.to(device)\n",
    "\n",
    "# Save tensors\n",
    "torch.save(paired_train_tensor.cpu(), 'paired_augmented_mnist_train.pt')\n",
    "torch.save(paired_test_tensor.cpu(), 'paired_augmented_mnist_test.pt')\n",
    "\n",
    "print(\"\\nDatasets saved as 'paired_augmented_mnist_train.pt' and 'paired_augmented_mnist_test.pt'\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "0cf3c533-2419-4b17-9151-58dacf25799e",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Move to device and save as .pt files\n",
    "paired_test_tensor = torch.load('paired_augmented_mnist_test.pt', weights_only=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "7a8311b5-84b8-4a0c-b416-0aa2d2b18468",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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uno/Z2ptbY26hQsXRl36vHN4eDjqSsmP7/SaDvjf1TK/eP7556Pu6quvjrrVq1dH3X333Rd1wP8uvbecPodvaWk5krfzK7q6uqIuvUYsJZ/v33DDDVG3ePHiqEuvOdP5a19fX9Q98cQTUVdK/twxnTsvW7Ys6m666aaoS+8xbt26NerS77AU9795/aXPxNPrqwsvvDDqLr744qhLnzGl12pPP/101JWSP4M7ePBg1KXHdzonTtea/ehHP4q69F5kKaWcdNJJUdfY2Bh16Xkh4S+uAwAAAAAAAAAAAABQKQvXAQAAAAAAAAAAAAColIXrAAAAAAAAAAAAAABUysJ1AAAAAAAAAAAAAAAqZeE6AAAAAAAAAAAAAACVsnAdAAAAAAAAAAAAAIBKWbgOAAAAAAAAAAAAAEClLFwHAAAAAAAAAAAAAKBSFq4DAAAAAAAAAAAAAFCppol+AyeSyZMnR91JJ50UdR/4wAfibc+cOTPqnnvuuahbt25d1PX19UXd+Ph41FWhs7Mz6mbPnh11zc3NUTcyMhJ1qcHBwajbuXNn/JoT+bvA/01HR0fUve9974u6d77znVF38803R923vvWtqBsbG4s6oFqTJuX/lrOhoSHqRkdHD/ftvC727t0bdbt37677ttPvO/2uN27cGHUPP/xw1L344otRl8694Nep9/VIKfkYlM730+M2ndvs2rUr6rq7u6OuCul3MzAwEHWvvvpq1P34xz+OuqlTp0bd6tWro66UUk499dSoW7BgQdSZ69Zfe3t71L31rW+Nuuuuu66u233Tm94UdaWU0tPTE3X/6T/9p6i77777oq6/vz/q0ns7s2bNirpa7q/s378/6tJjLO3cA+JYt2fPnqh78MEH49c855xzou6UU06JunTcTT/L+vXro66KOVU6ZvT29kZd+rxh+/btUdfS0hJ16XVden1aS2uuNHHSa6vLL7886q6++uqoS+9fpGNAKRO3H82YMSPqFi5cGHXp/f70flYp+T2t9JoO+N/VMv489thjUTd//vyoS69R586dG3VV3AOHY9nQ0FDUpXPu6dOnR116DzrdbrruqpRSNm3aFHU//OEPo27lypVRt3z58qhrbGyMuqambAngU089FXWllPLTn/406k4//fSo+93f/d2oO+OMM6Lu7W9/e9Q99NBDUXf77bdHXSmeE/L6mzZtWtSt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oqgrpZSNGzdG3ZNPPhl1Q0ND8baZOKOjo1GXXtdVMZfu7++PuoMHD0bdjh07om7Lli1RdyIaHx+va5dek5dSyvbt26MuvUZtaGiIt32i6+joiLr0nJceszNmzKhrl95DKKWUiy++OOrS+Vx6z+bCCy+MunQe8otf/CLqnn322agrpZQNGzZEXXqemTNnTtSlYzjwq9LzcnquTa9R3/3ud0fdG97whqhL74+VUsrPf/7zqOvt7Y1f83iRzpXSjvpLr6kfffTRqHvkkUei7tprr426P/iDP4i6Uko59dRToy69Buvr64u6dH6YPu9Mx9FNmzZFXSn5M950HglUL72+Sse06dOnR106lr744otRB/zv0muCF154IeoaGxujbu3atVFXSj43T+c26X2ldG1Rep+1tbU16tJ7SqXU/351eq8v/U3S+6Dpur62traoKyWfE7v2O76kY9BZZ50VdVdccUW87XTtZ7rPHTp0KOrS+1lPPfVU1KXrKNIx8liQ/ibpNXm6ZqWUUnbt2hV16drPdI6d8BfXAQAAAAAAAAAAAAColIXrAAAAAAAAAAAAAABUysJ1AAAAAAAAAAAAAAAqZeE6AAAAAAAAAAAAAACVsnAdAAAAAAAAAAAAAIBKWbgOAAAAAAAAAAAAAEClLFwHAAAAAAAAAAAAAKBSFq4DAAAAAAAAAAAAAFApC9cBAAAAAAAAAAAAAKhU00S/gaPZ2NhY1LW1tUXdpEnZvxNoasp+lunTp0ddKflnWb9+fdQNDg5G3a233hp1jz32WNS9+OKLUffqq69GXSmlNDY2Rl36HY6Pj8fbTjQ0NETd/fffH3Vvf/vb422feeaZUTdnzpyomzlzZrxt6i/dl3bv3h11O3fujLqPfexjUVdKKQ899FDU7d27N+qam5ujLh37li5dGnXz5s2LutmzZ0ddKaXs2bMnboHapPOaUkq57777ou7SSy+NussuuyzqzjvvvKhLx71p06ZF3aZNm6KulFLuuuuuunbptleuXBl1s2bNirr0N7nqqquirpRStm3bFnU7duyIuq1bt8bbpv7S+f7IyEjULViw4EjezmtKx7Xe3t6o2759e9Rt3Lgx6tJr3uHh4ag7EfX390fdrl274tc8dOhQ1O3fvz/qWltboy49dx3P0vsNc+fOjbqTTjop6s4+++yomzp1atRdd911UVdKfs30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AAAAAFgoB9cBAAAAAAAAAAAAAFioZTV4+PDhlNu3b1/KnTt3LuXWr1+fcmOMcfbs2ZR75JFHUu706dMpt3HjxpQ7dOhQyl1Nr33ta1OulvUXvvCFWa+3evXqlNuxY0fKLcKJEydSbtu2bSlX681DDz2UcsuXL0+5KTZt2pRyy5blLueybd68OeVWrVqVcgcOHEi5D37wgyk3Ri+H2n7qvR999NGUO3nyZMr9zb/5N1NuEX7mZ34m5X7v934v5VasWJFytT3WvmrlypUpN8YY69atS7mDBw+m3N69e1Ou9qe1TdUxs44LY4xx5syZlLtw4ULK1fpwuS5dupRytawuXrw4632nXLOW/alTp1KujlG1jX33d393yn37t397yk1R57of//jHU+7o0aMpN3f9nXK9a65pfxO7dOnSlKv1sKr96JIlS1Ku1v8x+thf601tU1eiln8tr9oHTemratut5V+fpdb1Y8eOpdz58+dTboq530udb77gBS9IuTqXe+pTn5pyr3/961NujDG+9Vu/NeX++T//5yl3xx13pNzXf/3Xp9zx48dT7g1veEPKTWlTtW7PXb8uV92zqW2szmenjI11/ln3EeZWx51qyhx+zZo1KfeVGPOuRN23rHP7MfoYXNtAzdW6XcfV+u6m1P/6G+u6opbNlZh7TltNmV/Utlufpc73a66OJ7VPq+PYGGN88pOfTLmnP/3p+Zpz+qt/9a+mXJ2v1HcyRq8Pi9izKKa852LKvvbcbWXuZ/li+/fvT7k6T639z/bt21NujDFuvfXWlFu7dm3K1T3e+o4eeOCBlKvznynft2rZ1GfeuXNnyj35yU9OufpN4s4770y5q6n2U1u3bk25+i2rzlemfGvYtWtXytX1TO1HvhLqe6p91ZT9v7nnlrX86xyo7kv82q/9Wsotwnvf+96Ue/vb355ydZ0/9772IvbUq1pv6vxi7u9qU/qquk6p55gWrf7eWga1TKfMFWubqPO+z3/+8yn31//6X0+597///Sn3zd/8zbPmxhjjpS99ac5+LXjXu96Vs+95z3tSro6tc+dqG6hnmupcboze7utc6StxnmrPnj0pV9dW9Zt9PZsyxhh/8id/knL3339/yj3hCU9IuTpHrudi63vfvXt3yo3R57q1v59bXdfVM01btmzJ967rqzrG1TKs3zE/+9nPplydb07pLx588MGUq2cFn/jEJ+Z7fzn+4zoAAAAAAAAAAAAAAAvl4DoAAAAAAAAAAAAAAAvl4DoAAAAAAAAAAAAAAAvl4DoAAAAAAAAAAAAAAAvl4DoAAAAAAAAAAAAAAAvl4DoAAAAAAAAAAAAAAAvl4DoAAAAAAAAAAAAAAAvl4DoAAAAAAAAAAAAAAAvl4DoAAAAAAAAAAAAAAAu1rAYffvjhlNu/f3/KrV27NuWuv/76lBtjjBUrVqTcY489lnLnzp1LuQsXLqTc8ePHU+5v/+2/nXK/+Zu/mXKLUMv6tttuW/Av+erzEz/xEylX28Dhw4dTrra9NWvWpNwYY5w4cSLlli1rXUn9jVdi586dKVfL4dFHH025X/zFX0y5McZ497vfnXKHDh1KuVqXdu/enXJLly5NuWc84xkptwj33Xdfyq1atSrljh49mnL33nvvrPfdunVryo0xxvbt21Pummva36TV9l3r4bp161Ju8+bNKVf7lTF6eS9fvjxfc5GWLFmScnWsre/81KlTKTfGGKdPn065Wo8uXbqUcitXrky52v+86lWvSrkp6pz42c9+dsqdOXMm5eo7qWVYx44NGzak3Bi9LdYxuD5zrV9VbaPnz5/P15ySLaa058tVy6Hm6nu6ePFiyo0xxrFjx3K2qP3p3M9cc/W+U7Jz5373d3835Z71rGel3Ete8pKUe+c735lyY4zxtKc9LeU+/OEPp9wDDzyQcnWd8rM/+7Oz3ndKvZm7DSxaHXequgdU5z9j9LKq+0pzt9nf/u3fTrlv+7ZvS7mbbrop5cYYY9u2bSlX6+WrX/3qfO851fnKlHZT50r1mnU/8uTJkylX1TH97Nmz+Zp1v6Ku/VavXp3vfblqHal1va6V6/WmZGuuln99n3P3pf/+3//7lBtjjE996lMp96EPfShfs/jABz6Qcp/5zGdSbu4xc4y+lq19wdxz4pqrbWrKntKU9ldMWSNdjrpnXPvrBx98MOWmzKlqtn5/eOihh1KujqF1b7nusUzZi7nxxhtTbuPGjSlX53M33HBDytX5xQc/+MFZc2OM8dznPjflDhw4kHIHDx5Mubr/vWPHjpSrv6/mxphWx4r63f1K1L212lfV3JS9p3rNuffIajv7ru/6rpRbhNtvvz3l3vjGN6ZcncfXMqxz9vqO6zx3jP5tra4T63elOpebsl4rpuzl13Zfn2XuOdoXO3LkSMrVbzi1bU/5nlrvXdcP9d51rlT7qR//8R9Pueuuuy7lxhjjyU9+cs7O6X//7/+dcnfccUfKve9970u5KWN3bWP1u1Xtw2u/V793VlP28GofWdczi+6nxhjjYx/7WMrVOW09w1LXiWOM8fnPfz7l6nqtjsu1LtUzPnVM3rVrV8qN0fcmvhLz8y/lV3/1V1OuroWmzMXvvvvulKvttr6Xuraq767Of6acFahr2Vpv5vxG6D+uAwAAAAAAAAAAAACwUA6uAwAAAAAAAAAAAACwUA6uAwAAAAAAAAAAAACwUA6uAwAAAAAAAAAAAACwUA6uAwAAAAAAAAAAAACwUA6uAwAAAAAAAAAAAACwUA6uAwAAAAAAAAAAAACwUA6uAwAAAAAAAAAAAACwUA6uAwAAAAAAAAAAAACwUMtq8NChQyl35MiRlNu4cWPKXbx4MeWmWLasPfbp06dTrpbN8uXLU+7o0aMp9w//4T9MuTHG+Kmf+qmUu+GGG/I1/6J5xzvekXLnz59Puf3796fco48+mnJnz55NuR07dqTcGGOsX78+5Wo7rXX7SqxcuTLllixZknIHDhxIuZMnT6bcGGNs3rw55a65pv1tUX33Dz/8cMqdOXMm5Q4ePJhyt956a8qNMcZdd92VcrUu7d69O+XuueeelHvooYdSrraJ66+/PuWmZOsYV9tK7dNq7ty5cylX6/WUe9fclHtfjhUrVqRcnYfU+nbhwoWUG2OMVatWpdzVeu/Pf/7zU67W8zqXG2OM5z3veSl37NixlKvjR60Pc9evOl6OMcbq1atTbu3atSlX60Oth7Wt1H60lvWUa166dCnlpsw7LtfSpUtTrtaR+p6m9FVVnVPVXH1PV1N9LzVXy6aq48Iv/dIvpVztc8cY4y1veUvK7dy5M+V27dqVcnUu/u53vzvl6juZ0o9Xc9eHy1X3lep84MSJEyk3pZ+qdf3UqVMpV99nHaNqG3vZy16Wcs985jNTbowxPvKRj+Ts1fDRj3405epasu4JjjF/+671oY5vdY5W6/+UfmruPm0RfeQXq/PuOveqdan2fVPuXX0lyvVLqWUzZRw7fPhwytVvIh/84AdT7od/+IdTrrbvOnYtYi+mmnt+Mfd3pSnrv7n3aRbxjexy1H5l+/btKVf3nsbo+zZ/+qd/mnL1m0tdy69ZsyblNmzYkHKbNm1KuTF626nzzbrvfvz48ZSr30JuuummlLv77rtTbowxnvvc56bcm970ppSr4+W1116bcvXd1TnVlHVxVevNI488Mvu9v1j9LVVt31P64DoHqs9Sx4ma+4M/+IOUe/nLX55y//pf/+uUG2OMX/iFX0i5ufeD575e7fumjHFbtmyZ9Zr1WWo9rOvE2qamzCHr9+ra9qa8l8tRf29V+/Up+1T1fdZv0nWOXPvS2mbf8IY3zHrfMeb/vl7fy4MPPphy9WxR/f5W56Vj9H2N2kfWNlvnxLVt199Xc2P0OlbXC1+J/ZTPfvazKVfnqtu2bUu5Kd/iH3jggZSr36XrGmffvn0pV+fni/jeXM29Z1P75z179qTc3HOvMXq/W8u79i01V/ctH3vssZSr3yXG6O+vzhPqmFR8dXxtBAAAAAAAAAAAAADga5aD6wAAAAAAAAAAAAAALJSD6wAAAAAAAAAAAAAALJSD6wAAAAAAAAAAAAAALJSD6wAAAAAAAAAAAAAALJSD6wAAAAAAAAAAAAAALJSD6wAAAAAAAAAAAAAALJSD6wAAAAAAAAAAAAAALJSD6wAAAAAAAAAAAAAALNSyGlyzZk274LJ2ydOnT6fc4cOHU26MMTZt2pRy1157bcodOnQo5epv3LZtW8qtXbs25R566KGUG2OMv/bX/lrK1ff8vOc9L+X+xb/4Fyl34403ptwi3HPPPSn3+7//+ym3Y8eOlHv00UdTbvny5SlX6//WrVtTbowxVqxYkXLHjh1LuSVLluR7X676W2pdX7p06azXG2OM7du3p9yGDRtSrtalPXv2pNypU6dSbvXq1Sk3xcc+9rGUO3nyZMrV97Jz586UO3/+fMrV8WPv3r0pN8YYZ8+eTblab1auXJlydVxftWpVytU2VecJU7NF7Xcv18WLF1Outu1aN+r8Yowx1q9fn3K1HtW++ejRo7Ner7rjjjtytta32keeO3cu5Wr/U+tvbYtTyvrChQspV+tirYcnTpxIudqf1fnPNdf0vwGu2dqe67NciUuXLn1VX2+MXq61vtffWPvxakpdqq7W+6vPMndZf+ADH0i5McZ4//vfn7PFc5/73JT7oz/6o5Sr40yt11PWYLW8a3+/6PVfHRtrn1nHp3rfMfraqr73Os7XeUidy734xS9Ouec85zkpN8YYr3jFK1Juy5YtKffUpz415f74j/845V7+8pen3Nz93hh9vVv7gbpGrG221uv6HFPmNXOX95kzZ/K9L9fc85C6fpjSB889t6y/sZZ/HZ9qn1ZzY/R9m927d6dcLZva39d3V+87ZX+l1u2530utD1PG66LOf8bo7bn+xkWspf5PDzzwQMpt3rw55R73uMel3JR11X333ZdydS+t7jdcf/31KVfrxyLm0XWcr33u5z73uZSre3i33HJLytX69T//5/9MuTH6N706v6jrhVo2ta+/7bbbUq7uj43R+9zapuq3gStR+8L6zb6WQe0vxuhjY60j9Zk3btyYcrfffnvK1fZY5xdjjLFu3bqUq+Vd1xq176v9QJ17TZlv1j5/7rM19Xr1O2t9x1PqTVV/Y20rl6u+91pW9R3V9jDF3N8zDh48mHJHjhxJufou6zxpjF436zPXvbk6H6jfrep8ZcqZoTrHqGVYx7da1rWtHDhwIOVqWY/RzwDO3e6vRF3j1POIc899x+jlWq9Zf2PtT48fP55ydV97yhmO2nbr3nv1yU9+MuXqenLuudwYY1x33XUpV9eo9XtDnXvVvuXhhx9OubpWG2OMJz3pSSlXzzLWNlD4j+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACzUshpct25dyq1fvz7lVqxYkXJbtmxJuTHG2LRpU8rV33j27NmUO3r0aModO3Ys5S5dupRyx48fT7kxxjhz5kzKrV27NuXuvPPOlPuBH/iBlNu3b1/K7dmzJ+UOHTqUcmOMsXHjxpSr9Wv79u0pt23btpS7cOFCyh0+fDjlan0do7eBkydPplyth1di6dKlKbdq1aqUq+9zyZIlKTfGGDfeeGPK1fZYy3X//v0pV9/7S17ykpSrzzvGGOfOnUu52sbre968eXPK7d69O+U2bNiQcgcPHky5MfrYVfuqZcvaFGDKWDPnfeu8Y0q2zilqvblctZ6fOnUq5eq8oZb9GL0M6jVrH3nx4sWU+6Ef+qGUe/vb355ytc2OMX8bq89c58517Fi+fHnK1Xo4JVvL5vz58ylX20DN1Xdy+vTplJty79pW1qxZk+99uWo51N9cc3UuN8YY11zT/g673nvuulQt4r5z17m51fpVc1NMqWPFhz70oZSr76TW60XUm7nfy6Lr15EjR1KulmnN1fFpjL62qvVy9erVs+bqs9Q1we23355yY4zxG7/xGzlb1PnF3G2n7ilN6c/qPlCdz9W5/dxzr/ocU/qKes3apuYeE76UlStXplxdJ9bclDpX69Lca5z6Pusz1/c5ZW1cy2busaaau6yn1JtajnOXTX2Wmpt7nTglO2VOsUg7d+5MubovUec/da96jDH27t2bcrXN3nzzzSl3/fXXp9yDDz6YcvUbYf3eMsYY1113XcrV91zV9zf3N6Ep7aaOC1erb577vnW8HKO3lRMnTqRc3Y+8EvUbQP0GWvuqmhujz/tq3Zx7P7her/6+KWVT16h1XK77snOvtes7nrIXU+tsbeN1rKlzuZqb+5vSGP0913nfotX6VutRPUsyZayt76mOZfV8Sp031HGnluGU7yj1+1bN1e/6Vb1e3aea8l20qvuHdQ4093q3jltT9qnm/u71lVgj3nTTTbNeb8eOHSk35ezHlDM0RW0XtS49+uijKVfb2datW1NujD5nmdt//s//OeV27dqVcrU9PvDAAyk3Rj9DVr8r1XV5HbtqH1mvV+cJY/Rzj3VdceDAgXzvL8d/XAcAAAAAAAAAAAAAYKEcXAcAAAAAAAAAAAAAYKEcXAcAAAAAAAAAAAAAYKEcXAcAAAAAAAAAAAAAYKEcXAcAAAAAAAAAAAAAYKEcXAcAAAAAAAAAAAAAYKEcXAcAAAAAAAAAAAAAYKEcXAcAAAAAAAAAAAAAYKEcXAcAAAAAAAAAAAAAYKGW1eDp06dTbvny5Sm3e/fulNuxY0fKjTHGsmXtcU6cOJFy9ZnPnDmTcseOHUu5Q4cOpdzx48dTbowx1q5dm3IrV65MufPnz6fc0aNHU+7cuXMpV+vXxYsXU26MMfbt25dyBw8eTLlab7Zt25ZyS5cuTbla/+u7G6M/y6lTp2a93pW47rrrUq62iZqr5T9G/421vtc+7ciRIylX6/qBAwdSbtWqVSk3Rn/mkydPzpqrNm7cmHKbN2+eNTdGL8cVK1bkaxa1DOuY9Oijj6bcNdfM/7d1tQxrG71ctZ6vXr065eoYOmVsPHv2bMrV97RkyZKUq31pvV6d19SyHmOMdevWpdylS5dSro7La9asSbn169enXO0r6lxuSrbWr5qr85DqwoULKVfXAGP091zbQB2PvhJqXa9q+16E+ixz5/5fUJ+ltp+rWYa1jtVc/Y1TxuE57zulDOcu77mf+YvVcaL2rfX5p6xr6zxt7vVprb9zz/UXMYev76/O9etcrs6VpuwFVHXeUMt77jYw9zup66Mp9567bK5ELYfabuvct/Y/U7J1fVXV91RzddyZMt7Ve9c+o77nOqeq/cUixuR6zSl1cc7r1XlCfY4pY9zcc+dFr5Ge8IQnpFz9bnXXXXel3D333JNyY4zxyCOPpFxdK9e2U9f9dX5R1boxRv82MPde2kMPPTRrbso3hKo+c51j17Zd603dj6x7glO+Gdd1QO37brnllnzvy1XH2vqba3lNmTfUdzr3uqmOZbUMa19a+58x5u936zg/d72p15uyX13H17n31GvfN/e8dMo5hXpGorbTub8jfLF6FqiuQ+taeRHr2tpH1vpWn2XTpk0pN/d3sDF6W5z7zFCdR27ZsiXlahlOMff3rTq3qfPDuc8y1no9Rm/3tf85fPhwvvfl2rlzZ8rVcyw333xzym3YsCHlxujnkPbv359ydY5Wx8b6PbzWzdomxri631CLb/zGb5z1elPmm//rf/2vlJv7TGftx2v//MQnPjHlppxp2r59e8rVM4V79uzJ9/5y/Md1AAAAAAAAAAAAAAAWysF1AAAAAAAAAAAAAAAWysF1AAAAAAAAAAAAAAAWysF1AAAAAAAAAAAAAAAWysF1AAAAAAAAAAAAAAAWysF1AAAAAAAAAAAAAAAWysF1AAAAAAAAAAAAAAAWysF1AAAAAAAAAAAAAAAWysF1AAAAAAAAAAAAAAAWalkNnjp1KuWWLFmScqtWrUq5CxcupNwYYxw5ciTljh49mnLHjx9Pufob165dm3IXL15MuTNnzqTcIu794IMPptw117S/jdi+fXvK3XDDDSm3adOmlBujP0utN4888kjKnT59OuUuXbqUcqtXr065bdu2pdwYY5w/fz7lanuu/ciVWL58ecrVdrtsWesma10fY4wTJ06k3IoVK1Ku9rsrV66cNVfr5pR+fMeOHSlX6/u+fftS7t577025LVu2pNx1112Xcrt27Uq5MXqfP3cftHnz5pSrY/Bdd92VcrVfGaOXTR3X65h5uWobq2V/8uTJlJsyb6jtto639Zlr267v8ty5cylXx7sxet9X1fGjztGqer1169bNet8x+jg491zp7NmzKVfr65S+os6B6m+cUmcvV51f1PKvda7ed4zefupvnLt9z23p0qU5O3ffMneumjLHrupvrPeuuVq36++rY/Ui2lS16DZV5w1r1qxJudq3TlnfzL2er/3A3ONOrUdT6lstm7rGr++55qq5+5Qx+nuu8/uaq/etc6W6n3I1x7evprG/llfdK6rte4ze/9Vyrb+xPnO1iL5qbnOvXebug2r7HmMx+4JzulrrmTF6edcxbtF19tixYyn3wAMPpNzhw4dTbsrYWPcJa79y8ODBlKv9480335xydR5S38kYY9x9990pV/fSapuo+5H1ndTvAlPaYv2NdY++3nv//v0pd+2116bc4x//+JR77LHHUm6MMT73uc+lXG0D9f1dibpPOGUPfG61v679f93bnrvfre22/r4xer9W5w21r6r1pqrj0ZQ1Tq03c+9TzT2Xq+NM7ZvH6M9S1x9z71t+sTqHP3ToUMot4rnmXvutX78+5eo4UdvY1VyL1Ho5pY8san2o77jWwzH6Oan6zayeV6r1sI5bdf91ythR5x21Lk4ZPxbttttuS7m6j7CIfaq6Rq3fup/2tKelXJ2ff/rTn065T33qUyk3Rl/L/tiP/VjK3X///SlX3189O1JzU77Fz30ebu4+qK47b7nllpSr48IY/dxcrQ9z7lP5j+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACyUg+sAAAAAAAAAAAAAACzUkkuXLl262j8CAAAAAAAAAAAAAICvXf7jOgAAAAAAAAAAAAAAC+XgOgAAAAAAAAAAAAAAC+XgOgAAAAAAAAAAAAAAC+XgOgAAAAAAAAAAAAAAC+XgOgAAAAAAAAAAAAAAC+XgOgAAAAAAAAAAAAAAC+XgOgAAAAAAAAAAAAAAC+XgOgAAAAAAAAAAAAAAC+XgOgAAAAAAAAAAAAAAC/X/AfVi4DCafsAFAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 3000x600 with 20 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Optional: Visualize a few examples\n",
    "def visualize_examples(tensor, n=3, save_name=None):\n",
    "    import matplotlib.pyplot as plt\n",
    "    \n",
    "    fig, axes = plt.subplots(2, n, figsize=(3*n, 6))\n",
    "    choices = np.random.choice(len(tensor[0]), n)\n",
    "    for i, idx in enumerate(choices):\n",
    "        # Display View 1 (rotation+scale)\n",
    "        axes[0, i].imshow(tensor[0, idx].reshape(28, 28), cmap='gray')\n",
    "        # axes[0, i].set_title(f\"View 1: Rotation+Scale\")\n",
    "        axes[0, i].axis('off')\n",
    "        \n",
    "        # Display View 2 (Perlin noise)\n",
    "        axes[1, i].imshow(tensor[1, idx].reshape(28, 28), cmap='gray')\n",
    "        # axes[1, i].set_title(f\"View 2: Perlin Noise\")\n",
    "        axes[1, i].axis('off')\n",
    "    \n",
    "    plt.tight_layout()\n",
    "    if save_name:\n",
    "        plt.savefig(save_name, dpi=300, format='pdf', bbox_inches='tight')\n",
    "\n",
    "    plt.show()\n",
    "\n",
    "# Uncomment to visualize\n",
    "visualize_examples(paired_test_tensor, 10, 'mnist_si_examples.pdf')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 99,
   "id": "7a7552bf-4b0a-4915-a659-51608f3b43a8",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Teacher dataset:\n",
    "data_params = {\n",
    "    'Nx': 500,\n",
    "    'Ny': 500,\n",
    "    'ncom': 10,\n",
    "    'test_size': 128,\n",
    "    'samples': 2**18,\n",
    "    'mi': 4,\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 101,
   "id": "6206d00e-2363-4025-b529-62f489286b12",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Define a Teacher Model for X and Y\n",
    "teacher_model_x = teacher(dz=data_params['ncom'], output_dim=data_params['Nx'])\n",
    "teacher_model_y = teacher(dz=data_params['ncom'], output_dim=data_params['Ny'])\n",
    "\n",
    "for param_x in teacher_model_x.parameters():\n",
    "    param_x.requires_grad_(False)  # Freeze\n",
    "for param_y in teacher_model_y.parameters():\n",
    "    param_y.requires_grad_(False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 103,
   "id": "6e77208a-e04d-49a3-b89f-560e8174f4b7",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_X_teacher, train_Y_teacher = sample_correlated_data(ncom=data_params['ncom'], total_size=data_params['Nx'], batch_size=data_params['samples'], info_tot=data_params['mi'], mlp_x=teacher_model_x, mlp_y=teacher_model_y)\n",
    "test_X_teacher, test_Y_teacher = sample_correlated_data(ncom=data_params['ncom'], total_size=data_params['Nx'], batch_size=data_params['test_size'], info_tot=data_params['mi'], mlp_x=teacher_model_x, mlp_y=teacher_model_y)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 115,
   "id": "0ab23918-cb24-4848-827c-b9dd2d1511fa",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([2, 262144, 500])"
      ]
     },
     "execution_count": 115,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.stack([train_X_teacher,train_Y_teacher])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 128,
   "id": "1fbcfed5-0cc1-46d9-a33c-93c0875c9ee1",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Save tensors\n",
    "torch.save(torch.stack([train_X_teacher,train_Y_teacher]), 'paired_teacher_train.pt')\n",
    "torch.save(torch.stack([test_X_teacher,test_Y_teacher]), 'paired_teacher_test.pt')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "866ef8a5-0259-40f6-b470-0daa3761f98d",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.12.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
