{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Training example\n",
    "Using MNIST to demonstrate the capabilities of the DeepCAE as proposed in https://arxiv.org/abs/2402.18164."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "import logging\n",
    "from models.data_utils.dataloaders import get_flattened_mnist_dataloaders\n",
    "train_loader, test_loader = get_flattened_mnist_dataloaders(batch_size=64, 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": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "DeepCAE(\n",
      "  (encoder): Sequential(\n",
      "    (0): Linear(in_features=1024, out_features=512, bias=True)\n",
      "    (1): Tanh()\n",
      "    (2): Linear(in_features=512, out_features=256, bias=True)\n",
      "    (3): Tanh()\n",
      "  )\n",
      "  (decoder): Sequential(\n",
      "    (0): Linear(in_features=256, out_features=512, bias=True)\n",
      "    (1): Tanh()\n",
      "    (2): Linear(in_features=512, out_features=1024, bias=True)\n",
      "    (3): Tanh()\n",
      "  )\n",
      ")\n"
     ]
    }
   ],
   "source": [
    "# Construct model\n",
    "from models.architectures.DeepCAE.model import DeepCAE\n",
    "\n",
    "model = DeepCAE(input_dim=1024)\n",
    "print(model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Train model\n",
    "from models.architectures.DeepCAE.training import DeepCAETrainer\n",
    "\n",
    "trainer = DeepCAETrainer(model=model, print_loss_every=50, lambda_c=1e-4)\n",
    "trainer.train(train_loader=train_loader, test_loader=test_loader, epochs=10, lr=1e-4, checkpointing=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "\n",
    "# Save model\n",
    "name = \"DeepCAE_contractive_60_epochs_mnist\"\n",
    "torch.save(model.state_dict(), f\"./model_saves/{name}.pt\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Load model back in\n",
    "model_test = torch.load(f\"./model_saves/{name}.pt\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Visualize reconstructions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/8t/j2v08q4544q0cfzh_2p0sy_80000gr/T/ipykernel_83496/3552565978.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))\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))\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()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
