{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "22d6b492-5f42-4f19-91ae-18594c47a38d",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import seaborn as sns\n",
    "import torch\n",
    "from matplotlib import pyplot as plt\n",
    "import matplotlib\n",
    "matplotlib.use(\"pgf\")\n",
    "matplotlib.rcParams.update({\n",
    "    \"pgf.texsystem\": \"pdflatex\",\n",
    "    'font.family': 'serif',\n",
    "    'text.usetex': True,\n",
    "    'pgf.rcfonts': False,\n",
    "})\n",
    "\n",
    "import beanmachine.ppl.experimental.gg_algebra as gga\n",
    "from beanmachine.ppl.experimental.gg_algebra import gauss_ens\n",
    "\n",
    "sns.set_style(\"darkgrid\")\n",
    "sns.set_context(\"paper\", font_scale = 1.)\n",
    "\n",
    "TEXTWIDTH = 362.12\n",
    "\n",
    "def set_size(width, fraction=1):\n",
    "    \"\"\" Set aesthetic figure dimensions to avoid scaling in latex.\n",
    "\n",
    "    Parameters\n",
    "    ----------\n",
    "    width: float\n",
    "            Width in pts\n",
    "    fraction: float\n",
    "            Fraction of the width which you wish the figure to occupy\n",
    "\n",
    "    Returns\n",
    "    -------\n",
    "    fig_dim: tuple\n",
    "            Dimensions of figure in inches\n",
    "    \"\"\"\n",
    "    # Width of figure\n",
    "    fig_width_pt = width * fraction\n",
    "\n",
    "    # Convert from pt to inches\n",
    "    inches_per_pt = 1 / 72.27\n",
    "\n",
    "    # Golden ratio to set aesthetic figure height\n",
    "    golden_ratio = (5 ** 0.5 - 1) / 2\n",
    "\n",
    "    # Figure width in inches\n",
    "    fig_width_in = fig_width_pt * inches_per_pt\n",
    "    # Figure height in inches\n",
    "    fig_height_in = fig_width_in * golden_ratio\n",
    "\n",
    "    return fig_width_in, fig_height_in"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1ec11fe1-1dea-4805-ac67-759e74d3edf9",
   "metadata": {},
   "source": [
    "Consider a random variable $y^\\top A z$ with all entries $\\sim N(0,1)$.\n",
    "\n",
    "The true distribution is an additive convolution of normal-power distributions (https://sci-hub.hkvisa.net/10.1007/s11749-006-0030-x)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "9a6462f3-f856-4085-a16b-1bbc979865f8",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "ename": "AttributeError",
     "evalue": "'FacetGrid' object has no attribute 'show'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mAttributeError\u001b[0m                            Traceback (most recent call last)",
      "\u001b[1;32m/workspaces/HTA_neurips/experiments/GGA_RLA_JLT.ipynb Cell 3'\u001b[0m in \u001b[0;36m<cell line: 10>\u001b[0;34m()\u001b[0m\n\u001b[1;32m      <a href='vscode-notebook-cell://dev-container%2B633a5c55736572735c4665796e6d616e204c69616e675c4f6e6544726976655c446f63756d656e74735c32312d32325c4854415f6e6575726970735c6578706572696d656e7473/workspaces/HTA_neurips/experiments/GGA_RLA_JLT.ipynb#ch0000003vscode-remote?line=5'>6</a>\u001b[0m     samples[i] \u001b[39m=\u001b[39m \u001b[39mabs\u001b[39m(\n\u001b[1;32m      <a href='vscode-notebook-cell://dev-container%2B633a5c55736572735c4665796e6d616e204c69616e675c4f6e6544726976655c446f63756d656e74735c32312d32325c4854415f6e6575726970735c6578706572696d656e7473/workspaces/HTA_neurips/experiments/GGA_RLA_JLT.ipynb#ch0000003vscode-remote?line=6'>7</a>\u001b[0m         np\u001b[39m.\u001b[39mrandom\u001b[39m.\u001b[39mrandn(\u001b[39m1\u001b[39m, k) \u001b[39m@\u001b[39m np\u001b[39m.\u001b[39mrandom\u001b[39m.\u001b[39mrandn(k, k) \u001b[39m@\u001b[39m np\u001b[39m.\u001b[39mrandom\u001b[39m.\u001b[39mrandn(k, \u001b[39m1\u001b[39m)\n\u001b[1;32m      <a href='vscode-notebook-cell://dev-container%2B633a5c55736572735c4665796e6d616e204c69616e675c4f6e6544726976655c446f63756d656e74735c32312d32325c4854415f6e6575726970735c6578706572696d656e7473/workspaces/HTA_neurips/experiments/GGA_RLA_JLT.ipynb#ch0000003vscode-remote?line=7'>8</a>\u001b[0m     )\u001b[39m.\u001b[39mitem()\n\u001b[1;32m      <a href='vscode-notebook-cell://dev-container%2B633a5c55736572735c4665796e6d616e204c69616e675c4f6e6544726976655c446f63756d656e74735c32312d32325c4854415f6e6575726970735c6578706572696d656e7473/workspaces/HTA_neurips/experiments/GGA_RLA_JLT.ipynb#ch0000003vscode-remote?line=8'>9</a>\u001b[0m fg \u001b[39m=\u001b[39m sns\u001b[39m.\u001b[39mdisplot(data\u001b[39m=\u001b[39msamples, kind\u001b[39m=\u001b[39m\u001b[39m\"\u001b[39m\u001b[39mhist\u001b[39m\u001b[39m\"\u001b[39m, stat\u001b[39m=\u001b[39m\u001b[39m\"\u001b[39m\u001b[39mdensity\u001b[39m\u001b[39m\"\u001b[39m)\n\u001b[0;32m---> <a href='vscode-notebook-cell://dev-container%2B633a5c55736572735c4665796e6d616e204c69616e675c4f6e6544726976655c446f63756d656e74735c32312d32325c4854415f6e6575726970735c6578706572696d656e7473/workspaces/HTA_neurips/experiments/GGA_RLA_JLT.ipynb#ch0000003vscode-remote?line=9'>10</a>\u001b[0m fg\u001b[39m.\u001b[39;49mshow()\n",
      "\u001b[0;31mAttributeError\u001b[0m: 'FacetGrid' object has no attribute 'show'"
     ]
    }
   ],
   "source": [
    "N = 5000\n",
    "k = 5\n",
    "\n",
    "samples = np.zeros(N)\n",
    "for i in range(N):\n",
    "    samples[i] = abs(\n",
    "        np.random.randn(1, k) @ np.random.randn(k, k) @ np.random.randn(k, 1)\n",
    "    ).item()\n",
    "fg = sns.displot(data=samples, kind=\"hist\", stat=\"density\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8f3eb415-992e-427d-9158-39dea1b8875b",
   "metadata": {},
   "source": [
    "Using GGA, we can compute a generalized Gamma approximation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "1bd67822-4b93-4279-b21e-8266e778e926",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "c x^0.6667 exp(-1.5 * x^0.6667)"
      ]
     },
     "execution_count": 30,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "y = gauss_ens(k, 1)\n",
    "z = gauss_ens(k, 1)\n",
    "A = gauss_ens(k, k)\n",
    "\n",
    "tail = (y.T @ A @ z).item()\n",
    "tail"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f8ec0b00-4508-4bb6-a53c-69f92fbeac90",
   "metadata": {},
   "source": [
    "NOTE: in GGA, multiplication doesn't distribute over sums"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "968412b1-3aac-4933-b6f7-2c11d7fdd55e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "c x^0.6667 exp(-1.5 * x^0.6667)"
      ]
     },
     "execution_count": 31,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "sum(gga.normal(0, 1) * gga.normal(0, 1) for _ in range(k)) * gga.normal(0, 1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "04d1dc93-2841-4915-86b5-cdd85c1e3d27",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "c x^(-0.6667) exp(-1.5 * x^0.6667)"
      ]
     },
     "execution_count": 32,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "sum(gga.normal(0, 1) * gga.normal(0, 1) * gga.normal(0, 1) for _ in range(k))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ba197a32-6a03-4acb-b5ef-c8f582a276f1",
   "metadata": {},
   "source": [
    "The resulting GG approximation is given by:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "b0dd9058-1673-4af9-ae6a-7487912bebf8",
   "metadata": {},
   "outputs": [],
   "source": [
    "q = gga.make_ggdist(tail)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "2fee0f06-b485-4e4c-8b4e-13b298b707d4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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7drFr3/9a370ox/x3nvvAWCaJmlpafTu3RvTNDly5Ai/+tWvmDhxImVlZW1RrPDjnccZwFG13c801R5TH6eIhIeA93F6lJaWsmbNGlatWsWBAwcAd1harVZuuukmxo8fzw033ADA559/zmuvvca6dev4/PPP+d3vfsejjz7aWkULX04TV4B/VYrs4Vl2TzVOEQkPAQ1Ol8vFv//9b1atWsWHH36Iw+HwNr2mp6dzxx13MHbsWJKSkuocd+WVV/L888+zbNkyFi5cSG5uroKzFZhOM6DNtACWGAsRyZGqcYpI2AhIcB4+fJhVq1axevVqiouLAXftMjY2lptvvpk77riDgQMHnvc8t99+OwsXLiQ/Pz8QxZJzuQI7MMjD2l2LIIhI+AhIcP7kJz/BMAxv7bJfv36MHz+eW2+9lYSEhGafJy4uDqBZA4RKS0vJzs4mNzeX/Px8YmNj6d+/PxMnTmT48OE+v4fy8nI+/PBDNm7cyO7du/nuu++w2+0kJydz+eWXc8cddzB06NDznmfPnj0sXbqUrVu3cuLECbp06cI111zD9OnT6du3r8/lCiiXiSvANU5wj6yt+ddpTNPEaMGcUBGRUBCwptqEhARGjRrFnXfeSVZWll/niIiI4I033jjvfvv372fq1KmUlJQAEB8fT1lZGZs2bWLTpk1MmTLF56be2267ja+//tr796ioKKxWK/n5+eTn57Nu3TpuvfVWFi5cSGRkw9+2tWvXMnfuXOx2d7NlYmIiBQUFrF27lg8++ICnn36aW265xadyBZLpbKUaZw8rps3EWeIgMsUa+AuIiASRgHyMLly4kI0bN/Lf//3ffoemxzXXXMM111zT6HabzcbMmTMpKSkhIyOD1atXs337drZv387s2bMxDIPly5ezatUqn67rcDjIyMhg7ty5fPDBB+zatYsdO3awYcMGxo8fD7iDccmSJQ0ev3//fm9ojhw5ko0bN/LZZ5+xceNGRo4cid1uZ86cORw8eNCncgVUI32cUVZLnZWEfF1NSI8XE5FwEpDgHDt2LNHR0YE41XmtXLmSo0ePEhcXR3Z2trf5MyYmhhkzZjB58mQAlixZ4q35NcfChQtZs2YNd999N5dccon39Z49e/LEE0/w05/+FHA/Qq26urre8c8//zx2u53+/fuzaNEiUlNTAUhNTWXRokX069cPm83G888/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",
      "text/plain": [
       "<Figure size 500x500 with 1 Axes>"
      ]
     },
     "execution_count": 34,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import pandas as pd\n",
    "import torch\n",
    "\n",
    "xs = torch.logspace(-25, np.log10(25), steps=1000)\n",
    "sns.lineplot(\n",
    "    ax=fg.ax,\n",
    "    data=pd.DataFrame(\n",
    "        {\n",
    "            \"x\": xs,\n",
    "            \"q(x)\": q.log_prob(xs).exp(),\n",
    "        }\n",
    "    ),\n",
    "    x=\"x\",\n",
    "    y=\"q(x)\",\n",
    "    color=\"m\",\n",
    ")\n",
    "fg.fig"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "575383e8-b958-4285-9a08-59c32248b07e",
   "metadata": {},
   "source": [
    "## Loop for figure"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "491bdddf-7b45-4c89-b79a-e229837660d9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "c exp(-1.5 * x^0.6667)\n",
      "c x^0.6667 exp(-1.5 * x^0.6667)\n",
      "c x^2.333 exp(-1.5 * x^0.6667)\n"
     ]
    }
   ],
   "source": [
    "N = 5000\n",
    "\n",
    "ks = [3, 5, 10]\n",
    "fig, axs = plt.subplots(1, len(ks), figsize=(5.2, 2.0), sharex=True, sharey=True)\n",
    "for i, k in enumerate(ks):\n",
    "    samples = np.zeros(N)\n",
    "    for j in range(N):\n",
    "        samples[j] = abs(\n",
    "            np.random.randn(1, k) @ np.random.randn(k, k) @ np.random.randn(k, 1)\n",
    "        ).item()\n",
    "    sns.histplot(ax=axs[i], data=samples, stat=\"density\")\n",
    "\n",
    "    y = gauss_ens(k, 1)\n",
    "    z = gauss_ens(k, 1)\n",
    "    A = gauss_ens(k, k)\n",
    "    tail = (y.T @ A @ z).item()\n",
    "    print(tail)\n",
    "    q = gga.make_ggdist(tail)\n",
    "    xs = torch.logspace(-25, np.log10(samples.max()), steps=1000)\n",
    "    sns.lineplot(\n",
    "        ax=axs[i],\n",
    "        data=pd.DataFrame(\n",
    "            {\n",
    "                \"x\": xs,\n",
    "                \"q(x)\": q.log_prob(xs).exp(),\n",
    "            }\n",
    "        ),\n",
    "        x=\"x\",\n",
    "        y=\"q(x)\",\n",
    "        color=\"y\",\n",
    "    )\n",
    "    axs[i].set(title=f\"k={k}\", xlim=[0., 25.], ylim=[0., .3], xlabel='')\n",
    "axs[-1].legend(['q(x)','$\\\\vert x^\\\\top A y \\\\vert$'])\n",
    "fig.tight_layout()\n",
    "fig.savefig('normal-power-sum.pgf')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "774e39ec",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "interpreter": {
   "hash": "83b589253c6ae2165fd99d3b5e434b8a0ff74c98e791d87ced25152a201010fd"
  },
  "kernelspec": {
   "display_name": "Python 3.10.4 ('gga')",
   "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.10.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
