{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 79,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from itertools import product\n",
    "from scipy.optimize import minimize"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "class LinearMixtureMDP():\n",
    "    def __init__(self, action_space, delta, xi_norm, seed=1):\n",
    "        np.random.seed(seed)\n",
    "        self.state_space = ['x_1', 'x_2', 'x_3', 'x_4', 'x_5']\n",
    "        self.action_space = action_space\n",
    "        self.initial_state = 'x_1'\n",
    "        self.nu = [np.zeros(4),\n",
    "                   np.array([0,0,0,1]),\n",
    "                   np.array([0,0,0,1])]\n",
    "        self.p = 0.1\n",
    "        # self.theta = np.array(([self.p, 1 - 2*self.p, self.p], [self.p, 1 - 2*self.p, self.p]))\n",
    "        self.theta = np.array(([0, 1 - self.p, self.p], [0, 1 - self.p, self.p]))\n",
    "        self.delta = delta\n",
    "        self.H = 3\n",
    "        Xi = np.full(len(self.action_space[0]), 1)\n",
    "        self.Xi = xi_norm * Xi / np.linalg.norm(Xi, 1)\n",
    "        self.xi_norm = xi_norm\n",
    "    \n",
    "    def reset(self):\n",
    "        self.S = [self.initial_state]\n",
    "        self.A = [] # save the action history\n",
    "        self.R = [] # save the reward history\n",
    "        self.h = 0 # reset the step to 0\n",
    "        self.current_state = self.initial_state # reset the current state to initial state\n",
    "\n",
    "    def psi(self, current_state, A):\n",
    "        if current_state == 'x_1':\n",
    "            psi = np.array([1 - self.delta - self.Xi @ A, 0, 0, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_2':\n",
    "            psi = np.array([0, 1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_3':\n",
    "            psi = np.array([0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_4':\n",
    "            psi = np.array([0, 0, 1, 0])\n",
    "        elif current_state == 'x_5':\n",
    "            psi = np.array([0, 0, 0, 1])\n",
    "        else:\n",
    "            raise ValueError(f\"Error: The value {current_state} is not allow!\")\n",
    "        return psi\n",
    "    \n",
    "    def add_state(self, s):\n",
    "        self.S.append(s)\n",
    "        self.current_state = s\n",
    "\n",
    "    def phi(self, current_state, A):\n",
    "        if current_state == 'x_1':\n",
    "            phi = np.array(([1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A, 0], \n",
    "                           [1 - self.delta - self.Xi @ A, 0, 0, self.delta + self.Xi @ A], \n",
    "                           [0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A]))\n",
    "        elif current_state == 'x_2':\n",
    "            phi = np.array(([0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A, 0],\n",
    "                            [0, 1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A],\n",
    "                            [0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A]))\n",
    "        elif current_state == 'x_4':\n",
    "            phi = np.array(([0, 0, 1, 0], [0, 0, 1, 0], [0, 0, 1, 0]))\n",
    "        elif current_state == 'x_5':\n",
    "            phi = np.array(([0, 0, 0, 1], [0, 0, 0, 1], [0, 0, 0, 1]))\n",
    "        return phi\n",
    "        \n",
    "    def update_state(self, phi, h):\n",
    "        # calculate the transition probability\n",
    "        prob = self.theta[h] @ phi\n",
    "        sprime = np.random.choice(range(1,5), size = 1, p = prob)[0]\n",
    "        return self.state_space[sprime] # return a string\n",
    "    \n",
    "    def next_state(self, phi, h):\n",
    "        next_state = self.update_state(phi, h)\n",
    "        self.add_state(next_state)\n",
    "        return next_state\n",
    "    \n",
    "    def generate_reward(self, psi):\n",
    "        reward = np.dot(psi, self.nu[self.h])\n",
    "        self.R.append(reward)\n",
    "        return reward\n",
    "    \n",
    "    def step(self, a):\n",
    "        self.A.append(a)\n",
    "        psi = self.psi(self.current_state, a)\n",
    "        r = self.generate_reward(psi)\n",
    "        if self.h < self.H-1:\n",
    "            phi = self.phi(self.current_state, a)\n",
    "            self.next_state(phi, self.h)\n",
    "            self.h += 1\n",
    "\n",
    "class LinearMixtureMDP_test():\n",
    "    \"\"\"Perturbed Environment\"\"\"\n",
    "    def __init__(self, nominal_MDP, q, seed=1):\n",
    "        np.random.seed(seed)\n",
    "        self.state_space = nominal_MDP.state_space\n",
    "        self.action_space = nominal_MDP.action_space\n",
    "        self.initial_state = 'x_1'\n",
    "        # self.theta = nominal_MDP.theta\n",
    "        self.nu = nominal_MDP.nu\n",
    "        self.delta = nominal_MDP.delta\n",
    "        self.Xi = nominal_MDP.Xi\n",
    "        self.xi_norm = nominal_MDP.xi_norm\n",
    "        self.H = 3\n",
    "        self.q = q\n",
    "        # self.theta = np.array(([self.q, 1 - self.q, 0], [nominal_MDP.p, 1 - 2*nominal_MDP.p, nominal_MDP.p]))\n",
    "        self.theta = np.array(([self.q, 1 - self.q, 0], [0, 1 - nominal_MDP.p, nominal_MDP.p]))\n",
    "\n",
    "    def reset(self):\n",
    "        self.S = [self.initial_state]\n",
    "        self.A = [] # save the action history\n",
    "        self.R = [] # save the reward history\n",
    "        self.h = 0 # reset the step to 0\n",
    "        self.current_state = self.initial_state # reset the current state to initial state\n",
    "\n",
    "    def psi(self, current_state, A):\n",
    "        if current_state == 'x_1':\n",
    "            psi = np.array([1 - self.delta - self.Xi @ A, 0, 0, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_2':\n",
    "            psi = np.array([0, 1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_3':\n",
    "            psi = np.array([0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_4':\n",
    "            psi = np.array([0, 0, 1, 0])\n",
    "        elif current_state == 'x_5':\n",
    "            psi = np.array([0, 0, 0, 1])\n",
    "        else:\n",
    "            raise ValueError(f\"Error: The value {current_state} is not allow!\")\n",
    "        return psi\n",
    "    \n",
    "    def add_state(self, s):\n",
    "        self.S.append(s)\n",
    "        self.current_state = s\n",
    "\n",
    "    def phi(self, current_state, A):\n",
    "        if current_state == 'x_1':\n",
    "            phi = np.array(([1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A, 0], \n",
    "                           [1 - self.delta - self.Xi @ A, 0, 0, self.delta + self.Xi @ A], \n",
    "                           [0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A]))\n",
    "        elif current_state == 'x_2':\n",
    "            phi = np.array(([0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A, 0],\n",
    "                            [0, 1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A],\n",
    "                            [0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A]))\n",
    "        elif current_state == 'x_4':\n",
    "            phi = np.array(([0, 0, 1, 0], [0, 0, 1, 0], [0, 0, 1, 0]))\n",
    "        elif current_state == 'x_5':\n",
    "            phi = np.array(([0, 0, 0, 1], [0, 0, 0, 1], [0, 0, 0, 1]))\n",
    "        return phi\n",
    "    \n",
    "    def update_state(self, phi, h):\n",
    "        # calculate the transition probability\n",
    "        prob = self.theta[h] @ phi\n",
    "        sprime = np.random.choice(range(1,5), size = 1, p = prob)[0]\n",
    "        return self.state_space[sprime] # return a string\n",
    "    \n",
    "    def next_state(self, phi, h):\n",
    "        next_state = self.update_state(phi, h)\n",
    "        self.add_state(next_state)\n",
    "        return next_state\n",
    "    \n",
    "    def generate_reward(self, psi):\n",
    "        reward = np.dot(psi, self.nu[self.h])\n",
    "        self.R.append(reward)\n",
    "        return reward\n",
    "    \n",
    "    def step(self, a):\n",
    "        self.A.append(a)\n",
    "        psi = self.psi(self.current_state, a)\n",
    "        r = self.generate_reward(psi)\n",
    "        if self.h < self.H-1:\n",
    "            phi = self.phi(self.current_state, a)\n",
    "            self.next_state(phi, self.h)\n",
    "            self.h += 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 81,
   "metadata": {},
   "outputs": [],
   "source": [
    "class DRVTR_TV():\n",
    "    def __init__(self, A, H, lam, dataset, Rho, delta, xi_norm):\n",
    "        self.lam = lam\n",
    "        self.H = H\n",
    "        self.action_space = A\n",
    "        self.w = [np.zeros(4) for _ in range(self.H)]\n",
    "        self.Lambda = [self.lam * np.diag(np.ones(3)) for _ in range(self.H)]\n",
    "        self.Rho = Rho\n",
    "        self.dataset = dataset\n",
    "        Xi = np.full(len(self.action_space[0]), 1)\n",
    "        self.Xi = xi_norm * Xi / np.linalg.norm(Xi, 1)\n",
    "        self.xi_norm = xi_norm\n",
    "        self.delta = delta\n",
    "        self.nu = [np.zeros(4),\n",
    "                   np.array([0,0,0,1]),\n",
    "                   np.array([0,0,0,1])]\n",
    "    \n",
    "    def psi(self, current_state, A):\n",
    "        if current_state == 'x_1':\n",
    "            psi = np.array([1 - self.delta - self.Xi @ A, 0, 0, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_2':\n",
    "            psi = np.array([0, 1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_3':\n",
    "            psi = np.array([0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_4':\n",
    "            psi = np.array([0, 0, 1, 0])\n",
    "        elif current_state == 'x_5':\n",
    "            psi = np.array([0, 0, 0, 1])\n",
    "        else:\n",
    "            raise ValueError(f\"Error: The value {current_state} is not allow!\")\n",
    "        return psi\n",
    "\n",
    "    def phi(self, current_state, A):\n",
    "        if current_state == 'x_1':\n",
    "            phi = np.array(([1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A, 0], \n",
    "                           [1 - self.delta - self.Xi @ A, 0, 0, self.delta + self.Xi @ A], \n",
    "                           [0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A]))\n",
    "        elif current_state == 'x_2':\n",
    "            phi = np.array(([0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A, 0],\n",
    "                            [0, 1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A],\n",
    "                            [0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A]))\n",
    "        elif current_state == 'x_4':\n",
    "            phi = np.array(([0, 0, 1, 0], [0, 0, 1, 0], [0, 0, 1, 0]))\n",
    "        elif current_state == 'x_5':\n",
    "            phi = np.array(([0, 0, 0, 1], [0, 0, 0, 1], [0, 0, 0, 1]))\n",
    "        return phi\n",
    "\n",
    "    def get_action(self, h, state):\n",
    "        Q_h = [self.get_Q_func(h, state, action) for action in self.action_space]\n",
    "        return self.action_space[np.argmax(Q_h)]\n",
    "\n",
    "    def get_Q_func(self, h, state, action):\n",
    "        if h == self.H-1:\n",
    "            reward_feature = self.psi(state, action)\n",
    "            reward_parameter = self.nu[h]\n",
    "            return np.dot(reward_feature, reward_parameter)\n",
    "        else:\n",
    "            phi_s_a = self.phi(state, action)\n",
    "            phi_V_h_plus_one_s_a = [np.dot(phi_s_a[i], self.V_hat[h + 1]) for i in range(3)]\n",
    "            theta_hat = self.theta_hat[h]\n",
    "            def Q_alpha(alpha):\n",
    "                first_part = np.dot(np.minimum(phi_V_h_plus_one_s_a, alpha), theta_hat)\n",
    "                second_part = self.Rho[h]*(alpha - np.min(np.minimum(phi_V_h_plus_one_s_a, alpha)))\n",
    "                return - first_part + second_part\n",
    "            result = minimize(Q_alpha, self.H/2, method = 'Nelder-Mead', bounds=[(0,self.H)])\n",
    "            reward_feature = self.psi(state, action)\n",
    "            reward_parameter = self.nu[h]\n",
    "            reward = np.dot(reward_feature, reward_parameter)\n",
    "            return np.min([reward - result.fun, self.H - h])\n",
    "\n",
    "    def update_Q(self):\n",
    "        # Backward induction\n",
    "        self.theta_hat = [None for _ in range(self.H-1)] # initialize parameter estimation\n",
    "        self.V_hat = [np.zeros(4) for _ in range(self.H)] # initialize future cumulative reward, i.e., value function\n",
    "        self.V_hat[2] = np.array([0, self.delta+self.xi_norm, 0, 1]) # the last stage value function is the maximum reward which is known\n",
    "        # Start From the Last But One Stage \n",
    "        for h in range(self.H-2, -1, -1):\n",
    "            # Prepare for Ridge Regression\n",
    "            Phi_V_h_plus_one = np.zeros((0,3))\n",
    "            V_h_plus_one_stack = np.zeros(0)\n",
    "            for n in range(self.dataset['k']):\n",
    "                state_temp = self.dataset['state'][n][h]\n",
    "                action_temp = self.dataset['action'][n][h]\n",
    "                phi_temp = self.phi(state_temp, action_temp) # prepare the basis modes for s_h^k a_h^k\n",
    "                phi_V_temp = [np.dot(phi_temp[i], np.array(self.V_hat[h+1])) for i in range(3)] # take expectation to get feature\n",
    "                feature_temp = phi_V_temp\n",
    "                self.Lambda[h] += np.outer(feature_temp, feature_temp)\n",
    "                Phi_V_h_plus_one = np.vstack((Phi_V_h_plus_one, feature_temp))\n",
    "                if self.dataset['state'][n][h+1] == 'x_2':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][0]\n",
    "                elif self.dataset['state'][n][h+1] == 'x_3':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][1]\n",
    "                elif self.dataset['state'][n][h+1] == 'x_4':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][2]\n",
    "                elif self.dataset['state'][n][h+1] == 'x_5':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][3]\n",
    "                else: \n",
    "                    raise ValueError(f\"Error: The value {self.dataset['state'][n][h+1]} is not allow!\")\n",
    "                V_h_plus_one_stack = np.hstack((V_h_plus_one_stack, V_h_plus_one_s_h_plus_one))\n",
    "\n",
    "            # update theta estimation\n",
    "            Lambda_h_inverse = np.linalg.inv(self.Lambda[h])\n",
    "            theta_hat_h =  Lambda_h_inverse @ Phi_V_h_plus_one.T @ V_h_plus_one_stack\n",
    "            if np.product(theta_hat_h) < 0:\n",
    "                theta_hat_h -= np.min(theta_hat_h)\n",
    "            self.theta_hat[h] = theta_hat_h / np.sum(theta_hat_h)\n",
    "            \n",
    "            # Calculate the Current Stage Value Function \n",
    "            V_h = []\n",
    "            for state in ['x_2', 'x_3', 'x_4', 'x_5']:\n",
    "                if state == 'x_3':\n",
    "                    V_h.append(0)\n",
    "                else:\n",
    "                    Q_h = [self.get_Q_func(h, state, action) for action in self.action_space]\n",
    "                    V_h.append(np.max(Q_h))\n",
    "            self.V_hat[h] = np.array(V_h)\n",
    "\n",
    "class DRVTR_KL():\n",
    "    def __init__(self, A, H, lam, dataset, Rho, delta, xi_norm):\n",
    "        self.lam = lam\n",
    "        self.H = H\n",
    "        self.action_space = A\n",
    "        self.w = [np.zeros(4) for _ in range(self.H)]\n",
    "        self.Lambda = [self.lam * np.diag(np.ones(3)) for _ in range(self.H)]\n",
    "        self.Rho = Rho\n",
    "        self.dataset = dataset\n",
    "        Xi = np.full(len(self.action_space[0]), 1)\n",
    "        self.Xi = xi_norm * Xi / np.linalg.norm(Xi, 1)\n",
    "        self.xi_norm = xi_norm\n",
    "        self.delta = delta\n",
    "        self.nu = [np.zeros(4),\n",
    "                   np.array([0,0,0,1]),\n",
    "                   np.array([0,0,0,1])]\n",
    "    \n",
    "    def psi(self, current_state, A):\n",
    "        if current_state == 'x_1':\n",
    "            psi = np.array([1 - self.delta - self.Xi @ A, 0, 0, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_2':\n",
    "            psi = np.array([0, 1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_3':\n",
    "            psi = np.array([0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_4':\n",
    "            psi = np.array([0, 0, 1, 0])\n",
    "        elif current_state == 'x_5':\n",
    "            psi = np.array([0, 0, 0, 1])\n",
    "        else:\n",
    "            raise ValueError(f\"Error: The value {current_state} is not allow!\")\n",
    "        return psi\n",
    "\n",
    "    def phi(self, current_state, A):\n",
    "        if current_state == 'x_1':\n",
    "            phi = np.array(([1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A, 0], \n",
    "                           [1 - self.delta - self.Xi @ A, 0, 0, self.delta + self.Xi @ A], \n",
    "                           [0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A]))\n",
    "        elif current_state == 'x_2':\n",
    "            phi = np.array(([0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A, 0],\n",
    "                            [0, 1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A],\n",
    "                            [0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A]))\n",
    "        elif current_state == 'x_4':\n",
    "            phi = np.array(([0, 0, 1, 0], [0, 0, 1, 0], [0, 0, 1, 0]))\n",
    "        elif current_state == 'x_5':\n",
    "            phi = np.array(([0, 0, 0, 1], [0, 0, 0, 1], [0, 0, 0, 1]))\n",
    "        return phi\n",
    "\n",
    "    def get_action(self, h, state):\n",
    "        Q_h = [self.get_Q_func(h, state, action) for action in self.action_space]\n",
    "        return self.action_space[np.argmax(Q_h)]\n",
    "\n",
    "    def get_Q_func(self, h, state, action):\n",
    "        if h == self.H-1:\n",
    "            reward_feature = self.psi(state, action)\n",
    "            reward_parameter = self.nu[h]\n",
    "            return np.dot(reward_feature, reward_parameter)\n",
    "        else:\n",
    "            if self.Rho[h] == 0:\n",
    "                phi_s_a = self.phi(state, action) # get the basis modes\n",
    "                phi_V_h_plus_one_s_a = [np.dot(phi_s_a[i], self.V_hat[h + 1]) for i in range(3)] # take expectations to get feature\n",
    "                theta_hat = self.theta_hat[h]\n",
    "                result = np.dot(phi_V_h_plus_one_s_a, theta_hat) # take expectation wrt i~theta\n",
    "                reward_feature = self.psi(state, action) \n",
    "                reward_parameter = self.nu[h] # calculate the immediate reward\n",
    "                reward = np.dot(reward_feature, reward_parameter) \n",
    "                return np.min([reward + result, self.H - h]) # reward + future cumulative reward = Qfunc\n",
    "            else:\n",
    "                phi_s_a = self.phi(state, action)\n",
    "                phi_V_h_plus_one_s_a = [np.dot(phi_s_a[i], self.V_hat[h + 1]) for i in range(3)]\n",
    "                theta_hat = self.theta_hat[h]\n",
    "                def Q_alpha(alpha):\n",
    "                    first_part = alpha * np.log(np.dot(theta_hat, np.exp(-np.array(phi_V_h_plus_one_s_a)/alpha)))\n",
    "                    second_part = self.Rho[h] * alpha \n",
    "                    return first_part + second_part\n",
    "                result = minimize(Q_alpha, self.H/2, method = 'Nelder-Mead', bounds=[(0.01, self.H/np.min([self.Rho[h], 1]))])\n",
    "                reward_feature = self.psi(state, action)\n",
    "                reward_parameter = self.nu[h]\n",
    "                reward = np.dot(reward_feature, reward_parameter)\n",
    "                return np.min([reward - result.fun, self.H - h])\n",
    "\n",
    "    def update_Q(self):\n",
    "        # Backward induction\n",
    "        self.theta_hat = [None for _ in range(self.H-1)] # initialize parameter estimation\n",
    "        self.V_hat = [np.zeros(4) for _ in range(self.H)] # initialize future cumulative reward, i.e., value function\n",
    "        self.V_hat[2] = np.array([0, self.delta+self.xi_norm, 0, 1]) # the last stage value function is the maximum reward which is known\n",
    "        # Start From the Last But One Stage \n",
    "        for h in range(self.H-2, -1, -1):\n",
    "            # Prepare for Ridge Regression\n",
    "            Phi_V_h_plus_one = np.zeros((0,3))\n",
    "            V_h_plus_one_stack = np.zeros(0)\n",
    "            for n in range(self.dataset['k']):\n",
    "                state_temp = self.dataset['state'][n][h]\n",
    "                action_temp = self.dataset['action'][n][h]\n",
    "                phi_temp = self.phi(state_temp, action_temp) # prepare the basis modes for s_h^k a_h^k\n",
    "                phi_V_temp = [np.dot(phi_temp[i], np.array(self.V_hat[h+1])) for i in range(3)] # take expectation to get feature\n",
    "                feature_temp = phi_V_temp\n",
    "                self.Lambda[h] += np.outer(feature_temp, feature_temp)\n",
    "                Phi_V_h_plus_one = np.vstack((Phi_V_h_plus_one, feature_temp))\n",
    "                if self.dataset['state'][n][h+1] == 'x_2':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][0]\n",
    "                elif self.dataset['state'][n][h+1] == 'x_3':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][1]\n",
    "                elif self.dataset['state'][n][h+1] == 'x_4':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][2]\n",
    "                elif self.dataset['state'][n][h+1] == 'x_5':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][3]\n",
    "                else: \n",
    "                    raise ValueError(f\"Error: The value {self.dataset['state'][n][h+1]} is not allow!\")\n",
    "                V_h_plus_one_stack = np.hstack((V_h_plus_one_stack, V_h_plus_one_s_h_plus_one))\n",
    "\n",
    "            # update theta estimation\n",
    "            Lambda_h_inverse = np.linalg.inv(self.Lambda[h])\n",
    "            theta_hat_h =  Lambda_h_inverse @ Phi_V_h_plus_one.T @ V_h_plus_one_stack\n",
    "            if np.product(theta_hat_h) < 0:\n",
    "                theta_hat_h -= np.min(theta_hat_h)\n",
    "            self.theta_hat[h] = theta_hat_h / np.sum(theta_hat_h)\n",
    "\n",
    "\n",
    "            # Calculate the Current Stage Value Function \n",
    "            V_h = []\n",
    "            for state in ['x_2', 'x_3', 'x_4', 'x_5']:\n",
    "                if state == 'x_3':\n",
    "                    V_h.append(0)\n",
    "                else:\n",
    "                    Q_h = [self.get_Q_func(h, state, action) for action in self.action_space]\n",
    "                    V_h.append(np.max(Q_h))\n",
    "            self.V_hat[h] = np.array(V_h)\n",
    "     \n",
    "class DRVTR_chi2():\n",
    "    def __init__(self, A, H, lam, dataset, Rho, delta, xi_norm):\n",
    "        self.lam = lam\n",
    "        self.H = H\n",
    "        self.action_space = A\n",
    "        self.w = [np.zeros(4) for _ in range(self.H)]\n",
    "        self.Lambda = [self.lam * np.diag(np.ones(3)) for _ in range(self.H)]\n",
    "        self.Rho = Rho\n",
    "        self.dataset = dataset\n",
    "        Xi = np.full(len(self.action_space[0]), 1)\n",
    "        self.Xi = xi_norm * Xi / np.linalg.norm(Xi, 1)\n",
    "        self.xi_norm = xi_norm\n",
    "        self.delta = delta\n",
    "        self.nu = [np.zeros(4),\n",
    "                   np.array([0,0,0,1]),\n",
    "                   np.array([0,0,0,1])]\n",
    "    \n",
    "    def psi(self, current_state, A):\n",
    "        if current_state == 'x_1':\n",
    "            psi = np.array([1 - self.delta - self.Xi @ A, 0, 0, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_2':\n",
    "            psi = np.array([0, 1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_3':\n",
    "            psi = np.array([0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_4':\n",
    "            psi = np.array([0, 0, 1, 0])\n",
    "        elif current_state == 'x_5':\n",
    "            psi = np.array([0, 0, 0, 1])\n",
    "        else:\n",
    "            raise ValueError(f\"Error: The value {current_state} is not allow!\")\n",
    "        return psi\n",
    "\n",
    "    def phi(self, current_state, A):\n",
    "        if current_state == 'x_1':\n",
    "            phi = np.array(([1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A, 0], \n",
    "                           [1 - self.delta - self.Xi @ A, 0, 0, self.delta + self.Xi @ A], \n",
    "                           [0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A]))\n",
    "        elif current_state == 'x_2':\n",
    "            phi = np.array(([0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A, 0],\n",
    "                            [0, 1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A],\n",
    "                            [0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A]))\n",
    "        elif current_state == 'x_4':\n",
    "            phi = np.array(([0, 0, 1, 0], [0, 0, 1, 0], [0, 0, 1, 0]))\n",
    "        elif current_state == 'x_5':\n",
    "            phi = np.array(([0, 0, 0, 1], [0, 0, 0, 1], [0, 0, 0, 1]))\n",
    "        return phi\n",
    "\n",
    "    def get_action(self, h, state):\n",
    "        Q_h = [self.get_Q_func(h, state, action) for action in self.action_space]\n",
    "        return self.action_space[np.argmax(Q_h)]\n",
    "\n",
    "    def get_Q_func(self, h, state, action):\n",
    "        if h == self.H-1:\n",
    "            reward_feature = self.psi(state, action)\n",
    "            reward_parameter = self.nu[h]\n",
    "            return np.dot(reward_feature, reward_parameter)\n",
    "        else:\n",
    "            if self.Rho[h] == 0:\n",
    "                phi_s_a = self.phi(state, action) # get the basis modes\n",
    "                phi_V_h_plus_one_s_a = [np.dot(phi_s_a[i], self.V_hat[h + 1]) for i in range(3)] # take expectations to get feature\n",
    "                theta_hat = self.theta_hat[h]\n",
    "                result = np.dot(phi_V_h_plus_one_s_a, theta_hat) # take expectation wrt i~theta\n",
    "                reward_feature = self.psi(state, action) \n",
    "                reward_parameter = self.nu[h] # calculate the immediate reward\n",
    "                reward = np.dot(reward_feature, reward_parameter) \n",
    "                return np.min([reward + result, self.H - h]) # reward + future cumulative reward = Qfunc\n",
    "            else:\n",
    "                phi_s_a = self.phi(state, action)\n",
    "                phi_V_h_plus_one_s_a = [np.dot(phi_s_a[i], self.V_hat[h + 1]) for i in range(3)]\n",
    "                theta_hat = self.theta_hat[h]\n",
    "                def Q_alpha(alpha):\n",
    "                    first_part = np.dot(np.minimum(phi_V_h_plus_one_s_a, alpha), theta_hat)\n",
    "                    # second_part = self.Rho[h]*(alpha - np.min(np.minimum(phi_V_h_plus_one_s_a, alpha)))\n",
    "                    first_order_term = np.dot(np.minimum(phi_V_h_plus_one_s_a, alpha), theta_hat)\n",
    "                    second_order_term = np.dot(np.minimum(phi_V_h_plus_one_s_a, alpha) ** 2, theta_hat)\n",
    "                    variance = np.max([second_order_term - first_order_term ** 2, 0])\n",
    "                    second_part= np.sqrt(self.Rho[h] * variance)\n",
    "                    return - first_part + second_part\n",
    "                result = minimize(Q_alpha, self.H/2, method = 'Nelder-Mead', bounds=[(0,self.H)])\n",
    "                reward_feature = self.psi(state, action)\n",
    "                reward_parameter = self.nu[h]\n",
    "                reward = np.dot(reward_feature, reward_parameter)\n",
    "                return np.min([reward - result.fun, self.H - h])\n",
    "\n",
    "    def update_Q(self):\n",
    "        # Backward induction\n",
    "        self.theta_hat = [None for _ in range(self.H-1)] # initialize parameter estimation\n",
    "        self.V_hat = [np.zeros(4) for _ in range(self.H)] # initialize future cumulative reward, i.e., value function\n",
    "        self.V_hat[2] = np.array([0, self.delta+self.xi_norm, 0, 1]) # the last stage value function is the maximum reward which is known\n",
    "        # Start From the Last But One Stage \n",
    "        for h in range(self.H-2, -1, -1):\n",
    "            # Prepare for Ridge Regression\n",
    "            Phi_V_h_plus_one = np.zeros((0,3))\n",
    "            V_h_plus_one_stack = np.zeros(0)\n",
    "            for n in range(self.dataset['k']):\n",
    "                state_temp = self.dataset['state'][n][h]\n",
    "                action_temp = self.dataset['action'][n][h]\n",
    "                phi_temp = self.phi(state_temp, action_temp) # prepare the basis modes for s_h^k a_h^k\n",
    "                phi_V_temp = [np.dot(phi_temp[i], np.array(self.V_hat[h+1])) for i in range(3)] # take expectation to get feature\n",
    "                feature_temp = phi_V_temp\n",
    "                self.Lambda[h] += np.outer(feature_temp, feature_temp)\n",
    "                Phi_V_h_plus_one = np.vstack((Phi_V_h_plus_one, feature_temp))\n",
    "                if self.dataset['state'][n][h+1] == 'x_2':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][0]\n",
    "                elif self.dataset['state'][n][h+1] == 'x_3':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][1]\n",
    "                elif self.dataset['state'][n][h+1] == 'x_4':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][2]\n",
    "                elif self.dataset['state'][n][h+1] == 'x_5':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][3]\n",
    "                else: \n",
    "                    raise ValueError(f\"Error: The value {self.dataset['state'][n][h+1]} is not allow!\")\n",
    "                V_h_plus_one_stack = np.hstack((V_h_plus_one_stack, V_h_plus_one_s_h_plus_one))\n",
    "\n",
    "            # update theta estimation\n",
    "            Lambda_h_inverse = np.linalg.inv(self.Lambda[h])\n",
    "            theta_hat_h =  Lambda_h_inverse @ Phi_V_h_plus_one.T @ V_h_plus_one_stack\n",
    "            if np.product(theta_hat_h) < 0:\n",
    "                theta_hat_h -= np.min(theta_hat_h)\n",
    "            self.theta_hat[h] = theta_hat_h / np.sum(theta_hat_h)\n",
    "            # Calculate the Current Stage Value Function \n",
    "            V_h = []\n",
    "            for state in ['x_2', 'x_3', 'x_4', 'x_5']:\n",
    "                if state == 'x_3':\n",
    "                    V_h.append(0)\n",
    "                else:\n",
    "                    Q_h = [self.get_Q_func(h, state, action) for action in self.action_space]\n",
    "                    V_h.append(np.max(Q_h))\n",
    "            self.V_hat[h] = np.array(V_h)\n",
    "         \n",
    "class VTR():\n",
    "    def __init__(self, A, H, lam, dataset, delta, xi_norm):\n",
    "        self.lam = lam\n",
    "        self.H = H\n",
    "        self.action_space = A\n",
    "        self.w = [np.zeros(4) for _ in range(self.H)]\n",
    "        self.Lambda = [self.lam * np.diag(np.ones(3)) for _ in range(self.H)]\n",
    "        self.dataset = dataset\n",
    "        Xi = np.full(len(self.action_space[0]), 1)\n",
    "        self.Xi = xi_norm * Xi / np.linalg.norm(Xi, 1)\n",
    "        self.xi_norm = xi_norm\n",
    "        self.delta = delta\n",
    "        self.nu = [np.zeros(4),\n",
    "                   np.array([0,0,0,1]),\n",
    "                   np.array([0,0,0,1])]\n",
    "    \n",
    "    def psi(self, current_state, A):\n",
    "        if current_state == 'x_1':\n",
    "            psi = np.array([1 - self.delta - self.Xi @ A, 0, 0, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_2':\n",
    "            psi = np.array([0, 1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_3':\n",
    "            psi = np.array([0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A])\n",
    "        elif current_state == 'x_4':\n",
    "            psi = np.array([0, 0, 1, 0])\n",
    "        elif current_state == 'x_5':\n",
    "            psi = np.array([0, 0, 0, 1])\n",
    "        else:\n",
    "            raise ValueError(f\"Error: The value {current_state} is not allow!\")\n",
    "        return psi\n",
    "\n",
    "    def phi(self, current_state, A):\n",
    "        if current_state == 'x_1':\n",
    "            phi = np.array(([1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A, 0], \n",
    "                           [1 - self.delta - self.Xi @ A, 0, 0, self.delta + self.Xi @ A], \n",
    "                           [0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A])) \n",
    "        elif current_state == 'x_2':\n",
    "            phi = np.array(([0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A, 0],\n",
    "                            [0, 1 - self.delta - self.Xi @ A, 0, self.delta + self.Xi @ A],\n",
    "                            [0, 0, 1 - self.delta - self.Xi @ A, self.delta + self.Xi @ A]))\n",
    "        elif current_state == 'x_4':\n",
    "            phi = np.array(([0, 0, 1, 0], [0, 0, 1, 0], [0, 0, 1, 0]))\n",
    "        elif current_state == 'x_5':\n",
    "            phi = np.array(([0, 0, 0, 1], [0, 0, 0, 1], [0, 0, 0, 1]))\n",
    "        return phi\n",
    "\n",
    "    def get_action(self, h, state):\n",
    "        Q_h = [self.get_Q_func(h, state, action) for action in self.action_space]\n",
    "        return self.action_space[np.argmax(Q_h)]\n",
    "\n",
    "    def get_Q_func(self, h, state, action):\n",
    "        if h == self.H-1: # if we are in the last stage, then the Qfunc is simply the reward func\n",
    "            reward_feature = self.psi(state, action) \n",
    "            reward_parameter = self.nu[h]\n",
    "            return np.dot(reward_feature, reward_parameter)\n",
    "        else:\n",
    "            phi_s_a = self.phi(state, action) # get the basis modes\n",
    "            phi_V_h_plus_one_s_a = [np.dot(phi_s_a[i], self.V_hat[h + 1]) for i in range(3)] # take expectations to get feature\n",
    "            theta_hat = self.theta_hat[h]\n",
    "            result = np.dot(phi_V_h_plus_one_s_a, theta_hat) # take expectation wrt i~theta\n",
    "            reward_feature = self.psi(state, action) \n",
    "            reward_parameter = self.nu[h] # calculate the immediate reward\n",
    "            reward = np.dot(reward_feature, reward_parameter) \n",
    "            return np.min([reward + result, self.H - h]) # reward + future cumulative reward = Qfunc\n",
    "\n",
    "    def update_Q(self):\n",
    "        # Backward induction\n",
    "        self.theta_hat = [None for _ in range(self.H-1)] # initialize parameter estimation\n",
    "        self.V_hat = [np.zeros(4) for _ in range(self.H)] # initialize future cumulative reward, i.e., value function\n",
    "        self.V_hat[2] = np.array([0, self.delta+self.xi_norm, 0, 1]) # the last stage value function is the maximum reward which is known\n",
    "        # Start From the Last But One Stage \n",
    "        for h in range(self.H-2, -1, -1):\n",
    "            # Prepare for Ridge Regression\n",
    "            Phi_V_h_plus_one = np.zeros((0,3))\n",
    "            V_h_plus_one_stack = np.zeros(0)\n",
    "            for n in range(self.dataset['k']):\n",
    "                state_temp = self.dataset['state'][n][h]\n",
    "                action_temp = self.dataset['action'][n][h]\n",
    "                phi_temp = self.phi(state_temp, action_temp) # prepare the basis modes for s_h^k a_h^k\n",
    "                phi_V_temp = [np.dot(phi_temp[i], np.array(self.V_hat[h+1])) for i in range(3)] # take expectation to get feature\n",
    "                feature_temp = phi_V_temp\n",
    "                self.Lambda[h] += np.outer(feature_temp, feature_temp)\n",
    "                Phi_V_h_plus_one = np.vstack((Phi_V_h_plus_one, feature_temp))\n",
    "                if self.dataset['state'][n][h+1] == 'x_2':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][0]\n",
    "                elif self.dataset['state'][n][h+1] == 'x_3':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][1]\n",
    "                elif self.dataset['state'][n][h+1] == 'x_4':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][2]\n",
    "                elif self.dataset['state'][n][h+1] == 'x_5':\n",
    "                    V_h_plus_one_s_h_plus_one = self.V_hat[h+1][3]\n",
    "                else: \n",
    "                    raise ValueError(f\"Error: The value {self.dataset['state'][n][h+1]} is not allow!\")\n",
    "                V_h_plus_one_stack = np.hstack((V_h_plus_one_stack, V_h_plus_one_s_h_plus_one))\n",
    "\n",
    "            # update theta estimation\n",
    "            Lambda_h_inverse = np.linalg.inv(self.Lambda[h])\n",
    "            theta_hat_h =  Lambda_h_inverse @ Phi_V_h_plus_one.T @ V_h_plus_one_stack\n",
    "            if np.product(theta_hat_h) < 0:\n",
    "                theta_hat_h -= np.min(theta_hat_h)\n",
    "            self.theta_hat[h] = theta_hat_h / np.sum(theta_hat_h)\n",
    "            # Calculate the Current Stage Value Function \n",
    "            V_h = []\n",
    "            for state in ['x_2', 'x_3', 'x_4', 'x_5']:\n",
    "                if state == 'x_3':\n",
    "                    V_h.append(0)\n",
    "                else:\n",
    "                    Q_h = [self.get_Q_func(h, state, action) for action in self.action_space]\n",
    "                    V_h.append(np.max(Q_h))\n",
    "            self.V_hat[h] = np.array(V_h) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 82,
   "metadata": {},
   "outputs": [],
   "source": [
    "def Offline_Dataset_Collection(sample_size, env, seed=1):\n",
    "    np.random.seed(seed)\n",
    "    history = {'k': 0, 'r': [], 'state': [], 'action': []}\n",
    "    epoch = sample_size\n",
    "    for t in range(epoch):\n",
    "        env.reset()\n",
    "        for h in range(env.H):\n",
    "            random_action_index = np.random.choice(range(0, len(env.action_space)), size = 1)[0]\n",
    "            action = env.action_space[random_action_index]\n",
    "            env.step(action)\n",
    "            \n",
    "        # log the trajectory\n",
    "        history['r'].append(env.R)\n",
    "        history['state'].append(env.S)\n",
    "        history['action'].append(env.A)\n",
    "        history['k'] += 1\n",
    "    return history\n",
    "def train_once(dataset, action_space, H, lam, delta, xi_norm):\n",
    "    agent = VTR(A=action_space, H=H, lam=lam, dataset=dataset, delta=delta, xi_norm=xi_norm)\n",
    "    agent.update_Q()\n",
    "    return agent\n",
    "def train_once_DR_TV(dataset, action_space, H, lam, Rho, delta, xi_norm):\n",
    "    agent = DRVTR_TV(A=action_space, H=H, lam=lam, dataset=dataset, Rho=Rho, delta=delta, xi_norm=xi_norm)\n",
    "    agent.update_Q()\n",
    "    return agent\n",
    "def train_once_DR_KL(dataset, action_space, H, lam, Rho, delta, xi_norm):\n",
    "    agent = DRVTR_KL(A=action_space, H=H, lam=lam, dataset=dataset, Rho=Rho, delta=delta, xi_norm=xi_norm)\n",
    "    agent.update_Q()\n",
    "    return agent\n",
    "def train_once_DR_chi2(dataset, action_space, H, lam, Rho, delta, xi_norm):\n",
    "    agent = DRVTR_chi2(A=action_space, H=H, lam=lam, dataset=dataset, Rho=Rho, delta=delta, xi_norm=xi_norm)\n",
    "    agent.update_Q()\n",
    "    return agent"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "T1 = 500\n",
    "H = 3\n",
    "rho_TV = 0.7\n",
    "rho_KL = 10\n",
    "rho_chi2 = 20\n",
    "lam = 0.1\n",
    "actions = list(product([-1, 1], repeat=4))\n",
    "action_space = [np.array(action) for action in actions]\n",
    "delta = 0.3\n",
    "xi_norm = 0.2\n",
    "Rho_TV = [rho_TV, 0]\n",
    "Rho_KL = [rho_KL, 0]\n",
    "Rho_chi2 = [rho_chi2, 0]\n",
    "replication = 10\n",
    "agent_dic = {}\n",
    "DR_agent_TV_dic = {}\n",
    "DR_agent_KL_dic = {}\n",
    "DR_agent_chi2_dic = {}\n",
    "env = LinearMixtureMDP(action_space, delta, xi_norm)\n",
    "\n",
    "\n",
    "for rep in range(replication):\n",
    "    Offline_Dataset = Offline_Dataset_Collection(T1, env, seed=rep)\n",
    "    agent = train_once(dataset=Offline_Dataset, action_space=action_space, H=H, lam=lam, delta=delta, xi_norm=xi_norm)\n",
    "    DR_agent_TV = train_once_DR_TV(dataset=Offline_Dataset, action_space=action_space, H=H, lam=lam, Rho=Rho_TV, delta=delta, xi_norm=xi_norm)\n",
    "    DR_agent_KL = train_once_DR_KL(dataset=Offline_Dataset, action_space=action_space, H=H, lam=lam, Rho=Rho_KL, delta=delta, xi_norm=xi_norm)\n",
    "    DR_agent_chi2 = train_once_DR_chi2(dataset=Offline_Dataset, action_space=action_space, H=H, lam=lam, Rho=Rho_chi2, delta=delta, xi_norm=xi_norm)\n",
    "    \n",
    "    agent_dic[str(rep)] = agent\n",
    "    DR_agent_TV_dic[str(rep)] = DR_agent_TV\n",
    "    DR_agent_KL_dic[str(rep)] = DR_agent_KL\n",
    "    DR_agent_chi2_dic[str(rep)] = DR_agent_chi2\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 99,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "Points = 10\n",
    "Perturbation = [x / Points for x in range(Points+1)]\n",
    "# Perturbation = [0.5]\n",
    "T2 = 100\n",
    "R_VTR = []\n",
    "R_DRVTR_TV = []\n",
    "R_DRVTR_KL = []\n",
    "R_DRVTR_chi2 = []\n",
    "env = LinearMixtureMDP(action_space, delta, xi_norm)\n",
    "for q in Perturbation:\n",
    "    REWARD = 0\n",
    "    REWARD_DR_TV = 0\n",
    "    REWARD_DR_KL = 0\n",
    "    REWARD_DR_chi2 = 0\n",
    "    for rep in range(replication):\n",
    "        reward = 0\n",
    "        reward_DR_TV = 0\n",
    "        reward_DR_KL = 0\n",
    "        reward_DR_chi2 = 0\n",
    "        env_test = LinearMixtureMDP_test(env, q=q, seed=rep)\n",
    "        env_test_DR_TV = LinearMixtureMDP_test(env, q=q, seed=rep)\n",
    "        env_test_DR_KL = LinearMixtureMDP_test(env, q=q, seed=rep)\n",
    "        env_test_DR_chi2 = LinearMixtureMDP_test(env, q=q, seed=rep)\n",
    "        agent = agent_dic[str(rep)]\n",
    "        DR_agent_TV = DR_agent_TV_dic[str(rep)]\n",
    "        DR_agent_KL = DR_agent_KL_dic[str(rep)]\n",
    "        DR_agent_chi2 = DR_agent_chi2_dic[str(rep)]\n",
    "\n",
    "\n",
    "        for t in range(T2):\n",
    "            env_test.reset()\n",
    "            env_test_DR_TV.reset()\n",
    "            env_test_DR_KL.reset()\n",
    "            env_test_DR_chi2.reset()\n",
    "            for h in range(H):\n",
    "                # VTR\n",
    "                current_state = env_test.current_state\n",
    "                action = agent.get_action(h, current_state)\n",
    "                env_test.step(action)\n",
    "            \n",
    "                # DRVTR_TV\n",
    "                current_state_DR_TV = env_test_DR_TV.current_state\n",
    "                action_DR_TV = DR_agent_TV.get_action(h, current_state_DR_TV)\n",
    "                env_test_DR_TV.step(action_DR_TV)\n",
    "\n",
    "                # DRVTR_KL\n",
    "                current_state_DR_KL = env_test_DR_KL.current_state\n",
    "                action_DR_KL = DR_agent_KL.get_action(h, current_state_DR_KL)\n",
    "                env_test_DR_KL.step(action_DR_KL)\n",
    "\n",
    "                # DRVTR_chi2\n",
    "                current_state_DR_chi2 = env_test_DR_chi2.current_state\n",
    "                action_DR_chi2 = DR_agent_chi2.get_action(h, current_state_DR_chi2)\n",
    "                env_test_DR_chi2.step(action_DR_chi2)\n",
    "\n",
    "            reward += np.sum(env_test.R) / T2   \n",
    "            reward_DR_TV += np.sum(env_test_DR_TV.R) / T2  \n",
    "            reward_DR_KL += np.sum(env_test_DR_KL.R) / T2  \n",
    "            reward_DR_chi2 += np.sum(env_test_DR_chi2.R) / T2  \n",
    "             \n",
    "        \n",
    "        REWARD += reward / replication \n",
    "        REWARD_DR_TV += reward_DR_TV / replication \n",
    "        REWARD_DR_KL += reward_DR_KL / replication \n",
    "        REWARD_DR_chi2 += reward_DR_chi2 / replication \n",
    "\n",
    "    R_VTR.append(REWARD)\n",
    "    R_DRVTR_TV.append(REWARD_DR_TV)\n",
    "    R_DRVTR_KL.append(REWARD_DR_KL)\n",
    "    R_DRVTR_chi2.append(REWARD_DR_chi2)\n",
    "\n",
    "plt.plot(Perturbation, R_VTR, label = 'VTR')\n",
    "plt.plot(Perturbation, R_DRVTR_TV, label = 'DRVTR_TV')\n",
    "plt.plot(Perturbation, R_DRVTR_KL, label = 'DRVTR_KL')\n",
    "plt.plot(Perturbation, R_DRVTR_chi2, label = 'DRVTR_chi2')\n",
    "plt.legend(fontsize=16)\n",
    "plt.xlabel('Perturbation', size=16)\n",
    "plt.ylabel('Average reward', size=16)\n",
    "plt.savefig(f'robustness_{delta}_{xi_norm}_{rho_TV}_{rho_KL}_{rho_chi2}.pdf', dpi=1000, bbox_inches='tight', pad_inches=0.0)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 85,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'/Users/liuzhishuai/Desktop'"
      ]
     },
     "execution_count": 85,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import os\n",
    "os.getcwd()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([ 1,  1,  1, -1])"
      ]
     },
     "execution_count": 72,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "DR_agent_TV.get_action(0, 'x_1')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 73,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[0.0727603765739645,\n",
       " 0.3448969016864375,\n",
       " 0.3448969016864375,\n",
       " 0.6170334267989102,\n",
       " 0.3448969016864375,\n",
       " 0.6170334267989102,\n",
       " 0.6170334267989102,\n",
       " 0.7166451371621465,\n",
       " 0.3448969016864375,\n",
       " 0.6170334267989102,\n",
       " 0.6170334267989102,\n",
       " 0.7166451371621465,\n",
       " 0.6170334267989102,\n",
       " 0.7166451371621465,\n",
       " 0.7166451371621466,\n",
       " 0.5449225362991749]"
      ]
     },
     "execution_count": 73,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "[DR_agent_TV.get_Q_func(0, 'x_1', action) for action in action_space]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 74,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[array([0.2565022, 0.5      , 0.2434978]),\n",
       " array([0.23192265, 0.71384541, 0.05423194])]"
      ]
     },
     "execution_count": 74,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "DR_agent_TV.theta_hat"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 75,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([[0.1, 0.8, 0.1],\n",
       "       [0.1, 0.8, 0.1]])"
      ]
     },
     "execution_count": 75,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "env.theta"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 76,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[array([0.2565022, 0.5      , 0.2434978]),\n",
       " array([0.23192265, 0.71384541, 0.05423194])]"
      ]
     },
     "execution_count": 76,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "DR_agent_KL.theta_hat"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 77,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[array([0.2565022, 0.5      , 0.2434978]),\n",
       " array([0.23192265, 0.71384541, 0.05423194])]"
      ]
     },
     "execution_count": 77,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "DR_agent_chi2.theta_hat"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 78,
   "metadata": {},
   "outputs": [],
   "source": [
    "vec = np.array([1,2,3,4])\n",
    "vec = vec / np.sum(vec)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([0.1, 0.2, 0.3, 0.4])"
      ]
     },
     "execution_count": 26,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "vec"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.7.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
