{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5a74f617",
   "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 \n",
    "from matplotlib.patches import Rectangle"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b759a4d8",
   "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": "db79c743",
   "metadata": {},
   "outputs": [],
   "source": [
    "n1=100\n",
    "n2=100\n",
    "eps1 = 3/(2*n1)  \n",
    "eps2 = 3/(2*n2)\n",
    "x1 = np.linspace(-1.5,1.5,n1).reshape(n1,1)\n",
    "xt = cart(x1,x1)\n",
    "xt2 = cart(xt,xt)\n",
    "xt = cart(xt2,xt)\n",
    "del xt2,x1\n",
    "print(xt.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1a02f24b",
   "metadata": {},
   "outputs": [],
   "source": [
    "eps2,eps1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "fae57799",
   "metadata": {},
   "outputs": [],
   "source": [
    "n1=100\n",
    "n2=100\n",
    "eps1 = 3/(2*n1)  \n",
    "eps2 = 3/(2*n2)\n",
    "x1 = np.linspace(-3,2,n1).reshape(n1,1)\n",
    "xt = cart(x1,x1)\n",
    "xt2 = cart(xt,xt)\n",
    "xuns = cart(xt2,xt)\n",
    "del xt2,x1\n",
    "print(xt.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0e9d898c",
   "metadata": {},
   "outputs": [],
   "source": [
    "xinit"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "99edcd9d",
   "metadata": {},
   "outputs": [],
   "source": [
    "xuns"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5f0c392b",
   "metadata": {},
   "outputs": [],
   "source": [
    "#Controller\n",
    "model = tf.keras.Sequential()\n",
    "model.add(layers.Dense(32, use_bias=True, activation = 'relu',input_shape=(6,),kernel_regularizer=regularizers.L1(0.01)))\n",
    "model.add(layers.Dense(32, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "model.add(layers.Dense(50, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.01)))\n",
    "model.add(layers.Dense(50, 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(3,use_bias=True, activation='linear'))\n",
    "model.add(tf.keras.layers.Lambda(lambda x: x * 10))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "17d00e9c",
   "metadata": {},
   "outputs": [],
   "source": [
    "#Barrier\n",
    "modelb = tf.keras.Sequential()\n",
    "modelb.add(layers.Dense(32, use_bias=True, activation = 'relu',input_shape=(3,),kernel_regularizer=regularizers.L1(0.01)))\n",
    "modelb.add(layers.Dense(32, 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",
    "\n",
    "#modelb.add(layers.Dense(200, 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",
    "m#odelb.add(layers.Dense(200, 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": "5330290e",
   "metadata": {},
   "outputs": [],
   "source": [
    "opt1 = tf.keras.optimizers.Adam(1e-6)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4acfb144",
   "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": "fa8756b8",
   "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": "471e983f",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2fddaccf",
   "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.max(eigenvalues1 ))) * np.sqrt((np.max(eigenvalues2 )))\n",
    "    \n",
    "    return lip / (np.power(2,nm-1))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "aef58fed",
   "metadata": {},
   "outputs": [],
   "source": [
    "opt1 = tf.keras.optimizers.Adam(1e-3)\n",
    "opt2 = tf.keras.optimizers.Adam(5e-5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "079dd23c",
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "epochs = 5000000\n",
    "eta = 0.01\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",
    "lipc=1\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(tes))\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+0.1*lipc)*eps2\n",
    "    et2 = (lipb)*eps1\n",
    "    if (len(te)==0 or len(tes)==0)  and  np.max(y_1)<-et2 and np.min(y_2)>=et2 and i!=0:\n",
    "        break\n",
    "    if i %10 ==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(te),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",
    "        u3 = tf.reshape( model(xt, training=True)[:,2],[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<=0)[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",
    "        x3 = tf.reshape(xt[:,2].astype(np.float32),[xt.shape[0],1])\n",
    "        \n",
    "        x4 = tf.reshape(xt[:,3].astype(np.float32),[xt.shape[0],1])\n",
    "        x5 = tf.reshape(xt[:,4].astype(np.float32),[xt.shape[0],1])\n",
    "        x6 = tf.reshape(xt[:,5].astype(np.float32),[xt.shape[0],1])\n",
    "\n",
    "        x1n = x1+ tau*(x1)\n",
    "        x2n = x2+ tau*(x2)\n",
    "        x3n = x3 + tau*(x3)\n",
    "        \n",
    "        x4n = x4+ tau*(3*x1 +2*u2)\n",
    "        x5n = x5+ tau*(-2*u1)\n",
    "        x6n = x6 - tau*(x3)\n",
    "        \n",
    "        xnext = tf.reshape(tf.stack([x1n,x2n,x3n,x4n,x5n,x6n],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",
    "        if i%50 ==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=False)\n",
    "        if len(te2)!=0:\n",
    "            l_c += tf.reduce_mean(tf.square(tf.subtract(y_nextc,-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.00005*( tf.reduce_mean(tf.square(tf.subtract(x2n,1))) + 0.00005*tf.reduce_mean(tf.square(tf.subtract(x1n,1))))\n",
    "        l_b =tf.reduce_mean(tf.square(tf.subtract(eta,eta)))\n",
    "        if len(te)!=0:\n",
    "            \n",
    "            l_b += tf.reduce_mean(tf.square(tf.gather(tf.subtract(y_next,-eta),te)))\n",
    "        #l_c =tf.reduce_mean(tf.square(tf.subtract(eta,eta)))\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",
    "        gradsc = tape1.gradient(l_c,model.trainable_variables)\n",
    "        try:\n",
    "            cgradc = [tf.clip_by_norm(g, 2)  for g in gradsc]\n",
    "            opt2.apply_gradients(zip(cgradc,model.trainable_variables))\n",
    "        except:\n",
    "            print('passed')\n",
    "            pass\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",
    "        gradsb = tape0.gradient(l_b,modelb.trainable_variables)\n",
    "                \n",
    "        cgradb = [tf.clip_by_norm(g, 2)  for g in gradsb]\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": "4cddec8c",
   "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
}
