{
  "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": "475d7a93-0e4e-485f-a5a5-dc8efc0271ea"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "--2022-09-22 09:45:54--  https://raw.githubusercontent.com/senya-ashukha/quantile-regression-dqn-pytorch/master/logger.py\n",
            "Resolving raw.githubusercontent.com (raw.githubusercontent.com)... 185.199.108.133, 185.199.110.133, 185.199.111.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: 2770 (2.7K) [text/plain]\n",
            "Saving to: ‘logger.py.3’\n",
            "\n",
            "\rlogger.py.3           0%[                    ]       0  --.-KB/s               \rlogger.py.3         100%[===================>]   2.71K  --.-KB/s    in 0s      \n",
            "\n",
            "2022-09-22 09:45:54 (36.2 MB/s) - ‘logger.py.3’ saved [2770/2770]\n",
            "\n",
            "--2022-09-22 09:45:54--  https://raw.githubusercontent.com/senya-ashukha/quantile-regression-dqn-pytorch/master/rl_utils.py\n",
            "Resolving raw.githubusercontent.com (raw.githubusercontent.com)... 185.199.110.133, 185.199.111.133, 185.199.108.133, ...\n",
            "Connecting to raw.githubusercontent.com (raw.githubusercontent.com)|185.199.110.133|:443... connected.\n",
            "HTTP request sent, awaiting response... 200 OK\n",
            "Length: 1162 (1.1K) [text/plain]\n",
            "Saving to: ‘rl_utils.py.3’\n",
            "\n",
            "rl_utils.py.3       100%[===================>]   1.13K  --.-KB/s    in 0s      \n",
            "\n",
            "2022-09-22 09:45:55 (56.9 MB/s) - ‘rl_utils.py.3’ 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": "4f297b0c-f863-45fb-9182-618d97425df5"
      },
      "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: importlib-metadata>=4.8.0 in /usr/local/lib/python3.7/dist-packages (from gym[box2d]) (4.12.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: 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: box2d-py==2.3.5 in /usr/local/lib/python3.7/dist-packages (from gym[box2d]) (2.3.5)\n",
            "Requirement already satisfied: swig==4.* in /usr/local/lib/python3.7/dist-packages (from gym[box2d]) (4.0.2)\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",
            "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"
          ]
        }
      ]
    },
    {
      "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": "42f33f2f-fd5e-4fa5-c8d7-bb2887634887"
      },
      "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": "c2d73b25-6095-48e2-da54-c7bdbd4b4943"
      },
      "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": "f5481e1f-e122-4e75-90f4-7dd2a13c79d4"
      },
      "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 [43:14<00:00,  1.30s/it]\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",
        "        Ttheta_tgt = rewards + gamma * (1 - dones) * Znext_max\n",
        "        atoms_tgt = torch.cat( (theta_tgt, Ttheta_tgt), dim=1)\n",
        "        atoms_tgt = atoms_tgt.detach().numpy()\n",
        "\n",
        "        # probabilities\n",
        "        probas = ( (1.0 - mixture_ratio) / num_avar ) * torch.ones_like( theta_tgt )\n",
        "        probas = torch.cat((probas, (mixture_ratio / num_avar)*torch.ones_like(Znext_max)), dim=1)\n",
        "        probas = probas.detach().numpy()\n",
        "\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 2N 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": "e3383296-7961-46dd-ddec-774c8aed84a2"
      },
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "Text(0.5, 1.0, 'MAD-DQN')"
            ]
          },
          "metadata": {},
          "execution_count": 11
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 432x288 with 1 Axes>"
            ],
            "image/png": "iVBORw0KGgoAAAANSUhEUgAAAY8AAAEWCAYAAACe8xtsAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjIsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+WH4yJAAAgAElEQVR4nO2deZwUxfXAv293uW8EAQE5BBQQVFhRFO8Lj8R4G2PwSozGKzH+EoxG8b7iEc9o1HgkarxPVEAULxABkUNREAFBUJTbBfaq3x/ds9sz0zPTPdM9PbP7vp/PwkxNddXrq17Ve1WvxBiDoiiKovihJGoBFEVRlOJDlYeiKIriG1UeiqIoim9UeSiKoii+UeWhKIqi+EaVh6IoiuIbVR6KoiiKb1R5KIqiKL5R5aEogIgsEZFKEemUkP6JiBgR6e1IG2en7ZGQ93QRqRGRTfbf1yLybxEZkKHud0Rki4hsFJENIjJTRMaKSLOEfINE5GURWW/nnSwiezp+723LNT7huP+IyDjfF0VR0qDKQ1Hq+Rr4ZeyLiAwBWjoziIgAY4A19v+JTDXGtAbaAQcDm4GZIrJzhrrPN8a0AboBfwJOBsbb9SEiOwAfAHOBPsB2wIvARBEZkVDWHiKyV+bTVZTsUeWhKPU8TrxCOA14LCHPPlgN/IXAySLS1K0gY0yNMeYrY8zvgSnAOC8CGGN+Msa8A/wcGAkcaf80DksxXWaMWWOM2WiMuRP4D3BTQjE3A9d5qU9RskWVh6LUMw1oKyIDRaQUq/f/n4Q8pwGvAE/b33/modznsZSOZ4wxy4AZjuMOAZ5xyfo0sI+INHek3QsMEJGD/dSpKH5Q5aEo8cRGH4cAnwMrYj+ISEvgBOAJY0wV8CzupqtEvgU6ZiGL87hOwEqXPCuB0oTyN2ONPK7Nok5F8YQqD0WJ53HgFOB0kk1WxwDVQMwh/V/gcBHpnKHM7lg+EkTknw6H+l+9Hgf8gGUuS6QbYIAfE9IfBLqIiJeRkaL4RpWHojgwxizFcpwfgWVucnIa0BpYJiKrsMxITbCUTTqOAd6zyz/HGNPa/rs+1QEi0hMYHjsOmIQ16knkRGCaMWZrwnlUAlcB1wCSQT5F8Y0qD0VJ5izgQGPMT4607sBBwFHArvbfLljO6iTTlYiUikgfEbkL2B+rIc+IiLQUkf2Al4Dp1I9yrgL2EpHrRKSjiLQRkQuAM4ArUhT3ONAcGO2lbkXxgyoPRUnAniU1IyF5H2C2MWaCMWZV7A+4ExjqmIo7UkQ2ARuAd4C2wO7GmLkZqr1bRDYC3wF3AM8Bo40xtbZMC4FRWAprCbAOa1RxjDFmUorzqMFSLNn4WxQlLaI7CSpK8SEiPbBmh11pjHkoanmUxoeOPBSlCDHGLAcOB7qJSOuo5VEaHzryUBRFUXyjIw9FURTFN2VRCxAmnTp1Mr17945aDEVRlKJi5syZPxhj0q5fatDKo3fv3syYkThpRlEURUmHiCzNlEfNVoqiKIpvVHkoiqIovlHloSiKovhGlYeiKIriG1UeiqIoim9UeSiKoii+UeWhKIqi+EaVh6IoShEx6bPvWLV+S9RiqPJQFEUJitUbt2bOlAPGGH7z2AyOu+9DamsNS3/8KfNBIaHKQ1EKmCU/RNc4KOmZtWwti77fWPf902/Wsft1k3hu5nLfZVVUVrO+oiptno1bqlhr51mxbjN3v72I/W55J06GfKLKQ6lj5tK13D/lq6jFUGwmffYd+//9HcbPXZnV8dU1tZ7z1tYaTnt4OteP/5zlayuyqg9g/eYqNm2tBmBrdQ2ponY/+N5ihlz5ZlZ1zFuxnvveCe85nbt8PWt+qsyY79h7P+Tg296t+/75yg0ATP96TapDUrLfLe+wy9UTWLBqAx8s+sE1z+7XTWLYNRPrvsfqWRmRCUuVh1LHcfd9yA2vLwikrP975lN++cC0QMpyo7bWsH5z+p5asTPv2/UALLAbJT/MXLqWfpe9zrTFP8alr1i3mTU/VXLnWwuZu3w9VTW11NYaNm6pZsqXq3ng3cWMuulteo99jcenLnEte8Dlr/PAu8mN9+bKGna5agI7X/kmC7/byI6Xv8ED7y52LePa1z5n49bqlMolxsYtVXyybG1c2lF3vc9Nbyxg9catjLppMl+t3pS2DDe+WVPBxf+bTWV1vILdWl3Dz+5+n2HXTOTxaUv5fsOWpONmLo1XDrFzqLH//+SbeHm9EDN3jb7jPX714Edxv9XUGlau38yWKu+dgXygykMJhWdmLmdqQsMVBOs3VzHkyjc56YGp7HLVBEbf8S53vrUw6/IqKqtZV1FJZbXViL4xbyW1tYbvNmzhh03e7dczl65h8epNLPree0NWUVnNT3Yv3Y3aWqsxKikRNlfWsKWqJun3mMliS1UN81ZYyuD8J2bxxEfLAHjdHrVs2lrN+U/MYu8bJzPsmoncNvFLfnb3+/S/7HXGPj8HJLn+p2csZ11FJe8vjO8JV1bXcv345E7Gd46GdvY36wB4afa3aa9BTW165XH2YzM55t4P2VJVQ02t4cInP6n77fV5K1m+djMPv/815zw+k498PG9jn5/D85+sSBol1Dra57+9OC+pId/n5rc57r6p/Pax+oCrt074kqHj3uTFT1YA8OV31jMwc+la/vzspxkVpJvyc97rO99ayMgbJqctY+X6zRx77wf8aD+ze984mUufz7TzcW6o8lCKihlL1rBxazUfL7F6dwtWbeS2iV/y7perAauR3LCliorKam54/XMqKusb5wue/KSuUQWrxzjoijfZ9eqJDLj8de6b8hXn/GcWz85azh7Xv0X5tdbW4H97cR69x76WJMuKdZv5YtVGNm6p4rj7pnLgrVM4+LYpns9lyLgJDE5juon1ZGtrDQOveIPdrp4Y9/tdkxdx8G3v8sWqjfztxXkcddf7vPX597w6ZyXPzbLs7o9OtYKjPjV9Ga/OcTd/PT1jOSUuyqOkRDjzkY859aGPmPjZd1TX1PLge/UjiZlL63vYL3/6LY98uKTuu9cRbFVN+oZ1lmPUsXj1Jl7+NFkZfbJsHW/MX8VJD0yjqqaWL1ZtpPfY1/ja4S964qNldc7lHzZtZckPlmmuJqFhr034vnL9FjZuqaL32Nd4+uNv6tInfvZd3OcNW+qfyRi/fugjnp6xnIrKeKWfyLqKZBPZTn97g95jX2PeivVxdbmxpaqW4++byqxl6+ru+4p1m3ly+rK0x+WKKo8GwNzl6wN1mp3/xCymfuW9F1dTaxh9x7tMmL8qMBkSWbx6E796cBpnPeoeYn/Mw9MBGHHdJIaOm8CgK97k/imLuebVz+vyvPLpt/z1hbls2GKZu2YlmEPm22aixBHH49Pco1PvfeNkDrvj3YyNw8Yt9X6AGC9+sqKu1z1z6dq6XnNtreHwf7zHG/NWcs/blmnozsmLANicMPKY8uX3AFzx0jzmLLdkv278Z0n173vz2xkb6TfnJzdQGzdXMWuZNYL47WMzuH78Aq59rf56Hnffh1TV1HLvO4u48MlP4pRHzGdQXVubZBoCKLW1VZXd1a+qqeW9havj8ry94Hu22scak3z+416eD8BnDrPeFS/N57A7LD/E9eM/56et1QwZ9yZ/fWEuB946hcrqWsqvncSKdZsBOO3h6XWyrquopDphJCRS71P413vuJjg3qmpqaVJqNa+x8/9mTQX//uDrpLwbtqQefR511/tx5+fGda99Vnc+GQZygaLKowHws7vfj3Pc5cqrc1Zy+r+nZ8y3cv1mfty0lY1bqliwaiOXPPNpYDJ87nhh1vxUyYG3TuGDRekV2ncbtiQ15E9OX8Y9by+KGzkMHTeBCfNX8dbn38flHT/XUn5TvqhvxI66672Msn5qm2hibNhSFWeOGjJuAkPHWSOMaYt/ZMiVb/KH/82u+/24+z7kJNs/VFFVw+crN3DeE5/gxocOZ2qs0fvo6zV88Z3VefhmzeakY5atqeCmN9KPBNzu3eKEmV4PuzR85ddO4uY3vkhZ7pffbWLA5a8npceUx0uzv6X32Nf44/9m8+uHpseNZu6aXG+ONFh+GSduDaWzt725soa/vjC37riaWuMqy7BrJrK5soZdr57I+U/MivutRKTOopeqXY5deyf9L3u9zie32zUT+WHTVg65fQpXvfIZ6yuqqKyu5dYJX1BRWR1nivNDrN4lP9ZPcLgxIJ+lF1R5KFkz8obJDLdNO05ynV762pyVHP6P93h97koWrNoQN8MkHalmudzyZnLj9sInK7g3xYydjxzlzFtRr8RmLVtbZ5/euKXeWZ/YqA4dNyFJ5lhDd+dbC9mYxs+x0u5BpvIF3O9wQDsbjajIdtJCma08/vex1djHTGpOJ32LpqV1n7/8bhNrXcw76aiorGbxam/P4hq77PcS/DvrN1fx1gKrk+HHn5XIMfd+UO/wFnjio6XcNXkR973zVZJS9EqqNSVO02yYNOidBJXgqayu5blZyzmpvGfybzW1/PnZT3l6hv957mA5CZuWlvDpcqsnv3RNBW9/8X2Go+q5wEcP7vV5/k1sx977IQBfXX8EC1bV9zbdzDJbq2t5avqyJFNLOo688z3mf5veRFFZXYsxhgw+2IJjzU+VdGzVtO57bIQoCZ76F2d/yx0n78ayHytoVlavPH5xzwe+66yorKkb4WRi7xtTO6SD6M07R4SlJcI6W+neZZskg+SvL4TrKI+hykNJiTGGWkPcC3jvO4u4Y9JCmpXVD1pjvfEtVbVJisMYw9IfK3hu1nL+ePAAShxljZ+7kgue/IR54w6jpMRyEp41qk9dg9vK0fMsJJ6Z8Q3btW9R932ri/IAGJsw2+WDRT+kXT+QSXEATF38I3+f8AXtWzTNmLeQ+MU9H/Dunw/g6lc+ixuplZUmN+6H3DaFhTn08mM4FXwhMe7l+SxY5X/6daGhyqMBYozhi+82slPXtp6PSXQKb62u5ai73uer1ZtYcM3hdekxZ/LFT9fbyI+7b2rKcmtqDfv//R0A+m3bmqN37V73201vLKCm1vDhVz9Q3qsjAA+9/zUH7rQtAH97aT7HDuueVGbUJCqFVMojkcRpn9kSc6QXE8vWVFBba5JMfE1Kki3nQSiOQubZLFagFyLq82iAPPzBEkbf8R4fL/G+0vVvL85LSpv/7Qa2VNVSU2uoslcr+5mFBfHD8ouemk3vsa+x2TZZLLVt9k5FBDB5Qb2p6vlZK3zVFwW52MIbEztd8UZS2nQfz6hSWKjyKGKW/viT6/qD+SusaZvLEhyqy9dWZBV64pDbptD/MmuWylceHZAx/uGygO+bBBnWb67CpJzLojQU3HxDSvGiZqsiJuUowDYjJy54GnXT2wAsufFIX/UkTtnMlbUudv9icwArSmNHRx55oqbWcNFTnzBn+brMmT2S2N4aY8XACcvU47a6Nxsqa2o5+7H4xX4N3c6tKF55/aJ9ohbBE6o88sTK9Zt5afa3nPufWZkzeyRxZPHPKYvjYuAE3ZnPdjFTIr9+aDoTEkIunHh/aqe7X246bggPjin3dcxz5470PSJTlDDo3KZZ3PdHzxzB8F4dIpImNao8iphEU0/iKuJMAdkaKh1bNaNnx5a+jhluz/ZSlKhJnLzcpnkZ2/t8nvOBKo8iJpNqaKS6gxKBti3UnacUJyLx6qNUvC10zDeqPIqZDNrB+etD7yfHJYoxeUH6qJ3FRokI3dq1oFPr4lpIpyiQPPIoUeWhZMui7ze5bnifaWBx6fNzeWaGFUb6mlfro63+Y9JCDv/He8xZvo5v121mxhL/m9cUNPa7NrCb90WS+aRN8+IbFQ3p3i5qERolHVs1pX+X1lGL4UrxPcUFzPqKKjZVVtPdEboiCGJ7RCQ6dL2YpR54dzEnJMShun3SlwD8/G7/8YKKgVhPLXH4XyhceGB/rhv/eeaMWAEEE8OEh0nLpqUZQ8wr4eJ8bMdfuA/Nm5S67dWVkXYtmgQmkxuRjTxEpKeIvC0in4nIfBG5yE7vKCITRWSh/X8HO11E5E4RWSQic0RkWFSyp2LfW95OG2AtaLw4xBd+v4lfPxRMWIxiIRY+K2rVcfEhA1zTm7jEc3LjggP7MWfcoWnz3H3Kbr7lSkfiDL4YBaqH4/jNqD55qee2E3cJtXxnsEiPcR0jIUqzVTXwJ2PMIGBP4DwRGQSMBd4yxvQH3rK/AxwO9Lf/zgbuy7/I6ck2PPX0r9ckbfXpBa8d0sQw08XKr/bY3lO+2MsX9Yt3zG7ucblKS729dn86dEdaNk1vHNhvQGffcqXjoIFdXNMLuA3zRJAx0o4d1iOwslyRFJ8LjMiUhzFmpTFmlv15I/A50B04GnjUzvYo8Av789HAY8ZiGtBeRLrlWexQOPH+qZyaxeigsU2m8tr7rRt5RNxdTlV9kEot6HNs27yMz64+LNAy80W69yFoJZsvEkPWFxIF4TAXkd7AbsBHQBdjTGyz5VVArCvUHfjGcdhyOy2xrLNFZIaIzFi9enXizwVHLmsxGts6Dq8vUqxBzdfIo2vb5q7p6WbJHDbYvYfvl7w1LcVgt0pD0Eo2zAWlTlGjHj2nI3LlISKtgeeAPxhj4oLcG6t19NVCGmMeMMaUG2PKO3cu/N7GCf8MbmV1Q8fri1T/8uXnzXvmnJEZ5Ejmrl8G47ILuk03xl1JF3AbVkcxyOiFOKtVASvtSJWHiDTBUhz/NcY8byd/FzNH2f/H4nOvAJzThnrYaUXNjKXZTZOdt2I9s78JLk5WMeD1RSrJ88gjlVjpRkpNy+JfvbGH75Rd3QE3mca4n08Bt2F1pOtlFoH4dTifcz9yP3eueycmLKKcbSXAQ8DnxpjbHD+9DJxmfz4NeMmRPsaedbUnsN5h3mp0HHXX+3X7Pivx1Ps88lNfKqWWSnn16tgKiDd9ZKvo8naO+akmNIpB+bnhZ4FgvkcpUa7z2Bv4NTBXRGbbaX8FbgSeFpGzgKXAifZv44EjgEVABXBGfsVVosbruxHLF/nKXJfq9+nfiVH9O7lkLdLWrYAI2gVYIt5nNAaJpPySnnw/75EpD2PM+6S+NAe55DfAeaEKVQRUVtfyvxnfZM7YAPHvMM/Py5SqFrf6BwW86j2MU3Q3WxWucnvxvL1p1bSUJ6enfi+yUc4lIinXvYSJ81L7uez5vkORO8yVeKpraln0/SZ6j32NBas2JP1+3ztfuW4Zq9RT12gX3sAjdd5szVZB+zwwRecw37Vne/p3aZM2TzbXNyp9Gb9I0I/ZKgxpUqPKo4BY81Ml/S57nRP++SEAr7hsvrS2InkXvlw5LuxFTwHh2WyV8H/YpF7P4V2CbEdJYcy2ykc9hcrUSw+MWoQ4/Fz2fJutVHkUEN+u2wzA2orsVqpny0UH9c9rfWGT75coVe/fl8khwVRxyaHuoU2S6w4eV7NVwDUdMsj7OpfLjxyYc31epW/qWP0flR8qa7NVQt6w14Gp8ihg8mVuLWlgT0Gh9JJdGx8Po5RXzh/F+QdGo9ANhWei2n9Hb+u1TJrJut4nWzhbbm/HhEndM+RBlnwruwbWbBQuiYpg6Y8/UZPFVI4wehORz0ryiNdTz/fppFzn4ePtytrnoXN1A6UQTjOokUfYaEj2CFjyw0/s//d3uPDAfmnz5WueR2khx0DIgkJRhv4c5o6FYRHOsOnUupmrQgq6Hj/llQUyNPZ/BoXwFPl5FtTn0UBx3teV9sZO075ek/G4mUsz58mVAmlrAyPvI48U6a4vcyqHdLZ1B3iuNx8/lD8e0t9Vliifkd6dWoVa/h596vevz7bXHyRO85MfQ0NiHzDsUakqjwiI2Wa9dPh/+9jMhGODpxD2SBaBXtu0TJsnnU3bSd5HHqnMVtk6zH2okiAbiBPLe9KsrDQvjWahmBYh/nlxXvuCcph7ePTVYd4IiN3TxIfzw6/i993YWlWbF3lybYBS7Vvhh526tmXK/x2QczlQOJFIfSkBZ6OV4rA+IffA01HsK+A9S18QI496mpWVej9OzVYNE7dOQOK9vn78grjvD3/wNWt+il/XEUZnItfG1uvOePmj+KfqupEv35SrzyPiW/zcuXtlzBPEuxH1eeaCrjBvBNSNPIr4QXVSaL3SQtkMyg9OvVBo1xOCf1b9nuPwXh1yqy/NCUT9mIzo3THue6bndtee7V3TdeTRCIjFyymURqIQ5PBin/Xas8z7IsEAqvPy4ke5+VcYYVDyiVfpJcXnUEl0dGfKnnKtUCDSeEaVR4Rk0+jk+6UrRqLuScZo3sSHvdpjvn+NKWfSxftlJ5CSkShGq8mzpNLn9zW7L0RUeURAwTX/OT5z2Sq0Pxwczirq/IcnCaAMD+s8RIRDBnWh37atA6jRH1GbrXKuz+sK83DFSFGnv1pTKTgNjNhAcVoc6sxWWdzthrptufO82rdsklNZ+Z8Gmn2FT/x2D44f7i0wZWPbsz5I/O4Fk08S68z0PKX6VX0ejYG6qbqFQa7PXJBt2udXj2bapUnbufiimBzle+3Qib+fsEvUYmQk6GtaRLcodIIaKetsqwaKm2nH7zPzw6atAUlTGLgN11s0LfXlK3Aj347DQm0HWzXN7To6KdRzdJJuZOZ5I7G49TYFdNYepnKrz6MREFMkfm/1WY/OCMVf4leOEb070ql107rv2crk91n3arYplNhWfnCeW6rTjNJoVYSXNCsKLKiuK6kUoc62aqDE+TzsheN+ezbfrKlIaliGdG+Xo2SF0cPy4nT32njm+2wK4PI1WIb28P58Z/scH7jTtlkdFxS+xU7p9MhVEn+o8sgTbg2f33u9paomqaTmTSK4hVL3T2B4GVR43U8638owiJlD2UbVzRdBi+T1HHt2TB/vzEk2EwpmX3EIZ43qU/c9God55kp/NnS7zOXofh4Nn9gj7vdBraisqRu1xAik4coifyAL43zm975I0LcoxYHPtjFIM1fQCnnHLm0DLS9b2rdsWhAj70wcEPHoyA1VHnnC2Suq70FH0AIHgEhQaxv85fdstrILjmqFcFgMyzFERy4EfYq5TscOiyiiLfiusUD2mVflkSec9zuX2FYvzF4R9z2IXrZfOUpE4o7J1/KDhjDyyOVSXfuLnX3lL+DL4JnAziGbdR4N4QKGiCqPCMnm2aysjrdbRdJTkjBiHQWXK+8+j4CrS6Ukc53C7GSnrm185Y/KspPvGWbO0yzLUy/kOI+LRDOhU3UbKPENQmyFee7lBuN78BkegYSRR5avuN9GPts9zH+2S2ZnYy4UYwfVf2iYwj/LIBSN85ksDWT728zsvF0w/h9dJNgIqE2xGVQ2RLGmIYwqw4yqW2xhPYK6vunO2u9zE/zoyuMoMthqfRH2PjW97Z0z/XaiWjYLbgSaC6o88ob1sqxYt5mPFv8IFNDII4sy4g7JU9vsdapu3leYF8FsnURiyiOSqd4FTJzZykV53HDskMDqyva1ueX4XbjooHCCivpBn5w84Wz3Hp26FCjM+fxeCbvBPP+Afklp3hcJ+pPtxfP29pU/DA4Z1CWv9eVrV8J8EvROgk0SzFZ9O7XixPKeuVdiU78dtT86t2mmyqOxE/TisqgIIjxJYhmXHLZjcj1Z+jwykWpntlT07Ngivj4Px4w9fKe0v7dr0YT+eQy1HrtGUc1g87P4Lwi8iu98nxJHHpMv2T+t0t2paxtOLPfv/M5q5B/9a6/KI1+4vqMFssyjEB5EL3h1zIftB0rX4MZ6hG2al8WlR3GJ09UZu0ZhjeYycdDALjx37sicIygHxRO/3YPjhsU3/F5WdTt59YJR3Hy89wjJzuf59pP8RVYuhE5j0SkPERktIl+IyCIRGRu1PLkQxO2PyvoQv84jfRPkyUTipRXLcuQxcodt4mKAXfOLndlvQGdvhQG/HJHeVOGsL9Yon7F3nxS584cXh3mUTdDwXh3p2q453du3yJzZA7HGOJvZdXvt0IlbT4xvwM8/MNl0GgSjB3dl8p/2c5ithGN268E9pwzj1QtG1eU7dc/t44773b5905bbpnkZ27VrzuVHDgxcZjeKSnmISClwD3A4MAj4pYgMilYqb7i1r0H0HgIpI4smxE+17Vq4ryZ21hug7kgaebRqWsYrjpfSyuOxMKDXNq3S/u52/aLvF6Yndv6p7mOfTsnnvNv2/sx7XgmqEx2bxTiiT8dAyvP7bnnNf9tJu9C3c72JMnbYkUO7sbPdyVly45Fc+4t45/ylRwxkyY1HJpW3Q+dWfHHtaMpKS/jw0oM4cmg3IPx5LEWlPIARwCJjzGJjTCXwFHB0xDJ5wnU/jwDKjaqR8tPw79A5fePrFa/TO2MjnVTvcrPSEm44dmjaMib/qX6f8EJUBNlOI21aZr3yJfY1SmXic7vWh+/cNas6/bLfgM78br/0vWw3YiJHNhr3mK9lU8ukGdQM8upaQ7Oy+um7+Vo4XGzKozvwjeP7cjutDhE5W0RmiMiM1atXB1Lp1uoaPlz0QyBlOQlmmm0Qoxf/x/gJY/2vMeX+K3ChNqCX7Zhh3enarnnaPM6eYYzRg90bzyAWTPrl3T8fwAu/38v/gXUNbHqzVZQrY/521EAGdfO/cC6m8NI1ntu0apryt2Il1bUKW4UUm/LIiDHmAWNMuTGmvHNn73btdFz76uec8uBHfPbthhzkSk4rxB6tV5x21Uzn0b5lU7ZzaazDCoyYfFz8kU1K/T32sbAgLdLszDey7zb+BcuBbu1asNv22QdKjF2C1CMPb+UcPDD4KcbOuv10jpwx4546e8+432LlDMpyNfcz54zMmCfbzc1y7f8l+mryRbEpjxWA03vZw04LlYXfbwRgXUVloOUWwowJsBr/G30ufipzNMBe2pn2LdP3+LytMI+mP/zLEdvzh4P7c8VRlnvt3P13SMrz6JkjmDPu0LrvUd7adA7+mCKtUxop5HQ3s+b/pPzc8/pzgz0DVua7987sR8n2fc72uOfO3YtP/nZInRmsvrysivNNsSmPj4H+ItJHRJoCJwMvh11pEC9NoY88Dt+5W6jlt06Yugrh7ecRFH07teKk8p40LSvhDwcPoEOrpiy58Uh+tUevuHwili+hbfPMYcbzoQAvPiR5jUx9/db/pRl9Ht7qCoTYzdAAACAASURBVKOhyrbMoML+DMzCZJYNuT4Jw3t1oEMaM5w6zB0YY6qB84E3gc+Bp40x88Oud6odTiSXm+FqCy8Q7SE5btARRHvobbZVfrXH5Ev256bj0zvWE8nlWlz188H07dSK7X0uoLviqEFx06Gdje8VRw3iodPqfU4x8ep8HqlGHl6Vhx9BPRKE2SoXnj1nJO//5YDcCgFapTFzAlx3zM706dSKzq2b5VyXk3w1K0WlPACMMeONMQOMMTsYY66LWh43qmpqk9I2V9YkpUVhBgiCFtmEBnedquyviHyvSvaK+1Rd//d2r36dmHzJ/r5Dr585qg9fXX9Eyt8OcvFLFMI6jxjBBYKM+RBST502Bt77c3rF0KpZGT06eH/WWqZQEpkU34E7deHtS/avmwFXbBSn1BHhtTfW/7LXmfrVj3Fpx/9zalK+YF6a3HvjgndZDh64LRccGB9XJ1sJOrby1+M6aKcuPHduFjOMlDoSzWZep+rmM/ZW1u9Fmgcxdp61xgTeCXnlglGuG3VFFs1ZfR7FzdSvMk/tfXbm8jxIEhwlAg+etjtDerTLnDmBo3eLX/VbViIcvWt9mtf3bHiEW7GmIl2MrihI13YkypfSbOX4/PnVozlueA93J3pIDVXM4f3rPXtlyFlPvUnO+t+54VUszWtkZj/s0Lk1p/qQs6GgyiMk7py8iB82bY1aDE+kcnkkbhiUKhy1lx7WKSO25y+jd2KwPVXyZ7tsl3MHKdfIsI+csTsPn577GhSnFKfuuT3lvTpwyh7bp8wfJYm3KpVpxZkv3RTlsOjStjlLbjzS12rx2oSpry+dXx8tWepGHsHJ2NhR5eEDvw7bKV8Es0gxHcGEoXZvQPxGmwXYs6/1sj84ppwvrz2c+VcdVlfHufvvwFmjrJhPiQonG2f4vv07+T7Gyf47bsuBOwVrjtm2TXOePXcvOrcJ1gnqBy9O5tgK9W1TyFkoU3X94IwVBcStuo71M4ptY7Bs0BXmBYjf527uivWMn7syHGHygFdHnvOyxBbglZUKTctKaNUs9Rx0ZyPndm0/vuzgtPVmaiSdTtIwKZT1On7o1LoZNx8/lH+fvrvr78XYxsZEdrsdsXAs+Rx5iEhcoMOGRvLkeyUwHvlwCY98uMQ1mFlQBPUuZNsAJo5Oqmssifyu4nYj6N77xD/uy7I1FYGWWYh4uZMlImk3NipC3eEwWyVfgTB9HqkwxtQFOswn+erLqPLIA2GOPjq1zl+snsT37p1L9qdTm2b85dk5dWk1dteuLIM/wpD/aaL9u7Shf5c2mTP6xMt5FNzgJIM8xTjyqI/blfxTTKEU5XkVKKo88sB/P1oaWtmJZqF80tslbHd1rbXGxW3/ZwjWHltM9utCEzXzXIMCE9gDdSMPl2esbvOrQrsRRYz6PHyQ7WO3aWvyAsFCo1XT0qSQ24kvoZfec7U98igtyfxoxW8olbnsQiXddXn1glEZF6UFjZf7lMlMWYz3I90K83qzVf7kiQpdYV6AZNtr2VpV+MpDRLjv1OFp83g5/UPtxWRuUXT9lpWJQnFUp5Nj5+7tCnJlfKYrF2Ubm+2z4QyMmIhzkaASDKo8AmDT1moeev/rlL8vWLUxtLrDnJaXTdv8+/378emVh7Jt2/TKQwkPL89Epn3e3TpKbodEtojaRZZDB1kj5wEufi1pTCOPPHWqPCkPEblIRNqKxUMiMktEDs18ZMPC7bnbUlXDzle+yTWvfpZ3eQoF55qAkhJJue2sM69I4YwcGiOZLr3bs54YTdhLOZ5kycI86sZxw3uw4JrRrht5qc8jeLyOPM40xmwADgU6AL8GbgxNqiIi3YjDjf13DGaDqkycvlfvnMsIM2Jqpp6vt7K0IXDFk88j/e9ul7ZF01JG9cttYWZQpBI/VVBJNVsFj1flEbtXRwCP22HQG1+30eW5c4uWm45HzhgRkDDpaZum95+O9/9ygO+Q4H7ex/o9FxLLCP6lLqSRTRSipLukmUxbUSnm8l4d2L5j5j3v/d5bdZgHj1flMVNEJmApjzdFpA2QHHe8ETH1qx/pPfY15q5YH6kcQTdKPTq0rAtZEWrj67Fo53a3hUjvbQrQGe7h2maaquu1jU2nhLIZpTx77l6hhCiPxUFrGsDiVcXC65U8CxgL7G6MqQCaAmeEJlWB4rTtv/Pl9wBMW/xjquyhcI1L6Oeo8dNJTdWjTVWE31GQe505F5HEgTttC8CYkb2DLzwgnEqkT8KanIwdA4/XLF1MsgLoe9TRp1MrLj5kAA+MST+jsCFQECvMRWRYQlLfQjIFREnMhrq1ujAHYE1yjDjrRhDRVeviDyW8/tk08FE+i0EotXwy8Y/7YrD2moHsRx5h7uboZ88Qv7deRLjwoP6ZMzYkQjbRZRp53Gr/3QNMAx4A/gV8ZKc1aJ6avox3v6yPjGsM9B77Ghc/PZvSHBquTNtTpiOxVud3525yv9mnb9Z11JWdUFnrFKvZYw2KJyd9QFuFQv0o5vIjB/LB2ANzL9AHsZHH7r29hwzPF26Xtqy0JC7eWOZFgu4tj5eJDv84eVfPeZ1cdkRhmyiLhYKIqmuMOcAYcwCwEhhujCk3xgwHdgNW5EPAKBn7/FzGPDw9Kf35WSs8hHdIzVt/2j/7gx0MSQi65tzfIpdRQqoOi9sUSCexcOxeCPLx7tOpFd3btwiwxMzsO6Azi68/IquNsfJFLua6VId62SI39hwm6o7DBruPLNSYERIhX1evPo8djTFzY1+MMfOABt1NcI44YjhfxpIctEfXDKuv0xF70YZ0b8ez545MevEeHFMeWBhoZ9GDt2ubZDeP4cvn4XMsvUcfa0e5Ns3KGFpgDXUuz0CYBGHOS3VPE/frduvlltmhaYKYjq0ULl6j6s0VkQeB/9jffwXMSZO/6HEbcTjJxWwVBK2blcVtdhPj4AD2mnYzWXT1tGI88zVJFX8olVJp17JJqCHt802YM2BvOX5oVht4uXHjcUO46KnZSekDu7XlpdnfpjzuwTHlDr9WPEGee7Et1+jevgUr1m3OT2V5apq8jjxOB+YDF9l/n9EIZ1s5iarXmbwaN70c+w7IflGif/2Y+Y1O5TBXcueE8p7079ImkCt79K7dXdPP3qdvSsd2x1ZNOXhQl7rOhw48LN78w77RbAoVscMcESkFXjfG3G6MOcb+u90YsyVc0QoP573w+mL0TWHquf2kXXKSxUv9n119GA+fVr9H9wnDe3gq2+8zd80vdubE8h6+tnRNGnnk8KCnuhaNve1yuy4xR3+2lJQIe++wTdo89bdS4vwcue4578QtflWhsmPXNnRoZe27k2mfm2Iio9nKGFMjIrUi0s4YE+2KuIhxmnO8mq3aNK+/xM6X55jdevDH/33qW4ZYtfX7NaemZdNc9/qoL71fl9TO8i5tm3Pz8d6UYRjmhmIzYeQLt+ty36nD2LilOqdyvYZzF4F7ThlGRVUNt034kt8fsAOvz1uVU90x7j5lNw68dUrGfJMu3o+2zQtj26L//maP/E7xDllPeb2qm7D8HhOBn2KJxpgLQ5GqCLjh9QWe8vXu1IpPl1s695Xzo93P2Otoya3RueTQHQORwRkYUck/zcpKadY6t/U6mUOX1IdGLystoW1pCeN+PjinOhNp09xb+J1+26afIZhP9s53XLCQO1Velcfz9p/igS5tm/Hdhq0AXPXzwXUOxkHbtc257FjokKE9w5t5VOeXcDTwQexJDk7FlLBIMJDSlUKgPn6Zvx6Cn2egVbPcF6wqueFJeRhjHg1bkIbE9h1b1imPIFZlOxnQpQ2vXjCKnbraNl8f72chOKndFFO+6ixWJv5xXyp8BuCMkrrIySGFkZp/1WEBmGSVXPF0B0SkP3ADMAiom7NpjMl9GXMRMXnB957yORvpEhHGjOwVqLNw5+7ZjToKyVQU5DTOQjqvMOhfRM5hSL+XeBC0siMd7N1vG04ZkbzHiJIfvKrvfwNXArcDB2BN02104Smf+vgbbxkd70yJCFcf7R7M8LYTd+Hip+ud5nv27ci0xWtyETG9WH7jAYUhhHq3QydqZZrCMpmSbMX972/2zPJI79xx0q6eVtUXJAWywryFMeYtQIwxS40x44CGs3IrYCTF50QOG9w17rvbor9M9AgjLEeIDXwUZqtCoBBMhvmibp1HxHIEwS92687onbtmzliIRL3Ow2ariJQAC0XkfBE5BiicaQwFhjMsQ7pGMvG3VOE/0h3jtjVomqN95A0naq1J6UzN/knXwUxh4jU8SWt7Km0DWgLRKPCqPC4CWgIXAsOBU4HTsq1URG4RkQUiMkdEXhCR9o7fLhWRRSLyhYgc5kgfbactEpGx2dYdBl3aNov77nxn0jXAzgb0kTN2z2oTnDBWuuejLY5dlnf/74AQKwmv6EImKmX6kL0gdeQO21BWIpw5qk9Snnt/NYxrjo6ftvuvMeX89Yidii7MfWPHq89jjTFmE9Z6jyDCkkwELjXGVIvITcClwF9EZBBwMjAY2A6YJCID7GPuAQ4BlgMfi8jLxpjPApAlZ245fhdmLFnDnZMXAd5NMtl07HMZDXg9dEj3dsxZvp4OLbPbyjYd29lmtt7bWKOs2Gy0IBzmr104qmDt07FRZTGtjPbD5UcOZLftOwCwbZvmLHJsD+DkiCHdktK6tWvB2fvuEKp8jYl8mYS9dnUfFpGvROQpETlPRIbkUqkxZoIxJrbMdRoQi5txNPCUMWarMeZrYBEwwv5bZIxZbIypBJ6y8xYENcaw3471YR+ysW8bvHWUs3kuLj9yIE//bmRc2tn7pp4od8XPBvHSeXvTa5vMZjS/HDKoC0/+ds+6vT+CfNAHb9eOHRxh4wd2tdbV5DtcuxsHD+rCqxeM4oRybyFiciGf/qRjh1kxsA4amHtATqW48KQ8jDH7YYVgvwtoD7wmIkFNCzoTeN3+3B1wTmlabqelSk9CRM4WkRkiMmP16uSw6mFQXWMY3qsDj545wpbB23Fe8k2/7KAcJLP4zT59GdGnY0JaskkhRrOyUnYJKDqrGyN32CYvgSXPGtWHl87bm5EZYjHli527t4t098MwGNy9HUtuPNKTv05pWHhd5zEK2Mf+aw+8CryX4ZhJgNs0hcuMMS/ZeS4DqoH/+pA5LcaYB7B2PKS8vDx06+/wXh3Yd0CnWN2Ad9NS0gjF5bBmpQn7J+TQ9sQO/eWI7Qtu9o+fG/XgmHI6tGrK3ZMXps1XUiKhKkGl0bqVCpp8+by8+jzeAWZiLRQcb5uO0mKMOTjd7yJyOnAUcJCpD5azAujpyNaD+h0LU6VHymNnjqibYuslWKETT4ogwLczVt/Abm1C3YvaD9mcXmzPkuuOGcLtE79kVL/sw843RGIdgzYFEhBQaZh49Xl0Aq4GRgJviMgkEbkm20pFZDTwZ+DnxpgKx08vAyeLSDMR6QP0B6YDHwP9RaSPiDTFcqq/nG39QdLKsa+3M+hfqi03ncQ1nGna8hfP29txTPbapEcHazZL59bNMuTMP5mD7SWzXfsW3HLCLlnNUmvItGhaylU/H8wz54zMnFlpcOTLMuo1ttU6EVmM1fvvAewF5DIV526gGTDRNvFMM8acY4yZLyJPY202VQ2cZ4ypARCR84E3gVLgYWPM/BzqDwXnyOOeU4axpbo2bX6nectgXBWDCOzasz2d2zRj9catOcn323360q9zaw4auC2rN+VWVlA0NB9AoXCaPSEhF57//V5sY+9DUQg8OKacNRUZjR5KnvDq81gMLADeB+4DzvBiukqFMaZfmt+uA65zSR8PjM+2znxQv4+BUFZaQusMkWizaTZzaWtLSySQbWrDoDCMaI2Tl8/fm85tkkejw+ypt+no1s7L9sTBEOSzO+ni/UKLzPu7ffty/7uLQym7kPBqFO1njEnfjVZS7t2cCi+KIKx+eaE4zAtDisbN0B7ZTSoYM7JX0Y4cw9zn44KD+nP/u4sj26o6X3hWHiJyH9DFGLOziAzF8ldcG6JsRYfv2VZF+uIFSbsWTdh/x85p150ohYk+ve60alrKBQf248ihyQsi80HLJqWUCPz1yIGh1uNVefwL+D/gfgBjzBwReQJQ5eGgbhOcLN4qY9yPa+gKpqREeOSMEVGLoSiBISL8KaCdN7OhrLSExTeEH7fWq/JoaYyZntCQ5bYRcoMkt2ii+VQTsZlhnVoXjkNUUYqJf5++e86TWIoZr8rjBxHZAbt1FJHjgZWhSVWkmBxGHk5aNCllc1WNXWa04a1/t29fmhVovChFiZIDdto2c6YGjFflcR7Wqu2dRGQF8DXwq9CkKlLqHea5NfW/338Hbp34pWvZQeFVxkuPCNduqihKceJ1ncdi4GARaYW1sLACa6He0hBlKzpyGXk418g5j9cprIqiFCJpFyKISFt7f427ReQQLKVxGla02xPzIWAxEfMjeN0EJ/7YeqWhmxspilLoZBp5PA6sBaYCvwUuwzK/H2OMmR2ybAXPbxI2u9nFni9/zG6uAX8z4mZKUkWiKEohkkl59DXGDAEQkQexnOTbG2O2hC5ZEXD4kPigwT07tmTJjblPkXPTFw17wq6iKMVGpohyVbEPdoyp5ao4nOShSdeRh6IoBUimkccuIrLB/ixAC/u7AMYY0zZU6QqcINfvZRNVNhfatrBu/XkHpAwzpiiKkpK0ysMYoxP80xCE7ti9dwc+XrLWKs+lwHYJ+4gHpWOalZUGYmJTFKVxohsh5EAQoUPatUhe4a1OcqWQ6drO2hO+Z8eWEUuiRIluNZYDQXs81CmuFAOHDe7CY2eOYFS/TlGLokSIKo8cCNTnEVxRihIqIsK+A3Tr38aOmq1SUOohFn8Qe2K4LQwslP3FFUVRUqEjjxTUenA8BB0tvdc2rQDY3sWWnK6um48byi49s9vQR1EUJRtUeaQgCqf1scO606NDC0b06cjFT3/qWZ4Td+8ZsmSKoijxqNnKBa9rLoIYeezT33I69unUChFhj77bNPgNoBRFKX505OFCrcdRRxA+j1/v2YvRg7uybdvmOZelKIqSL3Tk4UKNR+0RxABBRFRxKIpSdKjycMGLsxyCd5ino29ny5nepFRNWoqiRI+arVzwPPIIeVmfUzndf2o5s5atZZvWzUKtU1EUxQs68nChOoXy+MfJu8Z9D3Pk8e8zdmfKJQfUfW/Xskmj3zNZUZTCQUceLtR6HHl4NW9lwwE7qqJQFKVw0ZGHCzXG4LbAPFFXVFbX5kcgRVGUAkOVhwudWjdj8Q2Zw5VvVeWhKEojRZVHDujIQ1GUxooqjxzYpnXyXhyKoiiNAVUePkiMdrtT10a9C6+iKI2YSJWHiPxJRIyIdLK/i4jcKSKLRGSOiAxz5D1NRBbaf6flQ74OCVvAKoqiKBaRTdUVkZ7AocAyR/LhQH/7bw/gPmAPEekIXAmUY+2bNFNEXjbGrA1TxtISHZgpiqK4EWXreDvwZ+I30TsaeMxYTAPai0g34DBgojFmja0wJgKjwxawfcLIQ/cWVxRFsYhEeYjI0cAKY8ynCT91B75xfF9up6VKdyv7bBGZISIzVq9enZOcndQhriiK4kpoZisRmQR0dfnpMuCvWCarwDHGPAA8AFBeXp7TWKFWZ+IqiqK4EpryMMYc7JYuIkOAPsCn9qZHPYBZIjICWAE4t8XrYaetAPZPSH8ncKET6NZeQ6UriqK4kXezlTFmrjFmW2NMb2NMbywT1DBjzCrgZWCMPetqT2C9MWYl8CZwqIh0EJEOWKOWN8OW9bpjhiTIHnaNiqIoxUGhBUYcDxwBLAIqgDMAjDFrROQa4GM739XGmDVhC9O6WaFdHiVszj+gH7OWhTqJT1EaBJG3jvboI/bZAOelyPcw8HCexFIaKZcctmPUIihKUaALGRRFURTfqPLwgbo8FEVRLFR5KIqiKL5R5aEoiqL4RpWHD4zO1VUURQFUeSiKoihZoMpDURRF8Y0qD0VRFMU3qjwycPXRg+s+q8dDURTFQpVHBkYPdgsMrCiK0rhR5ZGBbdtqZF1FUZREVHn4Qe1WiqIogCoPRVEUJQtUeSiKoii+UeWhKIqi+Cby/TyKgRd+vxcbtlSzav3mqEVRFEUpCHTk4YHdtu/AfgM6x6VdcdSgiKRRFEWJHlUeWXDC8B6cOapP1GIoiqJEhiqPLBCJWgJFUZRoUeWhKIqi+EaVh6IoiuIbVR4+0L2gFEVRLFR5+CCmOwR1eiiK0rhR5aEoiqL4RpWHoiiK4htVHlmgU3UVRWnsqPLwgTrMFUVRLFR5KIqiKL5R5aEoiqL4RpVHFqjPQ1GUxk5kykNELhCRBSIyX0RudqRfKiKLROQLETnMkT7aTlskImOjkVpRFEWBiPbzEJEDgKOBXYwxW0VkWzt9EHAyMBjYDpgkIgPsw+4BDgGWAx+LyMvGmM/yKbfRTcwVRVGA6DaDOhe40RizFcAY872dfjTwlJ3+tYgsAkbYvy0yxiwGEJGn7Lx5VR71qN1KUZTGTVRmqwHAPiLykYhMEZHd7fTuwDeOfMvttFTpSYjI2SIyQ0RmrF69OlChdaquoiiKRWgjDxGZBHR1+ekyu96OwJ7A7sDTItI3iHqNMQ8ADwCUl5drc68oihICoSkPY8zBqX4TkXOB540xBpguIrVAJ2AF0NORtYedRpp0RVEUJc9EZbZ6ETgAwHaINwV+AF4GThaRZiLSB+gPTAc+BvqLSB8RaYrlVH85EsnRqbqKoihROcwfBh4WkXlAJXCaPQqZLyJPYznCq4HzjDE1ACJyPvAmUAo8bIyZH43oiqIoSiTKwxhTCZya4rfrgOtc0scD40MWTVEURfGArjD3gXrfFUVRLFR5+MGeq6suD0VRGjuqPBRFURTfqPJQFEVRfKPKIwt0qq6iKI0dVR6KoiiKb1R5KIqiKL5R5eEDnaqrKIpiocrDB2Ul1uVqWloasSSKoijRElV4kqLk+OE9WPrjT5x/YL+oRVEURYkUVR4+aFpWwqVHDIxaDEVRlMhRs5WiKIriG1UeiqIoim9UeSiKoii+UeWhKIqi+EaVh6IoiuIbVR6KoiiKb1R5KIqiKL5R5aEoiqL4RoxpuBGbRGQ1sDSHIjoBPwQkTpAUqlygsmVDocoFKls2FKpc4F22XsaYzukyNGjlkSsiMsMYUx61HIkUqlygsmVDocoFKls2FKpcEKxsarZSFEVRfKPKQ1EURfGNKo/0PBC1ACkoVLlAZcuGQpULVLZsKFS5IEDZ1OehKIqi+EZHHoqiKIpvVHkoiqIovlHl4YKIjBaRL0RkkYiMjaD+niLytoh8JiLzReQiO32ciKwQkdn23xGOYy615f1CRA4LUbYlIjLXrn+GndZRRCaKyEL7/w52uojInbZcc0RkWIhy7ei4LrNFZIOI/CGqayYiD4vI9yIyz5Hm+zqJyGl2/oUiclpIct0iIgvsul8QkfZ2em8R2ey4dv90HDPcfg4W2bJLSLL5vn9hvL8pZPufQ64lIjLbTs/bdUvTVoT/rBlj9M/xB5QCXwF9gabAp8CgPMvQDRhmf24DfAkMAsYBl7jkH2TL2QzoY8tfGpJsS4BOCWk3A2Ptz2OBm+zPRwCvAwLsCXyUx3u4CugV1TUD9gWGAfOyvU5AR2Cx/X8H+3OHEOQ6FCizP9/kkKu3M19COdNtWcWW/fCQrpmv+xfW++smW8LvtwJX5Pu6pWkrQn/WdOSRzAhgkTFmsTGmEngKODqfAhhjVhpjZtmfNwKfA93THHI08JQxZqsx5mtgEdZ55IujgUftz48Cv3CkP2YspgHtRaRbHuQ5CPjKGJMuukCo18wY8y6wxqVOP9fpMGCiMWaNMWYtMBEYHbRcxpgJxphq++s0oEe6MmzZ2hpjphmr5XnMcS6BypaGVPcvlPc3nWz26OFE4Ml0ZYRx3dK0FaE/a6o8kukOfOP4vpz0DXeoiEhvYDfgIzvpfHu4+XBsKEp+ZTbABBGZKSJn22ldjDEr7c+rgC4RyOXkZOJf5KivWQy/1ykKGc/E6pnG6CMin4jIFBHZx07rbsuSL7n83L8ortk+wHfGmIWOtLxft4S2IvRnTZVHASMirYHngD8YYzYA9wE7ALsCK7GGyvlmlDFmGHA4cJ6I7Ov80e5RRTb/W0SaAj8HnrGTCuGaJRH1dXJDRC4DqoH/2kkrge2NMbsBFwNPiEjbPItVkPcvgV8S31nJ+3VzaSvqCOtZU+WRzAqgp+N7Dzstr4hIE6yH4b/GmOcBjDHfGWNqjDG1wL+oN7PkTWZjzAr7/++BF2wZvouZo+z/v8+3XA4OB2YZY76z5Yz8mjnwe53yJqOInA4cBfzKbmywTUI/2p9nYvkSBtgyOE1bYT5vfu9fXu+riJQBxwL/c8ic1+vm1laQh2dNlUcyHwP9RaSP3Ys9GXg5nwLYNtSHgM+NMbc50p3+gmOA2MyPl4GTRaSZiPQB+mM55oKWq5WItIl9xnK0zrPrj83OOA14ySHXGHuGx57AesdQOizieoFRX7ME/F6nN4FDRaSDba451E4LFBEZDfwZ+LkxpsKR3llESu3PfbGu0WJbtg0isqf9rI5xnEvQsvm9f/l+fw8GFhhj6sxR+bxuqdoK8vGs5eLpb6h/WDMSvsTqMVwWQf2jsIaZc4DZ9t8RwOPAXDv9ZaCb45jLbHm/IICZLynk6os1e+VTYH7s2gDbAG8BC4FJQEc7XYB7bLnmAuUhX7dWwI9AO0daJNcMS4GtBKqw7MdnZXOdsHwQi+y/M0KSaxGWvTv2rP3TznucfZ9nA7OAnznKKcdqyL8C7saOVhGCbL7vXxjvr5tsdvojwDkJefN23UjdVoT+rGl4EkVRFMU3arZSFEVRfKPKQ1EURfGNKg9FURTFN6o8FEVRFN+o8lAURVF8o8pDUTwgIjUSH7U3bbRWETlHRMYEUO8SEemUazmKEjQ6VVdRPCAivRcQOQAAAiRJREFUm4wxrSOodwnWXPwf8l23oqRDRx6KkgP2yOBmsfZomC4i/ez0cSJyif35QrH2W5gjIk/ZaR1F5EU7bZqIDLXTtxGRCWLtzfAg1qKuWF2n2nXMFpH7RaTU/ntERObZMvwxgsugNEJUeSiKN1okmK1Ocvy23hgzBGvF8B0ux44FdjPGDAXOsdOuAj6x0/6KFZ4b4ErgfWPMYKzYYdsDiMhA4CRgb2PMrkAN8CusgIHdjTE72zL8O8BzVpSUlEUtgKIUCZvtRtuNJx3/3+7y+xzgvyLyIvCinTYKK4wFxpjJ9oijLdamQ8fa6a+JyFo7/0HAcOBjK5wRLbCC3b0C9BWRu4DXgAnZn6KieEdHHoqSOybF5xhHYsUTGobV+GfTaRPgUWPMrvbfjsaYccbauGcX4B2sUc2DWZStKL5R5aEouXOS4/+pzh9EpAToaYx5G/gL0A5oDbyHZXZCRPYHfjDWPgzvAqfY6YdjbQkKVpC740VkW/u3jiLSy56JVWKMeQ64HEtBKUroqNlKUbzRQkRmO76/YYyJTdftICJzgK1YIeGdlAL/EZF2WKOHO40x60RkHPCwfVwF9eGzrwKeFJH5wIfAMgBjzGcicjnWLo4lWNFdzwM2A/+20wAuDe6UFSU1OlVXUXJAp9IqjRU1WymKoii+0ZGHoiiK4hsdeSiKoii+UeWhKIqi+EaVh6IoiuIbVR6KoiiKb1R5KIqiKL75fyYfyVWxLkIAAAAAAElFTkSuQmCC\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('MAD-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": "633d6c51-b781-4699-d87c-afcfcad62e7d"
      },
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "Text(0.5, 1.0, 'MAD-DQN')"
            ]
          },
          "metadata": {},
          "execution_count": 13
        },
        {
          "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": [
        "signal = smooth(sum_rewards, window_len=600, window='hanning')\n",
        "plt.plot(signal)\n",
        "plt.xlabel('Episodes')\n",
        "plt.ylabel('Rewards')\n",
        "plt.title('MAD-DQN')"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 14,
      "metadata": {
        "id": "LNEk4vS1c8W5"
      },
      "outputs": [],
      "source": [
        "# save to npy file\n",
        "np.save(\"MAD-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
}