{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# NPG primal-dual comparison for finite constrained MDPs (conservative)"
   ]
  },
  {
   "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\\n   \\nReferences:\\n1) NPG primal-dual   \\n   Natural policy gradient primal-dual method for constrained Markov decision processes\\n   https://proceedings.neurips.cc/paper_files/paper/2020/file/5f7695debd8cde8db5abcb9f161b49ea-Paper.pdf\\n2) NPG PID primal-dual method\\n   Responsive Safety in Reinforcement Learning by PID Lagrangian Methods\\n   https://arxiv.org/abs/2007.03964\\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 + 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\n",
    "   \n",
    "References:\n",
    "1) NPG primal-dual   \n",
    "   Natural policy gradient primal-dual method for constrained Markov decision processes\n",
    "   https://proceedings.neurips.cc/paper_files/paper/2020/file/5f7695debd8cde8db5abcb9f161b49ea-Paper.pdf\n",
    "2) NPG PID primal-dual method\n",
    "   Responsive Safety in Reinforcement Learning by PID Lagrangian Methods\n",
    "   https://arxiv.org/abs/2007.03964\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"
   ]
  },
  {
   "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 = 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": 14,
   "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": 15,
   "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": 16,
   "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"
     ]
    }
   ],
   "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": 17,
   "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": 18,
   "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": [
    "## NPG primal dual method "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Policy search via 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 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",
    "    dual = 0\n",
    "    reward_value = []\n",
    "    utility_value = []\n",
    "    conserv = 0.1\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",
    "        # natural gradient \n",
    "        qvals = qvals_reward + dual*qvals_utility\n",
    "    \n",
    "        # natural gradient ascent\n",
    "        theta += stepsize*avals(qvals,prob)\n",
    "    \n",
    "        # dual desceent \n",
    "        violation_gradient = ell(qvals_utility,prob,rho) - conserv\n",
    "        dual -= stepsize*violation_gradient\n",
    "        dual = proj(dual,gamma)\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"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Regularized NPG primal dual method "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "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",
    "    conserv = 0.1\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 desceent \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",
    "        \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": 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 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",
    "    conserv = 0.1\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 primal PID dual method "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Policy search via PID-based NPG primal dual method \n",
    "Input: n, m, prob_transition, gamma, reward, utility, stepsize (PID), total_iterates   \n",
    "Output: function values of reward and utility  \n",
    "'''\n",
    "\n",
    "def PID_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, stepsize_p, stepsize_d, total_iterates):\n",
    "    \n",
    "    conserv = 0.1\n",
    "    theta = np.random.uniform(0,1,size=n*m)\n",
    "    dual = 0\n",
    "    dual_i = 0 \n",
    "    reward_value = []\n",
    "    utility_value = []\n",
    "    violation_momentum = 0\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",
    "        # natural gradient \n",
    "        qvals = qvals_reward + dual*qvals_utility\n",
    "    \n",
    "        # natural gradient ascent\n",
    "        theta += stepsize*avals(qvals,prob)\n",
    "    \n",
    "        # PID dual desceent \n",
    "        violation_gradient = ell(qvals_utility,prob,rho)  - conserv\n",
    "        dual_i -= violation_gradient\n",
    "        if dual_i >= 0:\n",
    "            dual_i = dual_i\n",
    "        else:\n",
    "            dual_i = 0\n",
    "\n",
    "        # violation momentum\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",
    "    \n",
    "        violation_momentum = ell(qvals_utility,prob,rho) - violation_gradient\n",
    "    \n",
    "        if violation_momentum >= 0:\n",
    "            violation_momentum = violation_momentum\n",
    "        else:\n",
    "            violation_momentum = 0\n",
    "    \n",
    "        dual = stepsize_p*violation_gradient + stepsize*dual_i + stepsize_p*violation_momentum\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"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "# call NPG primal dual method\n",
    "\n",
    "total_iterates = 1060\n",
    "stepsize = 0.1\n",
    "\n",
    "reward_value, utility_value = NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates)\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 PID-based NPG primal dual method\n",
    "\n",
    "stepsize = 0.1 # K_i\n",
    "stepsize_p = 0.2 # K_p\n",
    "stepsize_d = 0.01 # K_d\n",
    "\n",
    "reward_value_pid, utility_value_pid =  PID_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, stepsize_p, stepsize_d, total_iterates)\n"
   ]
  },
  {
   "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 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",
    "reward_value = np.array(reward_value)\n",
    "reward_value_reg = np.array(reward_value_reg)\n",
    "reward_value_opt = np.array(reward_value_opt)\n",
    "reward_value_pid = np.array(reward_value_pid)\n",
    "# print(reward_value_opt)\n",
    "\n",
    "plt.plot(num_grads,reward_value[::num_every], \"c-.\", linewidth=2)\n",
    "plt.plot(num_grads,reward_value_pid[::num_every], \"k:\",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_primal_dual_conserv_comparison_reward.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 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",
    "utility_value = np.array(utility_value)\n",
    "utility_value_reg = np.array(utility_value_reg)\n",
    "utility_value_opt = np.array(utility_value_opt)\n",
    "utility_value_pid = np.array(utility_value_pid)\n",
    "\n",
    "plt.plot(num_grads,utility_value[::num_every], \"c-.\", linewidth=2)\n",
    "plt.plot(num_grads,utility_value_pid[::num_every], \"k:\", 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_primal_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
}
