{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "75480f52",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import time\n",
    "from scipy.stats import norm\n",
    "from scipy.stats import poisson\n",
    "import math \n",
    "from scipy.optimize import minimize\n",
    "from tqdm import tqdm\n",
    "%matplotlib inline\n",
    "%config Completer.use_jedi = False"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "93af41c5",
   "metadata": {},
   "outputs": [],
   "source": [
    "def Mean_Sample(alpha,phi,N,r,V):\n",
    "    p = np.random.dirichlet(phi,N)\n",
    "    X = p@(r+lamb*V) \n",
    "#     print(int(np.ceil(N*alpha))-1)\n",
    "    return np.mean(X)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "f0d43cb0",
   "metadata": {},
   "outputs": [],
   "source": [
    "def VaR_Sample(alpha,phi,N,r,V): #Monto Carlo estimation for VaR\n",
    "    p = np.random.dirichlet(phi,N)\n",
    "    X = p@(r+lamb*V)\n",
    "    X = sorted(X)\n",
    "#     print(int(np.ceil(N*alpha))-1)\n",
    "    return X[int(np.ceil(N*alpha))-1]\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "c619c1cf",
   "metadata": {},
   "outputs": [],
   "source": [
    "def CVaR_Sample(alpha,phi,N,r,V): #Monto Carlo estimation for CVaR\n",
    "    p = np.random.dirichlet(phi,N)\n",
    "    X = p@(r+lamb*V)\n",
    "    X = sorted(X)\n",
    "    \n",
    "    return np.mean(X[0:int(np.ceil(N*alpha))])\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "d3bfaf35",
   "metadata": {},
   "outputs": [],
   "source": [
    "def demand(mass,n): #demand distribution, mass function p\n",
    "    return np.random.choice(A,n,p = mass)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "ff8d16dc",
   "metadata": {},
   "outputs": [],
   "source": [
    "def BRQL(risk,alpha,phi_0):\n",
    "    if risk == \"VAR\":\n",
    "        sample = VaR_Sample\n",
    "    elif risk == \"CVAR\":\n",
    "        sample = CVaR_Sample\n",
    "    elif risk == \"MEAN\":\n",
    "        sample = Mean_Sample\n",
    "    Q1 = np.zeros((SS,AA)) #Q value\n",
    "    Q2 = np.zeros((SS,AA))\n",
    "    Q_total = []\n",
    "    policy_total = []\n",
    "    for i in range(SS):\n",
    "        if S[i] > 0:\n",
    "            for j in range(AA-S[i],AA):\n",
    "                Q1[i,j] = -math.inf\n",
    "                Q2[i,j] = -math.inf\n",
    "    N_sample = np.ones((SS,AA)).astype(int)*N\n",
    "    N_next = np.copy(N_sample)\n",
    "    V = np.max(Q2,axis = 1)\n",
    "    phi = np.copy(phi_0)\n",
    "    learn_rate = 10\n",
    "    for t in range(T):\n",
    "        for l in range(m_t):\n",
    "            learn_rate+=1\n",
    "            for j in range(SS):\n",
    "                for k in range(AA-np.max([S[j],0])):\n",
    "                    index_s = range(np.max([j,K])+k-K,np.max([j,K])+k+1)\n",
    "                    Q2[j,k] = (1 - 1/learn_rate**(2/3))*Q1[j,k] + 1/learn_rate**(2/3)* sample(alpha,phi[j,k,index_s],N_sample[j,k],r[j,k,index_s],V[index_s])\n",
    "            Q1 = np.copy(Q2)\n",
    "            V = np.max(Q2,axis = 1)\n",
    "        Q_total.append(Q1)\n",
    "        policy_total.append(np.argmax(Q2,axis = 1))\n",
    "        \n",
    "    return Q_total,policy_total"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "5a068cab",
   "metadata": {},
   "outputs": [],
   "source": [
    "def f(j,k,V):\n",
    "    return r[j,k] + lamb*V"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "3cb8b04e",
   "metadata": {},
   "outputs": [],
   "source": [
    "def lamb_f(f,j,k,V,s,lamb_tran):\n",
    "    return np.max(- f(j,k,V) - lamb_tran*np.abs( S - np.tile(S[s],SS)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "589b72fa",
   "metadata": {},
   "outputs": [],
   "source": [
    "def WASS_Q_Learning(epsilon,phi,T): #phi denote the transition distribution instead of the posterior \n",
    "    Q1 = np.zeros((SS,AA)) #Q value\n",
    "    Q2 = np.zeros((SS,AA))\n",
    "    Q_total = []\n",
    "    policy_total = []\n",
    "    for i in range(SS):\n",
    "        if S[i] > 0:\n",
    "            for j in range(AA-S[i],AA):\n",
    "                Q1[i,j] = -math.inf\n",
    "                Q2[i,j] = -math.inf\n",
    "    V = np.max(Q2,axis = 1)\n",
    "    learn_rate = 1\n",
    "    lamb_0 = np.ones((SS,AA))\n",
    "    for t in (range(T)):\n",
    "        learn_rate+=1\n",
    "        for j in range(SS):\n",
    "            for k in range(AA-np.max([S[j],0])):\n",
    "                lamb_t = minimize( lambda lamb_tran: -( np.sum([phi[j,k,s]*(-lamb_f(f,j,k,V,s,lamb_tran)) for s in range(SS)])- epsilon * lamb_tran),x0 = lamb_0[j,k],bounds = [(0,None)]).x\n",
    "                lamb_0[j,k] = lamb_t\n",
    "                Q2[j,k] = (1 - 1/learn_rate)*Q1[j,k] + 1/learn_rate* (-np.sum([phi[j,k,s]*lamb_f(f,j,k,V,s,lamb_t) for s in range(SS)])- epsilon * lamb_t )\n",
    "            Q1 = np.copy(Q2)\n",
    "            V = np.max(Q2,axis = 1)\n",
    "        Q_total.append(Q1)\n",
    "        policy_total.append(np.argmax(Q2,axis = 1))\n",
    "    return Q_total,policy_total"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "593fd414",
   "metadata": {},
   "outputs": [],
   "source": [
    "def KL_Q_Learning(epsilon,phi,T):\n",
    "    Q1 = np.zeros((SS,AA)) #Q value\n",
    "    Q2 = np.zeros((SS,AA))\n",
    "    Q_total = []\n",
    "    policy_total = []\n",
    "    for i in range(SS):\n",
    "        if S[i] > 0:\n",
    "            for j in range(AA-S[i],AA):\n",
    "                Q1[i,j] = -math.inf\n",
    "                Q2[i,j] = -math.inf\n",
    "    V = np.max(Q2,axis = 1)\n",
    "    learn_rate = 1\n",
    "    beta_0 = np.ones((SS,AA))\n",
    "    for t in (range(T)):\n",
    "        learn_rate+=1\n",
    "        for j in range(SS):\n",
    "            for k in range(AA-np.max([S[j],0])):\n",
    "                beta_t = minimize( lambda beta: beta* np.log( np.dot(phi[j,k],np.exp(-f(j,k,V)/beta) ))+epsilon*beta,x0 = beta_0[j,k],bounds = [(0,None)]).x\n",
    "                beta_0[j,k] = beta_t\n",
    "                Q2[j,k] = (1 - 1/learn_rate)*Q1[j,k] + 1/learn_rate* ( -beta_t* np.log( np.dot(phi[j,k],np.exp(-f(j,k,V)/beta_t) )) - epsilon*beta_t)\n",
    "            Q1 = np.copy(Q2)\n",
    "            V = np.max(Q2,axis = 1)\n",
    "        Q_total.append(Q1)\n",
    "        policy_total.append(np.argmax(Q2,axis = 1))\n",
    "    return Q_total,policy_total"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "18bb6f0d",
   "metadata": {},
   "outputs": [],
   "source": [
    "def run_simulation(risk_level,radius):\n",
    "    policy_cvar = []\n",
    "    policy_var = []\n",
    "    policy_mean = []\n",
    "    policy_wass =[]\n",
    "    policy_kl = []\n",
    "    for rep in tqdm(range(rep_num)):\n",
    "        phi_0 = np.zeros((SS,AA,SS))\n",
    "        phi_1 = np.zeros((SS,AA,SS))\n",
    "        data = demand(p_demand,sample_size)\n",
    "        for i in range(SS):\n",
    "            for j in range(AA - np.max([S[i],0])):\n",
    "                for k in range(SS):\n",
    "                    phi_0[i,j,k] = np.sum(data == np.max([S[i],0]) + A[j] - S[k]) +1\n",
    "                phi_1[i,j] = phi_0[i,j]/np.sum(phi_0[i,j])\n",
    "        Q_wasserstein,pol_wasserstein = WASS_Q_Learning(radius,phi_1,10)\n",
    "        Q_kl,pol_kl = KL_Q_Learning(radius,phi_1,10)\n",
    "        Q_bayesian_CVAR,pol_bayesian_CVAR = BRQL(\"CVAR\",risk_level,phi_0)\n",
    "        Q_bayesian_VAR,pol_bayesian_VAR = BRQL(\"VAR\",risk_level,phi_0)\n",
    "        Q_bayesian_MEAN,pol_bayesian_MEAN = BRQL(\"MEAN\",risk_level,phi_0)\n",
    "        pol_kl=np.concatenate((pol_kl,np.tile(pol_kl[-1],(T-10,1))),axis=0)\n",
    "        pol_wasserstein=np.concatenate((pol_wasserstein,np.tile(pol_wasserstein[-1],(T-10,1))),axis=0)\n",
    "        policy_cvar.append(pol_bayesian_CVAR)\n",
    "        policy_var.append(pol_bayesian_VAR)\n",
    "        policy_mean.append(pol_bayesian_MEAN)\n",
    "        policy_wass.append(pol_wasserstein)\n",
    "        policy_kl.append(pol_kl)\n",
    "    V_cvar = np.zeros((5,rep_num,SS))\n",
    "    V_var = np.zeros((5,rep_num,SS))\n",
    "    V_mean = np.zeros((5,rep_num,SS))\n",
    "    b_cvar = np.zeros(SS)\n",
    "    b_var = np.zeros(SS)\n",
    "    b_mean = np.zeros(SS)\n",
    "    P_cvar = np.zeros((SS,SS))\n",
    "    P_var = np.zeros((SS,SS))\n",
    "    P_mean = np.zeros((SS,SS))\n",
    "    b_kl = np.zeros(SS)\n",
    "    b_wass = np.zeros(SS)\n",
    "    V_wass = np.zeros((5,rep_num,SS))\n",
    "    V_kl = np.zeros((5,rep_num,SS))\n",
    "    P_kl = np.zeros((SS,SS))\n",
    "    P_wass = np.zeros((SS,SS))\n",
    "    for ii in range(1,6):\n",
    "        for i in range(SS):\n",
    "            for j in range(AA - np.max([S[i],0])):\n",
    "                for k in range(np.max([S[i],0])+j,np.max([S[i],0])+j+K+1):\n",
    "                    demand_i = np.max([S[i],0]) + j -S[k]\n",
    "                    True_phi[i,j,k] = poisson.pmf(demand_i,lamb_poi-1+ 0.5 * ii)\n",
    "                True_phi[i,j] = True_phi[i,j]/np.sum(True_phi[i,j])\n",
    "        for rep in tqdm(range(rep_num)):\n",
    "            for i in range(SS):\n",
    "                b_cvar[i] = np.dot(True_phi[i,policy_cvar[rep][-1][i]],r[i,policy_cvar[rep][-1][i]])\n",
    "                P_cvar[i] = True_phi[i,policy_cvar[rep][-1][i]]\n",
    "                b_var[i] = np.dot(True_phi[i,policy_var[rep][-1][i]],r[i,policy_var[rep][-1][i]])\n",
    "                P_var[i] = True_phi[i,policy_var[rep][-1][i]]\n",
    "                b_mean[i] = np.dot(True_phi[i,policy_mean[rep][-1][i]],r[i,policy_mean[rep][-1][i]])\n",
    "                P_mean[i] = True_phi[i,policy_mean[rep][-1][i]]\n",
    "                b_kl[i] = np.dot(True_phi[i,policy_kl[rep][-1][i]],r[i,policy_kl[rep][-1][i]])\n",
    "                P_kl[i] = True_phi[i,policy_kl[rep][-1][i]]\n",
    "                b_wass[i] = np.dot(True_phi[i,policy_wass[rep][-1][i]],r[i,policy_wass[rep][-1][i]])\n",
    "                P_wass[i] = True_phi[i,policy_wass[rep][-1][i]]\n",
    "            V_cvar[ii-1,rep] = np.linalg.solve(np.eye(SS)-lamb*P_cvar,b_cvar)\n",
    "            V_var[ii-1,rep] = np.linalg.solve(np.eye(SS)-lamb*P_var,b_var)\n",
    "            V_mean[ii-1,rep] = np.linalg.solve(np.eye(SS)-lamb*P_mean,b_mean)\n",
    "            V_kl[ii-1,rep] = np.linalg.solve(np.eye(SS)-lamb*P_kl,b_kl)\n",
    "            V_wass[ii-1,rep] = np.linalg.solve(np.eye(SS)-lamb*P_wass,b_wass)\n",
    "    plot_cvar = np.mean(np.mean(V_cvar[:,:,:],axis=2),axis=1)\n",
    "    plot_var= np.mean(np.mean(V_var[:,:,:],axis=2),axis=1)\n",
    "    plot_mean= np.mean(np.mean(V_mean[:,:,:],axis=2),axis=1)\n",
    "    plot_wass = np.mean(np.mean(V_wass[:,:,:],axis=2),axis=1)\n",
    "    plot_kl = np.mean(np.mean(V_kl[:,:,:],axis=2),axis=1)\n",
    "    scale = 1.96/np.sqrt(rep_num)\n",
    "    data = [plot_mean,plot_cvar,plot_var,plot_kl,plot_wass]\n",
    "    step = 0.5\n",
    "    X = np.arange(1,6)*step-1.5+lamb_poi\n",
    "    fig = plt.figure(dpi=100, figsize=(6,4))\n",
    "    #添加子图区域\n",
    "    # ax = fig.add_axes([0,0,1,1])\n",
    "    #绘制柱状图\n",
    "    plt.bar(X - 0.375*step, data[0], color = 'red', width = 0.15*step,label = \"BRQL-mean\")\n",
    "    plt.bar(X - 0.225*step, data[1], color = 'black', width = 0.15*step,label = \"BRQL-CVaR\")\n",
    "    plt.bar(X -0.075*step, data[2], color = 'green', width = 0.15*step,label=\"BRQL-VaR\")\n",
    "    plt.bar(X +0.075*step, data[3], color = 'purple', width = 0.15*step,label=\"DRQL-KL\")\n",
    "    plt.bar(X +0.225*step, data[4], color = 'blue', width = 0.15*step,label=\"DRQL-Wass\")\n",
    "    plt.legend()\n",
    "    plt.xlabel(\"Poisson parameter\")\n",
    "    plt.ylabel(\"Value function in different environments\")\n",
    "    fig.savefig(\"poisson_T_\"+str(T)+\"fixed_data_K_\"+str(K)+\"risk_level_\"+str(risk_level)+\"radius_\"+str(radius)+\".png\")\n",
    "    plt.show()\n",
    "    return policy_cvar, policy_var, policy_mean, policy_wass ,policy_kl "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "39f7a268",
   "metadata": {},
   "outputs": [],
   "source": [
    "K= 10\n",
    "S = np.arange(-K,K+1) #State space\n",
    "A = np.arange(K+1) # action space\n",
    "SS = len(S)\n",
    "AA = len(A)\n",
    "c = 1.5 # order cost\n",
    "price = 3 #price for each order\n",
    "h = 1 #holding cost\n",
    "p = 1 # penalty for lost orders\n",
    "phi_0 = np.ones((SS,AA,SS))\n",
    "N = 10\n",
    "r = np.zeros((SS,AA,SS))\n",
    "lamb=0.9\n",
    "for i in range(SS):\n",
    "    for j in range(AA - np.max([S[i],0])):\n",
    "        for k in range(np.max([S[i],0])+j,np.max([S[i],0])+j+K+1):\n",
    "            r[i,j,k] = -(c*A[j]  + h*np.max([S[k],0]) + p* np.max([-S[k],0])) + price*(np.max([S[i],0])+A[j] - np.max([S[k],0]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "c355601e",
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [1:04:22<00:00, 38.63s/it]\n",
      "100%|██████████| 100/100 [00:00<00:00, 3863.94it/s]\n",
      "100%|██████████| 100/100 [00:00<00:00, 5007.65it/s]\n",
      "100%|██████████| 100/100 [00:00<00:00, 4855.92it/s]\n",
      "100%|██████████| 100/100 [00:00<00:00, 4996.97it/s]\n",
      "100%|██████████| 100/100 [00:00<00:00, 4972.21it/s]\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAhQAAAFzCAYAAABmY5CaAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjYuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8o6BhiAAAACXBIWXMAAA9hAAAPYQGoP6dpAABXe0lEQVR4nO3deVwV1f8/8NcFZN93UET0griBJmZYyaIYmqa5LxkmWO4aoqZmoqloprjlWoKWu6gVFooKqBjmRljwUTNU+ggfXEFQFmF+f/hlfl4BZS73Cldfz8djHjJnzpx533Ee3DczZ86RCYIggIiIiKgWtOo6ACIiItJ8TCiIiIio1phQEBERUa0xoSAiIqJaY0JBREREtcaEgoiIiGqNCQURERHVGhMKIiIiqjWdug5A3crLy3Hjxg2YmJhAJpPVdThEREQaQxAE3L9/H46OjtDSevY9iJc+obhx4wacnJzqOgwiIiKNlZWVhUaNGj2zzkufUJiYmAB4fDJMTU3rOBoiIiLNkZ+fDycnJ/G79Fle+oSi4jGHqakpEwoiIiIl1KTLADtlEhERUa0xoSAiIqJaY0JBREREtfbS96GoCUEQ8OjRI5SVldV1KFRHtLW1oaOjw1eLiYiU9MonFCUlJcjOzsaDBw/qOhSqY4aGhnBwcICurm5dh0JEpHFe6YSivLwcmZmZ0NbWhqOjI3R1dfkX6itIEASUlJTg5s2byMzMhKur63MHcCEiIkWvdEJRUlKC8vJyODk5wdDQsK7DoTpkYGCABg0a4Nq1aygpKYG+vn5dh0REpFH4ZxjAv0YJAK8DIqLa4G9QIiIiqjUmFERERFRrTCiIiIio1phQVEcme3GLEkaMGAGZTCYuVlZWCAwMRFpa2hMf4f9vNzY2hqenJ6Kjoyu1VVZWhsjISHh4eEBfXx/m5ubo3r07kpOTFepFR0fD3NxcqXiJiNTmBf3epWerdUKRn5+P/fv3IyMjQxXxkASBgYHIzs5GdnY2jhw5Ah0dHfTs2VOhTlRUFLKzs/HHH39g0KBB+Oijj3Dw4EFxuyAIGDx4MObNm4eJEyciIyMDSUlJcHJygq+vL/bv3/+CPxUREWkiyQnFwIEDsXr1agDAw4cP4eXlhYEDB8LDwwMxMTEqD5Cqp6enB3t7e9jb26Nt27aYPn06srKycPPmTbGOubk57O3t0axZM8ycOROWlpY4dOiQuH3Xrl3Ys2cPtmzZgpCQELi4uMDT0xMbNmzAe++9h5CQEBQWFioV39WrVyGTybBr1y68/fbbMDAwQIcOHXDp0iWcPn0aXl5eMDY2RmBgoELMwONEqEWLFtDX14e7uzvWrFmjsH369Olwc3ODoaEhmjZtitmzZ6O0tFTcHh4ejrZt2+L7779HkyZNYGZmhsGDB+P+/ftKfRYiIno2yQnFsWPH8PbbbwMA9u3bB0EQcO/ePaxcuRLz589XeYBUMwUFBdi6dSvkcjmsrKwqbS8rK8OuXbtw584dNGjQQCzftm0b3Nzc0KtXr0r7TJkyBbdv30Z8fHytYpszZw4+//xznDt3Djo6OhgyZAimTZuGFStW4Pjx47hy5Qq++OILsf7GjRsxa9YsLFiwABkZGVi4cCFmz56NzZs3i3VMTEwQHR2N9PR0rFixAhs3bkRkZKTCca9cuYL9+/cjNjYWsbGxSEpKwqJFi2r1WYiIqBqCRPr6+sL169cFQRCE4cOHC9OnTxcEQRCuXbsmGBkZSW1O7fLy8gQAQl5eXqVtDx8+FNLT04WHDx9W3hF4cYsSgoKCBG1tbcHIyEgwMjISAAgODg7C2bNnn/gIEPT19QUjIyNBW1tbACBYWloKly9fFuu4u7sLvXv3rvIYd+7cEQAIixcvFgRBEKKiogQzM7Max5iZmSkAEL799luxbPv27QIA4ciRI2JZRESE0Lx5c3HdyclJ2LZtm0JbX375peDt7V3tsb766iuhffv24vqcOXMEQ0NDIT8/XyybOnWq0LFjx2rbeOb1QET11wv6vfsqetZ36NMk36FwcnLCb7/9hsLCQsTFxaFbt24AgLt373J0wRfMz88PqampSE1NxalTp9CtWzd0794d165dE+tERkYiNTUV8fHxaNu2LSIjIyGXyyUdpyZzW4wePRrGxsbi8iQPDw/xZzs7OwBAmzZtFMpyc3MBADdv3kRWVhaCg4MV2ps/fz6uXLki7rNnzx689dZbsLe3h7GxMWbPno3r168rHLdJkyYwMTER1x0cHMTjEBGRakkeenvy5MkYNmwYjI2N4ezsDF9fXwCPH4U8+SVB6mdkZKSQHLRv3x5mZmbYuHGj+PjJ3t4ecrkccrkcu3fvRrt27eDl5YWWLVsCAFxdXZGenl5l+xUdbd3c3J4by7x58xAWFlblticfsVTMlfJ0WXl5OQCI/27cuBEdO3ZUaEdbWxsAkJKSgsGDB2Pu3Ll45513YGZmhh07dmDp0qXVHvfp4xARkWpJTijGjh2Ljh074vr16wgICBCHK27atCkWLFig8gCp5mQyGbS0tPDw4cMqt8vlcvTr1w8zZszAjz/+CAAYMmQIhg4dip9//rlSP4qlS5fC0dERAQEBzz22ra0tbG1ta/0Z7Ozs0LBhQ/zzzz8YNmxYlXWSk5Ph7OyMWbNmiWVP3pUhIqIXT3JCUfGXaPv27RXK/f39sWTJEnTq1EllwdGzFRcXIycnB8DjR06rV69GQUFBlR0sK0yZMgWenp44c+YMvLy8MHjwYOzatQtBQUFYsmQJunTpgvz8fHzzzTeIjY1FXFycwl/6ZWVlSE1NVWhTV1dXvOOhCuHh4Zg4cSJMTU3RvXt3FBcX48yZM7h79y5CQ0Mhl8tx/fp17NixAx06dMCBAwewb98+lR2fiIikk9yHYu7cuSgoKKhU/uDBA8ydO1clQdULL7JbppLi4uLg4OAABwcHdOzYEadPn8bu3bvFx1BVadOmDbp27Sq+VSGTybB7927MnDkTkZGRaN68OTw9PbFnzx6cP38efn5+CvsXFBSgXbt2CkuPHj2U/gxVCQkJwbfffovo6Gi0adMGPj4+iI6OhouLCwCgd+/e+PTTTzF+/Hi0bdsWJ0+exOzZs1UaAxERSSMTBGnfaFpaWvjf//4HGxsbhfKjR49i0KBBlcYTqGv5+fkwMzNDXl4eTE1NFbYVFRUhMzMTLi4u7FD6hHPnzqFr164IDg7GkiVL6jqcF4bXA5GGUmbky1r8MfcqedZ36NNq/MjDwsJCHMbZzc1N7FwHPL4NXlBQgNGjRysfNdUbr732Go4cOYIff/wRV65cQbNmzeo6JCIiqudqnFAsX74cgiBg5MiRmDt3LszMzMRturq6aNKkCby9vdUSJL14FY8ziIiIaqLGCUVQUBAAwMXFBZ06dar0Sh4REdHLTjZX+uMVYc6r8XhF8lsePj4+KC8vx6VLl5Cbm1vpvf7OnTurLDgiIiLSDJITipSUFAwdOhTXrl3D0/05ZTIZysrKVBYcERERaQbJCcXo0aPh5eWFAwcOwMHBQaFzJhEREb2aJCcUly9fxp49eyTPB0FERFRfKP3HcLhKw3ipSB7YqmPHjvj777/VEQsRERFpKMkJxYQJEzBlyhRER0fj7NmzSEtLU1ikWLt2LTw8PGBqagpTU1N4e3vj119/FbePGDFCHPuiYnnjjTekhkxERERqJvmRR79+/QAAI0eOFMtkMhkEQZDcKbNRo0ZYtGiR+Phk8+bN6N27N86fP49WrVoBAAIDAxEVFSXuU5OptFXhRfYNkThYKRERUb0j+Q5FZmZmpeWff/4R/5WiV69e6NGjB9zc3ODm5oYFCxbA2NgYKSkpYh09PT3Y29uLi6WlpdSQX0pP372xsrJCYGCgwl2iJ7cbGxvD09MT0dHRldoqKytDZGQkPDw8oK+vD3Nzc3Tv3h3JyckK9aKjo2Fubi451oSEBPTo0QNWVlYwNDREy5YtMWXKFPz3v/9FTEwMtLW1cf369Sr3dXd3x8SJE597jMTExErnw9/fv9JnICIi9ZCcUDg7Oz9zUVZZWRl27NiBwsJChRE3ExMTYWtrCzc3N4waNQq5ubnPbKe4uBj5+fkKy8sqMDAQ2dnZyM7OxpEjR6Cjo4OePXsq1ImKikJ2djb++OMPDBo0CB999BEOHjwobhcEAYMHD8a8efMwceJEZGRkICkpCU5OTvD19cX+/ftrFeP69evRtWtX2NvbIyYmBunp6Vi3bh3y8vKwdOlSvPfee7CyssLmzZsr7ZucnIyLFy8iODi4xse7ePEisrOzkZiYCBsbG7z77rvPvWaIiEgFBCVs2bJF6NSpk+Dg4CBcvXpVEARBiIyMFPbv3y+5rbS0NMHIyEjQ1tYWzMzMhAMHDojbduzYIcTGxgoXLlwQfvrpJ8HT01No1aqVUFRUVG17c+bMEQBUWvLy8irVffjwoZCeni48fPiw0raq2lDXooygoCChd+/eCmXHjh0TAAi5ubniZ9i3b59CHUtLSyE0NFRc37FjhwBA+Omnnyodo2/fvoKVlZVQUFAgCIIgREVFCWZmZjWOMSsrS9DV1RUmT55c5fa7d+8KgiAIoaGhQtOmTYXy8nKF7SNHjhTat28vCIIgLF26VGjdurVgaGgoNGrUSBgzZoxw//59sW5CQoIAQGxTEB5fW9V9tqo863ogonpMiXmelf6dHS590WR5eXnVfoc+TfIdirVr1yI0NBQ9evTAvXv3xD4T5ubmWL58udTm0Lx5c6SmpiIlJQVjxoxBUFAQ0tPTAQCDBg3Cu+++i9atW6NXr1749ddfcenSJRw4cKDa9mbMmIG8vDxxycrKkhyTJiooKMDWrVshl8thZWVVaXtZWRl27dqFO3fuKAybvm3bNri5uaFXr16V9pkyZQpu376N+Ph4pWLavXs3SkpKMG3atCq3Vzw+CQ4Oxj///IOkpCRxW2FhIXbt2iXendDS0sLKlSvx559/YvPmzTh69Gi17QLAgwcPxL43HCae6pRMJn0h0kCSO2WuWrUKGzduRJ8+fbBo0SKx3MvLC2FhYZID0NXVFTtlenl54fTp01ixYgXWr19fqa6DgwOcnZ1x+fLlatvT09ODnp6e5Dg0UWxsLIyNjQE8/gJ2cHBAbGwstLT+f544ZMgQaGtro6ioCGVlZbC0tERISIi4/dKlS2jRokWV7VeUX7p0San4Ll++DFNTUzg4ODyzXsuWLdGxY0dERUXB19cXALBr1y6UlZVhyJAhAIDJkyeL9V1cXPDll19izJgxWLNmjUJbjRo1AvA4oRAEAe3bt0eXLl2Uip+IiGpOqU6ZVc1Cqaenh8LCwloHJAgCiouLq9x2+/ZtZGVlPfcL6lXh5+eH1NRUpKam4tSpU+jWrRu6d++Oa9euiXUiIyORmpqK+Ph4tG3bFpGRkZIHJavJmzWjR4+GsbGxuAAQ3/ypieDgYOzZswf3798HAGzatAl9+/YV72IkJCQgICAADRs2hImJCT788EPcvn270jV3/PhxnDt3Dtu3b4ezszOio6N5h4KI6AWQnFC4uLggNTW1Uvmvv/6Kli1bSmpr5syZOH78OK5evYoLFy5g1qxZSExMxLBhw1BQUICwsDD89ttvuHr1KhITE9GrVy9YW1vj/ffflxr2S8nIyAhyuRxyuRyvv/46vvvuOxQWFmLjxo1iHXt7e8jlcvj5+WH37t0YN26c+EgJAFxdXRXWn5SRkQEAcHNze24s8+bNE5ObiuvDzc0NeXl5yM7Ofu7+gwcPhkwmw86dO/H333/jxIkT4uOOa9euoUePHmjdujViYmJw9uxZfPPNNwCA0tJShXZcXFzg5uaGQYMGYe7cuXj//ferTVCJiEh1JCcUU6dOxbhx47Bz504IgoDff/8dCxYswMyZMzF16lRJbf3vf//D8OHD0bx5c3Tp0gWnTp1CXFwcAgICoK2tjQsXLqB3795wc3NDUFAQ3Nzc8Ntvv8HExERq2K8EmUwGLS0tPHz4sMrtcrkc/fr1w4wZM8SyIUOG4PLly/j5558r1V+6dCkcHR0REBDw3GPb2tqKyU3FHZD+/ftDV1cXX331VZX73Lt3T/zZxMQEAwYMQFRUFDZt2oSmTZuKjz/OnDmDR48eYenSpXjjjTfg5uaGGzduPDem4cOHo7y8vNJjESIiUj3JfSg++ugjPHr0CNOmTcODBw8wdOhQNGzYECtWrMDgwYMltfXdd99Vu83AwEDh9UaqrLi4GDk5OQCAu3fvYvXq1SgoKKiyg2WFKVOmwNPTE2fOnIGXlxcGDx6MXbt2ISgoCEuWLEGXLl2Qn5+Pb775BrGxsYiLi1N4ZFBWVlbpDpWurm6Vd6ecnJwQGRmJ8ePHIz8/Hx9++CGaNGmCf//9F1u2bIGxsTGWLl0q1g8ODsbbb7+N9PR0hIWFiY9LmjVrhkePHmHVqlXo1asXkpOTsW7duueeHy0tLUyePBnz58/HJ598AkNDw+fuQ0RESqrN6yQ3b94U/ve//9WmCbV71isvmvyaYFBQkMKrTCYmJkKHDh2EPXv2iHVQxWujgiAIAQEBQvfu3cX10tJSYcmSJUKrVq0EXV1dAYBgaWkp/PXXXwr7RUVFVfkalbOz8zNjjY+PF9555x3BwsJC0NfXF9zd3YWwsDDhxo0bleo2b95c0NLSErKyshTKly1bJjg4OAgGBgbCO++8I2zZskXhNdGqXhsVBEEoKCgQLCwshMWLFz8zRkHQ7OuB6jElXmkkifjaqNpIeW1UJggv97jP+fn5MDMzQ15eHkxNTRW2FRUVITMzEy4uLtDX16+jCOufc+fOoWvXrggODsaSJUvqOpwXhtcDqYUyr4Eq+WtZNlf6sYQ5L8FXgBLnWOmXc8Ol76LJ5/hZ36FPk9yH4vbt2xg3bhxatmwJa2trWFpaKiyk+V577TUcOXIERkZGuHLlSl2HQ0REGkByH4oPPvgAV65cQXBwMOzs7F7oJFr04rRr167K14OJiIiqIjmhOHHiBE6cOAFPT091xENE9MpT+g+1cJWGQSSJ5Ece7u7u1b6WSERERK8myQnFmjVrMGvWLCQlJeH27duvzMyeREREVD3JjzzMzc2Rl5cHf39/hXLh/4ZZrpgsjIiIiF4dkhOKYcOGQVdXF9u2bWOnTCIiIgKgRELx559/4vz582jevLk64iEiIiINJLkPhZeXF7KystQRCxEREWkoyXcoJkyYgEmTJmHq1Klo06ZNpamhPTw8VBZcXVJmxDllafIoakRERIASdygGDRqEjIwMjBw5Eh06dEDbtm3Rrl078V96MUaMGAGZTCYuVlZWCAwMRFpamljnye3Gxsbw9PREdHR0pbbKysoQGRkJDw8P6Ovrw9zcHN27d0dycrJCvejoaJibmz83tpKSElhbW2P+/PlVbo+IiIC1tTVKSkqe21aTJk3Ez2BgYAB3d3csWbIEL/mI8UREGkdyQpGZmVlp+eeff8R/6cUJDAxEdnY2srOzceTIEejo6KBnz54KdaKiopCdnY0//vgDgwYNwkcffaQwi6sgCBg8eDDmzZuHiRMnIiMjA0lJSXBycoKvry/2798vOS5dXV188MEHiI6OrvKLPyoqCsOHD4eurm6N2ps3bx6ys7ORkZGBsLAwzJw5Exs2bJAcFxERqY/khMLZ2fmZC704enp6sLe3h729Pdq2bYvp06cjKysLN2/eFOuYm5vD3t4ezZo1w8yZM2FpaYlDhw6J23ft2oU9e/Zgy5YtCAkJgYuLCzw9PbFhwwa89957CAkJQWFhoeTYgoODceXKFRw7dkyh/Pjx47h8+TKCg4Nx+vRpBAQEwNraGmZmZvDx8cG5c+cqtWViYgJ7e3s0adIEISEh8PDwUPgMRERU9yT3oQCAS5cuITExEbm5uSgvL1fY9sUXX6gkMJKmoKAAW7duhVwuh5WVVaXtZWVliImJwZ07dxT6vWzbtg1ubm7o1atXpX2mTJmCvXv3Ij4+Hn369JEUT5s2bdChQwdERUXBx8dHLN+0aRNef/11tG7dGkePHkVQUBBWrlwJAFi6dCl69OiBy5cvw8TEpFKbgiAgKSkJGRkZcHV1lRQPERGpl+SEYuPGjRgzZgysra1hb2+vMA6FTCZjQvECxcbGwtjYGABQWFgIBwcHxMbGQkvr/994GjJkCLS1tVFUVISysjJYWloiJCRE3H7p0iW0aNGiyvYryi9duqRUfCNHjkRYWBhWr14NY2NjFBQUYPfu3Vi2bBkAVBocbf369bCwsEBSUpLCo5vp06fj888/R0lJCUpLS6Gvr4+JEycqFRMREamH5Ece8+fPx4IFC5CTk4PU1FScP39eXKq6XU3q4+fnh9TUVKSmpuLUqVPo1q0bunfvjmvXrol1IiMjkZqaivj4eLRt2xaRkZGQy+WSjlOTvg6jR4+GsbGxuACPk5ny8nLs3LkTALBz506xzwYA5ObmYvTo0XBzc4OZmRnMzMxQUFCA69evK7Q9depUpKamIikpCX5+fpg1axY6deok6TMQEZF6Sb5DcffuXQwYMEAdsZBERkZGCslB+/btYWZmho0bN4pvWNjb20Mul0Mul2P37t1o164dvLy80LJlSwCAq6sr0tPTq2w/IyMDAODm5vbcWObNm4ewsDCFMjMzM/Tv3x9RUVEIDg5GVFQU+vfvD1NTUwCP31S5efMmli9fDmdnZ+jp6cHb27vS2x/W1tbiZ4iJiYFcLscbb7yBrl271vBMERGRukm+QzFgwAB2iKunZDIZtLS0qp0NVi6Xo1+/fpgxY4ZYNmTIEFy+fBk///xzpfpLly6Fo6MjAgICnntsW1tb8Uv/ySQnODgYycnJiI2NRXJyMoKDg8Vtx48fx8SJE9GjRw+0atUKenp6uHXr1jOPY2FhgQkTJiAsLIyvjhIR1SOS71DI5XLMnj0bKSkpVQ5sxWfbL05xcTFycnIAPL5ztHr1ahQUFFTZwbLClClT4OnpiTNnzsDLywuDBw/Grl27EBQUhCVLlqBLly7Iz8/HN998g9jYWMTFxSn8H5eVlSE1NVWhTV1dXfGOx9N8fHwgl8vx4YcfQi6Xo3PnzuI2uVyO77//Hl5eXsjPz8fUqVNhYGDw3M89btw4LF68GDExMejfv/9z6xMRkfpJTig2bNgAY2NjJCUlISkpSWGbTCZ7aRIKTRi9Mi4uDg4ODgAev1rp7u6O3bt3w9fXt9p92rRpg65du+KLL77AL7/8AplMht27d2P58uWIjIzE2LFjUVJSAktLS5w/f75SolBQUFBpADNnZ2dcvXq12mOOHDkSM2fOxNSpUxXKN23ahI8//hjt2rVD48aNsXDhwkqPTapiY2OD4cOHIzw8HH379lXohEpERHVDJrzk943z8/NhZmaGvLw88dl9haKiImRmZsLFxQX6+vp1FGH9c+7cOXTt2hXBwcFYsmRJXYfzwvB6ILVQYkZmpQf+D5e+iyb88fRcPMdq86zv0KfV6k87QRD4HPsl9Nprr+HIkSMwMjLClStX6jocIiLSAEolFFu2bEGbNm1gYGAAAwMDeHh44Pvvv1d1bFSH2rVrh/DwcDRr1qyuQyEiIg0guQ/FsmXLMHv2bIwfPx5vvvkmBEFAcnIyRo8ejVu3buHTTz9VR5xERERUj0lOKFatWoW1a9fiww8/FMt69+6NVq1aITw8nAkFERHRK0jyI4/s7OwqRyns1KkTsrOzJbW1du1aeHh4wNTUFKampvD29savv/4qbhcEAeHh4XB0dISBgQF8fX3x119/SQ2ZiIiI1ExyQiGXy7Fr165K5Tt37pQ8YVOjRo2waNEinDlzBmfOnIG/vz969+4tJg1fffUVli1bhtWrV+P06dOwt7dHQEAA7t+/LzVsIiIiUiPJjzzmzp2LQYMG4dixY3jzzTchk8lw4sQJHDlypMpE41meHoBpwYIFWLt2LVJSUtCyZUssX74cs2bNQt++fQEAmzdvhp2dHbZt24ZPPvlEauhERESkJpLvUPTr1w+///47rK2tsX//fuzduxfW1tb4/fff8f777ysdSFlZGXbs2IHCwkJ4e3sjMzMTOTk56Natm1hHT08PPj4+OHnyZLXtFBcXIz8/X2EhomrIZMotRERPkXSHorS0FB9//DFmz56NH374QSUBXLhwAd7e3igqKoKxsTH27duHli1bikmDnZ2dQn07OzuF2TSfFhERgblz56okNiIiIqoZSQlFgwYNsG/fPsyePVtlATRv3hypqam4d+8eYmJiEBQUpDCkt+ypv4YEQahU9qQZM2YgNDRUXM/Pz4eTk5PkuObKXlxSMkeY88KORVRXZHOl39nQ5BEGiV41kh95vP/++9i/f7/KAtDV1YVcLoeXlxciIiLg6emJFStWwN7eHgDEya8q5ObmVrpr8SQ9PT3xrZGK5WU0YsQIyGQyyGQyNGjQAHZ2dggICMCmTZtQXl4u1mvSpIlYz8DAAO7u7liyZEmVI5xu3rwZr7/+OoyMjGBiYoLOnTsjNjZWoU5iYiJkMhnu3btX41ibNGmC5cuXi+uCIGDKlCkwMTHB0aNHAQC+vr6YPHmypHNAdafimpKyENHLTanZRr/88kucPHkS7du3h5GRkcL22k4OJggCiouL4eLiAnt7e8THx4uTUZWUlCApKQmLFy+u1TFeFoGBgYiKikJZWRn+97//IS4uDpMmTcKePXvw008/QUfn8X/vvHnzMGrUKBQVFeHw4cMYM2YMTE1NFTq2hoWFYfXq1Zg/fz769OmD0tJS/PDDD+jduzdWrFiB8ePHqyTmsrIyjBo1Cj///DOOHj2KDh06qKRdIiKqW5ITim+//Rbm5uY4e/Yszp49q7BN6myjM2fORPfu3eHk5IT79+9jx44dSExMRFxcHGQyGSZPnoyFCxfC1dUVrq6uWLhwIQwNDTF06FCpYb+U9PT0xDs5DRs2xGuvvYY33ngDXbp0QXR0NEJCQgA8nom0ol5ISAjWrl2LQ4cOiQlFSkoKli5dipUrV2LChAli+wsWLEBRURFCQ0PRu3dvpR4dPam4uBhDhgzB6dOncezYMbRo0aJW7RERUf0hOaHIzMxU2cH/97//Yfjw4cjOzoaZmRk8PDwQFxeHgIAAAMC0adPw8OFDjB07Fnfv3kXHjh1x6NAhmJiYqCyGl42/vz88PT2xd+9eMaGoIAgCkpKSkJGRoTBmyPbt22FsbFzlq7hTpkzBsmXLEBMTU6tHEgUFBXj33XeRlZWF5ORkNG7cWOm2iIio/pGcUKjSd99998ztMpkM4eHhCA8PfzEBvSTc3d2RlpYmrk+fPh2ff/45SkpKUFpaCn19fYU7SZcuXUKzZs2gq6tbqS1HR0eYmZnh0qVLtYrpyy+/hImJCdLT02Fra1urtoiIqP6pUUIRGhqKL7/8EkZGRgpvUFRl2bJlKgmMlPf0mzBTp07FiBEjcPPmTcyaNQv+/v5VDp/+rPaqSjaetnDhQixcuFBcT09PF+9EdOvWDYcPH8bChQsVOmgSEVHVlOnLXEV/+xemRgnF+fPnUVpaKv5cHfbkrh8yMjLg4uIirltbW0Mul0MulyMmJgZyuRxvvPEGunbtCgBwdXXFiRMnUFJSUilxuHHjBvLz8+Hm5vbc444ePRoDBw4U1x0dHcWfu3TpgokTJ6J3794oKyvDqlWravsxiYioHqlRQpGQkFDlz1T/HD16FBcuXKh21lcLCwtMmDABYWFhOH/+PGQyGYYMGYJVq1Zh/fr1Cp0yAeDrr7+Gvr4+Bg0a9NxjW1pawtLSstrtAQEBiI2NRa9evVBeXo7Vq1czCSUieklI7kOxefNm9O/fv9LrovTiFRcXIycnR+G10YiICPTs2VNhevmnjRs3DosXL0ZMTAz69+8Pb29vTJo0CVOnTkVJSYnCa6MrV65EdHQ0rKysFNq4cOFCpc6xbdu2fW7M/v7+OHDgAHr27AlBEPDNN9+IScXNmzeRmpqqUN/e3l58Q4WI6g9Nux1P6ic5oQgLC8PYsWPRq1cvfPDBBwgMDBTHO3iZaMLolXFxcXBwcICOjg4sLCzg6emJlStXIigoCFpa1Y9ZZmNjg+HDhyM8PBx9+/aFlpYWli9fDg8PD6xZswaff/45ioqKoKuri6NHj6Jz586V2qiqrKrBsqri6+uLX375Be+++y7Ky8uxdu1aAMC2bduwbds2hbpz5sxhp1wiIg0gE2r6LfB/Hj16hLi4OGzfvh0//vgjDAwMMGDAAHzwwQeSOvq9KPn5+TAzM0NeXl6lUTOLioqQmZkJFxcX6Ovr11GE9dPVq1fh4+MDb29vbN26Fdra2nUdktq9kteDko+clNorXPouL8XQ20qcY6UfBIZL30XZc1yv7lDwHP//Y6n4HD/rO/Rpkofe1tHRQc+ePbF161bk5uZi+fLluHbtGvz8/NCsWTOlg6b6pUmTJkhMTIS7u3ulxxBERERPq9WzCkNDQ7zzzju4e/curl27hoyMDFXFRfWAi4sLHzcQEVGNSL5DAQAPHjzA1q1b0aNHDzg6OiIyMhJ9+vTBn3/+qer4iIiISANIvkMxZMgQ/PzzzzA0NMSAAQOQmJhYL/tOEBER0YsjOaGQyWTYuXMn3nnnnZfy7Q4iIiKSTnJG8PRrfURERERK3WI4cuQIjhw5gtzcXJSXlyts27Rpk0oCIyIiIs0hOaGYO3cu5s2bBy8vLzg4OHDoZCIiIpKeUKxbtw7R0dEYPny4OuIhIiIiDSQ5oSgpKXkl3up4kTdeOL49ERFpOsnjUISEhLBjZj0wYsQIyGQyyGQyNGjQAHZ2dggICMCmTZsU+rU0adJErGdgYAB3d3csWbKkynk3Nm/ejNdffx1GRkYwMTFB586dERsbq1AnMTERMpkM9+7de26M//nPfyCTyXDq1CmF8o4dO0JPTw8PHjwQy0pKSmBoaIgNGzZIPBNElclk0hciqh3JdyiKioqwYcMGHD58GB4eHmjQoIHC9mXLlqksOHq2wMBAREVFKcw2OmnSJOzZswc//fST+FrvvHnzMGrUKBQVFeHw4cMYM2YMTE1N8cknn4hthYWFYfXq1Zg/f77CbKO9e/fGihUrMH78eMnxubu7w8HBAQkJCejYsSMAoKCgAOfPn4ednR1OnjyJrl27AgBOnTqFhw8fws/PTwVnhoiIXjTJdyjS0tLQtm1baGlp4c8//8T58+fFhXM+vFh6enqwt7dHw4YN8dprr2HmzJn48ccf8euvvyI6OlqsZ2JiAnt7ezRp0gQhISHw8PDAoUOHxO0pKSlYunQplixZgrCwMMjlcrRo0QILFizA5MmTERoaiqysLKVi9PX1RWJiorh+/PhxuLm54b333lMoT0xMRMOGDeHq6orTp08jICAA1tbWMDMzg4+PD86dO6fQbnh4OBo3bgw9PT04Ojpi4sSJ4rY1a9bA1dUV+vr6sLOzQ//+/ZWKnYiIak7yHYqEhAR1xEEq4u/vD09PT+zduxchISEK2wRBQFJSEjIyMuDq6iqWb9++HcbGxgp3LCpMmTIFy5YtQ0xMDCZPniw5Hj8/P3z66ad49OgRdHR0kJCQAF9fX3Tu3BkrVqwQ6yUkJIh3J+7fv4+goCCsXLkSALB06VL06NEDly9fhomJCfbs2YPIyEjs2LEDrVq1Qk5ODv744w8AwJkzZzBx4kR8//336NSpE+7cuYPjx49LjpuIiKRRai4PAPj7779x8OBBPHz4EACqfCZPdcPd3R1Xr14V16dPnw5jY2Po6enBz88PgiAo/EV/6dIlNGvWDLq6upXacnR0hJmZGS5duqRULL6+vigsLMTp06cBPL4T4ePjAx8fH5w5cwYPHjxASUkJUlJSxITC398fH3zwAVq0aIEWLVpg/fr1ePDgAZKSkgAA169fh729Pbp27YrGjRvj9ddfx6hRo8RtRkZG6NmzJ5ydndGuXTuFz0pEROohOaG4ffs2unTpAjc3N/To0QPZ2dkAHnfWnDJlisoDJOkEQVAYH2Tq1KlITU1FUlIS/Pz8MGvWLElv6giCUGWy8bSFCxfC2NhYXK5fvw5XV1c0atQIiYmJyM/Px/nz5+Hj4wM7Ozu4uLggOTkZKSkpePjwIfz9/QEAubm5GD16NNzc3GBmZgYzMzMUFBTg+vXrAIABAwbg4cOHaNq0KUaNGoV9+/bh0aNHAICAgAA4OzujadOmGD58OLZu3arQ+ZOIiNRDckLx6aefokGDBrh+/ToMDQ3F8kGDBiEuLk6lwZFyMjIy4OLiIq5bW1tDLpfD29sbMTExiIyMxOHDh8Xtrq6uuHLlCkpKSiq1dePGDeTn58PNze25xx09ejRSU1PFxdHREcDjuxQJCQk4fvw4XF1dYWtrCwDw8fFBQkICEhIS4OzsjCZNmgB4/AbL2bNnsXz5cpw8eRKpqamwsrIS43NycsLFixfxzTffwMDAAGPHjkXnzp1RWloKExMTnDt3Dtu3b4eDgwO++OILeHp61uitFCIiUp7khOLQoUNYvHgxGjVqpFDu6uqKa9euqSwwUs7Ro0dx4cIF9OvXr8rtFhYWmDBhAsLCwsTHVEOGDEFBQQHWr19fqf7XX38NfX19DBo06LnHtrS0hFwuF5eKt0z8/Pxw8uRJxMfHw9fXV6zv4+ODxMREJCYmincngMcdNydOnIgePXqgVatW0NPTw61btxSOZWBggPfeew8rV65EYmIifvvtN1y4cAEAoKOjg65du+Krr75CWloarl69iqNHjz43fiIiUp7kTpmFhYUKdyYq3Lp1C3p6eioJimqmuLgYOTk5Cq+NRkREoGfPnvjwww+r3W/cuHFYvHgxYmJi0L9/f3h7e2PSpEmYOnUqSkpKFF4bXblyJaKjo2FlZaXQxoULF2BiYqJQ1rZt2yqP5+fnh8LCQmzatAkbN24Uy318fDBixAhoa2tj5MiRYrlcLsf3338PLy8v5OfnY+rUqTAwMBC3R0dHo6ysDB07doShoSG+//57GBgYwNnZGbGxsfjnn3/QuXNnWFhY4JdffkF5eTmaN28u5dQSEZFEkhOKzp07Y8uWLfjyyy8BPJ7OvLy8HEuWLHmpxhDQhD6mcXFxcHBwgI6ODiwsLODp6YmVK1ciKCgIWlrV33yysbHB8OHDER4ejr59+0JLSwvLly+Hh4cH1qxZg88//xxFRUXQ1dXF0aNH0blz50ptVFVWXcdcFxcXODs749q1a/Dx8RHLGzZsiMaNG+PKlSsK186mTZvw8ccfo127dmjcuDEWLlyIsLAwcbu5uTkWLVqE0NBQlJWVoU2bNvj5559hZWUFc3Nz7N27F+Hh4SgqKoKrqyu2b9+OVq1a1eicEhGRcmSCxNcz0tPT4evri/bt2+Po0aN477338Ndff+HOnTtITk5Gs2bN1BWrUvLz82FmZoa8vDyYmpoqbCsqKkJmZiZcXFygr69fRxHWT1evXoWPjw+8vb2xdetWaGtr13VIavdKXg9KDhGp1F7h0ncR5iiX2SvzsdT2R4QSwSg9cGe49F14jiUKl76LJp/jZ32HPk1yH4qWLVsiLS0Nr7/+OgICAlBYWIi+ffvi/Pnz9S6ZIOU1adIEiYmJcHd354BlRET0XJIfeQCAvb095s6dW+uDR0REYO/evfjPf/4DAwMDdOrUCYsXL1Z43j1ixAhs3rxZYb+OHTsiJSWl1senZ3NxcUF4eHhdh0FEajRXpuzv8jkqjYM0n9IDW6lCUlISxo0bh5SUFMTHx+PRo0fo1q0bCgsLFeoFBgYiOztbXH755Zc6ipiIiIiqotQdClV5etyKqKgo2Nra4uzZswqd/irmrCAiIqL6qU7vUDwtLy8PwOPxDJ6UmJgIW1tbuLm5YdSoUcjNza22jeLiYuTn5yssz8NhwwngdUBEVBt1eofiSYIgIDQ0FG+99RZat24tlnfv3h0DBgyAs7MzMjMzMXv2bPj7++Ps2bNVjnsRERFR4/4dFVOvP3jwQGGcA3o1VQzRXXFdUN3j830izSE5ofD398fevXthbm6uUJ6fn48+ffooPSLh+PHjkZaWhhMnTiiUPzlCY+vWreHl5QVnZ2ccOHAAffv2rdTOjBkzEBoaqhCXk5NTlcfU1taGubm5eMfD0NBQYQ4MejUIgoAHDx4gNzcX5ubmr8QrskREqiY5oUhMTKxyzoeioiKlp4meMGECfvrpJxw7dqzSkN5Pc3BwgLOzMy5fvlzldj09PUkjdlb0zXjWYxR6NZibm7OvDhGp3Ktyp63GCUVaWpr4c3p6OnJycsT1srIyxMXFoWHDhpIOLggCJkyYgH379iExMVFhQqvq3L59G1lZWXBwcJB0rOrIZDI4ODjA1tYWpaWlKmmTNE+DBg14Z4KIqBZqnFC0bdsWMpkMMplMYSKnCgYGBli1apWkg48bNw7btm3Djz/+CBMTEzFJMTMzg4GBAQoKChAeHo5+/frBwcEBV69excyZM2FtbY33339f0rGeR1tbm18oRERESqpxQpGZmQlBENC0aVP8/vvvsLGxEbfp6urC1tZW8hfy2rVrAUBhBkrg8eujFZNGXbhwAVu2bMG9e/fg4OAAPz8/7Ny5s9LEVERERFR3apxQODs7AwDKy8tVdvDnvaZnYGCAgwcPqux49GpQpmMtXxklIqodpV4bvXTpEhITE5Gbm1spwfjiiy9UEhjRiySbq0QSouSEP0RELyPJCcXGjRsxZswYWFtbw97eXuGvQZlMxoSCiIjoFSQ5oZg/fz4WLFiA6dOnqyMeIiIi0kCSh96+e/cuBgwYoI5YiIiISENJTigGDBiAQ4cOqSMWIkUymXJLPfcSfiQiIumPPORyOWbPno2UlBS0adOm0rwHEydOVFlwREREpBkkJxQbNmyAsbExkpKSkJSUpLBNJpMxoSAiInoFSU4oMjMz1REHERERaTDJfSgqlJSU4OLFi3j06JEq4yEiIiINJDmhePDgAYKDg2FoaIhWrVrh+vXrAB73nVi0aJHKAyQiIqL6T/IjjxkzZuCPP/5AYmIiAgMDxfKuXbtizpw5+Oyzz1QaIFF99apMSUxEVBOSE4r9+/dj586deOONNxRGyWzZsiWuXLmi0uCIiIhIM0h+5HHz5k3Y2tpWKi8sLFRqUiYiIiLSfJITig4dOuDAgQPiekUSsXHjRnh7e6suMiIiItIYkh95REREIDAwEOnp6Xj06BFWrFiBv/76C7/99lulcSmIiIjo1SD5DkWnTp1w8uRJPHjwAM2aNcOhQ4dgZ2eH3377De3bt1dHjERERFTPSbpDUVpaio8//hizZ8/G5s2b1RUTERERaRhJdygaNGiAffv2qSsWIiIi0lCSH3m8//772L9/vxpCISIiIk2l1GyjX375JU6ePIn27dvDyMhIYTsnByMiInr1SE4ovv32W5ibm+Ps2bM4e/aswjbONkpERPRq4myjREREVGucbZSIiIhqjbONEhERUa1JTiienG1UX19fLO/atSt27typ0uCIiIhIM3C2USIiIqo1zjZKREREtcbZRomIiKjWJCcUERERmDVrFsaMGSPONhoQEIDo6GgsWLBAclsdOnSAiYkJbG1t0adPH1y8eFGhjiAICA8Ph6OjIwwMDODr64u//vpLathERESkRkrNNpqcnKyS2UaTkpIwbtw4pKSkID4+Ho8ePUK3bt1QWFgo1vnqq6+wbNkyrF69GqdPn4a9vT0CAgJw//59qaETERGRmtSoU2ZoaCi+/PJLGBkZ4dixY+jUqZNKZhuNi4tTWI+KioKtrS3Onj2Lzp07QxAELF++HLNmzULfvn0BAJs3b4adnR22bduGTz75pNYxEBERUe3V6A7FqlWrUFBQAADw8/PDnTt31BJMXl4eAMDS0hLA41E5c3Jy0K1bN7GOnp4efHx8cPLkySrbKC4uRn5+vsJCRERE6lWjOxRNmjTBypUr0a1bNwiCgN9++w0WFhZV1u3cubNSgQiCgNDQULz11lto3bo1ACAnJwcAYGdnp1DXzs4O165dq7KdiIgIzJ07V6kYiIiISDk1SiiWLFmC0aNHIyIiAjKZDO+//36V9WQyGcrKypQKZPz48UhLS8OJEyeqbPdJgiBU+4rqjBkzEBoaKq7n5+fDyclJqZiIiIioZmqUUPTp0wd9+vRBQUEBTE1NcfHixSrHolDWhAkT8NNPP+HYsWNo1KiRWG5vbw/g8Z0KBwcHsTw3N7fSXYsKenp60NPTU1lsRERE9Hw16kMRGhqKwsJCGBsbIyEhAS4uLjAzM6tykUIQBIwfPx579+7F0aNH4eLiorDdxcUF9vb2iI+PF8tKSkqQlJSETp06SToWERERqY/kTpn+/v4q65Q5btw4/PDDD9i2bRtMTEyQk5ODnJwcPHz4EMDjRx2TJ0/GwoULsW/fPvz5558YMWIEDA0NMXToUJXEQERERLVXp50y165dCwDw9fVVKI+KisKIESMAANOmTcPDhw8xduxY3L17Fx07dsShQ4dgYmJS4+MQERGRetVpp0xBEJ5bRyaTITw8HOHh4TVul4iIiF6setEpk4iIiDSbpOnLn+yUqaMjeeZzIiIieknVKCvIz8+HqakpAKBdu3Z48OBBtXUr6hEREdGro0YJhYWFBbKzs2Frawtzc/MqB5WqGGxK2YGtSLXmypQbLXSOMEfFkRAR0augRgnF0aNHxfk1EhIS1BoQERERaZ4aJRQ+Pj5V/kxEREQE1DChSEtLq3GDHh4eSgdDREREmqlGCUXbtm0hk8meOSlXBfah0GzP+e+tUg2GEyEiopdcjYbezszMxD///IPMzEzExMTAxcUFa9aswfnz53H+/HmsWbMGzZo1Q0xMjLrjJSIionqoRnconJ2dxZ8HDBiAlStXokePHmKZh4cHnJycMHv2bPTp00flQRIREVH9VqM7FE+6cOFCpVlBgcczg6anp6skKCIiItIskhOKFi1aYP78+SgqKhLLiouLMX/+fLRo0UKlwREREZFmkDx+9rp169CrVy84OTnB09MTAPDHH39AJpMhNjZW5QESERFR/Sc5oXj99deRmZmJH374Af/5z38gCAIGDRqEoUOHwsjISB0xEhERUT2n1AxfhoaG+Pjjj1UdCxEREWkoyX0oiIiIiJ7GhIKIiIhqjQkFERER1RoTCiIiIqo1pTplAkBJSQlyc3NRXl6uUN64ceNaB0VERESaRXJCcfnyZYwcORInT55UKK+YOIyTgz3b8yZXq4rA2beIiKiek5xQjBgxAjo6OoiNjYWDg4NSX5BERET0cpGcUKSmpuLs2bNwd3dXRzxUBdlc6UlbOMJVHwgREVE1JHfKbNmyJW7duqWOWIiIiEhDSU4oFi9ejGnTpiExMRG3b99Gfn6+wkJERESvHsmPPLp27QoA6NKli0I5O2USERG9uiQnFAkJCeqIg4iIiDSY5ITCx8dHZQc/duwYlixZgrNnzyI7Oxv79u1Dnz59xO0jRozA5s2bFfbp2LEjUlJSVBYDERER1V6NEoq0tDS0bt0aWlpaSEtLe2ZdDw+PGh+8sLAQnp6e+Oijj9CvX78q6wQGBiIqKkpc19XVrXH7RERE9GLUKKFo27YtcnJyYGtri7Zt20Imk1U52JLUPhTdu3dH9+7dn1lHT08P9vb2NW6TiIiIXrwaJRSZmZmwsbERf36REhMTYWtrC3Nzc/j4+GDBggWwtbV9oTEQERHRs9UooXB2dq7yZ3Xr3r07BgwYAGdnZ2RmZmL27Nnw9/fH2bNnoaenV+U+xcXFKC4uFtfV9iorRwglIiISKT052IswaNAg8efWrVvDy8sLzs7OOHDgAPr27VvlPhEREZg7d+6LCpGIiIigYdOXOzg4wNnZGZcvX662zowZM5CXlycuWVlZLzBCIiKiV1O9vkPxtNu3byMrKwsODg7V1tHT06v2cQgRERGpR50mFAUFBfj777/F9czMTKSmpsLS0hKWlpYIDw9Hv3794ODggKtXr2LmzJmwtrbG+++/X4dRExER0dOUTihKSkqQm5uL8vJyhfLGjRvXuI0zZ87Az89PXA8NDQUABAUFYe3atbhw4QK2bNmCe/fuwcHBAX5+fti5cydMTEyUDZuIiIjUQHJCcfnyZYwcORInT55UKFdmLg9fX98qx7OocPDgQanhERERUR2QnFCMGDECOjo6iI2NhYODA2R8fZKIiOiVJzmhSE1NxdmzZ+Hu7q6OeIiIiEgDSX5ttGXLlrh165Y6YiEiIiINJTmhWLx4MaZNm4bExETcvn0b+fn5CgsRERG9eiQ/8ujatSsAoEuXLgrlynTKJCIiopeD5IQiISFBHXEQERGRBpOcUPj4+KgjDiIiItJgSg1sde/ePXz33XfIyMiATCZDy5YtMXLkSJiZmak6PiIiItIAkjtlnjlzBs2aNUNkZCTu3LmDW7duYdmyZWjWrBnOnTunjhiJiIionpN8h+LTTz/Fe++9h40bN0JH5/Hujx49QkhICCZPnoxjx46pPEgiIiKq3yQnFGfOnFFIJgBAR0cH06ZNg5eXl0qDIyIiIs0g+ZGHqakprl+/Xqk8KyuLk3YRERG9oiQnFIMGDUJwcDB27tyJrKws/Pvvv9ixYwdCQkIwZMgQdcRIRERE9ZzkRx5ff/01ZDIZPvzwQzx69AgA0KBBA4wZMwaLFi1SeYBERERU/0lOKHR1dbFixQpERETgypUrEAQBcrkchoaG6oiPiIiINIBS41AAgKGhIdq0aaPKWIiIiEhD1Sih6Nu3L6Kjo2Fqaoq+ffs+s+7evXtVEhgRERFpjholFGZmZpDJZAAev+VR8TMRERERUMOEIioqSvw5OjpaXbEQERGRhpL82qi/vz/u3btXqTw/Px/+/v6qiImIiIg0jOSEIjExESUlJZXKi4qKcPz4cZUERURERJqlxm95pKWliT+np6cjJydHXC8rK0NcXBwaNmyo2uiIiIhII9Q4oWjbti1kMhlkMlmVjzYMDAywatUqlQZHREREmqHGCUVmZiYEQUDTpk3x+++/w8bGRtymq6sLW1tbaGtrqyVIIiIiqt9qnFA4OzsDAMrLy9UWDBEREWkmyZ0yIyIisGnTpkrlmzZtwuLFi1USFBEREWkWyQnF+vXr4e7uXqm8VatWWLdunUqCIiIiIs0iOaHIycmBg4NDpXIbGxtkZ2erJCgiIiLSLJITCicnJyQnJ1cqT05OhqOjo6S2jh07hl69esHR0REymQz79+9X2C4IAsLDw+Ho6AgDAwP4+vrir7/+khoyERERqZnkhCIkJASTJ09GVFQUrl27hmvXrmHTpk349NNPMWrUKEltFRYWwtPTE6tXr65y+1dffYVly5Zh9erVOH36NOzt7REQEID79+9LDZuIiIjUSPL05dOmTcOdO3cwduxYccRMfX19TJ8+HTNmzJDUVvfu3dG9e/cqtwmCgOXLl2PWrFniDKebN2+GnZ0dtm3bhk8++URq6ERERKQmku9QyGQyLF68GDdv3kRKSgr++OMP3LlzB1988YVKA8vMzEROTg66desmlunp6cHHxwcnT56sdr/i4mLk5+crLERERKRekhOKCsbGxujQoQNat24NPT09VcYEAOLQ3nZ2dgrldnZ2CsN+Py0iIgJmZmbi4uTkpPLYiIiISJHkRx6FhYVYtGgRjhw5gtzc3EoDXf3zzz8qCw54fEfkSYIgVCp70owZMxAaGiqu5+fnM6kgIiJSM8kJRUhICJKSkjB8+HA4ODg888u9Nuzt7QFUfk01Nze30l2LJ+np6anljgkRERFVT3JC8euvv+LAgQN488031RGPyMXFBfb29oiPj0e7du0AACUlJUhKSuKInERERPWM5ITCwsIClpaWKjl4QUEB/v77b3E9MzMTqampsLS0ROPGjTF58mQsXLgQrq6ucHV1xcKFC2FoaIihQ4eq5PhERESkGpITii+//BJffPEFNm/eDENDw1od/MyZM/Dz8xPXK/o+BAUFITo6GtOmTcPDhw8xduxY3L17Fx07dsShQ4dgYmJSq+MSERGRaklOKJYuXYorV67Azs4OTZo0QYMGDRS2nzt3rsZt+fr6QhCEarfLZDKEh4cjPDxcaphERET0AklOKPr06aOGMIiIiEiTSU4o5syZo444iIiISIMpPbAVERERUQXJdyi0tLSeOfZEWVlZrQIiIiIizSM5odi3b5/CemlpKc6fP4/Nmzdj7ty5KguMiIiINIfkhKJ3796Vyvr3749WrVph586dCA4OVklgREREpDlU1oeiY8eOOHz4sKqaIyIiIg2ikoTi4cOHWLVqFRo1aqSK5oiIiEjDKDX09pOdMgVBwP3792FoaIgffvhBpcERERGRZpCcUCxfvlxhXUtLCzY2NujYsSMsLCxUFRcRERFpkBolFH379kV0dDRMTU0hk8kwaNAgThFOREREohr1oYiNjUVhYSEA4KOPPkJeXp5agyIiIiLNUqM7FO7u7pgxYwb8/PwgCAJ27doFU1PTKut++OGHKg2QiIiI6r8aJRTr1q1DaGgoDhw4AJlMhs8//7zK0TJlMhkTCiIioldQjRKKTp06ISUlBcDjTpiXLl2Cra2tWgMjIiIizSF5HIrMzEzY2NioIxYiIiLSUJJfG3V2dlZHHERERKTBOH05ERER1RoTCiIiIqo1JhRERERUa0olFI8ePcLhw4exfv163L9/HwBw48YNFBQUqDQ4IiIi0gySO2Veu3YNgYGBuH79OoqLixEQEAATExN89dVXKCoqwrp169QRJxEREdVjku9QTJo0CV5eXrh79y4MDAzE8vfffx9HjhxRaXBERESkGSTfoThx4gSSk5Ohq6urUO7s7Iz/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",
      "text/plain": [
       "<Figure size 600x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "#####    #Test for fixed small number of data\n",
    "sample_size = 30\n",
    "N = 10\n",
    "T = 20\n",
    "m_t = 10\n",
    "rep_num = 100\n",
    "risk_level = 0.2\n",
    "radius = 0.1\n",
    "True_phi = np.zeros((SS,AA,SS)) #True environment\n",
    "lamb_poi=3\n",
    "for i in range(SS):\n",
    "    for j in range(AA - np.max([S[i],0])):\n",
    "        for k in range(np.max([S[i],0])+j,np.max([S[i],0])+j+K+1):\n",
    "            demand_i = np.max([S[i],0])+j - S[k]\n",
    "            True_phi[i,j,k] = poisson.pmf(demand_i,lamb_poi)\n",
    "        True_phi[i,j]  = True_phi[i,j]/np.sum(True_phi[i,j])\n",
    "p_demand=np.zeros(AA)\n",
    "for i in range(AA):\n",
    "    p_demand[i] = poisson.pmf(i,lamb_poi)\n",
    "p_demand/=np.sum(p_demand)\n",
    "pol = run_simulation(risk_level,radius)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.10.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
