{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "env: CUDA_VISIBLE_DEVICES=7\n",
      "env: XLA_PYTHON_CLIENT_PREALLOCATE=false\n",
      "env: XLA_PYTHON_CLIENT_ALLOCATOR=platform\n"
     ]
    }
   ],
   "source": [
    "%env CUDA_VISIBLE_DEVICES=7\n",
    "%env XLA_PYTHON_CLIENT_PREALLOCATE = false\n",
    "%env XLA_PYTHON_CLIENT_ALLOCATOR=platform"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "True"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import dotenv\n",
    "dotenv.load_dotenv()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 168,
   "metadata": {},
   "outputs": [],
   "source": [
    "# TODO: All this stuff from cleverhans (modified); include MIT license!\n",
    "#  I use the ensemble version; one_hot not needed anymore; ensemble version is my modification\n",
    "\n",
    "import typing\n",
    "import jax.numpy as jnp\n",
    "from jax import grad, vmap\n",
    "from jax.nn import log_softmax as logsoftmax\n",
    "import jax\n",
    "\n",
    "def fast_gradient_method_ensemble(\n",
    "    model_fn, x, eps, norm, clip_min=None, clip_max=None, y=None, targeted=False, aggregation: typing.Literal[\"mean\", \"min\"] = \"min\",\n",
    "):\n",
    "    \"\"\"\n",
    "    JAX implementation of the Fast Gradient Method.\n",
    "    :param model_fn: a callable that takes an input tensor and returns the model logits.\n",
    "    :param x: input tensor.\n",
    "    :param eps: epsilon (input variation parameter); see https://arxiv.org/abs/1412.6572.\n",
    "    :param norm: Order of the norm (mimics NumPy). Possible values: np.inf or 2.\n",
    "    :param clip_min: (optional) float. Minimum float value for adversarial example components.\n",
    "    :param clip_max: (optional) float. Maximum float value for adversarial example components.\n",
    "    :param y: (optional) Tensor with one-hot true labels. If targeted is true, then provide the\n",
    "              target one-hot label. Otherwise, only provide this parameter if you'd like to use true\n",
    "              labels when crafting adversarial samples. Otherwise, model predictions are used\n",
    "              as labels to avoid the \"label leaking\" effect (explained in this paper:\n",
    "              https://arxiv.org/abs/1611.01236). Default is None. This argument does not have\n",
    "              to be a binary one-hot label (e.g., [0, 1, 0, 0]), it can be floating points values\n",
    "              that sum up to 1 (e.g., [0.05, 0.85, 0.05, 0.05]).\n",
    "    :param targeted: (optional) bool. Is the attack targeted or untargeted?\n",
    "              Untargeted, the default, will try to make the label incorrect.\n",
    "              Targeted will instead try to move in the direction of being more like y.\n",
    "    :param aggregation: (optional) str. How to aggregate the losses across models.\n",
    "    :return: a tensor for the adversarial example\n",
    "    \"\"\"\n",
    "    if norm not in [jnp.inf, 2]:\n",
    "        raise ValueError(\"Norm order must be either np.inf or 2.\")\n",
    "\n",
    "    if y is None:\n",
    "        # Using model predictions as ground truth to avoid label leaking\n",
    "        x_labels = jnp.argmax(model_fn(x), -1)\n",
    "        y = jax.nn.one_hot(x_labels, 10)\n",
    "\n",
    "    def loss_adv(image, label):\n",
    "        pred = model_fn(image[None])[:, 0]\n",
    "        per_model_losses = -jnp.sum(logsoftmax(pred) * label, axis=-1)\n",
    "        if targeted:\n",
    "            per_model_losses = -per_model_losses\n",
    "\n",
    "        if aggregation == \"mean\":\n",
    "            return jnp.mean(per_model_losses)\n",
    "        elif aggregation == \"min\":\n",
    "            return jnp.min(per_model_losses)\n",
    "        else:\n",
    "            raise ValueError(f\"Invalid aggregation method: {aggregation}\")\n",
    "\n",
    "    grads_fn = vmap(grad(loss_adv), in_axes=(0, 1), out_axes=0)\n",
    "    grads = grads_fn(x, y)\n",
    "\n",
    "    axis = list(range(1, len(grads.shape)))\n",
    "    avoid_zero_div = 1e-12\n",
    "    if norm == jnp.inf:\n",
    "        perturbation = eps * jnp.sign(grads)\n",
    "    elif norm == 1:\n",
    "        raise NotImplementedError(\"L_1 norm has not been implemented yet.\")\n",
    "    elif norm == 2:\n",
    "        square = jnp.maximum(\n",
    "            avoid_zero_div, jnp.sum(jnp.square(grads), axis=axis, keepdims=True)\n",
    "        )\n",
    "        perturbation = grads / jnp.sqrt(square)\n",
    "\n",
    "    adv_x = x + perturbation\n",
    "\n",
    "    # If clipping is needed, reset all values outside of [clip_min, clip_max]\n",
    "    if (clip_min is not None) or (clip_max is not None):\n",
    "        # We don't currently support one-sided clipping\n",
    "        assert clip_min is not None and clip_max is not None\n",
    "        adv_x = jnp.clip(adv_x, a_min=clip_min, a_max=clip_max)\n",
    "\n",
    "    return adv_x\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 169,
   "metadata": {},
   "outputs": [],
   "source": [
    "import settings\n",
    "import data\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 170,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "os.chdir(os.pardir)\n",
    "import architectures\n",
    "import jax_mnist\n",
    "os.chdir(\"eval\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 178,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Loading raw data\n"
     ]
    }
   ],
   "source": [
    "# Config\n",
    "num_canaries = 1\n",
    "global_seed = 0xB0BA73A\n",
    "sample_non_canaries = True\n",
    "# Victim model training config\n",
    "learning_rate = 0.1\n",
    "momentum = 0.9\n",
    "num_epochs = 10\n",
    "batch_size = 128\n",
    "standardize = True\n",
    "architecture = architectures.MLP(width=20)\n",
    "# Adversarial attack config\n",
    "attack_eps = 0.3\n",
    "attack_norm = jnp.inf\n",
    "attack_num_models = 8  # do LiRA-style training set splits for attack models, and then take intersection of adversarial examples\n",
    "attack_aggregation = \"min\"\n",
    "\n",
    "# Baseline generator setup\n",
    "dataset_loader = settings.data.MNISTLoader()\n",
    "\n",
    "# Load raw data\n",
    "print(\"Loading raw data\")\n",
    "dataset_loader.prepare_raw_data()\n",
    "train_images_full, train_targets_full = dataset_loader.load_train_data()\n",
    "test_images, test_targets = dataset_loader.load_val_data()\n",
    "canary_indices, _ = data.select_canary_indices(\n",
    "    num_canaries=num_canaries,\n",
    "    num_samples=train_images_full.shape[0],\n",
    "    global_seed=global_seed,\n",
    ")\n",
    "\n",
    "# Method args\n",
    "num_canaries = num_canaries\n",
    "image_shape = (1, 28, 28)\n",
    "num_classes = 10\n",
    "replaced_images = train_images_full[canary_indices]\n",
    "replaced_targets = train_targets_full[canary_indices]\n",
    "global_seed = global_seed"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 179,
   "metadata": {},
   "outputs": [],
   "source": [
    "import jax"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 180,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Train a model\n",
    "key = jax.random.PRNGKey(global_seed)\n",
    "\n",
    "# Prepare images\n",
    "\n",
    "images_mean_std = dataset_loader.dataset_mean_std\n",
    "dataset_mean = jnp.array(images_mean_std[0], dtype=jnp.float32)\n",
    "dataset_std = jnp.array(images_mean_std[1], dtype=jnp.float32)\n",
    "\n",
    "def standardize_images(images):\n",
    "    return (images - dataset_mean.reshape(1, 1, 1, -1)) / dataset_std.reshape(1, 1, 1, -1)\n",
    "\n",
    "def prepare_data(images, targets, images_mean_std, standardize=True):\n",
    "    # Move training data to jax\n",
    "    # Code expects (N, H, W, C) in [0, 1], but images are torch (N, C, H, W)\n",
    "    images = jnp.array(images.permute(0, 2, 3, 1))\n",
    "    if targets is not None:\n",
    "        targets = jnp.array(targets, dtype=jnp.int16)\n",
    "    else:\n",
    "        targets = None\n",
    "\n",
    "    # Standardize with given statistics\n",
    "    if standardize:\n",
    "        images = standardize_images(images)\n",
    "\n",
    "    return images, targets\n",
    "\n",
    "images_train, targets_train = prepare_data(train_images_full, train_targets_full, images_mean_std, standardize=standardize)\n",
    "\n",
    "# Initialize models\n",
    "key, *keys_init = jax.random.split(key, attack_num_models + 1)\n",
    "image_shape_hwc = images_train.shape[1:]\n",
    "init_state = jax_mnist.create_train_state(\n",
    "    keys_init,\n",
    "    learning_rate=learning_rate,\n",
    "    momentum=momentum,\n",
    "    num_models=attack_num_models,\n",
    "    arch=architecture,\n",
    "    image_shape=image_shape_hwc,\n",
    "    dp_params=None,\n",
    ")\n",
    "del keys_init\n",
    "\n",
    "# Build epoch permutations\n",
    "key, key_perms = jax.random.split(key)\n",
    "epoch_perms = jax_mnist.generate_model_perms(\n",
    "    key=key_perms,\n",
    "    num_models=attack_num_models,\n",
    "    num_epochs=num_epochs,\n",
    "    num_samples=images_train.shape[0],\n",
    "    batch_size=batch_size,\n",
    "    sample_non_canaries=sample_non_canaries,\n",
    "    canary_idx=None,\n",
    ")\n",
    "del key_perms\n",
    "\n",
    "# Train\n",
    "trained_state, _, _, _ = jax_mnist.train_model(\n",
    "    state=init_state,\n",
    "    epoch_perms=epoch_perms,\n",
    "    train_images=images_train,\n",
    "    train_targets=targets_train,\n",
    "    test_images=None,\n",
    "    test_targets=None,\n",
    "    use_dp=False,\n",
    "    verbose=False,\n",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 181,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Generate untargeted adversarial examples for all test images\n",
    "# All inputs to the adversarial attack are unstandardized to no mess up eps; things will get standardized on the fly\n",
    "images_test, targets_test = prepare_data(test_images, test_targets, images_mean_std, standardize=False)\n",
    "\n",
    "def model_fn(images):\n",
    "    images = standardize_images(images)\n",
    "    return jax_mnist.flatten_over_models(jax_mnist.get_logits(trained_state, images))\n",
    "\n",
    "\n",
    "# Find adversarial examples for every model x every test image (untargeted)\n",
    "images_adversarial = fast_gradient_method_ensemble(\n",
    "    model_fn,\n",
    "    x=images_test,\n",
    "    eps=attack_eps,\n",
    "    norm=attack_norm,\n",
    "    clip_min=0.0,\n",
    "    clip_max=1.0,\n",
    "    aggregation=attack_aggregation,\n",
    ")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 182,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy original: 0.9332000017166138\n",
      "Test accuracy adversarial: 0.3762750029563904\n",
      "Found 4089 test images that are adversarial for all models\n"
     ]
    }
   ],
   "source": [
    "# For every model x sample pair, determine if the adversarial example has a flipped label\n",
    "logits_original = jax_mnist.flatten_over_models(jax_mnist.get_logits(trained_state, standardize_images(images_test)))\n",
    "preds_original = jnp.argmax(logits_original, axis=-1)\n",
    "\n",
    "logits_adversarial = jax_mnist.flatten_over_models(jax_mnist.get_logits(trained_state, standardize_images(images_adversarial)))\n",
    "assert logits_adversarial.shape == logits_original.shape\n",
    "preds_adversarial = jnp.argmax(logits_adversarial, axis=-1)\n",
    "assert preds_adversarial.shape == preds_original.shape\n",
    "\n",
    "test_accuracy_original = jnp.mean(preds_original == targets_test)\n",
    "test_accuracy_adversarial = jnp.mean(preds_adversarial == targets_test)\n",
    "\n",
    "print(f\"Test accuracy original: {test_accuracy_original}\")\n",
    "print(f\"Test accuracy adversarial: {test_accuracy_adversarial}\")\n",
    "\n",
    "preds_mismatch = (preds_adversarial != preds_original)\n",
    "intersection_indices = jnp.argwhere(jnp.all(preds_mismatch, axis=0))[:, 0]\n",
    "\n",
    "print(f\"Found {len(intersection_indices)} test images that are adversarial for all models\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 183,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import numpy as np\n",
    "\n",
    "# Convert images back to expected format\n",
    "selected_images = images_adversarial[intersection_indices]\n",
    "# No need to unstandardize, because we never standardized the adversarial images\n",
    "\n",
    "# Convert to torch float32 and NHWC to NCHW\n",
    "selected_images = torch.from_numpy(np.array(selected_images)).to(torch.float32)\n",
    "selected_images = selected_images.permute((0, 3, 1, 2))\n",
    "assert selected_images.shape == (len(intersection_indices), *train_images_full.shape[1:])\n",
    "assert selected_images.min() >= 0 and selected_images.max() <= 1\n",
    "assert isinstance(selected_images, torch.Tensor)\n",
    "\n",
    "# Use original clean labels as targets for the canary\n",
    "selected_targets = test_targets[np.array(intersection_indices)]\n",
    "assert isinstance(selected_targets, torch.Tensor)\n",
    "assert selected_targets.shape == (len(intersection_indices),)\n",
    "assert selected_targets.min() >= 0 and selected_targets.max() < num_classes\n",
    "assert selected_targets.dtype == torch.int64\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 184,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[8 8 8 3 8 3 8 8]\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAiIAAAGzCAYAAAASZnxRAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjEsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvc2/+5QAAAAlwSFlzAAAPYQAAD2EBqD+naQAAPExJREFUeJzt3Xl0FFXexvGnQ0iTtSEsWQRCAsi+KAgisgkSUBAERxZ1CCqoE0RARVEUYVRcUBkZQBEHFAEFh+XAq8wwrC6AMwgi44AQybDvkkDYArnvH7zplyZrt6RvSL6fc/ocqK5b9etKVefJrdt9HcYYIwAAAAsCbBcAAABKL4IIAACwhiACAACsIYgAAABrCCIAAMAagggAALCGIAIAAKwhiAAAAGsIIgAAwBqrQeSll16Sw+Hwqe3MmTPlcDiUmpp6dYu6TGpqqhwOh2bOnOlT+/bt28vhcMjhcKhbt25XtzgUa4sWLXL/7B0Oh/71r39ZqeO3nsPXivbt26t9+/Y+tXU4HHrppZd8apuUlOT+GTds2NCnbeDaVb58effPf8iQIbbLuWb5FET+/e9/6/7779d1110np9Op2NhY3Xffffr3v/99teu75tWtW1ezZs3SU089lec6KSkpKleunM+/sGbPni2Hw6GwsDCfahw0aJDPYSkzM1P169eXw+HQhAkTCtXm1KlTGjZsmKpWrSqn06l69epp6tSpXu9bKr6vvXnz5po1a5YGDx7sU11X+s9//iOHw6Fy5crpxIkTV2WbuDoqVaqkWbNm6bXXXvNYnpWVpffee09NmzZVWFiYoqKi1LVrV3377beF2m5aWppGjhyp2rVrKzg4WHFxcXrooYe0e/fuQrXfsWOH+vbtq6pVqyokJER169bVuHHjdPr06QLbLly4UImJiYqNjZXT6VTVqlV1zz33aOvWrYXatyTNmzdPN998s8qXL6+KFSuqXbt2+p//+Z9CtR0+fLhuvPFGRUZGKiQkRPXq1dNLL72kU6dOFaq9v47dtGnTNGvWrEJt80rZf0wX9KhRo4ZP2/e3V155xfdAbrz017/+1QQFBZno6Gjz/PPPm+nTp5vRo0ebmJgYExQUZBYsWFDobWVmZpozZ854W4IxxpgLFy6YM2fOmKysLJ/aF8auXbuMJDNjxgyf2rdr1860a9euwPW6d+9uQkNDjSTzz3/+06t9nDx50sTGxprQ0FATGhrqdY3//Oc/TWBgoClXrpy58847vW7/1ltvuWt/8803C1z/woUL5pZbbjFBQUFm+PDhZsqUKaZHjx5GknnllVe82ve18NpnzJjh08/1Ss8995yJjo42TqfTfPDBB4Vu91vP4WvFuXPnzLlz53xqK8mMGTPGp7YDBgwwcXFxuT43YsQII8ncf//95v333zevv/66SUhIMIGBgWbDhg35bvfixYvmpptuMqGhoebpp582H3zwgXnmmWdMeHi4ue6660x6enq+7Xfv3m3Kly9v4uLizPjx4837779vkpKSjCRz1113Ffi6xo4da/r06WNee+01M336dPPyyy+bhIQEExwcbDZv3lxg+3fffddIMnfeeaeZOnWqeeedd0yTJk2MJPPXv/61wPatW7c2Q4cONe+++66ZNm2aeeyxx4zT6TStW7c2Fy9ezLetjWMnySQnJxf4ui6XkpJiZs2a5fFwOp2mTZs2HssWLlzo1XZt2LNnjwkJCTGhoaGmQYMGXrf3Kojs3LnThISEmLp165rDhw97PHfkyBFTt25dExoaalJSUvLdzqlTp7wu1AZ/BJFly5aZoKAgM3r0aJ9+YT3zzDOmTp065r777vP6l3FWVpZp1aqVefDBB01cXJzXv4wPHTpkXC6XGTduXKGDyLx584wk8+GHH3os7927tylXrpw5dOhQofd/Lbz2qxFEsrKyTI0aNcyIESPM3Xffbdq3b1/otraCiL+u8YyMjN+8jaIIIpmZmSY4ONjcc889Hst/+eUXI8kMHTo03+1+8803RpL585//7LH8L3/5i5FU4B98r7zyipFktm7d6rH897//vZFkjh8/nm/73Bw8eNAEBgaaRx55pMB1a9eubW666SaPPxTT0tJMWFhYoYJQbiZMmGAkmXXr1uW7no1j50sQyU1oaKgZMGBAvutkZmb6HLqLSp8+fcxtt91m2rVr51MQ8erWzJtvvqnTp09r2rRpqly5ssdzlSpV0vvvv6+MjAy98cYb7uXZ40B++ukn9e/fXxUqVNCtt97q8dzlzpw5o6FDh6pSpUoKDw/XXXfdpX379uW4j5vbGJEaNWqoW7du+vrrr9WiRQuVK1dOCQkJ+vjjjz32cfz4cT311FNq1KiRwsLCFBERoa5du+qHH34o8BhkZmZq27ZtOnDgQGEPW77beuKJJ/TEE0+oZs2aXrffsWOH3nnnHb399tsKDAz0uv2sWbO0detWvfLKK163laRnn31WderU0f3331/oNl999ZUkqW/fvh7L+/btq7Nnz2rx4sWF2s61+Np99c033yg1NVV9+/ZV3759tXbtWu3duzfHeidOnFBSUpJcLpfKly+vAQMG5LiNM2HCBDkcDv33v//N0X7UqFEKCgrSr7/+6l62YcMGdenSRS6XSyEhIWrXrp2++eYbj3b5XeMHDx7UwIED3bfhYmJi1KNHD4/rdvHixbrzzjvdtwFq1qypP/7xj7p48aLHftq3b6+GDRtq48aNatu2rUJCQvTcc8+5n7t8jMj58+f14osvqlmzZnK5XAoNDVWbNm20atWqQh3zbdu2FbobPzeZmZk6c+aMoqKiPJZXqVJFAQEBCg4Ozrd9enq6JOVoHxMTI0m/qX1AQICCgoIKfhFXqFKlikJCQgp1azA9PV1VqlTxeH+PiIhQWFhYgbXnJfsWRUH7L47HzlfZY7wmTJigiRMnqmbNmnI6nfrpp5/yHCe5evVqORwOrV692mN5Ya5lyftzf+3atfr88881ceJEH17hJV4FkSVLlqhGjRpq06ZNrs+3bdtWNWrUyPU+4O9+9zudPn1ar776qgYNGpTnPpKSkjRp0iTdcccdev311xUcHKw777yz0DXu3LlT99xzj26//Xa99dZbqlChgpKSkjzGr/zyyy9atGiRunXrprfffltPP/20fvzxR7Vr10779+/Pd/v79u1TvXr1NGrUqELXlJeJEyfq119/1ejRo31qP2zYMHXo0EF33HGH121PnjypZ555Rs8995yio6O9bv/dd9/po48+0sSJE70acHzu3DmVKVMmx8UcEhIiSdq4cWOhtnMtvnZfzZ49WzVr1tRNN92k7t27KyQkRHPnzvVYxxijHj16aNasWbr//vv18ssva+/evRowYIDHevfee68cDofmzZuXYz/z5s1T586dVaFCBUnSypUr1bZtW6Wnp2vMmDF69dVXdeLECd1222367rvvcrTP7Rrv3bu3Fi5cqIEDB2rKlCkaOnSoTp486fFGN3PmTIWFhWnEiBH605/+pGbNmunFF1/Us88+m2Mfx44dU9euXdW0aVNNnDhRHTp0yPWYpaena/r06Wrfvr1ef/11vfTSSzpy5IgSExO1efPm/A+4pHr16un3v/99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ydG7eezYMRMZGWnuvfder/dtjDGzZ882AQEBBX5qx8axUxH3iFzeU5+tsD0i3lzL3nj00UdNrVq1PHoei7xHxOVyKSYmRlu2bMl3vS1btui6665TRESEx3JfByh5q0yZMrkuN8a4//3qq69qxIgRatu2rT755BP97W9/0/Lly9WgQYMcf+UVhZEjR6pNmzaKj49XamqqUlNT3X8hHThwIN+BQmlpaXr55Zc1aNAgpaenu9ufOnVKxhilpqbq8OHDebZfuXKlli1bpieeeMLdNjU1VRcuXNCZM2eUmpqaozfrchMmTND58+fVp08fd9vsgZO//vqrUlNTdf78+Xxff9u2bfXLL79o06ZN+vrrr7Vv3z7dfPPNkqTrr7++RL92b6Snp2vJkiXatWuXateu7X7Ur19fp0+f1pw5czzO68KKjY1VmzZt3ONE1q9fr927d6tPnz7udbKvgzfffFPLly/P9XHld7fkdo0PGzZMP//8s8aPH69y5crphRdeUL169bRp0yZJl8YgtGvXTj/88IPGjRunJUuWaPny5e4epyuvx8K+j3zyySdKSkpSzZo19eGHH2rZsmVavny5brvtNr9c42vXrtXWrVt11113eSyvXbu26tWrl+sgwcvNnDlTZ8+ezfH9NtnbK6j9lClTdMMNN6hq1ao52p8+fdp9/L1RoUIF3XbbbZo9e3a+6/3yyy9atmxZjtceGRmpW2+9tcDa89KrVy9J0qeffprvesXx2P1WuZ33eY1Pu7LHypdruSA7duzQtGnTNHToUO3fv9/9fnj27FllZmYqNTVVx48fL/T2vPq4Qbdu3fTBBx/o66+/do+Kv9xXX32l1NRUPfLII95s1i0uLk5ZWVnuN95she3KK6zPP/9cHTp00Icffuix/MSJE6pUqdJV3Vdudu/erf/+97+Kj4/P8dxdd90ll8uV58jwX3/9VadOndIbb7zh8emkbPHx8erRo4cWLVqU576l/7+oL7dv3z7Fx8frnXfe0bBhw/Js/+uvv6pBgwY5nnv11Vf16quvatOmTWratGmu7bOVKVPGY51//OMfkpTvQLKS8toLa8GCBTp79qymTp2a47zcvn27Ro8erW+++Ua33nqr4uLitGLFCp06dcrjTWX79u25brtPnz76wx/+oO3bt+uzzz5TSEiIunfv7n4++1NcERERBQ7uK0jNmjX15JNP6sknn9SOHTvUtGlTvfXWW/rkk0+0evVqHTt2TAsWLFDbtm3dbXbt2vWb9vn5558rISFBCxYs8HjDHjNmzG/abmEdOnRIUs5fCtKlT9RcuHChwPbGmBztMzMzJalQ7bMHHfvSPi9nzpxRWlpagfuWfH/teTl37pyysrIKtf/ieOyutuwar/xdceUn4q7mtZxt3759ysrK0tChQzV06NAcz8fHx+uJJ54o9CdpvAoiTz/9tD755BM98sgjWrt2rSpWrOh+7vjx43r00UcVEhKip59+2pvNuiUmJur555/XlClT9M4777iXT5o0yaft5aVMmTI5/pKcP3++9u3bp1q1auXbNjMzUykpKe4eIl9MmzYtxzf0rVy5UpMmTdKECRNUt27dPNtWqVJFCxcuzLH83Xff1bp16zR37tx867rttttybT948GDFxcXp+eefz3f089ChQ9WzZ0+PZYcPH9YjjzyipKQk9ejRI9eAlZ8jR47o9ddfV+PGjfO9UEria8/PJ598ooSEhFzHV507d06vvfaaZs+erVtvvVV33HGHpk2bpqlTp7qvv4sXL+Z57fTu3VuPP/645s6dq/nz56tbt24enwZo1qyZatasqQkTJqh///45/mI6cuRIgfeFT58+rYCAAJUrV869rGbNmgoPD9e5c+ck/X8P5uXX4/nz591jhnx1+Xazg8iGDRu0bt06Va9evcD227ZtU0hISKHWzU12z96nn36qLl26uJd///332r59e4Gfmrn++utljNG8efOUlJTkXp79aakbbrihwPZ///vf9fPPP3v0Ms6dO1cBAQFq3Lhxvu0PHz6sKlWqeCxLTU3VihUr1Lx583zb1qpVSwEBAfrss8/0yCOPuI//3r179dVXX+X6R+zlTpw4odDQUJUtW9Zj+fTp0yWpwP3bPnb+kh0w1q5d6/7j5+LFi5o2bZrHet5ey4U59xs2bJjre+no0aN18uRJ/elPf/LqKym8CiK1a9fWRx99pPvuu0+NGjXSQw895L698OGHH+ro0aOaO3euT9+JIV06YL1799bEiRN17Ngx3XzzzVqzZo1+/vlnSXl3RXmrW7duGjdunAYOHKhbbrlFP/74o2bPnq2EhIQC22Z/fHfAgAE+D1jt3LlzjmXZqbZdu3b5XmghISE5fhlKl+Y2+e6773J97nLVq1fP9QQbNmyYoqKiCmx/44036sYbb/RYlv059gYNGhTYXrr0Glu1aqVatWrp4MGDmjZtmk6dOqWlS5e6BxHnpiS8dulS1/HAgQM1Y8YMjzfKy+3fv1+rVq3K9a8N6dKgzcTERM2fP1/vvvuuunfvrtatW+vZZ59Vamqq6tevrwULFuT512OVKlXUoUMHvf322zp58qTHbRlJCggI0PTp09W1a1c1aNBAAwcO1HXXXad9+/Zp1apVioiI0JIlS/J9nT///LM6duyoe++9V/Xr11dgYKAWLlyoQ4cOub9H5pZbblGFChU0YMAADR06VA6HQ7NmzfLpltPlunXrpgULFujuu+/WnXfeqV27dum9995T/fr1CzXgsV69emrXrp3PA1abNWum22+/XR999JHS09PVuXNnHThwQJMmTVJwcHCevW7ZkpKSNGHCBD3yyCPatGmTGjRooO+//17Tp09XgwYNdPfdd+fb/umnn9aXX36pNm3aaMiQIapYsaKWLl2qL7/8Ug8//LBiY2Pzbd+oUSN17NhRTZs2VYUKFbRjxw59+OGHyszMzPFV9leqXLmyHnzwQU2fPl0dO3ZUr169dPLkSU2ZMkVnzpwp8KsPVq9eraFDh+qee+5R7dq1df78eX311VdasGCBmjdvXuB399g+dtkcDsdvOocK0qBBA918880aNWqUjh8/rsjISH366ac5emy8vZYLc+5XqlQp1/e77B6Qwr4Xunk9qsQYs2XLFtOvXz8TExNjypYta6Kjo02/fv3Mjz/+mGPd7IFsR44cyfO5y2VkZJjk5GQTGRlpwsLCTM+ePc327duNJPPaa6+518trsGpugw6vHMh29uxZ8+STT5qYmBgTHBxsWrdubdatW5djvaL8+O6VfuugRm8/wnolX75dNJu3g1WHDx9uEhISjNPpNJUrVzb9+/cv8Nt481OcX3tuP9dJkyYZSWbZsmV5bjf7Y84rVqzIc53swXeLFy82xlwaDPjAAw+YiIgI43K5zAMPPGA2bdqU54DrDz74wEgy4eHheU61sGnTJtOrVy9TsWJF43Q6TVxcnLn33ns96srrGj969KhJTk52f+Oyy+UyLVu2NPPmzfNY75tvvjE333yzCQ4ONrGxsWbkyJHmb3/7m8eAO2PyHwh35bWWlZVlXn31VRMXF2ecTqe54YYbzNKlS3MdXKoi+vju6dOnzbhx40z9+vVN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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "test_idx = 18\n",
    "\n",
    "current_mispredictions = preds_adversarial[:, intersection_indices[test_idx]]\n",
    "current_original_predictions = preds_original[:, intersection_indices[test_idx]]\n",
    "current_label = selected_targets[test_idx]\n",
    "print(current_mispredictions)\n",
    "plt.imshow(selected_images[test_idx].permute(1, 2, 0))\n",
    "plt.title(f\"Original: {current_original_predictions}, Adversarial: {current_mispredictions}, True: {current_label}\")\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 80,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(8, 10000)"
      ]
     },
     "execution_count": 80,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "preds_original.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": ".venv",
   "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.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
