{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# White-box Attack on CIFAR10"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import sys\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torchattacks\n",
    "import numpy as np"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Load model and data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "[Data loaded]\n",
      "[Model loaded]\n",
      "Acc: 100.00 %\n"
     ]
    }
   ],
   "source": [
    "import robustbench\n",
    "from robustbench.data import load_cifar10\n",
    "from robustbench.utils import load_model, clean_accuracy\n",
    "\n",
    "images, labels = load_cifar10(n_examples=5, data_dir='provide_some_data_directory_here')\n",
    "print('[Data loaded]')\n",
    "\n",
    "device = \"cuda\"\n",
    "model = load_model('Standard', norm='Linf', model_dir='provide_some_data_directory_here').to(device)\n",
    "acc = clean_accuracy(model, images.to(device), labels.to(device))\n",
    "print('[Model loaded]')\n",
    "print('Acc: %2.2f %%'%(acc*100))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Sparse Adversarial Attack with group norms"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from torchattacks import PGD\n",
    "from utils import imshow, get_pred\n",
    "import custom_proj\n",
    "import importlib\n",
    "\n",
    "\n",
    "importlib.reload(custom_proj)\n",
    "\n",
    "atk = custom_proj.PGDGroup(model, alpha=8/225, steps=500, random_start=False, k=50, D=np.sqrt(1/16.))\n",
    "print(atk)\n",
    "results = atk(images, labels)\n",
    "adv_images, delta, costs = results['adv_images'], results['delta'], results['costs']\n",
    "delta.shape\n",
    "delta.shape\n",
    "idx = 4\n",
    "imshow(delta[idx:idx+1])\n",
    "plt.figure()\n",
    "plt.plot(costs)\n",
    "idx = 3\n",
    "pre = get_pred(model, adv_images[idx:idx+1], device)\n",
    "plt.figure()\n",
    "imshow(adv_images[idx], title=\"True:%d, Pre:%d\"%(labels[idx], pre))\n",
    "pre = get_pred(model, adv_images, device)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Adversarial attack with a similar l2 norm, but enforced globally"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from torchattacks import PGD\n",
    "from utils import imshow, get_pred\n",
    "import custom_proj\n",
    "import importlib\n",
    "\n",
    "\n",
    "importlib.reload(custom_proj)\n",
    "\n",
    "atk = custom_proj.PGDvanillal2(model, alpha=8/225, steps=500, random_start=False, k=50, D=1.)\n",
    "print(atk)\n",
    "results = atk(images, labels)\n",
    "adv_images_global, delta_global, costs = results['adv_images'], results['delta'], results['costs']\n",
    "delta_global.shape\n",
    "delta_global.shape\n",
    "idx = 4\n",
    "imshow(delta_global[idx:idx+1])\n",
    "plt.figure()\n",
    "plt.plot(costs)\n",
    "idx = 3\n",
    "pre = get_pred(model, adv_images_global[idx:idx+1], device)\n",
    "plt.figure()\n",
    "imshow(adv_images_global[idx], title=\"True:%d, Pre:%d\"%(labels[idx], pre))\n",
    "pre = get_pred(model, adv_images_global, device)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# We now print the difference in l2 norms of all the images"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "max_pix = torch.max(torch.sqrt((delta_global[idx:idx+1][0]**2).sum(axis=0)).cpu())\n",
    "print(max_pix)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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AMByCGszwhoAYDmEtRnfWQMAYHF01gAAS/K17rgzCGsAgOV0Nqh9LeiZBgcAwOLorAEAlkNnbUZYA71cT73eN3wbYW3GNDgAABbnUVgXFBRozJgxcjgccjgciouL029/+1v384ZhKDs7WyEhIerXr58mT56sEydOdHnRAADfdu08684svsSjsB46dKhWr16to0eP6ujRo5o6dapmzpzpDuQXX3xRubm5Wr9+vY4cOaKgoCAlJibq/Pnz3VI8AMA3EdZmHoX1jBkz9N3vfld333237r77bv33f/+3br/9dh0+fFiGYSgvL08rVqzQrFmzFBkZqTfffFN///vftWXLlu6qHwAAn9fh76ybm5u1bds2Xbx4UXFxcaqsrJTL5VJSUpJ7jN1u16RJk1RaWtrmdhobG9XQ0GBaAAC9G521mcdhffz4cd1+++2y2+1KS0vTjh07NGrUKLlcLklSYGCgaXxgYKD7udbk5OTI6XS6l9DQUE9LAgD4GMLazOOwHjlypCoqKnT48GH98Ic/1Pz583Xy5En38zabzTTeMIwW674uKytL9fX17qW6utrTkgAAPoawNvP4PGt/f3/dddddkqSYmBgdOXJEL7/8sv7zP/9TkuRyuRQcHOweX1tb26Lb/jq73S673e5pGQAA9BqdPs/aMAw1NjYqPDxcQUFBKikpcT/X1NSkffv2KT4+vrNvAwDoReiszTzqrJ955hklJycrNDRU58+f17Zt27R37169//77stlsysjI0KpVqzRixAiNGDFCq1at0je+8Q3Nmzevu+oHAPggrmBm5lFY/+Uvf1FqaqpqamrkdDo1ZswYvf/++0pMTJQkPf3007p06ZLS09P1t7/9TePHj9eePXs0YMCAbikeAIDewKOwfv3112/4vM1mU3Z2trKzsztTEwCgl6OzNuNGHgAAyyGszbiRBwAAFkdnDQCwHDprM8IaAGA5hLUZ0+AAAFgcnTUAwHLorM3orAEAluOtK5jl5+crPDxcAQEBio6O1oEDB244ft++fYqOjlZAQIC+/e1v6xe/+EWLMdu3b9eoUaNkt9s1atQo7dixw+O6CGsAgOV4I6yLioqUkZGhFStWqLy8XBMmTFBycrKqqqpaHV9ZWanvfve7mjBhgsrLy/XMM8/oRz/6kbZv3+4ec+jQIaWkpCg1NVWffPKJUlNTNWfOHH388cce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",
      "text/plain": [
       "<Figure size 640x480 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.colors as mcolors\n",
    "\n",
    "cmap = plt.get_cmap('gray')\n",
    "plt.figure()\n",
    "im = plt.imshow(torch.sqrt((delta_global[idx:idx+1][0]**2).sum(axis=0)).cpu(), cmap=cmap, vmin=0., vmax=max_pix)\n",
    "for i in range(1, 4):\n",
    "    plt.axhline(i * 32 / 4 - 0.5, color='white', linewidth=1)\n",
    "# Add vertical lines\n",
    "for i in range(1, 4):\n",
    "    plt.axvline(i * 32 / 4 - 0.5, color='white', linewidth=1)\n",
    "ax = plt.gca()\n",
    "cbar = plt.colorbar(im, ax=ax, pad=0.03)  # Adjust the pad as needed for positioning"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 500x1500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.colors as mcolors\n",
    "\n",
    "cmap = plt.get_cmap('gray')\n",
    "\n",
    "norm = mcolors.LogNorm(vmin=1e-4, vmax=1, clip=True)\n",
    "plt.figure(figsize = (5, 15))\n",
    "im = plt.imshow(torch.sqrt((delta[idx:idx+1][0]**2).sum(axis=0)).cpu(), cmap=cmap, vmin=0., vmax=max_pix)\n",
    "for i in range(1, 4):\n",
    "    plt.axhline(i * 32 / 4 - 0.5, color='white', linewidth=1)\n",
    "\n",
    "# Add vertical lines\n",
    "for i in range(1, 4):\n",
    "    plt.axvline(i * 32 / 4 - 0.5, color='white', linewidth=1)\n",
    "\n",
    "cbar = plt.colorbar(im, ax=ax, pad=0.03)  # Adjust the pad as needed for positioning"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(0.2500, device='cuda:0')\n",
      "tensor(0.4711, device='cuda:0')\n"
     ]
    }
   ],
   "source": [
    "print(custom_proj.max_l2_norm_in_regions(delta[3]))\n",
    "print(custom_proj.max_l2_norm_in_regions(delta_global[3]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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mke6T34onFQ2oWRantCdzshrAcAomlAy3d9UWo4ew8efNOeei7IJ+RXpyL19urHTaflyusomFAzki2tjE5RPEgC0njiFIy2yO3YyLU8iH0lNSJOgrm5Crjxd2h8IAgBOtMXQcvS0YGxtE5Ivp7oJKR5WHEd3/3HC8nn7ej8X1yYSudP4z+IHE4iIyBo2ISIisoZNiIiIrGETIiIia9iEiIjIGjYhIiKyhk2IiIisYRMiIiJrRuzJqvnBQoSCQVFt0i8/AdWfkZ84CQD5+aXiWs/RjZ1WnNzq+nQnCRooEhPGTzhzXY1ET+517rryE9wAwFOc3OoZ3Um5juJ5lDHKs/0d+d1Dc/5uyN+//lwk4YNseQN+RbpGn/zkbQBwjOK5qG4VIpVSnDSdlC8jAPh98nl3d/Wcuf4YsY7cqSDqE2cVc9GKd3SIa41yHfYqOkC+/7i4NpWSz4NHQkREZI2qCTU0NOC2225DOBxGWVkZHnjgAXz00UcDah555BE4jjPgcscddwzppImIaGxQNaGmpiYsXboU27dvR2NjI9LpNGpra9HT0zOg7t5778Xx48ezl7feemtIJ01ERGOD6j2hd955Z8D/169fj7KyMuzatQt333139vZQKIRoNDo0MyQiojHrkt4T6uzsBACUlg58837Lli0oKyvD1KlT8eijj6Kt7fxJwYlEArFYbMCFiIiuDBfdhIwxWLFiBe68805Mnz49e3tdXR1efvllbN68Gc8//zx27NiBBQsWIJEY/FNADQ0NiEQi2UtVVdXFTomIiEaZi/6I9rJly7B37168//77A25fvHhx9t/Tp0/HrFmzUF1djTfffBOLFi06Z5yVK1dixYoV2f/HYjE2IiKiK8RFNaEnnngCb7zxBrZu3YpJkyZdsLaiogLV1dU4ePDgoD8PhUIIheTfkEhERGOHqgkZY/DEE0/gtddew5YtW1BTU5Pzd9rb29HS0oKKioqLniQREY1NqveEli5dip///OfYsGEDwuEwWltb0drair4zZ2h3d3fjhz/8IX7729/i8OHD2LJlC+677z5MmDABDz744LAsABERjV6qI6F169YBAObPnz/g9vXr1+ORRx6Bz+fDvn378NJLL6GjowMVFRW455578MorryAclkfrEBHRlUH9ctyF5OfnY9OmTZc0obO8VAKe44lqO2Pn/wj4lxX581Xz0MSNpZUfNkwq8pXiyW7V2KlUT+6iMwoK+z8IcrrjID4/eShnvTFJ1VwSybi41kvK8gLPSmVk+wgApFOKgDcAjiPPyPNy3DcGcK8FAHz88Q788Y8fi37FGHkeXF+vbl9Jp+UZhp6Rr28A8DLy9WKgGxuQb8++nhsALMOBfe/gwIH9Oeu12XGaSD3tOvQn5fe3VEKXHdc1cZy4dvK1EXFt2lPkYooriYiIhhibEBERWcMmRERE1rAJERGRNWxCRERkDZsQERFZwyZERETWsAkREZE1bEJERGQNmxAREVlz0d8nNNw++WQnAn5Zj9x7RB4LEw7mqeYRMPIYjLQqvAMA5F9hkcnoonIynrw+EJwJAGg58gd8fPD3Oes9b/AvKDyfVFqxDnWLCSjiVfzC/Sk7tCOPnPH55Nu+s7N/3NMdH+Pzk/tk4/sD4vE9RZQRAMTjilgl5diOK18vriNfRgBwXXnEUyKeyl7H+3LvZEYZraNJbdI+SiShuP+UFanGrpkzS1w7vlg+bn980O9EtTwSIiIia9iEiIjIGjYhIiKyhk2IiIisYRMiIiJr2ISIiMgaNiEiIrKGTYiIiKxhEyIiImvYhIiIyBo2ISIismbEZsf5TB58xieqDSlinhxPl09lPHkolOsqe7qbEZd6igw7AAj45ZvW7way135BHldKNxW4Rr7OXUe5SzryJC4vI1/fAODKdj8AQEYxtuf1L6PjhOA4sizDjCIjL6HcVzy/PMPQ8+ky1aDIVDPK+0/GaLZnJnttkM5Z7TiKjQ/ABORzTwV0Y5fUVIlrr51zs2rsgJMvrv18z2/FtemUfNvwSIiIiKxhEyIiImvYhIiIyBo2ISIisoZNiIiIrGETIiIia9iEiIjIGjYhIiKyhk2IiIisYRMiIiJrRmxsTzqVAIysR6YTPeJxHZ8sJuWsVCohL87oIoFcRbmniigBXCPPS+lLxrPXPfG+nPUZbSxMWh71ksnonhflheSxI4rkm/65KJbT55PHB5196ue4gDQdJp1MysdXrG8AcDx5fUgRB9U/uHy9OOpIIPk6yfOb7HVhIPd9I6WI6wKAZFj+uDLxlhtVY19zwxRxbW9Li2rs3+96T1xbmOwU12Yy8m3JIyEiIrKGTYiIiKxhEyIiImvYhIiIyBo2ISIisoZNiIiIrGETIiIia9iEiIjIGjYhIiKyhk2IiIisYRMiIiJrRmx2HPwQz84flOdT+YK6TCgnoOjTnrKne8LgMADaTaWZieP6s9euL/ffcTxFThqAQIF87mWlZaqxXcjnkk7rMu8y3vBkx00oLgEAlI8rRdfEctHvpFLy/dZRrBMA8Dx5LmE6k1aN3dEhzxvzlGN7/qC4tsvvZK9PB3Kvn0DlRNVcpt58s7h2YlmlauxDu/aKa1v3HlCNHVDs4wWKx9lMRl7LIyEiIrJG1YTWrVuHGTNmoLi4GMXFxZgzZw7efvvt7M+NMaivr0dlZSXy8/Mxf/587N+/f8gnTUREY4OqCU2aNAnPPvssdu7ciZ07d2LBggW4//77s43mueeew+rVq7F27Vrs2LED0WgUCxcuRFdX17BMnoiIRjdVE7rvvvvwzW9+E1OnTsXUqVPxd3/3dygqKsL27dthjMGaNWvw9NNPY9GiRZg+fTpefPFF9Pb2YsOGDcM1fyIiGsUu+j2hTCaDjRs3oqenB3PmzEFzczNaW1tRW1ubrQmFQpg3bx62bdt23nESiQRisdiACxERXRnUTWjfvn0oKipCKBTCY489htdeew3Tpk1Da2srAKC8fOCnfcrLy7M/G0xDQwMikUj2UlVVpZ0SERGNUuomdN1112HPnj3Yvn07vv/972PJkiU4cOC/PxbofOnrfI0x59z2RStXrkRnZ2f20qL8eloiIhq91OcJBYNBXHvttQCAWbNmYceOHfjRj36Ev/qrvwIAtLa2oqKiIlvf1tZ2ztHRF4VCIYRCIe00iIhoDLjk84SMMUgkEqipqUE0GkVjY2P2Z8lkEk1NTZg7d+6l/hkiIhqDVEdCTz31FOrq6lBVVYWuri5s3LgRW7ZswTvvvAPHcbB8+XKsWrUKU6ZMwZQpU7Bq1SoUFBTgoYceGq75ExHRKKZqQidOnMDDDz+M48ePIxKJYMaMGXjnnXewcOFCAMCTTz6Jvr4+PP744zh9+jRmz56Nd999F+FwWD0xN+XCZ4SxNgl57EgGfap5GCTFtX7IY0QAwAf5y5DuBd5XG4wmcsafTGWvA4ncy+s6nm4uKfk67Oo6rRrb8Wmij+T7CQAYT76cGSOPnOmOdwMAOrs70d7ZLvodV/GiheNo1gl0r4foNj3SivsPXF2kVsYvX85wxfjsdUlvNGf9VTdPU83FVayY3b96VzV239Hzf7Dry/zK6CO/K1+HGSPfPppaVRP62c9+dsGfO46D+vp61NfXa4YlIqIrFLPjiIjIGjYhIiKyhk2IiIisYRMiIiJr2ISIiMgaNiEiIrKGTYiIiKxhEyIiImvYhIiIyBp1ivZwM2fiHtIZeQyGp4hXUSbOwEARVaHNNFHUG3Vsj3zsVCqNWCyGVCot+j2jje3RxH0otjsA6NaKbmxdbI9mfafOrO+UeHk1gTbKXUU3uHIX121PXWyPl5EvaCp5Zp0nU8ikc8c3pRPy2CtAF9sj+ftf5GkeC5X3n4yRr8OM4n5/drsbwX3fMZKqy+jo0aP8YjsiojGgpaUFkyZNumDNiGtCnufh2LFjCIfDA74MLxaLoaqqCi0tLSguLrY4w+HF5Rw7roRlBLicY81QLKcxBl1dXaisrITrXvhdnxH3cpzruhfsnMXFxWN6BziLyzl2XAnLCHA5x5pLXc5IJCKq4wcTiIjIGjYhIiKyZtQ0oVAohGeeeQahkPyL4EYjLufYcSUsI8DlHGsu93KOuA8mEBHRlWPUHAkREdHYwyZERETWsAkREZE1bEJERGTNqGlCL7zwAmpqapCXl4dbb70Vv/nNb2xPaUjV19fDcZwBl2g0antal2Tr1q247777UFlZCcdx8Prrrw/4uTEG9fX1qKysRH5+PubPn4/9+/fbmewlyLWcjzzyyDnb9o477rAz2YvU0NCA2267DeFwGGVlZXjggQfw0UcfDagZC9tTspxjYXuuW7cOM2bMyJ6QOmfOHLz99tvZn1/ObTkqmtArr7yC5cuX4+mnn8bu3btx1113oa6uDkeOHLE9tSF144034vjx49nLvn37bE/pkvT09GDmzJlYu3btoD9/7rnnsHr1aqxduxY7duxANBrFwoUL0dXVdZlnemlyLScA3HvvvQO27VtvvXUZZ3jpmpqasHTpUmzfvh2NjY1Ip9Oora1FT09PtmYsbE/JcgKjf3tOmjQJzz77LHbu3ImdO3diwYIFuP/++7ON5rJuSzMK3H777eaxxx4bcNv1119v/vqv/9rSjIbeM888Y2bOnGl7GsMGgHnttdey//c8z0SjUfPss89mb4vH4yYSiZgf//jHFmY4NL68nMYYs2TJEnP//fdbmc9waWtrMwBMU1OTMWbsbs8vL6cxY3N7GmPMuHHjzE9/+tPLvi1H/JFQMpnErl27UFtbO+D22tpabNu2zdKshsfBgwdRWVmJmpoafOc738GhQ4dsT2nYNDc3o7W1dcB2DYVCmDdv3pjbrgCwZcsWlJWVYerUqXj00UfR1tZme0qXpLOzEwBQWloKYOxuzy8v51ljaXtmMhls3LgRPT09mDNnzmXfliO+CZ08eRKZTAbl5eUDbi8vL0dra6ulWQ292bNn46WXXsKmTZvwk5/8BK2trZg7dy7a29ttT21YnN12Y327AkBdXR1efvllbN68Gc8//zx27NiBBQsWIJFI2J7aRTHGYMWKFbjzzjsxffp0AGNzew62nMDY2Z779u1DUVERQqEQHnvsMbz22muYNm3aZd+WIy5F+3ycL31TlzHmnNtGs7q6uuy/b7rpJsyZMwfXXHMNXnzxRaxYscLizIbXWN+uALB48eLsv6dPn45Zs2ahuroab775JhYtWmRxZhdn2bJl2Lt3L95///1zfjaWtuf5lnOsbM/rrrsOe/bsQUdHB37xi19gyZIlaGpqyv78cm3LEX8kNGHCBPh8vnM6cFtb2zmdeiwpLCzETTfdhIMHD9qeyrA4+8m/K227AkBFRQWqq6tH5bZ94okn8MYbb+C9994b8JUrY217nm85BzNat2cwGMS1116LWbNmoaGhATNnzsSPfvSjy74tR3wTCgaDuPXWW9HY2Djg9sbGRsydO9fSrIZfIpHAhx9+iIqKCttTGRY1NTWIRqMDtmsymURTU9OY3q4A0N7ejpaWllG1bY0xWLZsGV599VVs3rwZNTU1A34+VrZnruUczGjcnoMxxiCRSFz+bTnkH3UYBhs3bjSBQMD87Gc/MwcOHDDLly83hYWF5vDhw7anNmR+8IMfmC1btphDhw6Z7du3m29961smHA6P6mXs6uoyu3fvNrt37zYAzOrVq83u3bvNp59+aowx5tlnnzWRSMS8+uqrZt++fea73/2uqaioMLFYzPLMdS60nF1dXeYHP/iB2bZtm2lubjbvvfeemTNnjrnqqqtG1XJ+//vfN5FIxGzZssUcP348e+nt7c3WjIXtmWs5x8r2XLlypdm6datpbm42e/fuNU899ZRxXde8++67xpjLuy1HRRMyxph//Md/NNXV1SYYDJqvfvWrAz4yORYsXrzYVFRUmEAgYCorK82iRYvM/v37bU/rkrz33nsGwDmXJUuWGGP6P9b7zDPPmGg0akKhkLn77rvNvn377E76IlxoOXt7e01tba2ZOHGiCQQCZvLkyWbJkiXmyJEjtqetMtjyATDr16/P1oyF7ZlrOcfK9vyzP/uz7OPpxIkTzde//vVsAzLm8m5LfpUDERFZM+LfEyIiorGLTYiIiKxhEyIiImvYhIiIyBo2ISIisoZNiIiIrGETIiIia9iEiIjIGjYhIiKyhk2IiIisYRMiIiJr2ISIiMia/w/vRY/t/im7xwAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import torchvision\n",
    "\n",
    "image_data = images[idx].cpu()\n",
    "img = torchvision.utils.make_grid(image_data.data, normalize=True)\n",
    "npimg = img.numpy()\n",
    "fig = plt.figure()\n",
    "plt.imshow(np.transpose(npimg,(1,2,0)))\n",
    "# Add horizontal lines\n",
    "for i in range(1, 4):\n",
    "    plt.axhline(i * 32 / 4 - 0.5, color='white', linewidth=1)\n",
    "# Add vertical lines\n",
    "for i in range(1, 4):\n",
    "    plt.axvline(i * 32 / 4 - 0.5, color='white', linewidth=1)\n",
    "# Show the plot\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "interpreter": {
   "hash": "1ceb8aea646a0c712ed5db194d127de24ece80f87032283552cbe7de982c3798"
  },
  "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.11.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
