{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Codes for development of LNN 2 for linear path following problem\n",
    "# -*- coding: utf-8 -*-\n",
    "import torch \n",
    "import torch.nn.functional as F\n",
    "import numpy as np\n",
    "import timeit \n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "#define activation functions in the hidden and output layers\n",
    "def acti_h(x):\n",
    "    return torch.tanh(x)\n",
    "\n",
    "def acti_o(x):\n",
    "    return x**2"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Neural network model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Net(torch.nn.Module):\n",
    "    \n",
    "    def __init__(self,n_input,n_hidden,n_output):\n",
    "        super(Net, self).__init__()\n",
    "        torch.manual_seed(2)\n",
    "        self.layer1 = torch.nn.Linear(n_input, n_hidden,bias=False)\n",
    "        self.layer2 = torch.nn.Linear(n_hidden,n_output,bias=False)\n",
    "        self.layer3 = torch.nn.Linear(1,n_output,bias=False)\n",
    "        self.layer4 = torch.nn.Linear(1,n_output,bias=False)\n",
    "\n",
    "    def forward(self,x):\n",
    "        h = torch.tanh(self.layer1(x))  # torch.tanh as activation function\n",
    "        out_1 = self.layer2(h) ** 2  # Squaring function at output layer\n",
    "        out_21 = (self.layer3(x[:, :1])) ** 2  # Squaring function for x_1\n",
    "        out_22 = (self.layer4(x[:, 1:])) ** 2  # Squaring function for x_2\n",
    "        return out_1 + out_21 + out_22\n",
    "    \n",
    "    def compute_jacobian(self, x):\n",
    "        \"\"\"\n",
    "        Compute the Jacobian ∂y/∂x manually (no autograd),\n",
    "        where y is the scalar output and x ∈ R^2 is the input.\n",
    "        \"\"\"\n",
    "        W1 = self.layer1.weight.detach()  # (n_hidden, 2)\n",
    "        W2 = self.layer2.weight.detach()  # (1, n_hidden)\n",
    "        W3 = self.layer3.weight.detach()  # (1, 1)\n",
    "        W4 = self.layer4.weight.detach()  # (1, 1)\n",
    "\n",
    "        x1 = x[:, :1]  # (batch, 1)\n",
    "        x2 = x[:, 1:]  # (batch, 1)\n",
    "\n",
    "        z1 = self.layer1(x)               # (batch, n_hidden)\n",
    "        h = torch.tanh(z1)                # (batch, n_hidden)\n",
    "        out1 = self.layer2(h)             # (batch, 1)\n",
    "        dy_dout1 = 2 * out1               # (batch, 1)\n",
    "\n",
    "        # ∂h/∂x = (1 - tanh^2(z1)) * W1\n",
    "        sech2 = 1 - torch.tanh(z1) ** 2   # (batch, n_hidden)\n",
    "        d_out1_dx = torch.zeros(x.shape[0], x.shape[1])  # (batch, 2)\n",
    "        for i in range(x.shape[0]):\n",
    "            d_h_dx = (sech2[i].unsqueeze(1) * W1)         # (n_hidden, 2)\n",
    "            d_out1_dx[i] = (W2 @ d_h_dx).squeeze() * dy_dout1[i]\n",
    "\n",
    "        # ∂(layer3(x1)^2)/∂x1 = 2 * (W3 * x1) * W3\n",
    "        d_out21_dx1 = 2 * (self.layer3(x1)) * W3          # (batch, 1)\n",
    "        d_out22_dx2 = 2 * (self.layer4(x2)) * W4          # (batch, 1)\n",
    "\n",
    "        # Fill Jacobian rows\n",
    "        J = torch.zeros(x.shape[0], x.shape[1])           # (batch, 2)\n",
    "        J[:, 0] = d_out1_dx[:, 0] + d_out21_dx1[:, 0]\n",
    "        J[:, 1] = d_out1_dx[:, 1] + d_out22_dx2[:, 0]\n",
    "\n",
    "        return J  # Shape: (batch_size, 2)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Dynamical system of Linear path following"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def f_value(x): #x = [x1,x2]\n",
    "    #Dynamics\n",
    "    y = []\n",
    "    c = 2\n",
    "    v = 6\n",
    "    \n",
    "    for r in range(0,len(x)): \n",
    "        x1 = x[r][0] \n",
    "        x2 = x[r][1]\n",
    "        dx1 = v * torch.sin(x2)\n",
    "        dx2 = -x2 - c*v*torch.sin(x2)*x1/x2\n",
    "        f = [ dx1, \n",
    "              dx2]\n",
    "        y.append(f) \n",
    "    y = torch.tensor(y)\n",
    "    return y "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Neural network hyperparameter Options & training dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<torch._C.Generator at 0x7f9585583ed0>"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "'''\n",
    "For learning \n",
    "'''\n",
    "N = 500             # sample size\n",
    "D_in = 2            # input dimension\n",
    "H1 = 4            # hidden dimension\n",
    "D_out = 1           # output dimension\n",
    "torch.manual_seed(10)  "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2.   3.14]\n"
     ]
    }
   ],
   "source": [
    "Num = 20\n",
    "\n",
    "x1_list = np.linspace(-2, 2, Num, endpoint=True)\n",
    "x2_list = np.linspace(-3.14, 3.14, Num, endpoint=True) \n",
    "\n",
    "x_start = list()\n",
    "\n",
    "for x1 in x1_list:   \n",
    "    for x2 in x2_list:\n",
    "        x_start.append(np.array([x1, x2]))\n",
    "\n",
    "print(x_start[-1])\n",
    "# convert to np.arrays\n",
    "x = torch.FloatTensor(np.array(x_start)).requires_grad_() #LNN input\n",
    "x_0 = torch.zeros([1, 2])\n",
    "\n",
    "#we need to allow gradient calculation for the input\n",
    "x_real = torch.FloatTensor(np.array(x_start))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Train and optimization process of LNN"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "start = timeit.default_timer()\n",
    "\n",
    "#define function to obtain the optimal LNN under one set of weight boundary parameters\n",
    "def search_LNN_p(min_a, min_b, min_c):\n",
    "    pho_list = list()\n",
    "    NUM_list = list()\n",
    "    out_iters = 0\n",
    "    valid = False\n",
    "    #develop model\n",
    "    model = Net(D_in,H1, D_out)\n",
    "    L = []\n",
    "    t = 0\n",
    "    max_iters = 10\n",
    "    learning_rate = 0.01\n",
    "    optimizer = torch.optim.Adam(model.parameters(), lr=learning_rate)\n",
    "    \n",
    "    #Assign initial guess for the parameter\n",
    "    with torch.no_grad():       \n",
    "    #the parameters will NOT be updated in the training phase\n",
    "        model.layer3.weight = torch.nn.Parameter(torch.ones_like(model.layer3.weight) * min_b) # CA^2\n",
    "        model.layer4.weight = torch.nn.Parameter(torch.ones_like(model.layer4.weight) * min_c) # T^2\n",
    "      \n",
    "    # Clamp all weights to be non-negative       \n",
    "    with torch.no_grad():\n",
    "        for layer in [model.layer1, model.layer2]:\n",
    "            for param in layer.parameters():\n",
    "                param.data.clamp_(min = min_a) \n",
    "\n",
    "\n",
    "    #train model, and search for the stability region after each epoch\n",
    "    while out_iters < max_iters:\n",
    "        V_candidate = model(x)\n",
    "        X0 = model(x_0)\n",
    "        # Initialize a tensor to store the Jacobians for each sample\n",
    "        jacobian = torch.zeros(Num*Num, 2)  \n",
    "        jacobian = model.compute_jacobian(x)\n",
    "        # Compute lie derivative of V : L_V = ∑∂V/∂xᵢ*dx\n",
    "        dx = f_value(x) #accurate first-principles model       \n",
    "        L_V = torch.diagonal(torch.mm(jacobian,dx.t()),0)\n",
    "\n",
    "        print(\"Number of negative values:\", (L_V <= 0).sum().item())\n",
    "\n",
    "        Lyapunov_risk = (F.relu(((L_V + 0.001*abs(V_candidate).T).T))).mean() #loss function\n",
    "        \n",
    "\n",
    "        if out_iters%5 == 0:\n",
    "            print(out_iters, \"Lyapunov Risk=\",Lyapunov_risk.item())\n",
    "        \n",
    "\n",
    "        L.append(Lyapunov_risk.item())\n",
    "        optimizer.zero_grad()\n",
    "        Lyapunov_risk.backward()\n",
    "        optimizer.step() \n",
    "        \n",
    "      # Clamp weight parameters in the hidden and output layers to be non-negative during training   \n",
    "        with torch.no_grad():\n",
    "            for layer in [model.layer1, model.layer2]:\n",
    "                for param in layer.parameters():\n",
    "                    param.data.clamp_(min = min_a) \n",
    "\n",
    "\n",
    "        out_iters+=1\n",
    "        \n",
    "        #Search for the biggest pho via global search method\n",
    "        start = timeit.default_timer()\n",
    "        Num_points = 0\n",
    "        Index_pho = 0\n",
    "        Flag = True\n",
    "        pho = min(V_candidate)[0]\n",
    "        while(Flag):\n",
    "            pho = pho+min(V_candidate)[0] #update pho\n",
    "            V_point_list = list()\n",
    "            #get region with stable if and only if\n",
    "            for i in range (len(x_real)):  \n",
    "                if V_candidate[i][0] < pho:\n",
    "                    if L_V[i] > - 0.001*V_candidate[i][0]:  #NO POINTS WITHIN THE REGION IS UNSTABLE\n",
    "                        Flag =False\n",
    "                        V_point_list = list()  #set the list as empty\n",
    "                        break\n",
    "                    else:\n",
    "                        V_point_list.append(x_real[i])\n",
    "            #Check the boundary, via the magnitude of x1 and x2\n",
    "            for k in V_point_list: \n",
    "                if abs(k[0])>=2 or abs(k[1])>=3.14:\n",
    "                    #print(k)\n",
    "                    Flag =False\n",
    "                    V_point_list = list()  #set the list as empty\n",
    "                    break \n",
    "            K = len(V_point_list)  #if number of samples increase, update the value of pho\n",
    "            if K > Num_points:\n",
    "                Num_points = K\n",
    "                Index_pho = pho\n",
    "        pho_list.append(Index_pho)\n",
    "        NUM_list.append(Num_points)\n",
    "        \n",
    "        end = timeit.default_timer()\n",
    "        \n",
    "    print(min_a, min_b, min_c)\n",
    "    print(max(NUM_list),NUM_list.index(max(NUM_list)),pho_list[NUM_list.index(max(NUM_list))])\n",
    "    return max(NUM_list)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Bayesian Optimization "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[93m--- Optimizing Number of points ---\u001b[0m\n",
      "|   iter    |  target   |   min_a   |   min_b   |   min_c   |\n",
      "-------------------------------------------------------------\n",
      "Number of negative values: 390\n",
      "0 Lyapunov Risk= 4.3829459173139185e-06\n",
      "Number of negative values: 400\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/wulab2/.local/lib/python3.8/site-packages/torch/autograd/__init__.py:251: UserWarning: CUDA initialization: The NVIDIA driver on your system is too old (found version 11060). Please update your GPU driver by downloading and installing a new version from the URL: http://www.nvidia.com/Download/index.aspx Alternatively, go to: https://pytorch.org to install a PyTorch version that has been compiled with your version of the CUDA driver. (Triggered internally at ../c10/cuda/CUDAFunctions.cpp:108.)\n",
      "  Variable._execution_engine.run_backward(  # Calls into the C++ engine to run the backward pass\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.0019960425587510337 0.06598978939358487 0.04939549651064031\n",
      "258 1 tensor(0.0209, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m1        \u001b[0m | \u001b[0m258.0    \u001b[0m | \u001b[0m0.001996 \u001b[0m | \u001b[0m0.06599  \u001b[0m | \u001b[0m0.0494   \u001b[0m |\n",
      "Number of negative values: 242\n",
      "0 Lyapunov Risk= 0.014732892625033855\n",
      "Number of negative values: 240\n",
      "Number of negative values: 236\n",
      "Number of negative values: 238\n",
      "Number of negative values: 238\n",
      "Number of negative values: 238\n",
      "5 Lyapunov Risk= 0.013014588505029678\n",
      "Number of negative values: 240\n",
      "Number of negative values: 240\n",
      "Number of negative values: 242\n",
      "Number of negative values: 242\n",
      "0.007875049978766315 0.08019782273069231 0.03453333447543775\n",
      "0 0 0\n",
      "| \u001b[0m2        \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.007875 \u001b[0m | \u001b[0m0.0802   \u001b[0m | \u001b[0m0.03453  \u001b[0m |\n",
      "Number of negative values: 268\n",
      "0 Lyapunov Risk= 0.022508084774017334\n",
      "Number of negative values: 270\n",
      "Number of negative values: 282\n",
      "Number of negative values: 288\n",
      "Number of negative values: 300\n",
      "Number of negative values: 322\n",
      "5 Lyapunov Risk= 0.0030201165936887264\n",
      "Number of negative values: 392\n",
      "Number of negative values: 396\n",
      "Number of negative values: 374\n",
      "Number of negative values: 354\n",
      "0.0028369961259166574 0.08216849597815173 0.09623254183153347\n",
      "10 7 tensor(0.0026, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m3        \u001b[0m | \u001b[0m10.0     \u001b[0m | \u001b[0m0.002837 \u001b[0m | \u001b[0m0.08217  \u001b[0m | \u001b[0m0.09623  \u001b[0m |\n",
      "Number of negative values: 280\n",
      "0 Lyapunov Risk= 0.004902464337646961\n",
      "Number of negative values: 298\n",
      "Number of negative values: 334\n",
      "Number of negative values: 366\n",
      "Number of negative values: 358\n",
      "Number of negative values: 348\n",
      "5 Lyapunov Risk= 0.0007157042855396867\n",
      "Number of negative values: 334\n",
      "Number of negative values: 336\n",
      "Number of negative values: 342\n",
      "Number of negative values: 348\n",
      "0.008771733083946737 0.04220355429620801 0.05508956129711129\n",
      "6 3 tensor(0.0008, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m4        \u001b[0m | \u001b[0m6.0      \u001b[0m | \u001b[0m0.008772 \u001b[0m | \u001b[0m0.0422   \u001b[0m | \u001b[0m0.05509  \u001b[0m |\n",
      "Number of negative values: 270\n",
      "0 Lyapunov Risk= 0.005334069952368736\n",
      "Number of negative values: 274\n",
      "Number of negative values: 276\n",
      "Number of negative values: 274\n",
      "Number of negative values: 274\n",
      "Number of negative values: 278\n",
      "5 Lyapunov Risk= 0.0038870815187692642\n",
      "Number of negative values: 278\n",
      "Number of negative values: 280\n",
      "Number of negative values: 280\n",
      "Number of negative values: 280\n",
      "0.00686628305820415 0.07414318242846102 0.04332256793113555\n",
      "0 0 0\n",
      "| \u001b[0m5        \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.006866 \u001b[0m | \u001b[0m0.07414  \u001b[0m | \u001b[0m0.04332  \u001b[0m |\n",
      "Number of negative values: 390\n",
      "0 Lyapunov Risk= 3.142700279568089e-06\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.0016671123893238812 0.06488634707179301 0.04894814713413217\n",
      "256 1 tensor(0.0201, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m6        \u001b[0m | \u001b[0m256.0    \u001b[0m | \u001b[0m0.001667 \u001b[0m | \u001b[0m0.06489  \u001b[0m | \u001b[0m0.04895  \u001b[0m |\n",
      "Number of negative values: 346\n",
      "0 Lyapunov Risk= 0.00025719768018461764\n",
      "Number of negative values: 398\n",
      "Number of negative values: 394\n",
      "Number of negative values: 390\n",
      "Number of negative values: 382\n",
      "Number of negative values: 376\n",
      "5 Lyapunov Risk= 7.888509571785107e-05\n",
      "Number of negative values: 376\n",
      "Number of negative values: 382\n",
      "Number of negative values: 384\n",
      "Number of negative values: 392\n",
      "0.0001 0.06417956533854523 0.05740687198222966\n",
      "8 3 tensor(0.0009, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m7        \u001b[0m | \u001b[0m8.0      \u001b[0m | \u001b[0m0.0001   \u001b[0m | \u001b[0m0.06418  \u001b[0m | \u001b[0m0.05741  \u001b[0m |\n",
      "Number of negative values: 378\n",
      "0 Lyapunov Risk= 4.228706893627532e-05\n",
      "Number of negative values: 398\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.006327901440836785 0.06442716975261455 0.04743046291480434\n",
      "262 2 tensor(0.0196, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m8        \u001b[0m | \u001b[95m262.0    \u001b[0m | \u001b[95m0.006328 \u001b[0m | \u001b[95m0.06443  \u001b[0m | \u001b[95m0.04743  \u001b[0m |\n",
      "Number of negative values: 318\n",
      "0 Lyapunov Risk= 0.0008435804629698396\n",
      "Number of negative values: 324\n",
      "Number of negative values: 342\n",
      "Number of negative values: 352\n",
      "Number of negative values: 360\n",
      "Number of negative values: 362\n",
      "5 Lyapunov Risk= 4.893482764600776e-05\n",
      "Number of negative values: 364\n",
      "Number of negative values: 356\n",
      "Number of negative values: 358\n",
      "Number of negative values: 358\n",
      "0.003618584641257641 0.06042418027547408 0.04159455202354583\n",
      "4 5 tensor(0.0002, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m9        \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.003619 \u001b[0m | \u001b[0m0.06042  \u001b[0m | \u001b[0m0.04159  \u001b[0m |\n",
      "Number of negative values: 398\n",
      "0 Lyapunov Risk= 6.837379373791919e-08\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 366\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.0073013623202537355 0.06541587176999332 0.051309465541339194\n",
      "248 1 tensor(0.0209, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m10       \u001b[0m | \u001b[0m248.0    \u001b[0m | \u001b[0m0.007301 \u001b[0m | \u001b[0m0.06542  \u001b[0m | \u001b[0m0.05131  \u001b[0m |\n",
      "Number of negative values: 396\n",
      "0 Lyapunov Risk= 1.8012925693255966e-06\n",
      "Number of negative values: 400\n",
      "Number of negative values: 398\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.00968335554764932 0.058811582813709434 0.04946463895688504\n",
      "254 3 tensor(0.0200, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m11       \u001b[0m | \u001b[0m254.0    \u001b[0m | \u001b[0m0.009683 \u001b[0m | \u001b[0m0.05881  \u001b[0m | \u001b[0m0.04946  \u001b[0m |\n",
      "Number of negative values: 258\n",
      "0 Lyapunov Risk= 0.04967370629310608\n",
      "Number of negative values: 256\n",
      "Number of negative values: 256\n",
      "Number of negative values: 256\n",
      "Number of negative values: 258\n",
      "Number of negative values: 270\n",
      "5 Lyapunov Risk= 0.028328562155365944\n",
      "Number of negative values: 280\n",
      "Number of negative values: 286\n",
      "Number of negative values: 318\n",
      "Number of negative values: 380\n",
      "0.0001 0.01 0.1\n",
      "10 9 tensor(0.0027, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m12       \u001b[0m | \u001b[0m10.0     \u001b[0m | \u001b[0m0.0001   \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 324\n",
      "0 Lyapunov Risk= 0.0008383054519072175\n",
      "Number of negative values: 376\n",
      "Number of negative values: 384\n",
      "Number of negative values: 384\n",
      "Number of negative values: 374\n",
      "Number of negative values: 372\n",
      "5 Lyapunov Risk= 0.00011470183380879462\n",
      "Number of negative values: 372\n",
      "Number of negative values: 376\n",
      "Number of negative values: 378\n",
      "Number of negative values: 384\n",
      "0.009854144628330466 0.05852444901059572 0.056302292304539814\n",
      "6 2 tensor(0.0007, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m13       \u001b[0m | \u001b[0m6.0      \u001b[0m | \u001b[0m0.009854 \u001b[0m | \u001b[0m0.05852  \u001b[0m | \u001b[0m0.0563   \u001b[0m |\n",
      "Number of negative values: 400\n",
      "0 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.004631417265389613 0.06042000042187318 0.04998711062617212\n",
      "242 0 tensor(0.0198, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m14       \u001b[0m | \u001b[0m242.0    \u001b[0m | \u001b[0m0.004631 \u001b[0m | \u001b[0m0.06042  \u001b[0m | \u001b[0m0.04999  \u001b[0m |\n",
      "Number of negative values: 220\n",
      "0 Lyapunov Risk= 0.002491979394108057\n",
      "Number of negative values: 226\n",
      "Number of negative values: 254\n",
      "Number of negative values: 260\n",
      "Number of negative values: 296\n",
      "Number of negative values: 372\n",
      "5 Lyapunov Risk= 3.209813257853966e-06\n",
      "Number of negative values: 390\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 292\n",
      "Number of negative values: 286\n",
      "Number of negative values: 282\n",
      "0.0001 0.01 0.01\n",
      "2 0 tensor(1.7448e-05, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m15       \u001b[0m | \u001b[0m2.0      \u001b[0m | \u001b[0m0.0001   \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.01     \u001b[0m |\n",
      "Number of negative values: 318\n",
      "0 Lyapunov Risk= 0.0011942199198529124\n",
      "Number of negative values: 326\n",
      "Number of negative values: 326\n",
      "Number of negative values: 332\n",
      "Number of negative values: 332\n",
      "Number of negative values: 326\n",
      "5 Lyapunov Risk= 0.0006594772567041218\n",
      "Number of negative values: 330\n",
      "Number of negative values: 334\n",
      "Number of negative values: 336\n",
      "Number of negative values: 336\n",
      "0.005442412258083299 0.08734260223899526 0.05831929471494512\n",
      "4 7 tensor(0.0004, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m16       \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.005442 \u001b[0m | \u001b[0m0.08734  \u001b[0m | \u001b[0m0.05832  \u001b[0m |\n",
      "Number of negative values: 260\n",
      "0 Lyapunov Risk= 0.03934847190976143\n",
      "Number of negative values: 260\n",
      "Number of negative values: 258\n",
      "Number of negative values: 270\n",
      "Number of negative values: 274\n",
      "Number of negative values: 282\n",
      "5 Lyapunov Risk= 0.016556009650230408\n",
      "Number of negative values: 298\n",
      "Number of negative values: 330\n",
      "Number of negative values: 386\n",
      "Number of negative values: 374\n",
      "0.008846205725400125 0.04692411372970065 0.09788847786702917\n",
      "6 8 tensor(0.0021, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m17       \u001b[0m | \u001b[0m6.0      \u001b[0m | \u001b[0m0.008846 \u001b[0m | \u001b[0m0.04692  \u001b[0m | \u001b[0m0.09789  \u001b[0m |\n",
      "Number of negative values: 272\n",
      "0 Lyapunov Risk= 0.006636417470872402\n",
      "Number of negative values: 274\n",
      "Number of negative values: 274\n",
      "Number of negative values: 272\n",
      "Number of negative values: 276\n",
      "Number of negative values: 278\n",
      "5 Lyapunov Risk= 0.005305823870003223\n",
      "Number of negative values: 278\n",
      "Number of negative values: 280\n",
      "Number of negative values: 280\n",
      "Number of negative values: 280\n",
      "0.005667202114398779 0.08621748694363306 0.05020228359898451\n",
      "0 0 0\n",
      "| \u001b[0m18       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.005667 \u001b[0m | \u001b[0m0.08622  \u001b[0m | \u001b[0m0.0502   \u001b[0m |\n",
      "Number of negative values: 270\n",
      "0 Lyapunov Risk= 0.011494061909615993\n",
      "Number of negative values: 276\n",
      "Number of negative values: 284\n",
      "Number of negative values: 310\n",
      "Number of negative values: 362\n",
      "Number of negative values: 362\n",
      "5 Lyapunov Risk= 0.00026298273587599397\n",
      "Number of negative values: 352\n",
      "Number of negative values: 334\n",
      "Number of negative values: 326\n",
      "Number of negative values: 326\n",
      "0.007989413057315715 0.03789227346663301 0.06360490865766633\n",
      "0 0 0\n",
      "| \u001b[0m19       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.007989 \u001b[0m | \u001b[0m0.03789  \u001b[0m | \u001b[0m0.0636   \u001b[0m |\n",
      "Number of negative values: 388\n",
      "0 Lyapunov Risk= 8.047656592680141e-06\n",
      "Number of negative values: 396\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.009646856404543193 0.053387533944397006 0.04613896892654326\n",
      "254 4 tensor(0.0173, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m20       \u001b[0m | \u001b[0m254.0    \u001b[0m | \u001b[0m0.009647 \u001b[0m | \u001b[0m0.05339  \u001b[0m | \u001b[0m0.04614  \u001b[0m |\n",
      "Number of negative values: 378\n",
      "0 Lyapunov Risk= 4.114294279133901e-05\n",
      "Number of negative values: 392\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 398\n",
      "5 Lyapunov Risk= 8.104346989057376e-07\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.009414517325907484 0.04706260490202406 0.040505525323379574\n",
      "244 9 tensor(0.0127, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m21       \u001b[0m | \u001b[0m244.0    \u001b[0m | \u001b[0m0.009415 \u001b[0m | \u001b[0m0.04706  \u001b[0m | \u001b[0m0.04051  \u001b[0m |\n",
      "Number of negative values: 248\n",
      "0 Lyapunov Risk= 0.0021577037405222654\n",
      "Number of negative values: 256\n",
      "Number of negative values: 262\n",
      "Number of negative values: 294\n",
      "Number of negative values: 342\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 334\n",
      "Number of negative values: 314\n",
      "Number of negative values: 312\n",
      "0.002613196453286354 0.017690482369735437 0.014582305639612611\n",
      "262 5 tensor(0.0018, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m22       \u001b[0m | \u001b[0m262.0    \u001b[0m | \u001b[0m0.002613 \u001b[0m | \u001b[0m0.01769  \u001b[0m | \u001b[0m0.01458  \u001b[0m |\n",
      "Number of negative values: 254\n",
      "0 Lyapunov Risk= 0.0019702583085745573\n",
      "Number of negative values: 262\n",
      "Number of negative values: 288\n",
      "Number of negative values: 310\n",
      "Number of negative values: 366\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 358\n",
      "Number of negative values: 350\n",
      "0.004542022074600522 0.023626881005590645 0.018158608066423314\n",
      "262 5 tensor(0.0029, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m23       \u001b[0m | \u001b[0m262.0    \u001b[0m | \u001b[0m0.004542 \u001b[0m | \u001b[0m0.02363  \u001b[0m | \u001b[0m0.01816  \u001b[0m |\n",
      "Number of negative values: 272\n",
      "0 Lyapunov Risk= 0.0012992310803383589\n",
      "Number of negative values: 294\n",
      "Number of negative values: 332\n",
      "Number of negative values: 360\n",
      "Number of negative values: 368\n",
      "Number of negative values: 332\n",
      "5 Lyapunov Risk= 6.408517947420478e-05\n",
      "Number of negative values: 310\n",
      "Number of negative values: 310\n",
      "Number of negative values: 322\n",
      "Number of negative values: 390\n",
      "0.01 0.017512717397945222 0.018804906505218218\n",
      "6 3 tensor(9.0700e-05, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m24       \u001b[0m | \u001b[0m6.0      \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.01751  \u001b[0m | \u001b[0m0.0188   \u001b[0m |\n",
      "Number of negative values: 226\n",
      "0 Lyapunov Risk= 0.003302366938441992\n",
      "Number of negative values: 234\n",
      "Number of negative values: 242\n",
      "Number of negative values: 248\n",
      "Number of negative values: 258\n",
      "Number of negative values: 262\n",
      "5 Lyapunov Risk= 0.0008416585042141378\n",
      "Number of negative values: 260\n",
      "Number of negative values: 264\n",
      "Number of negative values: 268\n",
      "Number of negative values: 268\n",
      "0.0001 0.023960874408402128 0.013172273188839673\n",
      "2 0 tensor(3.1970e-05, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m25       \u001b[0m | \u001b[0m2.0      \u001b[0m | \u001b[0m0.0001   \u001b[0m | \u001b[0m0.02396  \u001b[0m | \u001b[0m0.01317  \u001b[0m |\n",
      "Number of negative values: 274\n",
      "0 Lyapunov Risk= 0.0010127845453098416\n",
      "Number of negative values: 312\n",
      "Number of negative values: 348\n",
      "Number of negative values: 374\n",
      "Number of negative values: 382\n",
      "Number of negative values: 332\n",
      "5 Lyapunov Risk= 7.675361121073365e-05\n",
      "Number of negative values: 312\n",
      "Number of negative values: 316\n",
      "Number of negative values: 336\n",
      "Number of negative values: 400\n",
      "0.0015881544451506844 0.01923412868589741 0.019703716532725034\n",
      "238 9 tensor(0.0028, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m26       \u001b[0m | \u001b[0m238.0    \u001b[0m | \u001b[0m0.001588 \u001b[0m | \u001b[0m0.01923  \u001b[0m | \u001b[0m0.0197   \u001b[0m |\n",
      "Number of negative values: 224\n",
      "0 Lyapunov Risk= 0.0026996899396181107\n",
      "Number of negative values: 238\n",
      "Number of negative values: 254\n",
      "Number of negative values: 262\n",
      "Number of negative values: 270\n",
      "Number of negative values: 296\n",
      "5 Lyapunov Risk= 0.0001242654980160296\n",
      "Number of negative values: 336\n",
      "Number of negative values: 380\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.0034729635185387297 0.016752824699881295 0.011992711001254612\n",
      "256 8 tensor(0.0012, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m27       \u001b[0m | \u001b[0m256.0    \u001b[0m | \u001b[0m0.003473 \u001b[0m | \u001b[0m0.01675  \u001b[0m | \u001b[0m0.01199  \u001b[0m |\n",
      "Number of negative values: 308\n",
      "0 Lyapunov Risk= 0.0006896060658618808\n",
      "Number of negative values: 334\n",
      "Number of negative values: 366\n",
      "Number of negative values: 390\n",
      "Number of negative values: 390\n",
      "Number of negative values: 338\n",
      "5 Lyapunov Risk= 7.135380292311311e-05\n",
      "Number of negative values: 332\n",
      "Number of negative values: 350\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.0033235216889222824 0.02531128153024486 0.023636294477894525\n",
      "246 9 tensor(0.0044, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m28       \u001b[0m | \u001b[0m246.0    \u001b[0m | \u001b[0m0.003324 \u001b[0m | \u001b[0m0.02531  \u001b[0m | \u001b[0m0.02364  \u001b[0m |\n",
      "Number of negative values: 256\n",
      "0 Lyapunov Risk= 0.0021789956372231245\n",
      "Number of negative values: 266\n",
      "Number of negative values: 280\n",
      "Number of negative values: 300\n",
      "Number of negative values: 320\n",
      "Number of negative values: 344\n",
      "5 Lyapunov Risk= 3.99622404074762e-05\n",
      "Number of negative values: 366\n",
      "Number of negative values: 388\n",
      "Number of negative values: 400\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 400\n",
      "0.007743496114585722 0.02941623505723045 0.020556420279398087\n",
      "264 8 tensor(0.0036, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m29       \u001b[0m | \u001b[95m264.0    \u001b[0m | \u001b[95m0.007743 \u001b[0m | \u001b[95m0.02942  \u001b[0m | \u001b[95m0.02056  \u001b[0m |\n",
      "Number of negative values: 286\n",
      "0 Lyapunov Risk= 0.0011879135854542255\n",
      "Number of negative values: 300\n",
      "Number of negative values: 336\n",
      "Number of negative values: 380\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 374\n",
      "Number of negative values: 348\n",
      "0.004509441277720761 0.03353083343362212 0.025369599202900932\n",
      "262 4 tensor(0.0057, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m30       \u001b[0m | \u001b[0m262.0    \u001b[0m | \u001b[0m0.004509 \u001b[0m | \u001b[0m0.03353  \u001b[0m | \u001b[0m0.02537  \u001b[0m |\n",
      "Number of negative values: 324\n",
      "0 Lyapunov Risk= 0.00045448061428032815\n",
      "Number of negative values: 352\n",
      "Number of negative values: 376\n",
      "Number of negative values: 392\n",
      "Number of negative values: 382\n",
      "Number of negative values: 340\n",
      "5 Lyapunov Risk= 9.554490679875016e-05\n",
      "Number of negative values: 350\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.01 0.030189390862717054 0.027971274131922342\n",
      "248 9 tensor(0.0063, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m31       \u001b[0m | \u001b[0m248.0    \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.03019  \u001b[0m | \u001b[0m0.02797  \u001b[0m |\n",
      "Number of negative values: 262\n",
      "0 Lyapunov Risk= 0.002852467354387045\n",
      "Number of negative values: 262\n",
      "Number of negative values: 270\n",
      "Number of negative values: 280\n",
      "Number of negative values: 284\n",
      "Number of negative values: 292\n",
      "5 Lyapunov Risk= 0.0007237500976771116\n",
      "Number of negative values: 286\n",
      "Number of negative values: 288\n",
      "Number of negative values: 294\n",
      "Number of negative values: 296\n",
      "0.01 0.03738058526044638 0.023240089159910923\n",
      "2 0 tensor(7.0053e-05, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m32       \u001b[0m | \u001b[0m2.0      \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.03738  \u001b[0m | \u001b[0m0.02324  \u001b[0m |\n",
      "Number of negative values: 352\n",
      "0 Lyapunov Risk= 0.00017339082842227072\n",
      "Number of negative values: 374\n",
      "Number of negative values: 386\n",
      "Number of negative values: 392\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.0034188851726831993 0.02987692895521817 0.030501256858838824\n",
      "246 5 tensor(0.0073, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m33       \u001b[0m | \u001b[0m246.0    \u001b[0m | \u001b[0m0.003419 \u001b[0m | \u001b[0m0.02988  \u001b[0m | \u001b[0m0.0305   \u001b[0m |\n",
      "Number of negative values: 260\n",
      "0 Lyapunov Risk= 0.0021085545886307955\n",
      "Number of negative values: 268\n",
      "Number of negative values: 282\n",
      "Number of negative values: 296\n",
      "Number of negative values: 316\n",
      "Number of negative values: 326\n",
      "5 Lyapunov Risk= 0.00010253483196720481\n",
      "Number of negative values: 340\n",
      "Number of negative values: 344\n",
      "Number of negative values: 348\n",
      "Number of negative values: 352\n",
      "0.0005438729092449356 0.031234025558638738 0.021385726544103836\n",
      "4 7 tensor(4.7226e-05, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m34       \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.0005439\u001b[0m | \u001b[0m0.03123  \u001b[0m | \u001b[0m0.02139  \u001b[0m |\n",
      "Number of negative values: 322\n",
      "0 Lyapunov Risk= 0.0005608092760667205\n",
      "Number of negative values: 348\n",
      "Number of negative values: 372\n",
      "Number of negative values: 388\n",
      "Number of negative values: 368\n",
      "Number of negative values: 336\n",
      "5 Lyapunov Risk= 9.862283332040533e-05\n",
      "Number of negative values: 352\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.008122426427332302 0.026639513988285478 0.02565067389388108\n",
      "254 9 tensor(0.0054, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m35       \u001b[0m | \u001b[0m254.0    \u001b[0m | \u001b[0m0.008122 \u001b[0m | \u001b[0m0.02664  \u001b[0m | \u001b[0m0.02565  \u001b[0m |\n",
      "Number of negative values: 350\n",
      "0 Lyapunov Risk= 0.0002049130416708067\n",
      "Number of negative values: 378\n",
      "Number of negative values: 396\n",
      "Number of negative values: 398\n",
      "Number of negative values: 380\n",
      "Number of negative values: 362\n",
      "5 Lyapunov Risk= 2.8119391572545283e-05\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.004682999103717378 0.03595633231322111 0.03173385018047343\n",
      "250 9 tensor(0.0080, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m36       \u001b[0m | \u001b[0m250.0    \u001b[0m | \u001b[0m0.004683 \u001b[0m | \u001b[0m0.03596  \u001b[0m | \u001b[0m0.03173  \u001b[0m |\n",
      "Number of negative values: 374\n",
      "0 Lyapunov Risk= 7.326235208893195e-05\n",
      "Number of negative values: 378\n",
      "Number of negative values: 390\n",
      "Number of negative values: 398\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.00813368044044381 0.03928884427590702 0.03868642486748145\n",
      "246 4 tensor(0.0117, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m37       \u001b[0m | \u001b[0m246.0    \u001b[0m | \u001b[0m0.008134 \u001b[0m | \u001b[0m0.03929  \u001b[0m | \u001b[0m0.03869  \u001b[0m |\n",
      "Number of negative values: 352\n",
      "0 Lyapunov Risk= 0.00019184278789907694\n",
      "Number of negative values: 372\n",
      "Number of negative values: 380\n",
      "Number of negative values: 390\n",
      "Number of negative values: 398\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 396\n",
      "Number of negative values: 388\n",
      "0.009168602643861191 0.0315167481063207 0.03649018990099349\n",
      "254 7 tensor(0.0108, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m38       \u001b[0m | \u001b[0m254.0    \u001b[0m | \u001b[0m0.009169 \u001b[0m | \u001b[0m0.03152  \u001b[0m | \u001b[0m0.03649  \u001b[0m |\n",
      "Number of negative values: 222\n",
      "0 Lyapunov Risk= 0.030821511521935463\n",
      "Number of negative values: 222\n",
      "Number of negative values: 222\n",
      "Number of negative values: 222\n",
      "Number of negative values: 224\n",
      "Number of negative values: 224\n",
      "5 Lyapunov Risk= 0.028979897499084473\n",
      "Number of negative values: 224\n",
      "Number of negative values: 224\n",
      "Number of negative values: 224\n",
      "Number of negative values: 224\n",
      "0.0014278238055605084 0.09534771301269847 0.024274724007999612\n",
      "0 0 0\n",
      "| \u001b[0m39       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.001428 \u001b[0m | \u001b[0m0.09535  \u001b[0m | \u001b[0m0.02427  \u001b[0m |\n",
      "Number of negative values: 282\n",
      "0 Lyapunov Risk= 0.006394452881067991\n",
      "Number of negative values: 292\n",
      "Number of negative values: 316\n",
      "Number of negative values: 372\n",
      "Number of negative values: 378\n",
      "Number of negative values: 358\n",
      "5 Lyapunov Risk= 0.0003177727048750967\n",
      "Number of negative values: 348\n",
      "Number of negative values: 340\n",
      "Number of negative values: 338\n",
      "Number of negative values: 338\n",
      "0.0025268210137301557 0.048366366132118616 0.059897571883420826\n",
      "4 4 tensor(0.0003, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m40       \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.002527 \u001b[0m | \u001b[0m0.04837  \u001b[0m | \u001b[0m0.0599   \u001b[0m |\n",
      "Number of negative values: 334\n",
      "0 Lyapunov Risk= 0.00035064734402112663\n",
      "Number of negative values: 362\n",
      "Number of negative values: 368\n",
      "Number of negative values: 368\n",
      "Number of negative values: 372\n",
      "Number of negative values: 382\n",
      "5 Lyapunov Risk= 1.7699900126899593e-05\n",
      "Number of negative values: 392\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.0021295458821954747 0.03366461633588372 0.039707279254232654\n",
      "250 7 tensor(0.0126, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m41       \u001b[0m | \u001b[0m250.0    \u001b[0m | \u001b[0m0.00213  \u001b[0m | \u001b[0m0.03366  \u001b[0m | \u001b[0m0.03971  \u001b[0m |\n",
      "Number of negative values: 382\n",
      "0 Lyapunov Risk= 2.3722517653368413e-05\n",
      "Number of negative values: 384\n",
      "Number of negative values: 398\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 398\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.000633181338900593 0.042371325601307075 0.038749447249269035\n",
      "254 9 tensor(0.0123, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m42       \u001b[0m | \u001b[0m254.0    \u001b[0m | \u001b[0m0.0006332\u001b[0m | \u001b[0m0.04237  \u001b[0m | \u001b[0m0.03875  \u001b[0m |\n",
      "Number of negative values: 320\n",
      "0 Lyapunov Risk= 0.0005008410080336034\n",
      "Number of negative values: 350\n",
      "Number of negative values: 374\n",
      "Number of negative values: 384\n",
      "Number of negative values: 360\n",
      "Number of negative values: 340\n",
      "5 Lyapunov Risk= 7.259233098011464e-05\n",
      "Number of negative values: 382\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 400\n",
      "0.00817400181014424 0.02665947260555636 0.026169200517869404\n",
      "246 9 tensor(0.0054, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m43       \u001b[0m | \u001b[0m246.0    \u001b[0m | \u001b[0m0.008174 \u001b[0m | \u001b[0m0.02666  \u001b[0m | \u001b[0m0.02617  \u001b[0m |\n",
      "Number of negative values: 288\n",
      "0 Lyapunov Risk= 0.002195433946326375\n",
      "Number of negative values: 316\n",
      "Number of negative values: 358\n",
      "Number of negative values: 352\n",
      "Number of negative values: 342\n",
      "Number of negative values: 338\n",
      "5 Lyapunov Risk= 0.0005596743430942297\n",
      "Number of negative values: 346\n",
      "Number of negative values: 354\n",
      "Number of negative values: 368\n",
      "Number of negative values: 386\n",
      "0.004709284018894173 0.0247110462162633 0.04152813147937709\n",
      "6 2 tensor(0.0004, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m44       \u001b[0m | \u001b[0m6.0      \u001b[0m | \u001b[0m0.004709 \u001b[0m | \u001b[0m0.02471  \u001b[0m | \u001b[0m0.04153  \u001b[0m |\n",
      "Number of negative values: 312\n",
      "0 Lyapunov Risk= 0.0009906303603202105\n",
      "Number of negative values: 362\n",
      "Number of negative values: 374\n",
      "Number of negative values: 366\n",
      "Number of negative values: 360\n",
      "Number of negative values: 366\n",
      "5 Lyapunov Risk= 0.0001563294936204329\n",
      "Number of negative values: 370\n",
      "Number of negative values: 380\n",
      "Number of negative values: 390\n",
      "Number of negative values: 400\n",
      "0.001398769590296213 0.039402833360974394 0.044829282854685876\n",
      "254 9 tensor(0.0163, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m45       \u001b[0m | \u001b[0m254.0    \u001b[0m | \u001b[0m0.001399 \u001b[0m | \u001b[0m0.0394   \u001b[0m | \u001b[0m0.04483  \u001b[0m |\n",
      "Number of negative values: 378\n",
      "0 Lyapunov Risk= 2.6963558411807753e-05\n",
      "Number of negative values: 396\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 398\n",
      "Number of negative values: 394\n",
      "Number of negative values: 398\n",
      "0.0025944841363335545 0.04732646488613161 0.04448249593121601\n",
      "258 6 tensor(0.0166, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m46       \u001b[0m | \u001b[0m258.0    \u001b[0m | \u001b[0m0.002594 \u001b[0m | \u001b[0m0.04733  \u001b[0m | \u001b[0m0.04448  \u001b[0m |\n",
      "Number of negative values: 256\n",
      "0 Lyapunov Risk= 0.02126043289899826\n",
      "Number of negative values: 266\n",
      "Number of negative values: 270\n",
      "Number of negative values: 272\n",
      "Number of negative values: 294\n",
      "Number of negative values: 316\n",
      "5 Lyapunov Risk= 0.0014209088403731585\n",
      "Number of negative values: 368\n",
      "Number of negative values: 354\n",
      "Number of negative values: 332\n",
      "Number of negative values: 316\n",
      "0.01 0.01 0.0708438296588689\n",
      "4 6 tensor(0.0004, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m47       \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.07084  \u001b[0m |\n",
      "Number of negative values: 218\n",
      "0 Lyapunov Risk= 0.017677120864391327\n",
      "Number of negative values: 220\n",
      "Number of negative values: 220\n",
      "Number of negative values: 220\n",
      "Number of negative values: 220\n",
      "Number of negative values: 220\n",
      "5 Lyapunov Risk= 0.015734754502773285\n",
      "Number of negative values: 220\n",
      "Number of negative values: 220\n",
      "Number of negative values: 220\n",
      "Number of negative values: 220\n",
      "0.0001 0.06560366626103087 0.01\n",
      "0 0 0\n",
      "| \u001b[0m48       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.0001   \u001b[0m | \u001b[0m0.0656   \u001b[0m | \u001b[0m0.01     \u001b[0m |\n",
      "Number of negative values: 294\n",
      "0 Lyapunov Risk= 0.0010869719553738832\n",
      "Number of negative values: 318\n",
      "Number of negative values: 330\n",
      "Number of negative values: 352\n",
      "Number of negative values: 384\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 390\n",
      "Number of negative values: 384\n",
      "0.004115543042333076 0.04836615467315246 0.03381476218641649\n",
      "264 5 tensor(0.0099, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m49       \u001b[0m | \u001b[0m264.0    \u001b[0m | \u001b[0m0.004116 \u001b[0m | \u001b[0m0.04837  \u001b[0m | \u001b[0m0.03381  \u001b[0m |\n",
      "Number of negative values: 254\n",
      "0 Lyapunov Risk= 0.004503373056650162\n",
      "Number of negative values: 260\n",
      "Number of negative values: 260\n",
      "Number of negative values: 264\n",
      "Number of negative values: 258\n",
      "Number of negative values: 264\n",
      "5 Lyapunov Risk= 0.0026082461699843407\n",
      "Number of negative values: 266\n",
      "Number of negative values: 268\n",
      "Number of negative values: 268\n",
      "Number of negative values: 268\n",
      "0.0001 0.049355273689113036 0.02715558562186497\n",
      "0 0 0\n",
      "| \u001b[0m50       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.0001   \u001b[0m | \u001b[0m0.04936  \u001b[0m | \u001b[0m0.02716  \u001b[0m |\n",
      "Number of negative values: 292\n",
      "0 Lyapunov Risk= 0.001201893319375813\n",
      "Number of negative values: 310\n",
      "Number of negative values: 324\n",
      "Number of negative values: 346\n",
      "Number of negative values: 372\n",
      "Number of negative values: 390\n",
      "5 Lyapunov Risk= 1.2125989314881735e-06\n",
      "Number of negative values: 396\n",
      "Number of negative values: 392\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.009906569224303528 0.04881484789069467 0.033922657104765885\n",
      "262 8 tensor(0.0098, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m51       \u001b[0m | \u001b[0m262.0    \u001b[0m | \u001b[0m0.009907 \u001b[0m | \u001b[0m0.04881  \u001b[0m | \u001b[0m0.03392  \u001b[0m |\n",
      "Number of negative values: 398\n",
      "0 Lyapunov Risk= 2.886271444069166e-09\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 398\n",
      "5 Lyapunov Risk= 5.045430384598149e-07\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.01 0.1 0.08007021046044706\n",
      "256 4 tensor(0.0537, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m52       \u001b[0m | \u001b[0m256.0    \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.08007  \u001b[0m |\n",
      "Number of negative values: 400\n",
      "0 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "5 Lyapunov Risk= 0.0\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "Number of negative values: 400\n",
      "0.0019446811596267172 0.09809826369280511 0.07814957509382334\n",
      "238 0 tensor(0.0429, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m53       \u001b[0m | \u001b[0m238.0    \u001b[0m | \u001b[0m0.001945 \u001b[0m | \u001b[0m0.0981   \u001b[0m | \u001b[0m0.07815  \u001b[0m |\n",
      "Number of negative values: 280\n",
      "0 Lyapunov Risk= 0.0009122740011662245\n",
      "Number of negative values: 318\n",
      "Number of negative values: 350\n",
      "Number of negative values: 378\n",
      "Number of negative values: 370\n",
      "Number of negative values: 322\n",
      "5 Lyapunov Risk= 0.000128516141558066\n",
      "Number of negative values: 314\n",
      "Number of negative values: 330\n",
      "Number of negative values: 394\n",
      "Number of negative values: 400\n",
      "0.0014566734534298515 0.0191729635038893 0.0201263590551137\n",
      "242 9 tensor(0.0031, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m54       \u001b[0m | \u001b[0m242.0    \u001b[0m | \u001b[0m0.001457 \u001b[0m | \u001b[0m0.01917  \u001b[0m | \u001b[0m0.02013  \u001b[0m |\n",
      "Number of negative values: 254\n",
      "0 Lyapunov Risk= 0.008597949519753456\n",
      "Number of negative values: 254\n",
      "Number of negative values: 256\n",
      "Number of negative values: 250\n",
      "Number of negative values: 254\n",
      "Number of negative values: 254\n",
      "5 Lyapunov Risk= 0.0068756407126784325\n",
      "Number of negative values: 258\n",
      "Number of negative values: 258\n",
      "Number of negative values: 258\n",
      "Number of negative values: 258\n",
      "0.00824316443914667 0.0694178156266787 0.03492757507749461\n",
      "0 0 0\n",
      "| \u001b[0m55       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.008243 \u001b[0m | \u001b[0m0.06942  \u001b[0m | \u001b[0m0.03493  \u001b[0m |\n",
      "=============================================================\n",
      "Final result: {'target': 264.0, 'params': {'min_a': 0.007743496114585722, 'min_b': 0.02941623505723045, 'min_c': 0.020556420279398087}}\n",
      "Time consumed: 198.1597410510294\n"
     ]
    }
   ],
   "source": [
    "from sklearn.datasets import make_classification\n",
    "from sklearn.model_selection import cross_val_score\n",
    "from sklearn.ensemble import RandomForestClassifier as RFC\n",
    "from sklearn.svm import SVC\n",
    "\n",
    "from bayes_opt import BayesianOptimization\n",
    "from bayes_opt.util import Colours\n",
    "\n",
    "\n",
    "def optimize_region():\n",
    "    \"\"\"Apply Bayesian Optimization to stability region of LNN.\"\"\"\n",
    "\n",
    "    optimizer = BayesianOptimization(\n",
    "        f=search_LNN_p,\n",
    "        pbounds={\"min_a\": (1e-4, 1e-2), \"min_b\": (1e-2, 1e-1), \"min_c\": (1e-2, 1e-1)},   \n",
    "        random_state=1234,\n",
    "        verbose=2 \n",
    "    )\n",
    "    optimizer.maximize(n_iter=50)\n",
    "\n",
    "    print(\"Final result:\", optimizer.max)\n",
    "\n",
    "start = timeit.default_timer()\n",
    "\n",
    "if __name__ == \"__main__\":\n",
    "\n",
    "    print(Colours.yellow(\"--- Optimizing Number of points ---\"))\n",
    "    optimize_region()\n",
    "end = timeit.default_timer()\n",
    "print(\"Time consumed: \" + str(end - start))\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Optimal LNN obtained and its performance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "#optimal parameters\n",
    "min_a = 0.007743496114585722     \n",
    "min_b = 0.02941623505723045  \n",
    "min_c = 0.020556420279398087"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 256\n",
      "0 Lyapunov Risk= 0.0021789956372231245\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.1522, 0.0077],\n",
      "        [0.1839, 0.0077],\n",
      "        [0.2921, 0.1583],\n",
      "        [0.0077, 0.0077]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.1042, 0.0077, 0.0557, 0.0232]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0294]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0206]])\n",
      "Number of negative values: 266\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.1422, 0.0077],\n",
      "        [0.1739, 0.0077],\n",
      "        [0.2822, 0.1487],\n",
      "        [0.0077, 0.0077]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.0942, 0.0077, 0.0459, 0.0134]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0294]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0206]])\n",
      "Number of negative values: 280\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.1324, 0.0077],\n",
      "        [0.1639, 0.0077],\n",
      "        [0.2727, 0.1396],\n",
      "        [0.0077, 0.0077]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.0845, 0.0077, 0.0363, 0.0077]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0294]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0206]])\n",
      "Number of negative values: 300\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.1228, 0.0077],\n",
      "        [0.1540, 0.0077],\n",
      "        [0.2638, 0.1313],\n",
      "        [0.0077, 0.0077]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.0751, 0.0077, 0.0271, 0.0077]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0294]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0206]])\n",
      "Number of negative values: 320\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/wulab2/.local/lib/python3.8/site-packages/torch/autograd/__init__.py:251: UserWarning: CUDA initialization: The NVIDIA driver on your system is too old (found version 11060). Please update your GPU driver by downloading and installing a new version from the URL: http://www.nvidia.com/Download/index.aspx Alternatively, go to: https://pytorch.org to install a PyTorch version that has been compiled with your version of the CUDA driver. (Triggered internally at ../c10/cuda/CUDAFunctions.cpp:108.)\n",
      "  Variable._execution_engine.run_backward(  # Calls into the C++ engine to run the backward pass\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.1138, 0.0077],\n",
      "        [0.1445, 0.0077],\n",
      "        [0.2557, 0.1240],\n",
      "        [0.0077, 0.0077]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.0663, 0.0077, 0.0187, 0.0077]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0294]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0206]])\n",
      "Number of negative values: 344\n",
      "5 Lyapunov Risk= 3.99622404074762e-05\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.1056, 0.0077],\n",
      "        [0.1355, 0.0077],\n",
      "        [0.2484, 0.1176],\n",
      "        [0.0077, 0.0077]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.0583, 0.0077, 0.0110, 0.0077]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0294]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0206]])\n",
      "Number of negative values: 366\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.0983, 0.0077],\n",
      "        [0.1274, 0.0077],\n",
      "        [0.2420, 0.1119],\n",
      "        [0.0077, 0.0077]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.0512, 0.0077, 0.0077, 0.0077]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0294]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0206]])\n",
      "Number of negative values: 388\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.0918, 0.0077],\n",
      "        [0.1202, 0.0077],\n",
      "        [0.2363, 0.1070],\n",
      "        [0.0077, 0.0077]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.0449, 0.0077, 0.0077, 0.0077]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0294]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0206]])\n",
      "Number of negative values: 400\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.0860, 0.0077],\n",
      "        [0.1138, 0.0077],\n",
      "        [0.2312, 0.1026],\n",
      "        [0.0077, 0.0077]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.0393, 0.0077, 0.0077, 0.0077]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0294]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0206]])\n",
      "0.007743496114585722 0.02941623505723045 0.020556420279398087\n",
      "[2, 0, 0, 0, 0, 0, 0, 4, 264]\n"
     ]
    }
   ],
   "source": [
    "pho_list = list()\n",
    "NUM_list = list()\n",
    "out_iters = 0\n",
    "valid = False\n",
    "#develop model\n",
    "model = Net(D_in,H1, D_out)\n",
    "L = []\n",
    "t = 0\n",
    "max_iters = 9\n",
    "learning_rate = 0.01\n",
    "optimizer = torch.optim.Adam(model.parameters(), lr=learning_rate)\n",
    "\n",
    "#Assign initial guess for the parameter\n",
    "with torch.no_grad():       \n",
    "#the parameters will NOT be updated in the training phase\n",
    "    model.layer3.weight = torch.nn.Parameter(torch.ones_like(model.layer3.weight) * min_b) # CA^2\n",
    "    model.layer4.weight = torch.nn.Parameter(torch.ones_like(model.layer4.weight) * min_c) # T^2\n",
    "\n",
    "# Clamp all weights to be non-negative       \n",
    "with torch.no_grad():\n",
    "    for layer in [model.layer1, model.layer2]:\n",
    "        for param in layer.parameters():\n",
    "            param.data.clamp_(min = min_a) \n",
    "\n",
    "\n",
    "#train model, and search for the stability region after each epoch\n",
    "while out_iters < max_iters:\n",
    "    V_candidate = model(x)\n",
    "    X0 = model(x_0)\n",
    "    # Initialize a tensor to store the Jacobians for each sample\n",
    "    jacobian = torch.zeros(Num*Num, 2)  \n",
    "    jacobian = model.compute_jacobian(x)\n",
    "    # Compute lie derivative of V : L_V = ∑∂V/∂xᵢ*dx\n",
    "    dx = f_value(x) #accurate first-principles model\n",
    "\n",
    "    L_V = torch.diagonal(torch.mm(jacobian,dx.t()),0)\n",
    "\n",
    "    print(\"Number of negative values:\", (L_V <= 0).sum().item())\n",
    "\n",
    "    Lyapunov_risk = (F.relu(((L_V + 0.001*abs(V_candidate).T).T))).mean()\n",
    "    \n",
    "    if out_iters%5 == 0:\n",
    "        print(out_iters, \"Lyapunov Risk=\",Lyapunov_risk.item())\n",
    "\n",
    "\n",
    "    L.append(Lyapunov_risk.item())\n",
    "    optimizer.zero_grad()\n",
    "    Lyapunov_risk.backward()\n",
    "    optimizer.step() \n",
    "\n",
    "#             # Clamp all weights to be non-negative       \n",
    "    with torch.no_grad():\n",
    "        for layer in [model.layer1, model.layer2]:\n",
    "            for param in layer.parameters():\n",
    "                param.data.clamp_(min = min_a) \n",
    "         \n",
    "        \n",
    "    #print weight parameters in LNN\n",
    "    for name, param in model.named_parameters():\n",
    "        print(f\"{name}: {param.shape}\")\n",
    "        print(param.data)\n",
    "\n",
    "\n",
    "    out_iters+=1\n",
    "\n",
    "    #Search for the biggest pho via global search method\n",
    "    start = timeit.default_timer()\n",
    "    Num_points = 0\n",
    "    Index_pho = 0\n",
    "    Flag = True\n",
    "    pho = min(V_candidate)[0]\n",
    "    while(Flag):\n",
    "        pho = pho+min(V_candidate)[0]\n",
    "        V_point_list = list()\n",
    "        #get region with stable if and only if\n",
    "        for i in range (len(x_real)):  \n",
    "            if V_candidate[i][0] < pho:\n",
    "                if L_V[i] > - 0.001*V_candidate[i][0]:  #NO POINTS WITHIN THE REGION IS UNSTABLE\n",
    "                    Flag =False\n",
    "                    V_point_list = list()  #set the list as empty\n",
    "                    break\n",
    "                else:\n",
    "                    V_point_list.append(x_real[i])\n",
    "        #Check the boundary, via the magnitude \n",
    "        for k in V_point_list: \n",
    "            if abs(k[0])>=2 or abs(k[1])>=3.14:\n",
    "                #print(k)\n",
    "                Flag =False\n",
    "                V_point_list = list()  #set the list as empty\n",
    "                break \n",
    "        K = len(V_point_list)  #if number of samples increase, update the value of pho\n",
    "        if K > Num_points:\n",
    "            Num_points = K\n",
    "            Index_pho = pho\n",
    "    pho_list.append(Index_pho)\n",
    "    NUM_list.append(Num_points)\n",
    "\n",
    "    end = timeit.default_timer()\n",
    "print(min_a, min_b, min_c)\n",
    "print(NUM_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "number of V > 0 points:  400\n",
      "number of V < 0 points:  0\n"
     ]
    }
   ],
   "source": [
    "#identify the CE with V < 0\n",
    "V_point_list = list()\n",
    "uV_point_list = list()\n",
    "for i in range (len(x_real)):\n",
    "    if V_candidate[i][0] > 0:\n",
    "        V_point_list.append(x_real[i])\n",
    "    else:\n",
    "        uV_point_list.append(x_real[i]) \n",
    "        print(x_real[i],V_candidate[i][0])\n",
    "print(\"number of V > 0 points: \", len(V_point_list))\n",
    "print(\"number of V < 0 points: \", len(uV_point_list))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "number of \\dot V < 0 points:  400\n",
      "number of \\dot V > 0 points:  0\n"
     ]
    }
   ],
   "source": [
    "#identify the CE with \\dot V >0\n",
    "dV_point_list = list()\n",
    "udV_point_list = list()\n",
    "for i in range (len(x_real)):\n",
    "    if L_V[i] < - 0.001* V_candidate[i][0]: #L_V[i] <0: \n",
    "        dV_point_list.append(x_real[i])\n",
    "    else:\n",
    "        udV_point_list.append(x_real[i]) \n",
    "print(\"number of \\dot V < 0 points: \", len(dV_point_list))\n",
    "print(\"number of \\dot V > 0 points: \", len(udV_point_list))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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cli9Gp55ZgWXXXYNN900jaefns3IkSOZOHFiVt+PPPIIgwYNYtasWdx552TuvPMlbrvtRf73f39NXd3LLdZ/+eWXmTRpEgsWLODNN9/kueee46KLLmL33XfnwQfruPnmOl56aR6vvfYuzzwzn1deeYWzzjor2sFMR7pxoaQ2ZQIrE1iZwAnJBM6hHnTUTOC33357q/H4FSvcx469wr///Rt9xgz34cMP87vu+j9fscL9/fff97333tvd3adMmeIDBgzwa6+9tqnO/1tvveX9+x/Q1P8LL6z3k0/+rvfrN8gHDhziFRUVvmzZMn/rrbe8d+/eTX0+8sgjfvzxx7u7+8EHH+Z33z3DJ058xLt27eH77DPE9913iO+77/5+9tlnt/CfOgdw5JFH+iuvvOITJkzyc8/9r6b9P/vsn/gll9zsK1a477jjju4efHaOPPLIpu2MGzfO7777bnd37927j0+bttxnzHBfuHCx77HHXn7KKd/z++9/wjdt2pTT++hepnMAScjElD6/emUCJyQTOIcss6hZrD169GDVqlVb6des+Yidd+7Z9FqnTl14913Ydttt2bhxIwCnnXYaU6dOZfvtt+fYY4/lb3/7GwAbNmzp/4knprBq1XLuvnsWU6bMobq6mnXr1gFgtnVZ68a/168P9O7OF794FPfeO4cpU+bwwAMLuOuuu9LuQ+McwNNPP83AgQNZsyaIjKm4tzwGjU9Bg633bdOmLfqdd+7GvffOZfjwUdx+++2ce+656Q9kBEo6AMSdSSm99B1ZnzULNYcss6hZrJWVley2225NJ/Dlyz/ihReeZOjQr2TVv/nmm+y1115cdNFFHH/88cybN4+qqirWrt1SO7+hYQ3du3+OTp068/zzdSxOqWW9ZMkSXnjhBQAeeOABvvKVoL/G4DFo0JeYO/c53nknGNNfs2Ytr7/+evqdaMagQYfwzDN/Zt26T/j007VMn/4Qw4YdknMm7/bbb9mPlStXsnnzZg4//ETGjr2G2bNn57aRLJR0AIg7k1J66TuyPmsWag5Zam3JYv3973/P1VdfzdChQ/nudw/n3HOvoFevvbPq//CHPzBw4ECGDh3K/Pnz+fa3v02PHj0YNuzLnHLKQG6++VK++tUxLFw4k1NPHcQTT/ye/v37N+n3228/br31Vvbff39Wr17NBRdcAGxJwOrWbReuuOK3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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "# Extracting the x and y coordinates from the list Stable\n",
    "x_values = [point[0] for point in dV_point_list]\n",
    "y_values = [point[1] for point in dV_point_list]\n",
    "\n",
    "\n",
    "# Extracting the x and y coordinates from the list Unstable\n",
    "x_values_u = [point[0] for point in udV_point_list]\n",
    "y_values_u = [point[1] for point in udV_point_list]\n",
    "\n",
    "\n",
    "# Plotting the points\n",
    "plt.scatter(x_values, y_values, color='blue', label='Stable Points')\n",
    "\n",
    "\n",
    "# Plotting the points\n",
    "plt.scatter(x_values_u, y_values_u, color='red', label='Unstable Points')\n",
    "\n",
    "# Adding labels and title\n",
    "plt.xlabel('X-axis')\n",
    "plt.ylabel('Y-axis')\n",
    "plt.title('Plot of 2D unstable/Unstable Points')\n",
    "\n",
    "# Displaying the plot\n",
    "plt.legend()\n",
    "plt.grid(True)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "number of points within levelset:  264\n",
      "number of points beyond levelset:  136\n"
     ]
    },
    {
     "data": {
      "image/png": "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vb3dfdGibW39iy/mHu+h3tvb3det8169evmYMWP80EMP9bFjx/pDDz0UpO9KwWPtZehbW1t96tSpO0hbW1t911128TEHH+yHjBjhl8+cWXL8Sy+91O+///4dxl//4ovbxm698UZ/6Ac/CNp+yxYsSOQNWM7x7DTe/0oCZ7gpCVw9/0kmgZctXFiefskS90WLfO6Vf/bhe212s62+77C3fe7X/hqk36EtWeI777zzttUWLFjgxxxzTJC+KwUngDL0pSaAqRMnui9a5B0PPugH7ruvt//3fxcdv9jzX79mzbZxL/vsZ/2a884L0i+7995E3oDlTABpvP8bLQlc14eA0k7h1bo+MQPvvFOe/m9/49Z792Dm14bz1xf74W48v6oPM68cWt5u8N/+Vtbj69evZ/fdd9/29zXXXMMHPvABDj3lFC77zncA+PKNN3Ldj3+8TT9r1iyuv/563J2LLrqID37wg4wePZrbbrsNgLa2NiafeSan/t//yyGnnsrpX/pS7hsXsKCtjUMOOYRx48Zx5513Fvcfrb/zgAGMP+QQnnn+eRY/+SQfOv10Dj30UE4++WRee+01AGbMmMEdd9wBwIgRI7jssssYd9ppfKilhRXPPsuzL7zAjfPnc+1PfsJhp57Kb37zG26//XZGjRrFmDFjOOaYY8rffjWSxE1bnxUPpajrCSDtFF6t6xMz0Lt3efqddmLWt4eyafP262/a3HvbddHd6Ys9/uabb3LYYYdxyCGHcPbZZ3PppZcCsHDhQp5++mn+9Kc/sfj222lfsYIHH3mEz3z0o/zol78EYGufPsybN49p06Zx5513snjxYn73u9/xwAMPcNFFF7FmzRoAHn3qKa674AKW/fSnrHzhBR5asoTNb73FZ7/2Ne655x7a29t58cUXi/uPTsK+8vrr/GHpUkYecABnXH45V59/Po899hijR4/miiuuKCgdMmQIj/z0p5w1fTqz585lxD77cO4pp3D+pz7F4jvuYOLEiVx55ZXcd999LFmyhLvvvrv87VcjSdy09VnxUIq6ngDSTuHVuj4xA3nftksydCjPvVT4Q6isb0BDh+6YhO3VC4YOZcCAASxevJgVK1awYMECzjjjDNydhQsXsnDhQsaOHcu4T3+aFc8+y9PPP8+IffZh8KBBPPrUUyx86inGjh3L4MGD+e1vf8unPvUpevfuTXNzM5MmTWLRokUAHD5+PMP23ptevXpx2MEH8+wLL7DiuefYf//9OeiggzAzpk2bVtT+bxYvZuy0aRz/hS9w8fTpDGtu5vUNG5h00kkATJ8+nQcffLCg9uMf/zgMHcphY8bw7AsvvLvALLddgKOOOooZM2bw3e9+l3cK7ZUNHbptEtpGDSVx09ZnxUMp6noCOP10uOmmXHoOcj9vuikshdepN2s8fWIGdt65PP3gwew3dGvBRWV9Axo8ODdm5zfZnXbK/T148HarHXHEEaxbt461a9fi7lxyySUsXryYxY8/zjOLF3PWJz4BwNmnnMIPf/1rbp4/n8985jPdDt+vqWnb+L179WKLGey9N/TtW4Z5mHjMMTz60EO033Yb555ySk7Xp88O/guO3a8fDB5M73792BIdStqWBI70N954I1/5yld4/vnnGT9+PK90vUJo8OB3t2EKb8C03/8V///E7AF65qEkhU4MZLUpCVw9kvS1bNmysteN40qMQuSfBF6+fLkPHjzYt2zZ4vfdd58ffvjhvmHDBnd3X7Vqlb/00kvu7v7WW2/5wQcf7Pvvv79v2bLF3d3nz5/vxx9/vL/22mv+8ssv+3777edr1qzZ4eTu5z73Ob/55pv9zTff9H333defeeYZd3f/5Cc/WfwkcIHHDz30UH/wwQfd3f2yyy7zL37xi+7uPn36dL/99tvd3X348OG+du1ad3dva2vzSZMmubv77Nmz/ctf/vK2vjo9uLtPmDDBH3300R3GC3mtQmjE932lJJEErutaQKL26VpLZdiwrVx1Va+KvwF1ngOA3JegW265hd69e3P88cezfPlyjjjiCACampqYO3cu73nPe9hpp51oaWlht912o3d0HuPkk0/m97//PUceeSS9e/fmG9/4BnvttRcrVqwoOG7//v256aabmDp1KgMHDmTixIls2LChbN+33HIL5557Lps2beKAAw7g5ptvLlv7j//4j5x66qncdddd3HDDDVx77bU8/fTTuDvHHnssY8aMKbsvUScUmhWy2rQHUD2ysgfQlTRr7rzzzjs+ZswYf+qpp3ZYVm+1gDrRHkB2UC0gIVJi2bJlHHjggRx77LEcdNBBadsRIhbqfgJQKYja1melFMT7N21i5c9/zjf//d9TGT+tUhApV4KoeX2cHlQKQqUgqloKIqnxFy0q87BCgqUg4tb3tBRE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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "##identify the stability region\n",
    "iV_point_list = list()\n",
    "uV_point_list = list()\n",
    "n = 0.00363 #pho obtained\n",
    "for i in range (len(x_real)):\n",
    "    if V_candidate[i][0] > n:\n",
    "        iV_point_list.append(x_real[i])\n",
    "    else:\n",
    "        uV_point_list.append(x_real[i]) \n",
    "print(\"number of points within levelset: \", len(uV_point_list))\n",
    "print(\"number of points beyond levelset: \", len(iV_point_list))\n",
    "\n",
    "# Extracting the x and y coordinates from the list Beyond levelset\n",
    "x_values = [point[0] for point in iV_point_list]\n",
    "y_values = [point[1] for point in iV_point_list]\n",
    "\n",
    "\n",
    "# Extracting the x and y coordinates from the list Within levelset\n",
    "x_values_b = [point[0] for point in uV_point_list]\n",
    "y_values_b = [point[1] for point in uV_point_list]\n",
    "\n",
    "# Plotting the points\n",
    "plt.scatter(x_values, y_values, color='Blue', label='Beyond Points')\n",
    "plt.scatter(x_values_b, y_values_b, color='Red', label='In Points')\n",
    "\n",
    "# Adding labels and title\n",
    "plt.xlabel('X-axis')\n",
    "plt.ylabel('Y-axis')\n",
    "plt.title('V <' + str(n))\n",
    "\n",
    "# Displaying the plot\n",
    "plt.legend()\n",
    "plt.grid(True)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "#plot the heatmap for V and \\dot V\n",
    "key_heatmap = 20"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x432 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "def convert_to_numpy(tensor_list):\n",
    "    return np.array([tensor.numpy() for tensor in tensor_list])\n",
    "\n",
    "x_values = [point[0] for point in V_point_list]\n",
    "y_values = [point[1] for point in V_point_list]\n",
    "\n",
    "# Convert the tensor lists to 2D arrays (grids for heatmaps)\n",
    "grid1 = convert_to_numpy(x_values).reshape(key_heatmap, key_heatmap)\n",
    "grid2 = convert_to_numpy(y_values).reshape(key_heatmap, key_heatmap)\n",
    "\n",
    "# Detach the tensor from the computation graph and convert to NumPy array\n",
    "grid3 = V_candidate.detach().numpy().reshape(key_heatmap, key_heatmap)\n",
    "\n",
    "# Create a 2D heatmap\n",
    "plt.figure(figsize=(8, 6))\n",
    "\n",
    "# Create the heatmap using Z values\n",
    "plt.imshow(grid3, cmap='coolwarm', extent=[grid1.min(), grid1.max(), grid2.min(), grid2.max()], origin='lower', aspect='auto')\n",
    "\n",
    "# Add a color bar to indicate the mapping of Z values to color\n",
    "plt.colorbar(label='V')\n",
    "\n",
    "# Add labels and title\n",
    "plt.xlabel('X-axis')\n",
    "plt.ylabel('Y-axis')\n",
    "plt.title('Heatmap of V at (X, Y)')\n",
    "\n",
    "# Show the plot\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x432 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "def convert_to_numpy(tensor_list):\n",
    "    return np.array([tensor.numpy() for tensor in tensor_list])\n",
    "\n",
    "x_values = [point[0] for point in V_point_list]\n",
    "y_values = [point[1] for point in V_point_list]\n",
    "\n",
    "# Convert the tensor lists to 2D arrays (grids for heatmaps)\n",
    "grid1 = convert_to_numpy(x_values).reshape(key_heatmap, key_heatmap)\n",
    "grid2 = convert_to_numpy(y_values).reshape(key_heatmap, key_heatmap)\n",
    "\n",
    "# Detach the tensor from the computation graph and convert to NumPy array\n",
    "grid3 = L_V.detach().numpy().reshape(key_heatmap, key_heatmap)\n",
    "\n",
    "# Create a 2D heatmap\n",
    "plt.figure(figsize=(8, 6))\n",
    "\n",
    "# Create the heatmap using Z values\n",
    "plt.imshow(grid3, cmap='coolwarm', extent=[grid1.min(), grid1.max(), grid2.min(), grid2.max()], origin='lower', aspect='auto')\n",
    "\n",
    "# Add a color bar to indicate the mapping of Z values to color\n",
    "plt.colorbar(label='dV')\n",
    "\n",
    "# Add labels and title\n",
    "plt.xlabel('X-axis')\n",
    "plt.ylabel('Y-axis')\n",
    "plt.title('Heatmap of dV at (X, Y)')\n",
    "\n",
    "# Show the plot\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Save data for matlab plot"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Save to text file\n",
    "np.savetxt('V_PF_tanh.txt', V_candidate.detach().numpy(), fmt='%.6f')  # value of V for each sample\n",
    "# Save to text file\n",
    "np.savetxt('LV_PF_tanh.txt', L_V.detach().numpy(), fmt='%.6f')  # value of \\\\dot V for each sample"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "from scipy.io import savemat\n",
    "\n",
    "\n",
    "# Convert tensors to NumPy array\n",
    "array_data1 = np.array([t.numpy() for t in iV_point_list]) #points outside the stability region\n",
    "array_data2 = np.array([t.numpy() for t in uV_point_list]) #points within the stability region\n",
    "\n",
    "\n",
    "# Save as .mat\n",
    "savemat('unstable_PF_tanh.mat', {'data': array_data1})\n",
    "savemat('stable_PF_tanh.mat', {'data': array_data2})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "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.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
