{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Notebook for Adversarial Example of ViT on MNIST\n",
    "### Figure paper for Adv Attacks FGSM"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "from torchvision import datasets\n",
    "import numpy as np\n",
    "import torch\n",
    "from torchvision import transforms\n",
    "from torch.utils.data import DataLoader\n",
    "from vit_pytorch import ViT  # ViT model\n",
    "\n",
    "\n",
    "BATCH_SIZE  =32\n",
    "mnist_transforms = transforms.Compose([transforms.ToTensor()])\n",
    "train_val_dataset = datasets.MNIST(root=\"./datasets/\", train=True, download=False, transform=mnist_transforms)\n",
    "test_dataset = datasets.MNIST(root=\"./datasets/\", train=False, download=False, transform=mnist_transforms)\n",
    "train_size = int(0.9 * len(train_val_dataset))\n",
    "val_size = len(train_val_dataset) - train_size\n",
    "train_dataset, val_dataset = torch.utils.data.random_split(dataset=train_val_dataset, lengths=[train_size, val_size])\n",
    "train_dataloader = DataLoader(dataset=train_dataset, batch_size=BATCH_SIZE, shuffle=True)\n",
    "val_dataloader = DataLoader(dataset=val_dataset, batch_size=BATCH_SIZE, shuffle=True)\n",
    "test_dataloader = DataLoader(dataset=test_dataset, batch_size=BATCH_SIZE, shuffle=True)\n",
    "\n",
    "\n",
    "device = 'cuda' if torch.cuda.is_available() else 'cpu'\n",
    "\n",
    "\n",
    "from torch import nn\n",
    "\n",
    "class LeNet5(nn.Module):\n",
    "    def __init__(self):\n",
    "        super().__init__()\n",
    "        self.feature = nn.Sequential(\n",
    "            #1\n",
    "            nn.Conv2d(in_channels=1, out_channels=6, kernel_size=5, stride=1, padding=2),   # 28*28->32*32-->28*28\n",
    "            nn.Tanh(),\n",
    "            nn.AvgPool2d(kernel_size=2, stride=2),  # 14*14\n",
    "            \n",
    "            #2\n",
    "            nn.Conv2d(in_channels=6, out_channels=16, kernel_size=5, stride=1),  # 10*10\n",
    "            nn.Tanh(),\n",
    "            nn.AvgPool2d(kernel_size=2, stride=2),  # 5*5\n",
    "            \n",
    "        )\n",
    "        self.classifier = nn.Sequential(\n",
    "            nn.Flatten(),\n",
    "            nn.Linear(in_features=16*5*5, out_features=120),\n",
    "            nn.Tanh(),\n",
    "            nn.Linear(in_features=120, out_features=84),\n",
    "            nn.Tanh(),\n",
    "            nn.Linear(in_features=84, out_features=10),\n",
    "        )\n",
    "        \n",
    "    def forward(self, x):\n",
    "        return self.classifier(self.feature(x))\n",
    "    \n",
    "\n",
    "### Load trained models\n",
    "\n",
    "model_vit_v1 = ViT(\n",
    "    image_size=28,       # MNIST images are 28x28\n",
    "    patch_size=4,        # 4 patches per dimension (28/7)\n",
    "    num_classes=10,      # 10 output classes for digits 0-9\n",
    "    dim=64,             # Embedding dimension\n",
    "    depth=6,             # Number of transformer layers\n",
    "    heads=8,             # Number of attention heads\n",
    "    mlp_dim=128,         # Hidden dimension in MLP layers\n",
    "    dropout=0.1,         # Dropout rate in transformer\n",
    "    emb_dropout=0.1,     # Dropout rate for embeddings\n",
    "    channels = 1\n",
    ").to(device)\n",
    "\n",
    "\n",
    "model_vit_v2 = ViT(\n",
    "    image_size=28,       # MNIST images are 28x28\n",
    "    patch_size=7,        # 4 patches per dimension (28/7)\n",
    "    num_classes=10,      # 10 output classes for digits 0-9\n",
    "    dim=64,             # Embedding dimension\n",
    "    depth=6,             # Number of transformer layers\n",
    "    heads=8,             # Number of attention heads\n",
    "    mlp_dim=128,         # Hidden dimension in MLP layers\n",
    "    dropout=0.1,         # Dropout rate in transformer\n",
    "    emb_dropout=0.1,     # Dropout rate for embeddings\n",
    "    channels = 1\n",
    ").to(device)\n",
    "\n",
    "model_lenet5 = LeNet5().to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_45419/2789388921.py:9: FutureWarning: You are using `torch.load` with `weights_only=False` (the current default value), which uses the default pickle module implicitly. It is possible to construct malicious pickle data which will execute arbitrary code during unpickling (See https://github.com/pytorch/pytorch/blob/main/SECURITY.md#untrusted-models for more details). In a future release, the default value for `weights_only` will be flipped to `True`. This limits the functions that could be executed during unpickling. Arbitrary objects will no longer be allowed to be loaded via this mode unless they are explicitly allowlisted by the user via `torch.serialization.add_safe_globals`. We recommend you start setting `weights_only=True` for any use case where you don't have full control of the loaded file. Please open an issue on GitHub for any issues related to this experimental feature.\n",
      "  model_lenet5.load_state_dict(torch.load(MODEL_SAVE_PATH))\n",
      "/tmp/ipykernel_45419/2789388921.py:17: FutureWarning: You are using `torch.load` with `weights_only=False` (the current default value), which uses the default pickle module implicitly. It is possible to construct malicious pickle data which will execute arbitrary code during unpickling (See https://github.com/pytorch/pytorch/blob/main/SECURITY.md#untrusted-models for more details). In a future release, the default value for `weights_only` will be flipped to `True`. This limits the functions that could be executed during unpickling. Arbitrary objects will no longer be allowed to be loaded via this mode unless they are explicitly allowlisted by the user via `torch.serialization.add_safe_globals`. We recommend you start setting `weights_only=True` for any use case where you don't have full control of the loaded file. Please open an issue on GitHub for any issues related to this experimental feature.\n",
      "  model_vit_v2.load_state_dict(torch.load(MODEL_SAVE_PATH))\n",
      "/tmp/ipykernel_45419/2789388921.py:25: FutureWarning: You are using `torch.load` with `weights_only=False` (the current default value), which uses the default pickle module implicitly. It is possible to construct malicious pickle data which will execute arbitrary code during unpickling (See https://github.com/pytorch/pytorch/blob/main/SECURITY.md#untrusted-models for more details). In a future release, the default value for `weights_only` will be flipped to `True`. This limits the functions that could be executed during unpickling. Arbitrary objects will no longer be allowed to be loaded via this mode unless they are explicitly allowlisted by the user via `torch.serialization.add_safe_globals`. We recommend you start setting `weights_only=True` for any use case where you don't have full control of the loaded file. Please open an issue on GitHub for any issues related to this experimental feature.\n",
      "  model_vit_v1.load_state_dict(torch.load(MODEL_SAVE_PATH))\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from pathlib import Path\n",
    "\n",
    "MODEL_PATH = Path(\"models\")\n",
    "MODEL_PATH.mkdir(parents=True, exist_ok=True)\n",
    "MODEL_NAME = \"LENeT5_mnist.pth\"\n",
    "MODEL_SAVE_PATH = MODEL_PATH / MODEL_NAME\n",
    "\n",
    "\n",
    "model_lenet5.load_state_dict(torch.load(MODEL_SAVE_PATH))\n",
    "\n",
    "\n",
    "MODEL_PATH = Path(\"models\")\n",
    "MODEL_PATH.mkdir(parents=True, exist_ok=True)\n",
    "MODEL_NAME = \"ViT_v2_mnist.pth\"\n",
    "MODEL_SAVE_PATH = MODEL_PATH / MODEL_NAME\n",
    "\n",
    "model_vit_v2.load_state_dict(torch.load(MODEL_SAVE_PATH))\n",
    "\n",
    "\n",
    "MODEL_PATH = Path(\"models\")\n",
    "MODEL_PATH.mkdir(parents=True, exist_ok=True)\n",
    "MODEL_NAME = \"ViT_v1_mnist.pth\"\n",
    "MODEL_SAVE_PATH = MODEL_PATH / MODEL_NAME\n",
    "\n",
    "model_vit_v1.load_state_dict(torch.load(MODEL_SAVE_PATH))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "from art.attacks.evasion import FastGradientMethod\n",
    "from art.estimators.classification import PyTorchClassifier\n",
    "import torch.nn as nn\n",
    "from torch.optim import Adam\n",
    "from torchmetrics import Accuracy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 112,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy on benign test examples: 98.03%\n"
     ]
    }
   ],
   "source": [
    "model_id = model_vit_v1\n",
    "\n",
    "loss_fn = nn.CrossEntropyLoss()\n",
    "optimizer = Adam(model_id.parameters(), lr=0.001)\n",
    "classifier = PyTorchClassifier(model=model_id,clip_values=(0,1),loss=loss_fn,optimizer=optimizer,input_shape=(1,28,28),nb_classes=10)\n",
    "nb_correct=0\n",
    "for x,y in test_dataloader:\n",
    "    y_pred = classifier.predict(x)\n",
    "    nb_correct+=np.sum(np.argmax(y_pred,axis=1)==y.detach().cpu().numpy())\n",
    "\n",
    "print(\"Accuracy on benign test examples: {}%\".format(nb_correct/len(test_dataset)*100))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 113,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy on adversarial test examples with eps:0.0001 : 97.74000000000001%\n",
      "Accuracy on adversarial test examples with eps:0.0003 : 95.48%\n",
      "Accuracy on adversarial test examples with eps:0.0005 : 89.60000000000001%\n",
      "Accuracy on adversarial test examples with eps:0.001 : 65.34%\n",
      "Accuracy on adversarial test examples with eps:0.005 : 36.08%\n",
      "Accuracy on adversarial test examples with eps:0.01 : 32.06%\n",
      "Accuracy on adversarial test examples with eps:0.05 : 20.919999999999998%\n",
      "Accuracy on adversarial test examples with eps:0.1 : 13.08%\n"
     ]
    }
   ],
   "source": [
    "epsilons = [0.0001,0.0003,0.0005,0.001,0.005,0.01,0.05,0.1]\n",
    "epsilons_CNN = list(np.linspace(0.01,0.1,5))\n",
    "accuracies = []\n",
    "REPORTS= []\n",
    "for eps in epsilons:\n",
    "    nb_missclassified_FGSM=0\n",
    "    nb_missclassified_PGD=0\n",
    "\n",
    "\n",
    "    REPORT = {'original':{'y':[],'x':[]},'attack_FGSM':{'y':[],'x':[]},'attack_PGD':{'y':[],'x':[]}}\n",
    "\n",
    "\n",
    "    attack = FastGradientMethod(estimator=classifier, eps=eps)\n",
    "    nb_correct_attack=0\n",
    "    torch.manual_seed(10)\n",
    "    for x,y in test_dataloader:\n",
    "        REPORT['original']['x'].append(x)\n",
    "\n",
    "        x_test_fgm = attack.generate(x=x.detach().cpu().numpy())\n",
    "        REPORT['attack_FGSM']['x'].append(x_test_fgm)\n",
    "\n",
    "        #y_pred = classifier.predict(x)\n",
    "        REPORT['original']['y'].append(y.detach().cpu().numpy())\n",
    "        y_pred = classifier.predict(x_test_fgm)\n",
    "        REPORT['attack_FGSM']['y'].append(np.argmax(y_pred,axis=1))\n",
    "\n",
    "        nb_correct_attack+=np.sum(np.argmax(y_pred,axis=1)==y.detach().cpu().numpy())\n",
    "\n",
    "    print(\"Accuracy on adversarial test examples with eps:{} : {}%\".format(eps,nb_correct_attack/len(test_dataset) * 100))\n",
    "\n",
    "\n",
    "\n",
    "    REPORT['original']['y'] = np.concatenate(REPORT['original']['y'])\n",
    "    REPORT['original']['x'] = np.concatenate(REPORT['original']['x'])\n",
    "\n",
    "    REPORT['attack_FGSM']['y'] = np.concatenate(REPORT['attack_FGSM']['y'])\n",
    "    REPORT['attack_FGSM']['x'] = np.concatenate(REPORT['attack_FGSM']['x'])\n",
    "\n",
    "    REPORTS.append(REPORT)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 121,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 300x500 with 6 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "c1_list = [5,7,9]\n",
    "c2_list = [6,1,3]\n",
    "\n",
    "x_false = []\n",
    "x_true =[]\n",
    "\n",
    "eps_summary = 0.0005\n",
    "\n",
    "REPORT_plot = REPORTS[epsilons.index(eps_summary)]\n",
    "### get index of c1 in original\n",
    "#print(REPORT)\n",
    "row = 0\n",
    "fig,ax = plt.subplots(3,2,figsize=(3,5))\n",
    "for c1,c2 in zip(c1_list,c2_list):\n",
    "    #x_false = []\n",
    "    #x_true =[]\n",
    "    ind = REPORT_plot['original']['y'] == c1\n",
    "\n",
    "    x_true = REPORT_plot['original']['x'][ind]\n",
    "    ### get index in pertubed of c1 but with c2 predictions\n",
    "    ind_wrong = REPORT_plot['attack_FGSM']['y'][ind] == c2\n",
    "    ind_wrong\n",
    "    x_false_FGSM = REPORT_plot['attack_FGSM']['x'][ind][ind_wrong]\n",
    "\n",
    "\n",
    "    ax[row,0].imshow(x_false_FGSM.mean(axis=0)[0,:,:],cmap='gray')\n",
    "    ax[row,0].set_title(f\"{c1} -> {c2}\")\n",
    "    ax[row,1].imshow(np.abs(x_true.mean(axis=0)[0,:,:]-x_false_FGSM.mean(axis=0)[0,:,:])**4,cmap='plasma')\n",
    "    #ax[row,2].imshow(np.abs(x_true.mean(axis=0)[0,:,:].numpy()-x_false_PGD.mean(axis=0)[0,:,:].numpy())**4,cmap='plasma')\n",
    "    ax[row,0].set_xticks([])\n",
    "    ax[row,1].set_xticks([])\n",
    "    ax[row,0].set_yticks([])\n",
    "    ax[row,1].set_yticks([])\n",
    "    row+=1\n",
    "\n",
    "fig.tight_layout()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 122,
   "metadata": {},
   "outputs": [],
   "source": [
    "fig.savefig('Adv_example_VIT.pdf')"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "torch",
   "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.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
