{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "fb6837e9",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import os\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "b145dee9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'22 54 1063 1067 12 18 23 59 188 307 1043 1459 1475 1489 1497 1503 40499 44125 44131 45062 44157 1462 44160 29 37 53 49 1504 1494 41143 44126 40981 41168 44091 44158 44123 44090 40922 44161 45714'"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "datalist = [\n",
    "    22, 54, 1063, 1067, 12, 18, 23, 59, 188, 307, 1043, 1459, 1475, 1489, 1497, 1503, 40499, 44125,\n",
    "    44131, 45062, 44157, 1462, 44160, 29, 37, 53, 49, 1504, 1494, 41143, 44126, 40981, 41168, 44091, 44158, 44123, 44090,\n",
    "    40922, 44161, 45714]\n",
    "\" \".join(map(str, datalist))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "087a333b",
   "metadata": {},
   "outputs": [],
   "source": [
    "# result = pd.read_csv(\"/home/SemiTab/result-summary/result.csv\", index_col=0)\n",
    "# result = result[result[\"model\"].isin([\n",
    "#     \"sslbinshuffling-lr\", \"sslbinsampling-lr\", \"sslshuffling-lr\"])].reset_index(drop=True)\n",
    "# result = result[[\"trial\", \"shots\", \"data_id\", \"model\", \"acc\"]]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "7e9af2d6",
   "metadata": {},
   "outputs": [],
   "source": [
    "# p = pd.read_csv(\"/home/SemiTab/result-summary/result-numshots.csv\", index_col=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "6a7afd3d",
   "metadata": {},
   "outputs": [],
   "source": [
    "# result = result[~result[\"data_id\"].isin([1492, 4153])]\n",
    "# result[\"model\"] = result[\"model\"].replace(\"sslbinshuffling-lr\", \"sslbinshuffling\")\n",
    "# result[\"model\"] = result[\"model\"].replace(\"sslbinsampling-lr\", \"sslbinsampling\")\n",
    "\n",
    "# pd.concat([result, p], axis=0).to_csv(\"/home/SemiTab/result-summary/result-numshots.csv\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5b1e9d31",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "8096b653",
   "metadata": {},
   "outputs": [],
   "source": [
    "def update(model):\n",
    "    result = pd.read_csv(\"/home/SemiTab/result-summary/result-numshots.csv\", index_col=0)\n",
    "    i = len(result)\n",
    "    print(i)\n",
    "    for trial in range(10):\n",
    "        for shot in [10, 20, 30]:\n",
    "            for d in datalist:\n",
    "                if len(result[(result[\"trial\"] == trial) & (result[\"shots\"] == shot) & (result[\"model\"] == model) & (result[\"data_id\"] == d)]) > 0:\n",
    "                    pass\n",
    "                else:\n",
    "                    fname = f'/home/SemiTab/results/seed={trial}/shot={shot}/model={model}/data={d}/performance.npy'                \n",
    "                    try:\n",
    "                        perf = np.load(fname, allow_pickle=True).item()\n",
    "                        if model.startswith(\"ssl\"):\n",
    "                            result.loc[i] = [trial, shot, d, model, perf[\"Test\"][\"lr\"][0]]\n",
    "                            i += 1\n",
    "                        else:\n",
    "                            result.loc[i] = [trial, shot, d, model, perf[\"Test\"][0]]\n",
    "                            i += 1\n",
    "                    except Exception:\n",
    "                        print(trial, shot, d)\n",
    "    #                     if not (((shot >= 10) and (d in [1492, 4153]))):\n",
    "    #                         todo = True\n",
    "                        pass\n",
    "    result.to_csv(\"/home/SemiTab/result-summary/result-numshots.csv\")\n",
    "    print(result.shape)\n",
    "    return result"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "1b973980",
   "metadata": {},
   "outputs": [],
   "source": [
    "def update2(model, datalist=[49], shots=[10, 20, 30], trials=range(100)):\n",
    "    result = pd.read_csv(\"/home/SemiTab/result-summary/result-numshots.csv\", index_col=0)\n",
    "    for trial in trials:\n",
    "        for shot in shots:\n",
    "            for d in datalist:\n",
    "                fname = f'/home/SemiTab/results/seed={trial}/shot={shot}/model={model}/data={d}/performance.npy'                \n",
    "                try:\n",
    "                    perf = np.load(fname, allow_pickle=True).item()\n",
    "                    if model.startswith(\"ssl\"):\n",
    "                        result.loc[\n",
    "                            (result[\"trial\"] == trial) & (result[\"shots\"] == shot) & (\n",
    "                                result[\"data_id\"] == d) & (result[\"model\"] == model), \"acc\"] = perf[\"Test\"][\"lr\"][0]\n",
    "                    else:\n",
    "                        result.loc[\n",
    "                            (result[\"trial\"] == trial) & (result[\"shots\"] == shot) & (\n",
    "                                result[\"data_id\"] == d) & (result[\"model\"] == model), \"acc\"] = perf[\"Test\"][0]\n",
    "                except Exception:\n",
    "    #                     if not (((shot >= 10) and (d in [1492, 4153]))):\n",
    "    #                         todo = True\n",
    "                    pass\n",
    "    result.to_csv(\"/home/SemiTab/result-summary/result-numshots.csv\")\n",
    "    print(result.shape)\n",
    "    return result"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "fea54353",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'12 18 22 23 29 37 49 53 54 59 188 307 1043 1063 1067 1459 1462 1475 1489 1494 1497 1503 1504 40499 40922 40981 41143 41168 44090 44091 44123 44125 44126 44131 44157 44158 44160 44161 45062 45714'"
      ]
     },
     "execution_count": 18,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "\" \".join(map(str, sorted(datalist)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "77fe58e0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "95176\n",
      "(95976, 5)\n"
     ]
    }
   ],
   "source": [
    "for model in [\"sslshuffling\"]: #, \"sslbinsampling\"]:\n",
    "    result = update(model)\n",
    "result = result[~result[\"data_id\"].isin([1492, 4153])]\n",
    "result[\"model\"] = result[\"model\"].replace(\"sslbinshuffling-lr\", \"sslbinshuffling\")\n",
    "result[\"model\"] = result[\"model\"].replace(\"sslbinsampling-lr\", \"sslbinsampling\")\n",
    "\n",
    "# result[result[\"model\"] == \"sslbinsampling\"].shape\n",
    "\n",
    "# result = update2(\"sslbinshuffling\", datalist=datalist, shots=[10, 20, 30], trials=range(100))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "id": "4b1b5674",
   "metadata": {},
   "outputs": [],
   "source": [
    "result = result[result[\"trial\"] < 10]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "32da2950",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "86fc4ddd",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.666132420189583\n",
      "0.6815132447427428\n"
     ]
    }
   ],
   "source": [
    "s = 30\n",
    "print(result[(result[\"shots\"] == s) & (result[\"model\"] == \"sslshuffling\")][\"acc\"].mean())\n",
    "print(result[(result[\"shots\"] == s) & (result[\"model\"] == \"sslbinshuffling\")][\"acc\"].mean())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "2aa5721a",
   "metadata": {},
   "outputs": [],
   "source": [
    "q = result.groupby([\"shots\", \"model\", \"data_id\"]).agg(\n",
    "    acc_mean=('acc', 'mean')).reset_index()\n",
    "q1 = q[(q[\"model\"] == \"mlp\") & (q[\"shots\"] == s)].sort_values(\"data_id\").reset_index(drop=True)\n",
    "q2 = q[(q[\"model\"] == \"sslbinshuffling\") & (q[\"shots\"] == s)].sort_values(\"data_id\").reset_index(drop=True)\n",
    "q1[\"diff\"] = q1[\"acc_mean\"] - q2[\"acc_mean\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "a8ddaf63",
   "metadata": {},
   "outputs": [],
   "source": [
    "# q1.sort_values(\"diff\", ascending=False)\n",
    "# q2.sort_values(\"acc_mean\")\n",
    "# q1.sort_values(\"acc_mean\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d989eba5",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 398,
   "id": "f5532c87",
   "metadata": {},
   "outputs": [],
   "source": [
    "# result = result[~result[\"model\"].str.startswith(\"sslbin\")]\n",
    "# result.to_csv(\"/home/SemiTab/result-summary/result-numshots.csv\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "248e3d5b",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "3bb1338b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([43, 44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 57, 58, 59,\n",
       "       60, 61, 62, 63, 64, 65, 66, 67, 68, 69, 70, 71, 72, 73, 74, 75])"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "dt = result.groupby([\"shots\", \"model\", \"trial\"]).agg(count=('acc', lambda x: x.notna().sum())).reset_index()\n",
    "# dt[(dt[\"count\"] < 40) & (dt[\"model\"] != \"sslbinsampling\")].head(5)\n",
    "dt[(dt[\"count\"] < 40) & (dt[\"model\"] == \"sslbinsampling\") & (dt[\"shots\"] == 10)][\"trial\"].values"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "affdacdf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "44161 45714\n"
     ]
    }
   ],
   "source": [
    "vv = result[(result[\"model\"] == \"sslshuffling\") & (result[\"shots\"] == 10) & (\n",
    "    result[\"trial\"] == 43)][\"data_id\"]\n",
    "vv = vv.tolist() + [1492, 4153]\n",
    "print(\" \".join(map(str, [v for v in datalist if v not in vv])))\n",
    "# print(vv)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 91,
   "id": "80592c94",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99\n"
     ]
    }
   ],
   "source": [
    "vv = result[(result[\"model\"] == \"sslbinsampling\") & (result[\"shots\"] == 10) & (\n",
    "    result[\"data_id\"] == 1503)][\"trial\"].values.tolist()\n",
    "print(\" \".join(map(str, [v for v in range(100) if v not in vv])))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "be449994",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "3af53f4c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>shots</th>\n",
       "      <th>model</th>\n",
       "      <th>acc_mean</th>\n",
       "      <th>acc_std</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>18</th>\n",
       "      <td>20</td>\n",
       "      <td>sslbinsampling</td>\n",
       "      <td>0.663222</td>\n",
       "      <td>0.169840</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>19</th>\n",
       "      <td>20</td>\n",
       "      <td>sslbinshuffling</td>\n",
       "      <td>0.652290</td>\n",
       "      <td>0.183541</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>17</th>\n",
       "      <td>20</td>\n",
       "      <td>mlp</td>\n",
       "      <td>0.650278</td>\n",
       "      <td>0.195172</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>16</th>\n",
       "      <td>20</td>\n",
       "      <td>lr</td>\n",
       "      <td>0.636427</td>\n",
       "      <td>0.175734</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>15</th>\n",
       "      <td>20</td>\n",
       "      <td>knn</td>\n",
       "      <td>0.588256</td>\n",
       "      <td>0.170446</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "    shots            model  acc_mean   acc_std\n",
       "18     20   sslbinsampling  0.663222  0.169840\n",
       "19     20  sslbinshuffling  0.652290  0.183541\n",
       "17     20              mlp  0.650278  0.195172\n",
       "16     20               lr  0.636427  0.175734\n",
       "15     20              knn  0.588256  0.170446"
      ]
     },
     "execution_count": 38,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "p[(p[\"shots\"] == 20)].sort_values(\"acc_mean\", ascending=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 92,
   "id": "d011c080",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "6ade4e01",
   "metadata": {},
   "outputs": [],
   "source": [
    "q1 = result.groupby([\"shots\", \"model\", \"data_id\"]).agg(\n",
    "    acc_mean=('acc', 'mean')).reset_index()\n",
    "\n",
    "import copy\n",
    "rankdata = copy.deepcopy(q1)\n",
    "\n",
    "for shot in [1, 5, 10, 20, 30]:\n",
    "    for data in datalist:\n",
    "        q = q1[(q1[\"data_id\"] == data) & (q1[\"shots\"] == shot)]\n",
    "        q = q.reset_index(drop=True)\n",
    "    \n",
    "        q['rank'] = q[\"acc_mean\"].rank(ascending=False, method=\"min\")\n",
    "\n",
    "        for i, row in q.iterrows():\n",
    "            rankdata.loc[(rankdata[\"data_id\"] == row[\"data_id\"]) & \n",
    "                         (rankdata[\"shots\"] == shot) & \n",
    "                         (rankdata[\"acc_mean\"] == row[\"acc_mean\"]), 'rank'] = row['rank']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1e32441c",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 114,
   "id": "0fdcb9ad",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([   23,    53,    54,   307,  1504, 41143, 44090, 44125, 44158,\n",
       "       44161])"
      ]
     },
     "execution_count": 114,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# rankdata[(rankdata[\"shots\"] == 10) & (rankdata[\"model\"] == \"sslbinshuffling\") & (rankdata[\"rank\"] > 4)][\"data_id\"].unique()\n",
    "rankdata[(rankdata[\"shots\"] == 10) & (rankdata[\"model\"] == \"mlp\") & (rankdata[\"rank\"] == 1)][\"data_id\"].unique()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 115,
   "id": "230fa550",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>shots</th>\n",
       "      <th>model</th>\n",
       "      <th>data_id</th>\n",
       "      <th>acc_mean</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>508</th>\n",
       "      <td>10</td>\n",
       "      <td>mlp</td>\n",
       "      <td>44090</td>\n",
       "      <td>0.727994</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>468</th>\n",
       "      <td>10</td>\n",
       "      <td>lr</td>\n",
       "      <td>44090</td>\n",
       "      <td>0.709222</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>588</th>\n",
       "      <td>10</td>\n",
       "      <td>sslbinshuffling</td>\n",
       "      <td>44090</td>\n",
       "      <td>0.687436</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>548</th>\n",
       "      <td>10</td>\n",
       "      <td>sslbinsampling</td>\n",
       "      <td>44090</td>\n",
       "      <td>0.678758</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>428</th>\n",
       "      <td>10</td>\n",
       "      <td>knn</td>\n",
       "      <td>44090</td>\n",
       "      <td>0.662663</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "     shots            model  data_id  acc_mean\n",
       "508     10              mlp    44090  0.727994\n",
       "468     10               lr    44090  0.709222\n",
       "588     10  sslbinshuffling    44090  0.687436\n",
       "548     10   sslbinsampling    44090  0.678758\n",
       "428     10              knn    44090  0.662663"
      ]
     },
     "execution_count": 115,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "q1[(q1[\"shots\"] == 10) & (q1[\"data_id\"] == 44090)].sort_values(\"acc_mean\", ascending=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1c95cf7b",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9a0e1d3f",
   "metadata": {},
   "outputs": [],
   "source": [
    "result = pd.read_csv(\"/home/SemiTab/result-summary/result.csv\", index_col=0)\n",
    "result = result[result[\"model\"].isin([\n",
    "    \"sslbinshuffling-lr\", \"sslbinsampling-lr\", \"sslshuffling-lr\"])].reset_index(drop=True)\n",
    "result = result[[\"trial\", \"shots\", \"data_id\", \"model\", \"acc\"]]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "35e27932",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 371,
   "id": "14c20256",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 38,
   "id": "c72188b5",
   "metadata": {},
   "outputs": [],
   "source": [
    "p = result[result[\"trial\"] < 10].groupby([\"shots\", \"model\"]).agg(\n",
    "    acc_mean=('acc', 'mean'), acc_std=('acc', 'std')).reset_index()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 40,
   "id": "e20dd792",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 288x216 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "labels = {\"lr\": \"LR\", \"knn\": \"kNN\", \"mlp\": \"MLP\", \n",
    "          \"sslshuffling\": \"CL+Shuffling\",\n",
    "          \"sslbinshuffling\": \"Ours\", #\"CL+Range-limited Shuffling\", \n",
    "          \"sslbinsampling\": \"CL+Range-limited Sampling\"}\n",
    "\n",
    "plt.figure(figsize=(4, 3))\n",
    "plt.gca().set_axisbelow(True)\n",
    "plt.grid()\n",
    "for model in [\"lr\", \"knn\", \"mlp\", \"sslshuffling\", \"sslbinshuffling\"]: #, \"sslbinsampling\"]:\n",
    "    q = p[p[\"model\"] == model]\n",
    "    plt.plot(q[\"shots\"], q[\"acc_mean\"], label=labels[model], marker=\"o\")\n",
    "plt.legend(loc=\"lower right\")\n",
    "plt.xlabel(\"Shots\", fontsize=12)\n",
    "plt.ylabel(\"Average Accuracy\", fontsize=12)\n",
    "plt.savefig(\"figs/discussion-shots.png\", dpi=300, bbox_inches=\"tight\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "829440c8",
   "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.8.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
