{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Analysis of the different models for the two-body-problem\n",
    "\n",
    "In this jupyter notebook, we provide an easy tool to sumarise the results and compare the different models we tested. Here, you can find the precision of the different models, and an comparison of the conservation of different quantities."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.animation as animation\n",
    "import torch, time, sys\n",
    "import numpy as np\n",
    "import scipy.integrate\n",
    "solve_ivp = scipy.integrate.solve_ivp\n",
    "\n",
    "EXPERIMENT_DIR = '.'\n",
    "sys.path.append(EXPERIMENT_DIR)\n",
    "\n",
    "\n",
    "from data import get_dataset, get_field, get_trajectory, dynamics_fn, hamiltonian_fn, matrices\n",
    "#from data import potential_energy, kinetic_energy, total_energy\n",
    "\n",
    "import modelloader\n",
    "\n",
    "from sklearn.linear_model import LinearRegression, Ridge, Lasso\n",
    "from sklearn.metrics import mean_squared_error, r2_score\n",
    "from sklearn.preprocessing import PolynomialFeatures\n",
    "from sklearn.decomposition import PCA\n",
    "\n",
    "\n",
    "from utils import from_pickle"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "A first view on the defintion of S, D and A"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The matrix S is:\n",
      "[[ 0.97522402  0.22121958]\n",
      " [-0.22121958  0.97522402]]\n",
      "\n",
      "The matrix D is:\n",
      "[[1.   0.  ]\n",
      " [0.   1.96]]\n",
      "\n",
      "The matrix A is:\n",
      "[[ 1.04698058 -0.2071091 ]\n",
      " [-0.2071091   1.91301942]]\n"
     ]
    }
   ],
   "source": [
    "S, D=matrices()\n",
    "A=np.matmul(np.transpose(S),np.matmul(D,S))\n",
    "print('The matrix S is:')\n",
    "print(S)\n",
    "print()\n",
    "print('The matrix D is:')\n",
    "print(D)\n",
    "print()\n",
    "print('The matrix A is:')\n",
    "print(A)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Defining some constants"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "DIM=2\n",
    "\n",
    "def get_args():\n",
    "    return {'input_dim': 2*DIM, \n",
    "         'hidden_dim': 200,\n",
    "         'name': 'harm_osc',\n",
    "         'seed': 2}\n",
    "\n",
    "class ObjectView(object):\n",
    "    def __init__(self, d): self.__dict__ = d"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# We load all the different models:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "args = ObjectView(get_args())\n",
    "np.random.seed(args.seed)\n",
    "\n",
    "model_hnn=modelloader.benchmark_loader(args, baseline=False)\n",
    "model_scnn_1=modelloader.pnn_loader(args, test_dim=DIM, number=1)\n",
    "model_scnn_2=modelloader.pnn_loader(args, test_dim=DIM, number=2)\n",
    "model_scnn_3=modelloader.pnn_loader(args, test_dim=DIM, number=3)\n",
    "model_scnn_4=modelloader.pnn_loader(args, test_dim=DIM, number=4)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "In the following, you will find the analytic expression for the conserved quantity and for the Hamiltonian. Note, that for lower prefactors $\\alpha$ the conserved quantities will have more and more noise. Therefore, we displayed the conserved quantities from Model 1 in the paper. <br>\n",
    "The coordinates in the physical system are defined as the following: <br>\n",
    "x0 -> q1 <br>\n",
    "x1 -> q2 <br>\n",
    "x2 -> p1 <br>\n",
    "x3 -> p2 <br>\n",
    "Side note: The conserved quantities look different because superpositions of them are always allowed as conserved quantities"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "data = get_dataset(samples=100,test_split=1.)\n",
    "x = torch.tensor( data['x'], requires_grad=True, dtype=torch.float32)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# The analytic expressions for the conserved quantities for Model 1:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Coordinate 1\n",
      "x0^2 0.67\n",
      "x0 x1 -3.6\n",
      "x1^2 8.23\n",
      "x2^2 0.46\n",
      "x2 x3 -1.78\n",
      "x3^2 4.2\n",
      "____________________________________________________________________________________________________\n",
      "\n",
      "Coordinate 2\n",
      "x0^2 -4.53\n",
      "x0 x1 -1.74\n",
      "x1^2 -0.9\n",
      "x2^2 -4.5\n",
      "x2 x3 -1.88\n",
      "x3^2 -0.57\n",
      "____________________________________________________________________________________________________\n",
      "\n"
     ]
    }
   ],
   "source": [
    "\n",
    "x=torch.FloatTensor(data['x'])\n",
    "batch_size=np.shape(x)[0]\n",
    "y=model_scnn_1.forward_1(x)[0]\n",
    "y=y.detach().numpy()\n",
    "x=x.numpy()\n",
    "\n",
    "for j in range(DIM):\n",
    "    y1=y[:,j:j+1]\n",
    "    \n",
    "    for i in range(3):\n",
    "        polynomial_features= PolynomialFeatures(degree=i)\n",
    "        x_poly = polynomial_features.fit_transform(x)\n",
    "\n",
    "        model_2 = LinearRegression()\n",
    "        model_2.fit(x_poly, y1)\n",
    "        y_poly_pred = model_2.predict(x_poly)\n",
    "        rmse = np.sqrt(mean_squared_error(y1,y_poly_pred))\n",
    "        r2 = r2_score(y1,y_poly_pred)\n",
    "        if r2>0.95:\n",
    "            print('Coordinate '+ str(j+1))\n",
    "            for k in range(len(polynomial_features.get_feature_names())):\n",
    "                if np. abs(model_2.coef_.flatten()[k])>0.05:\n",
    "                    print(polynomial_features.get_feature_names()[k], np.round(model_2.coef_.flatten()[k],2))\n",
    "            print('_'*100)\n",
    "            print()\n",
    "            break\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# And for the Hamiltonian:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "x0^2 1.02\n",
      "x0 x1 -0.41\n",
      "x1^2 1.89\n",
      "x2^2 0.97\n",
      "x3^2 0.99\n",
      "____________________________________________________________________________________________________\n",
      "\n"
     ]
    }
   ],
   "source": [
    "\n",
    "x=torch.FloatTensor(data['x'])\n",
    "batch_size=np.shape(x)[0]\n",
    "y=model_scnn_1.forward_12(x)\n",
    "y=y.detach().numpy()\n",
    "x=x.numpy()\n",
    "\n",
    "\n",
    "for i in range(3):\n",
    "    polynomial_features= PolynomialFeatures(degree=i)\n",
    "    x_poly = polynomial_features.fit_transform(x)\n",
    "\n",
    "    model_2 = LinearRegression()\n",
    "    model_2.fit(x_poly, y)\n",
    "    y_poly_pred = model_2.predict(x_poly)\n",
    "    rmse = np.sqrt(mean_squared_error(y,y_poly_pred))\n",
    "    r2 = r2_score(y,y_poly_pred)\n",
    "    if r2>0.95:\n",
    "        for k in range(len(polynomial_features.get_feature_names())):\n",
    "            if np. abs(model_2.coef_.flatten()[k])>0.05:\n",
    "                print(polynomial_features.get_feature_names()[k], np.round(model_2.coef_.flatten()[k],2))\n",
    "        print('_'*100)\n",
    "        print()\n",
    "        break\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# The analytic expressions for the conserved quantities for Model 2:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Coordinate 1\n",
      "x0^2 0.62\n",
      "x0 x1 -3.03\n",
      "x1^2 7.04\n",
      "x2^2 0.45\n",
      "x2 x3 -1.48\n",
      "x3^2 3.6\n",
      "____________________________________________________________________________________________________\n",
      "\n",
      "Coordinate 2\n",
      "x0^2 -4.46\n",
      "x0 x1 -1.69\n",
      "x1^2 -0.91\n",
      "x2^2 -4.44\n",
      "x2 x3 -1.84\n",
      "x3^2 -0.58\n",
      "____________________________________________________________________________________________________\n",
      "\n"
     ]
    }
   ],
   "source": [
    "\n",
    "x=torch.FloatTensor(data['x'])\n",
    "batch_size=np.shape(x)[0]\n",
    "y=model_scnn_2.forward_1(x)[0]\n",
    "y=y.detach().numpy()\n",
    "x=x.numpy()\n",
    "\n",
    "for j in range(DIM):\n",
    "    y1=y[:,j:j+1]\n",
    "    \n",
    "    for i in range(3):\n",
    "        polynomial_features= PolynomialFeatures(degree=i)\n",
    "        x_poly = polynomial_features.fit_transform(x)\n",
    "\n",
    "        model_2 = LinearRegression()\n",
    "        model_2.fit(x_poly, y1)\n",
    "        y_poly_pred = model_2.predict(x_poly)\n",
    "        rmse = np.sqrt(mean_squared_error(y1,y_poly_pred))\n",
    "        r2 = r2_score(y1,y_poly_pred)\n",
    "        if r2>0.95:\n",
    "            print('Coordinate '+ str(j+1))\n",
    "            for k in range(len(polynomial_features.get_feature_names())):\n",
    "                if np. abs(model_2.coef_.flatten()[k])>0.05:\n",
    "                    print(polynomial_features.get_feature_names()[k], np.round(model_2.coef_.flatten()[k],2))\n",
    "            print('_'*100)\n",
    "            print()\n",
    "            break\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# And for the Hamiltonian:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "x0^2 1.04\n",
      "x0 x1 -0.41\n",
      "x1^2 1.9\n",
      "x2^2 0.99\n",
      "x3^2 0.99\n",
      "____________________________________________________________________________________________________\n",
      "\n"
     ]
    }
   ],
   "source": [
    "\n",
    "x=torch.FloatTensor(data['x'])\n",
    "batch_size=np.shape(x)[0]\n",
    "y=model_scnn_2.forward_12(x)\n",
    "y=y.detach().numpy()\n",
    "x=x.numpy()\n",
    "\n",
    "\n",
    "for i in range(3):\n",
    "    polynomial_features= PolynomialFeatures(degree=i)\n",
    "    x_poly = polynomial_features.fit_transform(x)\n",
    "\n",
    "    model_2 = LinearRegression()\n",
    "    model_2.fit(x_poly, y)\n",
    "    y_poly_pred = model_2.predict(x_poly)\n",
    "    rmse = np.sqrt(mean_squared_error(y,y_poly_pred))\n",
    "    r2 = r2_score(y,y_poly_pred)\n",
    "    if r2>0.95:\n",
    "        for k in range(len(polynomial_features.get_feature_names())):\n",
    "            if np. abs(model_2.coef_.flatten()[k])>0.05:\n",
    "                print(polynomial_features.get_feature_names()[k], np.round(model_2.coef_.flatten()[k],2))\n",
    "        print('_'*100)\n",
    "        print()\n",
    "        break\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# The analytic expressions for the conserved quantities for Model 3:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Coordinate 1\n",
      "x2 -0.06\n",
      "x0^2 0.19\n",
      "x0 x1 -0.51\n",
      "x1^2 1.27\n",
      "x2^2 0.16\n",
      "x2 x3 -0.23\n",
      "x3^2 0.65\n",
      "____________________________________________________________________________________________________\n",
      "\n",
      "Coordinate 2\n",
      "x0^2 -2.42\n",
      "x0 x1 0.69\n",
      "x1^2 -3.91\n",
      "x2^2 -2.33\n",
      "x2 x3 -0.15\n",
      "x3^2 -2.05\n",
      "____________________________________________________________________________________________________\n",
      "\n"
     ]
    }
   ],
   "source": [
    "\n",
    "x=torch.FloatTensor(data['x'])\n",
    "batch_size=np.shape(x)[0]\n",
    "y=model_scnn_3.forward_1(x)[0]\n",
    "y=y.detach().numpy()\n",
    "x=x.numpy()\n",
    "\n",
    "for j in range(DIM):\n",
    "    y1=y[:,j:j+1]\n",
    "    \n",
    "    for i in range(3):\n",
    "        polynomial_features= PolynomialFeatures(degree=i)\n",
    "        x_poly = polynomial_features.fit_transform(x)\n",
    "\n",
    "        model_2 = LinearRegression()\n",
    "        model_2.fit(x_poly, y1)\n",
    "        y_poly_pred = model_2.predict(x_poly)\n",
    "        rmse = np.sqrt(mean_squared_error(y1,y_poly_pred))\n",
    "        r2 = r2_score(y1,y_poly_pred)\n",
    "        if r2>0.95:\n",
    "            print('Coordinate '+ str(j+1))\n",
    "            for k in range(len(polynomial_features.get_feature_names())):\n",
    "                if np. abs(model_2.coef_.flatten()[k])>0.05:\n",
    "                    print(polynomial_features.get_feature_names()[k], np.round(model_2.coef_.flatten()[k],2))\n",
    "            print('_'*100)\n",
    "            print()\n",
    "            break\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# And for the Hamiltonian:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "x0^2 1.04\n",
      "x0 x1 -0.41\n",
      "x1^2 1.9\n",
      "x2^2 1.0\n",
      "x3^2 0.99\n",
      "____________________________________________________________________________________________________\n",
      "\n"
     ]
    }
   ],
   "source": [
    "\n",
    "x=torch.FloatTensor(data['x'])\n",
    "batch_size=np.shape(x)[0]\n",
    "y=model_scnn_3.forward_12(x)\n",
    "y=y.detach().numpy()\n",
    "x=x.numpy()\n",
    "\n",
    "\n",
    "for i in range(3):\n",
    "    polynomial_features= PolynomialFeatures(degree=i)\n",
    "    x_poly = polynomial_features.fit_transform(x)\n",
    "\n",
    "    model_2 = LinearRegression()\n",
    "    model_2.fit(x_poly, y)\n",
    "    y_poly_pred = model_2.predict(x_poly)\n",
    "    rmse = np.sqrt(mean_squared_error(y,y_poly_pred))\n",
    "    r2 = r2_score(y,y_poly_pred)\n",
    "    if r2>0.95:\n",
    "        for k in range(len(polynomial_features.get_feature_names())):\n",
    "            if np. abs(model_2.coef_.flatten()[k])>0.05:\n",
    "                print(polynomial_features.get_feature_names()[k], np.round(model_2.coef_.flatten()[k],2))\n",
    "        print('_'*100)\n",
    "        print()\n",
    "        break\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# The analytic expressions for the conserved quantities for Model 4:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Coordinate 1\n",
      "x0 -0.06\n",
      "x0^2 0.87\n",
      "x0 x1 0.25\n",
      "x1^2 0.36\n",
      "x2^2 0.86\n",
      "x2 x3 0.32\n",
      "x3^2 0.2\n",
      "____________________________________________________________________________________________________\n",
      "\n",
      "Coordinate 2\n",
      "x0^2 -1.32\n",
      "x0 x1 0.52\n",
      "x1^2 -2.43\n",
      "x2^2 -1.26\n",
      "x3^2 -1.27\n",
      "____________________________________________________________________________________________________\n",
      "\n"
     ]
    }
   ],
   "source": [
    "\n",
    "x=torch.FloatTensor(data['x'])\n",
    "batch_size=np.shape(x)[0]\n",
    "y=model_scnn_4.forward_1(x)[0]\n",
    "y=y.detach().numpy()\n",
    "x=x.numpy()\n",
    "\n",
    "for j in range(DIM):\n",
    "    y1=y[:,j:j+1]\n",
    "    \n",
    "    for i in range(3):\n",
    "        polynomial_features= PolynomialFeatures(degree=i)\n",
    "        x_poly = polynomial_features.fit_transform(x)\n",
    "\n",
    "        model_2 = LinearRegression()\n",
    "        model_2.fit(x_poly, y1)\n",
    "        y_poly_pred = model_2.predict(x_poly)\n",
    "        rmse = np.sqrt(mean_squared_error(y1,y_poly_pred))\n",
    "        r2 = r2_score(y1,y_poly_pred)\n",
    "        if r2>0.95:\n",
    "            print('Coordinate '+ str(j+1))\n",
    "            for k in range(len(polynomial_features.get_feature_names())):\n",
    "                if np. abs(model_2.coef_.flatten()[k])>0.05:\n",
    "                    print(polynomial_features.get_feature_names()[k], np.round(model_2.coef_.flatten()[k],2))\n",
    "            print('_'*100)\n",
    "            print()\n",
    "            break\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# And for the Hamiltonian:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "x0^2 1.05\n",
      "x0 x1 -0.41\n",
      "x1^2 1.89\n",
      "x2^2 1.0\n",
      "x3^2 0.99\n",
      "____________________________________________________________________________________________________\n",
      "\n"
     ]
    }
   ],
   "source": [
    "\n",
    "x=torch.FloatTensor(data['x'])\n",
    "batch_size=np.shape(x)[0]\n",
    "y=model_scnn_4.forward_12(x)\n",
    "y=y.detach().numpy()\n",
    "x=x.numpy()\n",
    "\n",
    "\n",
    "for i in range(3):\n",
    "    polynomial_features= PolynomialFeatures(degree=i)\n",
    "    x_poly = polynomial_features.fit_transform(x)\n",
    "\n",
    "    model_2 = LinearRegression()\n",
    "    model_2.fit(x_poly, y)\n",
    "    y_poly_pred = model_2.predict(x_poly)\n",
    "    rmse = np.sqrt(mean_squared_error(y,y_poly_pred))\n",
    "    r2 = r2_score(y,y_poly_pred)\n",
    "    if r2>0.95:\n",
    "        for k in range(len(polynomial_features.get_feature_names())):\n",
    "            if np. abs(model_2.coef_.flatten()[k])>0.05:\n",
    "                print(polynomial_features.get_feature_names()[k], np.round(model_2.coef_.flatten()[k],2))\n",
    "        print('_'*100)\n",
    "        print()\n",
    "        break\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "# The analytic expressions for the Hamiltonian of the HNN:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "x0^2 1.05\n",
      "x0 x1 -0.41\n",
      "x1^2 1.91\n",
      "x2^2 1.0\n",
      "x3^2 1.0\n",
      "____________________________________________________________________________________________________\n",
      "\n"
     ]
    }
   ],
   "source": [
    "\n",
    "x=torch.FloatTensor(data['x'])\n",
    "y=model_hnn.forward(x)[1]\n",
    "y=y.detach().numpy()\n",
    "x=x.numpy()\n",
    "\n",
    "\n",
    "for i in range(3):\n",
    "    polynomial_features= PolynomialFeatures(degree=i)\n",
    "    x_poly = polynomial_features.fit_transform(x)\n",
    "\n",
    "    model_2 = LinearRegression()\n",
    "    model_2.fit(x_poly, y)\n",
    "    y_poly_pred = model_2.predict(x_poly)\n",
    "    rmse = np.sqrt(mean_squared_error(y,y_poly_pred))\n",
    "    r2 = r2_score(y,y_poly_pred)\n",
    "    if r2>0.95:\n",
    "        for k in range(len(polynomial_features.get_feature_names())):\n",
    "            if np. abs(model_2.coef_.flatten()[k])>0.05:\n",
    "                print(polynomial_features.get_feature_names()[k], np.round(model_2.coef_.flatten()[k],2))\n",
    "        print('_'*100)\n",
    "        print()\n",
    "        break\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "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.7.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
