{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# PBO on LQR\n",
    "\n",
    "## Define paramters"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T15:07:02.219924Z",
     "iopub.status.busy": "2022-09-19T15:07:02.218976Z",
     "iopub.status.idle": "2022-09-19T15:07:02.930633Z",
     "shell.execute_reply": "2022-09-19T15:07:02.930167Z"
    }
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "WARNING:absl:No GPU/TPU found, falling back to CPU. (Set TF_CPP_MIN_LOG_LEVEL=0 and rerun for more info.)\n"
     ]
    }
   ],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "\n",
    "import warnings\n",
    "warnings.simplefilter(action='ignore', category=FutureWarning)\n",
    "import jax\n",
    "import os\n",
    "import json\n",
    "\n",
    "parameters = json.load(open(\"parameters.json\"))\n",
    "env_seed = parameters[\"env_seed\"]\n",
    "\n",
    "# Sample collection\n",
    "max_discrete_state = parameters[\"max_discrete_state\"]\n",
    "n_discrete_states = parameters[\"n_discrete_states\"]\n",
    "max_discrete_action = parameters[\"max_discrete_action\"]\n",
    "n_discrete_actions = parameters[\"n_discrete_actions\"]\n",
    "n_samples = n_discrete_states * n_discrete_actions\n",
    "\n",
    "# Weights collection\n",
    "n_weights = parameters[\"n_weights\"]\n",
    "weights_filtering = True\n",
    "\n",
    "# Q function\n",
    "max_action_on_max = parameters[\"max_action_on_max\"]\n",
    "n_actions_on_max = parameters[\"n_actions_on_max\"]\n",
    "\n",
    "# Trainings\n",
    "max_bellman_iterations = parameters[\"max_bellman_iterations\"]\n",
    "training_steps = parameters[\"training_steps\"]\n",
    "fitting_steps = parameters[\"fitting_steps_pbo\"]\n",
    "batch_size_samples = parameters[\"batch_size_samples\"]\n",
    "batch_size_weights = parameters[\"batch_size_weights\"]\n",
    "initial_weight_std = parameters[\"initial_weight_std\"]\n",
    "learning_rate = {\"first\": parameters[\"starting_lr_pbo\"], \"last\": parameters[\"ending_lr_pbo\"], \"duration\": training_steps * fitting_steps * n_samples // batch_size_samples}\n",
    "\n",
    "# Visualisation of errors and performances\n",
    "max_bellman_iterations_validation = max_bellman_iterations + 5\n",
    "\n",
    "# Search for an unused seed\n",
    "max_used_seed = 0\n",
    "if not os.path.exists(\"figures/data/PBO_custom_linear/\"):\n",
    "    os.makedirs(\"figures/data/PBO_custom_linear/\")\n",
    "for file in os.listdir(\"figures/data/PBO_custom_linear/\"):\n",
    "    if int(file.split(\"_\")[0]) == max_bellman_iterations and int(file.split(\"_\")[2]) == n_samples and int(file.split(\"_\")[3][:-4]) > max_used_seed:\n",
    "        max_used_seed = int(file.split(\"_\")[3][:-4])\n",
    "max_used_seed\n",
    "\n",
    "# keys\n",
    "env_key = jax.random.PRNGKey(env_seed)\n",
    "seed = max_used_seed + 1\n",
    "key = jax.random.PRNGKey(seed)\n",
    "shuffle_key, q_network_key, pbo_network_key = jax.random.split(key, 3)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Define environment"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T15:07:02.933363Z",
     "iopub.status.busy": "2022-09-19T15:07:02.933008Z",
     "iopub.status.idle": "2022-09-19T15:07:03.162666Z",
     "shell.execute_reply": "2022-09-19T15:07:03.162179Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Transition: s' = As + Ba\n",
      "Transition: s' = 0.7204725742340088s + -0.5264108180999756a\n",
      "Reward: Qs² + Ra² + 2 Ssa\n",
      "Reward: -0.13297832012176514s² + -0.8039400577545166a² + 0.2581009864807129sa\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "\n",
    "from pbo.environments.linear_quadratic import LinearQuadraticEnv\n",
    "\n",
    "\n",
    "discrete_states = np.linspace(-max_discrete_state, max_discrete_state, n_discrete_states)\n",
    "state_box_half_size = max_discrete_state / n_discrete_states\n",
    "discrete_states_boxes = np.linspace(\n",
    "    -max_discrete_state - state_box_half_size, max_discrete_state + state_box_half_size, n_discrete_states + 1\n",
    ")\n",
    "\n",
    "discrete_actions = np.linspace(-max_discrete_action, max_discrete_action, n_discrete_actions)\n",
    "action_box_half_size = max_discrete_action / n_discrete_actions\n",
    "discrete_actions_boxes = np.linspace(\n",
    "    -max_discrete_action - action_box_half_size, max_discrete_action + action_box_half_size, n_discrete_actions + 1\n",
    ")\n",
    "\n",
    "env = LinearQuadraticEnv(env_key, max_init_state=max_discrete_state)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Collect samples"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Samples on the mesh"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T15:07:03.166327Z",
     "iopub.status.busy": "2022-09-19T15:07:03.166179Z",
     "iopub.status.idle": "2022-09-19T15:07:03.307473Z",
     "shell.execute_reply": "2022-09-19T15:07:03.306980Z"
    }
   },
   "outputs": [],
   "source": [
    "import jax.numpy as jnp\n",
    "\n",
    "from pbo.sample_collection.replay_buffer import ReplayBuffer\n",
    "\n",
    "\n",
    "n_samples = n_discrete_states * n_discrete_actions\n",
    "replay_buffer = ReplayBuffer()\n",
    "\n",
    "for state in discrete_states:\n",
    "    for action in discrete_actions:\n",
    "        env.reset(jnp.array([state]))\n",
    "        next_state, reward, absorbing, _ = env.step(jnp.array([action]))\n",
    "\n",
    "        replay_buffer.add(jnp.array([state]), jnp.array([action]), reward, next_state, absorbing)\n",
    "\n",
    "replay_buffer.cast_to_jax_array()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Visualize samples"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T15:07:03.310097Z",
     "iopub.status.busy": "2022-09-19T15:07:03.309916Z",
     "iopub.status.idle": "2022-09-19T15:07:03.920563Z",
     "shell.execute_reply": "2022-09-19T15:07:03.920095Z"
    }
   },
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from pbo.sample_collection.count_samples import count_samples\n",
    "from pbo.utils.two_dimesions_mesh import TwoDimesionsMesh\n",
    "\n",
    "\n",
    "samples_count, n_outside_boxes, _ = count_samples(replay_buffer.states, replay_buffer.actions, discrete_states_boxes, discrete_actions_boxes, replay_buffer.rewards)\n",
    "samples_visu_mesh = TwoDimesionsMesh(discrete_states, discrete_actions, sleeping_time=0)\n",
    "\n",
    "samples_visu_mesh.set_values(samples_count, zeros_to_nan=True)\n",
    "samples_visu_mesh.show(\n",
    "    f\"Samples repartition, \\n{int(100 * n_outside_boxes / n_samples)}% are outside the box.\"\n",
    ")\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Collect weights"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T15:07:03.922880Z",
     "iopub.status.busy": "2022-09-19T15:07:03.922702Z",
     "iopub.status.idle": "2022-09-19T15:07:04.161744Z",
     "shell.execute_reply": "2022-09-19T15:07:04.159992Z"
    }
   },
   "outputs": [],
   "source": [
    "from pbo.sample_collection.dataloader import SampleDataLoader\n",
    "from pbo.networks.learnable_q import LQRQ\n",
    "from pbo.weights_collection.weights_buffer import WeightsBuffer\n",
    "\n",
    "\n",
    "data_loader_samples = SampleDataLoader(replay_buffer, batch_size_samples, shuffle_key)\n",
    "q = LQRQ(\n",
    "    n_actions_on_max=n_actions_on_max,\n",
    "    max_action_on_max=max_action_on_max,\n",
    "    network_key=q_network_key,\n",
    "    zero_initializer=True\n",
    ")\n",
    "\n",
    "validation_initial_weight = q.to_weights(q.params)\n",
    "\n",
    "weights_buffer = WeightsBuffer()\n",
    "weights_buffer.add(validation_initial_weight)\n",
    "\n",
    "q_random = LQRQ(\n",
    "    n_actions_on_max=n_actions_on_max,\n",
    "    max_action_on_max=max_action_on_max,\n",
    "    network_key=q_network_key,\n",
    "    zero_initializer=False\n",
    ")\n",
    "\n",
    "while len(weights_buffer) < n_weights:\n",
    "    weights = q_random.random_init_weights()\n",
    "    if not weights_filtering or q_random.to_params(weights)[\"LQRQNet\"][\"m\"] < 0:\n",
    "        weights_buffer.add(weights)\n",
    "\n",
    "weights_buffer.cast_to_jax_array()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Train linear PBO"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T15:07:04.168158Z",
     "iopub.status.busy": "2022-09-19T15:07:04.167754Z",
     "iopub.status.idle": "2022-09-19T15:07:09.088935Z",
     "shell.execute_reply": "2022-09-19T15:07:09.088481Z"
    }
   },
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "537726300224405b96d524ec16b8b974",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/3200 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0. 0. 0.]\n",
      "[-0.15485708  0.13041067 -0.77121866]\n",
      "[-0.17500959  0.179418   -0.87100869]\n",
      "[-0.17580572  0.18135403 -0.87495089]\n",
      "[-0.17583065  0.18141468 -0.87507439]\n",
      "[-0.17583142  0.18141656 -0.8750782 ]\n",
      "[-0.17583145  0.18141662 -0.87507832]\n",
      "[-0.17583145  0.18141662 -0.87507832]\n",
      "Optimal weights\n",
      "[-0.22707027  0.19779846 -0.8541705 ]\n"
     ]
    },
    {
     "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": [
    "from tqdm.notebook import tqdm\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from pbo.sample_collection.dataloader import SampleDataLoader\n",
    "from pbo.weights_collection.dataloader import WeightsDataLoader\n",
    "from pbo.networks.learnable_pbo import CustomLinearPBO\n",
    "\n",
    "\n",
    "data_loader_samples = SampleDataLoader(replay_buffer, batch_size_samples, shuffle_key)\n",
    "data_loader_weights = WeightsDataLoader(weights_buffer, batch_size_weights, shuffle_key)\n",
    "pbo_custom_linear = CustomLinearPBO(\n",
    "    q=q,\n",
    "    max_bellman_iterations=max_bellman_iterations,\n",
    "    network_key=pbo_network_key,\n",
    "    learning_rate=learning_rate,\n",
    "    initial_weight_std=initial_weight_std,\n",
    ")\n",
    "importance_iteration = jnp.ones(max_bellman_iterations + 1)\n",
    "\n",
    "cumulative_l2_loss = np.zeros((training_steps, fitting_steps)) * np.nan\n",
    "\n",
    "for training_step in tqdm(range(training_steps)):\n",
    "    params_target = pbo_custom_linear.params\n",
    "    for fitting_step in range(fitting_steps):\n",
    "        data_loader_weights.shuffle()\n",
    "        for batch_weights in data_loader_weights:\n",
    "            data_loader_samples.shuffle()\n",
    "            for batch_samples in data_loader_samples:\n",
    "                pbo_custom_linear.params, pbo_custom_linear.optimizer_state, l2_loss = pbo_custom_linear.learn_on_batch(\n",
    "                    pbo_custom_linear.params, params_target, pbo_custom_linear.optimizer_state, batch_weights, batch_samples, importance_iteration\n",
    "                )\n",
    "        cumulative_l2_loss[training_step, fitting_step] = l2_loss\n",
    "\n",
    "weights = np.zeros((max_bellman_iterations_validation + 1, q.weights_dimension))\n",
    "\n",
    "\n",
    "batch_iterated_weights = validation_initial_weight.reshape((1, -1))\n",
    "for bellman_iteration in range(max_bellman_iterations_validation + 1):\n",
    "    weights[bellman_iteration] = batch_iterated_weights[0]\n",
    "    print(weights[bellman_iteration])\n",
    "\n",
    "    batch_iterated_weights = pbo_custom_linear(pbo_custom_linear.params, batch_iterated_weights)\n",
    "\n",
    "print(\"Optimal weights\")\n",
    "print(env.optimal_weights)\n",
    "\n",
    "for training_step in range(0, training_steps, max(training_steps // 10, 1)):\n",
    "    plt.plot(cumulative_l2_loss[training_step], label=f\"Training step {training_step}\")\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Save data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T15:07:09.090819Z",
     "iopub.status.busy": "2022-09-19T15:07:09.090678Z",
     "iopub.status.idle": "2022-09-19T15:07:09.159325Z",
     "shell.execute_reply": "2022-09-19T15:07:09.158830Z"
    }
   },
   "outputs": [],
   "source": [
    "if not os.path.exists(\"figures/data/PBO_custom_linear/\"):\n",
    "    os.mkdir(\"figures/data/PBO_custom_linear/\")\n",
    "np.save(f\"figures/data/PBO_custom_linear/{max_bellman_iterations}_W_{n_samples}_{seed}.npy\", weights)\n",
    "np.save(f\"figures/data/PBO_custom_linear/{max_bellman_iterations}_Pi_{n_samples}_{seed}.npy\", env.greedy_V(weights))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Visualise"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T15:07:09.161383Z",
     "iopub.status.busy": "2022-09-19T15:07:09.161245Z",
     "iopub.status.idle": "2022-09-19T15:07:09.462814Z",
     "shell.execute_reply": "2022-09-19T15:07:09.462304Z"
    }
   },
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 504x504 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "from pbo.utils.lqr_weights_viewer import visualize\n",
    "\n",
    "weights[np.abs(weights) > 20] = np.nan\n",
    "\n",
    "visualize(weights, env.optimal_weights)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Understand"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T15:07:09.464669Z",
     "iopub.status.busy": "2022-09-19T15:07:09.464528Z",
     "iopub.status.idle": "2022-09-19T15:07:09.478051Z",
     "shell.execute_reply": "2022-09-19T15:07:09.477590Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Current loss: 0.050572187\n",
      "Loss with optimal params: 0.013090131\n"
     ]
    }
   ],
   "source": [
    "ideal_params = {\"CustomLinearPBONet\": {\"bias\": None, \"slope\": None}}\n",
    "ideal_params[\"CustomLinearPBONet\"][\"bias\"] = jnp.array([env.Q, env.S, env.R]).reshape((1, 3))\n",
    "ideal_params[\"CustomLinearPBONet\"][\"slope\"] = jnp.array([env.A ** 2, env.A * env.B,env.B ** 2]).reshape((1, 3))\n",
    "\n",
    "print(\"Current loss:\", pbo_custom_linear.learn_on_batch(\n",
    "    pbo_custom_linear.params, params_target, pbo_custom_linear.optimizer_state, batch_weights, batch_samples, importance_iteration\n",
    ")[-1])\n",
    "print(\"Loss with optimal params:\", pbo_custom_linear.learn_on_batch(\n",
    "    ideal_params, params_target, pbo_custom_linear.optimizer_state, batch_weights, batch_samples, importance_iteration\n",
    ")[-1])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T15:07:09.479805Z",
     "iopub.status.busy": "2022-09-19T15:07:09.479668Z",
     "iopub.status.idle": "2022-09-19T15:07:09.491477Z",
     "shell.execute_reply": "2022-09-19T15:07:09.491014Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Current bias: [-0.15485708  0.13041067 -0.77121866]\n"
     ]
    }
   ],
   "source": [
    "print(\"Current bias:\", pbo_custom_linear.params[\"CustomLinearPBONet\"][\"bias\"][0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T15:07:09.493268Z",
     "iopub.status.busy": "2022-09-19T15:07:09.493130Z",
     "iopub.status.idle": "2022-09-19T15:07:09.504815Z",
     "shell.execute_reply": "2022-09-19T15:07:09.504301Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Optimal bias: [-0.13297832  0.1290505  -0.80394006]\n"
     ]
    }
   ],
   "source": [
    "print(\"Optimal bias:\", jnp.array([env.Q, env.S, env.R]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T15:07:09.506585Z",
     "iopub.status.busy": "2022-09-19T15:07:09.506450Z",
     "iopub.status.idle": "2022-09-19T15:07:09.518117Z",
     "shell.execute_reply": "2022-09-19T15:07:09.517656Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Current slope: [ 0.151745   -0.36901706  0.7514024 ]\n"
     ]
    }
   ],
   "source": [
    "print(\"Current slope:\", pbo_custom_linear.params[\"CustomLinearPBONet\"][\"slope\"][0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T15:07:09.519832Z",
     "iopub.status.busy": "2022-09-19T15:07:09.519697Z",
     "iopub.status.idle": "2022-09-19T15:07:09.531197Z",
     "shell.execute_reply": "2022-09-19T15:07:09.530735Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Optimal slope: [ 0.51908076 -0.37926456  0.27710834]\n"
     ]
    }
   ],
   "source": [
    "print(\"Optimal slope:\", jnp.array([env.A ** 2, env.A * env.B, env.B ** 2]))"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3.8.10 ('env_cpu': venv)",
   "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.10"
  },
  "vscode": {
   "interpreter": {
    "hash": "af5525a3273d35d601ae265c5d3634806dd61a1c4d085ae098611a6832982bdb"
   }
  },
  "widgets": {
   "application/vnd.jupyter.widget-state+json": {
    "state": {
     "085317a41faa47ac9474a11263c75755": {
      "model_module": "@jupyter-widgets/controls",
      "model_module_version": "1.5.0",
      "model_name": "ProgressStyleModel",
      "state": {
       "_model_module": "@jupyter-widgets/controls",
       "_model_module_version": "1.5.0",
       "_model_name": "ProgressStyleModel",
       "_view_count": null,
       "_view_module": "@jupyter-widgets/base",
       "_view_module_version": "1.2.0",
       "_view_name": "StyleView",
       "bar_color": null,
       "description_width": ""
      }
     },
     "24b416bed013469ba4f2b1675a0cfd9b": {
      "model_module": "@jupyter-widgets/base",
      "model_module_version": "1.2.0",
      "model_name": "LayoutModel",
      "state": {
       "_model_module": "@jupyter-widgets/base",
       "_model_module_version": "1.2.0",
       "_model_name": "LayoutModel",
       "_view_count": null,
       "_view_module": "@jupyter-widgets/base",
       "_view_module_version": "1.2.0",
       "_view_name": "LayoutView",
       "align_content": null,
       "align_items": null,
       "align_self": null,
       "border": null,
       "bottom": null,
       "display": null,
       "flex": null,
       "flex_flow": null,
       "grid_area": null,
       "grid_auto_columns": null,
       "grid_auto_flow": null,
       "grid_auto_rows": null,
       "grid_column": null,
       "grid_gap": null,
       "grid_row": null,
       "grid_template_areas": null,
       "grid_template_columns": null,
       "grid_template_rows": null,
       "height": null,
       "justify_content": null,
       "justify_items": null,
       "left": null,
       "margin": null,
       "max_height": null,
       "max_width": null,
       "min_height": null,
       "min_width": null,
       "object_fit": null,
       "object_position": null,
       "order": null,
       "overflow": null,
       "overflow_x": null,
       "overflow_y": null,
       "padding": null,
       "right": null,
       "top": null,
       "visibility": null,
       "width": null
      }
     },
     "29aaebe637524d13967065ef9dc817d4": {
      "model_module": "@jupyter-widgets/base",
      "model_module_version": "1.2.0",
      "model_name": "LayoutModel",
      "state": {
       "_model_module": "@jupyter-widgets/base",
       "_model_module_version": "1.2.0",
       "_model_name": "LayoutModel",
       "_view_count": null,
       "_view_module": "@jupyter-widgets/base",
       "_view_module_version": "1.2.0",
       "_view_name": "LayoutView",
       "align_content": null,
       "align_items": null,
       "align_self": null,
       "border": null,
       "bottom": null,
       "display": null,
       "flex": null,
       "flex_flow": null,
       "grid_area": null,
       "grid_auto_columns": null,
       "grid_auto_flow": null,
       "grid_auto_rows": null,
       "grid_column": null,
       "grid_gap": null,
       "grid_row": null,
       "grid_template_areas": null,
       "grid_template_columns": null,
       "grid_template_rows": null,
       "height": null,
       "justify_content": null,
       "justify_items": null,
       "left": null,
       "margin": null,
       "max_height": null,
       "max_width": null,
       "min_height": null,
       "min_width": null,
       "object_fit": null,
       "object_position": null,
       "order": null,
       "overflow": null,
       "overflow_x": null,
       "overflow_y": null,
       "padding": null,
       "right": null,
       "top": null,
       "visibility": null,
       "width": null
      }
     },
     "2d7460b248cc417e829fc32ba96015f1": {
      "model_module": "@jupyter-widgets/controls",
      "model_module_version": "1.5.0",
      "model_name": "DescriptionStyleModel",
      "state": {
       "_model_module": "@jupyter-widgets/controls",
       "_model_module_version": "1.5.0",
       "_model_name": "DescriptionStyleModel",
       "_view_count": null,
       "_view_module": "@jupyter-widgets/base",
       "_view_module_version": "1.2.0",
       "_view_name": "StyleView",
       "description_width": ""
      }
     },
     "2e8fd64d178348a7903d62d12e566cec": {
      "model_module": "@jupyter-widgets/controls",
      "model_module_version": "1.5.0",
      "model_name": "HTMLModel",
      "state": {
       "_dom_classes": [],
       "_model_module": "@jupyter-widgets/controls",
       "_model_module_version": "1.5.0",
       "_model_name": "HTMLModel",
       "_view_count": null,
       "_view_module": "@jupyter-widgets/controls",
       "_view_module_version": "1.5.0",
       "_view_name": "HTMLView",
       "description": "",
       "description_tooltip": null,
       "layout": "IPY_MODEL_29aaebe637524d13967065ef9dc817d4",
       "placeholder": "​",
       "style": "IPY_MODEL_7b5e517a3a324d769c447f896527f66b",
       "value": " 3200/3200 [00:04&lt;00:00, 838.45it/s]"
      }
     },
     "4cc2254c2a5d4b66aa58fa09291976d8": {
      "model_module": "@jupyter-widgets/base",
      "model_module_version": "1.2.0",
      "model_name": "LayoutModel",
      "state": {
       "_model_module": "@jupyter-widgets/base",
       "_model_module_version": "1.2.0",
       "_model_name": "LayoutModel",
       "_view_count": null,
       "_view_module": "@jupyter-widgets/base",
       "_view_module_version": "1.2.0",
       "_view_name": "LayoutView",
       "align_content": null,
       "align_items": null,
       "align_self": null,
       "border": null,
       "bottom": null,
       "display": null,
       "flex": null,
       "flex_flow": null,
       "grid_area": null,
       "grid_auto_columns": null,
       "grid_auto_flow": null,
       "grid_auto_rows": null,
       "grid_column": null,
       "grid_gap": null,
       "grid_row": null,
       "grid_template_areas": null,
       "grid_template_columns": null,
       "grid_template_rows": null,
       "height": null,
       "justify_content": null,
       "justify_items": null,
       "left": null,
       "margin": null,
       "max_height": null,
       "max_width": null,
       "min_height": null,
       "min_width": null,
       "object_fit": null,
       "object_position": null,
       "order": null,
       "overflow": null,
       "overflow_x": null,
       "overflow_y": null,
       "padding": null,
       "right": null,
       "top": null,
       "visibility": null,
       "width": null
      }
     },
     "537726300224405b96d524ec16b8b974": {
      "model_module": "@jupyter-widgets/controls",
      "model_module_version": "1.5.0",
      "model_name": "HBoxModel",
      "state": {
       "_dom_classes": [],
       "_model_module": "@jupyter-widgets/controls",
       "_model_module_version": "1.5.0",
       "_model_name": "HBoxModel",
       "_view_count": null,
       "_view_module": "@jupyter-widgets/controls",
       "_view_module_version": "1.5.0",
       "_view_name": "HBoxView",
       "box_style": "",
       "children": [
        "IPY_MODEL_8ee475cfdae043db90df3f1b700db0de",
        "IPY_MODEL_6343de5c3bcf417c964991b66c555ccf",
        "IPY_MODEL_2e8fd64d178348a7903d62d12e566cec"
       ],
       "layout": "IPY_MODEL_d59b7c0bea75432b913efb256d61bb7b"
      }
     },
     "6343de5c3bcf417c964991b66c555ccf": {
      "model_module": "@jupyter-widgets/controls",
      "model_module_version": "1.5.0",
      "model_name": "FloatProgressModel",
      "state": {
       "_dom_classes": [],
       "_model_module": "@jupyter-widgets/controls",
       "_model_module_version": "1.5.0",
       "_model_name": "FloatProgressModel",
       "_view_count": null,
       "_view_module": "@jupyter-widgets/controls",
       "_view_module_version": "1.5.0",
       "_view_name": "ProgressView",
       "bar_style": "success",
       "description": "",
       "description_tooltip": null,
       "layout": "IPY_MODEL_24b416bed013469ba4f2b1675a0cfd9b",
       "max": 3200.0,
       "min": 0.0,
       "orientation": "horizontal",
       "style": "IPY_MODEL_085317a41faa47ac9474a11263c75755",
       "value": 3200.0
      }
     },
     "7b5e517a3a324d769c447f896527f66b": {
      "model_module": "@jupyter-widgets/controls",
      "model_module_version": "1.5.0",
      "model_name": "DescriptionStyleModel",
      "state": {
       "_model_module": "@jupyter-widgets/controls",
       "_model_module_version": "1.5.0",
       "_model_name": "DescriptionStyleModel",
       "_view_count": null,
       "_view_module": "@jupyter-widgets/base",
       "_view_module_version": "1.2.0",
       "_view_name": "StyleView",
       "description_width": ""
      }
     },
     "8ee475cfdae043db90df3f1b700db0de": {
      "model_module": "@jupyter-widgets/controls",
      "model_module_version": "1.5.0",
      "model_name": "HTMLModel",
      "state": {
       "_dom_classes": [],
       "_model_module": "@jupyter-widgets/controls",
       "_model_module_version": "1.5.0",
       "_model_name": "HTMLModel",
       "_view_count": null,
       "_view_module": "@jupyter-widgets/controls",
       "_view_module_version": "1.5.0",
       "_view_name": "HTMLView",
       "description": "",
       "description_tooltip": null,
       "layout": "IPY_MODEL_4cc2254c2a5d4b66aa58fa09291976d8",
       "placeholder": "​",
       "style": "IPY_MODEL_2d7460b248cc417e829fc32ba96015f1",
       "value": "100%"
      }
     },
     "d59b7c0bea75432b913efb256d61bb7b": {
      "model_module": "@jupyter-widgets/base",
      "model_module_version": "1.2.0",
      "model_name": "LayoutModel",
      "state": {
       "_model_module": "@jupyter-widgets/base",
       "_model_module_version": "1.2.0",
       "_model_name": "LayoutModel",
       "_view_count": null,
       "_view_module": "@jupyter-widgets/base",
       "_view_module_version": "1.2.0",
       "_view_name": "LayoutView",
       "align_content": null,
       "align_items": null,
       "align_self": null,
       "border": null,
       "bottom": null,
       "display": null,
       "flex": null,
       "flex_flow": null,
       "grid_area": null,
       "grid_auto_columns": null,
       "grid_auto_flow": null,
       "grid_auto_rows": null,
       "grid_column": null,
       "grid_gap": null,
       "grid_row": null,
       "grid_template_areas": null,
       "grid_template_columns": null,
       "grid_template_rows": null,
       "height": null,
       "justify_content": null,
       "justify_items": null,
       "left": null,
       "margin": null,
       "max_height": null,
       "max_width": null,
       "min_height": null,
       "min_width": null,
       "object_fit": null,
       "object_position": null,
       "order": null,
       "overflow": null,
       "overflow_x": null,
       "overflow_y": null,
       "padding": null,
       "right": null,
       "top": null,
       "visibility": null,
       "width": null
      }
     }
    },
    "version_major": 2,
    "version_minor": 0
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
