{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "%matplotlib inline"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import evaluation.evaluate as evaluate\n",
    "import matplotlib.pyplot as plt\n",
    "import math_utils.matrix_utils as matrix_utils\n",
    "import models.regression_models as reg_models\n",
    "import models.PLDS as PLDS\n",
    "import numpy as np\n",
    "import os\n",
    "import PGLDM.PGLDM as PGLDM\n",
    "import PGLDM.SDP_optimize as SDP_optimize\n",
    "import visualization.visualization as viz_utils"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "data = np.load(os.path.join(os.getcwd(), 'sample_data', 'sample_sys.npz'), allow_pickle=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "### Prepare the data.\n",
    "Y, Z, params = data['Y'], data['Z'], data['params'][()]\n",
    "Ytrain, Ztrain = Y[:int(1e6), :].astype('float64'), Z[:int(1e6), :]\n",
    "Ytest, Ztest = Y[-int(1e6):, :].astype('float64'), Z[-int(1e6):, :]\n",
    "\n",
    "##### Without loss of generality, demean the Gaussian observations.\n",
    "Zmean = np.mean(Ztrain, axis=0, keepdims=True)\n",
    "Ztrain -= Zmean\n",
    "Ztest -= Zmean"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The optimal value is 46.354580132833554 with status: optimal\n",
      "The optimal value is 6.152852655540296 with status: optimal\n"
     ]
    }
   ],
   "source": [
    "n1 = nx = 4\n",
    "i = 10 # horizon\n",
    "\n",
    "### PGLDM Stage 1 only.\n",
    "pgldm_params = PGLDM.PGLDM(i, nx, Ytrain.T, Z=Ztrain.T, n1=n1, Z_horizon=i)\n",
    "pgldm_params = SDP_optimize.optimize(pgldm_params)\n",
    "\n",
    "### PLDSID.\n",
    "pldsid_params = PGLDM.PGLDM(i, nx, Ytrain.T, n1=0)\n",
    "pldsid_params = SDP_optimize.optimize(pldsid_params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "### Extracting shared and disjoint modes in Y, as well as the unencoded (i.e., disjoint) modes in Z.\n",
    "shared_eigs = np.linalg.eigvals(matrix_utils.extract_block(params['A'], params['rev_statesz']))\n",
    "disjoint_dims = np.delete(params['rev_statesy'], params['rev_statesz'])\n",
    "y_only_eigs = np.linalg.eigvals(matrix_utils.extract_block(params['A'], disjoint_dims))\n",
    "z_only_eigs = np.linalg.eigvals(params['eps_sysz']['A'])\n",
    "\n",
    "pgldm_eigs = np.linalg.eigvals(pgldm_params['A'])\n",
    "pldsid_eigs = np.linalg.eigvals(pldsid_params['A'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "### Visualizing the identified modes by both learning algorithms vs. ground truth modes.\n",
    "colors = { # Colors for visualizing the modes.\n",
    "  'shared': '#39b54a', 'y only': '#e5542c', 'z only': '#0071bd', 'pgldm': '#39b54a', 'pldsid': '#e5542c'}\n",
    "\n",
    "viz_utils.plot_eigenvalues(\n",
    "  [shared_eigs, y_only_eigs, z_only_eigs, pgldm_eigs, pldsid_eigs],\n",
    "  'n1=nx=4', labels=['shared', 'y only', 'z only', 'pgldm', 'pldsid'],\n",
    "   markers=['o', 'o', 'o', 'x', 'x'],\n",
    "   edgecolors=[colors['shared'], colors['y only'], colors['z only'], None, None],\n",
    "   facecolors=['None', 'None', 'None', colors['pgldm'], colors['pldsid']],\n",
    "   alpha_vals=[0.4, 0.4, 0.4, 0.8, 0.8],\n",
    "   bound_lims_to_circle=False, show_legend=True, legend_location='upper left', fig=None, ax=None)\n",
    "plt.xlim([0.8, 1.0])\n",
    "plt.ylim([-0.35, 0.35])\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Warning: no baseline b is provided...will set b=0.\n"
     ]
    }
   ],
   "source": [
    "### Demonstarting decoding performance. This block will take some time to run as we're testing on 1e6 contiguous\n",
    "### samples and the point-process filter runs recursively.\n",
    "\n",
    "#### PGLDM\n",
    "pgldm_sys = PLDS.update_missing_params_and_construct_PLDS(pgldm_params)\n",
    "\n",
    "other_out = {'Z': pgldm_params['Cz']}\n",
    "pgldm_y_pred, _, other_est = pgldm_sys.predict(Ytest, other_out)\n",
    "pgldm_z_pred = other_est['Z']\n",
    "pgldm_z_evalres = evaluate.evaluate_results(Ztest, pgldm_z_pred, ['CC'])['CC']\n",
    "\n",
    "#### Handle prediction covariances to compute AUC.\n",
    "Pp = other_est.get('Pp', None) # Prediction error covariance.\n",
    "assert (Pp is not None)\n",
    "log_rate_covariances = Pp @ pgldm_params['C'].T # (time, nx, nr)\n",
    "log_rate_covariances = np.transpose(log_rate_covariances, (0, 2, 1)) @ pgldm_params['C'].T # (time, nr, nr)\n",
    "pgldm_y_evalres = evaluate.evaluate_results(\n",
    "                      Ytest, pgldm_y_pred, ['AUC'], predicted_covariances=log_rate_covariances)['AUC']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Warning: no baseline b is provided...will set b=0.\n"
     ]
    }
   ],
   "source": [
    "#### PLDSID\n",
    "pldsid_sys = PLDS.update_missing_params_and_construct_PLDS(pldsid_params)\n",
    "\n",
    "##### First fit Cz.\n",
    "_, x_pred, _ = pldsid_sys.predict(Ytrain, {}) # predict xTrain\n",
    "reg_kwargs = {'fit_intercept': True}\n",
    "regression = reg_models.RegressionModel(reg_models.RegressionMethod.OLS_REG)\n",
    "## Must be (num_samples, num_features)\n",
    "regression.fit(x_pred, Ztrain, **reg_kwargs)\n",
    "reg_weights, reg_intercept = regression.weights()\n",
    "pldsid_params['Cz'] = reg_weights\n",
    "\n",
    "##### Now predict test data.\n",
    "other_out = {'Z': pldsid_params['Cz']}\n",
    "pldsid_y_pred, _, other_est = pldsid_sys.predict(Ytest, other_out)\n",
    "pldsid_z_pred = other_est['Z']\n",
    "\n",
    "pldsid_z_evalres = evaluate.evaluate_results(Ztest, pldsid_z_pred, ['CC'])['CC']\n",
    "\n",
    "#### Handle prediction covariances to compute AUC.\n",
    "Pp = other_est.get('Pp', None) # Prediction error covariance.\n",
    "assert (Pp is not None)\n",
    "log_rate_covariances = Pp @ pldsid_params['C'].T # (time, nx, nr)\n",
    "log_rate_covariances = np.transpose(log_rate_covariances, (0, 2, 1)) @ pldsid_params['C'].T # (time, nr, nr)\n",
    "pldsid_y_evalres = evaluate.evaluate_results(\n",
    "                    Ytest, pldsid_y_pred, ['AUC'], predicted_covariances=log_rate_covariances)['AUC']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "PGLDM Z prediction CC 0.462162531444924, PLDSID Z prediction CC 0.4096155726706072\n",
      "PGLDM Y prediction AUC 0.544758912909446, PLDSID Y prediction AUC 0.5592364828837525\n"
     ]
    }
   ],
   "source": [
    "print(f'PGLDM Z prediction CC {pgldm_z_evalres.mean()}, PLDSID Z prediction CC {pldsid_z_evalres.mean()}')\n",
    "print(f'PGLDM Y prediction AUC {pgldm_y_evalres.mean()}, PLDSID Y prediction AUC {pldsid_y_evalres.mean()}')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Including stage 2 and using the true shared and disjoint latent state dimensions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The optimal value is 6.034635240563187 with status: optimal\n",
      "The optimal value is 2.548239739182375 with status: optimal\n"
     ]
    }
   ],
   "source": [
    "n1 = 4; nx = 8\n",
    "i = 10 # horizon\n",
    "\n",
    "# PGDLM Stage 1 + 2 only.\n",
    "pgldm_params = PGLDM.PGLDM(i, nx, Ytrain.T, Z=Ztrain.T, n1=n1, Z_horizon=i)\n",
    "pgldm_params = SDP_optimize.optimize(pgldm_params)\n",
    "\n",
    "# PLDSID.\n",
    "pldsid_params = PGLDM.PGLDM(i, nx, Ytrain.T, n1=0)\n",
    "pldsid_params = SDP_optimize.optimize(pldsid_params)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "### Visualizing the identified modes by both learning algorithms vs. ground truth modes.\n",
    "\n",
    "pgldm_eigs = np.linalg.eigvals(pgldm_params['A'])\n",
    "pldsid_eigs = np.linalg.eigvals(pldsid_params['A'])\n",
    "\n",
    "viz_utils.plot_eigenvalues(\n",
    "  [shared_eigs, y_only_eigs, z_only_eigs, pgldm_eigs, pldsid_eigs],\n",
    "  'n1=4, nx=16', labels=['shared', 'y only', 'z only', 'pgldm', 'pldsid'],\n",
    "   markers=['o', 'o', 'o', 'x', 'x'],\n",
    "   edgecolors=[colors['shared'], colors['y only'], colors['z only'], None, None],\n",
    "   facecolors=['None', 'None', 'None', colors['pgldm'], colors['pldsid']],\n",
    "   alpha_vals=[0.4, 0.4, 0.4, 0.8, 0.8],\n",
    "   bound_lims_to_circle=False, show_legend=True, legend_location='upper left', fig=None, ax=None)\n",
    "plt.xlim([0.8, 1.0])\n",
    "plt.ylim([-0.35, 0.35])\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "psid-python",
   "language": "python",
   "name": "psid-python"
  },
  "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.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
