{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "78a475d7",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from joblib import Parallel, delayed\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.stats import truncnorm\n",
    "import pandas as pd\n",
    "from joblib import Parallel, delayed"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "ed0d3507",
   "metadata": {},
   "outputs": [],
   "source": [
    "np.random.seed(0)\n",
    "\n",
    "#Hole=[5,7,11,12]\n",
    "T=80 # Time Horizon\n",
    "# K=130\n",
    "N=1000 #  Iteration number of outer loop \n",
    "alpha=0.4*np.ones(N)\n",
    "epsilon_start = 1.0\n",
    "epsilon_end = 0.1\n",
    "epsilon_decay_rate = 0.998\n",
    "num_multi_rep=10\n",
    "#Hole=[5,6,7,17,18,19]\n",
    "\n",
    "# initial_gauss_set=[0,1,2,3,4,5,6,7,8,9]\n",
    "# safe_s0=[0,1,2,3,4,5,9,10,11,12,13,14]\n",
    "\n",
    "\n",
    "# 8*8 parameter\n",
    "# nA=8\n",
    "# nS = 64\n",
    "# nrow = ncol = 8\n",
    "# e_risk=1.5\n",
    "\n",
    "\n",
    "# 12*12 parameter\n",
    "nA=8\n",
    "nS = 144\n",
    "nrow = 12\n",
    "ncol = 12\n",
    "\n",
    "e_risk=0.75\n",
    "safe_s0=[0,1,2,3,4,5]\n",
    "initial_gauss_set=[0,1,2,3,4,5,12,13,14,15,16,17,24,25,26,27,28,29]\n",
    "\n",
    "# e_risk=0.75\n",
    "# safe_s0=[0,1,2,3,4,5,12,13,14,15,16,17]\n",
    "# initial_gauss_set=[0,1,2,3,4,5,12,13,14,15,16,17,24,25,26,27,28,29]\n",
    "\n",
    "# e_risk=0.75\n",
    "# safe_s0=[0,1,2,3,4,5,12,13,14,15,16,17,24,25,28,29]\n",
    "# initial_gauss_set=[0,1,2,3,4,5,12,13,14,15,16,17,24,25,26,27,28,29]\n",
    "# 16*16 parameter\n",
    "# nA=8\n",
    "# nS = 256\n",
    "# nrow = 16\n",
    "# ncol = 16\n",
    "# e_risk=0.75\n",
    "\n",
    "# safe_s0=[0,1,2,3,4,5,6,16,17,18,19,20,21,22,35,36,37,38]\n",
    "# initial_gauss_set=[0,1,2,3,4,16,17,18,19,20,32,33,34,35,36]\n",
    "\n",
    "c_Qlearning=20\n",
    "# gamma=0.97\n",
    "#n=10  sample number of transition kernel\n",
    " # data number for post##change every time\n",
    "\n",
    "# V=50 #number of comparasion\n",
    "# mt=30\n",
    "\n",
    "\n",
    "\"\"\"\"\"\n",
    "V=50 #number of comparasion\n",
    "mt=50\n",
    "K=40\n",
    "\"\"\"\"\"\n",
    "# 8*8 parameter\n",
    "# cost_s=0\n",
    "# cost_d=1\n",
    "# cost_h=2\n",
    "\n",
    "\n",
    "# 16*16 parameter\n",
    "cost_s=0\n",
    "cost_A=0.5\n",
    "cost_B=1\n",
    "cost_C=1.5\n",
    "cost_D=2\n",
    "# beta=0.7\n",
    "# nu=0.5\n",
    "# prior=np.ones(4)\n",
    "sigma_noise=0\n",
    "LEFT = 0\n",
    "DOWN = 1\n",
    "RIGHT = 2\n",
    "UP = 3\n",
    "#action in {(+-1,+-2),(+-2,+-1)}\n",
    "MAPS = {\n",
    "    \"4x4\": [\"SFFF\", \"FHFH\", \"FFFH\", \"HFFG\"],\n",
    "    \"5x5\": [\n",
    "        \"SFFFF\",\n",
    "        \"HHHFF\",\n",
    "        \"FFFFF\",\n",
    "        \"FFHHH\",\n",
    "        \"FFFFG\",\n",
    "    ],\n",
    "    \"8x8\": [\n",
    "        \"SBFFFFDD\",\n",
    "        \"DBBBFFBD\",\n",
    "        \"DDDDBFBD\",\n",
    "        \"DDDBFFBD\",\n",
    "        \"DBBFBBBB\",\n",
    "        \"DBFBDDDD\",\n",
    "        \"BBFFBBBD\",\n",
    "        \"DDBFFFFG\",\n",
    "    ],\n",
    "    \"12x12\":[\n",
    "        \"SFFFFFFFFFFF\",\n",
    "        \"AAAAAAAAAAFF\",\n",
    "        \"BBCCBBBBBAFF\",\n",
    "        \"BAAAAAAAAAFF\",\n",
    "        \"AFFFFFFFFFFF\",\n",
    "        \"FFFFFFFFFFFF\",\n",
    "        \"FFAAAAAAAAAA\",\n",
    "        \"FABBBBAABBBB\",\n",
    "        \"FAAACCAAABCC\",\n",
    "        \"FFAABBAAAABB\",\n",
    "        \"FFFFAAFFFFAA\",\n",
    "        \"FFFFFFFFFFFG\",\n",
    "    ],\n",
    "    \"16x16\":[\n",
    "        \"SFFFFFFFFFFFFFFF\",\n",
    "        \"AAAFFFFFFFFFFFFF\",\n",
    "        \"BBBAFFFFFFAAAFFF\",\n",
    "        \"CDCBAFFFFABBBAFF\",\n",
    "        \"BBBAFFFFABBAAFFF\",\n",
    "        \"AAAFFFFAABAAFFFF\",\n",
    "        \"FFFFFAAABBAFFFFF\",\n",
    "        \"FFFFABBBBAFFFFFF\",\n",
    "        \"FFFABCDCBAAAAAAA\",\n",
    "        \"FFFABBBBBAABBBBB\",\n",
    "        \"FFFAAAAAAFABCCCC\",\n",
    "        \"FFFFFFFFFFFABCDD\",\n",
    "        \"AAFFFFFFFFFFABCD\",\n",
    "        \"ABAFFFFAAFFFABCC\",\n",
    "        \"FAFFFFABBAFFFABB\",\n",
    "        \"FFFFFABBBBAFFFFG\",\n",
    "    ],\n",
    "}\n",
    "desc = MAPS[\"12x12\"]\n",
    "desc = np.asarray(desc, dtype=\"c\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "70ebd175",
   "metadata": {},
   "outputs": [],
   "source": [
    "def to_row_col(state):\n",
    "    return int((state - state% ncol)/ ncol), state% ncol\n",
    "        \n",
    "def to_s(row, col):\n",
    "    return row * ncol + col\n",
    "\n",
    "def inc(row, col, a):\n",
    "    if a == LEFT:\n",
    "        col = max(col - 1, 0)\n",
    "    elif a == DOWN:\n",
    "        row = min(row + 1, nrow - 1)\n",
    "    elif a == RIGHT:\n",
    "        col = min(col + 1, ncol - 1)\n",
    "    elif a == UP:\n",
    "        row = max(row - 1, 0)\n",
    "    return (row, col)\n",
    "\n",
    "\n",
    "def inc_random(row, col, a):#a=(+-1,+-2)\n",
    "    choice=np.zeros(2)\n",
    "    if a[0]>=0:\n",
    "        choice[0]=RIGHT\n",
    "    else:\n",
    "        choice[0]=LEFT\n",
    "    if a[1]>=0:\n",
    "        choice[1]=UP\n",
    "    else:\n",
    "        choice[1]=DOWN\n",
    "    action=np.random.choice(choice,p=np.absolute(a)/3)\n",
    "    return inc(row, col,action)\n",
    "\n",
    "def action_to_number(a):\n",
    "    if abs(abs(a[0])-1)<=1e-5:\n",
    "        i=0\n",
    "    else: \n",
    "        i=4\n",
    "    if a[0]>=0 and a[1]>=0:\n",
    "        j=0\n",
    "    elif    a[0]>=0 and a[1]<0:\n",
    "        j=1\n",
    "    elif    a[0]<0 and a[1]>=0:\n",
    "        j=2\n",
    "    else:j=3\n",
    "    return i+j\n",
    "\n",
    "def number_to_action(n):\n",
    "    i,j=int((n - n% 4)/ 4), n% 4\n",
    "    if i==0 and j==0:\n",
    "        h=np.array([1,2])\n",
    "    elif i==0 and j==1:\n",
    "        h=np.array([1,-2])\n",
    "    elif i==0 and j==2:\n",
    "        h=np.array([-1,2])\n",
    "    elif i==0 and j==3:\n",
    "        h=np.array([-1,-2])        \n",
    "    elif i==1 and j==0:\n",
    "        h=np.array([2,1])\n",
    "    elif i==1 and j==1:\n",
    "        h=np.array([2,-1])\n",
    "    elif i==1 and j==2:\n",
    "        h=np.array([-2,1])\n",
    "    else:\n",
    "        h=np.array([-2,-1])  \n",
    "    return h \n",
    "\n",
    "def order_to_state(n):\n",
    "    return [int((n - n%  nrow)/ nrow), n%  nrow]\n",
    "\n",
    "\n",
    "def compare_set(A,B):\n",
    "    l1=len(A)\n",
    "    l2=len(B)\n",
    "    if l1!=l2:\n",
    "        return False   \n",
    "    else: \n",
    "        for i in range(l1):\n",
    "            if A[i]!=B[i]:\n",
    "                return False\n",
    "    return True\n",
    "\n",
    "def sa_to_s_random(s,a):\n",
    "    row,col=to_row_col(s)\n",
    "    (new_row,new_col)=inc_random(row,col,number_to_action(a))\n",
    "    return int(to_s(new_row,new_col))\n",
    "\n",
    "def stateset_to_grid(S):\n",
    "    grid=2*np.ones((nrow,ncol))\n",
    "    for s in S:\n",
    "        grid[to_row_col(s)[0],to_row_col(s)[1]]=0\n",
    "    \n",
    "    return grid\n",
    "\n",
    "def plot_set(S,color='Blues'):\n",
    "    plt.rcParams['font.sans-serif'] = ['SimHei']  \n",
    "    plt.rcParams['axes.unicode_minus'] = False  \n",
    "    plt.figure(figsize=(8, 8))\n",
    "    plt.imshow(S,cmap=color) \n",
    "    plt.colorbar()\n",
    "    plt.show()\n",
    "\n",
    "def plot_state_set(S,color='Blues'):\n",
    "    plot_set(stateset_to_grid(S),color)\n",
    "    \n",
    "\n",
    "def plot_route(dataset,color='Blues'):\n",
    "    S=[]\n",
    "    l=len(dataset)\n",
    "    if l !=0:\n",
    "        for i in range(l):\n",
    "            S.append(dataset[i][0])\n",
    "            if dataset[i][0]==nS-1:\n",
    "                break\n",
    "        if dataset[l-1][0]!=nS-1:     \n",
    "            S.append(dataset[l-1][2])\n",
    "    plot_state_set(S,color)\n",
    "\n",
    "def distance_p2s(s,S):\n",
    "    g=np.zeros(len(S))\n",
    "    for i in range(len(S)):\n",
    "        r1,c1=to_row_col(s)\n",
    "        r2,c2=to_row_col(S[i])\n",
    "        g[i]=np.sqrt((r1-r2)**2+(c1-c2)**2)\n",
    "    return np.min(g)\n",
    "\n",
    "def distance_p2p(s1,s2):\n",
    "    r1,c1=to_row_col(s1)\n",
    "    r2,c2=to_row_col(s2)\n",
    "    np.sqrt((r1-r2)**2+(c1-c2)**2)\n",
    "    return np.min(np.sqrt((r1-r2)**2+(c1-c2)**2)) \n",
    "\n",
    "def gauss_fit_initial(mu_input,cov_input,dataset):\n",
    "    mu=np.copy(mu_input)\n",
    "    cov=np.copy(cov_input)\n",
    "    new_mu=np.zeros(nS)\n",
    "    new_cov=np.zeros((nS,nS))\n",
    "    for i in range(len(dataset)):\n",
    "        s=dataset[i][0]\n",
    "        row,col=to_row_col(s)\n",
    "        y=dataset[i][3]\n",
    "        new_mu=mu+cov[:,s]/(cov[s,s])*(y-mu[s])#+sigma_noise)\n",
    "        new_mu[s]=y\n",
    "        for j in range(nS):\n",
    "            for k in range(nS):\n",
    "                new_cov[j,k]=cov[j,k]-cov[j,s]*cov[k,s]/(cov[s,s])#+sigma_noise)\n",
    "                    #print(str(i)+','+str(j)+','+str(k)+':'+str(new_cov[j,k]-cov[j,k]))\n",
    "        new_cov[:,s]=0\n",
    "        new_cov[s,:]=0\n",
    "        mu=np.copy(new_mu)\n",
    "        cov=np.copy(new_cov)\n",
    "\n",
    "    return mu, cov\n",
    "\n",
    "def gauss_fit(mu_input,cov_input,dataset,state_visit):\n",
    "    mu=np.copy(mu_input)\n",
    "    cov=np.copy(cov_input)\n",
    "    new_mu=np.zeros(nS)\n",
    "    new_cov=np.zeros((nS,nS))\n",
    "    visit_set=np.copy(state_visit)\n",
    "    for s in initial_gauss_set:\n",
    "        visit_set[to_row_col(s)[0],to_row_col(s)[1]]=1\n",
    "    for i in range(len(dataset)):\n",
    "        s=dataset[i][0]\n",
    "        row,col=to_row_col(s)\n",
    "        if  visit_set[row,col]==0:\n",
    "            visit_set[row,col]+=1\n",
    "            y=dataset[i][3]\n",
    "            new_mu=mu+cov[:,s]/(cov[s,s])*(y-mu[s])#+sigma_noise)\n",
    "            new_mu[s]=y\n",
    "            for j in range(nS):\n",
    "                for k in range(nS):\n",
    "                    new_cov[j,k]=cov[j,k]-cov[j,s]*cov[k,s]/(cov[s,s])#+sigma_noise)\n",
    "                    #print(str(i)+','+str(j)+','+str(k)+':'+str(new_cov[j,k]-cov[j,k]))\n",
    "                    #print(str(i)+','+str(j)+','+str(k)+':'+str(cov[j,s]*cov[s,k]/(cov[s,s]+sigma_noise)))\n",
    "            new_cov[:,s]=0\n",
    "            new_cov[s,:]=0\n",
    "            mu=np.copy(new_mu)\n",
    "            cov=np.copy(new_cov)\n",
    "\n",
    "    return mu, cov\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "def sa_to_order_random(s,a):\n",
    "    return s*nA+action_to_number(a)\n",
    "\n",
    "\n",
    "\n",
    "def gauss_fit_random(mu_input,cov_input,dataset):\n",
    "    mu=np.copy(mu_input)\n",
    "    cov=np.copy(cov_input)\n",
    "    new_mu=np.zeros(nS)\n",
    "    new_cov=np.zeros((nS,nS))\n",
    "    for i in range(len(dataset)):\n",
    "        s=dataset[i][0]\n",
    "        a=dataset[i][1]\n",
    "        y=dataset[i][3]\n",
    "        order_sa=sa_to_order_random(s,number_to_action(a))\n",
    "        new_mu=mu+cov[:,order_sa]*(y-mu)/(cov[order_sa,order_sa]+sigma_noise)\n",
    "        for j in range(nS):\n",
    "            for k in range(nS):\n",
    "                for c in range(nA):\n",
    "                    for d in range(nA):\n",
    "                        order_1=sa_to_order_random(j,number_to_action(c))\n",
    "                        order_2=sa_to_order_random(k,number_to_action(d))\n",
    "                        new_cov[order_1,order_2]=cov[order_1,order_2]-cov[order_1,order_sa]*cov[order_sa,order_2]/(cov[order_sa,order_sa]+sigma_noise)\n",
    "                #print(str(i)+','+str(j)+','+str(k)+':'+str(new_cov[j,k]-cov[j,k]))\n",
    "                #print(str(i)+','+str(j)+','+str(k)+':'+str(cov[j,s]*cov[s,k]/(cov[s,s]+sigma_noise)))\n",
    "        mu=np.copy(new_mu)\n",
    "        cov=np.copy(new_cov)\n",
    "    return mu, cov\n",
    "#     return 0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c6779553",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "6c830c53",
   "metadata": {},
   "outputs": [],
   "source": [
    "map_type=np.zeros((nrow,ncol))\n",
    "\n",
    "s_A=[]\n",
    "s_B=[]\n",
    "s_C=[]\n",
    "s_D=[]\n",
    "for s in range(nS):\n",
    "    row,col=to_row_col(s)\n",
    "    if desc[row,col]==b'A':\n",
    "        map_type[row,col]=cost_A\n",
    "        s_A.append(s)\n",
    "    elif desc[row,col]==b'B':\n",
    "        map_type[row,col]=cost_B\n",
    "        s_B.append(s)\n",
    "    elif desc[row,col]==b'C':\n",
    "        map_type[row,col]=cost_C\n",
    "        s_C.append(s)\n",
    "    elif desc[row,col]==b'D':\n",
    "        map_type[row,col]=cost_D\n",
    "        s_D.append(s)\n",
    "\n",
    "map_type[0,0]=0\n",
    "map_type[nrow-1,ncol-1]=0\n",
    "\n",
    "true_cost=np.zeros((nS,nA))\n",
    "for s in range(nS):\n",
    "    row,col=to_row_col(s)\n",
    "\n",
    "\n",
    "    if desc[row,col]==b'A':\n",
    "        true_cost[s]=cost_A\n",
    "    elif desc[row,col]==b'B':\n",
    "        true_cost[s]=cost_B\n",
    "    elif desc[row,col]==b'C':\n",
    "        true_cost[s]=cost_C\n",
    "    elif desc[row,col]==b'D':\n",
    "        true_cost[s]=cost_D\n",
    "\n",
    "r=np.zeros((nS,nA))\n",
    "r[nS-1,1]=1\n",
    "\n",
    "def count_route_holes(dataset):\n",
    "    num=0\n",
    "    l=len(dataset)\n",
    "    for i in range(l):\n",
    "        if map_type[to_row_col(dataset[i][0])]>e_risk:\n",
    "            num=num+1\n",
    "    if l>0 and map_type[to_row_col(dataset[l-1][2])]>e_risk:\n",
    "            num=num+1\n",
    "    return num"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "d649d53c",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "s_next_old=np.zeros((nS,4))\n",
    "for i in range(nS):\n",
    "    for j in range(4):\n",
    "        new_row,new_col=inc(to_row_col(i)[0],to_row_col(i)[1],j)\n",
    "        s_next_old[i,j]=int(to_s(new_row,new_col))\n",
    "        \n",
    "def Qlearning(rtilde,T,n_visit,total_visit,weight,set,unsafe_set):\n",
    "    Q=np.zeros((T,nS,nA))\n",
    "    Q[T-1]=rtilde\n",
    "    for t in range(T-2,-1,-1):  \n",
    "        Q_max=np.zeros((nS,nA))\n",
    "        for s in range(nS):\n",
    "            for a in range(nA):\n",
    "                prob=number_to_action(a)/3\n",
    "                choice=np.zeros(2)\n",
    "                if prob[0]>=0:\n",
    "                    choice[0]=RIGHT\n",
    "                else:\n",
    "                    choice[0]=LEFT\n",
    "                if prob[1]>=0:\n",
    "                    choice[1]=UP\n",
    "                else:\n",
    "                    choice[1]=DOWN\n",
    "                Q_max[s,a]+=abs(prob[0])*np.max(Q[t+1,int(s_next_old[s,int(choice[0])]),:])\n",
    "                Q_max[s,a]+=abs(prob[1])*np.max(Q[t+1,int(s_next_old[s,int(choice[1])]),:])\n",
    "        Q[t]=rtilde+Q_max\n",
    "    pi=np.zeros((nS,nA))\n",
    "    bound_set=[]\n",
    "    for s in range(nS):\n",
    "        if distance_p2s(s,set)<1.1:\n",
    "            bound_set.append(s)\n",
    "    for s in set:\n",
    "        for a in range(nA):\n",
    "            if n_visit[s,a]==0:\n",
    "                if unsafe_set[s,a]==0:\n",
    "                    Q[0,s,a]+=weight*np.sqrt(np.log(total_visit)/(n_visit[s,a]+1))\n",
    "    for s in range(nS):\n",
    "        pi[s,np.argmax(Q[0,s])]=1\n",
    "    return pi"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "84eaaa8f",
   "metadata": {},
   "outputs": [],
   "source": [
    "def final_route(pi):\n",
    "    s=0\n",
    "    a=np.random.choice([0,1,2,3,4,5,6,7],p=pi[s])\n",
    "    n_violate=0\n",
    "    final_data=[]\n",
    "    t=0\n",
    "    #while (s_next[s,a] in S and t<T and true_cost[s,a]*(1+noise) <e_risk ):\n",
    "    while ( t<T and n_violate<0.5 ):\n",
    "        row,col=to_row_col(s)\n",
    "        new_row,new_col=inc_random(row,col,number_to_action(a))\n",
    "        new_s=to_s(new_row,new_col)\n",
    "        # noise=truncnorm.rvs(-0.3, 0.3, size=1)[0]\n",
    "        # noise=0  \n",
    "        #if true_cost[s,a]*(1+noise)>e_risk:\n",
    "        if map_type[to_row_col(new_s)]>e_risk:\n",
    "            n_violate+=1\n",
    "        final_data.append([s,a,new_s,true_cost[s,a]])\n",
    "        #if true_cost[s,a]!=0:\n",
    "            #dataset.append([s,a,new_s,true_cost[s,a]+noise])\n",
    "        #else:\n",
    "            #dataset.append([s,a,new_s,0])\n",
    "        #dataset.append([s,a,new_s,(true_cost[s,a]!=0)*(true_cost[s,a]+noise)])\n",
    "        s=new_s\n",
    "\n",
    "        #state_visit[n+1,to_row_col(s)[0],to_row_col(s)[1]]+=1\n",
    "        a=np.random.choice([0,1,2,3,4,5,6,7],p=pi[s])\n",
    "        t+=1\n",
    "    return final_data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "c186a1ce",
   "metadata": {},
   "outputs": [],
   "source": [
    "def one_rep():\n",
    "    pi=np.zeros((nS,nA))    \n",
    "    count_violation=0\n",
    "    S0=[]\n",
    "    policy_list=[]\n",
    "    unsafe_sa=np.zeros((nS,nA))\n",
    "    unsafe_set=-np.ones(nS)\n",
    "    for i in range(nS):\n",
    "        if map_type[to_row_col(i)[0],to_row_col(i)[1]]>e_risk:\n",
    "            unsafe_set[i]+=1\n",
    "    state_visit=np.zeros((N+1,nrow,ncol))\n",
    "    sa_visit=np.zeros((N+1,nS,nA))\n",
    "    for s in [0,1,2,3,4,5,12,13,14,15,16,17,24,27,28,29]:\n",
    "        S0.append([s,0])\n",
    "        S0.append([s,4])\n",
    "        pi[s,1]=0.5\n",
    "        pi[s,5]=0.5\n",
    "    for s in [25,26]:\n",
    "        S0.append([s,2])\n",
    "        S0.append([s,6])\n",
    "        pi[s,2]=0.5\n",
    "        pi[s,6]=0.5\n",
    "    pi[nS-1]=np.array([0,1,0,0,0,0,0,0])    \n",
    "\n",
    "    S_list=[]\n",
    "    data_list=[]\n",
    "    S0=np.copy(safe_s0)\n",
    "    S_list.append(S0)\n",
    "    policy_list.append(pi)\n",
    "    S=np.copy(S_list[-1])\n",
    "    dataset=[]\n",
    "\n",
    "    t=0\n",
    "    s=0     #first episode\n",
    "    a=np.random.choice([0,1,2,3,4,5,6,7],p=pi[s])\n",
    "    mu_0=np.zeros(nS)\n",
    "    cov_0=np.zeros((nS,nS))\n",
    "    cov_list=np.zeros((N+1,nS,nS))\n",
    "    sigma_list=np.zeros((N+1,nrow,ncol))\n",
    "    y_predict_list=np.zeros((N+1,nrow,ncol))\n",
    "\n",
    "\n",
    "    for  i in range(nS):\n",
    "        for j in range(nS):\n",
    "            cov_0[i,j]=np.exp(-distance_p2p(i,j)**2/6)\n",
    "    total_visit=1\n",
    "    n_violate=0\n",
    "\n",
    "    while (s in S and t<T and n_violate<1.5):\n",
    "        row,col=to_row_col(s)\n",
    "        state_visit[0,row,col]+=1\n",
    "        sa_visit[0,s,a]+=1\n",
    "        new_row,new_col=inc_random(row,col,number_to_action(a))\n",
    "        new_s=to_s(new_row,new_col)\n",
    "        if map_type[new_row,new_col]>e_risk:\n",
    "            n_violate+=1\n",
    "            count_violation+=1\n",
    "            unsafe_sa[s,a]+=1\n",
    "            unsafe_set[new_s]+=1\n",
    "        dataset.append([s,a,new_s,true_cost[s,a]])\n",
    "        s=new_s\n",
    "        total_visit+=1\n",
    "        a=np.random.choice([0,1,2,3,4,5,6,7],p=pi[s])\n",
    "        t+=1\n",
    "\n",
    "    data_list.append(dataset)\n",
    "    data_len=len(dataset)\n",
    "    dataset_0=[]\n",
    "    for s in initial_gauss_set:\n",
    "        dataset_0.append([s,0,0,true_cost[s,0]])\n",
    "\n",
    "    y_predict,cov=gauss_fit_initial(mu_0,cov_0,dataset_0) #initial GP\n",
    "    new_y,new_cov=gauss_fit(y_predict,cov,dataset,np.zeros((nrow,ncol)))\n",
    "    y_predict=np.copy(new_y)\n",
    "    cov=np.copy(new_cov)\n",
    "    sigma=np.diag(cov)\n",
    "    visit_set=[]\n",
    "    visit_set_list=[]\n",
    "    for s in range(nS):\n",
    "        if state_visit[0,to_row_col(s)[0],to_row_col(s)[1]]>0:\n",
    "            visit_set.append(s)\n",
    "    visit_set_list.append(visit_set)\n",
    "    y_predict_list[0]=np.copy(y_predict.reshape(nrow,ncol))\n",
    "    sigma_list[0]=np.copy(sigma.reshape(nrow,ncol))\n",
    "\n",
    "\n",
    "    for n in range(N):\n",
    "        state_visit[n+1]=np.copy(state_visit[n])\n",
    "        sa_visit[n+1]=np.copy(sa_visit[n])\n",
    "        B=np.zeros(nS)\n",
    "        for i in range(nS):\n",
    "            B[i]=y_predict[i]+alpha[n]*sigma[i]\n",
    "            \n",
    "        new_S=np.copy(np.where(B<=e_risk)[0])\n",
    "        test_bool=compare_set(new_S,S)\n",
    "        if test_bool:\n",
    "            test=y_predict+alpha[n]*sigma\n",
    "            for s in S:\n",
    "                test[s]=1e6\n",
    "            for s in visit_set:\n",
    "                test[s]=1e6\n",
    "            for s in range(nS):\n",
    "                if distance_p2s(s,S)>1.1:\n",
    "                    test[s]=1e6\n",
    "            min_index=np.argmin(test)\n",
    "            if test[min_index] !=1e6:\n",
    "                B[min_index]=0\n",
    "            S=np.copy(np.where(B<=e_risk)[0])\n",
    "        else: \n",
    "            S=np.copy(new_S)\n",
    "        rtilde=np.zeros((nS,nA))\n",
    "        S_list.append(S)\n",
    "\n",
    "        for s in range(nS):\n",
    "            for a in range(nA):\n",
    "                if s in S:\n",
    "                    rtilde[s,a]=r[s,a]\n",
    "                # else: \n",
    "                #     rtilde[s,a]=-T\n",
    "                if unsafe_set[s]>0:\n",
    "                    rtilde[s,a]=-T*T\n",
    "\n",
    "        rtilde[nS-2,5]=T\n",
    "        rtilde[nS-1,0]=-T\n",
    "        rtilde[nS-1,2]=-T\n",
    "        rtilde[nS-1,3]=-T\n",
    "        rtilde[nS-1,4]=-T\n",
    "        rtilde[nS-1,6]=-T\n",
    "        rtilde[nS-1,7]=-T\n",
    "        rtilde[nS-1,1]=T\n",
    "        rtilde[nS-1,5]=T\n",
    "        #Q-learning \n",
    "        pi=Qlearning(rtilde,T,sa_visit[n+1],total_visit,c_Qlearning,S,unsafe_sa)\n",
    "        pi[nS-1]=np.array([0,1,0,0,0,0,0,0])    \n",
    "        policy_list.append(pi)\n",
    "        dataset=[]\n",
    "        t=0\n",
    "        #random start\n",
    "        start_set=[]\n",
    "        unvisited_set=[]\n",
    "        for s in range(nS):\n",
    "            row,col=to_row_col(s)\n",
    "            if map_type[row,col]<=e_risk and state_visit[n,row,col]>0:\n",
    "                start_set.append(s)\n",
    "            if state_visit[n,row,col]==0:\n",
    "                unvisited_set.append(s)\n",
    "        s=np.random.choice(start_set,p=np.ones(len(start_set))/len(start_set))\n",
    "        a=np.random.choice([0,1,2,3,4,5,6,7],p=pi[s])\n",
    "        n_violate=0\n",
    "\n",
    "        while (s in S and t<T and n_violate<1.5 ):\n",
    "            row,col=to_row_col(s)\n",
    "            state_visit[n+1,row,col]+=1\n",
    "            sa_visit[n+1,s,a]+=1\n",
    "            total_visit+=1\n",
    "            new_row,new_col=inc_random(row,col,number_to_action(a))\n",
    "            new_s=to_s(new_row,new_col)\n",
    "            if map_type[ new_row,new_col]>e_risk:\n",
    "                n_violate+=1\n",
    "                count_violation+=1\n",
    "                unsafe_sa[s,a]+=1\n",
    "                unsafe_set[new_s]+=1\n",
    "            dataset.append([s,a,new_s,true_cost[s,a]])\n",
    "            s=new_s\n",
    "                        \n",
    "            epsilon = max(epsilon_end, epsilon_start * (epsilon_decay_rate ** n))\n",
    "            if np.random.random() > epsilon:\n",
    "                a=np.random.choice([0,1,2,3,4,5,6,7],p=pi[s])\n",
    "            else:\n",
    "                possible_action=np.ones(nA)\n",
    "                for i in range(nA):\n",
    "                    if unsafe_sa[s,i]>0:\n",
    "                        possible_action[i]=0  \n",
    "                a=np.random.choice([0,1,2,3,4,5,6,7],size=1,p=possible_action/np.sum(possible_action))[0]\n",
    "                # a=np.random.choice([0,1,2,3,4,5,6,7],p=pi[s])\n",
    "            t+=1\n",
    "        row,col=to_row_col(s)\n",
    "        state_visit[n+1,row,col]+=1\n",
    "        data_list.append(dataset)\n",
    "        if len(dataset)==0: print('No data')\n",
    "        else:\n",
    "            new_y,new_cov=gauss_fit(y_predict,cov,dataset,state_visit[n])\n",
    "            y_predict=np.copy(new_y)\n",
    "            cov=np.copy(new_cov)\n",
    "            sigma=np.diag(cov)\n",
    "\n",
    "        y_predict_list[n+1]=np.copy(y_predict.reshape(nrow,ncol))\n",
    "        sigma_list[n+1]=np.copy(sigma.reshape(nrow,ncol))\n",
    "        visit_set=[]\n",
    "        for s in range(nS):\n",
    "            if state_visit[n+1,to_row_col(s)[0],to_row_col(s)[1]]>0:\n",
    "                visit_set.append(s)\n",
    "        visit_set_list.append(visit_set)  \n",
    "\n",
    "    final_state_visit=np.copy(state_visit[N])\n",
    "    total_data_number=0\n",
    "    for i in range(len(data_list)):\n",
    "        total_data_number+=len(data_list[i])\n",
    "\n",
    "    total_violation=0\n",
    "    for s in range(nS):\n",
    "        row,col=to_row_col(s)\n",
    "        for a in range(nA):\n",
    "            if    map_type[row,col]>e_risk:\n",
    "                total_violation+=np.sum(sa_visit[N,s,a])\n",
    "    return final_state_visit,total_data_number,total_violation,pi,count_violation\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "a49eb1f3",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "final_state_multi=np.zeros((num_multi_rep,nrow,ncol))\n",
    "total_data_number_multi=np.zeros(num_multi_rep)\n",
    "total_violation_multi=np.zeros(num_multi_rep)\n",
    "pi_multi=np.zeros((num_multi_rep,nS,nA))\n",
    "count_vio_multi=np.zeros(num_multi_rep)\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "1f195398",
   "metadata": {},
   "outputs": [],
   "source": [
    "# for i in range(num_multi_rep):\n",
    "#     test1,test2,test3,test4,test5=one_rep()\n",
    "#     final_state_multi[i]=np.copy(test1)\n",
    "#     total_data_number_multi[i]=np.copy(test2)\n",
    "#     total_violation_multi[i]=np.copy(test3)\n",
    "#     pi_multi[i]=np.copy(test4)\n",
    "#     count_vio_multi[i]=np.copy(test5)\n",
    "\n",
    "np.random.seed(0)\n",
    "mul_calls = [delayed(one_rep)() for rep in range(num_multi_rep)]\n",
    "result_mul_delayed = Parallel(n_jobs=-3)(mul_calls)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "dcaa5e80",
   "metadata": {},
   "outputs": [],
   "source": [
    "for i in range(num_multi_rep):\n",
    "    final_state_multi[i]=np.copy(result_mul_delayed[i][0])\n",
    "    total_data_number_multi[i]=np.copy(result_mul_delayed[i][1])\n",
    "    total_violation_multi[i]=np.copy(result_mul_delayed[i][2])\n",
    "    pi_multi[i]=np.copy(result_mul_delayed[i][3])\n",
    "    count_vio_multi[i]=np.copy(result_mul_delayed[i][4])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "69e47025",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(array([0.00052131, 0.00027657, 0.00043479, 0.00013942, 0.00019506,\n",
       "        0.00088936, 0.00031735, 0.00219144, 0.00024462, 0.00018158]),\n",
       " array([121., 116., 114., 134., 108., 100., 122., 110., 115.,  96.]))"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "total_violation_multi/total_data_number_multi,count_vio_multi"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bcc6a941",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(0.004281075676406123, 4.18928483576542e-06)"
      ]
     },
     "execution_count": 17,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "ratio=count_vio_multi/total_data_number_multi\n",
    "np.mean(ratio),np.var(ratio)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "a1fc11d0",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.000695004737220271"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "a=4.8303158475861796e-06\n",
    "np.sqrt(a)/np.sqrt(10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "7e96df41",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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iu5lnIV9kyV3AAA5IrkDADLLUihJkjsAgBzR3AEA5IjmDgDIrsL/WQ6lEkecwLDsfffdFzNnzoy6uro477zzYvv27RERsW3btujo6IgzzjgjVqxYEcPDwyO/57HHHotzzz03WltbY82aNeO6n+YOAOAk2blzZ1xxxRVx0003xfPPPx9nn312XHnllTE0NBRLliyJCy64IJ566qno6emJDRs2REREX19fLF26NLq6umLz5s2xcePG2LRpU8n31NwBAJlVE4WKH+Oxffv2uOmmm+KjH/1onHnmmfHpT386tm7dGvfff3/09/fHmjVrYtasWbFq1aq44447IiJi48aNMW3atFi5cmXMnj07rrvuupFzpTBbFgBgnIrF4qjXDQ0N0dDQkPjc4sWLR71+5plnYvbs2dHd3R2dnZ0xceLEiIiYO3du9PT0REREd3d3zJ8/Pwqvrql34YUXxle+8pWSa5PcAQCMU1tbWzQ3N48cq1evHvP3HDhwIL7xjW/E8uXLo1gsRnt7+8i5QqEQtbW1sWfPnsS5pqam2L17d8m1Se4AgMxKa2/Z3t7eaGpqGnn/WKndf3f99dfH6aefHldeeWV89atfTfyeCRMmxODgYNTV1Y06d/T9UmnuAADGqampaVRzN5ZHHnkkbr311njiiSfitNNOi5aWlti2bduozwwMDER9fX20tLREX19f4v1SGZYFADLr6A4VlTzGa9euXdHV1RW33nprzJkzJyIiOjo6YvPmzaM+MzQ0FC0tLYlzW7dujbPOOqv072T8JQIAUIp9+/bF4sWL4+KLL45ly5bF3r17Y+/evTFv3rwoFotx5513RkTEqlWr4qKLLora2tpYunRpPP744/HQQw/FwYMH4+abb44FCxaUfE/DsgBAZtUUClFTwYfuxnuvBx98MHp6eqKnpyfWrVs38v6uXbti/fr10dXVFStWrIiampp49NFHIyKitbU11q5dGwsXLozGxsaYPHnyyBp4pdDcAQCcJBdffPGonSdea8aMGbFz587YsmVLdHZ2xpQpU0bOLV++PBYsWBA7duyIefPmRWNjY8n31NwBAKRk6tSpsWjRomOea29vH7UkSqk0dwBAZqW1FEo1M6ECACBHJHcAQGbVRIUnVIxzb9k0SO4AAHJEcwcAkCOGZQGAzDKhIklyBwCQI5I7ACCzaqKySVUWUrEs1AgA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",
      "text/plain": [
       "<Figure size 800x800 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plot_set(np.minimum(np.mean(final_state_multi, axis=0),500))"
   ]
  }
 ],
 "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.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
