{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Codes for development of LNN 2 for Van der pol system\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)\n",
    "        "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Dynamical system of Van der Pol"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def f_value(x): #obtain \\dot x_1 and \\dot x_2\n",
    "    y = []\n",
    "    \n",
    "    for r in range(0,len(x)): \n",
    "        x1 = x[r][0]  \n",
    "        x2 = x[r][1]\n",
    "        dx1 = x2\n",
    "        dx2 = -x1 + (x1*x1-1)*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 0x7fa736c16ed0>"
      ]
     },
     "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. 2.]\n"
     ]
    }
   ],
   "source": [
    "Num = 20\n",
    "\n",
    "x1_list = np.linspace(-2, 2, Num, endpoint=True)\n",
    "x2_list = np.linspace(-2, 2, 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",
    "    \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.1\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) # x1^2\n",
    "        model.layer4.weight = torch.nn.Parameter(torch.ones_like(model.layer4.weight) * min_c) # x2^2\n",
    "\n",
    "      \n",
    "    # Clamp weight parameters in the hidden and output layers 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",
    "        \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",
    "\n",
    "        Lyapunov_risk = (F.relu(((L_V + 0.001*abs(V_candidate).T).T))).mean() #loss function for LNN\n",
    "\n",
    "        if out_iters%5 == 0:\n",
    "            print(out_iters, \"Lyapunov Risk=\",Lyapunov_risk.item())\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",
    "            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])>=2:\n",
    "                    Flag =False\n",
    "                    V_point_list = list()  \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: 242\n",
      "0 Lyapunov Risk= 0.00671195425093174\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: 212\n",
      "Number of negative values: 212\n",
      "Number of negative values: 210\n",
      "Number of negative values: 210\n",
      "Number of negative values: 210\n",
      "5 Lyapunov Risk= 0.006085623055696487\n",
      "Number of negative values: 210\n",
      "Number of negative values: 210\n",
      "Number of negative values: 210\n",
      "Number of negative values: 210\n",
      "0.01996042558751034 0.06598978939358487 0.04939549651064031\n",
      "8 1 tensor(0.0003, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m1        \u001b[0m | \u001b[0m8.0      \u001b[0m | \u001b[0m0.01996  \u001b[0m | \u001b[0m0.06599  \u001b[0m | \u001b[0m0.0494   \u001b[0m |\n",
      "Number of negative values: 218\n",
      "0 Lyapunov Risk= 0.011715477332472801\n",
      "Number of negative values: 222\n",
      "Number of negative values: 220\n",
      "Number of negative values: 216\n",
      "Number of negative values: 218\n",
      "Number of negative values: 218\n",
      "5 Lyapunov Risk= 0.009491742588579655\n",
      "Number of negative values: 220\n",
      "Number of negative values: 218\n",
      "Number of negative values: 220\n",
      "Number of negative values: 226\n",
      "0.07875049978766316 0.08019782273069231 0.03453333447543775\n",
      "8 9 tensor(0.0009, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m2        \u001b[0m | \u001b[0m8.0      \u001b[0m | \u001b[0m0.07875  \u001b[0m | \u001b[0m0.0802   \u001b[0m | \u001b[0m0.03453  \u001b[0m |\n",
      "Number of negative values: 232\n",
      "0 Lyapunov Risk= 0.021048778668045998\n",
      "Number of negative values: 230\n",
      "Number of negative values: 222\n",
      "Number of negative values: 212\n",
      "Number of negative values: 218\n",
      "Number of negative values: 210\n",
      "5 Lyapunov Risk= 0.021414365619421005\n",
      "Number of negative values: 214\n",
      "Number of negative values: 214\n",
      "Number of negative values: 214\n",
      "Number of negative values: 218\n",
      "0.028369961259166576 0.08216849597815173 0.09623254183153347\n",
      "108 9 tensor(0.0147, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m3        \u001b[0m | \u001b[95m108.0    \u001b[0m | \u001b[95m0.02837  \u001b[0m | \u001b[95m0.08217  \u001b[0m | \u001b[95m0.09623  \u001b[0m |\n",
      "Number of negative values: 268\n",
      "0 Lyapunov Risk= 0.008892777375876904\n",
      "Number of negative values: 250\n",
      "Number of negative values: 234\n",
      "Number of negative values: 228\n",
      "Number of negative values: 228\n",
      "Number of negative values: 226\n",
      "5 Lyapunov Risk= 0.012199081480503082\n",
      "Number of negative values: 224\n",
      "Number of negative values: 224\n",
      "Number of negative values: 218\n",
      "Number of negative values: 216\n",
      "0.08771733083946738 0.04220355429620801 0.05508956129711129\n",
      "134 0 tensor(0.0090, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m4        \u001b[0m | \u001b[95m134.0    \u001b[0m | \u001b[95m0.08772  \u001b[0m | \u001b[95m0.0422   \u001b[0m | \u001b[95m0.05509  \u001b[0m |\n",
      "Number of negative values: 240\n",
      "0 Lyapunov Risk= 0.009523462504148483\n",
      "Number of negative values: 242\n",
      "Number of negative values: 238\n",
      "Number of negative values: 240\n",
      "Number of negative values: 242\n",
      "Number of negative values: 242\n",
      "5 Lyapunov Risk= 0.007354037836194038\n",
      "Number of negative values: 240\n",
      "Number of negative values: 248\n",
      "Number of negative values: 256\n",
      "Number of negative values: 258\n",
      "0.06866283058204149 0.07414318242846102 0.04332256793113555\n",
      "8 8 tensor(0.0008, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m5        \u001b[0m | \u001b[0m8.0      \u001b[0m | \u001b[0m0.06866  \u001b[0m | \u001b[0m0.07414  \u001b[0m | \u001b[0m0.04332  \u001b[0m |\n",
      "Number of negative values: 268\n",
      "0 Lyapunov Risk= 0.00889339204877615\n",
      "Number of negative values: 250\n",
      "Number of negative values: 234\n",
      "Number of negative values: 228\n",
      "Number of negative values: 228\n",
      "Number of negative values: 226\n",
      "5 Lyapunov Risk= 0.012249409221112728\n",
      "Number of negative values: 224\n",
      "Number of negative values: 224\n",
      "Number of negative values: 218\n",
      "Number of negative values: 216\n",
      "0.08804085455144912 0.0413922620468548 0.055072772701722644\n",
      "136 0 tensor(0.0091, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m6        \u001b[0m | \u001b[95m136.0    \u001b[0m | \u001b[95m0.08804  \u001b[0m | \u001b[95m0.04139  \u001b[0m | \u001b[95m0.05507  \u001b[0m |\n",
      "Number of negative values: 240\n",
      "0 Lyapunov Risk= 0.02128443494439125\n",
      "Number of negative values: 248\n",
      "Number of negative values: 260\n",
      "Number of negative values: 258\n",
      "Number of negative values: 236\n",
      "Number of negative values: 226\n",
      "5 Lyapunov Risk= 0.020932381972670555\n",
      "Number of negative values: 214\n",
      "Number of negative values: 210\n",
      "Number of negative values: 210\n",
      "Number of negative values: 202\n",
      "0.09953611724799079 0.023175780059772205 0.0927821363510978\n",
      "68 2 tensor(0.0133, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m7        \u001b[0m | \u001b[0m68.0     \u001b[0m | \u001b[0m0.09954  \u001b[0m | \u001b[0m0.02318  \u001b[0m | \u001b[0m0.09278  \u001b[0m |\n",
      "Number of negative values: 194\n",
      "0 Lyapunov Risk= 0.008402272127568722\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 208\n",
      "Number of negative values: 206\n",
      "Number of negative values: 212\n",
      "5 Lyapunov Risk= 0.007742299232631922\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "Number of negative values: 218\n",
      "Number of negative values: 218\n",
      "0.1 0.015397041945197336 0.03222030581129184\n",
      "12 1 tensor(0.0004, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m8        \u001b[0m | \u001b[0m12.0     \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.0154   \u001b[0m | \u001b[0m0.03222  \u001b[0m |\n",
      "Number of negative values: 256\n",
      "0 Lyapunov Risk= 0.010645643807947636\n",
      "Number of negative values: 258\n",
      "Number of negative values: 252\n",
      "Number of negative values: 242\n",
      "Number of negative values: 226\n",
      "Number of negative values: 218\n",
      "5 Lyapunov Risk= 0.011553185060620308\n",
      "Number of negative values: 216\n",
      "Number of negative values: 212\n",
      "Number of negative values: 210\n",
      "Number of negative values: 208\n",
      "0.07644995495103142 0.030267361910377582 0.06679018529885676\n",
      "82 0 tensor(0.0055, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m9        \u001b[0m | \u001b[0m82.0     \u001b[0m | \u001b[0m0.07645  \u001b[0m | \u001b[0m0.03027  \u001b[0m | \u001b[0m0.06679  \u001b[0m |\n",
      "Number of negative values: 264\n",
      "0 Lyapunov Risk= 0.01230818871408701\n",
      "Number of negative values: 264\n",
      "Number of negative values: 238\n",
      "Number of negative values: 228\n",
      "Number of negative values: 228\n",
      "Number of negative values: 226\n",
      "5 Lyapunov Risk= 0.018202155828475952\n",
      "Number of negative values: 220\n",
      "Number of negative values: 214\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "0.1 0.042494565482526565 0.06610972704744802\n",
      "114 0 tensor(0.0097, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m10       \u001b[0m | \u001b[0m114.0    \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.04249  \u001b[0m | \u001b[0m0.06611  \u001b[0m |\n",
      "Number of negative values: 222\n",
      "0 Lyapunov Risk= 0.02285587787628174\n",
      "Number of negative values: 208\n",
      "Number of negative values: 196\n",
      "Number of negative values: 194\n",
      "Number of negative values: 196\n",
      "Number of negative values: 198\n",
      "5 Lyapunov Risk= 0.02331484481692314\n",
      "Number of negative values: 198\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "0.006505446426490665 0.1 0.1\n",
      "92 8 tensor(0.0136, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m11       \u001b[0m | \u001b[0m92.0     \u001b[0m | \u001b[0m0.006505 \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 226\n",
      "0 Lyapunov Risk= 0.02294486202299595\n",
      "Number of negative values: 240\n",
      "Number of negative values: 232\n",
      "Number of negative values: 222\n",
      "Number of negative values: 220\n",
      "Number of negative values: 216\n",
      "5 Lyapunov Risk= 0.022992152720689774\n",
      "Number of negative values: 216\n",
      "Number of negative values: 216\n",
      "Number of negative values: 212\n",
      "Number of negative values: 210\n",
      "0.04900760471298754 0.1 0.1\n",
      "94 0 tensor(0.0148, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m12       \u001b[0m | \u001b[0m94.0     \u001b[0m | \u001b[0m0.04901  \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 206\n",
      "0 Lyapunov Risk= 0.02335430681705475\n",
      "Number of negative values: 222\n",
      "Number of negative values: 196\n",
      "Number of negative values: 200\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "5 Lyapunov Risk= 0.024047331884503365\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "Number of negative values: 204\n",
      "0.002482835080973008 0.06389580085106157 0.1\n",
      "40 2 tensor(0.0063, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m13       \u001b[0m | \u001b[0m40.0     \u001b[0m | \u001b[0m0.002483 \u001b[0m | \u001b[0m0.0639   \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 268\n",
      "0 Lyapunov Risk= 0.009717443026602268\n",
      "Number of negative values: 228\n",
      "Number of negative values: 224\n",
      "Number of negative values: 230\n",
      "Number of negative values: 228\n",
      "Number of negative values: 226\n",
      "5 Lyapunov Risk= 0.0086595444008708\n",
      "Number of negative values: 220\n",
      "Number of negative values: 218\n",
      "Number of negative values: 214\n",
      "Number of negative values: 210\n",
      "0.09989760810890969 0.044066434904464956 0.04729367207293804\n",
      "102 0 tensor(0.0068, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m14       \u001b[0m | \u001b[0m102.0    \u001b[0m | \u001b[0m0.0999   \u001b[0m | \u001b[0m0.04407  \u001b[0m | \u001b[0m0.04729  \u001b[0m |\n",
      "Number of negative values: 226\n",
      "0 Lyapunov Risk= 0.023108821362257004\n",
      "Number of negative values: 238\n",
      "Number of negative values: 244\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 234\n",
      "Number of negative values: 224\n",
      "Number of negative values: 232\n",
      "5 Lyapunov Risk= 0.02231801114976406\n",
      "Number of negative values: 230\n",
      "Number of negative values: 220\n",
      "Number of negative values: 210\n",
      "Number of negative values: 208\n",
      "0.05415464541246608 0.06831212902637031 0.1\n",
      "110 6 tensor(0.0165, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m15       \u001b[0m | \u001b[0m110.0    \u001b[0m | \u001b[0m0.05415  \u001b[0m | \u001b[0m0.06831  \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 244\n",
      "0 Lyapunov Risk= 0.023405473679304123\n",
      "Number of negative values: 258\n",
      "Number of negative values: 238\n",
      "Number of negative values: 226\n",
      "Number of negative values: 216\n",
      "Number of negative values: 210\n",
      "5 Lyapunov Risk= 0.025334034115076065\n",
      "Number of negative values: 204\n",
      "Number of negative values: 198\n",
      "Number of negative values: 190\n",
      "Number of negative values: 194\n",
      "0.0844746488783796 0.07698003696956762 0.1\n",
      "102 0 tensor(0.0149, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m16       \u001b[0m | \u001b[0m102.0    \u001b[0m | \u001b[0m0.08447  \u001b[0m | \u001b[0m0.07698  \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 256\n",
      "0 Lyapunov Risk= 0.0247685257345438\n",
      "Number of negative values: 258\n",
      "Number of negative values: 242\n",
      "Number of negative values: 230\n",
      "Number of negative values: 222\n",
      "Number of negative values: 212\n",
      "5 Lyapunov Risk= 0.03459645435214043\n",
      "Number of negative values: 208\n",
      "Number of negative values: 206\n",
      "Number of negative values: 204\n",
      "Number of negative values: 196\n",
      "0.1 0.1 0.1\n",
      "136 0 tensor(0.0252, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m17       \u001b[0m | \u001b[0m136.0    \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 242\n",
      "0 Lyapunov Risk= 0.010808300226926804\n",
      "Number of negative values: 226\n",
      "Number of negative values: 226\n",
      "Number of negative values: 224\n",
      "Number of negative values: 230\n",
      "Number of negative values: 234\n",
      "5 Lyapunov Risk= 0.008883553557097912\n",
      "Number of negative values: 240\n",
      "Number of negative values: 242\n",
      "Number of negative values: 246\n",
      "Number of negative values: 246\n",
      "0.05667202114398779 0.08621748694363306 0.05020228359898451\n",
      "0 0 0\n",
      "| \u001b[0m18       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.05667  \u001b[0m | \u001b[0m0.08622  \u001b[0m | \u001b[0m0.0502   \u001b[0m |\n",
      "Number of negative values: 258\n",
      "0 Lyapunov Risk= 0.009974614717066288\n",
      "Number of negative values: 266\n",
      "Number of negative values: 234\n",
      "Number of negative values: 226\n",
      "Number of negative values: 218\n",
      "Number of negative values: 214\n",
      "5 Lyapunov Risk= 0.015316779725253582\n",
      "Number of negative values: 210\n",
      "Number of negative values: 210\n",
      "Number of negative values: 208\n",
      "Number of negative values: 204\n",
      "0.07989413057315715 0.03789227346663301 0.06360490865766633\n",
      "104 0 tensor(0.0069, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m19       \u001b[0m | \u001b[0m104.0    \u001b[0m | \u001b[0m0.07989  \u001b[0m | \u001b[0m0.03789  \u001b[0m | \u001b[0m0.0636   \u001b[0m |\n",
      "Number of negative values: 266\n",
      "0 Lyapunov Risk= 0.018580995500087738\n",
      "Number of negative values: 242\n",
      "Number of negative values: 238\n",
      "Number of negative values: 236\n",
      "Number of negative values: 232\n",
      "Number of negative values: 228\n",
      "5 Lyapunov Risk= 0.02140428125858307\n",
      "Number of negative values: 226\n",
      "Number of negative values: 222\n",
      "Number of negative values: 222\n",
      "Number of negative values: 214\n",
      "0.1 0.1 0.07777567364769196\n",
      "88 0 tensor(0.0137, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m20       \u001b[0m | \u001b[0m88.0     \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.07778  \u001b[0m |\n",
      "Number of negative values: 196\n",
      "0 Lyapunov Risk= 0.0016610738821327686\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "5 Lyapunov Risk= 0.0002332979638595134\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "0.001 0.01 0.01\n",
      "92 1 tensor(0.0001, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m21       \u001b[0m | \u001b[0m92.0     \u001b[0m | \u001b[0m0.001    \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.01     \u001b[0m |\n",
      "Number of negative values: 208\n",
      "0 Lyapunov Risk= 0.002500171074643731\n",
      "Number of negative values: 242\n",
      "Number of negative values: 234\n",
      "Number of negative values: 234\n",
      "Number of negative values: 232\n",
      "Number of negative values: 234\n",
      "5 Lyapunov Risk= 0.0005342354997992516\n",
      "Number of negative values: 228\n",
      "Number of negative values: 226\n",
      "Number of negative values: 226\n",
      "Number of negative values: 226\n",
      "0.026131964532863546 0.017690482369735437 0.014582305639612611\n",
      "34 1 tensor(0.0001, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m22       \u001b[0m | \u001b[0m34.0     \u001b[0m | \u001b[0m0.02613  \u001b[0m | \u001b[0m0.01769  \u001b[0m | \u001b[0m0.01458  \u001b[0m |\n",
      "Number of negative values: 216\n",
      "0 Lyapunov Risk= 0.024385983124375343\n",
      "Number of negative values: 228\n",
      "Number of negative values: 230\n",
      "Number of negative values: 212\n",
      "Number of negative values: 210\n",
      "Number of negative values: 220\n",
      "5 Lyapunov Risk= 0.022909726947546005\n",
      "Number of negative values: 208\n",
      "Number of negative values: 190\n",
      "Number of negative values: 202\n",
      "Number of negative values: 234\n",
      "0.028567878414364575 0.01 0.1\n",
      "114 2 tensor(0.0187, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m23       \u001b[0m | \u001b[0m114.0    \u001b[0m | \u001b[0m0.02857  \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 210\n",
      "0 Lyapunov Risk= 0.024619944393634796\n",
      "Number of negative values: 218\n",
      "Number of negative values: 204\n",
      "Number of negative values: 204\n",
      "Number of negative values: 206\n",
      "Number of negative values: 198\n",
      "5 Lyapunov Risk= 0.02339942753314972\n",
      "Number of negative values: 194\n",
      "Number of negative values: 184\n",
      "Number of negative values: 200\n",
      "Number of negative values: 204\n",
      "0.001 0.01 0.1\n",
      "40 2 tensor(0.0060, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m24       \u001b[0m | \u001b[0m40.0     \u001b[0m | \u001b[0m0.001    \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 218\n",
      "0 Lyapunov Risk= 0.023983927443623543\n",
      "Number of negative values: 236\n",
      "Number of negative values: 240\n",
      "Number of negative values: 224\n",
      "Number of negative values: 226\n",
      "Number of negative values: 226\n",
      "5 Lyapunov Risk= 0.022770674899220467\n",
      "Number of negative values: 238\n",
      "Number of negative values: 202\n",
      "Number of negative values: 190\n",
      "Number of negative values: 202\n",
      "0.046545027237027456 0.030520450792466323 0.1\n",
      "128 2 tensor(0.0204, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m25       \u001b[0m | \u001b[0m128.0    \u001b[0m | \u001b[0m0.04655  \u001b[0m | \u001b[0m0.03052  \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 220\n",
      "0 Lyapunov Risk= 0.02425193041563034\n",
      "Number of negative values: 238\n",
      "Number of negative values: 246\n",
      "Number of negative values: 228\n",
      "Number of negative values: 228\n",
      "Number of negative values: 240\n",
      "5 Lyapunov Risk= 0.02169649675488472\n",
      "Number of negative values: 216\n",
      "Number of negative values: 198\n",
      "Number of negative values: 198\n",
      "Number of negative values: 208\n",
      "0.05485446873876997 0.01 0.1\n",
      "132 2 tensor(0.0224, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m26       \u001b[0m | \u001b[0m132.0    \u001b[0m | \u001b[0m0.05485  \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 200\n",
      "0 Lyapunov Risk= 0.0027858028188347816\n",
      "Number of negative values: 246\n",
      "Number of negative values: 242\n",
      "Number of negative values: 244\n",
      "Number of negative values: 244\n",
      "Number of negative values: 242\n",
      "5 Lyapunov Risk= 0.00042929299524985254\n",
      "Number of negative values: 246\n",
      "Number of negative values: 246\n",
      "Number of negative values: 250\n",
      "Number of negative values: 250\n",
      "0.0347296351853873 0.016752824699881295 0.011992711001254612\n",
      "38 2 tensor(0.0001, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m27       \u001b[0m | \u001b[0m38.0     \u001b[0m | \u001b[0m0.03473  \u001b[0m | \u001b[0m0.01675  \u001b[0m | \u001b[0m0.01199  \u001b[0m |\n",
      "Number of negative values: 220\n",
      "0 Lyapunov Risk= 0.015011276118457317\n",
      "Number of negative values: 228\n",
      "Number of negative values: 244\n",
      "Number of negative values: 222\n",
      "Number of negative values: 214\n",
      "Number of negative values: 214\n",
      "5 Lyapunov Risk= 0.01522725448012352\n",
      "Number of negative values: 214\n",
      "Number of negative values: 224\n",
      "Number of negative values: 230\n",
      "Number of negative values: 222\n",
      "0.04195063927388454 0.01 0.07967050894082686\n",
      "126 9 tensor(0.0133, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m28       \u001b[0m | \u001b[0m126.0    \u001b[0m | \u001b[0m0.04195  \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.07967  \u001b[0m |\n",
      "Number of negative values: 200\n",
      "0 Lyapunov Risk= 0.01258467324078083\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "5 Lyapunov Risk= 0.011046778410673141\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "0.001 0.1 0.01\n",
      "0 0 0\n",
      "| \u001b[0m29       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.001    \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.01     \u001b[0m |\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 248\n",
      "0 Lyapunov Risk= 0.022121487185359\n",
      "Number of negative values: 240\n",
      "Number of negative values: 234\n",
      "Number of negative values: 228\n",
      "Number of negative values: 220\n",
      "Number of negative values: 216\n",
      "5 Lyapunov Risk= 0.02624426782131195\n",
      "Number of negative values: 210\n",
      "Number of negative values: 204\n",
      "Number of negative values: 196\n",
      "Number of negative values: 198\n",
      "0.07798737262167 0.09916042371675901 0.09660036506234009\n",
      "128 0 tensor(0.0206, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m30       \u001b[0m | \u001b[0m128.0    \u001b[0m | \u001b[0m0.07799  \u001b[0m | \u001b[0m0.09916  \u001b[0m | \u001b[0m0.0966   \u001b[0m |\n",
      "Number of negative values: 228\n",
      "0 Lyapunov Risk= 0.005873735062777996\n",
      "Number of negative values: 206\n",
      "Number of negative values: 188\n",
      "Number of negative values: 204\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "5 Lyapunov Risk= 0.00651800399646163\n",
      "Number of negative values: 210\n",
      "Number of negative values: 198\n",
      "Number of negative values: 188\n",
      "Number of negative values: 182\n",
      "0.001 0.01 0.051342271433036664\n",
      "32 0 tensor(0.0008, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m31       \u001b[0m | \u001b[0m32.0     \u001b[0m | \u001b[0m0.001    \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.05134  \u001b[0m |\n",
      "Number of negative values: 222\n",
      "0 Lyapunov Risk= 0.01657187007367611\n",
      "Number of negative values: 234\n",
      "Number of negative values: 236\n",
      "Number of negative values: 208\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "5 Lyapunov Risk= 0.017111584544181824\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "Number of negative values: 220\n",
      "0.03518343812663534 0.0309214202473764 0.0840010338858199\n",
      "100 2 tensor(0.0116, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m32       \u001b[0m | \u001b[0m100.0    \u001b[0m | \u001b[0m0.03518  \u001b[0m | \u001b[0m0.03092  \u001b[0m | \u001b[0m0.084    \u001b[0m |\n",
      "Number of negative values: 226\n",
      "0 Lyapunov Risk= 0.023502221331000328\n",
      "Number of negative values: 250\n",
      "Number of negative values: 258\n",
      "Number of negative values: 242\n",
      "Number of negative values: 250\n",
      "Number of negative values: 236\n",
      "5 Lyapunov Risk= 0.020495397970080376\n",
      "Number of negative values: 236\n",
      "Number of negative values: 224\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "0.07162925171262043 0.03923320841911201 0.0994266208673896\n",
      "140 2 tensor(0.0252, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m33       \u001b[0m | \u001b[95m140.0    \u001b[0m | \u001b[95m0.07163  \u001b[0m | \u001b[95m0.03923  \u001b[0m | \u001b[95m0.09943  \u001b[0m |\n",
      "Number of negative values: 220\n",
      "0 Lyapunov Risk= 0.02407538704574108\n",
      "Number of negative values: 250\n",
      "Number of negative values: 254\n",
      "Number of negative values: 240\n",
      "Number of negative values: 242\n",
      "Number of negative values: 246\n",
      "5 Lyapunov Risk= 0.02086665853857994\n",
      "Number of negative values: 228\n",
      "Number of negative values: 208\n",
      "Number of negative values: 204\n",
      "Number of negative values: 206\n",
      "0.06717476826715342 0.024060028709311537 0.1\n",
      "140 5 tensor(0.0222, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m34       \u001b[0m | \u001b[0m140.0    \u001b[0m | \u001b[0m0.06717  \u001b[0m | \u001b[0m0.02406  \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 220\n",
      "0 Lyapunov Risk= 0.023653531447052956\n",
      "Number of negative values: 240\n",
      "Number of negative values: 250\n",
      "Number of negative values: 226\n",
      "Number of negative values: 222\n",
      "Number of negative values: 238\n",
      "5 Lyapunov Risk= 0.021859265863895416\n",
      "Number of negative values: 232\n",
      "Number of negative values: 216\n",
      "Number of negative values: 202\n",
      "Number of negative values: 200\n",
      "0.058076450483831504 0.044927935088895664 0.1\n",
      "132 6 tensor(0.0215, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m35       \u001b[0m | \u001b[0m132.0    \u001b[0m | \u001b[0m0.05808  \u001b[0m | \u001b[0m0.04493  \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 248\n",
      "0 Lyapunov Risk= 0.024227488785982132\n",
      "Number of negative values: 256\n",
      "Number of negative values: 262\n",
      "Number of negative values: 256\n",
      "Number of negative values: 238\n",
      "Number of negative values: 222\n",
      "5 Lyapunov Risk= 0.0240907184779644\n",
      "Number of negative values: 212\n",
      "Number of negative values: 204\n",
      "Number of negative values: 206\n",
      "Number of negative values: 204\n",
      "0.1 0.054401470796423046 0.1\n",
      "66 0 tensor(0.0094, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m36       \u001b[0m | \u001b[0m66.0     \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.0544   \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 220\n",
      "0 Lyapunov Risk= 0.019216150045394897\n",
      "Number of negative values: 234\n",
      "Number of negative values: 250\n",
      "Number of negative values: 228\n",
      "Number of negative values: 230\n",
      "Number of negative values: 228\n",
      "5 Lyapunov Risk= 0.01800617016851902\n",
      "Number of negative values: 242\n",
      "Number of negative values: 228\n",
      "Number of negative values: 212\n",
      "Number of negative values: 202\n",
      "0.05635850191821514 0.02434798270396605 0.08989926257002957\n",
      "134 2 tensor(0.0183, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m37       \u001b[0m | \u001b[0m134.0    \u001b[0m | \u001b[0m0.05636  \u001b[0m | \u001b[0m0.02435  \u001b[0m | \u001b[0m0.0899   \u001b[0m |\n",
      "Number of negative values: 206\n",
      "0 Lyapunov Risk= 0.01875121518969536\n",
      "Number of negative values: 206\n",
      "Number of negative values: 204\n",
      "Number of negative values: 202\n",
      "Number of negative values: 198\n",
      "Number of negative values: 192\n",
      "5 Lyapunov Risk= 0.016284344717860222\n",
      "Number of negative values: 192\n",
      "Number of negative values: 192\n",
      "Number of negative values: 184\n",
      "Number of negative values: 184\n",
      "0.1 0.1 0.01\n",
      "4 0 tensor(0.0002, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m38       \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.01     \u001b[0m |\n",
      "Number of negative values: 200\n",
      "0 Lyapunov Risk= 0.011689942330121994\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "5 Lyapunov Risk= 0.009848663583397865\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "0.014278238055605082 0.09534771301269847 0.024274724007999612\n",
      "0 0 0\n",
      "| \u001b[0m39       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.01428  \u001b[0m | \u001b[0m0.09535  \u001b[0m | \u001b[0m0.02427  \u001b[0m |\n",
      "Number of negative values: 250\n",
      "0 Lyapunov Risk= 0.024008583277463913\n",
      "Number of negative values: 248\n",
      "Number of negative values: 240\n",
      "Number of negative values: 230\n",
      "Number of negative values: 220\n",
      "Number of negative values: 214\n",
      "5 Lyapunov Risk= 0.03020405024290085\n",
      "Number of negative values: 206\n",
      "Number of negative values: 200\n",
      "Number of negative values: 198\n",
      "Number of negative values: 198\n",
      "0.08889887597578258 0.1 0.1\n",
      "132 0 tensor(0.0231, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m40       \u001b[0m | \u001b[0m132.0    \u001b[0m | \u001b[0m0.0889   \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 226\n",
      "0 Lyapunov Risk= 0.024191509932279587\n",
      "Number of negative values: 252\n",
      "Number of negative values: 262\n",
      "Number of negative values: 242\n",
      "Number of negative values: 250\n",
      "Number of negative values: 242\n",
      "5 Lyapunov Risk= 0.020215561613440514\n",
      "Number of negative values: 232\n",
      "Number of negative values: 216\n",
      "Number of negative values: 212\n",
      "Number of negative values: 210\n",
      "0.0753605396089892 0.01 0.1\n",
      "144 2 tensor(0.0263, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m41       \u001b[0m | \u001b[95m144.0    \u001b[0m | \u001b[95m0.07536  \u001b[0m | \u001b[95m0.01     \u001b[0m | \u001b[95m0.1      \u001b[0m |\n",
      "Number of negative values: 226\n",
      "0 Lyapunov Risk= 0.017545104026794434\n",
      "Number of negative values: 230\n",
      "Number of negative values: 256\n",
      "Number of negative values: 248\n",
      "Number of negative values: 248\n",
      "Number of negative values: 244\n",
      "5 Lyapunov Risk= 0.014886260032653809\n",
      "Number of negative values: 238\n",
      "Number of negative values: 220\n",
      "Number of negative values: 212\n",
      "Number of negative values: 206\n",
      "0.07278390713305066 0.010269372108763577 0.08566087705940396\n",
      "152 4 tensor(0.0180, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m42       \u001b[0m | \u001b[95m152.0    \u001b[0m | \u001b[95m0.07278  \u001b[0m | \u001b[95m0.01027  \u001b[0m | \u001b[95m0.08566  \u001b[0m |\n",
      "Number of negative values: 220\n",
      "0 Lyapunov Risk= 0.01677185483276844\n",
      "Number of negative values: 234\n",
      "Number of negative values: 232\n",
      "Number of negative values: 208\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "5 Lyapunov Risk= 0.017304135486483574\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "Number of negative values: 220\n",
      "0.03492499376064322 0.031430123758574464 0.0844901625145773\n",
      "100 2 tensor(0.0117, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m43       \u001b[0m | \u001b[0m100.0    \u001b[0m | \u001b[0m0.03492  \u001b[0m | \u001b[0m0.03143  \u001b[0m | \u001b[0m0.08449  \u001b[0m |\n",
      "Number of negative values: 224\n",
      "0 Lyapunov Risk= 0.019152037799358368\n",
      "Number of negative values: 236\n",
      "Number of negative values: 256\n",
      "Number of negative values: 236\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 236\n",
      "Number of negative values: 246\n",
      "5 Lyapunov Risk= 0.01687653176486492\n",
      "Number of negative values: 238\n",
      "Number of negative values: 220\n",
      "Number of negative values: 210\n",
      "Number of negative values: 206\n",
      "0.06449086952085749 0.011659686277788345 0.08939317863354185\n",
      "154 6 tensor(0.0202, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m44       \u001b[0m | \u001b[95m154.0    \u001b[0m | \u001b[95m0.06449  \u001b[0m | \u001b[95m0.01166  \u001b[0m | \u001b[95m0.08939  \u001b[0m |\n",
      "Number of negative values: 228\n",
      "0 Lyapunov Risk= 0.014415163546800613\n",
      "Number of negative values: 218\n",
      "Number of negative values: 252\n",
      "Number of negative values: 236\n",
      "Number of negative values: 234\n",
      "Number of negative values: 244\n",
      "5 Lyapunov Risk= 0.013025360181927681\n",
      "Number of negative values: 250\n",
      "Number of negative values: 238\n",
      "Number of negative values: 214\n",
      "Number of negative values: 210\n",
      "0.06100827235183976 0.01 0.07809453867051783\n",
      "140 6 tensor(0.0132, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m45       \u001b[0m | \u001b[0m140.0    \u001b[0m | \u001b[0m0.06101  \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.07809  \u001b[0m |\n",
      "Number of negative values: 226\n",
      "0 Lyapunov Risk= 0.020602628588676453\n",
      "Number of negative values: 244\n",
      "Number of negative values: 260\n",
      "Number of negative values: 238\n",
      "Number of negative values: 246\n",
      "Number of negative values: 242\n",
      "5 Lyapunov Risk= 0.01761561632156372\n",
      "Number of negative values: 240\n",
      "Number of negative values: 218\n",
      "Number of negative values: 212\n",
      "Number of negative values: 210\n",
      "0.06991095779835467 0.01 0.09256162626607382\n",
      "152 5 tensor(0.0215, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m46       \u001b[0m | \u001b[0m152.0    \u001b[0m | \u001b[0m0.06991  \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.09256  \u001b[0m |\n",
      "Number of negative values: 258\n",
      "0 Lyapunov Risk= 0.009807578288018703\n",
      "Number of negative values: 268\n",
      "Number of negative values: 234\n",
      "Number of negative values: 224\n",
      "Number of negative values: 218\n",
      "Number of negative values: 214\n",
      "5 Lyapunov Risk= 0.015330817550420761\n",
      "Number of negative values: 212\n",
      "Number of negative values: 210\n",
      "Number of negative values: 208\n",
      "Number of negative values: 204\n",
      "0.08038221226372332 0.03818513941505107 0.06282568770486477\n",
      "104 0 tensor(0.0069, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m47       \u001b[0m | \u001b[0m104.0    \u001b[0m | \u001b[0m0.08038  \u001b[0m | \u001b[0m0.03819  \u001b[0m | \u001b[0m0.06283  \u001b[0m |\n",
      "Number of negative values: 246\n",
      "0 Lyapunov Risk= 0.01620558649301529\n",
      "Number of negative values: 252\n",
      "Number of negative values: 254\n",
      "Number of negative values: 248\n",
      "Number of negative values: 244\n",
      "Number of negative values: 234\n",
      "5 Lyapunov Risk= 0.014917979016900063\n",
      "Number of negative values: 216\n",
      "Number of negative values: 216\n",
      "Number of negative values: 216\n",
      "Number of negative values: 208\n",
      "0.07151169145598699 0.051709376741580146 0.08346314964899883\n",
      "132 3 tensor(0.0159, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m48       \u001b[0m | \u001b[0m132.0    \u001b[0m | \u001b[0m0.07151  \u001b[0m | \u001b[0m0.05171  \u001b[0m | \u001b[0m0.08346  \u001b[0m |\n",
      "Number of negative values: 218\n",
      "0 Lyapunov Risk= 0.013308384455740452\n",
      "Number of negative values: 214\n",
      "Number of negative values: 214\n",
      "Number of negative values: 214\n",
      "Number of negative values: 214\n",
      "Number of negative values: 214\n",
      "5 Lyapunov Risk= 0.012932832352817059\n",
      "Number of negative values: 214\n",
      "Number of negative values: 214\n",
      "Number of negative values: 214\n",
      "Number of negative values: 214\n",
      "0.001 0.1 0.0711464524860748\n",
      "8 1 tensor(0.0007, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m49       \u001b[0m | \u001b[0m8.0      \u001b[0m | \u001b[0m0.001    \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.07115  \u001b[0m |\n",
      "Number of negative values: 236\n",
      "0 Lyapunov Risk= 0.018081652000546455\n",
      "Number of negative values: 230\n",
      "Number of negative values: 264\n",
      "Number of negative values: 248\n",
      "Number of negative values: 252\n",
      "Number of negative values: 246\n",
      "5 Lyapunov Risk= 0.015630528330802917\n",
      "Number of negative values: 230\n",
      "Number of negative values: 220\n",
      "Number of negative values: 212\n",
      "Number of negative values: 206\n",
      "0.07816530904044379 0.03473731069511101 0.08735017691716995\n",
      "154 3 tensor(0.0192, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m50       \u001b[0m | \u001b[0m154.0    \u001b[0m | \u001b[0m0.07817  \u001b[0m | \u001b[0m0.03474  \u001b[0m | \u001b[0m0.08735  \u001b[0m |\n",
      "Number of negative values: 232\n",
      "0 Lyapunov Risk= 0.018132951110601425\n",
      "Number of negative values: 228\n",
      "Number of negative values: 256\n",
      "Number of negative values: 244\n",
      "Number of negative values: 250\n",
      "Number of negative values: 240\n",
      "5 Lyapunov Risk= 0.016039270907640457\n",
      "Number of negative values: 242\n",
      "Number of negative values: 228\n",
      "Number of negative values: 220\n",
      "Number of negative values: 214\n",
      "0.06841077616239988 0.03499173155659766 0.08767634802087945\n",
      "154 5 tensor(0.0197, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m51       \u001b[0m | \u001b[0m154.0    \u001b[0m | \u001b[0m0.06841  \u001b[0m | \u001b[0m0.03499  \u001b[0m | \u001b[0m0.08768  \u001b[0m |\n",
      "Number of negative values: 206\n",
      "0 Lyapunov Risk= 0.003825790947303176\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "5 Lyapunov Risk= 0.00227556680329144\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "0.001 0.045708239206319425 0.01\n",
      "0 0 0\n",
      "| \u001b[0m52       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.001    \u001b[0m | \u001b[0m0.04571  \u001b[0m | \u001b[0m0.01     \u001b[0m |\n",
      "Number of negative values: 226\n",
      "0 Lyapunov Risk= 0.01841186359524727\n",
      "Number of negative values: 232\n",
      "Number of negative values: 256\n",
      "Number of negative values: 250\n",
      "Number of negative values: 246\n",
      "Number of negative values: 248\n",
      "5 Lyapunov Risk= 0.015734994783997536\n",
      "Number of negative values: 232\n",
      "Number of negative values: 224\n",
      "Number of negative values: 214\n",
      "Number of negative values: 212\n",
      "0.07419100190665982 0.024471365576612742 0.08793485079958172\n",
      "160 4 tensor(0.0200, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m53       \u001b[0m | \u001b[95m160.0    \u001b[0m | \u001b[95m0.07419  \u001b[0m | \u001b[95m0.02447  \u001b[0m | \u001b[95m0.08793  \u001b[0m |\n",
      "Number of negative values: 262\n",
      "0 Lyapunov Risk= 0.007397968787699938\n",
      "Number of negative values: 242\n",
      "Number of negative values: 238\n",
      "Number of negative values: 236\n",
      "Number of negative values: 232\n",
      "Number of negative values: 230\n",
      "5 Lyapunov Risk= 0.006309147458523512\n",
      "Number of negative values: 230\n",
      "Number of negative values: 232\n",
      "Number of negative values: 232\n",
      "Number of negative values: 238\n",
      "0.06287753743683126 0.061662337281211746 0.0480056960410474\n",
      "56 0 tensor(0.0034, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m54       \u001b[0m | \u001b[0m56.0     \u001b[0m | \u001b[0m0.06288  \u001b[0m | \u001b[0m0.06166  \u001b[0m | \u001b[0m0.04801  \u001b[0m |\n",
      "Number of negative values: 222\n",
      "0 Lyapunov Risk= 0.010340294800698757\n",
      "Number of negative values: 230\n",
      "Number of negative values: 224\n",
      "Number of negative values: 218\n",
      "Number of negative values: 224\n",
      "Number of negative values: 230\n",
      "5 Lyapunov Risk= 0.007934329099953175\n",
      "Number of negative values: 228\n",
      "Number of negative values: 222\n",
      "Number of negative values: 216\n",
      "Number of negative values: 208\n",
      "0.08243164439146669 0.0694178156266787 0.03492757507749461\n",
      "12 6 tensor(0.0008, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m55       \u001b[0m | \u001b[0m12.0     \u001b[0m | \u001b[0m0.08243  \u001b[0m | \u001b[0m0.06942  \u001b[0m | \u001b[0m0.03493  \u001b[0m |\n",
      "=============================================================\n",
      "Final result: {'target': 160.0, 'params': {'min_a': 0.07419100190665982, 'min_b': 0.024471365576612742, 'min_c': 0.08793485079958172}}\n",
      "Time consumed: 83.87414462398738\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-3, 1e-1), \"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.07419100190665982 \n",
    "min_b = 0.024471365576612742  \n",
    "min_c = 0.08793485079958172"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 226\n",
      "0 Lyapunov Risk= 0.01841186359524727\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.2622, 0.0742],\n",
      "        [0.2939, 0.0742],\n",
      "        [0.4021, 0.0742],\n",
      "        [0.1742, 0.0742]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.2142, 0.1742, 0.1742, 0.0742]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0245]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0879]])\n",
      "Number of negative values: 232\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.2567, 0.0742],\n",
      "        [0.2920, 0.0742],\n",
      "        [0.4718, 0.0742],\n",
      "        [0.1714, 0.0742]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.1436, 0.1072, 0.1022, 0.0742]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0245]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0879]])\n",
      "Number of negative values: 256\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.2699, 0.0742],\n",
      "        [0.3123, 0.0742],\n",
      "        [0.5485, 0.0742],\n",
      "        [0.1804, 0.0742]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.0752, 0.0742, 0.0742, 0.0742]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0245]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0879]])\n",
      "Number of negative values: 250\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.2942, 0.0742],\n",
      "        [0.3475, 0.0742],\n",
      "        [0.6243, 0.0742],\n",
      "        [0.2099, 0.0742]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.0742, 0.0742, 0.0742, 0.0742]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0245]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0879]])\n",
      "Number of negative values: 246\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.3241, 0.0742],\n",
      "        [0.3898, 0.0742],\n",
      "        [0.6973, 0.0742],\n",
      "        [0.2495, 0.0742]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.0742, 0.0742, 0.0742, 0.0742]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0245]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0879]])\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"
     ]
    }
   ],
   "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 = 5\n",
    "learning_rate = 0.1\n",
    "optimizer = torch.optim.Adam(model.parameters(), lr=learning_rate)\n",
    "\n",
    "#Assign initial guess for the parameter\n",
    "with torch.no_grad():\n",
    "\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 weight parameters in the hidden and output layers 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",
    "\n",
    "    Lyapunov_risk = (F.relu(((L_V + 0.001*abs(V_candidate).T).T))).mean()\n",
    "   \n",
    "\n",
    "    if out_iters%10 == 0:\n",
    "        print(out_iters, \"Lyapunov Risk=\",Lyapunov_risk.item())\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",
    "    #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",
    "    out_iters+=1"
   ]
  },
  {
   "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:  246\n",
      "number of \\dot V > 0 points:  154\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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BSt37WLkyJEMGTKEE044gX//e593MzkUF0OLFhzUuTOHlZay5osveHXePAadfz79+/fnpz/9Kf/5z38AGDFiBPPmzXNlxUyaNIkBP/kJx110Eetranh74UJmvvkm106bxsDzzmPlypVMmzaNI488kqOPPppzz03wvqaAx69JeHmQTBecCeqHgalJ6pxOw8ns9/203aRcT48+qlpW5txRlJX5nwmM06tIRvSN5nDJcP9eRO0KqXtP/VNPVYbav5fe6zyGffyeeqrStz7Rr9GNG/f+sF24cO88qu+Jdo8GEt1RbN++XQcMGKCHH364Xn755Tpnzpx6fVm3brohuqyqNTU1qqpaW1urJ510ki50f2WXlZXpzTffrKqqDz30kJ7+3e+qLlzo3FH88peqGzfqySefrB999JGqqr777rtaXl5eb0ODO4orr1TduFFXPv+8Htipk659+WUt7d5dly9frqqqY8aM0TvvvFNVVU866SSdO3euqqoCOnPmTFVVvXbCBP3dlVc6dxRnnKEzHnigvq9u3brV3z1t3rw58Ulx93/bunUNT4AHQe8ownQUx+O82H0RzjuPF+A85TQe59WKUWdyN7AS55WbjQ47aVMdhUtzTPaVScyuYDQHuxJdZLxIdVjM66mn2tparays1BtuuEFLSkr0wQcfdPuLGVJS1XvuuUcHDRqk/fv3165du+rjjz9eX2/lypWqqrpr1y7t3LmzqqreeOONescdd+j27du1TZs2OmDAgPrSt2/ffex68MEHtWvXrjpgwAAdNmyY/uMf/9AFCxboCSecUF/3lVde0TPPPFNVGzqKVq1aaV1dnaqqVlRU6MUXX6yqqhdeeKHOmDGjXn/qqafqj370I33kkUd0+/btTTpe8QR1FKEF3KkzQZ00Wsc1/srsWGQYRrr59NNg6/3SsmVLRowYwYgRI+jfvz8PPfQQY8eObVDnk08+YfLkycydO5dOnToxduzYBvEDsXEg8YGDdXV1dOzYkQULFjRqyznnnMNdd91Vv7xw4UJf+7DffvvV99uyZUtqa2sT1nv++ed54403ePbZZ7nllltYvHgxRUXZvXRbrifDMDLGN74RbL0fli9fzscff1y/vGDBAsrKygBo164d27dvB2Dbtm0ccMABdOjQgfXr1/PCCy80aOeJJ56o/zt8+PAG29q3b8+hhx7KjBkzAGfk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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "#Plot the CEs\n",
    "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",
    "# 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",
    "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:  160\n",
      "number of points beyond levelset:  240\n"
     ]
    },
    {
     "data": {
      "image/png": "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KnuNAoIfVTfccAN9+/bl5JNPZvny5Rx77LEAtLS0cOONN3LAAQfQv39/2tvbecMb3rDDxHb66afzl7/8heOOO46+ffvyjW98gwMPPJAVK1bkHXfgwIFcd911nHrqqey1116ccMIJvPTSS7HjvuGGG7jgggt45ZVXGDlyJNdff31s7Xvf+16mTZvGb3/7W773ve9x9dVX8/jjj+PunHTSSYwdOzZ2X6JBiMog9dx0RFE9auWIoidp1lTatm2bjx071h977LHdljVSracsOqKoHRqq1pMQjcqyZcs49NBDOemkkzjssMPSDkeIsqBEEVLqw8lr5eHqzap/4QV45BF48MHg79atpel7XK+NrX/llSP5zW9W81//9a1Uxi9Wv3Vrafq097/0pX1/9UrUoUY9N5XwUAmPcpXwKLc+3ymeao4fpX/22c1F61XCo/FLeOiIguAC6Cuv7DrvlVfimbWkr5w+NBP3ytq1sH37rvPcg/nF6rdvby69e/H6F1+szc+P9PH0cVCiIH0LvfT55ycp4ZFkvvTl1auER33r46BEQfoWeunzz09SwiPJfOnLq1cJj/rWx0GJgvQt9NKXXsKjZwmOnAoURemzJTziPAQmV3/55TM47bQRnH32OM45Zzx/+ctfCmqPO+64guMDfPvb3+aVnucWYsYfh0MO2VEuqii9SnjUtz4WURcv6rnpwUXN+eCiJ792o7924DDfbuZb24YmCiLquT+5zuw4+lNPne6zZt3iGza4z5s3z8eMGbPLelF+hULPHcp1UCeNPy4bN24uWq8HF+nBRXXZZLirHjVjuKtQDY9soujs7PTJkyf7mWee6Ycffrj/27/9m2/fvn239XNLZLz66qu+5557urv7t771LR81apS/5S1v8auvvjp2/9/5znd8jz328NGjR/uUKVN869atPn36dB81apSPHj3aZ8+eXdL7yyLDXTIaMa5CiUIlPERjUOjWjzJVR3v44Yd59NFHOfjgg3n729/O/fffz/HHHx+5/u23386YMWPo6uri+uuv569//SubN2/mne98J5MnT+boo4/utf9Pf/rTzJ49m87OToYMGUJXVxdr165l6dKlAPzjH/8oy3sTohC6RiEagyrc+jFp0iTa2tqCcuDjxrFmzZq861100UWMGzeO6667jh//+Mfcd999nH766QwaNIiWlhbOOOMM7r33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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.02  # 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",
    "\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",
    "\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": "markdown",
   "metadata": {},
   "source": [
    "# Save data for matlab plot"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Save to text file\n",
    "np.savetxt('V_VDP_tanh.txt', V_candidate.detach().numpy(), fmt='%.6f')  # value of V for each sample\n",
    "# Save to text file\n",
    "np.savetxt('LV_VDP_tanh.txt', L_V.detach().numpy(), fmt='%.6f')  # value of \\dot V for each sample"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "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_VDP_tanh.mat', {'data': array_data1})\n",
    "savemat('stable_VDP_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
}
