{
 "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 revise_array_up(arr, x):\n",
    "    revised_arr = [num if num <= x else x for num in arr]\n",
    "    return revised_arr\n",
    "\n",
    "def revise_array_low(arr, x):\n",
    "    revised_arr = [num if num >= x else x for num in arr]\n",
    "    return revised_arr\n",
    "\n",
    "def theta_to_policy_cutoff(theta,n,m):\n",
    "    \n",
    "    # revise theta to be in some interval [-C, C]\n",
    "    C = 10000\n",
    "    revised_theta = revise_array_up(theta, C)\n",
    "    revised_theta = revise_array_low(revised_theta, -C)\n",
    "    \n",
    "#     print('Theta',theta)\n",
    "    \n",
    "#     print('Revised Theta',revised_theta)\n",
    "    \n",
    "    prob = []\n",
    "    for i in range(n):\n",
    "        norm = np.sum(np.exp(revised_theta[m*i:m*(i+1)]))\n",
    "        for j in range(m*i,m*(i+1)):\n",
    "            prob.append(np.exp(revised_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",
    "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": 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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "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": 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 = 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 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": 16,
   "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": 17,
   "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": 18,
   "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": 19,
   "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": 20,
   "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": 21,
   "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": 22,
   "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": 23,
   "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",
    "    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_cutoff(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",
    "        qvals = qvals_reward + dual*qvals_utility + tau*qvals_entropy\n",
    "    \n",
    "        # natural gradient ascent\n",
    "        theta += stepsize*avals(qvals,prob)\n",
    "    \n",
    "        # dual desceent \n",
    "        violation_gradient = ell(qvals_utility,prob,rho)\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",
    "        \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": 24,
   "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",
    "    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)  \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) \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": 25,
   "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",
    "    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)\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": 26,
   "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",
    "    theta = np.random.uniform(0,1,size=n*m)\n",
    "    dual = 0\n",
    "    dual_up = 10/(1-gamma)\n",
    "    dual_low = 0\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)\n",
    "    \n",
    "        if violation_gradient <= -0.0001:\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": 27,
   "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",
    "    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)\n",
    "    \n",
    "        dual -= stepsize*violation_gradient\n",
    "        dual = proj_median(dual,gamma)\n",
    "\n",
    "    return reward_value, utility_value"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "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": 29,
   "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 iterate\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_comparison_reward.png',dpi=300)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "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 value vs policy iterate\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_comparison_utility.png',dpi=300)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "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
}
