{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Codes for development of LNN 1 for Van der pol system\n",
    "# -*- coding: utf-8 -*-\n",
    "import torch \n",
    "import torch.nn.functional as F\n",
    "import numpy as np\n",
    "import timeit \n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "#define activation functions in the hidden and output layers\n",
    "def acti_h(x):\n",
    "    return torch.tanh(x)\n",
    "\n",
    "def acti_o(x):\n",
    "    return x**2"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Neural network model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "class Net(torch.nn.Module):\n",
    "    \n",
    "    def __init__(self,n_input,n_hidden,n_output):\n",
    "        super(Net, self).__init__()\n",
    "        torch.manual_seed(2)\n",
    "        self.layer1 = torch.nn.Linear(n_input, n_hidden,bias=False)\n",
    "        self.layer2 = torch.nn.Linear(n_hidden,n_output,bias=False)\n",
    "        self.layer3 = torch.nn.Linear(1,n_output,bias=False)\n",
    "        self.layer4 = torch.nn.Linear(1,n_output,bias=False)\n",
    "\n",
    "    def forward(self,x):\n",
    "        h = torch.tanh(self.layer1(x))  # torch.tanh as activation function\n",
    "        out_1 = self.layer2(h) ** 2  # Squaring function at output layer\n",
    "        out_21 = (self.layer3(x[:, :1])) ** 2  # Squaring function for x_1\n",
    "        out_22 = (self.layer4(x[:, 1:])) ** 2  # Squaring function for x_2\n",
    "        return out_1 + out_21 + out_22\n",
    "    \n",
    "    def compute_jacobian(self, x):\n",
    "        \"\"\"\n",
    "        Compute the Jacobian ∂y/∂x manually (no autograd),\n",
    "        where y is the scalar output and x ∈ R^2 is the input.\n",
    "        \"\"\"\n",
    "        W1 = self.layer1.weight.detach()  # (n_hidden, 2)\n",
    "        W2 = self.layer2.weight.detach()  # (1, n_hidden)\n",
    "        W3 = self.layer3.weight.detach()  # (1, 1)\n",
    "        W4 = self.layer4.weight.detach()  # (1, 1)\n",
    "\n",
    "        x1 = x[:, :1]  # (batch, 1)\n",
    "        x2 = x[:, 1:]  # (batch, 1)\n",
    "\n",
    "        z1 = self.layer1(x)               # (batch, n_hidden)\n",
    "        h = torch.tanh(z1)                # (batch, n_hidden)\n",
    "        out1 = self.layer2(h)             # (batch, 1)\n",
    "        dy_dout1 = 2 * out1               # (batch, 1)\n",
    "\n",
    "        # ∂h/∂x = (1 - tanh^2(z1)) * W1\n",
    "        sech2 = 1 - torch.tanh(z1) ** 2   # (batch, n_hidden)\n",
    "        d_out1_dx = torch.zeros(x.shape[0], x.shape[1])  # (batch, 2)\n",
    "        for i in range(x.shape[0]):\n",
    "            d_h_dx = (sech2[i].unsqueeze(1) * W1)         # (n_hidden, 2)\n",
    "            d_out1_dx[i] = (W2 @ d_h_dx).squeeze() * dy_dout1[i]\n",
    "\n",
    "        # ∂(layer3(x1)^2)/∂x1 = 2 * (W3 * x1) * W3\n",
    "        d_out21_dx1 = 2 * (self.layer3(x1)) * W3          # (batch, 1)\n",
    "        d_out22_dx2 = 2 * (self.layer4(x2)) * W4          # (batch, 1)\n",
    "\n",
    "        # Fill Jacobian rows\n",
    "        J = torch.zeros(x.shape[0], x.shape[1])           # (batch, 2)\n",
    "        J[:, 0] = d_out1_dx[:, 0] + d_out21_dx1[:, 0]\n",
    "        J[:, 1] = d_out1_dx[:, 1] + d_out22_dx2[:, 0]\n",
    "\n",
    "        return J  # Shape: (batch_size, 2)\n",
    "        "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Dynamical system of Van der Pol"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def f_value(x): #obtain \\dot x_1 and \\dot x_2\n",
    "    y = []\n",
    "    \n",
    "    for r in range(0,len(x)): \n",
    "        x1 = x[r][0]  \n",
    "        x2 = x[r][1]\n",
    "        dx1 = x2\n",
    "        dx2 = -x1 + (x1*x1-1)*x2\n",
    "        f = [ dx1, \n",
    "              dx2]\n",
    "        y.append(f) \n",
    "    y = torch.tensor(y)\n",
    "    return y "
   ]
  },
  {
   "cell_type": "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.00265241])\n",
    "x_std = np.array([1.16720222, 1.16907296])\n",
    "\n",
    "y_mean = np.array([-0.00265241,  0.00127373])\n",
    "y_std = np.array([1.16907296, 1.8885173 ])"
   ]
  },
  {
   "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_VDP\"))\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 0x7fa68965aed0>"
      ]
     },
     "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. 2.]\n"
     ]
    }
   ],
   "source": [
    "Num = 20\n",
    "\n",
    "x1_list = np.linspace(-2, 2, Num, endpoint=True)\n",
    "x2_list = np.linspace(-2, 2, Num, endpoint=True) \n",
    "\n",
    "x_start = list()\n",
    "\n",
    "for x1 in x1_list:   \n",
    "    for x2 in x2_list:\n",
    "        x_start.append(np.array([x1, x2]))\n",
    "\n",
    "print(x_start[-1])\n",
    "# convert to np.arrays\n",
    "x = torch.FloatTensor(np.array(x_start)).requires_grad_() #LNN input\n",
    "x_0 = torch.zeros([1, 2])\n",
    "\n",
    "#we need to allow gradient calculation for the input\n",
    "x_real = torch.FloatTensor(np.array(x_start))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([[-2.0000, -2.0000],\n",
      "        [-2.0000, -1.7895],\n",
      "        [-2.0000, -1.5789],\n",
      "        [-2.0000, -1.3684],\n",
      "        [-2.0000, -1.1579],\n",
      "        [-2.0000, -0.9474],\n",
      "        [-2.0000, -0.7368],\n",
      "        [-2.0000, -0.5263],\n",
      "        [-2.0000, -0.3158],\n",
      "        [-2.0000, -0.1053],\n",
      "        [-2.0000,  0.1053],\n",
      "        [-2.0000,  0.3158],\n",
      "        [-2.0000,  0.5263],\n",
      "        [-2.0000,  0.7368],\n",
      "        [-2.0000,  0.9474],\n",
      "        [-2.0000,  1.1579],\n",
      "        [-2.0000,  1.3684],\n",
      "        [-2.0000,  1.5789],\n",
      "        [-2.0000,  1.7895],\n",
      "        [-2.0000,  2.0000],\n",
      "        [-1.7895, -2.0000],\n",
      "        [-1.7895, -1.7895],\n",
      "        [-1.7895, -1.5789],\n",
      "        [-1.7895, -1.3684],\n",
      "        [-1.7895, -1.1579],\n",
      "        [-1.7895, -0.9474],\n",
      "        [-1.7895, -0.7368],\n",
      "        [-1.7895, -0.5263],\n",
      "        [-1.7895, -0.3158],\n",
      "        [-1.7895, -0.1053],\n",
      "        [-1.7895,  0.1053],\n",
      "        [-1.7895,  0.3158],\n",
      "        [-1.7895,  0.5263],\n",
      "        [-1.7895,  0.7368],\n",
      "        [-1.7895,  0.9474],\n",
      "        [-1.7895,  1.1579],\n",
      "        [-1.7895,  1.3684],\n",
      "        [-1.7895,  1.5789],\n",
      "        [-1.7895,  1.7895],\n",
      "        [-1.7895,  2.0000],\n",
      "        [-1.5789, -2.0000],\n",
      "        [-1.5789, -1.7895],\n",
      "        [-1.5789, -1.5789],\n",
      "        [-1.5789, -1.3684],\n",
      "        [-1.5789, -1.1579],\n",
      "        [-1.5789, -0.9474],\n",
      "        [-1.5789, -0.7368],\n",
      "        [-1.5789, -0.5263],\n",
      "        [-1.5789, -0.3158],\n",
      "        [-1.5789, -0.1053],\n",
      "        [-1.5789,  0.1053],\n",
      "        [-1.5789,  0.3158],\n",
      "        [-1.5789,  0.5263],\n",
      "        [-1.5789,  0.7368],\n",
      "        [-1.5789,  0.9474],\n",
      "        [-1.5789,  1.1579],\n",
      "        [-1.5789,  1.3684],\n",
      "        [-1.5789,  1.5789],\n",
      "        [-1.5789,  1.7895],\n",
      "        [-1.5789,  2.0000],\n",
      "        [-1.3684, -2.0000],\n",
      "        [-1.3684, -1.7895],\n",
      "        [-1.3684, -1.5789],\n",
      "        [-1.3684, -1.3684],\n",
      "        [-1.3684, -1.1579],\n",
      "        [-1.3684, -0.9474],\n",
      "        [-1.3684, -0.7368],\n",
      "        [-1.3684, -0.5263],\n",
      "        [-1.3684, -0.3158],\n",
      "        [-1.3684, -0.1053],\n",
      "        [-1.3684,  0.1053],\n",
      "        [-1.3684,  0.3158],\n",
      "        [-1.3684,  0.5263],\n",
      "        [-1.3684,  0.7368],\n",
      "        [-1.3684,  0.9474],\n",
      "        [-1.3684,  1.1579],\n",
      "        [-1.3684,  1.3684],\n",
      "        [-1.3684,  1.5789],\n",
      "        [-1.3684,  1.7895],\n",
      "        [-1.3684,  2.0000],\n",
      "        [-1.1579, -2.0000],\n",
      "        [-1.1579, -1.7895],\n",
      "        [-1.1579, -1.5789],\n",
      "        [-1.1579, -1.3684],\n",
      "        [-1.1579, -1.1579],\n",
      "        [-1.1579, -0.9474],\n",
      "        [-1.1579, -0.7368],\n",
      "        [-1.1579, -0.5263],\n",
      "        [-1.1579, -0.3158],\n",
      "        [-1.1579, -0.1053],\n",
      "        [-1.1579,  0.1053],\n",
      "        [-1.1579,  0.3158],\n",
      "        [-1.1579,  0.5263],\n",
      "        [-1.1579,  0.7368],\n",
      "        [-1.1579,  0.9474],\n",
      "        [-1.1579,  1.1579],\n",
      "        [-1.1579,  1.3684],\n",
      "        [-1.1579,  1.5789],\n",
      "        [-1.1579,  1.7895],\n",
      "        [-1.1579,  2.0000],\n",
      "        [-0.9474, -2.0000],\n",
      "        [-0.9474, -1.7895],\n",
      "        [-0.9474, -1.5789],\n",
      "        [-0.9474, -1.3684],\n",
      "        [-0.9474, -1.1579],\n",
      "        [-0.9474, -0.9474],\n",
      "        [-0.9474, -0.7368],\n",
      "        [-0.9474, -0.5263],\n",
      "        [-0.9474, -0.3158],\n",
      "        [-0.9474, -0.1053],\n",
      "        [-0.9474,  0.1053],\n",
      "        [-0.9474,  0.3158],\n",
      "        [-0.9474,  0.5263],\n",
      "        [-0.9474,  0.7368],\n",
      "        [-0.9474,  0.9474],\n",
      "        [-0.9474,  1.1579],\n",
      "        [-0.9474,  1.3684],\n",
      "        [-0.9474,  1.5789],\n",
      "        [-0.9474,  1.7895],\n",
      "        [-0.9474,  2.0000],\n",
      "        [-0.7368, -2.0000],\n",
      "        [-0.7368, -1.7895],\n",
      "        [-0.7368, -1.5789],\n",
      "        [-0.7368, -1.3684],\n",
      "        [-0.7368, -1.1579],\n",
      "        [-0.7368, -0.9474],\n",
      "        [-0.7368, -0.7368],\n",
      "        [-0.7368, -0.5263],\n",
      "        [-0.7368, -0.3158],\n",
      "        [-0.7368, -0.1053],\n",
      "        [-0.7368,  0.1053],\n",
      "        [-0.7368,  0.3158],\n",
      "        [-0.7368,  0.5263],\n",
      "        [-0.7368,  0.7368],\n",
      "        [-0.7368,  0.9474],\n",
      "        [-0.7368,  1.1579],\n",
      "        [-0.7368,  1.3684],\n",
      "        [-0.7368,  1.5789],\n",
      "        [-0.7368,  1.7895],\n",
      "        [-0.7368,  2.0000],\n",
      "        [-0.5263, -2.0000],\n",
      "        [-0.5263, -1.7895],\n",
      "        [-0.5263, -1.5789],\n",
      "        [-0.5263, -1.3684],\n",
      "        [-0.5263, -1.1579],\n",
      "        [-0.5263, -0.9474],\n",
      "        [-0.5263, -0.7368],\n",
      "        [-0.5263, -0.5263],\n",
      "        [-0.5263, -0.3158],\n",
      "        [-0.5263, -0.1053],\n",
      "        [-0.5263,  0.1053],\n",
      "        [-0.5263,  0.3158],\n",
      "        [-0.5263,  0.5263],\n",
      "        [-0.5263,  0.7368],\n",
      "        [-0.5263,  0.9474],\n",
      "        [-0.5263,  1.1579],\n",
      "        [-0.5263,  1.3684],\n",
      "        [-0.5263,  1.5789],\n",
      "        [-0.5263,  1.7895],\n",
      "        [-0.5263,  2.0000],\n",
      "        [-0.3158, -2.0000],\n",
      "        [-0.3158, -1.7895],\n",
      "        [-0.3158, -1.5789],\n",
      "        [-0.3158, -1.3684],\n",
      "        [-0.3158, -1.1579],\n",
      "        [-0.3158, -0.9474],\n",
      "        [-0.3158, -0.7368],\n",
      "        [-0.3158, -0.5263],\n",
      "        [-0.3158, -0.3158],\n",
      "        [-0.3158, -0.1053],\n",
      "        [-0.3158,  0.1053],\n",
      "        [-0.3158,  0.3158],\n",
      "        [-0.3158,  0.5263],\n",
      "        [-0.3158,  0.7368],\n",
      "        [-0.3158,  0.9474],\n",
      "        [-0.3158,  1.1579],\n",
      "        [-0.3158,  1.3684],\n",
      "        [-0.3158,  1.5789],\n",
      "        [-0.3158,  1.7895],\n",
      "        [-0.3158,  2.0000],\n",
      "        [-0.1053, -2.0000],\n",
      "        [-0.1053, -1.7895],\n",
      "        [-0.1053, -1.5789],\n",
      "        [-0.1053, -1.3684],\n",
      "        [-0.1053, -1.1579],\n",
      "        [-0.1053, -0.9474],\n",
      "        [-0.1053, -0.7368],\n",
      "        [-0.1053, -0.5263],\n",
      "        [-0.1053, -0.3158],\n",
      "        [-0.1053, -0.1053],\n",
      "        [-0.1053,  0.1053],\n",
      "        [-0.1053,  0.3158],\n",
      "        [-0.1053,  0.5263],\n",
      "        [-0.1053,  0.7368],\n",
      "        [-0.1053,  0.9474],\n",
      "        [-0.1053,  1.1579],\n",
      "        [-0.1053,  1.3684],\n",
      "        [-0.1053,  1.5789],\n",
      "        [-0.1053,  1.7895],\n",
      "        [-0.1053,  2.0000],\n",
      "        [ 0.1053, -2.0000],\n",
      "        [ 0.1053, -1.7895],\n",
      "        [ 0.1053, -1.5789],\n",
      "        [ 0.1053, -1.3684],\n",
      "        [ 0.1053, -1.1579],\n",
      "        [ 0.1053, -0.9474],\n",
      "        [ 0.1053, -0.7368],\n",
      "        [ 0.1053, -0.5263],\n",
      "        [ 0.1053, -0.3158],\n",
      "        [ 0.1053, -0.1053],\n",
      "        [ 0.1053,  0.1053],\n",
      "        [ 0.1053,  0.3158],\n",
      "        [ 0.1053,  0.5263],\n",
      "        [ 0.1053,  0.7368],\n",
      "        [ 0.1053,  0.9474],\n",
      "        [ 0.1053,  1.1579],\n",
      "        [ 0.1053,  1.3684],\n",
      "        [ 0.1053,  1.5789],\n",
      "        [ 0.1053,  1.7895],\n",
      "        [ 0.1053,  2.0000],\n",
      "        [ 0.3158, -2.0000],\n",
      "        [ 0.3158, -1.7895],\n",
      "        [ 0.3158, -1.5789],\n",
      "        [ 0.3158, -1.3684],\n",
      "        [ 0.3158, -1.1579],\n",
      "        [ 0.3158, -0.9474],\n",
      "        [ 0.3158, -0.7368],\n",
      "        [ 0.3158, -0.5263],\n",
      "        [ 0.3158, -0.3158],\n",
      "        [ 0.3158, -0.1053],\n",
      "        [ 0.3158,  0.1053],\n",
      "        [ 0.3158,  0.3158],\n",
      "        [ 0.3158,  0.5263],\n",
      "        [ 0.3158,  0.7368],\n",
      "        [ 0.3158,  0.9474],\n",
      "        [ 0.3158,  1.1579],\n",
      "        [ 0.3158,  1.3684],\n",
      "        [ 0.3158,  1.5789],\n",
      "        [ 0.3158,  1.7895],\n",
      "        [ 0.3158,  2.0000],\n",
      "        [ 0.5263, -2.0000],\n",
      "        [ 0.5263, -1.7895],\n",
      "        [ 0.5263, -1.5789],\n",
      "        [ 0.5263, -1.3684],\n",
      "        [ 0.5263, -1.1579],\n",
      "        [ 0.5263, -0.9474],\n",
      "        [ 0.5263, -0.7368],\n",
      "        [ 0.5263, -0.5263],\n",
      "        [ 0.5263, -0.3158],\n",
      "        [ 0.5263, -0.1053],\n",
      "        [ 0.5263,  0.1053],\n",
      "        [ 0.5263,  0.3158],\n",
      "        [ 0.5263,  0.5263],\n",
      "        [ 0.5263,  0.7368],\n",
      "        [ 0.5263,  0.9474],\n",
      "        [ 0.5263,  1.1579],\n",
      "        [ 0.5263,  1.3684],\n",
      "        [ 0.5263,  1.5789],\n",
      "        [ 0.5263,  1.7895],\n",
      "        [ 0.5263,  2.0000],\n",
      "        [ 0.7368, -2.0000],\n",
      "        [ 0.7368, -1.7895],\n",
      "        [ 0.7368, -1.5789],\n",
      "        [ 0.7368, -1.3684],\n",
      "        [ 0.7368, -1.1579],\n",
      "        [ 0.7368, -0.9474],\n",
      "        [ 0.7368, -0.7368],\n",
      "        [ 0.7368, -0.5263],\n",
      "        [ 0.7368, -0.3158],\n",
      "        [ 0.7368, -0.1053],\n",
      "        [ 0.7368,  0.1053],\n",
      "        [ 0.7368,  0.3158],\n",
      "        [ 0.7368,  0.5263],\n",
      "        [ 0.7368,  0.7368],\n",
      "        [ 0.7368,  0.9474],\n",
      "        [ 0.7368,  1.1579],\n",
      "        [ 0.7368,  1.3684],\n",
      "        [ 0.7368,  1.5789],\n",
      "        [ 0.7368,  1.7895],\n",
      "        [ 0.7368,  2.0000],\n",
      "        [ 0.9474, -2.0000],\n",
      "        [ 0.9474, -1.7895],\n",
      "        [ 0.9474, -1.5789],\n",
      "        [ 0.9474, -1.3684],\n",
      "        [ 0.9474, -1.1579],\n",
      "        [ 0.9474, -0.9474],\n",
      "        [ 0.9474, -0.7368],\n",
      "        [ 0.9474, -0.5263],\n",
      "        [ 0.9474, -0.3158],\n",
      "        [ 0.9474, -0.1053],\n",
      "        [ 0.9474,  0.1053],\n",
      "        [ 0.9474,  0.3158],\n",
      "        [ 0.9474,  0.5263],\n",
      "        [ 0.9474,  0.7368],\n",
      "        [ 0.9474,  0.9474],\n",
      "        [ 0.9474,  1.1579],\n",
      "        [ 0.9474,  1.3684],\n",
      "        [ 0.9474,  1.5789],\n",
      "        [ 0.9474,  1.7895],\n",
      "        [ 0.9474,  2.0000],\n",
      "        [ 1.1579, -2.0000],\n",
      "        [ 1.1579, -1.7895],\n",
      "        [ 1.1579, -1.5789],\n",
      "        [ 1.1579, -1.3684],\n",
      "        [ 1.1579, -1.1579],\n",
      "        [ 1.1579, -0.9474],\n",
      "        [ 1.1579, -0.7368],\n",
      "        [ 1.1579, -0.5263],\n",
      "        [ 1.1579, -0.3158],\n",
      "        [ 1.1579, -0.1053],\n",
      "        [ 1.1579,  0.1053],\n",
      "        [ 1.1579,  0.3158],\n",
      "        [ 1.1579,  0.5263],\n",
      "        [ 1.1579,  0.7368],\n",
      "        [ 1.1579,  0.9474],\n",
      "        [ 1.1579,  1.1579],\n",
      "        [ 1.1579,  1.3684],\n",
      "        [ 1.1579,  1.5789],\n",
      "        [ 1.1579,  1.7895],\n",
      "        [ 1.1579,  2.0000],\n",
      "        [ 1.3684, -2.0000],\n",
      "        [ 1.3684, -1.7895],\n",
      "        [ 1.3684, -1.5789],\n",
      "        [ 1.3684, -1.3684],\n",
      "        [ 1.3684, -1.1579],\n",
      "        [ 1.3684, -0.9474],\n",
      "        [ 1.3684, -0.7368],\n",
      "        [ 1.3684, -0.5263],\n",
      "        [ 1.3684, -0.3158],\n",
      "        [ 1.3684, -0.1053],\n",
      "        [ 1.3684,  0.1053],\n",
      "        [ 1.3684,  0.3158],\n",
      "        [ 1.3684,  0.5263],\n",
      "        [ 1.3684,  0.7368],\n",
      "        [ 1.3684,  0.9474],\n",
      "        [ 1.3684,  1.1579],\n",
      "        [ 1.3684,  1.3684],\n",
      "        [ 1.3684,  1.5789],\n",
      "        [ 1.3684,  1.7895],\n",
      "        [ 1.3684,  2.0000],\n",
      "        [ 1.5789, -2.0000],\n",
      "        [ 1.5789, -1.7895],\n",
      "        [ 1.5789, -1.5789],\n",
      "        [ 1.5789, -1.3684],\n",
      "        [ 1.5789, -1.1579],\n",
      "        [ 1.5789, -0.9474],\n",
      "        [ 1.5789, -0.7368],\n",
      "        [ 1.5789, -0.5263],\n",
      "        [ 1.5789, -0.3158],\n",
      "        [ 1.5789, -0.1053],\n",
      "        [ 1.5789,  0.1053],\n",
      "        [ 1.5789,  0.3158],\n",
      "        [ 1.5789,  0.5263],\n",
      "        [ 1.5789,  0.7368],\n",
      "        [ 1.5789,  0.9474],\n",
      "        [ 1.5789,  1.1579],\n",
      "        [ 1.5789,  1.3684],\n",
      "        [ 1.5789,  1.5789],\n",
      "        [ 1.5789,  1.7895],\n",
      "        [ 1.5789,  2.0000],\n",
      "        [ 1.7895, -2.0000],\n",
      "        [ 1.7895, -1.7895],\n",
      "        [ 1.7895, -1.5789],\n",
      "        [ 1.7895, -1.3684],\n",
      "        [ 1.7895, -1.1579],\n",
      "        [ 1.7895, -0.9474],\n",
      "        [ 1.7895, -0.7368],\n",
      "        [ 1.7895, -0.5263],\n",
      "        [ 1.7895, -0.3158],\n",
      "        [ 1.7895, -0.1053],\n",
      "        [ 1.7895,  0.1053],\n",
      "        [ 1.7895,  0.3158],\n",
      "        [ 1.7895,  0.5263],\n",
      "        [ 1.7895,  0.7368],\n",
      "        [ 1.7895,  0.9474],\n",
      "        [ 1.7895,  1.1579],\n",
      "        [ 1.7895,  1.3684],\n",
      "        [ 1.7895,  1.5789],\n",
      "        [ 1.7895,  1.7895],\n",
      "        [ 1.7895,  2.0000],\n",
      "        [ 2.0000, -2.0000],\n",
      "        [ 2.0000, -1.7895],\n",
      "        [ 2.0000, -1.5789],\n",
      "        [ 2.0000, -1.3684],\n",
      "        [ 2.0000, -1.1579],\n",
      "        [ 2.0000, -0.9474],\n",
      "        [ 2.0000, -0.7368],\n",
      "        [ 2.0000, -0.5263],\n",
      "        [ 2.0000, -0.3158],\n",
      "        [ 2.0000, -0.1053],\n",
      "        [ 2.0000,  0.1053],\n",
      "        [ 2.0000,  0.3158],\n",
      "        [ 2.0000,  0.5263],\n",
      "        [ 2.0000,  0.7368],\n",
      "        [ 2.0000,  0.9474],\n",
      "        [ 2.0000,  1.1579],\n",
      "        [ 2.0000,  1.3684],\n",
      "        [ 2.0000,  1.5789],\n",
      "        [ 2.0000,  1.7895],\n",
      "        [ 2.0000,  2.0000]], requires_grad=True)\n",
      "tensor([[-2.0000e+00, -4.0000e+00],\n",
      "        [-1.7895e+00, -3.3684e+00],\n",
      "        [-1.5789e+00, -2.7368e+00],\n",
      "        [-1.3684e+00, -2.1053e+00],\n",
      "        [-1.1579e+00, -1.4737e+00],\n",
      "        [-9.4737e-01, -8.4211e-01],\n",
      "        [-7.3684e-01, -2.1053e-01],\n",
      "        [-5.2632e-01,  4.2105e-01],\n",
      "        [-3.1579e-01,  1.0526e+00],\n",
      "        [-1.0526e-01,  1.6842e+00],\n",
      "        [ 1.0526e-01,  2.3158e+00],\n",
      "        [ 3.1579e-01,  2.9474e+00],\n",
      "        [ 5.2632e-01,  3.5789e+00],\n",
      "        [ 7.3684e-01,  4.2105e+00],\n",
      "        [ 9.4737e-01,  4.8421e+00],\n",
      "        [ 1.1579e+00,  5.4737e+00],\n",
      "        [ 1.3684e+00,  6.1053e+00],\n",
      "        [ 1.5789e+00,  6.7368e+00],\n",
      "        [ 1.7895e+00,  7.3684e+00],\n",
      "        [ 2.0000e+00,  8.0000e+00],\n",
      "        [-2.0000e+00, -2.6150e+00],\n",
      "        [-1.7895e+00, -2.1513e+00],\n",
      "        [-1.5789e+00, -1.6877e+00],\n",
      "        [-1.3684e+00, -1.2241e+00],\n",
      "        [-1.1579e+00, -7.6046e-01],\n",
      "        [-9.4737e-01, -2.9684e-01],\n",
      "        [-7.3684e-01,  1.6679e-01],\n",
      "        [-5.2632e-01,  6.3041e-01],\n",
      "        [-3.1579e-01,  1.0940e+00],\n",
      "        [-1.0526e-01,  1.5577e+00],\n",
      "        [ 1.0526e-01,  2.0213e+00],\n",
      "        [ 3.1579e-01,  2.4849e+00],\n",
      "        [ 5.2632e-01,  2.9485e+00],\n",
      "        [ 7.3684e-01,  3.4122e+00],\n",
      "        [ 9.4737e-01,  3.8758e+00],\n",
      "        [ 1.1579e+00,  4.3394e+00],\n",
      "        [ 1.3684e+00,  4.8030e+00],\n",
      "        [ 1.5789e+00,  5.2667e+00],\n",
      "        [ 1.7895e+00,  5.7303e+00],\n",
      "        [ 2.0000e+00,  6.1939e+00],\n",
      "        [-2.0000e+00, -1.4072e+00],\n",
      "        [-1.7895e+00, -1.0929e+00],\n",
      "        [-1.5789e+00, -7.7854e-01],\n",
      "        [-1.3684e+00, -4.6421e-01],\n",
      "        [-1.1579e+00, -1.4988e-01],\n",
      "        [-9.4737e-01,  1.6446e-01],\n",
      "        [-7.3684e-01,  4.7879e-01],\n",
      "        [-5.2632e-01,  7.9312e-01],\n",
      "        [-3.1579e-01,  1.1075e+00],\n",
      "        [-1.0526e-01,  1.4218e+00],\n",
      "        [ 1.0526e-01,  1.7361e+00],\n",
      "        [ 3.1579e-01,  2.0504e+00],\n",
      "        [ 5.2632e-01,  2.3648e+00],\n",
      "        [ 7.3684e-01,  2.6791e+00],\n",
      "        [ 9.4737e-01,  2.9934e+00],\n",
      "        [ 1.1579e+00,  3.3078e+00],\n",
      "        [ 1.3684e+00,  3.6221e+00],\n",
      "        [ 1.5789e+00,  3.9364e+00],\n",
      "        [ 1.7895e+00,  4.2508e+00],\n",
      "        [ 2.0000e+00,  4.5651e+00],\n",
      "        [-2.0000e+00, -3.7673e-01],\n",
      "        [-1.7895e+00, -1.9303e-01],\n",
      "        [-1.5789e+00, -9.3309e-03],\n",
      "        [-1.3684e+00,  1.7437e-01],\n",
      "        [-1.1579e+00,  3.5807e-01],\n",
      "        [-9.4737e-01,  5.4177e-01],\n",
      "        [-7.3684e-01,  7.2547e-01],\n",
      "        [-5.2632e-01,  9.0917e-01],\n",
      "        [-3.1579e-01,  1.0929e+00],\n",
      "        [-1.0526e-01,  1.2766e+00],\n",
      "        [ 1.0526e-01,  1.4603e+00],\n",
      "        [ 3.1579e-01,  1.6440e+00],\n",
      "        [ 5.2632e-01,  1.8277e+00],\n",
      "        [ 7.3684e-01,  2.0114e+00],\n",
      "        [ 9.4737e-01,  2.1951e+00],\n",
      "        [ 1.1579e+00,  2.3788e+00],\n",
      "        [ 1.3684e+00,  2.5625e+00],\n",
      "        [ 1.5789e+00,  2.7462e+00],\n",
      "        [ 1.7895e+00,  2.9299e+00],\n",
      "        [ 2.0000e+00,  3.1136e+00],\n",
      "        [-2.0000e+00,  4.7645e-01],\n",
      "        [-1.7895e+00,  5.4818e-01],\n",
      "        [-1.5789e+00,  6.1992e-01],\n",
      "        [-1.3684e+00,  6.9165e-01],\n",
      "        [-1.1579e+00,  7.6338e-01],\n",
      "        [-9.4737e-01,  8.3511e-01],\n",
      "        [-7.3684e-01,  9.0684e-01],\n",
      "        [-5.2632e-01,  9.7857e-01],\n",
      "        [-3.1579e-01,  1.0503e+00],\n",
      "        [-1.0526e-01,  1.1220e+00],\n",
      "        [ 1.0526e-01,  1.1938e+00],\n",
      "        [ 3.1579e-01,  1.2655e+00],\n",
      "        [ 5.2632e-01,  1.3372e+00],\n",
      "        [ 7.3684e-01,  1.4090e+00],\n",
      "        [ 9.4737e-01,  1.4807e+00],\n",
      "        [ 1.1579e+00,  1.5524e+00],\n",
      "        [ 1.3684e+00,  1.6241e+00],\n",
      "        [ 1.5789e+00,  1.6959e+00],\n",
      "        [ 1.7895e+00,  1.7676e+00],\n",
      "        [ 2.0000e+00,  1.8393e+00],\n",
      "        [-2.0000e+00,  1.1524e+00],\n",
      "        [-1.7895e+00,  1.1308e+00],\n",
      "        [-1.5789e+00,  1.1092e+00],\n",
      "        [-1.3684e+00,  1.0876e+00],\n",
      "        [-1.1579e+00,  1.0660e+00],\n",
      "        [-9.4737e-01,  1.0445e+00],\n",
      "        [-7.3684e-01,  1.0229e+00],\n",
      "        [-5.2632e-01,  1.0013e+00],\n",
      "        [-3.1579e-01,  9.7973e-01],\n",
      "        [-1.0526e-01,  9.5816e-01],\n",
      "        [ 1.0526e-01,  9.3658e-01],\n",
      "        [ 3.1579e-01,  9.1500e-01],\n",
      "        [ 5.2632e-01,  8.9342e-01],\n",
      "        [ 7.3684e-01,  8.7185e-01],\n",
      "        [ 9.4737e-01,  8.5027e-01],\n",
      "        [ 1.1579e+00,  8.2869e-01],\n",
      "        [ 1.3684e+00,  8.0711e-01],\n",
      "        [ 1.5789e+00,  7.8554e-01],\n",
      "        [ 1.7895e+00,  7.6396e-01],\n",
      "        [ 2.0000e+00,  7.4238e-01],\n",
      "        [-2.0000e+00,  1.6510e+00],\n",
      "        [-1.7895e+00,  1.5547e+00],\n",
      "        [-1.5789e+00,  1.4585e+00],\n",
      "        [-1.3684e+00,  1.3623e+00],\n",
      "        [-1.1579e+00,  1.2661e+00],\n",
      "        [-9.4737e-01,  1.1698e+00],\n",
      "        [-7.3684e-01,  1.0736e+00],\n",
      "        [-5.2632e-01,  9.7740e-01],\n",
      "        [-3.1579e-01,  8.8118e-01],\n",
      "        [-1.0526e-01,  7.8495e-01],\n",
      "        [ 1.0526e-01,  6.8873e-01],\n",
      "        [ 3.1579e-01,  5.9251e-01],\n",
      "        [ 5.2632e-01,  4.9628e-01],\n",
      "        [ 7.3684e-01,  4.0006e-01],\n",
      "        [ 9.4737e-01,  3.0383e-01],\n",
      "        [ 1.1579e+00,  2.0761e-01],\n",
      "        [ 1.3684e+00,  1.1139e-01],\n",
      "        [ 1.5789e+00,  1.5162e-02],\n",
      "        [ 1.7895e+00, -8.1061e-02],\n",
      "        [ 2.0000e+00, -1.7729e-01],\n",
      "        [-2.0000e+00,  1.9723e+00],\n",
      "        [-1.7895e+00,  1.8201e+00],\n",
      "        [-1.5789e+00,  1.6679e+00],\n",
      "        [-1.3684e+00,  1.5157e+00],\n",
      "        [-1.1579e+00,  1.3635e+00],\n",
      "        [-9.4737e-01,  1.2113e+00],\n",
      "        [-7.3684e-01,  1.0590e+00],\n",
      "        [-5.2632e-01,  9.0684e-01],\n",
      "        [-3.1579e-01,  7.5463e-01],\n",
      "        [-1.0526e-01,  6.0242e-01],\n",
      "        [ 1.0526e-01,  4.5021e-01],\n",
      "        [ 3.1579e-01,  2.9800e-01],\n",
      "        [ 5.2632e-01,  1.4579e-01],\n",
      "        [ 7.3684e-01, -6.4149e-03],\n",
      "        [ 9.4737e-01, -1.5862e-01],\n",
      "        [ 1.1579e+00, -3.1083e-01],\n",
      "        [ 1.3684e+00, -4.6304e-01],\n",
      "        [ 1.5789e+00, -6.1525e-01],\n",
      "        [ 1.7895e+00, -7.6746e-01],\n",
      "        [ 2.0000e+00, -9.1967e-01],\n",
      "        [-2.0000e+00,  2.1163e+00],\n",
      "        [-1.7895e+00,  1.9268e+00],\n",
      "        [-1.5789e+00,  1.7373e+00],\n",
      "        [-1.3684e+00,  1.5477e+00],\n",
      "        [-1.1579e+00,  1.3582e+00],\n",
      "        [-9.4737e-01,  1.1687e+00],\n",
      "        [-7.3684e-01,  9.7915e-01],\n",
      "        [-5.2632e-01,  7.8962e-01],\n",
      "        [-3.1579e-01,  6.0009e-01],\n",
      "        [-1.0526e-01,  4.1056e-01],\n",
      "        [ 1.0526e-01,  2.2102e-01],\n",
      "        [ 3.1579e-01,  3.1491e-02],\n",
      "        [ 5.2632e-01, -1.5804e-01],\n",
      "        [ 7.3684e-01, -3.4757e-01],\n",
      "        [ 9.4737e-01, -5.3710e-01],\n",
      "        [ 1.1579e+00, -7.2664e-01],\n",
      "        [ 1.3684e+00, -9.1617e-01],\n",
      "        [ 1.5789e+00, -1.1057e+00],\n",
      "        [ 1.7895e+00, -1.2952e+00],\n",
      "        [ 2.0000e+00, -1.4848e+00],\n",
      "        [-2.0000e+00,  2.0831e+00],\n",
      "        [-1.7895e+00,  1.8749e+00],\n",
      "        [-1.5789e+00,  1.6667e+00],\n",
      "        [-1.3684e+00,  1.4585e+00],\n",
      "        [-1.1579e+00,  1.2503e+00],\n",
      "        [-9.4737e-01,  1.0421e+00],\n",
      "        [-7.3684e-01,  8.3394e-01],\n",
      "        [-5.2632e-01,  6.2575e-01],\n",
      "        [-3.1579e-01,  4.1755e-01],\n",
      "        [-1.0526e-01,  2.0936e-01],\n",
      "        [ 1.0526e-01,  1.1664e-03],\n",
      "        [ 3.1579e-01, -2.0703e-01],\n",
      "        [ 5.2632e-01, -4.1522e-01],\n",
      "        [ 7.3684e-01, -6.2341e-01],\n",
      "        [ 9.4737e-01, -8.3161e-01],\n",
      "        [ 1.1579e+00, -1.0398e+00],\n",
      "        [ 1.3684e+00, -1.2480e+00],\n",
      "        [ 1.5789e+00, -1.4562e+00],\n",
      "        [ 1.7895e+00, -1.6644e+00],\n",
      "        [ 2.0000e+00, -1.8726e+00],\n",
      "        [-2.0000e+00,  1.8726e+00],\n",
      "        [-1.7895e+00,  1.6644e+00],\n",
      "        [-1.5789e+00,  1.4562e+00],\n",
      "        [-1.3684e+00,  1.2480e+00],\n",
      "        [-1.1579e+00,  1.0398e+00],\n",
      "        [-9.4737e-01,  8.3161e-01],\n",
      "        [-7.3684e-01,  6.2341e-01],\n",
      "        [-5.2632e-01,  4.1522e-01],\n",
      "        [-3.1579e-01,  2.0703e-01],\n",
      "        [-1.0526e-01, -1.1664e-03],\n",
      "        [ 1.0526e-01, -2.0936e-01],\n",
      "        [ 3.1579e-01, -4.1755e-01],\n",
      "        [ 5.2632e-01, -6.2575e-01],\n",
      "        [ 7.3684e-01, -8.3394e-01],\n",
      "        [ 9.4737e-01, -1.0421e+00],\n",
      "        [ 1.1579e+00, -1.2503e+00],\n",
      "        [ 1.3684e+00, -1.4585e+00],\n",
      "        [ 1.5789e+00, -1.6667e+00],\n",
      "        [ 1.7895e+00, -1.8749e+00],\n",
      "        [ 2.0000e+00, -2.0831e+00],\n",
      "        [-2.0000e+00,  1.4848e+00],\n",
      "        [-1.7895e+00,  1.2952e+00],\n",
      "        [-1.5789e+00,  1.1057e+00],\n",
      "        [-1.3684e+00,  9.1617e-01],\n",
      "        [-1.1579e+00,  7.2664e-01],\n",
      "        [-9.4737e-01,  5.3710e-01],\n",
      "        [-7.3684e-01,  3.4757e-01],\n",
      "        [-5.2632e-01,  1.5804e-01],\n",
      "        [-3.1579e-01, -3.1491e-02],\n",
      "        [-1.0526e-01, -2.2102e-01],\n",
      "        [ 1.0526e-01, -4.1056e-01],\n",
      "        [ 3.1579e-01, -6.0009e-01],\n",
      "        [ 5.2632e-01, -7.8962e-01],\n",
      "        [ 7.3684e-01, -9.7915e-01],\n",
      "        [ 9.4737e-01, -1.1687e+00],\n",
      "        [ 1.1579e+00, -1.3582e+00],\n",
      "        [ 1.3684e+00, -1.5477e+00],\n",
      "        [ 1.5789e+00, -1.7373e+00],\n",
      "        [ 1.7895e+00, -1.9268e+00],\n",
      "        [ 2.0000e+00, -2.1163e+00],\n",
      "        [-2.0000e+00,  9.1967e-01],\n",
      "        [-1.7895e+00,  7.6746e-01],\n",
      "        [-1.5789e+00,  6.1525e-01],\n",
      "        [-1.3684e+00,  4.6304e-01],\n",
      "        [-1.1579e+00,  3.1083e-01],\n",
      "        [-9.4737e-01,  1.5862e-01],\n",
      "        [-7.3684e-01,  6.4149e-03],\n",
      "        [-5.2632e-01, -1.4579e-01],\n",
      "        [-3.1579e-01, -2.9800e-01],\n",
      "        [-1.0526e-01, -4.5021e-01],\n",
      "        [ 1.0526e-01, -6.0242e-01],\n",
      "        [ 3.1579e-01, -7.5463e-01],\n",
      "        [ 5.2632e-01, -9.0684e-01],\n",
      "        [ 7.3684e-01, -1.0590e+00],\n",
      "        [ 9.4737e-01, -1.2113e+00],\n",
      "        [ 1.1579e+00, -1.3635e+00],\n",
      "        [ 1.3684e+00, -1.5157e+00],\n",
      "        [ 1.5789e+00, -1.6679e+00],\n",
      "        [ 1.7895e+00, -1.8201e+00],\n",
      "        [ 2.0000e+00, -1.9723e+00],\n",
      "        [-2.0000e+00,  1.7729e-01],\n",
      "        [-1.7895e+00,  8.1061e-02],\n",
      "        [-1.5789e+00, -1.5162e-02],\n",
      "        [-1.3684e+00, -1.1139e-01],\n",
      "        [-1.1579e+00, -2.0761e-01],\n",
      "        [-9.4737e-01, -3.0383e-01],\n",
      "        [-7.3684e-01, -4.0006e-01],\n",
      "        [-5.2632e-01, -4.9628e-01],\n",
      "        [-3.1579e-01, -5.9251e-01],\n",
      "        [-1.0526e-01, -6.8873e-01],\n",
      "        [ 1.0526e-01, -7.8495e-01],\n",
      "        [ 3.1579e-01, -8.8118e-01],\n",
      "        [ 5.2632e-01, -9.7740e-01],\n",
      "        [ 7.3684e-01, -1.0736e+00],\n",
      "        [ 9.4737e-01, -1.1698e+00],\n",
      "        [ 1.1579e+00, -1.2661e+00],\n",
      "        [ 1.3684e+00, -1.3623e+00],\n",
      "        [ 1.5789e+00, -1.4585e+00],\n",
      "        [ 1.7895e+00, -1.5547e+00],\n",
      "        [ 2.0000e+00, -1.6510e+00],\n",
      "        [-2.0000e+00, -7.4238e-01],\n",
      "        [-1.7895e+00, -7.6396e-01],\n",
      "        [-1.5789e+00, -7.8554e-01],\n",
      "        [-1.3684e+00, -8.0711e-01],\n",
      "        [-1.1579e+00, -8.2869e-01],\n",
      "        [-9.4737e-01, -8.5027e-01],\n",
      "        [-7.3684e-01, -8.7185e-01],\n",
      "        [-5.2632e-01, -8.9342e-01],\n",
      "        [-3.1579e-01, -9.1500e-01],\n",
      "        [-1.0526e-01, -9.3658e-01],\n",
      "        [ 1.0526e-01, -9.5816e-01],\n",
      "        [ 3.1579e-01, -9.7973e-01],\n",
      "        [ 5.2632e-01, -1.0013e+00],\n",
      "        [ 7.3684e-01, -1.0229e+00],\n",
      "        [ 9.4737e-01, -1.0445e+00],\n",
      "        [ 1.1579e+00, -1.0660e+00],\n",
      "        [ 1.3684e+00, -1.0876e+00],\n",
      "        [ 1.5789e+00, -1.1092e+00],\n",
      "        [ 1.7895e+00, -1.1308e+00],\n",
      "        [ 2.0000e+00, -1.1524e+00],\n",
      "        [-2.0000e+00, -1.8393e+00],\n",
      "        [-1.7895e+00, -1.7676e+00],\n",
      "        [-1.5789e+00, -1.6959e+00],\n",
      "        [-1.3684e+00, -1.6241e+00],\n",
      "        [-1.1579e+00, -1.5524e+00],\n",
      "        [-9.4737e-01, -1.4807e+00],\n",
      "        [-7.3684e-01, -1.4090e+00],\n",
      "        [-5.2632e-01, -1.3372e+00],\n",
      "        [-3.1579e-01, -1.2655e+00],\n",
      "        [-1.0526e-01, -1.1938e+00],\n",
      "        [ 1.0526e-01, -1.1220e+00],\n",
      "        [ 3.1579e-01, -1.0503e+00],\n",
      "        [ 5.2632e-01, -9.7857e-01],\n",
      "        [ 7.3684e-01, -9.0684e-01],\n",
      "        [ 9.4737e-01, -8.3511e-01],\n",
      "        [ 1.1579e+00, -7.6338e-01],\n",
      "        [ 1.3684e+00, -6.9165e-01],\n",
      "        [ 1.5789e+00, -6.1992e-01],\n",
      "        [ 1.7895e+00, -5.4818e-01],\n",
      "        [ 2.0000e+00, -4.7645e-01],\n",
      "        [-2.0000e+00, -3.1136e+00],\n",
      "        [-1.7895e+00, -2.9299e+00],\n",
      "        [-1.5789e+00, -2.7462e+00],\n",
      "        [-1.3684e+00, -2.5625e+00],\n",
      "        [-1.1579e+00, -2.3788e+00],\n",
      "        [-9.4737e-01, -2.1951e+00],\n",
      "        [-7.3684e-01, -2.0114e+00],\n",
      "        [-5.2632e-01, -1.8277e+00],\n",
      "        [-3.1579e-01, -1.6440e+00],\n",
      "        [-1.0526e-01, -1.4603e+00],\n",
      "        [ 1.0526e-01, -1.2766e+00],\n",
      "        [ 3.1579e-01, -1.0929e+00],\n",
      "        [ 5.2632e-01, -9.0917e-01],\n",
      "        [ 7.3684e-01, -7.2547e-01],\n",
      "        [ 9.4737e-01, -5.4177e-01],\n",
      "        [ 1.1579e+00, -3.5807e-01],\n",
      "        [ 1.3684e+00, -1.7437e-01],\n",
      "        [ 1.5789e+00,  9.3309e-03],\n",
      "        [ 1.7895e+00,  1.9303e-01],\n",
      "        [ 2.0000e+00,  3.7673e-01],\n",
      "        [-2.0000e+00, -4.5651e+00],\n",
      "        [-1.7895e+00, -4.2508e+00],\n",
      "        [-1.5789e+00, -3.9364e+00],\n",
      "        [-1.3684e+00, -3.6221e+00],\n",
      "        [-1.1579e+00, -3.3078e+00],\n",
      "        [-9.4737e-01, -2.9934e+00],\n",
      "        [-7.3684e-01, -2.6791e+00],\n",
      "        [-5.2632e-01, -2.3648e+00],\n",
      "        [-3.1579e-01, -2.0504e+00],\n",
      "        [-1.0526e-01, -1.7361e+00],\n",
      "        [ 1.0526e-01, -1.4218e+00],\n",
      "        [ 3.1579e-01, -1.1075e+00],\n",
      "        [ 5.2632e-01, -7.9312e-01],\n",
      "        [ 7.3684e-01, -4.7879e-01],\n",
      "        [ 9.4737e-01, -1.6446e-01],\n",
      "        [ 1.1579e+00,  1.4988e-01],\n",
      "        [ 1.3684e+00,  4.6421e-01],\n",
      "        [ 1.5789e+00,  7.7854e-01],\n",
      "        [ 1.7895e+00,  1.0929e+00],\n",
      "        [ 2.0000e+00,  1.4072e+00],\n",
      "        [-2.0000e+00, -6.1939e+00],\n",
      "        [-1.7895e+00, -5.7303e+00],\n",
      "        [-1.5789e+00, -5.2667e+00],\n",
      "        [-1.3684e+00, -4.8030e+00],\n",
      "        [-1.1579e+00, -4.3394e+00],\n",
      "        [-9.4737e-01, -3.8758e+00],\n",
      "        [-7.3684e-01, -3.4122e+00],\n",
      "        [-5.2632e-01, -2.9485e+00],\n",
      "        [-3.1579e-01, -2.4849e+00],\n",
      "        [-1.0526e-01, -2.0213e+00],\n",
      "        [ 1.0526e-01, -1.5577e+00],\n",
      "        [ 3.1579e-01, -1.0940e+00],\n",
      "        [ 5.2632e-01, -6.3041e-01],\n",
      "        [ 7.3684e-01, -1.6679e-01],\n",
      "        [ 9.4737e-01,  2.9684e-01],\n",
      "        [ 1.1579e+00,  7.6046e-01],\n",
      "        [ 1.3684e+00,  1.2241e+00],\n",
      "        [ 1.5789e+00,  1.6877e+00],\n",
      "        [ 1.7895e+00,  2.1513e+00],\n",
      "        [ 2.0000e+00,  2.6150e+00],\n",
      "        [-2.0000e+00, -8.0000e+00],\n",
      "        [-1.7895e+00, -7.3684e+00],\n",
      "        [-1.5789e+00, -6.7368e+00],\n",
      "        [-1.3684e+00, -6.1053e+00],\n",
      "        [-1.1579e+00, -5.4737e+00],\n",
      "        [-9.4737e-01, -4.8421e+00],\n",
      "        [-7.3684e-01, -4.2105e+00],\n",
      "        [-5.2632e-01, -3.5789e+00],\n",
      "        [-3.1579e-01, -2.9474e+00],\n",
      "        [-1.0526e-01, -2.3158e+00],\n",
      "        [ 1.0526e-01, -1.6842e+00],\n",
      "        [ 3.1579e-01, -1.0526e+00],\n",
      "        [ 5.2632e-01, -4.2105e-01],\n",
      "        [ 7.3684e-01,  2.1053e-01],\n",
      "        [ 9.4737e-01,  8.4211e-01],\n",
      "        [ 1.1579e+00,  1.4737e+00],\n",
      "        [ 1.3684e+00,  2.1053e+00],\n",
      "        [ 1.5789e+00,  2.7368e+00],\n",
      "        [ 1.7895e+00,  3.3684e+00],\n",
      "        [ 2.0000e+00,  4.0000e+00]])\n",
      "tensor([[-1.9349, -3.2335],\n",
      "        [-1.7520, -2.8712],\n",
      "        [-1.5593, -2.4425],\n",
      "        [-1.3590, -1.9555],\n",
      "        [-1.1534, -1.4191],\n",
      "        [-0.9446, -0.8424],\n",
      "        [-0.7343, -0.2341],\n",
      "        [-0.5236,  0.3971],\n",
      "        [-0.3133,  1.0432],\n",
      "        [-0.1037,  1.6957],\n",
      "        [ 0.1051,  2.3472],\n",
      "        [ 0.3135,  2.9910],\n",
      "        [ 0.5218,  3.6231],\n",
      "        [ 0.7305,  4.2423],\n",
      "        [ 0.9401,  4.8505],\n",
      "        [ 1.1509,  5.4514],\n",
      "        [ 1.3624,  6.0477],\n",
      "        [ 1.5737,  6.6393],\n",
      "        [ 1.7829,  7.2203],\n",
      "        [ 1.9872,  7.7789],\n",
      "        [-1.9458, -2.3318],\n",
      "        [-1.7624, -2.0397],\n",
      "        [-1.5691, -1.6932],\n",
      "        [-1.3682, -1.3000],\n",
      "        [-1.1619, -0.8684],\n",
      "        [-0.9524, -0.4070],\n",
      "        [-0.7413,  0.0762],\n",
      "        [-0.5297,  0.5734],\n",
      "        [-0.3183,  1.0771],\n",
      "        [-0.1076,  1.5799],\n",
      "        [ 0.1027,  2.0750],\n",
      "        [ 0.3124,  2.5568],\n",
      "        [ 0.5221,  3.0225],\n",
      "        [ 0.7321,  3.4730],\n",
      "        [ 0.9428,  3.9130],\n",
      "        [ 1.1542,  4.3502],\n",
      "        [ 1.3660,  4.7930],\n",
      "        [ 1.5771,  5.2475],\n",
      "        [ 1.7857,  5.7147],\n",
      "        [ 1.9892,  6.1886],\n",
      "        [-1.9563, -1.3950],\n",
      "        [-1.7722, -1.1661],\n",
      "        [-1.5782, -0.8971],\n",
      "        [-1.3764, -0.5964],\n",
      "        [-1.1693, -0.2723],\n",
      "        [-0.9587,  0.0674],\n",
      "        [-0.7464,  0.4156],\n",
      "        [-0.5335,  0.7666],\n",
      "        [-0.3207,  1.1150],\n",
      "        [-0.1083,  1.4563],\n",
      "        [ 0.1035,  1.7868],\n",
      "        [ 0.3150,  2.1041],\n",
      "        [ 0.5262,  2.4082],\n",
      "        [ 0.7374,  2.7025],\n",
      "        [ 0.9487,  2.9937],\n",
      "        [ 1.1603,  3.2906],\n",
      "        [ 1.3715,  3.6024],\n",
      "        [ 1.5814,  3.9358],\n",
      "        [ 1.7883,  4.2930],\n",
      "        [ 1.9898,  4.6697],\n",
      "        [-1.9653, -0.4808],\n",
      "        [-1.7805, -0.3129],\n",
      "        [-1.5855, -0.1195],\n",
      "        [-1.3828,  0.0899],\n",
      "        [-1.1745,  0.3064],\n",
      "        [-0.9627,  0.5226],\n",
      "        [-0.7491,  0.7332],\n",
      "        [-0.5347,  0.9351],\n",
      "        [-0.3204,  1.1271],\n",
      "        [-0.1065,  1.3090],\n",
      "        [ 0.1068,  1.4818],\n",
      "        [ 0.3195,  1.6476],\n",
      "        [ 0.5317,  1.8094],\n",
      "        [ 0.7435,  1.9721],\n",
      "        [ 0.9549,  2.1419],\n",
      "        [ 1.1659,  2.3256],\n",
      "        [ 1.3761,  2.5290],\n",
      "        [ 1.5844,  2.7553],\n",
      "        [ 1.7892,  3.0040],\n",
      "        [ 1.9886,  3.2704],\n",
      "        [-1.9720,  0.3553],\n",
      "        [-1.7861,  0.4561],\n",
      "        [-1.5901,  0.5700],\n",
      "        [-1.3862,  0.6865],\n",
      "        [-1.1767,  0.7967],\n",
      "        [-0.9638,  0.8942],\n",
      "        [-0.7489,  0.9758],\n",
      "        [-0.5335,  1.0418],\n",
      "        [-0.3182,  1.0949],\n",
      "        [-0.1035,  1.1397],\n",
      "        [ 0.1105,  1.1812],\n",
      "        [ 0.3235,  1.2241],\n",
      "        [ 0.5358,  1.2726],\n",
      "        [ 0.7474,  1.3306],\n",
      "        [ 0.9583,  1.4017],\n",
      "        [ 1.1685,  1.4887],\n",
      "        [ 1.3777,  1.5931],\n",
      "        [ 1.5848,  1.7147],\n",
      "        [ 1.7884,  1.8506],\n",
      "        [ 1.9863,  1.9958],\n",
      "        [-1.9756,  1.0660],\n",
      "        [-1.7883,  1.0870],\n",
      "        [-1.5910,  1.1129],\n",
      "        [-1.3860,  1.1340],\n",
      "        [-1.1755,  1.1427],\n",
      "        [-0.9617,  1.1345],\n",
      "        [-0.7464,  1.1086],\n",
      "        [-0.5306,  1.0679],\n",
      "        [-0.3154,  1.0175],\n",
      "        [-0.1010,  0.9636],\n",
      "        [ 0.1123,  0.9119],\n",
      "        [ 0.3247,  0.8665],\n",
      "        [ 0.5363,  0.8299],\n",
      "        [ 0.7472,  0.8032],\n",
      "        [ 0.9575,  0.7872],\n",
      "        [ 1.1674,  0.7819],\n",
      "        [ 1.3763,  0.7870],\n",
      "        [ 1.5834,  0.8011],\n",
      "        [ 1.7869,  0.8216],\n",
      "        [ 1.9848,  0.8448],\n",
      "        [-1.9760,  1.6138],\n",
      "        [-1.7871,  1.5444],\n",
      "        [-1.5885,  1.4772],\n",
      "        [-1.3825,  1.4056],\n",
      "        [-1.1715,  1.3249],\n",
      "        [-0.9577,  1.2328],\n",
      "        [-0.7428,  1.1300],\n",
      "        [-0.5281,  1.0195],\n",
      "        [-0.3141,  0.9056],\n",
      "        [-0.1011,  0.7923],\n",
      "        [ 0.1108,  0.6824],\n",
      "        [ 0.3219,  0.5770],\n",
      "        [ 0.5326,  0.4756],\n",
      "        [ 0.7431,  0.3772],\n",
      "        [ 0.9536,  0.2812],\n",
      "        [ 1.1641,  0.1873],\n",
      "        [ 1.3739,  0.0962],\n",
      "        [ 1.5820,  0.0079],\n",
      "        [ 1.7865, -0.0780],\n",
      "        [ 1.9851, -0.1628],\n",
      "        [-1.9737,  1.9739],\n",
      "        [-1.7833,  1.8167],\n",
      "        [-1.5837,  1.6648],\n",
      "        [-1.3774,  1.5155],\n",
      "        [-1.1668,  1.3666],\n",
      "        [-0.9539,  1.2170],\n",
      "        [-0.7405,  1.0667],\n",
      "        [-0.5275,  0.9165],\n",
      "        [-0.3154,  0.7676],\n",
      "        [-0.1043,  0.6201],\n",
      "        [ 0.1062,  0.4733],\n",
      "        [ 0.3164,  0.3255],\n",
      "        [ 0.5268,  0.1741],\n",
      "        [ 0.7378,  0.0174],\n",
      "        [ 0.9493, -0.1453],\n",
      "        [ 1.1611, -0.3127],\n",
      "        [ 1.3724, -0.4827],\n",
      "        [ 1.5817, -0.6527],\n",
      "        [ 1.7872, -0.8204],\n",
      "        [ 1.9861, -0.9846],\n",
      "        [-1.9702,  2.1374],\n",
      "        [-1.7788,  1.9153],\n",
      "        [-1.5791,  1.7044],\n",
      "        [-1.3732,  1.5041],\n",
      "        [-1.1636,  1.3128],\n",
      "        [-0.9522,  1.1287],\n",
      "        [-0.7405,  0.9500],\n",
      "        [-0.5293,  0.7749],\n",
      "        [-0.3188,  0.6019],\n",
      "        [-0.1088,  0.4294],\n",
      "        [ 0.1010,  0.2549],\n",
      "        [ 0.3113,  0.0757],\n",
      "        [ 0.5223, -0.1109],\n",
      "        [ 0.7341, -0.3068],\n",
      "        [ 0.9467, -0.5120],\n",
      "        [ 1.1596, -0.7244],\n",
      "        [ 1.3717, -0.9405],\n",
      "        [ 1.5816, -1.1562],\n",
      "        [ 1.7873, -1.3681],\n",
      "        [ 1.9862, -1.5747],\n",
      "        [-1.9679,  2.1090],\n",
      "        [-1.7763,  1.8606],\n",
      "        [-1.5769,  1.6269],\n",
      "        [-1.3718,  1.4066],\n",
      "        [-1.1632,  1.1969],\n",
      "        [-0.9529,  0.9945],\n",
      "        [-0.7422,  0.7966],\n",
      "        [-0.5317,  0.6009],\n",
      "        [-0.3215,  0.4059],\n",
      "        [-0.1116,  0.2106],\n",
      "        [ 0.0985,  0.0137],\n",
      "        [ 0.3092, -0.1865],\n",
      "        [ 0.5207, -0.3917],\n",
      "        [ 0.7329, -0.6029],\n",
      "        [ 0.9457, -0.8197],\n",
      "        [ 1.1586, -1.0403],\n",
      "        [ 1.3707, -1.2618],\n",
      "        [ 1.5805, -1.4817],\n",
      "        [ 1.7862, -1.6991],\n",
      "        [ 1.9854, -1.9154],\n",
      "        [-1.9691,  1.8964],\n",
      "        [-1.7773,  1.6624],\n",
      "        [-1.5779,  1.4411],\n",
      "        [-1.3729,  1.2288],\n",
      "        [-1.1643,  1.0213],\n",
      "        [-0.9538,  0.8147],\n",
      "        [-0.7427,  0.6065],\n",
      "        [-0.5316,  0.3960],\n",
      "        [-0.3210,  0.1840],\n",
      "        [-0.1106, -0.0281],\n",
      "        [ 0.0996, -0.2384],\n",
      "        [ 0.3101, -0.4459],\n",
      "        [ 0.5211, -0.6502],\n",
      "        [ 0.7326, -0.8513],\n",
      "        [ 0.9447, -1.0495],\n",
      "        [ 1.1571, -1.2451],\n",
      "        [ 1.3689, -1.4389],\n",
      "        [ 1.5788, -1.6333],\n",
      "        [ 1.7851, -1.8325],\n",
      "        [ 1.9852, -2.0427],\n",
      "        [-1.9741,  1.4984],\n",
      "        [-1.7811,  1.3105],\n",
      "        [-1.5804,  1.1292],\n",
      "        [-1.3740,  0.9488],\n",
      "        [-1.1639,  0.7640],\n",
      "        [-0.9520,  0.5711],\n",
      "        [-0.7397,  0.3690],\n",
      "        [-0.5279,  0.1592],\n",
      "        [-0.3171, -0.0544],\n",
      "        [-0.1071, -0.2666],\n",
      "        [ 0.1022, -0.4721],\n",
      "        [ 0.3116, -0.6670],\n",
      "        [ 0.5213, -0.8494],\n",
      "        [ 0.7319, -1.0195],\n",
      "        [ 0.9435, -1.1799],\n",
      "        [ 1.1557, -1.3346],\n",
      "        [ 1.3679, -1.4894],\n",
      "        [ 1.5787, -1.6513],\n",
      "        [ 1.7860, -1.8282],\n",
      "        [ 1.9873, -2.0282],\n",
      "        [-1.9805,  0.9073],\n",
      "        [-1.7847,  0.7850],\n",
      "        [-1.5812,  0.6632],\n",
      "        [-1.3723,  0.5350],\n",
      "        [-1.1601,  0.3945],\n",
      "        [-0.9468,  0.2387],\n",
      "        [-0.7339,  0.0678],\n",
      "        [-0.5224, -0.1144],\n",
      "        [-0.3124, -0.3014],\n",
      "        [-0.1038, -0.4856],\n",
      "        [ 0.1042, -0.6599],\n",
      "        [ 0.3123, -0.8196],\n",
      "        [ 0.5212, -0.9633],\n",
      "        [ 0.7315, -1.0929],\n",
      "        [ 0.9433, -1.2130],\n",
      "        [ 1.1562, -1.3303],\n",
      "        [ 1.3694, -1.4518],\n",
      "        [ 1.5814, -1.5850],\n",
      "        [ 1.7898, -1.7370],\n",
      "        [ 1.9920, -1.9140],\n",
      "        [-1.9854,  0.1200],\n",
      "        [-1.7859,  0.0726],\n",
      "        [-1.5793,  0.0231],\n",
      "        [-1.3681, -0.0361],\n",
      "        [-1.1547, -0.1105],\n",
      "        [-0.9410, -0.2025],\n",
      "        [-0.7286, -0.3110],\n",
      "        [-0.5180, -0.4318],\n",
      "        [-0.3092, -0.5583],\n",
      "        [-0.1018, -0.6836],\n",
      "        [ 0.1053, -0.8017],\n",
      "        [ 0.3130, -0.9092],\n",
      "        [ 0.5220, -1.0057],\n",
      "        [ 0.7329, -1.0932],\n",
      "        [ 0.9455, -1.1757],\n",
      "        [ 1.1595, -1.2574],\n",
      "        [ 1.3737, -1.3428],\n",
      "        [ 1.5863, -1.4361],\n",
      "        [ 1.7950, -1.5419],\n",
      "        [ 1.9970, -1.6651],\n",
      "        [-1.9884, -0.8554],\n",
      "        [-1.7858, -0.8243],\n",
      "        [-1.5772, -0.7927],\n",
      "        [-1.3648, -0.7679],\n",
      "        [-1.1510, -0.7553],\n",
      "        [-0.9375, -0.7578],\n",
      "        [-0.7256, -0.7756],\n",
      "        [-0.5156, -0.8066],\n",
      "        [-0.3073, -0.8468],\n",
      "        [-0.1000, -0.8918],\n",
      "        [ 0.1072, -0.9372],\n",
      "        [ 0.3153, -0.9797],\n",
      "        [ 0.5250, -1.0175],\n",
      "        [ 0.7365, -1.0499],\n",
      "        [ 0.9498, -1.0772],\n",
      "        [ 1.1642, -1.1003],\n",
      "        [ 1.3785, -1.1208],\n",
      "        [ 1.5908, -1.1420],\n",
      "        [ 1.7987, -1.1690],\n",
      "        [ 1.9995, -1.2088],\n",
      "        [-1.9924, -1.9959],\n",
      "        [-1.7882, -1.8839],\n",
      "        [-1.5784, -1.7633],\n",
      "        [-1.3654, -1.6421],\n",
      "        [-1.1510, -1.5274],\n",
      "        [-0.9371, -1.4246],\n",
      "        [-0.7247, -1.3367],\n",
      "        [-0.5140, -1.2644],\n",
      "        [-0.3051, -1.2056],\n",
      "        [-0.0971, -1.1564],\n",
      "        [ 0.1107, -1.1114],\n",
      "        [ 0.3192, -1.0651],\n",
      "        [ 0.5291, -1.0131],\n",
      "        [ 0.7406, -0.9525],\n",
      "        [ 0.9536, -0.8824],\n",
      "        [ 1.1675, -0.8042],\n",
      "        [ 1.3809, -0.7214],\n",
      "        [ 1.5920, -0.6400],\n",
      "        [ 1.7987, -0.5681],\n",
      "        [ 1.9980, -0.5156],\n",
      "        [-2.0006, -3.2632],\n",
      "        [-1.7956, -3.0679],\n",
      "        [-1.5847, -2.8540],\n",
      "        [-1.3702, -2.6320],\n",
      "        [-1.1542, -2.4118],\n",
      "        [-0.9386, -2.2009],\n",
      "        [-0.7245, -2.0039],\n",
      "        [-0.5124, -1.8213],\n",
      "        [-0.3023, -1.6501],\n",
      "        [-0.0937, -1.4846],\n",
      "        [ 0.1143, -1.3178],\n",
      "        [ 0.3226, -1.1435],\n",
      "        [ 0.5317, -0.9578],\n",
      "        [ 0.7423, -0.7599],\n",
      "        [ 0.9542, -0.5525],\n",
      "        [ 1.1668, -0.3409],\n",
      "        [ 1.3789, -0.1328],\n",
      "        [ 1.5889,  0.0626],\n",
      "        [ 1.7944,  0.2350],\n",
      "        [ 1.9926,  0.3734],\n",
      "        [-2.0134, -4.6124],\n",
      "        [-1.8073, -4.3364],\n",
      "        [-1.5946, -4.0323],\n",
      "        [-1.3778, -3.7127],\n",
      "        [-1.1594, -3.3882],\n",
      "        [-0.9414, -3.0663],\n",
      "        [-0.7254, -2.7505],\n",
      "        [-0.5119, -2.4400],\n",
      "        [-0.3011, -2.1312],\n",
      "        [-0.0925, -1.8186],\n",
      "        [ 0.1150, -1.4969],\n",
      "        [ 0.3222, -1.1629],\n",
      "        [ 0.5302, -0.8163],\n",
      "        [ 0.7394, -0.4607],\n",
      "        [ 0.9500, -0.1026],\n",
      "        [ 1.1615,  0.2491],\n",
      "        [ 1.3727,  0.5838],\n",
      "        [ 1.5819,  0.8905],\n",
      "        [ 1.7866,  1.1580],\n",
      "        [ 1.9842,  1.3753],\n",
      "        [-2.0285, -6.0021],\n",
      "        [-1.8212, -5.6499],\n",
      "        [-1.6065, -5.2576],\n",
      "        [-1.3876, -4.8370],\n",
      "        [-1.1670, -4.3984],\n",
      "        [-0.9472, -3.9490],\n",
      "        [-0.7299, -3.4930],\n",
      "        [-0.5157, -3.0315],\n",
      "        [-0.3048, -2.5637],\n",
      "        [-0.0966, -2.0875],\n",
      "        [ 0.1101, -1.6013],\n",
      "        [ 0.3163, -1.1053],\n",
      "        [ 0.5233, -0.6019],\n",
      "        [ 0.7316, -0.0969],\n",
      "        [ 0.9414,  0.4015],\n",
      "        [ 1.1523,  0.8826],\n",
      "        [ 1.3631,  1.3350],\n",
      "        [ 1.5720,  1.7465],\n",
      "        [ 1.7766,  2.1054],\n",
      "        [ 1.9740,  2.4005],\n",
      "        [-2.0437, -7.3766],\n",
      "        [-1.8361, -6.9421],\n",
      "        [-1.6208, -6.4498],\n",
      "        [-1.4009, -5.9122],\n",
      "        [-1.1795, -5.3412],\n",
      "        [-0.9589, -4.7473],\n",
      "        [-0.7409, -4.1389],\n",
      "        [-0.5265, -3.5214],\n",
      "        [-0.3155, -2.8979],\n",
      "        [-0.1074, -2.2695],\n",
      "        [ 0.0989, -1.6367],\n",
      "        [ 0.3048, -1.0005],\n",
      "        [ 0.5114, -0.3639],\n",
      "        [ 0.7194,  0.2677],\n",
      "        [ 0.9292,  0.8860],\n",
      "        [ 1.1401,  1.4804],\n",
      "        [ 1.3511,  2.0388],\n",
      "        [ 1.5603,  2.5487],\n",
      "        [ 1.7651,  2.9971],\n",
      "        [ 1.9627,  3.3725]])\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.05 to ensure stability"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "start = timeit.default_timer()\n",
    "L_W = 0.07\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+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])>=2:\n",
    "                    Flag =False\n",
    "                    V_point_list = list()  \n",
    "                    break \n",
    "            K = len(V_point_list)  #if number of samples increase, update the value of pho\n",
    "            if K > Num_points:\n",
    "                Num_points = K\n",
    "                Index_pho = pho\n",
    "        pho_list.append(Index_pho)\n",
    "        NUM_list.append(Num_points)\n",
    "        \n",
    "        end = timeit.default_timer()\n",
    "    \n",
    "    print(min_a, min_b, min_c) \n",
    "    print(max(NUM_list),NUM_list.index(max(NUM_list)),pho_list[NUM_list.index(max(NUM_list))])\n",
    "    return max(NUM_list)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Bayesian Optimization "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 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: 243\n",
      "0 Lyapunov Risk= 0.007078364957123995\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/wulab2/.local/lib/python3.8/site-packages/torch/autograd/__init__.py:251: UserWarning: CUDA initialization: The NVIDIA driver on your system is too old (found version 11060). Please update your GPU driver by downloading and installing a new version from the URL: http://www.nvidia.com/Download/index.aspx Alternatively, go to: https://pytorch.org to install a PyTorch version that has been compiled with your version of the CUDA driver. (Triggered internally at ../c10/cuda/CUDAFunctions.cpp:108.)\n",
      "  Variable._execution_engine.run_backward(  # Calls into the C++ engine to run the backward pass\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "5 Lyapunov Risk= 0.006446039769798517\n",
      "Number of negative values: 212\n",
      "Number of negative values: 211\n",
      "Number of negative values: 211\n",
      "Number of negative values: 211\n",
      "0.01996042558751034 0.06598978939358487 0.04939549651064031\n",
      "0 0 0\n",
      "| \u001b[0m1        \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.01996  \u001b[0m | \u001b[0m0.06599  \u001b[0m | \u001b[0m0.0494   \u001b[0m |\n",
      "Number of negative values: 219\n",
      "0 Lyapunov Risk= 0.012355935759842396\n",
      "Number of negative values: 230\n",
      "Number of negative values: 233\n",
      "Number of negative values: 232\n",
      "Number of negative values: 232\n",
      "Number of negative values: 232\n",
      "5 Lyapunov Risk= 0.00833093747496605\n",
      "Number of negative values: 232\n",
      "Number of negative values: 232\n",
      "Number of negative values: 232\n",
      "Number of negative values: 232\n",
      "0.07875049978766316 0.08019782273069231 0.03453333447543775\n",
      "2 0 tensor(0.0002, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m2        \u001b[0m | \u001b[95m2.0      \u001b[0m | \u001b[95m0.07875  \u001b[0m | \u001b[95m0.0802   \u001b[0m | \u001b[95m0.03453  \u001b[0m |\n",
      "Number of negative values: 225\n",
      "0 Lyapunov Risk= 0.022011566907167435\n",
      "Number of negative values: 232\n",
      "Number of negative values: 226\n",
      "Number of negative values: 223\n",
      "Number of negative values: 214\n",
      "Number of negative values: 208\n",
      "5 Lyapunov Risk= 0.022438373416662216\n",
      "Number of negative values: 211\n",
      "Number of negative values: 212\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "0.028369961259166576 0.08216849597815173 0.09623254183153347\n",
      "92 0 tensor(0.0119, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m3        \u001b[0m | \u001b[95m92.0     \u001b[0m | \u001b[95m0.02837  \u001b[0m | \u001b[95m0.08217  \u001b[0m | \u001b[95m0.09623  \u001b[0m |\n",
      "Number of negative values: 270\n",
      "0 Lyapunov Risk= 0.009419572539627552\n",
      "Number of negative values: 251\n",
      "Number of negative values: 243\n",
      "Number of negative values: 246\n",
      "Number of negative values: 246\n",
      "Number of negative values: 245\n",
      "5 Lyapunov Risk= 0.009991819970309734\n",
      "Number of negative values: 242\n",
      "Number of negative values: 238\n",
      "Number of negative values: 239\n",
      "Number of negative values: 237\n",
      "0.08771733083946738 0.04220355429620801 0.05508956129711129\n",
      "126 0 tensor(0.0087, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m4        \u001b[0m | \u001b[95m126.0    \u001b[0m | \u001b[95m0.08772  \u001b[0m | \u001b[95m0.0422   \u001b[0m | \u001b[95m0.05509  \u001b[0m |\n",
      "Number of negative values: 238\n",
      "0 Lyapunov Risk= 0.010047347284853458\n",
      "Number of negative values: 245\n",
      "Number of negative values: 238\n",
      "Number of negative values: 236\n",
      "Number of negative values: 236\n",
      "Number of negative values: 237\n",
      "5 Lyapunov Risk= 0.006976007018238306\n",
      "Number of negative values: 238\n",
      "Number of negative values: 238\n",
      "Number of negative values: 239\n",
      "Number of negative values: 241\n",
      "0.06866283058204149 0.07414318242846102 0.04332256793113555\n",
      "0 0 0\n",
      "| \u001b[0m5        \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.06866  \u001b[0m | \u001b[0m0.07414  \u001b[0m | \u001b[0m0.04332  \u001b[0m |\n",
      "Number of negative values: 269\n",
      "0 Lyapunov Risk= 0.009420465677976608\n",
      "Number of negative values: 251\n",
      "Number of negative values: 242\n",
      "Number of negative values: 244\n",
      "Number of negative values: 244\n",
      "Number of negative values: 243\n",
      "5 Lyapunov Risk= 0.01035980973392725\n",
      "Number of negative values: 240\n",
      "Number of negative values: 235\n",
      "Number of negative values: 237\n",
      "Number of negative values: 238\n",
      "0.08795239081932799 0.04139068045194713 0.05513592457835365\n",
      "122 0 tensor(0.0080, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m6        \u001b[0m | \u001b[0m122.0    \u001b[0m | \u001b[0m0.08795  \u001b[0m | \u001b[0m0.04139  \u001b[0m | \u001b[0m0.05514  \u001b[0m |\n",
      "Number of negative values: 267\n",
      "0 Lyapunov Risk= 0.012745856307446957\n",
      "Number of negative values: 264\n",
      "Number of negative values: 240\n",
      "Number of negative values: 227\n",
      "Number of negative values: 224\n",
      "Number of negative values: 223\n",
      "5 Lyapunov Risk= 0.016343340277671814\n",
      "Number of negative values: 217\n",
      "Number of negative values: 217\n",
      "Number of negative values: 216\n",
      "Number of negative values: 217\n",
      "0.09150718917078049 0.05386506625486208 0.06755638443182578\n",
      "124 0 tensor(0.0112, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m7        \u001b[0m | \u001b[0m124.0    \u001b[0m | \u001b[0m0.09151  \u001b[0m | \u001b[0m0.05387  \u001b[0m | \u001b[0m0.06756  \u001b[0m |\n",
      "Number of negative values: 253\n",
      "0 Lyapunov Risk= 0.011481382884085178\n",
      "Number of negative values: 258\n",
      "Number of negative values: 233\n",
      "Number of negative values: 225\n",
      "Number of negative values: 221\n",
      "Number of negative values: 214\n",
      "5 Lyapunov Risk= 0.014908344484865665\n",
      "Number of negative values: 210\n",
      "Number of negative values: 206\n",
      "Number of negative values: 201\n",
      "Number of negative values: 199\n",
      "0.07106144354409763 0.045596501028649795 0.06843251373062197\n",
      "114 0 tensor(0.0082, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m8        \u001b[0m | \u001b[0m114.0    \u001b[0m | \u001b[0m0.07106  \u001b[0m | \u001b[0m0.0456   \u001b[0m | \u001b[0m0.06843  \u001b[0m |\n",
      "Number of negative values: 238\n",
      "0 Lyapunov Risk= 0.02429940551519394\n",
      "Number of negative values: 263\n",
      "Number of negative values: 251\n",
      "Number of negative values: 247\n",
      "Number of negative values: 223\n",
      "Number of negative values: 211\n",
      "5 Lyapunov Risk= 0.02330140396952629\n",
      "Number of negative values: 204\n",
      "Number of negative values: 201\n",
      "Number of negative values: 201\n",
      "Number of negative values: 200\n",
      "0.07702041271743958 0.056354764749223206 0.1\n",
      "74 3 tensor(0.0140, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m9        \u001b[0m | \u001b[0m74.0     \u001b[0m | \u001b[0m0.07702  \u001b[0m | \u001b[0m0.05635  \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 268\n",
      "0 Lyapunov Risk= 0.010992941446602345\n",
      "Number of negative values: 271\n",
      "Number of negative values: 266\n",
      "Number of negative values: 249\n",
      "Number of negative values: 237\n",
      "Number of negative values: 232\n",
      "5 Lyapunov Risk= 0.010096793062984943\n",
      "Number of negative values: 223\n",
      "Number of negative values: 220\n",
      "Number of negative values: 217\n",
      "Number of negative values: 215\n",
      "0.09852850712119339 0.054421985000440966 0.047799828774833034\n",
      "96 1 tensor(0.0061, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m10       \u001b[0m | \u001b[0m96.0     \u001b[0m | \u001b[0m0.09853  \u001b[0m | \u001b[0m0.05442  \u001b[0m | \u001b[0m0.0478   \u001b[0m |\n",
      "Number of negative values: 219\n",
      "0 Lyapunov Risk= 0.018349027261137962\n",
      "Number of negative values: 234\n",
      "Number of negative values: 246\n",
      "Number of negative values: 221\n",
      "Number of negative values: 221\n",
      "Number of negative values: 222\n",
      "5 Lyapunov Risk= 0.018129851669073105\n",
      "Number of negative values: 224\n",
      "Number of negative values: 229\n",
      "Number of negative values: 230\n",
      "Number of negative values: 238\n",
      "0.0515382038566491 0.012407265127804823 0.08677557826132952\n",
      "134 2 tensor(0.0156, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m11       \u001b[0m | \u001b[95m134.0    \u001b[0m | \u001b[95m0.05154  \u001b[0m | \u001b[95m0.01241  \u001b[0m | \u001b[95m0.08678  \u001b[0m |\n",
      "Number of negative values: 215\n",
      "0 Lyapunov Risk= 0.02458520419895649\n",
      "Number of negative values: 233\n",
      "Number of negative values: 234\n",
      "Number of negative values: 211\n",
      "Number of negative values: 213\n",
      "Number of negative values: 214\n",
      "5 Lyapunov Risk= 0.024761319160461426\n",
      "Number of negative values: 213\n",
      "Number of negative values: 213\n",
      "Number of negative values: 218\n",
      "Number of negative values: 234\n",
      "0.03137475061737858 0.014302875631558088 0.09975099284196638\n",
      "104 2 tensor(0.0141, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m12       \u001b[0m | \u001b[0m104.0    \u001b[0m | \u001b[0m0.03137  \u001b[0m | \u001b[0m0.0143   \u001b[0m | \u001b[0m0.09975  \u001b[0m |\n",
      "Number of negative values: 235\n",
      "0 Lyapunov Risk= 0.01016151811927557\n",
      "Number of negative values: 238\n",
      "Number of negative values: 246\n",
      "Number of negative values: 226\n",
      "Number of negative values: 224\n",
      "Number of negative values: 224\n",
      "5 Lyapunov Risk= 0.010271826758980751\n",
      "Number of negative values: 228\n",
      "Number of negative values: 229\n",
      "Number of negative values: 233\n",
      "Number of negative values: 233\n",
      "0.04215338066619073 0.01 0.06554657360340989\n",
      "130 2 tensor(0.0089, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m13       \u001b[0m | \u001b[0m130.0    \u001b[0m | \u001b[0m0.04215  \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.06555  \u001b[0m |\n",
      "Number of negative values: 244\n",
      "0 Lyapunov Risk= 0.011219033040106297\n",
      "Number of negative values: 236\n",
      "Number of negative values: 259\n",
      "Number of negative values: 251\n",
      "Number of negative values: 248\n",
      "Number of negative values: 252\n",
      "5 Lyapunov Risk= 0.010092221200466156\n",
      "Number of negative values: 251\n",
      "Number of negative values: 236\n",
      "Number of negative values: 230\n",
      "Number of negative values: 225\n",
      "0.06750583504560279 0.011603092127033043 0.06794876676656515\n",
      "146 4 tensor(0.0109, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m14       \u001b[0m | \u001b[95m146.0    \u001b[0m | \u001b[95m0.06751  \u001b[0m | \u001b[95m0.0116   \u001b[0m | \u001b[95m0.06795  \u001b[0m |\n",
      "Number of negative values: 224\n",
      "0 Lyapunov Risk= 0.020433850586414337\n",
      "Number of negative values: 237\n",
      "Number of negative values: 263\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 250\n",
      "Number of negative values: 252\n",
      "Number of negative values: 251\n",
      "5 Lyapunov Risk= 0.017580358311533928\n",
      "Number of negative values: 248\n",
      "Number of negative values: 231\n",
      "Number of negative values: 223\n",
      "Number of negative values: 214\n",
      "0.07908346177444549 0.01 0.0908987946502663\n",
      "144 2 tensor(0.0223, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m15       \u001b[0m | \u001b[0m144.0    \u001b[0m | \u001b[0m0.07908  \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.0909   \u001b[0m |\n",
      "Number of negative values: 267\n",
      "0 Lyapunov Risk= 0.004099172540009022\n",
      "Number of negative values: 254\n",
      "Number of negative values: 235\n",
      "Number of negative values: 239\n",
      "Number of negative values: 236\n",
      "Number of negative values: 233\n",
      "5 Lyapunov Risk= 0.00416423799470067\n",
      "Number of negative values: 230\n",
      "Number of negative values: 228\n",
      "Number of negative values: 225\n",
      "Number of negative values: 229\n",
      "0.057452725196818454 0.01 0.03829079118516173\n",
      "112 0 tensor(0.0031, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m16       \u001b[0m | \u001b[0m112.0    \u001b[0m | \u001b[0m0.05745  \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.03829  \u001b[0m |\n",
      "Number of negative values: 233\n",
      "0 Lyapunov Risk= 0.02545110508799553\n",
      "Number of negative values: 267\n",
      "Number of negative values: 260\n",
      "Number of negative values: 240\n",
      "Number of negative values: 221\n",
      "Number of negative values: 208\n",
      "5 Lyapunov Risk= 0.02787330187857151\n",
      "Number of negative values: 199\n",
      "Number of negative values: 198\n",
      "Number of negative values: 196\n",
      "Number of negative values: 195\n",
      "0.1 0.01 0.1\n",
      "154 1 tensor(0.0308, grad_fn=<AddBackward0>)\n",
      "| \u001b[95m17       \u001b[0m | \u001b[95m154.0    \u001b[0m | \u001b[95m0.1      \u001b[0m | \u001b[95m0.01     \u001b[0m | \u001b[95m0.1      \u001b[0m |\n",
      "Number of negative values: 241\n",
      "0 Lyapunov Risk= 0.011371358297765255\n",
      "Number of negative values: 229\n",
      "Number of negative values: 226\n",
      "Number of negative values: 226\n",
      "Number of negative values: 226\n",
      "Number of negative values: 227\n",
      "5 Lyapunov Risk= 0.008790985681116581\n",
      "Number of negative values: 227\n",
      "Number of negative values: 226\n",
      "Number of negative values: 228\n",
      "Number of negative values: 227\n",
      "0.05667202114398779 0.08621748694363306 0.05020228359898451\n",
      "0 0 0\n",
      "| \u001b[0m18       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.05667  \u001b[0m | \u001b[0m0.08622  \u001b[0m | \u001b[0m0.0502   \u001b[0m |\n",
      "Number of negative values: 260\n",
      "0 Lyapunov Risk= 0.010460175573825836\n",
      "Number of negative values: 266\n",
      "Number of negative values: 236\n",
      "Number of negative values: 227\n",
      "Number of negative values: 222\n",
      "Number of negative values: 217\n",
      "5 Lyapunov Risk= 0.014756176620721817\n",
      "Number of negative values: 212\n",
      "Number of negative values: 208\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "0.07989413057315715 0.03789227346663301 0.06360490865766633\n",
      "104 0 tensor(0.0069, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m19       \u001b[0m | \u001b[0m104.0    \u001b[0m | \u001b[0m0.07989  \u001b[0m | \u001b[0m0.03789  \u001b[0m | \u001b[0m0.0636   \u001b[0m |\n",
      "Number of negative values: 196\n",
      "0 Lyapunov Risk= 0.0017533894861117005\n",
      "Number of negative values: 200\n",
      "Number of negative values: 201\n",
      "Number of negative values: 201\n",
      "Number of negative values: 201\n",
      "Number of negative values: 201\n",
      "5 Lyapunov Risk= 0.0002446180151309818\n",
      "Number of negative values: 201\n",
      "Number of negative values: 201\n",
      "Number of negative values: 201\n",
      "Number of negative values: 201\n",
      "0.001 0.01 0.01\n",
      "20 1 tensor(2.8981e-05, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m20       \u001b[0m | \u001b[0m20.0     \u001b[0m | \u001b[0m0.001    \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.01     \u001b[0m |\n",
      "Number of negative values: 188\n",
      "0 Lyapunov Risk= 0.009259320795536041\n",
      "Number of negative values: 192\n",
      "Number of negative values: 194\n",
      "Number of negative values: 195\n",
      "Number of negative values: 197\n",
      "Number of negative values: 201\n",
      "5 Lyapunov Risk= 0.008710534311830997\n",
      "Number of negative values: 207\n",
      "Number of negative values: 207\n",
      "Number of negative values: 210\n",
      "Number of negative values: 212\n",
      "0.1 0.01 0.01\n",
      "4 6 tensor(7.7566e-05, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m21       \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.01     \u001b[0m |\n",
      "Number of negative values: 209\n",
      "0 Lyapunov Risk= 0.002641035243868828\n",
      "Number of negative values: 241\n",
      "Number of negative values: 230\n",
      "Number of negative values: 232\n",
      "Number of negative values: 232\n",
      "Number of negative values: 231\n",
      "5 Lyapunov Risk= 0.0005509047769010067\n",
      "Number of negative values: 232\n",
      "Number of negative values: 236\n",
      "Number of negative values: 236\n",
      "Number of negative values: 234\n",
      "0.026131964532863546 0.017690482369735437 0.014582305639612611\n",
      "10 1 tensor(4.3756e-05, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m22       \u001b[0m | \u001b[0m10.0     \u001b[0m | \u001b[0m0.02613  \u001b[0m | \u001b[0m0.01769  \u001b[0m | \u001b[0m0.01458  \u001b[0m |\n",
      "Number of negative values: 209\n",
      "0 Lyapunov Risk= 0.024453721940517426\n",
      "Number of negative values: 210\n",
      "Number of negative values: 188\n",
      "Number of negative values: 194\n",
      "Number of negative values: 206\n",
      "Number of negative values: 209\n",
      "5 Lyapunov Risk= 0.024884114041924477\n",
      "Number of negative values: 211\n",
      "Number of negative values: 211\n",
      "Number of negative values: 211\n",
      "Number of negative values: 211\n",
      "0.001 0.04483049711977374 0.1\n",
      "8 2 tensor(0.0014, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m23       \u001b[0m | \u001b[0m8.0      \u001b[0m | \u001b[0m0.001    \u001b[0m | \u001b[0m0.04483  \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 235\n",
      "0 Lyapunov Risk= 0.025326982140541077\n",
      "Number of negative values: 257\n",
      "Number of negative values: 249\n",
      "Number of negative values: 239\n",
      "Number of negative values: 220\n",
      "Number of negative values: 206\n",
      "5 Lyapunov Risk= 0.028206180781126022\n",
      "Number of negative values: 200\n",
      "Number of negative values: 198\n",
      "Number of negative values: 196\n",
      "Number of negative values: 195\n",
      "0.1 0.03078912421778787 0.1\n",
      "134 1 tensor(0.0220, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m24       \u001b[0m | \u001b[0m134.0    \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.03079  \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 259\n",
      "0 Lyapunov Risk= 0.026082796975970268\n",
      "Number of negative values: 257\n",
      "Number of negative values: 242\n",
      "Number of negative values: 232\n",
      "Number of negative values: 223\n",
      "Number of negative values: 217\n",
      "5 Lyapunov Risk= 0.03649451583623886\n",
      "Number of negative values: 210\n",
      "Number of negative values: 208\n",
      "Number of negative values: 206\n",
      "Number of negative values: 203\n",
      "0.1 0.1 0.1\n",
      "136 0 tensor(0.0252, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m25       \u001b[0m | \u001b[0m136.0    \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 238\n",
      "0 Lyapunov Risk= 0.023628147318959236\n",
      "Number of negative values: 240\n",
      "Number of negative values: 228\n",
      "Number of negative values: 224\n",
      "Number of negative values: 219\n",
      "Number of negative values: 213\n",
      "5 Lyapunov Risk= 0.025359561666846275\n",
      "Number of negative values: 212\n",
      "Number of negative values: 212\n",
      "Number of negative values: 216\n",
      "Number of negative values: 217\n",
      "0.06600250257672595 0.09958783755081284 0.09829415435619178\n",
      "98 0 tensor(0.0158, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m26       \u001b[0m | \u001b[0m98.0     \u001b[0m | \u001b[0m0.066    \u001b[0m | \u001b[0m0.09959  \u001b[0m | \u001b[0m0.09829  \u001b[0m |\n",
      "Number of negative values: 205\n",
      "0 Lyapunov Risk= 0.0029337427113205194\n",
      "Number of negative values: 251\n",
      "Number of negative values: 246\n",
      "Number of negative values: 241\n",
      "Number of negative values: 240\n",
      "Number of negative values: 242\n",
      "5 Lyapunov Risk= 0.0004329855437390506\n",
      "Number of negative values: 242\n",
      "Number of negative values: 245\n",
      "Number of negative values: 247\n",
      "Number of negative values: 246\n",
      "0.0347296351853873 0.016752824699881295 0.011992711001254612\n",
      "10 3 tensor(3.2928e-05, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m27       \u001b[0m | \u001b[0m10.0     \u001b[0m | \u001b[0m0.03473  \u001b[0m | \u001b[0m0.01675  \u001b[0m | \u001b[0m0.01199  \u001b[0m |\n",
      "Number of negative values: 267\n",
      "0 Lyapunov Risk= 0.019226552918553352\n",
      "Number of negative values: 238\n",
      "Number of negative values: 241\n",
      "Number of negative values: 243\n",
      "Number of negative values: 243\n",
      "Number of negative values: 242\n",
      "5 Lyapunov Risk= 0.01845277100801468\n",
      "Number of negative values: 249\n",
      "Number of negative values: 251\n",
      "Number of negative values: 253\n",
      "Number of negative values: 254\n",
      "0.1 0.1 0.0748618504575303\n",
      "58 0 tensor(0.0082, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m28       \u001b[0m | \u001b[0m58.0     \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.07486  \u001b[0m |\n",
      "Number of negative values: 249\n",
      "0 Lyapunov Risk= 0.016914639621973038\n",
      "Number of negative values: 268\n",
      "Number of negative values: 250\n",
      "Number of negative values: 231\n",
      "Number of negative values: 217\n",
      "Number of negative values: 216\n",
      "5 Lyapunov Risk= 0.025748251006007195\n",
      "Number of negative values: 215\n",
      "Number of negative values: 211\n",
      "Number of negative values: 205\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 200\n",
      "0.1 0.01 0.08002617152273353\n",
      "154 1 tensor(0.0197, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m29       \u001b[0m | \u001b[0m154.0    \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.08003  \u001b[0m |\n",
      "Number of negative values: 227\n",
      "0 Lyapunov Risk= 0.024075264111161232\n",
      "Number of negative values: 205\n",
      "Number of negative values: 192\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 201\n",
      "5 Lyapunov Risk= 0.024462953209877014\n",
      "Number of negative values: 201\n",
      "Number of negative values: 201\n",
      "Number of negative values: 201\n",
      "Number of negative values: 201\n",
      "0.001 0.1 0.1\n",
      "38 0 tensor(0.0057, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m30       \u001b[0m | \u001b[0m38.0     \u001b[0m | \u001b[0m0.001    \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 200\n",
      "0 Lyapunov Risk= 0.013159334659576416\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "5 Lyapunov Risk= 0.011531239375472069\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "Number of negative values: 200\n",
      "0.001 0.1 0.01\n",
      "0 0 0\n",
      "| \u001b[0m31       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.001    \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.01     \u001b[0m |\n",
      "Number of negative values: 255\n",
      "0 Lyapunov Risk= 0.024940799921751022\n",
      "Number of negative values: 268\n",
      "Number of negative values: 248\n",
      "Number of negative values: 229\n",
      "Number of negative values: 220\n",
      "Number of negative values: 212\n",
      "5 Lyapunov Risk= 0.032983917742967606\n",
      "Number of negative values: 206\n",
      "Number of negative values: 202\n",
      "Number of negative values: 199\n",
      "Number of negative values: 197\n",
      "0.09993728180781966 0.07656447399650791 0.09927364322146048\n",
      "114 0 tensor(0.0179, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m32       \u001b[0m | \u001b[0m114.0    \u001b[0m | \u001b[0m0.09994  \u001b[0m | \u001b[0m0.07656  \u001b[0m | \u001b[0m0.09927  \u001b[0m |\n",
      "Number of negative values: 221\n",
      "0 Lyapunov Risk= 0.010992193594574928\n",
      "Number of negative values: 201\n",
      "Number of negative values: 219\n",
      "Number of negative values: 205\n",
      "Number of negative values: 211\n",
      "Number of negative values: 212\n",
      "5 Lyapunov Risk= 0.011529386974871159\n",
      "Number of negative values: 211\n",
      "Number of negative values: 211\n",
      "Number of negative values: 202\n",
      "Number of negative values: 193\n",
      "0.004341461811462775 0.010541251342918046 0.06772678551324735\n",
      "20 2 tensor(0.0014, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m33       \u001b[0m | \u001b[0m20.0     \u001b[0m | \u001b[0m0.004341 \u001b[0m | \u001b[0m0.01054  \u001b[0m | \u001b[0m0.06773  \u001b[0m |\n",
      "Number of negative values: 250\n",
      "0 Lyapunov Risk= 0.008109977468848228\n",
      "Number of negative values: 222\n",
      "Number of negative values: 253\n",
      "Number of negative values: 245\n",
      "Number of negative values: 240\n",
      "Number of negative values: 242\n",
      "5 Lyapunov Risk= 0.007878188975155354\n",
      "Number of negative values: 242\n",
      "Number of negative values: 246\n",
      "Number of negative values: 246\n",
      "Number of negative values: 239\n",
      "0.05261199051959163 0.026919042942154234 0.058605825829340936\n",
      "140 8 tensor(0.0076, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m34       \u001b[0m | \u001b[0m140.0    \u001b[0m | \u001b[0m0.05261  \u001b[0m | \u001b[0m0.02692  \u001b[0m | \u001b[0m0.05861  \u001b[0m |\n",
      "Number of negative values: 218\n",
      "0 Lyapunov Risk= 0.017360342666506767\n",
      "Number of negative values: 247\n",
      "Number of negative values: 243\n",
      "Number of negative values: 236\n",
      "Number of negative values: 227\n",
      "Number of negative values: 235\n",
      "5 Lyapunov Risk= 0.016714975237846375\n",
      "Number of negative values: 235\n",
      "Number of negative values: 237\n",
      "Number of negative values: 235\n",
      "Number of negative values: 230\n",
      "0.04672899447958127 0.037515741155630354 0.08504941441353796\n",
      "104 3 tensor(0.0102, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m35       \u001b[0m | \u001b[0m104.0    \u001b[0m | \u001b[0m0.04673  \u001b[0m | \u001b[0m0.03752  \u001b[0m | \u001b[0m0.08505  \u001b[0m |\n",
      "Number of negative values: 250\n",
      "0 Lyapunov Risk= 0.007923448458313942\n",
      "Number of negative values: 218\n",
      "Number of negative values: 262\n",
      "Number of negative values: 245\n",
      "Number of negative values: 243\n",
      "Number of negative values: 244\n",
      "5 Lyapunov Risk= 0.0074675255455076694\n",
      "Number of negative values: 247\n",
      "Number of negative values: 246\n",
      "Number of negative values: 247\n",
      "Number of negative values: 245\n",
      "0.05697716824529382 0.01 0.0576618173452969\n",
      "134 7 tensor(0.0073, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m36       \u001b[0m | \u001b[0m134.0    \u001b[0m | \u001b[0m0.05698  \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.05766  \u001b[0m |\n",
      "Number of negative values: 272\n",
      "0 Lyapunov Risk= 0.005701357498764992\n",
      "Number of negative values: 267\n",
      "Number of negative values: 261\n",
      "Number of negative values: 251\n",
      "Number of negative values: 242\n",
      "Number of negative values: 235\n",
      "5 Lyapunov Risk= 0.004942701663821936\n",
      "Number of negative values: 227\n",
      "Number of negative values: 222\n",
      "Number of negative values: 218\n",
      "Number of negative values: 222\n",
      "0.0676114249838147 0.038313141754893795 0.040712143630811874\n",
      "120 0 tensor(0.0050, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m37       \u001b[0m | \u001b[0m120.0    \u001b[0m | \u001b[0m0.06761  \u001b[0m | \u001b[0m0.03831  \u001b[0m | \u001b[0m0.04071  \u001b[0m |\n",
      "Number of negative values: 242\n",
      "0 Lyapunov Risk= 0.02038051374256611\n",
      "Number of negative values: 265\n",
      "Number of negative values: 256\n",
      "Number of negative values: 235\n",
      "Number of negative values: 214\n",
      "Number of negative values: 210\n",
      "5 Lyapunov Risk= 0.027191655710339546\n",
      "Number of negative values: 208\n",
      "Number of negative values: 204\n",
      "Number of negative values: 198\n",
      "Number of negative values: 193\n",
      "0.1 0.018992969305192103 0.08892216241909906\n",
      "150 1 tensor(0.0213, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m38       \u001b[0m | \u001b[0m150.0    \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.01899  \u001b[0m | \u001b[0m0.08892  \u001b[0m |\n",
      "Number of negative values: 200\n",
      "0 Lyapunov Risk= 0.012258767150342464\n",
      "Number of negative values: 204\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "5 Lyapunov Risk= 0.010306349955499172\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "Number of negative values: 206\n",
      "0.014278238055605082 0.09534771301269847 0.024274724007999612\n",
      "0 0 0\n",
      "| \u001b[0m39       \u001b[0m | \u001b[0m0.0      \u001b[0m | \u001b[0m0.01428  \u001b[0m | \u001b[0m0.09535  \u001b[0m | \u001b[0m0.02427  \u001b[0m |\n",
      "Number of negative values: 206\n",
      "0 Lyapunov Risk= 0.019711581990122795\n",
      "Number of negative values: 206\n",
      "Number of negative values: 204\n",
      "Number of negative values: 204\n",
      "Number of negative values: 201\n",
      "Number of negative values: 200\n",
      "5 Lyapunov Risk= 0.018548941239714622\n",
      "Number of negative values: 196\n",
      "Number of negative values: 192\n",
      "Number of negative values: 192\n",
      "Number of negative values: 188\n",
      "0.1 0.1 0.01\n",
      "4 0 tensor(0.0002, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m40       \u001b[0m | \u001b[0m4.0      \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.1      \u001b[0m | \u001b[0m0.01     \u001b[0m |\n",
      "Number of negative values: 243\n",
      "0 Lyapunov Risk= 0.015018913894891739\n",
      "Number of negative values: 266\n",
      "Number of negative values: 253\n",
      "Number of negative values: 238\n",
      "Number of negative values: 225\n",
      "Number of negative values: 214\n",
      "5 Lyapunov Risk= 0.01729271374642849\n",
      "Number of negative values: 208\n",
      "Number of negative values: 203\n",
      "Number of negative values: 203\n",
      "Number of negative values: 204\n",
      "0.08544997108681095 0.01 0.07725949730204996\n",
      "66 1 tensor(0.0087, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m41       \u001b[0m | \u001b[0m66.0     \u001b[0m | \u001b[0m0.08545  \u001b[0m | \u001b[0m0.01     \u001b[0m | \u001b[0m0.07726  \u001b[0m |\n",
      "Number of negative values: 218\n",
      "0 Lyapunov Risk= 0.024634497240185738\n",
      "Number of negative values: 255\n",
      "Number of negative values: 251\n",
      "Number of negative values: 239\n",
      "Number of negative values: 234\n",
      "Number of negative values: 241\n",
      "5 Lyapunov Risk= 0.022124076262116432\n",
      "Number of negative values: 247\n",
      "Number of negative values: 238\n",
      "Number of negative values: 228\n",
      "Number of negative values: 220\n",
      "0.06911629845407473 0.0209554786652312 0.1\n",
      "144 2 tensor(0.0232, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m42       \u001b[0m | \u001b[0m144.0    \u001b[0m | \u001b[0m0.06912  \u001b[0m | \u001b[0m0.02096  \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 215\n",
      "0 Lyapunov Risk= 0.02466670796275139\n",
      "Number of negative values: 233\n",
      "Number of negative values: 234\n",
      "Number of negative values: 211\n",
      "Number of negative values: 213\n",
      "Number of negative values: 214\n",
      "5 Lyapunov Risk= 0.024816714227199554\n",
      "Number of negative values: 214\n",
      "Number of negative values: 212\n",
      "Number of negative values: 219\n",
      "Number of negative values: 236\n",
      "0.031721115466832046 0.014323080904046766 0.09991343617824135\n",
      "104 2 tensor(0.0141, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m43       \u001b[0m | \u001b[0m104.0    \u001b[0m | \u001b[0m0.03172  \u001b[0m | \u001b[0m0.01432  \u001b[0m | \u001b[0m0.09991  \u001b[0m |\n",
      "Number of negative values: 232\n",
      "0 Lyapunov Risk= 0.014260249212384224\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 253\n",
      "Number of negative values: 256\n",
      "Number of negative values: 241\n",
      "Number of negative values: 241\n",
      "Number of negative values: 246\n",
      "5 Lyapunov Risk= 0.013167248107492924\n",
      "Number of negative values: 239\n",
      "Number of negative values: 238\n",
      "Number of negative values: 231\n",
      "Number of negative values: 227\n",
      "0.05817289948190501 0.02080954257631082 0.07693952434710218\n",
      "116 5 tensor(0.0107, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m44       \u001b[0m | \u001b[0m116.0    \u001b[0m | \u001b[0m0.05817  \u001b[0m | \u001b[0m0.02081  \u001b[0m | \u001b[0m0.07694  \u001b[0m |\n",
      "Number of negative values: 224\n",
      "0 Lyapunov Risk= 0.024809155613183975\n",
      "Number of negative values: 263\n",
      "Number of negative values: 253\n",
      "Number of negative values: 250\n",
      "Number of negative values: 237\n",
      "Number of negative values: 218\n",
      "5 Lyapunov Risk= 0.022925948724150658\n",
      "Number of negative values: 209\n",
      "Number of negative values: 203\n",
      "Number of negative values: 199\n",
      "Number of negative values: 198\n",
      "0.08895791547092279 0.016855226331524975 0.09969436392801542\n",
      "90 2 tensor(0.0181, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m45       \u001b[0m | \u001b[0m90.0     \u001b[0m | \u001b[0m0.08896  \u001b[0m | \u001b[0m0.01686  \u001b[0m | \u001b[0m0.09969  \u001b[0m |\n",
      "Number of negative values: 220\n",
      "0 Lyapunov Risk= 0.023135073482990265\n",
      "Number of negative values: 253\n",
      "Number of negative values: 251\n",
      "Number of negative values: 238\n",
      "Number of negative values: 235\n",
      "Number of negative values: 242\n",
      "5 Lyapunov Risk= 0.020837707445025444\n",
      "Number of negative values: 248\n",
      "Number of negative values: 239\n",
      "Number of negative values: 229\n",
      "Number of negative values: 216\n",
      "0.06864415100182901 0.010933503057419818 0.09685019615253451\n",
      "144 2 tensor(0.0216, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m46       \u001b[0m | \u001b[0m144.0    \u001b[0m | \u001b[0m0.06864  \u001b[0m | \u001b[0m0.01093  \u001b[0m | \u001b[0m0.09685  \u001b[0m |\n",
      "Number of negative values: 254\n",
      "0 Lyapunov Risk= 0.017777960747480392\n",
      "Number of negative values: 267\n",
      "Number of negative values: 253\n",
      "Number of negative values: 227\n",
      "Number of negative values: 217\n",
      "Number of negative values: 216\n",
      "5 Lyapunov Risk= 0.02706565149128437\n",
      "Number of negative values: 215\n",
      "Number of negative values: 212\n",
      "Number of negative values: 208\n",
      "Number of negative values: 203\n",
      "0.09995395283150736 0.042282375956214015 0.08235296712685031\n",
      "118 1 tensor(0.0168, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m47       \u001b[0m | \u001b[0m118.0    \u001b[0m | \u001b[0m0.09995  \u001b[0m | \u001b[0m0.04228  \u001b[0m | \u001b[0m0.08235  \u001b[0m |\n",
      "Number of negative values: 219\n",
      "0 Lyapunov Risk= 0.024000301957130432\n",
      "Number of negative values: 241\n",
      "Number of negative values: 229\n",
      "Number of negative values: 233\n",
      "Number of negative values: 231\n",
      "Number of negative values: 231\n",
      "5 Lyapunov Risk= 0.023383401334285736\n",
      "Number of negative values: 230\n",
      "Number of negative values: 229\n",
      "Number of negative values: 228\n",
      "Number of negative values: 231\n",
      "0.04621655585008994 0.06256137178301109 0.1\n",
      "108 3 tensor(0.0159, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m48       \u001b[0m | \u001b[0m108.0    \u001b[0m | \u001b[0m0.04622  \u001b[0m | \u001b[0m0.06256  \u001b[0m | \u001b[0m0.1      \u001b[0m |\n",
      "Number of negative values: 267\n",
      "0 Lyapunov Risk= 0.00482504116371274\n",
      "Number of negative values: 242\n",
      "Number of negative values: 229\n",
      "Number of negative values: 220\n",
      "Number of negative values: 218\n",
      "Number of negative values: 215\n",
      "5 Lyapunov Risk= 0.005282716825604439\n",
      "Number of negative values: 210\n",
      "Number of negative values: 211\n",
      "Number of negative values: 211\n",
      "Number of negative values: 213\n",
      "0.04376509873014594 0.031534881494474275 0.044159332331657516\n",
      "144 0 tensor(0.0053, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m49       \u001b[0m | \u001b[0m144.0    \u001b[0m | \u001b[0m0.04377  \u001b[0m | \u001b[0m0.03153  \u001b[0m | \u001b[0m0.04416  \u001b[0m |\n",
      "Number of negative values: 243\n",
      "0 Lyapunov Risk= 0.008121070452034473\n",
      "Number of negative values: 236\n",
      "Number of negative values: 240\n",
      "Number of negative values: 221\n",
      "Number of negative values: 206\n",
      "Number of negative values: 205\n",
      "5 Lyapunov Risk= 0.008510066196322441\n",
      "Number of negative values: 205\n",
      "Number of negative values: 205\n",
      "Number of negative values: 206\n",
      "Number of negative values: 207\n",
      "0.0340372400018226 0.031114319773297875 0.059228434399184855\n",
      "90 2 tensor(0.0054, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m50       \u001b[0m | \u001b[0m90.0     \u001b[0m | \u001b[0m0.03404  \u001b[0m | \u001b[0m0.03111  \u001b[0m | \u001b[0m0.05923  \u001b[0m |\n",
      "Number of negative values: 258\n",
      "0 Lyapunov Risk= 0.007613749708980322\n",
      "Number of negative values: 241\n",
      "Number of negative values: 228\n",
      "Number of negative values: 226\n",
      "Number of negative values: 218\n",
      "Number of negative values: 217\n",
      "5 Lyapunov Risk= 0.008966578170657158\n",
      "Number of negative values: 215\n",
      "Number of negative values: 210\n",
      "Number of negative values: 211\n",
      "Number of negative values: 210\n",
      "0.0536558348070252 0.045719438354055966 0.05570115044302199\n",
      "136 0 tensor(0.0075, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m51       \u001b[0m | \u001b[0m136.0    \u001b[0m | \u001b[0m0.05366  \u001b[0m | \u001b[0m0.04572  \u001b[0m | \u001b[0m0.0557   \u001b[0m |\n",
      "Number of negative values: 236\n",
      "0 Lyapunov Risk= 0.005077539011836052\n",
      "Number of negative values: 243\n",
      "Number of negative values: 239\n",
      "Number of negative values: 241\n",
      "Number of negative values: 243\n",
      "Number of negative values: 242\n",
      "5 Lyapunov Risk= 0.0029464317485690117\n",
      "Number of negative values: 244\n",
      "Number of negative values: 238\n",
      "Number of negative values: 234\n",
      "Number of negative values: 231\n",
      "0.049114401135819524 0.04379726935543336 0.030627610021438934\n",
      "2 0 tensor(8.1687e-05, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m52       \u001b[0m | \u001b[0m2.0      \u001b[0m | \u001b[0m0.04911  \u001b[0m | \u001b[0m0.0438   \u001b[0m | \u001b[0m0.03063  \u001b[0m |\n",
      "Number of negative values: 253\n",
      "0 Lyapunov Risk= 0.004872307181358337\n",
      "Number of negative values: 216\n",
      "Number of negative values: 248\n",
      "Number of negative values: 235\n",
      "Number of negative values: 227\n",
      "Number of negative values: 226\n",
      "5 Lyapunov Risk= 0.005018813535571098\n",
      "Number of negative values: 228\n",
      "Number of negative values: 229\n",
      "Number of negative values: 229\n",
      "Number of negative values: 232\n",
      "0.03957757289913368 0.014416220164119567 0.04586571916158343\n",
      "108 0 tensor(0.0035, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m53       \u001b[0m | \u001b[0m108.0    \u001b[0m | \u001b[0m0.03958  \u001b[0m | \u001b[0m0.01442  \u001b[0m | \u001b[0m0.04587  \u001b[0m |\n",
      "Number of negative values: 263\n",
      "0 Lyapunov Risk= 0.007820012979209423\n",
      "Number of negative values: 245\n",
      "Number of negative values: 240\n",
      "Number of negative values: 236\n",
      "Number of negative values: 233\n",
      "Number of negative values: 228\n",
      "5 Lyapunov Risk= 0.006584337446838617\n",
      "Number of negative values: 227\n",
      "Number of negative values: 227\n",
      "Number of negative values: 227\n",
      "Number of negative values: 227\n",
      "0.06287753743683126 0.061662337281211746 0.0480056960410474\n",
      "2 9 tensor(0.0002, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m54       \u001b[0m | \u001b[0m2.0      \u001b[0m | \u001b[0m0.06288  \u001b[0m | \u001b[0m0.06166  \u001b[0m | \u001b[0m0.04801  \u001b[0m |\n",
      "Number of negative values: 226\n",
      "0 Lyapunov Risk= 0.010941926389932632\n",
      "Number of negative values: 240\n",
      "Number of negative values: 247\n",
      "Number of negative values: 249\n",
      "Number of negative values: 249\n",
      "Number of negative values: 249\n",
      "5 Lyapunov Risk= 0.006771409884095192\n",
      "Number of negative values: 249\n",
      "Number of negative values: 249\n",
      "Number of negative values: 249\n",
      "Number of negative values: 249\n",
      "0.08243164439146669 0.0694178156266787 0.03492757507749461\n",
      "2 0 tensor(0.0002, grad_fn=<AddBackward0>)\n",
      "| \u001b[0m55       \u001b[0m | \u001b[0m2.0      \u001b[0m | \u001b[0m0.08243  \u001b[0m | \u001b[0m0.06942  \u001b[0m | \u001b[0m0.03493  \u001b[0m |\n",
      "=============================================================\n",
      "Final result: {'target': 154.0, 'params': {'min_a': 0.1, 'min_b': 0.01, 'min_c': 0.1}}\n",
      "Time consumed: 63.101063114008866\n"
     ]
    }
   ],
   "source": [
    "from sklearn.datasets import make_classification\n",
    "from sklearn.model_selection import cross_val_score\n",
    "from sklearn.ensemble import RandomForestClassifier as RFC\n",
    "from sklearn.svm import SVC\n",
    "\n",
    "from bayes_opt import BayesianOptimization\n",
    "from bayes_opt.util import Colours\n",
    "\n",
    "\n",
    "def optimize_region():\n",
    "    \"\"\"Apply Bayesian Optimization to stability region of LNN.\"\"\"\n",
    "\n",
    "    optimizer = BayesianOptimization(\n",
    "        f=search_LNN_p,\n",
    "        pbounds={\"min_a\": (1e-3, 1e-1), \"min_b\": (1e-2, 1e-1), \"min_c\": (1e-2, 1e-1)},   \n",
    "        random_state=1234,\n",
    "        verbose=2\n",
    "    )\n",
    "    optimizer.maximize(n_iter=50)\n",
    "\n",
    "    print(\"Final result:\", optimizer.max)\n",
    "\n",
    "start = timeit.default_timer()\n",
    "\n",
    "if __name__ == \"__main__\":\n",
    "\n",
    "    print(Colours.yellow(\"--- Optimizing Number of points ---\"))\n",
    "    optimize_region()\n",
    "end = timeit.default_timer()\n",
    "print(\"Time consumed: \" + str(end - start))\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Optimal LNN obtained and its performance"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "#optimal parameters\n",
    "min_a = 0.1\n",
    "min_b = 0.01\n",
    "min_c = 0.1\n",
    "pho_obtained  = 0.0308"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of negative values: 233\n",
      "0 Lyapunov Risk= 0.02545110508799553\n",
      "layer1.weight: torch.Size([4, 2])\n",
      "tensor([[0.2622, 0.1000],\n",
      "        [0.2939, 0.1000],\n",
      "        [0.4021, 0.1000],\n",
      "        [0.2000, 0.1000]])\n",
      "layer2.weight: torch.Size([1, 4])\n",
      "tensor([[0.1000, 0.2000, 0.1000, 0.1000]])\n",
      "layer3.weight: torch.Size([1, 1])\n",
      "tensor([[0.0100]])\n",
      "layer4.weight: torch.Size([1, 1])\n",
      "tensor([[0.1000]])\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 = 1\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:  261\n",
      "number of \\dot V > 0 points:  139\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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WLFjB6NGjWLo0KEce+yx/Pvfe7ybyaGkBNq0Yb9u3Ti4rIzVn3/OS/PnM/icc6ioqOCnP/0p//nPfwAYOXIk8+fPd2UlTJ48mYE/+QlHn38+X9TV8cbChcx67TWumj6dQWefzYoVK5g+fTqHHXYYRxxxBGedleB9TQGPX4vw8iCZLjgT1A8A05LUOYWmk9nv+Gm7RbmeHnpItbzcuaMoL/c/ExinV5GM6JvN4ZLh/r1otCvF/tNt/uOPV4fav5fe6zyGdPoaefzxat/6RL9Gdf363b9sFy5snEj1PdHuoU90R7F161YdOHCgHnLIIXrJJZfo3LlzG/XlPXvquuiyqtbV1amqan19vR5//PG60P2VXV5erjfccIOqqt5///16yne/q7pwoXNH8ctfqq5fryeccIJ++OGHqqr61ltvaWVlZaMNTe4oLrtMdf16XfHMM7pv1666Zs4cLTvgAF22bJmqqo4dO1Zvu+02VVU9/vjjdd68eaqqCuisWbNUVfWqiRP1d5dd5txRnHqqzrz33sa+evbs2Xj3tHHjxsQnxd3/LWvXNjl+XgS9owjTURyD82L393HeebwA5ymnCTivVow6kzuAFTiv3Gx22Elb6ihcWmOyr0xidgWjNdiV0FF4kOqwmNdTT/X19VpdXa3XXnutlpaW6n333ef2FzOkpKp33nmnDh48WCsqKrRHjx76yCOPNNZbsWKFqqru2LFDu3Xrpqqq1113nd566626detWLS4u1oEDBzaW/v3772HXfffdpz169NCBAwfq8OHD9R//+IcuWLBAjz322Ma6L774op522mmq2tRRtGvXThsaGlRVtaqqSi+44AJVVT3vvPN05syZjfqTTjpJf/SjH+mDDz6oW7dubdHxiieoowgt4E6dCeqk0Tqu8ZdlxyLDMNLNJ58EW++Xtm3bMnLkSEaOHElFRQX3338/48aNa1Ln448/ZsqUKcybN4+uXbsybty4JvEDsXEg8YGDDQ0NdOnShQULFjRry5lnnsntt9/euLxw4UJf+7DXXns19tu2bVvq6+sT1nvmmWd49dVXeeqpp7jxxhtZtGgRRUXZvXRbrifDMDLGN74RbL0fli1bxkcffdS4vGDBAsrLywHo2LEjW7duBWDLli106NCBzp0788UXX/Dss882aefRRx9t/DtixIgm2zp16sRBBx3EzJkzAWfk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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:  154\n",
      "number of points beyond levelset:  246\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:  261\n",
      "number of \\dot V > 0 points:  139\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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WLFjB6NGjWLo0KEce+yx/Pvfe7ybyaGkBNq0Yb9u3Ti4rIzVn3/OS/PnM/icc6ioqOCnP/0p//nPfwAYOXIk8+fPd2UlTJ48mYE/+QlHn38+X9TV8cbChcx67TWumj6dQWefzYoVK5g+fTqHHXYYRxxxBGedleB9TQGPX4vw8iCZLjgT1A8A05LUOYWmk9nv+Gm7RbmeHnpItbzcuaMoL/c/ExinV5GM6JvN4ZLh/r1otCvF/tNt/uOPV4fav5fe6zyGdPoaefzxat/6RL9Gdf363b9sFy5snEj1PdHuoU90R7F161YdOHCgHnLIIXrJJZfo3LlzG/XlPXvquuiyqtbV1amqan19vR5//PG60P2VXV5erjfccIOqqt5///16yne/q7pwoXNH8ctfqq5fryeccIJ++OGHqqr61ltvaWVlZaMNTe4oLrtMdf16XfHMM7pv1666Zs4cLTvgAF22bJmqqo4dO1Zvu+02VVU9/vjjdd68eaqqCuisWbNUVfWqiRP1d5dd5txRnHqqzrz33sa+evbs2Xj3tHHjxsQnxd3/LWvXNjl+XgS9owjTURyD82L393HeebwA5ymnCTivVow6kzuAFTiv3Gx22Elb6ihcWmOyr0xidgWjNdiV0FF4kOqwmNdTT/X19VpdXa3XXnutlpaW6n333ef2FzOkpKp33nmnDh48WCsqKrRHjx76yCOPNNZbsWKFqqru2LFDu3Xrpqqq1113nd566626detWLS4u1oEDBzaW/v3772HXfffdpz169NCBAwfq8OHD9R//+IcuWLBAjz322Ma6L774op522mmq2tRRtGvXThsaGlRVtaqqSi+44AJVVT3vvPN05syZjfqTTjpJf/SjH+mDDz6oW7dubdHxiieoowgt4E6dCeqk0Tqu8ZdlxyLDMNLNJ58EW++Xtm3bMnLkSEaOHElFRQX3338/48aNa1Ln448/ZsqUKcybN4+uXbsybty4JvEDsXEg8YGDDQ0NdOnShQULFjRry5lnnsntt9/euLxw4UJf+7DXXns19tu2bVvq6+sT1nvmmWd49dVXeeqpp7jxxhtZtGgRRUXZvXRbrifDMDLGN74RbL0fli1bxkcffdS4vGDBAsrLywHo2LEjW7duBWDLli106NCBzp0788UXX/Dss882aefRRx9t/DtixIgm2zp16sRBBx3EzJkzAWfk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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:  154\n",
      "number of points beyond levelset:  246\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYoAAAEWCAYAAAB42tAoAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuNSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/xnp5ZAAAACXBIWXMAAAsTAAALEwEAmpwYAAA6K0lEQVR4nO2de5wU5ZX3v2eGmzCIMJpZw+igYkQZYGAIiJcwrCYSzMpqzBsVFTaaUVjjZrNko+u7Gk3cxNdZYkyihhhjFJToosYYxetM0KgJjHIRQUUEHGTlLjRIFOa8f1T10Mx01VT1rbqnz/fzeT7dVfX86jld1dNn6vI7JaqKYRiGYXhREnUAhmEYRn5jicIwDMPwxRKFYRiG4YslCsMwDMMXSxSGYRiGL5YoDMMwDF8sURiGYRi+WKIwigoRWSAiNyWZP1lE/ldEuqWx7p4ico+I7HTX9Z1O+v+r22+nq+uZsKxRRDa7y5aKyOR22otEZJ2I7BaRx0RkQMKyQSLypIhsd9f/83Q+l2FYojCKjd8CF4uItJt/CTBXVfeFWZmIVCRMfh84HqgCJgD/LiITPXRnAdcAZ7j9jwVuTOjyL8CRqnooUA/MEZEjXe1Q4JduzBXAHuCOBO0dwCbgSKAGGA/MCPO5DCMRSxRGsfEYUA6cHp8hIv2BrwD3BVmBiPQWkUtE5AWgMWHRVOAHqrpdVVcCvwKmeaxmKvBrVV2hqtuBHyT2VdVlCUlLge7AUe70FOAPqrpQVWPAfwLniUhfd/kxwEOquldV/xdYAAwN8tkMIxmWKIyiQlU/Bh4CLk2Y/X+AVaq61E8rIuNE5FfABlf/a6DWXdYf5z/4xHUsxfsHemiSvhUiUp4w3hMishf4C9AELE6mVdV3gU+Az7mzbgMucBPaQODLOMnCMFLCEoVRjPwWOF9EernTl7rzkiIi/0dEVgH3Au8Bw1T1i6o61008AGXu60cJ0o+AviSnLElfEvur6lfc6UnAM6ra6qFtP9ZCnGSyE2jBSTCPeX0+w+gMSxRG0aGqLwFbgH8UkeOAMcADPpJKYCCwBOc/+f9N0ifmvh6aMO9QYJfHOmNJ+tK+v6p+qqpPAV8SkXM8tG1jiUgJztHDI0Af4HCgP3CLRxyG0SmWKIxi5T6cI4mLgadV9UOvjqo6CydRPA9cB7SIyE9EZGRCn+3ARmBEgnQEsMJjtSuS9P1QVbd69O8GHJdMKyLHAj2Bt4EBwNHAz1X1b+76foNzVGIYKWGJwihW7gPOBL6Jz2mnOKq6U1Vnq+opOHcR7QX+ICLPt1vn/xWR/iIyxF33vT7jXyYiJ4nIYcD/jfcVkSEi8mUROUREuovIxcAXgD+52rnAP4jI6SLSB7gJeERVd6nqFpzTY9NFpJu77qnAsmCbxTCSoKrWrBVlw7lAvB3omaK+BBiXMN0TuAfn2sCHwHcSlh2Nc8ro6IR533H77cT5r7+nO/9EnAvYu4AdwCLg3HZjXwSsB3YDvwcGJCyrSfhsW3Au3ldEvb2tFW4TVXtwkWEYhuGNnXoyDMMwfLFEYRiGYfhiicIwDMPwxRKFYRiG4UuXrCh5+OGH66BBg1LS7t69mz59+mQ2oAxgcYXD4gqHxRWOrhhXc3PzFlU9IunCqG+7ykarra3VVGlsbExZm00srnBYXOGwuMLRFeMCFqvHb6qdejIMwzB8sURhGIZh+GKJwjAMw/DFEoVhGIbhiyUKwzAMw5fIEoWIHOU+QP5NEVkhIv+SpI+IyO0islpElonIqGzFM2PGXLp1G0RzczPdug1ixoy5KelFSkyfgn7uXBg0CEpKnNe54eQd9Nu2padPd/xC02/blp4+6u+P6dP7/eoUr9uhst1wHhs5yn3fF6eW/knt+kwCngIEOBn4S5B1h709dvr0OQq9FdCGhgYFFHrr9OlzQusPtMzq/W57y8X4XjQ2NqY9/pw5qr17q8KB1ru3Mz9V/axZjWnp0x3fS59sP+ZyfC/9rFmNKeuz+f0LcrtnFN//xLii/Ptrr0/l9ysOPrfHRu55aAvEKZX8xXbzfglcmDD9FnBkZ+sKmyhKS6vadtCBDY2WllaF1ie2TOr9/mByMb4XjY2NaY9fVXXwj1y8VQWTJ9U3NDSmpU93fC99sv2Yy/G99A0NjSnrs/n9C5Ioovj+J8YV5d9fe30qv19x/BJFXpQZF5FBOM/5rVbVnQnznwB+rM6jK3EfEvM9VV2cZB31QD1ARUVF7bx58wKP39zc3Pa+srKSlpaWtuna2tpQ+vZkSh+LxSgrK0vaJxfjexGLxXjrrbfSHN97WQB5Un1lZYyWlrKU9emO76VPth9zOb6XPr69UtNn7/t3wgkneH7vczG+lz5xP0b599den8rvV5wJEyY0q+ropAu9MkiuGs6D4puB85IsewI4LWH6eWB0Z+u0Iwo7orAjCjuisCOKYPo45KszW0S6A/OBuar6SJIuG4CjEqYr3XkZpb7+ZqB3u7m93fmmz7b+5puhdzt5797O/FT1JSXp6dMdv9D0Je1+CcLoo/7+mD49fSC8Mki2G84F6vuA23z6nM3BF7P/GmTdqdR6mj59jpaWVmlDQ4OWllaFvhAU14NkRd/Zf1bZHt+LeFzpjj9njvMfrIjzGvRCqpd+/vzGSMf30nvtx1yN78X8+Y1p6bP1/QtauyjX3//2cUX199den+rvl6r/EUWUieI0nEOkZcASt00CrgSu1APJ5BfAu8ByApx20hQTRZyuWOwrm1hc4bC4wmFxhSNbRQEjKzOuzgVq6aSPAv+cm4gMwzCMZJgz2zAMw/DFEoVhGIbhiyUKFyvhYSU8CrkEh5XwMH2XLOGRzWYlPHIXv5XwsBIeVsLDSngUZDPDnRnuzHBnhjsz3AXTx/FLFHbqCdi/f32o+abPrH69Rzev+abPL33U3x/Tp6cPgiUKoLT06FDzTZ9Z/dEe3bzmmz6/9FF/f0yfnj4IliiI3kJf7Hor4RG93kp4FK8+EF7npAq5WQkPK+FhJTzC6a2Eh5XwwC5mB6crWvOzicUVDosrHBZXOLJVwsNOPRmGYRi+WKIwDMMwfLFE4WLO7PT0kVuLC9yaHfXmM2d219CbMzuL1yjMmZ2eMzsvrMXt9I2zZkU6vpfenNnh9ObMDqc3Z3YWE4U5s1PXNzY25oe1uJ22saEh0vG99ObMDqc3Z3Y4vTmzs0jUzshC10duDS5wfYGHH/n3z/Rd3JktIveIyCYRecNjeZ2IfCQiS9x2fTbiiNoZWej6yK3BBa4v8PAj//6Zvus7s+8FJnbS50VVrXHbTdkIImpnZKHr88JaXMDW7HzYfObMLl59ILzOSeWqAYOANzyW1QFPhF2nObNz78yO3FrcTt84f36k43vpzZkdTm/O7HD6bDmzxVkeHSIyyE0G1UmW1QHzgRbgA2Cmqq7wWE89UA9QUVFRO2/evJTiicVilJWVpaTNJhZXOCyucFhc4eiKcU2YMKFZVUcnXeiVQXLV8D+iOBQoc99PAt4Jsk4r4ZE7LK5wWFzhsLjCUZQlPFR1p6rG3PdPAt1F5PCIwzIMwygq8jpRiMjfiYi478fgxLs1G2Ol62yM2lmbrj7yAIrcmR21s9uc2V1D3yWd2cCDwEbgU5zrEJcBVwJXusuvAlYAS4FXgVOCrDfXzuxcOGv9DinTHT+dFZgzO5w+6X5Mc3xzZpsz25zZKbRcO7Nz4az1+4NJd/x0VmDO7HD6pPsxzfHNmZ298b0wZ3YRUuzPfI48ANOnpY86/Kidxabv4s7sfKHYn/kceQCmT0sfdfhRO4tN3/Wd2XlBPj7zOZf6yAMocmd2Pji7zZldvPpAeJ2TKuQWhTM7287azs7Vpjt+qiswZ3Y4ved+THN8c2abM9uemZ2DRBGnKxppsonFFQ6LKxwWVziK0nBnGIZhRI8lCsMwDMMXSxSGYRiGL5YoXOIlEJqbC7KCg5XwKPISHlHX8MiXEhbFru+SJTyy1cJezE4sgRB3qEZcgcJKeFgJj5zG3zhrVsp6K+FhJTwKsoVNFIklEBJLGeRTBQgr4RFOX0wlPDKhb2xoSFlvJTxyP76X3kp4ZJGoSyBErY88ANMXtD7qEhSmtxIeOSHqEghR6yMPwPQFrY+6BIXprYRHTijwCg5WwqPIS3hEXcMj6hIUprcSHjm5RqF6oARCQ0NjPlSgsBIeVsIjK+N70Th/flp6K+FhJTwKrlkJj9xhcYXD4gqHxRWOLlnCQ0TuEZFNIvKGx3IRkdtFZLWILBORUbmO0TAMo9iJ+hrFvcBEn+VfBo53Wz1wZw5iMgzDMBKINFGo6kLAz0I7GbjPPTJ6FThMRI7MRiyF7syOPIB805szO5zenNldQp8tZ7Y4p6aiQ0QGAU+oanWSZU8AP1bVl9zp54HvqeriJH3rcY46qKioqJ03b17gGLZtg3XroLUVKitjtLSUUVICVVUwYEA4fZxM62OxGGVlZdEF4EEsFqPsk08iG99LHzvqKMq6dcufHRiPK9l+jHD/xfWxffsoe//9lPTr129j8+Z1QML4lHDEEVUcfXR6+gEDenh/73Mwvpc+cT9GMb6XvrKykpaWllD6OBMmTGhW1dFJF3pdvMhVAwYBb3gsewI4LWH6eWB0Z+ssNmd2lM5ec2aH05szO5zenNnh9MXqzN4AHJUwXenOyygFboyNPgDTF7U+amex6c2Z/ThwqXv308nAR6q6MdODFLgxNvoATF/U+qidxabv4s5sEXkQeAU4QURaROQyEblSRK50uzwJrAFWA78CZmQjjnwwxpqz2pzZkerNmV20+kB4nZMq5FaMzuyonL3mzA6nN2d2OL05s8PpzZmd5UQRpys6LrOJxRUOiyscFlc4uqQz2zAMw8h/LFEYhmEYvliiMAzDMHyxROFiJTy6mN5KeITTp1nCI93x86UERqHrs1XCI/ILz9loYS9mJz6bPu7MzvGz7TvV+16kykUAHjQ2NkY6vpe+cdas/NqBLkn3Yx5sv8ZZsyIbf/r0OQq92zmTe+v06XMCXZz106c7vheJcUUxvpf+gDM7uD4OdteTP1bCI3W9lfAIp++KJTzSHd9KeGROX6wlPHJCgVdQiD4A05s+DX3UJSyKXR8ESxREXgHBSniYvqj1UZewKHZ9ECxRkB8VFKyEh5XwiFSfRgmPdMePuoRFsesD4XVOqpCblfDIQgAeWAmPcPquWsIj3fGthIeV8CiIRBGnK1rzs4nFFQ6LKxwWVzishIdhGIYRCZYoDMMwDF8sUbiYM7uL6c2ZHU4fsTPbnN3mzPZswETgLZwHE12TZPk0YDOwxG2XB1mvObNz9wHMmR1Ob87scHpzZofTdzlnNlAKvAscC/QAlgInteszDfh52HWbMzt3H8Cc2eH05swOpzdndjh9V3RmjwFWq+oaVf0EmAdMjiKQAje2Rh+A6U0foT5qZ3Oh64MgTiLJPSJyPjBRVS93py8BxqrqVQl9pgE/wjn99Dbwr6r6vsf66oF6gIqKitp58+YFjmX5cvjkE+d9ZWWMlpYyAHr0gGHDwukTyaQ+FotRVlYWXQAexGIxyt57L7LxvfSxykrKNm3Knx0YjyvZfoxw/8X1sc98hrKWlsjG99LHjjnG+3vv0ty8HEiipwe1tZ2Pn4o+cT9GMb6XvrKykpa2/RhMH2fChAnNqjo62bJugdcSDX8AHlTVv4nIFcBvgb9P1lFVZwOzAUaPHq11dXWBB9mwAerrYc8eaGhoYubMOnr3htmzIchqEvVxMq1vamrC8zPlIgAPmpqaqBswILLxvfRNs2ZR95nP5M8OjMeVbD9GuP/i+qYPPqBu5sxA+k8//ZSWlhb27t3rzCgrg61bnRNGcUSgvBz69Ol8fB99v9JSevXq5SsfO7aMWGwrzhmXthVQVlZOeXnn46ei79evX1tcUYzvpe/fvz/9+vXz1ffq1YvKykq6d+/e6brb8Donle0GjAOeTpi+FrjWp38p8FGQdZszOwsBeGDO7HD6ruDMXrNmjW7evFlbW1sPzNyyRXXpUtVFi5zXLVvCxe+h37lzZyD52rVbdNGipbpo0SJdtGiprl0bbvyw+vZx5Xp8L/3GjRt99a2trbp582Zds2ZNh2Xk6cXsbsAa4BgOXMwe2q7PkQnvzwVeDbJuc2bnDosrHF0hrjfffPPgJJFFgiaKXFPIcbW2tuqbb77ZYb5foojs1JOq7hORq4CncY4W7lHVFSJykxvw48DVInIOsA/YhnMXlGEYESMiUYdgpEgq+y7SaxSq+iTwZLt51ye8vxbnlJRhGIYREebMdil0Z7Y5u82ZHakze+tWWLYMFi92Xrduzaq+tLSUmpoaRowYwahRo3jqqZdzOXwbTU1NfOUrX+mg//3vm+jXrx81NTWceOKJ3Hjjjb7rmTnzembPfs53/KamJl5++WXf+HfvTu3zd4rXOalCbsXmzE53fHNmmzM7jDO7w/ntLVtUm5udC9Hx1tzcdkG60+vsPnqvc+59+vRpe//QQwt01KgveA3fKZ2En5R4XI2NjfrFL57dQf/LXzrzVVVjsZgOHjxYm5ub0xr/hhtu0FtvvdVXv3HjzkDxh71GYUcUwHXXHXxnITjT111XHPrIA8iGvrW1sOPPtb61NXX9hg0d9a2tsGEDc+c6d+6uW+dkoHXrnOmDDlh89EFYt24nffv2b5u+//5bufjizzNu3HBuuOEGAK6//npuu+22tj7XXXcdP/3pT1FV/u3fvsvXvlbNBRcM45lnfgfAokVNTJxYx/nnn8+QIUOYMmVK/KYaFixYQG1tLaNGjeKRRx5h796O4atC/O7hPn36UFtby+rVq1myZAknn3wyw4cP59xzz2X79u1s2ADXXz+N55//HwDOOWcQd955A6edNophw4axatUq1q5dy1133cVPfvITampqePHFF3n44Yeprq7mlFNGcPnlX0h18wUi330UOaHAjanm7DZ9tPpkZjl3vl8OmzKlc70XH3/8MTU1Nezdu5eWlo3ceecLALz66jOsX/8Ov/3tX1FVbrrpHBYuXMg3vvENzjvvPL797W/T2trKvHnz+Otf/8ojjzzCypVLeOCBpezYsYWpUz/PqFHOj+7Kla/z6KMr+OxnP8upp57Kn//8Z0aPHs03v/lNHn/8cWpqavj617+OeniW4/O3bt3Kq6++yn/+539y4YUX8rOf/Yzx48dz/fXXc+ONN3Lxxbd10B522OHcf/9r/PWvd9DQ0MDdd9/NlVdeSVlZGTNdv8uwYcN4+umn2bhxILt27Qiz+UJjRxRE/sjgyPWRB2D6wtb36OE5P1AO8tF7ccghh7BkyRJWrVrFnXcu4IYbLkVVefXVZ/jLX55hypSRXHrpKFatWsU777zDoEGDKC8v5/XXX+eZZ55h5MiRlJeX89JLLzFp0oWUlpZSXl7BqFHjWbFiEQDDho2hsrKSkpISampqWLt2LatWreKYY45h8ODBiAgXX3wxXjcRLVnyIiNHjuRLX/oS11xzDZWVlezYsYPx48cDMHXqVBYuXJj0Y06YcB49ekBtbS1r165Nuv5TTz2VadOm8Yc//Ir9+/eH2XyhsURBfjyy2J65bc/MjlSfzjOzBw7sqC8pgYEDg+UgH30QJk0ax0cfbWH79s2oKtOmXcu8eUv485+XsHr1ai677DIALr/8cu69915+85vf8I1vfKNN379/x+FFoG/fnm3TpaWl7Nu3L+n4vXol148bdzqvv/46zc3NXHnllZ7xDxxIh2TTs2dPBg70H/euu+7ihz/8Ibt3v8+ll9ayY8eBK9ghNl8gQiUKESkRkUMzN3x+MGWKU62gqsqZrqpyptsOjUPoRQpPH3kA2dBXVRV2/LnWx7dZKvryckcT/xe2Rw9nurw8WA7z0Qdh8+ZVwH6OOKKccePO4g9/uIfy8hjl5bBhwwY2bdoEwLnnnsuCBQtYtGgRZ511FgCnn346f/zj76is3E8stpnXX1/IyJFjqKhI/h/5kCFDWLt2LWvWrAHgwQcfbAs3Mfxk+n79+tG/f39efPFFAO6//37Gjx9PeblTxaSbeyFABI46quPH79u3L7t27Wqbfvfddxk7diwNDTdRUXEE27a9n8rmC4bXVe54Ax4ADgX6AG8CLcB3O9NF2cyZnTssrnB0hbiS3THjRzrVRbzueiopKdERI0boiBEjdPjw4frEE0+0Lbvtttu0urpaq6ur9eSTT9bVq1e3Lbviiiv0e9/7Xtt0a2urzpw5U4cOHarV1dU6b948VXW2x9lnn93W75//+Z/1N7/5jaqqPvXUU3r88cfryJEj9eqrrz6oX5z2+jivv/66jh07VocNG6aTJ0/Wbdu2qarq1KlT9eGHH1ZV1aqqKt28ebOqqi5atEjHjx+vqqpvvfWWDhs2TEeMGKELFy7Uc889V6urq3Xo0KF69dVXa2tra2DHeNi7noIkiiXu6xTgv4HuwLLOdFE2SxS5w+IKR1eIK2yiSIdMlsrYv3+/jhgxQt9+++2011XIJTxUs3N7bHcR6Q78I/C4qn7KwWUODcMw8po333yTwYMHc8YZZ3D88cdHHU7BEeT22F8Ca3GK9i0UkSpgZzaDMgzDyCQnnXRS23UFIzydHlGo6u2qOlBVJ7lHKOuACTmILadYCY8C/wBWwqOoSnhkWl/g4We9hIfnEYWIXKyqc0TkOx5dZmU2lOiIu0fjxqC4exSC3fhR6PrIA8iGft06Z36hxp9r/aZNji4V/datjiZuT/7kkwPrCnLrjZ8+iBkgzfGzGX4h6IPgd0QRfzRSX4/WZciHCgpWwsNKeESqz1IJj0LQF3j4aeuD4HlEoaq/dF87lD0UkQx6/qIn6goIUesjD8D0ha1PoQRHPukLPPy09UHo9BqFiDSJyKCE6c8DizIXQvREXQEhan3kAZi+sPUplOBIV19WVhZKP23aNI455hhqamoYNWoUr7zyiq/8G984pdPwb7vtNvbs2RPFx8+oPghBbo/9EbBARGaIyM04d0H9UyYGF5GJIvKWiKwWkWuSLO8pIr9zl/8lMWFlknyooGAlPKyER6T6LJXwADq/0J5mCY+g+ltvvZUlS5bw4x//mCuuuMJXfu+9L3c6fDxR5Cj8rOkD4WWwSGxAHfApsBH4uyCaAOssBd4FjuXAM7NPatdnBnCX+/4C4HdB1p2K4S7uHm1oaIzi2fad6jszRKU7fqoraIsr2xsgpL5x/vxIx/fSe+7HiLdf4/z5gfVJDXdbtqguXeo8FGHp0gMPQwj6rAwPfWfPo2hsbNTx48frV//hH/SEQYP0orPO0tYlSzo8jCHR+fzxxx/rIYccoqqq//3f/61Dhw7VIUOG6ne/+5O24Tus/6tf1RNOOEEvuugibW1t1VtuuUW7d++u1dXVWldXpx9+uE/POWeqHnvsUB08uFp/8INZfps78OYLq9+4cWcgfTac2f8JLAfGAVcAq4CzO9MFWO844OmE6WuBa9v1eRoY577vBmwBpLN1mzM7d1hc4egKcYVyZldVHZwk4q2qKpA8SKI49NBD9f3339f9+/frySefrC+++GKH/omJ4qGHHtIxY8bo4sWLtbq6WmOxmO7atUtPOukkfe211wKtf+fOnQeV2li8eLGeeeaZbeNt37490OfLNNlyZgcx3JUDY1T1Y+AVEVkA3A38MZUjmAQGAu8nTLcAY736qOo+EfnIjWdL+5WJSD1QD1BRUUFTU1NKQcVisZS12cTiCofFFY4wcfXr1++g4nR+lK1fT7Iq3Lp+PbEA69i/f7/nWLt27WLPnj3U1tbSr18/du/ezdChQ1m5ciUjRow4qO+nn37KzJkzuemmmzj88MO5/fbbee6555g0aRKt7i1DZ599Ns8++yyDBw/udP3V1dWoKrFYjJ49e3LEEUewevVqrrjiCs466yzOOOOMwNsok/htr0T27t0b6nvYaaJQ1W+3m14HfDHwCDlCVWcDswFGjx6tdXV1Ka2nqamJVLXZxOIKh8UVjjBxrVy5kr59A94hf/TRB27qT0COPjrQOnbt2uXZr2/fvvTu3ZvevXu39enVqxfdu3fvoOnevTsNDQ2cf/75bfNeffVVevbs2da3Z8+e9OrVq23ab/2lpaWICGVlZfTt25e+ffuyfPlynn76ae677z6eeOIJ7rnnngAbKLP4ba9EevXqxciRIwOvN8hdT0eISIOIPCkiL8Rb4BG82QAclTBd6c5L2kdEugH9gEw/NhwwZ3bkAZgzu2s6s4NeaM+UNXnTJnj//UD6008/nccee4w9e/awfv1uHnzwUcrLT2fZsmBDJpb9fvvtLSxb1kpV1Ve55JIfsmjRaymFX3DO7ATmAr8DzgauBKYCmzMw9iLgeBE5BichXABc1K7P4+54rwDnAy+459IySj4YY82Zbc7sLunMjuuvu87xZRx9tJMkEtebrjM73j+u37cvkDV51KhRTJs2jdraMXz6KZxzzuWccMJIPvnEWVVnP7b19fVMnDiRI474LFdddRvf//4/tZ3G+ta3fsTWrV3HmS2d/e6KSLOq1orIMlUd7s5bpKqfT3twkUnAbTh3QN2jqjeLyE04F1UeF5FewP3ASGAbcIGqdlrZa/To0bp48eLAcQwadGDDNjQ0MXNmHeA8/MPjKYSe+kQyqfc7NZDu+OmsoKmpibpp07K/AULqmxoaqPvZz/JnB8bjSrYfczi+l77pW9+izn0Wc2f6lStXcuKJJx6YsWxZcndXjx4wfHjn4/vodx1zTOenUtIcPxV54imeLH780PrKyl20tPTtVN9hH9L2Wz86Wf8gRxSfuq8bReRs4ANgQABdp6jqk8CT7eZdn/B+L/C1TIzlR9TG1qj1kQdg+sLWR20tNmd2WvogBDHc/VBE+gH/BszEuePpXzMXQvREbWyNWh95AKYvbH3U1uI09QUefn44s1X1CVX9SFXfUNUJqlqrqo9nLoToyQdjrDmzzZkdqT6bzuw81xd4+PnjzI434LUw/aNq5sw2Z7Y5s/PEmR2UkM7sTI8fVt4+rix9/ND6nDuzca4dDGo373Wv/vnUzJmdOyyucHSFuAr1mdmZpNDjyuQzs38DPCMi17nPzIb03diGYRhGgeGZKFT1YWAUcCiwWERmAttE5Ds+T70zDMPIKmvXrqW6uvqged///vdpaGgIva4dO3Zwxx13pDRmnLq6OsLcju9X8jwZp5xySqfrjFeyzRadXcz+BNgN9KQLP+HOMIzsMXfuXAYNGkRJSQmDBg1ibuiyAdkjaKLINF4lz5Px8ssvd7q+yBKFiEwElgC9gVGqeoOq3hhvWYsoIoq9hEeXKwFiJTzC6bNUwmPu3LnU19ezbt06VJV169ZRX1/fMVmkW8Nizx7YuBEWL6bu85/ne1dfzZgxY/jc5z7Hiy++CMCKFSsYM2YMNTU1DB8+nHfeeYdrrrmGd999l2HDapg27bssXBhj7NgzGD58FMOGDeP3v/992xD79u1jypQpnHjiiVxyySUH/TDv2OGE/fOfP8OIEeMYPnwUX/va14jFYr5hf+ELX2D16tVs3QozZ87iuOOqGTy4mptvvq2tT/whTXGz5vnnn8+QIUOYMmUKqsrtt9/OBx98wLhxE5g48Wxef30/F144jerqaoYNG8ZPfvKTcNsyGV4XL4AXgaFey/O5hb2YnVgyv6Gh0bNkfhC9X8n9dPR+FxtzMb4XjY2N0QbgoW+cNSu/dqBL0v2YB9uvcdaswPoOF0K3bFFtbnZu2Ym35mbVLVu0qqpKgQ6tKrHMuI8+2cXZ9957T4cOHXqQ/ob6er316qtVFy3S8aNG6XemTFHdskX/+Mc/6hlnnKGqqldddZXOcT/T3/72N92zZ4++9957OmTI0LbhX3nlU21s/Eibm1XfemuzHnfccdra2qrvvfeeAvrSSy+pqurFF1+st956q6qqnnLKeL3//kX67LObdeTI03Xhwpg2N6tef/2P9cYbb+wQf/uS56NGjdE5cxbrccdV68KFMf3Tn3bpsceepC+8EKzk+ZYtqkceWaXPPrtZN27cqffdt1jHjj2z7c6nZCXPM3YxW1VPV9UV6aei/Ccfnm1fyPrIA0imb20tnA2YD/p4oaBU9Bs2dNS3tsKGDaz3cHcfNN9HnwyRdoXLN2wA1YPmnzdhAmzYQG1tLWvdMiTjxo3jv/7rv7jllltYt24dhxxyCACffpo4vHLHHf/B178+nHPOOZMNGzbw4YcfAnDUUUdx6qmnAvD1r3+dl156CaCtNtTy5a+yZs2bXHbZqVxwQQ1z5vyWdclKqwDf/e53qampYfbs2fzHf/ya1157ibq6cznkkD707l1GXd15LFjwYgfdmDFjqKyspKSkhJqaGtauXdthMw0ceCwtLWu46qpvsWDBAg499NCkMYQhiDO7yxN1BYRC10cegOmj1fvUkDjaw9190PyQNSjKy8vZvn37Qf227dzJ4Ycd1jarZ/fu8MknlJaWsm/fPgAuuugiHn/8cQ455BAmTZrECy84RbA1odzdU0/NZfv2zdx/fzNz5iyhoqKCvXv3Ah0TVHw6nmRUlbFjv8gDDyzhgQeW8Lvfvcmvf/3rpJ8hfo3i2Wefpaoq+UXy/fs7zuvZs2fb+/hna7+ZDj20Pw88sJSRI+u46667uPzyy5OuPwyWKIi+AkKh6yMPwPTR6n1qSNx88830buca7927Nzcnur5D1qAoKyvjyCOPbPuh3/bxxyx45RVOq6nx1a9Zs4Zjjz2Wq6++msmTJ7Ns2TL69u3Lxx8feNBPLPYRAwZ8hm7durNkSeNBRwTr169vu0Pp4Ycf5rTTTgMOuKKHDTuZpUv/zPvvrwZg377dvP3228k/W7swR448nT/96TH27t3Dxx/vpqnpUT7/+dM71cb1vXv3Zfdu53Ps2LGF1tZWJk78Kj/84Q957bVwJc+TYYmC/KigUMj6yAOwEh55W8JjypQpzJ49m6qqKkSEqqoqZs+ezZTEMuMp1KC47777+MEPfkBNTQ1/P2MGN3zzmxxXWemrf+ihh6iurqampoY33niDSy+9lPLycsaNO5ULLqjmpz/9Ll/+8hRWrlzMBRcM44UX7mPIkCFt+hNOOIFf/OIXnHjiiezYsYPp06cDzg91SQn0738EN9xwL9dddyEXXjicyy4bx6pVqwJtvpNOGsVXvjKNqVPHMG3aWM4993LOPDPYg4UGDoTzzqvn6qsn8tWvns2mTRu48so6pkyp4eKLL+ZHP/pRoPX44nXxopCblfDIXfxWwiOc3kp4eGAlPAqzhEchNyvhkTssrnB0hbishEfhx5XJEh6GYRiGEU2iEJEBIvKsiLzjvvb36LdfRJa4rUuVNjeMQkYTbxUyCopU9l1URxTXAM+r6vHA8+50Mj5W1Rq3nZPNgGbMmEu3boNobm6mW7dBzJgRzpka14uUFKTenNl5Fn8eO7N79erF1q1bD/7BSddZXeD6fAl/925/vaqydetWevXqFWr9QR6Fmg0mA3Xu+98CTcD3IoqFGTPmcued9YBjWtq/f507DXfc0fnD5QtdP3cu1Ncf8GytW+dMA0zpXJ7+CrKhX7fOmR/V+IWm37TpwHO3O9FXVlbS0tLC5s2bnRm7dzu/TImJY+NGKC+HPn06H99Hv7e0tPMftSyO76Xfu3dvW1wRDO+p//jjvWzf3stX36tXLyoT7xALgERxCCkiO1T1MPe9ANvj0+367cOpN7UP+LGqPuazznqgHqCioqJ23rx5geNpbl6OU//wwB+BQw9qa4eF0h9M5vSxWKyt5kumx1++3Pvh7sM6kcdiMcreey/1FaQbgIc+VllJ2aZNkY3vpU+6H3M4vpc+9pnPUNb2vc/9+F762DHHeH7vczG+lz5xP+bB7mvTV1bGaGkpC6WPM2HChGZVHZ10oddV7nQb8BzwRpI2GdjRru92j3UMdF+PBdYCxwUZO+xdTyBtNWgaGhoSatJIaP3BLXN6v7tS0h1fRA8q8xNvEkDe2NiY3grSDcBD39jQEOn4Xvqk+zEPtl9jQ0Ne7b+4PtDdWBFsv8S48mD3tWniterC6OMQxV1PqnqmqlYnab8HPhSRIwHc100e69jgvq7BOT0VzIESktLS5A5Ur/ldTW/ObNObPnV9gYcfiKguZj8OTHXfTwV+376DiPQXkZ7u+8OBU4E3sxFMff3NONXUE+ntzu/6enNm52H8heTMzof4I9QXePjB8DrUyGYDynHudnoH5xTVAHf+aOBu9/0pwHJgqft6WdD1p2K4mz59jpaWVmlDQ4OWllbp9OnhnK1xPUhW9J0dgqc7vjmzzZmdT/svrg9sBMzx9msfV758/VOtLKHqf+opkkSR7WbO7NxhcYXD4gqHxRWOdOLySxTmzDYMwzB8sURhGIZh+GKJwqXYndlR682ZXTjO7KyMX+D6fAm/uTnFygqd4XVOqpBb2GsU06fPUejdzkfRO/AF4UT9gZZZvd+5x1yM70VjY2Pa49szswvrmdm5jD/QOfcItl9iXHmw+9r0cR9FGH0c7GK2P87dQh0Nd6WlVaH1iS2Ter8/mFyM70VjY2Pa42tV1cF/JfFWlbq+saEhLX2643vpk+7HHI7vpU9quMuD7RcoUUSw/RLjyoPd16ZJNNwF1cfxSxR26gnYvz/5s4G95ps+s/rIn/lsetOnoS/w8ANhiYLondHFro/cmmp606ehL/DwA2GJguid0cWuN2d2HujNmZ2yvsDDD4bXOalCbubMzl388bjSHd+c2ebMNme2ObPzPlHE6YqOy2xicYXD4gqHxRUOc2YbhmEYkWCJwjAMw/DFEoVhGIbhiyUKFyvhEa0+3RIGVsLDSnhYCQ8r4ZHVi9lWwiPaEh7pljCwEh5WwsNKeOhBzuwuUcID+BqwAmgFRvv0mwi8BawGrgm6fivhUVglPNItYWAlPNLXWwmPcHor4ZEb3gDOAxZ6dRCRUuAXwJeBk4ALReSkbAQTdQmLYtenXYIg6hoIpi9qfYGHH4hIEoWqrlTVtzrpNgZYraprVPUTYB4wORvxRF3Cotj1aZcgiLoGgumLWl/g4QdCnCOOaBCRJmCmqi5Osux8YKKqXu5OXwKMVdWrPNZVD9QDVFRU1M6bNy9wHOvXb2Pz5nVAK5WVlbS0tAAlHHFEFUcfPSCU/gCZ1cdiMcrKyiIb34tYLMa2bZ+kNf62bbBuHbQmyEtKoKoKBnQuT7qC2FFHUdatW7AVpBtACH3S/ZjD8b30sX37KHv//cjG99LHevTw/N7nYnwvfeJ+zIPd16avrIzR0lIW7u/HZcKECc2qOjrpQq9zUuk24DmcU0zt2+SEPk14XKMAzgfuTpi+BPh5kLGthEfhlfBIt4SBlfCwEh5WwqOLlvDoJFGMA55OmL4WuDbIeq2ER+6wuMJhcYXD4gpHMZbwWAQcLyLHiEgP4ALg8YhjMgzDKDoiSRQicq6ItOAcNfxRRJ52539WRJ4EUNV9wFXA08BK4CFVXRFFvIZhGMVMVHc9PaqqlaraU1UrVPUsd/4Hqjopod+Tqvo5VT1OVTNZXb0D5swubL05s82Z3RWc1ebMzmEzZ3ZhObPT1Zsz25zZ6Yyfityc2V2gmTO7sJzZ6erNmZ2+vpid2anIzZldhETtTDZ9evrIra2mL2h9gYffdZ3Z+UbUzmTTp6eP3Npq+oLWF3j4OXFmW6IA6utvBto9nZze7nzT57s+6dPlS0oK5+n2+aAvafdTUGjxp6Ev8PDT1gfC65xUITdzZheeMztdvTmzzZmdzvhh5ebM7gLNnNm5w+IKh8UVDosrHMXozDYMwzDyAEsUhmEYhi+WKAzDMAxfLFG4WAmPwtZHXcEjb2o4WAkPK+GRgr5TvC5eFHKzEh7FVcIjWQmEWbMac1ZCwUp4WAkPK+FRgM1KeBRXCY9kJRDitwmmqg9VAsFKeGQtfivhEU5vJTyySNQlKEyfnj7yEghRB2D6tPQFHr6V8MgVUZegMH16+shLIEQdgOnT0hd4+FbCI1dEXYLC9Onpo67gEXkNBivhYSU8umIJD+BrwAqgFY9nZrv91gLLgSX4nD9r36yER/GV8GhfAmH+/Ma09KFLIFgJDyvhkfrwVsIj6aBwInAC0BQgURwedv1WwiN3WFzhsLjCYXGFI1slPLpl8OAkMKq6EkBEohjeMAzDCIE4iSSiwUWagJmquthj+XvAdpzbJX+pqrN91lUP1ANUVFTUzps3L6WYYrEYZWVlKWmzicUVDosrHBZXOLpiXBMmTGhW1dFJF3odaqTbgOeAN5K0yQl9mvA/9TTQff0MsBT4QpCx7RqFXaPI9TWKoHq7RhFOH/RUSq6/f+3jivr7n+7vl2oeXqNoG7yTRNGu7/dxjj4ynijMmW3O7Fw5a82ZHU4fJFFE8f1LjCvq73+6v19xCjJRAH2AvgnvXwYmBlmvObPNmZ1LZ3YYvTmzw+mDJIoovn+JcUX9/U/39yuOX6KIxEchIueKSAswDvijiDztzv+siDzpdqsAXhKRpcBfgT+q6oJsxBO1s9j0he3MNmd3tPqov3+Frg9CJIlCVR9V1UpV7amqFap6ljv/A1Wd5L5fo6oj3DZUVTNpHzmIqJ3Fpi9sZ7Y5u6PVR/39K3R9EMyZTfTOYtMXtjO7Szi7C9iZHfX3r9D1gfA6J1XIze56srue7K4nu+spDHbXk/81ish/1LPRzJmdOyyucFhc4bC4wpEtZ7adejIMwzB8sURhGIZh+GKJwjAMw/DFEoXLjBlz6dZtEM3NzXTrNogZM8I9nTyuFykxfQT69g+n37YtlDxjD7cvVP22bdGOH/X3p6voU/396hSvixeF3KyEh5XwKKQSHnlQwUNnzWqMbHy//W8lPMLpu2QJj2w1K+FhJTwKqYRHHlTw0IaGxsjG99v/VsIjnL5LlfDIN6K20Ju+uEt4FLs+6u9PseuDYImC6C30pi/uEh7Fro/6+1Ps+iBYoiB6C73pi7uERz7oo6zgEfX3p9j1gfA6J1XIzUp4WAmPQivhEXEFD50/vzHS8b32v5XwCKe3Eh5ZThRxuqI1P5tYXOGwuMJhcYXDSngYhmEYkWCJwjAMw/Alqifc3Soiq0RkmYg8KiKHefSbKCJvichqEbkmmzGZM7uw9ebMNme26buYMxv4EtDNfX8LcEuSPqXAu8CxQA9gKXBSkPWbM9uc2ebMNme2ObO7kDMbOBeYm2T+OODphOlrgWuDrNOc2ebMNme2ObPNmR1MH8cvUYizPDpE5A/A71R1Trv55wMTVfVyd/oSYKyqXuWxnnqgHqCioqJ23rx5gWNobm5ue19ZWUlLS0vbdG1tbSh9ezKlj8VilJWVRTa+F7FYjLfeeiuy8R19x3mVlTFaWsoIIE+qPzB+ZvXJ9mMux/fSx7dXNON7r+CEE07w/N4H0Wfr+5e4H6P8+2uvT+X3K86ECROaVXV00oVeGSTdBjwHvJGkTU7ocx3wKDgJq53+fODuhOlLgJ8HGduOKOyIwo4o7IjCjiiC6eMQxe2xqnqmqlYnab8HEJFpwFeAKW6Q7dkAHJUwXenOyzhROyNNb87sqPXmzC5efSC8Mkg2GzAReBM4wqdPN2ANcAwHLmYPDbJ+c2abM9uc2eH05sw2Zzb5djEbWA28Dyxx213u/M8CTyb0mwS8jXP303VB12/O7NxhcYXD4gqHxRWObDmzu2Xu2CQ4qjrYY/4HOMkhPv0k8GSu4jIMwzA6Ys5swzAMwxdLFIZhGIYvligMwzAMXyxRGIZhGL5E7szOBiKyGViXovxwYEsGw8kUFlc4LK5wWFzh6IpxVanqEckWdMlEkQ4isli9bOwRYnGFw+IKh8UVjmKLy049GYZhGL5YojAMwzB8sUTRkdlRB+CBxRUOiyscFlc4iiouu0ZhGIZh+GJHFIZhGIYvligMwzAMX4o+UYjIrSKySkSWicijInKYR7+JIvKWiKwWkWtyENfXRGSFiLSKiOftbiKyVkSWi8gSEVmcR3HlensNEJFnReQd97W/R7/97rZaIiKPZzEe388vIj1F5Hfu8r+IyKBsxRIyrmkisjlhG12eg5juEZFNIvKGx3IRkdvdmJeJyKhsxxQwrjoR+ShhW12fo7iOEpFGEXnT/Vv8lyR9MrvNvMrKFksDvgR0c9/fAtySpE8pTqnzYznwbIyTshzXicAJQBMw2qffWuDwHG6vTuOKaHv9P+Aa9/01yfajuyyWg23U6ecHZnCgvP4FOI8Dzoe4phHwSZIZjOsLwCjgDY/lk4CnAAFOBv6SJ3HVAU/kclu54x4JjHLf98V5FEP7/ZjRbVb0RxSq+oyq7nMnX8V5kl57xgCrVXWNqn4CzAMmZzmularq/TDqiAgYV863l7v+37rvfwv8Y5bH8yPI50+M93+AM0RE8iCunKOqC4FtPl0mA/epw6vAYSJyZB7EFQmqulFVX3Pf7wJWAgPbdcvoNiv6RNGOb+Bk4fYMxHnQUpwWOu6YqFDgGRFpFpH6qINxiWJ7VajqRvf9/wIVHv16ichiEXlVRP4xS7EE+fxtfdx/VD4CyrMUT5i4AL7qnq74HxE5KsnyXJPPf3/jRGSpiDwlIkNzPbh7ynIk8Jd2izK6zSJ5cFGuEZHngL9Lsug6PfAM7+uAfcDcfIorAKep6gYR+QzwrIiscv8TijqujOMXV+KEqqqIeN33XeVur2OBF0Rkuaq+m+lYC5g/AA+q6t9E5Aqco56/jzimfOU1nO9TTEQmAY8Bx+dqcBEpA+YD31bVndkcqygShaqe6bdcRKYBXwHOUPcEXzs2AIn/WVW687IaV8B1bHBfN4nIozinF9JKFBmIK+fbS0Q+FJEjVXWje4i9yWMd8e21RkSacP4by3SiCPL5431aRKQb0A/YmuE4Qselqokx3I1z7SdqsvJ9SpfEH2dVfVJE7hCRw1U168UCRaQ7TpKYq6qPJOmS0W1W9KeeRGQi8O/AOaq6x6PbIuB4ETlGRHrgXHzM2h0zQRGRPiLSN/4e58J80js0ckwU2+txYKr7firQ4chHRPqLSE/3/eHAqcCbWYglyOdPjPd84AWPf1JyGle789jn4Jz/jprHgUvdO3lOBj5KOM0YGSLyd/HrSiIyBuf3NNvJHnfMXwMrVXWWR7fMbrNcX7HPtwasxjmXt8Rt8TtRPgs8mdBvEs7dBe/inILJdlzn4pxX/BvwIfB0+7hw7l5Z6rYV+RJXRNurHHgeeAd4Dhjgzh8N3O2+PwVY7m6v5cBlWYynw+cHbsL5hwSgF/Cw+/37K3BstrdRwLh+5H6XlgKNwJAcxPQgsBH41P1uXQZcCVzpLhfgF27My/G5CzDHcV2VsK1eBU7JUVyn4VybXJbwuzUpm9vMSngYhmEYvhT9qSfDMAzDH0sUhmEYhi+WKAzDMAxfLFEYhmEYvliiMAzDMHyxRGEYIXGrd74nIgPc6f7u9KA01/tyRgI0jAxjt8caRgqIyL8Dg1W1XkR+CaxV1R9FHZdhZAM7ojCM1PgJcLKIfBvHANXQvoOIPOYWa1wRL9goIlXiPDPjcBEpEZEXReRL7rKY+3qkiCx0n3HwhoicnruPZRgdsSMKw0gRETkLWAB8SVWfTbJ8gKpuE5FDcMpnjFfVreI8DOgsHEf2YFW9wu0fU9UyEfk3oJeq3iwipUBvdcpJG0Yk2BGFYaTOl3FKPFR7LL9aROLlHY7CrSyqqncDh+KUXJiZRLcI+CcR+T4wzJKEETWWKAwjBUSkBvgiztPD/tW9wB1/JOaVIlIHnAmMU9URwOs49Z0Qkd4ceEBWWft1q1Mm/gs41T7vFZFLs/xxDMOXoigzbhiZxK3eeSfOcwDWi8itwI9VtSahz2Rgu6ruEZEhOAklzi04zz1ZB/wKp8R94vqrgBZV/ZVb7XYUcF82P5Nh+GFHFIYRnm8C6xOuS9wBnCgi4xP6LAC6ichK4Mc4p59w+3we55nec4FPROSf2q2/DlgqIq8DXwd+mrVPYhgBsIvZhmEYhi92RGEYhmH4YonCMAzD8MUShWEYhuGLJQrDMAzDF0sUhmEYhi+WKAzDMAxfLFEYhmEYvvx/xCQysXXwO4sAAAAASUVORK5CYII=\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": [
    "#The performance on the FP process calculated.\n",
    "# Save to text file\n",
    "np.savetxt('V_VDP_NN.txt', V_candidate.detach().numpy(), fmt='%.6f')  # value of V for each sample\n",
    "# Save to text file\n",
    "np.savetxt('LV_VDP_NN.txt', L_V.detach().numpy(), fmt='%.6f')  # value of \\dot V for each sample"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'np' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Input \u001b[0;32mIn [1]\u001b[0m, in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m      1\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01mscipy\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mio\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m savemat\n\u001b[1;32m      4\u001b[0m \u001b[38;5;66;03m# Convert tensors to NumPy array\u001b[39;00m\n\u001b[0;32m----> 5\u001b[0m array_data1 \u001b[38;5;241m=\u001b[39m \u001b[43mnp\u001b[49m\u001b[38;5;241m.\u001b[39marray([t\u001b[38;5;241m.\u001b[39mnumpy() \u001b[38;5;28;01mfor\u001b[39;00m t \u001b[38;5;129;01min\u001b[39;00m iV_point_list]) \u001b[38;5;66;03m#points outside the stability region\u001b[39;00m\n\u001b[1;32m      6\u001b[0m array_data2 \u001b[38;5;241m=\u001b[39m np\u001b[38;5;241m.\u001b[39marray([t\u001b[38;5;241m.\u001b[39mnumpy() \u001b[38;5;28;01mfor\u001b[39;00m t \u001b[38;5;129;01min\u001b[39;00m uV_point_list]) \u001b[38;5;66;03m#points within the stability region\u001b[39;00m\n\u001b[1;32m      9\u001b[0m \u001b[38;5;66;03m# Save as .mat\u001b[39;00m\n",
      "\u001b[0;31mNameError\u001b[0m: name 'np' is not defined"
     ]
    }
   ],
   "source": [
    "from scipy.io import savemat\n",
    "\n",
    "\n",
    "# Convert tensors to NumPy array\n",
    "array_data1 = np.array([t.numpy() for t in iV_point_list]) #points outside the stability region\n",
    "array_data2 = np.array([t.numpy() for t in uV_point_list]) #points within the stability region\n",
    "\n",
    "\n",
    "# Save as .mat\n",
    "savemat('unstable_VDP_NN.mat', {'data': array_data1})\n",
    "savemat('stable_VDP_NN.mat', {'data': array_data2})"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
