{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# PBO linear on the chain walk environment\n",
    "\n",
    "## Define parameters"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T13:02:12.844463Z",
     "iopub.status.busy": "2022-09-19T13:02:12.844255Z",
     "iopub.status.idle": "2022-09-19T13:02:13.575643Z",
     "shell.execute_reply": "2022-09-19T13:02:13.575118Z"
    }
   },
   "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",
    "# Weights collection\n",
    "n_weights = parameters[\"n_weights\"]\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",
    "add_infinity = True\n",
    "\n",
    "# Visualisation of errors and performances\n",
    "max_bellman_iterations_validation = max_bellman_iterations + 20\n",
    "\n",
    "# Search for an unused seed\n",
    "max_used_seed = 0\n",
    "if not os.path.exists(\"figures/data/PBO_linear/\"):\n",
    "    os.makedirs(\"figures/data/PBO_linear/\")\n",
    "for file in os.listdir(\"figures/data/PBO_linear/\"):\n",
    "    if int(file.split(\"_\")[0]) == max_bellman_iterations and int(file.split(\"_\")[2]) == n_weights and int(file.split(\"_\")[3][:-4]) > max_used_seed:\n",
    "        max_used_seed = int(file.split(\"_\")[3][:-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, 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-19T13:02:13.577869Z",
     "iopub.status.busy": "2022-09-19T13:02:13.577662Z",
     "iopub.status.idle": "2022-09-19T13:02:14.010279Z",
     "shell.execute_reply": "2022-09-19T13:02:14.009764Z"
    }
   },
   "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-19T13:02:14.014901Z",
     "iopub.status.busy": "2022-09-19T13:02:14.014717Z",
     "iopub.status.idle": "2022-09-19T13:02:15.344494Z",
     "shell.execute_reply": "2022-09-19T13:02:15.343998Z"
    }
   },
   "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-19T13:02:15.347749Z",
     "iopub.status.busy": "2022-09-19T13:02:15.347559Z",
     "iopub.status.idle": "2022-09-19T13:02:16.347537Z",
     "shell.execute_reply": "2022-09-19T13:02:16.346767Z"
    }
   },
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAZ8AAAEYCAYAAACDV/v0AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAAArxklEQVR4nO3deZxcVZ3+8c/Dvq8BhADCKCqKioKgDiigIiAK44gCjiwDIriBDA44Krih8BuVcUcEBBdQBkQyEjZBwFFAghMhiEJkkYQ1hB0Ekjy/P85pU6l0d1V1uquru593XveVqnvvufdUdXd965x77vfINhEREd201GhXICIiJp4En4iI6LoEn4iI6LoEn4iI6LoEn4iI6LoEn4iI6LoEn4iICUTS6ZIekDSjYd1ekm6WtEDS1oOU3UXSnyXNlHRMw/pNJV1X1/9U0nKt6pHgExExsZwB7NK0bgbwTuDqgQpJWhr4FrAr8FJgH0kvrZtPBE6y/ULgYeCgVpVI8ImImEBsXw3MbVp3i+0/tyi6DTDT9u22nwV+AuwhScBOwLl1vzOBPVvVY5lOKx4REd3x1h1X9kNz53dU5oYbn7kZ+FvDqlNsnzIM1ZkM3N3wfBawLbA28IjteQ3rJ7c6WIJPRESPmjN3PtddsmFHZZZd/y9/sz3gdZtekeATEdGzzHwvGO1K9JkNbNTwfMO67iFgDUnL1NZP3/pB5ZpPRESPMrAAd7SMoOuBzerItuWAvYEpLtmpfwW8q+63P3BBq4Ml+ERE9LAFHf5rRdLZwDXAiyXNknSQpH+SNAt4HXChpEvqvhtImgpQWzUfBi4BbgHOsX1zPezRwJGSZlKuAZ3Wsh6ZUiEioje96pXL+aqLntdRmdUn331DrvlERMQSGeGutFGT4BMR0aMMzE/wiYiIbkvLJyIiusrA/HF6XT7BJyKih/XMXT7DLMEnIqJHGeeaT0REdJlh/viMPQk+ERG9qmQ4GJ8SfCIiepaYj0a7EiMiwSciokcZWJBut4iI6La0fCIioqtKhoMEn4iI6LIFTvCJiIguSssnIiK6zoj543TatQSfiIgelm63iIjoqnS7RUTEKBDznW63iIjoopJeJ8EnIiK6LN1uERHRVXa63SIiYhQsSMsnIiK6qYx2G58tn/H5qiIixoXS7dbJ0vKI0umSHpA0o2HdWpIuk3Rb/X/NfsrtKGl6w/I3SXvWbWdIuqNh25at6pHgExHRo/pGu3WytOEMYJemdccAl9veDLi8Pl+0LvavbG9pe0tgJ+Ap4NKGXT7et9329FaVSPCJiOhh862OllZsXw3MbVq9B3BmfXwmsGeLw7wLuMj2Ux2+nL9L8ImI6FF9ud06WYBJkqY1LIe0car1bN9bH98HrNdi/72Bs5vWHS/pRkknSVq+1Qkz4CAiooct6Hyo9RzbWw/1fLYtacD5UyWtD7wcuKRh9ScoQWs54BTgaOBzg50nLZ+IiB7VN9qtw5bPUNxfg0pfcHlgkH3fDZxv+7m/19O+18UzwPeBbVqdMMEnIqJHmc6u97RzzWcAU4D96+P9gQsG2XcfmrrcGgKXKNeLZixebFEJPhERPWy4R7tJOhu4BnixpFmSDgJOAN4i6TbgzfU5kraWdGpD2U2AjYCrmg77Y0k3ATcBk4AvtKpHrvlERPQom2FPr2N7nwE2vamffacBBzc8vxOY3M9+O3VajwSfiIiepaTXiYiI7jLD3/LpFQk+ERE9bLzmdkvwiYjoUUYsGPoItp6W4BMR0cPS8omIiK4yQ8pwMCYk+ERE9CxlGu2IiOiutHwiImJUpOUTERFdZSstn4iI6L7cZBoREV1VptFOt1tERHSV0vKJiIjuKqPd0vKJiIguS4aDiIjoquR2i4iIUdHO7KRjUYJPRESPKjOZpuUTERFdlm63iIjoqnLNJ91uERHRZcntFhERXZX7fCIiYhSM32638fmqIiLGiQWoo6UVSadLekDSjIZ1a0m6TNJt9f81Byg7X9L0ukxpWL+ppOskzZT0U0nLtapHgk9ERI/qG2rdydKGM4BdmtYdA1xuezPg8vq8P0/b3rIu72hYfyJwku0XAg8DB7WqRIJPREQPW+ClOlpasX01MLdp9R7AmfXxmcCe7dZPkoCdgHM7KZ/gExER69m+tz6+D1hvgP1WkDRN0rWS9qzr1gYesT2vPp8FTG51wgw4iIjoUUPM7TZJ0rSG56fYPqXtc9qW5AE2P9/2bEn/AFwh6Sbg0U4rCAk+ERE9bQiTyc2xvXWHZe6XtL7teyWtDzzQ3062Z9f/b5d0JfAq4DxgDUnL1NbPhsDsVidMt1tERI/qu8+nk2WIpgD718f7Axc07yBpTUnL18eTgH8E/mjbwK+Adw1WvlmCT0REDxvuAQeSzgauAV4saZakg4ATgLdIug14c32OpK0lnVqLbg5Mk/QHSrA5wfYf67ajgSMlzaRcAzqtVT3S7RYR0auWrDXT/yHtfQbY9KZ+9p0GHFwf/xZ4+QDHvB3YppN6JPhERPQoM6RrPmNCgk9ERA9LbreIiOiqJBaNiIhRkeATERFdNcSbTMeEBJ+IiB6WAQcREdFdTrdbRER0WQYcRETEqEjwiYiIrsqAg4iIGBVO8ImIiG7LaLeIiOgqZ7RbRESMhnS7RUREl2XAQUREjIK0fCIioqtyk2lERHSfy6CD8SjBJyKih2WodUREdJUZv9d8lhrtCsTEJukzkn402vUYaZL+Q9Kpg2x/r6RLu1mnGAvKaLdOlrEiwWeCkrSdpN9KelTSXEm/kfSa0a7XeCBpB0mzGtfZ/qLtg+v2TSRZ0jIN239se+du1zV6n93ZMlak220CkrQa8AvgMOAcYDlge+CZ0azXSJG0jO153TpXN84TE0e63WI8eRGA7bNtz7f9tO1Lbd8IIOkFkq6Q9JCkOZJ+LGmNvsKS7pT0cUk3SnpS0mmS1pN0kaTHJf1S0pp1375v+YdIukfSvZKOGqhikl5bW2SPSPqDpB0ath0g6fZ6jjskvXeAY3xG0rmSfiTpMeAASavXet4rabakL0hauuG4v5H0zdoS/JOkNzUc70BJt9Tz3i7pAw3bdpA0S9LRku4DzgYuAjaQ9ERdNmjqXry6/v9I3f66Wof/bTju6yVdX+tzvaTXN2y7UtLna50fl3SppEmD/sRjTCqtGXW0tCLpdEkPSJrRsG4tSZdJuq3+v2Y/5baUdI2km+vf/nsatp1R/yan12XLVvVI8JmYbgXmSzpT0q79/KIJ+BKwAbA5sBHwmaZ9/hl4CyWQvZ3ygfsfwDqU36uPNu2/I7AZsDNwtKQ3N1dK0mTgQuALwFrAUcB5ktaRtDLwdWBX26sCrwemD/Ia9wDOBdYAfgycAcwDXgi8qtbj4Ib9twX+AkwCjgN+Jmmtuu0BYHdgNeBA4CRJr24o+7xa3+cD+wG7AvfYXqUu9zTV7Q31/zXq9mua3oe16vvwdWBt4KvAhZLWbtht31qXdSkt1wEDeoxtI3DN5wxgl6Z1xwCX294MuLw+b/YUsJ/tl9Xy/9X4pRT4uO0t6zK9VSUSfCYg248B21EG03wPeFDSFEnr1e0zbV9m+xnbD1I+/N7YdJhv2L7f9mzg18B1tv/P9t+A8ykf8I0+a/tJ2zcB3wf26adq/wJMtT3V9gLblwHTgN3q9gXAFpJWtH2v7ZsHeZnX2P657QWUoLEbcEStwwPAScDeDfs/APyX7eds/xT4M/C2+n5caPsvLq4CLqV0U/ZZABxX36+nB6lTu94G3Gb7h7bn2T4b+BMlyPf5vu1b6/nOAbYchvNGDxruaz62rwbmNq3eAzizPj4T2LOfcrfavq0+vofyN7POUF9Xgs8EZfsW2wfY3hDYgtLK+S+A2oX2k9o99RjwI0qLoNH9DY+f7uf5Kk37393w+K56vmbPB/aqXW6PSHqEEiTXt/0k8B7gUOBeSRdKeskgL7HxfM8Hlq3l+o77XUqroc9se5E/3b/XsbYOr1UZmPEIJZA1vh8P1qA7XDao5290FzC54fl9DY+fYvH3O8aJ4e52G8B6tu+tj+8D1htsZ0nbUFrcf2lYfXztjjtJ0vKtTpjgE9j+E6UpvkVd9UVKq+jltlejtEiW9KrnRg2PNwaau6KgBIwf2l6jYVnZ9gm1npfYfguwPqUl8L1BztcYSO6mDKaY1HDc1Wr3QZ/Jkhpf48bAPfWP6Dzgy5Q/0DWAqSz6fjR/32z1/bPV9nsoAbPRxsDsFuVinDGdBZ4afCZJmtawHNLROcuXsAF/RyWtD/wQOLD2LAB8AngJ8BpKF/TRrc6T4DMBSXqJpH+TtGF9vhGlG+zausuqwBPAo/U6zMeH4bSflrSSpJdRrlX8tJ99fgS8XdJbJS0taYV6QX/D2hrbo177eabWb0E/x1hM/UZ3KfAVSatJWkplUEVjV+K6wEclLStpL8q1rqmUb3fLAw8C8yTtSrleNJj7gbUlrT7A9gdr3f9hgO1TgRdJ2lfSMvXC7kspIxRbUhkQckA7+0bvc4cLMMf21g3LKW2c5v4aVPqCywP97aQyUvZC4JO2+z4vqN3gtv0MpVt9m1YnTPCZmB6nXGC/TtKTlKAzA/i3uv2zwKuBRym/aD8bhnNeBcykXMz8su3Fbqi0fTel7/k/KB/Qd1MC31J1OZLSKphLuQZ1WAfn348SSP4IPEwZjLB+w/brKAMi5gDHA++y/ZDtxymDJ86p5fYFpgx2otqSPBu4vXbzbdC0/al6jt/U7a9t2v4QZYDDvwEPAf8O7G57TqsXKWk5yiCFa1vtG2PACIx2G8AUYP/6eH/gguYd6u/W+cAPbJ/btK0vcIlyvWhGc/nFjuexdFdSjDmSNgHuAJbt1r02naqthINtbzfadVlSkrYDPmS7vwEdMcas8ILJ3vjEQzsqc9tex95ge+uBtks6G9iBct3yfsrozp9TvmBtTLm++G7bcyVtDRxq+2BJ/0Jp1TQO9DnA9nRJV1AGH4gyCvVQ208MVs/cEBcxjtj+X+B/W+4YY8Zw32Q6yBeTNzWvsD2NekuC7R9Rusb7O+ZOndYjwSciooeN186pBJ8YUbbvZMlHyo0o22dQRvtF9JTxnNU6wSciolcZGKfBJ6PdYsKRdLKkTw+y3ZJeOEznOkPSF4bpWFdKOrj1njGejNes1gk+41BNEni+StLPuyTt27DtlTUx4BxJRzasX1bSdfWenzGnkw9m24fa/vwI1GGR5KARw2IIN/qMBel2G5++BTxLSZGxJSUp5R9qLrQvUZJQ3gjcKOks2/dR7qE5r95rMywkLW17/nAdL2LiWaJ7d3paWj7jTM0A8M/Ap20/UYfeTgHeV3fZFLiiJgS9DdhY0vNrmZPaOP5/S7pPJdX/1TVjQd+2MyR9R9LUevPqjirTCZwn6UGVlOvN2a4bj726pB/Ufe+S9ClJS9Vti8x4qoYJ2SQdT0n0+U2VKQq+qeIkldTxj0m6SdIWDfX8QsOxPq4y1cI9kv61qU7LS/qypL9Kur922a3YT903B04GXlfr8EjD5jVVctE9XluXL2go9xKVFPZzJf1Z0rtb/AheIOl39TVdoIWZt5H0jtqqfaS2BDev64+u512mPj+s7rdCi3NFLxinLZ8En/HnRcA827c2rPsD0BckZgA7q6TW2YSSGPBrlHToz7Vx/IsomQDWBX5Pma6g0b6Uu/dXBX4L/E89/2TKfQRHSHrrAMf+BrA6Je3MGylZCQ5sVSHbn6Rk1v5wnaLgw5QUOG+gvB+rA++mZAtYhKRdKC3Bt9TX1TzVwwn1GFtSpmOYDBzbTx1uoSQ9vabWYY2GzXtTskasScnycHw998rAZcBZlPdzb+Dbkl46yMvdD/hXSnaGeZRpF5D0IkpWhSMoN/tNBf5H5a70/6SkJPqUpM0oufv+ZZiTocZI6F6Gg65L8Bl/VgEea1r3KCUYQPmgPYzSGvoY8I+UdDt31G/SV6nkNuuX7dNtP15zOH0GeKUWzWF2ge3f1ISDLwfWsf0528/avp2SDHTv5uOqTOy2N/CJevw7ga+wsMXWqefqa34JJZPHLQ1Zexu9mzI9wYyaOfszDXUScAjwMdtza6qdL/ZX/xbOt/27muHhxyyc/mB34E7b369TJ/wfJYnpgO8/JfFqX10/Dby7vnfvAS50mQrjOUoi1BWB19efxX6UNEFTgP9XzxVjwTht+eSaz/jzBGX+mkarUQIMtu+izo8jaSXgGkor4RuUZJ8XAjMkXW57kTk/6ofc8ZQPx3VYmNhzEiXAweJTGWzQ1AW1NKWV0mwSZdqDxqkEmqcRaJvtKyR9k3L96/mSfgYc5TKXUaMNgBuaztlnHWAl4AYtTHit+ho6MdD0B88Htm16f5ahZAweSPPUFMtS3rtFpmGwvUDS3dT3z/adkn5F+dl/q8P6x6gaO62ZTqTlM/7cCixTu1f6vJJF8zH1ORb4nu37Ka2UabYfBWZRupia7UtJ/PlmSlfWJnX9QNML3A3c0TRFwqq2d2NxcyitlcapBBqnEXiSEgj6PK+p/GLf+Wx/3fZWlIzQL6L/7Nz3svh0D411ehp4WUP9V7c90Nw5nX7vvBu4qun9WcX2YAlTm+v6XK3nItMw1FbbRtT3T9LbgNdRErv+Z4f1jNE0Tls+HQUflVT0zd+qo4fU7pifAZ+TtLKkf6QEjEW+TdfrCjsA36mr7gB2UpnNdDPgr/0cflXKtYOHKIHgiy2q8zvg8XrBe0WVaRK2kPSafuo9n5LY8HhJq9ZBEEeyMJfUdOANkjau3XyfaDrE/TRMUSDpNZK2lbQsJXD9jf6nYDgHOEDSS2tL8LiGOi2gdBOeJGndetzJg1yzuh/YsF5naccvKFMnvE9lqPuytd6bD1LmXxrq+jng3Ib37m2S3lRf879Rfla/lTQJOJWSo2t/yrQV/X0BiF40UYOPpLNU5kBZmXKx+o+ShmN+lxg5H6T09z9AuQh9mBefcvpbwOENQ6E/QbkmcDPwxTr8utkPKF07sylTEwyatr8ee3fKNY47KN/QT6W0mvrzEUqguJ2SHPMs4PR6rMso3YI3UrrJmue2+RrwLkkPS/o6pavxe5RpEO6iBMzFvvHbvogyg+sVlMEAVzTtcnRdf63KrK6/BF48QP2voLx/90lqOf1BvYa0M+Ua0j2U7rkTKfMHDeSHlFRA9wErUH5m2P4zZdK/b1De57cDb7f9LHAK5Vrc1Dpdw0HAqZLWBqij87ZvPlH0gL4MB50sY0TLKRUkTbe9paT3UuZ4OQa4wfYrulHBiIiJavlNNvTzjh3w7oR+/fWgowedUqFXtNPttmxtxu8JTKkjacZQ4y4iYgybqN1uwHeBO4GVgatrX3zziKHFSDq93uDXcka7iIgYwDjtdmsZfOqIocm2d6tzdN8F7NjGsc8AdlnSCkZETGRyZ8tY0fI+H0nLU1KvbNK0/+cGK2f7apUplCMiYijGWFdaJ9q5yfQCyg2EN1CGbkZERFeMra60TrQTfDa0PWLdZ5IOoaQwQcstt9Wy6607UqeKiBhxz949a47tdYbtgBO45fNbSS+3fdNIVMD2KZT7EFh+4408+agjRuI0ERFdccfhR93Veq8OTODgsx3lDvA7KN1uApz7fCIiumACB59dh3JgSWdT0rdMkjQLOM72aUM5VkTEhNSX4WAcameo9V3AGtR0HcAadV2rcvvYXt/2srY3TOCJiOjccA+17u8eTElr1UkNb6v/rzlA2f3rPrdJ2r9h/VYqEzbOlPR1NaSBH0g7ud0Op8xBsm5dfiTpI61fYkRELLHhz3BwBovfg3kMcLntzSiZz49pLlRnzT0O2BbYBjiuIUh9B3g/JSnxZv0cfzHtZDg4CNjW9rG2jwVeW08SERFjjO2rgblNq/cAzqyPz6SkU2v2VuCyOrHiw5RZeHeRtD6wmu1rXZKF/mCA8oto55qPgPkNz+czXmc3iojoMV3KWrBew0y/9wHr9bPPZBadzHBWXTe5Pm5eP6h2gs/3gesknV+f7wnk+k1ERDd0PuBgkqRpDc9Pqbe0tHc629LIh7yWwcf2VyVdSRlyDXBg5n+PiOiCoaXXmTOEKRXul7S+7XtrN9oD/ewzmzKCuc+GwJV1/YZN62fTwoDXfPpmLK0Xme6kzCj5I+Cuui4iIkZad6ZUmEKZ5Zb6/wX97HMJsLOkNetAg52BS2p33WOSXltHue03QPlFDNbyOYsyC+UNLPqSVJ//Q3+FIiJi+Ax3B1h/92ACJwDnSDqIMvPvu+u+WwOH2j7Y9lxJnweur4f6nO2+gQsfpIyiWxG4qC6DGjD42N69/r9px6+ukrQLZXrjpYFTbZ8w1GNFRExIwxx8bO8zwKY39bPvNODghuenU6e272e/LTqpRzv3+Vzezrp+9lka+BYlQ8JLgX0kvbSTykVETHjjdCbTAVs+klYAVqI0zdZk4fDq1WhjGB3lJqSZtm+vx/sJZSz5H5eoxhERE8RYmyCuE4Nd8/kAcASwAeW6T1/weQz4ZhvH7m9M+LadVzEiYgIbp7ndBrvm8zXga5I+YvsbI1WBxvl8gGfuOPyoGYPt38IkYE7Kj9nyvVCHlM/vwJKWf/ESlF3cBGz59FkgaQ3bjwDULrh9bH+7RbnZwEYNz/sd+904n4+kaUMYn/53KT+2y/dCHVI+vwPDUX6oZfs93jgNPu3kdnt/X+ABqDl92sntdj2wmaRNJS0H7E0ZSx4REe2aaAMOGiwtSTVhXN8otuVaFbI9T9KHKTcmLQ2cbvvmJaptRMREMkEHHPS5GPippO/W5x+gjRuIAGxPBaZ2UJ+28w+l/Lgs3wt1SPnRLd8LdRjt8osap8FHtUEz8A7SUpQBAX03IN0IPM/2h0a4bhERE9oKkzfyxocd2VGZ2z595A1Let2tG9qZyXQBcB0lv9s2wE7ALSNbrYiIgOGfybRXDHaT6YuAfeoyB/gpgO0du1O1iIgYrwa75vMn4NfA7rZnAkj6WFdqFRERxRhqzXRisODzTsrw6F9Juhj4CcM4g6mkl1DS7fSl6pkNTLHdtS69WofJwHW2n2hYv4vti9sovw1l7qXra966XYA/1YEWQ6nPD2zvN8Sy21G6RWfYvrSN/bcFbrH9mKQVKXO2v5qS/uiLth9tUf6jwPm27x5sv0HK9w2/v8f2LyXtC7ye0qV7iu3n2jjGP1B+TzeizLB7K3CW7ceGUqeInjPGutI6MeA1H9s/t7038BLgV5RUO+tK+o6knZfkpJKOZmEw+11dBJwt6ZglOXY9/oFt7PNRypwTHwFmSNqjYfMX2yh/HPB14DuSvkRJObQycIykT7ZRfkrT8j/AO/uet1H+dw2P31/PvypwXJvv4enAU/Xx14DVgRPruu+3Uf7zlBlufy3pg5LWaaNMo+8DbwMOl/RDYC/KtcXXAKe2Klx/ficDK9Qyy1OC0LWSduiwLlFJWrcH6rD2aNehp4zT+3xajnZbZOeS3WAv4D22F0u/3cFxbgVe1vzttn4bvtn2ZkM9dj3OX21v3GKfm4DX2X5C0ibAucAPbX9N0v/ZflUb5bekfOjdB2zY0Iq4zvYrWpT/PaWVcSrlV0bA2ZTWALavalH+73WUdD2wm+0HJa0MXGv75S3K32J787662H51w7bptrdsdX5gK+DNwHuAd1ByAJ4N/Mz24y3K32j7FZKWobR6N7A9v05G9Yc23r+bgC1rmZWAqbZ3kLQxcEGrn189xurAJyhTw69L+Tk8QPlSckLjzdWdknSR7V1b7LNaPf+GwEW2z2rY9m3bH2xR/nmUuVgWAMdSvkj9M6X1eHid5Guw8s2TQoryM3wV5bNh7uKlFjvG33sJ6vv5VcqXgRnAx2zf36L8CcCXbc+pc8ecU1/PssB+bfwd/B74GXC27b+0qm8/5bcG/pPyO/gJypeybSit6ENazdosaRXg3ynv+4bAs8BfgJNtn9FpfZqtsMFG3uT9nY12+/Pnxslot0a2H7Z9ypIEnmoBJWFps/XrtpYk3TjAchOwXhuHWKqvq832nZTJlXaV9FXa616cZ3u+7aeAv/R19dh+us3XsDXlD/2TwKO2rwSetn1Vqz+4vvqrzCi4NuWD4sF6/ieBeW2Un9HQQvxD/SPsG2jSssurnMoLbF9q+yDKz/PblK7H29us/3KU1tpKlJYXlGC+bBvlYWG38fLAKrVSf+2g/DnAw8AOtteyvTawY113TqvCkl49wLIV5YtJK9+n/K6dB+wt6TxJy9dtr22j/BmULzB3U3onngZ2o1yrPbmN8nMov4N9yzRKN/Tv6+N2NPYSfAW4F3g7JcPJd/stsai32e7Lo/aflC+2LwTeUo/XyprAGpTLA7+T9DFJ/X22DOTbwP8DLgR+C3zX9uqUbuhWKcQAfkz5fX8r8FlKb8j7gB0ltexBaUVMwNFuI+wI4HJJt7Ew8/XGwAuBD7d5jPUoP/CHm9aL8kvUyv2StrQ9HaC2gHanfPMZtNVQPStppRp8tvr7ycu3v5bBpw5hP0nSf9f/76ezn8fqLMw2bi2cf30V2gueB1MSx36K8iF0jaS7KT+PgwctWSxyjtqKnQJMqS2RVk6jDGpZmhKA/1vS7ZQP3Z+0Uf5U4HpJ1wHbU7oMqd1/Lb+xV5vYPrHpddwHnCjpX9sofz1wFf2/32u0Uf4Ftv+5Pv557a69QtI72igLsF5f0l9JH2x4Ld9QmZGylY9TPuQ/bvumepw7PPQJJLduaDGfJGn/wXaulpG0jO15wIq2rwewfWtDIB7Mw7aPAo6StD1ldO7vJd1CaQ21uuFzWdsXAUg60fa59fyXS/pyG+ffpKGF81VJ19v+fP1i90fgP9o4xuDGUEDpxKgEH9sX12/Y27DogIPrbc9v8zC/AFbpCx6NJF3ZRvn9aGoh1D+A/bQwm8Ng3mD7mVquMdgsy8K50FuyPQvYS9LbKNNVtFtukwE2LQD+qY3yjwIH1K6fTSm/C7NadZM0eM8gx35qoG0N+5wkqW/4/j2SfkDpwvue7d8NXrpkXZf0S2Bz4Cu2/1TXPwi8oc3XcJekfwfO7HvdktYDDmDR6UAGcgvwAdu3NW+ogbyV5SUt1ff7Y/t4SbOBq6ktuRYaey5+0LRt6VaFbX+l/gxOqvU9js4/6taVdCQlAK8mLUzFRXs9K98Gptbut4slfY3SjbYTML2Titj+NfBrSR+hBNX30DrbwN9UrmGvTvkSt6ftn0t6I2UQSytPStrO9v/WLw1za10W1C7kJTPGWjOdGK2WT98H9rVLUH7Ab3a2922j/KxBtv2mjfLPDLB+DkNIx277QkrTf4nUD/47Otj/MeAPQzjPrZ2W6ecY9zQ8foRy3a2T8jcDS5Iv8D2U7pWrtPBC+/2UFtxebZT/DAN/wH6kjfL/Q/mQ/WXfCttnSLoPaGcakwskrWL7Cduf6lsp6YXAn9so3/jl5x3AZZQu0E58j9J1CnAmZTqCB+v1qOltnP8btav8MOBFlM+kzYCfUwa1tLLY72H9AntxXVo5lNLttoDSk3KYpDMoX4bbSaB8KHCqpM0ov4v/Cn9vgX+rjfKtjdPg09GAg4iJQtKBttsZ9TduyqsMlnmB7RlLev6h1iHlF7Xi+ht50wM7G3Bwy5fG4YCDiAnksxOtvO2nbfdN5rik5x+OY0z08kAGHESMO5JuHGgTbYyYnOjle6EOY718W8ZQQOlEgk9MZEs6YnKil++FOoz18oMbYzeOdiLBJyayJR0xOdHL90Idxnr5lkaiK03S4ZQBFaKMMP2vpu0fB95bny5DGVW6ju25ku4EHqeMBpw31OtLGXAQEdGjVnzeRn7B+zobcHDzlwcfcCBpC8q9dNtQMjJcDBzqmkC6n/3fTslWsVN9fiflnq6OR/U2yoCDiIgeNgIDDjanpAB7qt7beBUlQe9A9qGkzRpWCT4REb1s+BOLzgC2l7R2zUayGyUp72Lq9l0oKaAaa3SppBskHTKEVwTkmk+MUTUVzb6UfucFwAeA11GmYxg0w4KkI9rZL2LUDW3AwSRJjbn5TmlMM2T7FkknApcCT1JuBh4om8Pbgd940SSz29meXW/MvkzSn2xf3WklE3xizJH0OmB34NW2n5E0CViOMtvuj1g4VcRAjmhzv4hRJdpL1NhkTqtBALZPo+RXpCZAHSjjy940dbnZnl3/f0DS+ZRrRx0Hn3S7xVi0PuUPrC+33hzgXZTM2r+S9CsAlbmnpkm6WdJn67qP9rPfzpKukfR7Sf9dk7Mi6QRJf1TJlt5OksmI4TcC8/n0pZNSmYLkncBZ/eyzOvBGyhQjfetWlrRq32NgZ0o3XsfS8omx6FLgWJV5oX4J/NT212uCyx0bRuF8sg4NXZqSRf0VzfvVVtOngDfbflJlosMjJX2LkqD1JbYtaY2uv8oIRixrwXkq07E8B3zI9iOSDgWw3Tcdxz8Bl7pM09JnPeD8mjN1GcrMwe3k0FtMgk+MOS7TX2xFmUphR+Cn6n/21nfXC6LLUFpLLwWa70h/bV3/m/oHtRxwDfAo8DfgNEm/oNzPEdF9IxB8bG/fz7qTm56fQZkzqnHd7cArh6MOCT4xJtXMxVcCV9asyItMYyFpU+Ao4DW2H66Zilfo51ACLrO9z2IbpG2AN1G69D5MyUAd0V3j9FbMXPOJMUfSi2sK+z5bAndR7rruS++/GmUkz6Mqc/Q0TmnduN+1wD/WaQj6+rRfVK/7rG57KvAxhunbXkRHOrzHJ4lFI0bWKpTZOtegTAg4EziEcjPcxZLusb2jpP+jzJZ6N9A4R9MpTfsdAJythTNnfooSoC6QtAKlddTZbeYRw2UMBZROJL1ORESPWmndjfzivTr73jP922NjPp+0fCIietk4bR8k+ERE9LCxdB2nEwk+ERG9KvP5RETEqEjwiYiIbhLpdouIiNGQ4BMREd2mcXo7TIJPRESvyoCDiIgYDbnmExER3ZfgExER3ZaWT0REdF+CT0REdNUYmyahEwk+ERG9LMEnIiK6KRkOIiJidOQm04iI6La0fCIioruS4SAiIkaDFox2DUZGgk9ERC8bpy2fpUa7AhERMTC5s6WtY0qHS5oh6WZJR/SzfQdJj0qaXpdjG7btIunPkmZKOmaorystn4iIXmWGfbSbpC2A9wPbAM8CF0v6he2ZTbv+2vbuTWWXBr4FvAWYBVwvaYrtP3Zaj7R8IiJ62Ai0fDYHrrP9lO15wFXAO9uszjbATNu3234W+Amwx1BeV4JPREQvc4cLTJI0rWE5pOmIM4DtJa0taSVgN2Cjfs78Okl/kHSRpJfVdZOBuxv2mVXXdSzdbhERPWqIGQ7m2N56oI22b5F0InAp8CQwHZjftNvvgefbfkLSbsDPgc06rskg0vKJiOhVdudLW4f1aba3sv0G4GHg1qbtj9l+oj6eCiwraRIwm0VbSRvWdR1LyyciooeNRIYDSevafkDSxpTrPa9t2v484H7blrQNpaHyEPAIsJmkTSlBZ29g36HUIcEnIqKXjcx9PudJWht4DviQ7UckHQpg+2TgXcBhkuYBTwN72zYwT9KHgUuApYHTbd88lAok+ERE9LCRaPnY3r6fdSc3PP4m8M0Byk4Fpi5pHRJ8IiJ6lYEF4zPFQYJPREQvG5+xJ8EnIqKXZUqFiIjovkwmFxER3ZaWT0REdFcmk4uIiG4r6XXGZ/RJ8ImI6GWZyTQiIrotLZ+IiOiuXPOJiIjuaz9T9ViT4BMR0cMy1DoiIrovLZ+IiOgqgzLaLSIiui4tn4iI6LrxGXsSfCIielnu84mIiO5L8ImIiK4ySa8TERHdJZxut4iIGAUJPhER0XUJPhER0VXj+JrPUqNdgYiIGJjsjpa2jikdLmmGpJslHdHP9vdKulHSTZJ+K+mVDdvurOunS5o21NeVlk9ERC8b5m43SVsA7we2AZ4FLpb0C9szG3a7A3ij7Ycl7QqcAmzbsH1H23OWpB5p+URE9Kw6pUInS2ubA9fZfsr2POAq4J2LnNX+re2H69NrgQ2H9WWR4BMR0bvMUILPJEnTGpZDmo46A9he0tqSVgJ2AzYapBYHARc11epSSTf0c+y2pdstIqKXdT7gYI7trQfaaPsWSScClwJPAtOB+f3tK2lHSvDZrmH1drZnS1oXuEzSn2xf3Wkl0/KJiOhhIzHgwPZptrey/QbgYeDWxc4rvQI4FdjD9kMNZWfX/x8AzqdcO+pYgk9ERC8b/ms+1FYLkjamXO85q2n7xsDPgPfZvrVh/cqSVu17DOxM6cbrWLrdIiJ6lYEFI3KT6XmS1gaeAz5k+xFJhwLYPhk4Flgb+LYkgHm1K2894Py6bhngLNsXD6UCCT4RET2r/dZMR0e1t+9n3ckNjw8GDu5nn9uBVzavH4oEn4iIXpb0OhER0XUJPhER0VUjd81n1CX4RET0LIPHZ2bRBJ+IiF6WbreIiOiqdLtFRMSoSMsnIiK6LsEnIiK6a2RuMu0FCT4REb3KwIKMdouIiG5LyyciIrouwSciIrrLGWodERFdZnAyHERERNel5RMREV2Xaz4REdFVdoZaR0TEKEjLJyIius1p+URERHclvU5ERHRbplSIiIhRkft8IiKimwx4nLZ8lhrtCkRExADs0vLpZGmDpMMlzZB0s6Qj+tkuSV+XNFPSjZJe3bBtf0m31WX/ob60tHwiInrYcLd8JG0BvB/YBngWuFjSL2zPbNhtV2CzumwLfAfYVtJawHHA1pSG2Q2Spth+uNN6pOUTEdHLhr/lszlwne2nbM8DrgLe2bTPHsAPXFwLrCFpfeCtwGW259aAcxmwy1BeVlo+ERE96nEevuSXPndSh8VWkDSt4fkptk9peD4DOF7S2sDTwG5A4/4Ak4G7G57PqusGWt+xBJ+IiB5le0itihbHvEXSicClwJPAdGD+cJ+nlXS7RURMMLZPs72V7TcADwO3Nu0yG9io4fmGdd1A6zuW4BMRMcFIWrf+vzHles9ZTbtMAfaro95eCzxq+17gEmBnSWtKWhPYua7rWLrdIiImnvPqNZ/ngA/ZfkTSoQC2TwamUq4FzQSeAg6s2+ZK+jxwfT3O52zPHUoF5HGaNygiInpXut0iIqLrEnwiIqLrEnwiIqLrEnwiIqLrEnwiIqLrEnwiIqLrEnwiIqLr/j/19LdWC5auCQAAAABJRU5ErkJggg==\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": [
    "## Collect weights"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T13:02:16.350336Z",
     "iopub.status.busy": "2022-09-19T13:02:16.350067Z",
     "iopub.status.idle": "2022-09-19T13:02:17.054728Z",
     "shell.execute_reply": "2022-09-19T13:02:17.053787Z"
    }
   },
   "outputs": [],
   "source": [
    "from pbo.weights_collection.weights_buffer import WeightsBuffer\n",
    "from pbo.networks.learnable_q import TableQ\n",
    "\n",
    "\n",
    "weights_buffer = WeightsBuffer()\n",
    "\n",
    "# Add initial validation weights\n",
    "q_random = q = TableQ(\n",
    "    n_states=n_states,\n",
    "    n_actions=n_actions,\n",
    "    gamma=gamma,\n",
    "    network_key=q_network_key,\n",
    "    zero_initializer=True\n",
    ")\n",
    "validation_initial_weight = q.to_weights(q.params)\n",
    "\n",
    "weights_buffer.add(validation_initial_weight)\n",
    "\n",
    "# Add randow weights\n",
    "q_random = q = TableQ(\n",
    "    n_states=n_states,\n",
    "    n_actions=n_actions,\n",
    "    gamma=gamma,\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",
    "    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-19T13:02:17.058586Z",
     "iopub.status.busy": "2022-09-19T13:02:17.058378Z",
     "iopub.status.idle": "2022-09-19T13:03:03.711637Z",
     "shell.execute_reply": "2022-09-19T13:03:03.710909Z"
    }
   },
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "13b63b4fd53b4133bc95f88dffd19735",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/400 [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 1 1 1 1 1 1 1 1 0 0 1 1 1 0 0 0 1 1]\n",
      "[1 0 0 1 1 1 1 1 1 0 0 1 1 0 1 0 0 0 1 1]\n",
      "[0 0 0 1 1 1 1 1 1 0 0 1 1 0 0 0 0 1 1 1]\n",
      "[0 0 0 0 1 1 0 1 1 0 0 1 1 0 0 1 1 1 1 1]\n",
      "[1 0 0 0 0 0 0 1 1 1 0 1 1 0 0 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 1 0 1 0 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 0 1 0 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n",
      "[1 0 0 0 0 0 0 0 0 0 1 1 1 1 1 1 1 1 1 0]\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAZEAAAEWCAYAAACnlKo3AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAABjQElEQVR4nO2deXxdVbn3v8/eZ0pykrZJmk7p3NKWltLSAh0UmWQWXhWuIMpQfBlEGVSqqFwLgspwr8KLXARBROQqgggCQpVSBhlLmVo6BEo607RpMw9net4/9j4nJycn80ly0q7v53M+e+2111p77XOS9axnDb8tqorBYDAYDD3BGugKGAwGg2HwYoyIwWAwGHqMMSIGg8Fg6DHGiBgMBoOhxxgjYjAYDIYeY4yIwWAwGHqMMSIGQxcQkX+IyPmZTtvNOhwtItsyXa7B0Bs8A10Bg6GvEJG6pNNcoBmIuueXqOofu1qWqp7cF2kNhsGOMSKG/RZVDcbDIlIOfENV/5WaTkQ8qhrpz7oZDPsLZjjLcMARHxYSke+LyKfA70RkmIg8JSK7RWSfGy5NyrNSRL7hhi8QkVdE5DY37ScicnIP004UkZdEpFZE/iUivxaRh7r4HDPce1WJyFoROT3p2iki8qFb7nYR+Z4bX+w+W5WI7BWRl0XEcq+NFpHH3O/gExG5Iqm8I0RklYjUiMguEfnvHv8Ahv0KY0QMByojgUJgPHAxzv/C79zzcUAjcGcH+Y8ENgDFwC3AfSIiPUj7MPAmUAQsA77elcqLiBf4O7AcKAG+DfxRRKa5Se7DGbLLB2YBK9z47wLbgOHACOCHgLqG5O/Ae8AY4DjgKhE50c13O3C7qhYAk4FHulJPw/6PMSKGA5UY8BNVbVbVRlWtVNXHVLVBVWuBm4DPdZB/s6req6pR4PfAKJxGuctpRWQccDjwn6oaUtVXgCe7WP8FQBD4hZt3BfAUcI57PQwcLCIFqrpPVVcnxY8CxqtqWFVfVkdA73BguKre4Ja3CbgXODsp3xQRKVbVOlV9vYv1NOznGCNiOFDZrapN8RMRyRWR34jIZhGpAV4ChoqI3U7+T+MBVW1wg8Fuph0N7E2KA9jaxfqPBraqaiwpbjOOFwHwZeAUYLOIvCgiC934W4GPgOUisklEfuDGjwdGu8NcVSJSheOlxA3jRcBBwHoReUtETutiPQ37OWZi3XCgkipf/V1gGnCkqn4qInOAd4D2hqgywU6gUERykwzJ2C7m3QGMFREryZCMAzYCqOpbwBnusNe3cIafxrpe1neB74rILGCFiLyFY7w+UdWp6W6mqmXAOe6w15eAR0WkSFXru/vQhv0L44kYDA75OPMgVSJSCPykr2+oqpuBVcAyEfG53sIXupj9DaABWCoiXhE52s37J7esc0VkiKqGgRqc4TtE5DQRmeLOyVTjLHmO4czL1LqLDXJExBaRWSJyuJvvayIy3DVYVW4dkr0gwwGKMSIGg8OvgBxgD/A68Gw/3fdcYCFQCdwI/BlnP0uHqGoIx2icjFPnu4DzVHW9m+TrQLk7NHepex+AqcC/gDrgNeAuVX3Bna85DZgDfOKW+VtgiJvvJGCtu/fmduBsVW3s+WMb9hfEvJTKYMgeROTPwHpV7XNPyGDIBMYTMRgGEBE5XEQmi4glIicBZwB/G+BqGQxdxkysGwwDy0jgrzj7RLYBl6nqOwNbJYOh65jhLIPBYDD0GDOcZTAYDIYec8ANZxUXF+uECRMGuhoGg8EwqHj77bf3qOrw1PgDzohMmDCBVatWDXQ1DAaDYVAhIpvTxZvhLIPBYDD0GGNEDAaDwdBjjBExGAwGQ4854OZE0hEOh9m2bRtNTU2dJzYMSgKBAKWlpXi93oGuisGwX2GMCLBt2zby8/OZMGEC7b9XyDBYUVUqKyvZtm0bEydOHOjqGAz7FVk9nCUi5SLygYi8KyJtllS5rzmtdq+/KyL/2ZP7NDU1UVRUZAzIfoqIUFRUZDxNg6EPGAyeyDGquqeD6y+raq9fkGMMyP6N+X0Nhr5hMBiRrGBfQwgBhuR4TYNkMBgMLlk9nIXz9rnlIvK2iFzcTpqFIvKeiPxDRGamSyAiF4vIKhFZtXv37u5XQpV99SG27G1gy94GwtHMvounsrKSOXPmMGfOHEaOHMmYMWMS56FQqMO8q1at4oorruj0HosWLcpUddtQVVXFXXfdlbHy/vKXvzBz5kwsyzIbQw2GLCerBRhFZIyqbheREuCfwLdV9aWk6wVATFXrROQU4Pb2Xu8ZZ/78+ZraMK1bt44ZM2Z0WBdVZXddM7tqmrEFRg/N6ROvZNmyZQSDQb73ve8l4iKRCB5P9jqN5eXlnHbaaaxZsyYj5a1btw7Lsrjkkku47bbbmD9/fsbK7ex3NhgM6RGRt1W1zT9jVnsiqrrdPVYAjwNHpFyvUdU6N/wM4BWR4r6oi4hQkh9gakkQn8dmy94GNldm3iuJc8EFF3DppZdy5JFHsnTpUt58800WLlzI3LlzWbRoERs2bABg5cqVnHaaMyW0bNkylixZwtFHH82kSZO44447EuUFg8FE+qOPPpozzzyT6dOnc+655xLvSDzzzDNMnz6defPmccUVVyTKTWbt2rUcccQRzJkzh9mzZ1NWVsYPfvADPv74Y+bMmcM111wDwK233srhhx/O7Nmz+clPnPcrlZeXJ+45Y8YMzjzzTBoaGtrcY8aMGUybNi2D36bBYOgrsrZ7KyJ5gKWqtW74BOCGlDQjgV2qqiJyBI5RrOzNfa//+1o+3FHTabpwNEYoGkMAn8fGY7XvkRw8uoCffCHtSFuHbNu2jVdffRXbtqmpqeHll1/G4/Hwr3/9ix/+8Ic89thjbfKsX7+eF154gdraWqZNm8Zll13WZm/EO++8w9q1axk9ejSLFy/m3//+N/Pnz+eSSy7hpZdeYuLEiZxzzjlp63T33Xdz5ZVXcu655xIKhYhGo/ziF79gzZo1vPvuuwAsX76csrIy3nzzTVSV008/nZdeeolx48axYcMG7rvvPhYvXsySJUu46667WnldBoNhcJHNnsgI4BUReQ94E3haVZ8VkUtF5FI3zZnAGjfNHTjvfe6X8TmvbZHjtRERmsNRmsJRMn3ns846C9u2Aaiuruass85i1qxZXH311axduzZtnlNPPRW/309xcTElJSXs2rWrTZojjjiC0tJSLMtizpw5lJeXs379eiZNmpTYR9GeEVm4cCE/+9nPuPnmm9m8eTM5OTlt0ixfvpzly5czd+5cDjvsMNavX09ZWRkAY8eOZfHixQB87Wtf45VXXun+F2MwGLKGrPVEVHUTcGia+LuTwncCd2byvt31GFSVPXUhdtU0IQKjhuQwLDczcyV5eXmJ8HXXXccxxxzD448/Tnl5OUcffXTaPH6/PxG2bZtIJNKjNO3x1a9+lSOPPJKnn36aU045hd/85jdMmjSpVRpV5dprr+WSSy5pFV9eXt7mezEr3QyGwU02eyKDAhFheL6fqSVBAh6bbfsaKK9sIBTJ7FxJdXU1Y8aMAeCBBx7IaNkA06ZNY9OmTZSXlwPw5z//OW26TZs2MWnSJK644grOOOMM3n//ffLz86mtrU2kOfHEE7n//vupq6sDYPv27VRUVACwZcsWXnvtNQAefvhhPvOZz2T8WQwGQ/9hjEiG8HttJg3PY/TQHOqbI5TtqqWyvplMja4tXbqUa6+9lrlz53bLc+gqOTk53HXXXZx00knMmzeP/Px8hgwZ0ibdI488wqxZs5gzZw5r1qzhvPPOo6ioiMWLFzNr1iyuueYaTjjhBL761a+ycOFCDjnkEM4888yEkZk2bRq//vWvmTFjBvv27eOyyy5rc4/HH3+c0tJSXnvtNU499VROPPHEjD+vwWDIDFm9xLcv6OkS3+7QHImybV8j9c0Rgn4PpcNy8HnsjJXfV9TV1REMBlFVLr/8cqZOncrVV1+dsfIzvRS4u5glvgZDzxmUS3wHK36PzaTiPMYMzaEhFGXjrjoq6zLnlfQV9957L3PmzGHmzJlUV1e3mdMwGAyGVIwnQt/2UEOuV1I3yLyS/RHjiRgMPcd4IgOEz2Mz0fVKGl2vZM8g8EoMBoOhKxgj0g+ICEVBP1NH5JPn97CjqpFNe+ppDkcHumoGg8HQK4wR6Ud8HosJRbmUDsulKRylrKKOPbXGKzEYDIOXrN1suL8iIhTm+Qj6PWyvamRHdSPVjWFKh+Xg95q5EoPBMLgwnsgAEfdKxg7L5dOK3Rw6Zw6zZh9qpOCT+K//+i9EhD17nHeSrV+/noULF+L3+7ntttsyfj+DwdB9jCcygIgIw/J8HDFjPM+++Do1TWF+e/stjCoeyrXfX5pI15EU/Pz587sklf7qq69mrN6pxI3IN7/5zYyVuXXrVpYvX864ceMScYWFhdxxxx387W9/y9h9DAZD7zCeSBbgtS3GF+UyrjCXSCzGnroQZ5/7dS655JIDUgoe4Oqrr+aWW25ppa1VUlLC4Ycf3kaV2GAwDBzGE0nlHz+ATz/IbJkjD4GTf9FhEhFhaK6Pojw/MduiMRRl16bNrHjxZfICvgNKCv6JJ55gzJgxHHpoG/1Ng8GQZRgjkmXYllCQ6yXP7+HIU/8PmyobGZEfo7GqivPPP5+ysjJEhHA4nDZ/XAre7/cnpOBLS0tbpYlLwQMJKfhgMNhGCv6ee+5pU/7ChQu56aab2LZtG1/60peYOrXtiySTpeDBkVMpKytj3LhxbaTg77jjjlZGpKGhgZ/97GcsX768B9+ewWDob4wRSaUTj6E/EBF8HotJowspCHj4tKaJZddcy2c++7n9Xgr+448/5pNPPkl4Idu2beOwww7jzTffZOTIkV2up8Fg6B/MnEgW47EsxhflMb4wl5qaGiRYyK6aJn73u99l/F7ZIgV/yCGHUFFRQXl5OeXl5ZSWlrJ69WpjQAyGLMUYkUHAkFwfP/nRD7jz5p/yuUVHUFHTQKa3J2aTFHx7fPrpp5SWlvLf//3f3HjjjZSWllJT0/mrjA0GQ9+R1QKMIlIO1AJRIJIq/iXOWMjtwClAA3CBqq7uqMz+FmDMNNWNYbbvayQaU4bn+ykp8GNl6O2ARgreYDC0x2AWYDxGVeekqzxwMjDV/VwM/E+/1mwAGJLj5aARQYbmeqmobeKjijoaQpl5SZWRgjcYDN1lsE+snwE8qI479bqIDBWRUaq6c6Ar1pd4bIuxhbkMyfGyvaqRjyvqGZ7vo6Qg0Cuv5Oqrr86o55HKhAkTBswLMRgMfUO2eyIKLBeRt0Xk4jTXxwBbk863uXGtEJGLRWSViKzavXt3H1W1/ynI8TK1JO6VNPPRrsx5JQaDwdAVst2IfEZVD8MZtrpcRI7qSSGqeo+qzlfV+cOHD89sDQeYuFcyoTiPqCofV9Sxs7qRWCx757oMBsP+Q1YbEVXd7h4rgMeBI1KSbAfGJp2XunEHHAUBZ65kWJ6P3bXNlFXUUd9svBKDwdC3ZK0REZE8EcmPh4ETgNQB9SeB88RhAVC9v8+HdIRtWZQOy2VicR6qyse769hRZbwSg8HQd2StEQFGAK+IyHvAm8DTqvqsiFwqIpe6aZ4BNgEfAfcCmZOR7UcqKysT0u+ZkILPD3iZOiJIUZ6PPXWOV7Jg4cI+q3+mpeDfffddFixYwJw5c5g/fz5vvvkmYKTgDYZsJKv3ifQF2b5PZNmyZQSDwVZ6Uh1JwXdGXVOYbfsaCUVjFAf9jCgIYFuZ2VcSJ9P7P0444QSuvvpqTj75ZJ555hluueUWVq5cSUVFBZs3b+Zvf/sbw4YNayPc2BnZ9DsbDIONwbxP5IDkggsu4NJLL+21FHww4GXqiHwWTi9lT10zf/zbM3z2qM9ltRS8iCR2oldXVzN69GjASMEbDNnIYN8nknFufvNm1u9dn9EypxdO5/tHfL/b+bZt28arr76Kbdu9loIXYFJxkHeBd999hxWvvc2caZM46rOfyTop+F/96leceOKJfO973yMWi/XpC7UMBkPvMJ5IFnPWWWdh285716urqznrrLOYNWsWV199NWvXrk2bJy4FX1xcnJCCjxMMeBgzNJe58w7HP2Q4H++uZ+YhsykvL2f9+vVtpODTsXDhQn72s59x8803s3nzZnJyctqkSZaCP+yww1i/fj1lZWUAbaTgX3nllTb5/+d//odf/vKXbN26lV/+8pdcdNFF3fjWDAZDf2I8kRR64jH0FXl5eYnwddddxzHHHNNrKXjbEgrycpg8PMi2fY3UhWJUVNcTjcW6VKe+loIH+P3vf8/tt98OOIb0G9/4RpfqZjAY+h/jiQwSqqurGTPG2Yz/wAMP9Lq8PL+HqSVBcrw29c1R7KFj+PjjgZeCBxg9ejQvvvgiACtWrEj74iuDwZAdGE9kkLB06VLOP/98brzxRk499dSMlGlZQp7fQ0m+n5zcXJb+9BaOP+FEhuQHOfzww9PmeeSRR/jDH/6A1+tl5MiR/PCHP6SwsDAhBX/yySdz6623sm7dOha6y4qDwSAPPfQQtm0npOCXLFnCwQcfnFYK/t577+XKK68kEokQCAQSb1j89NNPmT9/PjU1NViWxa9+9Ss+/PBDCgoKMvJ9GAyG7mOW+GKWfgLEYsqmnXtoVC+2Jfz39d9n5vRpRgreYDAAZomvoRMsS/j7Iw9x7qmf4/Sjj2RnRSWnnvV1ItGuzZUYDIYDE+OJYHqoqcRUqahpYndtCI8tjBmaQ0HO4N+bYX5ng6HnGE/E0GUsEUYOyWFySR62JZRX1rN1b4PxSgwGQxuMETG0S67Pw5SSICUFAaoawmysqKO6MTzQ1TIYDFmEMSKGDrFEGFkQYEpJHh5L2FxZz5ZK45UYDAYHY0QMXSLH9UpGFASobgqzcVcd1Q0dKwwbDIb9H2NEsoBMS8GnY9GiRb2upyXCiIIAU4YH8drC5r0NbK6sZ3fl3oxKwf/lL39h5syZWJZF8iKIf/7zn8ybN49DDjmEefPmsWLFisS1H/3oR4wdO5ZgMJixehgMhi6gqp1+gM/jvK9jjnt+cVfyZeNn3rx5msqHH37YJm6g+MlPfqK33nprq7hwODxAtWmfaCymu6ob9f1tVbr89fd1+oyDNRaLZaTsDz/8UNevX6+f+9zn9K233krEr169Wrdv366qqh988IGOHj06ce21117THTt2aF5eXoflGgyGngGs0jRtalc9kSXANcDXRORYYE7mzZkhmUxJwceJ99BXrlzJ0Ucf3Wsp+MPmzuX4zxyBVO/k9p8v45NNm5g5+1C++11Hkbc3UvAzZsxg2rRpbeLnzp2bkIWfOXMmjY2NNDc3A7BgwQJGjRrVsy/bYDD0mK7KntSqahXwPRH5BZBeE2M/4NOf/YzmdZmVgvfPmM7IH/6w2/kyKQWfzDvvvMPatWsZPXo0ixcv7rUU/P/75W2cvHE9f372ZSyBx558mo0bN/ZYCr4rPPbYYxx22GGtBCcNBkP/01VP5Ol4QFV/ADzYN9VpjYjYIvKOiDyV5toFIrJbRN51P/ud1GumpeDjHHHEEZSWlmJZFnPmzOm1FLyI4LGEqSVB/B6bJ59+ln88t5w5PZSC74y1a9fy/e9/n9/85jfdzmswGDJLlzwRVX0i5fz/9U112nAlsA5oT2Hvz6r6rUzesCceQ1/RF1LwXU3THh1JwQe8NpOH55Hrs1ly+dV85esXMnpIDkNzvYhIl6XgO2Lbtm188Ytf5MEHH2Ty5MndymswGDJPt1ZnicMiEblORO4TkSdE5C8i8msRuVxEpmSqYiJSCpwK/DZTZQ5mMi0Fn8q0adPYtKn3UvAiwhe/cArPPvYw0eZGtu5r4PU1H7F9x6dA16Tg26OqqopTTz2VX/ziFwlvxmAwDCxdMiIiYonIBcDNwDTgCeAHwJnA+cBNwFvAKSJyvYicmIG6/QpYCnS0q+3LIvK+iDwqImM7qP/FIrJKRFbt3r07A1Xrf5YuXcq1117L3Llzu+U5dJWcnBzuuusuTjrpJObNm0d+fj5Dhgxpk+6RRx5h1qxZzJkzhzVr1nDeeedRVFSUkIK/5pprOOGEEzj33HM59/TPc/aJn+HSC87l/fJPqWoIJaTgZ8yYwb59+9JKwT/++OOUlpby2muvceqpp3Liic6f05133slHH33EDTfckFgCHX9PydKlSyktLaWhoYHS0lKWLVuW8e/IYDC0pVMBRhEpBE4ElqtqZZcKFZkMzFTVJ3tUKZHTgFNU9ZsicjTwPVU9LSVNEVCnqs0icgnwFVU9trOyjQBj+9TV1REMBlFVLr/8cqZOnZoRKfjmcJRtVY1s/GgTVy05mw8+WIPP0/9blMzvbDD0nN4IMFar6v921YAAqOrHqvqkiPS0pVgMnC4i5cCfgGNF5KGUe1SqarN7+ltgXg/vZXC59957mTNnDjNnzqS6urrN6217it9rM6k4jxEFfmIKZbtqqaxvprMOjMFgyH4yIgUvIsNxJr+3qGpGFfo68ERGqepON/xF4PuquqCz8ownMrCEIlHn3e7NEYJ+D6XDcvB57H65t/mdDYae054n0qvX47rDSJ8BGoEqYIyIVAO3quonvSm7nfvdgLNr8kngChE5HYgAe4ELMn0/Q+bxeWwmFuextz7EzuomNu6qY9SQAIV5vm6v1DIYDANPb9+x/paqtlqsLyJ5wOhelptAVVcCK93wfybFXwtcm6n7GPoPEaEo6Cc/4GHbvka2VzVS1RimdFgO/n7ySgwGQ2bo1eymqq4Wkf8UkRFJcfWqWtb7qhn2d+JeSemwHJpCUcp21bGnzsyVGAyDid56IgC/Ay4TkXzgj6q6OgNlGg4QRITCPD9Bv5ftVY3sqGqkujFM6dAc/F7jlRgM2U4m1llegWOMLJy9HYZuMlik4Nujqqqq11LwPo/FhKJcSofl8sRfH2PmrFlYlsVbb73VKt3777/PwoULmTlzJocccghNTU2AkYI3GAaMdNK+3fkAFwO3AUf3tqz++Bgp+MzzySef6MyZMzNW3nvvr9F/vbZa5y9YrH997kVtDEVU1fkeDjnkEH333XdVVXXPnj0aiTjXjBS8wdC30Esp+I5YB6wALhKRtzpLbOga2S4FP2fOHGbPnk1ZWRk/+MEP+Pjjj5kzZw7XXHMN0Dsp+NmHzOTYI+fg99qEojE+qqhjd20Tzz33HLNnz+bQQw8FoKioKCFQaaTgDYaBobdLfAPABGAf8Bv3iIgEVLWp17UbAF5+ZCN7ttZltMzisUE++x8HdTvfYJGC/8UvfsGaNWt49913AVi+fDllZWW9koKPKwOPL8wl6Pews7qJV1evIaZw4oknsnv3bs4++2yWLl3a7e/VYDBkjt56Ij8HDgNGup9TROQuzO7xjDBYpOBTWb58OcuXL2duBqTgPbbF+KJcxhXmEgqHefHll/nVb+7j5Zdf5vHHH+f555/vML/BYOhbeuWJqOrVrrbWQpwd6y+o6q0ZqdkA0ROPoa8YbFLwcVSVa6+9to1sSk+l4EWEobk+Dp0+iU2LPkPYE2RHfYwTTjyJ1atXc9xxx3W5/gaDIbP0ek5EVfeq6tPq6Gut6jyHoScMFil4cIab7r//furqnGHB7du3J9R2eyMFf+rJJ/PJxnUUB6CxKcxzz7/AqAlTCEc7Eno2GAx9Sa+NiIgcn4mKGDpmsEnBf/WrX2XhwoUccsghnHnmmQkj0xsp+GHDhvGd73yHz39uMV895SjmzJnL7IXHsG5nDf/3W1cxeswYIwVvMPQzPRJgFJH/Bd4BBFikqmckXfuSGz9NVX+WqYpmCiPA2D59JQUfp7y8nNNOO401a9ZkrMzGUJSapjC1TREaQo5x9doW+QEP+QEvQb8H23KGzMzvbDD0nEwLMN6lqi+7Bc9KubZDVV8Xkek9LNswQNx77738/ve/JxQKMXfu3IxJwfclOT6bHJ/NiAIIR2PUNkWobQpT3RBmb30IESHo95Af8BCJmWEvgyHTZEQKfjBhPJEDg5gqDc0Rapoi1DZFaI5E2bVlE7e8XsuxM0o4dloJ88YPw2P3/8uxDIbBSF9JwZ+oqs+liR+vqpt7U7bB0BssEYIBL8GAs0emORylebeXkgI/9738Cb95cRNDcrx87qDhHDu9hM8dNJxheb4BrrXBMPjokRERkXE4y3pPEpEhqvpI0rUC4GwReUtVV2SongZDr/B7bYIBD3/8xgJqm8K8UraH59dXsHJDBU++twNLYN74YRwzvYTjpo/goBFB834Tg6EL9NQT2QO8DhzsHhOoao2IPKOqH/S2cgZDX5Af8HLyIaM4+ZBRxGLK+9urWbFuFys2VHDLsxu45dkNjBmaw7HTSzh2RgkLJxURMIrCBkNaemREVLUB2Cwif1PVLcnXRGQsME5ExqrqM72pnIjYwCpgu7Z9Pa4feBBnd3wl8BVVLe/N/QwHHpYlzBk7lDljh/KdE6bxaXUTL2yoYMX6Ch59ext/eH0zAa/FZ6YUc8z0Eo6dXsKoIW136RsMByq9fSnVO2nitrqbD3tlQFyuxBF4TMdFwD5VnQL8Erg5A/cbEIwUfGuuueYapk+fzuzZs/niF79IVVVV4trPf/5zpkyZwrRp03juuZbpuCVLllBSUsKsWamLBbvHyCEBzjliHPeeN593/vPz/H7JEXxl/ljWf1rLjx5fw8Kfr+CU21/mtuc2sHrLPqKxA2thisGQSo9XZ4nIWar6FxGZpapr3LhDgQtxPIPNqvpgjysmUgr8HrgJ+E4aT+Q5YJmqviYiHuBTYLh28kDZvjpr2bJlBIPBVqKEkUgEjycT7w/rGzK9/2P58uUce+yxeDwevv/97wNw88038+GHH3LOOefw5ptvsmPHDo4//ng2btyIbdu89NJLBINBzjvvvHbr0ZvfWVUpq6hjxfoKVqyr4G3XgBTl+fjctOEcN30Enz2omIKAt/PCDIZBSHurs3rkiYjIIqBORH4MzEy6NBSYDFQDp4tI2+3IXedXwFKgvcX9Y4CtAKoace9Z1E59LxaRVSKyavfu3b2oUv9xIEvBn3DCCQmjuWDBArZt2wbAE088wdlnn43f72fixIlMmTKFN998E4CjjjqKwsLC3n/x7SAiHDQin0s/N5lHLl3I2z8+ntvPnsNnpxazYn0Flz+8msNu+Cfn3PM6v315E5t2Z1YJ2mDIVnravW0CpgBhYHg8UlVfFBEvsAhYDdzTk8JF5DSgQlXfFpGje1jHBKp6T7wu8+fP79BTeeGBe6jYvKm3t2xFyfhJHHPBxd3OdyBLwce5//77+cpXvgI4GlwLFixIXCstLWX79u3d/VozwtBcH2fMGcMZc8YQicZ4Z2tVwku58el13Pj0OiYU5XLs9BEcO72EIyYW4vOYPSmG/Y+eTqyvBlaLyPdU9c6Ua/8C/tXLei3G8WROAQJAgYg8pKpfS0qzHRgLbHOHs4bgDKPtN6RKwZ9//vmUlZUhIoTD4bR54lLwfr8/IQVfWlraKk1cCh5ISMEHg8E2UvD33NO2D7Bw4UJuuukmtm3bxpe+9CWmTp3aJk2yFDw4ciplZWWMGzeujRT8HXfc0a4Ruemmm/B4PJx77rld+boGDI9tcfiEQg6fUMj3T5rOtn0NvLC+gufXV/DQG5u5/9+fEPR7+OxUZ3L+mGklDM/3d16wwTAI6O1A+6TOk3QfVb0WuBbA9US+l2JAAJ4EzgdeA84EVnQ2H9IVeuIx9BUHshT8Aw88wFNPPcXzzz+fSDNmzBi2bt2aSLNt27aEsnE2UTosl68vnMDXF06gIRTh1Y8qeX59BS+sr+Afaz4F4NCxQzl2WgnHzShh5ugCsyfFMGjprREZ73oLCTK0KistInIDznt+nwTuA/4gIh8Be4Gz++q+2UB/SsFPmDChS1LwW7Zs4f333+fQQw9tIwV/3XXXce655xIMBtm+fXtiSC0uBb9w4cJ2peCfffZZbrnlFl588UVyc3MT8aeffjpf/epX+c53vsOOHTsoKyvjiCOOyPA3kVlyfR6OP3gExx88AlXlw501rFhXwYoNFfzq+Y388l8bKcn3c+z0Eo6ZXsJnphST58/eRRQGQyrd/msVkdHAcUAu8AhQAvTZOkdVXQmsdMP/mRTfBJzVV/fNNpYuXcr555/PjTfeyKmnnprx8pOl4PPy8jj88MPTpnvkkUf4wx/+gNfrZeTIkfzwhz+ksLAwIQV/8sknc+utt7Ju3ToWLlwIOJP6Dz30ELZtJ6TglyxZwsEHH5xWCv5b3/oWzc3NfP7znwecyfW7776bmTNn8h//8R8cfPDBeDwefv3rXyeG+8455xxWrlzJnj17KC0t5frrr+eiiy7K+PfUG0SEmaOHMHP0EL593FT21DWzcsNuXlhfwdPv7+RPb23FZ1ssmFzEsdOGc9yMEYwtzO28YINhAOnWEl8ROQFn2e1KoBk4FMeYXKiqr/ZFBTNNti/xHUgGoxR8d8jm3zkUibGqfK8zOb++gk176gGYWhJ0ds5PN4KRhoElUwKMNwKfVdWPkgpeCNwjIhcBDfE9I4bBx2CUgt9f8HksFk0pZtGUYn582sF8sqfeNSi7uP/fn/CblzZREPDwuWklHGcEIw1ZRHc9kXdVdU6a+IOAp4BmVT0kc9XLPMYTOXAZrL9zqmDknroQlsBh44Y5svbTS5g2It9Mzhv6lEx5Ik0iMlxVW+3YU9WNIhLFmSsxGAwZJK1gpOulJAtGHjPd2Tm/cLIRjDT0H901IrcCf3MlT3bEI0WkGMcLqcho7QwGQytaCUZ+/iB21TQl9qQ89vZ2Hnp9CwGvxeLJxQkvxQhGGvqSbhkRVX3MVc99TUTeBt4DfMB/4MyXGAyGfmREQYCzjxjH2UeMoykc5Y1P9rJi3S6edw0LwIxRBRznLiGeM3Zo4p3zBkMm6JEAo4jk4uzLmAXUAE+r6lsZrlufYOZEDlwOpN9ZVfmooo7n3dVeb292BCML83wcPc15m+NRBw03gpGGLpNRAUZVbVDV+1X1O6q6bLAYkGzFSMG3piMpeHA2LAaDQW677bZEXKak4PcXRISpccHIS1oEI49yBSO/9fA7CcHIe1/axMe768iA4IPhAKTHUvCDlWz3RIwUfPtS8HHOPPNMRIQjjzwy8T31tRT8/kQkGuPdrVWOl7Kugg27HLWBCUW5idcDG8FIQyoZ9UQMfY+Rgm8rBQ/wt7/9jYkTJzJz5sxWefpaCn5/wmNbzHfFIp+7+ihe+f4x/PSMmUwozuOPb2zha/e9wWE//SeX/uFtHlm1ld21zQNdZUMW06vurYh8QVX/nqnKZANVf/+Y0I76jJbpG53H0C9M7nY+IwXfWgq+rq6Om2++mX/+85+thrIMvSOdYOSKDY6X8uxaVzCydEhC1n7m6AIsMzlvcOntGMlNwH5lRLIJIwXfWgp+2bJlXH311QmvypB5WglG/h9HMDK+hDhZMPKYaSUcO8MIRhp6b0T2u+5ITzyGvsJIwbeWgn/jjTd49NFHWbp0KVVVVViWRSAQ4Fvf+laX62/oOsmCkd86diqVrmDkivUVPPPBTv68yhGMPHJSIcdNL+HY6SMYV2QEIw80emtEDqxZ+QHESMHDyy+/nAjHFyAYA9J/FAX9fHleKV+eV0o4GuOt8r0JWftlf/+QZX//kCklQY6bXsLhEwo5aEQ+pcNyzNDXfo7xQwcJRgr+7g7rPxik4PcnvLbFosnFLJrcWjDyhfUVCcFIgIDXYvLwIFNLgkwdkc+UkiAHjchnXGGu2fS4n9CrJb4i8r6qzs5gffqcbF/iO5AYKXhDJqhrjrDh01o+qqilbFcdZRV1fFRRx/aqxkQan8diUnEeU0fkOwamJMjUEUHGF+XhNXL3WUmmBBhT2dXL/IYswkjBGzJB0O9h3vhhzBs/rFV8XXOEjyrqKNtV6xwr6nh36z7+/l5Chg+PJUwszmPqiCBTS/ITxwnFufg9RlQyG8nazYYiEgBeAvw4xu5RVf1JSpoLcEQht7tRd6rqbzsq13giBy7md85OGkIRNu2up6yilo276ijbVcdHFbVs2dtAzG2ebEsYX5Trei2OcZlSEmTy8KBRLO4n+soT6UuagWNVtU5EvMArIvIPVX09Jd2fVdXMrhoMg5Rcn4dZY4Ywa8yQVvFN4WjCuDgeTB1lFbX8a10FUde6WALjCnOZkvBaHCMzuSSPXF82N2/7D1n7LavjItW5p173k51uk8FgyDgBr83Bows4eHRBq/jmSJTyPQ2UuXMuztBYLS9urCAcbWkiSoflMNWdyJ+SNLEfNPtaMkpGvk0RGQ4UAFtUNf0uuJ6VawNvA1OAX6vqG2mSfVlEjgI2Aler6tY05VwMXAwwbty4TFXPYDAMAH6PzbSR+Uwbmd8qPhyNsbmyITGhv9Gdf/n3R5WEorFEutFDAkxJmdCfUpLPkByjaNwTeit7cgmwGGgCqoAxIlIN3Kqqn/S2cqoaBeaIyFDgcRGZlfIO978D/6uqzW5dfg8cm6ace4B7wJkT6W29DAZD9uG1LaaUOHMlJyWJOUeiMbbua6RsV21ipVhZRS1/fKOSpnCLcRlR4GdqSdxrcedeSoLmXfad0FtP5C1gJHC3qu4CEJE8YHRvK5aMqlaJyAvAScCapPjKpGS/BW7J5H37i8rKSo47znmz8Keffopt2wwfPhyAN998E5+v/T/iVatW8eCDD7YSW0zHokWLePXVVzNX6SSqqqp4+OGH+eY3v5mR8q655hr+/ve/4/P5mDx5Mr/73e8YOnQo4XCYb3zjG6xevZpIJMJ5553HtddeCzhS8E899RQlJSUDtoTYkJ14bIuJxXlMLM7jhCTdzlhM2V7VyEbXuMQn9B9ZtZWGUDSRrjjocwxL0mqxqSOCFOX5zHvtycDqLBEZCyzBGc56SFXfyUjFnCGysGtAcoDlwM2q+lRSmlGqutMNfxH4vqou6KjcbF+dZaTg25eCf/jhh3nyySf505/+RENDAwcffDArV65kwoQJRgrekDFiMWVnTVNiKXLcyHy0q47a5haJoGG5XsdzSZrQnzoiSEm+f780Ln25OusKnJVUFnA7cFQGygQYBfzenRexgEdU9SkRuQFYpapPAleIyOlABNgLXJChew84F1xwAYFAgHfeeYfFixdz9tlnc+WVV9LU1EROTg6/+93vmDZtGitXruS2227jqaeeYtmyZWzZsoVNmzaxZcsWrrrqqsQLq4LBIHV1daxcuZJly5ZRXFzMmjVrmDdvHg899BAiwjPPPMN3vvMd8vLyWLx4MZs2beKpp55qVa+1a9dy4YUXEgqFiMViPPbYY1x33XUJKfjPf/7z3Hrrrdx666088sgjNDc388UvfpHrr7+e8vJyTjrpJObNm8fq1auZOXMmDz74YCtpE3Ck4OMsWLCARx99FHC0nOrr64lEIjQ2NuLz+SgocCZdjzrqKMrLy/vwFzEcKFiWMGZoDmOG5nD0tJJEvKqyq6Y5MaHvDI3V8vT7O6lubJkKLgh4Epso4xP6U0uCjBoS2C+NSyaMSBlwEPCEqmZse7Oqvg/MTRP/n0nha4FrM3VPgH/84x98+umnmSySkSNHcvLJJ3c7n5GCby0Ff+aZZ/LEE08watQoGhoa+OUvf2neIWLoN0SEkUMCjBwS4LNThyfiVZU9daHEnEvcyPzzw1386a2WdT5Bv4fJ8cn8pHmXMUMHt75YJozIOmAbcJGI3Kqq6UWXDN3GSMG3loJ/8803sW2bHTt2sG/fPj772c9y/PHHt1ERNhj6ExFheL6f4fl+Fk0pbnWtsq45sTu/ZSnybh59u+VFazle251zCbpDY47nMnaQ6ItlwohMBPYBv3GPg5qeeAx9hZGCby0F//DDD3PSSSfh9XopKSlh8eLFrFq1yhgRQ9ZSFPRTFPRz5KSiVvFVDaGEcYlvonxtUyV/fWd7Io3P44hXHuTOucQ3VI4vzMWTRfpivV3iG1DVB9uJb+pN2YbWGCl4Z4/PihUr+PrXv059fT2vv/46V111VWa/CIOhHxia62P+hELmT2g9HFvTFObjJMNSVlHHqvJ9PPFui76Y1xYmFQfbTOhPKMrD5+l/49JbT+TnIqLAhzj7RCa6nz8C/+5l2YYkjBT83Vx++eVceOGFzJw5E1XlwgsvZPZsR0TaSMEb9gcKAl7mjhvG3HGtxSvrmyN8vLuu1YT+B9uqeeaDncQX2HosYUJxXmLOJb6hcmJxXp/qi2ViiW8hsBBniW+Zqq7qJMuAku1LfAcSIwVvMAwuGkNRPt7dMt8Sl4Epr6xPiFdaAuOL8phSEuT602cyemhOj+7VZ0t8VXUv8HRvyzEMPEYK3mAYXOT47HbFK8sr6xOeS9muWjbuqiWvD0QpMy4FLyKlwGwAVX0mo4VngB57IrU7IRIC2+t8LG/r8H64/nt/w3giBkPPybgnIiJnqepfkvWsRORQ4EKgEtjc49pmI9EQNNdCrB19SSuNcUk9F9sYG4PBsF/RIyMiIouAOhH5Mc5mw/gg91BgMrAJ+IKI5Knq/2SiogPO0PHOURViEYiGHYMSDSWFwxBthlAdaLRtGWK19WDSGpvsWb5nMBgMHdFTT6QJR549DCS2bqrqi+4LpBYB7+Aq5+5XiLQ09h0RiyUZlpAbjrSEQ/XOtXSvSLE8nRsby2O8GoPBMOD0yIio6mpgtYh8T1XvTLn2L+BfmajcoMaywPKDx99+GlWIRVOMTaS1dxNucOLaIJ0Pn1lesPbPV4dqNIpGImgkglgWWBZi287RMp6cwdBf9Haq3mwV7g0iYHuorKruWAre63G8mISxCUPMMTSr3l7Ng3/6K3f89BrQWJp72GB7WXTaubz63ON9sjAgk1Lwqsp1P/4xTz75JJYIw4uK+O2vfsWoomIefvQv/Nfdd6OxGMHcXG6/7jpmT5sGwCXXXcezL73E8MJCVv3tCcS2wLKdo20jlkW0qopdv7gZqyAfO5iPlZ+PnR/ECuZj5Qex8924YBDxduJpHsCoKuFYmMZIIw3hBhoiDYlwY6SRhkgDUY1iYWGJhYhgiYWFE7bFbhNnidXqI7TEiUiirHRx8fy22Il86cpv776C7JfCiP1Fr1ZnicjTwK+T47JxRVYy2b5PpFdS8LFoyvxMqLXhiV9LRy8WBnR1/4eqouEwRCJoOOx80oRramspCAYBuOuPf2Tdxx9z5w0/5fUP3mfG9OkUFhXz3EsvcsNtt/HaypUQi/HSyy+Tl5PDhd/8Ju+uXIlGoxCNobEoRKNoLMbGbdvwfee7aGNjp1+lBAKOYQm2GBYrPz8pzjU6iXBB1hmieGPfXkOfHO7IILRKF26kMdJIRLsulTMYSGfQkg2SLXabuHiedMatjWFMMXjxcpINWxvDiIVtddEwpql7os5J9zx/5vkM8Q/p/AtJQ1/tE3mEpDkRzDvQM0aPpOBv+GnnUvAvvOBIwRcVsmbtWubNOYSH7rkD0QjPPPtPvnPdz8jLCbD48EPZtHkbTz2Y9LIrsVi7cTMXXv1jQuEIsZjy2B9/x3U33eJIwc+ezXHHHMPN11/Pbb/8JY8+8QTNzc2cfvzxXHf55ZRv3swZl17K3IMP5t1165gxZQq//dnPyCsoQDwerJwcpKCAohEjEa8H8XppzsvDU1hIYPo0jp4+LVGVxSeeyPbvfhfbNTbHnHqqo83l8eAdOTLtd+qNRJj+zmo0HCZaV0esro5YbS3R2jpidbVEa2uJJcLutbpaYjXOMfzpp25cXUYNkQSDaF4OoRwP4RwvTQGbpoBFQwAaNNRhQ5/a6KfzCLqKx/KQ68kl15tLjieHXI9zHJ47PBFOvpaaLtebmwjblk1MY6gqMY0RwwlHNdomLqaxlg+xVvmUlOtp8ilKNBZtlVZVW5WVuG9K+fHzdHVNvW+r8tPUK235aZ4p+bsIx8LtP1Pyd9XVe5LmelL5Xz7oyz02Iu3+3XQ3g4iMBo4DcoH1qvpiRms0wGzc+FNq69ZltMz84AwOOui6bufrEyl4Ed55993WUvAffOJIwV9zfYsU/Nlngz8fLZoK4SY0FELDIe564C9cft7XOOe0U2huDBGNRLn+/17CB+9+wGsPPwzAPx55hI1r1/LyQ38AS/jy5d/i3++/w7jx49hYXs69d/8Pnznqc1x02WXc/8ILXHPNNW2e40c/+hEPPvggQ4YM4YUXXmhz/b777uuxWKZ4vXiGDYNhwzpPnISq0hxtpjHSSH1TDY1VlTRWV9JUvZdwdRWhmioitTUJY0R9PdQ1IPWN2A078OxoxtsQxtcUIdAYxRduv8/lwZGA8HvA7wef+/H6BX9AyMvxEs71Es3xE83zE8vLgbxcJFiMlR/Ek1+AnV+Ar2AogZyg07h7kxr8NOfezhaLGAxp6JYREZETcN5jvhLnRVSXikgucIGqvpb56h3Y9LUUvKpy6OzZbNqwgRxg4tixlObkENq6jTOPO477Hn6Ypo+2tJprOXz6LG6+8x62bt3FF085halTJmM1hcG28A4fgljKyvfeYsWbr7Pw7P8AlLr6esq3rmfy+ALGjh7JUYeMgH0b+PoXjuKO3z7MNRed2Wb47Kbrr+Omn97Az2++hTvvvJPrr78+UYcXXniB++67j1deeSXtd6CqaXttzdFmVmxZ0eGQTfKwTbp0sXTzTsnkuJ8S8Fm+REOd6ylq05PPkwD5YZv8kE0wLOQ2Q24TBJpj+Buj+Boj+BrD5DaEsBqasOqboL4BrasnttfxnLSxsuP6AOL3t54HSvKIJJhPKD9ItNXQnBO284MtQ3MdvKLZcGDTXU/kRuCzqvpRPEJEFgL3ishFQH184+FgpSceQ1/RGyl4jcWwLYvmmhoiVVWgSnjnTsK7duGNRmnasAENR6CmhqadOwkPGYKGQkQqKxGPx1k5Zll4CgsTQ0t4vJx39dV85ktf4pl//IP/c+mlCSl48XjxjBjrVCJQwLU/uq5FNkVjEI1QvukjxPZAQamzMMATQCyBcCM017RZGKDAV44/jC98/Vss/fa5RCyb9z7cwIVLLuMPj/6Oek8NNVVVxFyDsaVqK83RZj6s/DDtd1PZWMlVb1/VJt5n+VoNxcQb+pG5I1v11lOHbNob2omHPVbfv9K420Nzbji8a1fL0FxDQ6f3Eb+/lQGKz/20N0fkGToUu6gYT3ERVn6+mbjej+nuX7kv2YAAqOprIvIl4Ckc7+SQTFRMRALAS4DfreejqvqTlDR+4EFgHs4u+a+oankm7p9txKXgVZXf3X8/qBKtrXN6o83NhLZvJ1JVRaSpiaZ165wlsKEQ4a1bCcdiTvp9+5yJbRGsvDzE68XKy8NTVMQhxx5L+dVX82luLhMnTuSvN9yAlZODd1Tr+YVNmzYxecqU7knB79jpDKn5ctmydRsvv7ue+Qvm8/snnuOwzx7FnoISIrEIkViYaDTMR2UfUTppLFGN8dC/XmTM1ImUE2bnli0s+fql/OzXP2P4+BJCoXo8qnhx3p8cDIewVSmORrHcOOcjWEAkBn+uVnLFJlc85FheciwPHtsHIY+z98ZWsJrBioHdDFadG+9tfUyE4/GepHAX07a53pW0rZds93RoLhmNRIjV1bmGJ/OGSHw+7KIiPO7HLi7C4xoYJ74lbA8Zkr1LtBNL8iPOZuJYpOU8cYw4naF4OHEtNV9KfJt8KeW2ypdSfpfyufc6/f9BcHjnz9oNumtEmkRkuKruTo5U1Y0iEsWZK8kUzcCxqlrnbmB8RUT+oaqvJ6W5CNinqlNE5GzgZuArGaxDv6KqTuPvrlaK1tYS/nQXGglz1de+xje+9z1u+PGPOemzn0UjEUKby4ns2UMsFCJWU+v8cdk21pAhiMfjTDKXluKfPBksi8DBB+OrqMDKycHnDnFZgQBWTg55hYXcddddnHzyyd2Sgr/22msZMmwICxYt4OCZB3P8icez7KZlfOHML3D4kYejKLl5udx6962oKBOnTOTm22/mwyUfMvmgyXx72bfZVb8LEcFjefCIh9t++ks++egTLMti7Lix3H7n7YwtGMttd9xGbVUtt/3wVlDF47FZ9dI/AeWcCy5m5cuvsqdyL/MOO4nrr/0uF513jvOPj4IqPk8NM8Yd1bJKLRZpvXQ6FnW8omi45Z8ykTZ55Vv8Hzbc/obRPkM6MF7JBsduveLOspOGCz2tNrSKZWNbXuxUQzbUC0XxtHlgD23X6KlaxJojRBvDxBpCRKrriO6rIrKviui+aiL7qolU1RLeUkbTB6uJVNdDNM3QoG3hyc/BLgjgyY9//NhBH56g1z168OR5sHMshFj7DWaHDW1y491evmjrxruzocz+Jv67xjsXYrf8JpaddM3jXrPbX53ZC7q1xFdEvgx8BzhLVXckxRcD/1LVORmvoVN+LvAKcJmqvpEU/xywzPWGPMCnwHDt4KEGaomvxmKJJaytlriGI2ikZYkrqVUXcQ2CF/F6k4aWPO6517megd5bXAo+Goty+eWXM2nyJC674jKisSgRjbjeQoSoRluF2/u6bctOGAbbstm5dSfnn3k+/179bzzica5ZHmyxsfthU2Sf/c7xhqc9I9PKIEW6lzYRF24xeGmvd5S2vfIHzjiqQjQkRJtsIk0WkSardbi5dbzG0gyHiWIHBE8OeHIFO9fCk2vjybOx82w8eV7X8HjxBP2I19O1hjZxLenYKp+VFPa4ckae7uWTdu7Txhi0l6//vbWMLPFV1cfcIaTXRORt4D3AB/wHznxJRhERG3gbR2Ll18kGxGUMsNWtW0REqoEiYE9KORcDF4PzdrxMk/AeUvY8kLz/Id0raC3LMQDusJJjKFzj4IbxeHo9nhxf5heJRYhoxDEKbjjZKNx757389U9/JRwKM+OQGXzj2m+wpWZLohwRSTT+XstLwBNInNuW3SacWu9obhTbsikMFKZWcXATb4w6UicYjKQax9R9SMlqC6nGCVIa2tYNtFgePG4nw99hA22jIsRqa4nsqSRauYdIZSWRPZVEKvcQ3VPpnFfuIbSnksiWSrQp/fJre+hQdyitoO2w2rDWw2qWWUjQZXq02dD1DM4GZgE1wNOq+laG65Z8v6HA48C3kyfuRWQNcJKqbnPPPwaOVNU9aQui555ItL4ebW5uszGOcBiNtXVzxbYdjyHuKSQf3Q+W1WMDEdNYwkNoYxTSeA4degsdGIJkbyG+cWmwkk2bSg19g6oSq29IMjZ7iCYbnUS4kuiePcTamcuxCgqS5nCKnXBiDqd1vJXTs5c8DTYyutlQVRuA+3tdq67fr0pEXgBOokUxGGA7MBbY5g5nDcGZYM84kYoKYvX1AAkvwfL7kWDQWc2UPLTk9XZ7eCnhLXRgCBLhWLTdTWRxbyE+lBT3FpKHljryFgyGwYyIYAfzsIN5+MaP7zR9rLExYVDiHk50b2tD07xhA/WVlcRqatKWYeXmpjE0SQsH3Gt2UTFWXu5+9z/XJSPiSrrXi0hQVev6ulLuPYcDYdeA5ACfx5k4T+ZJ4HzgNeBMYEVH8yG9wTtmTMv8RBf/CLrqLcTDnXkLtmV3ahQGu7dgMPQniUUmKXup0hELhdrxalqG1Zo/+YToW6uIVlWlLUMCgdZDaSnDanFjM5iWRnfVExkmIhcCHwHP9mF9khkF/N6dF7GAR1T1KRG5AVilqk8C9wF/EJGPgL04Q2x9QiTmeBZClJhEWzyDDjyHaKyL3oIdaGMUkiecLfN+EYNhwLF8PqxRo/COGtVpWg2Hiezd12oOJ1q5p8XDqdxDeNs2Gt97j+i+fc7KyhTE623xcNIujW4ZVhvIpdFdNSLHARcA94tIiapW9F2VHFT1fWBumvj/TAo3AWf1dV0A9lbWYEWcr0uBmESIWVGiEiEqUbBjYCmWxyZgJ3kHxlswGA44xOvFO6IE74iSTtNqNEp03770cziu0QlXVND04YdE9u6FaJrOqceDZ9iwjo1OcTH+iRMzrj7QVSPyJrAEGNsfBiQb8QyBaLgZS23smA0xL0S97jJydV7PlYRlW9gewbItLI8gHgs8FuoBbCDJhlRWVnYsBd/Bj75q1SoefPBB7rjjjnbTACxatIhXX321J4/eKZmUggdnd/4TTzyBZVmUlJTwwAMPMHr0aABWrlzJVVddRTgcpri4mBdfdKTblixZwlNPPUVJSUmnasIGQzYhtu3MmxQXw7RpHabVWIxodbU7h7O3/WG1jz8mumePs/gnicnLn8OX4RWqvZKCH4z0xT4RVSUaiRGLKNFojGhEiUViRCMxolEnnIpli2toXGPjsbBt4aaf/5T8gvxWooRdloIfILoqBd9VampqKCgoAOCOO+7gww8/5O6776aqqopFixbx7LPPMm7cOCoqKigpcXp6L730EsFgkPPOO6/depjVWYYDCVVtszQ6eMwxWP6eLUVvb3VWtwbR3P0WhhREBI/XxpfjISfoIzjUT0FxDsNG5lE8JsjwcfkUjQkydEQuBUUB8ob48eV4sCwhEorSUBuibm8T1bsbaawNU1/VzNlnncuF513E/HmHc9UV3+GlF/7NgiMXMGfOXBYtXMSGDRsAp2d+2mmnAc67SJYsWcLRRx/NpEmTWnknQVcyfeXKlRx99NGceeaZTJ8+nXPPPTcxof/MM88wffp05s2bxxVXXJEoN5m1a9dyxBFHMGfOHGbPnk1ZWRk/+MEPHCn4OXMSxu/WW2/l8MMPZ/bs2fzkJ45aTXl5eeKeM2bM4Mwzz6QhzRLLuAEBqK+vTwz9Pfzww3zpS19K7PWJGxCAo446isLC/Wz/icHQC0QEu6AA/6SJ5B5+OAUnndRjA9IR3e3eXsr++N70JK4r28aaus7fFdEdZgVz+OnUUmxPeputqsSiSiwaw5/rwZfjwfZY7Ni5g2f+9i9ELWpqqvnrw8/g8Xh48ZUX+O5VS3nwvoep29dEJByjoaaZSCjKunXrWPH8Curq65g+fXpbKXjgnXfeaS0F/+9/O1Lwl1zSIgV/zjlp63r33Xdz5ZVXcu655xIKhYhGo/ziF79gzZo1vPvuuwAsX76csrIy3nzzTVSV008/nZdeeolx48axYcMG7rvvPhYvXsySJUu46667Wr2AK046KfiNGzcSDoc5+uijqa2t5corr+S8887rxS9jMGQvqorGNDGaEYuqM8oRjYfdY7RlFCQWjY+CKLGYG++mi0WVGYtG4cvJ7KhGd0szs8F9gIhgewTbY+Hx2fgCHrx+m69+7WxKxg5BVWnYvJfLrlzCR2VlIEI4FMbySOKPpm5fM80NEY7+zPHU7g4jEqCosJiNa8oZN85R122qDxMJRTn88MMZM2YMIsKcOXMoLy8nGAwyadIkJk6cCMA555zDPfe07S8sXLiQm266iW3btvGlL32JqVOntkmzfPlyli9fzty5zrqIuro6ysrKGDduHGPHjmXx4sUAfO1rX+OOO+5Ia0RuuukmbrrpJn7+858npOAjkQhvv/02zz//PI2NjSxcuJAFCxZw0EEHZey3MOyfdNQgJzeyqQ1zLD48ndJwt4TjQ9YpeVrdJ5boJLYuq3XaqJsmuaxMM/HQ4gE3Ivv9BMpPp3a+Xry/iEvBiwg/WfYTjjvuWJ544m8JKfihJbkUFOfgy/FQXBokkO8lLyeP4LAA0UgMj20TDodpagijCjV7Gqnd24Soh91b67BtIdKs1FY10FgbIhZVQk0RbI/V7p6Vr371qxx55JE8/fTTnHLKKQkp+GRUlWuvvbZFCt6lvLy8zaq0zlapnXvuuZxyyilcf/31lJaWUlRURF5eHnl5eRx11FG89957xohkARpTws1RQk0RQk1RouFY255yu41panznDXK6PMkGoVWePmqQk7HcRTS23TK/adlOx7Bl/tM5enx223j33HYX4tgpedorKxHfKk88Xds8vkDm51a7W+KhIpJu26YAqqoFaa4ZMkBcCh7ggQceaHPd+QN2/kBzC5zVXJbHYmhJLsPH5iMChaPyyBvqx+O1yM33Eo0oqkokFGN0yXg2bdrE+6vWMW7seB783R8JNUXY92m9O/nv/KFu3lrO5MmT+fa3v911Kfjt2xNDalu2bOG1115j4cKFPPzww3zmM59p8yxlZWUJD+eJJ55g+vTpAJxxxhl861vfIhKJEAqFeOONN7j66qsz+j0fSLQ0/E7jH045dhQfTrkebu76a3g7o+MGOd5wtjTI/tyBaZDj97EsOaCX7HdXgLHvpVYNaVm6dCnnn38+N954I6eeemqPyogPldkei+CwAACBPC/5hQHGTi3h17++k3MvOovc3FwOmzMP23bmcEJNUWJRZ6ngQw88zKOP/xmPx0tJyQi+edGVeDWXIw9fwMyDZ3LiiSdxyy238OE5H7Jw4ULAmdR/6KGHsG2badOm8etf/5olS5Zw8MEHc9lll7Wp5w9+8AM2bNiAZVmMHz+eu+++G4AZM2Zw0kknMXv2bCzL4hvf+AazZs0CnOG3lStXsmfPHkpLS7n++uu56KKLevQ9ZTMaU8KhKKHGKOHmCKHGKKHmCGH3mByf3NAnN/yJYxcbfst2erDegPP34wvYBII+CoptfAEbrxsXP/oCHmyvldQgpzbcaQyCaZD7nFg0ivRCr689Ol3iKyKjgENUdXmXC3UkSxYDf1dtR+RpgBgoKfjBQFwKXlW5/PLLmTp1aqKn3+EyZnfYIZXUZczbtm/hy//xRd57932n0bD6t7EYqN9Z1enxp+vJt23o2+n1N0YINbsNfxdGZixL8ObY+PwefDk23pSjz+9pfT3JQCQbA2/AGXoxDXt2oqqEmxqp27ePhqp91Ffvo37fXuqr9rX5NNRUc/FdvyO/sLhH9+qxAKOq7hSRqIh8F+fP91VgtaqGUm5QAhwJHAGsUdU/96imhgHj3nvv5fe//z2hUIi5c+e2mtOIL2PGmz5vfIVZ1DUsyWPgkVCU5oYYdVXNRKPK3p2ukKUlLftkUvfMDICRSX2esNtohxojiWO6oZz0xiDp2MWGXyxxG/GkHn+uh/zCQKtGPXHswEiYhn9wE4tGaaipdg2AaxT2ucagah91ieNeIs3NbfJbtoe8ocPIGzqU/OLhjJoyjdyhw7A97fwD94LuvpRqJHAycDRQiGOEFAgB5cBzwPOpBiabMJ7IwJBsZBJH14uJT3ym/i2KlTSu3WpTphOWlKEPVUXVGfJxVuO0rMrRGGwoW0+0oqClV+8ek41E8rEr/xrpGv50wzsd9fTj8bbXNPz7O6GmxvSewr7WXkRjTQ2a5k2K/rw88oYMI29YoWsk3M+wQid+6FDyhhUSyAtmXEurV1LwIlKoqntV9VPgd+6nvbQ5InKUqr7U8+r2P6pq/oH7kORlzOlQVWIxbfFgkgxMNBIj3NT2nSgizvi6aouxaA9VpakuwqrHP0aENA29TXCoH2+OB5/f2TjqTTm2iQ/YeEzDf8ATi0VprKlxjUGKgdi31zEOrqEINze1yW/ZNrlDh5E3ZBj5xcMZOeUg1zgUOkZhaIvB8GThy7I6NSIiUgRUuO9W39uFMscBL+AoRA0KAoEAlZWVFBUVmQZhgBBxVsTYNnj9bf90Euv8k9bux70aEccjEMs1LK6HIpYTD7Cvai9FIwq4+PbZeHym4Td0TripibqUoaT40FLykFJDdXV6ryE3L9H4j5g0leCwYeQOGUZwWKFjNNxPTjA/416DRhVtjhBrihJriqBNUWLNEQJThiHezN6rq6uzevUfJyK+bB7iKi0tZdu2bezevXugq2LoIwKBAOMnjsPrHTR9G0Mf0MprSPUWWhmKKsJpXrNr2Ta5Q4aSN3QY+YVFjJw0JWkoaRh5w5xj7tBheH3dlxhRVTQccxr9pgja7BwThiAlvuV6FE06aritUQMYufRwPIWBbterI7pqRNqME4jIBcB3gC+o6uZO8l8PXOvmW6yq/+5OJfsar9eb2KltMBgGH+GmpiTDkGZIqaqK+up9NFRXpX2dtS8n151nGMqIiVPSzzl04jW06v3vCdPc1ESsOZIwCLHm1g19zPUOWhuMKHQwLBtHfDZWwEYCHqyAjZXjwRrqxwp4kICN5Y9f87jpbCy/Bzs/88NhPdq+KCI/Bm4AGoA3ROQLnbxj/TlXvLEemAFklRExGAzZh8ZiNNbWULdvb2L4KNlTaEgyGKHGtl6DWBZ5Q4aSO3QYwcJCSiZOdoaUEsbBMQy5Q4bgsXzt9/5rosQqIoSaK2lq2tXt3n8rbHEadb8ncbSH+vEG8pyGPuBpfT1uJAIexN9ylAFcuZhKt4yIOAPJ/wNcDPwSuAX4O7BSRL6mqo+nyXMfUI3zgqnXVfXaXtfaYDAMWsLNTY5nkDLpnOpFtO815CQMwPDxk5l8SBF5eYXk5Q0lxx8k4A/i9+TgES8a0ta9/8Yosap477+WhuYq6vq4959sHNgPl153x4jkAg8ApwJLVfU2ABH5HPC/wF9EZCnwdHImVb3IfUf6YcDhIvIbVW0tqpQGERkLPAiMwBlOu0dVb09JczTwBPCJG/VXVb2hG89kMAxK1FmS5i5rjrnLmmMtS5oT4VhLulgsJX3rvLRK0356Yo5CbPweJOdx0zY31LderZQ05xBqdOT/bfHitXx4LT8+K0CwoIhg3jCKc0eRO6yAgC+IzxvAZwfw4MNWG4laEIq19P53xWBX8jfTTIRmIslR+2HvP5vozsT6M8A04DxV/WP8gqo2isgXgV8BtwLHpmZW1UacIazuDGNFgO+q6moRyQfeFpF/quqHKeleVtW2L77IMH//5S/YvuFDZ4WBCIggCEhcRFBwDm3jnTj3mtsLEWkd35XyWsLuNdzyhDbxHd0r0RNKhNsvj3gZndS/5Vm7+H30uv4t5ac2mk4Dm76hTG5ku9xopkvXJm3bBpz2yu5qupSGu039BxALC68VcAyAHcAXD1v+lnjLj9+by1DfOHyeaXiDAbz5Xmz1IFFBtJ1GudH90Lb3L3kerEL7gO/9ZxPd8UTGA6eq6r9SL6izgP9KEdkE/FcmKqaqO4GdbrhWRNYBY4BUI9IvjJo6DX9urrsBzd0Y1yqs7mnLprnUeCfs5HPSaEp56fOllpfIl1ReTGNp4t17Ee+1tq57S5qO7kVL/dPk7UmZnX4fSfHJz9LevRw9IMvdfGi5y3tbwrh6QU58PGwlNivGw0jn6Ui5h2V5EK+Vkiclf0/q0Oo+bfO2SWelpm1btjhroJ2wWlhRQaIWVkSwIk5YwmBFBCIgYUHCIGEgDBJyXwMdUqQzMSOL1o27P6l3b3r/+xVdMSJNwG+Bu1V1dUcJVfV2ESnHmXTPGCIyAWdO5Y00lxeKyHvADuB7qro2Tf6LceZxEm/F6y7zT/tij/IZDH2BxhQNRYk1RhKfxJh/k3veGEmEY03ueWPL5HEHpYOF07jnuIZgqMeZAwh4kBzXCOQ4H4mHAzZWjhcJ2AnDatj/yfp3rItIEHgRuElV/5pyrQCIqWqdiJwC3K6qbd+SlEQ62RODob9J7AdobGnknWMUbQg7yz+T4hMGwY3XpkinelyJHn/axt7TYiDix9x4Ghvx2cYIGFrRK9mTgUJEvMBjwB9TDQiAqtYkhZ8RkbtEpFhV9/RnPQ0HJhqOterpt+79R5N6/+mvd7YiSHyW2/N3Gnq7wI93RNwg2C2NfxuD4AwPmaEgQ3+QtUbEXU58H7BOVf+7nTQjgV2qqiJyBGABlf1YTcMgRqPuKp823kDroZ/Ww0VJRiDSyeS2R1o19FauF6soJynObusNJAyCjdiZlacwGPqCrDUiOO8j+TrwgYi868b9EEebC1W9GzgTuExEIjjrOc7WbB+f209JfO2p376mBDq53ubX6yx/TFsN/bTp+TclzRukeAMa6sQIWILljv/HvQHvEH/roaGU68kGIdMaRQZDNpK1RkRVX6ETzS5VvRO4sz/qs/fPG2jenPRm4G42mqnRXW1UU68nGtlMldPD/F15P8aAIzirf3JbVv54inNSGnu7xQAkJo5dj8BMDhsMnZK1RiTb8JTkto1MbV9SG5zU08T+jAyV0938khKRWp12rnf1Pm3a227Wo222zurh7DFJDA0lTSJbOR5nctjMCxgMfYoxIl2k4JixA10Fg8FgyDrMoK3BYDAYeowxIgaDwWDoMWY4q4ts2bKFxsbGhPyE5cpLWEmSFKlxHV3ranqDwWDIZowR6SIrVqygvLy83++bSYPUX+kH4t6pHyBtfE/TGYNuMKTHGJEu8oUvfIHm5mZXAlvbHNPFdXRtoNNHIpGM3/tA2KLTHYPTF0ZsINN2p0yv10tOTg45OTnk5uYmwoFAwBjk/QxjRLpIUVHRQFch62l5p0TfGsjUa6n37ugz0GkzVWbydzCQde0uIpIwKKkGJjmceu7z+YzxyVKMETFkjOQeq23bA1wbQ3+QzuDEYjHC4TANDQ00NjYmPunOa2pq2LVrFw0NDYTD4XbvY9t2uwamo3Ov19tfX8UBizEiBoOhx7Q3X+Tz+cjLy+tWWZFIpEODkxzeu3dvIhyNti9r7/F4umxw4uFAIIDHY5rGrmK+KYPBkBV4PB7y8/PJz8/vVr5QKNShwUm+tnv37kQ4PhSaDp/P1yWDk3wtJycHyzrwdk0YI2IwGAY1Pp8Pn8/HkCFDupxHVQmFQh0anOTzqqoqGhsbaWpq6nAuKBAIdHu+x+/3D2rjY4yIwWA44BAR/H4/fr+fYcOGdTlfLBajubm50+G2+LGyspKGhgaam5s7rEtP5nuyZbGBMSIGg8HQRSzLSjTm3SEajdLU1NTpcFvyYoPGxkZCoVCHdenufE8wGMy412OMiMFgMPQxtm2Tl5fXq8UGnRmgffv2sWPHDhobG4lEImnLu+KKKygsLMzEIyUwRsRgMBiylJ4uNkhdYh0PB4PBzNcx4yVmCBEZCzwIjMB5BdI9qnp7ShoBbgdOARqAC1R1dX/X1WAwGLIJr9fLkCFDurXYoKdkrREBIsB3VXW1iOQDb4vIP1X1w6Q0JwNT3c+RwP+4R4PBYDD0A1m7rkxVd8a9ClWtBdYBY1KSnQE8qA6vA0NFZFQ/V9VgMBgOWLLZE0kgIhOAucAbKZfGAFuTzre5cTv7p2YGg8HQe1RjxGLNxGJNRKONRKNNxGKNRGNNxKJOnHOtiWiskVi0kWis2T02OmliTe65mycWL6cl/4IjnyUQGJ3Rume9ERGRIPAYcJWq1vSwjIuBiwHGjRuXwdoZDIb9mVgs4jTCSQ1zLJrUOKc24PE0boOfaORjzU5DHm/8Y41JxsL59AQRH7YdwLZysGy/e8zBsvz4fMXueQDbCrjxgQx/Q1luRETEi2NA/qiqf02TZDuQ/PLzUjeuFap6D3APwPz583ukV75mzZXU1L6PbQfx2HnYnjw8drDlmIjLw/YEE3EeT557DGLbQSzLnxUbhAyGwYoj9BgiGk1qjFs18qm98tRjci+9bW89OV61fVHIjrCsHGw7gGUFsN3G27YD2HYuXl8RluXHtnPaNP5OY+8YhXj+eHxyOfHyRQZe6DRrjYi78uo+YJ2q/nc7yZ4EviUif8KZUK9W1T4ZysrPPxhEiEbqiUTrCYV20xApJxqtIxKpJxZr7FI5IrZjcJIMi6eVAcpz44Kt4lqMUvIxNyv+iAwGANWo2wA3pzTGLT3xjoZeUo1A20a9pRxoX/eqPUTs9I27FcDjKcDvH+E05pbfabhTGvXkHn3rRr11436gdRSz1ogAi4GvAx+IyLtu3A+BcQCqejfwDM7y3o9wlvhe2FeVGT/+kg6vO/9ADUQidUSj9S3HaF3C8DjHOqJt4uppDu1K5IlG61FtX5k0GcvKSTIseWk9pbaGKr2nZFn+THxVhgHGkWMPucMkyQ16s9MIx+MTQzLNSefx603Eom3TJ9K6ccnl9rzX7nMb87YNtcc3vKUhT2r0bbtl2MbpteckeveJHr2bLn5NxHtANe79RdYaEVV9BejwF1dHCe3y/qlRx4jYeDz5eDzd2xSUDqcRaHa9nLhRqm9llNo1VNF6mpo/JdpQl8jT1fFWEW9rw5JscNqJ87TyqlqOjpeUtYv/+o34mLrzCSWNgTcneuHxBr/dxrpV4x5q6cknrje3bvxj7es0dQXL8jvDKJbfbcD9iTiPnYvlK3TP/YkGPn7djse3GXpJauStHGzbn1VDMoaek7VG5EBGRNzx0wA+X3Gvy4vFIkSjDa2NkusFpYtzvCXHSEUiNTQ17XS8J9eYdXUooWXYLtkbCrYZoms9v5TeU7IsX6++A2f1S1JjHU3qTSf1ylsa95ZGvKVxb9tYJ/fEU3v8sVhTlz3KdIh4khpzv9tYt5x7fMGWhtzyu419oE3jbicZhURPP977TzIAlhtveuuG7mCMyAGAZXmwrAK83oJel+V4SY2ul1OXZIxco+Mam5bhuroko1RPuGl7q2G9WKx9gblkRHwpw3aOwbHsXDQWSmncW/f443E9R1p62nZSA+w2zB5PPpZV3H7jbftbpY/37G0rySgkGvOkBt8y/56G7Mf8lRq6heMl5WLbucDwXpcXi4WThuuSPKBkTynZQ0rylMLhKqLNO7HEl2iYPZ6CNj3zlsbcMQCJxttO6sW36qUn9ehtPyKmd24wtIcxIoYBxbK8WNZQvN6hA10Vg8HQA8zMp8FgMBh6jDEiBoPBYOgxZjiri5TVNxFSJceyyLHFPVr4RMx4ucFgOGAxRqSLXLtxG69U1bWJt4Ac2yLHsggkGZdc95jT6igE0sZb5FjSYbzXGCuDwZCFGCPSRa6dNIpdoTCN0RiNMXWPsTRHTZxXR6J8Gg3TGIvRlBQf0u7Ld9lCwrAELKuNR5SbZHgClrQ1RqmGKW28hdcyhspgMHQdY0S6yLwh3Xs3ckdEYkpTzDE6DUnGpymtUWptmBpjbeOrwlF2NocT15vc8sI9MFYe11gF0nhRyee5cWPWoRfV4p3lpsR7jLEy7AfE1Pk/i8ScY1iViCrh5POYElYS8RE3Pp4mkq6MpHByng7LiJE2XSipvL/NncLoQO827qZijMgA4LGEoGUTpG/lHsKx1oapoZWBUprSelPte1l7w1Eak4xV/BjtgS6yV6RleK+dob9EuL1428JvWVg4+jgiYCGJMPF4xL0WT+emcT+Ik8YSWsW3SuemaX3eEmcl3y+lfCulHFLypb2fW1jy8yTfL/E8g3SIs73GNxRLbVRpt0HsaePbpow0jW9qeenuFY5pD2Qge4ZXBI8IXgvnmDh3wl4RPElhnwh5XguPCD5LEnn6YqTBGJH9GK8leC2bfE/fG6u23lOMhmSDlRTf3vBf/LgnHKaxqW18f/3DDkbSGbdkA4VrSFsMW1uj6KZKMW6t80Fbo5hq3ASIKW0a31BSIz2QjW+84fV00PjG06U22B2XQat0vqSyk/OlLaOde/lEsCW7OwvGiBh6TdxYFfShsVK3B9mU4i01ucZF3TQtYVAUTeR34mNoIqxJ+TRNPk3Kpzi9Z9rJF0vJR6t7qHu9bT7cclPv1dV6xZ+n+/VKqVNSPhJ5030PaeqanM7NZ7fTIKY2qqmNb2qvOrXxbVNGmsY3Xka2N777C8aIGAYFIoJfBL9lMWSgK2MwGBKYzYYGg8Fg6DHGiBgMBoOhxxgjYjAYDIYek7VGRETuF5EKEVnTzvWjRaRaRN51P//Z33U0GAyGA51snlh/ALgTeLCDNC+r6mn9Ux2DwWAwpJK1noiqvgTsHeh6GAwGg6F9staIdJGFIvKeiPxDRGa2l0hELhaRVSKyavfu3f1ZP4PBYNivGcxGZDUwXlUPBf4f8Lf2EqrqPao6X1XnDx/e+1e6GgwGg8FBtAciff2FiEwAnlLVWV1IWw7MV9U9naTbDWzuYZWKgQ7LH0TsL8+yvzwHmGfJVvaXZ+ntc4xX1Ta98GyeWO8QERkJ7FJVFZEjcLyqys7ypfsSunHPVao6v6f5s4n95Vn2l+cA8yzZyv7yLH31HFlrRETkf4GjgWIR2Qb8BPACqOrdwJnAZSISARqBszWb3SqDwWDYD8laI6Kq53Ry/U6cJcAGg8FgGCAG88T6QHDPQFcgg+wvz7K/PAeYZ8lW9pdn6ZPnyOqJdYPBYDBkN8YTMRgMBkOPMUbEYDAYDD3GGJE0iMhJIrJBRD4SkR+kue4XkT+7199w97NkHV14jgtEZHeSiOU3BqKeXaELgpwiIne4z/q+iBzW33XsCvuTsKiIjBWRF0TkQxFZKyJXpkmT9b9LF59jUPwuIhIQkTddJY+1InJ9mjSZbb9U1XySPoANfAxMAnzAe8DBKWm+Cdzths8G/jzQ9e7hc1wA3DnQde3i8xwFHAasaef6KcA/cF7xvQB4Y6Dr3MPnOBpng+2A17ULzzIKOMwN5wMb0/yNZf3v0sXnGBS/i/s9B92wF3gDWJCSJqPtl/FE2nIE8JGqblLVEPAn4IyUNGcAv3fDjwLHSfa9zLkrzzFo0M4FOc8AHlSH14GhIjKqf2rXdbrwHIMGVd2pqqvdcC2wDhiTkizrf5cuPsegwP2e69xTr/tJXT2V0fbLGJG2jAG2Jp1vo+0fVCKNqkaAaqCoX2rXdbryHABfdocZHhWRsf1TtT6hq887GOiSsGg24Q6JzMXp+SYzqH6XDp4DBsnvIiK2iLwLVAD/VNV2f5NMtF/GiBzY/B2YoKqzgX/S0jsxDBxdFhbNFkQkCDwGXKWqNQNdn57SyXMMmt9FVaOqOgcoBY4QkU61B3uDMSJt2Q4k98hL3bi0aUTEAwyhC7pd/Uynz6Gqlara7J7+FpjXT3XrC7ryu2U9qloTH45Q1WcAr4gUD3C12kVEvDgN7x9V9a9pkgyK36Wz5xhsvwuAqlYBLwAnpVzKaPtljEhb3gKmishEEfHhTDw9mZLmSeB8N3wmsELdWaosotPnSBmbPh1nLHiw8iRwnrsaaAFQrao7B7pS3UVERsbHp6UbwqIDgVvP+4B1qvrf7STL+t+lK88xWH4XERkuIkPdcA7weWB9SrKMtl9Zq501UKhqRES+BTyHs8LpflVdKyI3AKtU9UmcP7g/iMhHOJOkZw9cjdPTxee4QkROByI4z3HBgFW4E6RzQc5ncFYCfQQ0ABcOTE07pgvPMZiERRcDXwc+cMfgAX4IjINB9bt05TkGy+8yCvi9iNg4hu4RVX2qL9svI3tiMBgMhh5jhrMMBoPB0GOMETEYDAZDjzFGxGAwGAw9xhgRg8FgMPQYY0QMBoPB0GOMETEYOkFErhCRdSLyRxE5XVxFZBH5PyJycFK6C0RkdNL5b5Ov90G9rhKR3L4q32DoCmaJr8HQCSKyHjheVbelxD+Ao+z6qHu+Evieqq7qp3qVA/NVdU9/3M9gSIfxRAyGDhCRu3Hk9P8hIle73sadIrIIZ5f/re77Jb4PzAf+6J7niMhKEZnvllMnIje5An6vi8gIN36ye/6BiNwoInVp6pAnIk+7edeIyFdE5ApgNPCCiLzgpjtBRF4TkdUi8hdXCwoRKReRW9x7vCkiU/rjuzMcGBgjYjB0gKpeCuwAjlHVXybFv4ojH3GNqs5R1ZuBVcC57nljSlF5wOuugN9LwP91428HblfVQ3AUbtNxErBDVQ9V1VnAs6p6R1K9jnF1nH6M4zEd5tblO0llVLv3uBP4Vc++DYOhLcaIGAz9Qwh4yg2/DUxwwwuBv7jhh9vJ+wHweRG5WUQ+q6rVadIsAA4G/u1Kd5wPjE+6/r9Jx4U9eQCDIR1GO8tg6B/CSVpLUbrxv6eqG8V5rewpwI0i8ryq3pCSTHDeHXFOe8W0EzYYeoXxRAyGnlOL8zrV9s67wuvAl91wWiE8d8VXg6o+BNyK83rd1Pu9DiyOz3e48ygHJRXzlaTja92so8HQLsYTMRh6zp+Ae91J7jOBB4C7RaSRrg8ZXQU8JCI/Ap7FectcKofgTODHgDBwmRt/D/CsiOxw50UuAP5XRPzu9R/jvC8cYJiIvA80A+15KwZDtzFLfA2GAcTd59GoqioiZwPnqOoZGb5HOWYpsKGPMJ6IwTCwzAPudF94VAUsGdjqGAzdw3giBoPBYOgxZmLdYDAYDD3GGBGDwWAw9BhjRAwGg8HQY4wRMRgMBkOPMUbEYDAYDD3m/wMNoWYJwSWEfwAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "from tqdm.notebook import tqdm\n",
    "\n",
    "from pbo.sample_collection.dataloader import SampleDataLoader\n",
    "from pbo.weights_collection.dataloader import WeightsDataLoader\n",
    "from pbo.networks.learnable_pbo import LinearPBO\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_linear = LinearPBO(\n",
    "    q=q,\n",
    "    max_bellman_iterations=max_bellman_iterations,\n",
    "    add_infinity=add_infinity,\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",
    "l2_losses = np.ones((training_steps, fitting_steps)) * np.nan\n",
    "\n",
    "for training_step in tqdm(range(training_steps)):\n",
    "    params_target = pbo_linear.params\n",
    "\n",
    "    for fitting_step in range(fitting_steps):\n",
    "        cumulative_l2_loss = 0\n",
    "\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_linear.params, pbo_linear.optimizer_state, l2_loss = pbo_linear.learn_on_batch(\n",
    "                    pbo_linear.params, params_target, pbo_linear.optimizer_state, batch_weights, batch_samples, importance_iteration\n",
    "                )\n",
    "                cumulative_l2_loss += l2_loss\n",
    "\n",
    "        l2_losses[training_step, fitting_step] = cumulative_l2_loss\n",
    "\n",
    "q_functions = np.zeros((max_bellman_iterations_validation + 2, n_states, n_actions))\n",
    "bellman_iteration_functions = np.zeros((max_bellman_iterations_validation + 2, n_states, n_actions))\n",
    "v_functions = np.zeros((max_bellman_iterations_validation + 2, n_states))\n",
    "\n",
    "batch_iterated_weights = validation_initial_weight.reshape((1, -1))\n",
    "for bellman_iteration in range(max_bellman_iterations_validation + 1):\n",
    "    q_i = env.discretize(q, batch_iterated_weights[0], 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",
    "    batch_iterated_weights = pbo_linear(pbo_linear.params, batch_iterated_weights)\n",
    "\n",
    "q_pbo_fixed_point = env.discretize(q, pbo_linear.fixed_point(pbo_linear.params), states, actions)\n",
    "policy_fixed_point = q_pbo_fixed_point.argmax(axis=1)\n",
    "\n",
    "q_functions[max_bellman_iterations_validation + 1] = q_pbo_fixed_point\n",
    "bellman_iteration_functions[max_bellman_iterations_validation + 1] = env.apply_bellman_operator(q_pbo_fixed_point)\n",
    "v_functions[max_bellman_iterations_validation + 1] = env.value_function(policy_fixed_point)\n",
    "\n",
    "for training_step in range(0, training_steps, max(training_steps // 10, 1)):\n",
    "    plt.plot(l2_losses[training_step], label=f\"Training step {training_step + 1}\")\n",
    "\n",
    "plt.legend()\n",
    "plt.xlabel(\"fitting step\")\n",
    "plt.ylabel(r\"$\\sum_{k} \\left(\\Gamma^*Q_{\\Gamma_{\\phi_i}^k(w)} - Q_{\\Gamma_{\\phi_{i + 1}}^{k + 1}(w)} \\right)^2$\")\n",
    "plt.title(\"Training losses\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Save data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {
    "execution": {
     "iopub.execute_input": "2022-09-19T13:03:03.714743Z",
     "iopub.status.busy": "2022-09-19T13:03:03.714355Z",
     "iopub.status.idle": "2022-09-19T13:03:03.769037Z",
     "shell.execute_reply": "2022-09-19T13:03:03.768510Z"
    }
   },
   "outputs": [],
   "source": [
    "np.save(f\"figures/data/PBO_linear/{max_bellman_iterations}_Q_{n_weights}_{seed}.npy\", q_functions)\n",
    "np.save(f\"figures/data/PBO_linear/{max_bellman_iterations}_BI_{n_weights}_{seed}.npy\", bellman_iteration_functions)\n",
    "np.save(f\"figures/data/PBO_linear/{max_bellman_iterations}_V_{n_weights}_{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": {
     "13b63b4fd53b4133bc95f88dffd19735": {
      "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_30516b042ea94a978166c2153454c4e5",
        "IPY_MODEL_db78b00eaae446afafe6d90ba7958c58",
        "IPY_MODEL_a492fbc0f8444653b87e4af047c96977"
       ],
       "layout": "IPY_MODEL_413271dc328f4af78a9f24478b9290bc"
      }
     },
     "30516b042ea94a978166c2153454c4e5": {
      "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_ba26b712e6cd4594afbc7e735e7db015",
       "placeholder": "​",
       "style": "IPY_MODEL_76b5a738a3ca408bb0760412b9b9e7b6",
       "value": "100%"
      }
     },
     "3d6b3cab1ff34fedb17a4efe7d2877a4": {
      "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
      }
     },
     "413271dc328f4af78a9f24478b9290bc": {
      "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
      }
     },
     "4ea24fa6190441b29064480d78a293f4": {
      "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": ""
      }
     },
     "76b5a738a3ca408bb0760412b9b9e7b6": {
      "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": ""
      }
     },
     "859be453b9694987b4ae0b92b5c368dc": {
      "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
      }
     },
     "a492fbc0f8444653b87e4af047c96977": {
      "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_859be453b9694987b4ae0b92b5c368dc",
       "placeholder": "​",
       "style": "IPY_MODEL_4ea24fa6190441b29064480d78a293f4",
       "value": " 400/400 [00:29&lt;00:00, 15.05it/s]"
      }
     },
     "ab992494579249f180c2830cead0abe1": {
      "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": ""
      }
     },
     "ba26b712e6cd4594afbc7e735e7db015": {
      "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
      }
     },
     "db78b00eaae446afafe6d90ba7958c58": {
      "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_3d6b3cab1ff34fedb17a4efe7d2877a4",
       "max": 400.0,
       "min": 0.0,
       "orientation": "horizontal",
       "style": "IPY_MODEL_ab992494579249f180c2830cead0abe1",
       "value": 400.0
      }
     }
    },
    "version_major": 2,
    "version_minor": 0
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
