{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Sensitivity of optimistic NPG primal-dual 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 + optimistic primal-dual method (OGDA version)\\n   (Natural) policy gradient + optimistic primal-dual method (OMWU version)\\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 + optimistic primal-dual method (OGDA version)\n",
    "   (Natural) policy gradient + optimistic primal-dual method (OMWU version)\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: Discrete probability distributions p, q : array-like, dtype=float, shape=n\n",
    "Output: Kullback-Leibler divergence D(p||q)      \n",
    "\"\"\"\n",
    "def kl_div(p, q):\n",
    "    p = np.asarray(p, dtype=np.float)\n",
    "    q = np.asarray(q, dtype=np.float)\n",
    "    \n",
    "    return np.sum(np.where(p != 0, p * np.log(p / q), 0))"
   ]
  },
  {
   "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(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": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Input: theta as an array and \n",
    "Ouput: array of probabilites corresponding to each state: [\\pi_{s_1}(.), ...., \\pi_{s_n}(.)]\n",
    "'''\n",
    "def theta_to_policy_naive(theta,n,m):\n",
    "    prob = []\n",
    "    for i in range(n):\n",
    "        norm = np.sum(theta[m*i:m*(i+1)])\n",
    "        for j in range(m*i,m*(i+1)):\n",
    "            prob.append(theta[j]/norm)\n",
    "            \n",
    "    return np.asarray(prob)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "\"\"\"\n",
    "License: BSD\n",
    "Author: Mathieu Blondel\n",
    "Implements three algorithms for projecting a vector onto the simplex: sort, pivot and bisection.\n",
    "For details and references, see the following paper:\n",
    "Large-scale Multiclass Support Vector Machine Training via Euclidean Projection onto the Simplex\n",
    "Mathieu Blondel, Akinori Fujino, and Naonori Ueda.\n",
    "ICPR 2014.\n",
    "http://www.mblondel.org/publications/mblondel-icpr2014.pdf\n",
    "\"\"\"\n",
    "\n",
    "def projection_simplex_sort(v, z=1):\n",
    "    n_features = v.shape[0]\n",
    "    u = np.sort(v)[::-1]\n",
    "    cssv = np.cumsum(u) - z\n",
    "    ind = np.arange(n_features) + 1\n",
    "    cond = u - cssv / ind > 0\n",
    "    rho = ind[cond][-1]\n",
    "    theta = cssv[cond][-1] / float(rho)\n",
    "    w = np.maximum(v - theta, 0)\n",
    "    return w\n",
    "\n",
    "\n",
    "def projection_simplex_pivot(v, z=1, random_state=None):\n",
    "    rs = np.random.RandomState(random_state)\n",
    "    n_features = len(v)\n",
    "    U = np.arange(n_features)\n",
    "    s = 0\n",
    "    rho = 0\n",
    "    while len(U) > 0:\n",
    "        G = []\n",
    "        L = []\n",
    "        k = U[rs.randint(0, len(U))]\n",
    "        ds = v[k]\n",
    "        for j in U:\n",
    "            if v[j] >= v[k]:\n",
    "                if j != k:\n",
    "                    ds += v[j]\n",
    "                    G.append(j)\n",
    "            elif v[j] < v[k]:\n",
    "                L.append(j)\n",
    "        drho = len(G) + 1\n",
    "        if s + ds - (rho + drho) * v[k] < z:\n",
    "            s += ds\n",
    "            rho += drho\n",
    "            U = L\n",
    "        else:\n",
    "            U = G\n",
    "    theta = (s - z) / float(rho)\n",
    "    return np.maximum(v - theta, 0)\n",
    "\n",
    "\n",
    "def projection_simplex_bisection(v, z=1, tau=0.0001, max_iter=1000):\n",
    "    func = lambda x: np.sum(np.maximum(v - x, 0)) - z\n",
    "    lower = np.min(v) - z / len(v)\n",
    "    upper = np.max(v)\n",
    "\n",
    "    for it in range(max_iter):\n",
    "        midpoint = (upper + lower) / 2.0\n",
    "        value = func(midpoint)\n",
    "\n",
    "        if abs(value) <= tau:\n",
    "            break\n",
    "\n",
    "        if value <= 0:\n",
    "            upper = midpoint\n",
    "        else:\n",
    "            lower = midpoint\n",
    "\n",
    "    return np.maximum(v - midpoint, 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Input: theta as an array and \n",
    "Ouput: array of probabilites corresponding to each state: [\\pi_{s_1}(.), ...., \\pi_{s_n}(.)]\n",
    "'''\n",
    "def project_to_policy(theta,n,m):\n",
    "    prob = []\n",
    "    prob_pers = []\n",
    "    for i in range(n):\n",
    "#         norm = np.sum(np.exp(theta[m*i:m*(i+1)]))\n",
    "        prob_pers = projection_simplex_sort(theta[m*i:m*(i+1)], z=1)\n",
    "        for j in range(m):\n",
    "            prob.append(prob_pers[j])\n",
    "            \n",
    "    return np.asarray(prob)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Get \\Pi_{\\pi}((s) -> (s,a)) in R^{|S| x |S||A|} matrix corresponding to the policy \\pi using the prob vector\n",
    "'''\n",
    "def get_Pi(prob,n,m):\n",
    "    Pi = np.zeros((n,n*m))\n",
    "    for i in range(n):\n",
    "        Pi[i,i*m:(i+1)*m] = prob[i*m:(i+1)*m]\n",
    "    \n",
    "    return Pi"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Input: probability vector, state, action\n",
    "Output: \\nabla_{\\theta} \\pi_{\\theta}(s,a)\n",
    "\n",
    "States go from 0 to n-1 and actons from 0 to m-1\n",
    "'''\n",
    "def grad_state_action(prob,state,action):\n",
    "    grad = np.zeros(n*m)\n",
    "    for j in range(0,m):\n",
    "        if j == action:\n",
    "            grad[m*state + j] = prob[m*state + j]*(1-prob[m*state + j])\n",
    "        else:\n",
    "            grad[m*state + j] = -prob[m*state + action]*prob[m*state + j]\n",
    "            \n",
    "    return grad\n",
    "\n",
    "def grad_state(qvals,prob,state):\n",
    "    grad = np.sum([qvals[state*m + i]*grad_state_action(prob,state,i) for i in range(0,m)],axis=0)\n",
    "    return grad\n",
    "\n",
    "def grad(qvals,prob,d_pi):\n",
    "    grad = np.sum([d_pi[i]*grad_state(qvals,prob,i) for i in range(0,n)],axis=0)\n",
    "    return grad"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Input: probability vector\n",
    "Output: Fisher information matrix\n",
    "        \\nabla_{\\theta} \\pi_{\\theta}(s,a) x {\\nabla_{\\theta} \\pi_{\\theta}(s,a)}^T\n",
    "'''\n",
    "def Fisher_info(prob,d_pi):\n",
    "    qvals_one = np.ones(n*m)\n",
    "    grad = np.sum([d_pi[i]*grad_state(qvals_one,prob,i) for i in range(0,n)],axis=0)\n",
    "    fisher = np.outer(grad,grad)+1e-3*np.identity(n*m)\n",
    "    return fisher"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "The overall reward function \\ell(\\theta)\n",
    "'''\n",
    "def ell(qvals,prob,rho):\n",
    "    V = np.zeros(n)\n",
    "    for i in range(n):\n",
    "        V[i] = np.sum([qvals[i*m + j]*prob[i*m + j] for j in range(m)])\n",
    "    \n",
    "    ell = np.dot(V,rho)\n",
    "    return ell"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "The overall reward advantage function \\ell(\\theta)\n",
    "'''\n",
    "def avals(qvals,prob):\n",
    "    V = np.zeros(n)\n",
    "    for i in range(n):\n",
    "        V[i] = np.sum([qvals[i*m + j]*prob[i*m + j] for j in range(m)])\n",
    "    \n",
    "    A = qvals\n",
    "    for i in range(n):\n",
    "        for j in range(m):\n",
    "            A[i*m + j] = qvals[i*m + j] - V[i]\n",
    "        \n",
    "    return A"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "The projection function\n",
    "Input: a scalar \n",
    "Output: a scalar in the interval [0 C]\n",
    "'''\n",
    "def proj(scalar,gamma):\n",
    "    offset = 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": 15,
   "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": 16,
   "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": 17,
   "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": 18,
   "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": 19,
   "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",
    "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": [
    "## Optimistic NPG primal dual method (OGDA version)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "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, optimal_policy_LP):\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",
    "    policy_distance = []\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",
    "            policy_norm_squared = np.inner(theta_h - optimal_policy_LP, theta_h - optimal_policy_LP) \n",
    "            policy_distance.append(policy_norm_squared)\n",
    "                \n",
    "    return reward_value, utility_value, policy_distance"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Optimistic NPG primal dual method (OMWU 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 OMWU_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates, optimal_policy_LP):\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",
    "    policy_distance = []\n",
    "    for k in range(total_iterates):\n",
    "    \n",
    "        # policy evaluation of theta_h\n",
    "        prob_h = theta_to_policy(theta_h,n,m)\n",
    "    \n",
    "        Pi_h = get_Pi(prob_h,n,m)\n",
    "        mat = np.identity(n*m) - gamma*np.matmul(prob_transition,Pi_h)\n",
    "        qvals_reward_h = np.dot(np.linalg.inv(mat),reward)\n",
    "        qvals_utility_h = np.dot(np.linalg.inv(mat),utility)\n",
    "    \n",
    "        # optimistic step for (theta, dual)\n",
    "        prob = theta_to_policy(theta,n,m)\n",
    "\n",
    "        Pi = get_Pi(prob,n,m)\n",
    "        mat = np.identity(n*m) - gamma*np.matmul(prob_transition,Pi)\n",
    "        qvals_reward = np.dot(np.linalg.inv(mat),reward)\n",
    "        qvals_utility = np.dot(np.linalg.inv(mat),utility)\n",
    "\n",
    "        P_theta = np.matmul(Pi,prob_transition)\n",
    "        d_pi = (1-gamma)*np.dot(np.transpose((np.linalg.inv(np.identity(n) - gamma*P_theta))),rho)\n",
    "    \n",
    "        # natural gradient \n",
    "        qvals = qvals_reward + dual*qvals_utility\n",
    "    \n",
    "        # natural gradient ascent\n",
    "        theta = theta_h + stepsize*avals(qvals,prob)\n",
    "    \n",
    "        # dual descent \n",
    "        violation_gradient = ell(qvals_utility,prob,rho) \n",
    "        dual = dual_h - stepsize*violation_gradient\n",
    "        dual = proj(dual,gamma)\n",
    "\n",
    "        # optimistic step for (theta_h, dual_h)\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_h = theta_h + stepsize*avals(qvals,prob)\n",
    "    \n",
    "        # dual desceent \n",
    "        violation_gradient = ell(qvals_utility,prob,rho)\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_h,prob,rho)\n",
    "            avg_utility = ell(qvals_utility_h,prob,rho)\n",
    "            reward_value.append(avg_reward)\n",
    "            utility_value.append(avg_utility)\n",
    "            \n",
    "            policy_kl_distance = kl_div(optimal_policy_LP,prob)\n",
    "            policy_distance.append(policy_kl_distance)\n",
    "                \n",
    "    return reward_value, utility_value, policy_distance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "# call optimistic NPG primal dual method\n",
    "\n",
    "total_iterates = 2000\n",
    "\n",
    "stepsize = 0.2\n",
    "\n",
    "reward_value_opt_3, utility_value_opt_3, policy_distance_opt_3 = Opt_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates, optimal_policy_LP)\n",
    "\n",
    "reward_value_OMWU_3, utility_value_OMWU_3, policy_distance_OMWU_3 = OMWU_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates, optimal_policy_LP)\n",
    "\n",
    "stepsize = 0.1\n",
    "\n",
    "reward_value_opt_2, utility_value_opt_2, policy_distance_opt_2 = Opt_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates, optimal_policy_LP)\n",
    "\n",
    "reward_value_OMWU_2, utility_value_OMWU_2, policy_distance_OMWU_2 = OMWU_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates, optimal_policy_LP)\n",
    "\n",
    "stepsize = 0.05\n",
    "\n",
    "reward_value_opt_1, utility_value_opt_1, policy_distance_opt_1 = Opt_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates, optimal_policy_LP)\n",
    "\n",
    "reward_value_OMWU_1, utility_value_OMWU_1, policy_distance_OMWU_1 = OMWU_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates, optimal_policy_LP)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# plot reward values\n",
    "\n",
    "f_value = plt.figure\n",
    "\n",
    "num_every = 20\n",
    "num_grads = np.arange(0, total_iterates, num_every)\n",
    "\n",
    "# case #1\n",
    "reward_value_opt_1 = np.array(reward_value_opt_1)\n",
    "reward_value_OMWU_1 = np.array(reward_value_OMWU_1)\n",
    "\n",
    "# absolute difference to the optimal reward value from LP\n",
    "error_reward_value_opt_1 = np.absolute(reward_value_opt_1 + res.fun)\n",
    "error_reward_value_OMWU_1 = np.absolute(reward_value_OMWU_1 + res.fun)\n",
    "\n",
    "# case #2\n",
    "reward_value_opt_2 = np.array(reward_value_opt_2)\n",
    "reward_value_OMWU_2 = np.array(reward_value_OMWU_2)\n",
    "\n",
    "# absolute difference to the optimal reward value from LP\n",
    "error_reward_value_opt_2 = np.absolute(reward_value_opt_2 + res.fun)\n",
    "error_reward_value_OMWU_2 = np.absolute(reward_value_OMWU_2 + res.fun)\n",
    "\n",
    "# case #3\n",
    "reward_value_opt_3 = np.array(reward_value_opt_3)\n",
    "reward_value_OMWU_3 = np.array(reward_value_OMWU_3)\n",
    "\n",
    "# absolute difference to the optimal reward value from LP\n",
    "error_reward_value_opt_3 = np.absolute(reward_value_opt_3 + res.fun)\n",
    "error_reward_value_OMWU_3 = np.absolute(reward_value_OMWU_3 + res.fun)\n",
    "\n",
    "# convert y-axis to Logarithmic scale\n",
    "plt.yscale(\"log\")\n",
    "\n",
    "plt.plot(num_grads,error_reward_value_OMWU_1[::num_every], \"k:\",linewidth=2)\n",
    "plt.plot(num_grads,error_reward_value_OMWU_2[::num_every], \"b-.\",linewidth=2)\n",
    "plt.plot(num_grads,error_reward_value_OMWU_3[::num_every], \"r-\",linewidth=2)\n",
    "# plt.plot(num_grads,error_reward_value_opt_1[::num_every], \"k:\",linewidth=2)\n",
    "# plt.plot(num_grads,error_reward_value_opt_2[::num_every], \"b-.\",linewidth=2)\n",
    "# plt.plot(num_grads,error_reward_value_opt_3[::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_OMWU.png',dpi=300)\n",
    "# get_f_value.savefig('NPG_primal_dual_sensitivity_reward.png',dpi=300)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "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",
    "# case #1\n",
    "utility_value_opt_1 = np.array(utility_value_opt_1)\n",
    "utility_value_OMWU_1 = np.array(utility_value_OMWU_1)\n",
    "\n",
    "# absolute difference to the optimal utility value 0\n",
    "error_utility_value_opt_1 = np.absolute(utility_value_opt_1)\n",
    "error_utility_value_OMWU_1 = np.absolute(utility_value_OMWU_1)\n",
    "\n",
    "# case #2\n",
    "utility_value_opt_2 = np.array(utility_value_opt_2)\n",
    "utility_value_OMWU_2 = np.array(utility_value_OMWU_2)\n",
    "\n",
    "# absolute difference to the optimal utility value 0\n",
    "error_utility_value_opt_2 = np.absolute(utility_value_opt_2)\n",
    "error_utility_value_OMWU_2 = np.absolute(utility_value_OMWU_2)\n",
    "\n",
    "# case #3\n",
    "utility_value_opt_3 = np.array(utility_value_opt_3)\n",
    "utility_value_OMWU_3 = np.array(utility_value_OMWU_3)\n",
    "\n",
    "# absolute difference to the optimal utility value 0\n",
    "error_utility_value_opt_3 = np.absolute(utility_value_opt_3)\n",
    "error_utility_value_OMWU_3 = np.absolute(utility_value_OMWU_3)\n",
    "\n",
    "# convert y-axis to Logarithmic scale\n",
    "plt.yscale(\"log\")\n",
    "\n",
    "plt.plot(num_grads,error_utility_value_OMWU_1[::num_every], \"k:\", linewidth=2)\n",
    "plt.plot(num_grads,error_utility_value_OMWU_2[::num_every], \"b-.\", linewidth=2)\n",
    "plt.plot(num_grads,error_utility_value_OMWU_3[::num_every], \"r-\", linewidth=2)\n",
    "# plt.plot(num_grads,error_utility_value_opt_1[::num_every], \"k:\", linewidth=2)\n",
    "# plt.plot(num_grads,error_utility_value_opt_2[::num_every], \"b-.\", linewidth=2)\n",
    "# plt.plot(num_grads,error_utility_value_opt_3[::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_OMWU.png',dpi=300)\n",
    "# get_g_value.savefig('NPG_primal_dual_sensitivity_utility.png',dpi=300)\n",
    "\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "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_opt_1 = np.array(policy_distance_opt_1)\n",
    "policy_distance_OMWU_1 = np.array(policy_distance_OMWU_1)\n",
    "\n",
    "# case #2\n",
    "policy_distance_opt_2 = np.array(policy_distance_opt_2)\n",
    "policy_distance_OMWU_2 = np.array(policy_distance_OMWU_2)\n",
    "\n",
    "# case #3\n",
    "policy_distance_opt_3 = np.array(policy_distance_opt_3)\n",
    "policy_distance_OMWU_3 = np.array(policy_distance_OMWU_3)\n",
    "\n",
    "# convert y-axis to Logarithmic scale\n",
    "plt.yscale(\"log\")\n",
    "\n",
    "plt.plot(num_grads,policy_distance_OMWU_1[::num_every], \"k:\", linewidth=2)\n",
    "plt.plot(num_grads,policy_distance_OMWU_2[::num_every], \"b-.\", linewidth=2)\n",
    "plt.plot(num_grads,policy_distance_OMWU_3[::num_every], \"r-\", linewidth=2)\n",
    "# plt.plot(num_grads,policy_distance_opt_1[::num_every], \"k:\", linewidth=2)\n",
    "# plt.plot(num_grads,policy_distance_opt_2[::num_every], \"b-.\", linewidth=2)\n",
    "# plt.plot(num_grads,policy_distance_opt_3[::num_every], \"r-\", 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_OMWU.png',dpi=300)\n",
    "# p_distance.savefig('NPG_primal_dual_squared_policy_distance.png',dpi=300)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "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
}
