{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Codes for development of LNN 1 for  path following 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): #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": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Net1(torch.nn.Module):\n",
    "    \n",
    "    def __init__(self,n_input,n_hidden,n_output):\n",
    "        super(Net1, self).__init__()\n",
    "        torch.manual_seed(2)\n",
    "        self.layer1 = torch.nn.Linear(n_input, n_hidden)\n",
    "        self.layer2 = torch.nn.Linear(n_hidden, n_output)\n",
    "       \n",
    "\n",
    "    def forward(self,x):\n",
    "        acti_h = torch.nn.Tanh()\n",
    "        \n",
    "        h = acti_h(self.layer1(x))\n",
    "        out = self.layer2(h)\n",
    "        \n",
    "        return out"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "x_mean = np.array([ 0.00281402, -0.00416428])\n",
    "x_std = np.array([1.16720222, 1.83544454])\n",
    "\n",
    "y_mean = np.array([-0.01695862, -0.01011195])\n",
    "y_std = np.array([4.22776451, 9.52885124])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "#import the model\n",
    "basic = Net1(2,16,2)\n",
    "basic.load_state_dict(torch.load(\"model_PF\"))\n",
    "\n",
    "def predict_fn(model, x):\n",
    "    x_tensor = torch.tensor(x)\n",
    "    x_norm = (x_tensor - x_mean) / x_std\n",
    "    # Predict normalized output\n",
    "    y_pred_norm = model(x_norm.to(dtype=torch.float32))\n",
    "    # Re-normalize to real scale\n",
    "    y_real = y_pred_norm.detach().numpy() * y_std + y_mean\n",
    "    return torch.tensor(y_real).to(dtype=torch.float32)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Neural network hyperparameter Options & training dataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<torch._C.Generator at 0x7f7b8dc9fed0>"
      ]
     },
     "execution_count": 8,
     "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": 9,
   "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": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([[-2.0000, -3.1400],\n",
      "        [-2.0000, -2.8095],\n",
      "        [-2.0000, -2.4789],\n",
      "        [-2.0000, -2.1484],\n",
      "        [-2.0000, -1.8179],\n",
      "        [-2.0000, -1.4874],\n",
      "        [-2.0000, -1.1568],\n",
      "        [-2.0000, -0.8263],\n",
      "        [-2.0000, -0.4958],\n",
      "        [-2.0000, -0.1653],\n",
      "        [-2.0000,  0.1653],\n",
      "        [-2.0000,  0.4958],\n",
      "        [-2.0000,  0.8263],\n",
      "        [-2.0000,  1.1568],\n",
      "        [-2.0000,  1.4874],\n",
      "        [-2.0000,  1.8179],\n",
      "        [-2.0000,  2.1484],\n",
      "        [-2.0000,  2.4789],\n",
      "        [-2.0000,  2.8095],\n",
      "        [-2.0000,  3.1400],\n",
      "        [-1.7895, -3.1400],\n",
      "        [-1.7895, -2.8095],\n",
      "        [-1.7895, -2.4789],\n",
      "        [-1.7895, -2.1484],\n",
      "        [-1.7895, -1.8179],\n",
      "        [-1.7895, -1.4874],\n",
      "        [-1.7895, -1.1568],\n",
      "        [-1.7895, -0.8263],\n",
      "        [-1.7895, -0.4958],\n",
      "        [-1.7895, -0.1653],\n",
      "        [-1.7895,  0.1653],\n",
      "        [-1.7895,  0.4958],\n",
      "        [-1.7895,  0.8263],\n",
      "        [-1.7895,  1.1568],\n",
      "        [-1.7895,  1.4874],\n",
      "        [-1.7895,  1.8179],\n",
      "        [-1.7895,  2.1484],\n",
      "        [-1.7895,  2.4789],\n",
      "        [-1.7895,  2.8095],\n",
      "        [-1.7895,  3.1400],\n",
      "        [-1.5789, -3.1400],\n",
      "        [-1.5789, -2.8095],\n",
      "        [-1.5789, -2.4789],\n",
      "        [-1.5789, -2.1484],\n",
      "        [-1.5789, -1.8179],\n",
      "        [-1.5789, -1.4874],\n",
      "        [-1.5789, -1.1568],\n",
      "        [-1.5789, -0.8263],\n",
      "        [-1.5789, -0.4958],\n",
      "        [-1.5789, -0.1653],\n",
      "        [-1.5789,  0.1653],\n",
      "        [-1.5789,  0.4958],\n",
      "        [-1.5789,  0.8263],\n",
      "        [-1.5789,  1.1568],\n",
      "        [-1.5789,  1.4874],\n",
      "        [-1.5789,  1.8179],\n",
      "        [-1.5789,  2.1484],\n",
      "        [-1.5789,  2.4789],\n",
      "        [-1.5789,  2.8095],\n",
      "        [-1.5789,  3.1400],\n",
      "        [-1.3684, -3.1400],\n",
      "        [-1.3684, -2.8095],\n",
      "        [-1.3684, -2.4789],\n",
      "        [-1.3684, -2.1484],\n",
      "        [-1.3684, -1.8179],\n",
      "        [-1.3684, -1.4874],\n",
      "        [-1.3684, -1.1568],\n",
      "        [-1.3684, -0.8263],\n",
      "        [-1.3684, -0.4958],\n",
      "        [-1.3684, -0.1653],\n",
      "        [-1.3684,  0.1653],\n",
      "        [-1.3684,  0.4958],\n",
      "        [-1.3684,  0.8263],\n",
      "        [-1.3684,  1.1568],\n",
      "        [-1.3684,  1.4874],\n",
      "        [-1.3684,  1.8179],\n",
      "        [-1.3684,  2.1484],\n",
      "        [-1.3684,  2.4789],\n",
      "        [-1.3684,  2.8095],\n",
      "        [-1.3684,  3.1400],\n",
      "        [-1.1579, -3.1400],\n",
      "        [-1.1579, -2.8095],\n",
      "        [-1.1579, -2.4789],\n",
      "        [-1.1579, -2.1484],\n",
      "        [-1.1579, -1.8179],\n",
      "        [-1.1579, -1.4874],\n",
      "        [-1.1579, -1.1568],\n",
      "        [-1.1579, -0.8263],\n",
      "        [-1.1579, -0.4958],\n",
      "        [-1.1579, -0.1653],\n",
      "        [-1.1579,  0.1653],\n",
      "        [-1.1579,  0.4958],\n",
      "        [-1.1579,  0.8263],\n",
      "        [-1.1579,  1.1568],\n",
      "        [-1.1579,  1.4874],\n",
      "        [-1.1579,  1.8179],\n",
      "        [-1.1579,  2.1484],\n",
      "        [-1.1579,  2.4789],\n",
      "        [-1.1579,  2.8095],\n",
      "        [-1.1579,  3.1400],\n",
      "        [-0.9474, -3.1400],\n",
      "        [-0.9474, -2.8095],\n",
      "        [-0.9474, -2.4789],\n",
      "        [-0.9474, -2.1484],\n",
      "        [-0.9474, -1.8179],\n",
      "        [-0.9474, -1.4874],\n",
      "        [-0.9474, -1.1568],\n",
      "        [-0.9474, -0.8263],\n",
      "        [-0.9474, -0.4958],\n",
      "        [-0.9474, -0.1653],\n",
      "        [-0.9474,  0.1653],\n",
      "        [-0.9474,  0.4958],\n",
      "        [-0.9474,  0.8263],\n",
      "        [-0.9474,  1.1568],\n",
      "        [-0.9474,  1.4874],\n",
      "        [-0.9474,  1.8179],\n",
      "        [-0.9474,  2.1484],\n",
      "        [-0.9474,  2.4789],\n",
      "        [-0.9474,  2.8095],\n",
      "        [-0.9474,  3.1400],\n",
      "        [-0.7368, -3.1400],\n",
      "        [-0.7368, -2.8095],\n",
      "        [-0.7368, -2.4789],\n",
      "        [-0.7368, -2.1484],\n",
      "        [-0.7368, -1.8179],\n",
      "        [-0.7368, -1.4874],\n",
      "        [-0.7368, -1.1568],\n",
      "        [-0.7368, -0.8263],\n",
      "        [-0.7368, -0.4958],\n",
      "        [-0.7368, -0.1653],\n",
      "        [-0.7368,  0.1653],\n",
      "        [-0.7368,  0.4958],\n",
      "        [-0.7368,  0.8263],\n",
      "        [-0.7368,  1.1568],\n",
      "        [-0.7368,  1.4874],\n",
      "        [-0.7368,  1.8179],\n",
      "        [-0.7368,  2.1484],\n",
      "        [-0.7368,  2.4789],\n",
      "        [-0.7368,  2.8095],\n",
      "        [-0.7368,  3.1400],\n",
      "        [-0.5263, -3.1400],\n",
      "        [-0.5263, -2.8095],\n",
      "        [-0.5263, -2.4789],\n",
      "        [-0.5263, -2.1484],\n",
      "        [-0.5263, -1.8179],\n",
      "        [-0.5263, -1.4874],\n",
      "        [-0.5263, -1.1568],\n",
      "        [-0.5263, -0.8263],\n",
      "        [-0.5263, -0.4958],\n",
      "        [-0.5263, -0.1653],\n",
      "        [-0.5263,  0.1653],\n",
      "        [-0.5263,  0.4958],\n",
      "        [-0.5263,  0.8263],\n",
      "        [-0.5263,  1.1568],\n",
      "        [-0.5263,  1.4874],\n",
      "        [-0.5263,  1.8179],\n",
      "        [-0.5263,  2.1484],\n",
      "        [-0.5263,  2.4789],\n",
      "        [-0.5263,  2.8095],\n",
      "        [-0.5263,  3.1400],\n",
      "        [-0.3158, -3.1400],\n",
      "        [-0.3158, -2.8095],\n",
      "        [-0.3158, -2.4789],\n",
      "        [-0.3158, -2.1484],\n",
      "        [-0.3158, -1.8179],\n",
      "        [-0.3158, -1.4874],\n",
      "        [-0.3158, -1.1568],\n",
      "        [-0.3158, -0.8263],\n",
      "        [-0.3158, -0.4958],\n",
      "        [-0.3158, -0.1653],\n",
      "        [-0.3158,  0.1653],\n",
      "        [-0.3158,  0.4958],\n",
      "        [-0.3158,  0.8263],\n",
      "        [-0.3158,  1.1568],\n",
      "        [-0.3158,  1.4874],\n",
      "        [-0.3158,  1.8179],\n",
      "        [-0.3158,  2.1484],\n",
      "        [-0.3158,  2.4789],\n",
      "        [-0.3158,  2.8095],\n",
      "        [-0.3158,  3.1400],\n",
      "        [-0.1053, -3.1400],\n",
      "        [-0.1053, -2.8095],\n",
      "        [-0.1053, -2.4789],\n",
      "        [-0.1053, -2.1484],\n",
      "        [-0.1053, -1.8179],\n",
      "        [-0.1053, -1.4874],\n",
      "        [-0.1053, -1.1568],\n",
      "        [-0.1053, -0.8263],\n",
      "        [-0.1053, -0.4958],\n",
      "        [-0.1053, -0.1653],\n",
      "        [-0.1053,  0.1653],\n",
      "        [-0.1053,  0.4958],\n",
      "        [-0.1053,  0.8263],\n",
      "        [-0.1053,  1.1568],\n",
      "        [-0.1053,  1.4874],\n",
      "        [-0.1053,  1.8179],\n",
      "        [-0.1053,  2.1484],\n",
      "        [-0.1053,  2.4789],\n",
      "        [-0.1053,  2.8095],\n",
      "        [-0.1053,  3.1400],\n",
      "        [ 0.1053, -3.1400],\n",
      "        [ 0.1053, -2.8095],\n",
      "        [ 0.1053, -2.4789],\n",
      "        [ 0.1053, -2.1484],\n",
      "        [ 0.1053, -1.8179],\n",
      "        [ 0.1053, -1.4874],\n",
      "        [ 0.1053, -1.1568],\n",
      "        [ 0.1053, -0.8263],\n",
      "        [ 0.1053, -0.4958],\n",
      "        [ 0.1053, -0.1653],\n",
      "        [ 0.1053,  0.1653],\n",
      "        [ 0.1053,  0.4958],\n",
      "        [ 0.1053,  0.8263],\n",
      "        [ 0.1053,  1.1568],\n",
      "        [ 0.1053,  1.4874],\n",
      "        [ 0.1053,  1.8179],\n",
      "        [ 0.1053,  2.1484],\n",
      "        [ 0.1053,  2.4789],\n",
      "        [ 0.1053,  2.8095],\n",
      "        [ 0.1053,  3.1400],\n",
      "        [ 0.3158, -3.1400],\n",
      "        [ 0.3158, -2.8095],\n",
      "        [ 0.3158, -2.4789],\n",
      "        [ 0.3158, -2.1484],\n",
      "        [ 0.3158, -1.8179],\n",
      "        [ 0.3158, -1.4874],\n",
      "        [ 0.3158, -1.1568],\n",
      "        [ 0.3158, -0.8263],\n",
      "        [ 0.3158, -0.4958],\n",
      "        [ 0.3158, -0.1653],\n",
      "        [ 0.3158,  0.1653],\n",
      "        [ 0.3158,  0.4958],\n",
      "        [ 0.3158,  0.8263],\n",
      "        [ 0.3158,  1.1568],\n",
      "        [ 0.3158,  1.4874],\n",
      "        [ 0.3158,  1.8179],\n",
      "        [ 0.3158,  2.1484],\n",
      "        [ 0.3158,  2.4789],\n",
      "        [ 0.3158,  2.8095],\n",
      "        [ 0.3158,  3.1400],\n",
      "        [ 0.5263, -3.1400],\n",
      "        [ 0.5263, -2.8095],\n",
      "        [ 0.5263, -2.4789],\n",
      "        [ 0.5263, -2.1484],\n",
      "        [ 0.5263, -1.8179],\n",
      "        [ 0.5263, -1.4874],\n",
      "        [ 0.5263, -1.1568],\n",
      "        [ 0.5263, -0.8263],\n",
      "        [ 0.5263, -0.4958],\n",
      "        [ 0.5263, -0.1653],\n",
      "        [ 0.5263,  0.1653],\n",
      "        [ 0.5263,  0.4958],\n",
      "        [ 0.5263,  0.8263],\n",
      "        [ 0.5263,  1.1568],\n",
      "        [ 0.5263,  1.4874],\n",
      "        [ 0.5263,  1.8179],\n",
      "        [ 0.5263,  2.1484],\n",
      "        [ 0.5263,  2.4789],\n",
      "        [ 0.5263,  2.8095],\n",
      "        [ 0.5263,  3.1400],\n",
      "        [ 0.7368, -3.1400],\n",
      "        [ 0.7368, -2.8095],\n",
      "        [ 0.7368, -2.4789],\n",
      "        [ 0.7368, -2.1484],\n",
      "        [ 0.7368, -1.8179],\n",
      "        [ 0.7368, -1.4874],\n",
      "        [ 0.7368, -1.1568],\n",
      "        [ 0.7368, -0.8263],\n",
      "        [ 0.7368, -0.4958],\n",
      "        [ 0.7368, -0.1653],\n",
      "        [ 0.7368,  0.1653],\n",
      "        [ 0.7368,  0.4958],\n",
      "        [ 0.7368,  0.8263],\n",
      "        [ 0.7368,  1.1568],\n",
      "        [ 0.7368,  1.4874],\n",
      "        [ 0.7368,  1.8179],\n",
      "        [ 0.7368,  2.1484],\n",
      "        [ 0.7368,  2.4789],\n",
      "        [ 0.7368,  2.8095],\n",
      "        [ 0.7368,  3.1400],\n",
      "        [ 0.9474, -3.1400],\n",
      "        [ 0.9474, -2.8095],\n",
      "        [ 0.9474, -2.4789],\n",
      "        [ 0.9474, -2.1484],\n",
      "        [ 0.9474, -1.8179],\n",
      "        [ 0.9474, -1.4874],\n",
      "        [ 0.9474, -1.1568],\n",
      "        [ 0.9474, -0.8263],\n",
      "        [ 0.9474, -0.4958],\n",
      "        [ 0.9474, -0.1653],\n",
      "        [ 0.9474,  0.1653],\n",
      "        [ 0.9474,  0.4958],\n",
      "        [ 0.9474,  0.8263],\n",
      "        [ 0.9474,  1.1568],\n",
      "        [ 0.9474,  1.4874],\n",
      "        [ 0.9474,  1.8179],\n",
      "        [ 0.9474,  2.1484],\n",
      "        [ 0.9474,  2.4789],\n",
      "        [ 0.9474,  2.8095],\n",
      "        [ 0.9474,  3.1400],\n",
      "        [ 1.1579, -3.1400],\n",
      "        [ 1.1579, -2.8095],\n",
      "        [ 1.1579, -2.4789],\n",
      "        [ 1.1579, -2.1484],\n",
      "        [ 1.1579, -1.8179],\n",
      "        [ 1.1579, -1.4874],\n",
      "        [ 1.1579, -1.1568],\n",
      "        [ 1.1579, -0.8263],\n",
      "        [ 1.1579, -0.4958],\n",
      "        [ 1.1579, -0.1653],\n",
      "        [ 1.1579,  0.1653],\n",
      "        [ 1.1579,  0.4958],\n",
      "        [ 1.1579,  0.8263],\n",
      "        [ 1.1579,  1.1568],\n",
      "        [ 1.1579,  1.4874],\n",
      "        [ 1.1579,  1.8179],\n",
      "        [ 1.1579,  2.1484],\n",
      "        [ 1.1579,  2.4789],\n",
      "        [ 1.1579,  2.8095],\n",
      "        [ 1.1579,  3.1400],\n",
      "        [ 1.3684, -3.1400],\n",
      "        [ 1.3684, -2.8095],\n",
      "        [ 1.3684, -2.4789],\n",
      "        [ 1.3684, -2.1484],\n",
      "        [ 1.3684, -1.8179],\n",
      "        [ 1.3684, -1.4874],\n",
      "        [ 1.3684, -1.1568],\n",
      "        [ 1.3684, -0.8263],\n",
      "        [ 1.3684, -0.4958],\n",
      "        [ 1.3684, -0.1653],\n",
      "        [ 1.3684,  0.1653],\n",
      "        [ 1.3684,  0.4958],\n",
      "        [ 1.3684,  0.8263],\n",
      "        [ 1.3684,  1.1568],\n",
      "        [ 1.3684,  1.4874],\n",
      "        [ 1.3684,  1.8179],\n",
      "        [ 1.3684,  2.1484],\n",
      "        [ 1.3684,  2.4789],\n",
      "        [ 1.3684,  2.8095],\n",
      "        [ 1.3684,  3.1400],\n",
      "        [ 1.5789, -3.1400],\n",
      "        [ 1.5789, -2.8095],\n",
      "        [ 1.5789, -2.4789],\n",
      "        [ 1.5789, -2.1484],\n",
      "        [ 1.5789, -1.8179],\n",
      "        [ 1.5789, -1.4874],\n",
      "        [ 1.5789, -1.1568],\n",
      "        [ 1.5789, -0.8263],\n",
      "        [ 1.5789, -0.4958],\n",
      "        [ 1.5789, -0.1653],\n",
      "        [ 1.5789,  0.1653],\n",
      "        [ 1.5789,  0.4958],\n",
      "        [ 1.5789,  0.8263],\n",
      "        [ 1.5789,  1.1568],\n",
      "        [ 1.5789,  1.4874],\n",
      "        [ 1.5789,  1.8179],\n",
      "        [ 1.5789,  2.1484],\n",
      "        [ 1.5789,  2.4789],\n",
      "        [ 1.5789,  2.8095],\n",
      "        [ 1.5789,  3.1400],\n",
      "        [ 1.7895, -3.1400],\n",
      "        [ 1.7895, -2.8095],\n",
      "        [ 1.7895, -2.4789],\n",
      "        [ 1.7895, -2.1484],\n",
      "        [ 1.7895, -1.8179],\n",
      "        [ 1.7895, -1.4874],\n",
      "        [ 1.7895, -1.1568],\n",
      "        [ 1.7895, -0.8263],\n",
      "        [ 1.7895, -0.4958],\n",
      "        [ 1.7895, -0.1653],\n",
      "        [ 1.7895,  0.1653],\n",
      "        [ 1.7895,  0.4958],\n",
      "        [ 1.7895,  0.8263],\n",
      "        [ 1.7895,  1.1568],\n",
      "        [ 1.7895,  1.4874],\n",
      "        [ 1.7895,  1.8179],\n",
      "        [ 1.7895,  2.1484],\n",
      "        [ 1.7895,  2.4789],\n",
      "        [ 1.7895,  2.8095],\n",
      "        [ 1.7895,  3.1400],\n",
      "        [ 2.0000, -3.1400],\n",
      "        [ 2.0000, -2.8095],\n",
      "        [ 2.0000, -2.4789],\n",
      "        [ 2.0000, -2.1484],\n",
      "        [ 2.0000, -1.8179],\n",
      "        [ 2.0000, -1.4874],\n",
      "        [ 2.0000, -1.1568],\n",
      "        [ 2.0000, -0.8263],\n",
      "        [ 2.0000, -0.4958],\n",
      "        [ 2.0000, -0.1653],\n",
      "        [ 2.0000,  0.1653],\n",
      "        [ 2.0000,  0.4958],\n",
      "        [ 2.0000,  0.8263],\n",
      "        [ 2.0000,  1.1568],\n",
      "        [ 2.0000,  1.4874],\n",
      "        [ 2.0000,  1.8179],\n",
      "        [ 2.0000,  2.1484],\n",
      "        [ 2.0000,  2.4789],\n",
      "        [ 2.0000,  2.8095],\n",
      "        [ 2.0000,  3.1400]], requires_grad=True)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([[-9.5553e-03,  3.1522e+00],\n",
      "        [-1.9563e+00,  5.5947e+00],\n",
      "        [-3.6912e+00,  8.4351e+00],\n",
      "        [-5.0266e+00,  1.1507e+01],\n",
      "        [-5.8178e+00,  1.4619e+01],\n",
      "        [-5.9791e+00,  1.7567e+01],\n",
      "        [-5.4932e+00,  2.0151e+01],\n",
      "        [-4.4126e+00,  2.2187e+01],\n",
      "        [-2.8544e+00,  2.3525e+01],\n",
      "        [-9.8707e-01,  2.4056e+01],\n",
      "        [ 9.8707e-01,  2.3726e+01],\n",
      "        [ 2.8544e+00,  2.2533e+01],\n",
      "        [ 4.4126e+00,  2.0534e+01],\n",
      "        [ 5.4932e+00,  1.7837e+01],\n",
      "        [ 5.9791e+00,  1.4592e+01],\n",
      "        [ 5.8178e+00,  1.0983e+01],\n",
      "        [ 5.0266e+00,  7.2102e+00],\n",
      "        [ 3.6912e+00,  3.4772e+00],\n",
      "        [ 1.9563e+00, -2.4210e-02],\n",
      "        [ 9.5553e-03, -3.1278e+00],\n",
      "        [-9.5553e-03,  3.1509e+00],\n",
      "        [-1.9563e+00,  5.3016e+00],\n",
      "        [-3.6912e+00,  7.8081e+00],\n",
      "        [-5.0266e+00,  1.0522e+01],\n",
      "        [-5.8178e+00,  1.3271e+01],\n",
      "        [-5.9791e+00,  1.5875e+01],\n",
      "        [-5.4932e+00,  1.8151e+01],\n",
      "        [-4.4126e+00,  1.9938e+01],\n",
      "        [-2.8544e+00,  2.1100e+01],\n",
      "        [-9.8707e-01,  2.1541e+01],\n",
      "        [ 9.8707e-01,  2.1211e+01],\n",
      "        [ 2.8544e+00,  2.0109e+01],\n",
      "        [ 4.4126e+00,  1.8286e+01],\n",
      "        [ 5.4932e+00,  1.5838e+01],\n",
      "        [ 5.9791e+00,  1.2900e+01],\n",
      "        [ 5.8178e+00,  9.6357e+00],\n",
      "        [ 5.0266e+00,  6.2251e+00],\n",
      "        [ 3.6912e+00,  2.8502e+00],\n",
      "        [ 1.9563e+00, -3.1740e-01],\n",
      "        [ 9.5553e-03, -3.1291e+00],\n",
      "        [-9.5553e-03,  3.1496e+00],\n",
      "        [-1.9563e+00,  5.0084e+00],\n",
      "        [-3.6912e+00,  7.1811e+00],\n",
      "        [-5.0266e+00,  9.5368e+00],\n",
      "        [-5.8178e+00,  1.1924e+01],\n",
      "        [-5.9791e+00,  1.4182e+01],\n",
      "        [-5.4932e+00,  1.6152e+01],\n",
      "        [-4.4126e+00,  1.7690e+01],\n",
      "        [-2.8544e+00,  1.8676e+01],\n",
      "        [-9.8707e-01,  1.9027e+01],\n",
      "        [ 9.8707e-01,  1.8696e+01],\n",
      "        [ 2.8544e+00,  1.7685e+01],\n",
      "        [ 4.4126e+00,  1.6037e+01],\n",
      "        [ 5.4932e+00,  1.3838e+01],\n",
      "        [ 5.9791e+00,  1.1207e+01],\n",
      "        [ 5.8178e+00,  8.2882e+00],\n",
      "        [ 5.0266e+00,  5.2400e+00],\n",
      "        [ 3.6912e+00,  2.2233e+00],\n",
      "        [ 1.9563e+00, -6.1058e-01],\n",
      "        [ 9.5553e-03, -3.1304e+00],\n",
      "        [-9.5553e-03,  3.1483e+00],\n",
      "        [-1.9563e+00,  4.7152e+00],\n",
      "        [-3.6912e+00,  6.5542e+00],\n",
      "        [-5.0266e+00,  8.5517e+00],\n",
      "        [-5.8178e+00,  1.0577e+01],\n",
      "        [-5.9791e+00,  1.2489e+01],\n",
      "        [-5.4932e+00,  1.4153e+01],\n",
      "        [-4.4126e+00,  1.5441e+01],\n",
      "        [-2.8544e+00,  1.6252e+01],\n",
      "        [-9.8707e-01,  1.6512e+01],\n",
      "        [ 9.8707e-01,  1.6181e+01],\n",
      "        [ 2.8544e+00,  1.5261e+01],\n",
      "        [ 4.4126e+00,  1.3789e+01],\n",
      "        [ 5.4932e+00,  1.1839e+01],\n",
      "        [ 5.9791e+00,  9.5146e+00],\n",
      "        [ 5.8178e+00,  6.9407e+00],\n",
      "        [ 5.0266e+00,  4.2549e+00],\n",
      "        [ 3.6912e+00,  1.5963e+00],\n",
      "        [ 1.9563e+00, -9.0377e-01],\n",
      "        [ 9.5553e-03, -3.1317e+00],\n",
      "        [-9.5553e-03,  3.1470e+00],\n",
      "        [-1.9563e+00,  4.4220e+00],\n",
      "        [-3.6912e+00,  5.9272e+00],\n",
      "        [-5.0266e+00,  7.5666e+00],\n",
      "        [-5.8178e+00,  9.2291e+00],\n",
      "        [-5.9791e+00,  1.0797e+01],\n",
      "        [-5.4932e+00,  1.2153e+01],\n",
      "        [-4.4126e+00,  1.3193e+01],\n",
      "        [-2.8544e+00,  1.3828e+01],\n",
      "        [-9.8707e-01,  1.3997e+01],\n",
      "        [ 9.8707e-01,  1.3666e+01],\n",
      "        [ 2.8544e+00,  1.2837e+01],\n",
      "        [ 4.4126e+00,  1.1540e+01],\n",
      "        [ 5.4932e+00,  9.8396e+00],\n",
      "        [ 5.9791e+00,  7.8220e+00],\n",
      "        [ 5.8178e+00,  5.5933e+00],\n",
      "        [ 5.0266e+00,  3.2697e+00],\n",
      "        [ 3.6912e+00,  9.6933e-01],\n",
      "        [ 1.9563e+00, -1.1970e+00],\n",
      "        [ 9.5553e-03, -3.1330e+00],\n",
      "        [-9.5553e-03,  3.1458e+00],\n",
      "        [-1.9563e+00,  4.1288e+00],\n",
      "        [-3.6912e+00,  5.3003e+00],\n",
      "        [-5.0266e+00,  6.5815e+00],\n",
      "        [-5.8178e+00,  7.8816e+00],\n",
      "        [-5.9791e+00,  9.1041e+00],\n",
      "        [-5.4932e+00,  1.0154e+01],\n",
      "        [-4.4126e+00,  1.0944e+01],\n",
      "        [-2.8544e+00,  1.1404e+01],\n",
      "        [-9.8707e-01,  1.1482e+01],\n",
      "        [ 9.8707e-01,  1.1151e+01],\n",
      "        [ 2.8544e+00,  1.0413e+01],\n",
      "        [ 4.4126e+00,  9.2918e+00],\n",
      "        [ 5.4932e+00,  7.8403e+00],\n",
      "        [ 5.9791e+00,  6.1294e+00],\n",
      "        [ 5.8178e+00,  4.2458e+00],\n",
      "        [ 5.0266e+00,  2.2846e+00],\n",
      "        [ 3.6912e+00,  3.4237e-01],\n",
      "        [ 1.9563e+00, -1.4901e+00],\n",
      "        [ 9.5553e-03, -3.1342e+00],\n",
      "        [-9.5553e-03,  3.1445e+00],\n",
      "        [-1.9563e+00,  3.8356e+00],\n",
      "        [-3.6912e+00,  4.6733e+00],\n",
      "        [-5.0266e+00,  5.5963e+00],\n",
      "        [-5.8178e+00,  6.5341e+00],\n",
      "        [-5.9791e+00,  7.4115e+00],\n",
      "        [-5.4932e+00,  8.1546e+00],\n",
      "        [-4.4126e+00,  8.6960e+00],\n",
      "        [-2.8544e+00,  8.9801e+00],\n",
      "        [-9.8707e-01,  8.9672e+00],\n",
      "        [ 9.8707e-01,  8.6366e+00],\n",
      "        [ 2.8544e+00,  7.9885e+00],\n",
      "        [ 4.4126e+00,  7.0434e+00],\n",
      "        [ 5.4932e+00,  5.8409e+00],\n",
      "        [ 5.9791e+00,  4.4368e+00],\n",
      "        [ 5.8178e+00,  2.8983e+00],\n",
      "        [ 5.0266e+00,  1.2995e+00],\n",
      "        [ 3.6912e+00, -2.8459e-01],\n",
      "        [ 1.9563e+00, -1.7833e+00],\n",
      "        [ 9.5553e-03, -3.1355e+00],\n",
      "        [-9.5553e-03,  3.1432e+00],\n",
      "        [-1.9563e+00,  3.5424e+00],\n",
      "        [-3.6912e+00,  4.0463e+00],\n",
      "        [-5.0266e+00,  4.6112e+00],\n",
      "        [-5.8178e+00,  5.1866e+00],\n",
      "        [-5.9791e+00,  5.7189e+00],\n",
      "        [-5.4932e+00,  6.1552e+00],\n",
      "        [-4.4126e+00,  6.4475e+00],\n",
      "        [-2.8544e+00,  6.5560e+00],\n",
      "        [-9.8707e-01,  6.4523e+00],\n",
      "        [ 9.8707e-01,  6.1218e+00],\n",
      "        [ 2.8544e+00,  5.5644e+00],\n",
      "        [ 4.4126e+00,  4.7949e+00],\n",
      "        [ 5.4932e+00,  3.8415e+00],\n",
      "        [ 5.9791e+00,  2.7441e+00],\n",
      "        [ 5.8178e+00,  1.5508e+00],\n",
      "        [ 5.0266e+00,  3.1438e-01],\n",
      "        [ 3.6912e+00, -9.1155e-01],\n",
      "        [ 1.9563e+00, -2.0765e+00],\n",
      "        [ 9.5553e-03, -3.1368e+00],\n",
      "        [-9.5553e-03,  3.1419e+00],\n",
      "        [-1.9563e+00,  3.2493e+00],\n",
      "        [-3.6912e+00,  3.4194e+00],\n",
      "        [-5.0266e+00,  3.6261e+00],\n",
      "        [-5.8178e+00,  3.8391e+00],\n",
      "        [-5.9791e+00,  4.0263e+00],\n",
      "        [-5.4932e+00,  4.1559e+00],\n",
      "        [-4.4126e+00,  4.1990e+00],\n",
      "        [-2.8544e+00,  4.1319e+00],\n",
      "        [-9.8707e-01,  3.9375e+00],\n",
      "        [ 9.8707e-01,  3.6070e+00],\n",
      "        [ 2.8544e+00,  3.1403e+00],\n",
      "        [ 4.4126e+00,  2.5464e+00],\n",
      "        [ 5.4932e+00,  1.8422e+00],\n",
      "        [ 5.9791e+00,  1.0515e+00],\n",
      "        [ 5.8178e+00,  2.0333e-01],\n",
      "        [ 5.0266e+00, -6.7074e-01],\n",
      "        [ 3.6912e+00, -1.5385e+00],\n",
      "        [ 1.9563e+00, -2.3697e+00],\n",
      "        [ 9.5553e-03, -3.1381e+00],\n",
      "        [-9.5553e-03,  3.1406e+00],\n",
      "        [-1.9563e+00,  2.9561e+00],\n",
      "        [-3.6912e+00,  2.7924e+00],\n",
      "        [-5.0266e+00,  2.6410e+00],\n",
      "        [-5.8178e+00,  2.4916e+00],\n",
      "        [-5.9791e+00,  2.3337e+00],\n",
      "        [-5.4932e+00,  2.1565e+00],\n",
      "        [-4.4126e+00,  1.9506e+00],\n",
      "        [-2.8544e+00,  1.7078e+00],\n",
      "        [-9.8707e-01,  1.4227e+00],\n",
      "        [ 9.8707e-01,  1.0922e+00],\n",
      "        [ 2.8544e+00,  7.1625e-01],\n",
      "        [ 4.4126e+00,  2.9792e-01],\n",
      "        [ 5.4932e+00, -1.5717e-01],\n",
      "        [ 5.9791e+00, -6.4107e-01],\n",
      "        [ 5.8178e+00, -1.1442e+00],\n",
      "        [ 5.0266e+00, -1.6559e+00],\n",
      "        [ 3.6912e+00, -2.1655e+00],\n",
      "        [ 1.9563e+00, -2.6629e+00],\n",
      "        [ 9.5553e-03, -3.1394e+00],\n",
      "        [-9.5553e-03,  3.1394e+00],\n",
      "        [-1.9563e+00,  2.6629e+00],\n",
      "        [-3.6912e+00,  2.1655e+00],\n",
      "        [-5.0266e+00,  1.6559e+00],\n",
      "        [-5.8178e+00,  1.1442e+00],\n",
      "        [-5.9791e+00,  6.4107e-01],\n",
      "        [-5.4932e+00,  1.5717e-01],\n",
      "        [-4.4126e+00, -2.9792e-01],\n",
      "        [-2.8544e+00, -7.1625e-01],\n",
      "        [-9.8707e-01, -1.0922e+00],\n",
      "        [ 9.8707e-01, -1.4227e+00],\n",
      "        [ 2.8544e+00, -1.7078e+00],\n",
      "        [ 4.4126e+00, -1.9506e+00],\n",
      "        [ 5.4932e+00, -2.1565e+00],\n",
      "        [ 5.9791e+00, -2.3337e+00],\n",
      "        [ 5.8178e+00, -2.4916e+00],\n",
      "        [ 5.0266e+00, -2.6410e+00],\n",
      "        [ 3.6912e+00, -2.7924e+00],\n",
      "        [ 1.9563e+00, -2.9561e+00],\n",
      "        [ 9.5553e-03, -3.1406e+00],\n",
      "        [-9.5553e-03,  3.1381e+00],\n",
      "        [-1.9563e+00,  2.3697e+00],\n",
      "        [-3.6912e+00,  1.5385e+00],\n",
      "        [-5.0266e+00,  6.7074e-01],\n",
      "        [-5.8178e+00, -2.0333e-01],\n",
      "        [-5.9791e+00, -1.0515e+00],\n",
      "        [-5.4932e+00, -1.8422e+00],\n",
      "        [-4.4126e+00, -2.5464e+00],\n",
      "        [-2.8544e+00, -3.1403e+00],\n",
      "        [-9.8707e-01, -3.6070e+00],\n",
      "        [ 9.8707e-01, -3.9375e+00],\n",
      "        [ 2.8544e+00, -4.1319e+00],\n",
      "        [ 4.4126e+00, -4.1990e+00],\n",
      "        [ 5.4932e+00, -4.1559e+00],\n",
      "        [ 5.9791e+00, -4.0263e+00],\n",
      "        [ 5.8178e+00, -3.8391e+00],\n",
      "        [ 5.0266e+00, -3.6261e+00],\n",
      "        [ 3.6912e+00, -3.4194e+00],\n",
      "        [ 1.9563e+00, -3.2493e+00],\n",
      "        [ 9.5553e-03, -3.1419e+00],\n",
      "        [-9.5553e-03,  3.1368e+00],\n",
      "        [-1.9563e+00,  2.0765e+00],\n",
      "        [-3.6912e+00,  9.1155e-01],\n",
      "        [-5.0266e+00, -3.1438e-01],\n",
      "        [-5.8178e+00, -1.5508e+00],\n",
      "        [-5.9791e+00, -2.7441e+00],\n",
      "        [-5.4932e+00, -3.8415e+00],\n",
      "        [-4.4126e+00, -4.7949e+00],\n",
      "        [-2.8544e+00, -5.5644e+00],\n",
      "        [-9.8707e-01, -6.1218e+00],\n",
      "        [ 9.8707e-01, -6.4523e+00],\n",
      "        [ 2.8544e+00, -6.5560e+00],\n",
      "        [ 4.4126e+00, -6.4475e+00],\n",
      "        [ 5.4932e+00, -6.1552e+00],\n",
      "        [ 5.9791e+00, -5.7189e+00],\n",
      "        [ 5.8178e+00, -5.1866e+00],\n",
      "        [ 5.0266e+00, -4.6112e+00],\n",
      "        [ 3.6912e+00, -4.0463e+00],\n",
      "        [ 1.9563e+00, -3.5424e+00],\n",
      "        [ 9.5553e-03, -3.1432e+00],\n",
      "        [-9.5553e-03,  3.1355e+00],\n",
      "        [-1.9563e+00,  1.7833e+00],\n",
      "        [-3.6912e+00,  2.8459e-01],\n",
      "        [-5.0266e+00, -1.2995e+00],\n",
      "        [-5.8178e+00, -2.8983e+00],\n",
      "        [-5.9791e+00, -4.4368e+00],\n",
      "        [-5.4932e+00, -5.8409e+00],\n",
      "        [-4.4126e+00, -7.0434e+00],\n",
      "        [-2.8544e+00, -7.9885e+00],\n",
      "        [-9.8707e-01, -8.6366e+00],\n",
      "        [ 9.8707e-01, -8.9672e+00],\n",
      "        [ 2.8544e+00, -8.9801e+00],\n",
      "        [ 4.4126e+00, -8.6960e+00],\n",
      "        [ 5.4932e+00, -8.1546e+00],\n",
      "        [ 5.9791e+00, -7.4115e+00],\n",
      "        [ 5.8178e+00, -6.5341e+00],\n",
      "        [ 5.0266e+00, -5.5963e+00],\n",
      "        [ 3.6912e+00, -4.6733e+00],\n",
      "        [ 1.9563e+00, -3.8356e+00],\n",
      "        [ 9.5553e-03, -3.1445e+00],\n",
      "        [-9.5553e-03,  3.1342e+00],\n",
      "        [-1.9563e+00,  1.4901e+00],\n",
      "        [-3.6912e+00, -3.4237e-01],\n",
      "        [-5.0266e+00, -2.2846e+00],\n",
      "        [-5.8178e+00, -4.2458e+00],\n",
      "        [-5.9791e+00, -6.1294e+00],\n",
      "        [-5.4932e+00, -7.8403e+00],\n",
      "        [-4.4126e+00, -9.2918e+00],\n",
      "        [-2.8544e+00, -1.0413e+01],\n",
      "        [-9.8707e-01, -1.1151e+01],\n",
      "        [ 9.8707e-01, -1.1482e+01],\n",
      "        [ 2.8544e+00, -1.1404e+01],\n",
      "        [ 4.4126e+00, -1.0944e+01],\n",
      "        [ 5.4932e+00, -1.0154e+01],\n",
      "        [ 5.9791e+00, -9.1041e+00],\n",
      "        [ 5.8178e+00, -7.8816e+00],\n",
      "        [ 5.0266e+00, -6.5815e+00],\n",
      "        [ 3.6912e+00, -5.3003e+00],\n",
      "        [ 1.9563e+00, -4.1288e+00],\n",
      "        [ 9.5553e-03, -3.1458e+00],\n",
      "        [-9.5553e-03,  3.1330e+00],\n",
      "        [-1.9563e+00,  1.1970e+00],\n",
      "        [-3.6912e+00, -9.6933e-01],\n",
      "        [-5.0266e+00, -3.2697e+00],\n",
      "        [-5.8178e+00, -5.5933e+00],\n",
      "        [-5.9791e+00, -7.8220e+00],\n",
      "        [-5.4932e+00, -9.8396e+00],\n",
      "        [-4.4126e+00, -1.1540e+01],\n",
      "        [-2.8544e+00, -1.2837e+01],\n",
      "        [-9.8707e-01, -1.3666e+01],\n",
      "        [ 9.8707e-01, -1.3997e+01],\n",
      "        [ 2.8544e+00, -1.3828e+01],\n",
      "        [ 4.4126e+00, -1.3193e+01],\n",
      "        [ 5.4932e+00, -1.2153e+01],\n",
      "        [ 5.9791e+00, -1.0797e+01],\n",
      "        [ 5.8178e+00, -9.2291e+00],\n",
      "        [ 5.0266e+00, -7.5666e+00],\n",
      "        [ 3.6912e+00, -5.9272e+00],\n",
      "        [ 1.9563e+00, -4.4220e+00],\n",
      "        [ 9.5553e-03, -3.1470e+00],\n",
      "        [-9.5553e-03,  3.1317e+00],\n",
      "        [-1.9563e+00,  9.0377e-01],\n",
      "        [-3.6912e+00, -1.5963e+00],\n",
      "        [-5.0266e+00, -4.2549e+00],\n",
      "        [-5.8178e+00, -6.9407e+00],\n",
      "        [-5.9791e+00, -9.5146e+00],\n",
      "        [-5.4932e+00, -1.1839e+01],\n",
      "        [-4.4126e+00, -1.3789e+01],\n",
      "        [-2.8544e+00, -1.5261e+01],\n",
      "        [-9.8707e-01, -1.6181e+01],\n",
      "        [ 9.8707e-01, -1.6512e+01],\n",
      "        [ 2.8544e+00, -1.6252e+01],\n",
      "        [ 4.4126e+00, -1.5441e+01],\n",
      "        [ 5.4932e+00, -1.4153e+01],\n",
      "        [ 5.9791e+00, -1.2489e+01],\n",
      "        [ 5.8178e+00, -1.0577e+01],\n",
      "        [ 5.0266e+00, -8.5517e+00],\n",
      "        [ 3.6912e+00, -6.5542e+00],\n",
      "        [ 1.9563e+00, -4.7152e+00],\n",
      "        [ 9.5553e-03, -3.1483e+00],\n",
      "        [-9.5553e-03,  3.1304e+00],\n",
      "        [-1.9563e+00,  6.1058e-01],\n",
      "        [-3.6912e+00, -2.2233e+00],\n",
      "        [-5.0266e+00, -5.2400e+00],\n",
      "        [-5.8178e+00, -8.2882e+00],\n",
      "        [-5.9791e+00, -1.1207e+01],\n",
      "        [-5.4932e+00, -1.3838e+01],\n",
      "        [-4.4126e+00, -1.6037e+01],\n",
      "        [-2.8544e+00, -1.7685e+01],\n",
      "        [-9.8707e-01, -1.8696e+01],\n",
      "        [ 9.8707e-01, -1.9027e+01],\n",
      "        [ 2.8544e+00, -1.8676e+01],\n",
      "        [ 4.4126e+00, -1.7690e+01],\n",
      "        [ 5.4932e+00, -1.6152e+01],\n",
      "        [ 5.9791e+00, -1.4182e+01],\n",
      "        [ 5.8178e+00, -1.1924e+01],\n",
      "        [ 5.0266e+00, -9.5368e+00],\n",
      "        [ 3.6912e+00, -7.1811e+00],\n",
      "        [ 1.9563e+00, -5.0084e+00],\n",
      "        [ 9.5553e-03, -3.1496e+00],\n",
      "        [-9.5553e-03,  3.1291e+00],\n",
      "        [-1.9563e+00,  3.1740e-01],\n",
      "        [-3.6912e+00, -2.8502e+00],\n",
      "        [-5.0266e+00, -6.2251e+00],\n",
      "        [-5.8178e+00, -9.6357e+00],\n",
      "        [-5.9791e+00, -1.2900e+01],\n",
      "        [-5.4932e+00, -1.5838e+01],\n",
      "        [-4.4126e+00, -1.8286e+01],\n",
      "        [-2.8544e+00, -2.0109e+01],\n",
      "        [-9.8707e-01, -2.1211e+01],\n",
      "        [ 9.8707e-01, -2.1541e+01],\n",
      "        [ 2.8544e+00, -2.1100e+01],\n",
      "        [ 4.4126e+00, -1.9938e+01],\n",
      "        [ 5.4932e+00, -1.8151e+01],\n",
      "        [ 5.9791e+00, -1.5875e+01],\n",
      "        [ 5.8178e+00, -1.3271e+01],\n",
      "        [ 5.0266e+00, -1.0522e+01],\n",
      "        [ 3.6912e+00, -7.8081e+00],\n",
      "        [ 1.9563e+00, -5.3016e+00],\n",
      "        [ 9.5553e-03, -3.1509e+00],\n",
      "        [-9.5553e-03,  3.1278e+00],\n",
      "        [-1.9563e+00,  2.4210e-02],\n",
      "        [-3.6912e+00, -3.4772e+00],\n",
      "        [-5.0266e+00, -7.2102e+00],\n",
      "        [-5.8178e+00, -1.0983e+01],\n",
      "        [-5.9791e+00, -1.4592e+01],\n",
      "        [-5.4932e+00, -1.7837e+01],\n",
      "        [-4.4126e+00, -2.0534e+01],\n",
      "        [-2.8544e+00, -2.2533e+01],\n",
      "        [-9.8707e-01, -2.3726e+01],\n",
      "        [ 9.8707e-01, -2.4056e+01],\n",
      "        [ 2.8544e+00, -2.3525e+01],\n",
      "        [ 4.4126e+00, -2.2187e+01],\n",
      "        [ 5.4932e+00, -2.0151e+01],\n",
      "        [ 5.9791e+00, -1.7567e+01],\n",
      "        [ 5.8178e+00, -1.4619e+01],\n",
      "        [ 5.0266e+00, -1.1507e+01],\n",
      "        [ 3.6912e+00, -8.4351e+00],\n",
      "        [ 1.9563e+00, -5.5947e+00],\n",
      "        [ 9.5553e-03, -3.1522e+00]])\n",
      "tensor([[-1.5880e-01,  4.2273e+00],\n",
      "        [-1.9776e+00,  6.3236e+00],\n",
      "        [-3.7005e+00,  8.8888e+00],\n",
      "        [-5.0213e+00,  1.1790e+01],\n",
      "        [-5.7501e+00,  1.4786e+01],\n",
      "        [-5.8542e+00,  1.7590e+01],\n",
      "        [-5.3791e+00,  1.9963e+01],\n",
      "        [-4.3729e+00,  2.1777e+01],\n",
      "        [-2.8847e+00,  2.2991e+01],\n",
      "        [-1.0209e+00,  2.3562e+01],\n",
      "        [ 1.0085e+00,  2.3389e+01],\n",
      "        [ 2.9206e+00,  2.2353e+01],\n",
      "        [ 4.4587e+00,  2.0417e+01],\n",
      "        [ 5.4695e+00,  1.7686e+01],\n",
      "        [ 5.9018e+00,  1.4387e+01],\n",
      "        [ 5.7525e+00,  1.0809e+01],\n",
      "        [ 5.0253e+00,  7.2380e+00],\n",
      "        [ 3.7435e+00,  3.9143e+00],\n",
      "        [ 2.0125e+00,  9.9154e-01],\n",
      "        [ 6.5968e-02, -1.4855e+00],\n",
      "        [-1.2670e-01,  3.8764e+00],\n",
      "        [-1.9598e+00,  5.6922e+00],\n",
      "        [-3.7012e+00,  7.9589e+00],\n",
      "        [-5.0413e+00,  1.0576e+01],\n",
      "        [-5.7850e+00,  1.3329e+01],\n",
      "        [-5.8945e+00,  1.5932e+01],\n",
      "        [-5.4150e+00,  1.8124e+01],\n",
      "        [-4.3994e+00,  1.9755e+01],\n",
      "        [-2.9027e+00,  2.0793e+01],\n",
      "        [-1.0336e+00,  2.1241e+01],\n",
      "        [ 9.9976e-01,  2.1048e+01],\n",
      "        [ 2.9175e+00,  2.0109e+01],\n",
      "        [ 4.4627e+00,  1.8356e+01],\n",
      "        [ 5.4796e+00,  1.5847e+01],\n",
      "        [ 5.9139e+00,  1.2777e+01],\n",
      "        [ 5.7606e+00,  9.4239e+00],\n",
      "        [ 5.0239e+00,  6.0746e+00],\n",
      "        [ 3.7304e+00,  2.9722e+00],\n",
      "        [ 1.9915e+00,  2.6688e-01],\n",
      "        [ 4.7224e-02, -2.0046e+00],\n",
      "        [-9.8662e-02,  3.6092e+00],\n",
      "        [-1.9426e+00,  5.1438e+00],\n",
      "        [-3.6983e+00,  7.0984e+00],\n",
      "        [-5.0539e+00,  9.4048e+00],\n",
      "        [-5.8104e+00,  1.1880e+01],\n",
      "        [-5.9253e+00,  1.4253e+01],\n",
      "        [-5.4424e+00,  1.6248e+01],\n",
      "        [-4.4187e+00,  1.7693e+01],\n",
      "        [-2.9149e+00,  1.8553e+01],\n",
      "        [-1.0426e+00,  1.8867e+01],\n",
      "        [ 9.9225e-01,  1.8639e+01],\n",
      "        [ 2.9133e+00,  1.7789e+01],\n",
      "        [ 4.4647e+00,  1.6228e+01],\n",
      "        [ 5.4879e+00,  1.3966e+01],\n",
      "        [ 5.9251e+00,  1.1159e+01],\n",
      "        [ 5.7690e+00,  8.0636e+00],\n",
      "        [ 5.0239e+00,  4.9651e+00],\n",
      "        [ 3.7197e+00,  2.1055e+00],\n",
      "        [ 1.9742e+00, -3.6954e-01],\n",
      "        [ 3.3230e-02, -2.4305e+00],\n",
      "        [-7.5284e-02,  3.4184e+00],\n",
      "        [-1.9272e+00,  4.6741e+00],\n",
      "        [-3.6934e+00,  6.3075e+00],\n",
      "        [-5.0610e+00,  8.2797e+00],\n",
      "        [-5.8279e+00,  1.0445e+01],\n",
      "        [-5.9475e+00,  1.2555e+01],\n",
      "        [-5.4618e+00,  1.4335e+01],\n",
      "        [-4.4309e+00,  1.5590e+01],\n",
      "        [-2.9214e+00,  1.6274e+01],\n",
      "        [-1.0478e+00,  1.6453e+01],\n",
      "        [ 9.8603e-01,  1.6177e+01],\n",
      "        [ 2.9083e+00,  1.5406e+01],\n",
      "        [ 4.4647e+00,  1.4040e+01],\n",
      "        [ 5.4943e+00,  1.2045e+01],\n",
      "        [ 5.9352e+00,  9.5306e+00],\n",
      "        [ 5.7774e+00,  6.7268e+00],\n",
      "        [ 5.0251e+00,  3.9081e+00],\n",
      "        [ 3.7114e+00,  1.3123e+00],\n",
      "        [ 1.9604e+00, -9.2019e-01],\n",
      "        [ 2.3697e-02, -2.7663e+00],\n",
      "        [-5.6883e-02,  3.2952e+00],\n",
      "        [-1.9144e+00,  4.2764e+00],\n",
      "        [-3.6881e+00,  5.5837e+00],\n",
      "        [-5.0643e+00,  7.2035e+00],\n",
      "        [-5.8393e+00,  9.0279e+00],\n",
      "        [-5.9625e+00,  1.0844e+01],\n",
      "        [-5.4740e+00,  1.2385e+01],\n",
      "        [-4.4367e+00,  1.3445e+01],\n",
      "        [-2.9225e+00,  1.3961e+01],\n",
      "        [-1.0493e+00,  1.4008e+01],\n",
      "        [ 9.8122e-01,  1.3679e+01],\n",
      "        [ 2.9025e+00,  1.2973e+01],\n",
      "        [ 4.4628e+00,  1.1799e+01],\n",
      "        [ 5.4987e+00,  1.0085e+01],\n",
      "        [ 5.9440e+00,  7.8908e+00],\n",
      "        [ 5.7857e+00,  5.4106e+00],\n",
      "        [ 5.0272e+00,  2.9003e+00],\n",
      "        [ 3.7050e+00,  5.8896e-01],\n",
      "        [ 1.9497e+00, -1.3894e+00],\n",
      "        [ 1.8247e-02, -3.0170e+00],\n",
      "        [-4.3519e-02,  3.2290e+00],\n",
      "        [-1.9048e+00,  3.9417e+00],\n",
      "        [-3.6833e+00,  4.9217e+00],\n",
      "        [-5.0654e+00,  6.1758e+00],\n",
      "        [-5.8462e+00,  7.6335e+00],\n",
      "        [-5.9718e+00,  9.1227e+00],\n",
      "        [-5.4803e+00,  1.0402e+01],\n",
      "        [-4.4367e+00,  1.1259e+01],\n",
      "        [-2.9185e+00,  1.1614e+01],\n",
      "        [-1.0472e+00,  1.1539e+01],\n",
      "        [ 9.7794e-01,  1.1156e+01],\n",
      "        [ 2.8962e+00,  1.0507e+01],\n",
      "        [ 4.4591e+00,  9.5155e+00],\n",
      "        [ 5.5012e+00,  8.0891e+00],\n",
      "        [ 5.9517e+00,  6.2378e+00],\n",
      "        [ 5.7936e+00,  4.1107e+00],\n",
      "        [ 5.0300e+00,  1.9365e+00],\n",
      "        [ 3.7003e+00, -7.0353e-02],\n",
      "        [ 1.9416e+00, -1.7835e+00],\n",
      "        [ 1.6442e-02, -3.1891e+00],\n",
      "        [-3.5021e-02,  3.2088e+00],\n",
      "        [-1.8984e+00,  3.6594e+00],\n",
      "        [-3.6797e+00,  4.3137e+00],\n",
      "        [-5.0655e+00,  5.1941e+00],\n",
      "        [-5.8500e+00,  6.2644e+00],\n",
      "        [-5.9770e+00,  7.3981e+00],\n",
      "        [-5.4820e+00,  8.3893e+00],\n",
      "        [-4.4319e+00,  9.0330e+00],\n",
      "        [-2.9100e+00,  9.2341e+00],\n",
      "        [-1.0416e+00,  9.0512e+00],\n",
      "        [ 9.7633e-01,  8.6203e+00],\n",
      "        [ 2.8896e+00,  8.0191e+00],\n",
      "        [ 4.4538e+00,  7.2009e+00],\n",
      "        [ 5.5019e+00,  6.0624e+00],\n",
      "        [ 5.9579e+00,  4.5700e+00],\n",
      "        [ 5.8010e+00,  2.8219e+00],\n",
      "        [ 5.0331e+00,  1.0094e+00],\n",
      "        [ 3.6969e+00, -6.7362e-01],\n",
      "        [ 1.9358e+00, -2.1108e+00],\n",
      "        [ 1.7780e-02, -3.2907e+00],\n",
      "        [-3.1034e-02,  3.2222e+00],\n",
      "        [-1.8952e+00,  3.4174e+00],\n",
      "        [-3.6775e+00,  3.7500e+00],\n",
      "        [-5.0652e+00,  4.2530e+00],\n",
      "        [-5.8521e+00,  4.9212e+00],\n",
      "        [-5.9795e+00,  5.6750e+00],\n",
      "        [-5.4805e+00,  6.3532e+00],\n",
      "        [-4.4237e+00,  6.7708e+00],\n",
      "        [-2.8978e+00,  6.8224e+00],\n",
      "        [-1.0328e+00,  6.5467e+00],\n",
      "        [ 9.7650e-01,  6.0789e+00],\n",
      "        [ 2.8830e+00,  5.5238e+00],\n",
      "        [ 4.4472e+00,  4.8674e+00],\n",
      "        [ 5.5008e+00,  4.0112e+00],\n",
      "        [ 5.9628e+00,  2.8865e+00],\n",
      "        [ 5.8077e+00,  1.5379e+00],\n",
      "        [ 5.0363e+00,  1.1019e-01],\n",
      "        [ 3.6943e+00, -1.2309e+00],\n",
      "        [ 1.9318e+00, -2.3812e+00],\n",
      "        [ 2.1733e-02, -3.3315e+00],\n",
      "        [-3.1051e-02,  3.2569e+00],\n",
      "        [-1.8948e+00,  3.2030e+00],\n",
      "        [-3.6768e+00,  3.2190e+00],\n",
      "        [-5.0650e+00,  3.3452e+00],\n",
      "        [-5.8532e+00,  3.6028e+00],\n",
      "        [-5.9805e+00,  3.9581e+00],\n",
      "        [-5.4773e+00,  4.3009e+00],\n",
      "        [-4.4133e+00,  4.4775e+00],\n",
      "        [-2.8830e+00,  4.3808e+00],\n",
      "        [-1.0213e+00,  4.0270e+00],\n",
      "        [ 9.7850e-01,  3.5374e+00],\n",
      "        [ 2.8767e+00,  3.0315e+00],\n",
      "        [ 4.4397e+00,  2.5274e+00],\n",
      "        [ 5.4983e+00,  1.9432e+00],\n",
      "        [ 5.9664e+00,  1.1875e+00],\n",
      "        [ 5.8136e+00,  2.5218e-01],\n",
      "        [ 5.0393e+00, -7.7163e-01],\n",
      "        [ 3.6922e+00, -1.7539e+00],\n",
      "        [ 1.9290e+00, -2.6067e+00],\n",
      "        [ 2.7756e-02, -3.3227e+00],\n",
      "        [-3.4471e-02,  3.2999e+00],\n",
      "        [-1.8968e+00,  3.0025e+00],\n",
      "        [-3.6773e+00,  2.7081e+00],\n",
      "        [-5.0649e+00,  2.4613e+00],\n",
      "        [-5.8538e+00,  2.3057e+00],\n",
      "        [-5.9809e+00,  2.2508e+00],\n",
      "        [-5.4736e+00,  2.2398e+00],\n",
      "        [-4.4022e+00,  2.1603e+00],\n",
      "        [-2.8668e+00,  1.9130e+00],\n",
      "        [-1.0078e+00,  1.4930e+00],\n",
      "        [ 9.8222e-01,  9.9860e-01],\n",
      "        [ 2.8711e+00,  5.5023e-01],\n",
      "        [ 4.4317e+00,  1.9260e-01],\n",
      "        [ 5.4948e+00, -1.3224e-01],\n",
      "        [ 5.9689e+00, -5.2530e-01],\n",
      "        [ 5.8187e+00, -1.0415e+00],\n",
      "        [ 5.0419e+00, -1.6474e+00],\n",
      "        [ 3.6903e+00, -2.2562e+00],\n",
      "        [ 1.9271e+00, -2.8009e+00],\n",
      "        [ 3.5324e-02, -3.2769e+00],\n",
      "        [-4.0638e-02,  3.3384e+00],\n",
      "        [-1.9006e+00,  2.8023e+00],\n",
      "        [-3.6786e+00,  2.2041e+00],\n",
      "        [-5.0648e+00,  1.5909e+00],\n",
      "        [-5.8541e+00,  1.0250e+00],\n",
      "        [-5.9813e+00,  5.5487e-01],\n",
      "        [-5.4706e+00,  1.7751e-01],\n",
      "        [-4.3920e+00, -1.7234e-01],\n",
      "        [-2.8505e+00, -5.7577e-01],\n",
      "        [-9.9322e-01, -1.0538e+00],\n",
      "        [ 9.8739e-01, -1.5362e+00],\n",
      "        [ 2.8664e+00, -1.9143e+00],\n",
      "        [ 4.4237e+00, -2.1268e+00],\n",
      "        [ 5.4906e+00, -2.2055e+00],\n",
      "        [ 5.9706e+00, -2.2490e+00],\n",
      "        [ 5.8231e+00, -2.3486e+00],\n",
      "        [ 5.0439e+00, -2.5288e+00],\n",
      "        [ 3.6882e+00, -2.7523e+00],\n",
      "        [ 1.9257e+00, -2.9783e+00],\n",
      "        [ 4.3948e-02, -3.2078e+00],\n",
      "        [-4.8905e-02,  3.3599e+00],\n",
      "        [-1.9056e+00,  2.5888e+00],\n",
      "        [-3.6802e+00,  1.6936e+00],\n",
      "        [-5.0642e+00,  7.2288e-01],\n",
      "        [-5.8538e+00, -2.4574e-01],\n",
      "        [-5.9818e+00, -1.1290e+00],\n",
      "        [-5.4689e+00, -1.8793e+00],\n",
      "        [-4.3837e+00, -2.5110e+00],\n",
      "        [-2.8358e+00, -3.0784e+00],\n",
      "        [-9.7884e-01, -3.6106e+00],\n",
      "        [ 9.9346e-01, -4.0664e+00],\n",
      "        [ 2.8627e+00, -4.3588e+00],\n",
      "        [ 4.4161e+00, -4.4226e+00],\n",
      "        [ 5.4862e+00, -4.2671e+00],\n",
      "        [ 5.9717e+00, -3.9790e+00],\n",
      "        [ 5.8267e+00, -3.6733e+00],\n",
      "        [ 5.0453e+00, -3.4273e+00],\n",
      "        [ 3.6857e+00, -3.2571e+00],\n",
      "        [ 1.9244e+00, -3.1548e+00],\n",
      "        [ 5.3193e-02, -3.1301e+00],\n",
      "        [-5.8676e-02,  3.3522e+00],\n",
      "        [-1.9113e+00,  2.3489e+00],\n",
      "        [-3.6816e+00,  1.1636e+00],\n",
      "        [-5.0628e+00, -1.5409e-01],\n",
      "        [-5.8526e+00, -1.5136e+00],\n",
      "        [-5.9824e+00, -2.8017e+00],\n",
      "        [-5.4689e+00, -3.9247e+00],\n",
      "        [-4.3785e+00, -4.8462e+00],\n",
      "        [-2.8242e+00, -5.5864e+00],\n",
      "        [-9.6611e-01, -6.1731e+00],\n",
      "        [ 9.9955e-01, -6.5910e+00],\n",
      "        [ 2.8598e+00, -6.7817e+00],\n",
      "        [ 4.4093e+00, -6.6891e+00],\n",
      "        [ 5.4821e+00, -6.3084e+00],\n",
      "        [ 5.9728e+00, -5.7097e+00],\n",
      "        [ 5.8300e+00, -5.0179e+00],\n",
      "        [ 5.0461e+00, -4.3532e+00],\n",
      "        [ 3.6828e+00, -3.7857e+00],\n",
      "        [ 1.9229e+00, -3.3464e+00],\n",
      "        [ 6.2700e-02, -3.0589e+00],\n",
      "        [-6.9452e-02,  3.3036e+00],\n",
      "        [-1.9173e+00,  2.0703e+00],\n",
      "        [-3.6825e+00,  6.0186e-01],\n",
      "        [-5.0603e+00, -1.0508e+00],\n",
      "        [-5.8502e+00, -2.7858e+00],\n",
      "        [-5.9827e+00, -4.4650e+00],\n",
      "        [-5.4704e+00, -5.9541e+00],\n",
      "        [-4.3769e+00, -7.1684e+00],\n",
      "        [-2.8169e+00, -8.0897e+00],\n",
      "        [-9.5664e-01, -8.7349e+00],\n",
      "        [ 1.0045e+00, -9.1079e+00],\n",
      "        [ 2.8574e+00, -9.1822e+00],\n",
      "        [ 4.4035e+00, -8.9228e+00],\n",
      "        [ 5.4789e+00, -8.3225e+00],\n",
      "        [ 5.9742e+00, -7.4351e+00],\n",
      "        [ 5.8332e+00, -6.3826e+00],\n",
      "        [ 5.0465e+00, -5.3147e+00],\n",
      "        [ 3.6793e+00, -4.3522e+00],\n",
      "        [ 1.9211e+00, -3.5692e+00],\n",
      "        [ 7.2203e-02, -3.0099e+00],\n",
      "        [-8.0863e-02,  3.2033e+00],\n",
      "        [-1.9233e+00,  1.7418e+00],\n",
      "        [-3.6825e+00, -2.8978e-03],\n",
      "        [-5.0562e+00, -1.9773e+00],\n",
      "        [-5.8461e+00, -4.0697e+00],\n",
      "        [-5.9822e+00, -6.1213e+00],\n",
      "        [-5.4733e+00, -7.9640e+00],\n",
      "        [-4.3793e+00, -9.4693e+00],\n",
      "        [-2.8152e+00, -1.0578e+01],\n",
      "        [-9.5215e-01, -1.1288e+01],\n",
      "        [ 1.0067e+00, -1.1613e+01],\n",
      "        [ 2.8546e+00, -1.1560e+01],\n",
      "        [ 4.3986e+00, -1.1122e+01],\n",
      "        [ 5.4768e+00, -1.0305e+01],\n",
      "        [ 5.9765e+00, -9.1490e+00],\n",
      "        [ 5.8367e+00, -7.7655e+00],\n",
      "        [ 5.0468e+00, -6.3179e+00],\n",
      "        [ 3.6755e+00, -4.9688e+00],\n",
      "        [ 1.9190e+00, -3.8387e+00],\n",
      "        [ 8.1524e-02, -2.9985e+00],\n",
      "        [-9.2681e-02,  3.0416e+00],\n",
      "        [-1.9290e+00,  1.3531e+00],\n",
      "        [-3.6815e+00, -6.6049e-01],\n",
      "        [-5.0504e+00, -2.9423e+00],\n",
      "        [-5.8400e+00, -5.3716e+00],\n",
      "        [-5.9803e+00, -7.7734e+00],\n",
      "        [-5.4770e+00, -9.9521e+00],\n",
      "        [-4.3856e+00, -1.1741e+01],\n",
      "        [-2.8199e+00, -1.3041e+01],\n",
      "        [-9.5426e-01, -1.3822e+01],\n",
      "        [ 1.0044e+00, -1.4102e+01],\n",
      "        [ 2.8502e+00, -1.3913e+01],\n",
      "        [ 4.3942e+00, -1.3288e+01],\n",
      "        [ 5.4761e+00, -1.2252e+01],\n",
      "        [ 5.9801e+00, -1.0846e+01],\n",
      "        [ 5.8411e+00, -9.1634e+00],\n",
      "        [ 5.0472e+00, -7.3657e+00],\n",
      "        [ 3.6714e+00, -5.6457e+00],\n",
      "        [ 1.9166e+00, -4.1690e+00],\n",
      "        [ 9.0597e-02, -3.0397e+00],\n",
      "        [-1.0482e-01,  2.8098e+00],\n",
      "        [-1.9344e+00,  8.9564e-01],\n",
      "        [-3.6795e+00, -1.3792e+00],\n",
      "        [-5.0428e+00, -3.9529e+00],\n",
      "        [-5.8317e+00, -6.6971e+00],\n",
      "        [-5.9765e+00, -9.4238e+00],\n",
      "        [-5.4807e+00, -1.1917e+01],\n",
      "        [-4.3953e+00, -1.3977e+01],\n",
      "        [-2.8314e+00, -1.5467e+01],\n",
      "        [-9.6437e-01, -1.6326e+01],\n",
      "        [ 9.9581e-01, -1.6565e+01],\n",
      "        [ 2.8426e+00, -1.6239e+01],\n",
      "        [ 4.3895e+00, -1.5419e+01],\n",
      "        [ 5.4766e+00, -1.4164e+01],\n",
      "        [ 5.9853e+00, -1.2523e+01],\n",
      "        [ 5.8466e+00, -1.0572e+01],\n",
      "        [ 5.0482e+00, -8.4581e+00],\n",
      "        [ 3.6674e+00, -6.3898e+00],\n",
      "        [ 1.9140e+00, -4.5722e+00],\n",
      "        [ 9.9463e-02, -3.1475e+00],\n",
      "        [-1.1732e-01,  2.5004e+00],\n",
      "        [-1.9396e+00,  3.6214e-01],\n",
      "        [-3.6765e+00, -2.1654e+00],\n",
      "        [-5.0334e+00, -5.0148e+00],\n",
      "        [-5.8208e+00, -8.0503e+00],\n",
      "        [-5.9704e+00, -1.1075e+01],\n",
      "        [-5.4836e+00, -1.3857e+01],\n",
      "        [-4.4076e+00, -1.6173e+01],\n",
      "        [-2.8495e+00, -1.7848e+01],\n",
      "        [-9.8354e-01, -1.8789e+01],\n",
      "        [ 9.7890e-01, -1.8994e+01],\n",
      "        [ 2.8299e+00, -1.8533e+01],\n",
      "        [ 4.3831e+00, -1.7516e+01],\n",
      "        [ 5.4779e+00, -1.6042e+01],\n",
      "        [ 5.9923e+00, -1.4177e+01],\n",
      "        [ 5.8538e+00, -1.1985e+01],\n",
      "        [ 5.0502e+00, -9.5925e+00],\n",
      "        [ 3.6639e+00, -7.2047e+00],\n",
      "        [ 1.9116e+00, -5.0579e+00],\n",
      "        [ 1.0827e-01, -3.3345e+00],\n",
      "        [-1.3033e-01,  2.1072e+00],\n",
      "        [-1.9449e+00, -2.5294e-01],\n",
      "        [-3.6728e+00, -3.0240e+00],\n",
      "        [-5.0224e+00, -6.1317e+00],\n",
      "        [-5.8075e+00, -9.4341e+00],\n",
      "        [-5.9616e+00, -1.2728e+01],\n",
      "        [-5.4851e+00, -1.5772e+01],\n",
      "        [-4.4216e+00, -1.8323e+01],\n",
      "        [-2.8738e+00, -2.0174e+01],\n",
      "        [-1.0123e+00, -2.1199e+01],\n",
      "        [ 9.5199e-01, -2.1377e+01],\n",
      "        [ 2.8100e+00, -2.0790e+01],\n",
      "        [ 4.3734e+00, -1.9579e+01],\n",
      "        [ 5.4791e+00, -1.7887e+01],\n",
      "        [ 6.0009e+00, -1.5807e+01],\n",
      "        [ 5.8631e+00, -1.3400e+01],\n",
      "        [ 5.0539e+00, -1.0764e+01],\n",
      "        [ 3.6614e+00, -8.0904e+00],\n",
      "        [ 1.9098e+00, -5.6325e+00],\n",
      "        [ 1.1729e-01, -3.6114e+00],\n",
      "        [-1.4408e-01,  1.6256e+00],\n",
      "        [-1.9504e+00, -9.5351e-01],\n",
      "        [-3.6687e+00, -3.9578e+00],\n",
      "        [-5.0099e+00, -7.3057e+00],\n",
      "        [-5.7917e+00, -1.0850e+01],\n",
      "        [-5.9497e+00, -1.4384e+01],\n",
      "        [-5.4842e+00, -1.7662e+01],\n",
      "        [-4.4363e+00, -2.0424e+01],\n",
      "        [-2.9034e+00, -2.2437e+01],\n",
      "        [-1.0509e+00, -2.3545e+01],\n",
      "        [ 9.1371e-01, -2.3705e+01],\n",
      "        [ 2.7806e+00, -2.3001e+01],\n",
      "        [ 4.3583e+00, -2.1605e+01],\n",
      "        [ 5.4790e+00, -1.9700e+01],\n",
      "        [ 6.0108e+00, -1.7414e+01],\n",
      "        [ 5.8746e+00, -1.4812e+01],\n",
      "        [ 5.0598e+00, -1.1967e+01],\n",
      "        [ 3.6605e+00, -9.0436e+00],\n",
      "        [ 1.9091e+00, -6.2991e+00],\n",
      "        [ 1.2688e-01, -3.9863e+00]])\n"
     ]
    }
   ],
   "source": [
    "print(x)\n",
    "print(f_value(x))\n",
    "print(predict_fn(basic, x.detach().numpy()))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Train and optimization process of LNN"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Previous weight for \\dot V: 0.001, now : 0.2 to ensure stability"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "start = timeit.default_timer()\n",
    "L_W = 0.2\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",
    "    #dx = f_value(x) #accurate first-principles model\n",
    "    dx = predict_fn(basic, x.detach().numpy()) #use NN to predict dx\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",
    "        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 + L_W *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+3.5*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] > - L_W *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",
    "                    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": 12,
   "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: 400\n",
      "0 Lyapunov Risk= 3.4601374210296854e-08\n",
      "Number of negative values: 338\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: 283\n",
      "Number of negative values: 341\n",
      "Number of negative values: 338\n",
      "Number of negative values: 338\n",
      "5 Lyapunov Risk= 0.00070539879379794\n",
      "Number of negative values: 338\n",
      "Number of negative values: 338\n",
      "Number of negative values: 338\n",
      "Number of negative values: 338\n",
      "0.0019960425587510337 0.06598978939358487 0.04939549651064031\n",
      "0 0 0\n",
      "| \u001b[0m1        \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.001996 \u001b[0m | \u001b[0m0.06599  \u001b[0m | \u001b[0m0.0494   \u001b[0m |\n",
      "Number of negative values: 240\n",
      "0 Lyapunov Risk= 0.01608171872794628\n",
      "Number of negative values: 239\n",
      "Number of negative values: 239\n",
      "Number of negative values: 239\n",
      "Number of negative values: 239\n",
      "Number of negative values: 239\n",
      "5 Lyapunov Risk= 0.013757128268480301\n",
      "Number of negative values: 239\n",
      "Number of negative values: 239\n",
      "Number of negative values: 239\n",
      "Number of negative values: 239\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: 272\n",
      "0 Lyapunov Risk= 0.02423989027738571\n",
      "Number of negative values: 292\n",
      "Number of negative values: 302\n",
      "Number of negative values: 286\n",
      "Number of negative values: 289\n",
      "Number of negative values: 309\n",
      "5 Lyapunov Risk= 0.008836857974529266\n",
      "Number of negative values: 353\n",
      "Number of negative values: 291\n",
      "Number of negative values: 374\n",
      "Number of negative values: 327\n",
      "0.0028369961259166574 0.08216849597815173 0.09623254183153347\n",
      "0 0 0\n",
      "| \u001b[0m3        \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.002837 \u001b[0m | \u001b[0m0.08217  \u001b[0m | \u001b[0m0.09623  \u001b[0m |\n",
      "Number of negative values: 282\n",
      "0 Lyapunov Risk= 0.005450710654258728\n",
      "Number of negative values: 237\n",
      "Number of negative values: 316\n",
      "Number of negative values: 272\n",
      "Number of negative values: 268\n",
      "Number of negative values: 267\n",
      "5 Lyapunov Risk= 0.011097891256213188\n",
      "Number of negative values: 267\n",
      "Number of negative values: 267\n",
      "Number of negative values: 267\n",
      "Number of negative values: 267\n",
      "0.008771733083946737 0.04220355429620801 0.05508956129711129\n",
      "0 0 0\n",
      "| \u001b[0m4        \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.008772 \u001b[0m | \u001b[0m0.0422   \u001b[0m | \u001b[0m0.05509  \u001b[0m |\n",
      "Number of negative values: 265\n",
      "0 Lyapunov Risk= 0.006473808083683252\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\n",
      "5 Lyapunov Risk= 0.004417670425027609\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\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: 224\n",
      "0 Lyapunov Risk= 0.002772039733827114\n",
      "Number of negative values: 282\n",
      "Number of negative values: 282\n",
      "Number of negative values: 282\n",
      "Number of negative values: 282\n",
      "Number of negative values: 282\n",
      "5 Lyapunov Risk= 0.00027191429398953915\n",
      "Number of negative values: 282\n",
      "Number of negative values: 282\n",
      "Number of negative values: 282\n",
      "Number of negative values: 282\n",
      "0.0034827093848423954 0.010144987744947 0.010400377967918246\n",
      "0 0 0\n",
      "| \u001b[0m6        \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.003483 \u001b[0m | \u001b[0m0.01014  \u001b[0m | \u001b[0m0.0104   \u001b[0m |\n",
      "Number of negative values: 254\n",
      "0 Lyapunov Risk= 0.05034428462386131\n",
      "Number of negative values: 382\n",
      "Number of negative values: 230\n",
      "Number of negative values: 290\n",
      "Number of negative values: 275\n",
      "Number of negative values: 261\n",
      "5 Lyapunov Risk= 0.04668359458446503\n",
      "Number of negative values: 256\n",
      "Number of negative values: 256\n",
      "Number of negative values: 256\n",
      "Number of negative values: 256\n",
      "0.007979896539540803 0.010517429483049304 0.09986724963800271\n",
      "0 0 0\n",
      "| \u001b[0m7        \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.00798  \u001b[0m | \u001b[0m0.01052  \u001b[0m | \u001b[0m0.09987  \u001b[0m |\n",
      "Number of negative values: 219\n",
      "0 Lyapunov Risk= 0.04052175208926201\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "5 Lyapunov Risk= 0.03809802234172821\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "0.006786755558005269 0.09940897266543024 0.010251926456269676\n",
      "0 0 0\n",
      "| \u001b[0m8        \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.006787 \u001b[0m | \u001b[0m0.09941  \u001b[0m | \u001b[0m0.01025  \u001b[0m |\n",
      "Number of negative values: 254\n",
      "0 Lyapunov Risk= 0.05001682788133621\n",
      "Number of negative values: 379\n",
      "Number of negative values: 237\n",
      "Number of negative values: 345\n",
      "Number of negative values: 270\n",
      "Number of negative values: 262\n",
      "5 Lyapunov Risk= 0.045121144503355026\n",
      "Number of negative values: 258\n",
      "Number of negative values: 257\n",
      "Number of negative values: 256\n",
      "Number of negative values: 256\n",
      "0.008762124711222325 0.011139856062642312 0.09967151367963017\n",
      "0 0 0\n",
      "| \u001b[0m9        \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.008762 \u001b[0m | \u001b[0m0.01114  \u001b[0m | \u001b[0m0.09967  \u001b[0m |\n",
      "Number of negative values: 220\n",
      "0 Lyapunov Risk= 0.04033037647604942\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "5 Lyapunov Risk= 0.037945766001939774\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "0.009436224926558708 0.0993089807855446 0.010573033087018503\n",
      "0 0 0\n",
      "| \u001b[0m10       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.009436 \u001b[0m | \u001b[0m0.09931  \u001b[0m | \u001b[0m0.01057  \u001b[0m |\n",
      "Number of negative values: 255\n",
      "0 Lyapunov Risk= 0.04993504658341408\n",
      "Number of negative values: 386\n",
      "Number of negative values: 227\n",
      "Number of negative values: 266\n",
      "Number of negative values: 282\n",
      "Number of negative values: 264\n",
      "5 Lyapunov Risk= 0.045137375593185425\n",
      "Number of negative values: 257\n",
      "Number of negative values: 256\n",
      "Number of negative values: 256\n",
      "Number of negative values: 256\n",
      "0.0013180424970097235 0.011315616667882281 0.09913156010867247\n",
      "0 0 0\n",
      "| \u001b[0m11       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.001318 \u001b[0m | \u001b[0m0.01132  \u001b[0m | \u001b[0m0.09913  \u001b[0m |\n",
      "Number of negative values: 219\n",
      "0 Lyapunov Risk= 0.04078194499015808\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "5 Lyapunov Risk= 0.03834252059459686\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "0.005606128711299322 0.09987029623614033 0.010783227899360027\n",
      "0 0 0\n",
      "| \u001b[0m12       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.005606 \u001b[0m | \u001b[0m0.09987  \u001b[0m | \u001b[0m0.01078  \u001b[0m |\n",
      "Number of negative values: 255\n",
      "0 Lyapunov Risk= 0.050054579973220825\n",
      "Number of negative values: 389\n",
      "Number of negative values: 227\n",
      "Number of negative values: 266\n",
      "Number of negative values: 281\n",
      "Number of negative values: 263\n",
      "5 Lyapunov Risk= 0.04577542468905449\n",
      "Number of negative values: 257\n",
      "Number of negative values: 256\n",
      "Number of negative values: 256\n",
      "Number of negative values: 256\n",
      "0.0024492937190581264 0.010092288520294718 0.09922875712802567\n",
      "0 0 0\n",
      "| \u001b[0m13       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.002449 \u001b[0m | \u001b[0m0.01009  \u001b[0m | \u001b[0m0.09923  \u001b[0m |\n",
      "Number of negative values: 219\n",
      "0 Lyapunov Risk= 0.04067925363779068\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "5 Lyapunov Risk= 0.03827682137489319\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "0.008281779376648413 0.09969985655240184 0.010478055225853175\n",
      "0 0 0\n",
      "| \u001b[0m14       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.008282 \u001b[0m | \u001b[0m0.0997   \u001b[0m | \u001b[0m0.01048  \u001b[0m |\n",
      "Number of negative values: 255\n",
      "0 Lyapunov Risk= 0.05064636841416359\n",
      "Number of negative values: 385\n",
      "Number of negative values: 227\n",
      "Number of negative values: 266\n",
      "Number of negative values: 282\n",
      "Number of negative values: 265\n",
      "5 Lyapunov Risk= 0.04525007680058479\n",
      "Number of negative values: 257\n",
      "Number of negative values: 256\n",
      "Number of negative values: 256\n",
      "Number of negative values: 256\n",
      "0.0006049867001657468 0.010257288896854424 0.09964507934400871\n",
      "0 0 0\n",
      "| \u001b[0m15       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.000605 \u001b[0m | \u001b[0m0.01026  \u001b[0m | \u001b[0m0.09965  \u001b[0m |\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 284\n",
      "0 Lyapunov Risk= 0.017493994906544685\n",
      "Number of negative values: 283\n",
      "Number of negative values: 345\n",
      "Number of negative values: 299\n",
      "Number of negative values: 300\n",
      "Number of negative values: 326\n",
      "5 Lyapunov Risk= 0.005565973464399576\n",
      "Number of negative values: 382\n",
      "Number of negative values: 301\n",
      "Number of negative values: 353\n",
      "Number of negative values: 374\n",
      "0.009503294942758395 0.09978543856529694 0.09987018776501727\n",
      "62 9 tensor(0.0190, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m16       \u001b[0m | \u001b[95m62.0     \u001b[0m | \u001b[95m0.009503 \u001b[0m | \u001b[95m0.09979  \u001b[0m | \u001b[95m0.09987  \u001b[0m |\n",
      "Number of negative values: 305\n",
      "0 Lyapunov Risk= 0.007893087342381477\n",
      "Number of negative values: 262\n",
      "Number of negative values: 400\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.012924608774483204\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "0.009653224693082321 0.09983013396043913 0.09037323371100936\n",
      "218 2 tensor(0.0555, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m17       \u001b[0m | \u001b[95m218.0    \u001b[0m | \u001b[95m0.009653 \u001b[0m | \u001b[95m0.09983  \u001b[0m | \u001b[95m0.09037  \u001b[0m |\n",
      "Number of negative values: 269\n",
      "0 Lyapunov Risk= 0.008020247332751751\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\n",
      "5 Lyapunov Risk= 0.006111751310527325\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\n",
      "Number of negative values: 283\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.012152480892837048\n",
      "Number of negative values: 259\n",
      "Number of negative values: 386\n",
      "Number of negative values: 280\n",
      "Number of negative values: 272\n",
      "Number of negative values: 274\n",
      "5 Lyapunov Risk= 0.010745026171207428\n",
      "Number of negative values: 325\n",
      "Number of negative values: 267\n",
      "Number of negative values: 304\n",
      "Number of negative values: 323\n",
      "0.007989413057315715 0.03789227346663301 0.06360490865766633\n",
      "4 2 tensor(0.0009, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m19       \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.007989 \u001b[0m | \u001b[0m0.03789  \u001b[0m | \u001b[0m0.0636   \u001b[0m |\n",
      "Number of negative values: 332\n",
      "0 Lyapunov Risk= 0.0024283931124955416\n",
      "Number of negative values: 254\n",
      "Number of negative values: 379\n",
      "Number of negative values: 310\n",
      "Number of negative values: 300\n",
      "Number of negative values: 300\n",
      "5 Lyapunov Risk= 0.007679000496864319\n",
      "Number of negative values: 300\n",
      "Number of negative values: 300\n",
      "Number of negative values: 300\n",
      "Number of negative values: 300\n",
      "0.01 0.1 0.08386617654406507\n",
      "0 0 0\n",
      "| \u001b[0m20       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.08387  \u001b[0m |\n",
      "Number of negative values: 300\n",
      "0 Lyapunov Risk= 0.010016088373959064\n",
      "Number of negative values: 264\n",
      "Number of negative values: 391\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.01322544738650322\n",
      "Number of negative values: 291\n",
      "Number of negative values: 292\n",
      "Number of negative values: 300\n",
      "Number of negative values: 313\n",
      "0.008285818209866581 0.09961058551714637 0.0924181614355622\n",
      "90 2 tensor(0.0237, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m21       \u001b[0m | \u001b[0m90.0     \u001b[0m | \u001b[0m0.008286 \u001b[0m | \u001b[0m0.09961  \u001b[0m | \u001b[0m0.09242  \u001b[0m |\n",
      "Number of negative values: 245\n",
      "0 Lyapunov Risk= 0.0024786314461380243\n",
      "Number of negative values: 304\n",
      "Number of negative values: 304\n",
      "Number of negative values: 304\n",
      "Number of negative values: 304\n",
      "Number of negative values: 304\n",
      "5 Lyapunov Risk= 0.00020876186317764223\n",
      "Number of negative values: 304\n",
      "Number of negative values: 304\n",
      "Number of negative values: 304\n",
      "Number of negative values: 304\n",
      "0.002613196453286354 0.017690482369735437 0.014582305639612611\n",
      "0 0 0\n",
      "| \u001b[0m22       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.002613 \u001b[0m | \u001b[0m0.01769  \u001b[0m | \u001b[0m0.01458  \u001b[0m |\n",
      "Number of negative values: 306\n",
      "0 Lyapunov Risk= 0.007573953829705715\n",
      "Number of negative values: 261\n",
      "Number of negative values: 400\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.012677903287112713\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "0.01 0.09890483107808543 0.0894758674736823\n",
      "244 2 tensor(0.0629, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m23       \u001b[0m | \u001b[95m244.0    \u001b[0m | \u001b[95m0.01     \u001b[0m | \u001b[95m0.0989   \u001b[0m | \u001b[95m0.08948  \u001b[0m |\n",
      "Number of negative values: 306\n",
      "0 Lyapunov Risk= 0.0072581591084599495\n",
      "Number of negative values: 259\n",
      "Number of negative values: 400\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.012349743396043777\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "0.008882323857862534 0.0981417701478719 0.08856946191394448\n",
      "4 2 tensor(0.0018, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m24       \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.008882 \u001b[0m | \u001b[0m0.09814  \u001b[0m | \u001b[0m0.08857  \u001b[0m |\n",
      "Number of negative values: 303\n",
      "0 Lyapunov Risk= 0.008476227521896362\n",
      "Number of negative values: 262\n",
      "Number of negative values: 400\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.01310955174267292\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "0.01 0.09879535064142213 0.09040724836000921\n",
      "208 2 tensor(0.0526, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m25       \u001b[0m | \u001b[0m208.0    \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.0988   \u001b[0m | \u001b[0m0.09041  \u001b[0m |\n",
      "Number of negative values: 301\n",
      "0 Lyapunov Risk= 0.009198123589158058\n",
      "Number of negative values: 263\n",
      "Number of negative values: 399\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.013324720785021782\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 291\n",
      "Number of negative values: 295\n",
      "0.009604327852045855 0.09986319262961597 0.09180305221417266\n",
      "122 2 tensor(0.0322, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m26       \u001b[0m | \u001b[0m122.0    \u001b[0m | \u001b[0m0.009604 \u001b[0m | \u001b[0m0.09986  \u001b[0m | \u001b[0m0.0918   \u001b[0m |\n",
      "Number of negative values: 229\n",
      "0 Lyapunov Risk= 0.003008416621014476\n",
      "Number of negative values: 390\n",
      "Number of negative values: 378\n",
      "Number of negative values: 377\n",
      "Number of negative values: 377\n",
      "Number of negative values: 377\n",
      "5 Lyapunov Risk= 1.4755050870007835e-05\n",
      "Number of negative values: 377\n",
      "Number of negative values: 377\n",
      "Number of negative values: 377\n",
      "Number of negative values: 377\n",
      "0.0034729635185387297 0.016752824699881295 0.011992711001254612\n",
      "4 1 tensor(3.1773e-05, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m27       \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.003473 \u001b[0m | \u001b[0m0.01675  \u001b[0m | \u001b[0m0.01199  \u001b[0m |\n",
      "Number of negative values: 307\n",
      "0 Lyapunov Risk= 0.0067543284967541695\n",
      "Number of negative values: 260\n",
      "Number of negative values: 400\n",
      "Number of negative values: 300\n",
      "Number of negative values: 293\n",
      "Number of negative values: 291\n",
      "5 Lyapunov Risk= 0.011932547204196453\n",
      "Number of negative values: 291\n",
      "Number of negative values: 291\n",
      "Number of negative values: 291\n",
      "Number of negative values: 291\n",
      "0.01 0.1 0.08922519388592472\n",
      "4 2 tensor(0.0018, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m28       \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.08923  \u001b[0m |\n",
      "Number of negative values: 301\n",
      "0 Lyapunov Risk= 0.009218498133122921\n",
      "Number of negative values: 262\n",
      "Number of negative values: 397\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.01334611140191555\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 291\n",
      "Number of negative values: 293\n",
      "0.009042462104329702 0.09855471227662799 0.09099192322891382\n",
      "122 2 tensor(0.0316, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m29       \u001b[0m | \u001b[0m122.0    \u001b[0m | \u001b[0m0.009042 \u001b[0m | \u001b[0m0.09855  \u001b[0m | \u001b[0m0.09099  \u001b[0m |\n",
      "Number of negative values: 301\n",
      "0 Lyapunov Risk= 0.009511577896773815\n",
      "Number of negative values: 263\n",
      "Number of negative values: 396\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.013473897241055965\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 291\n",
      "Number of negative values: 298\n",
      "0.009661867272102843 0.09857281334199333 0.09136131667864777\n",
      "122 2 tensor(0.0318, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m30       \u001b[0m | \u001b[0m122.0    \u001b[0m | \u001b[0m0.009662 \u001b[0m | \u001b[0m0.09857  \u001b[0m | \u001b[0m0.09136  \u001b[0m |\n",
      "Number of negative values: 289\n",
      "0 Lyapunov Risk= 0.0036619838792830706\n",
      "Number of negative values: 231\n",
      "Number of negative values: 288\n",
      "Number of negative values: 272\n",
      "Number of negative values: 269\n",
      "Number of negative values: 268\n",
      "5 Lyapunov Risk= 0.009200194850564003\n",
      "Number of negative values: 268\n",
      "Number of negative values: 268\n",
      "Number of negative values: 268\n",
      "Number of negative values: 268\n",
      "0.0074648773139264725 0.040196663038117274 0.05086642157959664\n",
      "0 0 0\n",
      "| \u001b[0m31       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.007465 \u001b[0m | \u001b[0m0.0402   \u001b[0m | \u001b[0m0.05087  \u001b[0m |\n",
      "Number of negative values: 298\n",
      "0 Lyapunov Risk= 0.010530881583690643\n",
      "Number of negative values: 265\n",
      "Number of negative values: 390\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.01340172253549099\n",
      "Number of negative values: 291\n",
      "Number of negative values: 294\n",
      "Number of negative values: 302\n",
      "Number of negative values: 357\n",
      "0.009667526433248086 0.09859466392200207 0.09244545612235923\n",
      "90 2 tensor(0.0236, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m32       \u001b[0m | \u001b[0m90.0     \u001b[0m | \u001b[0m0.009668 \u001b[0m | \u001b[0m0.09859  \u001b[0m | \u001b[0m0.09245  \u001b[0m |\n",
      "Number of negative values: 304\n",
      "0 Lyapunov Risk= 0.008118296042084694\n",
      "Number of negative values: 262\n",
      "Number of negative values: 400\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.01312772836536169\n",
      "Number of negative values: 289\n",
      "Number of negative values: 289\n",
      "Number of negative values: 289\n",
      "Number of negative values: 289\n",
      "0.009841301273594323 0.09803915324249028 0.08954302441309904\n",
      "218 2 tensor(0.0545, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m33       \u001b[0m | \u001b[0m218.0    \u001b[0m | \u001b[0m0.009841 \u001b[0m | \u001b[0m0.09804  \u001b[0m | \u001b[0m0.08954  \u001b[0m |\n",
      "Number of negative values: 283\n",
      "0 Lyapunov Risk= 0.017690394073724747\n",
      "Number of negative values: 283\n",
      "Number of negative values: 344\n",
      "Number of negative values: 299\n",
      "Number of negative values: 300\n",
      "Number of negative values: 328\n",
      "5 Lyapunov Risk= 0.005419944413006306\n",
      "Number of negative values: 378\n",
      "Number of negative values: 301\n",
      "Number of negative values: 354\n",
      "Number of negative values: 374\n",
      "0.00957715680291234 0.09932256183573722 0.09979515566722251\n",
      "62 9 tensor(0.0189, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m34       \u001b[0m | \u001b[0m62.0     \u001b[0m | \u001b[0m0.009577 \u001b[0m | \u001b[0m0.09932  \u001b[0m | \u001b[0m0.0998   \u001b[0m |\n",
      "Number of negative values: 301\n",
      "0 Lyapunov Risk= 0.009155355393886566\n",
      "Number of negative values: 261\n",
      "Number of negative values: 395\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.012980777770280838\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 291\n",
      "Number of negative values: 299\n",
      "0.007048172696746997 0.09860068662783958 0.0908076702371191\n",
      "84 2 tensor(0.0217, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m35       \u001b[0m | \u001b[0m84.0     \u001b[0m | \u001b[0m0.007048 \u001b[0m | \u001b[0m0.0986   \u001b[0m | \u001b[0m0.09081  \u001b[0m |\n",
      "Number of negative values: 302\n",
      "0 Lyapunov Risk= 0.008529909886419773\n",
      "Number of negative values: 260\n",
      "Number of negative values: 398\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.01301438920199871\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 291\n",
      "0.007795539833886065 0.09936882392801395 0.09065653196549121\n",
      "122 2 tensor(0.0316, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m36       \u001b[0m | \u001b[0m122.0    \u001b[0m | \u001b[0m0.007796 \u001b[0m | \u001b[0m0.09937  \u001b[0m | \u001b[0m0.09066  \u001b[0m |\n",
      "Number of negative values: 304\n",
      "0 Lyapunov Risk= 0.008343838155269623\n",
      "Number of negative values: 262\n",
      "Number of negative values: 400\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.013065513223409653\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "0.01 0.09957802242921768 0.09074097580774204\n",
      "208 2 tensor(0.0530, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m37       \u001b[0m | \u001b[0m208.0    \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.09958  \u001b[0m | \u001b[0m0.09074  \u001b[0m |\n",
      "Number of negative values: 305\n",
      "0 Lyapunov Risk= 0.0077164750546216965\n",
      "Number of negative values: 262\n",
      "Number of negative values: 400\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.01274515874683857\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "0.009345939153887884 0.09954932988108428 0.0899823861373177\n",
      "218 2 tensor(0.0552, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m38       \u001b[0m | \u001b[0m218.0    \u001b[0m | \u001b[0m0.009346 \u001b[0m | \u001b[0m0.09955  \u001b[0m | \u001b[0m0.08998  \u001b[0m |\n",
      "Number of negative values: 222\n",
      "0 Lyapunov Risk= 0.03246121481060982\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",
      "Number of negative values: 224\n",
      "5 Lyapunov Risk= 0.029988456517457962\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: 284\n",
      "0 Lyapunov Risk= 0.007033824920654297\n",
      "Number of negative values: 247\n",
      "Number of negative values: 358\n",
      "Number of negative values: 279\n",
      "Number of negative values: 271\n",
      "Number of negative values: 269\n",
      "5 Lyapunov Risk= 0.012306437827646732\n",
      "Number of negative values: 269\n",
      "Number of negative values: 269\n",
      "Number of negative values: 269\n",
      "Number of negative values: 269\n",
      "0.0025268210137301557 0.048366366132118616 0.059897571883420826\n",
      "0 0 0\n",
      "| \u001b[0m40       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.002527 \u001b[0m | \u001b[0m0.04837  \u001b[0m | \u001b[0m0.0599   \u001b[0m |\n",
      "Number of negative values: 301\n",
      "0 Lyapunov Risk= 0.00897559430450201\n",
      "Number of negative values: 262\n",
      "Number of negative values: 398\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.013307136483490467\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 291\n",
      "0.008927528174256625 0.09926991525390079 0.09115668467631265\n",
      "122 2 tensor(0.0318, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m41       \u001b[0m | \u001b[0m122.0    \u001b[0m | \u001b[0m0.008928 \u001b[0m | \u001b[0m0.09927  \u001b[0m | \u001b[0m0.09116  \u001b[0m |\n",
      "Number of negative values: 305\n",
      "0 Lyapunov Risk= 0.0075933849439024925\n",
      "Number of negative values: 261\n",
      "Number of negative values: 400\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.012651546858251095\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "0.009461518336302856 0.09917194116972225 0.08962202731861603\n",
      "4 2 tensor(0.0018, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m42       \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.009462 \u001b[0m | \u001b[0m0.09917  \u001b[0m | \u001b[0m0.08962  \u001b[0m |\n",
      "Number of negative values: 301\n",
      "0 Lyapunov Risk= 0.009565978311002254\n",
      "Number of negative values: 264\n",
      "Number of negative values: 396\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.013352887704968452\n",
      "Number of negative values: 290\n",
      "Number of negative values: 291\n",
      "Number of negative values: 293\n",
      "Number of negative values: 300\n",
      "0.009162387303867562 0.09972318153319099 0.09207682487975408\n",
      "122 2 tensor(0.0323, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m43       \u001b[0m | \u001b[0m122.0    \u001b[0m | \u001b[0m0.009162 \u001b[0m | \u001b[0m0.09972  \u001b[0m | \u001b[0m0.09208  \u001b[0m |\n",
      "Number of negative values: 304\n",
      "0 Lyapunov Risk= 0.008235421031713486\n",
      "Number of negative values: 260\n",
      "Number of negative values: 400\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.012814926914870739\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "0.008028026235981188 0.09889897003068872 0.09006507096910579\n",
      "16 2 tensor(0.0046, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m44       \u001b[0m | \u001b[0m16.0     \u001b[0m | \u001b[0m0.008028 \u001b[0m | \u001b[0m0.0989   \u001b[0m | \u001b[0m0.09007  \u001b[0m |\n",
      "Number of negative values: 263\n",
      "0 Lyapunov Risk= 0.008564367890357971\n",
      "Number of negative values: 269\n",
      "Number of negative values: 269\n",
      "Number of negative values: 269\n",
      "Number of negative values: 269\n",
      "Number of negative values: 269\n",
      "5 Lyapunov Risk= 0.006534228101372719\n",
      "Number of negative values: 269\n",
      "Number of negative values: 269\n",
      "Number of negative values: 269\n",
      "Number of negative values: 269\n",
      "0.007348275454339711 0.08125182270746652 0.045812820879455124\n",
      "0 0 0\n",
      "| \u001b[0m45       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.007348 \u001b[0m | \u001b[0m0.08125  \u001b[0m | \u001b[0m0.04581  \u001b[0m |\n",
      "Number of negative values: 228\n",
      "0 Lyapunov Risk= 0.0103532075881958\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "5 Lyapunov Risk= 0.0076818629167973995\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "0.00033175774362835894 0.0517045490601026 0.017022450865244627\n",
      "0 0 0\n",
      "| \u001b[0m46       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.0003318\u001b[0m | \u001b[0m0.0517   \u001b[0m | \u001b[0m0.01702  \u001b[0m |\n",
      "Number of negative values: 300\n",
      "0 Lyapunov Risk= 0.009574931114912033\n",
      "Number of negative values: 262\n",
      "Number of negative values: 392\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.013105925172567368\n",
      "Number of negative values: 290\n",
      "Number of negative values: 291\n",
      "Number of negative values: 295\n",
      "Number of negative values: 301\n",
      "0.007746065472005494 0.09885900129682337 0.09146217655555666\n",
      "90 2 tensor(0.0233, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m47       \u001b[0m | \u001b[0m90.0     \u001b[0m | \u001b[0m0.007746 \u001b[0m | \u001b[0m0.09886  \u001b[0m | \u001b[0m0.09146  \u001b[0m |\n",
      "Number of negative values: 228\n",
      "0 Lyapunov Risk= 0.027215609326958656\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "5 Lyapunov Risk= 0.024832898750901222\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "Number of negative values: 230\n",
      "0.0042673987390171265 0.09604319794278653 0.03425198640355429\n",
      "0 0 0\n",
      "| \u001b[0m48       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.004267 \u001b[0m | \u001b[0m0.09604  \u001b[0m | \u001b[0m0.03425  \u001b[0m |\n",
      "Number of negative values: 300\n",
      "0 Lyapunov Risk= 0.009577377699315548\n",
      "Number of negative values: 263\n",
      "Number of negative values: 396\n",
      "Number of negative values: 300\n",
      "Number of negative values: 290\n",
      "Number of negative values: 289\n",
      "5 Lyapunov Risk= 0.013503555208444595\n",
      "Number of negative values: 288\n",
      "Number of negative values: 290\n",
      "Number of negative values: 291\n",
      "Number of negative values: 297\n",
      "0.009494816213182423 0.0964865235799032 0.09016506658109563\n",
      "122 2 tensor(0.0310, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m49       \u001b[0m | \u001b[0m122.0    \u001b[0m | \u001b[0m0.009495 \u001b[0m | \u001b[0m0.09649  \u001b[0m | \u001b[0m0.09017  \u001b[0m |\n",
      "Number of negative values: 259\n",
      "0 Lyapunov Risk= 0.04102921858429909\n",
      "Number of negative values: 363\n",
      "Number of negative values: 337\n",
      "Number of negative values: 284\n",
      "Number of negative values: 352\n",
      "Number of negative values: 250\n",
      "5 Lyapunov Risk= 0.04612984135746956\n",
      "Number of negative values: 345\n",
      "Number of negative values: 282\n",
      "Number of negative values: 268\n",
      "Number of negative values: 266\n",
      "0.003043733699752048 0.03407612648075811 0.0944051140480775\n",
      "16 4 tensor(0.0050, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m50       \u001b[0m | \u001b[0m16.0     \u001b[0m | \u001b[0m0.003044 \u001b[0m | \u001b[0m0.03408  \u001b[0m | \u001b[0m0.09441  \u001b[0m |\n",
      "Number of negative values: 262\n",
      "0 Lyapunov Risk= 0.017725525423884392\n",
      "Number of negative values: 263\n",
      "Number of negative values: 343\n",
      "Number of negative values: 282\n",
      "Number of negative values: 305\n",
      "Number of negative values: 299\n",
      "5 Lyapunov Risk= 0.004914699122309685\n",
      "Number of negative values: 341\n",
      "Number of negative values: 327\n",
      "Number of negative values: 321\n",
      "Number of negative values: 394\n",
      "0.008923888067846867 0.01577613608141365 0.06586808894142808\n",
      "38 2 tensor(0.0051, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m51       \u001b[0m | \u001b[0m38.0     \u001b[0m | \u001b[0m0.008924 \u001b[0m | \u001b[0m0.01578  \u001b[0m | \u001b[0m0.06587  \u001b[0m |\n",
      "Number of negative values: 222\n",
      "0 Lyapunov Risk= 0.02892250567674637\n",
      "Number of negative values: 226\n",
      "Number of negative values: 226\n",
      "Number of negative values: 226\n",
      "Number of negative values: 226\n",
      "Number of negative values: 226\n",
      "5 Lyapunov Risk= 0.026464682072401047\n",
      "Number of negative values: 226\n",
      "Number of negative values: 226\n",
      "Number of negative values: 226\n",
      "Number of negative values: 226\n",
      "0.0022801800298233353 0.09141548026143473 0.025571992006304542\n",
      "0 0 0\n",
      "| \u001b[0m52       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.00228  \u001b[0m | \u001b[0m0.09142  \u001b[0m | \u001b[0m0.02557  \u001b[0m |\n",
      "Number of negative values: 307\n",
      "0 Lyapunov Risk= 0.006959323771297932\n",
      "Number of negative values: 260\n",
      "Number of negative values: 400\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 291\n",
      "5 Lyapunov Risk= 0.012152663432061672\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "Number of negative values: 290\n",
      "0.009841954354987534 0.09904305432115469 0.08885507045680235\n",
      "4 2 tensor(0.0018, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m53       \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.009842 \u001b[0m | \u001b[0m0.09904  \u001b[0m | \u001b[0m0.08886  \u001b[0m |\n",
      "Number of negative values: 298\n",
      "0 Lyapunov Risk= 0.010443134233355522\n",
      "Number of negative values: 265\n",
      "Number of negative values: 390\n",
      "Number of negative values: 300\n",
      "Number of negative values: 291\n",
      "Number of negative values: 290\n",
      "5 Lyapunov Risk= 0.013309232890605927\n",
      "Number of negative values: 291\n",
      "Number of negative values: 294\n",
      "Number of negative values: 301\n",
      "Number of negative values: 349\n",
      "0.009212004894020057 0.0985697254966542 0.09230603270606536\n",
      "90 2 tensor(0.0236, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m54       \u001b[0m | \u001b[0m90.0     \u001b[0m | \u001b[0m0.009212 \u001b[0m | \u001b[0m0.09857  \u001b[0m | \u001b[0m0.09231  \u001b[0m |\n",
      "Number of negative values: 258\n",
      "0 Lyapunov Risk= 0.009679052978754044\n",
      "Number of negative values: 257\n",
      "Number of negative values: 257\n",
      "Number of negative values: 257\n",
      "Number of negative values: 257\n",
      "Number of negative values: 257\n",
      "5 Lyapunov Risk= 0.007345122285187244\n",
      "Number of negative values: 257\n",
      "Number of negative values: 257\n",
      "Number of negative values: 257\n",
      "Number of negative values: 257\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': 244.0, 'params': {'min_a': 0.01, 'min_b': 0.09890483107808543, 'min_c': 0.0894758674736823}}\n",
      "Time consumed: 27.002823081973474\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": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "#optimal parameters\n",
    "min_a = 0.01  \n",
    "min_b = 0.09890483172613722\n",
    "min_c = 0.08947586689985391\n",
    "pho_obtained  = 0.0629"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 306\n",
      "0 Lyapunov Risk= 0.007573953829705715\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.2622, 0.0100],\n",
      "        [0.2939, 0.0100],\n",
      "        [0.4021, 0.0683],\n",
      "        [0.1100, 0.0100]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.2142, 0.1100, 0.1657, 0.1332]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0989]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0895]])\n",
      "Number of negative values: 261\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.2167, 0.0100],\n",
      "        [0.2246, 0.0100],\n",
      "        [0.3519, 0.0100],\n",
      "        [0.0444, 0.0100]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.1587, 0.0563, 0.1017, 0.0591]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0989]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.0895]])\n"
     ]
    }
   ],
   "source": [
    "#train performance\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 = 2\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",
    "dx = predict_fn(basic, x.detach().numpy()) #use NN to predict dx\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",
    "    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 + L_W *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": "markdown",
   "metadata": {},
   "source": [
    "# Performance on NN model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "V_candidate = model(x)\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",
    "L_V = torch.diagonal(torch.mm(jacobian,dx.t()),0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "number of V > 0 points:  400\n",
      "number of V < 0 points:  0\n",
      "number of \\dot V < 0 points:  396\n",
      "number of \\dot V > 0 points:  4\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))\n",
    "\n",
    "#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] < - L_W * 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": 17,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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/PMrueqqG7j11qk899wshg8fzoQJE9L6fvzxxxkwYAAzZ87knnsmcc89r3Lnna/wv//7a+rqXmux/muvvcbEiRNZsGABb7/9Ni+++CKXXXYZe++9N488Usdtt9Xx6qtzeeONZTz//HzmzZvHeeedF+1gJiPZuFCxNmUCKxNYmcBFkgmcQT3oqJnA77777nbj8StXuo8efY1///vjffp096FDj/J77/0/X7nS/YMPPvD999/f3d0nT57s/fr18xtvvLGpzv8777zjffse0tT/yy9v9NNP/6736TPA+/cf5BUVFb58+XJ/5513vGfPnk19Pv74437yySe7u/vhhx/l99033SdMeNw7d+7mBxwwyA88cJAfeODBfv7557fwnzgHcOyxx/q8efN83LiJfuGF/9W0/+ef/xO/4orbfOVK91133dXdg8/Oscce27SdMWPG+H333efu7j179vKpU1f49OnuCxcu9n322c/POON7/tBDT/uWLVsyeh/dy3QOoBgyMaXPrV6ZwEWSCZxBllnULNZu3bqxevXq7fRr137M7rt3b3qtQ4dOLFsGO+64I5s3bwbgrLPOYsqUKey8886ceOKJ/O1vfwNg06Zt/T/99GRWr17BfffNZPLk2VRXV7NhwwYAzLYva93498aNgd7d+eIXj+OBB2YzefJsHn54Affee2/SfWicA3juuefo378/a9cGkTER95bHoPEpaLD9vm3Zsk2/++5deOCBOQwdOoK77rqLCy+8MPmBjEBJB4C4Mymll74969NmoWaQZRY1i7WyspK99tqr6QS+YsXHvPzyMwwe/JW0+rfffpv99tuPyy67jJNPPpm5c+dSVVXFunXbauc3NKyla9fP0aFDR156qY7FCbWslyxZwssvvwzAww8/zFe+EvTXGDwGDPgSc+a8yHvvBWP6a9eu480330y+E80YMOAInn/+z2zYsJ5PP13HtGmPMmTIERln8u6887b9WLVqFVu3buXoo09l9OgbmDVrVmYbSUNJB4C4Mymll74969NmoWaQpdaWLNbf//73XH/99QwePJjvfvdoLrzwGnr02D+t/g9/+AP9+/dn8ODBzJ8/n29/+9t069aNIUO+zBln9Oe2267kq18dxcKFMzjzzAE8/fTv6du3b5P+oIMO4o477uDggw9mzZo1XHLJJcC2BKwuXfbgmmt+y9ix3+Sb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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": 18,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "number of points within levelset:  242\n",
      "number of points beyond levelset:  158\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of common tensors: 0\n"
     ]
    }
   ],
   "source": [
    "#identify the stability region\n",
    "iV_point_list = list()\n",
    "uV_point_list = list()\n",
    "n = pho_obtained # 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",
    "# 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",
    "plt.scatter(x_values_u, y_values_u, color='black', label='Unstable 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()\n",
    "\n",
    "# Count number of matching tensors (by value)\n",
    "count = 0\n",
    "for t1 in uV_point_list:\n",
    "    for t2 in udV_point_list:\n",
    "        if torch.equal(t1, t2):\n",
    "            count += 1\n",
    "            break  # Avoid double-counting\n",
    "\n",
    "print(\"Number of common tensors:\", count)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Performance on FP model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "dx = predict_fn(basic, x.detach().numpy()) #use NN to predict dx\n",
    "V_candidate = model(x)\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",
    "L_V = torch.diagonal(torch.mm(jacobian,dx.t()),0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "number of V > 0 points:  400\n",
      "number of V < 0 points:  0\n",
      "number of \\dot V < 0 points:  396\n",
      "number of \\dot V > 0 points:  4\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))\n",
    "\n",
    "#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] < - L_W * 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": 21,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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/PMrueqqG7j11qk899wshg8fzoQJE9L6fvzxxxkwYAAzZ87knnsmcc89r3Lnna/wv//7a+rqXmux/muvvcbEiRNZsGABb7/9Ni+++CKXXXYZe++9N488Usdtt9Xx6qtzeeONZTz//HzmzZvHeeedF+1gJiPZuFCxNmUCKxNYmcBFkgmcQT3oqJnA77777nbj8StXuo8efY1///vjffp096FDj/J77/0/X7nS/YMPPvD999/f3d0nT57s/fr18xtvvLGpzv8777zjffse0tT/yy9v9NNP/6736TPA+/cf5BUVFb58+XJ/5513vGfPnk19Pv74437yySe7u/vhhx/l99033SdMeNw7d+7mBxwwyA88cJAfeODBfv7557fwnzgHcOyxx/q8efN83LiJfuGF/9W0/+ef/xO/4orbfOVK91133dXdg8/Oscce27SdMWPG+H333efu7j179vKpU1f49OnuCxcu9n322c/POON7/tBDT/uWLVsyeh/dy3QOoBgyMaXPrV6ZwEWSCZxBllnULNZu3bqxevXq7fRr137M7rt3b3qtQ4dOLFsGO+64I5s3bwbgrLPOYsqUKey8886ceOKJ/O1vfwNg06Zt/T/99GRWr17BfffNZPLk2VRXV7NhwwYAzLYva93498aNgd7d+eIXj+OBB2YzefJsHn54Affee2/SfWicA3juuefo378/a9cGkTER95bHoPEpaLD9vm3Zsk2/++5deOCBOQwdOoK77rqLCy+8MPmBjEBJB4C4Mymll74969NmoWaQZRY1i7WyspK99tqr6QS+YsXHvPzyMwwe/JW0+rfffpv99tuPyy67jJNPPpm5c+dSVVXFunXbauc3NKyla9fP0aFDR156qY7FCbWslyxZwssvvwzAww8/zFe+EvTXGDwGDPgSc+a8yHvvBWP6a9eu480330y+E80YMOAInn/+z2zYsJ5PP13HtGmPMmTIERln8u6887b9WLVqFVu3buXoo09l9OgbmDVrVmYbSUNJB4C4Mymll74969NmoWaQpdaWLNbf//73XH/99QwePJjvfvdoLrzwGnr02D+t/g9/+AP9+/dn8ODBzJ8/n29/+9t069aNIUO+zBln9Oe2267kq18dxcKFMzjzzAE8/fTv6du3b5P+oIMO4o477uDggw9mzZo1XHLJJcC2BKwuXfbgmmt+y9ix3+Sb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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": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "number of points within levelset:  242\n",
      "number of points beyond levelset:  158\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of common tensors: 0\n"
     ]
    }
   ],
   "source": [
    "#identify the stability region\n",
    "iV_point_list = list()\n",
    "uV_point_list = list()\n",
    "n = pho_obtained # 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",
    "plt.scatter(x_values_u, y_values_u, color='black', label='Unstable 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()\n",
    "\n",
    "# Count number of matching tensors (by value)\n",
    "count = 0\n",
    "for t1 in uV_point_list:\n",
    "    for t2 in udV_point_list:\n",
    "        if torch.equal(t1, t2):\n",
    "            count += 1\n",
    "            break  # Avoid double-counting\n",
    "\n",
    "print(\"Number of common tensors:\", count)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Save data for matlab plot"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Save to text file\n",
    "np.savetxt('V_PF_NN.txt', V_candidate.detach().numpy(), fmt='%.6f')  # value of V for each sample\n",
    "# Save to text file\n",
    "np.savetxt('LV_PF_NN.txt', L_V.detach().numpy(), fmt='%.6f')  # value of \\dot V for each sample"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "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_NN.mat', {'data': array_data1})\n",
    "savemat('stable_PF_NN.mat', {'data': array_data2})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "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
}
