{"cells":[{"cell_type":"markdown","metadata":{"id":"4kEj-BFnmH37"},"source":["## Learning Dyanmics and Lyapunov function for Van der Pol"]},{"cell_type":"code","execution_count":null,"metadata":{},"outputs":[],"source":["# To install dReal, if you use Google CoLab\n","# If not CoLab, command out\n","import pkgutil\n","if not pkgutil.find_loader(\"dreal\"):\n","  !curl https://raw.githubusercontent.com/dreal/dreal4/master/setup/ubuntu/18.04/install.sh | bash\n","  !pip install dreal --upgrade"]},{"cell_type":"code","execution_count":null,"metadata":{"executionInfo":{"elapsed":785,"status":"ok","timestamp":1652561935247,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"e9EDzAWmmH38"},"outputs":[],"source":["# -*- coding: utf-8 -*-\n","from dreal import *\n","from Functions import *\n","import torch \n","import torch.nn.functional as F\n","import numpy as np\n","import timeit \n","import matplotlib.pyplot as plt\n","from tqdm import tqdm"]},{"cell_type":"code","execution_count":null,"metadata":{"executionInfo":{"elapsed":155,"status":"ok","timestamp":1652562319597,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"E90gNEmXo-1d"},"outputs":[],"source":["device = 'cuda' if torch.cuda.is_available() else 'cpu'"]},{"cell_type":"markdown","metadata":{"id":"-ABa4JuimH39"},"source":["## Actual dynamical system"]},{"cell_type":"code","execution_count":null,"metadata":{"executionInfo":{"elapsed":148,"status":"ok","timestamp":1652562040358,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"Yz7R7cTv71Fy"},"outputs":[],"source":["def f_value(x):\n","    y = torch.zeros_like(x)\n","    y[:,0] = - x[:,1]\n","    y[:,1] = x[:,0] + (x[:,0]**2-1)*x[:,1]\n","    return y"]},{"cell_type":"markdown","metadata":{"id":"bfmggjnvmH3-"},"source":["## NN for learning the dynamics f"]},{"cell_type":"code","execution_count":null,"metadata":{"executionInfo":{"elapsed":198,"status":"ok","timestamp":1652562043859,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"kADm4aVSmH3-"},"outputs":[],"source":["# NN for learning dynamics\n","class fNet(torch.nn.Module):\n","    def __init__(self,n_input, n_hidden1,  n_output):\n","        super().__init__()\n","        torch.manual_seed(2)\n","        self.layer1 = torch.nn.Linear(n_input, n_hidden1)\n","        self.layer2 = torch.nn.Linear(n_hidden1,n_output)   \n","        self.to(device)\n","\n","    def forward(self,x):\n","        sigmoid = torch.nn.Tanh()\n","        h_1 = sigmoid(self.layer1(x))\n","        out = self.layer2(h_1) \n","        return out"]},{"cell_type":"markdown","metadata":{"id":"FR__fZnQmH3-"},"source":["## Learning the dynamics with NNs"]},{"cell_type":"code","execution_count":null,"metadata":{"executionInfo":{"elapsed":13282,"status":"ok","timestamp":1652562067830,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"PV0wSHJomH3-"},"outputs":[],"source":["# generate training dataset\n","N_f = 3001\n","xx = np.linspace(-1.5,1.5, N_f, dtype = float)\n","x_f = []\n","for i in range(0,N_f): \n","    for j in range(0,N_f):\n","        x_f.append([xx[j],xx[i]])\n","\n","x_f = torch.tensor(x_f)\n","x_f = x_f.float()\n","\n","# target\n","t_f = f_value(x_f)  # actual dynamics\n","t_f = t_f.float()"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/","height":311},"executionInfo":{"elapsed":364477,"status":"ok","timestamp":1652556485718,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"Zz1me5ObmH3_","outputId":"97b44a40-4181-4f62-9639-f0aed5b0b192"},"outputs":[],"source":["# define parameters\n","max_iter = 1000\n","losses = []\n","# NN: 1 hidden layers with 100 neurons each layer \n","fnet = fNet(n_input = 2, n_hidden1 = 100,  n_output = 2)\n","optimizer = torch.optim.Adam(fnet.parameters(), lr = 0.1)\n","\n","loss_func = torch.nn.MSELoss(reduction='sum')\n","\n","for epoch in tqdm(range(max_iter)):\n","    x_f = x_f.to(device)\n","    t_f = t_f.to(device)\n","    y_nn = fnet(x_f)\n","    loss = loss_func(y_nn,t_f)\n","    losses.append(loss.item())\n","    \n","    optimizer.zero_grad()\n","    loss.backward()\n","    optimizer.step()\n","    \n","    with torch.no_grad():\n","        torch.cuda.empty_cache()\n","\n","plt.plot(losses)"]},{"cell_type":"code","execution_count":null,"metadata":{"executionInfo":{"elapsed":157,"status":"ok","timestamp":1652562174314,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"kLf9pcRbYVvb"},"outputs":[],"source":["# clear cache\n","with torch.no_grad():\n","    torch.cuda.empty_cache()"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/","height":236},"executionInfo":{"elapsed":151,"status":"error","timestamp":1652562430312,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"1ooVmMOBmH3_","outputId":"9eee1739-6198-481e-d3cf-df37675713a7"},"outputs":[],"source":["# train more epoches if needed as we need alpha to be very small\n","losses = []\n","optimizer = torch.optim.Adam(fnet.parameters(), lr = 0.001) # \n","loss_func = torch.nn.MSELoss(reduction='sum')\n","# fnet = fnet.to(device)\n","\n","for epoch in tqdm(range(50)):\n","    x_f = x_f.to(device)\n","    t_f = t_f.to(device)\n","    y_nn = fnet(x_f)\n","    loss = loss_func(y_nn,t_f)\n","    losses.append(loss.item())\n","    \n","    optimizer.zero_grad()\n","    loss.backward()\n","    optimizer.step()\n","\n","    with torch.no_grad():\n","        torch.cuda.empty_cache()\n","\n","plt.plot(losses)"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"elapsed":129,"status":"ok","timestamp":1652559609898,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"n8L5gb7VmH4A","outputId":"19d1f129-7ba0-4cc7-e17d-6b72d63205bb"},"outputs":[],"source":["losses[-1]"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/","height":236},"executionInfo":{"elapsed":158,"status":"error","timestamp":1652562463676,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"VyIUZ8KrLUsz","outputId":"9ae0a424-e7c5-44f9-cc91-3a5f7d000f5b"},"outputs":[],"source":["# generate testing dataset\n","r = 1.2 # region of interest\n","N_l = 2401\n","xl = np.linspace(-r,r,N_l, dtype = float)\n","x_l = []\n","for i in range(0,N_l): \n","    for j in range(0,N_l):\n","        x_l.append([xl[j],xl[i]])\n","\n","x_l = torch.tensor(x_l)\n","x_l = x_l.float()\n","\n","# target\n","t_l = f_value(x_l)  # actual dynamics\n","t_l = t_l.float()"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/","height":236},"executionInfo":{"elapsed":157,"status":"error","timestamp":1652562452360,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"0USgcnNdmH4A","outputId":"886aee0c-ec8b-4932-83f2-1835dced1ef0"},"outputs":[],"source":["# output of FNN\n","x_l = x_l.to(device)\n","t_l = t_l.to(device)\n","y_l = fnet(x_l)\n","\n","# maximum of loss\n","loss_all = torch.norm(y_l-t_l, dim = 1)\n","alpha = torch.max(loss_all)"]},{"cell_type":"code","execution_count":null,"metadata":{"executionInfo":{"elapsed":144,"status":"ok","timestamp":1652560291227,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"uKbFhq8RmH4A"},"outputs":[],"source":["# save the weights to calculate the Lipschitz constant with LipSDP\n","f_w1 = fnet.layer1.weight.data.cpu().numpy()\n","f_w2 = fnet.layer2.weight.data.cpu().numpy()\n","\n","np.savetxt(\"fw1.txt\", f_w1, fmt=\"%s\")\n","np.savetxt(\"fw2.txt\", f_w2, fmt=\"%s\")"]},{"cell_type":"code","execution_count":null,"metadata":{"executionInfo":{"elapsed":2830,"status":"ok","timestamp":1652562089830,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"8a6LRBvpmH4B"},"outputs":[],"source":["# # save the fNN\n","# torch.save(fnet.cpu(), 'VDP_fnet.pt')\n","\n","# # load fNN from a file\n","# fnet = torch.load('VDP_fnet.pt').to(device)"]},{"cell_type":"markdown","metadata":{"id":"_RD7mvVvmH4B"},"source":["## Plot the approximated results"]},{"cell_type":"code","execution_count":null,"metadata":{"executionInfo":{"elapsed":126,"status":"ok","timestamp":1652560515752,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"XOXnrgqmmH4C"},"outputs":[],"source":["# dataset for plot\n","N_p = 300\n","xp = np.linspace(-r,r,N_p, dtype = float).reshape(N_p,1)\n","xx = np.linspace(-r,r,N_p, dtype = float).reshape(N_p,1)\n","\n","for n in range(1): # for dim n\n","    xp = np.concatenate((xp,xx), axis=1 ) \n","\n","xp = torch.tensor(xp)\n","Xp = xp.float()\n"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/","height":541},"executionInfo":{"elapsed":435,"status":"ok","timestamp":1652560518473,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"I2wUcVsgmH4C","outputId":"2304f149-ebab-4e25-c353-7cff46ce5af3"},"outputs":[],"source":["# fnet_1 = fnet.to('cpu') # to plot, command out this line\n","yp = fnet_1(Xp)\n","tp = f_value(xp)\n","\n","with torch.no_grad():\n","    plt.figure(1)\n","    plt.plot(xp[:,0], yp[:,0],label='NN')\n","    plt.plot(xp[:,0], tp[:,0], label='Actual')\n","    plt.xlabel('x1 & x2 pair')\n","    plt.ylabel('$\\dot{x_1}$')\n","    plt.legend(loc = 0)\n","\n","    plt.figure(2)\n","    plt.plot(xp[:,0],yp[:,1],label='NN')\n","    plt.plot(xp[:,0],tp[:,1],label='Aactual')\n","    plt.xlabel('x1 & x2 pair')\n","    plt.ylabel('$\\dot{x_2}$')\n","    plt.legend()"]},{"cell_type":"markdown","metadata":{"id":"R0esL68wmH4C"},"source":["## Learned dynamics"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"17sN8Y_1mH4C"},"outputs":[],"source":["def f_learned(x):\n","    y = fnet(x)\n","    return y"]},{"cell_type":"markdown","metadata":{"id":"3CfzCAkTmH4C"},"source":["## Neural network model for Lyapunov function V"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"ePWlh9OcmH4C"},"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)\n","        self.layer2 = torch.nn.Linear(n_hidden,n_output)\n","        self.to(device) \n","        \n","    def forward(self,x):\n","        sigmoid = torch.nn.Tanh()\n","        h_1 = sigmoid(self.layer1(x))\n","        out = sigmoid(self.layer2(h_1))\n","        return out"]},{"cell_type":"markdown","metadata":{"id":"ll1sstmomH4D"},"source":["## Parameters"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"WaAJWAmSmH4D"},"outputs":[],"source":["'''\n","For learning \n","'''\n","N = 500            # sample size\n","D_in = 2            # input dimension\n","H1 = 6              # hidden dimension\n","D_out = 1           # output dimension\n","torch.manual_seed(10)  \n","\n","x = torch.Tensor(N, D_in).uniform_(-r, r)           \n","x_0 = torch.zeros([1, 2])\n","x_0 = x_0.to(device)\n","\n","'''\n","For verifying \n","'''\n","x1 = Variable(\"x1\")\n","x2 = Variable(\"x2\")\n","vars_ = [x1,x2]\n","config = Config()\n","config.use_polytope_in_forall = True\n","config.use_local_optimization = True\n","config.precision = 1e-2\n","beta = -0.02 # initial guess of beta\n","# Checking candidate V within a ball around the origin (ball_lb ≤ sqrt(∑xᵢ²) ≤ ball_ub)\n","ball_lb = 0.2\n","ball_ub = 1.2\n","\n","# parameters for beta\n","Kf = 3.4599\n","KF = 5.452\n","d = 5e-4\n","loss = 0.0085 # equals alpha above"]},{"cell_type":"markdown","metadata":{"id":"Z-vy4z21mH4D"},"source":["## Learning and Falsification"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/"},"executionInfo":{"elapsed":42438,"status":"ok","timestamp":1652482358200,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"rvQBaISPmH4D","outputId":"f223c95e-6812-40c8-dd51-cffe63a36a81","scrolled":true},"outputs":[],"source":["out_iters = 0\n","valid = False\n","while out_iters < 2 and not valid: \n","    start = timeit.default_timer()\n","    model = Net(D_in,H1, D_out)\n","    L = []\n","    i = 0 \n","    t = 0\n","    max_iters = 3000 # increase number of epoches if cannot find a valid LF\n","    learning_rate = 0.01\n","    optimizer = torch.optim.Adam(model.parameters(), lr=learning_rate)\n","\n","    # learned dynamics\n","    f_w1 = fnet.layer1.weight.data.cpu().numpy()\n","    f_w2 = fnet.layer2.weight.data.cpu().numpy()\n","    f_b1 = fnet.layer1.bias.data.cpu().numpy()\n","    f_b2 = fnet.layer2.bias.data.cpu().numpy()\n","\n","    f_h1 = []\n","    f_z1 = np.dot(vars_,f_w1.T)+f_b1\n","    for n in range(len(f_z1)):\n","        f_h1.append(tanh(f_z1[n]))\n","\n","    f_learn = np.dot(f_h1,f_w2.T)+f_b2\n","\n","    while i < max_iters and not valid: \n","        x = x.float()\n","        x = x.to(device)\n","        V_candidate = model(x)\n","        X0 = model(x_0)\n","        f = f_learned(x)\n","        Circle_Tuning = Tune(x)\n","        Circle_Tuning = Circle_Tuning.to(device)\n","        # Compute lie derivative of V : L_V = ∑∂V/∂xᵢ*fᵢ\n","        L_V = torch.diagonal(torch.mm(torch.mm(torch.mm(dtanh(V_candidate),model.layer2.weight)\\\n","                            *dtanh(torch.tanh(torch.mm(x,model.layer1.weight.t())+model.layer1.bias)),model.layer1.weight),f.t()),0)\n","        \n","        dVdx = torch.mm(torch.mm(dtanh(V_candidate),model.layer2.weight)\\\n","                            *dtanh(torch.tanh(torch.mm(x,model.layer1.weight.t())+model.layer1.bias)),model.layer1.weight)\n","\n","        # With tuning\n","        Lyapunov_risk = (F.relu(-V_candidate)+ 1.2*F.relu(L_V+0.2)).mean()\\\n","                    +((Circle_Tuning-V_candidate).pow(2)).mean()+ 1.2*(X0).pow(2) + 0.01*torch.norm(dVdx) \n","\n","\n","        print(i, \"Lyapunov Risk=\",Lyapunov_risk.item()) \n","        L.append(Lyapunov_risk.item())\n","        optimizer.zero_grad()\n","        Lyapunov_risk.backward()\n","        optimizer.step() \n","\n","        # save the weights and biases \n","        w1 = model.layer1.weight.data.cpu().numpy()\n","        w2 = model.layer2.weight.data.cpu().numpy()\n","        b1 = model.layer1.bias.data.cpu().numpy()\n","        b2 = model.layer2.bias.data.cpu().numpy()\n","        \n","        # Falsification with SMT solver\n","        if i % 10 == 0:\n","            \n","            # Candidate V\n","            z1 = np.dot(vars_,w1.T)+b1\n","\n","            a1 = []\n","            for j in range(0,len(z1)):\n","                a1.append(tanh(z1[j]))\n","            z2 = np.dot(a1,w2.T)+b2\n","            V_learn = tanh(z2.item(0))\n","\n","            print('===========Verifying==========')        \n","            start_ = timeit.default_timer() \n","            beta = -np.maximum(beta, -0.02) # in case beta is too negative and cannot return any results\n","            result= CheckLyapunov(vars_, f_learn, V_learn, ball_lb, ball_ub, config, beta) # SMT solver\n","            stop_ = timeit.default_timer() \n","\n","            if (result): \n","                print(\"Not a Lyapunov function. Found counterexample: \")\n","                print(result)\n","                x = x.to('cpu')\n","                x = AddCounterexamples(x,result,10)\n","            else:  \n","                # calculate norm of dVdx with the SMT solver\n","                M = 0.005 # lower bound of M\n","                violation = CheckdVdx(vars_, V_learn, ball_ub, config, M) \n","                while violation:\n","                    violation = CheckdVdx(vars_, V_learn, ball_ub, config, M)\n","                    if not violation:\n","                        dvdx_bound = np.sqrt(M)\n","                        print(dvdx_bound, \"is the norm of dVdx\")\n","                    M += 0.0001\n","                beta = -dvdx_bound*((Kf+KF)*d+loss) # update beta \n","                result_strict= CheckLyapunov(vars_, f_learn, V_learn, ball_lb, ball_ub, config, beta) # SMT solver\n","                if not result_strict:\n","                    valid = True\n","                    print(\"Satisfy conditions with beta = \", beta)\n","                    print(V_learn, \" is a Lyapunov function.\")\n","            t += (stop_ - start_)\n","            print('==============================') \n","        i += 1\n","\n","    stop = timeit.default_timer()\n","\n","\n","    np.savetxt(\"w1_vdp.txt\", model.layer1.weight.data.cpu(), fmt=\"%s\")\n","    np.savetxt(\"w2_vdp.txt\", model.layer2.weight.data.cpu(), fmt=\"%s\")\n","    np.savetxt(\"b1_vdp.txt\", model.layer1.bias.data.cpu(), fmt=\"%s\")\n","    np.savetxt(\"b2_vdp.txt\", model.layer2.bias.data.cpu(), fmt=\"%s\")\n","\n","    print('\\n')\n","    print(\"Total time: \", stop - start)\n","    print(\"Verified time: \", t)\n","    \n","    out_iters+=1"]},{"cell_type":"markdown","metadata":{"id":"hGddjfcDmH4F"},"source":["### Checking result with bounded beta if needed"]},{"cell_type":"code","execution_count":null,"metadata":{"colab":{"base_uri":"https://localhost:8080/","height":240},"executionInfo":{"elapsed":194,"status":"error","timestamp":1652545276184,"user":{"displayName":"Zhou Ruikun","userId":"16546965067278251021"},"user_tz":240},"id":"P7LXeTaomH4F","outputId":"cf546027-8e78-452c-aba7-148653b98126"},"outputs":[],"source":["beta = -0.02\n","start_ = timeit.default_timer() \n","result = CheckLyapunov(vars_, f_learn, V_learn, ball_lb, ball_ub, config, beta)\n","stop_ = timeit.default_timer() \n","\n","if (result): \n","    print(\"Not a Lyapunov function. Found counterexample: \")\n","    print(result)\n","else:  \n","    print(\"Satisfy conditions with beta = \",beta)\n","    print(V_learn, \" is a Lyapunov function.\")\n","t += (stop_ - start_)"]},{"cell_type":"markdown","metadata":{"id":"msUVAVa_mH4F"},"source":["## Checking with beta = 0 for actual dynamics"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"3qVY2J-5mH4F","outputId":"1233bf06-2986-4b30-86f6-2ee50f9a748c"},"outputs":[],"source":["beta = 0\n","f = [-x2,\n","        x1+(x1**2 - 1)*x2]\n","start_ = timeit.default_timer() \n","result = CheckLyapunov(vars_, f, V_learn, ball_lb, ball_ub, config, beta)\n","stop_ = timeit.default_timer() \n","if (result): \n","    print(\"Not a Lyapunov function. Found counterexample: \")\n","    print(result)\n","else:  \n","    print(\"Satisfy conditions with beta= \", beta)\n","    print(V_learn, \" is a Lyapunov function for actual dynamics.\")\n","t += (stop_ - start_)\n"]},{"cell_type":"code","execution_count":null,"metadata":{"id":"IdJprlZSmH4G"},"outputs":[],"source":[]}],"metadata":{"accelerator":"GPU","colab":{"collapsed_sections":[],"name":"Van_der_Pol_final_copy.ipynb","provenance":[]},"kernelspec":{"display_name":"Python 3","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.12"}},"nbformat":4,"nbformat_minor":0}
