{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "%matplotlib inline\n",
    "from notebook_init import *\n",
    "import time\n",
    "from tqdm import tqdm\n",
    "\n",
    "out_root = Path('out/consistency')\n",
    "makedirs(out_root, exist_ok=True)\n",
    "# rand = lambda : np.random.randint(np.iinfo(np.int32).max)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Loading ../models/checkpoints/stylegan2/stylegan2_ffhq_1024.pt\n"
     ]
    }
   ],
   "source": [
    "use_w = True\n",
    "inst = get_instrumented_model('StyleGAN2', 'ffhq', 'style', device, inst=inst, use_w=use_w)\n",
    "model = inst.model\n",
    "model.truncation = 1.0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Load directions\n",
    "gs_dir = np.load('./global_directions/ganspace_directions_ffhq.npy')#Note! Only ffhq is provided.\n",
    "gs_dir = torch.from_numpy(gs_dir).to(device)\n",
    "sf_dir = np.load('./global_directions/sefa_directions_ffhq.npy')#Note! Only ffhq is provided.\n",
    "sf_dir = torch.from_numpy(sf_dir).to(device)\n",
    "class compare_basis_config:\n",
    "    n_samples = 50\n",
    "    seed = 0\n",
    "    subspace_dim = 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.autograd.set_grad_enabled(True)\n",
    "eval_config = compare_basis_config()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Grassmannain Metric between two random O(n)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:29<00:00,  1.11it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 2\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:29<00:00,  1.12it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 3\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:30<00:00,  1.11it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 4\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:29<00:00,  1.11it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 5\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:30<00:00,  1.11it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 6\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:31<00:00,  1.09it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 7\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:30<00:00,  1.11it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 8\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:30<00:00,  1.10it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 9\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:29<00:00,  1.11it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 10\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:30<00:00,  1.10it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 11\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:30<00:00,  1.11it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 12\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:29<00:00,  1.12it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 13\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:28<00:00,  1.14it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 14\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:28<00:00,  1.13it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 15\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:29<00:00,  1.12it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 16\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:28<00:00,  1.13it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 17\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:28<00:00,  1.13it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 18\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:29<00:00,  1.11it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 19\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:28<00:00,  1.13it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 20\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:29<00:00,  1.12it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 21\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:34<00:00,  1.06it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 22\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:27<00:00,  1.14it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 23\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:26<00:00,  1.16it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 24\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:32<00:00,  1.09it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 25\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:27<00:00,  1.14it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 26\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:28<00:00,  1.13it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 27\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [01:30<00:00,  1.10it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluated dim 28\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 48%|████▊     | 48/100 [00:42<00:45,  1.13it/s]"
     ]
    }
   ],
   "source": [
    "eval_config.n_samples = 100\n",
    "\n",
    "for metric_type in ['geodesic', 'proj']:\n",
    "    '''\n",
    "    Grassmannain Metric between two Random Orthogonal Matrix\n",
    "    '''\n",
    "    consistency_list = []\n",
    "    time_list = []\n",
    "\n",
    "    for dim in range(1, 51):\n",
    "        print(f'Evaluated dim {dim}')\n",
    "        time.sleep(0.2)\n",
    "        eval_config.subspace_dim = dim\n",
    "\n",
    "        timer = time.time()\n",
    "        consistency = evaluate_random_basis_consistency(model, eval_config, metric_type = metric_type)\n",
    "        consistency_list.append(consistency)\n",
    "        time_list.append(round(time.time() - timer, 2))\n",
    "\n",
    "    plt.plot(consistency_list)\n",
    "    plt.savefig(f'./out/consistency/consistency_o(n)_metric_{metric_type}_n_{eval_config.n_samples}.png')\n",
    "    plt.show()\n",
    "\n",
    "    with open(f'./out/consistency/consistency_o(n)_metric_{metric_type}_n_{eval_config.n_samples}.dill', 'wb') as f:\n",
    "        pickle.dump(consistency_list, f)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Grassmannain Metric between two Random w"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "eval_config.n_samples = 1000\n",
    "\n",
    "for metric_type in ['geodesic', 'proj']:\n",
    "    '''\n",
    "    Grassmannain Metric between two Random w\n",
    "    '''\n",
    "    consistency_list = []\n",
    "    time_list = []\n",
    "    for dim in range(1, 51):\n",
    "        print(f'Evaluated dim {dim}')\n",
    "        time.sleep(0.2)\n",
    "        eval_config.subspace_dim = dim\n",
    "\n",
    "        timer = time.time()\n",
    "        consistency = evaluate_basis_consistency(model, eval_config, metric_type = metric_type)\n",
    "        consistency_list.append(consistency)\n",
    "        time_list.append(round(time.time() - timer, 2))\n",
    "\n",
    "    plt.plot(consistency_list)\n",
    "    plt.savefig(f'./out/consistency/consistency_random_metric_{metric_type}_n_{eval_config.n_samples}.png')\n",
    "    plt.show()\n",
    "    \n",
    "    with open(f'./out/consistency/consistency_random_metric_{metric_type}_n_{eval_config.n_samples}.dill', 'wb') as f:\n",
    "        pickle.dump(consistency_list, f)\n",
    "    "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Grassmannain Metric between two Close w"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "eval_config.n_samples = 1000\n",
    "\n",
    "for metric_type in ['geodesic', 'proj']:\n",
    "    '''\n",
    "    Grassmannain Metric between two Random w\n",
    "    '''\n",
    "    consistency_list = []\n",
    "    time_list = []\n",
    "    eps = 1e-1\n",
    "\n",
    "    for dim in range(1, 51):\n",
    "        print(f'Evaluated dim {dim}')\n",
    "        time.sleep(0.2)\n",
    "        eval_config.subspace_dim = dim\n",
    "\n",
    "        timer = time.time()\n",
    "        consistency = evaluate_basis_consistency_local(model, eval_config, eps = eps,  metric_type = metric_type)\n",
    "        consistency_list.append(consistency)\n",
    "        time_list.append(round(time.time() - timer, 2))\n",
    "\n",
    "\n",
    "    plt.plot(consistency_list)\n",
    "    plt.savefig(f'./out/consistency/consistency_local_metric_{metric_type}_eps_{eps}_n_{eval_config.n_samples}.png')\n",
    "    plt.show()\n",
    "    \n",
    "    with open(f'./out/consistency/consistency_local_metric_{metric_type}_eps_{eps}_n_{eval_config.n_samples}.dill', 'wb') as f:\n",
    "        pickle.dump(consistency_list, f)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Grassmannain Metric between random w and GANSpace"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "eval_config.n_samples = 1000\n",
    "global_basis = gs_dir.squeeze().t().cpu()\n",
    "\n",
    "for metric_type in ['geodesic', 'proj']:\n",
    "    '''\n",
    "    Grassmannain Metric between two Random w\n",
    "    '''\n",
    "    consistency_list = []\n",
    "    time_list = []\n",
    "\n",
    "    for dim in range(1, 51):\n",
    "        print(f'Evaluated dim {dim}')\n",
    "        time.sleep(0.2)\n",
    "        eval_config.subspace_dim = dim\n",
    "\n",
    "        timer = time.time()\n",
    "        consistency = evaluate_basis_consistency_to_global(model, eval_config, global_basis,  metric_type = metric_type)\n",
    "        consistency_list.append(consistency)\n",
    "        time_list.append(round(time.time() - timer, 2))\n",
    "\n",
    "\n",
    "    plt.plot(consistency_list)\n",
    "    plt.savefig(f'./out/consistency/consistency_to_global_ganspace_metric_{metric_type}_eps_{eps}_n_{eval_config.n_samples}.png')\n",
    "    plt.show()\n",
    "    \n",
    "    with open(f'./out/consistency/consistency_to_global_ganspace_metric_{metric_type}_n_{eval_config.n_samples}.dill', 'wb') as f:\n",
    "        pickle.dump(consistency_list, f)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Grassmannain Metric between random w and SeFa"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "eval_config.n_samples = 1000\n",
    "global_basis = sf_dir.squeeze().t().cpu()\n",
    "\n",
    "for metric_type in ['geodesic', 'proj']:\n",
    "    '''\n",
    "    Grassmannain Metric between two Random w\n",
    "    '''\n",
    "    consistency_list = []\n",
    "    time_list = []\n",
    "    \n",
    "    for dim in range(1, 51):\n",
    "        print(f'Evaluated dim {dim}')\n",
    "        time.sleep(0.2)\n",
    "        eval_config.subspace_dim = dim\n",
    "\n",
    "        timer = time.time()\n",
    "        consistency = evaluate_basis_consistency_to_global(model, eval_config, global_basis,  metric_type = metric_type)\n",
    "        consistency_list.append(consistency)\n",
    "        time_list.append(round(time.time() - timer, 2))\n",
    "        \n",
    "\n",
    "    plt.plot(consistency_list)\n",
    "    plt.savefig(f'./out/consistency/consistency_to_global_sefa_metric_{metric_type}_eps_{eps}_n_{eval_config.n_samples}.png')\n",
    "    plt.show()\n",
    "    \n",
    "    with open(f'./out/consistency/consistency_to_global_sefa_metric_{metric_type}_n_{eval_config.n_samples}.dill', 'wb') as f:\n",
    "        pickle.dump(consistency_list, f)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Ablation study for eps in Close W"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "eval_config.n_samples = 1000\n",
    "\n",
    "for metric_type in ['geodesic', 'proj']:\n",
    "    for eps in [1e-2, 1e-1, 0.25, 5e-1]:\n",
    "        consistency_list = []\n",
    "        time_list = []\n",
    "\n",
    "        for dim in range(1, 51):\n",
    "            print(f'Evaluated dim {dim}')\n",
    "            time.sleep(0.2)\n",
    "            eval_config.subspace_dim = dim\n",
    "\n",
    "            timer = time.time()\n",
    "            consistency = evaluate_basis_consistency_local(model, eval_config, eps = eps,  metric_type = metric_type)\n",
    "            consistency_list.append(consistency)\n",
    "            time_list.append(round(time.time() - timer, 2))\n",
    "\n",
    "\n",
    "        plt.plot(consistency_list)\n",
    "        plt.savefig(f'./out/consistency/consistency_local_metric_{metric_type}_eps_{eps}_n_{eval_config.n_samples}.png')\n",
    "        plt.show()\n",
    "\n",
    "        with open(f'./out/consistency/consistency_local_metric_{metric_type}_eps_{eps}_n_{eval_config.n_samples}.dill', 'wb') as f:\n",
    "            pickle.dump(consistency_list, f)\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "StyleGAN2: Optimized CUDA op FusedLeakyReLU not available, using native PyTorch fallback.\n",
      "StyleGAN2: Optimized CUDA op UpFirDn2d not available, using native PyTorch fallback.\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from localbasis_utils import compute_grsm_metric"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(512, 512)\n",
      "(512, 30)\n",
      "Dim 1 Metric [1.0, 8.458997098232027]\n",
      "Dim 2 Metric [1.0, 8.311872882066082]\n",
      "Dim 3 Metric [1.0, 8.16209713905398]\n",
      "Dim 4 Metric [1.0, 8.009521122207046]\n",
      "Dim 5 Metric [1.0, 7.853981633974483]\n",
      "Dim 6 Metric [1.0, 7.6952989809711845]\n",
      "Dim 7 Metric [1.0, 7.533274540746794]\n",
      "Dim 8 Metric [1.0, 7.367687846671537]\n",
      "Dim 9 Metric [1.0, 7.198293068896204]\n",
      "Dim 10 Metric [1.0, 7.024814731040727]\n",
      "Dim 11 Metric [1.0, 6.846942449383845]\n",
      "Dim 12 Metric [1.0, 6.664324407237549]\n",
      "Dim 13 Metric [1.0, 6.476559171707596]\n",
      "Dim 14 Metric [1.0, 6.283185307179586]\n",
      "Dim 15 Metric [1.0, 6.083668013960418]\n",
      "Dim 16 Metric [1.0, 5.877381679269498]\n",
      "Dim 17 Metric [1.0, 5.663586699569488]\n",
      "Dim 18 Metric [1.0, 5.441398092702653]\n",
      "Dim 19 Metric [1.0, 5.209742038047156]\n",
      "Dim 20 Metric [1.0, 4.967294132898051]\n",
      "Dim 21 Metric [1.0, 4.71238898038469]\n",
      "Dim 22 Metric [1.0, 4.442882938158366]\n",
      "Dim 23 Metric [1.0, 4.155936441033041]\n",
      "Dim 24 Metric [1.0, 3.8476494904855922]\n",
      "Dim 25 Metric [1.0, 3.5124073655203634]\n",
      "Dim 26 Metric [1.0, 3.141592653589793]\n",
      "Dim 27 Metric [1.0, 2.7206990463513265]\n",
      "Dim 28 Metric [1.0, 2.221441469079183]\n",
      "Dim 29 Metric [1.0, 1.5707963267948966]\n",
      "Dim 30 Metric [0.0, 0.0]\n"
     ]
    }
   ],
   "source": [
    "ambient_dim, k = 512, 30\n",
    "eye = np.eye(ambient_dim)\n",
    "print(eye.shape)\n",
    "basis_1 = eye[:, :k]\n",
    "basis_2 = np.zeros_like(basis_1)\n",
    "print(basis_1.shape)\n",
    "\n",
    "metric_score = []\n",
    "for common_dim in range(1, k+1):\n",
    "    basis_2[:, :common_dim] = eye[:, :common_dim]\n",
    "    basis_2[:, common_dim:] = eye[:, k:2*k-common_dim]\n",
    "    \n",
    "    metric_score_dim = []\n",
    "    for metric_type in ['proj', 'geodesic']:\n",
    "        metric  = compute_grsm_metric(basis_1, basis_2, d=k, metric_type=metric_type)\n",
    "        metric_score_dim.append(metric)\n",
    "        \n",
    "    metric_score.append(metric_score_dim)\n",
    "    #metric_score.append(metric_score_dim)\n",
    "    \n",
    "    print(f'Dim {common_dim} Metric {metric_score_dim}')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "metric_score = np.array(metric_score)\n",
    "\n",
    "for i, metric_type in enumerate(['proj', 'geodesic']):\n",
    "    plt.plot(metric_score[:, i])\n",
    "    plt.title(metric_type)\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "local_basis",
   "language": "python",
   "name": "local_basis"
  },
  "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.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
