{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# NPG dual comparison 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) Regularized (Natural) policy gradient + primal-dual method\\n   (Natural) policy gradient + optimistic primal-dual method\\n   Regularized (Natural) policy gradient + dual method\\n   Regularized (Natural) policy gradient + accelerated (Nesterov) dual method\\n   Regularized (Natural) policy gradient + accelerated (bisection) dual method\\n   \\nReferences:\\n1) Regularized NPG dual   \\n   Constrained Reinforcement Learning Has Zero Duality Gap\\n   https://proceedings.neurips.cc/paper_files/paper/2019/file/c1aeb6517a1c7f33514f7ff69047e74e-Paper.pdf\\n   Policy Optimization for Constrained MDPs with Provable Fast Global Convergence\\n   https://arxiv.org/abs/2111.00552\\n3) Regularized NPG Nesterov dual\\n   Faster Algorithm and Sharper Analysis for Constrained Markov Decision Process\\n   https://arxiv.org/abs/2110.10351 \\n4) Regularized NPG bisection dual\\n   A Dual Approach to Constrained Markov Decision Processes with Entropy Regularization\\n   https://proceedings.mlr.press/v151/ying22a/ying22a.pdf\\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) Regularized (Natural) policy gradient + primal-dual method\n",
    "   (Natural) policy gradient + optimistic primal-dual method\n",
    "   Regularized (Natural) policy gradient + dual method\n",
    "   Regularized (Natural) policy gradient + accelerated (Nesterov) dual method\n",
    "   Regularized (Natural) policy gradient + accelerated (bisection) dual method\n",
    "   \n",
    "References:\n",
    "1) Regularized NPG dual   \n",
    "   Constrained Reinforcement Learning Has Zero Duality Gap\n",
    "   https://proceedings.neurips.cc/paper_files/paper/2019/file/c1aeb6517a1c7f33514f7ff69047e74e-Paper.pdf\n",
    "   Policy Optimization for Constrained MDPs with Provable Fast Global Convergence\n",
    "   https://arxiv.org/abs/2111.00552\n",
    "3) Regularized NPG Nesterov dual\n",
    "   Faster Algorithm and Sharper Analysis for Constrained Markov Decision Process\n",
    "   https://arxiv.org/abs/2110.10351 \n",
    "4) Regularized NPG bisection dual\n",
    "   A Dual Approach to Constrained Markov Decision Processes with Entropy Regularization\n",
    "   https://proceedings.mlr.press/v151/ying22a/ying22a.pdf\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",
    "rho = np.ones(n)/n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Input: theta as an array and \n",
    "Ouput: 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": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Input: theta as an array and \n",
    "Ouput: 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": 6,
   "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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Input: theta as an array and \n",
    "Ouput: 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": 8,
   "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": 9,
   "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": 10,
   "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": 11,
   "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": 12,
   "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": 13,
   "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 = 10/(1-gamma)\n",
    "    if scalar < 0:\n",
    "        scalar = 0\n",
    "\n",
    "    if scalar > offset:\n",
    "        scalar = offset\n",
    "\n",
    "    return scalar"
   ]
  },
  {
   "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_median(scalar,gamma):\n",
    "    offset = 10/(1-gamma)\n",
    "    if scalar < 0:\n",
    "        scalar = 0\n",
    "\n",
    "    if scalar > offset:\n",
    "        scalar = offset\n",
    "\n",
    "    return scalar"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "The natural policy gradient(NPG) direction\n",
    "Input: current policy\n",
    "Output: NPG direction\n",
    "'''\n",
    "def NPG_sub(theta,n,m,reward,utility,prob_transition,gamma,rho,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",
    "    natural_gradient = avals(qvals,prob)\n",
    "    \n",
    "    return natural_gradient, qvals_reward, qvals_utility, prob"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "The regularized natural policy gradient(NPG) direction\n",
    "Input: current policy\n",
    "Output: regularized NPG direction\n",
    "'''\n",
    "def regularized_NPG_sub(theta,n,m,reward,utility,prob_transition,gamma,rho,dual,tau):\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",
    "    # entropy \n",
    "    qvals_entropy = np.dot(np.linalg.inv(mat),-np.log(prob))\n",
    "    \n",
    "    # natural gradient \n",
    "    qvals = qvals_reward + dual*qvals_utility + tau*qvals_entropy\n",
    "    natural_gradient = avals(qvals,prob)\n",
    "    \n",
    "    return natural_gradient, qvals_reward, qvals_utility, prob"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Policy Iteration to check feasibility "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "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": 18,
   "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": 19,
   "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": 20,
   "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": 21,
   "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",
    "eps = 0.001\n",
    "\n",
    "res = linprog(c, A_ub, b_ub, A_eq, b_eq, bounds)\n",
    "\n",
    "print('Optimal reward value:',-res.fun)\n",
    "\n",
    "print('Optimal policy:',theta_to_policy_naive(res.x,n,m))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Regularized NPG primal dual method "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Policy search via regularized NPG primal dual method \n",
    "Input: n, m, prob_transition, gamma, reward, utility, stepsize, tau, total_iterates   \n",
    "Output: function values of reward and utility  \n",
    "'''\n",
    "\n",
    "def Reg_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, tau, total_iterates):\n",
    "\n",
    "    conserv = 0.1\n",
    "    theta = np.random.uniform(0,1,size=n*m)\n",
    "    dual = 0\n",
    "    reward_value = []\n",
    "    utility_value = []\n",
    "    \n",
    "    for k in range(total_iterates):\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",
    "        # entropy \n",
    "        qvals_entropy = np.dot(np.linalg.inv(mat),-np.log(prob))\n",
    "    \n",
    "        # natural gradient \n",
    "        qvals = qvals_reward + dual*qvals_utility + tau*qvals_entropy\n",
    "    \n",
    "        # natural gradient ascent\n",
    "        theta += stepsize*avals(qvals,prob)\n",
    "    \n",
    "        # dual descent \n",
    "        violation_gradient = ell(qvals_utility,prob,rho) - conserv\n",
    "        dual = (1-stepsize*tau)*dual - stepsize*violation_gradient\n",
    "        dual = proj(dual,gamma)\n",
    "    \n",
    "    \n",
    "        if k % 1 == 0:\n",
    "        \n",
    "        # record iterates        \n",
    "            # record values\n",
    "            avg_reward = ell(qvals_reward,prob,rho)\n",
    "            avg_utility = ell(qvals_utility,prob,rho)\n",
    "            reward_value.append(avg_reward)\n",
    "            utility_value.append(avg_utility)\n",
    "            \n",
    "    return reward_value, utility_value\n",
    "    "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Optimistic NPG primal dual method (OGDA version)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "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 Opt_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates):\n",
    "\n",
    "# N = 1000\n",
    "# stepsize = 0.1\n",
    "# Parameters for line search\n",
    "# alpha = 1\n",
    "# beta = 0.7\n",
    "    conserv = 0.1\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",
    "    for k in range(total_iterates):\n",
    "    \n",
    "        # optimistic step for (theta, dual)  \n",
    "        Pi = get_Pi(theta,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,theta,rho)  - conserv \n",
    "    \n",
    "        # gradient \n",
    "        qvals = qvals_reward + dual*qvals_utility\n",
    "\n",
    "        # natural gradient ascent\n",
    "        theta = project_to_policy(theta_h + stepsize*qvals,n,m)\n",
    "    \n",
    "        # dual descent \n",
    "        dual = dual_h - stepsize*violation_gradient\n",
    "        dual = proj(dual,gamma)\n",
    "\n",
    "        # optimistic step for (theta_h, dual_h)    \n",
    "        Pi = get_Pi(theta,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,theta,rho)  - conserv\n",
    "        \n",
    "        # gradient \n",
    "        qvals = qvals_reward + dual*qvals_utility\n",
    "    \n",
    "        # natural gradient ascent\n",
    "        theta_h = project_to_policy(theta_h + stepsize*qvals,n,m)\n",
    "    \n",
    "        # dual desceent \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,theta_h,rho)\n",
    "            avg_utility = ell(qvals_utility,theta_h,rho)\n",
    "            reward_value.append(avg_reward)\n",
    "            utility_value.append(avg_utility)\n",
    "            \n",
    "    return reward_value, utility_value"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## NPG dual method "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Policy search via NPG dual method \n",
    "Input: n, m, prob_transition, gamma, reward, utility, stepsize, tau, total_iterates   \n",
    "Output: function values of reward and utility  \n",
    "'''\n",
    "def NPG_dual(n, m, prob_transition, gamma, reward, utility, stepsize, stepsize_npg, tau, total_iterates, NPG_iterates):\n",
    "\n",
    "    conserv = 0.1\n",
    "    theta = np.random.uniform(0,1,size=n*m)\n",
    "    dual = 0\n",
    "    reward_value = []\n",
    "    utility_value = []\n",
    "\n",
    "    for k in range(total_iterates):\n",
    "    \n",
    "    # natural gradient ascent as a RL oracle\n",
    "        for i in range(NPG_iterates):\n",
    "            natural_gradient, qvals_reward, qvals_utility, prob = regularized_NPG_sub(theta,n,m,reward,utility,prob_transition,gamma,rho,dual,tau)\n",
    "\n",
    "            # NPG update\n",
    "            theta += stepsize_npg*natural_gradient\n",
    "        \n",
    "            if k*i % 1 == 0:\n",
    "                # record values\n",
    "                avg_reward = ell(qvals_reward,prob,rho)\n",
    "                avg_utility = ell(qvals_utility,prob,rho)\n",
    "                reward_value.append(avg_reward)\n",
    "                utility_value.append(avg_utility)\n",
    "            \n",
    "        # dual descent \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_utility = np.dot(np.linalg.inv(mat),utility)\n",
    "        violation_gradient = ell(qvals_utility,prob,rho) - conserv \n",
    "    \n",
    "        dual -= stepsize*violation_gradient\n",
    "        dual = proj(dual,gamma)\n",
    "\n",
    "    return reward_value, utility_value"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Accelerated (bisection) NPG dual method "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Policy search via accelerated (bisection) NPG dual method \n",
    "Input: n, m, prob_transition, gamma, reward, utility, stepsize, tau, total_iterates   \n",
    "Output: function values of reward and utility  \n",
    "'''\n",
    "def NPG_dual_acc(n, m, prob_transition, gamma, reward, utility, stepsize, tau, stepsize_npg, total_iterates, NPG_iterates):\n",
    "\n",
    "    conserv = 0.1\n",
    "    theta = np.random.uniform(0,1,size=n*m)\n",
    "    dual = 0\n",
    "    dual_up = 10/(1-gamma)\n",
    "    dual_low = 0\n",
    "# tau = 0.01\n",
    "    reward_value = []\n",
    "    utility_value = []\n",
    "\n",
    "    for k in range(total_iterates):\n",
    "    \n",
    "        # dual exptrapolation \n",
    "        dual = (dual_up + dual_low)/2\n",
    "    \n",
    "        # natural gradient ascent as a RL oracle\n",
    "        for i in range(NPG_iterates):\n",
    "            \n",
    "            natural_gradient, qvals_reward, qvals_utility, prob = regularized_NPG_sub(theta,n,m,reward,utility,prob_transition,gamma,rho,dual,tau)\n",
    "            # NPG update\n",
    "            theta += stepsize_npg*natural_gradient\n",
    "        \n",
    "            if k*i % 1 == 0:\n",
    "                # record values\n",
    "                avg_reward = ell(qvals_reward,prob,rho)\n",
    "                avg_utility = ell(qvals_utility,prob,rho)\n",
    "                reward_value.append(avg_reward)\n",
    "                utility_value.append(avg_utility)\n",
    "            \n",
    "        # dual descent \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_utility = np.dot(np.linalg.inv(mat),utility)\n",
    "        violation_gradient = ell(qvals_utility,prob,rho) - conserv\n",
    "    \n",
    "        if violation_gradient <= -0.01:\n",
    "            dual_up = dual_up\n",
    "            dual_low = dual\n",
    "        else:\n",
    "            dual_low = dual_low\n",
    "            dual_up = dual\n",
    "    \n",
    "    return reward_value, utility_value"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Accelerated (Nesterov) NPG dual method "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Policy search via accelerated (Nesterov) NPG dual method \n",
    "Input: n, m, prob_transition, gamma, reward, utility, stepsize, tau, total_iterates   \n",
    "Output: function values of reward and utility  \n",
    "'''\n",
    "def NPG_dual_nesterov(n, m, prob_transition, gamma, reward, utility, stepsize, stepsize_npg, stepsize_acc, tau, total_iterates, NPG_iterates):\n",
    "\n",
    "    conserv = 0.1\n",
    "    theta = np.random.uniform(0,1,size=n*m)\n",
    "    dual = 0\n",
    "    dual_last = 0\n",
    "    reward_value = []\n",
    "    utility_value = []\n",
    "\n",
    "    for k in range(total_iterates):\n",
    "    \n",
    "        # dual exptrapolation \n",
    "        dual += stepsize_acc*(dual - dual_last)\n",
    "    \n",
    "        # natural gradient ascent as a RL oracle\n",
    "        for i in range(NPG_iterates):\n",
    "            natural_gradient, qvals_reward, qvals_utility, prob = regularized_NPG_sub(theta,n,m,reward,utility,prob_transition,gamma,rho,dual,tau)\n",
    "            # NPG update\n",
    "            theta += stepsize_npg*natural_gradient\n",
    "        \n",
    "            if k*i % 1 == 0:\n",
    "                # record values\n",
    "                avg_reward = ell(qvals_reward,prob,rho)\n",
    "                avg_utility = ell(qvals_utility,prob,rho)\n",
    "                reward_value.append(avg_reward)\n",
    "                utility_value.append(avg_utility)\n",
    "            \n",
    "        # dual descent \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_utility = np.dot(np.linalg.inv(mat),utility)\n",
    "        violation_gradient = ell(qvals_utility,prob,rho) - conserv\n",
    "    \n",
    "        dual -= stepsize*violation_gradient\n",
    "        dual = proj_median(dual,gamma)\n",
    "\n",
    "    return reward_value, utility_value"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "total_iterates = 1060\n",
    "stepsize = 0.1\n",
    "\n",
    "# call regularized NPG primal dual method\n",
    "\n",
    "tau = 0.08\n",
    "\n",
    "reward_value_reg, utility_value_reg = Reg_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, tau, total_iterates)\n",
    "\n",
    "# call optimistic NPG primal dual method\n",
    "\n",
    "reward_value_opt, utility_value_opt = Opt_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates)\n",
    "\n",
    "# call NPG dual methods\n",
    "\n",
    "stepsize = 0.1\n",
    "stepsize_npg = 0.1\n",
    "stepsize_acc = 0.1\n",
    " \n",
    "total_iterates = 53\n",
    "NPG_iterates = 20\n",
    "\n",
    "tau = 0.01\n",
    "\n",
    "reward_value_dual, utility_value_dual = NPG_dual(n, m, prob_transition, gamma, reward, utility, stepsize, stepsize_npg, tau, total_iterates, NPG_iterates)\n",
    "\n",
    "tau = 0.08\n",
    "\n",
    "reward_value_accdual, utility_value_accdual = NPG_dual_acc(n, m, prob_transition, gamma, reward, utility, stepsize, tau, stepsize_npg, total_iterates, NPG_iterates)\n",
    "\n",
    "reward_value_nesterovdual, utility_value_nesterovdual = NPG_dual_nesterov(n, m, prob_transition, gamma, reward, utility, stepsize, stepsize_npg, stepsize_acc, tau, total_iterates, NPG_iterates)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "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 value vs policy iteate \n",
    "total_iterates = 1060\n",
    "\n",
    "f_value = plt.figure\n",
    "\n",
    "num_every = 20\n",
    "num_grads = np.arange(0, total_iterates, num_every)\n",
    "\n",
    "reward_value_dual = np.array(reward_value_dual)\n",
    "reward_value_accdual = np.array(reward_value_accdual)\n",
    "reward_value_nesterovdual = np.array(reward_value_nesterovdual)\n",
    "reward_value_reg = np.array(reward_value_reg)\n",
    "reward_value_opt = np.array(reward_value_opt)\n",
    "\n",
    "plt.plot(num_grads,reward_value_accdual[::num_every], \"k:\", linewidth=2)\n",
    "plt.plot(num_grads,reward_value_dual[::num_every], \"m-.\", linewidth=2)\n",
    "plt.plot(num_grads,reward_value_nesterovdual[::num_every], \"c--\", linewidth=2)\n",
    "plt.plot(num_grads,reward_value_reg[::num_every], \"b--\",linewidth=2)\n",
    "plt.plot(num_grads,reward_value_opt[::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_dual_conserv_comparison_reward.png',dpi=300)\n",
    "\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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pRYeeughevbsybPPPqv6yxXFR2p9sVNKeQDoftYd68iGDRtYvHgxo0eP9lcIynlo2bIlGzdurDGJSA0hVZTaCfjhhw888ACff/45l19+ub9DUc7R8Ul8zZo1DB48mLKyMj9GpCiBLeATea9evRg3blyNGywrgUFKyfTp01m9evVJdyJSFOXcBfxaK0rgEkLw2Wef8f777/PEE0/4OxxFCVg+GUd+vnw5jvy7776juLiYq6++mvj4eJ/UqSiK0hDV6Thyf3ruueeYPHky+/fv93coSi1VVVUxdepUNm3a5O9QFCWgBHzXyqhRo+jQoQMJCQn+DkWppVdffZV3332X1atXs3PnTrRarb9DUpSAEPCJXPWtNh5//etf2bFjB4899phK4opyHgI+kSuNh9FoZOHChf4OQ1ECTsD3kefn51NSUoLb7fZ3KIqPrV27lpUrV/o7DEVp8AI+kV900UVER0erCSWNzPLlyxkwYACTJ09W9/9UlLMI+K6VqKgohBBqidRGZsiQIfTp04crrrhCTd9XlLMI+ER+4MABf4eg1AGdTsfvv/9e425DiqKcWsB3rSiN1/FJ3OFwqOsginIaKpErDd66detITU1lwYIF/g5FURqkgE7khYWF9O7dm2uuucbfoSh1aP/+/ezYsYP58+erm1EoyikEdAdkZWUlmzZtoqCgwN+hKHXo5ptvxmq1csstt6gLn4pyCgGdyBMTE9mwYYO/w1DqmBCCO++8099hKEqDFdBdK0ajkd69e9O790mLgSmNlMPh4L333lMXPhXlOD5L5EIIrRBiixBiia/qVJQTjRkzhrvuuosff/zR36EoSoPhy66V6cAeINyHdZ7R/v37+eSTT2jfvj0TJ06sr8MqfjR06FCys7NVX7miHMcnLXIhRAtgFPCeL+o7V2lpacyePVsNS2tCHnroIXbs2MFVV13l71AUpcHwVYv8n8CjQNjpdhBCTAWmAiQlJZGRkVHrgxqNRh588EFatmzpk/oURVECUa1v9SaEGA2MlFLeK4S4HHhESjn6TI/x5a3elKaprKyM9957j5tvvpnExER/h6Mo9aIub/V2KXCNEOIQsAgYKoT4Px/Uqyindc899/DII4/w5ptv+jsURfG7WnetSCkfBx4HOK5Ffmtt6z0X+fn55ObmkpCQoFplTcx9993H4cOHufzyy/0diqL4XUCP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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plot utility value vs policy iteate \n",
    "\n",
    "g_value = plt.figure\n",
    "\n",
    "num_every = 20\n",
    "num_grads = np.arange(0, total_iterates, num_every)\n",
    "\n",
    "utility_value_dual = np.array(utility_value_dual)\n",
    "utility_value_accdual = np.array(utility_value_accdual)\n",
    "utility_value_nesterovdual = np.array(utility_value_nesterovdual)\n",
    "utility_value_reg = np.array(utility_value_reg)\n",
    "utility_value_opt = np.array(utility_value_opt)\n",
    "\n",
    "plt.plot(num_grads,utility_value_accdual[::num_every], \"k:\", linewidth=2)\n",
    "plt.plot(num_grads,utility_value_dual[::num_every], \"m-.\", linewidth=2)\n",
    "plt.plot(num_grads,utility_value_nesterovdual[::num_every], \"c--\", linewidth=2)\n",
    "plt.plot(num_grads,utility_value_reg[::num_every], \"b--\", linewidth=2)\n",
    "plt.plot(num_grads,utility_value_opt[::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_dual_conserv_comparison_utility.png',dpi=300)\n",
    "\n",
    "plt.show()"
   ]
  }
 ],
 "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
}
