{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Sensitivity of regularized NPG primal-dual for finite constrained MDPs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "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": 48,
     "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": 49,
   "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": 50,
   "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": 51,
   "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": 52,
   "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": 53,
   "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": 54,
   "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": 55,
   "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": 56,
   "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": 57,
   "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": 58,
   "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": 59,
   "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": 60,
   "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": 61,
   "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": 62,
   "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": 63,
   "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": 64,
   "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": 65,
   "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": 71,
   "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_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",
    "            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": 72,
   "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": 73,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAXQAAAD4CAYAAAD8Zh1EAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjEsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+j8jraAAAgAElEQVR4nO3deXhU5dn48e+dnSRAEsIaIAEEZLEIBhHrWpBW6lKXulURWkXcl7dutS61ta19fVu0Win+UIsKLiBKrRuKClVB2WRfAhK2hJ2EhOxz//54ZphJSEICSSYzuT/Xda6Zec5zzrnnzJl7nvOcZURVMcYYE/oigh2AMcaYhmEJ3RhjwoQldGOMCROW0I0xJkxYQjfGmDARFawFp6amakZGRrAWb4wxIWnx4sV7VLV9deOCltAzMjJYtGhRsBZvjDEhSUSyaxpnXS7GGBMmLKEbY0yYsIRujDFhwhK6McaECUvoxhgTJiyhG2NMmLCEbowxYcISektWVgbLl0NBQf2n3brVTV8f8+bBCy/Anj31m66oCN54Ayoq6jfdgQOwcCHs2FG/6QAWLIDZs+s/HcD+/ZVfH8v6BcjPh23bjm3amTNh6VLweOo33cMPw113uc+qPvbuddNddx3s3Fm/aSsq4LXX4Nln6zedT1FR5ddffgnHeo3LnXfCQw9Bbm79plu2zE175ZX1304bkqoGZTjllFPU1GDfPtXdu+tePy9P9bHHVG+8se7TeDyq55yjCqp//Wv94vvRj9x0CxbUfZpdu1Tj491069fXb3mnnOKm++ADf1lxce3TVFSonnSSm+655/zlBw8efd1Onuymu+aayvP7059U9++vebr8fNUrrlDt3Nktx2fkSNVLL1XNza152nXrVK+9VnXuXH/Zv/7l4rjggtrj3b5ddfp0/+uSEtWEBDft9u21T1tSolpY6H89daqb7v77a5+uqoIC1dRU1chI1UOH/OVbtrh1V5s//MEt8667/GX5+aoXX6z6zTc1T+fxqD7+uGpKimpWlr/817928/vf/619uVu3qs6YUbksJUU1Kkp1w4bap63q3XfdMm+5pXL5nj31m08dAIu0hrxqCb2xffutS7g+Ho8bquPxqF53ndugHnrIX15a6r54Ndm6VTU21n2cvg3b41FdvLj22KZNc9MsX+4vW7VKdeXK2qf71a9U27at/GV480335a36fgLf63PPqV54YeWy3/xG9T//qX15v/+96tChqh9/7C+77DLVyy9X3bGj5ukWLlQdNMglRp877nCJ5/PPa54uJ0e1Y8fKPwR/+5tbV0OH1v75nXqqalyc6ief+MvT01Wjo1VXrKh5mY8+6ub/ox/5yyZOVE1MVH3llZqn27XLJe+oKP/637NHddw41fPOq1w3cDtUVf3oI9WePd0PVeD8HnlE9euv/WULF7pYArfBrCzV++5TLS/3l737rurLL/tfFxWpnnCCWyeBPyxlZarbtvlf5+aqDhjg4vF57DG3Pk47reb1rao6dqyr95e/+MteeME1HgK36/z8yvPJyXF1YmNVN2/2l0+ZovrPf1ZexsSJlRP83r2qt93mtkufnTtVn3iicmPl88/dtvDoo/4yj6fyD+gxsITeWPLyVL/7rubxd9/tVvHrr/vLHnxQ9Z57at5IFy9W7dBB9YYb3GuPxyXQc891LfeaPPdc5SQ1ZYpb9oMP+ss2bVJ9//3K0wW2pvbscV/wxET3JVZ1X76XXlKdP99fLy+vcqtr6VKXsBIS/DF++aX7Ir/zTs0xz53rYoyL8yfmggKXqKdN89cLTBqqroWdmOim27TJla1Z41p0Tz5Z8/LKylySA/fDpeq+6L/9reqoUZU/k6pfujVrVIcNU50921+2bp3q+PGVW/wrVx75w5adXXkbKC5WPf98lwB9Dh50LdTvv688bUFB5bh++UvVIUNU583zl11/vdsD8K2L6qxf79bXxRf7y95/362Lc8+tebryctWTT3b1fD+oFRUuUUPtPzYrVqh26aLav7//x8A3v9NPr/y+qn4f8vNdS/vbb/1lX3zhflC/+MJftmVL5dc+Vb8rP/2p2x7LyvxlV1/tGgZVP69AS5e699m2rfuBUnWfJ7ht8MCBmqf93e9cvcce85c9/bTbMz4OltAbSnm5PwEWF7vWVGKi6mefVV//uedUY2JUX3zRvV671iW+yEjVJUv89apuzL4NR9Uluk6dVFu1Ul20yJXl56v+7GfVb8g+Tz3lWm2vvuper1njWiPJyTXvBhYWql51levi8CW03Fy3mbRv71pv1cnJUf35z1XHjPG/hb8846Y7++yaY6yoUM+f/qwlTz/v70H5/ns3XZculbpViovdd/vw9yc7W3XmTP+8tmxx00VF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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_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": 74,
   "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": 75,
   "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()"
   ]
  },
  {
   "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
}
