{
 "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 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": "ac7d223f-9aea-4d83-96d1-54947293188f",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy.linalg as LA"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "17b6f1ee-8f68-415c-ad79-2ea5be953772",
   "metadata": {},
   "outputs": [],
   "source": [
    "import gc"
   ]
  },
  {
   "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=52\n",
    "eps = np.pi/(3*n1*2)  \n",
    "x1 = np.linspace(-np.pi/6,np.pi/6,n1).reshape(n1,1)\n",
    "xt1 = cart(x1,x1)\n",
    "xt = cart(xt1,xt1)\n",
    "del xt1,x1\n",
    "print(xt.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f38521da-8865-4992-94f4-58f8a65dd6b9",
   "metadata": {},
   "outputs": [],
   "source": [
    "eps"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8947d929-09f9-4b7f-9781-0d863f5e01bb",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "x1 = np.linspace(-np.pi/20,np.pi/20,n1).reshape(n1,1)\n",
    "xinit1 = cart(x1,x1)\n",
    "xinit = cart(xinit1,xinit1)\n",
    "del xinit1,x1\n",
    "print(xinit.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d9915ca0-e7fe-4957-a8f7-42419a39ff0e",
   "metadata": {},
   "outputs": [],
   "source": [
    "#n2 = int(n1/1.64)\n",
    "n2=31\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/6,np.pi/6,n2).reshape(n2,1)\n",
    "x1ss = cart(x1,x1)\n",
    "x1uns = np.concatenate((x1s,x2s))\n",
    "xuns1 = cart(x1uns,x1)\n",
    "xuns2 = cart(x1, x1uns)\n",
    "xuns3 = np.vstack((xuns1,xuns2))\n",
    "xuns4 = cart(xuns3,x1ss)\n",
    "xuns5 = cart(x1ss,xuns3)\n",
    "xuns = np.vstack((xuns5,xuns4))\n",
    "del xuns1, xuns2,xuns3,xuns4,xuns5,x1s,x2s,x1,x1uns,x1ss\n",
    "print(xuns.shape)"
   ]
  },
  {
   "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(200, use_bias=True, activation = 'relu',input_shape=(4,),kernel_regularizer=regularizers.L1(0.1)))\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.1)))\n",
    "model.add(layers.Dense(200, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.1)))\n",
    "model.add(layers.Dense(200, use_bias=True, activation = 'relu',input_shape=(2,),kernel_regularizer=regularizers.L1(0.01)))\n",
    "model.add(layers.Dense(200, use_bias=True, activation = 'relu',kernel_regularizer=regularizers.L1(0.1)))\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(2,use_bias=True, activation='linear'))\n",
    "#model.add(tf.keras.layers.Lambda(lambda x: x * 100))"
   ]
  },
  {
   "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(200, use_bias=True, activation = 'relu',input_shape=(4,),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(200, 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(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(20, 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(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(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,et2):\n",
    "    \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",
    "    ind1 = np.where(y_next-y_3>-etac)[0]\n",
    "    #ind2 = np.where(((y_next-y_3)>-etac) )[0]\n",
    "    ind2 = np.where( y_3<=et2)[0]\n",
    "    ind4 = np.where( y_next> -etac)[0]\n",
    "    #ind5 = np.where((np.abs(xt[:,0])>0.05)|(np.abs(xt[:,2])>0.05))[0]\n",
    "    #ind4 = tf.experimental.numpy.where(y_next-y_3<0)[0].numpy()\n",
    "    \n",
    "    #ind3 = np.intersect1d(ind1,ind2)\n",
    "    ind6 = np.intersect1d(ind2,ind4)\n",
    "    ind7 = np.intersect1d(ind1,ind6)\n",
    "    return ind7"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d88bcf08-dc0a-4bb5-8c1f-576ca02bf0c3",
   "metadata": {},
   "outputs": [],
   "source": [
    "def check2(xt,y_3,y_next,etac,et2):\n",
    "    ind1 = np.where(y_3<=et2)[0]\n",
    "    \n",
    "    \n",
    "    \n",
    "    ind2 = 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-5)\n",
    "opt2 = tf.keras.optimizers.Adam(5e-5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "7dcc94f9-ffd5-40bb-9ae1-8ed796ed3dad",
   "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": "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",
    "l_c=0\n",
    "\n",
    "ers1 = 1\n",
    "ers2 = 1\n",
    "te1 = 1\n",
    "l1=[1]\n",
    "l2=[1]\n",
    "for i in range(epochs):\n",
    "    \n",
    "    \n",
    "    # if i !=0:\n",
    "    #     #et = 0\n",
    "    #     te = check1(xt,y_3,y_next,et)\n",
    "    #     te2 = check1(xt,y_3,y_next_c,et)\n",
    "    #     tes = check2(xt,y_3,y_next,et)\n",
    "        \n",
    "    #     l3.append(len(te))\n",
    "        \n",
    "    # else:\n",
    "    #     te = [1]\n",
    "    #     #te2=[]\n",
    "    #     tes =[1]\n",
    "    #     l3.append(np.inf)\n",
    "    \n",
    "    lipb = calculate_lip2(modelb)\n",
    "    lipc = calculate_lip2(model)\n",
    "    et =lipb*(1+tau.numpy()*39*lipc)*eps\n",
    "    et2 = lipb*eps\n",
    "    if te1==0 and  max(l1)<=-et and min(l2)>et   and i!=0:\n",
    "        break\n",
    "    if i %1 ==0:\n",
    "        \n",
    "        print( lipb,lipc,l_b, l_c)\n",
    "        print(max(l1),min(l2),(te1),min(l3),et,et2)\n",
    "    \n",
    "    n=40\n",
    "    k = int(xt.shape[0]/n)\n",
    "    te = []\n",
    "    te1 = 0\n",
    "    ers1 = 0\n",
    "    ers2 = 0\n",
    "    l1=[]\n",
    "    l2=[]\n",
    "    if i%10 ==0:\n",
    "        print('B_switch')\n",
    "        model_copy= tf.keras.models.clone_model(modelb)\n",
    "    for j in range(n):\n",
    "        \n",
    "        with tf.GradientTape() as tape0, tf.GradientTape() as tape1:\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",
    "            \n",
    "            #batch = np.random.choice(xt.shape[0], batch_size)\n",
    "            #xt = xtt[batch]\n",
    "            #xinit = xinitt[batch]\n",
    "            #xuns = xunst[batch]\n",
    "            if j!=n-1:\n",
    "                u1 = tf.reshape( model(xt[j*k:(j+1)*k], training=True)[:,0],[k,1])\n",
    "                u2 = tf.reshape( model(xt[j*k:(j+1)*k], training=True)[:,1],[k,1])\n",
    "                y_1 =tf.reshape( modelb(xinit[j*k:(j+1)*k], training=True),[k,1])\n",
    "                y_3 = tf.reshape(modelb(xt[j*k:(j+1)*k], training=True),[k,1])\n",
    "                y_2 = tf.reshape(modelb(xuns[j*k:(j+1)*k], training=True),[k,1])\n",
    "                x1 = tf.reshape(xt[j*k:(j+1)*k,0].astype(np.float32),[k,1])\n",
    "                x2 = tf.reshape(xt[j*k:(j+1)*k,1].astype(np.float32),[k,1])\n",
    "                x3 = tf.reshape(xt[j*k:(j+1)*k,2].astype(np.float32),[k,1])\n",
    "                x4 = tf.reshape(xt[j*k:(j+1)*k,3].astype(np.float32),[k,1])\n",
    "    \n",
    "            else:\n",
    "                u1 = tf.reshape( model(xt[j*k::], training=True)[:,0],[xt[j*k::].shape[0],1])\n",
    "                u2 = tf.reshape( model(xt[j*k::], training=True)[:,1],[xt[j*k::].shape[0],1])\n",
    "                y_1 =tf.reshape( modelb(xinit[j*k::], training=True),[xt[j*k::].shape[0] ,1])\n",
    "                y_3 = tf.reshape(modelb(xt[j*k::], training=True),[xt[j*k::].shape[0] ,1])\n",
    "                y_2 = tf.reshape(modelb(xuns[j*k::], training=True),[xuns[j*k::].shape[0],1])\n",
    "                x1 = tf.reshape(xt[j*k::,0].astype(np.float32),[xt[j*k::].shape[0],1])\n",
    "                x2 = tf.reshape(xt[j*k::,1].astype(np.float32),[xt[j*k::].shape[0],1])\n",
    "                x3 = tf.reshape(xt[j*k::,2].astype(np.float32),[xt[j*k::].shape[0],1])\n",
    "                x4 = tf.reshape(xt[j*k::,3].astype(np.float32),[xt[j*k::].shape[0],1])\n",
    "    \n",
    "            ind1 = np.where(y_1<=et)[0]\n",
    "            er1 = np.where(y_1>=-et)\n",
    "            ers1 += len(er1)\n",
    "            er2 = np.where(y_2<et)\n",
    "            ers2 += len(er2)\n",
    "\n",
    "            \n",
    "            l1.append(np.max(y_1))\n",
    "            l2.append(np.min(y_2))\n",
    "            x2n = x2 + tau *(9.8*tf.math.sin(x1) - tf.math.sin(x1-x3)*x2*x2 + 30.0*u1)\n",
    "            \n",
    "            x4n = x4 + tau* (9.8*tf.math.sin(x3)+tf.math.sin(x1-x3)*x4*x4 +390.0*u2)\n",
    "            x3n =x3+ tau*(x4)\n",
    "            x1n = x1+ tau*(x2)\n",
    "            x1nn = x1n + tau*(x2n)\n",
    "            x3nn = x3n + tau*(x4n)\n",
    "            x2nn \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,x3n,x4n],axis=1),(u1.shape[0],4))\n",
    "            y_next =tf.reshape( modelb(xnext,training = True),[xnext.shape[0],1])\n",
    "    \n",
    "           \n",
    "            \n",
    "            y_next_c =tf.reshape( model_copy(xnext,training = False),[xnext.shape[0],1])\n",
    "            \n",
    "            te = check1(xt,y_3,y_next,et2,et2)\n",
    "            te2 = check1(xt,y_3,y_next_c,et,et2)\n",
    "            te3 = check1(xt,y_3,y_next,et,et2)\n",
    "            te1 += len(te3)\n",
    "            #l3.append(len(te))\n",
    "            #ind2 = np.where((tf.gather(y_next-y_3,np.where(y_3<=et)[0]))>=0)[0]\n",
    "    \n",
    "            \n",
    "    \n",
    "            y_nextc = tf.gather(y_next_c,te2)\n",
    "            y_3c = modelb(xt[te2], training=True)\n",
    "            \n",
    "            \n",
    "            l_b += tf.reduce_mean(tf.square(tf.subtract(y_1,-eta)))# +tf.reduce_max(tf.square(tf.subtract(y_1,-eta)))\n",
    "            \n",
    "            l_b+= tf.reduce_mean(tf.square(tf.subtract(y_2,eta))) #+tf.reduce_max(tf.square(tf.subtract(y_2,eta)))  \n",
    "            if len(te)!=0:\n",
    "                y_next = tf.gather(y_next,te)\n",
    "                y_3 = tf.gather(y_3,te)\n",
    "                l_b += tf.reduce_mean(tf.square(tf.subtract(y_next-y_3,-eta))) #+tf.reduce_max(tf.square(tf.subtract(y_next-y_3,-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",
    "            # if len(tes)<0:\n",
    "            #     l_c += tf.reduce_mean(tf.square(tf.subtract(y_next-y_3,-eta))) #+tf.reduce_max(tf.square(tf.subtract(y_next-y_3,-eta)))\n",
    "            # else:\n",
    "            if len(te2!=0):\n",
    "                l_c += tf.reduce_mean(tf.square(tf.subtract(y_3c,-eta))) #+tf.reduce_max(tf.square(tf.subtract(y_nextc-y_3c,-eta)))\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",
    "                #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))) + tf.reduce_max(tf.square(tf.subtract(x2n,0)))+ tf.reduce_mean(tf.square(tf.subtract(x4n,0))) + tf.reduce_max(tf.square(tf.subtract(x4n,0))))\n",
    "            +1000* ( tf.reduce_max(tf.square(tf.subtract(x1nn,0))) + tf.reduce_max(tf.square(tf.subtract(x1nn,0)))+ tf.reduce_max(tf.square(tf.subtract(x3nn,0))) + tf.reduce_max(tf.square(tf.subtract(x3nn,0))))\n",
    "            \n",
    "            if l_b!=0:\n",
    "                gradsb = tape0.gradient(l_b,modelb.trainable_variables)\n",
    "                cgradb = [tf.clip_by_norm(g, 2)  for g in gradsb]\n",
    "                opt1.apply_gradients(zip(cgradb,modelb.trainable_variables))\n",
    "            \n",
    "            gradsc = tape1.gradient(l_c,model.trainable_variables)\n",
    "            \n",
    "            cgradc = [tf.clip_by_norm(g, 2)  for g in gradsc]\n",
    "            \n",
    "            opt2.apply_gradients(zip(cgradc,model.trainable_variables))\n",
    "            \n",
    "            gc.collect()\n",
    "    l3.append(te1)\n",
    "            #tape0.reset()\n",
    "\n",
    "            #tape1.reset()\n",
    "t2 = time.perf_counter()\n",
    "tot = t2-t1\n",
    "print(f'Total runtime:{tot}s')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "08c94b79-5a87-4723-85df-6c69e26a2e25",
   "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
}
