{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fd90b39c-362d-4e61-be85-bc4b07a1ee3e",
   "metadata": {},
   "outputs": [],
   "source": [
    "import tensorflow as tf \n",
    "from tensorflow import keras\n",
    "from tensorflow.keras import layers\n",
    "import torch\n",
    "import matplotlib.pyplot as plt\n",
    "from numpy import linalg as LA\n",
    "from scipy import stats\n",
    "from functools import partial\n",
    "%matplotlib inline\n",
    "import numpy as np\n",
    "from tensorflow.keras.layers import Input, Dense, Concatenate, Add\n",
    "from tensorflow.keras import Model\n",
    "from tensorflow.keras import regularizers\n",
    "import time "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "70176290-386e-40bd-9c82-78670710ee8a",
   "metadata": {},
   "outputs": [],
   "source": [
    "def cart(x,y):\n",
    "    temp = np.zeros([x.shape[0]*y.shape[0],x.shape[1]+y.shape[1]])\n",
    "    #num1 = y.shape[1]\n",
    "    j=0\n",
    "    num2 = x.shape[0]\n",
    "    num1 = x.shape[1]\n",
    "    for i in range(temp.shape[0]):\n",
    "        if i!=0 and i%num2 == 0:\n",
    "            j+=1\n",
    "        temp[i,0:num1] = x[i%num2]\n",
    "        temp[i,num1::] = y[j]\n",
    "    return temp"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d2f50fe4-38b9-461d-a772-89b803b4ac6a",
   "metadata": {},
   "outputs": [],
   "source": [
    "n1=850\n",
    "n2=850\n",
    "eps1 = np.pi/(2*3*n1)  \n",
    "eps2 = np.pi/(2*3*n2)\n",
    "x1 = np.linspace(-np.pi/6,np.pi/6,n1).reshape(n1,1)\n",
    "x11 = np.linspace(-np.pi/6,np.pi/6,n2).reshape(n2,1)\n",
    "xt = cart(x1,x11)\n",
    "print(xt.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d27998a0-1c50-4d16-bd01-61702879659f",
   "metadata": {},
   "outputs": [],
   "source": [
    "eps2,eps1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8947d929-09f9-4b7f-9781-0d863f5e01bb",
   "metadata": {},
   "outputs": [],
   "source": [
    "n1=800\n",
    "x1 = np.linspace(-np.pi/12,np.pi/12,n1).reshape(n1,1)\n",
    "xinit = cart(x1,x1)\n",
    "print(xinit.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d9915ca0-e7fe-4957-a8f7-42419a39ff0e",
   "metadata": {},
   "outputs": [],
   "source": [
    "n2 = int(n1/2)\n",
    "x1s = np.linspace(-np.pi/4,-np.pi/6,n2).reshape(n2,1)\n",
    "x2s = np.linspace(np.pi/6,np.pi/4,n2).reshape(n2,1)\n",
    "x1 = np.linspace(-np.pi/4,np.pi/4,n2).reshape(n2,1)\n",
    "x1uns = np.concatenate((x1s,x2s))\n",
    "xuns1 = cart(x1uns,x1)\n",
    "xuns2 = cart(x1, x1uns)\n",
    "xuns = np.vstack((xuns1,xuns2))\n",
    "print(xuns.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e0b978e5-c8ef-4543-be43-ebc4e08d8de9",
   "metadata": {},
   "outputs": [],
   "source": [
    "xuns1.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7d499472-0396-4dce-b361-95595ec2eb5a",
   "metadata": {},
   "outputs": [],
   "source": [
    "np.vstack((xuns1,xuns2)).shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "abb42cf3-c9b6-4dbb-a75d-b018b143fd67",
   "metadata": {},
   "outputs": [],
   "source": [
    "xuns1.reshape(xuns1.shape[0]*2,1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ec5ca503-1ca7-4135-b454-4700cfd60b5a",
   "metadata": {},
   "outputs": [],
   "source": [
    "xt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7a0b12c4-fd57-401d-9550-0124dc736ab3",
   "metadata": {},
   "outputs": [],
   "source": [
    "#Controller\n",
    "model = tf.keras.Sequential()\n",
    "model.add(layers.Dense(10, use_bias=True, activation = 'relu',input_shape=(2,),kernel_regularizer=regularizers.L1(0.01)))\n",
    "model.add(layers.Dense(20, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "model.add(layers.Dense(20, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "model.add(layers.Dense(20, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "model.add(layers.Dense(20, use_bias=True, activation = 'relu',input_shape=(2,),kernel_regularizer=regularizers.L1(0.01)))\n",
    "model.add(layers.Dense(20, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.001)))\n",
    "model.add(layers.Dense(20, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.001)))\n",
    "#model.add(layers.Dense(20, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.001)))\n",
    "# model.add(layers.Dense(200, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "# model.add(layers.Dense(200, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "# model.add(layers.Dense(200, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "model.add(layers.Dense(1,use_bias=True, activation='linear'))\n",
    "model.add(tf.keras.layers.Lambda(lambda x: x * 10))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "edc33b91-fa0b-4843-87b8-a3ee8803a1c2",
   "metadata": {},
   "outputs": [],
   "source": [
    "#Barrier\n",
    "modelb = tf.keras.Sequential()\n",
    "modelb.add(layers.Dense(10, use_bias=True, activation = 'relu',input_shape=(2,),kernel_regularizer=regularizers.L1(0.1)))\n",
    "modelb.add(layers.Dense(20, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "modelb.add(layers.Dense(20, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "modelb.add(layers.Dense(20, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "\n",
    "modelb.add(layers.Dense(20, use_bias=True, activation = 'relu',input_shape=(2,),kernel_regularizer=regularizers.L1(0.01)))\n",
    "modelb.add(layers.Dense(10, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "#modelb.add(layers.Dense(20, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "#modelb.add(layers.Dense(20, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "\n",
    "# modelb.add(layers.Dense(50, use_bias=True, activation = 'relu',input_shape=(2,),kernel_regularizer=regularizers.L1(0.01)))\n",
    "# modelb.add(layers.Dense(50, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "# modelb.add(layers.Dense(200, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "# modelb.add(layers.Dense(200, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "# modelb.add(layers.Dense(50, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "# modelb.add(layers.Dense(50, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "modelb.add(layers.Dense(1,use_bias=True, activation='linear'))\n",
    "#model.add(tf.keras.layers.Lambda(lambda x: x * 100))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "460ab594-b94b-4153-ba00-c006508f5c03",
   "metadata": {},
   "outputs": [],
   "source": [
    "opt1 = tf.keras.optimizers.Adam(1e-6)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "47f8eea3-cf8b-4cda-8901-fca1ccad2349",
   "metadata": {},
   "outputs": [],
   "source": [
    "def check1(xt,y_3,y_next,etac,etac2):\n",
    "    #ind1 = np.where(y_3<=0)[0]\n",
    "    #ind1 = tf.experimental.numpy.where(y_3<=0)[0].numpy()\n",
    "    \n",
    "    #tf.experimental.numpy.where(y_next-y_3<0)[0].numpy()\n",
    "    #ind2 = tf.experimental.numpy.where(y_next-y_3>-etac)[0].numpy()\n",
    "    #ind2 = np.where(((y_next-y_3)>-etac) )[0]\n",
    "    ind2 = np.where( y_3<etac2)[0]\n",
    "    ind4 = np.where( y_next> -etac)[0]\n",
    "    #ind4 = np.where((np.abs(xt[:,0])>0.1)|(np.abs(xt[:,1])>0.1))[0]\n",
    "    #ind4 = tf.experimental.numpy.where(y_next-y_3<0)[0].numpy()\n",
    "    ind1 = np.where(y_next-y_3>-etac)[0]\n",
    "    #ind3 = np.intersect1d(ind1,ind2)\n",
    "    ind5 = np.intersect1d(ind2,ind4)\n",
    "    ind6 = np.intersect1d(ind1,ind5)\n",
    "    return ind6"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d88bcf08-dc0a-4bb5-8c1f-576ca02bf0c3",
   "metadata": {},
   "outputs": [],
   "source": [
    "def check2(xt,y_3,y_next,etac):\n",
    "    ind1 = np.where(y_3<=0)[0]\n",
    "    \n",
    "    \n",
    "    \n",
    "    ind2 = np.where(((y_next-y_3)>-etac) )[0]\n",
    "    ind4 = np.where((np.abs(xt[:,0])>0.1)|(np.abs(xt[:,1])>0.1))[0]\n",
    "    \n",
    "    \n",
    "    ind3 = np.intersect1d(ind1,ind2)\n",
    "    ind5 = np.intersect1d(ind3,ind4)\n",
    "    \n",
    "    return ind5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d56c5b38-90b8-49fd-9db0-6a132e0e1bdb",
   "metadata": {},
   "outputs": [],
   "source": [
    "opt1 = tf.keras.optimizers.Adam(1e-4)\n",
    "opt2 = tf.keras.optimizers.Adam(5e-5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ad1057c5-4c71-4086-8723-e1ccdc8a2cc1",
   "metadata": {},
   "outputs": [],
   "source": [
    "def calculate_lip2(m):\n",
    "    test = m.weights[0].numpy()\n",
    "    \n",
    "    nm = int(len(m.weights)/2)\n",
    "    \n",
    "    lip = 0\n",
    "    for i in range(2,nm*2,2):\n",
    "\n",
    "        \n",
    "       \n",
    "        \n",
    "        test = np.matmul(test,m.weights[i].numpy())\n",
    "        \n",
    "        if i!= nm*2-2:\n",
    "            test2 = m.weights[i].numpy()\n",
    "            for j in range(i+2,len(m.weights),2):\n",
    "                \n",
    "                test2 = np.matmul(test2,m.weights[j].numpy())\n",
    "\n",
    "           \n",
    "            eigenvalues2, _ = LA.eig(np.matmul(test2.T,test2))\n",
    "        else:\n",
    "            eigenvalues2=1\n",
    "        eigenvalues1, _ = LA.eig(np.matmul(test.T,test))\n",
    "        \n",
    "        \n",
    "        lip+= np.sqrt(np.abs(np.max(eigenvalues1 ))) * np.sqrt(np.abs(np.max(eigenvalues2 )))\n",
    "    \n",
    "    return lip / (np.power(2,nm-1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "22d1fbb1-2af8-434a-91f2-c362b93a0745",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4b03416c-49c1-45ef-87bf-ad9faf8be442",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "epochs = 5000000\n",
    "eta = 0.1\n",
    "#eps = 0.03\n",
    "tau = tf.constant(0.01)\n",
    "er1 = 1\n",
    "er2 = 1\n",
    "er3 =1\n",
    "y_1=1\n",
    "y_2=1\n",
    "y_3=1\n",
    "y_next= 10\n",
    "l3=[]\n",
    "l3.append(np.inf)\n",
    "t1 = time.perf_counter()\n",
    "te=[2,3]\n",
    "te2=[2,3]\n",
    "lr = 1e-6\n",
    "l_b=0\n",
    "et = 0.1\n",
    "et2=0.1\n",
    "l_c=0\n",
    "for i in range(epochs):\n",
    "    \n",
    "    \n",
    "    lipb = calculate_lip2(modelb)\n",
    "    lipc = calculate_lip2(model)\n",
    "    #lipc = 1\n",
    "    # for j in range(0,len(model.weights),2):\n",
    "    #     lipc*=np.max(np.abs(model.weights[j]))\n",
    "    #lipc,_ = calculate_lip(model)\n",
    "    if i !=0:\n",
    "        #et = 0\n",
    "        te = check1(xt,y_3,y_next,et,et2)\n",
    "        te2 = check1(xt,y_3,y_next_c,et,et2)\n",
    "        tes = check1(xt,y_3,y_next,et2,et2)\n",
    "        \n",
    "        l3.append(len(te))\n",
    "        \n",
    "    else:\n",
    "        te = [1]\n",
    "        #te2=[]\n",
    "        tes =[1]\n",
    "        l3.append(np.inf)\n",
    "    \n",
    "    #et = np.max(1.05*lipb*eps, 1.05*lipb*(1+lipc)*eps)\n",
    "    et =  lipb*(1+lipc)*eps2\n",
    "    et2 = (lipb)*eps1\n",
    "    if (len(tes)==0 )  and  np.max(y_1)<-et2 and np.min(y_2)>=et2 and i!=0:\n",
    "        break\n",
    "    if i %50 ==0:\n",
    "        \n",
    "        print( lipb,lipc,l_b, l_c)\n",
    "        print(np.max(y_1),len(np.where(y_1>-et2)[0]),np.min(y_2),len(np.where(y_2<et2)[0]),len(tes),len(tes),er3,min(l3),et,et2)\n",
    "    with tf.GradientTape() as tape0, tf.GradientTape() as tape1:\n",
    "        u1 = tf.reshape( model(xt, training=True)[:,0],[xt.shape[0],1])\n",
    "        #u2 = tf.reshape( model(xt, training=True)[:,1],[xt.shape[0],1])\n",
    "        y_1 =tf.reshape( modelb(xinit, training=True),[xinit.shape[0],1])\n",
    "        y_3 = tf.reshape(modelb(xt, training=True),[xt.shape[0],1])\n",
    "        y_2 = tf.reshape(modelb(xuns, training=True),[xuns.shape[0],1])\n",
    "\n",
    "        ind1 = np.where(y_1<=et)[0]\n",
    "        er1 = np.where(y_1>=-et2)\n",
    "\n",
    "        er2 = np.where(y_2<et2)\n",
    "\n",
    "        x1 = tf.reshape(xt[:,0].astype(np.float32),[xt.shape[0],1])\n",
    "        x2 = tf.reshape(xt[:,1].astype(np.float32),[xt.shape[0],1])\n",
    "\n",
    "        x1n = x1+ tau*(x2)\n",
    "        x2n = x2+ tau*(9.8*tf.math.sin(x1) +10*u1)\n",
    "        xnext = tf.reshape(tf.stack([x1n,x2n],axis=1),xt.shape)\n",
    "        y_next =tf.reshape( modelb(xnext,training = True),[xnext.shape[0],1])\n",
    "        \n",
    "        x1nn =x1n+tau*x2n\n",
    "        \n",
    "            \n",
    "        ind2 = np.where((tf.gather(y_next-y_3,np.where(y_3<=et)[0]))>=0)[0]\n",
    "\n",
    "        l_b =tf.reduce_mean(tf.square(tf.subtract(eta,eta)))\n",
    "        l_c =tf.reduce_mean(tf.square(tf.subtract(eta,eta)))\n",
    "        if i%100 ==0:\n",
    "            print('B_switch')\n",
    "            model_copy= tf.keras.models.clone_model(modelb)\n",
    "        y_next_c =tf.reshape( model_copy(xnext,training = False),[xnext.shape[0],1])\n",
    "        y_nextc = tf.gather(y_next_c,te2)\n",
    "        y_3c = model_copy(xt[te2], training=True)\n",
    "        try:\n",
    "            er3 = np.max((tf.gather(y_next-y_3,te)))\n",
    "            #er3 = tf.norm(tf.gather(xt-xnext,te),ord=np.inf)\n",
    "        except:\n",
    "            er3 = 0\n",
    "        #if np.max(y_1)>-et:\n",
    "        l_b += tf.reduce_mean(tf.square(tf.subtract(y_1,-eta))) \n",
    "        #if np.min(y_2)<=et:\n",
    "        l_b+= tf.reduce_mean(tf.square(tf.subtract(y_2,eta))) \n",
    "\n",
    "        # if np.max(y_1)<=-et and np.min(y_2)>et and len(te)<1500:\n",
    "        #     l_b += tf.reduce_mean(tf.square(tf.gather(tf.subtract(y_3,2*et),te)))\n",
    "        if len(te)!=0:\n",
    "            \n",
    "            l_b += tf.reduce_mean(tf.square(tf.gather(tf.subtract(y_next-y_3,-eta),te)))\n",
    "        if len(te2)!=0:\n",
    "            l_c += tf.reduce_mean(tf.square(tf.subtract(y_nextc-y_3c,-eta)))\n",
    "            #+tf.reduce_max(tf.square(tf.subtract(y_nextc-y_3c,-eta)))\n",
    "             \n",
    "        #l_c += tf.reduce_mean(tf.square(tf.subtract(y_nextc-y_3c,-eta))) +tf.reduce_max(tf.square(tf.subtract(y_nextc-y_\"3c,-eta)))\n",
    "        \n",
    "         #+tf.reduce_max(tf.square(tf.subtract(y_nextc-y_3c,-eta)))\n",
    "            #l_c+=tf.reduce_max(tf.square(tf.subtract(y_nextc-y_3c,-eta)))\n",
    "        l_c +=0.005*( tf.reduce_mean(tf.square(tf.subtract(x2n,0))) + 10*tf.reduce_mean(tf.square(tf.subtract(x1nn,0))))\n",
    "        gradsb = tape0.gradient(l_b,modelb.trainable_variables)\n",
    "        gradsc = tape1.gradient(l_c,model.trainable_variables)\n",
    "        \n",
    "        cgradb = [tf.clip_by_norm(g, 2)  for g in gradsb]\n",
    "        if gradsc!=None:\n",
    "            cgradc = [tf.clip_by_norm(g, 2)  for g in gradsc]\n",
    "            opt2.apply_gradients(zip(cgradc,model.trainable_variables))\n",
    "        opt1.apply_gradients(zip(cgradb,modelb.trainable_variables))\n",
    "        \n",
    "t2 = time.perf_counter()\n",
    "tot = t2-t1\n",
    "print(f'Total runtime:{tot}s')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7b4dad82-c2df-41b4-a5d6-123f01aa79d9",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.9.18"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
