{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "e1c143d1-0d08-4de2-8891-04e5ef37362b",
   "metadata": {},
   "source": [
    "# Simulation studies: ARPLN-ICA\n",
    "\n",
    "This notebook aims at reproducing the simulated studies of *Independent Component Discovery in Temporal count data*.\n",
    "\n",
    "Through 3 scenarios, we evaluate the recoverability of the mixing function $\\Gamma$ under finite sample regime. Comparison between ARPLN-ICA inferred with the AR variational proxy and MF is assessed, as well as standard ICA baselines (UwedgeICA / Picard)."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "067d64ef-e16a-4016-8f3c-c6fdc9c3d3d6",
   "metadata": {},
   "source": [
    "# Effect of $\\Gamma$ the mixing function"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "82656599-f53f-429c-af56-81e6e1b83f10",
   "metadata": {},
   "source": [
    "## Definition of the 3 scenarios\n",
    "\n",
    "First, we generate the 3 scenarios where only $\\Gamma$ varies. Seed 0 serves as a common seed for all scenarios to generate the model parameters; only $\\Gamma$ changes."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "25364465-e83d-4112-a525-4a937c1e10ba",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "from plntime.models.plnica import PLNICA\n",
    "\n",
    "seed = 0\n",
    "seeds = [0, 1, 2, 3]\n",
    "\n",
    "n = 150\n",
    "T = 20\n",
    "K = 12\n",
    "d = 5\n",
    "C = 1\n",
    "\n",
    "pcounts = torch.zeros((n, T, K))\n",
    "\n",
    "bases = {}\n",
    "\n",
    "for i, seed in enumerate(seeds):\n",
    "    base = PLNICA(\n",
    "        counts=pcounts,\n",
    "        latent_size=d,\n",
    "        n_dynamics=C,\n",
    "        offsets='zero',\n",
    "        bias=False,\n",
    "        seed=seed,\n",
    "    )\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        \n",
    "        if i == 0:\n",
    "            base.Gamma.W.data = torch.randn_like(base.Gamma.W.data) * 0.6 + 4.5\n",
    "        elif i == 1:\n",
    "            # High coherence, mostly active and balanced prevalence\n",
    "            base.Gamma.W.data = torch.abs(torch.randn_like(base.Gamma.W.data)) * 1.6 + torch.randn_like(base.Gamma.W.data) * 0.6\n",
    "            base.offsets = base.offsets + 0.7\n",
    "        elif i == 2:\n",
    "            # Moderate coherence, mostly active but imbalance prevalence\n",
    "            base.Gamma.W.data = torch.randn_like(base.Gamma.W.data) * 2.8 + (torch.randn_like(base.Gamma.W.data).abs().clamp_max(1.) * 0.9) + torch.randn_like(base.Gamma.W.data).abs() * 5.3\n",
    "            base.offsets = base.offsets + 1.5 # old 1.2\n",
    "        elif i == 3:\n",
    "            # Zero-inflation in observations, low-excitation\n",
    "            base.Gamma.W.data = torch.randn_like(base.Gamma.W.data) * 0.6 * torch.randn(1)\n",
    "            \n",
    "        if seed == 0:            \n",
    "            # --- Dynamics mixture (irrelevant if C=1, but keep it sane)\n",
    "            base.pi.data = torch.ones_like(base.pi.data) / base.pi.numel()\n",
    "        \n",
    "            # --- AR dynamics\n",
    "            rho = 0.95\n",
    "            base.B.data = torch.sigmoid(torch.ones_like(base.B.data) * rho + 3 * torch.randn_like(base.B.data))\n",
    "            base.B.data[0] = -0.5 * base.B.data[0]\n",
    "            base.B.data[3] = -0.5 * base.B.data[3]\n",
    "            base.B.data[-1] = -2.0 * base.B.data[3]\n",
    "        \n",
    "            # --- Process noise\n",
    "            base.psi.data = torch.ones_like(base.psi.data) * 0.08\n",
    "        \n",
    "            # --- Latent AR bias \n",
    "            base.b.data = torch.randn_like(base.b.data) * 0.5 + 0.5\n",
    "            base.b.data[0] += 2.0\n",
    "            base.b.data[1] += 0.4\n",
    "        \n",
    "            # --- Initial latent mean/cov\n",
    "            base.bar_b.data = base.b.data + torch.randn_like(base.bar_b.data) * 0.05\n",
    "            base.bar_psi.data = base.psi.data + torch.randn_like(base.bar_psi.data).abs() * 0.3\n",
    "        \n",
    "            base.offsets = 0. * torch.randn_like(base.offsets)\n",
    "            base.M.data = torch.randn_like(base.M.data)*0\n",
    "        else:\n",
    "            base_ref = bases[0]\n",
    "            base.pi.data = base_ref.pi.data.clone()\n",
    "            base.B.data = base_ref.B.data.clone()\n",
    "            base.psi.data = base_ref.psi.data.clone()\n",
    "            base.b.data = base_ref.b.data.clone()\n",
    "            base.bar_b.data = base_ref.bar_b.data.clone()\n",
    "            base.bar_psi.data = base_ref.bar_psi.data.clone()\n",
    "            base.M.data = base_ref.M.data.clone()\n",
    "\n",
    "        bases[seed] = base"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "505c2afa-1b90-43a0-8f51-411f5c9dd128",
   "metadata": {},
   "source": [
    "For each scenario, we visualise the statistics of interest:\n",
    "- support of the log-intensities indicate excitation.\n",
    "- Gram correlation coefficients indicate components entanglement.\n",
    "- Conditioning indicates the difficulty to numerically estimate $\\Gamma$ in standard numerical analysis."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "e65cf740-b2ee-4675-8db3-d9d1b85b0cc2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Gram correlations (mean offdiag): 0.6473861105249811\n",
      "Gram correlations (mean offdiag): 0.8709341367372953\n",
      "Gram correlations (mean offdiag): 0.5131865799084707\n"
     ]
    },
    {
     "data": {
      "image/png": "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sWKFhg8frk8++URNmjTJ0LbffPNNzZkzR0WKFFGrVq1UtGhRnTx5Ups3b1b16tVlsVgkXSuwH3nkER09elT33nuv2rdvr2PHjumnn35SRESEPvnkE/vr5XrvvPOOtmzZovDwcDVp0kSrV6/WtGnTlJycrBEjRji0nTx5smbNmqWSJUuqTZs2KliwoLZt26a3335bO3fu1Pvvv++0/h9//FGbNm1SixYt1LNnT8XHx0u6VtQ+9thj2rJli+6++2517dpVycnJioiI0KBBgzR27Fj17t07S3kzuu88zQLvOXTokHr27KnY2FiFh4frrrvuUlRUlL799lutXbtWX375pSpWrChJCgsL0+eff67IyEjVrFlTkmSz2ewnKtKOWU59HBYWdsMcmXnvbd++vTZs2KAlS5bo888/16OPPirp2km08ePHq1ixYnrzzTfl5+cnSTp79qx69+6tgwcP6u6771aPHj1ks9l08OBBffLJJ+rfv7/bT906d+4s6VpR3LlzZ/vv9oIFC9rbfPzxxzp69Khq166tUqVK6cKFC9q4caPGjRunQ4cO2U8elClTRkOGDNH06dNVpkwZ+7qla/VEerZt26YBAwYoOTlZ7dq1U5kyZfTHH39o7ty5WrdunRYsWOB09tus985bzsglYmJiDIvFYvTv398wDMPo0KGD0b59e/v8f//916hWrZrx2muvGYZhGDVq1DDCw8Md1jFt2jTDYrEYo0aNMmw2m336nj17jOrVqxv16tUzLl68aJ/+/vvvGxaLxXjppZcc1rN+/XrDYrEYFovF+Pbbb+3TY2NjjTp16hhNmzY1jh075rDMDz/8YFgsFuPVV191ek4Wi8V47733XD7vo0ePOk2Lj483OnToYNStW9e4dOmS0/pGjx7tcl0LFiwwLBaL8fzzzxtJSUn26VeuXDGefPJJw2KxGLt27XK57PX27dtnVK1a1WjSpIkRExPjMM9msxn//POP/bGn+7xu3bpGdHS0fXpiYqLRtm1bo2rVqg7rj4qKMqpVq2Y8+OCDRmxsrEOWmTNnGhaLxZg9e7Z92m+//Wbf54sWLcryPsps3szsO7OOV06Q+tpu3bq18f7777v8N3r0aMNisRhjx451WDZ1+vX7+9dffzUsFovRqVMnhz506tQpo1GjRi77UXh4uGGxWIyBAwcaV65ccVpX6nvTjaxZs8awWCxGhw4dnF6zycnJxunTp+2Pn3/+ecNisRiTJ092aLdu3TrDYrEYbdq0MVJSUpyea8uWLY1Tp07Zp589e9aoV6+eERoa6pB948aN9uwJCQn26TabzRg3bpxhsViMH3/80T7922+/NSwWi1G1alVj06ZNTs9typQphsViMd59912H/n7x4kWjS5cuRvXq1R1e45nNm5l9l9ksuDnS/u5OT58+fQyLxWJ8/fXXDtPnz59vWCwWo2/fvvZp58+fN6pWrWoMGDDAPm337t2GxWIx+vXrZ1gsFuPgwYP2eT179jRq1arl8HpyJ7PvvfHx8UabNm2MGjVqGHv37jWuXr1qPPLII4bVajXWr1/vsO6hQ4caFovFmDJlitN2T58+bSQnJ9sfh4eHO9Uxqb9zfvvtN5fZXdUMycnJxn//+1/j7rvvNo4fP+4wz2KxGL1793a5rtTfl++//759WkpKitG6dWvDYrE4Pbe33nrLsFgsxpgxYxymm/Xe6Q25brhIqoceekhRUVHauXOnpGtDJK5evZruUJGlS5cqT548euaZZ+x/VUpStWrV1LlzZ124cEG//PKLU/unn37aYT1NmzZVw4YNnda/bNkyxcfHa+TIkfa/MFO1b99e1atX1/Lly52WK168uJ566imXmcuWLes0LTg4WF26dNHFixedPipLz/z585U/f36NHz/e4S/loKAg+9kiV/nS+vrrr2Wz2TR8+HDdeeedDvP8/PwcLjDL7D5P1bdvX1WqVMn++LbbblOHDh1ks9m0Z88ehyxXr17V2LFjnS7OGDBggEJCQvTDDz84rb969eouXyue7qPM5M3ovjPreOUkR48e1fTp013+W7JkSYbX891330mSBg8erHz58tmnlyhRQn379k132TFjxjh8MtKwYUOVKVNGu3fvztC2v/zyS0nSiy++6PSaDQwMVLFixSRdOxO7fPlyFSlSRAMHDnRo17x5czVu3FhHjhzR9u3bnbYxaNAglShRwv44JCRErVq1UkJCgg4dOmSfPn/+fEnSa6+9pvz589un+/n52fusq9dYq1at1KhRI4dpNptNX331lcqVK6enn37aob8XKFBAgwcPVnJyssvhdhnNm9F9l5Us8I4TJ04oMjJSVapUUbdu3Rzm9ejRQ5UqVdJvv/1m/9SxUKFCuvvuu7Vt2zb73cZSz1an/s7+7bffJEmX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sKZOneryI8rExMRMFeCpF0+mvZ/0999/b/8Y7HqFChWSn5+f/vnnH5fr69OnjxITE/XSSy+5/KgrJibG6V7CrnTv3l0BAQF69913dfz4cYd5qR+zpcrsPs+snj17KjAwUK+99prTfVSlax/9pY4hywiz9pE7mdl3NztLbpb6qdeMGTMcrnM4ffq05s6de1O33atXL0nShAkT7MMaUl29etV+higoKEjt27fXuXPnNHPmTId269ev18aNG1W+fPl0x5DfSOotQF944QWX97c9ffq0oqOjM7SuwMBA9ejRQ8ePH9dbb73lsog4cOBAlj4pyOi+uxVZYK7SpUsrLCxMUVFRTveiXrBggaKjo9WgQQP7eOxUYWFhunz5subPn6+qVauqSJEikq5dE1O+fHnNmTMnw+OxU2X2vXfTpk2aM2eO7rnnHg0ZMkTh4eHq1auXtmzZolmzZtnbFStWTG3btrVfwJ3W2bNn7WO+3Ul9fqkXgF6vdOnSkpxrhi1btuibb75xub7ChQu7rRlcqVOnjsqVK6f169fr119/dZj34YcfKi4uTu3bt8/QBabZQa4eLiJdu9gg9R6ZN/L4448rIiJCy5YtU3R0tBo2bKizZ89q5cqVSklJ0WuvvebwEceAAQO0atUqLVy4UFFRUfb7ZP/4449q0aKF1q1b57D+kJAQTZkyRcOGDVOnTp3UtGlTVapUSUlJSTp+/Li2bNmi0NBQtxdupdWpUyd9/PHHev311xUZGanSpUtr//792rx5s9q2bauff/7ZoX1wcLBq1qyprVu36tlnn1X58uXl7++vTp06qUyZMurevbt27typJUuWaPv27WrUqJFKlCihs2fP6uDBg9q5c6cmT57sdNeLtKxWq1544QW9/vrr6tChg1q1aqUyZcro9OnT2rZtm5o3b64XX3zRo32eWRaLRePHj9fLL7+s++67T82bN1fZsmWVkJCgY8eO2e8n+uqrr2ZofWbtI3cys+9udpbcrFGjRurQoYN++OEHdezYUa1bt1ZSUpJWrlypWrVqae3atW6HcWVV8+bN1b9/f3366adq166dWrduraJFi+rUqVPavHmz+vfvb7/X7rPPPqutW7fqww8/1I4dO1S7dm0dP35cP/74o/Lly6c33njjhuOT09OsWTMNGjRIH3zwgdq2baumTZuqdOnSiouL05EjR/T7779r+PDhqly5cobW9/TTT2vv3r2aN2+eIiIiVK9ePftzO3DggP766y8tWLAg3XHv6cnMvrvZWZA5Bw4ccPsFT5UqVdITTzyhl19+WT179tTYsWO1du1aValSRVFRUVqzZo1CQkL08ssvOy0bFhamWbNmKTY2Vg8++KDTvIULF9p/zqjMvPfGxsZq9OjRCg4O1qRJk+zXEowePVpbt27VtGnT1LBhQ9WuXVuSNH78eEVFRemjjz7S+vXr1aBBAxmGocOHD2vjxo369ddf7cNKXAkLC5Ofn5+mTJmiqKgoFSxYUIUKFVLv3r0VHh6uMmXK6JNPPlFUVJTuuusuHTp0SOvWrVPr1q2dvsROunbGf+XKlRo0aJCqVasmf39/tWzZ0u0nzP7+/po4caIGDBigJ554wn6f7B07dmjLli0qV66c0xcBZWe5vsjOjLx58+rzzz/Xxx9/rBUrVuizzz5Tvnz5VL9+fT355JNOV+jnz59f8+bN05QpU7Rq1Srt3btXVapU0dSpU3Xx4kWnIluSWrRooSVLlmj27NnavHmzNm3apPz586tkyZLq0qWL05tAekqVKqX58+frnXfe0ebNm3X16lVVr15dn376qU6ePOlUZEvS22+/rYkTJ2rdunW6ePGiDMNQ3bp1VaZMGfn5+enNN99Us2bN9M0332jdunW6dOmSQkJCVL58eY0ePdrlrQld6d27t+666y7NmTNHGzZsUEJCgooWLaratWs7fLlMZve5J7p166aqVavqs88+09atW7V27VoVKFBApUuX1qOPPqr//Oc/GV6XmfvInYzuu1uRJTd76623VLlyZX377beaN2+eSpUqpX79+qlhw4b219DNMnr0aIWGhmr+/Pn66aefdOXKFRUvXlwNGjRw+KKrkJAQLVy4UB988IHWrFljv/agVatWGjJkSKau/XBn2LBhql+/vubOnavNmzfr4sWLKlKkiO68804NGTIk3Wtd0goKCtLHH3+sRYsWaenSpfr555+VlJSkYsWKqXLlyurevXuWM2d0392KLMi4f//91+2Y4NRvAa5UqZK+/fZbTZ8+XRs2bFBERIRuv/12denSRUOGDHG6Na50bVx2njx5lJycbB8ikiq1yM6fP7/LO0q5k5n33hdeeEGnT5/WO++843D3jLx582ry5Mnq2rWrRo0apaVLl6pAgQL2Pj179mz9+OOP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r/ZPyXf7qK/2T4k+mhP5ah/11f4p+W4fpX+mjyLbA7t371bfvn3tjydOnChJ6ty5s958801vxdJXX30lSerTp4/D9IkTJzp8lOMNZ8+e1ejRo/Xvv/+qYMGCslqtmj17tho3buzVXL7sn3/+0ciRIxUXF6eQkBDVrVtXCxcuVEhIiLej+TRf7Z+S7/ZR+qdn6KOe8dU+6qv9U6KPesIX+qefYRjGLdsaAAAAkAtwn2wAAADAZBTZAAAAgMkosgEAAACTUWQDAAAAJqPIBgAAAExGkQ0AAACYjCIbAAAAMBlFNgAAAGAyimwAAADAZHytug87evSoPvnkE23atEn//vuv8uTJI4vFovvvv1+PPPKIbrvtNm9HzNa2b9+uTZs2qV+/fipUqJC34yCboX/eXPRPZBV99Oaij94YRbaPWrdunYYNG6agoCB16tRJFotFycnJ+v333/XOO+/o77//1muvvebtmNnajh07NH36dHXu3Jk3CGQK/fPmo38iK+ijNx999MYosn1QTEyMRowYodKlS+vzzz9XiRIl7PN69eqlI0eOaN26dd4LCORi9E/At9FH4SsYk+2DPvnkE126dEkTJkxweHNIVb58efXr18/+eO/evRowYIDq1Kmj0NBQ9evXT3/88YfDMtOmTZPVatWhQ4f0zDPPqG7dumrQoIHeffddGYahkydPauDAgapTp44aN26sTz/91OXy0dHRGjZsmOrUqaOwsDC9/vrrunLlilPGzGQ6cuSInn/+edWrV09169bVmDFjlJiY6LTOU6dOacyYMWrUqJFq1Kih9u3ba9GiRR6tc9q0aXr77bclSa1atZLVapXVatWxY8dcHpMXX3xRYWFhWrFihdO8iIgIWa1W3rRzCfon/RO+jT5KH/UVnMn2QWvXrlXZsmVVp06dG7aNiopSr169FBwcrAEDBigwMFALFixQnz59NH/+fNWuXduh/YgRI1S5cmWNGjVKERER+vDDD1WkSBF9/fXXatCggZ555hl9//33euutt1SzZk3Vr1/fYfnhw4erTJkyGjVqlP744w/NmzdPFy5csHc2TzINHz5cd955p0aOHKm9e/fqm2++UUhIiJ599ll7mzNnzqhbt27y8/NTr169FBISovXr1+vFF19UfHy8Hn300Uyts02bNjp8+LB++OEHjRkzRrfffrskKSQkxOV+vu++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",
      "text/plain": [
       "<Figure size 850x400 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "sns.set_style('whitegrid')\n",
    "\n",
    "fig, axes = plt.subplots(1, len(bases)-1, figsize=(8.5, 4), sharey=True)\n",
    "\n",
    "titles = [\"Moderate coherence\", \"High coherence\", \"Low excitation\"]\n",
    "mappable = None\n",
    "\n",
    "for i, seed in enumerate([2, 1, 3]):\n",
    "    base = bases[seed]\n",
    "    with torch.no_grad():\n",
    "        G = base.Gamma.normalize()\n",
    "        Gram = (G.T @ G).abs().detach()\n",
    "        mask = torch.tril(torch.ones_like(Gram, dtype=torch.bool), diagonal=-1)\n",
    "        Gram_tril = Gram[mask]\n",
    "        print(\"Gram correlations (mean offdiag):\", Gram_tril.max().item())\n",
    "\n",
    "    hm = sns.heatmap(\n",
    "        Gram, ax=axes[i], vmin=0., vmax=1., cmap='Blues',\n",
    "        cbar=False, linewidth=0., rasterized=True\n",
    "    )\n",
    "    mappable = hm.collections[0]\n",
    "\n",
    "    xticks = axes[i].get_xticks()\n",
    "    axes[i].set_xticks(xticks)\n",
    "    axes[i].set_xticklabels([str(int(t) + 1) for t in xticks])\n",
    "\n",
    "    yticks = axes[i].get_yticks()\n",
    "    axes[i].set_yticks(yticks)\n",
    "    axes[i].set_yticklabels([str(int(t) + 1) for t in yticks])\n",
    "\n",
    "    axes[i].set_title(titles[i], fontsize=14)\n",
    "    axes[i].set_xlabel(r\"Component $\\gamma_j$\", fontsize=12)\n",
    "    if i == 0:\n",
    "        axes[i].set_ylabel(r\"Component $\\gamma_i$\", fontsize=12)\n",
    "\n",
    "cbar = fig.colorbar(mappable, ax=axes, orientation='horizontal', fraction=0.08, pad=0.25)\n",
    "cbar.set_label(r\"$|\\langle \\gamma_i\\,, \\gamma _j \\rangle|$\", fontsize=14)\n",
    "\n",
    "plt.savefig('./figures/coherence_gram.pdf', bbox_inches='tight')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "d5d8e104-c05e-41fb-a697-a14d4d4624c3",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAxYAAAEiCAYAAABkykQ1AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjkuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8hTgPZAAAACXBIWXMAAA9hAAAPYQGoP6dpAABJ5ElEQVR4nO3dd1gU5/o38O+C9KIiKFiw70JUpFlQlGhETdQYe6KoiT3WHGNUjp4osaBpnoglWGKsscToid1oYtTEFiNqDNgFFCwLImUpyj7vH3nZn+sC7jKLu4vfz3V5hXlm5pl7J9yz3FOekQkhBIiIiIiIiCSwMnUARERERERk+VhYEBERERGRZCwsiIiIiIhIMhYWREREREQkGQsLIiIiIiKSjIUFERERERFJxsKCiIiIiIgkY2FBRERERESSsbAgIiIiIiLJWFi8JBQKBWJiYkwdhllTKBT45JNPTB0GkUE6duyI6dOnl3nd0aNHGzmisjGnWIjoxZg+fTo6duz4wrYn5XhJ+mFh8QL98MMPUCgUUCgU+OOPP3TmCyEQFhYGhUJRob5g7927h5iYGMTHx5s6FCKzVnSMuHjxYrHzBw8ejO7du7/gqIjImJ6X5y+z3NxcxMTE4NSpU2Xu488//0RMTAwyMzONGBnpq5KpA3gZ2dnZYffu3QgODtZqP336NO7evQtbW1sTRVY+7t+/jyVLlqBWrVrw9fU1dThEFcr+/fshk8lMHQYRkcHmzJkDIYRmOjc3F0uWLMH48ePRqlWrMvV57tw5LFmyBL169YKrq6vWPB4vyx+vWJhAWFgY9u/fjydPnmi17969G02aNIGHh4eJItNPfn4+1Gq1qcOwSNx3ZGy2trawsbExdRgW4cmTJygoKDB1GET0/9nY2LzQk6k8XpY/FhYm0K1bN2RkZOC3337TtBUUFODAgQPo0aNHseuoVCosWLAAYWFhaNq0Kbp06YLVq1drVfpF/cyfPx+tW7dGQEAAxowZg7t37xbb57179xAZGYk2bdqgadOm6NatG77//nutZU6dOgWFQoE9e/Zg0aJFaNeuHZo3b47s7GxkZGRg4cKF6NGjBwICAhAYGIgRI0YgISFBa/2+ffsCACIjIzW3gv3www+aZc6fP4/hw4cjKCgIzZs3R0REBM6ePavXvszPz0dMTAy6dOmCZs2aITQ0FOPHj0dSUpLB+67IoUOH0L17d80+OXr0qNH3nb6fOyYmBgqFAomJiZg+fTqCg4MRFBSEyMhI5Obm6sT1v//9D3379kXz5s3RokULDBo0CMePH9da5tdff8XAgQPh7++PgIAAjBo1ClevXtVrf5P5Ke6e4YSEBERERMDPzw/t27fHsmXLsH37digUCty+fVunjz/++AN9+/ZFs2bN8Nprr2Hnzp16bVutVmPt2rXo0aMHmjVrhtatW2P48OFat3g8efIES5cuRadOndC0aVN07NgRX375ZYl/4OsTS2ZmJubNm6fJ6fDwcKxYsUKraL99+zYUCgVWr16Nb7/9Fp06dUKzZs1w/fp1AMD169cxceJEtGzZEs2aNUPv3r1x+PBhre0U3bJy9uxZREdHo3Xr1vD398e4ceOQnp6uE9evv/6KiIgIzfGwT58+2LVrl9YyUo539HL5+++/MWLECAQGBiIgIABDhw5FXFycZn5mZiZ8fX2xbt06TVt6ejp8fHzQqlUrre+4WbNmoW3bts/d5vO+2/Ly8tC1a1d07doVeXl5mvaMjAyEhobi7bffRmFhoab9eTnx9DMWt2/fRkhICABgyZIlmr8Xip4PTUhIwPTp0/Haa6+hWbNmaNu2LSIjI/Hw4UNNfzExMfj0008BAK+99pqmj6LjXnHHy+TkZM2xoHnz5ujfvz+OHDmitUzR9/nevXuxfPlytG/fHs2aNcPQoUORmJj43P36MuGtUCZQq1Yt+Pv7Y8+ePQgLCwMAHD16FFlZWXjjjTewfv16reWFEHj//fc1f6T7+vri2LFj+PTTT3Hv3j38+9//1iw7Y8YM/Pjjj+jevTsCAwNx8uRJjBo1SicGpVKJ/v37QyaTYdCgQXBzc8PRo0cxY8YMZGdn491339VaftmyZbCxscHw4cNRUFAAGxsbXLt2DYcOHULXrl1Ru3ZtKJVKbNmyBREREdizZw9q1KiBhg0bYuLEiVi8eDEGDBiAoKAgAEBgYCAA4MSJExg5ciSaNm2K8ePHQyaT4YcffsDQoUOxadMm+Pn5lbgfCwsLMXr0aJw4cQLdunXDkCFDkJOTg99++w1XrlyBt7e3QfsOAM6ePYuDBw9i4MCBcHJywvr16zFx4kT88ssvqFq1qtH2naGf+4MPPkDt2rUxefJk/P3339i2bRvc3Nzw0UcfaZZZsmQJYmJiEBAQgIkTJ8LGxgbnz5/HyZMnERoaCgDYuXMnpk+fjtDQUEyZMgW5ubn47rvvMHDgQOzYsQO1a9cucX/Ti5OdnV3sH66PHz9+7rr37t3D0KFDAQCjRo2Co6Mjtm3bVuJZwcTEREyaNAl9+/ZFr169sH37dkyfPh1NmjRB48aNS93WjBkz8MMPP6B9+/bo27cvCgsL8ccff+D8+fNo1qwZAGDmzJnYsWMHunTpgvfeew8XLlxAbGwsrl+/jqVLlxocS25uLiIiInDv3j28/fbb8PLywrlz5/Dll1/iwYMHmDFjhlafP/zwA/Lz89G/f3/Y2tqicuXKuHr1Kt555x3UqFEDI0eOhKOjI/bt24dx48YhJiYG4eHhWn3MnTsXrq6uGD9+PO7cuYO1a9fik08+wX//+1+t7fz73/9G48aNMXr0aLi4uCA+Ph7Hjh3TnDCScryjl8vVq1cxaNAgODk5YcSIEahUqRK2bNmCwYMHY8OGDWjevDlcXV3RuHFj/PHHHxgyZAiAf54vkMlkyMjIwLVr1zR5c/bsWc33b0n0+W6zt7fHwoUL8c4772DRokWIjIwEAHzyySfIyspCdHQ0rK2tAeiXE09zc3PD7NmzMXv2bISHh2vyUKFQAAB+//13JCcno3fv3vDw8MDVq1exdetWXLt2DVu3boVMJkN4eDhu3bqF3bt3IzIyUvO97ebmVuJnfvvtt5Gbm4vBgwejatWq2LFjB95//30sXrxY51iwcuVKyGQyDBs2DNnZ2Vi1ahWmTJmCbdu26fX/9aUg6IXZvn27kMvl4sKFC2LDhg0iICBA5ObmCiGEmDhxohg8eLAQQogOHTqIUaNGadb76aefhFwuF8uWLdPqb8KECUKhUIjExEQhhBDx8fFCLpeL2bNnay03efJkIZfLxeLFizVt//73v0Xbtm1Fenq61rL/+te/RFBQkCaukydPCrlcLl577TVNW5H8/HxRWFio1ZacnCyaNm0qlixZomm7cOGCkMvlYvv27VrLqtVq0blzZzFs2DChVqs17bm5uaJjx47ivffeK2lXCiGE+P7774VcLhdr1qzRmVfUn777Tggh5HK5aNKkiVZb0T5dv369pk3qvjPkcy9evFjI5XIRGRmpta1x48aJli1baqZv3bolfHx8xLhx43T+nxRtIzs7WwQHB4uZM2dqzX/w4IEICgrSaacXr+gYUdq/bt26aa3ToUMHMW3aNM30nDlzhEKhEH///bem7eHDh6Jly5ZCLpeL5ORkrXXlcrk4c+aMpi0tLU00bdpULFiwoNRYT5w4IeRyuZgzZ47OvKLfuaL8mTFjhtb8BQsWCLlcLk6cOGFwLEuXLhX+/v7i5s2bWn1+/vnnwtfXV6SkpAgh/jkWyeVyERgYKNLS0rSWHTp0qOjevbvIz8/XinnAgAGic+fOmrai/x/vvvuuVq7Onz9f+Pr6iszMTCGEEJmZmSIgIED069dP5OXlFbsvpB7vqOJ4+m+BkowdO1Y0adJEJCUladru3bsnAgICxKBBgzRtUVFRok2bNprp6OhoMWjQIBESEiI2bdokhPgn/xUKhVi7dm2pcen73SaEEF988YXw8fERZ86cEfv27RNyuVx8++23mvn65IQQQkybNk106NBBM52Wlqbz90qRZ/8GEUKI3bt36xw3Vq1apXOsK/Ls8XLevHk662dnZ4uOHTuKDh06aL5Pi77PX3/9da3jxtq1a4VcLheXL1/W2dbLirdCmcjrr7+O/Px8/PLLL8jOzsaRI0dKvA3q6NGjsLa2xuDBg7Xahw0bBiGE5ladX3/9FQB0lis6e1lECIGDBw+iY8eOEEIgPT1d8y80NBRZWVm4dOmS1jpvvfUW7O3ttdpsbW1hZfXPr1BhYSEePnwIR0dH1K9fH3///fdz90F8fDxu3bqFHj164OHDh5oYVCoVQkJCcObMmVKfRzh48CCqVq2KiIgInXlFD2fpu++KtGnTBt7e3pppHx8fODs7Izk5GYBx9l1ZPvfbb7+tNR0cHIyMjAzNbVWHDh2CWq3GuHHjNP9Pnt0Xv//+OzIzM9GtWzetuK2srNC8eXNJo3CQcX388cdYs2aNzr+iM3elOXbsGPz9/bUGSqhSpUqJx5dGjRppDSTh5uaG+vXra37nS3Lw4EHIZDKMHz9eZ17R71zRMem9997Tmj9s2DCt+YbEsn//fgQFBcHV1VXr97hNmzYoLCzEmTNntPrs3Lmz1tnKjIwMnDx5Eq+//rrmylB6ejoePnyI0NBQ3Lp1C/fu3dPqo+gsbpHg4GAUFhbizp07AIDffvsNOTk5GDVqFOzs7IrdF1KPd/TyKCwsxG+//YZOnTqhTp06mvbq1auje/fuOHv2rObYHxwcDKVSiRs3bgD451bC4OBgBAcHa0afPHv2LIQQOgPGPM3Q77bx48ejUaNGmDZtGqKiotCyZUvNVRNAv5ww1NPfo/n5+UhPT0fz5s0BQOd7V1+//vor/Pz8tPaNk5MTBgwYgDt37uDatWtay/fu3Vvr6m/Res87Xr5MeCuUibi5uSEkJAS7d+9GXl4eCgsL0aVLl2KXvXPnDqpXrw5nZ2et9oYNG2rmF/3XyspK6w9jAGjQoIHWdHp6OjIzM7FlyxZs2bKl2G0+extGcbfIqNVqrFu3Dps2bcLt27e17qusUqVKsf0+7datWwCAadOmlbhMVlYWKleuXOy8pKQk1K9fH5UqlfxrrO++K+Ll5aXTR+XKlTXD1hlj35Xlc9esWVNrftFIF48ePYKzszOSkpJgZWWl+VzFKdrus4VmkWf3EZmOn5+f5laip1WuXFnrfuLi3LlzB/7+/jrtzx4XipT0O//o0aNSt5OUlITq1auXmuslHZM8PDzg6uqqd/49HUtiYiIuX76suRf7Wc/Lv6SkJAgh8NVXX+Grr74qto+0tDTUqFFDM11S/hUdF4qe6Srt1jGpxzt6eaSnpyM3Nxf169fXmdewYUOo1WqkpqaicePGmtubzp49C09PT8THx+ODDz6Am5sbvvnmGwD/FBvOzs7w8fEpdZuGfLfZ2tpi/vz56Nu3L+zs7DB//nytgkGfnDBURkYGlixZgr179yItLU1rXlZWVpn6TElJ0RQnTyv6uyklJQVyuVzT/rxjAbGwMKnu3bvjP//5D5RKJdq3b68zLFp5KTor9uabb6JXr17FLvPsmdFnr1YAwNdff42vvvoKffr0waRJk1C5cmVYWVlh/vz5JT4Y/bSiZaZOnVriMLSOjo7P7ceYiu4NfVZRrMbYd2X53M9ehXi2L30ULfvpp58WO/JYSZ+dKrYX8f9d3zOU+sSiVqvRtm1bjBgxotj59erV05p+Nv+KcnjYsGFo165dsX08WwgZM//M6XhHlq9GjRqoXbs2zpw5g1q1akEIAX9/f7i5uWHevHm4c+cOzp49i4CAgBJ/j4GyfbcVDQySn5+PxMRErasr5eGDDz7AuXPnMHz4cPj6+sLR0RFqtRojRowwKBelMMaxoKJjYWFC4eHhmDVrFuLi4rBo0aISl6tVqxZOnDiB7OxsrbPKRZc+a9WqpfmvWq1GUlKS1lWKouWKuLm5wcnJCWq1Gm3atClz/AcOHECrVq0wf/58rfbMzEzNA1NAyX9UFB2EnJ2dyxSHt7c3zp8/j8ePH5c4fJy++05fxth3Uj93cby9vaFWq3H9+vUS/2gp2m61atWMtl0yP7Vq1Sp2lJKnR0ozBm9vbxw/fhwZGRklXrUoOiYlJiZqXU1TKpXIzMw0OP+KtqtSqSTnn42NjVHzD/jngdu6deuWul1j5j1VTG5ubnBwcMDNmzd15t24cQNWVlZaV/eCg4Nx5swZ1K5dW3P7ro+PD1xcXHDs2DH8/fffmDBhwnO3ach3W0JCApYuXYrevXsjISEBM2fOxK5du+Di4gJAv5woTkl/Lzx69AgnTpzAhAkTtG6/LLoSqE8fxalZs2aJ+7loPhmGz1iYkJOTE2bPno0JEyaU+kr79u3bo7CwEBs3btRq//bbbyGTydC+fXvNcgB0RpVau3at1rS1tTW6dOmCAwcO4MqVKzrbK240muJYW1vrVOn79u3TuT/ZwcEBgO6lwqZNm8Lb2xvffPMNcnJyDI6jc+fOePjwoc5+Af7v7IG++05fxth3Uj93cTp16gQrKyssXbpU5z7ton3Rrl07ODs7IzY2ttjRhcqyXTI/oaGhiIuL03rTfUZGhs6wp1J17twZQggsWbJEZ17R71zRqHfPHoPWrFmjNd8Qr7/+Os6dO4djx47pzMvMzNR5P9CzqlWrhpYtW2LLli24f/++zvyy5EFoaCicnJwQGxuL/Px8rXlF+6I88p4qJmtra7Rt2xaHDx/WGh5aqVRi9+7dCAoK0jpRFhwcjDt37mDv3r2ae/6trKwQEBCANWvW4PHjx88dEcqQ77bHjx8jMjIS1atXx4wZMxAdHQ2lUql1klGfnChOSX8vlHQ189ljy9N96HN7VFhYGC5cuIBz585p2lQqFbZu3YpatWqhUaNGz+2DtPGKhYmVdMnxaR07dkSrVq2waNEi3LlzBwqFAr/99hsOHz6MoUOHas4M+Pr6onv37ti0aROysrIQEBCAkydPFnv28sMPP8SpU6fQv39/9OvXD40aNcKjR49w6dIlnDhxAqdPn35uXK+++iqWLl2KyMhIBAQE4MqVK9i1a5fO5VBvb2+4urpi8+bNcHJygqOjI/z8/FCnTh3MnTsXI0eORPfu3dG7d2/UqFED9+7dw6lTp+Ds7Iyvv/66xO2/9dZb2LlzJ6Kjo3HhwgUEBQUhNzcXJ06cwDvvvINOnTrpve8MIXXfWVlZSfrcxalbty7GjBmDZcuWYeDAgejcuTNsbW1x8eJFVK9eHR9++CGcnZ0xe/ZsTJ06Fb1798Ybb7wBNzc3pKSk4Ndff0VgYCA+/vhjg/cHmZcRI0bgxx9/xHvvvYeIiAjNcLNeXl7IyMgw2ltnW7dujZ49e2L9+vVITExEu3btoFarcfbsWbRq1QoRERHw8fFBr169sGXLFmRmZqJFixa4ePEiduzYgU6dOqF169YGb3f48OH4+eefMWbMGPTq1QtNmjRBbm4urly5ggMHDuDw4cMlDi1ZZNasWRg4cCB69OiB/v37o06dOlAqlYiLi8Pdu3fx448/GhSTs7MzIiMjMXPmTPTt2xfdu3eHq6srEhISkJeXh4ULF5ZL3pNl2759e7EF8pAhQ/DBBx/g999/x8CBAzFw4EBYW1tjy5YtKCgo0BpmHICmaLh58yYmT56saW/RogWOHj0KW1tbvYYy1ve7bfny5YiPj8e3336ruToybtw4/Pe//0XXrl0RFhamV04Ux97eHo0aNcK+fftQr149VKlSBY0bN4ZcLkeLFi2watUqPH78GDVq1MBvv/1W7Ht5mjRpAgBYtGgR3njjDdjY2KBDhw7F3mo4atQo7NmzByNHjsTgwYNRuXJl7Ny5E7dv30ZMTEypt49R8VhYWAArKyssX74cixcvxt69e/HDDz+gVq1amDp1qmZ0lSLz589H1apVsWvXLhw+fBitWrXCihUrdM4Muru7Y9u2bVi6dCl++uknfPfdd6hSpQoaNWqEKVOm6BXXmDFjkJubi127dmHv3r145ZVXEBsbiy+++EJrORsbGyxYsABffvklZs+ejSdPniA6Ohp16tRBq1atsGXLFixbtgwbNmyASqWCh4cH/Pz8MGDAgFK3b21tjZUrV2L58uXYvXs3Dh48iCpVqiAwMFBzL6gh+05fxth3Uj53SSZNmoTatWtjw4YNWLRoERwcHKBQKNCzZ0/NMj169ED16tWxYsUKrF69GgUFBahRowaCg4PRu3fvMm2XzIuXlxfWrVuHuXPnIjY2Fm5ubhg0aBAcHBwwd+5cnRFapIiOjoZCocD333+PTz/9FC4uLmjatCkCAgI0y8ydOxe1a9fGjh07cOjQIbi7u2P06NHFjialDwcHB6xfvx6xsbHYv38/du7cCWdnZ9SrVw8TJkzQ3IpRmkaNGmH79u1YsmQJduzYgYyMDLi5ueGVV17BuHHjyhRXv379UK1aNaxYsQLLli1DpUqV0KBBA6332pRH3pPl+u6774pt7927Nxo3boyNGzfiiy++QGxsLIQQ8PPzw2effabzsHGDBg1QrVo1pKWlaV2ZKPrZz89Pr7db6/PddunSJcTGxiIiIkLrxMCoUaNw+PBhzJw5E3v27IGrq6teOVGcuXPnYs6cOYiOjsbjx48xfvx4yOVyfPHFF5gzZw42bdoEIQTatm2LlStX6jwr5efnh0mTJmHz5s04duwY1Go1Dh8+XGxh4e7ujs2bN+Ozzz7Dhg0bkJ+fD4VCga+//hqvvvrqc/cZ6ZIJPnFCRFThzZs3D1u2bMG5c+f4oD4REZULXuMhIqpg8vLytKYfPnyIH3/8EUFBQSwqiIio3PBWKCKiCmbAgAFo2bIlGjZsCKVSie3btyM7Oxtjx441dWhERFSB8VYoIqIK5ssvv8SBAwdw9+5dyGQyvPLKKxg/fjyHOSUionLFwoKIiIiIiCTjMxZERERERCQZCwsiIiIiIpLspSsshBAoLCws9c2PRGQ5mNNEFQfzmciyvXSFhVqtRlxcHNRqtalDISIjYE4TVRzMZyLL9tIVFkREREREZHxmVVjExsaiT58+CAgIQEhICMaOHYsbN25oLZOfn4+oqCi0atUKAQEBmDBhApRKpYkiJiIiIiIiwMwKi9OnT2PQoEHYunUr1qxZgydPnmD48OFQqVSaZebPn49ffvkF//3vf7F+/Xrcv38f48ePN2HURERERERkVm/eXr16tdb0ggULEBISgkuXLqFFixbIysrC9u3b8fnnnyMkJATAP4XGG2+8gbi4OPj7+5sgaiIiIiIiMqsrFs/KysoCAFSuXBkA8Ndff+Hx48dab49t2LAhatasibi4OFOESEREREREMLMrFk9Tq9WYP38+AgMDIZfLAQBKpRI2NjZwdXXVWrZatWp48OCBQf0LITicHZEZkclkktZnThOZD+YzUcWib06bbWERFRWFq1evYtOmTeXSv0qlgrW1dbn0TUSGc3JykrQ+c5rIfDCfiSoWfXPaLAuLTz75BEeOHMGGDRvg6empaXd3d8fjx4+RmZmpddUiLS0NHh4eBm3D0dGRBy2iCoQ5TVRxMJ+JLJNZFRZCCMyZMwc//fQT1q9fjzp16mjNb9q0KWxsbHDixAl06dIFAHDjxg2kpKQY/OC2TCaTfKmWiMwHc5pIW2pqKlJTU3Xavby84OXlZYKI9Md8JrJMZlVYREVFYffu3Vi2bBmcnJw0z024uLjA3t4eLi4u6NOnDxYsWIDKlSvD2dkZc+fORUBAAEeEIiIiekpsbCyioqJ02mfNmoXZs2e/+ICIqMIzq8Liu+++AwAMHjxYqz06Ohq9e/cGAPz73/+GlZUVJk6ciIKCAoSGhmLWrFkvPFYiIiJzNnr0aISHhyM0NBQAcPz4cTg4OJj91Qoislwy8ZINu1BYWKh55wXv3ySyfMxpopLl5OTA2dkZAJCdnS35oeryxnwmsmxm/R4LIiIiIiKyDCwsiIiIiIhIMhYWREREREQkGQsLIiIiIiKSjIUFERERERFJxsKCiIiIiIgkY2FBRERERESSsbAgIiIiIiLJWFgQEREREZFkLCyIiIiIiEgyFhZERERERCQZCwsiIiIiIpKMhQUREREREUnGwoKIiIiIiCRjYUFERERERJKxsCAiIiIiIskqmToAIiIiKllSUhKUSmWZ1s3NzdX8HBcXBwcHB0mxuLu7w9vbW1IfRFRxsbAgIiIyU0lJSfD19YVKpZLcV2hoqOQ+HB0dER8fz+KCiIrFwoKIiMhMKZVKqFQqfLRgGbwbyA1ePz8vF1OG9AAAfL5uF+zsy37FIunGFXw2fSyUSiULCyIqFgsLIiIiM+fdQI5Gr/gZvF6eKkfzc0OfprB3dDJmWEREWvjwNhERERERScbCgoiIiIiIJGNhQUREREREkrGwICIiIiIiyVhYEBERERGRZCwsiIiIiIhIMhYWREREREQkGQsLIiIiIiKSjIUFERERERFJxjdvE5FGamoqUlNTddq9vLzg5eVlgoiIiIjIUvCKBRFpxMbGIigoSOdfbGysqUMjIiIiM8crFkSkMXr0aISHhyM0NBQAcPz4cTg4OPBqBRERET0XCwsi0vDy8oKrq6tm2t/fH05OTiaMiIiIiCyFWd0KdebMGYwZMwahoaFQKBQ4dOiQ1vzp06dDoVBo/Rs+fLiJoiUiIiIioiJmdcVCpVJBoVCgT58+GD9+fLHLtGvXDtHR0ZppW1vbFxUeERERERGVwKwKi7CwMISFhZW6jK2tLTw8PF5QREREREREpA+zKiz0cfr0aYSEhMDV1RWtW7fGBx98gKpVq5o6LCIiIrOS/uAeUpNvaaavJ/wFO3sHuHnUgJtHDdMFRkQVlkUVFu3atUN4eDhq166N5ORkfPnllxg5ciS2bNkCa2trg/oSQkAIUU6RElmup/PiReaJTCaTtD5zmioiKb/Te7euxcbln2umpwzpAQAY9P4URIybKimm58XFfCaqWPTNaUmFRUFBwQt9xqFbt26an4se3u7UqZPmKoYhVCqVwcUI0ctApVJp/Sz1DwR9SR19ijlNFVFeXl6Z132j/1C07tBVp13q1Yq8vDyt40RxmM9EFYu+OS2psAgNDUWXLl3Qs2dPBAcHS+mqTOrUqYOqVasiMTHR4MLC0dGRBy2iYjx9ltDR0RGOjo4mjEZ/zGmqiOzt7cu8bnnd8mRvb1/uxwXmM5FlklRYdOnSBQcPHsT3338PLy8v9OjRA2+++SYaNmxorPhKdffuXWRkZJTpYW6ZTPbCzsQSWZKn88KS8sSSYiXSlzn+Tr+IXGM+E1kmSe+xmDNnDo4fP47FixejadOmWLNmDbp3747evXtj7dq1UCqVBvWXk5OD+Ph4xMfHAwBu376N+Ph4pKSkICcnBwsXLkRcXBxu376NEydOYOzYsahbty7atWsn5WMQEREREZFEkh/etrGxQXh4OMLDw5GdnY19+/Zh9+7dWLhwIT777DOEhITgzTffRHh4+HMv6f71118YMmSIZrrofRW9evXC7NmzceXKFezcuRNZWVmoXr062rZti0mTJvFdFkREREREJmbUUaGcnZ3Rr18/+Pj4YOXKlTh48CCOHTuGY8eOwcnJCf3798eECRNKvDezVatWuHz5con9r1692pjhEhERERGRkRitsEhOTsauXbuwa9cu3Lp1C1WqVEFERAR69uwJGxsbbN26FevXr8ft27cRExNjrM0SEREREZEZkFRYPHz4EHv37sWuXbtw/vx52NjY4NVXX8VHH32E9u3bo1Kl/+v+448/hqenJ5YtWyY5aCIqWVJSksHPNz0tNzdX83NcXBwcHBzK3Je7uzu8vb3LvD4RERFZDkmFRbt27fDkyRP4+/tj1qxZeOONN+Dq6lri8o0bN4abm5uUTRJRKZKSkuDr6/vcMeb1FRoaKml9R0dHxMfHs7ggIiJ6CUgqLEaPHo2ePXvq/UdDhw4d0KFDBymbJKJSKJVKqFQqfLRgGbwbyMvUR35eruYNvZ+v2wU7+7JdsUi6cQWfTR8LpVLJwoKIiOglIKmwqFOnDqysSh6x9vbt2/jjjz/w1ltvSdkMERnIu4EcjV7xK9O6eaoczc8NfZrC3lHaG3SJiIjo5SDpPRaRkZE4d+5cifMvXLiAyMhIKZsgIiIiIiILIKmwEEKUOl+lUsHa2lrKJoiIiIiIyAIYfCtUQkICEhISNNN//PEHCgsLdZbLzMzE5s2bUb9+fWkREhERERGR2TO4sDh06BCWLFkCAJDJZNiyZQu2bNlS7LKurq5YuHChtAiJiIiIiMjsGVxY9O/fH6+++iqEEOjXrx8mTpyI9u3bay0jk8ng4OAAb29vrXdZEBERERFRxWTwX/3Vq1dH9erVAQDr1q1Dw4YNUa1aNaMHRkRERERElkPS5YSWLVsaKw4iIiIiIrJgBhUWgwcPhpWVFVavXo1KlSphyJAhz11HJpNh7dq1ZQ6QiIiIiIjMn8FXLNRqtebn5w03q+8yRERERERk2QwqLNavX1/qNBERERERvZwkvSCPiIiIiIgIkPjwdkpKClJSUhAcHKxpS0hIwDfffIOCggJ0794dnTp1khwkERERERGZN0mFxdy5c6FSqfDtt98CAJRKJYYMGYLHjx/DyckJBw4cwFdffYXOnTsbI1YiKmfpD+4hNfmWZvp6wl+ws3eAm0cNuHnUMF1gREREZPYk3Qp14cIFtGnTRjO9c+dO5OXl4X//+x+OHj2KkJAQfPPNN5KDJKIXY+/WtZgypIdmesqQHpjQvxP2buXIbkRERFQ6SVcsHj16pPVyvCNHjqBFixbw9vYGAISHh2PRokXSIiSiF+aN/kPRukNXnXZerSAiIqLnkVRYuLm5ISUlBQCQmZmJuLg4TJkyRTO/sLAQT548kRYhEb0wvOWJiIiIykpSYdGmTRusX78ezs7OOHXqFIQQeO211zTzr127Bi8vL8lBEhERERGReZNUWHz44Ye4efMmFi5cCBsbG0ydOhV16tQBABQUFGDfvn3o0aPHc3ohIiIiIiJLJ6mwcHd3x+bNm5GVlQU7OzvY2tpq5qnVaqxduxaenp6SgyQiIiIiIvMmqbAo4uLiotNmb28PHx8fY3RPRERERERmTnJhUVhYiOPHjyM5ORmPHj2CEEJrvkwmw7hx46RuhoiIiIiIzJikwuLixYuYOHEi7t69q1NQFGFhQURERERU8UkqLKKiopCXl4elS5ciODgYrq6uxoqLiIiIiIgsiKTC4vLly/jXv/6Fjh07GiseIiIiIiKyQFZSVvb09CzxFigiIiIiInp5SCosRo4cia1btyI7O9tY8RARERERkQWSdCtUTk4OnJycEB4ejm7dusHT0xPW1tZay8hkMrz77rtSNkNERERERGZOUmGxcOFCzc8bNmwodhkWFkREREREFZ+kwuLw4cPGigMAcObMGaxevRp//fUXHjx4gKVLl6JTp06a+UIILF68GNu2bUNmZiYCAwMxe/Zs1KtXz6hxEBERERGRYSQVFrVq1TJWHAAAlUoFhUKBPn36YPz48TrzV65cifXr12PBggWoXbs2vvrqKwwfPhx79+6FnZ2dUWMhIiIiIiL9SX7zNgDcu3cPZ86cQVpaGrp06QJPT08UFhYiKysLLi4uOs9dlCQsLAxhYWHFzhNCYN26dXj//fc1VzE+/fRTtGnTBocOHUK3bt2M8VGIiIiIiKgMJBUWQggsWLAAGzduxJMnTyCTySCXy+Hp6QmVSoWOHTti4sSJRnnG4vbt23jw4AHatGmjaXNxcUHz5s1x7tw5gwsLIQSHyqUKxxx/p/XNNZlM9kK2Q2RJzPF3Wp9cYz4TVSz65rSkwmLVqlVYt24dRo4ciZCQELz33nuaeS4uLujcuTMOHjxolMLiwYMHAIBq1apptVerVg1KpdLg/lQqld5XUogsRV5enqlD0JGXlweVSvXc5ZycnCRthzlNFZGl5jTzmahi0TenJRUW27Ztw1tvvYXJkyfj4cOHOvMVCgWOHj0qZRPlxtHRkQctqnDs7e1NHYIOe3t7ODo6lvt2mNNUEb2sOc18JrJMkgqL1NRUBAQElDjfwcHBaC/P8/DwAACkpaWhevXqmva0tDT4+PgY3J9MJpN8qZbI3Jjj7/SLyjXmNFVE5vg7/SJyjflMZJkkvXm7WrVqSE1NLXH+pUuX4OXlJWUTGrVr14aHhwdOnDihacvOzsb58+dLLW6IiIiIiKj8SSoswsPDsXnzZiQnJ2vais4wHD9+HDt27EDXrl317i8nJwfx8fGIj48H8M8D2/Hx8UhJSYFMJsOQIUOwfPlyHD58GJcvX8bUqVNRvXp1rXddEBERERHRiyfpVqiJEyfi1KlT6NmzJ4KDgyGTybBy5Up89dVXiIuLg6+vL8aMGaN3f3/99ReGDBmimY6OjgYA9OrVCwsWLMDIkSORm5uLjz/+GJmZmQgKCsKqVav4DgsiIiIiIhOTVFi4uLhg69at+Oabb3DgwAHY2dnhzJkz8Pb2xrhx4zBixAiDHjxr1aoVLl++XOJ8mUyGSZMmYdKkSVLCJiIySGpqarG3fXp5eRntdk8iIiJLJ/kFefb29hg7dizGjh1rjHiIiMpNUlJSmYanjo2NxYoVK3TaR40ahdGjR5cpFnd3d3h7e5dpXSIiInMkqbAYMmQI3n//fYSEhBQ7/+TJk1i2bBnWrVsnZTNERJIlJSVB4eODvNxco/W5YsWKYgsOfdg7OOByQgKLCyIiqjAkFRanT59Gv379Spyfnp6OM2fOSNkEEZFRKJVK5OXmwr7VaFi51jRoXZGfBXVuOvLPfAMAsGsxDDIrG8jsXCCzczE4FnVmCvJOxUKpVLKwICKiCkPyrVCljTOdmJgo+e2bRETGZOVaE9Zu9QxaJ/+vHSi4tPP/pv9/gWHb5C3YNe1lxOiIiIgsl8GFxY4dO7Bjxw7N9PLly7F161ad5bKysnD58mW0b99eWoRERCZm07ADKtXUfV+OzKHKiw+GiIjITBlcWOTm5uLhw4ea6ZycHFhZ6b4Ow9HREW+//TbGjRsnLUIiIhOzcqgCsIggIiIqlcGFxcCBAzFw4EAAQMeOHTFjxgy89tprRg+MiIiIiIgsh6RnLH7++WdjxUFERERERBZM8sPbAJCdnY2UlBRkZmZCCKEzv0WLFsbYDBERERERmSlJhUV6ejrmzp2LgwcPorCwUGe+EAIymQzx8fFSNkNERERERGZOUmHx8ccf45dffsHgwYMRHBwMV1dXY8VFRERERET/X2pqKlJTU3Xavby84OXlZYKIdEkqLH777TcMHToUU6dONVY8RET0ErGEL0oiInMQGxuLqKgonfZZs2Zh9uzZLz6gYkgqLOzt7VGrVi1jxUJERC8ZS/iiJCIyB6NHj0Z4eDhCQ0MBAMePH4eDg4NZnYSRVFi8+eabOHToEAYNGmSseIiI6CViCV+URETmwMvLS+uxA39/fzg5OZkwIl2SCosuXbrgzJkzGD58OAYMGABPT09YW1vrLNekSRMpmyEiogrKEr4oiYhIP5IKi6IX5QHA77//rjOfo0IREREREb0cJBUW0dHRxoqDiIiIiIgsmKTColevXsaKg4iIiIiowktKSoJSqSzTurm5uZqf4+Li4ODgICkWd3d3eHt7S+rjaUZ58zYREREREZUuKSkJvr6+UKlUkvsqGvRCCkdHR8THxxutuDCosIiMjIRMJsOcOXNgbW2NyMjI564jk8kwf/78MgdIRERERFQRKJVKqFQqfLRgGbwbyA1ePz8vF1OG9AAAfL5uF+zsy37FIunGFXw2fSyUSqVpCotTp05BJpNBrVbD2toap06deu46MpmszMEREREREVU03g3kaPSKn8Hr5alyND839GkKe0fzGkXPoMLi559/LnWa/g/fJktERERELxM+Y1EKKQ/XxMbGYsWKFTrto0aNwujRow3uz9gP1xARGVNFfhiRiIj0Y9TC4tGjR5gwYQKmT5+OV155xZhdv3BJSUlQ+Pgg76kvPGNYsWJFsQXH89g7OOByQgK/LInI7CQlJcHHxxe5uebxMKKDgyMSEoz3MCIREenHqIXF48ePcfr0aTx69MiY3ZqEUqlEXm4u7FuNhpVrTYPXL7j+C57cOKLTXqnBq7Bt2MGgvtSZKcg7FWvUh2ssGW8zIzIvSqUSubkqBI+cAxev+gavX1iQh6MLRgAA2k9fBWtb+zLHkpV6E3+s/A+Pl0REJsBboZ7DyrUmrN3qGbyenUOvYgsImUMVWDlUkR7YSyw2NhZRUVE67bNmzcLs2bNffEBEBABw8aqPqnV9DF7vSf7/XRmu4q1AJTtpt0IREVVE6Q/uITX5lmb6esJfsLN3gJtHDbh51DBdYE9hYVFOrByqACwgysXo0aMRHh6uuWXi+PHjcHBw4NUKIiIiqrD2bl2Ljcs/10wXDTs76P0piBg31VRhaTFqYWFvb49evXqhevXqxuyWSIuXlxdcXV010/7+/nByMq/h1oiIiIiM6Y3+Q9G6Q1eddnO5WgEYubBwdnZGdHS0MbskIiIiInrpmdMtTyWRVFikpKSUOl8mk8HOzg5Vq1bli/KIiIiIiCowSYVFx44d9SoY7OzsEBQUhLFjxyIoKEjKJomIiIiIyAxJKizmzZuH9evXIzU1FT169EDdunUBAImJidi1axdq1aqF3r17IzExET/++COGDh2KVatWoXXr1kYJnoiIiIiIzIOkwuL+/ft4/PgxfvrpJ62HaQFg/PjxGDhwIPLy8jBjxgyMHTsWffr0wdKlS8tcWMTExGDJkiVabfXr18f+/fvL/BnINKS8pRcw7pt6+ZZeItPJzVAi58FtzXRG0mVY29rDvrI7HKq4mzAyIiIylKTCYvPmzXjvvfd0igoAqFKlCvr164d169ZhxIgRqFq1Knr37o3Vq1dL2SQaN26MNWvWaKatra0l9UcvnjHf0gtIf1Mv39JLZDo3f92OhB9XaqaLXpTn8+ZIvNJztKnCIiKiMpBUWGRkZGidOX6WSqVCenq6ZtrDw0PK5gD8U0gYox8yHalv6QWM96ZevqWXyLTqh/WBl3+YTrt9ZV6tICKyNJIKi2bNmmHdunXo2LEjFAqF1ryEhARs2LABfn5+mrbr16+jRg1pw2QlJiYiNDQUdnZ28Pf3x4cffoiaNWtK6pNMo6xv6QX4pl6iisKhCm95IiKqKCQVFjNnzsTQoUPRq1cv+Pv7az28HRcXB2dnZ8yYMQMAkJ+fj9OnT6NLly5l3p6fnx+io6NRv359PHjwAEuXLsWgQYOwa9cuODs7G9SXEAJCiFLnm5vnxWwpzPEzcN+WH333rdQhqZnTpmOOn4H7tvzos2/LO5+JLJU5/l4bM6clFRY+Pj748ccfsXLlShw7dgwXL14EANSsWRMDBw7EiBEj4OnpCeCfIWd37twpZXMIC/u/y+U+Pj5o3rw5OnTogH379qFfv34G9aVSqUp9PiMvL6/McZaXvLw8qFTGeS7BlLhvy48l71upb09nTpsO9235sdR9W975TGSpKnpOS37zdo0aNTBz5kyp3ZSJq6sr6tWrh6SkJIPXdXR0LPWgZW9ftnv2y5O9vT0cHR1NHYZk3Lfl52Xet8xp0+G+LT8v6759Xj4TWaqKntOSC4siOTk5uHv3LgDA09NT8tkKfbeZnJxcpoe5ZTJZqZd1zPFN4c+L2VKY42fgvi0/L2rfMqdNxxw/A/dt+XkR+7ai/P8jepY5/l4bM98kFxYXLlzAZ599hj///BNqtRoAYGVlhaCgIHz00Udo1qyZ5CCLLFy4EB06dEDNmjVx//59xMTEwMrKCt27dzfaNsj8cdx7IiIiIvMjqbA4f/48Bg8eDBsbG/Tt2xcNGzYE8M/oT3v27EFERATWr1+vNTKUFHfv3sXkyZORkZEBNzc3BAUFYevWrXBzczNK/2QZOO49ERERkfmRVFgsWrQINWrUwKZNm3RuR5owYQLeeecdLFq0SOuFdlK3R8Rx74mIiIjMj+QrFuPGjSv2GQd3d3f0798fy5Ytk7IJIh0c956IiIjI/FhJWtnKCoWFhSXOV6vVsLKStAkiIiIiIrIAkv7qDwgIwMaNG3Hnzh2deSkpKdi0aRMCAwOlbIKIiIiIiCyApFuhJk+ejEGDBuH1119HeHg46tWrBwC4efMmDh8+DCsrK3z44YfGiJOIiIiIiMyYpMLilVdewbZt27Bo0SL8/PPPyM3NBQA4ODigXbt2GD9+PKpWrWqUQImIiIiIyHxJfo9Fo0aNsHTpUqjVaqSnpwMA3NzcYGVlheXLl2Px4sWIj4+XHCgREREREZkvo71528rKCu7uHKmHiIiIiOhlxCGbiIiIiIhIMhYWREREREQkGQsLIiIiIiKSzOBnLC5duqT3svfv3ze0eyIiIiIiskAGFxZ9+vSBTCbTa1khhN7LEhERERGR5TK4sIiOji6POIiIiIiIyIIZXFj06tWrPOIgIiIiIiILxoe3iYiIiIhIMhYWREREREQkGQsLIiIiIiKSjIUFERERERFJxsKCiIiIiIgkY2FBRERERESSGTzcLFU8qampSE1N1Wn38vKCl5eXCSIiIiIiIkvDwoIQGxuLqKgonfZZs2Zh9uzZLz4gIpKEJwuIyFR4/Hm5sbCoQJKSkqBUKg1eLyQkBKtXr8bw4cMBAKtXr4adnR3c3d3x559/likWd3d3eHt7l2ldIpKGJwuIyFR4/Hm5sbCoIJKSkqDw8UVerkpyX0UFhhT2Do64nBDP4oJIAnM5WcATBUQvH3M5/gA8BlkSFhYVhFKpRF6uCrV6ToNdNcOS74nqER5nKpG690sAgNcbk2FVyQbWjpVRybGywbHkpyXhzv8WQqlU8kBAVEbmdLKAJwqIXi5JSUnw8fFFrhkcfwDAwcERCTwGWQQWFhWMXTVvOHg1Nmid+0fX4cGxDZrpogLDo10EqrcfYtT4iEg/5nKygCcKiF4+SqUSubkqBI+cAxev+gavX1iQh6MLRgAA2k9fBWtb+zLHkpV6E3+s/A+PQRaChQWhakA3uDQO0Wmv5OxmgmiI6Gk8WUBEpnog2sWrPqrW9TFondwMJXIe3NZpt6/sDocq7sYKjcwUCwuCjUs12LhUM3UYRGQkPFlAVLFY0gPRN3/djoQfV2qmi65c+Lw5Eq/0HG2qsOgFYWFBRFTB8GQBkfkp68PQgGUNyFA/rA+8/MN02u0r82rFy4CFBREREVE5MuZgDIB5D8jgUIW3PL3MWFgQERERlSMpgzEAQPqfe5ARt1envYr/G3AL7GZQXxyQgcoTCwsiIiI98a3CJEVZBmMAgOrtBxdbQFRyduNtj2RWWFgQlSP+EUJUsVjSQ7RUcfC5KbIUFllYbNy4EatXr8aDBw/g4+OD//znP/Dz8zN1WFSBlfWhu9jYWKxYsUKnfdSoURg9umyjY/ANpETSmctbhZnPRFSRWFxhsXfvXkRHRyMqKgrNmzfH2rVrMXz4cOzfvx/VqrGaJ+NLSkqCj68vclXGeegOAFasWFFswaEPB0dHJMTzDaREZWXMnJb6EC3zmYgqEosrLNasWYP+/fujT58+AICoqCgcOXIE27dvx6hRo0wcHVVESqUSuSoV+kV+AQ/vhgate3r3d/hjzxad9uBuA9Cy+zsGx/Ig6Tq2RX/Ih+6IJDCXnGY+E1FFY1GFRUFBAS5duqR1C4mVlRXatGmDc+fOmTAyehl4eDdErcZNDVrntSGTiv1jw8WtOlyrVTdWaERUBsxpIiLjsqjC4uHDhygsLNS55alatWq4ceOGQX3l5OTA2tq6xPm5ubkAgMKHiRCFBYYHa0TqzH8e/s3NzUVOTk6xyxTFm3vvOtRPTBtvflryP7GUEG9RrBnJl1H4OP+FxvasrLu3AOi3b1Ovx+NxgXHiTb+bjPS7yQavp0y+oYmptH17PeEvFOTnSQtSouSb1wCUvm+f5uTkJGl7zOny8bx8LpoHMKcNzenn5XPRPMDycrq881kTU3Iy0tLSJG3LWKpVq4Y6deoUO89c8hmomDl99+5d3L17t0z9K5XKYn+HqlWrBnd3w9/H4enpCU9PzxLnV/SctqjCwpguXLhQ6vx79+7BxtYO+X9884IiKp2NrR3u3r1b4pWZe/fuwdbODql7vnzBkRXP1q7keItiPbdmjgki01VarMA/8drZ2WPH55EvOLLi2dnZl7pv7ezt8d+PP3jxgRXDzr7kWJ8VGhoqaVvM6fKjT44wp8umtHwGLDenyzufgX/2zTsDBxmtOJTKxtYO323aiBo1aujMM6d8BipeTq9evRrffvvtiw2qBO+++26pz15V9Jy2qMKiatWqsLa21qks09LSDK4q/fz8nns25ML5OIs4E1LkfJzlxGtJsQJAXNw5i4k37pzlxGpMzOnyY0mxApaV03rF+hLmtD75HBcXh8cF+bALHgYrV9MO363OTEX+H9/A09MT/v7+xS5jaTliSfHOnDkTI0aMKFPfMTEx2LJF97mpAQMGYMKECQb397wrFkDFzmmLKixsbW3RpEkTnDhxAp06dQIAqNVqnDhxAhEREQb15eTk9NyDlo+PT5ljNQVLiteSYgUsK15LitWYmNOmZUmxApYVryXFaiz65LODgwMAwLpqXVi71XsBUZWs0NoWwD8xlXTLiKX9f7SkeBs2bIiGDQ0biKFIgwYNMHXqVJ328nzflCXtW0NZVGEBAO+99x6mTZuGpk2bws/PD2vXrkVubi569+5t6tCIiIiIyILwhbXGZXGFxRtvvIH09HQsXrwYDx48gK+vL1atWlWmB2yIiIiIiMg4LK6wAICIiAiDb30iIiIiIqLyY2XqAIiIiIiIyPKxsCAiIiIiIslYWBARERERkWQsLIiIiIiISDIWFkREREREJBkLCyIiIiIikswih5slIiIiAgB1ZkoZ1kmFOue+TruVU3VYuRr+srSyxEBUEbGwICIiIovj7u4OewcH5J2KNXUoAAB7Bwe+rJdeeiwsiIiIyOJ4e3vjckIClEqlwesmJCTgxo0bOu0NGjSAj49PmeJxd3eHt7d3mdYlqihYWBAREZFF8vb2LtMf84GBgeUQDRHx4W0iIiIiIpKMhQUREREREUnGwoKIiIiIiCRjYUFERERERJKxsCAiIiIiIslYWBARERERkWQv3XCzQggAQGFhoYkjIaJnWVlZQSaTGbQOc5rIPDGfiSoWfXL6pSss1Go1AODixYsmjoSInuXv7w9ra2uD1mFOE5kn5jNRxaJPTstE0emBl4RarcaTJ0/KdCaFiMpXWfKSOU1knpjPRBWLPnn50hUWRERERERkfHx4m4iIiIiIJGNhQUREREREkrGwICIiIiIiyVhYEBERERGRZCwsiIiIiIhIMhYWREREREQkGQsLIiIiIiKSjIVFOTlz5gzGjBmD0NBQKBQKHDp0yNQhlSg2NhZ9+vRBQEAAQkJCMHbsWNy4ccPUYRVr06ZN6NGjBwIDAxEYGIgBAwbg119/NXVYeluxYgUUCgXmzZtn6lB0xMTEQKFQaP3r2rWrqcMyG8zp8mHJOW3O+Qwwp0vDfC4/zOnyYwk5XcnUAVRUKpUKCoUCffr0wfjx400dTqlOnz6NQYMGoVmzZigsLMSXX36J4cOHY8+ePXB0dDR1eFo8PT0xZcoU1K1bF0II7Ny5E+PGjcOOHTvQuHFjU4dXqgsXLmDz5s1QKBSmDqVEjRs3xpo1azTT1tbWJozGvDCny4el5rQl5DPAnC4J87n8MKfLl7nnNAuLchIWFoawsDBTh6GX1atXa00vWLAAISEhuHTpElq0aGGiqIrXsWNHrel//etf+O677xAXF2fWB6ycnBx89NFHmDt3LpYvX27qcEpkbW0NDw8PU4dhlpjT5cMSc9pS8hlgTpeE+Vx+mNPly9xzmrdCkY6srCwAQOXKlU0cSekKCwuxZ88eqFQqBAQEmDqcUn3yyScICwtDmzZtTB1KqRITExEaGorXXnsNH374IVJSUkwdEhkBc9q4LCWfAeZ0RWQp+Qwwp8uDuec0r1iQFrVajfnz5yMwMBByudzU4RTr8uXLePvtt5Gfnw9HR0csXboUjRo1MnVYJdqzZw/+/vtvfP/996YOpVR+fn6Ijo5G/fr18eDBAyxduhSDBg3Crl274OzsbOrwqIyY08ZlKfkMMKcrIkvIZ4A5XV4sIadZWJCWqKgoXL16FZs2bTJ1KCWqX78+du7ciaysLBw4cADTpk3Dhg0bzPKglZqainnz5uGbb76BnZ2dqcMp1dO3Bfj4+KB58+bo0KED9u3bh379+pkwMpKCOW08lpTPAHO6IrKEfAaY0+XFEnKahQVpfPLJJzhy5Ag2bNgAT09PU4dTIltbW9StWxcA0LRpU1y8eBHr1q3DJ598YuLIdF26dAlpaWno3bu3pq2wsBBnzpzBxo0bcfHiRbN78KqIq6sr6tWrh6SkJFOHQmXEnDYuS85ngDlt6SwlnwHm9ItijjnNwoIghMCcOXPw008/Yf369ahTp46pQzKIWq1GQUGBqcMoVuvWrbFr1y6ttsjISDRo0AAjR4406wNWTk4OkpOTzfohMSoec7p8WHI+A8xpS2Xp+Qwwp8uLOeY0C4tykpOTo1VB3r59G/Hx8ahcuTJq1qxpwsh0RUVFYffu3Vi2bBmcnJzw4MEDAICLiwvs7e1NHJ22L774Au3bt4eXlxdycnKwe/dunD59WmfUDHPh7Oyscx+so6MjqlSpYnb3xy5cuBAdOnRAzZo1cf/+fcTExMDKygrdu3c3dWhmgTldPiwppy0pnwHmdGmYz+WHOV1+LCGnWViUk7/++gtDhgzRTEdHRwMAevXqhQULFpgqrGJ99913AIDBgwdrtUdHR2tdHjQHaWlpmDZtGu7fvw8XFxcoFAqsXr0abdu2NXVoFu/u3buYPHkyMjIy4ObmhqCgIGzduhVubm6mDs0sMKfLB3O6/DCnS8Z8Lj/M6fJjCTktE0IIUwdBRERERESWje+xICIiIiIiyVhYEBERERGRZCwsiIiIiIhIMhYWREREREQkGQsLIiIiIiKSjIUFERERERFJxsKCiIiIiIgkY2FBRERERESSsbAgIiIiIiLJKpk6ALIMDx48wNdff40jR47g3r17qFatGnx9fTF06FCEhISYOjyzMH36dGRmZmLZsmWmDoXouZjTz8ecJkvCnH4+5nT5Y2FBz3X79m288847cHV1xdSpUyGXy/HkyRMcP34cUVFR2L9/v6lDJCIDMKeJKhbmNJkNQfQcI0aMEO3atRM5OTk68x49eiSEEOLOnTtizJgxwt/fXwQEBIiJEyeKBw8eaJZbvHixePPNN8W2bdtEWFiY8Pf3F7NmzRJPnjwRK1asEG3atBGtW7cWy5Yt0+pfLpeLjRs3iuHDh4tmzZqJjh07in379mktk5CQIAYPHiyaNWsmWrZsKWbOnCmys7M186dNmybef/99sWrVKtG2bVvRsmVLMXv2bFFQUKBZJj8/XyxYsECEhoaK5s2bi759+4qTJ09q5m/fvl0EBQWJo0ePiq5duwp/f38xbNgwce/ePc3nk8vlWv+eXp/InDCnmdNUsTCnmdPmgoUFlerhw4dCoVCIr7/+usRlCgsLRc+ePcU777wjLl68KOLi4kSvXr1ERESEZpnFixcLf39/MWHCBHH16lVx+PBh0aRJEzFs2DAxZ84ccf36dfH9998LuVwu4uLiNOvJ5XLRsmVLsXXrVnHjxg2xaNEi4evrK65duyaEECInJ0e0bdtWjB8/Xly+fFn8/vvvomPHjmLatGmaPqZNmyYCAwPFxx9/LK5duyZ+/vln0bx5c7FlyxbNMjNmzBADBgwQZ86cEYmJiWLVqlWiadOm4ubNm0KIfw5YTZo0Ee+++664cOGC+Ouvv8Trr78uJk+eLIQQIjs7W0yaNEkMHz5c3L9/X9y/f1/k5+cb5f8BkTExp28KIZjTVHEwp28KIZjT5oKFBZXq/PnzQi6Xi4MHD5a4zPHjx4Wvr69ISUnRtF29elXI5XJx/vx5IcQ/B6zmzZuLrKwszTLDhg0THTp0EIWFhZq2Ll26iNjYWM20XC4XH3/8sdb2+vXrJ2bNmiWEEGLLli2iRYsWWmdpjhw5Inx8fDRnYqZNmyY6dOggnjx5ollm4sSJ4oMPPhBC/HMWx9fXV9y9e1drO0OHDhVffPGFEOKfA5ZcLheJiYma+Rs2bBBt2rTRTBedcSEyZ8xp5jRVLMxp5rQ54TMWVCohxHOXuX79Ojw9PeHl5aVpa9SoEVxdXXHjxg34+fkBAGrVqgVnZ2fNMu7u7rC2toaVlZVWW1pamlb/AQEBWtP+/v6Ij4/XbFuhUMDR0VEzPzAwEGq1Gjdv3oS7u7smHmtra80yHh4euHLlCgDgypUrKCwsRNeuXbW2U1BQgCpVqmimHRwc4O3trZmuXr26TqxE5o45XUUzzZymioA5XUUzzZw2PRYWVKq6detCJpPhxo0bkvuqVEn7100mkxXbplarJW9Ln20XHYxVKhWsra2xfft2rYMaAK0DYWl9EFkK5jRzmioW5jRz2pzwPRZUqipVqiA0NBQbN26ESqXSmZ+ZmYmGDRvi7t27SE1N1bRfu3ZNM0+quLg4renz589r+m3YsCEuX76sFduff/4JKysr1K9fX6/+fX19UVhYiPT0dNStW1frn4eHh95x2tjYlMvBlsiYmNPMaapYmNPMaXPCwoKea9asWVCr1ejXrx8OHDiAW7du4fr161i3bh0GDBiANm3aQC6XY8qUKbh06RIuXLiAqVOnomXLlmjWrJnk7e/fvx/ff/89bt68icWLF+PChQuIiIgAAPTo0QO2traYPn06rly5gpMnT2LOnDno2bOn5vLq89SvXx89evTA1KlTcfDgQSQnJ+PChQuIjY3FkSNH9I6zVq1auHz5Mm7cuIH09HQ8fvy4LB+XqNwxp/XDnCZLwZzWD3O6/PFWKHquOnXq4IcffsDXX3+NhQsX4v79+3Bzc0OTJk0we/ZsyGQyLFu2DHPmzEFERARkMhnatWuH//znP0bZ/oQJE7B3715ERUXBw8MDX3zxBRo1agTgn/spV69ejXnz5qFv375wcHBA586dMX36dIO2ER0djeXLl2PBggW4f/8+qlSpAn9/f7z66qt699G/f3+cPn0affr0gUqlwrp169CqVSuD4iB6EZjT+mFOk6VgTuuHOV3+ZII3n5EZUygUWLp0KTp16mTqUIjICJjTRBULc5qexluhiIiIiIhIMhYWREREREQkGW+FIiIiIiIiyXjFgoiIiIiIJGNhQUREREREkrGwICIiIiIiyVhYEBERERGRZCwsiIiIiIhIMhYWREREREQkGQsLIiIiIiKSjIUFERERERFJxsKCiIiIiIgk+38xuCRhdWSDFQAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 800x300 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import torch\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "from matplotlib.patches import Patch\n",
    "import seaborn as sns\n",
    "\n",
    "sns.set_style(\"whitegrid\")\n",
    "\n",
    "fig, axes = plt.subplots(1, len(bases) - 1, figsize=(8, 3), sharey=True)\n",
    "\n",
    "max_abs = 0.0\n",
    "\n",
    "for i, seed in enumerate([2, 1, 3]):\n",
    "    base = bases[seed]\n",
    "    ax = axes[i]\n",
    "\n",
    "    _, S_ref, _ = base.sample(500, seed=seed)\n",
    "\n",
    "    G_ref = base.Gamma.normalize() \n",
    "\n",
    "    eta_samples = S_ref.unsqueeze(-1) * G_ref.t().unsqueeze(0).unsqueeze(0)\n",
    "\n",
    "    logint_samples = eta_samples.sum(dim=-1).mean(dim=1).detach()\n",
    "\n",
    "    logint_mean = logint_samples.mean(dim=0).cpu().numpy()\n",
    "    logint_std = logint_samples.std(dim=0).cpu().numpy()\n",
    "\n",
    "    d = logint_mean.shape[0]\n",
    "    x = np.arange(d)\n",
    "\n",
    "    cmap = plt.cm.Blues_r\n",
    "    colors = cmap(np.linspace(0.15, 0.7, d))\n",
    "\n",
    "    ax.bar(\n",
    "        x, logint_mean,\n",
    "        yerr=logint_std, capsize=2,\n",
    "        color=colors,\n",
    "        edgecolor=\"black\", linewidth=1.0\n",
    "    )\n",
    "\n",
    "    ax.axhline(0.0, color=\"black\", linewidth=1.0, alpha=0.2)\n",
    "\n",
    "    ax.set_xlabel(\"Component\")\n",
    "    if i == 0:\n",
    "        ax.set_ylabel(\"Log-intensity\", fontsize=12)\n",
    "\n",
    "    ax.set_xticks(x)\n",
    "    ax.set_xticklabels([f\"{c+1}\" for c in range(d)])\n",
    "\n",
    "    ax.grid(axis=\"x\", alpha=0)\n",
    "    ax.grid(axis=\"y\", alpha=0.2)\n",
    "    if seed == 1:\n",
    "        ax.set_title('High coherence')\n",
    "    elif seed == 2:\n",
    "        ax.set_title('Moderate coherence')\n",
    "    elif seed == 3:\n",
    "        ax.set_title('Low excitation')\n",
    "\n",
    "\n",
    "fig.tight_layout()\n",
    "sns.despine(fig)\n",
    "plt.savefig('./figures/log_intensity_scenarios.pdf')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fbf12802-be39-4ceb-abef-0bd7890150a6",
   "metadata": {},
   "source": [
    "## Samples generation and fit\n",
    "\n",
    "We draw $n=150$ data points per scenario, then fit 10 ARPLN-ICA models with varying seeds on each of them to assess variability in recovery."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "7ee1c33c-3d0b-46d0-930f-6a41fdbd415a",
   "metadata": {},
   "outputs": [],
   "source": [
    "for i, (seed, base) in enumerate(bases.items()):\n",
    "    X, S, U = base.sample(n, seed=seed)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "fa964478-b348-4c26-b88a-b9177488453e",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Selected seeds: [453 793 209 309 740 578 895 545 436 678]\n",
      "Seed 1\n",
      "AR\n",
      "Loading from cache ./cache/GammaExp/AR-1-train_453.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-1-train_793.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-1-train_209.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-1-train_309.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-1-train_740.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-1-train_578.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-1-train_895.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-1-train_545.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-1-train_436.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-1-train_678.pkl\n",
      "Mean field\n",
      "Loading from cache ./cache/GammaExp/MF-1-train_453.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-1-train_793.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-1-train_209.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-1-train_309.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-1-train_740.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-1-train_578.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-1-train_895.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-1-train_545.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-1-train_436.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-1-train_678.pkl\n",
      "Seed 2\n",
      "AR\n",
      "Loading from cache ./cache/GammaExp/AR-2-train_453.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-2-train_793.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-2-train_209.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-2-train_309.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-2-train_740.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-2-train_578.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-2-train_895.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-2-train_545.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-2-train_436.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-2-train_678.pkl\n",
      "Mean field\n",
      "Loading from cache ./cache/GammaExp/MF-2-train_453.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-2-train_793.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-2-train_209.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-2-train_309.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-2-train_740.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-2-train_578.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-2-train_895.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-2-train_545.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-2-train_436.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-2-train_678.pkl\n",
      "Seed 3\n",
      "AR\n",
      "Loading from cache ./cache/GammaExp/AR-3-train_453.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-3-train_793.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-3-train_209.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-3-train_309.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-3-train_740.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-3-train_578.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-3-train_895.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-3-train_545.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-3-train_436.pkl\n",
      "Loading from cache ./cache/GammaExp/AR-3-train_678.pkl\n",
      "Mean field\n",
      "Loading from cache ./cache/GammaExp/MF-3-train_453.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-3-train_793.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-3-train_209.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-3-train_309.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-3-train_740.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-3-train_578.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-3-train_895.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-3-train_545.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-3-train_436.pkl\n",
      "Loading from cache ./cache/GammaExp/MF-3-train_678.pkl\n"
     ]
    }
   ],
   "source": [
    "# --------------------------------\n",
    "# Fit PLNICA models with different random seeds\n",
    "# --------------------------------\n",
    "\n",
    "import numpy as np\n",
    "import pickle\n",
    "import os\n",
    "\n",
    "n_seeds = 10\n",
    "np.random.seed(42)\n",
    "seeds_train = np.arange(0, 999)\n",
    "np.random.shuffle(seeds_train)\n",
    "seeds_train = seeds_train[:n_seeds]\n",
    "print('Selected seeds:', seeds_train)\n",
    "\n",
    "params = dict(\n",
    "    amortized_size=4,\n",
    "    n_layers_amortizer=2,\n",
    "    n_output_layers_amortizer=2,\n",
    "    shared_network_size=[16, 8],\n",
    "    n_output_layers_shared=2\n",
    ")\n",
    "\n",
    "models = {}\n",
    "losses = {}\n",
    "\n",
    "for seed in seeds:\n",
    "    if seed == 0:\n",
    "        continue\n",
    "    base = bases[seed]\n",
    "    X = base.sample(n_samples=base.counts.shape[0], seed=seed)[0]\n",
    "    print('Seed', seed)\n",
    "    models[seed] = {}\n",
    "    losses[seed] = {}\n",
    "    print('AR')\n",
    "    for s in seeds_train:\n",
    "        path = f'./cache/GammaExp/AR-{seed}-train_{s}.pkl'\n",
    "        if os.path.isfile(path):\n",
    "            print('Loading from cache', path)\n",
    "            with open(path, 'rb') as f:\n",
    "                model = pickle.load(f)\n",
    "            model.sample(10)\n",
    "        else:\n",
    "            model = PLNICA(\n",
    "                counts=X,\n",
    "                latent_size=base.d,\n",
    "                n_dynamics=base.C,\n",
    "                network_params=params,\n",
    "                offsets=base.offsets[:X.shape[0]],\n",
    "                seed=int(s),\n",
    "                bias=False,\n",
    "            )\n",
    "        \n",
    "            loss = model.fit(n_epochs=800, batch_size=X.shape[0])\n",
    "            \n",
    "            model.counts = model.counts[[0]]\n",
    "            with open(path, 'wb') as f:\n",
    "                pickle.dump(model, f)\n",
    "            \n",
    "        models[seed]['AR-' + str(s)] = model\n",
    "    \n",
    "    print('Mean field')\n",
    "    for s in seeds_train:\n",
    "        path = f'./cache/GammaExp/MF-{seed}-train_{s}.pkl'\n",
    "        if os.path.isfile(path):\n",
    "            print('Loading from cache', path)\n",
    "            with open(path, 'rb') as f:\n",
    "                model = pickle.load(f)\n",
    "        else:\n",
    "            model = PLNICA(\n",
    "                counts=X,\n",
    "                latent_size=base.d,\n",
    "                n_dynamics=base.C,\n",
    "                network_params=params,\n",
    "                offsets=base.offsets[:X.shape[0]],\n",
    "                mean_field=True,\n",
    "                seed=int(s),\n",
    "                bias=False\n",
    "            )\n",
    "        \n",
    "            loss = model.fit(n_epochs=800, batch_size=X.shape[0])\n",
    "            model.counts = model.counts[[0]]\n",
    "            with open(path, 'wb') as f:\n",
    "                pickle.dump(model, f)\n",
    "            \n",
    "        models[seed]['MF-' + str(s)] = model"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7cb4d188-72a5-4471-9cb9-586b3e3bac53",
   "metadata": {},
   "source": [
    "## Evaluation of the recovery\n",
    "\n",
    "**Evaluation of distribution recovery**: For each trained model, we sample $500$ samples (and similarly for the baseline) to assess the empirical sliced Wasserstein distance between the learned model and the baseline.\n",
    "\n",
    "**Evaluation of mixing recovery**: For each trained model, we align the inferred mixing to the true one and assess cosine similarity. We also report recovery with baseline ICA algorithms (UwedgeICA / Picard)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "0e2f33c9-411c-48a1-a0f4-e46870062a30",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "seeds = [1, 2, 3]\n",
    "# Posterior predictive check samples, for the sliced Wasserstein\n",
    "X_models = {}\n",
    "for seed in seeds:\n",
    "    X_models[seed] = {}\n",
    "    for key, model in models[seed].items():\n",
    "        X_models[seed][key] = model.sample(500)[0]\n",
    "    for s in seeds_train:\n",
    "        X_models[seed][f'Baseline-{s}'] = bases[seed].sample(500, seed=int(s))[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "8c5c96a5-065a-4333-908d-7835a42218b8",
   "metadata": {},
   "outputs": [],
   "source": [
    "import ot\n",
    "import warnings\n",
    "warnings.filterwarnings(\"ignore\")\n",
    "\n",
    "def method_from_name(name: str):\n",
    "    if name == \"Baseline\" or name.startswith(\"Baseline\"):\n",
    "        return \"Baseline\"\n",
    "    if name.startswith(\"AR-\"):\n",
    "        return \"AR\"\n",
    "    if name.startswith(\"MF-\"):\n",
    "        return \"MF\"\n",
    "    return None\n",
    "\n",
    "def sliced_wasserstein(\n",
    "    X_ref: torch.Tensor,\n",
    "    X_s: torch.Tensor,\n",
    "    n_projections: int = 1000,\n",
    "    p: int = 2,\n",
    "    seed: int = 0,\n",
    ") -> float:\n",
    "    \"\"\"\n",
    "    Compute Sliced Wasserstein distance between two datasets of time series.\n",
    "\n",
    "    X_ref, X_s: torch tensors of shape (n, T, K)\n",
    "    Flatten (T, K) -> d = T*K and compare empirical distributions in R^d.\n",
    "    \"\"\"\n",
    "    Xr = torch.flatten(X_ref, start_dim=1).detach().to(\"cpu\").double().numpy()\n",
    "    Xq = torch.flatten(X_s, start_dim=1).detach().to(\"cpu\").double().numpy()\n",
    "\n",
    "    a = np.full(Xr.shape[0], 1.0 / Xr.shape[0], dtype=np.float64)\n",
    "    b = np.full(Xq.shape[0], 1.0 / Xq.shape[0], dtype=np.float64)\n",
    "\n",
    "    return float(\n",
    "        ot.sliced_wasserstein_distance(\n",
    "            Xr, Xq, a=a, b=b, n_projections=n_projections, p=p, seed=seed\n",
    "        )\n",
    "    )\n",
    "\n",
    "def standardize_counts(X):\n",
    "    X_log = torch.log(X + 1e-2)\n",
    "    return (X_log - X_log.mean(dim=-1, keepdim=True))/X_log.std(dim=-1, keepdim=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "5a9da4bd-7cc9-48ae-afb5-d2171fd5c51b",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Computation of the sliced Wasserstein between the trained models and the baseline\n",
    "rows = []\n",
    "for seed in seeds:\n",
    "    X_ref = bases[seed].sample(500, seed=seed)[0]\n",
    "    for name, Xt in X_models[seed].items():\n",
    "        method = method_from_name(name)\n",
    "        if method is None:\n",
    "            continue\n",
    "\n",
    "        wass = sliced_wasserstein(standardize_counts(X_ref), standardize_counts(Xt))\n",
    "    \n",
    "        rows.append({\n",
    "            \"name\": name,\n",
    "            \"method\": method,\n",
    "            f\"Sliced Wasserstein-{seed}\": wass,\n",
    "        })\n",
    "\n",
    "df_wass = pd.DataFrame(rows)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "4c8ba06e-5523-4952-84b2-cc0226c1f4e6",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>Scenario</th>\n",
       "      <th>Seed</th>\n",
       "      <th>MethodLabel</th>\n",
       "      <th>Component</th>\n",
       "      <th>Cosine</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>High coherence</td>\n",
       "      <td>1</td>\n",
       "      <td>Picard</td>\n",
       "      <td>1</td>\n",
       "      <td>0.069403</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>High coherence</td>\n",
       "      <td>1</td>\n",
       "      <td>Picard</td>\n",
       "      <td>2</td>\n",
       "      <td>0.642390</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>High coherence</td>\n",
       "      <td>1</td>\n",
       "      <td>Picard</td>\n",
       "      <td>3</td>\n",
       "      <td>0.160894</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>High coherence</td>\n",
       "      <td>1</td>\n",
       "      <td>Picard</td>\n",
       "      <td>4</td>\n",
       "      <td>0.290026</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>High coherence</td>\n",
       "      <td>1</td>\n",
       "      <td>Picard</td>\n",
       "      <td>5</td>\n",
       "      <td>0.384099</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>295</th>\n",
       "      <td>Low excitation</td>\n",
       "      <td>3</td>\n",
       "      <td>UwedgeICA</td>\n",
       "      <td>1</td>\n",
       "      <td>0.374585</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>296</th>\n",
       "      <td>Low excitation</td>\n",
       "      <td>3</td>\n",
       "      <td>UwedgeICA</td>\n",
       "      <td>2</td>\n",
       "      <td>0.533539</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>297</th>\n",
       "      <td>Low excitation</td>\n",
       "      <td>3</td>\n",
       "      <td>UwedgeICA</td>\n",
       "      <td>3</td>\n",
       "      <td>0.538945</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>298</th>\n",
       "      <td>Low excitation</td>\n",
       "      <td>3</td>\n",
       "      <td>UwedgeICA</td>\n",
       "      <td>4</td>\n",
       "      <td>0.329077</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>299</th>\n",
       "      <td>Low excitation</td>\n",
       "      <td>3</td>\n",
       "      <td>UwedgeICA</td>\n",
       "      <td>5</td>\n",
       "      <td>0.845639</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>300 rows × 5 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "           Scenario  Seed MethodLabel  Component    Cosine\n",
       "0    High coherence     1      Picard          1  0.069403\n",
       "1    High coherence     1      Picard          2  0.642390\n",
       "2    High coherence     1      Picard          3  0.160894\n",
       "3    High coherence     1      Picard          4  0.290026\n",
       "4    High coherence     1      Picard          5  0.384099\n",
       "..              ...   ...         ...        ...       ...\n",
       "295  Low excitation     3   UwedgeICA          1  0.374585\n",
       "296  Low excitation     3   UwedgeICA          2  0.533539\n",
       "297  Low excitation     3   UwedgeICA          3  0.538945\n",
       "298  Low excitation     3   UwedgeICA          4  0.329077\n",
       "299  Low excitation     3   UwedgeICA          5  0.845639\n",
       "\n",
       "[300 rows x 5 columns]"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Computation of baseline ICA algorithm\n",
    "\n",
    "from baselines.ica_baseline import BatchedPicardICA, BatchedUwedgeICA\n",
    "import random\n",
    "\n",
    "\n",
    "scenario_label = {\n",
    "    0: \"Ill-conditioned\",\n",
    "    1: \"High coherence\",\n",
    "    2: \"Moderate coherence\",\n",
    "    3: \"Low excitation\",\n",
    "}\n",
    "\n",
    "rows = []\n",
    "for seed in seeds:\n",
    "    if seed == 0:\n",
    "        continue\n",
    "    Gamma_base = bases[seed].Gamma.normalize().detach()\n",
    "    G_ref = Gamma_base\n",
    "\n",
    "    for s in seeds_train:\n",
    "        \n",
    "        for competitor in ['Picard', 'UwedgeICA']:\n",
    "            X = bases[seed].sample(n_samples=bases[seed].counts.shape[0], seed=int(s))[0]\n",
    "            X_log = torch.log(X + 0.5)\n",
    "\n",
    "            torch.manual_seed(42)\n",
    "            np.random.seed(42)\n",
    "            random.seed(42)\n",
    "            \n",
    "            if competitor == 'Picard':\n",
    "                method = BatchedPicardICA(\n",
    "                    d=bases[seed].d\n",
    "                ).fit(X_log)\n",
    "            elif competitor == 'UwedgeICA':\n",
    "                method = BatchedUwedgeICA(\n",
    "                    d=bases[seed].d,\n",
    "                    window=10,\n",
    "                    lags=np.arange(1, 9),\n",
    "                    step=10,\n",
    "                    instantcov=True,\n",
    "                ).fit(X_log)\n",
    "            G = torch.tensor(method.get_mixing())\n",
    "            G = G / torch.linalg.norm(G, axis=0, keepdims=True)\n",
    "            ref = PLNICA(\n",
    "                        counts=X,\n",
    "                        latent_size=bases[seed].d,\n",
    "                        n_dynamics=bases[seed].C,\n",
    "                        network_params=params,\n",
    "                        offsets=bases[seed].offsets[:X.shape[0]],\n",
    "                        seed=int(s),\n",
    "                        bias=False,\n",
    "                    )\n",
    "            ref.Gamma.W.data = G\n",
    "            G_ref = base.Gamma.normalize().detach()\n",
    "            G_aligned, perm, sign, err = ref.identify(G_ref)\n",
    "            col_cos = (G_aligned * G_ref).sum(dim=0).abs().cpu().numpy()\n",
    "            for j, c in enumerate(col_cos, start=1):\n",
    "                rows.append({\n",
    "                    \"Scenario\": scenario_label[seed],\n",
    "                    \"Seed\": seed,\n",
    "                    \"MethodLabel\": competitor,\n",
    "                    \"Component\": j,\n",
    "                    \"Cosine\": float(c),\n",
    "                })\n",
    "\n",
    "df_competitors = pd.DataFrame(rows)\n",
    "df_competitors"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "50ebbad9-cac8-4fac-b0a4-a54953151c32",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Moderate coherence\n",
      "  Auto-regressive: 0.75 ( 0.16 )\n",
      "  Mean Field: 0.73 ( 0.15 )\n",
      "  UwedgeICA: 0.46 ( 0.22 )\n",
      "  Picard: 0.4 ( 0.17 )\n",
      "High coherence\n",
      "  Auto-regressive: 0.84 ( 0.12 )\n",
      "  Mean Field: 0.87 ( 0.12 )\n",
      "  UwedgeICA: 0.39 ( 0.14 )\n",
      "  Picard: 0.33 ( 0.2 )\n",
      "Low excitation\n",
      "  Auto-regressive: 0.58 ( 0.27 )\n",
      "  Mean Field: 0.56 ( 0.21 )\n",
      "  UwedgeICA: 0.53 ( 0.2 )\n",
      "  Picard: 0.6 ( 0.25 )\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 1220x200 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# ----------------------------\n",
    "# Select scenarios (seeds) and labels\n",
    "# ----------------------------\n",
    "seeds_keep = [2, 1, 3]  # e.g. [0, 2, 3] or [0,1,2,3]\n",
    "\n",
    "scenario_order = [scenario_label[s] for s in seeds_keep]\n",
    "\n",
    "scenario_order_nl = [s.replace(\" \", \"\\n\") for s in scenario_order]\n",
    "scenario_label_nl = {k: v.replace(\" \", \"\\n\") for k, v in scenario_label.items()}\n",
    "method_label = {\"AR\": \"Auto-regressive\", \"MF\": \"Mean field\"}\n",
    "\n",
    "# ----------------------------\n",
    "# Panel (a) data: Sliced Wasserstein\n",
    "# ----------------------------\n",
    "metrics_keep = [f\"Sliced Wasserstein-{seed}\" for seed in seeds_keep]\n",
    "\n",
    "df_ppc = df_wass.melt(\n",
    "    id_vars=[\"name\", \"method\"],\n",
    "    value_vars=metrics_keep,\n",
    "    var_name=\"metric\",\n",
    "    value_name=\"value\",\n",
    ")\n",
    "df_ppc[\"seed\"] = df_ppc[\"metric\"].str.split(\"-\").str[-1].astype(int)\n",
    "df_ppc = df_ppc[df_ppc[\"seed\"].isin(seeds_keep)].copy()\n",
    "df_ppc[\"Scenario\"] = df_ppc[\"seed\"].map(scenario_label)\n",
    "\n",
    "df_ppc = df_ppc[df_ppc[\"method\"].isin([\"AR\", \"MF\"])].copy()\n",
    "df_ppc[\"MethodLabel\"] = df_ppc[\"method\"].map(method_label)\n",
    "df_ppc[\"Scenario_nl\"] = df_ppc[\"Scenario\"].str.replace(\" \", \"\\n\")\n",
    "\n",
    "# ----------------------------\n",
    "# Panel (b) data: Cosines per component\n",
    "# ----------------------------\n",
    "rows = []\n",
    "for seed in seeds_keep:\n",
    "    G_ref = bases[seed].Gamma.normalize().detach()\n",
    "\n",
    "    for name, model in models[seed].items():\n",
    "        if name.startswith(\"AR-\"):\n",
    "            method = \"AR\"\n",
    "        elif name.startswith(\"MF-\"):\n",
    "            method = \"MF\"\n",
    "        else:\n",
    "            continue\n",
    "\n",
    "        Gamma_aligned, perm, sign, err = model.identify(G_ref)\n",
    "        col_cos = (G_ref * Gamma_aligned).sum(dim=0).abs().cpu().numpy()\n",
    "\n",
    "        for j, c in enumerate(col_cos, start=1):\n",
    "            rows.append({\n",
    "                \"Scenario\": scenario_label[seed],\n",
    "                \"Seed\": seed,\n",
    "                \"MethodLabel\": method_label[method],\n",
    "                \"Component\": j,\n",
    "                \"Cosine\": float(c),\n",
    "            })\n",
    "\n",
    "df_cos = pd.DataFrame(rows)\n",
    "df_cos = pd.concat((df_cos, df_competitors))\n",
    "\n",
    "#------------------------------\n",
    "\n",
    "sns.set_style(\"whitegrid\")\n",
    "\n",
    "n_sc = len(seeds_keep)\n",
    "fig = plt.figure(figsize=(12.2, 2.))\n",
    "\n",
    "gs = fig.add_gridspec(\n",
    "    nrows=1, ncols=2,\n",
    "    width_ratios=[1., 2.75],\n",
    "    wspace=0.2\n",
    ")\n",
    "\n",
    "ax_ppc = fig.add_subplot(gs[0, 0])\n",
    "\n",
    "gs_cos = gs[0, 1].subgridspec(\n",
    "    nrows=1, ncols=n_sc,\n",
    "    wspace=0.10\n",
    ")\n",
    "\n",
    "axes_cos = []\n",
    "axes_cos.append(fig.add_subplot(gs_cos[0, 0]))\n",
    "for i in range(1, n_sc):\n",
    "    axes_cos.append(fig.add_subplot(gs_cos[0, i], sharey=axes_cos[0]))\n",
    "\n",
    "cmap = plt.cm.viridis_r\n",
    "import cmasher\n",
    "method_palette = {\n",
    "    \"Auto-regressive\": cmap(0.25),\n",
    "    \"Mean field\":      cmap(0.70),\n",
    "    \"UwedgeICA\":       cmasher.pride(0.65),  \n",
    "    \"Picard\":          cmasher.pride(0.8), \n",
    "}\n",
    "\n",
    "# ----------------------------\n",
    "# (a) Sliced Wasserstein barplot\n",
    "# ----------------------------\n",
    "hue_order_ppc = [\"Auto-regressive\", \"Mean field\"]\n",
    "\n",
    "ax_ppc.grid(False)\n",
    "ax_ppc.yaxis.grid(True, alpha=0.3)\n",
    "\n",
    "sns.barplot(\n",
    "    data=df_ppc,\n",
    "    x=\"Scenario_nl\", y=\"value\",\n",
    "    order=scenario_order_nl,\n",
    "    hue=\"MethodLabel\",\n",
    "    hue_order=hue_order_ppc,\n",
    "    palette=method_palette,\n",
    "    errorbar=(\"ci\", 95),\n",
    "    width=0.7,\n",
    "    capsize=0.06, err_kws={'linewidth': 1.5, 'color': 'k'},\n",
    "    edgecolor=\"black\", linewidth=1.2,\n",
    "    ax=ax_ppc\n",
    ")\n",
    "\n",
    "ax_ppc.set_xlabel(\"\")\n",
    "ax_ppc.set_ylabel(\"Sliced Wasserstein\", fontsize=10)\n",
    "ax_ppc.set_yticks([0, 0.1, 0.2, 0.3])\n",
    "for lab in ax_ppc.get_xticklabels():\n",
    "    lab.set_fontsize(10)\n",
    "\n",
    "handles_ppc, labels_ppc = ax_ppc.get_legend_handles_labels()\n",
    "leg = ax_ppc.get_legend()\n",
    "if leg is not None:\n",
    "    leg.remove()\n",
    "\n",
    "# ----------------------------\n",
    "# (b) Cosine similarity boxplots per scenario\n",
    "# ----------------------------\n",
    "hue_order_cos = [\"Auto-regressive\", \"Mean field\", \"UwedgeICA\", \"Picard\"]\n",
    "\n",
    "for idx, (ax, seed) in enumerate(zip(axes_cos, seeds_keep)):\n",
    "    scen = scenario_label[seed]\n",
    "    scen_nl = scenario_label[seed]\n",
    "    sub = df_cos[df_cos[\"Scenario\"] == scen].copy()\n",
    "\n",
    "    sns.boxplot(\n",
    "        data=sub,\n",
    "        x=\"Component\",\n",
    "        y=\"Cosine\",\n",
    "        hue=\"MethodLabel\",\n",
    "        hue_order=hue_order_cos,\n",
    "        palette=method_palette,\n",
    "        width=0.72,\n",
    "        flierprops=dict(marker='x', markersize=3, alpha=0.3),\n",
    "        boxprops=dict(edgecolor=\"black\", linewidth=1.1),\n",
    "        whiskerprops=dict(color=\"black\", linewidth=1.1),\n",
    "        capprops=dict(color=\"black\", linewidth=1.1),\n",
    "        medianprops=dict(color=\"black\", linewidth=1.1),\n",
    "        ax=ax\n",
    "    )\n",
    "\n",
    "    sep_color = \"0.5\"  \n",
    "    n_comp = sub[\"Component\"].nunique()\n",
    "    for x in range(n_comp - 1):\n",
    "        ax.axvline(x + 0.5, linestyle=\"-\", color=sep_color, alpha=0.28,\n",
    "                   linewidth=1.0, zorder=0)\n",
    "\n",
    "    ax.grid(False)\n",
    "    ax.yaxis.grid(True, alpha=0.15)\n",
    "    ax.xaxis.grid(False)\n",
    "\n",
    "    print(scen)\n",
    "    print('  Auto-regressive:', sub[sub['MethodLabel'] == 'Auto-regressive']['Cosine'].mean().round(2),\n",
    "          '(', sub[sub['MethodLabel'] == 'Auto-regressive']['Cosine'].std().round(2), ')')\n",
    "    print('  Mean Field:', sub[sub['MethodLabel'] == 'Mean field']['Cosine'].mean().round(2),\n",
    "          '(', sub[sub['MethodLabel'] == 'Mean field']['Cosine'].std().round(2), ')')\n",
    "    print('  UwedgeICA:', sub[sub['MethodLabel'] == 'UwedgeICA']['Cosine'].mean().round(2),\n",
    "          '(', sub[sub['MethodLabel'] == 'UwedgeICA']['Cosine'].std().round(2), ')')\n",
    "    print('  Picard:', sub[sub['MethodLabel'] == 'Picard']['Cosine'].mean().round(2),\n",
    "          '(', sub[sub['MethodLabel'] == 'Picard']['Cosine'].std().round(2), ')')\n",
    "\n",
    "    ax.set_title(scen_nl, fontsize=10, pad=6)\n",
    "\n",
    "    ax.set_xlabel(\"Component\", fontsize=8)\n",
    "    ax.set_ylabel(\"Cosine similarity\" if idx == 0 else \"\", fontsize=10)\n",
    "\n",
    "    if idx != 0:\n",
    "        ax.tick_params(axis=\"y\", labelleft=False)\n",
    "    if idx == 0:\n",
    "        ax.set_yticks([0, 0.2, 0.4, 0.6, 0.8, 1.0])\n",
    "    ax.set_ylim(ymin=0, ymax=1.)\n",
    "\n",
    "    if idx == 0:\n",
    "        handles_cos, labels_cos = ax.get_legend_handles_labels()\n",
    "\n",
    "    leg = ax.get_legend()\n",
    "    if leg is not None:\n",
    "        leg.remove()\n",
    "\n",
    "# ----------------------------\n",
    "# Separate legends below each panel\n",
    "# ----------------------------\n",
    "y_legend = -0.1\n",
    "leg_ppc = fig.legend(\n",
    "    handles_ppc, labels_ppc,\n",
    "    loc=\"lower center\",\n",
    "    ncol=2,\n",
    "    bbox_to_anchor=(0.215, y_legend), \n",
    "    frameon=True,\n",
    "    fontsize=10,\n",
    "    handlelength=1.6,\n",
    "    columnspacing=1.6,\n",
    ")\n",
    "\n",
    "leg_cos = fig.legend(\n",
    "    handles_cos, labels_cos,\n",
    "    loc=\"lower center\",\n",
    "    ncol=4,\n",
    "    bbox_to_anchor=(0.633, y_legend), \n",
    "    frameon=True,\n",
    "    fontsize=10,\n",
    "    handlelength=1.6,\n",
    "    columnspacing=1.6,\n",
    ")\n",
    "fig.subplots_adjust(bottom=0.28)\n",
    "\n",
    "sns.despine(fig)\n",
    "plt.savefig(\"./figures/identifiability/identifiability_by_scenario.pdf\", bbox_inches=\"tight\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "eaac925f-0295-4f70-a850-c7d90fcccfde",
   "metadata": {},
   "source": [
    "# Varying the effective training size\n",
    "\n",
    "## Effect of $n$ on recovery\n",
    "\n",
    "In this additional study, we vary the number of training samples in the *moderate coherence* scenario to assess the impact of $n$ on the recovery and wall-clock time."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "a26b6ffc-8473-49c0-89cc-eec69b0859d1",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Selection of the scenario, seed = 2 is the moderate coherence\n",
    "\n",
    "seed = 2\n",
    "\n",
    "n = 200\n",
    "T = 20\n",
    "K = 12\n",
    "d = 5\n",
    "C = 1\n",
    "\n",
    "pcounts = torch.zeros((n, T, K))\n",
    "\n",
    "base = bases[seed]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "b2c832d0-3347-49be-804e-4848099f59a2",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Sample\n",
    "X, S, U = base.sample(base.offsets.shape[0], seed=seed)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "7974bbc8-e70a-46b8-ab0f-225228166d43",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Selected seeds: [453 793 209 309 740 578 895 545 436 678]\n",
      "Size 5\n",
      "AR\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n5-train_453.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n5-train_793.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n5-train_209.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n5-train_309.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n5-train_740.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n5-train_578.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n5-train_895.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n5-train_545.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n5-train_436.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n5-train_678.pkl\n",
      "Mean field\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n5-train_453.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n5-train_793.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n5-train_209.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n5-train_309.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n5-train_740.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n5-train_578.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n5-train_895.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n5-train_545.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n5-train_436.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n5-train_678.pkl\n",
      "Size 30\n",
      "AR\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n30-train_453.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n30-train_793.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n30-train_209.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n30-train_309.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n30-train_740.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n30-train_578.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n30-train_895.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n30-train_545.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n30-train_436.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n30-train_678.pkl\n",
      "Mean field\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n30-train_453.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n30-train_793.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n30-train_209.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n30-train_309.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n30-train_740.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n30-train_578.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n30-train_895.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n30-train_545.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n30-train_436.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n30-train_678.pkl\n",
      "Size 50\n",
      "AR\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n50-train_453.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n50-train_793.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n50-train_209.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n50-train_309.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n50-train_740.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n50-train_578.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n50-train_895.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n50-train_545.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n50-train_436.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n50-train_678.pkl\n",
      "Mean field\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n50-train_453.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n50-train_793.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n50-train_209.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n50-train_309.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n50-train_740.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n50-train_578.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n50-train_895.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n50-train_545.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n50-train_436.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n50-train_678.pkl\n",
      "Size 80\n",
      "AR\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n80-train_453.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n80-train_793.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n80-train_209.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n80-train_309.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n80-train_740.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n80-train_578.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n80-train_895.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n80-train_545.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n80-train_436.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n80-train_678.pkl\n",
      "Mean field\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n80-train_453.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n80-train_793.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n80-train_209.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n80-train_309.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n80-train_740.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n80-train_578.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n80-train_895.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n80-train_545.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n80-train_436.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n80-train_678.pkl\n",
      "Size 100\n",
      "AR\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n100-train_453.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n100-train_793.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n100-train_209.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n100-train_309.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n100-train_740.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n100-train_578.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n100-train_895.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n100-train_545.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n100-train_436.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n100-train_678.pkl\n",
      "Mean field\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n100-train_453.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n100-train_793.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n100-train_209.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n100-train_309.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n100-train_740.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n100-train_578.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n100-train_895.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n100-train_545.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n100-train_436.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n100-train_678.pkl\n",
      "Size 150\n",
      "AR\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n150-train_453.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n150-train_793.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n150-train_209.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n150-train_309.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n150-train_740.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n150-train_578.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n150-train_895.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n150-train_545.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n150-train_436.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/AR-n150-train_678.pkl\n",
      "Mean field\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n150-train_453.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n150-train_793.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n150-train_209.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n150-train_309.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n150-train_740.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n150-train_578.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n150-train_895.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n150-train_545.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n150-train_436.pkl\n",
      "Loading from cache ./cache/n_effect_scenario2/MF-n150-train_678.pkl\n"
     ]
    }
   ],
   "source": [
    "# --------------------------------\n",
    "# Fit PLNICA models with different random seeds\n",
    "# --------------------------------\n",
    "\n",
    "import numpy as np\n",
    "import pickle\n",
    "import os\n",
    "\n",
    "n_seeds = 10\n",
    "np.random.seed(42)\n",
    "seeds = np.arange(0, 999)\n",
    "np.random.shuffle(seeds)\n",
    "seeds = seeds[:n_seeds]\n",
    "print('Selected seeds:', seeds)\n",
    "\n",
    "n_list = [5, 30, 50, 80, 100, 150]\n",
    "\n",
    "params = dict(\n",
    "    amortized_size=4,\n",
    "    n_layers_amortizer=2,\n",
    "    n_output_layers_amortizer=2,\n",
    "    shared_network_size=[16, 8],\n",
    "    n_output_layers_shared=2\n",
    ")\n",
    "\n",
    "models = {}\n",
    "losses = {}\n",
    "\n",
    "for n_train in n_list:\n",
    "    print('Size', n_train)\n",
    "    X_train = X[:n_train]\n",
    "    models[n_train] = {}\n",
    "    losses[n_train] = {}\n",
    "    print('AR')\n",
    "    for s in seeds:\n",
    "        path = f'./cache/n_effect_scenario{seed}/AR-n{n_train}-train_{s}.pkl'\n",
    "        if os.path.isfile(path):\n",
    "            print('Loading from cache', path)\n",
    "            with open(path, 'rb') as f:\n",
    "                model = pickle.load(f)\n",
    "            model.sample(10)\n",
    "        else:\n",
    "            model = PLNICA(\n",
    "                counts=X_train,\n",
    "                latent_size=base.d,\n",
    "                n_dynamics=base.C,\n",
    "                network_params=params,\n",
    "                offsets=base.offsets[:X_train.shape[0]],\n",
    "                seed=int(s),\n",
    "                bias=False,\n",
    "            )\n",
    "        \n",
    "            loss = model.fit(n_epochs=800, batch_size=X.shape[0])\n",
    "            \n",
    "            model.counts = model.counts[[0]]\n",
    "            with open(path, 'wb') as f:\n",
    "                pickle.dump(model, f)\n",
    "            \n",
    "        models[n_train]['AR-' + str(s)] = model\n",
    "    \n",
    "    print('Mean field')\n",
    "    for s in seeds:\n",
    "        path = f'./cache/n_effect_scenario{seed}/MF-n{n_train}-train_{s}.pkl'\n",
    "        if os.path.isfile(path):\n",
    "            print('Loading from cache', path)\n",
    "            with open(path, 'rb') as f:\n",
    "                model = pickle.load(f)\n",
    "        else:\n",
    "            model = PLNICA(\n",
    "                counts=X_train,\n",
    "                latent_size=base.d,\n",
    "                n_dynamics=base.C,\n",
    "                network_params=params,\n",
    "                offsets=base.offsets[:X_train.shape[0]],\n",
    "                mean_field=True,\n",
    "                seed=int(s),\n",
    "                bias=False\n",
    "            )\n",
    "        \n",
    "            loss = model.fit(n_epochs=800, batch_size=X.shape[0])\n",
    "            model.counts = model.counts[[0]]\n",
    "            with open(path, 'wb') as f:\n",
    "                pickle.dump(model, f)\n",
    "            \n",
    "        models[n_train]['MF-' + str(s)] = model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "8ee29a7e-b2e8-4363-8e54-a4c84bc9f6fd",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# Trained models samples\n",
    "X_models = {}\n",
    "for n_train in n_list:\n",
    "    X_models[n_train] = {}\n",
    "    for key, model in models[n_train].items():\n",
    "        X_models[n_train][key] = model.sample(500)[0]\n",
    "    for s in seeds:\n",
    "        X_models[n_train][f'Baseline-{s}'] = base.sample(500, seed=int(s))[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "28fe1cd0-c102-4854-94ed-8700050014cf",
   "metadata": {},
   "outputs": [],
   "source": [
    "# -----------------------\n",
    "# Sliced wasserstein reference stats\n",
    "# -----------------------\n",
    "X_ref = base.sample(1000, seed=seed)[0]\n",
    "\n",
    "eps = 1e-8\n",
    "rows = []\n",
    "\n",
    "for n_train in n_list:\n",
    "    for name, Xt in X_models[n_train].items():\n",
    "        method = method_from_name(name)\n",
    "        if method is None:\n",
    "            continue\n",
    "\n",
    "        wass = sliced_wasserstein(standardize_counts(X_ref), standardize_counts(Xt))\n",
    "    \n",
    "        rows.append({\n",
    "            \"name\": name,\n",
    "            \"method\": method,\n",
    "            f\"Sliced Wasserstein-{n_train}\": wass,\n",
    "        })\n",
    "\n",
    "df_wass = pd.DataFrame(rows)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "6fe710e5-abe8-4fca-b3fd-b225e927ece8",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Cosine similarity summary:\n",
      "-----------------------\n",
      "Mean cosine AR: 0.7614 (0.1527)\n",
      "AR-5\n",
      "AR-30\n",
      "AR-50\n",
      "AR-80\n",
      "AR-100\n",
      "AR-150\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 500x150 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.patches as mpatches\n",
    "from matplotlib.offsetbox import AnchoredOffsetbox, TextArea, HPacker, DrawingArea\n",
    "\n",
    "fig, ax = plt.subplots(figsize=(5., 1.5))\n",
    "\n",
    "cmap = plt.cm.plasma\n",
    "hue_order = ['n = ' + x.split('-')[-1] + '' for x in list(df_wass.columns[2:])]\n",
    "metric_palette = {\n",
    "    lab: cmap(x) for lab, x in zip(hue_order, np.linspace(0.15, 0.85, len(hue_order)))\n",
    "}\n",
    "\n",
    "Gamma_base = base.Gamma.normalize().detach()\n",
    "G_ref = Gamma_base\n",
    "\n",
    "rows = []\n",
    "gram_error = {}\n",
    "\n",
    "for n_train in n_list:\n",
    "    for name, model in models[n_train].items():\n",
    "        if name.startswith(\"AR-\"):\n",
    "            group = f\"AR-{n_train}\"\n",
    "        elif name.startswith(\"MF-\"):\n",
    "            group = f\"MF-{n_train}\"\n",
    "            continue\n",
    "        else:\n",
    "            continue\n",
    "        if group not in gram_error:\n",
    "            gram_error[group] = []\n",
    "        Gamma_aligned, perm, sign, err = model.identify(G_ref)\n",
    "    \n",
    "        # per-column |cos|\n",
    "        col_cos = (G_ref * Gamma_aligned).sum(dim=0).abs().cpu().numpy()  # (d,)\n",
    "        d = len(col_cos)\n",
    "        for j in range(d):\n",
    "            rows.append({\"Model\": group, \"Component\": j + 1, \"Cosine\": col_cos[j]})\n",
    "    \n",
    "        # Gram error\n",
    "        Delta = (G_ref.T @ G_ref) - (Gamma_aligned.T @ Gamma_aligned)\n",
    "        rel_err = (torch.linalg.norm(Delta) / torch.linalg.norm(G_ref.T @ G_ref)).item()\n",
    "        gram_error[group].append(rel_err)\n",
    "\n",
    "df_long = pd.DataFrame(rows)\n",
    "\n",
    "df_long[\"n_label\"] = df_long[\"Model\"].str.replace(\"AR-\", \"n = \", regex=False)\n",
    "\n",
    "for n_train in n_list:\n",
    "    mu_ge_ar = float(np.mean(df_long[df_long['Model'] == f\"AR-{n_train}\"]['Cosine']))\n",
    "    sd_ge_ar = float(np.std(df_long[df_long['Model'] == f\"AR-{n_train}\"]['Cosine']))\n",
    "    mu_ge_mf = float(np.mean(df_long[df_long['Model'] == f\"MF-{n_train}\"]['Cosine']))\n",
    "    sd_ge_mf = float(np.std(df_long[df_long['Model'] == f\"MF-{n_train}\"]['Cosine']))\n",
    "print('Cosine similarity summary:')\n",
    "print(\"-----------------------\")\n",
    "print(f\"Mean cosine AR: {mu_ge_ar:.4f} ({sd_ge_ar:.4f})\")\n",
    "for model in df_long.Model.unique():\n",
    "    print(model)\n",
    "    for c in df_long.Component.unique():\n",
    "        values = df_long[(df_long['Component'] == c) & (df_long['Model'] == model)]['Cosine']\n",
    "        ci_lo = np.quantile(values, 0.05)\n",
    "        ci_hi = np.quantile(values, 0.95)\n",
    "        m = values.mean()\n",
    "\n",
    "sep_color = \"0.5\"  # grey (0=black, 1=white)\n",
    "for x in range(d - 1):\n",
    "    ax.axvline(x + 0.5, linestyle=\"-\", color=sep_color, alpha=0.28,\n",
    "                   linewidth=1.0, zorder=0)\n",
    "\n",
    "ax.grid(False)\n",
    "ax.yaxis.grid(True, alpha=0.15)\n",
    "ax.xaxis.grid(False)\n",
    "\n",
    "ax.xaxis.grid(False)\n",
    "sns.boxplot(\n",
    "    data=df_long,\n",
    "    x=\"Component\",\n",
    "    y=\"Cosine\",\n",
    "    hue=\"n_label\",\n",
    "    palette=metric_palette,\n",
    "    ax=ax,\n",
    "    flierprops=dict(marker='x', markersize=3, alpha=0.3),\n",
    "    width=0.7,\n",
    "    boxprops=dict(edgecolor=\"black\", linewidth=1.2),\n",
    "    whiskerprops=dict(color=\"black\", linewidth=1.2),\n",
    "    capprops=dict(color=\"black\", linewidth=1.2),\n",
    "    medianprops=dict(color=\"black\", linewidth=1.2),\n",
    ")\n",
    "\n",
    "ax.set_ylabel(\"Cosine similarity\", fontsize=10)\n",
    "ax.set_xlabel(\"Component\", fontsize=8)\n",
    "\n",
    "if ax.get_legend() is not None:\n",
    "    ax.get_legend().remove()\n",
    "\n",
    "legend_handles = [\n",
    "    mpatches.Patch(facecolor=metric_palette[lab], edgecolor=\"black\", label=lab)\n",
    "    for lab in hue_order\n",
    "]\n",
    "fig.legend(\n",
    "    handles=legend_handles,\n",
    "    labels=hue_order,\n",
    "    title=\"\",\n",
    "    frameon=True,\n",
    "    ncol=3,\n",
    "    loc=\"upper center\",\n",
    "    fontsize=10,\n",
    "    bbox_to_anchor=(0.5, -0.12),\n",
    ")\n",
    "\n",
    "sns.despine(fig)\n",
    "plt.savefig(f'./figures/identifiability/grouped_identifiability_benchmark_varying_n_simple_case_{seed}.pdf', bbox_inches='tight')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7a8d6632-bb26-43de-a136-ba1999f9235c",
   "metadata": {},
   "source": [
    "## Timings with $n$ and $T$\n",
    "\n",
    "We vary $n$ and $T$ in the *moderate scenario* to assess wall-clock time effects. For a fixed $n$ and $T$, we fit an ARPLN-ICA models as below and report the total time divided by the number of epochs required to reach the said time in the dictionary `timings` below."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "fc8b8952-320e-4574-9138-5475245cecc8",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 77%|███████▋  | 614/800 [00:45<00:13, 13.48it/s, Loss=0.884, Status=Converged]                                        \n"
     ]
    }
   ],
   "source": [
    "n_samples = 5\n",
    "T = 10\n",
    "\n",
    "base.T = T\n",
    "model = PLNICA(\n",
    "    counts=base.sample(n_samples)[0],\n",
    "    latent_size=base.d,\n",
    "    n_dynamics=base.C,\n",
    "    network_params=params,\n",
    "    offsets=base.offsets.repeat(20, 2)[:n_samples, :T],\n",
    "    seed=42,\n",
    "    bias=False,\n",
    "    device='cuda'\n",
    ")\n",
    "        \n",
    "loss = model.fit(n_epochs=800, batch_size=X.shape[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "ad26714e-a7dc-45a0-83d3-d933eae42b14",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 480x270 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.ticker as mticker\n",
    "import matplotlib as mpl\n",
    "\n",
    "# Timings reported in seconds / number of epochs over varying parameters in the above cell.\n",
    "timings = {\n",
    "    10:{\n",
    "        5: 36 / 496,\n",
    "        50: 32 / 454,\n",
    "        100: 33 / 457,\n",
    "        300: 37 / 496,\n",
    "        500: 42 / 502,\n",
    "        1000: 57 / 511,\n",
    "        1500: 68 / 504,\n",
    "        2000: 84 / 502,\n",
    "        2500: 76 / 500,\n",
    "        3000: 87 / 504,\n",
    "    },\n",
    "    20:{\n",
    "        5: 121 / 752,\n",
    "        50: 121 / 762,\n",
    "        100: 122 / 763,\n",
    "        300: 121 / 749,\n",
    "        500: 119 / 749,\n",
    "        1000: 210 / 758,\n",
    "        1500: 265 / 762,\n",
    "        2000: 319 / 762,\n",
    "        2500: 311 / 764,\n",
    "        3000: 352 / 765\n",
    "    },\n",
    "    40:{\n",
    "        5:  282 / 790,\n",
    "        50:  284 / 772,\n",
    "        100:  316 / 786,\n",
    "        300:  374 / 782,\n",
    "        500:  463 / 788,\n",
    "        1000:  643 / 789,\n",
    "        1500:  815 / 786,\n",
    "        2000:  986 / 784,\n",
    "        2500:  1027 / 787,\n",
    "        3000:  1167 / 782,\n",
    "    },\n",
    "}\n",
    "\n",
    "fig, ax = plt.subplots(figsize=(4.8, 2.7))\n",
    "for i, T_ref in enumerate(timings): \n",
    "    x = np.array(list(timings[T_ref].keys()))\n",
    "    y = np.array(list(timings[T_ref].values()))\n",
    "\n",
    "    df = pd.DataFrame({\"N\": x, \"sec_per_epoch\": y}).sort_values(\"N\")\n",
    "    \n",
    "    sns.set_theme(style=\"whitegrid\", context=\"paper\")\n",
    "    \n",
    "    sns.lineplot(\n",
    "        data=df, x=\"N\", y=\"sec_per_epoch\",\n",
    "        marker=\"o\", markersize=5, linewidth=2,\n",
    "        ax=ax, label='$T =' + str(int(T_ref)) + '$',\n",
    "        color=mpl.colormaps.get_cmap('plasma')((i+1)/(len(timings)+1))\n",
    "    )\n",
    "    \n",
    "    ax.set_xscale(\"log\")\n",
    "    ax.set_xlabel(\"Training set size ($n$)\")\n",
    "    ax.set_ylabel(\"Seconds per epoch\")\n",
    "    \n",
    "    major_ticks = df[\"N\"].to_list()\n",
    "    ax.set_xticks(major_ticks)\n",
    "    ax.xaxis.set_major_formatter(mticker.FuncFormatter(lambda v, pos: f\"{int(v)}\"))\n",
    "    \n",
    "    ax.xaxis.set_minor_locator(mticker.LogLocator(base=10.0, subs=np.arange(2, 10) * 0.1))\n",
    "    ax.xaxis.set_minor_formatter(mticker.NullFormatter())  # keep minor ticks unlabeled (clean look)\n",
    "    ax.tick_params(axis=\"x\", which=\"major\", labelsize=8, rotation=-45)\n",
    "\n",
    "ax.grid(axis='x', alpha=0.25)\n",
    "ax.grid(axis='y', alpha=0.1)\n",
    "fig.tight_layout()\n",
    "sns.despine(fig)\n",
    "plt.savefig('./figures/timings_moderate_coherence_case.pdf', bbox_inches='tight')"
   ]
  }
 ],
 "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.11.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
