{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6da84189",
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "sys.path.append(\"../\")\n",
    "\n",
    "import os\n",
    "\n",
    "import numpy as np\n",
    "import random\n",
    "import tensorflow as tf\n",
    "import numpy as np\n",
    "\n",
    "from models import DenseReparam, DenseWN\n",
    "from utils import plot_loss\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "GLOBAL_SEED = 0\n",
    "DATASPLIT_SEED = 0\n",
    "\n",
    "os.environ['PYTHONHASHSEED']=str(GLOBAL_SEED)\n",
    "os.environ['TF_CUDNN_DETERMINISTIC'] = '1'\n",
    "random.seed(GLOBAL_SEED)\n",
    "np.random.seed(GLOBAL_SEED)\n",
    "tf.random.set_seed(GLOBAL_SEED)\n",
    "\n",
    "#os.environ[\"CUDA_VISIBLE_DEVICES\"] = \"-1\" # disable GPU\n",
    "dtype = 'float64'\n",
    "tf.keras.backend.set_floatx(dtype)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "68ea1cee",
   "metadata": {},
   "outputs": [],
   "source": [
    "def levy_normalized(params):\n",
    "    w = 1.0 + (params - 1.0) / 4.0\n",
    "    res = np.sin(np.pi * w[:,0])**2 + np.sum(((w [:,0:-1] -1)**2) * (1.0+10.0*np.sin(np.pi * w[:,0:-1] + 1)**2), axis=-1) + ((w [:,-1] - 1.0)**2) * (1.0 + np.sin(2.0*np.pi*w[:,-1])**2)\n",
    "    return np.array(res, dtype=dtype)\n",
    "\n",
    "num_data_points = 50\n",
    "\n",
    "test_features = np.arange(-10.0, 10.1, 0.1).reshape(-1, 1)\n",
    "test_labels = levy_normalized(test_features).reshape(-1,1)\n",
    "\n",
    "train_features = np.random.uniform(low=-10.0, high=10.0, size=[num_data_points,1])\n",
    "epsilon = np.random.normal(loc=0.0, scale=0.1, size=[num_data_points,1])\n",
    "train_labels = levy_normalized(train_features).reshape(-1,1) + epsilon"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "40aeeb20",
   "metadata": {},
   "outputs": [],
   "source": [
    "units = 100\n",
    "n_epochs = 10000\n",
    "\n",
    "lr_gmp = 0.1\n",
    "lr_others = 0.01\n",
    "\n",
    "os.makedirs(\"./figs\", exist_ok=True)"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "id": "c5b9f677",
   "metadata": {},
   "source": [
    "# Geometric Parameterization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cbe51ec0",
   "metadata": {},
   "outputs": [],
   "source": [
    "def build_reparam_dnn_model(lr):\n",
    "    model = tf.keras.Sequential([\n",
    "        DenseReparam(units, activation='relu'),\n",
    "        tf.keras.layers.Dense(1)\n",
    "    ])\n",
    "    model.compile(loss='mean_absolute_error', optimizer=tf.keras.optimizers.Adam(learning_rate=lr), metrics=[tf.keras.metrics.MeanSquaredError()])\n",
    "    return model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f55e9f65",
   "metadata": {},
   "outputs": [],
   "source": [
    "reparam_dnn_model = build_reparam_dnn_model(lr=lr_gmp)\n",
    "test_pred_reparam_dnn_empty = reparam_dnn_model(test_features, training=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3de2317c",
   "metadata": {},
   "outputs": [],
   "source": [
    "reparam_locs = [np.zeros(units)]\n",
    "reg_weights = [reparam_dnn_model.get_weights()[1].squeeze()]\n",
    "print_reparam_locs = tf.keras.callbacks.LambdaCallback(on_epoch_end=lambda batch, logs: reparam_locs.append(\n",
    "    -1.0 * reparam_dnn_model.get_weights()[0][1,:] / (reparam_dnn_model.get_weights()[0][0,:] / np.abs(reparam_dnn_model.get_weights()[0][0,:]))\n",
    "))\n",
    "lr_decay = tf.keras.callbacks.ReduceLROnPlateau(monitor='loss', factor=0.1, patience=500, verbose=0, mode='min')\n",
    "early_stop = tf.keras.callbacks.EarlyStopping(monitor='loss', patience=1000, verbose=0, mode='min', restore_best_weights=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a62a81c0",
   "metadata": {},
   "outputs": [],
   "source": [
    "history_reparam_dnn = reparam_dnn_model.fit(\n",
    "    train_features,\n",
    "    train_labels,\n",
    "    batch_size=num_data_points,\n",
    "    verbose=0, \n",
    "    epochs=n_epochs,\n",
    "    callbacks=[print_reparam_locs, lr_decay, early_stop]\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dee07f0f",
   "metadata": {},
   "outputs": [],
   "source": [
    "test_pred_reparam_dnn = reparam_dnn_model(test_features, training=False)\n",
    "test_results_reparam_dnn = reparam_dnn_model.evaluate(test_features, test_labels, verbose=0)\n",
    "reparam_locs_np = np.stack(reparam_locs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3377cadc",
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.style.use('ggplot')\n",
    "n_subplots = 11\n",
    "plot_every = 500\n",
    "f, axes = plt.subplots(1+n_subplots, 1, gridspec_kw={'height_ratios': [7, *([0.5]*n_subplots)]} ,figsize=(7, 10))\n",
    "\n",
    "axes[0].scatter(train_features, train_labels, color='r', s=100, label=\"Training data\")\n",
    "axes[0].plot(test_features, test_labels, c='black', label='Ground-truth', linewidth=5)\n",
    "axes[0].plot(test_features, test_pred_reparam_dnn_empty, c='blue', label='Prediction before training', linewidth=3, linestyle=\"--\")\n",
    "axes[0].plot(test_features, test_pred_reparam_dnn, c='blue', label='Prediction after training', linewidth=3)\n",
    "axes[0].set_xticklabels([])\n",
    "axes[0].set_ylabel(\"$y$\", fontsize=12)\n",
    "axes[0].set_xlim(-10.5, 10.5)\n",
    "axes[0].set_ylim(-0.5, 18)\n",
    "axes[0].set_title(\"Test RMSE: {0:.3f}\".format(np.sqrt(test_results_reparam_dnn[1])), fontsize=20)\n",
    "\n",
    "\n",
    "for i in range(n_subplots):\n",
    "    axes[n_subplots-i].vlines(reparam_locs_np[i*plot_every:i*plot_every+1, :], 0, 1, color='darkgoldenrod', alpha=0.5, linestyles=\"solid\")\n",
    "    axes[n_subplots-i].set_yticks([])\n",
    "    axes[n_subplots-i].set_xlim(-10.5, 10.5)\n",
    "    axes[i].set_xticklabels([])\n",
    "    if i == n_subplots // 2 + 1:\n",
    "        axes[i].set_ylabel(\"Training (beginning --> end)\")\n",
    "axes[-1].set_xlabel(\"$x$\")\n",
    "\n",
    "\n",
    "plt.subplots_adjust(hspace=0.1)\n",
    "plt.savefig(\"./figs/levy_gmp.pdf\", format=\"pdf\", bbox_inches=\"tight\")"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "id": "e446a4a5",
   "metadata": {},
   "source": [
    "# Standard Parameterization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "98099b36",
   "metadata": {},
   "outputs": [],
   "source": [
    "def build_dnn_model(lr):\n",
    "    model = tf.keras.Sequential([\n",
    "        tf.keras.layers.Dense(units, activation='relu'),\n",
    "        tf.keras.layers.Dense(1)\n",
    "    ])\n",
    "    model.compile(loss='mean_absolute_error', optimizer=tf.keras.optimizers.Adam(lr), metrics=[tf.keras.metrics.MeanSquaredError()])\n",
    "    return model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "682b53f4",
   "metadata": {},
   "outputs": [],
   "source": [
    "dnn_model = build_dnn_model(lr=lr_others)\n",
    "test_pred_dnn_empty = dnn_model(test_features, training=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a41bc73a",
   "metadata": {},
   "outputs": [],
   "source": [
    "dnn_locs = [np.zeros(units)]\n",
    "print_dnn_weights = tf.keras.callbacks.LambdaCallback(on_epoch_end=lambda batch, logs: dnn_locs.append(-1.0 * dnn_model.get_weights()[1] / dnn_model.get_weights()[0].squeeze()))\n",
    "lr_decay = tf.keras.callbacks.ReduceLROnPlateau(monitor='loss', factor=0.1, patience=500, verbose=0, mode='min')\n",
    "early_stop = tf.keras.callbacks.EarlyStopping(monitor='loss', patience=1000, verbose=0, mode='min', restore_best_weights=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e31e4abf",
   "metadata": {},
   "outputs": [],
   "source": [
    "history_dnn = dnn_model.fit(\n",
    "    train_features,\n",
    "    train_labels,\n",
    "    batch_size=num_data_points,\n",
    "    verbose=0, \n",
    "    epochs=n_epochs,\n",
    "    callbacks=[print_dnn_weights, lr_decay, early_stop]\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "25301904",
   "metadata": {},
   "outputs": [],
   "source": [
    "dnn_locs_np = np.stack(dnn_locs)\n",
    "test_pred_dnn = dnn_model(test_features, training=False)\n",
    "test_results_dnn = dnn_model.evaluate(test_features, test_labels, verbose=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d603e2af",
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.style.use('ggplot')\n",
    "n_subplots = 11\n",
    "plot_every = dnn_locs_np.shape[0] // 11\n",
    "f, axes = plt.subplots(1+n_subplots, 1, gridspec_kw={'height_ratios': [7, *([0.5]*n_subplots)]} ,figsize=(7, 10))\n",
    "\n",
    "axes[0].scatter(train_features, train_labels, color='r', s=100, label=\"Training data\")\n",
    "axes[0].plot(test_features, test_labels, c='black', label='Ground-truth', linewidth=5)\n",
    "axes[0].plot(test_features, test_pred_dnn_empty, c='blue', label='Prediction before training', linewidth=3, linestyle=\"--\")\n",
    "axes[0].plot(test_features, test_pred_dnn, c='blue', label='Prediction after training', linewidth=3)\n",
    "axes[0].set_xticklabels([])\n",
    "axes[0].set_ylabel(\"$y$\", fontsize=12)\n",
    "axes[0].set_xlim(-10.5, 10.5)\n",
    "axes[0].set_ylim(-0.5, 18)\n",
    "axes[0].set_title(\"Test RMSE: {0:.3f}\".format(np.sqrt(test_results_dnn[1])), fontsize=20)\n",
    "axes[0].legend(fontsize=15, loc=\"upper center\")\n",
    "\n",
    "\n",
    "for i in range(n_subplots):\n",
    "    axes[n_subplots-i].vlines(dnn_locs_np[i*plot_every:i*plot_every+1, :], 0, 1, color='darkgoldenrod', alpha=0.5, linestyles=\"solid\")\n",
    "    axes[n_subplots-i].set_yticks([])\n",
    "    axes[n_subplots-i].set_xlim(-10.5, 10.5)\n",
    "    axes[i].set_xticklabels([])\n",
    "    if i == n_subplots // 2 + 1:\n",
    "        axes[i].set_ylabel(\"Training (beginning --> end)\")\n",
    "axes[-1].set_xlabel(\"$x$\")\n",
    "\n",
    "\n",
    "plt.subplots_adjust(hspace=0.1)\n",
    "plt.savefig(\"./figs/levy_sp.pdf\", format=\"pdf\", bbox_inches=\"tight\")"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "id": "9e353f04",
   "metadata": {},
   "source": [
    "# Batch Normalization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "cca46fa0",
   "metadata": {},
   "outputs": [],
   "source": [
    "def build_bn_dnn_model(lr):\n",
    "    model = tf.keras.Sequential([\n",
    "        tf.keras.layers.Dense(units),\n",
    "        tf.keras.layers.BatchNormalization(),\n",
    "        tf.keras.layers.ReLU(),\n",
    "        tf.keras.layers.Dense(1)\n",
    "    ])\n",
    "    model.compile(loss='mean_absolute_error', optimizer=tf.keras.optimizers.Adam(lr), metrics=[tf.keras.metrics.MeanSquaredError()])\n",
    "    return model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "05f46e6e",
   "metadata": {},
   "outputs": [],
   "source": [
    "bn_dnn_model = build_bn_dnn_model(lr=lr_others)\n",
    "test_pred_bn_dnn_empty = bn_dnn_model(test_features, training=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f4d108cd",
   "metadata": {},
   "outputs": [],
   "source": [
    "weights = bn_dnn_model.get_weights()\n",
    "bn_layer = bn_dnn_model.get_layer(\"batch_normalization\")\n",
    "mu = bn_layer.moving_mean.numpy()\n",
    "sigma = np.sqrt(bn_layer.moving_variance.numpy() + bn_layer.epsilon)\n",
    "gamma = bn_layer.gamma.numpy()\n",
    "beta = bn_layer.beta.numpy()\n",
    "\n",
    "w = weights[0].squeeze()\n",
    "b = weights[1] + sigma * beta / gamma - mu\n",
    "bn_dnn_locs = [-1.0 * b / w]\n",
    "\n",
    "def get_feature_locs(epoch, logs):\n",
    "    weights = bn_dnn_model.get_weights()\n",
    "    bn_layer = bn_dnn_model.get_layer(\"batch_normalization\")\n",
    "    mu = bn_layer.moving_mean.numpy()\n",
    "    sigma = np.sqrt(bn_layer.moving_variance.numpy() + bn_layer.epsilon)\n",
    "    gamma = bn_layer.gamma.numpy()\n",
    "    beta = bn_layer.beta.numpy()\n",
    "    \n",
    "    w = weights[0].squeeze()\n",
    "    b = weights[1] + sigma * beta / gamma - mu\n",
    "    bn_dnn_locs.append(-1.0 * b / w)\n",
    "\n",
    "print_bn_dnn_weights = tf.keras.callbacks.LambdaCallback(on_epoch_end=get_feature_locs)\n",
    "lr_decay = tf.keras.callbacks.ReduceLROnPlateau(monitor='loss', factor=0.1, patience=500, verbose=0, mode='min')\n",
    "early_stop = tf.keras.callbacks.EarlyStopping(monitor='loss', patience=1000, verbose=0, mode='min', restore_best_weights=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f0ec89d5",
   "metadata": {},
   "outputs": [],
   "source": [
    "history_bn_dnn = bn_dnn_model.fit(\n",
    "    train_features,\n",
    "    train_labels,\n",
    "    batch_size=num_data_points,\n",
    "    verbose=0, \n",
    "    epochs=n_epochs,\n",
    "    callbacks=[print_bn_dnn_weights, lr_decay, early_stop]\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7374c34f",
   "metadata": {},
   "outputs": [],
   "source": [
    "test_pred_bn_dnn = bn_dnn_model(test_features, training=False)\n",
    "test_results_bn_dnn = bn_dnn_model.evaluate(test_features, test_labels, verbose=0)\n",
    "bn_dnn_locs_np = np.stack(bn_dnn_locs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c6c7d445",
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.style.use('ggplot')\n",
    "n_subplots = 11\n",
    "plot_every = bn_dnn_locs_np.shape[0] // 11\n",
    "f, axes = plt.subplots(1+n_subplots, 1, gridspec_kw={'height_ratios': [7, *([0.5]*n_subplots)]} ,figsize=(7, 10))\n",
    "\n",
    "axes[0].scatter(train_features, train_labels, color='r', s=100, label=\"Training data\")\n",
    "axes[0].plot(test_features, test_labels, c='black', label='Ground-truth', linewidth=5)\n",
    "axes[0].plot(test_features, test_pred_bn_dnn_empty, c='blue', label='Prediction before training', linewidth=3, linestyle=\"--\")\n",
    "axes[0].plot(test_features, test_pred_bn_dnn, c='blue', label='Prediction after training', linewidth=3)\n",
    "axes[0].set_xticklabels([])\n",
    "axes[0].set_ylabel(\"$y$\", fontsize=12)\n",
    "axes[0].set_xlim(-10.5, 10.5)\n",
    "axes[0].set_ylim(-0.5, 18)\n",
    "axes[0].set_title(\"Test RMSE: {0:.3f}\".format(np.sqrt(test_results_bn_dnn[1])), fontsize=20)\n",
    "\n",
    "\n",
    "for i in range(n_subplots):\n",
    "    axes[n_subplots-i].vlines(bn_dnn_locs_np[i*plot_every:i*plot_every+1, :], 0, 1, color='darkgoldenrod', alpha=0.5, linestyles=\"solid\")\n",
    "    axes[n_subplots-i].set_yticks([])\n",
    "    axes[n_subplots-i].set_xlim(-10.5, 10.5)\n",
    "    axes[i].set_xticklabels([])\n",
    "    if i == n_subplots // 2 + 1:\n",
    "        axes[i].set_ylabel(\"Training (beginning --> end)\")\n",
    "axes[-1].set_xlabel(\"$x$\")\n",
    "\n",
    "plt.subplots_adjust(hspace=0.1)\n",
    "plt.savefig(\"figs/levy_batchnorm.pdf\", format=\"pdf\", bbox_inches=\"tight\")"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "id": "f5f2564d",
   "metadata": {},
   "source": [
    "# Weight Normalization"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2695b8de",
   "metadata": {},
   "outputs": [],
   "source": [
    "def build_wn_dnn_model(lr):\n",
    "    model = tf.keras.Sequential([\n",
    "        DenseWN(units, activation='relu'),\n",
    "        DenseWN(1)\n",
    "    ])\n",
    "    model.compile(loss='mean_absolute_error', optimizer=tf.keras.optimizers.Adam(lr), metrics=[tf.keras.metrics.MeanSquaredError()])\n",
    "    return model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b810c1d5",
   "metadata": {},
   "outputs": [],
   "source": [
    "wn_dnn_model = build_wn_dnn_model(lr=lr_others)\n",
    "test_pred_wn_dnn_empty = wn_dnn_model(test_features, training=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "540e78eb",
   "metadata": {},
   "outputs": [],
   "source": [
    "wn_dnn_locs = [np.zeros(units)]\n",
    "\n",
    "def get_feature_locs(epoch, logs):\n",
    "    weights = wn_dnn_model.get_weights()[0]\n",
    "    \n",
    "    w = weights[0] / np.abs(weights[0]) * weights[2]\n",
    "    b = weights[1]\n",
    "    wn_dnn_locs.append(-1.0 * b / w)\n",
    "\n",
    "print_wn_dnn_weights = tf.keras.callbacks.LambdaCallback(on_epoch_end=get_feature_locs)\n",
    "lr_decay = tf.keras.callbacks.ReduceLROnPlateau(monitor='loss', factor=0.1, patience=500, verbose=0, mode='min')\n",
    "early_stop = tf.keras.callbacks.EarlyStopping(monitor='loss', patience=1000, verbose=0, mode='min', restore_best_weights=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d9b0c670",
   "metadata": {},
   "outputs": [],
   "source": [
    "history_wn_dnn = wn_dnn_model.fit(\n",
    "    train_features,\n",
    "    train_labels,\n",
    "    batch_size=num_data_points,\n",
    "    verbose=0, \n",
    "    epochs=n_epochs,\n",
    "    callbacks=[print_wn_dnn_weights, lr_decay, early_stop]\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1f0cae3d",
   "metadata": {},
   "outputs": [],
   "source": [
    "test_pred_wn_dnn = wn_dnn_model(test_features, training=False)\n",
    "test_results_wn_dnn = wn_dnn_model.evaluate(test_features, test_labels, verbose=0)\n",
    "wn_dnn_locs_np = np.stack(wn_dnn_locs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "9001ef4c",
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.style.use('ggplot')\n",
    "n_subplots = 11\n",
    "plot_every = wn_dnn_locs_np.shape[0] // 11\n",
    "f, axes = plt.subplots(1+n_subplots, 1, gridspec_kw={'height_ratios': [7, *([0.5]*n_subplots)]} ,figsize=(7, 10))\n",
    "\n",
    "axes[0].scatter(train_features, train_labels, color='r', s=100, label=\"Training data\")\n",
    "axes[0].plot(test_features, test_labels, c='black', label='Ground-truth', linewidth=5)\n",
    "axes[0].plot(test_features, test_pred_wn_dnn_empty, c='blue', label='Prediction before training', linewidth=3, linestyle=\"--\")\n",
    "axes[0].plot(test_features, test_pred_wn_dnn, c='blue', label='Prediction after training', linewidth=3)\n",
    "axes[0].set_xticklabels([])\n",
    "axes[0].set_ylabel(\"$y$\", fontsize=12)\n",
    "axes[0].set_xlim(-10.5, 10.5)\n",
    "axes[0].set_ylim(-0.5, 18)\n",
    "axes[0].set_title(\"Test RMSE: {0:.3f}\".format(np.sqrt(test_results_wn_dnn[1])), fontsize=20)\n",
    "\n",
    "for i in range(n_subplots):\n",
    "    axes[n_subplots-i].vlines(wn_dnn_locs_np[i*plot_every:i*plot_every+1, :], 0, 1, color='darkgoldenrod', alpha=0.5, linestyles=\"solid\")\n",
    "    axes[n_subplots-i].set_yticks([])\n",
    "    axes[n_subplots-i].set_xlim(-10.5, 10.5)\n",
    "    axes[i].set_xticklabels([])\n",
    "    if i == n_subplots // 2 + 1:\n",
    "        axes[i].set_ylabel(\"Training (beginning --> end)\")\n",
    "axes[-1].set_xlabel(\"$x$\")\n",
    "\n",
    "plt.subplots_adjust(hspace=0.1)\n",
    "plt.savefig(\"./figs/levy_weightnorm.pdf\", format=\"pdf\", bbox_inches=\"tight\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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"
  },
  "vscode": {
   "interpreter": {
    "hash": "630694aee0ba86fd16b724d8128d1f0c1677f298824c71c2eb5a498d077f2247"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
