{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Sensitivity of optimistic NPG primal-dual for finite constrained MDPs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'\\nSet-up:\\n1) Softmax policy \\n2) Bounded rewards\\n3) Many states and actions\\n4) (Natural) policy gradient + primal-dual method\\n   Regularized (Natural) policy gradient + primal-dual method\\n   (Natural) policy gradient + optimistic primal-dual method\\n   (Natural) policy gradient + PID primal-dual method\\n5) Stochastic version\\n'"
      ]
     },
     "execution_count": 1,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "'''\n",
    "Set-up:\n",
    "1) Softmax policy \n",
    "2) Bounded rewards\n",
    "3) Many states and actions\n",
    "4) (Natural) policy gradient + optimistic primal-dual method (OMWU version)\n",
    "   (Natural) policy gradient + optimistic primal-dual method (ReLOAD version)\n",
    "   \n",
    "References:\n",
    "1) ReLOAD  \n",
    "   ReLOAD: Reinforcement Learning with Optimistic Ascent-Descent for Last-Iterate Convergence in Constrained MDPs\n",
    "   arXiv:2302.01275v2 \n",
    "'''"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "np.set_printoptions(formatter={'float': lambda x: \"{0:0.6f}\".format(x)})\n",
    "%matplotlib inline\n",
    "import matplotlib\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.optimize import linprog"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "## Random Seed\n",
    "np.random.seed(10) \n",
    "## Problem Setup\n",
    "gamma = 0.9\n",
    "n, m = 20, 5 # s, a\n",
    "'''\n",
    "Randomly generated probability transition matrix P((s,a) -> s') in R^{|S||A| x |S|}\n",
    "Each row sums up to one\n",
    "'''\n",
    "raw_transition = np.random.uniform(0,1,size=(n*m,n))\n",
    "prob_transition = raw_transition/raw_transition.sum(axis=1,keepdims=1)\n",
    "'''\n",
    "Random positive rewards\n",
    "'''\n",
    "reward = np.random.uniform(0,1,size=(n*m))\n",
    "\n",
    "'''\n",
    "Random utilities between -1 and +1\n",
    "'''\n",
    "utility = np.random.uniform(-1,1,size=(n*m))\n",
    "\n",
    "'''\n",
    "Start state distribution\n",
    "'''\n",
    "# uniform\n",
    "rho = np.ones(n)/n\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "\"\"\"\n",
    "Input: Discrete probability distributions p, q : array-like, dtype=float, shape=n\n",
    "Output: Kullback-Leibler divergence D(p||q)      \n",
    "\"\"\"\n",
    "def kl_div(p, q):\n",
    "    p = np.asarray(p, dtype=np.float)\n",
    "    q = np.asarray(q, dtype=np.float)\n",
    "    \n",
    "    return np.sum(np.where(p != 0, p * np.log(p / q), 0))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Input: theta as an array and \n",
    "Output: array of probabilites corresponding to each state: [\\pi_{s_1}(.), ...., \\pi_{s_n}(.)]\n",
    "'''\n",
    "def theta_to_policy(theta,n,m):\n",
    "    prob = []\n",
    "    for i in range(n):\n",
    "        norm = np.sum(np.exp(theta[m*i:m*(i+1)]))\n",
    "        for j in range(m*i,m*(i+1)):\n",
    "            prob.append(np.exp(theta[j])/norm)\n",
    "            \n",
    "    return np.asarray(prob)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Input: theta as an array and \n",
    "Output: array of probabilites corresponding to each state: [\\pi_{s_1}(.), ...., \\pi_{s_n}(.)]\n",
    "'''\n",
    "def theta_to_policy_naive(theta,n,m):\n",
    "    prob = []\n",
    "    for i in range(n):\n",
    "        norm = np.sum(theta[m*i:m*(i+1)])\n",
    "        for j in range(m*i,m*(i+1)):\n",
    "            prob.append(theta[j]/norm)\n",
    "            \n",
    "    return np.asarray(prob)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "\"\"\"\n",
    "License: BSD\n",
    "Author: Mathieu Blondel\n",
    "Implements three algorithms for projecting a vector onto the simplex: sort, pivot and bisection.\n",
    "For details and references, see the following paper:\n",
    "Large-scale Multiclass Support Vector Machine Training via Euclidean Projection onto the Simplex\n",
    "Mathieu Blondel, Akinori Fujino, and Naonori Ueda.\n",
    "ICPR 2014.\n",
    "http://www.mblondel.org/publications/mblondel-icpr2014.pdf\n",
    "\"\"\"\n",
    "\n",
    "def projection_simplex_sort(v, z=1):\n",
    "    n_features = v.shape[0]\n",
    "    u = np.sort(v)[::-1]\n",
    "    cssv = np.cumsum(u) - z\n",
    "    ind = np.arange(n_features) + 1\n",
    "    cond = u - cssv / ind > 0\n",
    "    rho = ind[cond][-1]\n",
    "    theta = cssv[cond][-1] / float(rho)\n",
    "    w = np.maximum(v - theta, 0)\n",
    "    return w\n",
    "\n",
    "\n",
    "def projection_simplex_pivot(v, z=1, random_state=None):\n",
    "    rs = np.random.RandomState(random_state)\n",
    "    n_features = len(v)\n",
    "    U = np.arange(n_features)\n",
    "    s = 0\n",
    "    rho = 0\n",
    "    while len(U) > 0:\n",
    "        G = []\n",
    "        L = []\n",
    "        k = U[rs.randint(0, len(U))]\n",
    "        ds = v[k]\n",
    "        for j in U:\n",
    "            if v[j] >= v[k]:\n",
    "                if j != k:\n",
    "                    ds += v[j]\n",
    "                    G.append(j)\n",
    "            elif v[j] < v[k]:\n",
    "                L.append(j)\n",
    "        drho = len(G) + 1\n",
    "        if s + ds - (rho + drho) * v[k] < z:\n",
    "            s += ds\n",
    "            rho += drho\n",
    "            U = L\n",
    "        else:\n",
    "            U = G\n",
    "    theta = (s - z) / float(rho)\n",
    "    return np.maximum(v - theta, 0)\n",
    "\n",
    "\n",
    "def projection_simplex_bisection(v, z=1, tau=0.0001, max_iter=1000):\n",
    "    func = lambda x: np.sum(np.maximum(v - x, 0)) - z\n",
    "    lower = np.min(v) - z / len(v)\n",
    "    upper = np.max(v)\n",
    "\n",
    "    for it in range(max_iter):\n",
    "        midpoint = (upper + lower) / 2.0\n",
    "        value = func(midpoint)\n",
    "\n",
    "        if abs(value) <= tau:\n",
    "            break\n",
    "\n",
    "        if value <= 0:\n",
    "            upper = midpoint\n",
    "        else:\n",
    "            lower = midpoint\n",
    "\n",
    "    return np.maximum(v - midpoint, 0)\n",
    "\n",
    "# if __name__ == '__main__':\n",
    "#     v = np.array([1.1, 0.2, 0.2])\n",
    "#     z = 1\n",
    "\n",
    "#     w = projection_simplex_sort(v, z)\n",
    "#     print(np.sum(w))\n",
    "#     print(w)\n",
    "\n",
    "#     w = projection_simplex_pivot(v, z)\n",
    "#     print(np.sum(w))\n",
    "#     print(w)\n",
    "\n",
    "#     w = projection_simplex_bisection(v, z)\n",
    "#     print(np.sum(w))\n",
    "#     print(w)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Input: theta as an array and \n",
    "Output: array of probabilites corresponding to each state: [\\pi_{s_1}(.), ...., \\pi_{s_n}(.)]\n",
    "'''\n",
    "def project_to_policy(theta,n,m):\n",
    "    prob = []\n",
    "    prob_pers = []\n",
    "    for i in range(n):\n",
    "#         norm = np.sum(np.exp(theta[m*i:m*(i+1)]))\n",
    "        prob_pers = projection_simplex_sort(theta[m*i:m*(i+1)], z=1)\n",
    "        for j in range(m):\n",
    "            prob.append(prob_pers[j])\n",
    "            \n",
    "    return np.asarray(prob)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Get \\Pi_{\\pi}((s) -> (s,a)) in R^{|S| x |S||A|} matrix corresponding to the policy \\pi using the prob vector\n",
    "'''\n",
    "def get_Pi(prob,n,m):\n",
    "    Pi = np.zeros((n,n*m))\n",
    "    for i in range(n):\n",
    "        Pi[i,i*m:(i+1)*m] = prob[i*m:(i+1)*m]\n",
    "    \n",
    "    return Pi"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Input: probability vector, state, action\n",
    "Output: \\nabla_{\\theta} \\pi_{\\theta}(s,a)\n",
    "\n",
    "States go from 0 to n-1 and actons from 0 to m-1\n",
    "'''\n",
    "def grad_state_action(prob,state,action):\n",
    "    grad = np.zeros(n*m)\n",
    "    for j in range(0,m):\n",
    "        if j == action:\n",
    "            grad[m*state + j] = prob[m*state + j]*(1-prob[m*state + j])\n",
    "        else:\n",
    "            grad[m*state + j] = -prob[m*state + action]*prob[m*state + j]\n",
    "            \n",
    "    return grad\n",
    "\n",
    "def grad_state(qvals,prob,state):\n",
    "    grad = np.sum([qvals[state*m + i]*grad_state_action(prob,state,i) for i in range(0,m)],axis=0)\n",
    "    return grad\n",
    "\n",
    "def grad(qvals,prob,d_pi):\n",
    "    grad = np.sum([d_pi[i]*grad_state(qvals,prob,i) for i in range(0,n)],axis=0)\n",
    "    return grad"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Input: probability vector\n",
    "Output: Fisher information matrix\n",
    "        \\nabla_{\\theta} \\pi_{\\theta}(s,a) x {\\nabla_{\\theta} \\pi_{\\theta}(s,a)}^T\n",
    "'''\n",
    "def Fisher_info(prob,d_pi):\n",
    "    qvals_one = np.ones(n*m)\n",
    "    grad = np.sum([d_pi[i]*grad_state(qvals_one,prob,i) for i in range(0,n)],axis=0)\n",
    "    fisher = np.outer(grad,grad)+1e-3*np.identity(n*m)\n",
    "    return fisher"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "The overall reward function \\ell(\\theta)\n",
    "'''\n",
    "def ell(qvals,prob,rho):\n",
    "    V = np.zeros(n)\n",
    "    for i in range(n):\n",
    "        V[i] = np.sum([qvals[i*m + j]*prob[i*m + j] for j in range(m)])\n",
    "    \n",
    "    ell = np.dot(V,rho)\n",
    "    return ell"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "The overall reward advantage function \\ell(\\theta)\n",
    "'''\n",
    "def avals(qvals,prob):\n",
    "    V = np.zeros(n)\n",
    "    for i in range(n):\n",
    "        V[i] = np.sum([qvals[i*m + j]*prob[i*m + j] for j in range(m)])\n",
    "    \n",
    "    A = qvals\n",
    "    for i in range(n):\n",
    "        for j in range(m):\n",
    "            A[i*m + j] = qvals[i*m + j] - V[i]\n",
    "        \n",
    "    return A"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "The projection function\n",
    "Input: a scalar \n",
    "Output: a scalar in the interval [0 C]\n",
    "'''\n",
    "def proj(scalar,gamma):\n",
    "    offset = 1000/(1-gamma)\n",
    "    if scalar < 0:\n",
    "        scalar = 0\n",
    "\n",
    "    if scalar > offset:\n",
    "        scalar = offset\n",
    "\n",
    "    return scalar"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Policy Iteration to check feasibility "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "raw_vec = np.random.uniform(0,1,size=(n,m))\n",
    "prob_vec = raw_vec/raw_vec.sum(axis=1,keepdims=1)\n",
    "init_policy = prob_vec.flatten()## Policy Iteration to get the optimal policy"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Policy iteration function\n",
    "'''\n",
    "def policy_iter(q_vals,n,m):\n",
    "    new_policy = np.zeros(n*m)\n",
    "    for i in range(n):\n",
    "        idx = np.argmax(q_vals[i*m:(i+1)*m])\n",
    "        new_policy[i*m + idx] = 1\n",
    "    \n",
    "    return new_policy       "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Starting policy [0.022861 0.022257 0.496791 0.417151 0.040940 0.260055 0.020741 0.239421\n",
      " 0.113255 0.366528 0.077009 0.252897 0.293405 0.147306 0.229383 0.234235\n",
      " 0.114754 0.214643 0.266094 0.170273 0.019934 0.289380 0.444991 0.163370\n",
      " 0.082325 0.184145 0.550146 0.115828 0.008301 0.141579 0.193071 0.128147\n",
      " 0.056816 0.408135 0.213831 0.164747 0.271480 0.219975 0.227132 0.116667\n",
      " 0.170695 0.097732 0.193326 0.297322 0.240925 0.204368 0.076399 0.025505\n",
      " 0.397828 0.295900 0.232097 0.206203 0.026091 0.237565 0.298044 0.098219\n",
      " 0.185154 0.083181 0.392678 0.240769 0.392924 0.090288 0.311557 0.079063\n",
      " 0.126168 0.106657 0.096738 0.356238 0.324030 0.116337 0.184156 0.268226\n",
      " 0.030368 0.122968 0.394283 0.506945 0.038054 0.101757 0.114022 0.239222\n",
      " 0.075823 0.464715 0.061802 0.241150 0.156510 0.268392 0.088919 0.131564\n",
      " 0.277810 0.233316 0.377841 0.216636 0.191506 0.064988 0.149028 0.467008\n",
      " 0.035160 0.104154 0.047115 0.346563]\n",
      "Final policy [0.000000 0.000000 0.000000 0.000000 1.000000 0.000000 1.000000 0.000000\n",
      " 0.000000 0.000000 0.000000 0.000000 0.000000 0.000000 1.000000 0.000000\n",
      " 1.000000 0.000000 0.000000 0.000000 0.000000 0.000000 0.000000 0.000000\n",
      " 1.000000 0.000000 0.000000 0.000000 1.000000 0.000000 0.000000 0.000000\n",
      " 0.000000 1.000000 0.000000 0.000000 1.000000 0.000000 0.000000 0.000000\n",
      " 0.000000 0.000000 0.000000 1.000000 0.000000 0.000000 0.000000 0.000000\n",
      " 0.000000 1.000000 0.000000 1.000000 0.000000 0.000000 0.000000 0.000000\n",
      " 0.000000 1.000000 0.000000 0.000000 0.000000 0.000000 0.000000 0.000000\n",
      " 1.000000 0.000000 0.000000 0.000000 0.000000 1.000000 1.000000 0.000000\n",
      " 0.000000 0.000000 0.000000 0.000000 1.000000 0.000000 0.000000 0.000000\n",
      " 0.000000 0.000000 1.000000 0.000000 0.000000 1.000000 0.000000 0.000000\n",
      " 0.000000 0.000000 0.000000 0.000000 0.000000 1.000000 0.000000 0.000000\n",
      " 1.000000 0.000000 0.000000 0.000000]\n"
     ]
    }
   ],
   "source": [
    "curr_policy = np.random.uniform(0,1,size=(n*m))\n",
    "new_policy = init_policy\n",
    "print('Starting policy',init_policy)\n",
    "\n",
    "while np.count_nonzero(curr_policy - new_policy) > 0:\n",
    "    curr_policy = new_policy\n",
    "    Pi = get_Pi(curr_policy,n,m)\n",
    "    mat = np.identity(n*m) - gamma*np.matmul(prob_transition,Pi)\n",
    "    q_vals_utility = np.dot(np.linalg.inv(mat),utility)\n",
    "    new_policy = policy_iter(q_vals_utility,n,m)\n",
    "    \n",
    "print('Final policy',new_policy)\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "5.556458336351498\n"
     ]
    }
   ],
   "source": [
    "ell_utility_star = ell(q_vals_utility,new_policy,rho)\n",
    "print(ell_utility_star)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Compute the optimal reward value from LP"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Optimal reward value: 8.163862517858446\n",
      "Optimal policy: [0.000000 1.000000 0.000000 0.000000 0.000000 0.000000 1.000000 0.000000\n",
      " 0.000000 0.000000 0.000000 0.000000 1.000000 0.000000 0.000000 0.000000\n",
      " 0.999999 0.000001 0.000000 0.000000 0.000000 1.000000 0.000000 0.000000\n",
      " 0.000000 0.000000 0.000000 0.000000 0.000000 1.000000 0.000000 0.000000\n",
      " 0.000000 1.000000 0.000000 1.000000 0.000000 0.000000 0.000000 0.000000\n",
      " 1.000000 0.000000 0.000000 0.000000 0.000000 0.000000 0.000000 0.000001\n",
      " 0.000000 0.999999 0.000000 1.000000 0.000000 0.000000 0.000000 0.000000\n",
      " 1.000000 0.000000 0.000000 0.000000 0.000000 0.000000 0.000000 1.000000\n",
      " 0.000000 0.000000 0.000000 0.000000 0.000000 1.000000 0.350830 0.000000\n",
      " 0.649170 0.000000 0.000000 0.000000 0.000000 0.000000 0.000000 1.000000\n",
      " 0.000000 0.000000 0.000000 1.000000 0.000000 0.000000 1.000000 0.000000\n",
      " 0.000000 0.000000 0.000000 0.000000 0.000000 1.000000 0.000000 0.000000\n",
      " 0.000000 0.000000 0.000000 1.000000]\n"
     ]
    }
   ],
   "source": [
    "## linear programming solver\n",
    "\n",
    "# minimize c @ x\n",
    "# \n",
    "# such that \n",
    "#          A_ub @ x <= b_ub\n",
    "#          A_eq @ x == b_eq\n",
    "#          lb <= x <= ub\n",
    "\n",
    "c = -reward\n",
    "A_ub = -utility.reshape(1, n*m)\n",
    "b_ub = np.zeros(1)\n",
    "\n",
    "prob_transition_lp = np.transpose(prob_transition)\n",
    "\n",
    "E_sum = np.full_like(prob_transition_lp, 0)\n",
    "for i in range(n):\n",
    "    E_sum[i,m*i:m*(i+1)] = np.ones(m)\n",
    "    \n",
    "A_eq = E_sum - gamma*prob_transition_lp\n",
    "b_eq = rho\n",
    "\n",
    "lb = np.zeros(n*m)\n",
    "ub = np.ones(n*m)\n",
    "ub = ub/(1-gamma)\n",
    "bounds = np.transpose([lb, ub])\n",
    "\n",
    "res = linprog(c, A_ub, b_ub, A_eq, b_eq, bounds)\n",
    "\n",
    "optimal_reward_value_LP = -res.fun\n",
    "optimal_policy_LP = theta_to_policy_naive(res.x,n,m)\n",
    "\n",
    "print('Optimal reward value:',optimal_reward_value_LP)\n",
    "\n",
    "print('Optimal policy:',optimal_policy_LP)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Optimistic NPG primal dual method (OMWU version)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Policy search via optimistic NPG primal dual method\n",
    "Input: n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates   \n",
    "Output: function values of reward and utility  \n",
    "'''\n",
    "\n",
    "def OMWU_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates, optimal_policy_LP):\n",
    "\n",
    "    theta = np.random.uniform(0,1,size=n*m)\n",
    "    theta_h = np.random.uniform(0,1,size=n*m)\n",
    "    dual = 0\n",
    "    dual_h = 0\n",
    "    reward_value = []\n",
    "    utility_value = []\n",
    "    policy_distance = []\n",
    "    for k in range(total_iterates):\n",
    "    \n",
    "        # policy evaluation of theta_h\n",
    "        prob = theta_to_policy(theta_h,n,m)\n",
    "    \n",
    "        Pi = get_Pi(prob,n,m)\n",
    "        mat = np.identity(n*m) - gamma*np.matmul(prob_transition,Pi)\n",
    "        qvals_reward_h = np.dot(np.linalg.inv(mat),reward)\n",
    "        qvals_utility_h = np.dot(np.linalg.inv(mat),utility)\n",
    "    \n",
    "        # optimistic step for (theta, dual)\n",
    "        prob = theta_to_policy(theta,n,m)\n",
    "\n",
    "        Pi = get_Pi(prob,n,m)\n",
    "        mat = np.identity(n*m) - gamma*np.matmul(prob_transition,Pi)\n",
    "        qvals_reward = np.dot(np.linalg.inv(mat),reward)\n",
    "        qvals_utility = np.dot(np.linalg.inv(mat),utility)\n",
    "\n",
    "        P_theta = np.matmul(Pi,prob_transition)\n",
    "        d_pi = (1-gamma)*np.dot(np.transpose((np.linalg.inv(np.identity(n) - gamma*P_theta))),rho)\n",
    "    \n",
    "        # natural gradient \n",
    "        qvals = qvals_reward + dual*qvals_utility\n",
    "    \n",
    "        # natural gradient ascent\n",
    "        theta = theta_h + stepsize*avals(qvals,prob)\n",
    "    \n",
    "        # dual desceent \n",
    "        violation_gradient = ell(qvals_utility,prob,rho) \n",
    "        dual = dual_h - stepsize*violation_gradient\n",
    "        dual = proj(dual,gamma)\n",
    "\n",
    "        # optimistic step for (theta_h, dual_h)\n",
    "        prob = theta_to_policy(theta,n,m)\n",
    "    \n",
    "        Pi = get_Pi(prob,n,m)\n",
    "        mat = np.identity(n*m) - gamma*np.matmul(prob_transition,Pi)\n",
    "        qvals_reward = np.dot(np.linalg.inv(mat),reward)\n",
    "        qvals_utility = np.dot(np.linalg.inv(mat),utility)\n",
    "    \n",
    "        P_theta = np.matmul(Pi,prob_transition)\n",
    "        d_pi = (1-gamma)*np.dot(np.transpose((np.linalg.inv(np.identity(n) - gamma*P_theta))),rho)\n",
    "    \n",
    "        # natural gradient \n",
    "        qvals = qvals_reward + dual*qvals_utility\n",
    "    \n",
    "        # natural gradient ascent\n",
    "        theta_h = theta_h + stepsize*avals(qvals,prob)\n",
    "    \n",
    "        # dual desceent \n",
    "        violation_gradient = ell(qvals_utility,prob,rho)\n",
    "        dual_h = dual_h - stepsize*violation_gradient\n",
    "        dual_h = proj(dual_h,gamma)\n",
    "    \n",
    "        if k % 1 == 0:\n",
    "            avg_reward = ell(qvals_reward_h,prob,rho)\n",
    "            avg_utility = ell(qvals_utility_h,prob,rho)\n",
    "            reward_value.append(avg_reward)\n",
    "            utility_value.append(avg_utility)\n",
    "            \n",
    "            policy_norm_squared = np.inner(prob - optimal_policy_LP, prob - optimal_policy_LP) \n",
    "            policy_distance.append(policy_norm_squared)\n",
    "                \n",
    "    return reward_value, utility_value, policy_distance"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Optimistic NPG primal dual method (ReLOAD version)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Policy search via optimistic NPG primal dual method\n",
    "Input: n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates   \n",
    "Output: function values of reward and utility  \n",
    "'''\n",
    "\n",
    "def ReLOAD_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates, optimal_policy_LP):\n",
    "\n",
    "    theta = np.random.uniform(0,1,size=n*m)\n",
    "    theta_h = np.random.uniform(0,1,size=n*m)\n",
    "    dual = 0\n",
    "    dual_h = 0\n",
    "    reward_value = []\n",
    "    utility_value = []\n",
    "    policy_distance = []\n",
    "    for k in range(total_iterates):\n",
    "    \n",
    "        # policy evaluation of theta_h\n",
    "        prob_h = theta_to_policy(theta_h,n,m)\n",
    "    \n",
    "        Pi_h = get_Pi(prob_h,n,m)\n",
    "        mat = np.identity(n*m) - gamma*np.matmul(prob_transition,Pi_h)\n",
    "        qvals_reward_h = np.dot(np.linalg.inv(mat),reward)\n",
    "        qvals_utility_h = np.dot(np.linalg.inv(mat),utility)\n",
    "        violation_gradient_h = ell(qvals_utility_h,prob_h,rho) \n",
    "        \n",
    "        # Fisher inverse\n",
    "        P_theta = np.matmul(Pi_h,prob_transition)\n",
    "        d_pi = (1-gamma)*np.dot(np.transpose((np.linalg.inv(np.identity(n) - gamma*P_theta))),rho)\n",
    "        Fisher_inv = np.linalg.pinv(Fisher_info(prob_h,d_pi))\n",
    "    \n",
    "        # optimistic step for (theta, dual)\n",
    "        prob = theta_to_policy(theta,n,m)\n",
    "\n",
    "        Pi = get_Pi(prob,n,m)\n",
    "        mat = np.identity(n*m) - gamma*np.matmul(prob_transition,Pi)\n",
    "        qvals_reward = np.dot(np.linalg.inv(mat),reward)\n",
    "        qvals_utility = np.dot(np.linalg.inv(mat),utility)\n",
    "        violation_gradient = ell(qvals_utility,prob,rho) \n",
    "    \n",
    "        # natural gradient \n",
    "        qvals = 2*qvals_reward_h + 2*dual_h*qvals_utility_h - qvals_reward - dual*qvals_utility\n",
    "        \n",
    "        # natural gradient ascent\n",
    "        theta = theta_h\n",
    "        theta_h = theta_h + stepsize*avals(qvals,prob)\n",
    "        \n",
    "        # dual gradient descent\n",
    "        dual = dual_h\n",
    "        dual_h = dual_h - stepsize*(2*violation_gradient_h - violation_gradient)\n",
    "        dual_h = proj(dual_h,gamma)\n",
    "        \n",
    "        if k % 1 == 0:\n",
    "            avg_reward = ell(qvals_reward_h,prob,rho)\n",
    "            avg_utility = ell(qvals_utility_h,prob,rho)\n",
    "            reward_value.append(avg_reward)\n",
    "            utility_value.append(avg_utility)\n",
    "            \n",
    "            policy_kl_distance = kl_div(optimal_policy_LP,prob)\n",
    "            policy_distance.append(policy_kl_distance)\n",
    "                \n",
    "    return reward_value, utility_value, policy_distance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "# call ReLOAD method\n",
    "\n",
    "total_iterates = 2000\n",
    "\n",
    "stepsize = 0.2\n",
    "\n",
    "reward_value_ReLOAD_3, utility_value_ReLOAD_3, policy_distance_ReLOAD_3 = ReLOAD_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates, optimal_policy_LP)\n",
    "\n",
    "stepsize = 0.1\n",
    "\n",
    "reward_value_ReLOAD_2, utility_value_ReLOAD_2, policy_distance_ReLOAD_2 = ReLOAD_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates, optimal_policy_LP)\n",
    "\n",
    "stepsize = 0.05\n",
    "\n",
    "reward_value_ReLOAD_1, utility_value_ReLOAD_1, policy_distance_ReLOAD_1 = ReLOAD_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates, optimal_policy_LP)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "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": [
    "# plot reward values\n",
    "\n",
    "f_value = plt.figure\n",
    "\n",
    "num_every = 20\n",
    "num_grads = np.arange(0, total_iterates, num_every)\n",
    "\n",
    "# case #1\n",
    "reward_value_ReLOAD_1 = np.array(reward_value_ReLOAD_1)\n",
    "error_reward_value_ReLOAD_1 = np.absolute(reward_value_ReLOAD_1 + res.fun)\n",
    "\n",
    "# case #2\n",
    "reward_value_ReLOAD_2 = np.array(reward_value_ReLOAD_2)\n",
    "error_reward_value_ReLOAD_2 = np.absolute(reward_value_ReLOAD_2 + res.fun)\n",
    "\n",
    "# case #3\n",
    "reward_value_ReLOAD_3 = np.array(reward_value_ReLOAD_3)\n",
    "error_reward_value_ReLOAD_3 = np.absolute(reward_value_ReLOAD_3 + res.fun)\n",
    "\n",
    "# convert y-axis to Logarithmic scale\n",
    "plt.yscale(\"log\")\n",
    "\n",
    "plt.plot(num_grads,error_reward_value_ReLOAD_1[::num_every], \"k:\",linewidth=2)\n",
    "plt.plot(num_grads,error_reward_value_ReLOAD_2[::num_every], \"b-.\",linewidth=2)\n",
    "plt.plot(num_grads,error_reward_value_ReLOAD_3[::num_every], \"r-\",linewidth=2)\n",
    "plt.grid(axis='y', color='0.85')\n",
    "plt.draw()\n",
    "get_f_value = plt.gcf()\n",
    "get_f_value.savefig('NPG_primal_dual_sensitivity_reward_ReLOAD.png',dpi=300)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "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": [
    "# plot utility values\n",
    "\n",
    "g_value = plt.figure\n",
    "\n",
    "num_every = 20\n",
    "num_grads = np.arange(0, total_iterates, num_every)\n",
    "\n",
    "# case #1\n",
    "utility_value_ReLOAD_1 = np.array(utility_value_ReLOAD_1)\n",
    "error_utility_value_ReLOAD_1 = np.absolute(utility_value_ReLOAD_1)\n",
    "\n",
    "# case #2\n",
    "utility_value_ReLOAD_2 = np.array(utility_value_ReLOAD_2)\n",
    "error_utility_value_ReLOAD_2 = np.absolute(utility_value_ReLOAD_2)\n",
    "\n",
    "# case #3\n",
    "utility_value_ReLOAD_3 = np.array(utility_value_ReLOAD_3)\n",
    "error_utility_value_ReLOAD_3 = np.absolute(utility_value_ReLOAD_3)\n",
    "\n",
    "# convert y-axis to Logarithmic scale\n",
    "plt.yscale(\"log\")\n",
    "\n",
    "plt.plot(num_grads,error_utility_value_ReLOAD_1[::num_every], \"k:\", linewidth=2)\n",
    "plt.plot(num_grads,error_utility_value_ReLOAD_2[::num_every], \"b-.\", linewidth=2)\n",
    "plt.plot(num_grads,error_utility_value_ReLOAD_3[::num_every], \"r-\", linewidth=2)\n",
    "plt.grid(axis='y', color='0.85')\n",
    "plt.draw()\n",
    "get_g_value = plt.gcf()\n",
    "get_g_value.savefig('NPG_primal_dual_sensitivity_utility_ReLOAD.png',dpi=300)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "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": [
    "# plot sqaured norm of policy gap  \n",
    "\n",
    "p_distance = plt.figure\n",
    "\n",
    "num_every = 20\n",
    "num_grads = np.arange(0, total_iterates, num_every)\n",
    "\n",
    "# case #1\n",
    "policy_distance_ReLOAD_1 = np.array(policy_distance_ReLOAD_1)\n",
    "\n",
    "# case #2\n",
    "policy_distance_ReLOAD_2 = np.array(policy_distance_ReLOAD_2)\n",
    "\n",
    "# case #3\n",
    "policy_distance_ReLOAD_3 = np.array(policy_distance_ReLOAD_3)\n",
    "\n",
    "# convert y-axis to Logarithmic scale\n",
    "plt.yscale(\"log\")\n",
    "\n",
    "plt.plot(num_grads,policy_distance_ReLOAD_1[::num_every], \"k:\", linewidth=2)\n",
    "plt.plot(num_grads,policy_distance_ReLOAD_2[::num_every], \"b-.\", linewidth=2)\n",
    "plt.plot(num_grads,policy_distance_ReLOAD_3[::num_every], \"r-\", linewidth=2)\n",
    "plt.grid(axis='y', color='0.85')\n",
    "plt.draw()\n",
    "p_distance = plt.gcf()\n",
    "p_distance.savefig('NPG_primal_dual_squared_policy_distance_ReLOAD.png',dpi=300)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
