{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# FQI on the chain walk environment\n",
    "\n",
    "## Define parameters"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T11:20:30.061112Z",
     "iopub.status.busy": "2022-09-19T11:20:30.060951Z",
     "iopub.status.idle": "2022-09-19T11:20:30.701625Z",
     "shell.execute_reply": "2022-09-19T11:20:30.701128Z"
    }
   },
   "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",
    "n_states = parameters[\"n_states\"]\n",
    "n_actions = 2\n",
    "sucess_probability = parameters[\"sucess_probability\"]\n",
    "gamma = parameters[\"gamma\"]\n",
    "env_seed = parameters[\"env_seed\"]\n",
    "\n",
    "# Sample collection\n",
    "n_repetitions = parameters[\"n_repetitions\"]\n",
    "n_samples = n_states * n_actions * n_repetitions\n",
    "\n",
    "# Trainings\n",
    "max_bellman_iterations = parameters[\"max_bellman_iterations\"]\n",
    "fitting_steps = parameters[\"fitting_steps_fqi\"]\n",
    "batch_size_samples = parameters[\"batch_size_samples\"]\n",
    "learning_rate = {\"first\": parameters[\"starting_lr_fqi\"], \"last\": parameters[\"ending_lr_fqi\"], \"duration\": fitting_steps * n_samples // batch_size_samples}\n",
    "\n",
    "# Search for an unused seed\n",
    "max_used_seed = 0\n",
    "if not os.path.exists(\"figures/data/FQI/\"):\n",
    "    os.makedirs(\"figures/data/FQI/\")\n",
    "for file in os.listdir(\"figures/data/FQI/\"):\n",
    "    if int(file.split(\"_\")[0]) == max_bellman_iterations and int(file.split(\"_\")[2][:-4]) > max_used_seed:\n",
    "        max_used_seed = int(file.split(\"_\")[2][:-4])\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, dummy_q_network_key, _ = jax.random.split(key, 3) # 3 keys are generated to be coherent with the other trainings"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Define environment"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T11:20:30.704179Z",
     "iopub.status.busy": "2022-09-19T11:20:30.703994Z",
     "iopub.status.idle": "2022-09-19T11:20:31.061868Z",
     "shell.execute_reply": "2022-09-19T11:20:31.061352Z"
    }
   },
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from pbo.environments.chain_walk import ChainWalkEnv\n",
    "\n",
    "\n",
    "states = np.arange(n_states)\n",
    "actions = np.arange(n_actions)\n",
    "states_boxes = np.arange(n_states + 1) - 0.5\n",
    "actions_boxes = np.arange(n_actions + 1) - 0.5\n",
    "\n",
    "env = ChainWalkEnv(env_key, n_states, sucess_probability, gamma)"
   ]
  },
  {
   "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-19T11:20:31.065428Z",
     "iopub.status.busy": "2022-09-19T11:20:31.065268Z",
     "iopub.status.idle": "2022-09-19T11:20:32.236395Z",
     "shell.execute_reply": "2022-09-19T11:20:32.235818Z"
    }
   },
   "outputs": [],
   "source": [
    "import jax.numpy as jnp\n",
    "from pbo.sample_collection.replay_buffer import ReplayBuffer\n",
    "\n",
    "\n",
    "replay_buffer = ReplayBuffer()\n",
    "\n",
    "for state in states:\n",
    "    for action in actions:\n",
    "        # Need to repeat the samples to capture the randomness\n",
    "        for _ in range(n_repetitions):\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-19T11:20:32.239083Z",
     "iopub.status.busy": "2022-09-19T11:20:32.238883Z",
     "iopub.status.idle": "2022-09-19T11:20:33.106419Z",
     "shell.execute_reply": "2022-09-19T11:20:33.105914Z"
    }
   },
   "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, states_boxes, actions_boxes, replay_buffer.rewards)\n",
    "samples_visu_mesh = TwoDimesionsMesh(states, 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",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Train FQI"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T11:20:33.108744Z",
     "iopub.status.busy": "2022-09-19T11:20:33.108448Z",
     "iopub.status.idle": "2022-09-19T11:20:55.494512Z",
     "shell.execute_reply": "2022-09-19T11:20:55.493991Z"
    }
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0]\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "622fb3e1c5954bc4a13acca0acd2fc07",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/2 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 0]\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXoAAAD4CAYAAADiry33AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAAAx7ElEQVR4nO2debgcdZnvP2/32bKRnSQkgQQIQoAkA0lA9iGyiRIcowMyTpyLF2auzL3z6DgTx3sVmVHR64ijch1Q0AAiAQTJYJBVZIckEJYQQkLMcrLnJDlZz9Ld7/2jqs+prtPdp7uq1/L9PM95Tu311q+rvvXW+771K1FVDMMwjOgSq7YBhmEYRnkxoTcMw4g4JvSGYRgRx4TeMAwj4pjQG4ZhRJyGahvgZ9SoUTpp0qRqm2EYhlFXLF++fJeqjs42r+aEftKkSSxbtqzaZhiGYdQVIrIh1zwL3RiGYUQcE3rDMIyIY0JvGIYRcWouRm8YRjTo7u6mtbWVjo6OapsSKVpaWpgwYQKNjY0Fr2NCbxhGWWhtbWXIkCFMmjQJEam2OZFAVWlra6O1tZXJkycXvJ6FbgzDKAsdHR2MHDnSRL6EiAgjR44s+inJhN4wjLJhIl96grRpdIS+8wA8801oXV5tSwzDMGqK6Ah9ogOe+y5seb3alhiGUQMMHjwYgPXr13PvvfeWdNvf+ta3MsbPOuuskmx39+7dXHTRRUyZMoWLLrqIPXv2lGS70RF6cQ9FU9W1wzCMmiKI0CcSibzz/UL/0ksvFW1XNm6++WbmzJnDmjVrmDNnDjfffHNJthshoXfjVib0hmF4WLBgAc8//zwzZszglltuIZlM8uUvf5lZs2Yxbdo0brvtNgCeffZZzj33XK644gqmTp0KwJVXXsnpp5/OySefzO23396zvcOHDzNjxgyuueYaoPfpQVX58pe/zCmnnMKpp57KokWLerZ9wQUXMG/ePE488USuueYasn3d75FHHmH+/PkAzJ8/n9/85jclaYPolFeaR28YNcs3/msl727ZV9JtTj3qCL7+8ZP7Xe7mm2/me9/7Ho8++igAt99+O0OHDmXp0qV0dnZy9tlnc/HFFwPw+uuv88477/SULt55552MGDGCw4cPM2vWLD75yU9y88038+Mf/5gVK1b02ddDDz3EihUrePPNN9m1axezZs3ivPPOA+CNN95g5cqVHHXUUZx99tm8+OKLnHPOORnrb9++nXHjxgEwduxYtm/fHrh9vETIozehNwyjf5544gnuuusuZsyYwRlnnEFbWxtr1qwBYPbs2Rn16T/84Q+ZPn06Z555Jps2bepZLhcvvPACV199NfF4nDFjxnD++eezdOnSnm1PmDCBWCzGjBkzWL9+fd5tiUjJqpai59GnktW1wzCMPhTieVcKVeVHP/oRl1xyScb0Z599lkGDBmWMP/XUU7z88ssMHDiQCy64INRbvs3NzT3D8Xg8ax5gzJgxbN26lXHjxrF161aOPPLIwPvzUpBHLyKXishqEVkrIguyzD9PRF4XkYSIzPNMnyEiL4vIShF5S0T+siRWZzUy7vw3j94wDA9Dhgxh//79PeOXXHIJP/nJT+ju7gbg/fff5+DBg33Wa29vZ/jw4QwcOJD33nuPV155pWdeY2Njz/pezj33XBYtWkQymWTnzp0899xzzJ49u2Bbr7jiChYuXAjAwoULmTt3bsHr5qNfoReROHArcBkwFbhaRKb6FtsIfA7wp7YPAX+tqicDlwI/EJFhIW3OYaiFbgzD6Mu0adOIx+NMnz6dW265hc9//vNMnTqV0047jVNOOYXrr78+q3d96aWXkkgkOOmkk1iwYAFnnnlmz7zrrruOadOm9SRj03ziE59g2rRpTJ8+nQsvvJDvfve7jB07tmBbFyxYwJNPPsmUKVN46qmnWLCgj18dCMmW+c1YQOTDwI2qeok7/hUAVf12lmV/ATyqqg/m2NabwDxVzRnomjlzpgb68EgyAf86Ev78f8P5Xy5+fcMwSsqqVas46aSTqm1GJMnWtiKyXFVnZlu+kNDNeGCTZ7zVnVYUIjIbaAI+yDLvOhFZJiLLdu7cWeym3Y2YR28YhpGNilTdiMg44G7gb1T7KrGq3q6qM1V15ujRWT95WMhO3I2Z0BuGYXgpROg3AxM94xPcaQUhIkcAvwW+qqqv9Ld8YEQAAbWqG8MwDC+FCP1SYIqITBaRJuAqYHEhG3eXfxi4K1fcvqTE4ubRG4Zh+OhX6FU1AdwAPA6sAu5X1ZUicpOIXAEgIrNEpBX4FHCbiKx0V/80cB7wORFZ4f7NKMeBAE6c3oTeMAwjg4JemFLVJcAS37SveYaX4oR0/OvdA9wT0sbCMaE3DMPoQ3S6QAATesMweqjHboofeOABTj75ZGKxGIHKzHMQQaHP/16AYRh/WtRTN8WnnHIKDz30UE9HaKUigkJvHr1hGL3UUzfFJ510Eh/60IdK3gbR6dQMnBJL69TMMGqPxxbAtrdLu82xp8Jl/X+Yo566KS4XEfPorbzSMIz81Es3xaUkYh69hW4MoyYpwPOuFLXcTXG5iJhHb0JvGEYm9dRNcbkwoTcMI9LUUzfFDz/8MBMmTODll1/m8ssv7/PUEZR+uymuNIG7KQb495Pg+Dkw98elNcowjKKxborLRzm6Ka4fzKM3DMPoQ7SEPmZCbxiG4SdaQm8evWHUFLUWGo4CQdrUhN4wjLLQ0tJCW1ubiX0JUVXa2tpoaWkpaj2rozcMoyxMmDCB1tZWAn8e1MhKS0sLEyb06Sw4Lyb0hmGUhcbGxoy3TI3qEb3QjfV1YxiGkUHEhN76ujEMw/ATMaG3/ugNwzD8REzoxTx6wzAMHxETekvGGoZh+DGhNwzDiDgRFHqrujEMw/ASLaGPWdWNYRiGn4KEXkQuFZHVIrJWRBZkmX+eiLwuIgkRmeebN19E1rh/80tleHZDLXRjGIbhp1+hF5E4cCtwGTAVuFpEpvoW2wh8DrjXt+4I4OvAGcBs4OsiMjy82bmMtfJKwzAMP4V49LOBtaq6TlW7gPuAud4FVHW9qr4F+N3pS4AnVXW3qu4BngQuLYHd2TGP3jAMow+FCP14YJNnvNWdVggFrSsi14nIMhFZFqoDJKujNwzD6ENNJGNV9XZVnamqM0ePHh18Q9bXjWEYRh8KEfrNwETP+AR3WiGEWbd4LHRjGIbRh0KEfikwRUQmi0gTcBWwuMDtPw5cLCLD3STsxe608mCdmhmGYfShX6FX1QRwA45ArwLuV9WVInKTiFwBICKzRKQV+BRwm4isdNfdDfwrzs1iKXCTO608mEdvGIbRh4I+PKKqS4Alvmlf8wwvxQnLZFv3TuDOEDYWjgm9YRhGH2oiGVsyrI7eMAyjDxETeiuvNAzD8BMxoY+RSiWqbYVhGEZNESmh37yvi3U79vPetn3VNsUwDKNmiJTQb9vXhaC8t3V/tU0xDMOoGSIl9EqMGClEqm2JYRhG7RApoU+JEMOqbgzDMLxESugVE3rDMAw/ERP6GDFJIRa7MQzD6CFSQp8kZh69YRiGj0gJvYo4ydhqG2IYhlFDREvo1Tx6wzAMP9ESehEG0cGAw1urbYphGEbNECmhTyEMlE4+8rsLq22KYRhGzRApoddoHY5hGEZJiJQypqJ1OIZhGCUhUspoQm8YhtGXSCmjWmGlYRhGH6Il9PZGrGEYRh8iJfQZoRv7pKBhGAYQMaHPCN3YJwUNwzCAiAl9pkdvQm8YhgEFCr2IXCoiq0VkrYgsyDK/WUQWufNfFZFJ7vRGEVkoIm+LyCoR+UqJ7c/AhN4wDKMv/Qq9iMSBW4HLgKnA1SIy1bfYtcAeVT0euAX4jjv9U0Czqp4KnA5cn74JlIOM7sxSyXLtxjAMo64oxKOfDaxV1XWq2gXcB8z1LTMXWOgOPwjMEadTeAUGiUgDMADoAsr25W41j94wDKMPhQj9eGCTZ7zVnZZ1GVVNAO3ASBzRPwhsBTYC31PV3f4diMh1IrJMRJbt3Lmz6INIk1FeaUJvGIYBlD8ZOxtIAkcBk4Evicix/oVU9XZVnamqM0ePHh14Zyn1Cr2FbgzDMKAwod8MTPSMT3CnZV3GDdMMBdqAzwC/U9VuVd0BvAjMDGt0LlSsjt4wDMNPIUK/FJgiIpNFpAm4CljsW2YxMN8dngc8o6qKE665EEBEBgFnAu+VwvBsZFTdWDLWMAwDKEDo3Zj7DcDjwCrgflVdKSI3icgV7mJ3ACNFZC3wRSBdgnkrMFhEVuLcMH6uqm+V+iDSpOyFKcMwjD40FLKQqi4Blvimfc0z3IFTSulf70C26eXC3ow1DMPoS6TejMX7vVhLxhqGYQARE/qYV9zNozcMwwAiJvRxr9BbMtYwDAOImNDHMI/eMAzDT7SEXhO9I1ZHbxiGAURM6DNCN5aMNQzDAKIm9Hg9egvdGIZhQMSEPmbJWMMwjD5ESujjXi/ePHrDMAwgYkIfs9CNYRhGHyIl9JaMNQzD6EukhD6zjt7KKw3DMCBiQr8rPqZ3xEI3hmEYQMSE/uHBV3Nn4lJnxKpuDMMwgIgJfULiPJk63Rkxj94wDAOImNCDp096S8YahmEABX54pF5QhZS69y7z6A3DMIAIevQ9nxM0oTcMwwAiJvSKJ3STMqE3DMOAiAk9CkksdGMYhuElWkIPpHqE3pKxhmEYEDGhV9Qj9ObRG4ZhQIFCLyKXishqEVkrIguyzG8WkUXu/FdFZJJn3jQReVlEVorI2yLSUkL7M0ipJWMNwzD89Cv0IhIHbgUuA6YCV4vIVN9i1wJ7VPV44BbgO+66DcA9wN+q6snABUB3yaz3kVLtFXp7M9YwDAMozKOfDaxV1XWq2gXcB8z1LTMXWOgOPwjMEREBLgbeUtU3AVS1TbV8wfOUJWMNwzD6UIjQjwc2ecZb3WlZl1HVBNAOjAROAFREHheR10Xkn8KbnJtUSj1vxprQG4ZhQPnfjG0AzgFmAYeAp0Vkuao+7V1IRK4DrgM4+uijA+/MCd2YR28YhuGlEI9+MzDRMz7BnZZ1GTcuPxRow/H+n1PVXap6CFgCnObfgarerqozVXXm6NGjiz8KFwvdGIZh9KUQoV8KTBGRySLSBFwFLPYtsxiY7w7PA55RVQUeB04VkYHuDeB84N3SmN6XVMqSsYZhGH76Dd2oakJEbsAR7Thwp6quFJGbgGWquhi4A7hbRNYCu3FuBqjqHhH5Ps7NQoElqvrbMh2LE7qxTs0MwzAyKChGr6pLcMIu3mlf8wx3AJ/Kse49OCWWZSejvNKE3jAMA4jam7FqXSAYhmH4iZTQJ63qxjAMow+REvrMN2NN6A3DMCBqQp+y8krDMAw/0RJ6tTdjDcMw/ERO6FP2cXDDMIwMIib0FroxDMPwEymhV2/Vjb0ZaxiGAURM6JMpK680DMPwEymhz/zClFbXGMMwjBohYkJvyVjDMAw/0RL6lKIWujEMw8ggWkLvRmtSxCwZaxiG4RIxoXeUPkXMPHrDMAyXSAl9Ov+qIib0hmEYLpES+qSr9ErMkrGGYRgukRL6zNCNlVcahmFAhIReVT2hG4vRG4ZhpImM0Kc8DnwKsaobwzAMlwgJfa/SW9WNYRhGL5EUekvGGoZh9BIZoffmXlMWozcMw+ghMkKf9ATpRVOw/Bfw7iPVM8gwDKNGKEjoReRSEVktImtFZEGW+c0issid/6qITPLNP1pEDojIP5bI7j54QzdDUu3OwJJ/KtfuDMMw6oZ+hV5E4sCtwGXAVOBqEZnqW+xaYI+qHg/cAnzHN//7wGPhzc2Nt+omhjvSPKScuzQMw6gLCvHoZwNrVXWdqnYB9wFzfcvMBRa6ww8Cc0REAETkSuCPwMqSWJyDVCrLC1Im9IZhGAUJ/Xhgk2e81Z2WdRlVTQDtwEgRGQz8M/CNfDsQketEZJmILNu5c2ehtmeQyvYmrAm9YRhG2ZOxNwK3qOqBfAup6u2qOlNVZ44ePTrQjhriMT5y0pGZE03oDcMwChL6zcBEz/gEd1rWZUSkARgKtAFnAN8VkfXAPwD/IiI3hDM5O0MHNPKz+bM4Y/KI3onNR5RjV4ZhGHVFQwHLLAWmiMhkHEG/CviMb5nFwHzgZWAe8IyqKnBuegERuRE4oKo/LoHdORHxjDQ0lXNXhmEYdUG/Qq+qCdcLfxyIA3eq6koRuQlYpqqLgTuAu0VkLbAb52ZQFWJepU8mqmWGYRhGzVCIR4+qLgGW+KZ9zTPcAXyqn23cGMC+oskU+q5K7NIwDKOmicybsWkyQjep7qrZYRiGUStEUOiFewf+lTOSDCn0+7bAOw+FN8owDKOKRE7oYwJ3Nnya1JhTIRUyRr/il/Dgf+u9Yex4Dza8DI9/FQ7syFz2uf8Ly36eOW3/Nnj7wQD7/RW8fpcz3PYBbHu7+G0A7Frj2LpjFXzwe/j55dB9ONi2gvLqbfDQ9cHXT3RCx77s83asgu3vOsNb34INLwXfTxj2bbEvmhk1TeSEXoC1Ow7w7vZD4T36ZDegvR8x+cPNcPeV8PKP4YNnMpdd+RtY7evl4c374NfXFi+uK34Jr9/tDD99Eyz++wDGA2uedGz92UWw6K9gwwtwYHuwbQWldRlseDH4+s9+G35xefZ5j/8L/O6fneE/fAd+16cbpvKzdyPccnK4YwTnhrX2aXhvCXQddKapwpYVxW9rxa/gV1cXv16yG+66Eja+Wvy6QUh29x6rl0o7I0HYvS68vlSQyAl9OhnbkYqHj9GnnwjS/xOdkOhwpyX7Lut/gkifCMU+WaSSvf3pdx8OfuKn99u1H7oO9G67kmRrl2LYvz33zan7MHS7v0eiszoX3qHdTpfYB4O90d3DSz+ERZ+F+67u7XV1w4tw+/mwc3Vx29ryOvzx+eJt6GiHdb+HzcuLXzcIz34bfv7RzGnbV8K3xjtPsrVK53649Qx459fVtqRgIif0bhc7JIiHL6/0C71XsPzi5RXnXOsXs1/vukGFMtt69Sb0+dYvVTuFId2eYds12QXdrnebvrEf3uv872gv0qaAbRH0fA3Kvq2wf6tv2hbnOtq/rTI2BKHroPN7HdpdbUsKJnJCH3Orbrq1FB69e/GmP2LivQCyiXo2L9+7nULRJKRSnuGAIpLtK1uVFkNNhRPBfMfvvblWTehLJI4ZTkT6vEv2nVfotsIIfaW+zpbNzkrfbILg/33qgILq6OuJdHml49GXSOizCbZffLIJUlChz/BUQwh9tvUqfQFluwGWav1UovcmHPaGEpQeMQ6574xzy/8kWez5kwomQv7zvdxkFfoK2xCEergZ+YigR58O3TSE/yH8F3E+oU8l++4vsEeWzDyZShm6qbQXEjp0k6Vds80Le0MJSjk8+j7nXQCPXlO9T4XF2lCpdvQ+uVbLhiCY0FeftEffXRKP3veDahavyzteKu8kIySRJfZfzHYKmVZOwtifXj9vjD4qoZtSevQBQzAV9+iz/LaVDh8FoSeUW8M2+oic0HclnHrmBCWM0YdNxhZ9wVkyNmP9XO2nyUwxrIY4pDyho1DbyXJuBb2JBA0nlSoMVSh1H7oxoa8aXUnnguumoQRVN2GTsQEvnFSiV0DChCRqIkafxyMvaP08YYgMjz7kfoJSlmSsz2MMErqp5HpBySb0QcOdlcRCN9WnK+GcKKWpusnzCF1IjD5U1U36aSBEkjGrR1/pqpv0hRvQ402vn7WCKJk5v56Tsd4ngnwhw0IIfYOooEePZp4b9eAtm9BXn66Ec9KUpOombzI2iyeS06OvUjI2m0BUIxnr/V/0+nmENKrJWL/YBRXsYsNJlS4bzLa/ehBR/xNXHRA9ofeGbkru0Xsvxmyhm1wefZCqiWTf4WKpiRh9yEfxfG0YqWRsvhh9pTz6KiRj/furpxh9LSeMfUSujj7Toy9RjL7gqhvfD58v7NDffksRe66F0E1YLzHfhZ+O36eXi0oyVj0vy/nnFbOteojR+/dXqjeNy0k9PHX4iJ5H7xX6Sr0Zq+osk7MLhCDJWG95ZSpY74hZE5j1FrrJE4bQLDfESvciWTKPPk+MPrBnHrAss6Ixet/+6kFE6yFh7COyQt9NHC11HX2uZGwurzNweVwq/ON7rv1WLRlbDo8+2feiC+tZF0stxuhDPwlUSOiznRuWjC0L0RN6N0af0AbE28VwEPwXTK5kbK6TM2jSxhtvDuM9ZFunasnYgPvNd/xZ26nCx1eyLhBKGaMPWu2V5cm1nGS7kdWDt1wP4SUfkRP6Tm/oBsJV3uTzrLw/cq6TM2wyVjWc95CrJLGSlCp0U0jVTZj9BKVU+83I/+Sp9irGpiAv6gVZLyhZc1/1IPR18NThI7JC3+0K/d4Dh4JvrNA3Y3N69CGrJrw19GE6qcpmU6VIhfQS+6u6Secvwu4nKKXy7vJVnlSqeqYmkrF1EBaph5uRj8gJfXcy06N/cOkfg2/Mn4zVLF58xnIlqLpR9XW9G0JIcsW1K0lYLzHX8adSgPbOq1bJW6lqz/MlJCsm9BUOSWR9Uq6D0sV6uBn5KEjoReRSEVktImtFpM/32kSkWUQWufNfFZFJ7vSLRGS5iLzt/r+wxPb3IV100e1Wjia6OkNsLF+MvpBkbIALzr/dUMnYWvDoQz7m9ncTTe+jWo/T5UjGlqL3Su/65V4vKFlfRKyD+HcUQzciEgduBS4DpgJXi8hU32LXAntU9XjgFuA77vRdwMdV9VRgPnB3qQzvj7RHn0iUIkZfaOgmV4y+GI/eJ2D1nowNm1zrr23T+6hWEq9kQp/Fo69052QVf2EqT5FDLXvL2Z7wa5xCPPrZwFpVXaeqXcB9wFzfMnOBhe7wg8AcERFVfUNVt7jTVwIDRKS5FIb3R4/Qd3cF30ifZGwuj95T2uet4w5y0voFLMyJXxOhm5AeWn+J7vRwtbyssH35pMkXow+aVK3LGH0dxL/r4WbkoxChHw9s8oy3utOyLqOqCaAdGOlb5pPA66raJ5YiIteJyDIRWbZzZ8iPLLt0qyP0yUSI0I0/wZfLo88Wr/cOFyM+3u0mE+G8h2w15fXm8eZMdHvGvZVVVUvGlsGjDxujL/acqfRTUd0mY+vARh8VScaKyMk44Zzrs81X1dtVdaaqzhw9enSoff36785i+sRhzhemgEGHtvazRh68nro3+Qf9e/cZ6xcj9F4B68w+veBtRcCjz5mM9Yx7b+b1diNLk7e8ssJVN5V66SxbiKla70MUQxRj9MBmYKJnfII7LesyItIADAXa3PEJwMPAX6vqB2EN7o/TjxnODX9+fE/o5m83/WPw1+LzdSyWq9+bsDXBpRSwXCWJlaRkVTd5QjeJjt7hir8ZW6qqm2xebdBYe4WrdYKSre3qwVuuh4Sxj0KEfikwRUQmi0gTcBWw2LfMYpxkK8A84BlVVREZBvwWWKCqL5bI5n5pjAsJ76ElA8bpvSdd3hhxP6JfzEnrPem9dpes6qaCJ6e/VDQIuW4UOdupTj36fOGLwLH2oN+MrYUYfQ2LaD3kEXz0K/RuzP0G4HFgFXC/qq4UkZtE5Ap3sTuAkSKyFvgikC7BvAE4HviaiKxw/44s+VH4aIzHaMETt/V6fMXg9azyhQ76E/2gMfrQHn2W/VayUiDbxzSKpZCqm0TIEFcYSiFM6U7xesb9uaEKVc9UvI7ekrGVoqBuilV1CbDEN+1rnuEO4FNZ1vs34N9C2lg0DTHh8dRMVqSOZUZsXaYQFEPG5/z6qfroGS6T0Ad6M7bKoZtc7VIo+Z4IMpLWVfToSxFTzvlGdUBBqZdOzfK9MFXLHn099MfjI3JvxgI0xGMkaOCXyY84E7oPB9tQoR59Ls81SPy2lMnYavd1E1rove3qC0N4x6vq0ZfAu8t1EwudVK1QbD8IGW825+miuRaph5uRj0gKfVPcOaxObQIg2R0wdOPtwqAqydiQnmrVPfocbVSK9Uv55BOGcgp94BefAgpRJV8EyvlEXA9CX6IEfAWJpNA3xAWADhoB6O4M69EHTMYGOWlzVZPUY3llrhtgkPXzhm6qWV6ZvuhDVPv0+WBNnvc3irEpcOimEh59jt+20l0lB6EebkY+Iin0ja7Qd+J49N0dB4NtyNuffKVi9BnVJGGTsTm+ylQpQnv0eW4UpUxah6EUycNSxuj9nb0VZUcFBSzXE3HQsFMlqYeEsY9ICn1DzA3dlMyjzxaj7ycuD8EevTMErM5DN/m+sVv0+nlCZ9WM0ZckGdtPjL6oHE+ep6BC1w3bnUMx+4IcdfS1LPR1YKOPSAp9Y0M6Ru8IfbKrFKGbPB5lf8nYwDH6kC8CVVvoM9oiiP35YvQlTFqHoRwx+j69VwZ8IqxUyCcI/b1ZXsvesgl9bdAYS8fondBNIqjQe5OxuS5G6D+xFLjqJqRHX1NVNyGfSPLdaMM++YShJEJfytBNDgekGDuqGaOv1gdkiqEe8gg+Iin0DfHM0E2iFMnYfDHiklbdlDD2XO03Y8tadZPDo6/ah0dChDtyHVtYoQ8aurGqm/zUg40+Iir0bjK2J3QToLwyIwbfXzK2v8fQUiRjS1R1U61kbDmrbiKbjA3iKIR4G7mSIYlSXjOVph5s9BFJoU/X0Q8YOAiAVJAXpvxC3kdo+ovLB6x+KGkyNptHX0fJ2HxhiFJ2/haGUnh3fZ4WfUnYoOdP0XX0NRC6qafeK2u5MshHQV0g1BstjXGe+dL57Nq1CxZBKsgLU/4Tsc+bmf2VVAZ8hI5sMjbsm7F5ciQZTz4V7r2yrFU3ATzHUKEbS8YWRD3kEXxEUugBjh09mI4Ox5NPBUnG+j8Eni90kM1zzfUh8f4oVTLW209Mru2Xm7BCX3DophaSsaUU+lLF6IPW0VczRl8HNer1cDPyEcnQTZqmphZSKmiQ3iv9J2KxXSAE9uhLFHvO9QRQUaEP2XtlwVU3HdmnV4KSxOj78+iL2HZQBwMyQ0ZBv+FQKLnsrIf4d0aRRpnbqUREWuibG+NOiWWg0E1/ydhc5ZXZhD5ojD5EMjaXOFQ0GRsijAC529g/z/vkU7WqmzAeve+m7P8ObdDQX9BkrH875SAKHj1U/kM3AYm40MfopJH3N+9k7Y4Dxa3cbzK2vxhjKvv8/ihVFwi5lq9WjD5Q1U2ONvaP13sytmZi9CF/r6L2Vccx+nxvbNco0Rb6eJxOGjl48CCLlm4sbmX/42TBydiwoZtcAhbwMbzP9GpV3YSM0eft68Ybuqn0pwTLIPSackICQao78t0ci1q3zOdJf9dPLVe0VLKdSkS0hb4xRqc20iJd7D3U3f8KXvwiky8ZmG24FEIfJiSR06OvoBCGDd3kK8+slU8J+sMsYbbhxfsUWalkbNhy2GLIGfqsA48+7HldBSIt9E3xGEliXBl/iTO33lXcyn3KK7MITToR09/HjYN2ShUmyVgLHn0Y7xLyX1ClSlqHpRzJ2PS0sH0l1WOMvi7q6M2jryliMeHY2DYALtrzQHEr+2Ps2cQ6W58X2T7eUJIXpkJ4Z7m2X25Cd4FQYNVNVZOxpQjdZPPoE8FEL1TVTQ0Ifb159JaMrS1aY2OLW6FPMjZPTXp/r3BXowuEmqu6CeLR50lo5/yUYJU8+lBVN7k8+rChmxBPgZUM3WiW37mmhd48+prjm92fAWBock9xK/ZJxua4GCG7qJckGRsi9hyJqpsgoZsqefSaCh6nz2azpiofow/bN1FR+4pIeWUt2+kh8kL/0+THuDVxBUdqGx/7wbPs7ygwKdsnGZvj8dq/bFbxD3ihhumVMWeMvoKPmiXt6yZf6KaaQl8CcczWNsluT2iwUlU3FRSw/rr5rnT1VDFUMmldIiIv9ACtOppGSXLNrh/w4uI7Clupv2Qs5H8LNmgcr1T14TWXjC1j1U01u0AoRU11tvWC3rzqPRlbbzH6Wk4aeyhI6EXkUhFZLSJrRWRBlvnNIrLInf+qiEzyzPuKO321iFxSQtsLZrOOAuDqht9z3rtfd7ot9nsM7a1wsK13vL83YwGSCVj7NHR5vkmbzQsL6tEXKvSpZKbt+ZYv9gLq2Jd/virsWgu7/whtH+Te14aXIFcvoqpwYGd+W2uhC4TDe2Drm7ntKGTf3R192yHbk0DQm1cYrzzbjXXXmt5zSxUO7Chum7nIuD7SBQya3XnKRvvmvud8Lrp971mEFWf/een9rYKQXj/sdvLQb6dmIhIHbgUuAlqBpSKyWFXf9Sx2LbBHVY8XkauA7wB/KSJTgauAk4GjgKdE5ATVypZGvJo6idsSl7NDh/F/Gn/Jjjs/zfC9K5ETP0rD1I/DpLPhp3Ng6Hj4/NOOeG95vXcD3sdoL4//C7x9f+a0Xe/DOw/B4DG90za+DPd8Ej72Axg2EfZvh9VLYMRk6NwPzUfA1hXQeQC2v+PZb5ZeGVc/BrFG2P2BY1N7K+x4F9Y9C2f8Hcy61tnmptfcFYWe7pLBuZD2rIdld8K46bDlDThpLkyc1ff4XvoRPPl1mPEZ53hOnw/DjnaP6VXY9Ars3QhLf+ZMGzgSLvzfjiAOOQr+8N3Mdln4cfjINxwb1jwBwyeBxOG9R2H9C3D987B3AwwYAV374e1fe47fU6/+5q+gbV3vvIMe8dEUvP+4c+M543oQgbcfdGzcsQqmXgEnfwKOu9DZ5iv/DwaOctrx7P8JDc3ODevADjjmw842N73mXNz3z3f29fevQ7wJNr3qnBtp3n4ARhzr/DaDx8CJl7s2Kax/HkafBPd+Gjr3wYe/AEdOBYllF7X0zWvAcKc9l94B409z7DzxY85xvf8EHNgGx81xzqc/+ytY/dvM3xqc9lj/ArQcATOvhYEjYM8GZ/rg0Y7N46ZnHksqCSsfhgc+5xzruV+CvZvgzXvhY7dA02D4/Tfh6kWw/jnYvw1GHAfTr4Kdq511JQbNQxy7Wo7o3fa7j8C7v/Hsyz3+NU86/2ONkOqGF/8DPnQ5PP/vsPM9+IufQvdBWP4LWPZz57c654sQi8PRZ8Lm5TDxDGcYnBvqI1+A934Lc2+FU+fB4huc6/v0zznLtS4DFFqGOvZvXeGcHyOP6/ubAGx7B7a93Tu+Zz38/DL40GVw7AXOtAM7nGtmwDDnWn/7fqcybMrFMPZUZ/8DhsH402HfVrjtXBgyDvZtgeufc3SoxIj20ymPiHwYuFFVL3HHvwKgqt/2LPO4u8zLItIAbANGAwu8y3qXy7W/mTNn6rJly0IdlJdJC3pP/KtOH8sXV87jSHZnLLOVUYxjFwCtMo4JurVn3iY5iom6hd0yjBG6l6T7EBTHEd709C0yhkEcYqju71n3MM0MoFes9zGY3bHhjEjt4Qjyd8mQJIagxFCSxDjIQPbGhnJ0anO/66VtA3io6eM0aJJjUhtp0i6mpD7gMC0Mofcp5DDNbI8d2Wdb41NbaSTRcxz7GURbbAQA41LbaaZ/D2R9bCI7ZRSzkm+QQohRfCdQSWJ00MLO2EiatZNx6gh7uu29yx1kIIM4RJwUrbGjSBHj6FQr62MT2SdDmJZ0/JMNsYk0aRfjdHvP+jtkFIdkQM+xbYxNQBEmploz7G6T4bRoJ4M41LNfb5un2RQbT5IYjdrNeN2W8/jaZDgjdU/GubVLRjBKd3Nr83/nC50/zVh+i4ylSxqZmNqcsd9uGmgk0WNTJ83sjI3imNSmnnl7ZCjtcgSjUrsZ7J4DXTSyLXYkR6c2sz42kUmpTWyRsQzVdrqlkXWxyZyWdJ5k0u3rJ/3bbo6NY0Rqd8Z5n94nQANJJqS2ALBXjkARmrSbXbGRHJPaBMBTjRfwke5nM7bfTYN7PaSIobwen95jk5ckMVpj41FgiB5kuO6hNTaeo1OtbIyN7/f6AThMC9tjo7POOyq1jSa66aCJljznf/qYvde6085jODrVSoI4rbGjGKIHGKm9hSLLh/w5p3/pN/3amA0RWa6qM7POK0Do5wGXqurn3fHPAmeo6g2eZd5xl2l1xz8AzgBuBF5R1Xvc6XcAj6nqg759XAdcB3D00UefvmHDhiDHmZUnVm7jcHeSdza3c8OFU9i75QPWPHE7gyaeyt6t6xjasZlB3bvZ2DiZwcm9DE20savxKOIkGJjcz29GXcecPQ8wqnsLuxvHsKNxAofjgzn1wEtsbZ7Ea0Mu4iN7FvHssL+gUbs4d+8j7G0czfjOD3h+6FxaUgc5IrGbTS0nMGfP/cQ1QVIa2Nh8AoOS+9jQciInHlrO7sYxJGlgfNcH7Gk4kiRxxneto61xLPvjw5nUsQqArU2TGJA6wO6GsTSnDvHGkPM5q30Jj434LINS+7msbSGH44PokhZWD/wzVg88vactphxawXntj5CUBp4e9ik+dPiNnv20pPpevB2xgTw64m/Y2ziaozrXcdGe+2hU5+Q+FBvMzsYJdMQGsqHlQ0w/8AJdsRa2N05kzcDpnLf3ETa1nMA7g87s2d7A5D7Oa3+EThnI80M/zln7HqNJD3NEYg9bmyYx9dBrbGw+gaHJNjpiA+mIDWRoYjc7G8dzwuE3erazvuVERnZv5w9D5zLj4Ascjg1iaGI3h2ODOKbzPfbFR7CvYTgTOj/oabPHRnyWlMQZkNzPZbvvYURie8+2hiV20dYwluM6nKepg7Ej2N8wnLFdznm4s3E8DdrNm4PPZlT3Vk45+Apd0sLWpkkc3fk+bw06i1MOvsK2pmMYnGrnzUFncWzHu0zoXNtjc2vz8Yzp2sSuxnEkJU6XtDA8sYOENDG6ezPtDaNobTqOztgATj34Eo3axaqBM3nxiMuZvf8phiV2Mjyxkz0Nozm6830A2htGcTA2hCO7N7OxeQqTO95lU/MUhiZ3s6VpEicdchymtsZx/NeIv+GYzvc4b+8jxEnSLc1saj6e5tRhhibbGJxsZ0vTZF4Y+jGu3PVTmrSDJA08PuIzbGk+liO7NnF++yM8NezTnL3vtwzv3sHaAdOYeug13hx8DssGX8hH9t7PpI5VdMQG8eiIz9HeMJITDq/g7PbfEqP3IX5T8wk8OfwvSRHjnH2PcuKh5QDsaJxAXBO8csQljO9cx7DkTsZ2bWR/fBjPD72Cy3bfQ1IaeHTEfA40DOeEQ29w3OG3eWvw2Zzd/igvDb2cM/Y9wfCE4wgoMV494mJWDTydy3bfzdiujexpOJLFI6/lyO5WzmpfwhMjrmJ/fARTDy3luMNv8dqQi5iz94Gs10P63NjbMJoNLScQ0xRn7H+SVQNPZ1zXevbFR5CQJjY3H8vZ7Y8SJ0lCmnh62DzaG0bx0d0LGZxsZ0fjRJpThxiadEJPqwbOZGT3NuIkGTRwIGdd+32IFZ8+rXmh91Jqj94wDONPgXxCX8htYzMw0TM+wZ2WdRk3dDMUaCtwXcMwDKOMFCL0S4EpIjJZRJpwkquLfcssBua7w/OAZ9R5VFgMXOVW5UwGpgCvYRiGYVSMfqtuVDUhIjcAjwNx4E5VXSkiNwHLVHUxcAdwt4isBXbj3Axwl7sfeBdIAF+odMWNYRjGnzr9xugrjcXoDcMwiidsjN4wDMOoY0zoDcMwIo4JvWEYRsQxoTcMw4g4NZeMFZGdQJhXY0eB259BbWF2FYfZVRy1ahfUrm1Rs+sYVc3ad0PNCX1YRGRZrsxzNTG7isPsKo5atQtq17Y/JbssdGMYhhFxTOgNwzAiThSF/vZqG5ADs6s4zK7iqFW7oHZt+5OxK3IxesMwDCOTKHr0hmEYhgcTesMwjIgTGaHv7wPmFbZlvYi8LSIrRGSZO22EiDwpImvc/8MrZMudIrLD/ThMelpWW8Thh24bviUip1XYrhtFZLPbbitE5KOeeRX5yLyITBSR34vIuyKyUkT+lzu9qm2Wx66qtpmItIjIayLypmvXN9zpk0XkVXf/i9wuznG7LF/kTn9VRCZV2K5fiMgfPe01w51esXPf3V9cRN4QkUfd8fK2l6rW/R9O98kfAMcCTcCbwNQq2rMeGOWb9l1ggTu8APhOhWw5DzgNeKc/W4CPAo/hfFH8TODVCtt1I/CPWZad6v6mzcBk97eOl8muccBp7vAQ4H13/1Vtszx2VbXN3OMe7A43Aq+67XA/cJU7/T+Bv3OH/wfwn+7wVcCiMrVXLrt+AczLsnzFzn13f18E7gUedcfL2l5R8ehnA2tVdZ2qdgH3AXOrbJOfucBCd3ghcGUldqqqz4Hva+i5bZkL3KUOrwDDRGRcBe3KxVzgPlXtVNU/AmtxfvNy2LVVVV93h/cDq4DxVLnN8tiVi4q0mXvc6S/dN7p/ClwIpD8Z6m+vdDs+CMwREamgXbmo2LkvIhOAy4GfueNCmdsrKkI/HtjkGW8l/0VQbhR4QkSWi/Phc4AxqrrVHd4GjKmOaXltqYV2vMF9dL7TE96qil3uY/Kf4XiDNdNmPrugym3mhiFWADuAJ3GeHvaqaiLLvnvscue3AyMrYZeqptvrm2573SIizX67sthcan4A/BOQcsdHUub2iorQ1xrnqOppwGXAF0TkPO9MdZ7DaqKutZZsAX4CHAfMALYC/14tQ0RkMPBr4B9UdZ93XjXbLItdVW8zVU2q6gycb0LPBk6stA3Z8NslIqcAX8GxbxYwAvjnStokIh8Ddqjq8kruNypCX1MfIVfVze7/HcDDOCf/9vSjoPt/R7Xsy2NLVdtRVbe7F2cK+Cm9oYaK2iUijThi+ktVfcidXPU2y2ZXrbSZa8te4PfAh3FCH+lPlXr33WOXO38o0FYhuy51Q2Cqqp3Az6l8e50NXCEi63FCzBcC/0GZ2ysqQl/IB8wrgogMEpEh6WHgYuAdMj+gPh94pBr2ueSyZTHw124FwplAuydcUXZ8MdFP4LRb2q6KfGTejX/eAaxS1e97ZlW1zXLZVe02E5HRIjLMHR4AXISTP/g9MM9dzN9e6XacBzzjPiFVwq73PDdrwYmDe9ur7L+jqn5FVSeo6iQcnXpGVa+h3O1VykxyNf9wsubv48QHv1pFO47FqXZ4E1iZtgUnrvY0sAZ4ChhRIXt+hfNI340T+7s2ly04FQe3um34NjCzwnbd7e73LfcEH+dZ/quuXauBy8po1zk4YZm3gBXu30er3WZ57KpqmwHTgDfc/b8DfM1zHbyGkwR+AGh2p7e442vd+cdW2K5n3PZ6B7iH3sqcip37HhsvoLfqpqztZV0gGIZhRJyohG4MwzCMHJjQG4ZhRBwTesMwjIhjQm8YhhFxTOgNwzAijgm9YRhGxDGhNwzDiDj/H8SS18KuTXxLAAAAAElFTkSuQmCC\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.networks.learnable_q import TableQ\n",
    "\n",
    "\n",
    "data_loader_samples = SampleDataLoader(replay_buffer, batch_size_samples, shuffle_key)\n",
    "q = TableQ(\n",
    "    n_states=n_states,\n",
    "    n_actions=n_actions,\n",
    "    gamma=gamma,\n",
    "    network_key=dummy_q_network_key,\n",
    "    zero_initializer=True,\n",
    "    learning_rate=learning_rate\n",
    ")\n",
    "\n",
    "q_functions = np.zeros((max_bellman_iterations + 1, n_states, n_actions))\n",
    "bellman_iteration_functions = np.zeros((max_bellman_iterations + 1, n_states, n_actions))\n",
    "v_functions = np.zeros((max_bellman_iterations + 1, n_states))\n",
    "\n",
    "params_target = q.params\n",
    "q_i = env.discretize(q, q.to_weights(q.params), states, actions)\n",
    "policy_q = q_i.argmax(axis=1)\n",
    "\n",
    "q_functions[0] = q_i\n",
    "v_functions[0] = env.value_function(policy_q)\n",
    "print(policy_q)\n",
    "cumulative_l2_loss = np.zeros((max_bellman_iterations, fitting_steps)) * np.nan\n",
    "\n",
    "for bellman_iteration in tqdm(range(1, max_bellman_iterations + 1)):\n",
    "    q.reset_optimizer()\n",
    "    \n",
    "    for step in range(fitting_steps):\n",
    "        data_loader_samples.shuffle()\n",
    "        for batch_samples in data_loader_samples:\n",
    "            q.params, q.optimizer_state, l2_loss = q.learn_on_batch(q.params, params_target, q.optimizer_state, batch_samples)\n",
    "        cumulative_l2_loss[bellman_iteration - 1, step] = l2_loss\n",
    "    q_i = env.discretize(q, q.to_weights(q.params), states, actions)\n",
    "    policy_q = q_i.argmax(axis=1)\n",
    "\n",
    "    q_functions[bellman_iteration] = q_i\n",
    "    bellman_iteration_functions[bellman_iteration] = env.apply_bellman_operator(q_i)\n",
    "    v_functions[bellman_iteration] = env.value_function(policy_q)\n",
    "    print(policy_q)\n",
    "\n",
    "    params_target = q.params\n",
    "\n",
    "for bellman_iteration in range(max_bellman_iterations):\n",
    "    plt.plot(cumulative_l2_loss[bellman_iteration], label=f\"Iteration {bellman_iteration}\")\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Save data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T11:20:55.496951Z",
     "iopub.status.busy": "2022-09-19T11:20:55.496804Z",
     "iopub.status.idle": "2022-09-19T11:20:55.507754Z",
     "shell.execute_reply": "2022-09-19T11:20:55.507376Z"
    }
   },
   "outputs": [],
   "source": [
    "np.save(f\"figures/data/FQI/{max_bellman_iterations}_Q_{seed}.npy\", q_functions)\n",
    "np.save(f\"figures/data/FQI/{max_bellman_iterations}_BI_{seed}.npy\", bellman_iteration_functions)\n",
    "np.save(f\"figures/data/FQI/{max_bellman_iterations}_V_{seed}.npy\", v_functions)"
   ]
  }
 ],
 "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": {
     "128b8067d1b2400ea17b08abe70f92e5": {
      "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": ""
      }
     },
     "2605a2ce1bb241898ee10567292b05f5": {
      "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_da5e05a229264afebc917251997a49c8",
       "max": 2.0,
       "min": 0.0,
       "orientation": "horizontal",
       "style": "IPY_MODEL_128b8067d1b2400ea17b08abe70f92e5",
       "value": 2.0
      }
     },
     "35f09f1cfb8544bcb4f422a80a6d4ad2": {
      "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_8e194c2ed0ad40c0914135dae1e8bb13",
       "placeholder": "​",
       "style": "IPY_MODEL_d72f6b4aca1a4446a61b6ddf58f60cdf",
       "value": " 2/2 [00:09&lt;00:00,  4.56s/it]"
      }
     },
     "622fb3e1c5954bc4a13acca0acd2fc07": {
      "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_66cada98ab704eaa944479fa219c9f5a",
        "IPY_MODEL_2605a2ce1bb241898ee10567292b05f5",
        "IPY_MODEL_35f09f1cfb8544bcb4f422a80a6d4ad2"
       ],
       "layout": "IPY_MODEL_6e818cfabf764d138024fe711414dab2"
      }
     },
     "66cada98ab704eaa944479fa219c9f5a": {
      "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_fc01e35c771941afad27b0abf1a558a4",
       "placeholder": "​",
       "style": "IPY_MODEL_bed6a41d60ef4d95ba14b056d937731d",
       "value": "100%"
      }
     },
     "6e818cfabf764d138024fe711414dab2": {
      "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
      }
     },
     "8e194c2ed0ad40c0914135dae1e8bb13": {
      "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
      }
     },
     "bed6a41d60ef4d95ba14b056d937731d": {
      "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": ""
      }
     },
     "d72f6b4aca1a4446a61b6ddf58f60cdf": {
      "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": ""
      }
     },
     "da5e05a229264afebc917251997a49c8": {
      "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
      }
     },
     "fc01e35c771941afad27b0abf1a558a4": {
      "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
}
