{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "67576132",
   "metadata": {},
   "outputs": [],
   "source": [
    "from math import inf\n",
    "import numpy as np\n",
    "import scipy\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import torch.optim as optim\n",
    "import gymnasium as gym\n",
    "import pandas as pd\n",
    "from datetime import datetime\n",
    "from sklearn.base import BaseEstimator\n",
    "from realkd.boosting import WeightUpdateMethod\n",
    "from pandas import qcut\n",
    "from realkd.rules import SquaredLoss, AdditiveRuleEnsemble, Rule\n",
    "from realkd.search import Context\n",
    "\n",
    "import warnings"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "8b8f01e2",
   "metadata": {},
   "outputs": [],
   "source": [
    "# warnings.filterwarnings('ignore')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "db545637",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "cuda\n"
     ]
    }
   ],
   "source": [
    "env = gym.make('MountainCar-v0', render_mode='rgb_array')\n",
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "print(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "7aa6c889",
   "metadata": {},
   "outputs": [],
   "source": [
    "column_names = ['p', 'v']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "24f75485",
   "metadata": {},
   "outputs": [],
   "source": [
    "np.random.seed(463647)\n",
    "torch.manual_seed(350907)\n",
    "num_rules=12\n",
    "action_space=3\n",
    "reg=10.0\n",
    "nn_actor_train_iterations=100\n",
    "rule_actor_train_iterations=1000"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "84732b72",
   "metadata": {},
   "outputs": [],
   "source": [
    "def orthonormalization(Q):\n",
    "    n, k = Q.shape\n",
    "    O = np.zeros(shape=(n, k))\n",
    "    q = Q[:, 0]\n",
    "    O[:, 0] = q / (norm(q) + 1e-6)\n",
    "    for i in range(1, k):\n",
    "        O_i = O[:, :i]\n",
    "        q = Q[:, i]\n",
    "        q_orth = q - O_i.dot(O_i.T.dot(q))\n",
    "        O[:, i] = q_orth / (norm(q_orth) + 1e-6)\n",
    "    return O"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "973f1f9b",
   "metadata": {},
   "outputs": [],
   "source": [
    "def softmax1(action, values, other_values, location):\n",
    "    all_values = np.insert(other_values, location, values, axis=1)\n",
    "    exps = np.exp(all_values)\n",
    "    res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
    "    return res\n",
    "\n",
    "def norm(x):\n",
    "    \"\"\"\n",
    "    Calculate the L-2 norm of a vector\n",
    "    :param x: the vector whose L-2 norm is to be calculated\n",
    "    :return: the L-2 norm of the vector\n",
    "    \"\"\"\n",
    "    return (x * x).sum() ** 0.5\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "5b0af6ac",
   "metadata": {},
   "outputs": [],
   "source": [
    "class PiLoss:\n",
    "    _instance = None\n",
    "\n",
    "    def __new__(cls):\n",
    "        if cls._instance is None:\n",
    "            cls._instance = super(PiLoss, cls).__new__(cls)\n",
    "        return cls._instance\n",
    "\n",
    "    @staticmethod\n",
    "    def __call__(action, values, advantage, other_values, current):\n",
    "        \"\"\"\n",
    "        :param action: the action chosen\n",
    "        :param values: the values provided by current model\n",
    "        :param other_values: the output by other models\n",
    "        :param current: the action represented by the current model\n",
    "        \"\"\"\n",
    "        sm = softmax1(action, values, other_values, current)+1e-6\n",
    "        return -np.log(sm) * advantage\n",
    "\n",
    "    @staticmethod\n",
    "    def g(action, values, advantage, other_values, current):\n",
    "        return np.where(action == current, -advantage * (1 - softmax1(current, values, other_values, current)),\n",
    "                        advantage * softmax1(current, values, other_values, current))\n",
    "\n",
    "    @staticmethod\n",
    "    def h(action, values, advantage, other_values, current):\n",
    "        sm = softmax1(current, values, other_values, current)\n",
    "        return advantage * sm * (1 - sm)\n",
    "\n",
    "    @staticmethod\n",
    "    def __repr__():\n",
    "        return 'pi_loss'\n",
    "\n",
    "    @staticmethod\n",
    "    def __str__():\n",
    "        return 'pi_loss'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "67942365",
   "metadata": {},
   "outputs": [],
   "source": [
    "loss_functions = {\n",
    "    'pi_loss': PiLoss()\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "693287f4",
   "metadata": {},
   "outputs": [],
   "source": [
    "def loss_function(loss):\n",
    "    \"\"\"Provides loss functions from string representation.\n",
    "\n",
    "    :param loss: string identifier of loss function loss function\n",
    "    :return: loss function matching corresponding to input string (or unchanged input if was already loss function)\n",
    "    \"\"\"\n",
    "    if callable(loss):\n",
    "        return loss\n",
    "    else:\n",
    "        return loss_functions[loss]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "ae0af97c",
   "metadata": {},
   "outputs": [],
   "source": [
    "def calc_risk(loss, action, rules, states, reg, advantage, others, current):\n",
    "    weights = np.array([rule.y for rule in rules])\n",
    "    risk = sum(loss(action, rules(states), advantage, others, current)) + reg * sum(weights * weights) / 2\n",
    "    return risk"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "d06dc96b",
   "metadata": {},
   "outputs": [],
   "source": [
    "class FullyCorrectiveA2C:\n",
    "    def __init__(self, loss='pi_loss', reg=1.0, solver='L-BFGS-B'):\n",
    "        self.loss = loss_function(loss)\n",
    "        self.reg = reg\n",
    "        self.solver = solver\n",
    "\n",
    "    @staticmethod\n",
    "    def get_risk(loss, action, q_mat, reg, advantage, others, current):\n",
    "        def sum_loss(weights):\n",
    "            return sum(loss(action, q_mat.dot(weights), advantage, others, current)) + reg * sum(weights * weights) / 2\n",
    "\n",
    "        return sum_loss\n",
    "\n",
    "    @staticmethod\n",
    "    def get_gradient(g, action, q_mat, reg, advantage, other, current):\n",
    "        def gradient(weights):\n",
    "            grad_vec = g(action, q_mat.dot(weights), advantage, other, current)\n",
    "            return q_mat.T.dot(grad_vec) + reg * weights\n",
    "\n",
    "        return gradient\n",
    "\n",
    "    @staticmethod\n",
    "    def get_hessian(h, action, q_mat, reg, advantage, other, current):\n",
    "        def hessian(weights):\n",
    "            h_vec = h(action, q_mat.dot(weights), advantage, other, current)\n",
    "            return q_mat.T.dot(np.diag(h_vec)).dot(q_mat) + np.diag([reg] * len(weights))\n",
    "\n",
    "        return hessian\n",
    "\n",
    "    def calc_weight(self, data, action, rules, advantage, other, current):\n",
    "        g = self.loss.g\n",
    "        h = self.loss.h\n",
    "        loss = self.loss\n",
    "        y = np.array(action)\n",
    "        q_mat = np.column_stack(\n",
    "            [rules[i].q(data) + np.zeros(len(data)) for i in range(len(rules))])\n",
    "        sum_loss = self.get_risk(loss, y, q_mat, self.reg, advantage, other, current)\n",
    "        gradient = self.get_gradient(g, y, q_mat, self.reg, advantage, other, current)\n",
    "        hessian = self.get_hessian(h, y, q_mat, self.reg, advantage, other, current)\n",
    "        if self.solver == 'GD':  # Gradient descent\n",
    "            w = np.array([r.y for r in rules])\n",
    "            old_w = np.ones_like(w) * (1.0 if len(w) - sum(w) > 1e-5 else 2.0)\n",
    "            i = 0\n",
    "            while norm(old_w - w) > 1e-3 and i < 50:\n",
    "                old_w = np.array(w)\n",
    "                if norm(gradient(w)) == 0:\n",
    "                    break\n",
    "                p = -gradient(w) / norm(gradient(w))\n",
    "                w += GoldenRatioSearch(sum_loss, old_w, p, gradient).run() * p\n",
    "                i += 1\n",
    "        elif self.solver == 'Line':\n",
    "            w = np.array([r.y for r in rules])\n",
    "            if norm(gradient(w)) != 0:\n",
    "                p = -gradient(w) / norm(gradient(w))\n",
    "                distance = GoldenRatioSearch(sum_loss, w, p, gradient).run()\n",
    "                w += distance * p\n",
    "        else:\n",
    "            w = np.array([r.y for r in rules])\n",
    "            w = scipy.optimize.minimize(sum_loss, w, method=self.solver, jac=gradient,  # hess=hessian,\n",
    "                                        options={'disp': False}).x\n",
    "\n",
    "        return w"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "2d0c04a8",
   "metadata": {},
   "outputs": [],
   "source": [
    "class ObjectFunction:\n",
    "    def __init__(self, data, target, predictions, loss, reg, advantage, other_values, current, rules=None):\n",
    "        self.loss = loss_function(loss)\n",
    "        self.reg = reg\n",
    "        predictions = np.zeros_like(\n",
    "            target) if predictions is None else predictions\n",
    "        g = np.array(self.loss.g(target, predictions, advantage, other_values, current))\n",
    "        h = np.array(self.loss.h(target, predictions, advantage, other_values, current)) + 1e-6\n",
    "        r = g / h\n",
    "        order = np.argsort(r)[::-1]\n",
    "        self.g = g[order]\n",
    "        self.h = h[order]\n",
    "        self.data = data.iloc[order].reset_index(drop=True)\n",
    "        self.target = target.iloc[order].reset_index(drop=True)\n",
    "        self.n = len(target)\n",
    "\n",
    "    def __call__(self, ext):\n",
    "        raise NotImplementedError()\n",
    "\n",
    "    def bound(self, ext):\n",
    "        raise NotImplementedError()\n",
    "\n",
    "    def search(self, method='greedy', verbose=False, **search_params):\n",
    "        from realkd.search import search_methods\n",
    "        ctx = Context.from_df(self.data, **search_params)\n",
    "        if verbose >= 2:\n",
    "            print(\n",
    "                f'Created search context with {len(ctx.attributes)} attributes')\n",
    "        return search_methods[method](ctx, self, self.bound, verbose=verbose, **search_params).run()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "eaec4413",
   "metadata": {},
   "outputs": [],
   "source": [
    "class OrthogonalBoostingObjective(ObjectFunction):\n",
    "    def __init__(self, data, target, advantage, other_values, current, predictions=None, loss=SquaredLoss, reg=1.0,\n",
    "                 rules=None,\n",
    "                 epsilon=1e-4, **kwargs):\n",
    "        super().__init__(data, target, predictions, loss, reg, advantage, other_values, current, rules)\n",
    "        self.rules = [] if rules is None else rules\n",
    "        self.loss = loss_function(loss)\n",
    "        self.reg = reg\n",
    "        self.epsilon = epsilon\n",
    "        predictions = np.zeros_like(\n",
    "            target) if predictions is None else predictions\n",
    "        g = np.array(self.loss.g(target, predictions, advantage, other_values, current))\n",
    "        self.n = len(target)\n",
    "        r = g\n",
    "        order = np.argsort(r)[::-1]\n",
    "        self.g = g[order]\n",
    "        self.data = data.iloc[order].reset_index(drop=True)\n",
    "        self.target = target.iloc[order].reset_index(drop=True)\n",
    "        if len(rules) != 0:\n",
    "            orth_basis = kwargs['orth_basis']\n",
    "            self.orth_basis = orth_basis[order]\n",
    "            self.g = self.g - self.orth_basis @ self.orth_basis.T @ self.g\n",
    "        else:\n",
    "            self.orth_basis = np.zeros(self.n)\n",
    "\n",
    "    def __call__(self, ext):\n",
    "        if len(ext) == 0:\n",
    "            return -inf\n",
    "        g_q = self.g[ext]\n",
    "        if len(self.rules) == 0:\n",
    "            h_q = self.h[ext]\n",
    "            return abs(g_q.sum()) / np.sqrt(h_q.sum())\n",
    "        length = self.fast_orth_norm(ext)\n",
    "        if length > 1e-4:\n",
    "            obj = abs(g_q.sum()) / (length + self.epsilon)\n",
    "        else:\n",
    "            obj = 0\n",
    "        return obj\n",
    "\n",
    "    def fast_orth_norm(self, ext):\n",
    "        deltas = self.orth_basis[ext]\n",
    "        length = len(ext)\n",
    "        okqi = abs(np.sum(deltas, axis=0))\n",
    "        q_para_norms = (okqi ** 2).sum()\n",
    "        q_orth_norms_sq = np.abs(length - q_para_norms)\n",
    "        return np.sqrt(q_orth_norms_sq)\n",
    "\n",
    "    def fast_para_norms_prefix(self, ext):\n",
    "        deltas = self.orth_basis[ext]\n",
    "        length = len(ext)\n",
    "        okqi = np.cumsum(deltas, axis=0)\n",
    "        q_para_norms = (okqi ** 2).sum(axis=1)\n",
    "        q_orth_norms_sq = np.abs(np.arange(1, length + 1) - q_para_norms)\n",
    "        q_orth_norms = np.sqrt(q_orth_norms_sq)\n",
    "        return q_orth_norms\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "626cceb7",
   "metadata": {},
   "outputs": [],
   "source": [
    "class GeneralRuleBoostingEstimator(BaseEstimator):\n",
    "    def __init__(self, num_rules, objective_function, weight_update_method, loss='squared', reg=1.0,\n",
    "                 search='greedy', max_col_attr=10,\n",
    "                 search_params=None, verbose=False):\n",
    "        if search_params is None:\n",
    "            search_params = {'order': 'bestboundfirst', 'apx': 1.0, 'max_depth': None, 'discretization': qcut,\n",
    "                             'max_col_attr': max_col_attr}\n",
    "        self.num_rules = num_rules\n",
    "        self.num_components = 500\n",
    "        self.objective = objective_function\n",
    "        self.objective_function = objective_function\n",
    "        self.max_col_attr = max_col_attr\n",
    "        self.weight_update_method = weight_update_method\n",
    "        self.loss = loss_function(loss)\n",
    "        self.reg = reg\n",
    "        self.weight_update_method.loss = loss\n",
    "        self.weight_update_method.reg = reg\n",
    "        self.verbose = verbose\n",
    "        self.search = search\n",
    "        self.rules_ = AdditiveRuleEnsemble([])\n",
    "        self.search_params = search_params\n",
    "        self.history = []\n",
    "        self.time = []\n",
    "\n",
    "    def set_reg(self, reg):\n",
    "        self.reg = reg\n",
    "        self.objective.reg = reg\n",
    "        self.weight_update_method.reg = reg\n",
    "\n",
    "    def fit(self, data, target, advantage, other_values, current, has_origin_rules=False, verbose=False):\n",
    "        if not has_origin_rules:\n",
    "            self.history = []\n",
    "            self.time = []\n",
    "            self.rules_.members = []\n",
    "            orth_basis = np.array([])\n",
    "        else:\n",
    "            q_mat = np.column_stack(\n",
    "                [self.rules_[i].q(data) + np.zeros(len(data)) for i in range(len(self.rules_))])\n",
    "            orth_basis = orthonormalization(q_mat)\n",
    "        num_components = 0\n",
    "        while len(self.rules_) < self.num_rules and num_components < self.num_components:\n",
    "            start_time = datetime.now()\n",
    "            # Search for a rule\n",
    "            scores = self.rules_(data)\n",
    "            obj = self.objective(data, target, advantage, other_values, current, predictions=scores,\n",
    "                                 loss=self.loss, reg=self.reg, rules=self.rules_, orth_basis=orth_basis)\n",
    "            q = obj.search(method=self.search, verbose=verbose,\n",
    "                           **self.search_params)\n",
    "            if hasattr(self.objective, 'opt_weight') and callable(getattr(self.objective, 'opt_weight')):\n",
    "                y = obj.opt_weight(q)\n",
    "            else:\n",
    "                y = 1.0  # np.random.random()\n",
    "            q_vec = q(data)\n",
    "            num_components += (1 + len(q))\n",
    "            if len(orth_basis) == 0:\n",
    "                basis = q_vec / norm(q_vec)\n",
    "                orth_basis = np.array([basis]).T\n",
    "            else:\n",
    "                basis = q_vec - orth_basis.dot(orth_basis.T.dot(q_vec))\n",
    "                basis = basis / (norm(basis) + 1e-6)\n",
    "                orth_basis = np.hstack((orth_basis, np.array([basis]).T))\n",
    "            rule = Rule(q, y)\n",
    "            if self.verbose:\n",
    "                print(rule)\n",
    "            self.rules_.append(rule)\n",
    "            # Calculate weights\n",
    "            weights = self.weight_update_method.calc_weight(\n",
    "                data, target, self.rules_, advantage, other_values, current)\n",
    "            for i in range(len(self.rules_)):\n",
    "                self.rules_[i].y = weights[i]\n",
    "            self.history.append(AdditiveRuleEnsemble(\n",
    "                [Rule(q=rule.q, y=rule.y) for rule in self.rules_.members]))\n",
    "            end_time = datetime.now()\n",
    "            self.time.append(str(end_time - start_time))\n",
    "        return self\n",
    "\n",
    "    def predict(self, data):\n",
    "        loss = loss_function(self.loss)\n",
    "        return loss.preidictions(self.rules_(data))\n",
    "\n",
    "    def decision_function(self, data):\n",
    "        return self.rules_(data)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "b50ce787",
   "metadata": {},
   "outputs": [],
   "source": [
    "# class Actor(nn.Module):\n",
    "#     def __init__(self, action_space, node=16):\n",
    "#         super(Actor, self).__init__()\n",
    "#         self.fc1 = nn.Linear(4, node)  \n",
    "# #         self.fc1_ = nn.Linear(node, node)\n",
    "#         self.fc2 = nn.Linear(node, action_space)\n",
    "\n",
    "#     def forward(self, state):\n",
    "#         x = F.relu(self.fc1(state))\n",
    "# #         x = F.relu(self.fc1_(x))\n",
    "#         x = self.fc2(x)\n",
    "#         return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "84b0ee0a",
   "metadata": {},
   "outputs": [],
   "source": [
    "def copy_rules(origin_rules):\n",
    "    rules = []\n",
    "    for n in range(len(origin_rules)):\n",
    "        rule = origin_rules[n]\n",
    "        rules.append(Rule(q=rule.q, y=rule.y))\n",
    "    return AdditiveRuleEnsemble(rules)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "bb3fcc8c",
   "metadata": {},
   "outputs": [],
   "source": [
    "class ActorRule():\n",
    "    def __init__(self, action_space=action_space, num_rules=num_rules):\n",
    "        self.action_space = action_space\n",
    "        self.model = [GeneralRuleBoostingEstimator(num_rules=num_rules, objective_function=OrthogonalBoostingObjective,\n",
    "                                                   weight_update_method=FullyCorrectiveA2C(), loss=PiLoss(),\n",
    "                                                   reg=reg, search='greedy', max_col_attr=10, verbose=False) for _ in\n",
    "                      range(action_space)]\n",
    "        self.best_model = [\n",
    "            GeneralRuleBoostingEstimator(num_rules=num_rules, objective_function=OrthogonalBoostingObjective,\n",
    "                                         weight_update_method=FullyCorrectiveA2C(), loss=PiLoss(),\n",
    "                                         reg=0.01, search='greedy', max_col_attr=20, verbose=False) for _ in\n",
    "            range(action_space)]\n",
    "\n",
    "    def __call__(self, state):\n",
    "        res = [self.model[i].rules_(state) for i in range(len(self.model))]\n",
    "        return np.array(res)\n",
    "\n",
    "    def predict(self, state):\n",
    "        res = [self.best_model[i].rules_(state) for i in range(len(self.model))]\n",
    "        return np.array(res)\n",
    "\n",
    "\n",
    "def update_rules(estimator: GeneralRuleBoostingEstimator, x, y, adv, other_values, current):\n",
    "    queries_lst = []\n",
    "    queries = {}\n",
    "    original_rules = copy_rules(estimator.rules_)\n",
    "    origin_risk = calc_risk(PiLoss(), y, original_rules, x, estimator.reg, adv, other_values, current)\n",
    "    for i in range(len(estimator.rules_)):\n",
    "        q_str = str(estimator.rules_[i].q)\n",
    "        if q_str not in queries:\n",
    "            queries[q_str] = estimator.rules_[i].y\n",
    "            queries_lst.append(estimator.rules_[i].q)\n",
    "        else:\n",
    "            queries[q_str] += estimator.rules_[i].y\n",
    "    for k in queries:\n",
    "        queries[k] = abs(queries[k])\n",
    "    min_weight_query = min(queries, key=queries.get)\n",
    "    rules = []\n",
    "    for i in range(len(queries_lst)):\n",
    "        q_str = str(queries_lst[i])\n",
    "        if q_str != min_weight_query:\n",
    "            rules.append(Rule(q=queries_lst[i], y=queries[str(queries_lst[i])]))\n",
    "    if len(rules) == 0:\n",
    "        rules = [Rule(q=queries_lst[0], y=queries[str(queries_lst[0])])]\n",
    "    new_rules = AdditiveRuleEnsemble(rules)\n",
    "    weights = FullyCorrectiveA2C(loss_function('pi_loss'), estimator.reg).calc_weight(x, y, new_rules, adv,\n",
    "                                                                                      other_values, current)\n",
    "    for i in range(len(new_rules)):\n",
    "        new_rules[i].y = weights[i]\n",
    "    estimator.rules_ = new_rules\n",
    "    estimator.fit(x, y, adv, other_values, current, has_origin_rules=True)\n",
    "    weights = FullyCorrectiveA2C(loss_function('pi_loss'), estimator.reg).calc_weight(x, y, estimator.rules_, adv,\n",
    "                                                                                      other_values, current)\n",
    "    for i in range(len(estimator.rules_)):\n",
    "        estimator.rules_[i].y = weights[i]\n",
    "    new_risk = calc_risk(PiLoss(), y, estimator.rules_, x, estimator.reg, adv, other_values, current)\n",
    "    if new_risk > origin_risk:\n",
    "        estimator.rules_ = copy_rules(original_rules)\n",
    "        print(\"origin\", origin_risk, 'new', new_risk, 'not updated')\n",
    "        # risk = sum(PiLoss.__call__(y, estimator.rules_[i].y))\n",
    "    return estimator"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "2824981f",
   "metadata": {},
   "outputs": [],
   "source": [
    "class Critic(nn.Module):\n",
    "    def __init__(self, node=16):\n",
    "        super(Critic, self).__init__()\n",
    "        self.fc1 = nn.Linear(len(column_names), node)\n",
    "        self.fc2 = nn.Linear(node, 1)\n",
    "\n",
    "    def forward(self, state):\n",
    "        x = F.relu(self.fc1(state))\n",
    "        x = self.fc2(x)\n",
    "        return x"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "6b6e2dd7",
   "metadata": {},
   "outputs": [],
   "source": [
    "gamma = 0.99\n",
    "actor = ActorRule()\n",
    "critic = Critic().to(device)\n",
    "critic_optimizer = optim.AdamW(critic.parameters(), lr=0.001)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "ae072cc5",
   "metadata": {},
   "outputs": [],
   "source": [
    "class ActorNet(nn.Module):\n",
    "    def __init__(self, hidden_dim=16):\n",
    "        super().__init__()\n",
    "\n",
    "        self.hidden = nn.Linear(len(column_names), hidden_dim)\n",
    "        self.output = nn.Linear(hidden_dim, 2)\n",
    "\n",
    "    def forward(self, s):\n",
    "        outs = self.hidden(s)\n",
    "        outs = F.relu(outs)\n",
    "        logits = self.output(outs)\n",
    "        return logits\n",
    "\n",
    "\n",
    "actor_func = ActorNet().to(device)\n",
    "value_func = critic"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "c565d5ef",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Run episode 99 with rewards -220.48690053820612\r"
     ]
    }
   ],
   "source": [
    "opt1 = torch.optim.AdamW(value_func.parameters(), lr=0.001)\n",
    "opt2 = torch.optim.AdamW(actor_func.parameters(), lr=0.001)\n",
    "def pick_sample(s):\n",
    "    with torch.no_grad():\n",
    "        #   --> size : (1, 4)\n",
    "        s_batch = np.expand_dims(s, axis=0)\n",
    "        s_batch = torch.tensor(s_batch, dtype=torch.float).to(device)\n",
    "        # Get logits from state\n",
    "        #   --> size : (1, 2)\n",
    "        logits = actor_func(s_batch)\n",
    "        #   --> size : (2)\n",
    "        logits = logits.squeeze(dim=0)\n",
    "        # From logits to probabilities\n",
    "        probs = F.softmax(logits, dim=-1)\n",
    "        # Pick up action's sample\n",
    "        a = torch.multinomial(probs, num_samples=1)\n",
    "        # Return\n",
    "        return a.tolist()[0]\n",
    "reward_records = []\n",
    "for i in range(nn_actor_train_iterations):\n",
    "    #\n",
    "    # Run episode till done\n",
    "    #\n",
    "    done = False\n",
    "    states = []\n",
    "    actions = []\n",
    "    rewards = []\n",
    "    s, _ = env.reset(seed=np.random.randint(0, 1e6))\n",
    "    while not done:\n",
    "        states.append(s.tolist())\n",
    "        a = pick_sample(s)\n",
    "        s, r, term, trunc, _ = env.step(a)\n",
    "        done = term or trunc\n",
    "        actions.append(a)\n",
    "        rewards.append(s[0]-0.5)\n",
    "\n",
    "    #\n",
    "    # Get cumulative rewards\n",
    "    #\n",
    "    cum_rewards = np.zeros_like(rewards)\n",
    "    reward_len = len(rewards)\n",
    "    for j in reversed(range(reward_len)):\n",
    "        cum_rewards[j] = rewards[j] + (cum_rewards[j+1]*gamma if j+1 < reward_len else 0)\n",
    "\n",
    "    #\n",
    "    # Train (optimize parameters)\n",
    "    #\n",
    "\n",
    "    # Optimize value loss (Critic)\n",
    "    opt1.zero_grad()\n",
    "    states = torch.tensor(states, dtype=torch.float).to(device)\n",
    "    cum_rewards = torch.tensor(cum_rewards, dtype=torch.float).to(device)\n",
    "    values = value_func(states)\n",
    "    values = values.squeeze(dim=1)\n",
    "    vf_loss = F.mse_loss(\n",
    "        values,\n",
    "        cum_rewards,\n",
    "        reduction=\"none\")\n",
    "    vf_loss.sum().backward()\n",
    "    opt1.step()\n",
    "\n",
    "    # Todo 2; RULE BASED UPDATES\n",
    "    with torch.no_grad():\n",
    "        values = value_func(states).squeeze(dim=1)\n",
    "    opt2.zero_grad()\n",
    "    actions = torch.tensor(actions, dtype=torch.int64).to(device)\n",
    "    advantages = cum_rewards - values\n",
    "#     print(advantages)\n",
    "    logits = actor_func(states)\n",
    "    log_probs = -F.cross_entropy(logits, actions, reduction=\"none\")\n",
    "    pi_loss = -log_probs * advantages\n",
    "    pi_loss.sum().backward()\n",
    "    opt2.step()\n",
    "    \n",
    "    # Output total rewards in episode (max 500)\n",
    "    print(\"Run episode {} with rewards {}\".format(i, sum(rewards)), end=\"\\r\")\n",
    "    if sum(rewards)>-200:\n",
    "        break\n",
    "    reward_records.append(sum(rewards))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "73589768",
   "metadata": {},
   "outputs": [],
   "source": [
    "def softmax(logits):\n",
    "    exp_logits = np.exp(logits)\n",
    "    sum_logits = np.sum(exp_logits)\n",
    "    return (exp_logits / sum_logits).reshape([-1])\n",
    "\n",
    "def pick_sample(s):\n",
    "    with torch.no_grad():\n",
    "        s_batch = np.expand_dims(s, axis=0)\n",
    "        logits = actor(pd.DataFrame(s_batch, columns=column_names))\n",
    "        probs = softmax(logits)\n",
    "        a = torch.multinomial(torch.tensor(probs), num_samples=1)\n",
    "        return a.tolist()[0]\n",
    "\n",
    "\n",
    "def train():\n",
    "    reward_records = []\n",
    "    best_rewards = -10000\n",
    "    # best_model = None\n",
    "    for i in range(rule_actor_train_iterations):\n",
    "\n",
    "        done = False\n",
    "        states = []\n",
    "        actions = []\n",
    "        rewards = []\n",
    "        s, _ = env.reset(seed=np.random.randint(0, 1e6))\n",
    "        if i == 2:\n",
    "            print('stop here')\n",
    "        while not done:\n",
    "            states.append(s.tolist())\n",
    "            a = pick_sample(s)\n",
    "            s, r, term, trunc, _ = env.step(a)\n",
    "            done = term or trunc\n",
    "            actions.append(a)\n",
    "            rewards.append((s[0]-0.5))\n",
    "\n",
    "        # if sum(rewards) > 480:\n",
    "        #     break\n",
    "        #\n",
    "        # Get cumulative rewards\n",
    "        #\n",
    "        cum_rewards = np.zeros_like(rewards)\n",
    "        reward_len = len(rewards)\n",
    "        for j in reversed(range(reward_len)):\n",
    "            cum_rewards[j] = rewards[j] + (cum_rewards[j + 1] * gamma if j + 1 < reward_len else 0)\n",
    "        #\n",
    "        # Train (optimize parameters)\n",
    "        #\n",
    "        # Optimize value loss (Critic)\n",
    "        critic_optimizer.zero_grad()\n",
    "        states = torch.tensor(states, dtype=torch.float).to(device)\n",
    "        cum_rewards = torch.tensor(cum_rewards, dtype=torch.float).to(device)\n",
    "        values = critic(states)\n",
    "        values = values.squeeze(dim=1)\n",
    "        vf_loss = F.mse_loss(\n",
    "            values,\n",
    "            cum_rewards,\n",
    "            reduction=\"none\")\n",
    "        vf_loss.sum().backward()\n",
    "        critic_optimizer.step()\n",
    "        print('============', i, '===========')\n",
    "#         print(len(actions), 'actions', actions)\n",
    "        reward_records.append(sum(rewards))\n",
    "        print(\"Run episode {} with rewards {}\".format(i, sum(rewards)))  # , end=\"\\r\")\n",
    "        if i > 0:\n",
    "            if sum(rewards) >= best_rewards:\n",
    "                best_rewards = sum(rewards)\n",
    "                for rr in range(actor.action_space):\n",
    "                    rules = []\n",
    "                    for n in range(len(actor.model[rr].rules_)):\n",
    "                        rule = actor.model[rr].rules_[n]\n",
    "                        rules.append(Rule(q=rule.q, y=rule.y))\n",
    "                    actor.best_model[rr].rules_ = AdditiveRuleEnsemble(rules)\n",
    "                print('best')\n",
    "        if np.average(reward_records[-5:]) > -150.0:\n",
    "            break\n",
    "        #     else:\n",
    "        #         for rr in range(actor.action_space):\n",
    "        #             rules = []\n",
    "        #             for n in range(len(actor.best_model[rr].rules_)):\n",
    "        #                 rule = actor.best_model[rr].rules_[n]\n",
    "        #                 rules.append(Rule(q=rule.q, y=rule.y))\n",
    "        #             actor.model[rr].rules_ = AdditiveRuleEnsemble(rules)\n",
    "        #         print('')\n",
    "\n",
    "        #\n",
    "        # Todo 2; RULE BASED UPDATES\n",
    "        with torch.no_grad():\n",
    "            values = critic(states).squeeze(dim=1)\n",
    "        actions = torch.tensor(actions, dtype=torch.int64).to(device)\n",
    "        advantages = cum_rewards - values\n",
    "        states_array = states.cpu().detach().numpy()\n",
    "        x = pd.DataFrame(states_array, columns=column_names)\n",
    "        y = pd.Series(actions.cpu().detach().numpy())\n",
    "#         advantages = (advantages-advantages.mean())/advantages.std()\n",
    "        adv = pd.Series(advantages.cpu().detach().numpy())\n",
    "        outputs = actor(x).T\n",
    "        # print('adv', advantages.cpu().detach().numpy())\n",
    "        # print('outputs', outputs.tolist())\n",
    "        for j in range(len(actor.model)):\n",
    "            m = actor.model[j]\n",
    "            masks = np.ones_like(outputs, dtype=bool)\n",
    "            masks[:, j] = False\n",
    "            other_values = outputs[masks].reshape(-1, outputs.shape[1] - 1)\n",
    "\n",
    "            if len(m.rules_) == 0:\n",
    "                m.fit(x, y, adv, other_values, j)\n",
    "            else:\n",
    "                for _ in range(1):\n",
    "                    update_rules(m, x, y, adv, other_values, j)\n",
    "        \n",
    "        # print(\n",
    "        #     \"{}\\n--------------\\n{}\\nRun episode {} with rewards {}\".format(actor.model[0].rules_,\n",
    "        #                                                                     actor.model[1].rules_,\n",
    "        #                                                                     i,\n",
    "        #                                                                     sum(rewards)))  # , end=\"\\r\")\n",
    "        \n",
    "    print(\"\\nDone\")\n",
    "    env.close()\n",
    "    return reward_records"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "b411aa5d",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 0 ===========\n",
      "Run episode 0 with rewards -203.8392645418644\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/1846393928.py:32: RuntimeWarning: invalid value encountered in sqrt\n",
      "  return abs(g_q.sum()) / np.sqrt(h_q.sum())\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/1846393928.py:32: RuntimeWarning: invalid value encountered in sqrt\n",
      "  return abs(g_q.sum()) / np.sqrt(h_q.sum())\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/1846393928.py:32: RuntimeWarning: invalid value encountered in sqrt\n",
      "  return abs(g_q.sum()) / np.sqrt(h_q.sum())\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 1 ===========\n",
      "Run episode 1 with rewards -199.77510887384415\n",
      "best\n",
      "stop here\n",
      "============ 2 ===========\n",
      "Run episode 2 with rewards -216.08222511410713\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 3 ===========\n",
      "Run episode 3 with rewards -188.63656221702695\n",
      "best\n",
      "============ 4 ===========\n",
      "Run episode 4 with rewards -201.99666353594512\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 5 ===========\n",
      "Run episode 5 with rewards -188.76777771115303\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 6 ===========\n",
      "Run episode 6 with rewards -216.56327086687088\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 7 ===========\n",
      "Run episode 7 with rewards -207.6582389473915\n",
      "============ 8 ===========\n",
      "Run episode 8 with rewards -211.35003089904785\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 9 ===========\n",
      "Run episode 9 with rewards -129.3398380400613\n",
      "best\n",
      "============ 10 ===========\n",
      "Run episode 10 with rewards -207.35418057441711\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 11 ===========\n",
      "Run episode 11 with rewards -210.95766350626945\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 12 ===========\n",
      "Run episode 12 with rewards -163.29150964180008\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 13 ===========\n",
      "Run episode 13 with rewards -180.42710748547688\n",
      "============ 14 ===========\n",
      "Run episode 14 with rewards -217.43968203663826\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 15 ===========\n",
      "Run episode 15 with rewards -207.13390338420868\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 16 ===========\n",
      "Run episode 16 with rewards -235.04290544986725\n",
      "============ 17 ===========\n",
      "Run episode 17 with rewards -192.41245371103287\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 18 ===========\n",
      "Run episode 18 with rewards -233.8218863606453\n",
      "============ 19 ===========\n",
      "Run episode 19 with rewards -190.92995017766953\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 20 ===========\n",
      "Run episode 20 with rewards -141.44658004661324\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 21 ===========\n",
      "Run episode 21 with rewards -203.9635288119316\n",
      "============ 22 ===========\n",
      "Run episode 22 with rewards -201.7753072977066\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 23 ===========\n",
      "Run episode 23 with rewards -198.60147489607334\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 24 ===========\n",
      "Run episode 24 with rewards -176.67167922854424\n",
      "============ 25 ===========\n",
      "Run episode 25 with rewards -218.3763048350811\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 26 ===========\n",
      "Run episode 26 with rewards -175.82891884446144\n",
      "============ 27 ===========\n",
      "Run episode 27 with rewards -221.37368229031563\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 28 ===========\n",
      "Run episode 28 with rewards -204.80158483982086\n",
      "============ 29 ===========\n",
      "Run episode 29 with rewards -205.4428251683712\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 30 ===========\n",
      "Run episode 30 with rewards -175.1840765022207\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 31 ===========\n",
      "Run episode 31 with rewards -228.6946205496788\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 32 ===========\n",
      "Run episode 32 with rewards -173.24335558712482\n",
      "============ 33 ===========\n",
      "Run episode 33 with rewards -209.54239374399185\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 34 ===========\n",
      "Run episode 34 with rewards -152.17294419091195\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 35 ===========\n",
      "Run episode 35 with rewards -234.12533417344093\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 36 ===========\n",
      "Run episode 36 with rewards -223.62724840641022\n",
      "============ 37 ===========\n",
      "Run episode 37 with rewards -203.01710492372513\n",
      "============ 38 ===========\n",
      "Run episode 38 with rewards -234.52759236097336\n",
      "============ 39 ===========\n",
      "Run episode 39 with rewards -192.07807344198227\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 40 ===========\n",
      "Run episode 40 with rewards -218.66338800720405\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 41 ===========\n",
      "Run episode 41 with rewards -204.53989952802658\n",
      "============ 42 ===========\n",
      "Run episode 42 with rewards -208.30157205462456\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 43 ===========\n",
      "Run episode 43 with rewards -201.9921447634697\n",
      "============ 44 ===========\n",
      "Run episode 44 with rewards -205.30636531114578\n",
      "============ 45 ===========\n",
      "Run episode 45 with rewards -207.78912934660912\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 46 ===========\n",
      "Run episode 46 with rewards -205.00042086839676\n",
      "============ 47 ===========\n",
      "Run episode 47 with rewards -208.83894285559654\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 48 ===========\n",
      "Run episode 48 with rewards -176.3820638358593\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 49 ===========\n",
      "Run episode 49 with rewards -234.6041458249092\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 50 ===========\n",
      "Run episode 50 with rewards -172.642872646451\n",
      "============ 51 ===========\n",
      "Run episode 51 with rewards -221.841733366251\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 52 ===========\n",
      "Run episode 52 with rewards -175.4364768564701\n",
      "============ 53 ===========\n",
      "Run episode 53 with rewards -220.96884658932686\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 54 ===========\n",
      "Run episode 54 with rewards -179.29933670163155\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 55 ===========\n",
      "Run episode 55 with rewards -206.33789606392384\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 56 ===========\n",
      "Run episode 56 with rewards -176.50503039360046\n",
      "============ 57 ===========\n",
      "Run episode 57 with rewards -220.0962735414505\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 58 ===========\n",
      "Run episode 58 with rewards -175.6297623217106\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 59 ===========\n",
      "Run episode 59 with rewards -205.46069169044495\n",
      "============ 60 ===========\n",
      "Run episode 60 with rewards -205.1636277139187\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 61 ===========\n",
      "Run episode 61 with rewards -204.13421896100044\n",
      "============ 62 ===========\n",
      "Run episode 62 with rewards -204.8262485563755\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 63 ===========\n",
      "Run episode 63 with rewards -234.7199685573578\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 64 ===========\n",
      "Run episode 64 with rewards -181.09976060688496\n",
      "============ 65 ===========\n",
      "Run episode 65 with rewards -237.28325435519218\n",
      "============ 66 ===========\n",
      "Run episode 66 with rewards -190.80046769976616\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 67 ===========\n",
      "Run episode 67 with rewards -236.76674979925156\n",
      "============ 68 ===========\n",
      "Run episode 68 with rewards -184.35426351428032\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 69 ===========\n",
      "Run episode 69 with rewards -217.26728071458638\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 70 ===========\n",
      "Run episode 70 with rewards -176.5256662964821\n",
      "============ 71 ===========\n",
      "Run episode 71 with rewards -233.76963007450104\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 72 ===========\n",
      "Run episode 72 with rewards -175.16907167434692\n",
      "============ 73 ===========\n",
      "Run episode 73 with rewards -219.8481003344059\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 74 ===========\n",
      "Run episode 74 with rewards -176.94596868753433\n",
      "============ 75 ===========\n",
      "Run episode 75 with rewards -219.86908811330795\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 76 ===========\n",
      "Run episode 76 with rewards -184.4087849855423\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 77 ===========\n",
      "Run episode 77 with rewards -235.31804859638214\n",
      "============ 78 ===========\n",
      "Run episode 78 with rewards -189.9949931204319\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 79 ===========\n",
      "Run episode 79 with rewards -235.46270576119423\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 80 ===========\n",
      "Run episode 80 with rewards -203.88736230134964\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 81 ===========\n",
      "Run episode 81 with rewards -234.8780625462532\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 82 ===========\n",
      "Run episode 82 with rewards -205.4967966377735\n",
      "============ 83 ===========\n",
      "Run episode 83 with rewards -205.32359221577644\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 84 ===========\n",
      "Run episode 84 with rewards -205.0408592224121\n",
      "============ 85 ===========\n",
      "Run episode 85 with rewards -202.9902845621109\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 86 ===========\n",
      "Run episode 86 with rewards -205.76077280100435\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 87 ===========\n",
      "Run episode 87 with rewards -78.48011961299926\n",
      "best\n",
      "============ 88 ===========\n",
      "Run episode 88 with rewards -205.41440215706825\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:3: RuntimeWarning: overflow encountered in exp\n",
      "  exps = np.exp(all_values)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: overflow encountered in exp\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n",
      "/tmp/ipykernel_267/4286230475.py:4: RuntimeWarning: invalid value encountered in divide\n",
      "  res = np.exp(all_values[np.arange(values.shape[0]), action]) / np.sum(exps, axis=1)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "============ 89 ===========\n",
      "Run episode 89 with rewards -83.19323232350871\n",
      "============ 90 ===========\n",
      "Run episode 90 with rewards -204.07604184746742\n",
      "origin 0.011429525822515487 new 0.011429525822515973 not updated\n",
      "============ 91 ===========\n",
      "Run episode 91 with rewards -157.72871918813325\n",
      "\n",
      "Done\n"
     ]
    }
   ],
   "source": [
    "rewards = train()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "3d20e8e0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "--------------------------------\n",
      "  +25.3111 if v<=-0.0006471572560258208\n",
      "  -10.5678 if p<=-0.5023447513580322\n",
      "  +53.8697 if \n",
      "  +51.0351 if v<=-0.004265955928713083\n",
      "  -14.5140 if p>=-0.5813989758491516 & v<=0.006803702469915152 & v>=-0.0008947865571826689\n",
      "  -11.6162 if p<=-0.41075775027275085 & v<=-0.000661600707098839\n",
      "  +17.6882 if v>=-1.5593287753289978e-05\n",
      "  +16.4293 if p<=-0.25240878462791444 & p>=-0.6275408387184143 & v<=0.01510303020477297 & v>=-0.011912779696285722\n",
      "  -17.9516 if p<=-0.4901819765567779 & p>=-0.615831172466278 & v<=0.007591811567544942 & v>=-0.004150395561009644 & v>=0.0012682238826528204\n",
      "  +21.4738 if v<=0.00040989847620949166\n",
      "   -2.3676 if p>=-0.5437554597854615 & v<=0.0016861518379300846 & v>=-0.0034104006830602885\n",
      "   +1.2316 if p<=-0.14024059176444992 & p>=-0.4494079709053038 & v<=0.023414026945829402 & v>=-0.01632060520350933\n",
      "--------------------------------\n",
      "  -19.7248 if p>=-0.5101138830184937 & v<=0.006803702469915152 & v>=-0.005573090445250273\n",
      "  -13.4848 if v>=0.0007145821116864693\n",
      "  +24.2199 if p>=-0.4848527312278747 & v<=0.00018940809495689726 & v>=-0.009227151423692702\n",
      "  -23.3706 if v<=0.003292182506993414\n",
      "   -6.6878 if v<=-9.685811237431741e-05\n",
      "  +24.7429 if p<=-0.4022643387317657 & v<=0.00020952874911017827\n",
      "  -26.5154 if p>=-0.6311534523963929 & v<=0.006872396916151048 & v>=-0.006245836243033409\n",
      "   -8.7515 if p>=-0.39260063171386717 & v>=0.0019062188919633629\n",
      "   +0.7214 if v<=-0.006146392878144977\n",
      "   -0.9839 if \n",
      "   +6.8736 if p>=-0.5437554597854615 & v<=0.0038646903820335875\n",
      "   +1.0000 if v>=-0.0010096890269778618\n",
      "--------------------------------\n",
      "  +26.2150 if p<=-0.5101138830184937 & v<=0.006803702469915152\n",
      "   +6.3636 if p>=-0.4848527312278747 & v>=-0.009227151423692702\n",
      "   +6.1251 if p>=-0.4855665504932403 & v>=0.003292182506993414\n",
      "  +24.0837 if \n",
      "  +27.5176 if p<=-0.4022643387317657 & v<=0.0012711382005363714\n",
      "  +21.3072 if v>=0.0007119119516573855\n",
      "  +18.6883 if v<=-0.0005520289996638883\n",
      "  +19.7662 if v>=0.0004626010428182804\n",
      "  +21.3412 if p<=-0.6329279899597166\n",
      "  +11.7750 if p>=-0.5767600297927856 & v>=0.0012682238826528204\n",
      "  +16.5113 if v>=0.00040989847620949166\n",
      "   +1.3610 if p<=-0.4494079709053038 & p>=-0.8744302272796631 & v<=0.023414026945829402\n"
     ]
    }
   ],
   "source": [
    "for i in range(action_space):\n",
    "    print('--------------------------------')\n",
    "    print(actor.best_model[i].rules_)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "0b443f87",
   "metadata": {},
   "outputs": [],
   "source": [
    "def test(num_test_episodes=100):\n",
    "    \"\"\"\n",
    "    Test the learned policy using the trained actor model.\n",
    "\n",
    "    Args:\n",
    "        num_test_episodes (int): Number of episodes to test the model.\n",
    "\n",
    "    Returns:\n",
    "        float: Average reward over the test episodes.\n",
    "    \"\"\"\n",
    "    total_reward = 0\n",
    "\n",
    "    for episode in range(num_test_episodes):\n",
    "        state = env.reset()[0]  # Reset the environment and get the initial state\n",
    "        episode_reward = 0\n",
    "\n",
    "        for t in range(1, 10000):  # Limit the number of time steps\n",
    "            # Convert state to tensor and predict action probabilities\n",
    "            #             state_tensor = torch.tensor(state, dtype=torch.float32).to(device)\n",
    "            action_probs = actor.predict(pd.DataFrame(np.array([state]), columns=column_names))\n",
    "            # Select action based on the highest probability\n",
    "            action = np.argmax(action_probs)\n",
    "            # Take the chosen action\n",
    "            next_state, reward, done, trunc, _ = env.step(action)\n",
    "            # Accumulate reward\n",
    "            episode_reward += reward\n",
    "            if done or trunc:\n",
    "                break\n",
    "            # Update state\n",
    "            state = next_state\n",
    "        total_reward += episode_reward\n",
    "        print(f\"Test Episode {episode + 1}, Reward: {episode_reward}\")\n",
    "    avg_reward = total_reward / num_test_episodes\n",
    "    print(f\"\\nAverage Reward over {num_test_episodes} Test Episodes: {avg_reward}\")\n",
    "    return avg_reward"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "ef9e3e26",
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test Episode 1, Reward: -127.0\n",
      "Test Episode 2, Reward: -101.0\n",
      "Test Episode 3, Reward: -119.0\n",
      "Test Episode 4, Reward: -87.0\n",
      "Test Episode 5, Reward: -96.0\n",
      "Test Episode 6, Reward: -85.0\n",
      "Test Episode 7, Reward: -115.0\n",
      "Test Episode 8, Reward: -110.0\n",
      "Test Episode 9, Reward: -84.0\n",
      "Test Episode 10, Reward: -116.0\n",
      "Test Episode 11, Reward: -90.0\n",
      "Test Episode 12, Reward: -84.0\n",
      "Test Episode 13, Reward: -120.0\n",
      "Test Episode 14, Reward: -127.0\n",
      "Test Episode 15, Reward: -117.0\n",
      "Test Episode 16, Reward: -101.0\n",
      "Test Episode 17, Reward: -92.0\n",
      "Test Episode 18, Reward: -173.0\n",
      "Test Episode 19, Reward: -88.0\n",
      "Test Episode 20, Reward: -97.0\n",
      "Test Episode 21, Reward: -119.0\n",
      "Test Episode 22, Reward: -115.0\n",
      "Test Episode 23, Reward: -126.0\n",
      "Test Episode 24, Reward: -161.0\n",
      "Test Episode 25, Reward: -87.0\n",
      "Test Episode 26, Reward: -119.0\n",
      "Test Episode 27, Reward: -109.0\n",
      "Test Episode 28, Reward: -121.0\n",
      "Test Episode 29, Reward: -124.0\n",
      "Test Episode 30, Reward: -89.0\n",
      "Test Episode 31, Reward: -116.0\n",
      "Test Episode 32, Reward: -176.0\n",
      "Test Episode 33, Reward: -95.0\n",
      "Test Episode 34, Reward: -119.0\n",
      "Test Episode 35, Reward: -85.0\n",
      "Test Episode 36, Reward: -122.0\n",
      "Test Episode 37, Reward: -122.0\n",
      "Test Episode 38, Reward: -116.0\n",
      "Test Episode 39, Reward: -101.0\n",
      "Test Episode 40, Reward: -119.0\n",
      "Test Episode 41, Reward: -92.0\n",
      "Test Episode 42, Reward: -116.0\n",
      "Test Episode 43, Reward: -119.0\n",
      "Test Episode 44, Reward: -99.0\n",
      "Test Episode 45, Reward: -115.0\n",
      "Test Episode 46, Reward: -98.0\n",
      "Test Episode 47, Reward: -90.0\n",
      "Test Episode 48, Reward: -89.0\n",
      "Test Episode 49, Reward: -101.0\n",
      "Test Episode 50, Reward: -124.0\n",
      "Test Episode 51, Reward: -119.0\n",
      "Test Episode 52, Reward: -105.0\n",
      "Test Episode 53, Reward: -172.0\n",
      "Test Episode 54, Reward: -95.0\n",
      "Test Episode 55, Reward: -116.0\n",
      "Test Episode 56, Reward: -120.0\n",
      "Test Episode 57, Reward: -85.0\n",
      "Test Episode 58, Reward: -123.0\n",
      "Test Episode 59, Reward: -84.0\n",
      "Test Episode 60, Reward: -116.0\n",
      "Test Episode 61, Reward: -85.0\n",
      "Test Episode 62, Reward: -116.0\n",
      "Test Episode 63, Reward: -116.0\n",
      "Test Episode 64, Reward: -118.0\n",
      "Test Episode 65, Reward: -126.0\n",
      "Test Episode 66, Reward: -133.0\n",
      "Test Episode 67, Reward: -116.0\n",
      "Test Episode 68, Reward: -122.0\n",
      "Test Episode 69, Reward: -124.0\n",
      "Test Episode 70, Reward: -115.0\n",
      "Test Episode 71, Reward: -84.0\n",
      "Test Episode 72, Reward: -125.0\n",
      "Test Episode 73, Reward: -116.0\n",
      "Test Episode 74, Reward: -122.0\n",
      "Test Episode 75, Reward: -166.0\n",
      "Test Episode 76, Reward: -116.0\n",
      "Test Episode 77, Reward: -119.0\n",
      "Test Episode 78, Reward: -122.0\n",
      "Test Episode 79, Reward: -115.0\n",
      "Test Episode 80, Reward: -116.0\n",
      "Test Episode 81, Reward: -115.0\n",
      "Test Episode 82, Reward: -165.0\n",
      "Test Episode 83, Reward: -89.0\n",
      "Test Episode 84, Reward: -116.0\n",
      "Test Episode 85, Reward: -170.0\n",
      "Test Episode 86, Reward: -133.0\n",
      "Test Episode 87, Reward: -116.0\n",
      "Test Episode 88, Reward: -92.0\n",
      "Test Episode 89, Reward: -119.0\n",
      "Test Episode 90, Reward: -86.0\n",
      "Test Episode 91, Reward: -116.0\n",
      "Test Episode 92, Reward: -100.0\n",
      "Test Episode 93, Reward: -93.0\n",
      "Test Episode 94, Reward: -124.0\n",
      "Test Episode 95, Reward: -115.0\n",
      "Test Episode 96, Reward: -116.0\n",
      "Test Episode 97, Reward: -85.0\n",
      "Test Episode 98, Reward: -135.0\n",
      "Test Episode 99, Reward: -166.0\n",
      "Test Episode 100, Reward: -131.0\n",
      "\n",
      "Average Reward over 100 Test Episodes: -114.02\n"
     ]
    }
   ],
   "source": [
    "avg_test_reward = test(num_test_episodes=100)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "31340922",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7f2c4ce814e0>]"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "# Generate recent 50 interval average\n",
    "average_reward = []\n",
    "for idx in range(len(rewards)):\n",
    "    avg_list = np.empty(shape=(1,), dtype=int)\n",
    "    if idx < 5:\n",
    "        avg_list = rewards[:idx+1]\n",
    "    else:\n",
    "        avg_list = rewards[idx-4:idx+1]\n",
    "    average_reward.append(np.average(avg_list))\n",
    "plt.plot(rewards)\n",
    "plt.plot(average_reward)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "9f18bae5",
   "metadata": {},
   "outputs": [],
   "source": [
    "x=[]\n",
    "for p in np.linspace(-1.2, 0.6, 100):\n",
    "    for v in np.linspace(0.07,-0.07,60):\n",
    "        x.append([p, v])\n",
    "res=pd.DataFrame(x, columns=column_names)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "44b0354e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>p</th>\n",
       "      <th>v</th>\n",
       "      <th>left</th>\n",
       "      <th>no acc</th>\n",
       "      <th>right</th>\n",
       "      <th>value</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>-1.2</td>\n",
       "      <td>0.070000</td>\n",
       "      <td>60.990082</td>\n",
       "      <td>-13.468705</td>\n",
       "      <td>103.009607</td>\n",
       "      <td>103.009607</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>-1.2</td>\n",
       "      <td>0.067627</td>\n",
       "      <td>60.990082</td>\n",
       "      <td>-13.468705</td>\n",
       "      <td>103.009607</td>\n",
       "      <td>103.009607</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>-1.2</td>\n",
       "      <td>0.065254</td>\n",
       "      <td>60.990082</td>\n",
       "      <td>-13.468705</td>\n",
       "      <td>103.009607</td>\n",
       "      <td>103.009607</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>-1.2</td>\n",
       "      <td>0.062881</td>\n",
       "      <td>60.990082</td>\n",
       "      <td>-13.468705</td>\n",
       "      <td>103.009607</td>\n",
       "      <td>103.009607</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>-1.2</td>\n",
       "      <td>0.060508</td>\n",
       "      <td>60.990082</td>\n",
       "      <td>-13.468705</td>\n",
       "      <td>103.009607</td>\n",
       "      <td>103.009607</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5995</th>\n",
       "      <td>0.6</td>\n",
       "      <td>-0.060508</td>\n",
       "      <td>151.689673</td>\n",
       "      <td>-23.447158</td>\n",
       "      <td>42.771938</td>\n",
       "      <td>151.689673</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5996</th>\n",
       "      <td>0.6</td>\n",
       "      <td>-0.062881</td>\n",
       "      <td>151.689673</td>\n",
       "      <td>-23.447158</td>\n",
       "      <td>42.771938</td>\n",
       "      <td>151.689673</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5997</th>\n",
       "      <td>0.6</td>\n",
       "      <td>-0.065254</td>\n",
       "      <td>151.689673</td>\n",
       "      <td>-23.447158</td>\n",
       "      <td>42.771938</td>\n",
       "      <td>151.689673</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5998</th>\n",
       "      <td>0.6</td>\n",
       "      <td>-0.067627</td>\n",
       "      <td>151.689673</td>\n",
       "      <td>-23.447158</td>\n",
       "      <td>42.771938</td>\n",
       "      <td>151.689673</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5999</th>\n",
       "      <td>0.6</td>\n",
       "      <td>-0.070000</td>\n",
       "      <td>151.689673</td>\n",
       "      <td>-23.447158</td>\n",
       "      <td>42.771938</td>\n",
       "      <td>151.689673</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>6000 rows × 6 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "        p         v        left     no acc       right       value\n",
       "0    -1.2  0.070000   60.990082 -13.468705  103.009607  103.009607\n",
       "1    -1.2  0.067627   60.990082 -13.468705  103.009607  103.009607\n",
       "2    -1.2  0.065254   60.990082 -13.468705  103.009607  103.009607\n",
       "3    -1.2  0.062881   60.990082 -13.468705  103.009607  103.009607\n",
       "4    -1.2  0.060508   60.990082 -13.468705  103.009607  103.009607\n",
       "...   ...       ...         ...        ...         ...         ...\n",
       "5995  0.6 -0.060508  151.689673 -23.447158   42.771938  151.689673\n",
       "5996  0.6 -0.062881  151.689673 -23.447158   42.771938  151.689673\n",
       "5997  0.6 -0.065254  151.689673 -23.447158   42.771938  151.689673\n",
       "5998  0.6 -0.067627  151.689673 -23.447158   42.771938  151.689673\n",
       "5999  0.6 -0.070000  151.689673 -23.447158   42.771938  151.689673\n",
       "\n",
       "[6000 rows x 6 columns]"
      ]
     },
     "execution_count": 30,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "results=pd.concat([res, pd.Series(actor.best_model[0].rules_(res), name='left'), \n",
    "           pd.Series(actor.best_model[1].rules_(res), name='no acc'), \n",
    "           pd.Series(actor.best_model[2].rules_(res), name='right')], axis=1)\n",
    "results1=pd.concat([results, \n",
    "                   pd.Series(results.apply(lambda row: max(row['left'], \n",
    "                                                           row['no acc'], \n",
    "                                                           row['right']), \n",
    "                                           axis=1), name='value')], \n",
    "                  axis=1)\n",
    "results1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "266a341a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x532.8 with 7 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.colors as mcolors\n",
    "import matplotlib.patches as patches\n",
    "fig, axs=plt.subplots(3,2,figsize=(6,7.4),sharex=True, sharey=True, gridspec_kw={'width_ratios':[1,1]})\n",
    "names=['left', 'no acc', 'right']\n",
    "xlabels=np.linspace(0.07,-0.07,3)\n",
    "ylabels=[rf'{x:.2f}' for x in np.linspace(-1.2, 0.6, 6)]\n",
    "min_value=min(results1[names].min())\n",
    "max_value=max(results1[names].max())\n",
    "# print(min_value, max_value)\n",
    "def calc_loc(value, pv):\n",
    "    if pv=='p':\n",
    "        return (value+1.2)/1.8*100\n",
    "    elif pv=='v':\n",
    "        return (0.07-value)/0.14*60\n",
    "norm=mcolors.Normalize(vmin=min_value, vmax=max_value)\n",
    "cmap=plt.get_cmap('viridis')\n",
    "# def tostr(rule):\n",
    "    \n",
    "for i in range(3):\n",
    "    im=axs[i][0].imshow(results1[names[i]].to_numpy().reshape(100, -1).T, norm=norm)\n",
    "    axs[i][0].set_yticks(np.linspace(0,60,3), labels=xlabels)\n",
    "    axs[i][0].set_xticks(np.linspace(0,100,6), labels=ylabels)\n",
    "    axs[i][0].set_title(names[i])\n",
    "    axs[i][0].set_ylabel('v')\n",
    "    if i<3:\n",
    "        for j in range(len(actor.best_model[i].rules_)):\n",
    "            ru = actor.best_model[i].rules_[j]\n",
    "            if abs(abs(ru.y)-17.952)<=0.001:\n",
    "                continue\n",
    "            if abs(abs(ru.y)-16.429)<=0.001:\n",
    "                continue\n",
    "            if abs(ru.y)>=10 and len(ru.q)>0:\n",
    "                conditions=str(ru.q).split(' & ')\n",
    "                \n",
    "                vmin=-0.069\n",
    "                vmax=0.069\n",
    "                pmin=-1.19\n",
    "                pmax=0.59\n",
    "                rule_str=rf'{ru.y:.3f} if '\n",
    "                query_str=[]\n",
    "                for cond in conditions:\n",
    "                    feature=cond[0]\n",
    "                    sign=cond[1:3]\n",
    "                    value=float(cond[3:])\n",
    "                    if feature=='p' and sign=='<=' and value<=pmax:\n",
    "                        pmax=value\n",
    "                        query_str+=[rf'p$\\leq${value:.3f}']\n",
    "                    elif feature=='p' and sign=='>=' and value>=pmin:\n",
    "                        pmin=value\n",
    "                        query_str+=[rf'p$\\geq${value:.3f}']\n",
    "                    elif feature=='v' and sign=='<=' and value<=vmax:\n",
    "                        vmax=value\n",
    "                        query_str+=[rf'v$\\leq${value:.3f}']\n",
    "                    elif feature=='v' and sign=='>=' and value>=vmin:\n",
    "                        vmin=value\n",
    "                        query_str+=[rf'v$\\geq${value:.3f}']\n",
    "                colors=[np.random.uniform(0,1) for _ in range(3)]\n",
    "                lwdth=np.random.uniform(1,2)\n",
    "#                 print(colors, lwdth)\n",
    "                axs[i][1].add_patch(patches.Rectangle((calc_loc(pmin,'p'),\n",
    "                                                      calc_loc(vmax, 'v')),\n",
    "                                                     calc_loc(pmax,'p')-calc_loc(pmin,'p'),\n",
    "                                                     calc_loc(vmin,'v')-calc_loc(vmax,'v'),\n",
    "                                                     alpha=1,linewidth=lwdth,\n",
    "                                                      edgecolor=colors,\n",
    "                                                      facecolor='none',\n",
    "                                                    label=rule_str+' & '.join(query_str)))\n",
    "                axs[i][1].set_title('rules for '+names[i])\n",
    "        axs[i][1].legend(loc=(1.1,0 if i==2 else -0.1 if i==0 else -0), fontsize=9, ncol=1 if i>0 else 1)\n",
    "                                \n",
    "axs[2][0].set_xlabel('p')\n",
    "axs[2][1].set_xlabel('p')\n",
    "fig.colorbar(im, ax=axs, orientation='horizontal', fraction=0.04, pad=0.12)\n",
    "plt.savefig('ac_rules_mountaincar.pdf', bbox_inches='tight', pad_inches=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "ae2409ef",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 288x216 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.colors as mcolors\n",
    "import matplotlib.patches as patches\n",
    "fig, axs=plt.subplots(1,1,figsize=(4,3),sharex=True, sharey=True)#, gridspec_kw={'width_ratios':[1]})\n",
    "names=['left', 'no acc', 'right', 'value']\n",
    "xlabels=np.linspace(0.07,-0.07,3)\n",
    "ylabels=[rf'{x:.2f}' for x in np.linspace(-1.2, 0.6, 6)]\n",
    "min_value=min(results1[names].min())\n",
    "max_value=max(results1[names].max())\n",
    "# print(min_value, max_value)\n",
    "norm=mcolors.Normalize(vmin=min_value, vmax=max_value)\n",
    "cmap=plt.get_cmap('viridis')\n",
    "im=axs.imshow(results1[names[3]].to_numpy().reshape(100, -1).T, norm=norm)\n",
    "axs.set_yticks(np.linspace(0,60,3), labels=xlabels)\n",
    "axs.set_xticks(np.linspace(0,100,6), labels=ylabels)\n",
    "axs.set_title('predicted values')\n",
    "axs.set_ylabel('v')\n",
    "axs.set_xlabel('p') \n",
    "fig.colorbar(im, ax=axs, orientation='horizontal', pad=0.24)\n",
    "plt.savefig('values_mountaincar.pdf', bbox_inches='tight', pad_inches=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "30c5e192",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 288x216 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "results2=pd.concat([results1,pd.Series(results1[['left', 'no acc', 'right']].idxmax(axis=1), name='action')],axis=1)\n",
    "actions=results2['action'].map({'left':1, 'right':3, 'no acc':2}).to_numpy().reshape(100,-1).T\n",
    "fig, axs=plt.subplots(1,1,figsize=(4,3),sharex=True)\n",
    "im=axs.imshow(actions)\n",
    "axs.set_yticks(np.linspace(0,60,3), labels=xlabels)\n",
    "axs.set_xticks(np.linspace(0,100,6), labels=ylabels)\n",
    "axs.set_title('predicted actions (10 rules)')\n",
    "axs.set_ylabel('v')\n",
    "axs.set_xlabel('p') \n",
    "cmap=plt.get_cmap('viridis')\n",
    "norm=mcolors.Normalize(vmin=1, vmax=3)\n",
    "axs.add_patch(patches.Rectangle((-10,-10),1,1, alpha=1,linewidth=lwdth,\n",
    "                                                      edgecolor=cmap(norm(1)),\n",
    "                                                      facecolor=cmap(norm(1)),\n",
    "                                                    label='left'))\n",
    "\n",
    "axs.add_patch(patches.Rectangle((-10,-10),1,1, alpha=1,linewidth=lwdth,\n",
    "                                                      edgecolor=cmap(norm(3)),\n",
    "                                                      facecolor=cmap(norm(3)),\n",
    "                                                    label='right'))\n",
    "\n",
    "axs.add_patch(patches.Rectangle((-10,-10),1,1, alpha=1,linewidth=lwdth,\n",
    "                                                      edgecolor=cmap(norm(2)),\n",
    "                                                      facecolor=cmap(norm(2)),\n",
    "                                                    label='no acc'))\n",
    "axs.legend()\n",
    "plt.savefig('actions_mountaincar.pdf', bbox_inches='tight', pad_inches=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "b215130d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.colorbar.Colorbar at 0x7f2c4c319780>"
      ]
     },
     "execution_count": 34,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 360x432 with 5 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.colors as mcolors\n",
    "fig, axs=plt.subplots(4,1,figsize=(5,6),sharex=True)#, gridspec_kw={'width_ratios':[1,1,1,1.25]})\n",
    "names=['left', 'no acc', 'right', 'value']\n",
    "xlabels=np.linspace(-0.07,0.07,3)\n",
    "ylabels=[rf'{x:.2f}' for x in np.linspace(-1.2, 0.6, 6)]\n",
    "min_value=min(results1[names].min())\n",
    "max_value=max(results1[names].max())\n",
    "# print(min_value, max_value)\n",
    "norm=mcolors.Normalize(vmin=min_value, vmax=max_value)\n",
    "for i in range(4):\n",
    "    im=axs[i].imshow(results1[names[i]].to_numpy().reshape(100, -1).T, norm=norm)\n",
    "    axs[i].set_yticks(np.linspace(0,30,3), labels=xlabels)\n",
    "    axs[i].set_xticks(np.linspace(0,100,6), labels=ylabels)\n",
    "    axs[i].set_title(names[i])\n",
    "    axs[i].set_ylabel('v')\n",
    "axs[3].set_xlabel('p')\n",
    "fig.colorbar(im, ax=axs)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "53a9c53d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "ax = plt.gca()\n",
    "ax.imshow(results1['right'].to_numpy().reshape(100, 60))\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "59cc7271",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "ax = plt.gca()\n",
    "ax.imshow(results1['value'].to_numpy().reshape(100, 60))\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e16aecc9",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "afd0d2e6",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "792caeb8",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "296aa2bf",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fa5efe1d",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "58e2db1b",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8385f8b3",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c1d27da3",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fb4df6e4",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.10.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
