{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/opt/conda/lib/python3.8/site-packages/tqdm/auto.py:22: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
      "  from .autonotebook import tqdm as notebook_tqdm\n",
      "WARNING:root:Cuda kernels could not loaded -> no CUDA support!\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "import yaml\n",
    "from ml_collections import ConfigDict\n",
    "from tqdm import tqdm\n",
    "\n",
    "import torch\n",
    "import torch_geometric\n",
    "\n",
    "from utils.data import load_dataset, make_dataset_splits, load_dataset_splits\n",
    "from utils.split import SplitManager, node_induced_subgraph\n",
    "from utils.storage import TensorHash\n",
    "from utils.model import load_model_class, accuracy, load_model_instance, create_model_instance\n",
    "from utils.attack import load_attack_class\n",
    "\n",
    "from robust_diffusion.data import SparseGraph\n",
    "from robust_diffusion.data import count_edges_for_idx\n",
    "from robust_diffusion.helper import utils as robust_utils\n",
    "from robust_diffusion.train import train"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "## Experiment configs\n",
    "dataset_name = \"cora_ml\"\n",
    "model_name = \"GCN\"\n",
    "recreate_splits = False\n",
    "n_splits = 50\n",
    "\n",
    "training_split = None\n",
    "validation_split = None\n",
    "training_split_type = None\n",
    "validation_split_type = None\n",
    "\n",
    "model_params = None\n",
    "epsilon = 0.1\n",
    "\n",
    "attack_name = \"PRBCD\"\n",
    "attack_params = None\n",
    "\n",
    "inductive = False"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "n_splits=5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Experiment Started\n",
      "Loading dataset = cora_ml\n"
     ]
    }
   ],
   "source": [
    "\n",
    "## Loading general configs (like dataset_root, etc.) and initial parameters\n",
    "general_config = yaml.safe_load(open(\"conf/general-config.yaml\"))\n",
    "default_dataset_configs = yaml.safe_load(open(\"conf/data-configs.yaml\")).get(\"configs\").get(\"default\")\n",
    "default_model_configs = yaml.safe_load(open(\"conf/model-configs.yaml\")).get(\"configs\")\n",
    "default_attack_configs = yaml.safe_load(open(\"conf/attack-configs.yaml\")).get(\"configs\")\n",
    "\n",
    "# extracting configs \n",
    "dataset_root = general_config.get(\"dataset_root\", \"data/\")\n",
    "splits_root = general_config.get(\"splits_root\", \"splits/\")\n",
    "models_root = general_config.get(\"models_root\", \"models/\")\n",
    "results_root = general_config.get(\"results_root\", \"results/\")\n",
    "reports_root = general_config.get(\"reports_root\", \"reports/\")\n",
    "    \n",
    "device = 'cuda' if torch.cuda.is_available() else 'cpu'\n",
    "\n",
    "\n",
    "print(\"Experiment Started\")\n",
    "# Trains the specified model on the given graph and saves the model artifacts, and the splits.\n",
    "\n",
    "print(\"Loading dataset =\", dataset_name)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Found 5 splits, creating 0 more splits\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "0it [00:00, ?it/s]\n"
     ]
    }
   ],
   "source": [
    "dataset_splits = [split_record for split_record in os.listdir(splits_root) if split_record.split(\"-\")[0] == dataset_name]\n",
    "creating_splits = max(n_splits - len(dataset_splits), 0)\n",
    "\n",
    "# creating remaining needed dataset splits\n",
    "print(f\"Found {len(dataset_splits)} splits, creating {creating_splits} more splits\")\n",
    "for i in tqdm(range(creating_splits)):\n",
    "    torch.cuda.empty_cache()\n",
    "    data = make_dataset_splits(dataset_name=dataset_name, \n",
    "                            training_split=training_split, validation_split=validation_split, \n",
    "                            training_split_type=training_split_type, validation_split_type=validation_split_type, \n",
    "                            inductive=inductive, \n",
    "                            default_dataset_configs=default_dataset_configs, dataset_root=dataset_root, splits_root=splits_root, device=device)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Training GCN model on cora_ml dataset for 5 splits\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|          | 0/5 [00:00<?, ?it/s]/opt/conda/lib/python3.8/site-packages/torch_geometric/data/in_memory_dataset.py:157: UserWarning: It is not recommended to directly access the internal storage format `data` of an 'InMemoryDataset'. If you are absolutely certain what you are doing, access the internal storage via `InMemoryDataset._data` instead to suppress this warning. Alternatively, you can access stacked individual attributes of every graph via `dataset.{attr_name}`.\n",
      "  warnings.warn(msg)\n",
      "100%|██████████| 5/5 [00:02<00:00,  2.24it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Mean accuracy: 0.81517493724823, std: 0.01838192529976368\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "dataset_splits = [split_record for split_record in os.listdir(splits_root) if split_record.split(\"-\")[0] == dataset_name][:n_splits]\n",
    "print(f\"Training {model_name} model on {dataset_name} dataset for {n_splits} splits\")\n",
    "\n",
    "accs = []\n",
    "for split_file in tqdm(dataset_splits):\n",
    "    split_code = split_file.split(\"-\")[1].replace(\".pt\", \"\")\n",
    "\n",
    "    data = load_dataset_splits(\n",
    "        dataset_name, split_code, inductive=inductive, \n",
    "        dataset_root=dataset_root, splits_root=splits_root, device=device)\n",
    "\n",
    "    training_attr = data[\"training_attr\"]\n",
    "    training_adj = data[\"training_adj\"]\n",
    "    labels = data[\"labels\"]\n",
    "    training_idx = data[\"training_idx\"]\n",
    "    validation_idx = data[\"validation_idx\"]\n",
    "    test_attr = data[\"test_attr\"]\n",
    "    test_adj = data[\"test_adj\"]\n",
    "    test_mask = data[\"test_mask\"]\n",
    "    dataset_info = data[\"dataset_info\"]\n",
    "    split_name = data[\"split_name\"]\n",
    "\n",
    "    try:\n",
    "        model_instance = load_model_instance(\n",
    "            model_name=model_name, model_params=model_params, \n",
    "            test_attr=test_attr, test_adj=test_adj, labels=labels, test_mask=test_mask, split_name=split_name, dataset_info=dataset_info, inductive=inductive,\n",
    "            models_root=models_root,\n",
    "            default_model_configs=default_model_configs, device=device)\n",
    "    except FileNotFoundError as e:\n",
    "        print(e)\n",
    "        print(\"Creating model from scratch\")\n",
    "        model_instance = create_model_instance(\n",
    "            model_name=model_name, model_params=model_params, dataset_info=dataset_info, \n",
    "            training_attr=training_attr, training_adj=training_adj, labels=labels, training_idx=training_idx, validation_idx=validation_idx,\n",
    "            test_attr=test_attr, test_adj=test_adj, test_mask=test_mask, inductive=inductive, split_name=split_name,\n",
    "            models_root=models_root, \n",
    "            default_model_configs=default_model_configs, \n",
    "            device=device)\n",
    "\n",
    "    model = model_instance[\"model\"]\n",
    "    acc = model_instance[\"accuracy\"]\n",
    "    model_params = model_instance[\"model_params\"]\n",
    "    model_storage_name = model_instance[\"model_storage_name\"]\n",
    "    accs.append(acc)\n",
    "\n",
    "acc_mean = torch.mean(torch.tensor(accs))\n",
    "acc_std = torch.std(torch.tensor(accs))\n",
    "\n",
    "print(f\"Mean accuracy: {acc_mean}, std: {acc_std}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'0ae4e175b2'"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "TensorHash().hash_model_params(\"GCN\", model_params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "lr: 0.01\n",
       "max_epochs: 3000\n",
       "n_classes: 7\n",
       "n_features: 2879\n",
       "n_filters: 64\n",
       "patience: 200\n",
       "weight_decay: 0.001"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model_params"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'[\"GCN\", {\"lr\": 0.01, \"max_epochs\": 3000, \"n_classes\": 7, \"n_features\": 2879, \"n_filters\": 64, \"patience\": 200, \"weight_decay\": 0.001}]'"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import json\n",
    "json.dumps([model_name, model_params.to_dict()])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'0ae4e175b2'"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "TensorHash().hash_model_params(\"GCN\", model_params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'[\"GCN\", {\"lr\": 0.01, \"max_epochs\": 3000, \"n_classes\": 7, \"n_features\": 2879, \"n_filters\": 64, \"patience\": 200, \"weight_decay\": 0.001}]'"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import json\n",
    "json.dumps([model_name, model_params.to_dict()])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "True"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "'[\"GCN\", {\"lr\": 0.01, \"max_epochs\": 3000, \"n_classes\": 7, \"n_features\": 2879, \"n_filters\": 64, \"patience\": 200, \"weight_decay\": 0.001}]' == '[\"GCN\", {\"lr\": 0.01, \"max_epochs\": 3000, \"n_classes\": 7, \"n_features\": 2879, \"n_filters\": 64, \"patience\": 200, \"weight_decay\": 0.001}]'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'0x2bce7824369f93ab'"
      ]
     },
     "execution_count": 13,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "hex(abs(hash(\"mamad\")))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'0x4990b5fea40a4b05'"
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "hex(abs(hash(\"mamad\")))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[0;31mSignature:\u001b[0m \u001b[0mhash\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mobj\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m/\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mDocstring:\u001b[0m\n",
      "Return the hash value for the given object.\n",
      "\n",
      "Two objects that compare equal must also have the same hash value, but the\n",
      "reverse is not necessarily true.\n",
      "\u001b[0;31mType:\u001b[0m      builtin_function_or_method\n"
     ]
    }
   ],
   "source": [
    "hash?"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "import hashlib"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'51a23605b6'"
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "hex(int(hashlib.sha256(\"mamad\".encode()).hexdigest(), 16))[-10:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "['cora_ml_0x4282f4c8d6664093.pt',\n",
       " 'cora_ml_0x783050dc9e05033d.pt',\n",
       " 'cora_ml_0x5bb188d276063ba2.pt',\n",
       " 'cora_ml_0x3040d782ac323ca3.pt',\n",
       " 'cora_ml_0x403dfec936ccb112.pt',\n",
       " 'cora_ml_0x790cafb76908c696.pt',\n",
       " 'cora_ml_0x76b6eca487a4b65b.pt',\n",
       " 'cora_ml_0x1c7f0f28c14320.pt',\n",
       " 'cora_ml_0x4efe51b865a0947d.pt',\n",
       " 'cora_ml_0x28a825bd30ba6645.pt',\n",
       " 'cora_ml-0x2aaeb60e1d1bc7b8.pt',\n",
       " 'cora_ml_0x77c40ec4e714a5d3.pt',\n",
       " 'cora_ml_0x399d1feafcb8b5cb.pt',\n",
       " 'cora_ml_0x773b32cb00631a3f.pt',\n",
       " 'cora_ml_0x1f8a49c5ad8b8640.pt',\n",
       " 'cora_ml_0x67d6c795effc3d90.pt',\n",
       " 'cora_ml_0x75c8d1ff6cfa92ce.pt',\n",
       " 'cora_ml_0x72cb3105726b8a4.pt',\n",
       " 'cora_ml_0x2a1842326d6818bc.pt',\n",
       " 'cora_ml_0x41fe821fa16b8577.pt',\n",
       " 'cora_ml_0x21cdf494f4785f06.pt',\n",
       " 'cora_ml_0x3c8e4aff4f1d6fe9.pt',\n",
       " 'cora_ml_0xe39194776b68492.pt',\n",
       " 'cora_ml_0x7c6841345166f78e.pt',\n",
       " 'cora_ml_0x4f6f92348a63ddcb.pt',\n",
       " 'cora_ml_0x32cd22775819fdc6.pt',\n",
       " 'cora_ml_0x5aa602f8f77c86ab.pt',\n",
       " 'cora_ml_0x90fa01f5a76969f.pt',\n",
       " 'cora_ml_0x66a03ee16f021e6f.pt',\n",
       " 'cora_ml-0x3c1f44bf1ddec0f4.pt',\n",
       " 'cora_ml_0x6495456c8be6c51c.pt',\n",
       " 'cora_ml_0x5ad255f0fb06e7f8.pt',\n",
       " 'cora_ml_0x25771787074afae6.pt',\n",
       " 'cora_ml_0x63091e89a8b23f07.pt',\n",
       " 'cora_ml_0xe043a8487dbfe6a.pt',\n",
       " 'cora_ml_0x4b12f5ec781bc45e.pt',\n",
       " 'cora_ml_0x4f33a46178790e87.pt',\n",
       " 'cora_ml_0x34e1af2f4b69139c.pt',\n",
       " 'cora_ml_0x70c8574c25d9bf8f.pt',\n",
       " 'cora_ml_0x451516faed14e999.pt',\n",
       " 'cora_ml_0x31298c179918a5b0.pt',\n",
       " 'cora_ml_0x3e3c9e95527585b3.pt',\n",
       " 'cora_ml_0x15bb53329f05fcee.pt',\n",
       " 'cora_ml_0x4549516858c82680.pt',\n",
       " 'cora_ml_0x372d9cb4e6bb040a.pt',\n",
       " 'cora_ml_0x4a0769eef4e4dae7.pt',\n",
       " 'cora_ml_0x4cd92589bde7f68f.pt',\n",
       " 'cora_ml_0x438cc9e7c416e647.pt',\n",
       " 'cora_ml_0x5ecce5f708aa8372.pt',\n",
       " 'cora_ml_0x33e91bebf090b83c.pt']"
      ]
     },
     "execution_count": 23,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "[split_record for split_record in os.listdir(splits_root)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "['cora_ml-0x2aaeb60e1d1bc7b8.pt', 'cora_ml-0x3c1f44bf1ddec0f4.pt']"
      ]
     },
     "execution_count": 22,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "[split_record for split_record in os.listdir(splits_root) if split_record.split(\"-\")[0] == dataset_name]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "# Loading the dataset, creating splits, and saving them (for both transductive and inductive)\n",
    "\n",
    "# data = load_dataset_splits(dataset_name, \n",
    "#                         \"0x30d202b2fcf2b06\",\n",
    "#                             inductive=inductive, dataset_root=dataset_root, splits_root=splits_root, device=device)\n",
    "training_attr = data[\"training_attr\"]\n",
    "training_adj = data[\"training_adj\"]\n",
    "labels = data[\"labels\"]\n",
    "training_idx = data[\"training_idx\"]\n",
    "validation_idx = data[\"validation_idx\"]\n",
    "test_attr = data[\"test_attr\"]\n",
    "test_adj = data[\"test_adj\"]\n",
    "test_mask = data[\"test_mask\"]\n",
    "dataset_info = data[\"dataset_info\"]\n",
    "split_name = data[\"split_name\"]\n",
    "\n",
    "\n",
    "# Loading and training the model\n",
    "model_instance = create_model_instance(\n",
    "    model_name=model_name, model_params=model_params, dataset_info=dataset_info, \n",
    "    training_attr=training_attr, training_adj=training_adj, labels=labels, training_idx=training_idx, validation_idx=validation_idx,\n",
    "    test_attr=test_attr, test_adj=test_adj, test_mask=test_mask, inductive=inductive, split_name=split_name,\n",
    "    models_root=models_root, \n",
    "    default_model_configs=default_model_configs, \n",
    "    device=device)\n",
    "\n",
    "# model_instance = load_model_instance(\n",
    "#     model_storage_name='GCN-0xbd14035a4016352-tr-cora_ml-0x30d202b2fcf2b06', \n",
    "#     model_name=model_name, model_params=model_params, \n",
    "#     test_attr=test_attr, test_adj=test_adj, labels=labels, test_mask=test_mask, dataset_info=dataset_info, inductive=inductive,\n",
    "#     models_root=models_root,\n",
    "#     default_model_configs=default_model_configs, device=device)\n",
    "\n",
    "model = model_instance[\"model\"]\n",
    "acc = model_instance[\"accuracy\"]\n",
    "print(\"Accuracy (Clean): \", acc)\n",
    "model_params = model_instance[\"model_params\"]\n",
    "model_storage_name = model_instance[\"model_storage_name\"]\n",
    "\n",
    "idx_attack = test_mask.nonzero(as_tuple=True)[0].cpu().numpy()\n",
    "n_feasible_edges = count_edges_for_idx(test_adj.cpu(), idx_attack) / (2)\n",
    "n_attack_edges = (n_feasible_edges * epsilon).int().item()\n",
    "\n",
    "if attack_params is None:\n",
    "    attack_params = ConfigDict(default_attack_configs.get(attack_name))\n",
    "attack_params.device = device\n",
    "adversary = load_attack_class(attack_name)(\n",
    "    attr=test_attr, adj=test_adj, labels=labels, model=model, \n",
    "    idx_attack=test_mask.nonzero(as_tuple=True)[0].cpu().numpy(),\n",
    "    data_device=device, make_undirected=True, binary_attr=False,\n",
    "    **attack_params.to_dict())\n",
    "adversary.attack(n_attack_edges)\n",
    "pert_adj, pert_attr = adversary.get_pertubations()\n",
    "adv_acc = accuracy(model, pert_attr, pert_adj, labels, test_mask)\n",
    "\n",
    "attack_config_hash = TensorHash.hash_model_params(model_name=attack_name, model_params=attack_params)\n",
    "epsilon_str = str(epsilon).replace(\".\", \"_\")\n",
    "attack_storage_name = f\"{attack_name}-eps{epsilon_str}-{attack_config_hash}-{model_storage_name}\"\n",
    "\n",
    "try:\n",
    "    os.makedirs(results_root, exist_ok=True)\n",
    "    torch.save(pert_adj, os.path.join(results_root, f\"{attack_storage_name}-adj.pt\"))\n",
    "except RuntimeError as e:\n",
    "    print(f\"Error saving perturbed adj: {e}\")\n",
    "\n",
    "print(\"Accuracy (Perturbed):\", adv_acc)\n",
    "report = {\n",
    "    \"model\": model_name,\n",
    "    \"dataset\": dataset_name,\n",
    "    \"dataset_info\": dataset_info.to_dict(),\n",
    "    \"model_params\": model_params,\n",
    "    \"clean_accuracy\": acc,\n",
    "    \"setting\": \"inductive\" if inductive else \"transductive\",\n",
    "    \"attack\": attack_name,\n",
    "    \"attack_params\": attack_params,\n",
    "    \"attack_accuracy\": adv_acc,\n",
    "    \"epsilon\": epsilon,\n",
    "    \"n_attack_edges\": n_attack_edges,\n",
    "    \"model_storage_name\": model_storage_name,\n",
    "    \"attack_storage_name\": attack_storage_name\n",
    "}\n",
    "\n",
    "try:\n",
    "    os.makedirs(reports_root, exist_ok=True)\n",
    "    torch.save(report, os.path.join(reports_root, f\"{attack_storage_name}-report.pt\"))\n",
    "except RuntimeError as e:\n",
    "    print(f\"Error saving report: {e}\")\n",
    "\n",
    "print(\"Experiment Finished\")\n",
    "\n",
    "\n",
    "# if __name__ == '__main__':\n",
    "#     ex.run_commandline()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "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": 2
}
