{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Sensitivity of regularized NPG primal-dual for finite constrained MDPs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "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'"
      ]
     },
     "execution_count": 25,
     "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",
    "'''"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "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": 27,
   "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": 28,
   "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": 29,
   "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": 30,
   "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": 31,
   "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": 32,
   "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": 33,
   "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": 34,
   "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": 35,
   "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": 36,
   "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": 37,
   "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": 38,
   "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": 39,
   "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": 40,
   "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": 41,
   "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": 42,
   "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",
    "optimal_reward_value_LP = -res.fun\n",
    "optimal_policy_LP = theta_to_policy_naive(res.x,n,m)\n",
    "\n",
    "print('Optimal reward value:',optimal_reward_value_LP)\n",
    "\n",
    "print('Optimal policy:',optimal_policy_LP)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Regularized NPG primal dual method "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "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,optimal_policy_LP):\n",
    "\n",
    "    theta = np.random.uniform(0,1,size=n*m)\n",
    "    dual = 0\n",
    "    reward_value = []\n",
    "    utility_value = []\n",
    "    policy_distance = []\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) \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",
    "            policy_norm_squared = np.inner(prob - optimal_policy_LP, prob - optimal_policy_LP) \n",
    "            policy_distance.append(policy_norm_squared)\n",
    "            \n",
    "    return reward_value, utility_value, policy_distance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "metadata": {},
   "outputs": [],
   "source": [
    "# call regularized NPG primal dual method\n",
    "\n",
    "total_iterates = 2000\n",
    "\n",
    "stepsize = 0.1\n",
    "# stepsize = 0.01\n",
    "\n",
    "tau = 0.1\n",
    "reward_value_reg_3, utility_value_reg_3, policy_distance_reg_3 = Reg_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, tau, total_iterates, optimal_policy_LP)\n",
    "\n",
    "tau = 0.05\n",
    "reward_value_reg_2, utility_value_reg_2, policy_distance_reg_2 = Reg_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, tau, total_iterates, optimal_policy_LP)\n",
    "\n",
    "tau = 0.01\n",
    "reward_value_reg_1, utility_value_reg_1, policy_distance_reg_1 = Reg_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, tau, total_iterates, optimal_policy_LP)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 45,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXQAAAD4CAYAAAD8Zh1EAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjEsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+j8jraAAAgAElEQVR4nO3deXxU5dXA8d/JHpIAYQlLgLC7K0IqWsUWxA2ty6u4VVQUEbV9FbVV69LFvrW2trZqK1LrSt3ApVZxwxWtoLIji6wxLGFLyL7Pef94ZphJSEICSe7McL6fz/1k5rnPvffMnZszZ557Z0ZUFWOMMZEvxusAjDHGtA5L6MYYEyUsoRtjTJSwhG6MMVHCEroxxkSJOK823K1bN+3fv79XmzfGmIi0YMGCnaravaF5niX0/v378/XXX3u1eWOMiUgiktPYPBtyMcaYKGEJ3RhjooQldGOMiRKW0I0xJkpYQjfGmChhCd0YY6KEJXRjjIkSltBN85WWwmuvQW5u85dZtQrefRd27GjZtior3bZ+/3soKWn+crt2wQcfwKJFLdteVRVMnw6//jW09Culy8rgo49atkxAURFUV+/fsj5f3fsFBfu3nrvvhtdfb/njXrgQJk2CX/0Kamtbtuxbb8Gtt7rnuaVycmD58pYvB3s/xu3b3TGzPx55BF55BcrLW7bczp37t73mUFVPphEjRqipp7ZW9eWXVXNymte/vFx1yhTV009X/fbb5m/nhRdUTzpJ9ZFHmr9MdbVqZqYqqM6c2fzlZsxwywwc2PxlVN02wMXp8zV/uXfeccuddlrLtvf662657Oy67ZWVTS9XVKTarZtqQoJqXl6w/Y036t6vr7ZW9cILVePjVWfPDrbPnq16112qO3Y0vuyGDapnnOGe+4DcXNWUFNVJk9y6G1NYqDptWt19euGF7rE/80zjy6m6ZUL3xwcfuOVOPbXp5Rpy1FFu2Y8/DrZ99507zpoya5Zb7swzg23l5arnnaf67rtNx/7oo6qHHaa6cmWw/Y47VGNiVG+7rentFhWpfvRR8H5lpXvuRFSLi5tetr5Fi5p+jvYB+FobyauW0MNFRYXqxRerJiWpbtvWvGW2b3eJUkS1oKD523rwQffUT53ashgnTVIdM6ZuMigr27tfVVXwdm2t6vjxqtdcE2zz+Rp+jKGPoazMJfP33w+27dql+tpreyf48vLg7S+/dDH+/OfBtq1bVX/967pxqaqWlNSNacIE1eeeC7Z9951q376qTz7Z9IvKOeeojhypumRJMJ4uXdzU1AvtAw+4ZPKHPwRj+N733HNz//2NLxd40Tr66GDbc8+pxsaqXnRR48vV1qoeeaRb9qWXgu0zZ7oXstD9mJ9fd9mNG10SveGGYFtxser//Z9LUAGbNqnec4/q7t3Btpoatw8rKoJtDz+sevfdqmvXBttGjVI95BDVpUvrbjt032/bptqxo+rllwfbn3/ePaYhQ/Z+jkNdfrnrd/fdwbYpU1QTE91x05i8PPeinZoaPEYLC1XvvLPuca2q+uqrqlu21F323ntVn3462FZdXXf/tJAldC/s3OmqhtBqOyen8QqkokL1lFNUBw1yiSvgscfq3q9vxQrVhx4K3q+uVr3uOtVvvgm2+Xyqq1YF7+fmqn76qeq6dcG2Bx9U/cc/6sZzzTWq8+YF2+pXq5s2qWZkqN53n0sWtbWuuuzbV3Xz5sZjfuwx1c6dXWJSdf+EN92kmpXl9ltjpkxxh+xddwXjOesst8+a+gc56yy33B//GNzeLbe42JuqhO+/3y131lnB5LFqleqPfqS6Zk2wX2lp3eW2b3dV6+jRweVWrlS99lpX1QbU1Oz9wvb556pXXRVMrhUV7vFecUXdfo8/vveya9aorl8fvL96terVV9c9FqZNc8m7qaRZWekKheuvD1aSS5eqxsW5F6n6yT7Udde5fXbllcG2n/3Mtd13X+PL7d7tnscuXYJJs6bGVdQnnli3oq2ftPPzVX//+7rvdF56SXX4cNcekJvrKvz6y9cvSm6+WbV//7r77bTTVE84we3Txnz6qXucQ4cG22bPdm09e+59nOwnS+htYdeuppPW1Ve73XvGGcG20aPdgVFU1PAyhYV1/9GmT3frOOool6irq90wyT33NL7dBx4IViuBF48JE1wVEvqWMdSSJW6Zrl2D/ziBKn7gwMarnkcfdX3OOy/Ydv75qqBVv/tjw8v4fK5ih2A1XFGhetxx7i3sG280/tj+/ndXKS1fHmw76SSXaF54Ya/uVVX+PDV3rurgwcEXNZ9PdexY987mX/9qfHs+nxsyCk2c06a52M85p/HlAssWFgbv//e/brkBA1r2Ft3nU01Lc8s29eLTkAsucMs9/HCwrbZ238MaX33lqtGjjqrb/tJL7t1OUz7/3B3nocMa06e7x93Uc6vqXkhCq31Vl+BBdc6cppet75FH3HI33tiy5datc+90wBUsAYWF+x76++IL9//9k58E22prXQyffNKyocMmNJXQxc1vf9nZ2RrRX87173/D+efDtGkwefLe83fvhp/8BP7wB+jd250469/fnTyaNw8GD4ZnnnEn055+GkT2XsemTXDWWXDbbTBhAnzzDRx9tOv77bcwcODey5SXU3vV1SwfcxMDLzuetDRg6lQX54wZcMEFe7pOmACzZ8P48XD/gOmkz38HXn3VzayuhmuvJef8m7n56WEUF0NcHCQkwOmnw7XXutt89JGLIyuLNWtgxq/W8vErO/lSjuess+Cii9xDSEkJhlhT5eO/v36fGTtOZ/VqWL0aelR+x7XjNnPRQyeQkRHsqwpffAEPPQRvvw19u5XTZ0gyDz8Mhx2G2ycZGdC9O9XVMH8+vP++m778En73O/j5z/0rCtnHvlXfsuCTEqZ/PZw1a+Dii+H66/3x1bhzlV26BONYs8Y95Vmr3mF85zlwyy3QuzdPPgnHHOOmuDh3bvXrr91TfP31kJzslv/PUzvpMHsWQ7/fjW4/Pp3kjDQA1q6FxETo29f1W7gQ7rsPBg2CMWNg1ChIm/EY5QmdyDvuHAYclbonpokT3TnFww93++LQQ91hFRfnzpfGz/0QPv6YuSln8MfPv09ZmTt8jj8eRoyA1FSIjw8+zt274eqrITYWesfvYNTWl8j7nxs5Zphw9NGQlgYx/ssoPv7Y7Y+1a906hgxxU1YW9OwJRx3l+lVUwLS/1ZK7OYaycqG83PU/7DA44ggYORI6d3Z9162Dzz5zx0NurjtmBusaEnp1pfshXRg/3j02gA8/dP8e27a5accO6NTJbf/kk+F7/bZBbi7laRkszu/Hhg2wYQMUFrpDwedzj3/q1OD2ly9350eL80rZvWILKyoGsrs4lk6d3D4791zXr7LSHQ/5+cFtFxW5qawMbroJBgxwfVetcudA4+LcurdsgY4d3fG2v0RkgapmNzizsUzf1lPEVegzZrgTJ4G3Z4ETQqEnphYurDtOWN+aNarz5un27ap/vj1PK5I6unWEjhOHyMlRvePWKp0/P9hWeNtvtPi5uuPIBQWuAAhYvNitNjnZFfMla7ao7tqlBQV13/XdeafrB27ofupUF8rWrcHV796t2qNHsF9gGjjQFbeBgv7bb13BW79fYN2BoeSlS93ISkP9ROoOqV57rRsdaKhvaL+XX3ajIqmpe/cLDGuruvNfhx7qRkz69q3b7+abg/3efde9WRg2zL3zTkkJ9uve3Y0EqLoh+EB7Wpob/k5ODrYtXhxc55gxdbeXmBgsPm+/PdjvP/+p2y82NligJycHnxefz41a1X+8MTGqHTq4YeWAv/2t4X0I7rkNqK1teJ2BKfRceOCNYENTly511xm6T+pP06cH+wbeFDY0JSbWHXVp6JgMTPfeG+wXONfd2BT6Jipw+qKh6dprg/0WLWp6naFvIC+7bO/5I0fqAaGJCt2zr8+NKGVlcPPN7qV2xAi45BJXPr37Lowd6/o8+CDcfjvfXHgvd1f9kqlTXaVQx+DBbEoazNhRsHp1D+Yxnd/eU8mQU04BgkVkdTU88ICr1Kqq4snZAs8/76qK8xbcw8oZcNMmV5GIuCv78vNddTd4sAv32GPdlXv33QdPPNGLs86CF190FetPf+rCufVW+NGPXPU7c6b7+9BDbt7jj7s3Hp06wX/+466I8/kgL8/FtmqVq3wDlcaQIXDKKa5CuuUWV4XMnAkvv+yqrUGDXL9Bg9xViEOGuPUfc4yrLPPyXHUW6Adu2cJCV0FOmeLeFVRWuqowK4s9++zOO111B25dY8fCqafCD37g4ghYtszFvWqVu9+/P1x+Ofzwh8GKCmDJEvdGavHiYFvnzjBuHJx3ntsPsbHuarWrroK5c932v/rK9T3sMDjpJOjQIbj8pEnQqxd8+qmr6ior3ZSeHqx6AbKz3X5btMhVoV99BcXFrorv1ctdOZrqL9Jnz4aVK4PT6tWwfr17/levDq7zjDPcFaBxce4YmTcPli517ya6dg32i4mBWbNcJVlU5Cr2lSvdfli+3D1HAWPHuuNg0CB3vK5Z46ZNm4LxBdZ5991u2ykp7h1LWRmsWOHeXIXuo+OOc8fToYe656a83B13O3e6bYTup3POccdRz55u6t7d9c3Jce9qAkRg+HD3JnLAAHcsxca69vLyusfHscdCUpJ7J9Knj5s6d3b7YtiwYL/KShdj167uzWFGhvs/6djRPb7AsQkwdCh8//vuXV96OmRmundUbaaxTN/WU9hV6CtX1j1xWN+8eW5srJFxsJoPP9GamDj9Lb/Y80r84ot1+6xb5yo+cBcb3HBD3eHUyy5zVeQxxwRfzS++OHheMj/fDRk3VBVkZ9c9x6mq+tlne1cdEyc2/PAWLnTxnHiiaqdOda8mq6+62p0/Peywuif0G7sSK/RiElV3Xqk5V229+KK7aKCpc0klJaq//KXqs8/WHfJsSGmpq5pfftntm6ZiyMtTnT/f7dPmDJ9u2uTetDVnmNvnc7Fs2bLv4eziYne6prnDrxUVTZ9D31+tNPxrWgEHOoYuIlOBSYACy4CJqloRMj8ReBYYAewCLlbVjU2tM6zG0L/91g0uduvmBtri4117QYF7Oe7ZE3CV0u7dwfHOgPx8uPRSWPFeLltj+3LxxW4sd9Ei92qvCmef7aqjvDxXibz9dt0x2sWLXYUQMHAg/OMf7o1AKJ8Pnn3WVZEVFW465BBXFSck7P3QfD546SXX/6KLXLWyL4H0H2MfOzMm7BzQGDqQCWwAkv33XwauqtfnBmCa//YlwEv7Wm9YVehLl7oPzRx/fLAtMPB4ww1aXe0u6OjSxY1R3nlncKh8yRLVPn1c127dgpVt6FB6RUWwQj755MYvcvnwQ9Uf/MBd5VW/qjXGGNXWGUOPA5JFpBroAGypN/9c4Ff+27OAR0VE/BsPf0cd5QZ6Qz/CO3gwJCeTl1vN2GFuvC/g/vvdWOjcuW68LCbGna1/+WXo18/1SUwM9o+NhTffdOOIo0c3XEmDmzd6dOs/PGPMwWGfCV1VN4vIg8B3QDnwnqq+V69bJpDr718jIoVAV6DOlxaIyGRgMkBmZiY5OY3+NJ73Bg1i87srGXt6X8rLoV+/au66q4CuXX3cdltXzj23kO++KwVgxow4eveuQdWdlGnIkUe6v1u3tlP8xpiDzj7H0EUkHXgFuBjYDcwEZqnqjJA+y4EzVHWT//46YKSqNvotNGExhv7ss26g+9573SnoelTh73931zM//rg7Aw6ukE9KavjScWOMaUtNjaE357TXWGCDqu5Q1WrgVeD79fpsBvr6NxYHdMKdHA1fFRVw++3wl7/AO+802EUEbrzRff4nkMzBXZpkydwYE26aM4b+HXC8iHTADbmcAtQvrd8ArgS+AC4EPgz78fOkJHeB9axZ5J92CZePc1ernHiiG/OeNMldQxppVJXy8nKKiooQETIyMpCQVx9VpaSkZM9UXV1Njx496NKly55+tbW1FBQUUFpaSllZGeXl5XTq1ImMjAxS/RcZV1RUUFxcvGd+eXk5iYmJdO7cmfT0dOLi4va0V1RUUFlZSUWFuzAqLS2NtLQ0EhMTqa6upqqqqs5UW1tLUlISHTp0IDExEZ/PR01NDbW1tXv619bWEh8fT2JiIon+ExY+n29P3+rqampqaoiJiSE+Pp74+HhEZM/JI5/PR21tLTU1NQDExcURFxdHTMilPYE+tbW1qCqxsbHExMQQExOzZ1+FrktV68yvv98D8QXm1V9P6ATs6Sf1qofQPoF+oX+b0ycwr6F/0/rbM60vMTGRzMzMVl9vcy9b/DVuyKUGWIS7hPEu3NnWN0QkCXgOOBbIBy5R1fVNrTMshlxwH1oYO9Zd1hcqLc190OXoo1tnO+Xl5WzZsoWsrCzi4oKvo5s3b+bNN98kJyeHvLw8duzYQZ8+fRg5ciTZ2dls376dzz77jM8//5zc3FwKCgooKCggNTWVI488kqOOOoqEhARWrFjBypUr2bRpE7Uh303doUMHBg0aRNeuXdm6dSu5ubmUlZXtFV9SUhIZGRkUFxeze/fuBv/RwR2IgeRqjNk/I0eOZN68efu1bFNDLgfnd7nU1kJsLD6f+46Gr75y1fgDD7gh9c8/d59Ue/xxV63vD5/Px6xZs3jmmWdYuXIlGzduRFVJS0tj1KhRjBgxgk8++YS5c+c2mjz3V1JSEmlpadTU1FDQwI8eJCcn07FjR1JTU4mLi2Pr1q0UFRXV6dO5c2fS0tL2VMmFhYVs376dcv+VQImJiaSmppKSkkJycjJJSUlUVlbuecFRVZKSkkhOTiYxMZGkpCSSkpLw+XyUlJRQXFxMVVUVCQkJeyroQLUdExNDRUUFZWVlVFRUEBsbS1xcHLGxscTHx5OQkEBsbCzV1dVUVlZSWVlZp+KNj4/fU3H7fD6qq6uprq52H7wIqY4D6wT2VOu1tbV1qtnY2FhiY2MRkT2VuK/eD0sEKvfAO4BAJR5Qv3IH9uoT2F5oRV6/yg60hfZrqNJuTp/A9kL/NraepjSnT2uuK1piGjZsGK8GvjephSyhh1J1n8UdOhQefJD3FnXnzjvdD6j4Pz/UzNUoS5cu5ZVXXuGVV15hzZo1nHTSSZx55plkZGTwhz/8gRUrVuzpHxsbS7du3di2bVud9SQmJjJu3DiOPfZYevToQbdu3Vi3bh3z589nwYIFpKenM2rUKEaNGsWhhx5Keno6nTt3pqCggOXLl7Ns2TKqq6s57LDDOPzww+nfv/+eIQiAgoIC1q1bR35+Pr1796Zv37507NhxrwOwqKiIHTt20KlTJ9LT0/ckuvrKysqIi4sjobFrL40xbcoSeqjly90XiHTt6q4xTE7G52vZpyK3bt3KFVdcwZw5c5rs169fP26//XbGjBnDwIEDSUhIYNOmTXz88ccsXLiQ4cOHc84559Ax9AsljDGmCZbQ61u/nvx539LlsjP22XXOnDksW7aME044gREjRvDhhx8yYcIEduzYQXp6OuPHj+eCCy5g2LBhfPzxx7z99tusX7+eH//4x1x11VVWyRpjWpUl9Ho2bHDfEnfGGW6opaHhr5qaGu644w7+9Kc/7WlLSUmhtNR9mGjMmDHMmDGDXr16tVfYxhhzwNehR4f//heeegpwvyeh6r4cq6Fknp+fz7hx4/jTn/5EXFwcF198MUOHDqW0tJSYmBjuu+8+3nvvPUvmxpiwcnB8H/rOne7rDgsKqO3Wg6eeGgfANdfs3XXXrl2ceOKJrF69mu7duzNz5kx+8IMfALBlyxbi4uLICP1JHWOMCRMHR0Lv1g1++1uYM4eP4k8jN9d92b0/T+9RWVnJ+eefz+rVqzniiCOYPXs2/QLftgX07t27nQM3xpjmO3iGXG64AV55hX8+417DJk6se2WLqjJp0iTmzp1L7969eeedd+okc2OMCXcHR4Xut2On8Nprbtz8yitdW01NDStWrODJJ59kxowZdOjQgf/85z/06dPH22CNMaaFoj+h//zn7kcSb7+dJd8NJT7efdS/Z88qLrjgUt599909V66ICC+++CLDm/OzPsYYE2aiP6HPmuWuU7z1VsaOdT8UDDBz5sw9H70dMGAAxx13HBMmTOCss87yMFhjjNl/0Z/Qn37a/cT5kCGA+44WgMsu+xsAjz32GFOmTPEmNmOMaUVRf1JUR53MFV/+hEcfjyfwXUgLFy7kiy++oFOnTkyYMMHbAI0xppVEfUL//HN47jn4xS8gL8+1/e1vrjqfOHEiKSkpHkZnjDGtJ7qHXF59ldLnttOXcZxzRT9693YfHHr++ecBuOGGGzwO0BhjWk90V+jTpnH669czjMVk+7/54KmnnqKiooIzzjiDIf5xdWOMiQbRndDHj2dW6lUs4liys92PGPz9738H4MYbb/Q4OGOMaV1RndDzL7iW8SVPsSu5L4ceCu+++y4bNmygf//+nHnmmV6HZ4wxrSqqE/qCBe7vsGEQFwePP/44ANddd12jv8hjjDGRKnpPiq5eTZ/tJUy54jAGHtlhz48xx8XFMXHiRK+jM8aYVrfPCl1EDhGRxSFTkYjcXK/PD0WkMKTPvW0XcjP95S8cdnk2jw17nJ/9DP75z3/i8/k4//zz6dGjh9fRGWNMq9tnha6qq4FhACISC2wGXmug61xVPbt1wzsAPXvCEUfA0UdTW1vLE088AcDkyZM9DswYY9pGS4dcTgHWqWpOWwTTmgpv/iXvH/FLvjcYlr/zFrm5uQwaNIgxY8Z4HZoxxrSJlib0S4AXGpl3gogsAbYAt6nqN/U7iMhkYDJAZmYmOTlt97rw6adJXHFFD0aMqCA9/S8AjB8/ntzc3DbbpjHGeKnZCV1EEoBzgDsbmL0QyFLVEhEZB7wO7PWpHVWdDkwH9yPRWVlZ+xX0PpWVsXljPADHHFPN009/SHx8PFOnTrWfjzPGRK2WXLZ4JrBQVbfVn6GqRapa4r89G4gXkW6tFGPLPfcct/4yhd9wD7W1X+Hz+Tj77LMtmRtjolpLEvqlNDLcIiI9RUT8t4/zr3fXgYe3nwoKiNdqYvCRl/cWAKeeeqpn4RhjTHto1pCLiKQApwLXhbRNAVDVacCFwPUiUgOUA5eoqrZ+uM2zc9IdZN45lY7J1chCN/JjJ0ONMdGuWQldVUuBrvXapoXcfhR4tHVD238LFkAVifQ9tIZFi/Lo1asXQ4cO9TosY4xpU1H50f/AhSzJyRsBGD16NP4RIWOMiVpRmdAnfXYVNRdcxJCUewCX0I0xJtpF53e5vPUWsTt3Mq9zZ8ASujHm4BCVFTovv8zG++9n7e7d9O3bl4EDB3odkTHGtLmorNDPf3g0S5YMpZYnGDPmJBs/N8YcFKIyoX/5JWzZkglU2XCLMeagEXUJvXb1WsZvfYulHMVHbLOEbow5aERdQi/56Cv+ojfzEueSM7AP/fr18zokY4xpF1GX0LenDuQ5bmQhPfj+99O8DscYY9pN1CX0dd1G8lNGAnO4K6vS63CMMabdRN1li1u37rlFZmaml6EYY0y7irqEfnTqekZmPkUs71hCN8YcVKJuyGXEn3/MvM3zOBHIzLzF63CMMabdRF2FTteu5MXEsA3o3bu319EYY0y7ibqE/sHUV+nlG8GGmBT7hSJjzEEl6hL6NdcI8CXdu48gNjbW63CMMabdRFVCV4Vt29xD6tPHkrkx5uASVQm97LOFrKwYxDNcRr9+6V6HY4wx7SqqEvrulVvpTw7d2UyfPnbJojHm4BJVly2uGzCWUaxD+IrJmRu9DscYY9rVPit0ETlERBaHTEUicnO9PiIiD4vIWhFZKiLD2y7kxm3NT2QDA1lPjH2oyBhz0Nlnha6qq4FhACISC2wGXqvX7UxgiH8aCTzm/9uugh/7z6N37yPae/PGGOOplo6hnwKsU9Wceu3nAs+qMw/oLCK9WiXCFrgx9jH+kZrJkdxnFbox5qDT0jH0S4AXGmjPBHJD7m/yt20N7SQik4HJAJmZmeTk1H9dODAZM19gUskWXgFqa2tbff3GGBPOmp3QRSQBOAe4c383pqrTgekA2dnZmpWVtb+ralDxDTdw69y55KSmcvjhh7fquo0xJty1ZMjlTGChqm5rYN5moG/I/T7+tnY14cXT+DMvUt1zRHtv2hhjPNeShH4pDQ+3ALwBXOG/2uV4oFBVtzbSt8189FEqcDG9etl3uBhjDj7NGnIRkRTgVOC6kLYpAKo6DZgNjAPWAmXAxFaPdB+qdpcxtuhtNtOT/v1T2nvzxhjjuWYldFUtBbrWa5sWcluBG1s3tJbZteg7XuFC1tCfp/pc6mUoxhjjiaj5pOiOgjjmcR7b7ENFxpiDVNQk9I1xg/kfXgPe4rXMaq/DMcaYdhc1X86Vlxe4ZT8ObYw5OEVNQu+TUUVywhvAPEvoxpiDUtQk9NM/+wVlVefyM3mSHj16eB2OMca0u6hJ6KW7dgEQ17Gj/fScMeagFDUJfc6PbiSGPnwwyD7yb4w5OEVNQv/Nb/qg5BKbeKbXoRhjjCeiJqHv3u3+9uqV7G0gxhjjkai5Dv13eVNJpZT5Kdleh2KMMZ6ImoQ+quoz+rKJ9V1O8DoUY4zxRNQk9MvlWTprEecM2eV1KMYY44moSOgVFfCpbzRQxdV93vE6HGOM8URUnBQtLNxziy5d0r0MxRhjPBMVCb0zu/lN+gguYjTp6ZbQjTEHp6gYckncnss9BQv5BuhsCd0Yc5CKigpdO3bkkZgYXgC6dOnidTjGGOOJqKjQ5+b04H99DxIX9zW/TbYPFhljDk5RUaF/8UU5MJX4+HFeh2KMMZ6Jigq9ZOMWBpJPTVLhvjsbY0yUalaFLiKdRWSWiKwSkZUickK9+T8UkUIRWeyf7m2bcBt25PwZrGMwd9a83p6bNcaYsNLcCv2vwDuqeqGIJAAdGugzV1XPbr3Qmi+/IpEN9KeoQ0cvNm+MMWFhnxW6iHQCTgb+CaCqVaq6u60Da4lnO13BQDbw70FjvQ7FGGM805wKfQCwA3hKRI4BFgA3qWppvX4niMgSYAtwm6p+U39FIjIZmAyQmZlJTk7OAQUfsHNnDQDJyRWttk5jjIk0zUnoccBw4KeqOqEUBWwAABChSURBVF9E/grcAdwT0mchkKWqJSIyDngdGFJ/Rao6HZgOkJ2drVlZWQcaPwCpqQuAtWRlpdBa6zTGmEjTnJOim4BNqjrff38WLsHvoapFqlrivz0biBeRbq0aaRP+WXIJyxjCKemr22uTxhgTdvaZ0FU1D8gVkUP8TacAK0L7iEhPERH/7eP8622377HtumsXRwIdO9pJUWPMwau5Hyz6KfAvEVkKDAN+JyJTRGSKf/6FwHL/GPrDwCWqqq0fbsPuHjacowE53H4g2hhz8GrWZYuquhio/9tu00LmPwo82opxNVtNDcz46D2ggI49VnoRgjHGhIWI/+i/+y70GCDGvgvdGHNQi/iP/hdtK+cRfsZ24klP/5nX4RhjjGciPqEXf5fPT/gbeXSlU/rvvA7HGGM8E/EJfWtRLE/wV3xs4FH76lxjzEEs4sfQvyuM4xH+lycST/c6FGOM8VTEJ/S8vAoAkpIqPI7EGGO8FfFDLod3XMxYrieuW1fgPK/DMcYYz0R8Qh+45h3e503er+rrdSjGGOOpiB9yyY+N5QNge0aG16EYY4ynIj6hv1R+MmO5ijlHefLbGsYYEzYiPqF/8snhwFOUlR3qdSjGGOOpiE/opcXuIXTvHu9xJMYY462IT+iP7byJUjpw0u4lXodijDGeiviE3qG2nA6U0zEjzetQjDHGUxGf0E/nDVIpJmb0970OxRhjPBXR16H7fFDrS6OUGHpn2VfnGmMObhGd0EtKAGqACrp3t4RujDm4RXRCT0tTXieJUoT0tEKvwzHGGE9FdEIv3bWLc1GqUBJSUrwOxxhjPBXRJ0ULior4EfDTzp1BxOtwjDHGU81K6CLSWURmicgqEVkpIifUmy8i8rCIrBWRpSIyvG3CrevtOfAmS3gl9sH22JwxxoS15g65/BV4R1UvFJEEoEO9+WcCQ/zTSOAx/982lZNTCRxNTExeW2/KGGPC3j4Tuoh0Ak4GrgJQ1Sqgql63c4FnVVWBef6Kvpeqbm3leOuoWZfDZD6lKCYfOK0tN2WMMWGvORX6AGAH8JSIHAMsAG5S1dKQPplAbsj9Tf62OgldRCYDkwEyMzPJyck5gNChw6qFPMBdzCn9Hjk5lx3QuowxJtI1J6HHAcOBn6rqfBH5K3AHcE9LN6aq04HpANnZ2ZqVldXSVdSxNTaTfzCJ4oxYxh7guowxJtI156ToJmCTqs7335+FS/ChNgOhPxnUx9/Wpr6WQ5nMP5hzyI/aelPGGBP29pnQVTUPyBWRQ/xNpwAr6nV7A7jCf7XL8UBhW4+fA5SVufA7dozoqy+NMaZVNPcql58C//Jf4bIemCgiUwBUdRowGxgHrAXKgIltEOteBqXPYxcfMqhP//bYnDHGhLVmJXRVXQxk12ueFjJfgRtbMa5m+d/tD3Mqa/ly55XAxe29eWOMCSsRPVZR5vOxHYjp0sXrUIwxxnMRndB/2+V0epDNtlPO8joUY4zxXEQn9GXL7gS+orTUKnRjjInohF5bmwRA166JHkdijDHei+iE/lLNNcxjJL3LtnkdijHGeC5ivw/d54Nj+IZBrGdbl2SvwzHGGM9FbEIvL4ezeIvO5PHBoUO8DscYYzwXsQm9qKiW1RwKdCU53X5P1BhjInYMfceOcgBESomJidiHYYwxrSZiK/ROiTv4P7KoSOqE+zYCY4w5uEVsQvfl5/EL8tlRVeh1KMYYExYidqyiuKaGXwDPde/udSjGGBMWIrZCn7eyI/fzHAOSt3KL18EYY0wYiNgKfe1aAS6nvPxYr0MxxpiwELEVes22XYxkHvGxbf7DSMYYExEitkLvu/ZL5nEC9xY+7nUoxhgTFiI2oedXJzOf48hN6eN1KMYYExYiNqF/mnwCxzOfp4de4XUoxhgTFiI2oZeWur+pqd7GYYwx4SJiE3qHDtuAz8jIqPQ6FGOMCQvNSugislFElonIYhH5uoH5PxSRQv/8xSJyb+uHWtcvSu5lG6O4svrVtt6UMcZEhJZctjhaVXc2MX+uqp59oAE1V2JxMRnA2oSE9tqkMcaEtYgdcnl0yHB6kMC2Mad4HYoxxoSF5iZ0Bd4TkQUiMrmRPieIyBIReVtEjmil+Br13tz72U4lW4sHtPWmjDEmIjR3yOUkVd0sIhnA+yKySlU/DZm/EMhS1RIRGQe8Duz1M0L+F4PJAJmZmeTk5Ox34FVVbqjF5ys6oPUYY0y0aFZCV9XN/r/bReQ14Djg05D5RSG3Z4vI30WkW/0xd1WdDkwHyM7O1qysrP0O/JfVt9KdIg7tfCEHsh5jjIkW+xxyEZEUEUkL3AZOA5bX69NTRMR/+zj/ene1frhB5+lsruUJuif72nIzxhgTMZpTofcAXvPn6zjgeVV9R0SmAKjqNOBC4HoRqQHKgUtUVdsoZnw+uIMH6ME2fnWYjaEbYwyAtGHebVJ2drZ+/fVel7Q3S0kJpKUBlFJYWEvHjh1bNTZjjAlXIrJAVbMbmheRly0WFwdehEpISUnxNBZjjAkXEfl96PFSzCWcSllsMrGxH3sdjjHGhIWITOhalMcLfEmhT7wOxRhjwkZEDrmUlZXxIjCnQwevQzHGmLARkRX68s3JXMrP6dullAu8DsYYY8JERCb0ZcsAHqCk5COvQzHGmLARkUMuhTvL6UApCfHVXodijDFhIyITeu9v5lJKKs8X3+11KMYYEzYiMqFXlVRRRjKVcYleh2KMMWEjIhP6R51+SApl3DXoNq9DMcaYsBGRCb2kxH1SNDXN40CMMSaMRGRCr6mpAopITfXme2iMMSYcRWRCv7XDr3ifTlybPtPrUIwxJmxEZELvkpfHWKBLZaXXoRhjTNiIyIT+5uDBjAW2jRzpdSjGGBM2IjKh/2veL/iA1WzUo70OxRhjwkZEJvTi4q7AUFJT7cu5jDEmICK/y2V8xZukovSSwV6HYowxYSMiE/pPav7JcJYwz/e816EYY0zYiMiE/i8u5TNOZvSQPl6HYowxYSPiEnptLfyZ2wEf3x22xetwjDEmbDTrpKiIbBSRZSKyWES+bmC+iMjDIrJWRJaKyPDWD9UpLd1zi9RU+4FoY4wJaEmFPlpVdzYy70xgiH8aCTzm/9vqYsTHkdxMMbGkpDzQFpswxpiI1FpDLucCz6qqAvNEpLOI9FLVra20/j3ia4tYxiNUAAkJD7X26o0xJmI1N6Er8J6IKPC4qk6vNz8TyA25v8nfViehi8hkYDJAZmYmOTk5LQ646LvviAF8MTF03I/ljTEmWjU3oZ+kqptFJAN4X0RWqeqnLd2Y/4VgOkB2drZmZWW1dBUsyo9lOGeS0a2WbfuxvDHGRKtmnRRV1c3+v9uB14Dj6nXZDPQNud/H39bqvvyyFphNaan9/JwxxoTaZ0IXkRQRSQvcBk4Dltfr9gZwhf9ql+OBwrYYPwcoKHA/DB1vPxBtjDF1NGfIpQfwmogE+j+vqu+IyBQAVZ0GzAbGAWuBMmBi24QLnZZ+wTpO44uKocCYttqMMcZEnH0mdFVdDxzTQPu0kNsK3Ni6oTUiv4CBbGCtZLTL5owxJlJE3Lctzu/4PQaxlvt7Xet1KMYYE1YiLqHvKk9iPYMo7NTF61CMMSasRFxCLy52f1PsU//GGFNHxCX0a3s+wkOkcWnG616HYowxYSXiEnq/jSu4mRIOKd3mdSjGGBNWIi6hL+zbl6nAtsMP9zoUY4wJKxGX0DcPGMBzXbtSdbT9QLQxxoQSdwl5+8vOztavv97rq9WNMcY0QUQWqGp2Q/MirkI3xhjTMEvoxhgTJSyhG2NMlLCEbowxUcISujHGRAlL6MYYEyUsoRtjTJSwhG6MMVHCsw8WicgOIGc/F+8G7GzFcFpTuMYWrnFB+MYWrnFB+MZmcbVcS2PLUtXuDc3wLKEfCBH5urFPSnktXGML17ggfGML17ggfGOzuFquNWOzIRdjjIkSltCNMSZKRGpCn+51AE0I19jCNS4I39jCNS4I39gsrpZrtdgicgzdGGPM3iK1QjfGGFOPJXRjjIkSEZfQReQMEVktImtF5I523nZfEflIRFaIyDcicpO//VcisllEFvuncSHL3OmPdbWInN7G8W0UkWX+GL72t3URkfdFZI3/b7q/XUTkYX9sS0VkeBvFdEjIflksIkUicrNX+0xEnhSR7SKyPKStxftIRK70918jIle2UVx/FJFV/m2/JiKd/e39RaQ8ZN9NC1lmhP8YWOuPXdogrhY/d23xf9tIbC+FxLVRRBb729tznzWWJ9r+OFPViJmAWGAdMBBIAJYAh7fj9nsBw/2304BvgcOBXwG3NdD/cH+MicAAf+yxbRjfRqBbvbY/AHf4b98BPOC/PQ54GxDgeGB+Oz1/eUCWV/sMOBkYDizf330EdAHW+/+m+2+nt0FcpwFx/tsPhMTVP7RfvfV86Y9V/LGf2QZxtei5a6v/24Ziqzf/T8C9HuyzxvJEmx9nkVahHwesVdX1qloFvAic214bV9WtqrrQf7sYWAlkNrHIucCLqlqpqhuAtbjH0J7OBZ7x334GOC+k/Vl15gGdRaRXG8dyCrBOVZv6hHCb7jNV/RTIb2CbLdlHpwPvq2q+qhYA7wNntHZcqvqeqtb4784D+jS1Dn9sHVV1nrqM8GzIY2m1uJrQ2HPXJv+3TcXmr7IvAl5oah1ttM8ayxNtfpxFWkLPBHJD7m+i6YTaZkSkP3AsMN/f9BP/26UnA2+laP94FXhPRBaIyGR/Ww9V3eq/nQf08Cg2gEuo+w8WDvsMWr6PvIjxalwVFzBARBaJyCciMsrflumPpT3iaslz58X+GgVsU9U1IW3tvs/q5Yk2P84iLaGHBRFJBV4BblbVIuAxYBAwDNiKe6vnhZNUdThwJnCjiJwcOtNfgXhynaqIJADnADP9TeGyz+rwch81RkTuAmqAf/mbtgL9VPVY4BbgeRHp2I4hheVzV8+l1C0e2n2fNZAn9mir4yzSEvpmoG/I/T7+tnYjIvG4J+lfqvoqgKpuU9VaVfUB/yA4RNCu8arqZv/f7cBr/ji2BYZS/H+3exEb7kVmoapu88cYFvvMr6X7qN1iFJGrgLOBH/uTAP4hjV3+2wtw49ND/TGEDsu0SVz78dy163MqInHA/wAvhcTcrvusoTxBOxxnkZbQvwKGiMgAf8V3CfBGe23cPy73T2Clqv45pD107Pl8IHDW/Q3gEhFJFJEBwBDcCZi2iC1FRNICt3En1Jb7YwicHb8S+HdIbFf4z7AfDxSGvB1sC3UqpnDYZyFauo/eBU4TkXT/cMNp/rZWJSJnAD8HzlHVspD27iIS6789ELeP1vtjKxKR4/3H6hUhj6U142rpc9fe/7djgVWqumcopT33WWN5gvY4zg7kbK4XE+6M8Le4V9i72nnbJ+HeJi0FFvunccBzwDJ/+xtAr5Bl7vLHupoDPHu+j9gG4q4eWAJ8E9g3QFfgA2ANMAfo4m8X4G/+2JYB2W0YWwqwC+gU0ubJPsO9qGwFqnFjktfszz7CjWmv9U8T2yiutbgx1MCxNs3f9wL/c7wYWAj8KGQ92bgEuw54FP+nwVs5rhY/d23xf9tQbP72p4Ep9fq25z5rLE+0+XFmH/03xpgoEWlDLsYYYxphCd0YY6KEJXRjjIkSltCNMSZKWEI3xpgoYQndGGOihCV0Y4yJEv8PmaKA9NOQsmEAAAAASUVORK5CYII=\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_reg_3 = np.array(reward_value_reg_3)\n",
    "reward_value_reg_2 = np.array(reward_value_reg_2)\n",
    "reward_value_reg_1 = np.array(reward_value_reg_1)\n",
    "\n",
    "# plt.plot(num_grads,reward_value[::num_every], \"c-.\", linewidth=2)\n",
    "plt.plot(num_grads,reward_value_reg_3[::num_every], \"k-\",linewidth=2)\n",
    "plt.plot(num_grads,reward_value_reg_2[::num_every], \"b--\",linewidth=2)\n",
    "plt.plot(num_grads,reward_value_reg_1[::num_every], \"r:\",linewidth=2)\n",
    "plt.grid(axis='y', color='0.85')\n",
    "plt.draw()\n",
    "get_f_value = plt.gcf()\n",
    "get_f_value.savefig('NPG_primal_dual_sensitivity_reward_tau.png',dpi=300)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "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_reg_3 = np.array(utility_value_reg_3)\n",
    "utility_value_reg_2 = np.array(utility_value_reg_2)\n",
    "utility_value_reg_1 = np.array(utility_value_reg_1)\n",
    "\n",
    "# plt.plot(num_grads,utility_value[::num_every], \"c-.\", linewidth=2)\n",
    "plt.plot(num_grads,utility_value_reg_3[::num_every], \"k-\", linewidth=2)\n",
    "plt.plot(num_grads,utility_value_reg_2[::num_every], \"b--\", linewidth=2)\n",
    "plt.plot(num_grads,utility_value_reg_1[::num_every], \"r:\", linewidth=2)\n",
    "plt.grid(axis='y', color='0.85')\n",
    "plt.draw()\n",
    "get_g_value = plt.gcf()\n",
    "get_g_value.savefig('NPG_primal_dual_sensitivity_utility_tau.png',dpi=300)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plot sqaured norm of policy gap  \n",
    "\n",
    "p_distance = plt.figure\n",
    "\n",
    "num_every = 20\n",
    "num_grads = np.arange(0, total_iterates, num_every)\n",
    "\n",
    "# case #1\n",
    "policy_distance_reg_1 = np.array(policy_distance_reg_1)\n",
    "\n",
    "# case #2\n",
    "policy_distance_reg_2 = np.array(policy_distance_reg_2)\n",
    "\n",
    "\n",
    "# case #3\n",
    "policy_distance_reg_3 = np.array(policy_distance_reg_3)\n",
    "\n",
    "# convert y-axis to Logarithmic scale\n",
    "plt.yscale(\"log\")\n",
    "\n",
    "plt.plot(num_grads,policy_distance_reg_1[::num_every], \"r:\", linewidth=2)\n",
    "plt.plot(num_grads,policy_distance_reg_2[::num_every], \"b--\", linewidth=2)\n",
    "plt.plot(num_grads,policy_distance_reg_3[::num_every], \"k-\", linewidth=2)\n",
    "plt.grid(axis='y', color='0.85')\n",
    "plt.draw()\n",
    "p_distance = plt.gcf()\n",
    "p_distance.savefig('NPG_primal_dual_squared_policy_distance_tau.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
}
