{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "4606d264",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Intel(R) Extension for Scikit-learn* enabled (https://github.com/intel/scikit-learn-intelex)\n",
      "Intel(R) Extension for Scikit-learn* enabled (https://github.com/intel/scikit-learn-intelex)\n"
     ]
    }
   ],
   "source": [
    "import sys\n",
    "sys.path.append('..')\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import torch\n",
    "import os\n",
    "import dgl\n",
    "from dgl import function as fn\n",
    "from dataset import load_graph_dataset\n",
    "from model_softmax import SimplifiedGraphNeuralNetwork, fast_hess, fast_hess_cuda, fast_get_inv_hvp_cuda\n",
    "from model_node_influence import NodeInfluenceSGC\n",
    "from gcn_with_node_flipping import gcn_with_node_flipping\n",
    "import tensorflow.compat.v1 as tf\n",
    "from graph_neural_networks import SGC_layer1, SGC_layer2\n",
    "from sklearn.preprocessing import OneHotEncoder\n",
    "from sklearn.metrics import log_loss\n",
    "from scipy.special import softmax, log_softmax\n",
    "from scipy.linalg import cho_solve, cho_factor\n",
    "from tqdm import tqdm\n",
    "import cupy as cp\n",
    "from random import choice\n",
    "import heapq\n",
    "from sklearnex import patch_sklearn, config_context\n",
    "patch_sklearn()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "1ec74038",
   "metadata": {},
   "outputs": [],
   "source": [
    "# dataname = 'cora'\n",
    "\n",
    "\n",
    "\n",
    "# dataname = 'pubmed'\n",
    "\n",
    "\n",
    "\n",
    "dataname = 'citeseer'\n",
    "\n",
    "\n",
    "\n",
    "# \"\"\"set up random seed, perturb ratio\"\"\"\n",
    "# perturb_ratio_list = [0.05, 0.1, 0.15, 0.2]\n",
    "# some_seed_list = [15, 42, 123, 211]\n",
    "# num_times_running = 5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "96ebe2cb",
   "metadata": {},
   "outputs": [],
   "source": [
    "perturb_ratio = 0.2\n",
    "num_perturb = int(40 * perturb_ratio)\n",
    "temp_acc_flip_list = []\n",
    "temp_acc_no_flip_list = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "35245350",
   "metadata": {},
   "outputs": [],
   "source": [
    "def index_to_mask(index, size):\n",
    "    mask = torch.zeros(size, dtype=torch.bool, device=index.device)\n",
    "    mask[index] = 1\n",
    "    return mask\n",
    "def random_splits_label_flip_attack(graph, labels, num_classes, seed):\n",
    "    # Set new random planetoid splits:\n",
    "    # * 20 * num_classes labels for training\n",
    "    # * 500 labels for validation\n",
    "    # * 1000 labels for testing\n",
    "    torch.manual_seed(seed)\n",
    "    \n",
    "    indices = []\n",
    "\n",
    "    for i in range(num_classes):\n",
    "        index = (labels == i).nonzero().view(-1)\n",
    "        index = index[torch.randperm(index.size(0))]\n",
    "        indices.append(index)\n",
    "\n",
    "    train_index = torch.cat([i[:20] for i in indices], dim=0)\n",
    "\n",
    "    rest_index = torch.cat([i[20:] for i in indices], dim=0)\n",
    "    rest_index = rest_index[torch.randperm(rest_index.size(0))]\n",
    "    \n",
    "    train_mask = index_to_mask(train_index, size=graph.num_nodes())\n",
    "    val_mask = index_to_mask(rest_index[:500], size=graph.num_nodes())\n",
    "    test_mask = index_to_mask(rest_index[500:1500], size=graph.num_nodes())\n",
    "\n",
    "    return train_mask, val_mask, test_mask\n",
    "\n",
    "def get_first_two_frequent(labels):\n",
    "    class_counts = np.bincount(labels)\n",
    "    a = np.argsort(class_counts)[-1]\n",
    "    b = np.argsort(class_counts)[-2]\n",
    "    return a, b"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "701dd4a6",
   "metadata": {},
   "outputs": [],
   "source": [
    "def generate_remove_index_train_all(from_indexes, to_indexes, train_mask):\n",
    "    train_index = torch.where(train_mask == 1)[0]\n",
    "    remove_from_list = []\n",
    "    remove_to_list = []\n",
    "    for i in tqdm(range(len(train_index))):\n",
    "        f_index = train_index[i]\n",
    "        to_index_list = torch.where(from_indexes == f_index)[0]\n",
    "        for to_index_e in to_index_list:\n",
    "            j = to_index_e\n",
    "            t_index = to_indexes[j]\n",
    "\n",
    "            remove_from_list.append(f_index)\n",
    "            remove_to_list.append(t_index)\n",
    "\n",
    "    return torch.tensor(remove_from_list), torch.tensor(remove_to_list)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "e2051936",
   "metadata": {},
   "outputs": [],
   "source": [
    "def generate_maxinfl_candidate(mat, t_lab):\n",
    "    new_t = []\n",
    "    max_infl = []\n",
    "    for i in range(len(t_lab)):\n",
    "        dif_arr = mat[i, t_lab[i]] - mat[i, :]\n",
    "        temp_max_infl_val = np.max(dif_arr)\n",
    "        temp_new_lab_val = np.where(dif_arr == temp_max_infl_val)[0][0]\n",
    "        new_t.append(temp_new_lab_val)\n",
    "        max_infl.append(temp_max_infl_val)\n",
    "    return np.array(new_t), np.array(max_infl)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "ae231d83",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  NumNodes: 3327\n",
      "  NumEdges: 9228\n",
      "  NumFeats: 3703\n",
      "  NumClasses: 6\n",
      "  NumTrainingSamples: 120\n",
      "  NumValidationSamples: 500\n",
      "  NumTestSamples: 1000\n",
      "Done loading data from cached files.\n"
     ]
    }
   ],
   "source": [
    "# graph, feat, labels, _, _, _, number_classes = load_graph_dataset(dataname)\n",
    "# train_mask, val_mask, test_mask = random_splits_label_flip_attack(graph, labels, number_classes, seed=42)\n",
    "\n",
    "graph, feat, labels, train_mask, val_mask, test_mask, number_classes = load_graph_dataset(dataname)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "7615db7d",
   "metadata": {},
   "outputs": [],
   "source": [
    "# df_flip_infl = pd.read_csv('result_flip_attack/cora_influence_flip/public_split_cora_pred_infl_flip_class.csv')\n",
    "# df_flip_infl = pd.read_csv('result_flip_attack/pubmed_influence_flip/public_split_pubmed_pred_infl_flip_class.csv')\n",
    "df_flip_infl = pd.read_csv('result_flip_attack/citeseer_influence_flip/public_split_citeseer_pred_infl_flip_class.csv')\n",
    "mat_flip_infl = np.array(df_flip_infl)[:, 1:-1]\n",
    "train_index = np.where(train_mask.numpy() == 1)[0]\n",
    "train_labels = labels[train_index].numpy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "23feb46b",
   "metadata": {},
   "outputs": [],
   "source": [
    "# # generate new flipped labels and it influences\n",
    "# # new_labels_list: new candidiate flip labels\n",
    "# new_labels_list, flip_infl_list = generate_maxinfl_candidate(mat_flip_infl, train_labels)\n",
    "\n",
    "# # get the index of labels to be flipped, in the training set\n",
    "# index_labels_to_flipped = np.argpartition(flip_infl_list, -num_perturb)[-num_perturb:]\n",
    "# # get the index of labels to be flipped, in the all set\n",
    "# all_index_labels_to_flipped = train_index[index_labels_to_flipped]\n",
    "\n",
    "# new_labels = labels.numpy().copy()\n",
    "# new_labels[all_index_labels_to_flipped] = new_labels_list[index_labels_to_flipped]\n",
    "# new_labels = torch.tensor(new_labels)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "89bcded1",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(array([  0,   1,   2,   3,   4,   5,   6,   7,   8,   9,  10,  11,  12,\n",
       "         13,  14,  15,  16,  17,  18,  19,  20,  21,  22,  23,  24,  25,\n",
       "         26,  27,  28,  29,  30,  31,  32,  33,  34,  35,  36,  37,  38,\n",
       "         39,  40,  41,  42,  43,  44,  45,  46,  47,  48,  49,  50,  51,\n",
       "         52,  53,  54,  55,  56,  57,  58,  59,  60,  61,  62,  63,  64,\n",
       "         65,  66,  67,  68,  69,  70,  71,  72,  73,  74,  75,  76,  77,\n",
       "         78,  79,  80,  81,  82,  83,  84,  85,  86,  87,  88,  89,  90,\n",
       "         91,  92,  93,  94,  95,  96,  97,  98,  99, 100, 101, 102, 103,\n",
       "        104, 105, 106, 107, 108, 109, 110, 111, 112, 113, 114, 115, 116,\n",
       "        117, 118, 119]),)"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.where(train_mask == 1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "9404ebce",
   "metadata": {},
   "outputs": [],
   "source": [
    "a, b = get_first_two_frequent(labels[test_mask])\n",
    "\n",
    "idx1 = np.where(labels[train_mask].numpy() == a)[0]\n",
    "idx2 = np.where(labels[train_mask].numpy() == b)[0]\n",
    "a_to_b_infl_list = mat_flip_infl[idx1, a] - mat_flip_infl[idx1, b]\n",
    "b_to_a_infl_list = mat_flip_infl[idx2, b] - mat_flip_infl[idx2, a]\n",
    "\n",
    "predicted_influence_combined = np.concatenate([a_to_b_infl_list, b_to_a_infl_list])\n",
    "predicted_influence_combined_sorted = np.sort(predicted_influence_combined)[::-1]\n",
    "threshold = predicted_influence_combined_sorted[num_perturb]\n",
    "\n",
    "new_labels_a = np.repeat(a, len(idx1))\n",
    "new_labels_b = np.repeat(b, len(idx2))\n",
    "\n",
    "idx_a_to_b = np.where(a_to_b_infl_list >= threshold)[0]\n",
    "idx_b_to_a = np.where(b_to_a_infl_list >= threshold)[0]\n",
    "\n",
    "new_labels_a[idx_a_to_b] = b\n",
    "new_labels_b[idx_b_to_a] = a\n",
    "\n",
    "perturb_labels = train_labels.copy()\n",
    "\n",
    "perturb_labels[idx1] = new_labels_a\n",
    "perturb_labels[idx2] = new_labels_b\n",
    "new_labels = labels.numpy().copy()\n",
    "new_labels[train_mask] = perturb_labels\n",
    "new_labels = torch.tensor(new_labels)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "78af2a8a",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  0%|                                                    | 0/25 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 54.80%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  4%|█▊                                          | 1/25 [00:09<03:47,  9.47s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 71.60%\n",
      "Test accuracy 54.90%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      "  8%|███▌                                        | 2/25 [00:18<03:34,  9.31s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 71.70%\n",
      "Test accuracy 54.80%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 12%|█████▎                                      | 3/25 [00:27<03:23,  9.26s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 71.10%\n",
      "Test accuracy 54.80%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 16%|███████                                     | 4/25 [00:36<03:13,  9.21s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 70.60%\n",
      "Test accuracy 54.60%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 20%|████████▊                                   | 5/25 [00:46<03:03,  9.18s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 70.50%\n",
      "Test accuracy 54.20%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 24%|██████████▌                                 | 6/25 [00:55<02:54,  9.19s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 70.00%\n",
      "Test accuracy 54.20%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 28%|████████████▎                               | 7/25 [01:04<02:46,  9.27s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 71.60%\n",
      "Test accuracy 54.60%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 32%|██████████████                              | 8/25 [01:14<02:38,  9.30s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 70.60%\n",
      "Test accuracy 52.90%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 36%|███████████████▊                            | 9/25 [01:23<02:28,  9.31s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 71.20%\n",
      "Test accuracy 54.40%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 40%|█████████████████▏                         | 10/25 [01:32<02:20,  9.35s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 70.60%\n",
      "Test accuracy 54.90%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 44%|██████████████████▉                        | 11/25 [01:42<02:10,  9.33s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 70.70%\n",
      "Test accuracy 54.80%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 48%|████████████████████▋                      | 12/25 [01:51<02:00,  9.30s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 71.40%\n",
      "Test accuracy 54.30%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 52%|██████████████████████▎                    | 13/25 [02:00<01:51,  9.29s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 71.30%\n",
      "Test accuracy 54.10%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 56%|████████████████████████                   | 14/25 [02:10<01:42,  9.30s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 70.80%\n",
      "Test accuracy 55.40%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 60%|█████████████████████████▊                 | 15/25 [02:19<01:32,  9.28s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 71.40%\n",
      "Test accuracy 54.40%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 64%|███████████████████████████▌               | 16/25 [02:28<01:23,  9.27s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 70.60%\n",
      "Test accuracy 54.50%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 68%|█████████████████████████████▏             | 17/25 [02:37<01:14,  9.28s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 71.80%\n",
      "Test accuracy 54.80%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 72%|██████████████████████████████▉            | 18/25 [02:47<01:05,  9.30s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 71.50%\n",
      "Test accuracy 55.10%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 76%|████████████████████████████████▋          | 19/25 [02:56<00:55,  9.32s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 70.10%\n",
      "Test accuracy 54.50%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 80%|██████████████████████████████████▍        | 20/25 [03:05<00:46,  9.34s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 71.60%\n",
      "Test accuracy 55.10%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 84%|████████████████████████████████████       | 21/25 [03:15<00:37,  9.35s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 70.50%\n",
      "Test accuracy 54.50%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 88%|█████████████████████████████████████▊     | 22/25 [03:24<00:28,  9.34s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 70.30%\n",
      "Test accuracy 54.60%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 92%|███████████████████████████████████████▌   | 23/25 [03:33<00:18,  9.34s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 70.80%\n",
      "Test accuracy 53.60%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\r",
      " 96%|█████████████████████████████████████████▎ | 24/25 [03:43<00:09,  9.37s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 71.10%\n",
      "Test accuracy 54.80%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████| 25/25 [03:52<00:00,  9.31s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test accuracy 71.20%\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "# train_mask, val_mask, test_mask = random_splits_label_flip_attack(graph, labels, number_classes, seed=15)\n",
    "\n",
    "\n",
    "for _ in tqdm(range(25)):\n",
    "    gcn_with_node_flip = gcn_with_node_flipping(graph= graph, features=feat, new_labels=new_labels, \n",
    "                                                train_mask=train_mask, val_mask=val_mask, test_mask=test_mask,\n",
    "                                               num_classes=number_classes, dropout=0.5)\n",
    "\n",
    "    gcn_without_node_flip = gcn_with_node_flipping(graph= graph, features=feat, new_labels=labels, \n",
    "                                                train_mask=train_mask, val_mask=val_mask, test_mask=test_mask,\n",
    "                                               num_classes=number_classes)\n",
    "\n",
    "    acc_flip = gcn_with_node_flip.train_evaluate()\n",
    "    acc_no_flip = gcn_without_node_flip.train_evaluate()\n",
    "\n",
    "\n",
    "    temp_acc_flip_list.append(acc_flip)\n",
    "    temp_acc_no_flip_list.append(acc_no_flip)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "feff2b91",
   "metadata": {},
   "outputs": [],
   "source": [
    "flip_df = pd.DataFrame([temp_acc_flip_list, temp_acc_no_flip_list]).T\n",
    "flip_df.columns = ['filped accuracy', 'original accuracy']\n",
    "flip_df.to_csv('result_flip_attack/'+ dataname + '/'+ 'public_split' + str(perturb_ratio)+'_final.csv', index = False)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ed38a324",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "9102f037",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.54544"
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "np.mean(temp_acc_flip_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "f39cbefd",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.2"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "perturb_ratio"
   ]
  }
 ],
 "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.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
