{
  "cells": [
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "LM0eqWBsyQ-O"
      },
      "source": [
        "# Atomic Distributional Reinforcement Learning"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 1,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "mYCcPiaKyQ-R",
        "outputId": "6183cbcf-3ad2-4389-a7c0-5fb88930feb1"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "--2022-09-22 09:46:15--  https://raw.githubusercontent.com/senya-ashukha/quantile-regression-dqn-pytorch/master/logger.py\n",
            "Resolving raw.githubusercontent.com (raw.githubusercontent.com)... 185.199.111.133, 185.199.108.133, 185.199.109.133, ...\n",
            "Connecting to raw.githubusercontent.com (raw.githubusercontent.com)|185.199.111.133|:443... connected.\n",
            "HTTP request sent, awaiting response... 200 OK\n",
            "Length: 2770 (2.7K) [text/plain]\n",
            "Saving to: ‘logger.py.4’\n",
            "\n",
            "\rlogger.py.4           0%[                    ]       0  --.-KB/s               \rlogger.py.4         100%[===================>]   2.71K  --.-KB/s    in 0s      \n",
            "\n",
            "2022-09-22 09:46:15 (40.3 MB/s) - ‘logger.py.4’ saved [2770/2770]\n",
            "\n",
            "--2022-09-22 09:46:15--  https://raw.githubusercontent.com/senya-ashukha/quantile-regression-dqn-pytorch/master/rl_utils.py\n",
            "Resolving raw.githubusercontent.com (raw.githubusercontent.com)... 185.199.108.133, 185.199.109.133, 185.199.110.133, ...\n",
            "Connecting to raw.githubusercontent.com (raw.githubusercontent.com)|185.199.108.133|:443... connected.\n",
            "HTTP request sent, awaiting response... 200 OK\n",
            "Length: 1162 (1.1K) [text/plain]\n",
            "Saving to: ‘rl_utils.py.4’\n",
            "\n",
            "rl_utils.py.4       100%[===================>]   1.13K  --.-KB/s    in 0s      \n",
            "\n",
            "2022-09-22 09:46:15 (93.4 MB/s) - ‘rl_utils.py.4’ saved [1162/1162]\n",
            "\n"
          ]
        }
      ],
      "source": [
        "!wget https://raw.githubusercontent.com/senya-ashukha/quantile-regression-dqn-pytorch/master/logger.py\n",
        "!wget https://raw.githubusercontent.com/senya-ashukha/quantile-regression-dqn-pytorch/master/rl_utils.py "
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "!pip install gym[box2d]"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "AQcxexSylxcz",
        "outputId": "e68da343-1a40-4762-bef2-b7aafa81ec4f"
      },
      "execution_count": 2,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Looking in indexes: https://pypi.org/simple, https://us-python.pkg.dev/colab-wheels/public/simple/\n",
            "Requirement already satisfied: gym[box2d] in /usr/local/lib/python3.7/dist-packages (0.25.2)\n",
            "Requirement already satisfied: cloudpickle>=1.2.0 in /usr/local/lib/python3.7/dist-packages (from gym[box2d]) (1.5.0)\n",
            "Requirement already satisfied: gym-notices>=0.0.4 in /usr/local/lib/python3.7/dist-packages (from gym[box2d]) (0.0.8)\n",
            "Requirement already satisfied: importlib-metadata>=4.8.0 in /usr/local/lib/python3.7/dist-packages (from gym[box2d]) (4.12.0)\n",
            "Requirement already satisfied: numpy>=1.18.0 in /usr/local/lib/python3.7/dist-packages (from gym[box2d]) (1.21.6)\n",
            "Requirement already satisfied: pygame==2.1.0 in /usr/local/lib/python3.7/dist-packages (from gym[box2d]) (2.1.0)\n",
            "Requirement already satisfied: swig==4.* in /usr/local/lib/python3.7/dist-packages (from gym[box2d]) (4.0.2)\n",
            "Requirement already satisfied: box2d-py==2.3.5 in /usr/local/lib/python3.7/dist-packages (from gym[box2d]) (2.3.5)\n",
            "Requirement already satisfied: typing-extensions>=3.6.4 in /usr/local/lib/python3.7/dist-packages (from importlib-metadata>=4.8.0->gym[box2d]) (4.1.1)\n",
            "Requirement already satisfied: zipp>=0.5 in /usr/local/lib/python3.7/dist-packages (from importlib-metadata>=4.8.0->gym[box2d]) (3.8.1)\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 3,
      "metadata": {
        "id": "7JuEc-DTyQ-S",
        "scrolled": false
      },
      "outputs": [],
      "source": [
        "import gym\n",
        "import torch\n",
        "import pickle\n",
        "import random\n",
        "import numpy as np\n",
        "import torch.nn as nn\n",
        "import torch.optim as optim\n",
        "\n",
        "from logger import Logger\n",
        "from rl_utils import ReplayMemory, huber"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 4,
      "metadata": {
        "id": "Ngt-yoqoj8Td",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "62c45424-a9d7-4ec3-f935-f75b822c3123"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Drive already mounted at /content/drive; to attempt to forcibly remount, call drive.mount(\"/content/drive\", force_remount=True).\n"
          ]
        }
      ],
      "source": [
        "from google.colab import drive\n",
        "drive.mount('/content/drive')"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 5,
      "metadata": {
        "id": "UwP0rCnyyQ-T"
      },
      "outputs": [],
      "source": [
        "class Network(nn.Module):\n",
        "    def __init__(self, len_state, num_avar, num_actions):\n",
        "        nn.Module.__init__(self)\n",
        "        \n",
        "        self.num_avar = num_avar\n",
        "        self.num_actions = num_actions\n",
        "        \n",
        "        self.layer1 = nn.Linear(len_state, 256)\n",
        "        self.layer2 = nn.Linear(256, num_actions*num_avar)  \n",
        "\n",
        "    def forward(self, x):\n",
        "        x = self.layer1(x)\n",
        "        x = torch.tanh(x)\n",
        "        x = self.layer2(x)\n",
        "        return x.view(-1, self.num_actions, self.num_avar)\n",
        "    \n",
        "    def select_action(self, state, eps):\n",
        "        if not isinstance(state, torch.Tensor): \n",
        "            state = torch.Tensor([state])    \n",
        "        action = torch.randint(0, 2, (1,))\n",
        "        if random.random() > eps:\n",
        "            action = self.forward(state).mean(2).max(1)[1]\n",
        "        return int(action)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 6,
      "metadata": {
        "id": "I_U-hBE1yQ-U"
      },
      "outputs": [],
      "source": [
        "eps_start, eps_end, eps_dec = 0.9, 0.1, 500 \n",
        "eps = lambda steps: eps_end + (eps_start - eps_end) * np.exp(-1. * steps / eps_dec)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 7,
      "metadata": {
        "id": "vAocoMGgyQ-U",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "bb150bf4-d365-4cc4-94c7-ac2842de1954"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/usr/local/lib/python3.7/dist-packages/gym/core.py:318: DeprecationWarning: \u001b[33mWARN: Initializing wrapper in old step API which returns one bool instead of two. It is recommended to set `new_step_api=True` to use new step API. This will be the default behaviour in future.\u001b[0m\n",
            "  \"Initializing wrapper in old step API which returns one bool instead of two. It is recommended to set `new_step_api=True` to use new step API. This will be the default behaviour in future.\"\n",
            "/usr/local/lib/python3.7/dist-packages/gym/wrappers/step_api_compatibility.py:40: DeprecationWarning: \u001b[33mWARN: Initializing environment in old step API which returns one bool instead of two. It is recommended to set `new_step_api=True` to use new step API. This will be the default behaviour in future.\u001b[0m\n",
            "  \"Initializing environment in old step API which returns one bool instead of two. It is recommended to set `new_step_api=True` to use new step API. This will be the default behaviour in future.\"\n"
          ]
        }
      ],
      "source": [
        "num_avar = 3\n",
        "#num_avar = 10\n",
        "\n",
        "#env_name = 'MountainCar-v0'\n",
        "#env_name = 'CartPole-v0'\n",
        "#env_name = 'CartPole-v1'\n",
        "#env_name = 'Acrobot-v1'\n",
        "env_name = 'LunarLander-v2'\n",
        "\n",
        "env = gym.make(env_name)\n",
        "\n",
        "memory = ReplayMemory(10000)\n",
        "logger = Logger('q-net', fmt={'loss': '.5f'})"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 8,
      "metadata": {
        "id": "a4imlInFyQ-V"
      },
      "outputs": [],
      "source": [
        "Z = Network(len_state=len(env.reset()), num_avar=num_avar, num_actions=env.action_space.n)\n",
        "Ztgt = Network(len_state=len(env.reset()), num_avar=num_avar, num_actions=env.action_space.n)\n",
        "optimizer = optim.Adam(Z.parameters(), 1e-3)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 9,
      "metadata": {
        "id": "cDPEQcEzyQ-W"
      },
      "outputs": [],
      "source": [
        "steps_done = 0\n",
        "running_reward = None\n",
        "gamma, batch_size = 0.99, 32 \n",
        "mixture_ratio = 0.8\n",
        "tau = torch.Tensor((2 * np.arange(Z.num_avar) + 1) / (2.0 * Z.num_avar)).view(1, -1)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 10,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "xL628xSuyQ-X",
        "outputId": "5ecf4b27-28e0-4411-b569-f2d0a7585d8b"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r  0%|          | 0/2001 [00:00<?, ?it/s]/usr/local/lib/python3.7/dist-packages/ipykernel_launcher.py:10: UserWarning: Creating a tensor from a list of numpy.ndarrays is extremely slow. Please consider converting the list to a single numpy.ndarray with numpy.array() before converting to a tensor. (Triggered internally at  ../torch/csrc/utils/tensor_new.cpp:201.)\n",
            "  # Remove the CWD from sys.path while we load stuff.\n",
            "100%|██████████| 2001/2001 [29:07<00:00,  1.14it/s]\n"
          ]
        }
      ],
      "source": [
        "import tqdm, sys\n",
        "sum_rewards = []\n",
        "\n",
        "for episode in tqdm.tqdm(range(2001)): \n",
        "    sum_reward = 0\n",
        "    state = env.reset()\n",
        "    while True:\n",
        "        steps_done += 1\n",
        "        \n",
        "        action = Z.select_action(torch.Tensor([state]), eps(steps_done))\n",
        "        next_state, reward, done, _ = env.step(action)\n",
        "\n",
        "        memory.push(state, action, next_state, reward, float(done))\n",
        "        sum_reward += reward\n",
        "        \n",
        "        if len(memory) < batch_size: break    \n",
        "        states, actions, rewards, next_states, dones = memory.sample(batch_size)\n",
        "\n",
        "        theta = Z(states)[np.arange(batch_size), actions]   # -->  Qi_(x,a)\n",
        "\n",
        "        # --------------------------------------------------------------------------\n",
        "        \n",
        "        theta_tgt = Ztgt(states)[np.arange(batch_size), actions]  # -->  Qi_tgt_(x,a)\n",
        "        Znext = Ztgt(next_states).detach()\n",
        "        #Znext_max = Znext[np.arange(batch_size), Znext.mean(2).max(1)[1]]  \n",
        "        Qnext_max = torch.max( torch.mean( Znext[np.arange(batch_size)] , dim=2 ) , dim=1)[0]\n",
        "        Qnext_max = torch.unsqueeze( Qnext_max, 1)\n",
        "        Ttheta_tgt = rewards + gamma * (1 - dones) * Qnext_max\n",
        "        atoms_tgt = torch.cat( (theta_tgt, Ttheta_tgt), dim=1)\n",
        "        atoms_tgt = atoms_tgt.detach().numpy()\n",
        "        # probabilities\n",
        "        probas = ( (1.0 - mixture_ratio) / num_avar ) * torch.ones_like( theta_tgt )\n",
        "        probas = torch.cat((probas, mixture_ratio*torch.ones_like(Qnext_max)), dim=1)\n",
        "        probas = probas.detach().numpy()\n",
        "        # sort atoms\n",
        "        sigma = np.argsort(atoms_tgt , axis=-1)\n",
        "        atoms_tgt = np.take_along_axis(atoms_tgt, sigma, axis=-1)\n",
        "        probas = np.take_along_axis(probas, sigma, axis=-1)       \n",
        "\n",
        "        i_window = np.arange( 1, num_avar + 1 ) /  num_avar   # avar integration segment\n",
        "        j_right = np.cumsum(probas, axis=-1)  # cumulative probabilities of the N+1 atoms\n",
        "        j_left = j_right - probas\n",
        "        \n",
        "        i_window = np.expand_dims( np.expand_dims( i_window, axis=1 ) , axis=0)\n",
        "        j_right = np.expand_dims( j_right, axis=1 )\n",
        "        j_left = np.expand_dims( j_left, axis=1 )\n",
        "\n",
        "        # compute avars\n",
        "        minij = np.minimum( i_window, j_right )\n",
        "        maxij = np.maximum( i_window - 1.0/ num_avar  , j_left )\n",
        "        lengths_inter = np.maximum( 0.0, minij - maxij )  # matrix of lengths of intersections of intervals [(i-1)/N, i/N] with [Pr_j-1, Pr_j]\n",
        "        dist_target = np.zeros( (batch_size, num_avar) )\n",
        "        for t in range(batch_size):\n",
        "          dist_target[t] = num_avar * np.dot(lengths_inter[t], atoms_tgt[t])\n",
        "\n",
        "        loss = torch.square( theta - torch.from_numpy(dist_target) )\n",
        "\n",
        "\n",
        "\n",
        "        #diff = Ttheta.t().unsqueeze(-1) - theta \n",
        "        #loss = huber(diff) * (tau - (diff.detach() < 0).float()).abs()\n",
        "        \n",
        "        # --------------------------------------------------------------------------\n",
        "        \n",
        "        loss = loss.mean()\n",
        "\n",
        "        optimizer.zero_grad()\n",
        "        loss.backward()\n",
        "        optimizer.step()\n",
        "        state = next_state\n",
        "        \n",
        "        if steps_done % 100 == 0:\n",
        "            Ztgt.load_state_dict(Z.state_dict())\n",
        "            \n",
        "        if done and episode % 50 == 0:\n",
        "            #logger.add(episode, steps=steps_done, running_reward=running_reward, loss=loss.data.numpy())\n",
        "            #logger.iter_info()\n",
        "            pass\n",
        "            \n",
        "        if done: \n",
        "            running_reward = sum_reward  if not running_reward else 0.2 * sum_reward + running_reward*0.8\n",
        "            sum_rewards.append(sum_reward)\n",
        "            break"
      ]
    },
    {
      "cell_type": "markdown",
      "metadata": {
        "id": "HoE8eQh3yQ-X"
      },
      "source": [
        "# Vizualization"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 11,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 312
        },
        "id": "sslXOgx4_yz-",
        "outputId": "b664aade-e43d-4c58-84eb-8c88cdb0016d"
      },
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "Text(0.5, 1.0, 'SAD-DQN')"
            ]
          },
          "metadata": {},
          "execution_count": 11
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 432x288 with 1 Axes>"
            ],
            "image/png": "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\n"
          },
          "metadata": {
            "needs_background": "light"
          }
        }
      ],
      "source": [
        "import matplotlib.pyplot as plt\n",
        "\n",
        "plt.plot(sum_rewards)\n",
        "plt.xlabel('Episodes')\n",
        "plt.ylabel('Rewards')\n",
        "plt.title('SAD-DQN')"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 12,
      "metadata": {
        "id": "Uc2uymzNcD3_"
      },
      "outputs": [],
      "source": [
        "def smooth(x,window_len=11,window='hanning'):\n",
        "    if window_len<3:\n",
        "        return x\n",
        "\n",
        "    s=np.r_[x[window_len-1:0:-1],x,x[-2:-window_len-1:-1]]\n",
        "    #print(len(s))\n",
        "    if window == 'flat': #moving average\n",
        "        w=np.ones(window_len,'d')\n",
        "    else:\n",
        "        w=eval('np.'+window+'(window_len)')\n",
        "\n",
        "    y=np.convolve(w/w.sum(),s,mode='valid')\n",
        "    return y"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 13,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 312
        },
        "id": "7OThowjycEKx",
        "outputId": "159701f9-99b5-45e9-b497-df1490eb76d7"
      },
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "Text(0.5, 1.0, 'SAD-DQN')"
            ]
          },
          "metadata": {},
          "execution_count": 13
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 432x288 with 1 Axes>"
            ],
            "image/png": "iVBORw0KGgoAAAANSUhEUgAAAY0AAAEWCAYAAACaBstRAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+WH4yJAAAgAElEQVR4nO3dd3xcd53v/9dH3ZZkdclV7nGNi6w4lRRCSIFdJ4GEUEO5v8DdZCm/u/eSLLDAsncX9t7AwlKWUEOAlCWBhBBSISTZVMl2XOMS27Jly5Jsdctqo8/9Y44cxbGtsa3RGY3ez8djHjPznZlzPscj663z/Z7zPebuiIiIxCIl7AJERGT0UGiIiEjMFBoiIhIzhYaIiMRMoSEiIjFTaIiISMwUGiIiEjOFhoiIxEyhITKImV1gZs+bWauZNZnZf5nZWYNezzGzDjP74zE+u8vMDptZu5m1BMv5lJkd9/+ZmV1sZv3BMjvMrNbM7hu8zuB9Zmb/08y2BevYbWb/bGYZg97zczNzM1s5qG2OmekMXhk2Cg2RgJlNAB4G/h0oBKYAXwW6B73tPcHzy8xs4jEW81fungtMB74OfB74yRCr3ufuOUAucA7wGvCsmV066D3fAW4CPhK870rgHcA9Ry2rCfinIdYncsoUGiJvOAPA3e9294i7H3b3x9193aD33Aj8B7AO+NDxFuTure7+EPA+4EYzWzzUyj2q1t3/Afgx8A0AM5sL/A3wQXd/wd373H0j0QB7l5ldNGgxdwJLjmoTGTYKDZE3bAUiZnanmV1pZgWDXzSz6cDFwK+C20eGWqC7vwzUAm87yVoeACrMLBu4FKgNljV42XuAF4F3DmruBP4Z+N8nuT6RmCg0RALu3gZcADjwI6DRzB4ys7LgLR8G1rn7JqLdQovMbHkMi95HtLvrZOwDDMgHioG647yvDig5qu2HQLmZXXmS6xQZkkJDZBB33+zuH3X3qcBiYDLwb8HLHyG6h4G77wX+QrS7aihTgCYzKx804N0Rw2ccaAEOAJOO875JweuDt6Eb+FpwExlWCg2R43D314CfA4vN7DxgLnCbme03s/3A2cAHzCzteMsIjoKaAjzn7rvdPWfgNsTqrwFWu/sh4E/AtMFHRQXLnkZ04PzpY3z+Z0T3Uq6NYVNFYqbQEAmY2Xwz+x9mNjV4Pg14P9FxgxuBJ4CFwLLgthgYR/RIpqOXNcHM3k20G+uX7r4+hvWbmU0xsy8D/w34ewB330p08P1XZnaOmaWa2SLgfuB54Mmjl+XufcCXiR69JTJsFBoib2gnuvfwkpkdIhoWG4D/AVwP/Lu77x902wncxZu7qH5vZu3AHuALwDeBjw2x3slBd1UH8ApwJnCxuz8+6D23ED2i6pdEB7s3ADXA1e7ef5zl3s3xx0JETonpyn0io4+ZfZVoF9aF7t4Sdj0ydig0REYpM7sF2O7uj4Zdi4wdCg0REYmZxjRERCRmxz1UMBkUFxf7jBkzwi5DRGRUqa6uPuDuR580CiR5aMyYMYOqqqqwyxARGVXMrOZ4r6l7SkREYqbQEBGRmCk0REQkZgoNERGJmUJDRERiptAQEZGYKTRERCRmSX2ehogMj67eCLXNndQc7KSxvZuO7j46eyIAZKSlkJ6aQv64dApzMijOzqQ4N4Oy3CxSUizkymW4KTRE5C3aunp5ZmsjL+44yOqaFrbUtxPpP7l56jLSUpheOJ7pRdnMKBrP7NIcFkyawLyyXMZlpMapcok3hYaIANDS2cPv19Xx+Mb9vLjjIL0RJzsjlWXl+fz3i2YzpzSH8qLxlE3IIiczjezgF39PpJ/u3n5aD/dy8FAPBzu6qW/vZk9TJ7sOHKLmYCfPbW+kqzd62Y8Ug5nF2Zw5JY+zZhZy9swiZpdkY6a9ktFAoSEyhkX6nee2H+C+qj08sbGenkg/s0qy+fgFM3nHgjIqygtIHaKLKS01hfEZUJCdwYzi7GO+p7/f2dtymI372thc18amujb+6/WD/G7tPgCKczI5d3YRl8wr4eJ5pRRmZwz7tsrwUGiIjEG7DhziN9W13L+6lrrWLgrGp/OBs8u5rnIqiybnDfv6UlKMaYXjmVY4nisWTwTA3dl1sJOXdhzkpZ1NPLvtAL9/dR9msHxaPm+fX8rb55exYFKu9kISSFJfT6OystI1YaFI1KHuPh5ZX8d/Vtfy8s4mUgwuOqOE6yqncemCUjLTwh1n6O931u9t5U+vNfDnLQ2sq20FYHJeFpcuKOPtC0o5d1YRWekaD4k3M6t298pjvqbQEEle7k5VTTP/WbWHP6yr41BPhFnF2by3cirXLp/KxLyssEs8rob2Lv78WgNPbW7g2W0HONwbYXxGKhfMKeYdC8q4eH4JpbmJW3/YIv0+ZNfi8SRkaJjZNOAXQBngwB3u/m0zKwTuBWYAu4Dr3b3Zovun3wauAjqBj7r76hOtQ6EhY9Wepk4eWL2XB9bUUnOwk+yMVN69ZDLXVU5lxfSCUdfd09Ub4YUdB3lqcz1PbW6grrULgDmlOaycWcjZMws5a0Yhk/PHhVxpeA50dFO1q5mXdh7k5Z1NlORm8vOPrTylZSVqaEwCJrn7ajPLBaqBq4GPAk3u/nUzuxUocPfPm9lVwN8SDY2zgW+7+9knWodCQ8aSxvZunthUz0Ov7uXFHU0AnDe7iPdUTOWKxRPJzkyOIUx3Z3NdO09vbeDlnU1U7Wqmo7sPgCn541gyNY+FkyawcPIEzijLZWJeFumpyXMec1+kn9rmw+w8eIjt9R2srW1h7e4W9rYcBiAzLYWK8gIunlfCJy+afUrrSMjQOJqZPQh8N7hd7O51QbA87e7zzOyHweO7g/dvGXjf8Zap0JBk1t0XYeO+Nl7Z2cQTm+qp3t2Me/Rw1muXT+GaiilMLRgfdplxF+l3Nte1RQOkpolN+9rYdbDzyOspBhMnZDGlYBzFOZkUZGdQlJ1BwfgMinKi94XZb9zCGjPp7ovQ2N79xq0jen/gyH0PBzq62ddymN7IG7+3p+SPY9m0fJZNy2d5eT5nTs077fGphA8NM5sBPAMsBna7e37QbkCzu+eb2cPA1939ueC1p4DPu3vVUcu6CbgJoLy8fEVNzXEvQCUyarg7+1q7WLO7mdU1LazZ08zGvW30RKLnPiycNIHLF03k8sVlzCvT0UbtXb28tr+dHY0d7G0+TG3LYfY2H+bgoR6aD/XQ3NnD8c5VHJ+ReiRQSnMzKZ2QRVluFmUTMimbkEVpcF84PmPIM97dnZbO3iMB0NDedSQUGgYFREN7N62He4+5jILx6RTnZFKSm0lxTiZTC8YxozibWcXZzCzOpign83T/ud7iRKER+v6qmeUA9wOfdfe2wT/s7u5mdlKp5u53AHdAdE9jOGsVGSldvRHW7219U0jUt3UDkJWewpIp+Xzs/BksLy+gojyf0gkaEB4sNyuds2ZExzmOpb/fj5yM2NzZQ1MQJgOh0tTZw8GOHva1dLF2TwsHOnresoy0FDsSKnnj0oHo4GxXb4SWzh6aO3tp6ex5017BgMy0FEonZFKam8XskhzOmVVESW4mpbnRcBi4FedkJlzXWqihYWbpRAPjV+7+QNBcb2aTBnVPNQTte4Fpgz4+NWgTGdX6Iv1sa+hgXW0Lr9a2sq62hdfq2ukL/hSeXjSec2cVBQFRwPxJuQn3i2S0SUkxCrIzKIjxJMKevn4OdHSzv62LhrYu6tu6qQ/uG9q7aO7swQDMyExNYWZxNhXjM8gfn/GWMCjNzSQnM23U7g2GFhpB19NPgM3u/s1BLz0E3Ah8Pbh/cFD7LWZ2D9GB8NYTjWeIJKKevn52HTzE5ro2Xt0TDYiN+9o43Bud/C83K40lU/O46cJZVJQXsKw8n+I4dD/IyclIS2Fy/rgxfXTWgDD3NM4HPgysN7O1QdvfEw2L+8zsE0ANcH3w2iNEj5zaTvSQ24+NbLkiQ+vvdw4c6qaupYu61i72tx6mrrWL3U2dbGvoYNeBQ0f2ILLSU1g0OY8bVk5j6dR8lkzNY0ZRtmaGlYQWWmgEA9rH+99x6THe78DNcS1KZAjuTkN7N9vqO3i9sYO9LYffFA71bV1v6cPOSE1hSsE45pTmcPmiMuaW5nJGWS5nlOWQpm4mGWVCHwgXSWQ9ff1U7WrilV3NVNU0sXZPC+1dfUdez0xLYVJeFhPzsjhrRiFlE7KYnJ/FxAlZTM4fx8S8LIqyM0Zt/7XI0RQaIkfpi/Tz1GsNPLK+jj+91kB7Vx9mMK8sl79aOpn5E3OZU5rDnNIcSnIyFQgypig0RAKN7d3c9WIN976ym/q2bgrGp3PFoom8c9FEVs4sPHJYpchYptCQMa+xvZs7nnmdu16sobuvn4vOKOGfrp7OJfNKNOYgchSFhoxZfZF+fvFCDd98YiudPX1cvXwKt1wyh1klOWGXJpKwFBoyJm2rb+ez965l4742LjyjhC//1UJmKyxEhqTQkDHF3fn1y7v5x99vIiczje9/sIIrF0/UYLZIjBQaMmb09PXzhd+u5z+ra3nb3GJuv36pLuIjcpIUGjImtHb28qlfVvPCjoN8+u1z+Ow7ztCZ1yKnQKEhSa+1s5cP/uRFtuxv55vXL+XaiqlhlyQyaik0JKkNBMbW/R3c8eFKLplfGnZJIqOaDkKXpHW4J8KNP3uZrfs7+OGHVygwRIaB9jQkKUX6nc/eu4ZXa1v4jw8pMESGi/Y0JCn9yyObeWxjPV9810IuXzQx7HJEkoZCQ5LOva/s5sfP7eSj583g4+fPCLsckaSi0JCksr62lS89uJEL5hTzpXcv1El7IsNMoSFJo/lQD5/6ZTXF2Rl85/3LSdV5GCLDTgPhkhSiA99raWzv5r5PnUthdkbYJYkkJe1pSFL49z9t4y9bG/nyXy9k2bT8sMsRSVoKDRn1XtxxkG8/tY1rK6bwgZXlYZcjktQUGjKqNR/q4XP3rmVGUTZfW7VYA98icaYxDRm13J3P37+OAx3dPPDfzyc7Uz/OIvEW6p6Gmf3UzBrMbMOgtq+Y2V4zWxvcrhr02m1mtt3MtpjZ5eFULYnily/t5vFN9Xz+ivmcOTUv7HJExoSwu6d+DlxxjPZvufuy4PYIgJktBG4AFgWf+b6ZpY5YpZJQtuxv558e3sRFZ5Tw8fNnhl2OyJgRami4+zNAU4xvXwXc4+7d7r4T2A6sjFtxkrC6eiP87d2ryc1K5/9et1TXxRAZQWHvaRzPLWa2Lui+KgjapgB7Br2nNmh7EzO7ycyqzKyqsbFxJGqVEfYvj2xma30Ht1+/lJLczLDLERlTEjE0fgDMBpYBdcDtJ/Nhd7/D3SvdvbKkpCQe9UmIntt2gDtfqOFj58/gojP0/YqMtIQLDXevd/eIu/cDP+KNLqi9wLRBb50atMkY0dbVy//6zavMKsnm81fMD7sckTEp4ULDzCYNenoNMHBk1UPADWaWaWYzgbnAyyNdn4Tna7/fxP62Lm6/bilZ6ToGQiQMoR7YbmZ3AxcDxWZWC3wZuNjMlgEO7AI+CeDuG83sPmAT0Afc7O6RMOqWkffkpnr+s7qWmy+ZzfLygqE/ICJxYe4edg1xU1lZ6VVVVWGXIaep+VAPl33rGYpzMnjwlvPJTNNehkg8mVm1u1ce6zWdQisJ70sPbqD1cA+/+PhKBYZIyBJuTENksIfX7ePhdXV85tK5LJw8IexyRMY8hYYkrIb2Lr70uw0snZrHpy6aHXY5IoJCQxKUu/P3D2ygsyfC7dcvIy1VP6oiiUD/EyUh3b96L09urud/Xj6POaU5YZcjIgGFhiScvS2H+epDG1k5s1CTEYokGIWGJJT+fufv7nuVfnf+73s1GaFIolFoSEL5+fO7eGHHQb707oWUF40PuxwROYpCQxLG9oZ2vvHoa1w6v5T3nTVt6A+IyIhTaEhC6I3087l7X2V8Rir/8p4zda1vkQSlM8IlIXz7yW2s39vKf3yogtLcrLDLEZHj0J6GhO7ZbY187+ntXLdiKlcsnjT0B0QkNAoNCVV9WxefvWctc0tz+MdVi8MuR0SGoO4pCU1fpJ9P372Gzp4I93yggnEZmoxQJNEpNCQ0//bkNl7a2cTt1y1lbllu2OWISAzUPSWheHJT/ZFxjPesmBp2OSISI4WGjLht9e189t61LJ6cx9eu1jiGyGii0JAR1dLZw3/7RRVZ6anc8ZEVuta3yCij0JAR0xfp52/vXsO+lsP88MMVTMobF3ZJInKSNBAuI+brf3yNZ7cd4BvvOZMV0wvDLkdEToH2NGRE3F9dy4+f28mN507nfWeVh12OiJyiUEPDzH5qZg1mtmFQW6GZPWFm24L7gqDdzOw7ZrbdzNaZWUV4lcvJWLunhdt+u55zZxXxxXcvDLscETkNYe9p/By44qi2W4Gn3H0u8FTwHOBKYG5wuwn4wQjVKKehoa2LT95VRWluJt/7YAXpumyryKgW6v9gd38GaDqqeRVwZ/D4TuDqQe2/8KgXgXwz00RFCayrN8Inf1lNe1cfP76xksLsjLBLEpHTlIh/9pW5e13weD9QFjyeAuwZ9L7aoO1NzOwmM6sys6rGxsb4VirH5e588XcbWLO7hW9ev5T5EyeEXZKIDINEDI0j3N0BP8nP3OHule5eWVJSEqfKZCh3Pr+L31TX8plL52rmWpEkkoihUT/Q7RTcNwTte4HBl3ObGrRJgnnh9YN87Q+buWxhGZ+5dG7Y5YjIMErE0HgIuDF4fCPw4KD2jwRHUZ0DtA7qxpIEUdvcyc2/Xs3M4my+ef1SUlJ0BT6RZBLqyX1mdjdwMVBsZrXAl4GvA/eZ2SeAGuD64O2PAFcB24FO4GMjXrCc0OGeCJ+8q5reSD93fHgFuVnpYZckIsMs1NBw9/cf56VLj/FeB26Ob0VyqtydL/x2PZvq2vjpjWcxqyQn7JJEJA4SsXtKRqH7V+/lgTV7+eylZ3DJ/NKwyxGROFFoyGl7vbGDf3hwA+fMKuSWt88JuxwRiSOFhpyWnr5+/vbXa8hMS+Hf3recVA18iyQ1zXIrp+X7T29nU10bP/pIJRPzssIuR0TiTHsacsq27G/ne3/ezqplk7lsYdnQHxCRUS+m0DCzz5jZhOAciZ+Y2Woze2e8i5PEFel3/tf968jNSucfNHOtyJgR657Gx929DXgnUAB8mOj5FDJG/aZ6D6/uaeFL715AUU5m2OWIyAiJNTQGRjevAu5y942D2mSM6eju4/88tpWK8nyuXvaWOSNFJInFGhrVZvY40dB4zMxygf74lRW+ls4eevqSehNP2Q+e3s6Bjm6+9O6FmOlvB5GxJNbQ+ATRiyGd5e6dQAZJPI3HrgOHOP/rf+J3azQf4tEOdHTzk+d2smrZZJaXF4RdjoiMsBMecnuMS6rOGgt/WU4vGs+skhy+//R2rq2YQpquNnfEj57dQU9fP5/W7LUiY9JQ52ncHtxnASuAdUTHMpYAVcC58SstPGbGLW+fwyfvquYP6+tYpX57AJoO9XDXCzW8e8lkZmtuKZEx6YR/Qrv7Je5+CVAHrAgubrQCWE6SX8visgVlnFGWw3f/tJ3+/pO6DlTS+ulzOzncG9FUISJjWKz9LvPcff3AE3ffACyIT0mJISXF+JuL57CtoYNntumysV29EX71Ug3vWFDGGWW5YZcjIiGJNTTWm9mPzezi4PYjol1VSe2qMydRnJPJXS/UhF1K6B5eV0dzZy8fPW9G2KWISIhiDY2PAhuBzwS3TSTx0VMDMtJS+MDZ5fxpSwO7D3aGXU5o3J07n9/FnNIczptdFHY5IhKiIUPDzFKBP7r7t9z9muD2LXfvGoH6QvfBs8tJNeOXL43dvY01e1pYv7eVG8+drvMyRMa4IUPD3SNAv5nljUA9CadsQhaXLSzj/upaeiNj82S/+17Zw/iMVK6pmBp2KSISslinRu8gOq7xBHBooNHdPx2XqhLMeyqm8scN+3lmayOXLhhbs7l29Ub4w/o6rlg0kZxMzaQvMtbF+lvggeA2Jl00r4TC7AweWL13zIXGk5vrae/q41rtZYgIMYaGu98Z70ISWXpqCn+9dDK/fmk3rZ295I1PD7ukEfPA6r1MnJDFuRoAFxFiv57GXDP7jZltMrMdA7d4FmZmu8xsvZmtNbOqoK3QzJ4ws23B/YhNfvSeiqn0RPp5ZEPdSK0ydAc6uvnL1kZWLZ+sy7iKCBD7Ibc/A34A9AGXAL8Afhmvoga5xN2XuXtl8PxW4Cl3nws8FTwfEYunTGB60Xj+uGH/SK0ydI9vrCfS76xaqmlURCQq1tAY5+5PAebuNe7+FeBd8SvruFYBA11ldwJXj9SKzYwrFk/k+e0HaO3sHanVhuqxjfuZXjSeBZN0BriIRMUaGt1mlgJsM7NbzOwaIN4z1jnwuJlVm9lNQVuZuw/0D+0H3jIqbWY3mVmVmVU1Ng7v9B9XLp5EX7/z5Ob6YV1uImo93Mvzrx/gikUTdW6GiBwRa2h8BhgPfJrobLcfAm6MV1GBC9y9ArgSuNnMLhz8ors70WDhqPY7gokVK0tKSoa1oCVT8piUl8WjG5O/i+rPrzXQG3HeuWhi2KWISAKJ9ZDbJnfvIHq+xohMH+Lue4P7BjP7LbASqDezSe5eZ2aTgIaRqGVASopx+aKJ3P3ybg5195GdxOctPLZxP6W5mSyflh92KSKSQGLd0/ipmb1uZveY2c1mdmY8izKz7OCSsphZNvBOYAPwEG/s4dwIPBjPOo7l8kUT6e7r57ntB0Z61SOmqzfC01sauXzRRFJ01JSIDBLreRoXmVkGcBZwMfAHM8tx98I41VUG/DboS08Dfu3uj5rZK8B9ZvYJoAa4Pk7rP67KGQXkZqbx9JYGLk/SrpsXdxzkcG+ESxeUhl2KiCSYmELDzC4A3hbc8oGHgWfjVZS77wCWHqP9IHBpvNYbi/TUFM6fU8zTWxpx96QcJP7L1kYy01I4Z5ZO6BORN4u1U/5poBr4F+ARd++JW0WjwCXzS3h043621Lczf+KEsMsZdn/Z2si5s4vISk8NuxQRSTCxjmkUA/9I9Jrgj5rZk2b2tfiVldguOiPabfP0luS7ot+epk52NB7iojOG98gzEUkOMYWGu7cAO4CdRK8XPhu48IQfSmIT87JYMGkCT28Z0YO3RsTTW6NBqNAQkWOJde6pHcDtQCHR6UTmuftF8Sws0V08r4SqXc20dyXX2eF/2dLItMJxzCzODrsUEUlAsXZPzXH3q9z9n939ubE+pgHwtrnF9PU7r+xqCruUYdPT18/zrx/gojNKknKAX0ROX8yhYWZPmdkGADNbYmZfjGNdCa+ivICMtBSe334w7FKGzau1LXT2RLhgjrqmROTYYg2NHwG3Ab0A7r4OuCFeRY0GWempVE4v4PnXkyc0Xnj9IGZwzqx4nX4jIqNdrKEx3t1fPqqtb7iLGW3Om13Epro2mg8lR2/dC68fZMHECeSPzwi7FBFJULGGxgEzm00wQaCZvZfoUVRj2sDV7F7cMfr3Nrp6I6ze3awr9InICcUaGjcDPwTmm9le4LPAp+JW1SixZGo+4zNSk6KLau2eFrr7+nUWuIicUKxzT+0A3hFMHpgCdBId06iJY20JLz01hZUzC3n+9dE/eeHAeMbKmRrPEJHjO+GehplNMLPbzOy7ZnYZ0bC4EdhOCJMFJqLzZhfxeuMh6tu6wi7ltLy44yCLJk8gb1x62KWISAIbqnvqLmAesB74/4A/A9cB17j7qjjXNioMdOe8vHP0nq/R1Rthze4WzlXXlIgMYajuqVnufiaAmf2Y6OB3ubuP7j+rh9HCSRMYn5HKK7ua+Kulk8Mu55SsrmmmJ9KvQXARGdJQexpH5shw9whQq8B4s7TUFCrKC3hlV3PYpZyyl3Y2kWJQOUPjGSJyYkOFxlIzawtu7cCSgcdm1jYSBY4GlTMKeG1/G62HR+c8VKt3NzNv4gQmZGk8Q0RO7ISh4e6p7j4huOW6e9qgx8l3IYlTtHJGIe7RX76jTaTfWbO7hRXTdS1wERlarOdpyAksK88nNcWoGoWTF26tb6eju4+K8oKwSxGRUUChMQzGZ6SxePIEXtk5+vY0BvaOVkxXaIjI0BQaw+SsGYWsrW2huy8SdiknpbqmmeKcDMoLx4ddioiMAgqNYVI5o5Cevn7W17aGXcpJWV3TzPLyAl0/Q0RiMupCw8yuMLMtZrbdzG4Nu54BlTOi3Tuj6dDbAx3d7DrYqa4pEYnZqAoNM0sFvgdcCSwE3m9mC8OtKqo4J5NZJdmj6kp+a3a3ABrPEJHYjarQAFYC2919R3DJ2XuAhJnOpHJ6Aat3N+PuYZcSk+qaZtJTjTOn5IVdioiMEqMtNKYAewY9rw3aEkJFeQEtnb3sOHAo7FJisrqmmYWT88hKTw27FBEZJUZbaAzJzG4ysyozq2psbBzRdVcE3TwD3T6JrDfSz6u1LazQ+RkichJGW2jsBaYNej41aDvC3e9w90p3rywpKRnR4uaU5JCbmTYqzgzftK+N7r5+jWeIyEkZbaHxCjDXzGaaWQbRC0E9FHJNR6SkGMvK81ldk/ihUR3UWKHpQ0TkJIyq0HD3PuAW4DFgM3Cfu28Mt6o3W15ecGRqjkRWvbuZyXlZTMobF3YpIjKKxHS510Ti7o8Aj4Rdx/FUlOfT7/DqnhbOn1McdjnHtaam+cgYjIhIrEbVnsZosHxa9BdxIndR7Ws5zL7WLo1niMhJU2gMs7zx6cwpzWHNnsQ9gmpgoF4z24rIyVJoxMHyafmsSeCT/KprmslKT2HhZF0SRUROjkIjDiqmF9Dc2cvOBD3Jb/XuFpZMzSc9VV+/iJwc/daIg4Fun9UJeJJfV2+EjXtbNZ4hIqdEoREHc0sT9yS/dbWt9PW7xjNE5JQoNOIgkU/yO3JSX7lO6hORk6fQiJNEPclv9e5mZhZnU5STGXYpIjIKKTTiZOAkv3UJdOitu7O6plldUyJyyhQacXLkJL8EGteoOdjJwUM9mm9KRE6ZQiNO8sanM7skO6GOoBoIMB05JSKnSqERRxXlBQl1kl91TTO5mWnMLc0NuxQRGaUUGnGUaCf5Vdc0s6w8n9QUC7sUERrKn5cAAA46SURBVBmlFBpxlEgn+bV39bKlvl2D4CJyWhQacTS3NIfcrLQj50aEae2eFtw1niEip0ehEUcpKUZFeQHVNU1hl0J1TTNmsFwn9YnIaVBoxFnl9AK21nfQ2tkbah3VNc3MK8slNys91DpEZHRTaMTZQHfQ6j3hdVFF+p01u1vUNSUip02hEWdLp0WPVqreFV5oDExnUjlDoSEip0ehEWfZmWksmJQb6mB4VbDuFeWFodUgIslBoTECKqcXsnZPC72R/lDWv7qmmeKcTKYVjgtl/SKSPBQaI2DF9AIO90bYXNcWyvqra5qpnF6AmU7qE5HTk3ChYWZfMbO9ZrY2uF016LXbzGy7mW0xs8vDrPNkDAxAh9FF1dDexe6mTg2Ci8iwSLjQCHzL3ZcFt0cAzGwhcAOwCLgC+L6ZpYZZZKwm549jcl7WkbGFkTRwIagKhYaIDINEDY1jWQXc4+7d7r4T2A6sDLmmmFVMLwjlSn7VNc1kpKWweMqEEV+3iCSfRA2NW8xsnZn91MwG/kSeAuwZ9J7aoO1NzOwmM6sys6rGxsaRqDUmldMLqGvtYm/L4RFdb1VNM0um5JGZNip2ykQkwYUSGmb2pJltOMZtFfADYDawDKgDbj+ZZbv7He5e6e6VJSUlcaj+1KyYHj3cdSTHNbp6I2zY26rxDBEZNmlhrNTd3xHL+8zsR8DDwdO9wLRBL08N2kaFBZNyGZ+RSvWuJv566eQRWee62lZ6I67QEJFhk3DdU2Y2adDTa4ANweOHgBvMLNPMZgJzgZdHur5TlZaawrJp+SM6GP7SjoMAnDVDJ/WJyPBIuNAA/tXM1pvZOuAS4HMA7r4RuA/YBDwK3OzukfDKPHlnzShkc10brYdHZvLCl3Y2MX9iLgXZGSOyPhFJfgkXGu7+YXc/092XuPtfu3vdoNf+t7vPdvd57v7HMOs8FefMKqLf4ZWd8Z8qvTfST3VNM+fMKor7ukRk7Ei40Ehmy8vzyUhL4cWg2yie1tW2crg3wtkz1TUlIsNHoTGCstJTWT4tnxd3xj80XgrWsVKhISLDSKExws6dXcTGffEf13hpRxNzS3MoysmM63pEZGxRaIywc2YV4XEe1+iL9FO1q4mzZ2kvQ0SGl0JjhC2bFv9xjY372jjUE+HsmRoEF5HhpdAYYVnpqVSUx3dc44UgkLSnISLDTaERgnNmxXdc47ltB5hXlktpblZcli8iY5dCIwTnzS7GHV54ffj3Ng73RHh5VxNvm1s87MsWEVFohGB5eT45mWk8s234Z+F9eVcTPX39vO2MxJmsUUSSh0IjBOmpKZw3u4i/bGnE3Yd12c9tayQjLYWVmm9KROJAoRGSi+aVsLflMK83HhrW5T677QArZxQyLkPXzxCR4afQCMmFc6PdR3/ZOnxdVA1tXby2v13jGSISNwqNkEwrHM/skuxhDY2ng2VdoNAQkThRaIToojNKeXHHQQ73DM8M709sqmdyXhYLJ+l64CISHwqNEL19fik9ff3DchTV4Z4Iz25r5B0LyzCzYahOROStFBohOntWIXnj0nl0w/7TXtZz2w/Q1dvPZQvLhqEyEZFjU2iEKD01hcsWlvHk5np6+vpPa1lPbNpPbmaa5psSkbhSaITsysUTae/q479eP3DKy+iN9PPk5gYunl9KRpq+UhGJH/2GCdkFc4vJyUzj0fWn3kX17LZGmg71sGrp5GGsTETkrRQaIctMS+WyhWX8cUMdXb2ndhTVb9fso2B8Ohdq6hARiTOFRgK4bsVU2rr6eGzjye9tdHT38cSm/bxrySR1TYlI3IXyW8bMrjOzjWbWb2aVR712m5ltN7MtZnb5oPYrgrbtZnbryFcdP+fMKmJqwTjuq9pz0p99+NV9dPX2c83yKXGoTETkzcL603QDcC3wzOBGM1sI3AAsAq4Avm9mqWaWCnwPuBJYCLw/eG9SSEkxrlsxjf/afpDdBztj/py78/PndzF/Yi4V5QVxrFBEJCqU0HD3ze6+5RgvrQLucfdud98JbAdWBrft7r7D3XuAe4L3Jo33nTWN9FTjJ8/tiPkzL+9s4rX97Xz0vBk6oU9ERkSidYJPAQb30dQGbcdrfwszu8nMqsysqrFx+K9XES8T87K4etkU7q3aw8GO7pg+86Nnd5A3Lp1Vy9Q1JSIjI26hYWZPmtmGY9ziuofg7ne4e6W7V5aUjK6jiT550Sy6evv58XM7h3zv6t3NPLm5gZsunKVp0EVkxKTFa8Hu/o5T+NheYNqg51ODNk7QnjTmlOZyzfIp/OS5nbz/rHLKi8Yf8339/c7X//gaRdkZfPS8GSNbpIiMaYnWPfUQcIOZZZrZTGAu8DLwCjDXzGaaWQbRwfKHQqwzbm69cj5pKcYXH9xAf/+xr+p39yu7eXlnE393+TyyM+OW+yIibxHWIbfXmFktcC7wBzN7DMDdNwL3AZuAR4Gb3T3i7n3ALcBjwGbgvuC9SadsQha3XbWAZ7Y28t0/b3/L6+tqW/jaw5s4b3YRN5w17RhLEBGJHxvua1QnksrKSq+qqgq7jJPm7vz/973Kb9fs5W8uns2nL51LZloKf97SwOfufZXcrDR++zfnU5KbGXapIpKEzKza3SuP9Zr6NhKQmfF/3ruEjNQUvv/069z5/C7GZaRxoKObuaU5/OTGsxQYIhIKhUaCSktN4RvvXcK1FVN4ZH0dh3oirJxRyNXLp2i6EBEJjUIjwZ09q4izZ+kaGSKSGPQnq4iIxEyhISIiMVNoiIhIzBQaIiISM4WGiIjETKEhIiIxU2iIiEjMFBoiIhKzpJ57yswagZrTWEQxcGCYykl02tbkNJa2FcbW9sZzW6e7+zEvSJTUoXG6zKzqeJN2JRtta3IaS9sKY2t7w9pWdU+JiEjMFBoiIhIzhcaJ3RF2ASNI25qcxtK2wtja3lC2VWMaIiISM+1piIhIzBQaIiISM4XGMZjZFWa2xcy2m9mtYdczHMxsl5mtN7O1ZlYVtBWa2RNmti24Lwjazcy+E2z/OjOrCLf6oZnZT82swcw2DGo76e0zsxuD928zsxvD2JahHGdbv2Jme4Pvd62ZXTXotduCbd1iZpcPak/4n3Mzm2ZmfzazTWa20cw+E7Qn3Xd7gm1NrO/W3XUbdANSgdeBWUAG8CqwMOy6hmG7dgHFR7X9K3Br8PhW4BvB46uAPwIGnAO8FHb9MWzfhUAFsOFUtw8oBHYE9wXB44Kwty3Gbf0K8HfHeO/C4Gc4E5gZ/Gynjpafc2ASUBE8zgW2BtuUdN/tCbY1ob5b7Wm81Upgu7vvcPce4B5gVcg1xcsq4M7g8Z3A1YPaf+FRLwL5ZjYpjAJj5e7PAE1HNZ/s9l0OPOHuTe7eDDwBXBH/6k/Ocbb1eFYB97h7t7vvBLYT/RkfFT/n7l7n7quDx+3AZmAKSfjdnmBbjyeU71ah8VZTgD2Dntdy4i9utHDgcTOrNrObgrYyd68LHu8HyoLHyfJvcLLbN9q3+5agS+anA901JNG2mtkMYDnwEkn+3R61rZBA361CY+y4wN0rgCuBm83swsEvenR/N2mPv0727QN+AMwGlgF1wO3hljO8zCwHuB/4rLu3DX4t2b7bY2xrQn23Co232gtMG/R8atA2qrn73uC+Afgt0V3Y+oFup+C+IXh7svwbnOz2jdrtdvd6d4+4ez/wI6LfLyTBtppZOtFfor9y9weC5qT8bo+1rYn23So03uoVYK6ZzTSzDOAG4KGQazotZpZtZrkDj4F3AhuIbtfAUSQ3Ag8Gjx8CPhIciXIO0DqoK2A0Odntewx4p5kVBF0A7wzaEt5RY07XEP1+IbqtN5hZppnNBOYCLzNKfs7NzICfAJvd/ZuDXkq67/Z425pw323YRwwk4o3oERhbiR6B8IWw6xmG7ZlF9AiKV4GNA9sEFAFPAduAJ4HCoN2A7wXbvx6oDHsbYtjGu4nuuvcS7cP9xKlsH/BxogOK24GPhb1dJ7GtdwXbsi74BTFp0Pu/EGzrFuDKQe0J/3MOXEC062kdsDa4XZWM3+0JtjWhvltNIyIiIjFT95SIiMRMoSEiIjFTaIiISMwUGiIiEjOFhoiIxEyhIRIDM4sMmmV07VAzh5rZp8zsI8Ow3l1mVny6yxEZLjrkViQGZtbh7jkhrHcX0XMNDoz0ukWORXsaIqch2BP4V4teq+RlM5sTtH/FzP4uePzp4BoJ68zsnqCt0Mx+F7S9aGZLgvYiM3s8uJ7Cj4merDawrg8F61hrZj80s9Tg9nMz2xDU8LkQ/hlkDFFoiMRm3FHdU+8b9Fqru58JfBf4t2N89lZgubsvAT4VtH0VWBO0/T3wi6D9y8Bz7r6I6Bxh5QBmtgB4H3C+uy8DIsAHiU5iN8XdFwc1/GwYt1nkLdLCLkBklDgc/LI+lrsH3X/rGK+vA35lZr8Dfhe0XQC8B8Dd/xTsYUwgeoGla4P2P5hZc/D+S4EVwCvRKYoYR3SSvt8Ds8zs34E/AI+f+iaKDE17GiKnz4/zeMC7iM6HVEH0l/6p/LFmwJ3uviy4zXP3r3j0gkJLgaeJ7sX8+BSWLRIzhYbI6XvfoPsXBr9gZinANHf/M/B5IA/IAZ4l2r2EmV0MHPDotROeAT4QtF9J9NKkEJ2c771mVhq8Vmhm04Mjq1Lc/X7gi0SDSSRu1D0lEptxZrZ20PNH3X3gsNsCM1sHdAPvP+pzqcAvzSyP6N7Cd9y9xcy+Avw0+Fwnb0zz/VXgbjPbCDwP7AZw901m9kWiV19MITrD7c3AYeBnQRvAbcO3ySJvpUNuRU6DDomVsUbdUyIiEjPtaYiISMy0pyEiIjFTaIiISMwUGiIiEjOFhoiIxEyhISIiMft/5FSqcXoB810AAAAASUVORK5CYII=\n"
          },
          "metadata": {
            "needs_background": "light"
          }
        }
      ],
      "source": [
        "signal = smooth(sum_rewards, window_len=600, window='hanning')\n",
        "plt.plot(signal)\n",
        "plt.xlabel('Episodes')\n",
        "plt.ylabel('Rewards')\n",
        "plt.title('SAD-DQN')"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 14,
      "metadata": {
        "id": "LNEk4vS1c8W5"
      },
      "outputs": [],
      "source": [
        "# save to npy file\n",
        "np.save(\"SAD-DQN_N\"+ str(num_avar) + \"_\" + env_name  +\".npy\", signal)"
      ]
    }
  ],
  "metadata": {
    "colab": {
      "collapsed_sections": [],
      "provenance": []
    },
    "kernelspec": {
      "display_name": "Python 3",
      "language": "python",
      "name": "python3"
    },
    "language_info": {
      "codemirror_mode": {
        "name": "ipython",
        "version": 3
      },
      "file_extension": ".py",
      "mimetype": "text/x-python",
      "name": "python",
      "nbconvert_exporter": "python",
      "pygments_lexer": "ipython3",
      "version": "3.7.6"
    }
  },
  "nbformat": 4,
  "nbformat_minor": 0
}