{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Load Model Instance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "from models.TransformerAE.model import TransformerAE\n",
    "import torch\n",
    "import torch.nn as nn"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "autoencoder = TransformerAE(10, False, [20, 4])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([32, 10, 1])"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "seq = 10\n",
    "batch = 32\n",
    "feature = 1\n",
    "\n",
    "# input = torch.rand(size=(seq, feature))  # Single sample\n",
    "input = torch.rand(size=(batch, seq, feature)) # Batched samples\n",
    "input.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([32, 4])\n"
     ]
    }
   ],
   "source": [
    "restored, latent = autoencoder.forward(input)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(torch.Size([32, 10]), torch.Size([32, 4]))"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "restored.shape, latent.shape"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import logging\n",
    "from models.dataloaders import get_flattened_mnist_dataloaders\n",
    "train_loader, test_loader = get_flattened_mnist_dataloaders(batch_size=128, size_share=0.2)\n",
    "\n",
    "# Setup logging\n",
    "logging.basicConfig(\n",
    "    level=logging.INFO, format=\"%(process)d - %(levelname)s - %(message)s\"\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "TransformerAE(\n",
      "  (encoder): Sequential(\n",
      "    (0): TransformerEncoderLayer(\n",
      "      (self_attn): MultiheadAttention(\n",
      "        (out_proj): NonDynamicallyQuantizableLinear(in_features=1, out_features=1, bias=True)\n",
      "      )\n",
      "      (linear1): Linear(in_features=1, out_features=512, bias=True)\n",
      "      (dropout): Dropout(p=0.1, inplace=False)\n",
      "      (linear2): Linear(in_features=512, out_features=1, bias=True)\n",
      "      (norm1): LayerNorm((1,), eps=1e-05, elementwise_affine=True)\n",
      "      (norm2): LayerNorm((1,), eps=1e-05, elementwise_affine=True)\n",
      "      (dropout1): Dropout(p=0.1, inplace=False)\n",
      "      (dropout2): Dropout(p=0.1, inplace=False)\n",
      "    )\n",
      "    (1): Flatten(start_dim=1, end_dim=-1)\n",
      "    (2): Linear(in_features=1024, out_features=256, bias=True)\n",
      "    (3): Tanh()\n",
      "  )\n",
      "  (decoder): Sequential(\n",
      "    (0): Unflatten(dim=1, unflattened_size=(256, 1))\n",
      "    (1): TransformerEncoderLayer(\n",
      "      (self_attn): MultiheadAttention(\n",
      "        (out_proj): NonDynamicallyQuantizableLinear(in_features=1, out_features=1, bias=True)\n",
      "      )\n",
      "      (linear1): Linear(in_features=1, out_features=512, bias=True)\n",
      "      (dropout): Dropout(p=0.1, inplace=False)\n",
      "      (linear2): Linear(in_features=512, out_features=1, bias=True)\n",
      "      (norm1): LayerNorm((1,), eps=1e-05, elementwise_affine=True)\n",
      "      (norm2): LayerNorm((1,), eps=1e-05, elementwise_affine=True)\n",
      "      (dropout1): Dropout(p=0.1, inplace=False)\n",
      "      (dropout2): Dropout(p=0.1, inplace=False)\n",
      "    )\n",
      "    (2): Flatten(start_dim=1, end_dim=-1)\n",
      "    (3): Linear(in_features=256, out_features=1024, bias=True)\n",
      "    (4): Tanh()\n",
      "  )\n",
      ")\n"
     ]
    }
   ],
   "source": [
    "# Construct model\n",
    "from models.TransformerAE.model import TransformerAE\n",
    "\n",
    "model = TransformerAE(input_dim=1024)\n",
    "print(model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "96598 - INFO - Step 10 loss: 0.05786\n",
      "96598 - INFO - Step 20 loss: 0.05844\n",
      "96598 - INFO - Step 30 loss: 0.05905\n",
      "96598 - INFO - Step 40 loss: 0.05651\n",
      "96598 - INFO - Step 50 loss: 0.05805\n",
      "96598 - INFO - Step 60 loss: 0.05544\n",
      "96598 - INFO - Step 70 loss: 0.05676\n",
      "96598 - INFO - Step 80 loss: 0.05723\n",
      "96598 - INFO - Step 90 loss: 0.05620\n",
      "96598 - INFO - Epoch [1/2], Train Loss: 0.056672, Test Loss: 0.053710\n",
      "\n",
      "96598 - INFO - Step 100 loss: 0.05461\n",
      "96598 - INFO - Step 110 loss: 0.05807\n",
      "96598 - INFO - Step 120 loss: 0.05619\n",
      "96598 - INFO - Step 130 loss: 0.05633\n",
      "96598 - INFO - Step 140 loss: 0.05661\n",
      "96598 - INFO - Step 150 loss: 0.05489\n",
      "96598 - INFO - Step 160 loss: 0.05645\n",
      "96598 - INFO - Step 170 loss: 0.05666\n",
      "96598 - INFO - Step 180 loss: 0.05476\n",
      "96598 - INFO - Epoch [2/2], Train Loss: 0.056688, Test Loss: 0.053688\n",
      "\n"
     ]
    }
   ],
   "source": [
    "from models.TransformerAE.training import TransformerAETrainer\n",
    "\n",
    "trainer = TransformerAETrainer(model=model, print_loss_every=10)\n",
    "trainer.train(train_loader=train_loader, test_loader=test_loader, epochs=2, lr=1e-4)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/h3/r2ql0vxd0w73xqwt1lr7s5r80000gr/T/ipykernel_96598/3775784357.py:6: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  recon, latent = model(torch.tensor(data).unsqueeze(-1))\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 300x900 with 10 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import torch \n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "data = next(iter(train_loader))\n",
    "with torch.no_grad():\n",
    "    recon, latent = model(torch.tensor(data).unsqueeze(-1))\n",
    "\n",
    "reconstructions = recon[:5]\n",
    "originals = data[:5]\n",
    "figsize = (3, 9)\n",
    "fig, axs = plt.subplots(5, 2, figsize=figsize)\n",
    "\n",
    "for i in range(5):\n",
    "    axs[i, 0].imshow(reconstructions[i].view(32, 32), cmap='gray')\n",
    "    axs[i, 0].set_xticks([])\n",
    "    axs[i, 0].set_yticks([])\n",
    "\n",
    "    axs[i, 1].imshow(originals[i].view(32, 32), cmap='gray')\n",
    "    axs[i, 1].set_xticks([])\n",
    "    axs[i, 1].set_yticks([])\n",
    "\n",
    "plt.subplots_adjust(wspace=0, hspace=0)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "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.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
