{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import warnings\n",
    "warnings.filterwarnings(\"ignore\")\n",
    "\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "from sklearn.metrics import f1_score,roc_auc_score\n",
    "from sklearn.svm import SVC\n",
    "\n",
    "\n",
    "from tqdm import tqdm\n",
    "import torch\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "from vit import ViT\n",
    "import os\n",
    "\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "\n",
    "signals = pd.read_csv(\"Eval/ptb_xl_signals_II.csv\")\n",
    "annos = pd.read_csv(\"Eval/ptb_xl_annos.csv\")\n",
    "\n",
    "signals = np.array(signals)\n",
    "annos = np.array(annos)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "NUM_BLOCKS = 4\n",
    "MODEL_DIM = 128\n",
    "NUM_HEADS = 4\n",
    "PATCH_SIZE = 10\n",
    "FS = 100\n",
    "L = 10\n",
    "C_IN = 1\n",
    "\n",
    "\n",
    "model = ViT(num_blocks=NUM_BLOCKS, num_heads=NUM_HEADS, model_dim=MODEL_DIM, \n",
    "            patch_size=PATCH_SIZE, in_channels=C_IN, fs=FS, l=L, do_prob=0.1)\n",
    "\n",
    "model = model.to(device)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# For Main Evaluation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      " Eval: clocs.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [31:13<00:00, 187.31s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6465303095698814\n",
      "0.012097354073528329\n",
      "\n",
      "F1: \n",
      "0.38524497715450756\n",
      "0.01661952746804477\n",
      "AUC\n",
      "0.7552356267618462\n",
      "0.011275302122963874\n",
      "\n",
      " Eval: cupid_0.5.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [25:54<00:00, 155.42s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.7096882931294679\n",
      "0.010963184108511596\n",
      "\n",
      "F1: \n",
      "0.4871299541109139\n",
      "0.015115744395635108\n",
      "AUC\n",
      "0.7999965095372817\n",
      "0.009870448766056882\n",
      "\n",
      " Eval: deaps.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [25:32<00:00, 153.23s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.7003661073773354\n",
      "0.011193691766760704\n",
      "\n",
      "F1: \n",
      "0.475763771344646\n",
      "0.018706053883964163\n",
      "AUC\n",
      "0.7963249443562997\n",
      "0.009915245473843922\n",
      "\n",
      " Eval: jepa_0.5.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [26:02<00:00, 156.21s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6777460858208345\n",
      "0.010211773137060268\n",
      "\n",
      "F1: \n",
      "0.4448053477747858\n",
      "0.017082723706213894\n",
      "AUC\n",
      "0.7736770771771789\n",
      "0.01054134882631811\n",
      "\n",
      " Eval: mae_0.5.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [24:33<00:00, 147.35s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6897598645745657\n",
      "0.011284138258234098\n",
      "\n",
      "F1: \n",
      "0.4617064125903365\n",
      "0.016013538446085624\n",
      "AUC\n",
      "0.7937008035894229\n",
      "0.01192434328910228\n",
      "\n",
      " Eval: mix_up.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [27:06<00:00, 162.70s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6605394867612281\n",
      "0.011638475845081453\n",
      "\n",
      "F1: \n",
      "0.41894188693283796\n",
      "0.017399316296437272\n",
      "AUC\n",
      "0.760882627413799\n",
      "0.01267353963104626\n",
      "\n",
      " Eval: pclr.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [28:19<00:00, 169.98s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6791479465496417\n",
      "0.010247894436270789\n",
      "\n",
      "F1: \n",
      "0.4456560802774551\n",
      "0.015005402082155821\n",
      "AUC\n",
      "0.7767460505686192\n",
      "0.011752196209865355\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "for weights in os.listdir(\"ViT Trained Models\"):\n",
    "    print(\"\\n Eval: \" + weights)\n",
    "    weighs_path = os.path.join(\"ViT Trained Models\", weights)\n",
    "    model.load_state_dict(torch.load(weighs_path))\n",
    "    model.eval()\n",
    "    print(\"Model Loaded\")\n",
    "\n",
    "\n",
    "\n",
    "    final_results = []\n",
    "    final_y = []\n",
    "    final_probs  = []\n",
    "\n",
    "    fold_accus = []\n",
    "    fold_f1 = []\n",
    "    fold_auc  = []\n",
    "\n",
    "\n",
    "    features = []\n",
    "    bs = 512\n",
    "\n",
    "    for i in range(0, signals.shape[0], bs):\n",
    "        tmp_signal = signals[i : i + bs, :]\n",
    "\n",
    "        with torch.no_grad():\n",
    "            tmp_strip = torch.tensor(tmp_signal).float().to(device)\n",
    "            tmp_features = model(tmp_strip).squeeze().detach().cpu().numpy()\n",
    "            features.extend(tmp_features)\n",
    "\n",
    "    features = np.array(features)\n",
    "\n",
    "\n",
    "    for fold in tqdm(range(1, 11)):\n",
    "\n",
    "        svc = SVC(gamma='auto', probability=True)\n",
    "\n",
    "        train_index = [i for i in range(annos.shape[0]) if annos[i, 2] != fold]\n",
    "        test_index = [i for i in range(annos.shape[0]) if annos[i, 2] == fold]\n",
    "\n",
    "        train_features = features[train_index, :]\n",
    "        train_annos = annos[train_index, 0]\n",
    "\n",
    "        test_features = features[test_index, :]\n",
    "        test_annos = annos[test_index, 0]\n",
    "\n",
    "        y_train = np.array(train_annos, dtype = np.int32)\n",
    "        y_val = np.array(test_annos, dtype = np.int32)\n",
    "\n",
    "        \n",
    "        scaler = StandardScaler()\n",
    "        train_features = scaler.fit_transform(train_features)\n",
    "        test_features = scaler.transform(test_features)\n",
    "\n",
    "        svc.fit(train_features, y_train)\n",
    "        pred = svc.predict(test_features)\n",
    "        probs = svc.predict_proba(test_features)\n",
    "\n",
    "        fold_accus.append(np.mean(np.array(y_val) == np.array(pred)))\n",
    "        fold_f1.append(f1_score(y_val, pred, average='macro'))  \n",
    "        fold_auc.append(roc_auc_score(y_val, probs, multi_class='ovr'))\n",
    "\n",
    "\n",
    "        final_results.extend(pred)\n",
    "        final_y.extend(y_val)\n",
    "        final_probs.extend(probs)\n",
    "\n",
    "    print(\"Accuracy: \")\n",
    "    print(np.mean(fold_accus))\n",
    "    print(np.std(fold_accus))\n",
    "    print(\"\")\n",
    "\n",
    "    print(\"F1: \")\n",
    "    print(np.mean(fold_f1))\n",
    "    print(np.std(fold_f1))\n",
    "\n",
    "    print(\"AUC\")\n",
    "    print(np.mean(fold_auc))\n",
    "    print(np.std(fold_auc))  "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# For Ablation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      " Eval: cupid_0.4.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [04:23<00:00, 26.36s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.7087821189722171\n",
      "0.013318631416230695\n",
      "\n",
      "F1: \n",
      "0.48591786033013645\n",
      "0.018763311263299786\n",
      "\n",
      " Eval: cupid_0.6.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [04:34<00:00, 27.46s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.7077603059694014\n",
      "0.012783966152518042\n",
      "\n",
      "F1: \n",
      "0.48472726819266043\n",
      "0.01853918897264641\n",
      "\n",
      " Eval: jepa_0.4.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [04:57<00:00, 29.77s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.689389158846953\n",
      "0.009219138513682942\n",
      "\n",
      "F1: \n",
      "0.45783014891879786\n",
      "0.013839369763546725\n",
      "\n",
      " Eval: jepa_0.6.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [05:23<00:00, 32.36s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6608814614075482\n",
      "0.010764724843715589\n",
      "\n",
      "F1: \n",
      "0.402377706552784\n",
      "0.01664682821155946\n",
      "\n",
      " Eval: jepa_spec_0.4.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [04:51<00:00, 29.13s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6938599501424983\n",
      "0.009880603967374831\n",
      "\n",
      "F1: \n",
      "0.4665426867333958\n",
      "0.015257393399893855\n",
      "\n",
      " Eval: jepa_spec_0.5.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [04:51<00:00, 29.15s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6947252306717545\n",
      "0.010733567852536444\n",
      "\n",
      "F1: \n",
      "0.46340547690983963\n",
      "0.016530246614840597\n",
      "\n",
      " Eval: jepa_spec_0.6.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [04:51<00:00, 29.16s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6839564580793349\n",
      "0.015331008760165748\n",
      "\n",
      "F1: \n",
      "0.45478442060554103\n",
      "0.022190167264569193\n",
      "\n",
      " Eval: mae_0.4.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [05:01<00:00, 30.15s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6870085001635481\n",
      "0.012432548843854094\n",
      "\n",
      "F1: \n",
      "0.455859781750267\n",
      "0.01550836925318474\n",
      "\n",
      " Eval: mae_0.6.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [05:01<00:00, 30.18s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6790417257826051\n",
      "0.015523883188212793\n",
      "\n",
      "F1: \n",
      "0.44514209154979045\n",
      "0.01875378991522289\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "for weights in os.listdir(\"ViT Models Ablation\"):\n",
    "    print(\"\\n Eval: \" + weights)\n",
    "    weighs_path = os.path.join(\"ViT Models Ablation\", weights)\n",
    "    model.load_state_dict(torch.load(weighs_path))\n",
    "    model.eval()\n",
    "    print(\"Model Loaded\")\n",
    "\n",
    "\n",
    "\n",
    "    final_results = []\n",
    "    final_y = []\n",
    "    final_probs  = []\n",
    "\n",
    "    fold_accus = []\n",
    "    fold_f1 = []\n",
    "    fold_auc  = []\n",
    "\n",
    "\n",
    "    features = []\n",
    "    bs = 512\n",
    "\n",
    "    for i in range(0, signals.shape[0], bs):\n",
    "        tmp_signal = signals[i : i + bs, :]\n",
    "\n",
    "        with torch.no_grad():\n",
    "            tmp_strip = torch.tensor(tmp_signal).float().to(device)\n",
    "            tmp_features = model(tmp_strip).squeeze().detach().cpu().numpy()\n",
    "            features.extend(tmp_features)\n",
    "\n",
    "    features = np.array(features)\n",
    "\n",
    "\n",
    "    for fold in tqdm(range(1, 11)):\n",
    "\n",
    "        svc = SVC(gamma='auto')\n",
    "\n",
    "        train_index = [i for i in range(annos.shape[0]) if annos[i, 2] != fold]\n",
    "        test_index = [i for i in range(annos.shape[0]) if annos[i, 2] == fold]\n",
    "\n",
    "        train_features = features[train_index, :]\n",
    "        train_annos = annos[train_index, 0]\n",
    "\n",
    "        test_features = features[test_index, :]\n",
    "        test_annos = annos[test_index, 0]\n",
    "\n",
    "        y_train = np.array(train_annos, dtype = np.int32)\n",
    "        y_val = np.array(test_annos, dtype = np.int32)\n",
    "\n",
    "        \n",
    "        scaler = StandardScaler()\n",
    "        train_features = scaler.fit_transform(train_features)\n",
    "        test_features = scaler.transform(test_features)\n",
    "\n",
    "        svc.fit(train_features, y_train)\n",
    "        pred = svc.predict(test_features)\n",
    "\n",
    "\n",
    "        fold_accus.append(np.mean(np.array(y_val) == np.array(pred)))\n",
    "        fold_f1.append(f1_score(y_val, pred, average='macro'))  \n",
    "\n",
    "        final_results.extend(pred)\n",
    "        final_y.extend(y_val)\n",
    "\n",
    "    print(\"Accuracy: \")\n",
    "    print(np.mean(fold_accus))\n",
    "    print(np.std(fold_accus))\n",
    "    print(\"\")\n",
    "\n",
    "    print(\"F1: \")\n",
    "    print(np.mean(fold_f1))\n",
    "    print(np.std(fold_f1))\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# For Icentia Ablation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      " Eval: cupid_icentia.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [04:24<00:00, 26.49s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.7065675335610427\n",
      "0.01350397867124158\n",
      "\n",
      "F1: \n",
      "0.4830480160922067\n",
      "0.01693301824966694\n",
      "\n",
      " Eval: jepa_icentia.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [04:42<00:00, 28.29s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6872350246306558\n",
      "0.01202336165963003\n",
      "\n",
      "F1: \n",
      "0.45105160962311697\n",
      "0.014074817316313751\n",
      "\n",
      " Eval: jepa_spec_icentia.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [04:39<00:00, 27.96s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6937927807942419\n",
      "0.012745198619085527\n",
      "\n",
      "F1: \n",
      "0.46258336121022037\n",
      "0.01673719027030332\n",
      "\n",
      " Eval: mae_icentia.pth\n",
      "Model Loaded\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [04:43<00:00, 28.39s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy: \n",
      "0.6937927807942419\n",
      "0.012745198619085527\n",
      "\n",
      "F1: \n",
      "0.46258336121022037\n",
      "0.01673719027030332\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "from vit import ViT\n",
    "\n",
    "\n",
    "NUM_BLOCKS = 4\n",
    "MODEL_DIM = 128\n",
    "NUM_HEADS = 4\n",
    "PATCH_SIZE = 10\n",
    "FS = 100\n",
    "L = 10\n",
    "C_IN = 1\n",
    "\n",
    "\n",
    "model = ViT(num_blocks=NUM_BLOCKS, num_heads=NUM_HEADS, model_dim=MODEL_DIM, \n",
    "            patch_size=PATCH_SIZE, in_channels=C_IN, fs=FS, l=L, do_prob=0.1)\n",
    "\n",
    "model = model.to(device)\n",
    "\n",
    "\n",
    "for weights in os.listdir(\"ViT Icentia\"):\n",
    "    print(\"\\n Eval: \" + weights)\n",
    "    weighs_path = os.path.join(\"ViT Icentia\", weights)\n",
    "    model.load_state_dict(torch.load(weighs_path))\n",
    "    model.eval()\n",
    "    print(\"Model Loaded\")\n",
    "\n",
    "\n",
    "\n",
    "    final_results = []\n",
    "    final_y = []\n",
    "    final_probs  = []\n",
    "\n",
    "    fold_accus = []\n",
    "    fold_f1 = []\n",
    "    fold_auc  = []\n",
    "\n",
    "\n",
    "    features = []\n",
    "    bs = 512\n",
    "\n",
    "    for i in range(0, signals.shape[0], bs):\n",
    "        tmp_signal = signals[i : i + bs, :]\n",
    "\n",
    "        with torch.no_grad():\n",
    "            tmp_strip = torch.tensor(tmp_signal).float().to(device)\n",
    "            tmp_features = model(tmp_strip).squeeze().detach().cpu().numpy()\n",
    "            features.extend(tmp_features)\n",
    "\n",
    "    features = np.array(features)\n",
    "\n",
    "\n",
    "    for fold in tqdm(range(1, 11)):\n",
    "\n",
    "        svc = SVC(gamma='auto')\n",
    "\n",
    "        train_index = [i for i in range(annos.shape[0]) if annos[i, 2] != fold]\n",
    "        test_index = [i for i in range(annos.shape[0]) if annos[i, 2] == fold]\n",
    "\n",
    "        train_features = features[train_index, :]\n",
    "        train_annos = annos[train_index, 0]\n",
    "\n",
    "        test_features = features[test_index, :]\n",
    "        test_annos = annos[test_index, 0]\n",
    "\n",
    "        y_train = np.array(train_annos, dtype = np.int32)\n",
    "        y_val = np.array(test_annos, dtype = np.int32)\n",
    "\n",
    "        \n",
    "        scaler = StandardScaler()\n",
    "        train_features = scaler.fit_transform(train_features)\n",
    "        test_features = scaler.transform(test_features)\n",
    "\n",
    "        svc.fit(train_features, y_train)\n",
    "        pred = svc.predict(test_features)\n",
    "\n",
    "        # probs = svc.predict_proba(test_features)\n",
    "\n",
    "        fold_accus.append(np.mean(np.array(y_val) == np.array(pred)))\n",
    "        fold_f1.append(f1_score(y_val, pred, average='macro'))  \n",
    "        # fold_auc.append(roc_auc_score(y_val, probs, multi_class='ovr'))\n",
    "\n",
    "\n",
    "        # print(accuracy_score(pred, y_val))\n",
    "\n",
    "        final_results.extend(pred)\n",
    "        final_y.extend(y_val)\n",
    "        # final_probs.extend(probs)\n",
    "\n",
    "    print(\"Accuracy: \")\n",
    "    print(np.mean(fold_accus))\n",
    "    print(np.std(fold_accus))\n",
    "    print(\"\")\n",
    "\n",
    "    print(\"F1: \")\n",
    "    print(np.mean(fold_f1))\n",
    "    print(np.std(fold_f1))"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "pytorch",
   "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.7.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
