{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# NPG primal-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) (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": null,
   "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": 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",
      "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": 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",
    "\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)\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",
    "    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",
    "        # 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) \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"
   ]
  },
  {
   "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",
    "    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 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",
    "    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)\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",
    "# total_iterates = 1000\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": "iVBORw0KGgoAAAANSUhEUgAAAXQAAAD4CAYAAAD8Zh1EAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjEsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+j8jraAAAgAElEQVR4nO3deXxU5bnA8d87mUkmC2Rl33dEEJEgUrXFHTcsLe5t1auXFmu1m7fS29veemsXu7jUVoq1VYt1X1BqlVZRpIgFlF2WgIQQQshG9sz63D/OmclMSCAJSSYzPN/P53wy886Zc94zM3nmnee873uMiKCUUir+OWJdAaWUUl1DA7pSSiUIDehKKZUgNKArpVSC0ICulFIJwhmrHefl5cnIkSNjtXullIpLGzZsKBeRfq09FrOAPnLkSNavXx+r3SulVFwyxhS29ZimXJRSKkFoQFdKqQShAV0ppRKEBnSllEoQGtCVUipBaEBXSqkEoQFdKaUSRMz6oSsVL8q9XlZXV7OrsZHzsrLI79MHY0yXbd8fDPLOkSNU+/1kOp2MS01lVGoqACLSpfsCOOLzccjrZWJ6erhsU10dE1JTcScldem+AKp8PrbW17Olvp6t9fUc9HjI79OH2VlZzOjblxRH17YrRYTDPh99kpJIs4/nwaIidjQ0ML9fPy7MyenS/UUq83pZX1vL+tpaPqqrQ0QYl5bG2NRUxqWmMjY1laEpKTi6+D0N0YB+MvL7obERPB5wOsHlal5aftBEIBCAYNB6ns939BIMNq/bwq6GBjbW1TExPZ3T+va1th8IWIvfby2h2yLWtkJL6H6EQDDIrsZGTklPt7ZlDPcWFrKtoYH6YJABycl8uV8/Ptu3L47IurfcZmiB8HZCS4Xfz7a6OrbX1bGtvp4ijwcjghHh3yIMdjo5t08fzs7IYLDL1epxH1cwSFlTE6urqvjgyBFqfD4cwSBGhEuysxmVmwsi7Kyv582qKs7LymJqRsbR701bS+hYI28D+5qaeLykhKEpKUwcPBhE8ASDvLxvHw5jGOhyMTQlhVFuN+PT03EmJTW/Nm0F3tDrHHo//X62VFezp66Ocq+XukAgvOp4ewk4HKxxOPjA6WRwairD0tM5JzsbR2gfoc+hMc3bj/zcBAJRx1vv97OnoYGCxkYO+/3UA1f178/YjAxwOJhZV0fNkSO4srMhKwuM4bDPxyeNjYxLSWGQ04kJvVahz0xrryVYr4O91AWDZIT+b4zh6QMHqAUEON0Y63MDlAeDVNifn35JSdw6cybccUfHPzfHYWJ1gYv8/Hw56UaKVlfD/v1RH/yogNbaUlkJxcVw8KD1N3Q7EICUFHC7o/9GfvhDH85QAG9shIYG8HrbrqPDgSQlQTCIifhHVEp1oZkzYe3aTj3VGLNBRPJbe0xb6J1RUgIffQQffwy7dkFmJgweDIMGWcvgwZCaCtu2wcaNsGmT9ffTT7u2HrW1nXpa0Bjq3W68LhfOQACX309qIIDx+61A3qJVjMMBSUnWEtmaDy1JSTQFg5T5fDSKkOpwMDQlBSNCfSBAvd9PmjFkOBwgQiVQFgyS4nIxIC2N1ORka9sRLZ9wizDUOrRtr6+nJhDglLQ0Mu0vnka7tZYkQoXfz36fj0Yg6HCQ5nQyOi2NfqmpVuqi5T6Aap/P2pbdCltXU0MQyElOJtflItPpJMleX5KSOOz3U+j1Uujz0WQM52ZlMdLtBqA+EMBpTFQaoT4Q4KDHQ7HXS4nXiz8YJOBwYBwORqSlMSEjg4HJyZjI1rC9eEXY0dDAgORkBiQnA1DY1MTO+nr6p6SQlpREmtMZ/hsAtjc1saW+njK/n6DDgcvh4K6hQ3HZ6Qe/CM7IX2L2bY8IBz0eDni9fFJfT6nXixHBEQyS43QyJTWVHJcLnwhjUlNJt7e3ua6Of9bWcvWgQQxLTwenk11eL1UijE5LI9flik4xhFq9fj91Xi+F9fUE/X6m2GmmGp+Ph4uLyTCGO4cOxWEMDcDbNTXUG0MtUAsc9PsRrNZwksPBhLQ0pvbpwzi3m0xjrM9xqFETatGH9i/C/sZGttbVscPj4UgwSNDhIGgMzqQkhqWm4nA48BuDHwgAfuCrgweT7nBAMMgjRUWUNTVx28CBDEtObr1FH/k5tj93ASBpyJCO/+O2w8nbQhexWq4uV9uPV1RYAXv3bti50wrKH30EpaWd22dKCowZY/1NSrLSHZF/XS7rtr1IUhJH0tLYnp3NmowMVqWnsz83l4N5eficTladcgqnuVzQ1GSlT5qamoNVKACHlrQ03vN6uXX/fvYEg6Q4HPzPyJGMcbvxiXBVXh59k5Jo8Hi44uOP2dDYyJaZMxmelnZ0GiZCicfDDz79lD8dOgTAoORk7h89mhsHDGgz97u4uJhFn37KEb8fB3DboEHcO2oUeS4X5T4fpV4vh7xeSr1eXquo4GuDB3NBdjYAlT4fGUlJJB8j7+oJBvljSQk/LSzkoP1rJNvpZKTbzQi3m28PHcq5WVmICFdu2cLfKiv59xlnMKNvX6D9eeumQIA3Kiu5JCcnHNxu2bGDpaWl/HnCBL40cCAAEz/8kJ2NjeHnTUlP5z8HDeJLAwaQ3dbn7xgu3byZNysrj7tensvFlwYM4OaBA49O17TDlro6/lJaytLSUkpa/KpbOXUqs+335DsFBfzmwAH+Z8QI7h01qsP7aWldTQ0Ldu1ilNvNy5MnA1Zuuv+aNVHruYzhkpwcrunXj7l5eWQ6O9c+FRE21dXxWkUFr5WXs6Gurs11i846i6H2l3dRUxMDk5NxdfE5gOM5Vgv95Azohw7B5ZdbwTk9HUJ5tdBSVmYF8qqq1p+fmQmnnw5nnAGTJkFdnZUGKSmxloMHrdbzKadY602dav2dMMEK1u2wvb6eGz/5hI0RH64k4DOZmXw2M5Mg8NPRo8OPHfR4GJyS0uq26gMBvrdnD787eBCAGX368OTEiVYeuhVBEbbX1zPZDgIBEX5TVMScnBx2NjSwqb6ezXV1bKqro9DjAax/ru8MG8b3hw+nTzuOscLn48f79vH74mIC9vMDIgRbWfeczEzenzbtuNtsqSkQ4LGSEn6xfz/FEQHptcmTuTIvD4C79+zhjyUlLB4/nmv79+/wPlq6bts2XiwrY9uZZzIhLQ2AH+zdy9b6eubk5HBJTk74hGdnfdrYyPNlZRQ1NVFqn+AMfQE2BoNclpPDLQMHcllu7jG/+NorIMLbVVU8d/gw1X4/GUlJfHfYsPDnY211NUnGdPnJYo/d8ADrxPFjJSXkuFzkOJ3kuFyMTU3tdBA/lmKPh49qa3EYg8tekh0OXMYwNSOjy0/idpQG9Eh1dTB7NmzYcPx1+/SB8eNh3DhrOe00K4iPGnXMVuuJerq0lAU7d9IQDJLncnFZTg6X5eZycXZ2qy26t6uquHTzZhYNH86P7RbSv6qrebGsjCqfj/erq9nb1ITTGH44YgSLhg/H2YEP5ZKDB/nqrl2tPuZ2OJiTk8P9o0czzg5gHbGjvp679+5leUUFALlOJwPt9MLA5GRGud18bfDgcKuoM0K9HgqbmihsauKzWVnh9EWVz4fDmC4NDJU+HzmdaHl3haBIt/WgUL2D5tBD/H649lormI8eDWvWWLnuI0es1viRI9aSnW0F8gEDOhW46/x+VlRVkWwMuS5XuFWR7XQeM5A2BQLcVVDAkpISAL40YACLx48P/5xvy9qaGvwiUS2HLXV1PHjgQPj+qWlpPHXKKZzRp0+Hj2dm375cmJ3N7oYGJqenc1pGBlMzMpians7Y1NQOfTm0NDE9ndenTKHa7yfNzvd2NWNMOAd9pp1WCelMyuN4YhXMAQ3mJ7mTp4UuAl/9Kjz2GOTmwgcfWK3uLnbE5+O8TZuiUiUhY1NT2T1zZvh+QUMDo1NTw/+EV27ZwvKKClKM4eFx4/jPQYPa/RP249paTklLC/cj3lxXxz+qqsh2OunvcnFRTk7MfyoqpU7cCbfQjTHfAm7DOqG8BbhFRJoiHk8BngKmAxXAtSKy7wTr3bXuu88K5m43vP56twTzOr+fS7dsYWNdHSNSUjg1PZ0Kn49Kv59Kn49+ES23hkCASevWkeV0sv+ss3AnJfFfw4axq6GBZyZN6nBLelqL9U/LyOC0TpwIU0rFr+MGdGPMEOBOYJKINBpjngeuA56IWO1WoEpExhpjrgN+AVzbDfXtnCefhP/5Hyt98swzMGtWl++iKRDgqq1bWVtTw/CUFFZNm8bwFnnfyF9D++wz5P1crnCr+tysLLbNmHFCKQx1fHV11nnx0lLr/PaIEdbpks7YssUaWpCTA3l51pKZ2fYYnJYaG63z6P37Q+j7t7zc6mDldltLnz5WZvBYP9aamqznHDnSPARhwACrFy1Afb11zNnZVv2OlcWrq7OWxkZru5FLXh5MmWKtFxry0Nrph9BHPVTnw4et/dfXWx2yUlIgLc06rowMq6dviM8X3bM0GGweQhFaH6whGvv2NXcMC3USS062ltzc6PchEDj6eDwe6/UdNsxax+OBggJrv8nJVj1bLqFjCr1OkcNJQovTGd1mLCxsrlvkMXQ5ETnmAgwBioAcrC+A5cDFLdZ5C5hl33YC5djpnLaW6dOnS49YsULE6bR6hf72t92yC08gIJdv2iSsXCkD//Uv2V1f367nBYNBOeLzdUudTlQw2Lnn1daKvPGGyKJFInfcIfLKK82P7dkj8v3vizz6qMjbb4vs3y8SCLS+nSefFLnzTpHLLhM5+2yRKVNERowQyc4Wufvu5vVKSqz7jz0msmqVyIYN1v7/9CeRn/5UxOttXnf+fJGMjNaHVn7xi83rFRVZ+164UOS220RuuknkhhtEPv95kRkzRD79tHnd668/elsOh0i/ftb+Qvx+kYsvFpk7V+TCC0UmTbKOJfScZcua1/3FL47eptMpkpsrMnly9Os0ZYpIamrrx/SjHzWv9/e/Rz/Wt6/1eo4bJzJ4sEhVVfO6F1zQ9hDUK69sXu/gQassLU0kPV3E7bbqaYxVvmJF87rf+U7b2xw+PPqYsrKiX8vIdX/96+b1nnuu7W2CSGVl87oXX9z2etde27zerl3H3uZ77zWv+81vtr3e+PHRx9S3b/NjM2bICQHWSxtx9bgtdBEpNsb8CtgPNAIrRGRFi9VCQR8R8RtjqoFcO7CHGWMWAAsAhgwZQmFhm5fG6xoeD0O+9CWcfj/VX/0qR6680vqq7EIBEe4sK+Nv9fVkOxw81a8frrIyOrKXI11Ul9A4ho4KBODDD90kJwv5+VY3xB07XMybN5CcnCC5uQFycoLk5QUYODDAgAF+Lr20gbw8q5Phhg3JvP9+KmvWuPn44xR8vuZKeL3VTJtmHeGHH7r56U8HRO3b7Q4ycqSfoUP9PPZYWbj+P//5ID75JLnV+hYV1VJYaPXDfv99N7/85YBW1wO44IIDDBhgjXitru5HXV0aKSlB+vcPkJsbpKbGQXFxEikp9eFtbt/u4uGHB7e5zbVrD2GM9TpNmpTB5z6XRnW1g6oqB5WVSdTWOigrg/37mygstMYsNDQYVqwYftS2XC6hf/8ABw9WUVjYAIDPl8Ho0X3xeAxNTYa6OoPH46CiAhwOP4WFxeHn19QMprHRhcslZGUF6NtXcLms5rHTWUthoXUup7TUzbBhOVRXO6itdVBTY6ipaa7Hjh0HGDTIep3S0vLIy3OTkiK43UJKirUkJwtjxjRRWFgNwN69TowZTEND6x+6kpJSCgutzGx6eh8mTMggNdXajtdrHVtTk6F//0D4dbIMxRgHIiY8O4PbHSQ1Vaiurqaw0BpQ19jo5tRTs/D7DX6/IRAAn8/YM1QYDh0qpqbGei2OHBmAMSnhY2k+Jujbt4HCQuszevhwEuPG9ScYNPh84PWaqKW4+HD4mEQyycvrg9Mp9rg4CY+PGzTIT2Hh4fAR5eUNxu02+HwGt9sb9VhXOu5JUWNMNvASVgrlCPAC8KKILI1YZyswR0QO2Pf3ADNFpLyVTQI9dFL0ySfh5put7oYff9z+38EdsLOhgTPtLpDvnH460zv72/0EVFTA978PTz9t/YwdMyZ6+dznrLQCWD9TKyqsn38lJfDcc/D889bP4YsvhrfestZbuRLOP7/tfW7e3PzT++yzrQ5DYL3E06fDBRdYP6OnTYNzzrEe27cPnnrK+k7dtctaDkd8rgsLYbgd8xYvtmZKmDjRSmf07Ru9hE5H7N5tHcOuXdbYr6YmGDjQSjcMHAh33w397Oujl5ZaP5kzM6O/+ESsn9qhDFl5OfzlL9bP48gBsW63dUxTphw7RePzWa+x3w9DhzaXvfOOlToIbWfwYOvY2vOx9His16OhAUaObC4vKbF+vmdktP/LPBiEmhorPdPUZA3FGDz42GmYtohYaRSIHscWMRC300KDLo81jUx7BQJdU6fe4IT6oRtjrsYK1rfa978CnCUit0es8xbwvyLygTHGCRwC+skxNt7tAV3EGsyzeTM88QTcdFO37eqj2loaAgHOycrqtn0cS1OTFbAPt/GlH3n4998P3/ve0euMGQNf/jL88IfN8yHV1lpjrELLoUPN08n86ldWYAR49FFrloMLLrC6+NsDCNulutoKysXF1nND21RKte5Ee7nsB84yxqRhpVwuAFpG4teAm4APgPnAO8cK5j1i5UormA8YANdd12WbrfP7ubewEAP8YswYgE717Y60bZtV1aFDrcA8/Ohf5vh8VtArKrJaok8/Da++agVAtxv+9CdrvFNmJuzZE71Mndq8ncxMGDvW2l5KClxxBVx/vdWqbjm9R6g1bB9mmxYu7PyxZ2ZCfr61KKVOUFvJ9cgF+DGwA9gK/AVIAe4F5tqPu7FSMQXAv4HRx9tmt58Uvfxy6wzEvfd26WY/qqkRx8qV4nr3XTnY1HRC22pqErnnnugTP6NGNT9+5IjIWWdZJ61CJ5oil9/85gQPRikVdzjGSdHEHFi0Y4c1j4rbbfUpCyVRO6nlUO6HDhzgM337hidz6owNG6w0yLZtVmt4zhxrsOrQofDCC9Y6waB1CD6ftc7gwVb3qmHD4MILrRTJCU4LopSKMyff0P+HHrL+fvnLJxTMRYTHS0r49p49PDNpEpfn5gJwV+hMVyft3291hff5rL6qTzwBn/nM0es5HPD++1bWaMiQtieGVEopSMSAXlFh9W4B+OY3O72ZA01N3LZzJ2/ZMy4ur6gIB/TOCA08SEmxcuQLF1q9AX7yE6tnSlsiZgpQSqljSryA/oc/WH3D5syxprbtIBHhyUOH+GZBAdWBANlOJ4+MG8f1nZha1e+3zs2+8AK8/DL88pdwyy3WYw8+mBhdqJRSvUdiBXSvFx55xLr97W93+On+YJBrt2/n5XKr+/yVubn8Yfx4BrUxz3hrduyAdetg1Sp45RXrB0PImjXNAV2DuVKqqyVWQH/uOWukxeTJ1lnDDnqjspKXy8vJTErit+PG8aVjXHUHrO+PJ5+Ea65p7j+9aJHVnTBk4kS4+mprsS++opRS3SJxAroI/OY31u1vfatTTeDn7JE53xs+nC/blw87lltvhaVLrRObs2dbZRddZOXGzzgD5s6FU0/V1rhSqmckTkB/913rmp/9+8MNN3T46Q2BAMvsVEt7LkX2l79YwTw0w1vI7bdbi1JK9bTEmaf1scesv7ff3jwpRwe8UVFBfTDIjD59GH2czt0FBc1B+/e/b56rRCmlYilxAnqxPQPd5z7Xqac/V1YGHL917vVaPwDq6qy8+K23dmp3SinV5RInoDc2Wn87MXTSFwzy7hFr+sxrjjMQ6Yc/tHqxDB8OS5Zoflwp1XskTg79BAK6y+Fg31lnsaa6mmHHSNc0NMBLL1kjOP/6V4jR5IpKKdWqxAnoTfYlTjuRPwdIT0riopycY66TlmbNwfLOO9Yc4Eop1Zuc9CmXpkAAT+iyKO3Qty98/vMd2oVSSvWIxAnooRZ6BwP6s4cPM+Bf/+JX+/e3uc6iRfD1rzd/ZyilVG+UOCmXULTtYMplfW0t1YEAfVq7dDnws5/Bz39u9TWfPx/OO+9EK6qUUt0jMQK6SKdTLo+MH8+dQ4cyIPnoCxL/7nfWtTqNsQYSaTBXSvVmiRHQfT4rqLtcnbrS7fhW5q/9y1/gjjus24sXd+lV7JRSqlskRg69k+mWolDevYVXXmmeFfGXv4QFC06kckop1TMSK6B3IN1S4vEwcu1aZm7YQCDiMnwi1gWPAgH4wQ/gu9/t6soqpVT3OG5AN8ZMMMZsjFhqjDHfbLHObGNMdcQ6P+y+KreiEz1cXiwrIwgMSE4mKWK4pzGwfLk1R8u993ZxPZVSqhsdN4cuIjuB0wGMMUlAMfBKK6u+LyJXdG312qkTKZfQVLnXtTJ3S0aGdYk4pZSKJx1NuVwA7BGRwu6oTKd1MOVS7ffzr5oako3hSvs6obW18L3vwb593VRHpZTqZh3t5XId8Ewbj80yxmwCDgLfFZFtLVcwxiwAFgAMGTKEwsKu+V5I2bePgUCTw0FpO7b5bztFM97lorK4mErgz3/uw/335/Duu008/3xpl9RLKaV6UrsDujEmGZgLLGrl4Y+AESJSZ4y5DHgVGNdyJRFZAiwByM/PlxEjRnSq0kcpKADAnZlJe7a53J5qd0ZODiNGjCAQsLopAtxzj7td21BKqd6mIymXS4GPROSo5quI1IhInX37DcBljMnrojoeXwdTLpvr6gA4LT0dgNdeg717YfRo67JxSikVjzoS0K+njXSLMWagsa+mbIw5095uRWvrdosOzrS4ub4egNMyMoDmS5HedVenxiUppVSv0K6UizEmHbgI+GpE2dcARGQxMB9YaIzxA43AdSIRnbu7Wwda6EERtkS00Netg9WrITOzeTCRUkrFo3YFdBGpB3JblC2OuP0I8EjXVq0DOhDQ9zY2Uh8MMjg5mbzkZO58wCpfsAD69OnGOiqlVDdLjLlcOpByaZlu+eY3rdGh3/hGt9VOKaV6RGIE9A600DfZ6Zap9gnRM8+EZ9rqiKmUUnHkpAvos7OyqA0EuCj72JebU0qpeJMYk3N1IOVyXnY2vxk7lu1PZ3PRRbBmTTfXTSmlekhiBPQO9kMPzaj4z39CqQ4KVUoliJMq5bK3sZFVR44wpDKTPXvSyMvTgURKqcSRGAG9nSmXFZWVLNy9m89tGgukkZ+vA4mUUokjMQJ6O1voI91uru3Xj4a9WQCccUZ3V0wppXrOSRXQ5+TmMic3l0t3Wfc1oCulEklinBTtQC8XEdiwwbqtAV0plUhOmhZ6pc/H+tpaTnVn8IMfJLN9O4wc2TPVU0qpnnDSBPT3q6v5/NatXJydzVt3Tu2hiimlVM85aVIu4TnQ7TlclFIq0SRGQG9HCz00h0v9W7m8+irU1PRExZRSquckRkAPtdCPEdBDsyy++vO+zJsH9lXolFIqYSRGQA+10NtIudQHAhQ0NpJU7aKkyEFaGowf34P1U0qpHpBYAb2NFvrW+noEGFZkXeb09NN1hKhSKvHEf0APBsHrtW6npLS6SuiEaPan2YD2P1dKJab4D+iRPVys61QfJZQ/D+yyLmqhAV0plYjiP6B3oIdL+XYrxz5tWrfXSimletxxA7oxZoIxZmPEUmOM+WaLdYwx5mFjTIExZrMxpufawMfp4SIiVsrFZ8hIMaSmwqRJPVY7pZTqMccdKSoiO4HTAYwxSUAx8EqL1S4FxtnLTOBR+2/3O04Pl/0eD9WBAP3SXOzYZvB4IDm5R2qmlFI9qqMplwuAPSJS2KL8KuApsawFsowxg7qkhsdznJRLeIRoejrGmPbM36WUUnGpo3O5XAc800r5EKAo4v4Bu6wkciVjzAJgAcCQIUMoLGz5vdBxyZ9+yiDAk5TEoVa2N8Dv51d5eSQ3Ortkf0op1Vu1O6AbY5KBucCizu5MRJYASwDy8/NlxIgRnd1UMztIp/TtS2vbGwGcCUyZAveVwXvvwYQJJ75bpZTqbTrSQr8U+EhEWruscjEwLOL+ULus+7Wjl0tDA2zfbvVqHD68R2qllFI9riMB/XpaT7cAvAbcYYx5FutkaLWIlLSxbtc6Ri8XEeGH+/bh2ZpOMNifyZOPe1EjpZSKW+0K6MaYdOAi4KsRZV8DEJHFwBvAZUAB0ADc0uU1bcsxerlU+Hz8pLAQ97tDgf46oEgpldDaFdBFpB7IbVG2OOK2AF/v2qq10zFSLg5j+PHIkbxanMvH6IAipVRii/+RosdIueS4XPxw5EgcBX0AHfKvlEps8R/QjzOwyOuFLVus26ef3kN1UkqpGIj/a4oeI+Xy75oaSht9PLO8D5WFyfTt28N1U0qpHhT/LfRjpFweOnCAuZ9soXZKJbfd1sP1UkqpHhb/Af0YKZcijweAYW3Mk66UUokkcQJ6Ky30Io8Hloxm2W/SKS/v4XoppVQPi/+A3kbKJShCsccDywfx8E+S8ftjUDellOpB8R/Q20i5lHq9+OocUOvC7YYBA2JQN6WU6kGJE9BbtNCLPB4otYL8yJFtXp1OKaUSRvwH9DZSLkUeDxxqDuhKKZXo4j+gt5FyKWpqCgf0rpilVymlervECejHSbkopVSii/+AHkq5tGihH/B4INPHsFN8jB8fg3oppVQPS9ih/0UeD9xQxtL7c/hsVlYMKqaUUj0rcVrorQV0dJSoUurkkTgt9BYpl/uGj2ZHqZfBya0H9K1bt1JZWcm5556L0T6NSqkEEP8t9DZSLtOPDOBn04cxc/rRh7hx40amT5/OTTfdFFVeUlKCda0OpZSKP/Ef0NtIuezbZ/1tbYRov379yM/PZ+jQoeHWucfjYezYsYwZM4bt27d3Y4WVUqp7xHfKxeeDQACSksDZfCib6+p4aqMfyGq1y+KQIUNYtWoV/ogJXnbv3o3b7WbkyJGccsop3V93pZTqYvHdQm8j3bKispLnt9QA0X3QDxw4EL6dlJRESsQJ08mTJ3P48GGWLVsWbrWXl5dz2223cfDgwe6pv1JKdaF2BXRjTJYx5kVjzA5jzCfGmFktHp9tjKk2xmy0lx92T3VbaCPdMiUjg1FHsoHmgF5YWMjkyZO5+eab8WMUaOEAABlsSURBVNg9YFpKSkqiT58+4fs/+MEPePzxx7n99tu7vOpKKdXV2ptyeQh4U0TmG2OSgbRW1nlfRK7ouqq1Qxs9XC7JySG3Aj6lOaBv2rQJr9dLVVUVycnJ7dr8okWLKC8v57777guXeTyeqJa9Ukr1FscN6MaYTOCzwM0AIuIFvN1brXY6xsUtQidFQwF97ty5rF+/nv79+7e7m+KIESN48cUXo8puu+02ysrKeOSRRxg7dmwnK66UUl2vPS30UUAZ8GdjzFRgA3CXiNS3WG+WMWYTcBD4rohsa7khY8wCYAFYJyYLCwtPqPKuvXsZDHiTkiiJ2Naqhga++5sMHKWpNDbWUVhoBfD09HTq6+upr29Z9fapqqritddeo6mpiUOHDuFyuU6o/kop1aVE5JgLkA/4gZn2/YeA/2uxTl8gw759GbD7eNudPn26nLA1a0RAZObMcJEnEBCzcqU4Vq4UXyAgc+bMkX/+858SDAZPfH8iUlZWJq+++mpU2QMPPCD79u3rku0rpdSxAOuljbjanpOiB4ADIvKhff9F4IwWXwo1IlJn334DcBlj8k7wu+b4Wkm5FHs8CDA4JYX62lrefPNNrrjiiqguiiciLy+Pq666Knx/7dq1fOtb32LGjBn4fL4u2YdSSnXGcVMuInLIGFNkjJkgIjuBC4CokTfGmIFAqYiIMeZMrN4zFd1S40it9HIp8nhgfTayfQD/mp/Egw8+SEVFRbelR7Kzs7nxxhsZPXp0eB+BQIDly5dz6aWXtvsErFJKnaj29nL5BvC03cNlL3CLMeZrACKyGJgPLDTG+IFG4Dr7p0H3aqWXSyigFz83kI/HwH//913dWoUJEyawdOnSqLLly5fz+c9/nnPPPZdVq1Z16/6VUiqkXQFdRDZi5dIjLY54/BHgkS6sV/u0knIpamqCUut+LC9sMWXKFC677LLw/aqqKv7v//6PG264gfz8li+lUkqduPge+t9WyuWQNaiooOCfrFuXyRlnnEFSUlKPVeuqq67iqquuisrbv/zyyzzwwAPs3LmTv/3tb4B1QrqiooK8vO4/3aCUSnyJMfS/ZcrFvvTcj398C7NmzYrZyUpnxPwyZ511FnfeeSfXX399uKygoID+/fszZ86cWFRPKZVg4ruF3krKpbDaC1XJOF3CtdecR0NDHe4WI0lj4dRTT+Whhx6KKtu6dSvJyclkZ2eHywKBALNnz2batGn8+te/1r7uSql2i++A3krKZb89vmjoMFi69KkYVKr95s2bR2VlJUeOHAmXbdmyhdWrV3PgwAEefvjhcPk999xDeno6Cxcu1BSNUqpV8R3QW6RcGgIBjtQLjK3l9CkZMaxY+6WlpZGW1jw1zsSJE1m5cmVUkPf7/Tz88MM0NjaycOHCcPljjz1GUVERN954IxMmTOjReiulep/ECOh2C/2AxwPj6xj5l238NDMTj2d03E2k5Xa7mT17dlRZIBDgscceY/fu3VGt86eeeorVq1dzzjnnhAP666+/zksvvcTVV1/N5Zdf3pNVV0rFWHyfFG2RcjlgT4s7xBimTp1KZmYmjaGgH8dSUlK48cYb+d///d+o8m9/+9vcc889nHFG88Ddd999lyeffJJNmzaFyzZt2sTkyZP51re+FfX8Q4cOEQgEurXuSqmekxgtdDvlMsLt5of9RpNWX0rFmDEYY0htZSbGRDFv3jzmzZsXVXbzzTczbtw4Zs1qnrJ+586dbNu2LWp2yEAgwIgRIwCoqakJ/5J56aWXqK2tZc6cOQwcOLAHjkIp1VUSI6DbQXtMaiorbhvOxo3DWbXqE04//eSbW2XKlClMmTIlquzyyy9nw4YNUX3xy8rKyM7OxhgTlZb69a9/zQcffMB7770XDui///3veeCBB7jrrru44447AKioqOCFF15g1KhRXHLJJeHn+3w+7ZmjVIzEd0BvpZfLvn1W8cCBaGCxpaenR6VlAAYOHMihQ4fweqOntp83bx6jR49m9OjR4bK9e/dSUFBAXV1duGz37t0sXLiQGTNmRAX0YcOGUVtby969exlgX6H7t7/9LWvWrOH222/n3HPPBWDPnj288847TJgwgc9+9rMABINBNm/eTFpaGuPHjw9vMxgM4nDEd3ZQqZ4Q3/8lLVIuLxWVc+gQOJ3C4MExrFccaTl52N13383SpUsZOnRouOxHP/oR27Zt46abbgqXZWVl8Z//+Z9ceeWV4TIRoba2loaGBjIzM8Plq1ev5tlnn6W4uDhc9uGHH7JgwQIWLw7PIEF9fT3Tpk076svnvPPOIykpiffeey9ctmTJEsaOHcv9998fLissLGT27NnccsstUc8PDeg6dOhQuOz5559n4cKFrFy5Mly2e/du/uu//os//vGPUc+/7777uPfee6MGqC1btoz777+fnTt3hss++eQTHnzwQd56661wWVNTE7/73e94/PHHo7a5bNkylixZElWnTZs28fjjj7Nu3bpwWVVVFU888QSvvvpq1PNffvllnnrqKWpra8Nl69atY+nSpezYsSNcVlJSwtNPP83bb78d9fxnn32Wv/71r1HnUFavXs0zzzzD/v37w2Wffvopzz77LGvXrg2Xeb1ennvuOV555ZWobb799ts8//zzlJeXh8t27NjBCy+8wObNm8Nl1dXVvPjii6xYsSLq+X//+9956aWXohoOGzdu5KWXXqKgoCBcdvjwYV5++WXef//9qOcvW7aMV155JeqY/v3vf/Pqq69Gffb279/PsmXL+Oijj6KO6bXXXuPvf/971DZXr17N66+/TmVlZbisoKCA5cuX88knn4TLamtr+dvf/hb1GQVYuXIlb7zxBg0NDeGybr1GcVvz6nb30iXzoZ99tjUf+qpVIiIy7oXNAiLGsVemT58ujY2NJ74P1SHBYFBqamqiytauXStLly6NmjN+zZo1ctttt8mSJUvCZZWVlTJ16lSZGTG/vYjIZz7zGQFkzZo14bL77rtPAFm0aFG4bPPmzQLIqaeeGvX8kSNHCiB79uwJl33jG98QQB566KFw2YoVKwSQCy+8MOr5KSkpAkR9nubPny+APP/88+GyJ554QgD58pe/HC6rqKgQQLKzs1s9ptWrV4fLfvaznwkg3/ve98JlW7duFUAmTZrU6jHt3bu3W4/pySef7LFjGjVqVI+8T7E+pqefflpOBMeYDz2hUi6n1eayG5DgXioqKnrFCNGTjTEm6kLbADNnzmTmzJlRZbNmzYo6cQvWVMQbN248apv/+te/8Pv9UWmX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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",
    "\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_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_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
}
