{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Sensitivity of EG NPG primal-dual for finite constrained MDPs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "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": 3,
     "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",
    "   (Natural) policy gradient + extragradient primal-dual method (OGDA version)\n",
    "'''"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "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": 5,
   "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": 6,
   "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": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "Input: theta as an array and \n",
    "Ouput: array of probabilites corresponding to each state: [\\pi_{s_1}(.), ...., \\pi_{s_n}(.)]\n",
    "'''\n",
    "def 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": 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 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": 9,
   "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": 10,
   "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": 11,
   "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": 12,
   "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": 13,
   "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": 14,
   "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": 15,
   "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": 16,
   "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": 17,
   "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": 18,
   "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": 19,
   "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": 20,
   "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": 21,
   "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": 22,
   "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": [
    "## EG NPG primal dual method "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "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 EG_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_h,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_h,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": 24,
   "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": 33,
   "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_EG_3, utility_value_EG_3, policy_distance_EG_3 = EG_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_EG_2, utility_value_EG_2, policy_distance_EG_2 = EG_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_EG_1, utility_value_EG_1, policy_distance_EG_1 = EG_NPG_primal_dual(n, m, prob_transition, gamma, reward, utility, stepsize, total_iterates, optimal_policy_LP)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAX0AAAD4CAYAAAAAczaOAAAABHNCSVQICAgIfAhkiAAAAAlwSFlzAAALEgAACxIB0t1+/AAAADh0RVh0U29mdHdhcmUAbWF0cGxvdGxpYiB2ZXJzaW9uMy4yLjEsIGh0dHA6Ly9tYXRwbG90bGliLm9yZy+j8jraAAAgAElEQVR4nOydd3hUxdfHvzedhBJI6IQSem9RRIqAIF2JAhakRhGRJuqrCAIiTZAmIr36AxEEpQuCSO+9CAkEkgCBhPSe7O73/WOy2ZTdzW6ymyxhPs9zn83eO+Xcm+TM3HPOnFFIQiKRSCTPB3aFLYBEIpFICg6p9CUSieQ5Qip9iUQieY6QSl8ikUieI6TSl0gkkucIh8IWwBienp6sXr16YYshkUgkzxQXLlx4SrKsvms2rfSrV6+O8+fPF7YYEolE8kyhKEqQoWs2ad5RFKW3oigrYmJiClsUiUQiKVLYpNInuYvk8FKlShW2KBKJRFKksEmlL5FIJBLrIJW+RCKRPEdIpS+RSCTPETap9KUjVyKRSKyDTSp96ciVSCQS62CTSt8iqNWATBstkUgkWSiSSv9BsWqAgwNObL9Y2KJIJBKJTVEklX5Sqrity//eLGRJJBKJxLawSaWfX0duqqs7AKCco4clxZJIJJJnHptU+vl15CplygMAHBMUS4olkUgkzzw2qfTzC0uJmX7y4+hClkQikUhsiyKp9O3KCKUffPU/pKWlFbI0EolEYjsUSaXv6CmUftT9YAQEBBSyNBKJRGI7FEml71JBKP2KLi5wdHQsZGkkEonEdiiSSt+tSmkAQBlFg9q1axeyNBKJRGI72KTSz2/IZgkvMdN3ToqGWm1JySQSieTZxiaVfn5DNh3SbfqlEI0jR65bUjSJRCJ5prFJpZ9v3IXSd0c0XnttEFJSUgpZIIlEIrENirTSL41IVK/eHOHh4YUskEQikdgGDoUtgFUoLRy5tTxjcOfO6kIWRiKRSGyHojnTT/cFKNHRMr2yRCKRZKJoKn1nZ6BYMUClAhIToZYhPBKJRAKgqCp9ACnFhF2/QaU1aNKkSSFLI5FIJLZBkVX6acWF0ldi3RAQECAjeCQSiQRFWOm7VBTO3I2LXRATEwNnZ+dClkgikUgKnwKL3lEUxQ3AzwBSAfxLcqM1+3PwEDP9ZtVLCvu+RCKRSPI301cUZY2iKGGKolzPdr6boii3FUW5oyjKV+mn3wTwO8kPAbyen35NIj1WH9Eyp75EIpFoya95Zx2AbplPKIpiD2AJgO4AGgB4V1GUBgCqAAhJL2b9cJp0pb9kRhj69x+DIUOGWL1LiUQisXXyZd4heVRRlOrZTr8I4A7JQABQFGUzgDcAPIBQ/JdhZLBRFGU4gOEAULlyZQQFBeVJNndFQSkAobcSsfXWSbi6/ofJkyfD3t4+T+1JJBJJUcAaNv3K0M3oAaHsWwH4EcBPiqL0BLDLUGWSKwCsAAAfHx9Wq1Ytb1Kk1yuNKIwdOxvvvlsC1atXh51dkfVdSyQSSa4UmCOXZAKAoaaUVRSlN4DetWrVynuH6akY3BGNUo06o1WrvDclkUgkRQVrTHsfAvDK9L1K+jmTyW9qZQBZMm0+fpz3ZiQSiaQoYQ2lfw5AbUVRaiiK4gTgHQA7zWkgv5uoAMii9O/fT8bixYvx/fff5709iUQiKQLkN2TzVwCnANRVFOWBoih+JFUARgHYD+A/AFtI3jCnXUvP9ENDiTFjxmDWrFmgTMAmkUieY/IbvfOugfN7AezNa7sWselnUvqxsS4YPXo06tati7S0NDg5OeW9XYlEInmGUWx55uvj48Pz58/nrXJEBODpiUiURqtakQgIsKxsEolEYqsoinKBpI++a0U3fjHdNFQKMXgSqilkYSQSicQ2sEmlbxFHroMDWLw47KEBEuIRGZmK06dPY9cug0sEJBKJpMhjk0rfIo5cAEomu/61a2Fo3bo1/Pz8pDNXIpE8txTNPXK1uLsDDx7AHdEAGqNNmzaoVasWUlJS4OLiUtjSSSQSSYFTtJV++qrcE7uiUOIVBcePHy9kgSQSiaRwsUnzjkVs+kBG2GYJtUyvLJFIJICNKn1L2fT15dRPS0vDjh078py9UyKRSJ5lbFLpW4x0pb9iTjR69RKnxo0bhz59+mDlypWFKJhEIpEUDs+F0n94MxoHDwIaDfD222+jXr168Pb2LmThJBKJpOCxSUeuRdIwABmO3MGvR6PvdHGqXbt2uHnzJhRFyV/bEolE8gxikzN9S9v0vUtHoXFjwM4OUBRFKnyJRPLcYpNK32IY2Rw9JiYGy5cvR3h4eAELJZFIJIXHc6H0Y0OiMWIEsGyZ7tLQoUMxYsQI/Pzzz4UknEQikRQ8z4XS10RGY/lyYPdu3aWxY8fixRdfRJs2bQpJOIlEIil4ngtHrmuqMO9kTq/8yiuv4PTp09K+L5FInitscqZvaUeuY3wUACAwEFCpdJelwpdIJM8bNqn0LUbJkoCiQImNRdXKaqhUQPaFuBEREZgwYQJmz55dODJKJBJJAVK0lb6dnVD8AJp5xwIA/P2zFrl16xZmz56NmTNnIi4urqAllEgkkgKlaCt9IMPE06hKTrs+ALRp0waTJk3CwYMHUaJEiYKWTiKRSAoUm3TkWhR3dyAoCPXKRwKooXev3O+++67AxZJIJJLCoOjP9CtUAADULvEEQM6ZvkQikTxP2KTSt1g+fQCoWBEAUM3xEQDDSv/cuXPw8/PDL7/8kv8+JRKJxEaxSaVvsZBNAKhUCQBQVh0KOzvg/n0gNTVnsZs3b2LNmjXYsGFD/vuUSCQSG8Umlb5FSZ/pOzx5hGrVRHrlwMCcxXr37o1p06Zh4cKFBSygRCKRFBxFX+mnz/QRGop69QAHByAkJGexMmXK4JtvvkHDhg0LVj6JRCIpQIp+9E76TB+PHmHBL0C5chnZGSQSieS54/mZ6T96hLp1c1f4v/zyC/r27YuEhATryyaRSCQFTNFX+ukhm3j8WBj0AajVQFSU/uJLlizBtm3bcOzYsQISUCKRSAqOoq/0nZ0BDw+h6cPDsXevsPh8+qn+4gMHDsTcuXPRtWvXgpVTIpFICoCib9MHhIknIgIIDYW3d3mEhwM3bwIkkD3R5ieffFI4MkokEkkBUPRn+kAWZ269ekLhnzmTU+FnJzo6Ghs3brS+fBKJRFJAFJjSVxTFW1GU1Yqi/F5QfWaQKWwTAOrXz13hp6SkoFWrVnj//fcRnWmP3cjISJC0lqQSiURiVUxS+oqirFEUJUxRlOvZzndTFOW2oih3FEX5ylgbJANJ+uVH2DyTaaafmYcPM3y7OXB2dsbw4cMBIEPJnz17Fo0bN8a8efOsJqpEIpFYE1Nn+usAdMt8QlEUewBLAHQH0ADAu4qiNFAUpbGiKLuzHeUsKrW5ZArb1NK3L1ClCvDXX4arffbZZ/D390fx4sUBAA8ePMCjR4+we/duqNVqa0oskUgkVsEkRy7Jo4qiVM92+kUAd0gGAoCiKJsBvEFyFoBeeRVIUZThAIYDQOXKlRGUfaurPODq4ICyABLv3kV4enuVK5cC4I6+fTVYtSocbdok663r5OSER+mDRcuWLbF69Wq0a9cODx48yLdcEolEUtDkJ3qnMoDMCQ0eAGhlqLCiKB4AZgBorijKhPTBIQckVwBYAQA+Pj6sVq1aPkRMp2lTAIBrdDS07c2fD8TGAuvW2WHYsPL44w+ge/fcmxo2bFj+5ZFIJJJCosAcuSQjSI4gWdOQwtdi0dTKQA5HLgDY2wOrVwMffwykpABvvAGcPGl6k3fv3sWsWbPw9OlTy8gokUgkBUB+lP5DAF6ZvldJP5dvLJpaGdC7KhcQW+guWQJ89BGQlgbMMjoUZWXUqFH4+uuvsW3bNsvIKJFIJAVAfpT+OQC1FUWpoSiKE4B3AOy0hFAWn+lrV+WqVEC2mbmiAN99Bzg5AXv26E+7rI8hQ4ZgwIABaNy4sWVklEgkkgLA1JDNXwGcAlBXUZQHiqL4kVQBGAVgP4D/AGwhecMSQll8pg8YDNsEgLJlgXfeESt0f/7ZtObefvtt/O9//8PLL79sORklEonEypik9Em+S7IiSUeSVUiuTj+/l2SddDv9DEsJZfGZPqDXrp+Z0aPF5+rVgEywKZFIiio2mYahoGf6AODjA7z0EhAdDWzaZFqTJHH69GksWrTIQkJKJBKJdbFJpW8V9CzQys6oUeJz8WJh6smN+Ph4dOzYEePGjZNx+xKJ5JnAJpW+Vcw72pm+AfMOAPTrB5Qvn5GFOVdKlCgBPz8/jBs3Dkq2ZD6rV6/Gf//9l/F9+fLlCDXSt0QikRQEii0nD/Px8eH58+ct09i2bSL3whtvAH/+abBYUBBQtWruCdmMcf36dTRv3hz29va4d+8e7ty5g/bt28Pb2xu3bt2Co6Nj3huXSCSSXFAU5QJJH33Xno98+kCujlwt+VkAHBISAldXV1SqVAl+fn5wcHBAxYoVkZKSgi5duqBNmzZISUmRSl8ikRQaz89M//59oEYNkWUtJCTX4kFBwNKlwIwZYvVubgQEBKBz586oUKECDh48iBIlSoBkDrOPRCKRWJtnbqavKEpvAL1r1apluUa1Nn3tqlw7w+4MEujaFbh9G6hbFxg6NPfmXV1dYWdnBwcHh4wMnFLhSyQSW8MmHblWCdl0dgbKlNG7Kjc7igJMngy89x7QoYOI2//yS6BXL+Dff/XXqVy5Mv79918cOHAA7u7uessEBwdj69atchMWiURSaNjkTN9qVKoEREaKsM1yxlP8v/eeOAARzbN8ORATI8aMDh301zGWEZQkWrRogYiICNy9exfe3t55vAmJRCLJOzY507caXun54W7fNquavT0wbZr4+fLlvHWtKAp69uyJnj17IjExMW+NSCQSST6xSaVvlTh9QDdFN7ZdlgFGjwZKlgSePBFugbywfv167N69G40aNcpbAxKJRJJPbFLpW8WmDwA9eojPffsMb45rAEUBmjUTP+d1ti+RSCSFjU0qfavRsKEw8Tx5Aly6ZHZ1Syh9krhz5w4SZFY3iURSCDxfSl9Rss72zcQSSr9Pnz6oXbs2Dh8+bFJ5GekjkUgsyfOl9AGd0t+71+yqllD6DRo0QNmyZREZGWm03OLFi+Hm5gY7O7uMuH+JRCLJLza5IjfT4qwPAwICLNt4fLzYRSstDQgLAzw9Ta6akgIULy5COOPiADc387tPTk6Gs7Nzrgu3wsPD8f3332PevHlISEiAq6ur+Z1JJJLnEmMrcm1ypm81Ry4gtHb79mLZ7YEDZlV1dgYaNBBVr13LW/cuLi4mrdQtW7YsZs2aBbVaDVdXV4SFhWHdunVy1i+RSPKFTSp9q2NBE09YGNC0KTBhgnntaDSaXOP1HR0dYZeeLqJdu3YYOnQoTp48aa7IEolEksHzrfT/+kvYaswgu9KfPx+4ehWYN08MAKawatUqeHp6Yu7cuTmuJSUloWPHjli4cCE0mcJK+/bti+7du8PB4flaRC2RSCzL86lB6tQBvL2BwEDg3DmxT6KJvPmmmNk3aybSMixdKs6npQHr1wNffJF7G56enoiKisKNGzn3kd+zZw/+/fdfxMfHY9y4cRnnZ8yw2BbEEonkOeb5nOkrisieBojNVcygWjWgUyeRu23ZMiA2FihbVlxbudK0bRa7dOmCwMBAbNmyJce1nj17YuvWrZgyZYpZckkkEokpPJ9KHwD69xefW7aYpqn1cPOm+Fy3DqhcGQgIAI4cyb2em5sbatSoofdasWLF0LdvX/TSDkrZ8Pf3x6lTp/Ikr0Qikdik0rda7p3MtG4tNlQJDgZOnzar6pYtwFtviXHj2jWge3fAz09cW7HCPDHMCZn9+++/UbduXYwZM8a8TiQSiSQdm1T6Vg3Z1GJnp5vt//abWVVv3AC2bweOHgUaNRLWIj8/8bltW67p+gEAgYGB6Nq1K7p16wZARPP06dMHEyZMQEREhN46bdu2RZUqVVC/fn2oVCqTZM0+qGjMzDkkkUiKGCRt9mjZsiWtypkzJEBWrEiqVCZXu3qV3LCBvHs36/nu3UVz8+bl3kZ0dDTt7Ozo6OjIhIQE/vPPPwTASpUqMT4+3mA9jUZjspwkOXz4cHbp0oXnz5/nxYsX2ahRI16+fNmsNiQSybMFgPM0oFdtcqZfYLzwgtg3NzQUOH7c5GqNGwMDB4oAoMwMHy5eIPRNpv/4I6vPuFSpUtizZw+Cg4Ph6uqKjh074siRI1ixYgXcjCz1NWcLRpLYuXMn/v77b7i4uOC7777D9evXMWzYMJPbkEgkRQubTMOgxaIboxviq6+A778HPv4Y+PnnfDWlUgFXrgAtW2Y9v3Mn8MYb4ue7d3MOFuaiVqsRHx8PU8xfT58+xT///IN+/fohLi4OV65cQcuWLWVaB4mkCPPMpWEoUN5+W3z+/rvQ2vnAwSGnwgdE/L6W5cuzXnv8+DECAwNN7uPYsWNwdnbG66+/blJ5T09P9O/fH4qioGTJkmjXrp1U+BLJc4xU+s2aAbVrA+Hhhnc9zwN79wKvvCI2Vf/tN2DkSHF+zRqRuE1Lhw4dULNmTaxatcqkdsuWLQu1Wo2kpCSLySqRSJ4fpNJXFKBfP/Hzrl0WaVKtFrl4jh4VC7gcHICffhLjy9On4qVCS+PGjeHp6YlOnTqZ1HadOnWQnJyMs2fPGi13//59vPDCC5g/f36W85cuXcKwYcPw/fffm31fEonk2UcqfQBID5s0N+umIeztxYKtuXMBbSYFRRFuA0CXugEAtmzZgpCQEHibaOi3s7ODs7NzruV27dqF8+fP43S2NQiRkZFYu3YtduzYYVJ/+iCJOXPm4Mcff5QhoBLJM4Z05AIicY6Hh0iSHxQEVK1qlW7i44FKlUQ3V6+KKCBrkZiYiH/++QceHh5o3bp1xvno6Ghs2rQJLVu2RKtWrfLcvqurK5KSkrB161b07dvXEiJLJBILYTOOXEVR+iiKslJRlN8URXmtIPs2iqOjSKgDAPv3W62b4sVFqCeQdbZvLtOnT0ebNm1w7Ngxg2VcXV3Rq1evLAofANzd3TFy5EizFX5aWlqWBWE9e/bE3r174evra57wEomkUDFZ6SuKskZRlDBFUa5nO99NUZTbiqLcURTlK2NtkPyT5IcARgB4O28iW4muXcWnFZU+oDPxrF0LfP21WCKg5do1ke05N/z9/XHy5ElYfFcxI+zduxeenp6YPHkyAGDr1q3o3r077O3tC0wGiUSSf8yZ6a8D0C3zCUVR7AEsAdAdQAMA7yqK0kBRlMaKouzOdpTLVHVSej3bQav0Dx7Md+imMRo1AgYPBpKTgVmzdMk+AbE+rHv33F0Ln332GY4cOWIwbPPo0aMYP348du/erff6/fv3sWTJEvz5558my33+/HnExMTA0dHR5Dq5oVarUSDmO4lEkoHJ+fRJHlUUpXq20y8CuEMyEAAURdkM4A2SswDkSBOpiOWkswHsI3lRXz+KogwHMBwAKleujKCgIFNFzB/29qhUvToc799H6M6dSNUXcG8hvv0WeP11Z6xaVQKvvpqEoKAEAMCNG+4ASuHixQjUrRtvsL67uzvc3d2RkJCAhISEHNf37duHBQsWID4+Ho31OA727duHUaNGoWPHjmjevLlJMn/wwQfo2rUrnJ2dM34n//33H1auXInq1aubnQQuLS0No0aNwuHDh7F+/focZiiJRGId8ruJSmUAIZm+PwBgzFg8GkBnAKUURalFcln2AiRXAFgBCEdutWrV8imiGfTsCSxZgopXr4rdUqxItWraLtwAiM3ZZ80Si4Pd3DwAeOS57TfffBNlypSBj48P9D2/Hj16YMiQIWjfvr3e64Zlzlo2ODgY27dvR61atfDDDz+YnSKiVq1aOHPmDCpXrgyVSoV79+6hc+fOJrchkUjMx6zonfSZ/m6SjdK/9wXQjeQH6d8HAmhFclS+hFKU3gB616pV68OCtFtn5Et46SXAhnPWR0VFYd26dSCJ8ePHW70/knoVukqlwqJFi9CjRw/Uq1fPLKUPCPNOcHAwoqOj0aJFC1SpUgX379+XfgKJJJ9YM3rnIQCvTN+rpJ/LFyyI1Mr66NhRrKQ6exaIiirYvjORmAhs3qw/cZu4nojx48djzpw5BSJP//790aVLF1y/nsWHDwcHB3z22WeoX7++2QofAOzt7VGjRg00a9YMTZo0QadOnRAbG2spsSUSiR7ya945B6C2oig1IJT9OwDey69QmWb6+W3KPEqUANq0EdtfzZkDTJwo4iwLmJdfFonbPDyALl1yXi9fvjxGjRoFLy8vvbPwo0ePQlEUtGjRwmDGzqSkJFy7dg0uLi5o0qSJQVnS0tKwf/9+xMXFoXTp0vm6Ly1Tp06Fq6sr/Pz84OHhAUVRcOnSJdjZ5T+CODg4GDdu3ED37t0tIKlEUgQxlHM5+wHgVwChANIgbPd+6ed7APAHcBfARFPbM+Wwej59fSxbJpLiA2SpUuS4cWRAQIGKMG2a6L5v37zVr1u3LgHwxo0bBsusWLGCADhgwIBc2wsPD+fOnTsNXl+3bh179+7N8PDwXNtSq9UsXrw4AfDJkye5ljeV5ORkBgQE0NHRke7u7oyKirJY2xLJswYskU+f5LskK5J0JFmF5Or083tJ1iFZk+QMC45HhcPw4SJDWps2QEwMsHChSMjWvTuwZ49hm4sFGTZM5OX/809gyRLg+nXD3T5+nHOL3xdeeAGtW7dGuXLl9FcC0KJFCzRq1MgkR66npyd69+5t8PrGjRuxa9cu/PPPP7m2pVKpsHDhQnz55Zc55COJU6dO4eDBg7m2k5mLFy/Cy8sL+/fvR7t27dC9e3eZkE4iMYSh0aAwDwC9AayoVauWRUc/s7lwgRw6lHRx0c3+v/qqQLr29dV1qd3c699/ddefPHnCLVsu0dVVzZkzC0Qkg+zfv58rVqzgnTt38tXO7t27CYAvv/yyWfW++OILAuBnn33G1NTUfMkgkRQFYGSmX+gK3thRKOYdfTx9Ss6aJR6XqytZAKaDqChhaXrvPbJyZdF1yZKkdqfDkSNHEijPqlVDCZAWtJRk4b333uOIESOYlJRknQ4y8fDhQ7Zt25Yff/yxWfU0Gg337NnDsLAwK0kmkTxbPHNK32Zm+tl59VXxyObPL9BuVSqyXz/RdYUK5H//kYsXL2bLli3Zo8dV7t2rK6tWq6lWq01u21j55ORkAqC9vb1ZbRpj69atPHToEBMTEy3SniGuXbvGAQMGMCIiwqr9SCS2yDOn9LWHzcz0tezYIR6Zt7dZG6lbguRksmNHZph7Vq3SX+7kyZO0s7Njly5dcm2zS5cudHBw4NmzZ/VeT0lJ4datW7l69Wqj7URHR3P37t3cm3n00YNGo2HZsmUJgIGBgbnKlx9at25NALnKJJEURYwpfZlP3xx69gSqVwcCA8XWWAWIs7PYXL1pU/F948aczt1bt4DIyChoNJqMHDkqlcjoeehQzjbt7OygUqkQERGht08nJyf07ds3143UAwMD0atXL3z55ZdGy6WmpuKtt97Cq6++iurVqxssl5KSgtTUVKNtAWKRmqenJ9q3b5/j2rBhw/D111+bteJYInkuMDQa2MJhczN9kvzhBzHV7ty5ULqPiCDXrydjY8V3rdnlq69IRSE3bSJTU1MZm15g5kzdy0lKSta2oqKimJycnG+Znjx5wm7dunHs2LH5bqtfv34EwF27duVa9sqVKwTABg0a5LvfzKjVam7fvl2ahiTPLHjWZvqKovRWFGVFTExMYYuSk2HDAFdXkY3z5s0C775MGWDQIMDePhFeXl4oU6YMSMLbWxh+RowAAgMdUaJECdy8CUydKuoNGAA4OWVty93d3eguXNeuXcMvv/ySYyVudsqVK4d9+/Zh4cKF+bw7oESJErC3t0eUCSuiGzVqhMePH5uVLdQUgoKCsHz5crRr1w7h4eEWbVsiKXQMjQa2cNjkTJ8kR4wQ0+eOHckHDwpFBI1GQzc3NwJgdHQ0NRqxmAsg69UT0T+tWonvfn66eioVuW2baQFIM2bMIAB++eWXFpH55s2bjImJMVomISHBIk7j5ORk3rhxg1euXDG7rlqtZv/+/akoCnfs2JFvWSSSggbSkWth/P1F/CRAlihBLlpU4I5dkgwJCclinomLI8uXDyNAliuXRECEe0ZH6+q8/74Q+4cfhNO3b9++nDVrlt72t2/fznfffZdbt27NVRaNRsPIyEijoZ01atQgAN66dcv0m8wjJ06cIAC+8MILeaofGhrK3bt3W1gqiaRgMKb0bdK8Y/PUri22uXrjDbHh7dixQL9+OZfGWpkqVapkMc8ULw40aTIFQBTCwlwAACtWAJnz1r39NlCzJlC+PPD06VP8/vvvOHr0aMZ1Umwedvgw4Ovri02bNpm0B26vXr1QpkwZg6tyU1JS4OHhAXd3d9SsWTNvN5yNRYsW4eOPP8alS5dyXKtSpQrq1KmDGjVqmNXmX3/9hTt37qBChQro2bOnReSUSGwKQ6NBYR6w1Th9ffz5p8jRA5D/+19hS8OLFy/y66+PsUQJNceMyXldrda9lDx69IibN2/muXPnqFaT27eTzZqJW7GzI82ZkA8aNIglSpTgli1bjJZT5fJGdO/ePfbu3Zvvv/9+rn127tzZomGZqampLFeuHAHw2rVrJMmYmBjOmTOHEydOtEgfEklBAGnesTJr1ohH6eFhvaWxeti1axf79evHdevW5biWPVLHGLdvky+8wIw1AHZ24nPgwDgmJCSY1EZaWprpHRohODiYAFihQoVcyx48eJA//vgjHz58aJG+nz59ykGDBtHHx4cajYYkGRcXRwB0cnKy2D1KJNZGKn1ro9GIEE6AfPvtAuv2559/JgAOGzbM7LoPHpATJohQTzc3IXqlSuTixeT589oBIJ5AaZOyZ1qK1NRU/vHHHwYXjOUFrQI3leyO5EmTJnHp0qUFkopCIrEEUukXBIGBIi8PIEw+BUBAQAB/+eWXjBTKGo2G8+fP54YNG3JVdF99pZvZA2T//mlZHL6vvaamosTSxaWnRaJppk6dSm9vb65du6Ic0h4AACAASURBVDbfbZmKn58fS5UqxT179hRYnxKJLWBM6UtHrqWoUUNscgsAuaxMtRS1atXC+++/jwYNGgAAYmJiMH78eHzyySe57mT1ySdizxhFSQAwGLNmhWRx+C5bZofo6BJITNxl0uYmN2/eRKdOnTBkyBC91x88eIDAwECkpKSYentGefLkCTZs2IAzZ84YLKNSqRATE4PQ0NBc2wsICMCZM2dMWh8gkTzTGBoNCvPAs+TIzUxaGlm8uJg6P3qU9dqHH5J5MMOYQ1RUFMeOHWvyytigIHLMmGkcNWqU2Xbx+Hhdxk9SxOADYJ06dTLOaTRkcLBwHkdHRzMgIMCkVa4HDx7krFmzePXqVYNl9u3bRwDsbGRl9JMnT/j06VOTzDvjx48nAM6ePTvHtbi4OB4+fJh///13ru1IJLYApHmnAOnSRTzWzFEsd+/q7Cjx8Rbt7ujRo5wzZ47VYt8TE8nNm4Xi1hITQzZsKG7nl1+05RJ54MCBLLt1bdhAenmJsdAc/Pz8CIDLli0zWObMmTN89913Da4xMJc5c+awRYsW/OOPP3Jc08b8t2jRwiJ9SSTWRir9gkS71+Ho0bpzK1bolL6Fo3sGDhxIAFy5cqVF2928eTObNm3G8uXDCQinLykUePfuutvx8CAN+Xl9fYXS16NHjbJ161Z+/vnnPHnyZP5uwkJERkbypZde4qhRowpbFInEJIwpfWnTtzRt24rP48d15zKnuExIsGh3PXr0wOjRo9GgQQNERETg/v37SExMNLl+SkoKQkJC8OTJkyzn7969iytXLuOVV/5EzZqAn584//nnwL59YtP2Vq2AiAhxLjskcOoUEBIC1K8PvPfeapQvHwI/v9y3MXzrrb7o0GEupk9vjblzTb6VHDx48AAffPABxo0bl/dGAJQuXRqnTp3C4sWL89WOlm+++Qbt27fHqFGjTMomKpFYFEOjgS0cz+RMPyGBdHAQwe4xMcIu4umpmxqnL/qxBvPmzSMAs7Jdfv/99wTA8ePHZzkfERHBc+fOMTAwkNodCENCRPYJR0fy6FGRjcLZWdzWwYPk77//zkmTJvHOnTsMDBTny5QRtn1397cIkK1aGc/quWcP2aKF7nE1baq/3MOHD3OsIYiNFZY0Lffv3ycAVqlSxWifGo2mQGPwe/bsSW9vb+7du9fscFJTCQsL4+PHj63StsT2gZzpFyCurkDLliLZ/cmTwNWrwNOnuuvx8Vbr2tHREVWrVkWlSpVMrlOpUiVUqlQJxYoVy3K+TJky8PHxQY0aNZCemh9Vqohb2rgRaNdOZKOYPFlc++gjYM2aXzF9+nRcvXoVp06J861bA4oCzJr1JkaP3ojffjMsS0CA2LLg4kXA01OFTz55iJMn9Zft3Lkz3NzccO3aNQBAVJR47PXrA0FBokzFihWxfPlyLF++3OgzCAoKgouLC1q3bm20XFxcHMLCwoyW0UdUVBS+/fbbjDewH3/8EZs2bULXrl1zjbLKCz/99BMqVqyIefPmWbxtSRHA0GhgC8czOdMnyc8/F9PUiRPJuXN101btlNjChIaG8vDhw0wxZxmuhUhJIRs1Erc2dOhpTp06ldevX+eoUeLc9Ommt6XdmKxly2QCLixfvrzBsk2aNKGjoyMjIiKoUpHduuke8Zo15t2D1lFrbEP2tWvX5nkh3PDhwwmA7733nt7rcXFxZrdpjKNHj9LBwYEffvihRdvNTm4pNSSFB561mb5N59M3hXbtxOexYyLvPgA4OIhPK8z027Zti44dOyIgIMBibU6bNg3fffcdoqOjjZZzcgK++EL8fO9eK0yZMgUNGzbMmOm//LLpfQYHi88WLRzg7V0JderUAUnoM3tfuXIFycnJKF26NCZOBP76S3ctszvFFF5++WUkJycbzctfvXp1ODk55WmdwdChQ+Hj44NJkybluDZz5kxUrlwZN27cMLtdQ7Rp0wZPnjzBihUrAIg3lOXLl0PogvyTkpKCkSNHYvDgwRZrU1KAGBoNbOF4Zmf6T5+KKaezM1msGDNy7wPkxo0W727AgAFs3bo1z507Z7E23d3dCcCkFAzR0aSTk0jjnJAgolLt7YVbIy5OxOjv2LGD48YFsU8f0lCKe+0L0syZ4vuOHWTZsuTgwYb73rxZ1LG3J7//Xvxct67u+tmzZ7l69Wr6+/uTFPb7+/fv88yZR2a9dKWlpeXL7m/Idj9y5EgC4EztTeeRvXv3ctasWRk7pmlRq9Vs2bJlriGw5nD79m26ubnRyckpI0R3zZo1HDt2rMxPZCNAhmwWAg0a6OwNjRuLnUwAcvlyq3XZuXNnNmvWjHczezNzISUlhS1btmTt2rUzFJNGo+GiRYs4YcIEk1MwXL1KxsYm8NKlS1yx4jYBkbGTFDH1AOjufpAA+euv+tvo359ZkpUeOya+t29vuN8RI8SjXrCATE3VjbFhYeL6kCFDCICr0neS1zquGzQ4Q4Bs144cODDnWjp9hIeL8nPmmPRITCIsLIxHjhzJVxsajYbNmzcnAC5ZsiTH9Q0bNtDX15fbtm3LVz+Z2b59O8+fP09SOP2LFy9OAFy4cKHF+sg84UhKSmKHDh34q6E/HkkWpNIvDIYP1yn9Tz8lx44VP8+fb7UuK1SoQAB8YOZuXsWKFSOAHLNEc9HaxqtU+YkAOXKkOH/t2jX27NmTzZsfJiBm5Pp46SXxiI4eFd+TksSK3uyT5E2bNrFp06acn/4s1WpdmVdeYZb0R2vXruXgwYN56NAhkuTOnTvp7t6egJqOjrpFZi1arOD169eN3l9cHFmrlihvSmDMH3/8wXHjxln0DUwfGo2G+/fvp6+vLxMTEw2WsSYHDhzg9u3bmaoN9cone/bsoZubW4aS37JlCwHQzc3NYllVtVjap2ILGFP6NmnTLxJo7foA0Lkz4OYmfrZi9M7Ro0dx4cIFlC9f3qx6J06cQGBgIFxdXfPVf+XKlVG/flukpr4KQETuAGIv2927d2PQoA4AdLb77GijbqpWBaZOnQovr7LYt28Fsge4+Pv748qVK4iIiAAA2Nkho4x2mcSJE+JzyJAhWLduHTp16oQ//vgDcXFx8PE5CMAOI0cCc+YA5csvxMWLM3O112/cuBxhYTvRqdOVHDLpY/PmzVi4cCP++su0vZTzGrOvKApee+01bN++PUcUVuYy+WX16tVYvXo14vX8DXfp0gW+vr5w1IZ65ZNDhw4hISEBt2/fBgD07t0bK1aswL1798yKTsuN3bt3o2zZsmIGnE5YWFiW70UOQ6OBLRzP9Ez//n0xJXRwEFPEmTPFdwvtN5uZ2NhYenl5sWTJkhaZ0T148ICHDh3i/fv3zao3e7a4Xe0Lzp07Wa9v3y7O9+6ds25yMjNy+aelkZMnTyYATp48OUfZyMhInj9/nleuBDEsLOubwN69op3WrXP24ePjQ6ATAbHeQGs9OHnyJP/3v/8ZfdNRq8kFCxYQAD/66CNTHgePHTtNd/dIuriojS7PUKlU7NixI93c3Bhv4TQdWjQaDS9fvsxZs2blKWtqWloaK1asSAC5mqMskZVVo9Fw9+7dVn1DUalUrFWrFgEwMDCQJLlx40aWLFlS7x4VluD69ev85ZdfrB75BGneKSSWLiU3bRI/L1okHreVlvJ7eHjQycmJTyyQ5mHVqlUEwCFDhphVT6vUAbJcOZ0yTk1NpVqtzsjTr2/B1Z074pqXl/geHh7O0NBQrlunZtOm+q1i8+eLOuPG6c7FxIi0D9rHoNFoGB0dzXv37nHBggWsWPEAAZEtw9R/PI1GbDI/ZEg8jxy5YvLGMpmfR25m+xdffJGKovDYsWMmta1l5cqV/PTTT3PdAF6j0bBq1aoEkKf0FqmpqVy/fj0HDBhgUBFrNBpOnDiRXl5eFvk7NIRGo+GZM2cs0tbt27c5ZsyYjL+F9evXEwAHDRpkkfazc/jw4YzQX2sOaFLp2wKrV4vHbaYiNZV///2XI0aM4IoVK8yuu2vXLo4dO5aHDx8mSW7bto3t27fnvHnzzGonOVk4UJOTNfT31/1BT5o0iYqicOLEhQTI0qVz1k1LEyt8s++dok1bpG/3xFmzxIzdmJsk+6pc7XaQ7u49uGjRIpJC5m+/JbMtSs5g/35Rp2JFsS4hLU38Ort2JY2ZsLU5ir7+2nAZLTdu3GBkZCTj40V/pprGO3bsSAB6E8VlZ9q0afzwww+zJMWzNL17985XpNDvv//OS5cuGR1YXnvttTwPXrmh0Wj4999/W1QhZ45oCg0NJQBOmTJFKn19R5FS+trYwn79rNL83r17CYCvvfaa2XW/+OILArBIxsr+/fuzWLFiGQMIqUtbvGDBQrq4iMdgqs946lQxM2/eXHz/5Zdf2Lt3b3733XdMTU2lRmM8i2dycjKLFy/O+vXrMy1Nk9E/UJIjR47kzZs3OXXqciqKhs7OwhKXHe3CrxkzxHe1WoSFGovAnTRpJRVFQycnTZaEdLlZb3bsIKtVI9u2JUNDjZclxcxxypQpBbq7mTEuXLjAEydO5EmhpaSkZIQK375922C5iRMn0tPTk1u3bs2TjBqNJiPyyNrcunWLFStW5A8//JBxzlyzaV6QSt8W2L1bPO7u3Q2XSUsz7T9dD4GBgVy4cGGuG5Pr48iRI5w3b55Fokz69etHADlC69LS0piSksI6dcRjMGWyeffuXSpKaQKki4uGKhU5c+ZMo6tbSTIgQLwZaE3viYmJXLJkCY8ceUSArFRJxaCgIGo0Gi5fvpwAWK6cP4GsGbFJsTk8IEJBnz4Vvobhw4dz6VIVRS6hnP0nJyfTwWEGAbJPH7HFYmqqmPFXrSrWMhgiMVG8QWjfLE6cyP05WZslS5ZwxowZVs/lExERwREjRvDVV181Wi42NjZfETfaSKDs+aYyo9FoePPmzTy9OWdm6tSpBMDhw4frva5WqxkZGZmvPvRhE0ofQH0AywD8DuBjU+oUKaV/+DCNBp2npYl4QwcH83MRW5j8LLAJDw9nbGyswZmedivhvXuznp87V8TLZzbV+vv709fXl66uEQSEMk9JSeG2bduoVmtoyF947x4zzEhqtTBfAWDNmuMJCKWq5ejRoxw1ahTfe++SXuvbwoXMYl7y8PBId/w9pqMjqShkVFTWOpGRsSxVKoYAmR4pyrQ0XXho9nvXsmzZMrZs2ZLr1/+VEXrq4KALYTXElSukqVGMaWlpPHTokFlbSGrXAOzfv9/kOrbM7Nmz6erqanRNQWpqKkuWLJnFyZsXNBoNDxw4oLeNI0eOsGrVquzbt2+e2zdEvpU+gDUAwgBcz3a+G4DbAO4A+MrEtuwA/M+UskVK6Z89y/SAcP3Xp09nhteveHHy5s2cZdLSRDt//KHfTmAhG2HNmjXp7u6esYrVkgwbJm4xu8lXu/eMPl3UpYsmS+w9KXb9cnEhO3TIWV6jEb6AS5fEzydOnKCvry+7dTtKQL/t/tw50X/2zdr69hXn09d2cfHixfz5558ZExPDtm3FtZ07s9bZtUvXVuaB6eRJ8vp1w7+m3r13EKjAMWPGMDVVDEAAOWCA/vIrV67kF1/8m/FnU6+eWBvx33/6y5O6HceaNGliuFA2Nm/ezLFjx5o8u37w4AF9fX3Zo0cPk/vICykpKXnePCgpKSlXh/zHH3/MQYMGWeX/gCSDg4OpKArr1Klj8ZXMllD67QG0yKz0AdgDuAvAG4ATgCsAGgBoDGB3tqNcep3XAewD8J4p/RYppX/zpnjcmXMEaDl/XhfrqF2hVKeOyG+QlESuXy/MQiVK6AaGZs1ErmNSaJa5c8X1nj1zekMzk5jI7FPkxMREHjlyJGM7wBEjRlBRFJO2NjQFPz8/vvHGGwwKCuLUqdTr3Dx1SjhH9Vm3Pv2UWWzqJPnPP+KckRxpOXjnHVFn9WoRodS9e3ceTM/FkJZGurmJ61oZNBphYgH0K9JJk6h3EHn9dXFez86LBrl4kekO5mRGR4sB3d9fey6nY1ej0aQvxttDQKSh0P5pvPCC4X6Sk5PZoUMHfvLJJ2aFVj56lHV7TGPExcXRzs6O9vb2BheLZefq1as8cOCAycrv4cOHdHV1paenp0VCRC1NTEwMnz59mmu5S5cuWUV+Y0rfwcRY/qOKolTPdvpFAHdIBgKAoiibAbxBchaAXgba2Qlgp6IoewBs0ldGUZThAIYDYrFPkHbFzjOOfVQUqgBQxcbiYaZ7UpKTUeGdd+CkUiF2yBBE/9//ocKbb8Lp1i2ktm4N+8ePYZ9ps+606tWhpKXB4fJlqHx8EDF7NkquWYNix46JAnv2AHv2IKlDByS1bw+1hwc0pUrB+do1uBw+DOfLl6GuWBHR48cjoU8fwN4eD+7fx+cdOqBymTKo+/ff+PTTT1G9enXExcUhLi7OrPsMCgrCokWLUKZMmYwEY3/99RcePnyIzz//PH0BmCdu3YpHUFBERr2KFcWRkqJbpKWlQoXiADxw9mwCgoJEmuqzZ8W5ChWytmOMBw/KASgGD49Q7Nx5Evv27YOiKChevDjKlSuH5s0r4PjxYvjjj3D06JGIkBAHhIZWRunSari4PMghV/36LgDK48CBFAQFPQYAPHyowu7dNWBvD7z66kMEBWlMku3JE0d07eoOLy8VoqOjEB39FE5OQK1aFXHnjhO2bn2CNm2SM8qnpaXhjTdGYfnybnByIo4ff4jgYAcMHVoO587Z4dChh6hVS6W3r3Xr1gEAQkJCTJJt5kx3rFhRCnXqpOLAgdw3mQeAn3/+GbVq1cLjx49hZyfWgN6544C1a0uiZ89EvPxycpbyM2bMwG+//YbPP/8co0aNyrV9kihVqhSKFy+OCxcuoFy5cibJFRMTg5IlS1olpXVmfvzxRyxduhRTpkzBO++8Y7Bc6dKlTf49WAxDo0H2A0B1ZJ3p9wWwKtP3gQB+MlK/A4AfASwH8IkpfRapmX5EhG7alpkvvtC9m2tnRXfvCoO0durWooWwV2jTKzx9KnwD2uuA2Kjl11/F4i/tlDW3o2FDsk8fakqVynrew0P4F774Qng2Dx4Uxm0/P/Ldd0XuYgNx2NoN0mvXrp1x7siRI9y2bRuTkpIYEyNSGJgzuTl1SoiVOb5/wgRxbupU/XUSEsivvhILwTKbU6KixKz54sWLGQ49AExISOC334o2x4wRZTdsEN9ff11X//79+9y3bx9v3brFhATmsOtPmBBKgHR1PaRXrk2bxEzc0O6W2U0/2vvMvPumFq28b7+tO6c1n02apL99lUpsYzlwoGnWwD///JObN//Djz9OYdu2upQXe/fm9GUYIjFRyOPoKGQrWVL4XTIzd+5c1q1bN9dUGJnJizPXx8eH1atXN7kflUrFEydO5LoOIjvasNJ//vnHpPJJSUkWTQcBSzhy86v0zTkA9AawolZ2A+uzjHbJqaNj1vO1a4vz//6b9fzZs+Rnn5GnT+v/70xOFuknAfLVV7N68sLDyR9+EAvB+vcXGT6HDxdG8ehoYS6qVi2Loo/w8GBM/fpZTUjGDkUh69cXg1W1amSlSmTt2lQ3bcpH9eox2NeXXLeOvHBBeGf37ye3bROmrGy21Lt3xdYDhvKBxcaKlbqOjmLxFalLzrZhg/46arUYu7QOYH3ExMTQx8eH9erVIymcrpndLh99JL5nzhWUfaVwdrt+48aJBMjOnfVr9aVLRfm33tIv0/Hjxzly5Ej+Lz3r3OnToryXV9Y/A7Va9ys8cEB3Xmv2ql5d/5/Nzp26X+HKlbl7f+vVq0cAWSK7tHslfPttrtUZHEzWrKnrs0YN8dmunRiAMmMszNMSgUPx8QksW7YsnZ2dTV5gN3/+fALgwIEDze7v6dOnJoWuLlu2jCVLluT3hpJS5QFrKf3WAPZn+j4BwART2zPlKFIzfY1GZ7fPvNmJdivFvK5g1JeRzBSSk4XGXLUq67RLoxG+gl27yMmTRZD6yy+TH35ILl4sjm7dRC5lUwYHQwNGzZpkmzZkr168/8pALsVH3FFjDPl//ydWL//9t3izSU0lNZoM5frbb0LMli3Fd2Mhjb6+zLDhm/KIsu90qY22OX5cV2bz5s3s3LkzV69eTTKrXV+jIX//XQxIhnSKdhtJd3fhR0hLE2P7vn1CCa5evZoAOCDde6tW6/wKFy7o2tm6dTuXLAnl4MFZo5i0g8Frr4kXwuxow0EBsnjxG0afi0ql4dixn7Fdu3ZMTtZtc/nvv7p7iI429CwTOHPmTNav/w8BkQn1+HExH6lQgWb5PObPF7ENp0+L7zt2iDmM1vwfHx+fq118zhyx1iMtTWWW8/fmzZv09vbmVEOvlBbgzz//zPI7twTWUvoOAAIB1IDOkdvQ1PZy6avozfRJUmtG0cblajQ6D1wh7HqVL+LihPfx5k2hyUJCyNu3hWbat0/kGnrrLarq1WOolxcfN24sbC0NGjBNyZSgx8RDrdgxHB68596U7NmTa5w+4jf4ljHzVoo3mIMHhR3o9m0xoFEXbjlokEi7UKMGuXat7hb279/PyZMn816mQa9VK1Hn11/Fp7NzRnN6OXiQWd4OTEGbqfPkSaH4ALJKFTHQ+Pv7c/78+VnSMWjfOLRpiCIiIgiAzs7OWZSxlqQk/f3evi3acXHREAgjYDh8lCT/+kvItXRpzmvakFJDO6OpVCq6ulYkEEcg69bQ+/Yx3QQmBgF/f3+DsepaExYgIr5iY8W+y4BIwf3222/T0dGRly5dMngff/8tyn/yic4h/uiRaUtiNBpNltl6UJCYIxkbY8yNxElJSeFNfdF6+SDfSh/ArwBCAaQBeADAL/18DwD+EFE8E01py5yjSM30SbHLCCBm56RQnNq//iLEiRMnuHTpUgYEBPDKlSsEwIYNG2Zcb90yhfVwk9eWHCF37ODG19bxIyzlod4LxGAxfLiwm3h6Zs3gZs6bRJUqjGv6MnehJ7e7vc+9tUfza0zniWGrxEK5y5c5sEcPAuAmbX4kilk3IET46iuhKIyR3a4fFhaWa06fTz4Rfbz9tu6F6a+/DJfXKkltlOWVK/fYu7cv+/Tpk+vvIjPa7N5+fuTs2WoCYpAzNNvXRiHp299FaworU8bw6uo+fUQ4afv2OZXg99/r3lxatWpFADyabUHCtWuiD3t7YZHUcuKE2HBHoyEHDx5MOzs7bjBk56N4g/r4Yw0XLRKaWqMRPpKuXc1/SdYOwB07Gp4M1K1bl61ateKjbJs0JCSIeVIeo0zNwiIz/cI4ipzS1y5H1cb/BQeL75UqFa5cFmbAgAEEwPXr1/P+/fscPXo0p02blnH91CnhstCaQN58UzyGzZsNNKjRkKmp1IQ+Js+fZ8APf/BjLOHyshPJoUPFG0SnTuSLLwpjdub4xVyORIAJZcqIjX7bt2dw+/e4wPlL7njtJ+FkOH5cOAWyOdmSMk2nu3cXO3y1bk1WqvQunZycGGDIkUDdXsDaI7etbJOTRZ6hGzdE4raSJcViMSOTW6rVYlHXxYvie1yc7kXz4kWxzMPTU+zvo91wJjNBQaSdnYaOjhq9lkeNRufPyJRhIMv1evXE9e3bDcup0WjYoUMHuri45AgRfu893QzdECEhIYwywaN87tx5enh4cPz48YyP15nMfv/deL29e8X9xcbG8dKlSxw/Xjx/QL9zPSwsjHZ2dixRokSOwX/tWt3v3JBpy1L5eJ45pV9kzTvNm4tHrnWKXb0qvmeaBRcF1qxZww8++CDHzM0QPj7MMHeYwm+/ifIGJ7ppacLkdPQov/XZyfexgecGLBBT9yFDxBSvYcOcUUu5HW5uVFWvzmN2dvzd2VlESi1aRG7ezEW+h1kbt1nWbTHt7OyyDArZiYnRvcBUrapzTmu5evUq161bp3cHtNhYoXQ6dhQOcEMsXiza166PWrZMfM+8riEggExL02+n0PoqgE0GbeBap7C3d06nLCkGjunTjedGIsXMN7tJJCBA+FYcHEQ7hrh3T+ywlp20NHEPWgfw7NmzCejSIWgd6pUr68+3RAorrLu7KGdn9yYdHByYnJzMs2d1kUj6Bo2YmJgMx3fmP4MzZ8RAqCiibuZEgWFhYWzXrl2WN+L88Mwpfe1R5Gb67dqJR66N1DlyRHxv06Zw5SoktI65smXFYzAllUBoqO6FyUjqlAzmzBFlP/jAQIH4ePHGdfmyMNCvX8+0qd/xepvhjH7ldWH/qFaNmTK15XpoSpcWU+hu3UTHU6aIbTJ37RLT7KdP2aunhvb2wt6cnffff5+AbovH7Pz3X3iujsvQUJHeWvv29H//J5So1pL1zz//0Nvbm/37989RNymJLFdOrIJ2d3/doLlKpRIvVoCwmGVHm7Fy9uzZeuVVq0XgGZDT5PHBB+L80KGG7/HMGaF8a9TIuXhNO+g1bSreOtRqNQMCAhiUPoKoVLrJxhdfGO5j61bhE2rUqAmbNWuWUf/HH0XdkiVz7huh5d9/xUt89rnPqlW6P5fFi5kuj4pubm4EYNKirtx45pR+kZ3pa9M1anMNaN/ze/UqXLmsSHh4OO/du5djc5CtW8Wtv/+++HR0zD12X6MRM2PtP4ye7WBzcOaMrrw+h6Q+tOsCSpTI1nlMDHnrFhN376Zm7VrhHR45UsRftm0rprwmRjVpXF2ZUrOe0HqDBoklysuWkXv2cNu333LoW28ZzHVTtmxZFitWLNcEaOHhWZVhUJAuXuDSpUsEwEaNGjEwMKuDe+VKIWbz5qRKlXtUDCD+tLXExup+l5UrVyYAg28Lfn5qurtnTTcVHKzzkxhJtkmVSmdCqlZtRkaY65Mnuhm6sTRW586JPhwcaHSTGzLnxjAajc4s2bSp/jUL2r/tzz/PeU37pgGI6DKSPHXqlMWypT5zSl97FLmZvjaRizbucN06neYrQmg0GoaFhdHf35/Tp08nAE6YMCFLmY0bdZuYA2K2Zgqff66rc0j/+qcsr4aRbwAAGBtJREFUpKbqyg8bZlofWifliBGmlc+CRiO0zoULwv6xdCn5zTfCe9qtm/AdmLoWwt1deG979SI//picOZPJa9eyR+nSrOHiQnU+8rWkpqby+vXrDAtLpYuLeAu4f18o6/r1RffaDeqN8fSp7iXI318o/ObNxdvYoUPkhAkTOGbMGINJy155pS+9vZvwbHrqEI1Gtw5Az0tIDrQZy4Fgvvuu2P/Bz0+cM8VRO2KEKFu2rG6jm2nTdG+hxoiK0i2zefFFMjAwgs2aNeMX6a8OKpVYU2nIr6/dV8nOzvI5FqXStxW0i6nWrBHftTGF+jxCzzD37t0jIDYu+eGHH1i1alW9GQ2vXtWZajp3tp48rq6ij59+Mr3OxYuGQx8NMWXKFLZu3Zq79dk6shMdLR7Avn3ifX/KFDEqdekiHkrmEdHQ4ewstE7nzsIeMmOGGE1PnhTGbBOdgsOGiZeNe/fES6iwdatN3shFuwp43DjRbfPmYhmGPgdxdipVqkQAvHcviP3762L4AdNy/YhBKjmjTtOmzHhz1L4lnDt3jq+//jqXL1+eo35srFjTAIgZv3Z2XqKEWESfnewbvwcF6UxcdetGEBjF1q19cxc8ncmTdb/Kw4fFm6mRGACTkUrfVtDG6v34o/g+ZYr4/s03hSqWpUlOTmbp0qXZtGnTXKMRYmNFZIqxSJT8EhBALlhgeMZlLgsWLGCXLl0ykrVp6dWrFwFwm6GlxWaQnJTEW8eOUXXmjJgGLloknBhvvSVWpmmXGxs7XF1FkECvXiK3xMKFwq9w82aWES3zr6hTJ63Sm0Zvb2+Ttj3UJosrVUo4RWNjc6ZZMERKSgovX75MjUbDJk1EO56e+sNEDfHokUjKlzlzyVdf6a7/9NNPBMChBhwEKpUuG4r2yO5OiY2NZYMGDejh4ZHjb/rePbFiWlu3efNIk0NBNRrd24b28PbeZfrNG+CZU/pF1qb/f/8nHrl2hypt4LSx/f4kNsfw4cMJgD///HOW8w8fPuThw4ctYpfV7merL4Ing7g4Ece5Zw/5889Cc/XtKwaFzBpQ36EoTC5fntfLlePlF14gZ89m4Nzf2RhXWNYtnk5O5ejm5mZyBkhfX5FKQ59tOzw8nAcPHsy1rRMnhEM3r1GLiYlikfm33+rSWJEirPPXX3/l8cxLq/WwcSNZvry4D32UK1eO9vb2DNFmt83EnTvC31SypPBXmYNKpUsroigPCUzMEeNvLsaUviKu2yY+Pj48f/58YYthOaZNA6ZMASZOBKZPBwYPBjZsANasAYYOLWzpJCZy+fJlPH78GE2aNEGlSpWs0keXLl0QGBiIzZs344UXXsg4P3bsWFy6dAkzZ85E27ZtjTcSHQ3cuyeOwEBx3L0L3LkjUpmq1QarskoVJHt5oViTJkCdOuKoVw+oXh1wMCk5bwaVK1fGo0ePEBAQgFq1aplV11R27tyJv/76C0OGDMGLL76Y53ZIwFACTn9/f3h5eaFYsWJ6ryclifqurub3q9GIX82JExtQoUI5vPLKKwb7MQVFUS6Q9NF3zbzfniR/FC8uPhMSxGd0tPgsXbpw5CkAfH19ERgYiE2bNqFhw4aFLY5FaNasmdX72L9/f0ZK4sycOnUK586dg0mTNXd3oHlzcWQnLQ2Rly/j/ObNaOjigooJSTiyKgCVEgNQxyEQyoMHKPbgAXDqVNZ6jo5A7dpiAKhXD6hfXxz16gFubnrFaNOmDUJCQnKk6R47dizCwsLwzTffoEGDBrnfjxH279+PpUuXombNmvlS+sYyLtepU8fgtRs3bmDJkiXo0qULfH19ze7Xzk481tq1B5ld11yk0i9ItP8U8fHiU5sn3929cOSxInPnzsWKFStw584dANCrwIoSV65cwbp169C+ffs8/dNnx9Dz2rZtG27fvp3/gcfREWVeeAGvZXqLaDxJ/GkqVVTiTcDfHwgIEJ+3bgG3bwMPHgA3b4ojO1WrAg0bAg0aiM/0n7ds2aJXhB07diAoKAjffPNN/u4FQN++fVGzZk106dIly/mQkBCsW7cObdq0QadOnfLdjyGOHz+OpUuXIi4uziK/f2tik0pfUZTeAHpb61Ww0HiOZvrx8fG4c+cOBg8ejHHjxqFGjRqFLZLFCAsLw7Zt2+Dm5oZBg8TM7MyZM1i4cCGio6Ot+k/v5eUFLy8vq7Tt6SmO/v3fA0nMnz8fXt27Zy0UHy8Ggdu3xUDw33/iuH0bCA4Wx759WevUqAE0agQ0aQI0biw+a9fG9u3bcenSJdStWzffsnfs2BEdO3bMcf7kyZOYPHkyevfunW+lHx0dnfF2si/bPbZv3x4//PBDvt9YVCoVtm7diuvXr2P69OlW2exF2vQLkj//BHx9gddfB3bsALy8xMzp/n2gWrXCls6iPHz4EHFxcfDy8oKbgdf+Z5XLly+jefPmaNSoEa5duwYAuHbtGvbt24eGDRuiZ8+e+e7jyZMn6NGjB5KTk3Hjxo18t2eIhw8fYvPmzXBzc8OIESOgVqtRvHhxJCcnIzIyEqVNnZCoVMIoffMmcOOG7rh1C0hLy1nexUW8CTRtCjRrJo6mTYGSJS16fxcvXsTGjRvRpEkTDB48OF9tqVQqlChRAsnJyYiKioK7Fd7QScLT0xORkZEIDg7O8wAvbfq2gnamrzXvFOGZfuXKlQtbBKtRpUoVfPjhh6hZs2bGucaNG6Nx48YW66NMmTK4evUqVCoVkpKSUKxYMRw+fBh79uxB9+7d8eqrr1qkn5CQEHz++edo0KABRowYAUVRcOrUKdy+fdt0hQ8I527duuLI/KaTlgb6++OL7t3hHhKC/+vWDU7//SfMRxcuiCMztWsLH0SLFkDLluIwUY5r167h/Pnz6Nq1a4aDvUWLFmjRooXp92H0Fh2wYcMGVKlSxWoTGUVRMHr0aDg4OMDJyckqfRR6eKaxo8jF6Z88KeKyWrUSGaG0y/FscGNnSxAdHc3Ro0dbdEeg54mLFy9mycMyadIkAuAkQ3sh5oGUlBQOHz6cGzZssFiGR328+OKLdHZ25tmzZ6lWqzmkTx9uGzeOqgULxOquFi0Mp7Dw9hZ5qH/4QSSyMbBDzeuvv04A3GwwXat1CA4O5urVq3njxo0C7dcYeNbi9LVHkVP6mbNqhoeLn8uUKWyprEJSUhLffPNNAmDNmjULWxyrotFo+Ouvv/Ls2bNWVZynTp3ijBkzco03t0VCQ0MzMmmePn2aAFi1atWszyslRSzDXb1aLGR86SX9ie7s7clmzcSqpnXrxNJbjYY//fQT33nnHR4+fJikWFC1b98+hpqyW0o+WLt2LQGwX79+Vu3HHIwpfZs07xRZR672lTAhQWfaKYKROwDg5OSE7du3AwDGjBlTyNJYnvDwcDx+/Bje3t64d+8e3n33XZQrVw5PnjyxeF9PnjxBuXLl8NJLL+Gll16yePuZmT9/PlQqFQYOHIiKFStarN0KFSpk/Fy3bl2sX78eKpUqq6PSyUnY9Zs2BYYNE+dUKuEbOHdOHGfOANeuAZcvi2PZMlHOwwOftGmDT9q0+f/2zj+4qura45+VBMGJCYbXSp3wO6TFDG+kQiu0Fp8lVeJUw9POC/Daotax7dABxnGsr3bewHQ6bwTUqWOr8AZF3hToD6XwSDKWQicyVfkhIkZsagxawvDDEoaI/HqQ9f7Y+8ZDyM3kJvece3Lv+sycuefue+7Z37PPueuus/Y+e7n9nD/Pzp07qaqqYtq0abz66qtpOY729nbWrFnDRx99xJIlSwDXuV5TU0NlZWVa6gidZP8GcViyztM/ckQ7nzPfudOtZ9sxBnjyySd11apVPc4tP1CZNm2aArp9+3bdv3+/zpkzRxcuXJj2elasWKGFhYWXZPdKN+fPn9eGhgZdtWqVjhkzRoG0p+9LK6dOudnRHnvMJVUYPvzyu4EhQ7Rt0iR9ftQofbam5tJHdPtV9anOKZD37duXln2GAQPN089agkM2s9zTB1i0aFGmJYRGeXk5J06coKOjg+uuu461a9eGUk9+fj6ffPIJmzdvJj8/n4qKCiZOnJjWOs6cOcMtt9xCXl4eTzzxBM3NzZSXl6e1jo6ODubOnUtjYyNvvvkmgwYN6vvOCgth+nS3gDPzBw7AX/7CxYYGTr/8MkWtrZTs3cs94IaR/uEPMHUqfP3rUFkJX/qSe9As5aoLWbJkCSUlJWlvo8hI9m8QhyXrPP2LFz/1RBKZt7/1rUyrCo0DBw7o1q1bL0k8bqTGxYsXtaGhQRsaGhTQG2+8MZR6Zs+erfPnz9djvZkas4+UlZV1xr6bk2Ue6SdTpkxxd2AvveTyNC5c6OL/iXRViaWoyCUBfvppNyd0H/ti2tradN++fSknQw8brCM3RhQWumZfvty9Jk3pNPBZsGCBAjpjxoxMSwmNdevW6a5du0LtwFVV3bFjh1ZXV1+Wl2AgsW3bNi0pKVEgtHDVM888o2VlZVpXV3fpB8ePuz+B+fM/zbwSXMrK3BTn9fUpzam9bt06BVJOUh82ZvTjxDXXaOfk49BzrrYBzrJlyxTQZcuWZVpKKJw5c0aLi4sV0KaeUjwNELZt26Zbt27V9vb20Oqora3V+++/X9va2kKro1ccPOhGCdXUuBF0XaeknjXL5b1Icudz7tw5Xbp0aecopMWLF0d8AD1jRj9OjBunnWEdcIkvspSOjg5taWkJ3QvOBPX19Tps2DAF9PZE9vEBzvXXX6+A1tfXZ1pKtFy44J6h+elPXQaY4B9AXp7qzTe7XAR//3vnVzo6OrSioqIzf0LcrvGejH4sZ8ESkTtEZOXJkyczLSX9JIZttra61yx8GjeBiDB27NhQ5g+JA21tbVRWVlJbW5tpKWnh7Nmz5OXl9WuWygFJfj5MmwY/+xns2eM6fn/1K7jtNvdZQwMsWuQmlPvqV+Gpp5AjR1i+fDm1tbXMmjVrQF3jNvdO1HzlK2662hEjnOFfuxbmzMm0KiNFPv74Yw4dOsTw4cNTm64gxpw4cYLTp09n9RQaKXPyJNTWwosvQl0dnD3ryvPyYMYM+Pa33bQTRUWZ1dmFnubeiaWnn9UkPP3Dh91rlhiMXKOoqIgJEyZkjcEHKCkpMYPflaFDYe5cZ/SPHXNO2p13ujuALVtcIqTPfc4lQXrlFRcUijlm9KMmYfQTWYuyeJy+YWQVRUXurnzjRjhyxD0NfNNNcPo0rF4NN9/sJpx7/HE4fjzTapNiRj9qEg9oJcgiT9EwcoZhw+D734ft211+gZ/8BEpLXdKZhx5y6/PmuT6CmGFGP2q6Tslqnr5hDGzKy+HnP3d5MTZuhKoqOH/e5b+ePNk9Obxhg0uEGwPM6EdNV0/fjL5hZAcFBS7eX1fnks8/+KBLCrN9O9x1l0sj+fzz7g8hg5jRj5qgp3/llTB4cOa0GIYRDuPGudh+ayv84hcuM15Tk5s9dPx4WLmy+4xiEWBGP2qCnr7F8w0juykqggULXKx/zRrn7R886PoDJkxwZYlBHRFhRj9qgp6+hXYMIzcYNAi+8x2XC2D9ejfKp6XFdfZOngzbtkUmJVKjLyKFIrJbRL4ZZb2xIujpm9E3jNwiLw9qaqCx0Q3zHDkS3nrLPehVXQ1tbeFL6M1GIvKciBwTkcYu5TNFpElEmkXkkV7s6sfAb/siNGsIevoW3jGM3KSgwHn5TU1u5M9VV8GmTa7zN2R66+mvBmYGC0QkH/glUAVUAHNEpEJE/llENndZrhGRbwD7gWNp1D/wsPCOYRgJrrzSjfHfvduFgF54waWFDJFeZc5S1VdEZEyX4i8DzaraAiAi64FqVf0v4LLwjYj8C1CI+4M4IyJ1qnrZwFUReQB4AKC0tJQPP/yw1wczEBh86hSJbKHtBQWcyLLjMwyjDwwZwtX33cfQFSs4+8MfcvT3v4eQJnHrT7rEUuBg4H0rcGOyjVX1UQARuQf4R3cG32+3ElgJbsK10aNH90NiDAk8nl08ciTF2XZ8hmH0jaVLYcMGhrzxBqNfey20iRgjH72jqqtVdXNP22T11Mo2ZNMwjO4oLnbxfYCHH3Zz+oRAf4z+IWBk4P0IX9ZvVPV/VfWBoUOHpmN38cJi+oZhJOPee2HSJPdQ17JloVTRH6O/CygXkbEicgUwG9iUDlHm6RuGkZPk57sneEeNgokTQ6mit0M21wGvAV8QkVYR+Z6qXgB+BLwMvAv8VlXfSYco8/QNw8hZpk93c/fcfXcou+/t6J1uexRUtQ6oS6uibKegwM23c+6cefqGYXTPoEGh7TqW0zBkdXgHPvX2zdM3DCNiYmn0szq8A66zpqrKxe0MwzAipD/j9ENDRO4A7hg/fnympYTD8uWZVmAYRo5inr5hGEYOEUujbxiGYYSDGX3DMIwcIpZGP+tH7xiGYWSIWBp9i+kbhmGEQyyNvmEYhhEOZvQNwzByiFgafYvpG4ZhhIOoaqY1JEVEPgL6mlrqM8A/0ignncRVm+lKnbhqi6suiK+2uOqC1LWNVtXPdvdBrI1+fxCR3ao6JdM6uiOu2kxX6sRVW1x1QXy1xVUXpFdbLMM7hmEYRjiY0TcMw8ghstnor8y0gB6IqzbTlTpx1RZXXRBfbXHVBWnUlrUxfcMwDONystnTNwzDMLpgRt8wDCOHyEqjLyIzRaRJRJpF5JGI6x4pIn8Wkf0i8o6ILPTli0XkkIjs9cvtge/8h9faJCK3hajtAxF529e/25cNE5EtIvKefy3x5SIiT3ld+0TkhhB1fSHQLntFpF1EFmWizUTkORE5JiKNgbKU20hE5vnt3xOReSFqWyYif/X1bxCRq335GBE5E2i7ZwPfmeyvg2avX0LQlfK5C+N3m0TbbwK6PhCRvb48yjZLZifCv9ZUNasWIB94HxgHXAG8BVREWP+1wA1+vQj4G1ABLAYe6mb7Cq9xMDDWa88PSdsHwGe6lC0FHvHrjwCP+fXbgXpAgKnAjgjP3xFgdCbaDJgO3AA09rWNgGFAi38t8eslIWm7FSjw648FtI0JbtdlPzu9XvH6q0LQldK5C+t32522Lp8/DvxnBtosmZ0I/VrLRk//y0Czqrao6nlgPVAdVeWqelhV9/j1j4F3gdIevlINrFfVc6p6AGjGHUNUVAMv+PUXgFmB8jXqeB24WkSujUDPDOB9Ve3pSezQ2kxVXwHauqkvlTa6Ddiiqm2qegLYAswMQ5uq/lFVL/i3rwMjetqH11esqq+rsxprAseTNl09kOzchfK77Umb99b/DVjX0z5CarNkdiL0ay0bjX4pcDDwvpWejW5oiMgY4IvADl/0I39r9lzito1o9SrwRxF5Q0Qe8GXDVfWwXz8CDM+AriCzufRHmOk2g9TbKFNtdx/OG0wwVkTeFJEGEfmaLyv1eqLQlsq5y0SbfQ04qqrvBcoib7MudiL0ay0bjX4sEJGrgBeBRaraDjwDlAGTgMO428qouUlVbwCqgPkiMj34ofdiMjaGV0SuAO4EfueL4tBml5DpNkqGiDwKXAB+7YsOA6NU9YvAg8BaESmOUFLszl03zOFSByPyNuvGTnQS1rWWjUb/EDAy8H6EL4sMERmEO5G/VtWXAFT1qKpeVNUO4L/5NBwRmV5VPeRfjwEbvIajibCNfz0Wta4AVcAeVT3qdWa8zTyptlGk+kTkHuCbwL97Q4EPnxz362/g4uWf9zqCIaBQtPXh3EXdZgXAXcBvApojbbPu7AQRXGvZaPR3AeUiMtZ7jrOBTVFV7uOEq4B3VfWJQHkwHv6vQGI0wSZgtogMFpGxQDmu0yjdugpFpCixjusAbPT1J3r85wEbA7q+60cNTAVOBm47w+ISzyvTbRYg1TZ6GbhVREp8WONWX5Z2RGQm8DBwp6qeDpR/VkTy/fo4XBu1eH3tIjLVX6vfDRxPOnWleu6i/t1WAn9V1c6wTZRtlsxOEMW11p8e6LguuJ7uv+H+qR+NuO6bcLdk+4C9frkd+B/gbV++Cbg28J1HvdYm+jkqoAdd43AjIt4C3km0C/BPwFbgPeBPwDBfLsAvva63gSkht1shcBwYGiiLvM1wfzqHgf/DxUe/15c2wsXXm/1yb4jamnEx3cS19qzf9m5/nvcCe4A7AvuZgjPC7wNP45/MT7OulM9dGL/b7rT58tXAD7psG2WbJbMToV9rNg2DYRhGDpGN4R3DMAwjCWb0DcMwcggz+oZhGDmEGX3DMIwcwoy+YRhGDmFG3zAMI4cwo28YhpFD/D/oVaWkRqgf8QAAAABJRU5ErkJggg==\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_EG_1 = np.array(reward_value_EG_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_EG_1 = np.absolute(reward_value_EG_1 + res.fun)\n",
    "\n",
    "# case #2\n",
    "reward_value_opt_2 = np.array(reward_value_opt_2)\n",
    "reward_value_EG_2 = np.array(reward_value_EG_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_EG_2 = np.absolute(reward_value_EG_2 + res.fun)\n",
    "\n",
    "# case #3\n",
    "reward_value_opt_3 = np.array(reward_value_opt_3)\n",
    "reward_value_EG_3 = np.array(reward_value_EG_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_EG_3 = np.absolute(reward_value_EG_3 + res.fun)\n",
    "\n",
    "# convert y-axis to Logarithmic scale\n",
    "plt.yscale(\"log\")\n",
    "\n",
    "plt.plot(num_grads,error_reward_value_EG_1[::num_every], \"k:\",linewidth=2)\n",
    "plt.plot(num_grads,error_reward_value_EG_2[::num_every], \"b-.\",linewidth=2)\n",
    "plt.plot(num_grads,error_reward_value_EG_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_EG.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": 35,
   "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_EG_1 = np.array(utility_value_EG_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_EG_1 = np.absolute(utility_value_EG_1)\n",
    "\n",
    "# case #2\n",
    "utility_value_opt_2 = np.array(utility_value_opt_2)\n",
    "utility_value_EG_2 = np.array(utility_value_EG_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_EG_2 = np.absolute(utility_value_EG_2)\n",
    "\n",
    "# case #3\n",
    "utility_value_opt_3 = np.array(utility_value_opt_3)\n",
    "utility_value_EG_3 = np.array(utility_value_EG_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_EG_3 = np.absolute(utility_value_EG_3)\n",
    "\n",
    "# convert y-axis to Logarithmic scale\n",
    "plt.yscale(\"log\")\n",
    "\n",
    "plt.plot(num_grads,error_utility_value_EG_1[::num_every], \"k:\", linewidth=2)\n",
    "plt.plot(num_grads,error_utility_value_EG_2[::num_every], \"b-.\", linewidth=2)\n",
    "plt.plot(num_grads,error_utility_value_EG_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_EG.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": 36,
   "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_EG_1 = np.array(policy_distance_EG_1)\n",
    "\n",
    "# case #2\n",
    "policy_distance_opt_2 = np.array(policy_distance_opt_2)\n",
    "policy_distance_EG_2 = np.array(policy_distance_EG_2)\n",
    "\n",
    "# case #3\n",
    "policy_distance_opt_3 = np.array(policy_distance_opt_3)\n",
    "policy_distance_EG_3 = np.array(policy_distance_EG_3)\n",
    "\n",
    "# convert y-axis to Logarithmic scale\n",
    "plt.yscale(\"log\")\n",
    "\n",
    "plt.plot(num_grads,policy_distance_EG_1[::num_every], \"k:\", linewidth=2)\n",
    "plt.plot(num_grads,policy_distance_EG_2[::num_every], \"b-.\", linewidth=2)\n",
    "plt.plot(num_grads,policy_distance_EG_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_EG.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
}
