{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "dad02194-9d83-4760-8681-3c345460221f",
   "metadata": {},
   "outputs": [],
   "source": [
    "# --! include root folder into PYTHONPATH --!\n",
    "\n",
    "import os\n",
    "import sys\n",
    "\n",
    "thisdir = os.getcwd()\n",
    "rootdir = os.path.abspath(os.path.join(thisdir, '..', '..'))\n",
    "sys.path.append(rootdir)\n",
    "\n",
    "# --! import python libraries and kind framework --!\n",
    "\n",
    "import time\n",
    "\n",
    "import torch\n",
    "import numpy as np\n",
    "\n",
    "from typing import Tuple\n",
    "from matplotlib import pyplot as plt\n",
    "\n",
    "import utils_data\n",
    "import utils_plot"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "96946911-26c2-479c-9ae3-d38cf6df3778",
   "metadata": {},
   "outputs": [],
   "source": [
    "# --! declare a class that implements online DMD as designed in https://github.com/haozhg/odmd --!\n",
    "\n",
    "class OnlineDMD:\n",
    "    \"\"\"OnlineDMD is a class that implements online dynamic mode decomposition\n",
    "    The time complexity (multiply–add operation for one iteration) is O(4n^2),\n",
    "    and space complexity is O(2n^2), where n is the state dimension.\n",
    "\n",
    "    Algorithm description:\n",
    "        At time step t, define two matrix X(t) = [x(1),x(2),...,x(t)],\n",
    "        Y(t) = [y(1),y(2),...,y(t)], that contain all the past snapshot pairs,\n",
    "        where x(t), y(t) are the n dimensional state vector, y(t) = f(x(t)) is\n",
    "        the image of x(t), f() is the dynamics.\n",
    "\n",
    "        Here, if the (discrete-time) dynamics are given by z(t) = f(z(t-1)),\n",
    "        then x(t), y(t) should be measurements correponding to consecutive\n",
    "        states z(t-1) and z(t).\n",
    "\n",
    "        We would like to update the DMD matrix A(t) recursively\n",
    "        by efficient rank-1 updating online DMD algrithm.\n",
    "        An exponential weighting factor can be used to place more weight on\n",
    "        recent data.\n",
    "\n",
    "    Usage:\n",
    "        odmd = OnlineDMD(n, weighting)\n",
    "        odmd.initialize(Xp, Yp) # optional\n",
    "        odmd.update(x, y)\n",
    "        evals, modes = odmd.computemodes()\n",
    "\n",
    "    properties:\n",
    "        n: state dimension\n",
    "        weighting: weighting factor in (0,1]\n",
    "        timestep: number of snapshot pairs processed (i.e., current time step)\n",
    "        A: DMD matrix, size n by n\n",
    "\n",
    "    methods:\n",
    "        initialize(Xp, Yp), initialize online DMD algorithm with first p\n",
    "                            snapshot pairs stored in (Xp, Yp), this func call is optional\n",
    "        update(x, y), update DMD computation when new snapshot pair (x,y)\n",
    "                            becomes available\n",
    "        computemodes(), compute and return DMD eigenvalues and DMD modes\n",
    "\n",
    "    Authors:\n",
    "        Hao Zhang\n",
    "        Clarence W. Rowley\n",
    "\n",
    "    References:\n",
    "        Zhang, Hao, Clarence W. Rowley, Eric A. Deem, and Louis N. Cattafesta.\n",
    "        \"Online dynamic mode decomposition for time-varying systems.\"\n",
    "        SIAM Journal on Applied Dynamical Systems 18, no. 3 (2019): 1586-1609.\n",
    "\n",
    "    Date created: April 2017\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(self, n: int, weighting: float = 0.9) -> None:\n",
    "        \"\"\"Creat an object for online DMD\n",
    "        Usage: odmd = OnlineDMD(n, weighting)\n",
    "\n",
    "        Args:\n",
    "            n (int): state dimension x(t) as in  z(t) = f(z(t-1)) or y(t) = f(t, x(t))\n",
    "            weighting (float, optional): exponential weighting factor\n",
    "                smaller value allows more adpative learning, \n",
    "                but too small weighting may result in model identification instability (relies only on limited recent snapshots). \n",
    "                Defaults to 0.9.\n",
    "        \"\"\"\n",
    "        assert isinstance(n, int) and n >= 1\n",
    "        weighting = float(weighting)\n",
    "        assert weighting > 0 and weighting <= 1\n",
    "\n",
    "        self.n = n\n",
    "        self.weighting = weighting\n",
    "        self.timestep = 0\n",
    "        self.A = np.zeros([n, n])\n",
    "        self._P = np.zeros([n, n])\n",
    "        # initialize\n",
    "        self._initialize()\n",
    "        self.ready = False\n",
    "\n",
    "    def _initialize(self) -> None:\n",
    "        \"\"\"Initialize online DMD with epsilon small (1e-15) ghost snapshot pairs before t=0\"\"\"\n",
    "        epsilon = 1e-15\n",
    "        alpha = 1.0 / epsilon\n",
    "        self.A = np.random.randn(self.n, self.n)\n",
    "        self._P = alpha * np.identity(self.n)  # inverse of cov(X)\n",
    "\n",
    "    def initialize(self, Xp: np.ndarray, Yp: np.ndarray) -> None:\n",
    "        \"\"\"Initialize online DMD with first p (p >= n) snapshot pairs stored in (Xp, Yp)\n",
    "        Usage: odmd.initialize(Xp, Yp)\n",
    "\n",
    "        Args:\n",
    "            Xp (np.ndarray): 2D array, shape (n, p), matrix [x(1),x(2),...x(p)]\n",
    "            Yp (np.ndarray): 2D array, shape (n, p), matrix [y(1),y(2),...y(p)]\n",
    "        \"\"\"\n",
    "\n",
    "        assert Xp is not None and Yp is not None\n",
    "        Xp, Yp = np.array(Xp), np.array(Yp)\n",
    "        assert Xp.shape == Yp.shape\n",
    "        assert Xp.shape[0] == self.n\n",
    "        # necessary condition for over-constrained initialization\n",
    "        p = Xp.shape[1]\n",
    "        assert p >= self.n and np.linalg.matrix_rank(Xp) == self.n\n",
    "\n",
    "        weight = np.sqrt(self.weighting) ** range(p - 1, -1, -1)\n",
    "        Xqhat, Yqhat = weight * Xp, weight * Yp\n",
    "        self.A = Yqhat.dot(np.linalg.pinv(Xqhat))\n",
    "        self._P = np.linalg.inv(Xqhat.dot(Xqhat.T)) / self.weighting\n",
    "        self.timestep += p\n",
    "\n",
    "        if self.timestep >= 2 * self.n:\n",
    "            self.ready = True\n",
    "\n",
    "    def update(self, x: np.ndarray, y: np.ndarray) -> None:\n",
    "        \"\"\"Update the DMD computation with a new pair of snapshots (x,y)\n",
    "        Here, if the (discrete-time) dynamics are given by z(t) = f(z(t-1)),\n",
    "        then (x,y) should be measurements correponding to consecutive states\n",
    "        z(t-1) and z(t).\n",
    "        Usage: odmd.update(x, y)\n",
    "\n",
    "        Args:\n",
    "            x (np.ndarray): 1D array, shape (n, ), x(t) as in y(t) = f(t, x(t))\n",
    "            y (np.ndarray): 1D array, shape (n, ), y(t) as in y(t) = f(t, x(t))\n",
    "        \"\"\"\n",
    "        assert x is not None and y is not None\n",
    "        x, y = np.array(x), np.array(y)\n",
    "        assert np.array(x).shape == np.array(y).shape\n",
    "        assert np.array(x).shape[0] == self.n\n",
    "\n",
    "        # compute P*x matrix vector product beforehand\n",
    "        Px = self._P.dot(x)\n",
    "        # compute gamma\n",
    "        gamma = 1.0 / (1 + x.T.dot(Px))\n",
    "        # update A\n",
    "        self.A += np.outer(gamma * (y - self.A.dot(x)), Px)\n",
    "        # update P, group Px*Px' to ensure positive definite\n",
    "        self._P = (self._P - gamma * np.outer(Px, Px)) / self.weighting\n",
    "        # ensure P is SPD by taking its symmetric part\n",
    "        self._P = (self._P + self._P.T) / 2\n",
    "\n",
    "        # time step + 1\n",
    "        self.timestep += 1\n",
    "\n",
    "        if self.timestep >= 2 * self.n:\n",
    "            self.ready = True\n",
    "\n",
    "    def computemodes(self) -> Tuple[np.ndarray, np.ndarray]:\n",
    "        \"\"\"Compute and return DMD eigenvalues and DMD modes at current time step\n",
    "        Usage: evals, modes = odmd.computemodes()\n",
    "\n",
    "        Raises:\n",
    "            Exception: if Model not ready! Have not seen enough data!\n",
    "\n",
    "        Returns:\n",
    "            Tuple[np.ndarray, np.ndarray]: DMD eigenvalues (2D array) and DMD modes (2D array)\n",
    "        \"\"\"\n",
    "        if not self.ready:\n",
    "            raise Exception(f\"Model not ready! Have not seen enough data!\")\n",
    "        evals, modes = np.linalg.eig(self.A)\n",
    "        return evals, modes"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "ecd57dc7-5872-4ed3-8dd3-575afb896cbf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1\n",
      "120\n"
     ]
    }
   ],
   "source": [
    "datadir            = '../../data/baselines/tesla_test'\n",
    "timeseries_nsample = 250_000\n",
    "timeseries         = utils_data.read_datafile(f'{datadir}/eval', timeseries_nsample)\n",
    "timeseries         = timeseries.reshape(-1, timeseries.shape[-1])\n",
    "\n",
    "sample_begin       = 159_585\n",
    "nsample            = 121\n",
    "timeseries         = timeseries[sample_begin:sample_begin+nsample, :]\n",
    "\n",
    "# --! normalize loaded data\n",
    "datanormalized = False\n",
    "if datanormalized:\n",
    "    mean         = torch.mean(timeseries, dim=0, keepdim=True)\n",
    "    timeseries   = timeseries - mean\n",
    "    scaler       = utils_data.minmax_scaler(feature_range=(-1, 1))\n",
    "    timeseries   = scaler.fit_transform(timeseries, dim=0)\n",
    "\n",
    "timeseries = timeseries.numpy().T\n",
    "x          = timeseries[:, :-1]\n",
    "y          = timeseries[:, 1:]\n",
    "\n",
    "n = x.shape[0]\n",
    "m = x.shape[1]\n",
    "\n",
    "print(n)\n",
    "print(m)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "572898eb-2f16-44ac-8b52-ea21d24a58de",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "inf >> the shape of matrix A in online DMD is (1, 1)\n"
     ]
    }
   ],
   "source": [
    "# --! run prediction and forecast loops using online DMD --!\n",
    "\n",
    "q    = 80\n",
    "odmd = OnlineDMD(n, 0.6)\n",
    "odmd.initialize(x[:, :q], y[:, :q])\n",
    "\n",
    "# --! create an output buffer\n",
    "odmd_o = np.empty((n, m))\n",
    "\n",
    "# --! define state x and save the current state x in the output buffer\n",
    "state        = x[:, 0]\n",
    "odmd_o[:, 0] = state\n",
    "\n",
    "# --! prediction start\n",
    "pre_start = 80\n",
    "\n",
    "# --! run a typical +1 prediction loop with a regular matrix A update\n",
    "for j in range(1, pre_start):\n",
    "\n",
    "    # --! update matrix A with a new measurement\n",
    "    meas  = x[:, j]\n",
    "    odmd.update(state, meas)\n",
    "\n",
    "    # --! make +1 prediction step using the updated matrix A\n",
    "    state = odmd.A.dot(state)\n",
    "    odmd_o[:, j] = state\n",
    "\n",
    "# --! define forecast horizon\n",
    "pre_horizon = 40\n",
    "pre_end     = pre_start + pre_horizon\n",
    "\n",
    "print(f'inf >> the shape of matrix A in online DMD is {odmd.A.shape}')\n",
    "\n",
    "# --! run a forecast loop\n",
    "for j in range(pre_start, pre_end):\n",
    "    state = odmd.A.dot(state)\n",
    "    odmd_o[:, j] = state"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "6177b5f8-544d-4006-999b-ab155d542b9e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAhYAAAE6CAYAAABKwRsDAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjkuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8hTgPZAAAACXBIWXMAAA9hAAAPYQGoP6dpAACIoklEQVR4nO3dd3hTZfvA8e9Jmqa7pXtPoKyyWvYGZYuKoOACceD6qThBRUFFnK++TgQV9VVwoCgiCsree1Nmd2npHnQnOb8/0kZKW2hLZvt8risX7cnJOXcOaXLnGfcjybIsIwiCIAiCYAQKSwcgCIIgCELLIRILQRAEQRCMRiQWgiAIgiAYjUgsBEEQBEEwGpFYCIIgCIJgNCKxEARBEATBaERiIQiCIAiC0YjEQhAEQRAEoxGJhSAIgiAIRiMSCyuTlpbGE088wZAhQ/Dw8ECSJL766itLh9UkX331FZIkkZSUZNg2dOhQhg4d2uRjvf766/z66691tm/atAlJkti0aVOz47QFkiQxb948s5zrn3/+QZIkJEkiJyenzv0JCQlMnDgRDw8PXFxcuP766zlw4ECjjl1ZWcmDDz5IQEAASqWS7t27Gzl669PQa9ccmvv3Zo2SkpLqvA/W9x5jLtOnTzf8nUiShLOzM+Hh4UyYMIGlS5dSUVFR5zFDhw5FkiQiIyOpr9j1li1bDMer73nW3BwcHPD392fYsGEsXLiQrKwsUz7VZhOJhZU5e/Ys3333Hfb29owdO9bS4RjNJ598wieffNLkxzX05tyzZ0927txJz549jRCdcPHiRe6//34CAwPrvT87O5tBgwZx+vRpvvzyS3788UfKy8sZOnQop06duurxP/30Uz777DNeeOEFtm3bxv/+9z9jPwWrY8nEorl/b7Zi3Lhx7Ny5k4CAAIuc39HRkZ07d7Jz505Wr17NK6+8grOzM/fffz+xsbGkpaXVeYyrqyuJiYls2LChzn1ffvklbm5uDZ5v6dKl7Ny5k7///puPP/6Y7t278+abb9KxY0f++ecfoz43o5AFsxkyZIg8bdq0K+6j1WoNP+/du1cG5KVLl5o2sGoajUYuLy+/5uMsXbpUBuTExMRrPpazs/NVr1lLBsgvv/xykx/38ssvy2FhYY3e/5FHHpF79Oghv/jiizIgZ2dn17r/mWeekVUqlZyUlGTYVlhYKHt7e8u33nrrVY9/3333yY6Ojo2OpzFKS0uNejxja+2vXWNJTEw06/vg1UybNk12dnau9761a9fKKpVK7tOnT63tQ4YMkTt37iz37dtXvv3222vdV1RUJDs5Ocn3339/nedZ8166d+/eOudKTk6WQ0JCZFdXVzkzM/Pan5gRiRYLK6NQXNt/ydChQ+nSpQtbt26lb9++ODo6EhQUxNy5c9FqtYb9apoX33rrLV577TUiIiJQq9Vs3LgRgH379jFhwgQ8PT1xcHCgR48e/Pjjj3XOt2vXLgYMGICDgwOBgYHMmTOHqqqqeuO6vGm2oqKCV155hY4dO+Lg4ICXlxfDhg1jx44dgL4boKSkhK+//trQFFhzjIa6QlatWkW/fv1wcnLC1dWV66+/np07d9baZ968eUiSxPHjx5k6dSru7u74+fkxY8YMCgsLr3qN//77b2688UaCg4NxcHCgbdu2zJw5s073QVPOU1RUxP3334+XlxcuLi6MHj2a06dPXzUWY9i6dSuLFy/m888/R6lU1rvPypUrGT58OGFhYYZtbm5uTJw4kd9//x2NRtPg8SVJ4vPPP6esrKxOc295eTlz5swhIiICe3t7goKCeOSRRygoKKh1jPDwcMaPH88vv/xCjx49cHBwYP78+QBkZmYyc+ZMgoODsbe3JyIigvnz59eJ6WqvN4CPP/6YwYMH4+vri7OzMzExMbz11lt1XtMHDx5k/Pjx+Pr6olarCQwMZNy4cYZvqld67danoddzfd0ACQkJTJkyhcDAQNRqNX5+fowYMYJDhw4Z9rn8763mOO+88w7/+c9/iIiIwMXFhX79+rFr16468SxZsoT27dujVqvp1KkTy5YtY/r06YSHhzf4HGrodDreeustOnTogFqtxtfXl7vvvrvOt/ia96q9e/cyaNAgnJyciIyM5I033kCn013xHA11tzb2eEVFRTz99NO1XndPPPEEJSUlV31+VzJy5Ejuv/9+du/ezZYtW+rcP2PGDH755Zdar+/vv/8egClTpjTpXKGhobz77rsUFxfz2WefXVPcxmZn6QAE48vMzGTKlCnMnj2bV155hT/++IPXXnuN/Px8Pvroo1r7fvDBB7Rv35533nkHNzc32rVrx8aNGxk9ejR9+vRh0aJFuLu78/3333PbbbdRWlrK9OnTAThx4gQjRowgPDycr776CicnJz755BOW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hvuvZPz48Rw7doyoqKh6X6s1iUVjXm/1/d/KssySJUuu+Jhu3brx3nvv4eHhUatgWFO+edfMtjhy5Eit7fUNpr1U+/btefHFF4mJiWl0sbIriY6Oxt/fv84MsJSUlFqzZxoyfPhwoO7/6d69e4mPj2/U/6mpjR8/nnPnzhm+5F1+a8zMl4b8/ffffP755/Tv37/OYPFLTZs2jRtuuIFnnnmGoKCgZp0rJSWFp59+Gnd3d2bOnNnckE1CtFhYoRUrVgD6aWWgn/rp4uICwKRJk676eC8vLx566CFSUlJo3749a9asYcmSJTz00EOEhoZe9fGfffYZY8aMYdSoUUyfPp2goCDy8vKIj4/nwIEDhm+4L774IqtWrWL48OG89NJLODk58fHHHzdqytbUqVNZunQpDz74IKdOnWLYsGHodDp2795Nx44dDVOvYmJi2LRpE7///jsBAQG4uroaWkwupVAoeOutt7jjjjsYP348M2fOpKKigrfffpuCggLeeOONq8bUGB06dCAqKorZs2cjyzKenp78/vvv/P33380+5siRIxk8eDDPPvssJSUlxMXFsX37dpMXkarvW05N//KAAQNqfXN7+umn+d///se4ceN45ZVXUKvVvPHGG5SXlze7Muj111/PqFGjeO655ygqKmLAgAGGWSE9evTgrrvuuuoxXnnlFf7++2/69+/PY489RnR0NOXl5SQlJbFmzRoWLVpEcHBwo15v119/Pfb29kydOpVnn32W8vJyPv30U/Lz82udc/Xq1XzyySfcdNNNhkqKNVMIr7/+esN+jX3tgr7b67rrrmPhwoW0adOGsLAw1q9fzy+//FJrvyNHjvDoo48yefJk2rVrh729PRs2bODIkSPNaim5nEKhYP78+cycOZNJkyYxY8YMCgoKmD9/PgEBAVdtXYyOjuaBBx7gww8/RKFQMGbMGMOskJCQEGbNmnXNMV6rJ554gp9//pnBgwcza9Ysunbtik6nIyUlhXXr1vHUU0/Rp0+fKx5Dp9MZpulWVFSQkpLCn3/+yY8//kjHjh3rnZp/qcDAwCYVTzt27JhhLEhWVhZbt25l6dKlKJVKVq5cWac70eIsN2609WlMgSxZ1hdFaujWmHN07txZ3rRpkxwXFyer1Wo5ICBAfv7552vN8qgZaf3222/Xe5zDhw/Lt956q+zr6yurVCrZ399f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",
      "text/plain": [
       "<Figure size 600x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# --! display prediction and forecast results --!\n",
    "\n",
    "dt = 0.001\n",
    "t  = np.arange(0., pre_end*dt, dt)\n",
    "\n",
    "plt.figure(figsize=(6,3))\n",
    "mino = np.min(odmd_o)\n",
    "maxo = np.max(odmd_o)\n",
    "plt.title(f'+1 prediction and +{pre_horizon} forecast using online DMD')\n",
    "for j in range(n):\n",
    "    plt.plot(t, timeseries[j, :pre_end], label='$x_{' + f'{j+1}' + '}$')\n",
    "    plt.plot(t, odmd_o[j, :pre_end], linestyle='dashed', label='$\\\\hat{x_{' + f'{j+1}' + '}}$')\n",
    "    plt.plot([t[pre_start], t[pre_start]], [mino, maxo], linestyle='dotted', color='gray')\n",
    "plt.legend()\n",
    "plt.xlabel('Time [s]')\n",
    "plt.ylim((-6, -1))\n",
    "plt.ylabel('Amplitude')\n",
    "\n",
    "plt.show()\n",
    "\n",
    "datasaved=True\n",
    "if datasaved:\n",
    "    savedir = '../../savedata/baselines/tesla'\n",
    "    truth = timeseries[:1, :pre_end]\n",
    "    step  = torch.arange(truth.shape[1]).reshape(-1, 1)\n",
    "    pre   = odmd_o[:1, pre_start:pre_end].T\n",
    "\n",
    "    savedata = np.expand_dims(np.concatenate([step, truth.T], axis=1), 0)\n",
    "    utils_data.write_datafile(f'{savedir}/kind_vs_odmd_truth2', savedata, delim=' ')\n",
    "\n",
    "    savedata = np.expand_dims(np.concatenate([step[pre_start:], pre], axis=1), 0)\n",
    "    utils_data.write_datafile(f'{savedir}/kind_vs_odmd_forecast2', savedata, delim=' ')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "20f4dece-5ff3-420a-bdce-93177569cdea",
   "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.13.0"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
