{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "f942bd1f",
   "metadata": {},
   "outputs": [],
   "source": [
    "import h5py\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "0bc7d408",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "<KeysViewHDF5 ['attack_successful', 'attacked_spiketrains', 'attacked_spiketrains_random', 'ground_truth', 'n_spikes_attk', 'n_spikes_orig', 'original_spiketrains', 'random_patch_successful_rate', 'sinabs_label', 'targeted_patch_successful_rate']>\n"
     ]
    }
   ],
   "source": [
    "with h5py.File(\"../Results/attacks_patches_ep10_lb4_num50_patchsize025.h5\", \"r\") as F:\n",
    "    idx = F['attack_successful'][()]\n",
    "    gt_all = F['ground_truth'][()]\n",
    "    gt = gt_all[idx]\n",
    "    sinabs_out = F['sinabs_label'][()]\n",
    "    spk_orig = F['n_spikes_orig'][idx]\n",
    "    spk_attk = F['n_spikes_attk'][idx]\n",
    "    print(F.keys())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "04708d14",
   "metadata": {},
   "outputs": [],
   "source": [
    "report = np.loadtxt(\"../Results/report_ep10_lb4_num50_patchsize0.025.csv\", skiprows=1, delimiter=\",\")\n",
    "\n",
    "(\n",
    "    _, gt2, chip_out, sim_out,\n",
    "    chip_out_attacked_targeted, chip_out_attacked_random,\n",
    "    sim_out_attacked_targeted, sim_out_attacked_random\n",
    ") = report.T\n",
    "\n",
    "assert np.all(gt == gt2)\n",
    "assert len(gt) == len(spk_orig) == len(spk_attk)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "3aba4ffb",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Fraction of cases where sinabs got the right label: 0.9\n",
      "Fraction of successful attacks: 0.9\n"
     ]
    }
   ],
   "source": [
    "print(\"Fraction of cases where sinabs got the right label:\", (sinabs_out == gt_all).mean())\n",
    "print(\"Fraction of successful attacks:\", idx.mean())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "859a467a",
   "metadata": {},
   "outputs": [],
   "source": [
    "class_labels = [\n",
    "    \"Hand Clap\",\n",
    "    \"RH Wave\",\n",
    "    \"LH Wave\",\n",
    "    \"RH CW\",\n",
    "    \"RH CCW\",\n",
    "    \"LH CW\",\n",
    "    \"LH CCW\",\n",
    "    \"Arm Roll\",\n",
    "    \"Air Drums\",\n",
    "    \"Air Guitar\",\n",
    "    \"Other\",\n",
    "]\n",
    "# class_labels = [\"-\"]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6ff210f0",
   "metadata": {},
   "source": [
    "# video"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "ff1f3c4d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "6.0 4.0\n",
      "3.0 4.0\n",
      "3.0 4.0\n",
      "7.0 4.0\n",
      "6.0 4.0\n",
      "9.0 4.0\n",
      "7.0 4.0\n",
      "10.0 4.0\n",
      "9.0 4.0\n",
      "3.0 4.0\n",
      "3.0 4.0\n",
      "3.0 4.0\n",
      "6.0 4.0\n",
      "7.0 4.0\n",
      "7.0 4.0\n",
      "1.0 4.0\n",
      "6.0 4.0\n",
      "3.0 4.0\n",
      "1.0 4.0\n",
      "6.0 4.0\n"
     ]
    }
   ],
   "source": [
    "from aermanager.preprocess import create_raster_from_xytp\n",
    "\n",
    "fps = 25\n",
    "slowdown = 10\n",
    "T = 0.2 # 200 ms\n",
    "pause = 10\n",
    "dt = 1000000/fps/slowdown\n",
    "length = int(slowdown*fps*T)\n",
    "bins = np.arange(129)\n",
    "\n",
    "F = h5py.File(\"../Results/attacks_patches_ep10_lb4_num50_patchsize025.h5\", \"r\")\n",
    "N = 20\n",
    "orig, attk = np.zeros((2, N*(length+pause), 128, 128))\n",
    "orig_labels, attk_labels = [], []\n",
    "\n",
    "attacked_IDs = sorted([int(k) for k in F['attacked_spiketrains'].keys()])\n",
    "\n",
    "i = 0\n",
    "for j, k in enumerate(attacked_IDs):\n",
    "    if i >= N: break\n",
    "\n",
    "    if chip_out[j] != 4 and chip_out_attacked_targeted[j] == 4:        \n",
    "        spk_orig = F['original_spiketrains'][str(k)][()]\n",
    "        spk_attk = F['attacked_spiketrains'][str(k)][()]\n",
    "        orig_labels.append(chip_out[j])\n",
    "        attk_labels.append(chip_out_attacked_targeted[j])\n",
    "\n",
    "        start = i*(length+pause)\n",
    "\n",
    "        raster_orig = create_raster_from_xytp(spk_orig, dt=dt, bins_x=bins, bins_y=bins)\n",
    "        orig[start:start+length] = raster_orig[:, 1] - raster_orig[:, 0]\n",
    "        raster_attk = create_raster_from_xytp(spk_attk, dt=dt, bins_x=bins, bins_y=bins)\n",
    "        attk[start:start+length] = raster_attk[:, 1] - raster_attk[:, 0]\n",
    "    \n",
    "        i += 1\n",
    "\n",
    "F.close()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "2ff98bee",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAsgAAAF2CAYAAABzg27uAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjQuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8rg+JYAAAACXBIWXMAAAsTAAALEwEAmpwYAAAaI0lEQVR4nO3debQkdX338c93hn0HFWQTiICKMEPYNKJRQY2GB5fEiEaj+AhGPa4JLomSYNSQGFBUnkSiEI0kChLD0RjFNRJcWDMzrC6gMDLIpsCAbCO/54+qO/7mMnfmAhcYMq/XOX2Y21VdVd1976/fXV3VVGstAADAYNaDvQEAALAqEcgAANARyAAA0BHIAADQEcgAANARyAAA0BHIq4Cq+vOq+vhMzzuNZbWq2nGa8x5RVSfOxHqnub5dquqcqqoHap0PNVW1/fgcrjH+/KWqesW9WM6jqurmqpo9A9t0VlU9/r4uB7h/VNW+VfXD8W/++Q/29sy0e/K6BisikGdYVR1cVedX1S+r6mdV9Q9VtcmKbtNa++vW2iHTWf49mffBUlWfqKr33sfFvCfJUW0V/qLuqvqvqlplnovW2nNaa59c2XxV9ZOqekZ3uytaaxu01n41A5txVJK/moHlAPfBOD79oqrWnjTpr5IcO/7Nn/pAB+XydrasamMpJAJ5RlXVnyb52yRvTbJxkicm2S7JV6tqrSlus8YDt4UPDVW1ZZKnJzl1hpd7t8d6VXr8V6VtuQ8+n+TpVfXIB3tDYHVVVdsneUqSluS5kyZvl+TCGVrP/4YxC5ZLIM+QqtooybuTvKG19uXW2p2ttZ8keVGS7ZO8bJzviKo6papOrKqbkhw8+R11Vb28qi6vquur6vB+j18/b/cR+yuq6oqquq6q3tktZ5+q+m5V3VBVV1XVsVOF+nLuzw5V9a2qWlxVX03y8EnTPzvuIb+xqk6f+Fi9ql6d5KVJ3jZ+hPeF8fp3VNWl4/IuqqoXrGD1z0xyXmvttm5921bV56rq2vFxOXa8flZVvWt8vK6pqn+uqo0nPT6vqqorknxj3MP/7ar6YFVdn+SIqlq7qo4aH8Orq+qjVbVut+7nVdW8qrppvA/Prqr3ZXgBOna8n8cu5zGcWP+rq2rR+Bwc1k1f3u/CxlV1/DjvlVX13hoPfaiq2eN2XldVlyU5YNL6ltkLU1WHVtXF3WO+R1V9Ksmjknxh3O631d0P1diqqj5fVT+vqh9V1aGTtvnk8XFeXFUXVtVeE9PH5+zcJL+zgucXuH+9PMn3knwiydLDrqrq0iS/kV///X93nDR//Pmgcb7/M455N1TVd6pqTreMn1TV26tqQZJbavk7Hj5UVQvHMfPcqnrKeP2zk/x5koPG9c2faiydahnjtNk1HG448ZpyblVtu5ztePK4jKfdlweT1VRrzWUGLkmenWRJkjWWM+2TST49/vuIJHcmeX6GNyjrjtedOE7fJcnNSZ6cZK0MH1nfmeQZ3e0n5t0+wx6Cj43LmZvk9iSPG6fvmWEv9hrjvBcneXO3XS3JjlPcn+8m+UCStZP8dpLFE+sdp//fJBuO049JMq+b9okk7520vD9IstV4nw9KckuSLadY998l+X/dz7OTzE/ywSTrJ1knyZO77fhRhkF/gySfS/KpSY/PP4+3WzfJwePz9IbxcVl3XO7nk2w23qcvJDlyXMY+SW7MEO2zkmyd5LHjtP9KcsgKficm1v/pcf27Jbl20nM5+Xfh35McN86/eZKzkvzxOP9rklySZNtxW785Ln+NydszPt5XJtk7SSXZMcl247SfTGzDpO2cWM7pSf5+fJx3H7d5v26bb0vyu+PzcmSS70263x9O8oEH+2/SxWV1vYxj4usyvAbcmWSLbtrkv/9lXgeS/GaSa5I8Yfwbf8V4m7W7288bx6F1p1j/y5I8bBxj/zTJz5KsM047It1ryXjd3cbSlSzjrUnOT/KYcXybm+Rh/f3J8Jq8MMk+D/bz4fLQvNiDPHMenuS61tqS5Uy7Ksvugf1ua+3U1tpdrbVbJ837wiRfaK2d0Vq7I8lfZPiDX5F3t9Zuba3NzxCSc5OktXZua+17rbUlbdibfVySp67sjlTVozKE1eGttdtba6dniMalWmsntNYWt9ZuzzDgzZ3Yc7s8rbXPttYWjff5pCQ/zBCfy7NJhiCfsE+GuH5ra+2W1tptrbUzxmkvzRBjl7XWbk7yZ0lePGmvxhHj7SYe60WttY+Mz9VtSV6d5C2ttZ+31hYn+eskLx7nfVWSE1prXx23/crW2iVT3c8pvHtc//lJ/inJS7ppS38XkmyUITzfPM5/TYZ4n9iWFyU5prW2sLX28wxxOpVDkry/tXZ2G/yotXb5yjZ03Auzb5K3j4/zvCQfz7BHasIZrbX/bMMxy5/K+PvWWZzhOQQeYFX15AyHUZzcWjs3yaVJ/vAeLOLVSY5rrZ3ZWvtVG85ruD3DzpYJHx7HocmvX0mS1tqJrbXrx9eeozPsSHnMPbkfK1nGIUne1Vr7/ji+zW+tXd/d/A8yvN49p7V21j1ZL0wQyDPnuiQPX97HTUm2HKdPWLiC5WzVT2+t/TLJ9VPPnmR4Zz3hlxn2pKaqdq6q/6jhUIibMoTfw5e3gOVswy9aa7d01y2Nq/Hjrb8ZP966KcMehaxo2TUcNjLxkd0NSXZdwfy/yLAnd8K2SS6f4s3HVv22jf9eI8kW3XWTH+/+50ckWS/Jud22fXm8fmLdl051v6apX9/l4zYvb9p2SdZMclW3Lcdl2JOcTPrdyLL3e7J7u91bJZl4o9CvZ+vu58m/b+tM+r3fMMkN92LdwH33iiRfaa1NvOb8a7rDLKZhuyR/OjEGjePQtpl63LqbqjpsPLzrxvH2G2d6rz3TXcbKxrc3Z3iDcME9WSf0BPLM+W6Gd9m/119ZVRskeU6Sr3dXr2iP8FVJtuluv26Gj5nujX/I8JH8Tq21jTIc+zWdr027KsmmVbV+d92jun//YZLnJXlGhkFr+4nNHf+7zP2rqu0yHAby+gwfg22S5IIVbMuCJDt3Py9M8qgp3nwsyjCg99u5JMnV3XWTH+/+5+uS3Jrk8a21TcbLxq21Dbp1P3qK7ZzuN2z0x8Y9atzm5S1jYYbfoYd327JRa23ia9OuWs6ypnJvt3tRks2qqn+D8qgMh2tM1+MyfJIBPIDG14sXJXnquGPkZ0nekuETvsmf9ExlYZL3dWPQJq219Vprn+7mmXIMGY8Vftu4HZuO4/2NmeL1YXnXTWMZKxrfkmEP8vOr6k0rmAdWSCDPkNbajRlO0vtIDSdxrVnDmcQnJ/lpho+ip+OUJAdW1ZNqOKHuiEwvapdnwyQ3Jbm5qh6b5LXTudH4Ufw5Sd5dVWuNH9kdOGm5t2fYs71ehj3TvaszHBM8Yf0MA+C1SVJVr8ywB3kqX02yR1WtM/58VoY4/JuqWr+q1qmqfcdpn07ylhpOKtxg3JaTptjbvLz7eleGeP9gVW0+bt/WVTVxktnxSV5ZVfvXcELg1uNjubz7OZXDq2q9Gk5kfGWSk6bYlquSfCXJ0VW10bi+R1fVxGExJyd5Y1VtU1WbJnnHCtb58SSHVdWeNdhxfKOywu1urS1M8p0kR46P85wMh5lM6zuwx+dszwzPIfDAen6SX2U4l2X38fK4JP+dZQ+T6k0eDz6W5DVV9YRx7Fi/qg6Y9KZ5RTbMsJPi2iRrVNVfZDh8rF/f9lU1a9J1/TasbBkfT/Keqtpp3MY5VdXvSFqUZP8kb6qqab3uwWQCeQa11t6fYS/tURnC9MwM73T3H4/Vnc4yLsxwAtlnMkThzRlOmJjW7Sc5LMPe3sUZBr3lhtkU/jDDSRo/T/KXGU50m/DPGT52vzLJRRnOlu4dn2SX8eO5U1trFyU5OsNe9qsznKz27alW3Fq7Osk3Muylznis64EZTry4IsMbjoPG2U/I8Obj9CQ/znBM8Rvuwf1MkrdnOKnle+MhI1/LeKzbePzaKzMcC3xjkm/l13usP5TkhTV81+iHV7D8b43L/3qG73b+ygrmfXmGkzMvynCoySkZDtFJhufwtAx7Z8/LcELicrXWPpvkfRk+Xl2c4SvzNhsnH5nkXePzc9hybv6SDJ8KLMpw0uBftta+toJt7h2Y5L9aa4tWOicw016R5J/a8N3mP5u4JDk2yUun+BTuiCSfHMeDF7XWzkly6HibX2QYuw6+B9twWobD1H6Q4XXitix7SMZnx/9eX1Xnjf+ePJaubBkfyLDD4CsZXmuPz3CS81KttSsyRPI7yncscy9Ua6vs/4eBLD1E44YMh0n8+EHenAdMVe2S4ds/9mkP0V/S8ROEHydZc7p7tB/qqurMJK9y7B8AD2UCeRVUVQdm2NtYGfa8PiHJHg/VUFxdrY6BDAD/GzjEYtX0vAwfby9KslOSF4tjAIAHhj3IAADQsQcZAAA6AhkAADrL+8qXqWeetWZba/Za99e2AKw27vjVHVly1533+DvOjcMAM+fWJb+8rrX2iMnX36NAXmv2Wtl5kxX9/x0AmI4f3HDvvgnPOAwwc+Zfd9bly7veIRYAANC5z4H8mrcekm9ectqU07fadsvMu/bMPOWZ+95t2l5P2iPzrj0zj37syv9vvdtsv3X+8ph35svzPp+zrzwj37j4y/ngJ9+fJ/z23svMV1V5wcuem09+8WM547Jv5MyFp+eU0/81r33bodlwow2y1tpr5ewrz8ghf/LKZW63w07bZ961Z+ajp3zkbus+bf4XcvjRf7bSbQR4MLzmrYdk3rVnLr187cL/zIf/5ejstMuOy8xnPAaYnnt0iMWDZfd95uQjn/5grrhsYf7hb/8xC3/y02z6sE2z/wFPy9+f/KE8dadn5ubFt6Sq8v6Pvy+//awn56QTTslxRx+fO26/M4/dbee85JAXZf0N189Rhx+TS87/fnbfe84y65i7z2659ZZbs+sej8+sWbNy1113JUm23OaR2WKrzTPvrPkPxl0HmJbFNy7O6w56c5IhhF/3jlfno5/9SF6w70G56YabZmw9xmNgdbDKB/La66ydv/3Y+3Lh/1yU17/kLVly56//h2Rf/49v5nMnfj5LlgzXHfSqF2a/A56W173oTTnz9LOXznfOt8/Nyf/0b9l9n2EQnnfWgjz3xQcss545e+2WL5/61Rzwwmdnp10ene9f8MMkydxx4J5/9vn36/0EuC+WLPlVzj93OK75/HMvyKKFV+VTXz4+++73xHzpc1+ZkXUYj4HVxSp/DPIzn7tftthq8xx1+DHLDMYTzvn2ubnt1tuTJC/745fkm186fZnBeMIdt9+Rs/77nCTD4LrxphvnN3beYen0uXvPyTnfPi8XL/j+0kE4GfZkXH/Nz3PFZQtn+q4B3G9+cOEQlVtsvcWMLdN4DKwuHrBAnjVrVmbPnr3MZdbsla9+zyftkWuuuiY/uvjSFc63xVabZ5vtt853vv7dlS5z/lkLkgyDbZJsuNEG2WGn7bLgnPOz4JzzM3fv3ZbOO3fvOZl/9oKVLhNgVfLIMYwXXbHobtOMxwAr9oAdYvGhE4+6V7fbfMtH5KqfXj2N+TZPklx15c9WOu+1V1+XKy9flLl7z8m/n/j5zNl7t9xw/Q1Z+OOfZsE55+dNh78+SbLOeutkp112zGn//tV7te0AD6TZs2cnSbbc9pH5s785LJec//1880un320+4zHAij1ggfx37/pg/ud785a57nFzH5fDj37Hym/c2vRXNM1Z55+9YOmeibl7z8mC8di9BedckG223zoP23yz7LDTDllzzTWcEAKs8jZ92CY592ffWfrzL66/IS991sG584477zav8RhgxR6wQL7isoW5aP4ly1y33vrrrfR211x1bXZ8+qOnMd81SZJHbjO94+3mn31+nv17z8rGm26UOXvtuvQ4uasXXZOrF12TuXvPyW/svH1uv+32u203wKpm8Y2L88e///rMmj07j3n8TnnLu9+YIz/6nhx8wKFpk6LWeAywYqv8SXrnfPu8bLHV5nn0Y3ZY4XxXL7omC3/80zzp6U+c1nLnnTU/s2bNym8+Yffsusfjs+CcX58VPRz3Nidz956Tixdcstw9MACrkiVLfpWL5l+SC867MP/2qVNz5Nv/LnP33i3PfN7+M7YO4zGwuljlA/lrX/hGrl50TQ57z1uyxhqz7zZ9ryftkXXWXTtJ8i//+Jnsd8DTste+e95tvrXWXit7P/nX1//woktz8+Jb8vsvf37WXmftXPg/Fy2dNv/s87P7PnOy2567Zt6ZTggBHnq++Nkv5UcXX5pXvuGPZmyZxmNgdTEjh1isudaaecaB+93t+nO/c959Xvbtt92etx/6zhz7mWPyiS9+LCedcEp+evmibLLZxtnvd5+a5/z+7+RpOz8rye056fhTsudv/WaO/fQHctIJ/5bvfevM3HnHkjxm151y0KtemNNPOyNnn3FukuSuu+7KBeddmH33/61ccv4Pln41UTLssfiTd78xs2bNyryzDMjAQ9Pxx3wiRx73nuzzlL2Wfq3afWE8BlYXMxLIG2y4fo464ci7XX/I816bRQuvus/Ln3fWgrxk/5fnVW8+OK//89dms0dslsU3Ls68M+fnNS98Y25efEuSpLWWtx3yzjz/pQfmBS99Xv7g4Bdk9uzZueKyhfniyV/Kvxz3mWWWO/+sBXniU/dZ5uO8JLl4wfez5M4lWWvttXylEPCQddqpX8tr3nZoDn7DH81IICfGY2D1UJNP3liR9dZcv+28ya734+YArB5+cMMF+eWdt9Q9vZ1xGGDmzL/urHNba3tNvn6VPwYZAAAeSAIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADoCGQAAOgIZAAA6AhkAADrVWpv+zFXXJrn8/tscgNXGdq21R9zTGxmHAWbUcsfiexTIAADwv51DLAAAoCOQAQCgI5ABAKAjkAEAoCOQAQCgI5ABAKAjkAEAoCOQAQCgI5ABAKDz/wH0gBQyzZwuZQAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 720x395.52 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from matplotlib import animation\n",
    "\n",
    "fig, ax = plt.subplots(1,2, figsize=(10, 5.5))\n",
    "\n",
    "MAX = 3 \n",
    "\n",
    "im = ax[0].imshow(\n",
    "    np.zeros((128, 128)),\n",
    "    interpolation='none',\n",
    "    vmin=-MAX, vmax=MAX,\n",
    "    cmap=plt.cm.twilight\n",
    ")\n",
    "ax[0].set_xticks([])\n",
    "ax[0].set_yticks([])\n",
    "im2 = ax[1].imshow(\n",
    "    np.zeros((128, 128)),\n",
    "    interpolation='none',\n",
    "    vmin=-MAX, vmax=MAX,\n",
    "    cmap=plt.cm.twilight\n",
    ")\n",
    "ax[1].set_xticks([])\n",
    "ax[1].set_yticks([])\n",
    "ax[0].set_title(\"Original data (correct prediction)\")\n",
    "ax[1].set_title(\"After attack\")\n",
    "\n",
    "t = ax[0].text(1, 5, \"Prediction:\", c='white', size=15)\n",
    "t2 = ax[1].text(1, 5, \"Prediction:\", c='white', size=15)\n",
    "\n",
    "# initialization function: plot the background of each frame\n",
    "def init():\n",
    "    im.set_data(np.zeros((128, 128)))\n",
    "    im2.set_data(np.zeros((128, 128)))\n",
    "    \n",
    "    return [im]\n",
    "\n",
    "# animation function.  This is called sequentially\n",
    "def animate(i):\n",
    "    im.set_array(orig[i])\n",
    "    im2.set_array(attk[i])\n",
    "    \n",
    "    orig_label = class_labels[int(orig_labels[i // (length + pause)])]\n",
    "    attk_label = class_labels[int(attk_labels[i // (length + pause)])]\n",
    "    t.set_text(orig_label)\n",
    "    t2.set_text(attk_label)\n",
    "    return [im]\n",
    "      \n",
    "fig.tight_layout()\n",
    "anim = animation.FuncAnimation(fig, animate, init_func=init,\n",
    "               frames=len(orig), blit=True)\n",
    "\n",
    "# save the animation as an mp4.  This requires ffmpeg or mencoder to be\n",
    "# installed.  The extra_args ensure that the x264 codec is used, so that\n",
    "# the video can be embedded in html5.  You may need to adjust this for\n",
    "# your system: for more information, see\n",
    "# http://matplotlib.sourceforge.net/api/animation_api.html\n",
    "anim.save('patches_animation.mp4', fps=fps, dpi=150, extra_args=['-vcodec', 'libx264'])\n",
    "\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5ea2941d",
   "metadata": {},
   "source": [
    "# static figure"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 104,
   "id": "36a43a26",
   "metadata": {},
   "outputs": [],
   "source": [
    "k = '0'\n",
    "N_cols = 5\n",
    "N_rows = 1\n",
    "sample_len_ms = 200\n",
    "dt = sample_len_ms / N_cols * 1000\n",
    "bins = np.arange(129)\n",
    "\n",
    "with h5py.File(\"../Results/attacks_patches_ep10_lb4_num50_patchsize025.h5\", \"r\") as F:\n",
    "    spk_orig = F['original_spiketrains'][k][()]\n",
    "    spk_attk = F['attacked_spiketrains'][k][()]\n",
    "    \n",
    "    raster_orig = create_raster_from_xytp(spk_orig, dt=dt, bins_x=bins, bins_y=bins).sum(1)\n",
    "    raster_attk = create_raster_from_xytp(spk_attk, dt=dt, bins_x=bins, bins_y=bins).sum(1)\n",
    "    raster_diff = raster_attk - raster_orig"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 112,
   "id": "9970df99",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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zm+dnb+QdAeSHlVfvzrTA2LQWQxtqAMO0AghhBBCCCEkEnqfg9ZUbkmVMEeoHCnHo7cdJ+2jjpDHPpOyRvOgPeoOqeu0i1qM9RnRJqnr1EcX9dtXos9Ba3M6vquDM4CGuks0eS1NOETIPWnit5dpX1fDE1KlLntsaySGkNbh0PlcJAmwjEabfjYWvafKVpDkMz8umrKlpJtsaLsBAMVWhNBzRW9a92jqWo7H40LHo9a6RRPXcTabte4cm81mC8n29EqnRUrXqmhbU/ptxA+vI1mGKGbQCCGEEEIIIYQ0PED74he/iNe+9rXYsmUL/vRP/9S5zbPPPou3ve1t2LJlC3bu3ImHH364ySZGT+h0eczhDlWUrK6TF154AT//8z+P8847b/7eQw89hJ07d2LLli1429vehueee8753Q996EPYsmULXvva1+Kuu+5qqsmtUlZrJvwjVp3GzPXXX4+tW7di27ZtuPTSS/HMM88AKKbTV73qVa3ptO5rHmJfXAu++2BVvHK8853vxNFHH41t27YtvH/11VfjhBNOwI4dO/DWt74VTzzxxPyzEBsaqvPUoe7qx6dRALjhhhtwwgknYOvWrbjmmmvm71OjpBG01rX9GwwG2nDgwAG9efNm/cADD+hnn31W79ixQ+/du1fbfOxjH9Oj0UhrrfVnP/tZffHFF6/bpk7G43HQdtPptNT+p9Pp/N8y+wk5Tp9x/X4AU52jU621/vM//3N96aWX6je/+c3z937jN35Df/azn9Vaaz0ajfRf/uVfrtv/3r179Y4dO/QzzzyjH3zwQb1582Z94MCB6n6U1gva0TpcrzGzzL3UNUI0um/fPr1p0yb99NNPa61XtfnpT396/ncMOiXdJtSW/v3f/72ezWZ669atC+/fdddd+vnnn9daa33NNdfoa665Rmsdrs0QnRMSolOfRnfv3q1/5Vd+RT/zzDNaa62///3va62pUVItPo1qrZubQduzZw+2bNmCzZs347DDDsMll1yC22+/fd12t99+Oy677DIAwEUXXYSvfOUr69bP+ru/+zucccYZuOCCC7B582Zce+21uPnmm3HKKadg+/bteOCBBwAAf/u3f4tt27bhpJNOwhvf+MagdobmRfjWtMnzdtlFFTh7UA9lz+u+fftw55134oorrpi/p7XG7t27cdFFFwEALrvsMnzhC19Y993bb78dl1xyCQ4//HAcf/zx2LJlC/bs2QNgURsvfvGLcfXVV2Pr1q341V/9VezZswe//Mu/jM2bN2PXrl0AgL179+KUU07BySefjB07duD++++f/66sNaB8dNET2+d758CBA/jP//xPHDhwAE8//TR+7ud+rhKdSl784hfjFa94RSmdhtLkTHoZ/XftnmmaN77xjfiZn/mZde+fffbZ2LBhNQX+F37hF7Bv3z4AYdoM1fkf/MEf4LLLLsMv/dIv4dWvfjVuvfVWXHPNNdi+fTvOOeccPP/88wCAa6+9FieeeCJ27NiBq666qsqfXxlF7hNqthg+jX784x/Htddei8MPPxwAcPTRRwOgRquijO3vm7YbG6B997vfxate9ar562OPPRbf/e53M7fbsGEDjjzySOzfv3/ddvfccw8+8YlP4J//+Z9x00034Tvf+Q727NmDK664AjfccAMA4IMf/CDuuusu3HPPPfMORR7LdBhCQ72a6CwPBoPeibkKrrzySnzkIx/BIYes3Rr79+/HS1/60nmHIkS79nZSG0899RTOOuss7N27Fy95yUvwvve9D1/60pdw22234f3vfz8A4BOf+ATe+9734lvf+ham0ymOPfbYhWNlVZ9zXfemQ16LaLxsu/qq72OOOQZXXXUVjjvuOLzyla/EkUceibPPPrsSnUqeeuopfPrTny6t05Dr02ShENrEOPnUpz6FN73pTQDCtBmqcwB44IEHsHv3buzatQvveMc7cOaZZ+Lee+/Fxo0bceedd2L//v247bbbsHfvXnz729/G+973vkJtr1NPct9F7pM+O66q5Dvf+Q7+4R/+ATt37sQZZ5yBb37zmwDS02jT1Pnc75u2ky0S8oY3vAGvfOUrcfjhh+M1r3kNzj77bADA9u3b53lrp59+On77t38bn/zkJ/HCCy+02FqSAnfccQeOPvro2o3AYYcdhnPOOQfAql7POOMMHHrooQvaPfXUU/Enf/In+PCHP4xHHnkEGzdurLVNJB1++MMf4vbbb8dDDz2E733ve3jqqafwmc98pvLjUKekbv74j/8YGzZswNvf/vZa9v+mN71prtkXXnhhQc8PP/wwjjzySPzUT/0ULr/8ctx666044ogjamkHSY8DBw7g8ccfx913340/+7M/w8UXX7wumqsKqFHio7EB2jHHHINHH310/nrfvn045phjMrc7cOAAfvSjH+FlL3vZuu3MtDMAHHLIIfPXhxxyCA4cOABg1bv7R3/0R3j00UcxGAycM3E2IZ6qZb1mTa0b1Tdvw7J87Wtfw65du7Bp0yZccskl2L17N97xjnfgZS97GZ544om5rkK0m7XdoYceCqUUAL92f/M3fxO7du3Cxo0bce6552L37t0L+8haZDOG697EjF0Mv7MNvvzlL+P444/HUUcdhUMPPRQXXnghvv71r0en0xivD8udx8Pf/M3f4I477sDNN98811mINkN1DmBBs7aeDxw4gA0bNmDPnj246KKLcMcdd8w7x6FUoQ/f85zaa5djjz0WF154IZRSOOWUU3DIIYfgscceS06jTcN1equjsQHaG97wBtx///146KGH8Nxzz+GWW27B+eefv267888/HzfeeCMA4POf/zzOOuusuWCL8sADD2Dnzp344Ac/iKOOOmrhplqGuoWVNwAzn3cxr6hNPvShD2Hfvn14+OGHccstt+Css87CZz7zGSilcOaZZ+Lzn/88AODGG2/EBRdcsO77559/Pm655RY8++yzeOihh3D//ffjlFNOKdWWBx98EJs3b8Z73vMeXHDBBfj2t7+91G8rQxXaqkqfqei8iXYed9xxuPvuu/H0009Da42vfOUreN3rXtdbnZL0+OIXv4iPfOQj2LVr18KMQIg2Q3Uewn/8x3/gRz/6Ec4991xcf/31uOeee8r/qJKwemicvOUtb8FXv/pVAKvhjs899xxe/vKX91KjpB0aG6Bt2LABf/EXf4Ff//Vfx+te9zpcfPHF2Lp1KwDg/e9//zxH7PLLL8f+/fuxZcsWXHfddd5y/CFcffXV2L59O7Zt24bTTjsNJ510Uul9GUO5TI5a6HdNroTPMJtZvphL6XeND3/4w7juuuuwZcsW7N+/H5dffjkAYNeuXfN8nK1bt+Liiy/GiSeeiHPOOQcf+9jH8KIXvajU8T73uc9h27ZtOPnkk3Hffffht37rt4K+V/XDvGwHoeoS+qnovIl27ty5ExdddBFe//rXY/v27fjJT34ytwmp6JT0g0svvRSnnnoq/uVf/gXHHnss/vqv/xoA8O53vxtPPvkkfu3Xfg0nn3wyfvd3fxdAtjbPPfdcfO973wPg13lRnnzySZx33nnYsWMHfvEXfxHXXXfd/LMYBkbyGR9De7qIT6PvfOc78eCDD2Lbtm245JJLcOONN0IpFZVGSbdRdcTUGobDoc4KxUoJOWtVNrF9MpkUqroHpNMxTQGl1Exrva66Rtd0WpVmZIeg6D6p33L0QaMkffqg09jWaIytPSnQB52StPFpFEi4SEjTGE/WMlXHilZiYtUxUoTRaITBYJA7U1t0sfOy1ZbaDgUm8cJr1126EpYX22AotvakTqoaTbXdKRGLDat1Bk0p9QMAj9R2AEKK8Wqt9VH2m9QpiQhqlKQAdUpSgDolsePUKFDzAI0QQgghhBBCSDgMcSSEEEIIIYSQSOAAjRBCCCGEEEIigQM0QgghhBBCCIkEDtAIIYQQQgghJBI4QCOEEEIIIYSQSPj/jS1OlVSvltQAAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 864x338.4 with 5 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib as mpl\n",
    "\n",
    "colors = [(1., 0., 0., 0.0), (1., 0., 0., 1.0)]\n",
    "nodes = [0, 1]\n",
    "cmap = mpl.colors.LinearSegmentedColormap.from_list(\n",
    "    \"transparent_red\", list(zip(nodes, colors)))\n",
    "\n",
    "fig = plt.figure(constrained_layout=True, figsize=(12, 4.7))\n",
    "spec = mpl.gridspec.GridSpec(ncols=N_cols, nrows=N_rows, figure=fig)\n",
    "axes = [fig.add_subplot(spec[i, j]) for i in range(N_rows) for j in range(N_cols)]\n",
    "time_labels = [\"%.1f ms\" % (dt*i/1000) for i in range(len(axes))]\n",
    "\n",
    "\n",
    "for i, ax in enumerate(axes):\n",
    "    ax.set_aspect(\"equal\")\n",
    "    ax.tick_params(axis='both',\n",
    "                   which='both',\n",
    "                   bottom=False,\n",
    "                   top=False,\n",
    "                   labelbottom=False,\n",
    "                   right=False,\n",
    "                   left=False,\n",
    "                   labelleft=False)\n",
    "    \n",
    "    ax.pcolormesh(raster_orig[i][::-1], vmin=0, vmax=5, cmap=plt.cm.gray_r)\n",
    "    ax.pcolormesh(raster_diff[i][::-1], vmin=0, vmax=5, cmap=cmap)\n",
    "    ax.text(2, 2, time_labels[i])\n",
    "\n",
    "i = 0 # i and k are not the same thing, determining this manually, sorry\n",
    "label_orig = class_labels[int(chip_out[i])]\n",
    "label_attk = class_labels[int(chip_out_attacked_targeted[i])]\n",
    "axes[0].set_ylabel(fr\"{label_orig}$\\rightarrow${label_attk}\")\n",
    "\n",
    "plt.savefig(\"patches_sample.png\", bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "df3a5a42",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "72c32930",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "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.9.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
