{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "sys.path.append('<anonymized>/hard_label_manifolds')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "from config import SEED, dataset_to_path, dataset_to_victim_path\n",
    "import os, argparse\n",
    "import numpy as np\n",
    "import json\n",
    "import utils\n",
    "import shutil\n",
    "import math\n",
    "from tqdm import tqdm\n",
    "from collections import defaultdict\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Baseline model accuracies"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "from victim_models.utils import init_classifier\n",
    "from sklearn.metrics import accuracy_score"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "state = {\n",
    "    \"seed\": SEED,\n",
    "    \"batch_size\": 128,\n",
    "    \"test_batch_size\": 128,\n",
    "    \"victim_architecture\": 'resnet50',\n",
    "    \"targeted\": True,\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|          | 0/79 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Loading generator version\n",
      "Loading generator version\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 79/79 [00:00<00:00, 79.60it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of MNIST: 99.32\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "for dataset in [\"MNIST\"]:\n",
    "    state[\"dataset_path\"] = dataset_to_path[dataset]\n",
    "    state['dataset'] = dataset\n",
    "    state['victim_path'] = dataset_to_victim_path[dataset]\n",
    "    model_wrapper, gen_dataset, target_loader = init_classifier(state)    \n",
    "    preds = []\n",
    "    actuals = []\n",
    "    \n",
    "    # Target loader is model's shuffled test loader\n",
    "    for xi, yi in tqdm(target_loader):\n",
    "        if type(yi) is not torch.Tensor:\n",
    "            yi = torch.tensor(yi)\n",
    "        \n",
    "        xi, yi = xi.cuda(), yi.cuda()\n",
    "        dec = model_wrapper.predict_label(xi)\n",
    "        \n",
    "        preds.extend(list(dec.cpu().detach().numpy()))\n",
    "        actuals.extend(list(yi.cpu().detach().numpy()))\n",
    "    \n",
    "    acc = accuracy_score(actuals, preds) * 100\n",
    "    print(f\"Accuracy of {dataset}: {acc:.2f}\")\n",
    "    "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Robust model accuracies"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "from config import robust_l2_mnist, robust_linf_cifar, dataset_to_path, dataset_to_victim_path"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "state = {\n",
    "    \"seed\": SEED,\n",
    "    \"batch_size\": 128,\n",
    "    \"test_batch_size\": 128,\n",
    "    \"victim_architecture\": 'resnet50',\n",
    "    \"targeted\": True,\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "from victim_models.utils import init_classifier"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trying MNIST_rob_manifold.\n",
      "Loading generator version\n",
      "Loading generator version\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 79/79 [00:00<00:00, 127.20it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of MNIST_rob_manifold: 98.62\n",
      "Trying MNIST_madry.\n",
      "Loading generator version\n",
      "Loading generator version\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 79/79 [00:00<00:00, 124.59it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of MNIST_madry: 99.26\n",
      "Trying MNIST_trades.\n",
      "Loading generator version\n",
      "Loading generator version\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 79/79 [00:00<00:00, 130.64it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of MNIST_trades: 98.79\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "for dataset in robust_l2_mnist:\n",
    "    print(f\"Trying {dataset}.\")\n",
    "    state[\"dataset_path\"] = dataset_to_path[dataset]\n",
    "    state['dataset'] = dataset\n",
    "    state['victim_path'] = dataset_to_victim_path[dataset]\n",
    "    \n",
    "    model_wrapper, gen_dataset, target_loader = init_classifier(state)    \n",
    "    preds = []\n",
    "    actuals = []\n",
    "    \n",
    "    # Target loader is model's shuffled test loader\n",
    "    for xi, yi in tqdm(target_loader):\n",
    "        if type(yi) is not torch.Tensor:\n",
    "            yi = torch.tensor(yi)\n",
    "        \n",
    "        xi, yi = xi.cuda(), yi.cuda()\n",
    "        dec = model_wrapper.predict_label(xi)\n",
    "        \n",
    "        preds.extend(list(dec.cpu().detach().numpy()))\n",
    "        actuals.extend(list(yi.cpu().detach().numpy()))\n",
    "    \n",
    "    acc = accuracy_score(actuals, preds) * 100\n",
    "    print(f\"Accuracy of {dataset}: {acc:.2f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trying CIFAR10_trades.\n",
      "Loading generator version\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Loading generator version\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 79/79 [00:15<00:00,  5.27it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of CIFAR10_trades: 84.92\n",
      "Trying CIFAR10_madry.\n",
      "Loading generator version\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Loading generator version\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "WARNING:tensorflow:From /opt/conda/envs/alae-data/lib/python3.6/site-packages/tensorflow/python/compat/v2_compat.py:96: disable_resource_variables (from tensorflow.python.ops.variable_scope) is deprecated and will be removed in a future version.\n",
      "Instructions for updating:\n",
      "non-resource variables are not supported in the long term\n",
      "WARNING:tensorflow:From /opt/conda/envs/alae-data/lib/python3.6/site-packages/tensorflow/python/ops/resource_variable_ops.py:1666: calling BaseResourceVariable.__init__ (from tensorflow.python.ops.resource_variable_ops) with constraint is deprecated and will be removed in a future version.\n",
      "Instructions for updating:\n",
      "If using Keras pass *_constraint arguments to layers.\n",
      "WARNING:tensorflow:From /opt/conda/envs/alae-data/lib/python3.6/site-packages/tensorflow/python/util/deprecation.py:507: UniformUnitScaling.__init__ (from tensorflow.python.ops.init_ops) is deprecated and will be removed in a future version.\n",
      "Instructions for updating:\n",
      "Use tf.initializers.variance_scaling instead with distribution=uniform to get equivalent behavior.\n",
      "INFO:tensorflow:Restoring parameters from <anonymized>/hard_label_manifolds/community/ckpt-madry/checkpoint-70000\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 79/79 [01:24<00:00,  1.07s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of CIFAR10_madry: 87.25\n",
      "Trying CIFAR10_interp.\n",
      "Loading generator version\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Loading generator version\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  1%|▏         | 1/79 [00:00<00:09,  7.81it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Normalization parameters were set above for the model.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 79/79 [00:10<00:00,  7.74it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of CIFAR10_interp: 90.25\n",
      "Trying CIFAR10_fs.\n",
      "Loading generator version\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Loading generator version\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  1%|▏         | 1/79 [00:00<00:09,  7.91it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Normalization parameters were set above for the model.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 79/79 [00:10<00:00,  7.69it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of CIFAR10_fs: 89.98\n",
      "Trying CIFAR10_sense.\n",
      "Loading generator version\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Loading generator version\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 79/79 [00:12<00:00,  6.44it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of CIFAR10_sense: 91.51\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "for dataset in robust_linf_cifar:\n",
    "    print(f\"Trying {dataset}.\")\n",
    "    state[\"dataset_path\"] = dataset_to_path[dataset]\n",
    "    state['dataset'] = dataset\n",
    "    state['victim_path'] = dataset_to_victim_path[dataset]\n",
    "    \n",
    "    model_wrapper, gen_dataset, target_loader = init_classifier(state)    \n",
    "    preds = []\n",
    "    actuals = []\n",
    "    \n",
    "    # Target loader is model's shuffled test loader\n",
    "    for xi, yi in tqdm(target_loader):\n",
    "        if type(yi) is not torch.Tensor:\n",
    "            yi = torch.tensor(yi)\n",
    "        \n",
    "        xi, yi = xi.cuda(), yi.cuda()\n",
    "        dec = model_wrapper.predict_label(xi)\n",
    "        \n",
    "        preds.extend(list(dec.cpu().detach().numpy()))\n",
    "        actuals.extend(list(yi.cpu().detach().numpy()))\n",
    "    \n",
    "    acc = accuracy_score(actuals, preds) * 100\n",
    "    print(f\"Accuracy of {dataset}: {acc:.2f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of CIFAR10_madry: 87.25\n"
     ]
    }
   ],
   "source": [
    "acc = accuracy_score(actuals, preds) * 100\n",
    "print(f\"Accuracy of {dataset}: {acc:.2f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "from config import robust_l2_cifar, robust_l2_imagenet, robust_linf_imagenet, \\\n",
    "                   dataset_to_path, dataset_to_victim_path"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "state = {\n",
    "    \"seed\": SEED,\n",
    "    \"batch_size\": 128,\n",
    "    \"test_batch_size\": 128,\n",
    "    \"victim_architecture\": 'resnet50',\n",
    "    \"targeted\": True,\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "from victim_models.utils import init_classifier"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trying CIFAR10_smooth110.\n",
      "Loading generator version\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "Loading generator version\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 79/79 [04:30<00:00,  3.43s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of CIFAR10_smooth110: 72.64\n",
      "Trying Imagenet_smooth50.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 391/391 [1:07:09<00:00, 10.31s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Accuracy of Imagenet_smooth50: 59.23\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "for dataset in robust_l2_cifar + robust_l2_imagenet:\n",
    "    print(f\"Trying {dataset}.\")\n",
    "    state[\"dataset_path\"] = dataset_to_path[dataset]\n",
    "    state['dataset'] = dataset\n",
    "    state['victim_path'] = dataset_to_victim_path[dataset]\n",
    "    \n",
    "    model_wrapper, gen_dataset, target_loader = init_classifier(state)    \n",
    "    preds = []\n",
    "    actuals = []\n",
    "    \n",
    "    # Target loader is model's shuffled test loader\n",
    "    for xi, yi in tqdm(target_loader):\n",
    "        if type(yi) is not torch.Tensor:\n",
    "            yi = torch.tensor(yi)\n",
    "        \n",
    "        xi, yi = xi.cuda(), yi.cuda()\n",
    "        dec = model_wrapper.predict_label(xi)\n",
    "        # print(dec, yi)\n",
    "        preds.extend(list(dec.cpu().detach().numpy()))\n",
    "        actuals.extend(list(yi.cpu().detach().numpy()))\n",
    "        \n",
    "    acc = accuracy_score(actuals, preds) * 100\n",
    "    print(f\"Accuracy of {dataset}: {acc:.2f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "metadata": {},
   "outputs": [],
   "source": [
    "from victim_models.utils import init_classifier"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Trying Imagenet_madry8.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|          | 0/391 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Normalization parameters were set above for the model.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|          | 1/391 [00:12<1:24:25, 12.99s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[364, 153, 854, 679, 236, 619, 312, 7, 695, 546]\n",
      "[377, 204, 544, 163, 165, 818, 765, 7, 695, 721]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  1%|          | 2/391 [00:23<1:19:13, 12.22s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[470, 942, 701, 491, 783, 779, 830, 153, 10, 306]\n",
      "[470, 942, 405, 491, 542, 779, 830, 204, 10, 306]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  1%|          | 3/391 [00:33<1:14:07, 11.46s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[982, 990, 869, 886, 650, 807, 770, 863, 991, 225]\n",
      "[578, 990, 869, 860, 818, 879, 728, 863, 991, 225]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  1%|          | 4/391 [00:42<1:09:10, 10.72s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[866, 511, 815, 87, 641, 987, 57, 294, 875, 590]\n",
      "[717, 511, 815, 87, 451, 987, 57, 294, 456, 590]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  1%|▏         | 5/391 [00:52<1:07:34, 10.50s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[102, 134, 320, 937, 785, 714, 945, 919, 238, 261]\n",
      "[677, 171, 320, 809, 785, 714, 945, 455, 241, 261]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  2%|▏         | 6/391 [01:01<1:05:05, 10.14s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[21, 703, 755, 366, 806, 808, 631, 617, 867, 221]\n",
      "[21, 128, 755, 297, 658, 879, 631, 529, 799, 205]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  2%|▏         | 7/391 [01:12<1:06:33, 10.40s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[34, 7, 943, 979, 669, 890, 383, 355, 371, 248]\n",
      "[57, 7, 943, 980, 931, 862, 383, 368, 371, 250]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  2%|▏         | 8/391 [01:21<1:04:06, 10.04s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[903, 750, 222, 876, 846, 872, 663, 780, 18, 837]\n",
      "[629, 855, 207, 647, 619, 872, 980, 914, 18, 965]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  2%|▏         | 9/391 [01:30<1:01:10,  9.61s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[271, 421, 553, 821, 697, 941, 268, 410, 646, 24]\n",
      "[278, 791, 553, 547, 411, 941, 268, 912, 646, 43]\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  2%|▏         | 9/391 [01:39<1:10:19, 11.05s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[720, 655, 581, 746, 992, 969, 949, 770, 744, 763]\n",
      "[631, 643, 581, 746, 947, 969, 928, 770, 657, 512]\n",
      "Accuracy of Imagenet_madry8: 47.81\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "# for dataset in ['Imagenet', *robust_linf_imagenet]:\n",
    "for dataset in [*robust_linf_imagenet]:\n",
    "    print(f\"Trying {dataset}.\")\n",
    "    state[\"dataset_path\"] = dataset_to_path[dataset]\n",
    "    state['dataset'] = dataset\n",
    "    state['victim_path'] = dataset_to_victim_path[dataset]\n",
    "    \n",
    "    model_wrapper, gen_dataset, target_loader = init_classifier(state)    \n",
    "    preds = []\n",
    "    actuals = []\n",
    "    \n",
    "    batches = 0\n",
    "    # Target loader is model's shuffled test loader\n",
    "    for xi, yi in tqdm(target_loader):\n",
    "        if type(yi) is not torch.Tensor:\n",
    "            yi = torch.tensor(yi)\n",
    "        \n",
    "        xi, yi = xi.cuda(), yi.cuda()\n",
    "        dec = model_wrapper.predict_label(xi)\n",
    "        # print(dec, yi)\n",
    "        pred = list(dec.cpu().detach().numpy())\n",
    "        preds.extend(pred)\n",
    "        actual = list(yi.cpu().detach().numpy())\n",
    "        actuals.extend(actual)\n",
    "        \n",
    "        batches += 1\n",
    "        if batches == 10:\n",
    "            break\n",
    "        \n",
    "    acc = accuracy_score(actuals, preds) * 100\n",
    "    print(f\"Accuracy of {dataset}: {acc:.2f}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "print(preds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "alae-data",
   "language": "python",
   "name": "alae-data"
  },
  "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.6.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
