{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 324,
   "id": "CxPbssuMmpfq",
   "metadata": {
    "id": "CxPbssuMmpfq",
    "tags": []
   },
   "outputs": [],
   "source": [
    "# !apt update\n",
    "# !apt install imagemagick\n",
    "# !apt install pdf2svg\n",
    "# !apt install texlive texlive-latex-extra"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 325,
   "id": "xMOrfL6pll78",
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/"
    },
    "id": "xMOrfL6pll78",
    "outputId": "3d3f1245-6100-416c-d094-06308748eb96",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Requirement already satisfied: powerlaw in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (1.5)\n",
      "Requirement already satisfied: flowtorch in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (0.8)\n",
      "Requirement already satisfied: seaborn-qqplot in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (0.5.0)\n",
      "Requirement already satisfied: scipy in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from powerlaw) (1.10.0)\n",
      "Requirement already satisfied: numpy in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from powerlaw) (1.24.2)\n",
      "Requirement already satisfied: mpmath in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from powerlaw) (1.3.0)\n",
      "Requirement already satisfied: matplotlib in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from powerlaw) (3.7.0)\n",
      "Requirement already satisfied: torch>=1.8.1 in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from flowtorch) (1.13.1.post200)\n",
      "Requirement already satisfied: seaborn in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from seaborn-qqplot) (0.12.2)\n",
      "Requirement already satisfied: tabulate in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from seaborn-qqplot) (0.9.0)\n",
      "Requirement already satisfied: pandas in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from seaborn-qqplot) (1.5.3)\n",
      "Requirement already satisfied: typing_extensions in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from torch>=1.8.1->flowtorch) (4.5.0)\n",
      "Requirement already satisfied: cycler>=0.10 in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from matplotlib->powerlaw) (0.11.0)\n",
      "Requirement already satisfied: pyparsing>=2.3.1 in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from matplotlib->powerlaw) (3.0.9)\n",
      "Requirement already satisfied: python-dateutil>=2.7 in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from matplotlib->powerlaw) (2.8.2)\n",
      "Requirement already satisfied: pillow>=6.2.0 in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from matplotlib->powerlaw) (9.4.0)\n",
      "Requirement already satisfied: importlib-resources>=3.2.0 in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from matplotlib->powerlaw) (5.12.0)\n",
      "Requirement already satisfied: fonttools>=4.22.0 in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from matplotlib->powerlaw) (4.39.3)\n",
      "Requirement already satisfied: kiwisolver>=1.0.1 in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from matplotlib->powerlaw) (1.4.4)\n",
      "Requirement already satisfied: contourpy>=1.0.1 in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from matplotlib->powerlaw) (1.0.7)\n",
      "Requirement already satisfied: packaging>=20.0 in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from matplotlib->powerlaw) (23.1)\n",
      "Requirement already satisfied: pytz>=2020.1 in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from pandas->seaborn-qqplot) (2023.3)\n",
      "Requirement already satisfied: zipp>=3.1.0 in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from importlib-resources>=3.2.0->matplotlib->powerlaw) (3.15.0)\n",
      "Requirement already satisfied: six>=1.5 in /home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages (from python-dateutil>=2.7->matplotlib->powerlaw) (1.16.0)\n"
     ]
    }
   ],
   "source": [
    "!pip install powerlaw flowtorch seaborn-qqplot"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 326,
   "id": "CMsWtie7o_-A",
   "metadata": {
    "cellView": "form",
    "id": "CMsWtie7o_-A",
    "tags": []
   },
   "outputs": [],
   "source": [
    "#@title gg_algebra.py\n",
    "\n",
    "import torch\n",
    "import torch.distributions as dist\n",
    "import numpy as np\n",
    "import scipy.special as sp\n",
    "\n",
    "def _eq(a,b):\n",
    "    return abs(a-b) < 1e-15\n",
    "\n",
    "def _leq(a,b):\n",
    "    return a - b < 1e-15\n",
    "\n",
    "def _geq(a,b):\n",
    "    return b - a < 1e-15\n",
    "\n",
    "\n",
    "class GGTail(object):\n",
    "    \n",
    "    def __init__(self, nu, sigma, rho):\n",
    "        #self._check(nu, sigma, rho)\n",
    "        self.nu    = nu\n",
    "        self.sigma = sigma\n",
    "        self.rho   = rho\n",
    "            \n",
    "    def __add__(self, other):\n",
    "        \"\"\"The addition operation (additive convolution)\"\"\"\n",
    "        if not isinstance(other, self.__class__):\n",
    "            return self._copy()\n",
    "        \n",
    "        nu_1, sigma_1, rho_1 = self.params()\n",
    "        nu_2, sigma_2, rho_2 = other.params()\n",
    "        \n",
    "        if _eq(rho_1, rho_2):\n",
    "            rho = 0.5*(rho_1+rho_2)\n",
    "            \n",
    "            if _eq(rho, 1):\n",
    "                # Exponential case\n",
    "                return self._rv(nu_1 + nu_2 + 1, \n",
    "                                min(sigma_1, sigma_2), 1)\n",
    "            elif rho > 1:\n",
    "                # Superexponential case\n",
    "                sigma  = sigma_1**(-1.0/(rho-1))\n",
    "                sigma += sigma_2**(-1.0/(rho-1))\n",
    "                sigma = sigma**(1 - rho)\n",
    "                return self._rv(nu_1 + nu_2 + 1 - rho_1/2,\n",
    "                                sigma, rho)\n",
    "            \n",
    "            else:\n",
    "                # Subexponential case\n",
    "                return max(self, other)._copy()\n",
    "        \n",
    "        else:\n",
    "            return max(self, other)._copy()\n",
    "        \n",
    "    def __mul__(self, other):\n",
    "        \"\"\"The multiplication operation (multiplicative convolution)\"\"\"\n",
    "        if self is other:\n",
    "            return self**2\n",
    "        if not isinstance(other, self.__class__):\n",
    "            if _eq(other, 0):\n",
    "                return 0\n",
    "            # Scalar multiplication\n",
    "            return self._rv(self.nu, \n",
    "                            self.sigma / abs(other)**self.rho,\n",
    "                            self.rho)\n",
    "        \n",
    "        nu_1, sigma_1, rho_1 = self.params()\n",
    "        nu_2, sigma_2, rho_2 = other.params()\n",
    "        \n",
    "        if _leq(rho_1,0) and rho_2  > 0:\n",
    "            return self._reg(abs(nu_1))\n",
    "        if rho_1 > 0 and _leq(rho_2, 0):\n",
    "            return self._reg(abs(nu_2))\n",
    "        if _eq(rho_1,0) and _eq(rho_2,0):\n",
    "            return self._reg(min(abs(nu_1),abs(nu_2)))\n",
    "        \n",
    "        mu = 1.0/abs(rho_1) + 1.0/abs(rho_2)\n",
    "        sigma  = (sigma_1*abs(rho_1))**(1.0/(mu*abs(rho_1)))\n",
    "        sigma *= (sigma_2*abs(rho_2))**(1.0/(mu*abs(rho_2)))\n",
    "        sigma *= mu\n",
    "        \n",
    "        if rho_1 < 0:\n",
    "            nu = nu_1/abs(rho_1) + nu_2/abs(rho_2) + 1/2\n",
    "            return self._rv(nu/mu, sigma, -1/mu)\n",
    "        else:\n",
    "            nu = nu_1/rho_1 + nu_2/rho_2 - 1/2\n",
    "            return self._rv(nu/mu, sigma, 1/mu)\n",
    "        \n",
    "    def __pow__(self, num):\n",
    "        \"\"\"The power operation\"\"\"\n",
    "        if (num < 0) and (_eq(self.rho, 0) or _leq((self.nu+1)/self.rho, 0)):\n",
    "            # Reciprocal by assuming bounded density near zero\n",
    "            return self._reg(2.0)\n",
    "        \n",
    "        return self._rv((self.nu+1)/num - 1, self.sigma, self.rho/num)\n",
    "    \n",
    "    def __and__(self, other):\n",
    "        \"\"\"Product of densities operation\"\"\"\n",
    "        assert isinstance(other, self.__class__)\n",
    "        nu_1, sigma_1, rho_1 = self.params()\n",
    "        nu_2, sigma_2, rho_2 = other.params()\n",
    "        nu = nu_1 + nu_2\n",
    "        if _eq(rho_1, rho_2):\n",
    "            rho = 0.5*(rho_1 + rho_2)\n",
    "            return self._rv(nu, sigma_1 + sigma_2, rho)\n",
    "        if rho_1 < rho_2:\n",
    "            return self._rv(nu, sigma_1, rho_1)\n",
    "        else:\n",
    "            return self._rv(nu, sigma_2, rho_2)\n",
    "        \n",
    "        \n",
    "    def __sub__(self, other):\n",
    "        \"\"\"The subtraction operation\"\"\"\n",
    "        return self + (-1.0 * other)\n",
    "    \n",
    "    def __rsub__(self, other):\n",
    "        \"\"\"The subtraction operation\"\"\"\n",
    "        return (-1.0 * self) + other\n",
    "        \n",
    "    def __truediv__(self, other):\n",
    "        \"\"\"The division operation (involving reciprocal)\"\"\"\n",
    "        return self * (other ** (-1))\n",
    "    \n",
    "    def __rtruediv__(self, other):\n",
    "        \"\"\"Reciprocal operation\"\"\"\n",
    "        return self ** (-1)\n",
    "    \n",
    "    def __abs__(self):\n",
    "        \"\"\"Absolute value operation\"\"\"\n",
    "        return self._copy()\n",
    "    \n",
    "    def log(self):\n",
    "        \"\"\"The logarithm operation\"\"\"\n",
    "        if self.nu < -1 and _leq(self.rho, 0):\n",
    "            return self._rv(0.0, abs(self.nu)-1, 1.0)\n",
    "        \n",
    "        return self._light()\n",
    "    \n",
    "    def exp(self):\n",
    "        \"\"\"The exponential operation\"\"\"\n",
    "        if _geq(self.rho, 1):\n",
    "            return self._reg(self.sigma + 1)\n",
    "        \n",
    "        return self._heavy()\n",
    "    \n",
    "        \n",
    "    __radd__ = __add__\n",
    "    __rmul__ = __mul__\n",
    "    \n",
    "    ### ORDERING ###\n",
    "    \n",
    "    def __ge__(self, other):\n",
    "        \"\"\"Comparing heaviness of the tail: does\n",
    "        self have a heavier tail than other?\"\"\"\n",
    "        nu_1, sigma_1, rho_1 =  self.params()\n",
    "        nu_2, sigma_2, rho_2 = other.params()\n",
    "        \n",
    "        if _leq(rho_1, 0) and _leq(rho_2, 0):\n",
    "            return (abs(nu_1) < abs(nu_2))\n",
    "        \n",
    "        if _eq(rho_1, rho_2):\n",
    "            if _eq(sigma_1, sigma_2):\n",
    "                return nu_1 > nu_2\n",
    "            return sigma_1 < sigma_2\n",
    "        \n",
    "        return (rho_1 < rho_2)\n",
    "        \n",
    "    def __eq__(self, other):\n",
    "        return _eq(self.nu,other.nu) and \\\n",
    "                _eq(self.sigma,other.sigma) and \\\n",
    "                _eq(self.rho,other.rho)\n",
    "    \n",
    "    def __gt__(self, other):\n",
    "        return (self.__ge__(other)) and not (self.__eq__(other))\n",
    "    \n",
    "    def __ne__(self, other):\n",
    "        return not (self.__eq__(other))\n",
    "    \n",
    "    def __lt__(self, other):\n",
    "        return not (self.__ge__(other))\n",
    "    \n",
    "    def __le__(self, other):\n",
    "        return other.__ge__(self)\n",
    "    \n",
    "    ### STRING REPRESENTATION ###\n",
    "    \n",
    "    def __str__(self):\n",
    "        \n",
    "        try:\n",
    "            if _eq(self.nu, -1) and _leq(self.rho, 0):\n",
    "                return \"super heavy tail\"\n",
    "            if np.isinf(self.rho):\n",
    "                return \"super light tail\"\n",
    "        except TypeError:\n",
    "            pass\n",
    "        \n",
    "        tail = \"c\"\n",
    "        try:\n",
    "            nu = '%s' % float('%.4g' % self.nu)\n",
    "            if not _eq(self.nu, 0):\n",
    "                if self.nu < 0:\n",
    "                    tail += \" x^({nu})\".format(nu=nu)\n",
    "                else:\n",
    "                    tail += \" x^{nu}\".format(nu=nu)\n",
    "        except TypeError:\n",
    "            tail += \" x^({nu})\".format(nu=self.nu)\n",
    "        try:\n",
    "            if _eq(self.sigma, 1.0):\n",
    "                sigma = \"\"\n",
    "            else:\n",
    "                sigma = '%s' % float('%.4g' % self.sigma)\n",
    "                sigma += \" * \"\n",
    "        except TypeError:\n",
    "            sigma = \"({sigma}) * \".format(sigma=str(self.sigma))\n",
    "        try:\n",
    "            if _eq(self.rho, 1.0):\n",
    "                rho = \"\"\n",
    "            else:\n",
    "                rho = '%s' % float('%.4g' % self.rho)\n",
    "                if self.rho < 0:\n",
    "                    rho = \"^({rho})\".format(rho=rho)\n",
    "                else:\n",
    "                    rho = \"^{rho}\".format(rho=rho)\n",
    "            if not _eq(self.rho, 0):\n",
    "                tail += \" exp(-{sigma}x{rho})\".format(\n",
    "                    sigma=sigma, rho=rho)\n",
    "        except TypeError:\n",
    "            tail += \" exp(-{sigma}x^({rho}))\".format(\n",
    "                    sigma=sigma, rho=self.rho)\n",
    "        \n",
    "        return tail\n",
    "    \n",
    "    def __repr__(self):\n",
    "        return str(self)\n",
    "    \n",
    "    ### HELPERS ###\n",
    "    \n",
    "    def _check(self, nu, sigma, rho):\n",
    "        \"\"\"Check if the parameters are sane\"\"\"\n",
    "        positivity = None\n",
    "        try:\n",
    "            positivity = sigma > 0\n",
    "        except TypeError:\n",
    "            pass\n",
    "        if positivity is not None:\n",
    "            assert positivity\n",
    "        if _leq(rho, 0):\n",
    "            assert _leq(nu, 1.0)\n",
    "            \n",
    "    def params(self):\n",
    "        \"\"\"Return a tuple of the class parameters\"\"\"\n",
    "        return (self.nu, self.sigma, self.rho)\n",
    "            \n",
    "    def _copy(self):\n",
    "        \"\"\"Return a copy of the current tail object\"\"\"\n",
    "        return self.__class__(self.nu, self.sigma, self.rho)\n",
    "                                  \n",
    "    def _rv(self,nu,sigma,rho):\n",
    "        \"\"\"Creates a new random variable with given parameters\"\"\"\n",
    "        return self.__class__(nu,sigma,rho)\n",
    "                \n",
    "    def _reg(self,nu):\n",
    "        \"\"\"Creates a new regularly-varying random variable with\n",
    "        parameter nu\"\"\"\n",
    "        return self.__class__(-nu,1,0)\n",
    "    \n",
    "    def _light(self):\n",
    "        \"\"\"Creates a new super-light random variable\"\"\"\n",
    "        return self.__class__(0.0, 1.0, np.inf)\n",
    "    \n",
    "    def _heavy(self):\n",
    "        \"\"\"Creates a new super-heavy random variable\"\"\"\n",
    "        return self._reg(1.0)\n",
    "    \n",
    "\n",
    "RegVar = lambda nu: GGTail(-nu, 1.0, 0.0)\n",
    "SuperHeavy = lambda: RegVar(1.0)\n",
    "SuperLight = lambda: GGTail(0.0, 1.0, np.inf)\n",
    "    \n",
    "benktander2 = lambda a,b: GGTail(2*b-2, a/b, b)\n",
    "betaprime   = lambda a,b,*vars: RegVar(b+1)\n",
    "burr        = lambda c,k: RegVar(c*k+1)\n",
    "cauchy      = lambda *args: RegVar(2.0)\n",
    "chi         = lambda k: GGTail(k-1,0.5,2)\n",
    "chi2        = lambda k: GGTail(k/2-1,0.5,1)\n",
    "dagum       = lambda a,*args: RegVar(a+1)\n",
    "davis       = lambda n,b,*args: GGTail(-1-n,b,1)\n",
    "exponential = lambda lam: GGTail(0,lam,1)\n",
    "fdistn      = lambda d1,d2: RegVar(d2/2+1)\n",
    "fisherz     = lambda d1,d2: GGTail(0,d2,1)\n",
    "frechet     = lambda a,lam,*args: GGTail(-1-a,lam**a,-a)\n",
    "gamma       = lambda a,b: GGTail(a-1,b,1)\n",
    "ggompertz   = lambda b,s: GGTail(0,b*s,1)\n",
    "hyperbolic  = lambda alpha,beta,lam,*args: GGTail(lam-1,alpha-beta,1)\n",
    "gennormal   = lambda alpha,beta,*args: GGTail(0, alpha**(-beta),beta)\n",
    "geostable   = lambda alpha: RegVar(alpha+1)\n",
    "gompertz    = lambda *args: SuperLight()\n",
    "gumbel      = lambda beta,*args: GGTail(0, 1.0/beta, 1)\n",
    "gumbel2     = lambda alpha,beta: GGTail(-alpha-1,beta,-alpha)\n",
    "holtsmark   = lambda: RegVar(5/2)\n",
    "hypsecant   = lambda: GGTail(0,np.pi/2,1)\n",
    "invchi2     = lambda k: GGTail(-k/2-1,1/2,-1)\n",
    "invgamma    = lambda alpha,beta: GGTail(-alpha-1,beta,-1)\n",
    "invnormal   = lambda *args: GGTail(-2,1/2,-2)\n",
    "levy        = lambda c,*args: GGTail(-3/2,c/2,-1)\n",
    "laplace     = lambda lam: GGTail(0,1/lam,1)\n",
    "logistic    = lambda lam: GGTail(0,1/lam,1)\n",
    "logcauchy   = lambda *args: SuperHeavy()\n",
    "loglaplace  = lambda lam,*args: RegVar(1/lam+1)\n",
    "loglogistic = lambda alpha,beta: RegVar(beta+1)\n",
    "logt        = lambda *args: SuperHeavy()\n",
    "lomax       = lambda alpha, *args: RegVar(alpha+1)\n",
    "maxboltz    = lambda sigma: GGTail(2, 1/(2*sigma**2), 2)\n",
    "normal      = lambda mu,sigma: GGTail(0, 1/(2*sigma**2), 2)\n",
    "pareto      = lambda alpha, *args: RegVar(alpha+1)\n",
    "rayleigh    = lambda sigma: GGTail(1, 1/(2*sigma**2), 2)\n",
    "rice        = lambda sigma, *args: GGTail(1/2, 1/(2*sigma**2), 2)\n",
    "skewnormal  = lambda mu,sigma: GGTail(0, 1/(2*sigma**2), 2)\n",
    "slash       = lambda: GGTail(-2,1/2,2)\n",
    "stable      = lambda alpha: RegVar(alpha+1)\n",
    "student     = lambda nu: RegVar(nu+1)\n",
    "tracywidom  = lambda beta: (-3*beta/4 - 1, 2*beta/3, 3/2)\n",
    "voigt       = lambda *args: RegVar(2)\n",
    "weibull     = lambda rho, lam: GGTail(rho-1, 1/lam**rho, rho)\n",
    "    \n",
    "############# FUNCTIONS #############\n",
    "    \n",
    "class TailFunction(object):\n",
    "    \n",
    "    def __init__(self, func, num_args):\n",
    "        self.func = func\n",
    "        self.num_args = num_args\n",
    "        assert num_args > 0\n",
    "        \n",
    "    def filter_args(self, args):\n",
    "        assert len(args) == self.num_args\n",
    "        list_vars = list(args)\n",
    "        idx = 0\n",
    "        for var in list_vars:\n",
    "            if not isinstance(var, GGTail):\n",
    "                list_vars.pop(idx)\n",
    "            else:\n",
    "                idx += 1\n",
    "        return list_vars\n",
    "        \n",
    "class PowerFunction(TailFunction):\n",
    "    \n",
    "    def __init__(self, alpha, func, num_args, const = 1.0):\n",
    "        super().__init__(func, num_args)\n",
    "        self.alpha = alpha\n",
    "        self.const = const\n",
    "            \n",
    "    def __call__(self, *args):\n",
    "        list_vars = self.filter_args(args)\n",
    "        if len(list_vars) == 0:\n",
    "            return self.func(*args)\n",
    "        else:\n",
    "            return self.const * max(list_vars)**self.alpha\n",
    "        \n",
    "class BoundedFunction(TailFunction):\n",
    "    \n",
    "    def __init__(self, bound, func, num_args):\n",
    "        super().__init__(func, num_args)\n",
    "        self.bound = bound\n",
    "        \n",
    "    def __call__(self, *args):\n",
    "        list_vars = self.filter_args(args)\n",
    "        if len(list_vars) == 0:\n",
    "            return self.func(*args)\n",
    "        else:\n",
    "            return self.bound\n",
    "        \n",
    "class BoundedFunction(TailFunction):\n",
    "    \n",
    "    def __init__(self, bound, func, num_args):\n",
    "        super().__init__(func, num_args)\n",
    "        self.bound = bound\n",
    "        \n",
    "    def __call__(self, *args):\n",
    "        list_vars = self.filter_args(args)\n",
    "        if len(list_vars) == 0:\n",
    "            return self.func(*args)\n",
    "        else:\n",
    "            return self.bound\n",
    "        \n",
    "class ExpFunction(TailFunction):\n",
    "    def __init__(self, scale, exponent, func, num_args):\n",
    "        super().__init__(func, num_args)\n",
    "        self.scale = scale\n",
    "        self.exponent = exponent\n",
    "        \n",
    "    def __call__(self, *args):\n",
    "        list_vars = self.filter_args(args)\n",
    "        if len(list_vars) == 0:\n",
    "            return self.func(*args)\n",
    "        else:\n",
    "            return (max(list_vars)**self.exponent * self.scale).exp()\n",
    "        \n",
    "class LogFunction(TailFunction):\n",
    "    def __init__(self, scale, func, num_args):\n",
    "        super().__init__(func, num_args)\n",
    "        self.scale = scale\n",
    "        \n",
    "    def __call__(self, *args):\n",
    "        list_vars = self.filter_args(args)\n",
    "        if len(list_vars) == 0:\n",
    "            return self.func(*args)\n",
    "        else:\n",
    "            return (max(list_vars).log() * self.scale)\n",
    "        \n",
    "LipschitzFunction = lambda func, num_args, const = 1.0: \\\n",
    "                        PowerFunction(1.0, func, num_args, const)\n",
    "    \n",
    "exp = ExpFunction(1.0, 1.0, np.exp, 1)\n",
    "log = LogFunction(1.0, np.log, 1)\n",
    "relu = LipschitzFunction(lambda x: max([x,0]), 1, 1.0)\n",
    "sin = BoundedFunction(1.0, np.sin, 1)\n",
    "cos = BoundedFunction(1.0, np.cos, 1)\n",
    "sinh = ExpFunction(0.5, 1.0, np.sinh, 1)\n",
    "cosh = ExpFunction(0.5, 1.0, np.cosh, 1)\n",
    "tanh = BoundedFunction(np.pi / 2, np.tanh, 1)\n",
    "arcsinh = LogFunction(1.0, np.arcsinh, 1)\n",
    "arccosh = LogFunction(1.0, np.arccosh, 1)\n",
    "sqrt = PowerFunction(0.5, np.sqrt, 1, 1.0)\n",
    "erf = BoundedFunction(1.0, sp.erf, 1)\n",
    "\n",
    "############# MATRICES #############\n",
    "\n",
    "def MatGGTail(size, distn):\n",
    "    matrix = np.empty(size, dtype=object)\n",
    "    for idx in range(size[0]):\n",
    "        for idy in range(size[1]):\n",
    "            matrix[idx,idy] = GGTail(*distn.params())\n",
    "    return matrix\n",
    "\n",
    "gauss_ens = lambda *size: MatGGTail(size, normal(0,1))\n",
    "\n",
    "class AbsTransform(dist.transforms.Transform):\n",
    "    r\"\"\"\n",
    "    Transform via the mapping :math:`y = |x|`.\n",
    "    \"\"\"\n",
    "    domain = dist.constraints.real\n",
    "    codomain = dist.constraints.positive\n",
    "\n",
    "    def __eq__(self, other):\n",
    "        return isinstance(other, AbsTransform)\n",
    "\n",
    "    def _call(self, x):\n",
    "        return x.abs()\n",
    "\n",
    "    def _inverse(self, y):\n",
    "        return y\n",
    "\n",
    "    def log_abs_det_jacobian(self, x, y):\n",
    "        return torch.tensor(2.0).log().expand(x.shape)\n",
    "\n",
    "\n",
    "def make_positive(d: dist.Distribution) -> dist.Distribution:\n",
    "    return dist.TransformedDistribution(\n",
    "        d,\n",
    "        [AbsTransform()],\n",
    "    )\n",
    "\n",
    "\n",
    "def make_ggdist(tail: GGTail) -> dist.Distribution:\n",
    "    if abs(tail.rho) < 0.1:\n",
    "        return make_positive(\n",
    "            dist.StudentT(\n",
    "                df=-(tail.nu + 1),\n",
    "            )\n",
    "        )\n",
    "    return dist.TransformedDistribution(\n",
    "        dist.Gamma(\n",
    "            concentration=(tail.nu + 1) / tail.rho,\n",
    "            rate=tail.sigma,\n",
    "        ),\n",
    "        [dist.transforms.PowerTransform(exponent=1.0 / tail.rho)],\n",
    "    )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 327,
   "id": "e7a51eb2-a916-4481-8917-2b13b8e23097",
   "metadata": {
    "id": "e7a51eb2-a916-4481-8917-2b13b8e23097",
    "tags": []
   },
   "outputs": [],
   "source": [
    "from typing import Dict\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import powerlaw\n",
    "import seaborn as sns\n",
    "import torch\n",
    "import torch.distributions as dist\n",
    "from matplotlib import pyplot as plt\n",
    "from tqdm.auto import tqdm\n",
    "\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 as bm\n",
    "# import beanmachine.ppl.experimental.gg_algebra as gga\n",
    "# from beanmachine.ppl.experimental.vi.variational_world import VariationalWorld\n",
    "# from beanmachine.ppl.world import World\n",
    "\n",
    "sns.set_style(\"darkgrid\")\n",
    "sns.set_context(\"paper\", font_scale = 1.)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 913,
   "id": "oK7QZj_AnAE7",
   "metadata": {
    "id": "oK7QZj_AnAE7",
    "tags": []
   },
   "outputs": [],
   "source": [
    "def make_samples(N):\n",
    "    #samples = dist.Exponential(1.0).sample((N,5)).sum(axis=1) * 10\n",
    "    samples = torch.stack([dist.StudentT(i).sample((N,))**2 for i in range(2,6)], axis=1).sum(axis=1)\n",
    "    return samples\n",
    "\n",
    "\n",
    "N = 1000000\n",
    "samps = make_samples(N)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 939,
   "id": "a54db829-5f05-4743-b722-2e3e33f6746d",
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 122,
     "referenced_widgets": [
      "f81ce42cf96f44bb825475f791703948",
      "0072a65d9c8643a187c17d656586cefb",
      "48a3467b5983445a839eca742537e195",
      "2c684b979a4f4148b9113fd69d582a9d",
      "ebaf9556ed42453ca590126ec8f25dea",
      "73d1bec1ef2c45ccb5dafff1ed2d44af",
      "f94489e61e9341558dc8f5f81bfd4fba",
      "cdfd007ea89a495f9042f73fb3dbc187",
      "726e4ff203f64fd7b1e18975265938ae",
      "ebe29fab349d41d28f3d3f07b67c6f45",
      "0b3159afb4844ccd940f9ba4d8692e98"
     ]
    },
    "id": "a54db829-5f05-4743-b722-2e3e33f6746d",
    "outputId": "75e9ae3d-0d3f-4bc4-a5a0-08366d916123",
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "c x^(-2.0)\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "3e7540062e294cc484c9b3f2fae60d05",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/201 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from tqdm.auto import tqdm\n",
    "\n",
    "import flowtorch.bijectors\n",
    "import flowtorch.distributions\n",
    "import flowtorch.parameters\n",
    "\n",
    "#tail = 10 * sum([exponential(1) for _ in range(5)])\n",
    "tail = sum([student(i)**2 for i in range(2, 6)])\n",
    "print(tail)\n",
    "\n",
    "# transform and train flow in log domain so base distribution supported on R\n",
    "q = make_ggdist(tail)\n",
    "base_dist = dist.TransformedDistribution(\n",
    "    q,\n",
    "    [dist.transforms.ExpTransform().inv],\n",
    ")\n",
    "\n",
    "\n",
    "#params = flowtorch.parameters.DenseAutoregressive(hidden_dims=(64,))\n",
    "bijectors = flowtorch.bijectors.AffineAutoregressive()#params_fn=params)\n",
    "#bijectors = flowtorch.bijectors.SplineAutoregressive(bound=10, count_bins=2)\n",
    "flow = flowtorch.distributions.Flow(base_dist, bijectors)\n",
    "opt = torch.optim.AdamW(flow.parameters(), lr=0.001)\n",
    "for idx in (pbar := tqdm(range(201))):\n",
    "    opt.zero_grad()\n",
    "\n",
    "    loss = 0.0\n",
    "    for i in range(10):\n",
    "        loss -= flow.log_prob(samps[torch.randint(N,(1,))].log())\n",
    "    loss = loss / 10\n",
    "\n",
    "    pbar.set_description(f\"epoch: {idx}, loss: {loss}\")\n",
    "\n",
    "    loss.backward()\n",
    "    opt.step()\n",
    "\n",
    "flow = dist.TransformedDistribution(\n",
    "    flow,\n",
    "    [dist.transforms.ExpTransform()]\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 940,
   "id": "i-lEEVBp2hx_",
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 468
    },
    "id": "i-lEEVBp2hx_",
    "outputId": "ededd6e0-aba3-42fa-ac58-54f795d6b437",
    "tags": []
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages/scipy/interpolate/_interpolate.py:701: RuntimeWarning: invalid value encountered in multiply\n",
      "  y_new = slope*(x_new - x_lo)[:, None] + y_lo\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<seaborn.axisgrid.PairGrid at 0x7fb3471f3e80>"
      ]
     },
     "execution_count": 940,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 300x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from seaborn_qqplot import pplot\n",
    "\n",
    "fig = pplot(data=pd.DataFrame({\n",
    "    'Target': samps,\n",
    "    'Flow': 1.6*base_dist.sample((1000000,)).exp()\n",
    "    #1.8*base_dist.sample((1000000,)).exp()\n",
    "}), x='Target', y='Flow', kind='qq', height=2, aspect=3/2)#, display_kws={\"identity\": True})\n",
    "fig.set(title=\"(i) GGA\")\n",
    "xx = np.linspace(0,10000,3)\n",
    "plt.plot(xx, xx, 'k--')\n",
    "plt.xlim([0,10000])\n",
    "plt.ylim([0,10000])\n",
    "fig.tight_layout()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 941,
   "id": "LJ6tDzTOwdGO",
   "metadata": {
    "id": "LJ6tDzTOwdGO",
    "tags": []
   },
   "outputs": [],
   "source": [
    "fig.savefig(\"gga_qq.png\", dpi=500)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fdb92743",
   "metadata": {
    "id": "fdb92743"
   },
   "source": [
    "# Compare vs Gaussian"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 943,
   "id": "oY50Smxsxu7v",
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 104,
     "referenced_widgets": [
      "def2376828aa4a14a575702349e0c27b",
      "b98ebd27b6834f74ace9468761148b60",
      "7c79b4f490834d9381a951011da270b0",
      "ec9c533c50ca4a77ba6b1ff9e384fb93",
      "ea08d940b991408cb58ea981e9803534",
      "58fee13b7c76483e802b7a0e0c9ca21a",
      "df4e853a026b40c999efed50489938dc",
      "b3522228b1b94327800ec8d7f7eb1cef",
      "bd39f72b67004725828ce5c49ada154d",
      "d3afeead45d04b71ad89545b7df8f10d",
      "334cbb56667a4b90b44275363c9dee17"
     ]
    },
    "id": "oY50Smxsxu7v",
    "outputId": "9a8801e0-7879-43bf-a1e0-bcfab67eb971",
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "f5c594531a96441480df9a3b6842ef0a",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/201 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# transform and train flow in log domain so base distribution supported on R\n",
    "q = dist.Normal(0,1)\n",
    "\n",
    "#params = flowtorch.parameters.DenseAutoregressive(hidden_dims=(64,))\n",
    "bijectors = flowtorch.bijectors.AffineAutoregressive()#params_fn=params)\n",
    "#bijectors = flowtorch.bijectors.SplineAutoregressive(bound=10, count_bins=2)\n",
    "flow = flowtorch.distributions.Flow(base_dist, bijectors)\n",
    "opt = torch.optim.AdamW(flow.parameters(), lr=0.001)\n",
    "for idx in (pbar := tqdm(range(201))):\n",
    "    opt.zero_grad()\n",
    "\n",
    "    loss = 0.0\n",
    "    for i in range(10):\n",
    "        loss -= flow.log_prob(samps[torch.randint(N,(1,))].log())\n",
    "    loss = loss / 10\n",
    "\n",
    "    pbar.set_description(f\"epoch: {idx}, loss: {loss}\")\n",
    "\n",
    "    loss.backward()\n",
    "    opt.step()\n",
    "\n",
    "flow = dist.TransformedDistribution(\n",
    "    flow,\n",
    "    [dist.transforms.ExpTransform()]\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 946,
   "id": "rDOv98QQiU-g",
   "metadata": {
    "colab": {
     "base_uri": "https://localhost:8080/",
     "height": 468
    },
    "id": "rDOv98QQiU-g",
    "outputId": "790cf30e-a979-43c6-f5d1-64de2e44bf1b",
    "tags": []
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/liam/.var/app/org.jupyter.JupyterLab/config/jupyterlab-desktop/jlab_server/lib/python3.8/site-packages/scipy/interpolate/_interpolate.py:701: RuntimeWarning: invalid value encountered in multiply\n",
      "  y_new = slope*(x_new - x_lo)[:, None] + y_lo\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<seaborn.axisgrid.PairGrid at 0x7fb347483c40>"
      ]
     },
     "execution_count": 946,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 300x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from seaborn_qqplot import pplot\n",
    "\n",
    "fig = pplot(data=pd.DataFrame({\n",
    "    'Target': samps,\n",
    "    'Flow': 3*flow.sample((1000000,))\n",
    "}), x='Target', y='Flow', kind='qq', height=2, aspect=3/2)#, display_kws={\"identity\": True})\n",
    "xx = np.linspace(0,10000,3)\n",
    "plt.plot(xx, xx, 'k--')\n",
    "fig.set(title=\"(ii) Gaussian\")\n",
    "plt.xlim([0,10000])\n",
    "plt.ylim([0,10000])\n",
    "fig.tight_layout()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 947,
   "id": "oxZLIdRflyPB",
   "metadata": {
    "id": "oxZLIdRflyPB",
    "tags": []
   },
   "outputs": [],
   "source": [
    "fig.savefig(\"gaussian_qq.png\", dpi=500)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 466,
   "id": "65f91bc5-a871-482a-b5b8-7c630392715b",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(array([  2.,   1.,   0.,   1.,   2.,   1.,   0.,   3.,   0.,   2.,   2.,\n",
       "          3.,   4.,   2.,   6.,  12.,   7.,   7.,   5.,  14.,  14.,  14.,\n",
       "         10.,  23.,  34.,  30.,  31.,  24.,  42.,  54.,  44.,  52.,  63.,\n",
       "         58.,  97.,  89.,  86., 109., 124., 135., 149., 151., 150., 152.,\n",
       "        211., 230., 229., 232., 249., 249., 276., 294., 298., 283., 294.,\n",
       "        274., 328., 308., 300., 322., 307., 303., 289., 303., 269., 246.,\n",
       "        227., 218., 178., 193., 162., 147., 144., 118., 112., 112.,  75.,\n",
       "         75.,  57.,  48.,  52.,  36.,  34.,  32.,  17.,  12.,  12.,   7.,\n",
       "          4.,   7.,   5.,   7.,   1.,   1.,   2.,   1.,   0.,   0.,   0.,\n",
       "          1.]),\n",
       " array([-2.90804482e+00, -2.82259727e+00, -2.73714972e+00, -2.65170193e+00,\n",
       "        -2.56625438e+00, -2.48080683e+00, -2.39535928e+00, -2.30991173e+00,\n",
       "        -2.22446394e+00, -2.13901639e+00, -2.05356884e+00, -1.96812129e+00,\n",
       "        -1.88267362e+00, -1.79722607e+00, -1.71177840e+00, -1.62633085e+00,\n",
       "        -1.54088330e+00, -1.45543563e+00, -1.36998808e+00, -1.28454041e+00,\n",
       "        -1.19909286e+00, -1.11364532e+00, -1.02819765e+00, -9.42750096e-01,\n",
       "        -8.57302487e-01, -7.71854877e-01, -6.86407268e-01, -6.00959659e-01,\n",
       "        -5.15512109e-01, -4.30064499e-01, -3.44616890e-01, -2.59169281e-01,\n",
       "        -1.73721701e-01, -8.82740989e-02, -2.82649999e-03,  8.26210976e-02,\n",
       "         1.68068692e-01,  2.53516287e-01,  3.38963896e-01,  4.24411476e-01,\n",
       "         5.09859085e-01,  5.95306695e-01,  6.80754304e-01,  7.66201854e-01,\n",
       "         8.51649463e-01,  9.37097073e-01,  1.02254462e+00,  1.10799229e+00,\n",
       "         1.19343984e+00,  1.27888751e+00,  1.36433506e+00,  1.44978261e+00,\n",
       "         1.53523028e+00,  1.62067783e+00,  1.70612550e+00,  1.79157305e+00,\n",
       "         1.87702060e+00,  1.96246827e+00,  2.04791594e+00,  2.13336349e+00,\n",
       "         2.21881104e+00,  2.30425858e+00,  2.38970613e+00,  2.47515392e+00,\n",
       "         2.56060147e+00,  2.64604902e+00,  2.73149657e+00,  2.81694412e+00,\n",
       "         2.90239191e+00,  2.98783946e+00,  3.07328701e+00,  3.15873456e+00,\n",
       "         3.24418211e+00,  3.32962990e+00,  3.41507745e+00,  3.50052500e+00,\n",
       "         3.58597255e+00,  3.67142010e+00,  3.75686789e+00,  3.84231544e+00,\n",
       "         3.92776299e+00,  4.01321077e+00,  4.09865808e+00,  4.18410587e+00,\n",
       "         4.26955318e+00,  4.35500097e+00,  4.44044876e+00,  4.52589607e+00,\n",
       "         4.61134386e+00,  4.69679117e+00,  4.78223896e+00,  4.86768675e+00,\n",
       "         4.95313406e+00,  5.03858185e+00,  5.12402916e+00,  5.20947695e+00,\n",
       "         5.29492474e+00,  5.38037205e+00,  5.46581984e+00,  5.55126715e+00,\n",
       "         5.63671494e+00]),\n",
       " <BarContainer object of 100 artists>)"
      ]
     },
     "execution_count": 466,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAicAAAGeCAYAAABPfaH9AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy88F64QAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAoL0lEQVR4nO3df2zT94H/8ZftxMnsOISCWDkaOinDoI1Lk4BaxFHGJbAqgtFrNdEv6663cnT8mHrpjuquQHYIEGitvmi3CESnUa4IujtRjSvh1Gb0qL4C3cFRWtQbKlkQ0ha1RbmjC5l/LHH84/tHauM4TrAdO37bfj6kCPLxx/bbftvOy++flkgkEhEAAIAhrPkuAAAAQDzCCQAAMArhBAAAGIVwAgAAjEI4AQAARiGcAAAAoxBOAACAUQgnAADAKIQTAABgFMIJAAAwSlm+C5CpP/zhjwqFwvkuhlGmT3eqv9+X72LgHqgn81FHhYF6Ml98HdlsVlVXfyml6xVsOAmFwgoGCSdRFsvIv6FQWOyWZC7qyXzUUWGgnsw3mTqiWwcAABiFcAIAAIxCOAEAAEYhnAAAAKMQTgAAgFEIJwAAwCiEEwAAYBTCCQAAMArhBAAAGIVwAgAAjEI4AQAARiGcAAAAoxBOAACAUQp2V2IAyIbozqmS2N0WMAThBEDJctU4VFlui/0+OBySd8CfxxIBkAgnAEqUxSJVltu0ZP85eYeCqqoo06UdLfJZ7n1dALlFOAFQ0rxDQXmHgvkuBoA4DIgFAABGIZwAAACjEE4AAIBRGHMCAGmyJAyaZQoykF2EEwBIQ+L0Y2lkCrLnDlOQgWwhnABAihKnH0uKTUH2WmhBAbKFcAIAaWL6MZBbDIgFAABGIZwAAACjEE4AAIBRCCcAAMAohBMAAGAUwgkAADAK4QQAABiFcAIAAIxCOAEAAEYhnAAAAKMQTgAAgFEIJwAQx2K5+2/0/wCmFhv/AYAke5lVgWBY993nkiTNmDHy7+BwSJ47/ntePz7IsDsxMDmEEwCQVG6zyF5m1ZL952I7DldVlOnSjhZ5LeMHjmioiYYZKfVAAyA5wgkAxPEOBWPhJBWJoSaVQANgYoQTAMiCdEMNgPERTgAgBxIH0yZrRUnlHKAUpT1b50c/+pEeffRRNTU1qbm5Wa+++mrssp6eHq1bt04PPfSQ1qxZoytXroy6bldXl1paWtTQ0KANGzaor69v8o8AAAwSPwZl5sy7P64ax6jzXDWOUZcnOwcoVWm3nPzVX/2Vdu7cqcrKSt26dUt//dd/rQcffFArV67Uli1b9NRTT+nEiRN65513tHXrVr377ruaNm2abt68qe3bt+vQoUNqamrSyy+/rG3btunEiRO5eFwAkBepDKy1WKTKclvag2+BUpF2y8lXv/pVVVZW3r0Bq1W/+93vdPnyZQ0ODmrjxo2y2+16/PHH9cADD+js2bOSpM7OTi1fvlxLly5VZWWl2tradPXqVfX29mbv0QCAIaJjUCYai5LKOUApymjMyYEDB3T8+HH98Y9/1Jw5c7R27VqdPXtWbrdbVuvdvLNgwQLduHFD0kiXT319feyympoazZ49Wz09PZo7d25GhWeBpLviF46Cuagnc6RTB1br3RaPbNznRLfDayM1vJfMN5k6yiicbNu2TX/7t3+rX//61zp37pyqq6vl8/nkcrlGnVddXS2PxyNJ8vv9SS/3+XyZFEHTpzszul6xi19rAeaingpD4sJsk5VKvfPaSA/Pl/kyqaOMZ+tYLBbV19frwoULOnjwoO6//355vd5R53g8HjmdIyHC4XBMeHm6+vt9CoXCmRW+CFksIy+Azz/30F9tMOopeya7Imu0LiaSOH5klqtC7724Iv07+0K03ie6b14bqeG9ZL7EOrLZrCk3LEx6KnEoFNLvfvc7Pfroozpy5IjC4XCsa+f69etav369JMntdqu7uzt2vYGBAd26dUtutzvj++YFOVYkwvNSCKinyXHVOFRZbov9nusVWaNjQpwVtnufPIFU6p3XRnp4vsyXSR2lNSDW4/HorbfektfrVTgc1gcffKB//ud/1tKlS/Xwww/Lbrfr6NGjCgQCOnPmjD755BOtWrVKkrR27VqdP39eFy9e1ODgoDo6OtTQ0JDxeBMApSl+psvCXb/Skv3nVFluY+wBUETSajmxWCz613/9V+3bt0/BYFBf/vKX9eyzz+q73/2uLBaLDh8+rPb2dnV0dKi2tlaHDh1STU2NJKmurk779u1Te3u7bt++rUWLFunAgQO5eEwASgAzXIDilVY4qaqq0rFjx8a9fP78+XrzzTfHvby1tVWtra3p3CUAjGoVSaeFhJ2CgcLE8vUAjJY4viTT67FTMFA4CCcAjJVsJdVUZswkXo/VV4HCQjgBYLz48SXpzJhhXApQmNJevh4AACCXaDkBgCnCkutAaggnAJBj0WXwWWodSA3hBAByLNvL4APFjjEnADBFogN0fQEG6QITIZwAAACjEE4AAIBRCCcAAMAoDIgFUBQy3X8HgHkIJwAKGtN0geJDOAFQ0BKn6Uqp7b8DwFyEEwBFIdP9dwCYhwGxAADAKIQTAABgFMIJAAAwCmNOABiH3XuB0kY4AWAUV41DleUMaAVKGeEEgDEsFqmy3Jaz3XtZqA0oDIQTAMaJTgvO1pRgFmoDCgvhBEDRY6E2oLAQTgCUDBZqAwoDU4kBAIBRCCcAAMAodOsAgEHiZxFFIvkrB5BPhBMAMECyGUWDwyF57vjzWCogPwgnAGCAxBlFVRVlurSjRV4LLSgoPYQTADBI/IwioFQxIBYAABiFcAIAAIxCOAEAAEYhnAAAAKMQTgAAgFEIJwAAwCiEEwAAYBTCCQAAMArhBAAAGIVwAgAAjEI4AQAARiGcAAAAoxBOAACAUdIKJ4FAQDt37lRzc7MaGxu1evVqdXZ2xi5vbm5WfX29GhsbY5fH6+rqUktLixoaGrRhwwb19fVl51EAAICiUZbOycFgULNmzdKxY8c0Z84cffjhh9q0aZNqa2vV2NgoSTp48KCWL18+5ro3b97U9u3bdejQITU1Nenll1/Wtm3bdOLEiew8EgAAUBTSajlxOBxqa2tTbW2trFarFi9erKamJl29evWe1+3s7NTy5cu1dOlSVVZWqq2tTVevXlVvb2/GhQcAAMUnrZaTRH6/X9euXdMzzzwTO/bSSy8pHA5r3rx5euGFF7Ro0SJJUk9Pj+rr62Pn1dTUaPbs2erp6dHcuXMzun+LZTKlLy7R54LnxGzU08R4XsbiOUmO95L5JlNHGYeTSCSi7du3q76+XsuWLZMkvfLKK1q4cKEk6dSpU3ruued05swZzZkzR36/Xy6Xa9RtVFdXy+fzZXT/06c7My16UZsxw3Xvk5B31BNSxWtlYjw/5sukjjIKJ5FIRLt27VJfX5+OHj0qyxexaPHixbFzvvOd7+jtt9/W+fPntX79ejkcDnm93lG34/F45HRmFjL6+30KhcIZXbcYWSwjL4DPP/coEsl3aTCeUq+nqmkOVZbbYr8PBUOqKLNNcA2U6mvlXkr9vVQIEuvIZrOm3LCQdjiJRCLavXu3Pv74Y73++utyOBwTFMyiyBevGrfbre7u7thlAwMDunXrltxud7pFiCtLxlctWpEIz0shKMV6slikynKbluw/J+9QULNcFXrvxRWx3yXFjuGuUnytpIPnx3yZ1FHa65zs2bNHH330kV577TVVVVXFjn/22We6cuWKAoGAAoGATp48qWvXrsW6fNauXavz58/r4sWLGhwcVEdHhxoaGjIebwKgMHmHgvIOBeULBEf9Hn8Md1kso3+AUpBWy8mnn36qX/ziF7Lb7VqxYkXs+KZNm7Ry5Urt3btXvb29Ki8vV11dnV599dVY+Kirq9O+ffvU3t6u27dva9GiRTpw4EBWHwwAFAt7mVWBYHhMf/3gcEieO/48lQqYGmmFkzlz5ug3v/nNuJefPn16wuu3traqtbU1nbsEgJJUbrPIXmYd1e1VVVGmSzta5LXQlYHiNqmpxACA3Ip2eQGlhL11AACAUQgnAADAKIQTAABgFMIJAAAwCuEEAAAYhXACAACMQjgBAABGIZwAAACjEE4AAIBRCCcAAMAohBMAAGAU9tYBkDMWS/L/A8BECCcAcsJV41BluS3fxQBQgAgnALLOYpEqy21asv9cbEfdWa4KvffiivwWDEBBIJwAyEhiN00kMvYc71AwFk6cFbSiAEgN4QRA2pJ12QwOh+Qd8EtifAmAySGcAEhLsi6b+xx2/fu2b2jmTFeeSwegGBBOAGQkscvGXmaNBRbGl+RWfMtUsu40oNARTgBkTTSwML4kN+xlVgWCYc2YcbeFanA4JM8dfx5LBWQf4QQACkS5zTKqhaqqokyXdrTIa6EFBcWFcAIABSa+Sw0oRixfDwAAjEI4AQAARiGcAAAAoxBOAACAUQgnAADAKIQTAABgFKYSA0CBS2UTRqCQEE4AoEAlWzFWYtVYFD7CCQAUqMQVYyWxaiyKAuEEAAocK8ai2DAgFgAAGIVwAgAAjEI4AQAARmHMCYCURKerJk5bBYBsI5wAuCdXjUOV5bZ8FwNAiSCcAJiQxSJVltti01VnuSr03osr8l0sAEWMMScAUhKdruoLMGUVQG4RTgAAgFEIJwAAwCiEEwAAYBTCCQAAMArhBAAAGCWtcBIIBLRz5041NzersbFRq1evVmdnZ+zynp4erVu3Tg899JDWrFmjK1eujLp+V1eXWlpa1NDQoA0bNqivry87jwIAABSNtMJJMBjUrFmzdOzYMX3wwQfavXu3du/eratXr2p4eFhbtmzRypUr9f777+u5557T1q1bNTAwIEm6efOmtm/frr179+rSpUt68MEHtW3btpw8KAAAULjSCicOh0NtbW2qra2V1WrV4sWL1dTUpKtXr+ry5csaHBzUxo0bZbfb9fjjj+uBBx7Q2bNnJUmdnZ1avny5li5dqsrKSrW1tenq1avq7e3NyQMDAACFaVIrxPr9fl27dk3PPPOMbty4IbfbLav1bt5ZsGCBbty4IWmky6e+vj52WU1NjWbPnq2enh7NnTs3o/tnj4+72PekMBRiPRVSWXFXsddbIb6XSs1k6ijjcBKJRLR9+3bV19dr2bJl+u///m+5XK5R51RXV8vj8UgaCTLJLvf5fBnd//TpzswKXuRmzHDd+yTkHfWEXCuV11ipPM5ClkkdZRROIpGIdu3apb6+Ph09elQWi0VOp1Ner3fUeR6PR07nSIhwOBwTXp6u/n6fQqFwRtctRhbLyAvg8889ikTyXRqMpxDrKVpmFJZCeo1lohDfS6UmsY5sNmvKDQtph5NIJKLdu3fr448/1uuvvy6HwyFJmjdvno4cOaJwOBzr2rl+/brWr18vSXK73eru7o7dzsDAgG7duiW3251uEeLKkvFVi1YkwvNSCKgn5FqpvMZK5XEWskzqKO11Tvbs2aOPPvpIr732mqqqqmLHH374Ydntdh09elSBQEBnzpzRJ598olWrVkmS1q5dq/Pnz+vixYsaHBxUR0eHGhoaMh5vAgAAilNaLSeffvqpfvGLX8hut2vFihWx45s2bdLmzZt1+PBhtbe3q6OjQ7W1tTp06JBqamokSXV1ddq3b5/a29t1+/ZtLVq0SAcOHMjmYwEAAEUgrXAyZ84c/eY3vxn38vnz5+vNN98c9/LW1la1tramc5cAAKDETGoqMQCgMCRO52ScBkxGOAGAIueqcaiy3Dbq2OBwSJ47/jyVCJgY4QQAipjFIlWW27Rk/zl5h4KSpKqKMl3a0SKvhRYUmIlwAgAlwDsUjIUTwHRpTyUGAADIJcIJAAAwCuEEAAAYhXACAACMQjgBAABGIZwAAACjEE4AAIBRCCcAAMAohBMAAGAUwgkAADAK4QQAABiFcAIAAIzCxn8AUIQsltH/AoWEcAIARcReZlUgGNaMGa58FwXIGOEEAIpIuc0ie5lVS/afk3coqFmuCr334op8FwtIC2NOAKAIeYeC8g4F5QsE810UIG2EEwAAYBTCCQAAMArhBAAAGIVwAgAAjEI4AQAARiGcAAAAoxBOAACAUQgnAADAKIQTAABgFMIJAAAwCuEEAAAYhXACAACMQjgBAABGIZwAAACjlOW7AACA/LBY7v4/EslfOYBEhBMAKDH2MqsCwbBmzHDFjg0Oh+S5489jqYC7CCcAUGLKbRbZy6xasv+cvENBVVWU6dKOFnkttKDADIQTAChR3qGgvEPBfBcDGIMBsQAAwCiEEwAAYBTCCQAAMArhBAAAGIVwAgAAjEI4AQAARkkrnJw4cUJPPvmkFi5cqB/+8IejLmtublZ9fb0aGxvV2Nio1atXj7q8q6tLLS0tamho0IYNG9TX1zf50gPICYtl9A8ATKW01jmZNWuWtm7dqv/8z/9Uf3//mMsPHjyo5cuXjzl+8+ZNbd++XYcOHVJTU5Nefvllbdu2TSdOnMi85AAykhg2EhfdctU4VFlum7oCAUCCtMLJN7/5TUnS9evXk4aT8XR2dmr58uVaunSpJKmtrU1/9md/pt7eXs2dOzedIozCN7q7os8Fz4nZ8l1PVdPGBo/B4ZB8fxhZttxikSrLbbGVQyVplqtC7724YqqLijwopM+PfL+XcG+TqaOsrhD70ksvKRwOa968eXrhhRe0aNEiSVJPT4/q6+tj59XU1Gj27Nnq6enJOJxMn+7MSpmLTfxeGTBXPuspPnjc57Dr37d9Y0x54lcOdVbQilIqCvHzoxDLXGoyqaOshZNXXnlFCxculCSdOnVKzz33nM6cOaM5c+bI7/fL5RpduOrqavl8vozvr7/fp1AoPKkyFxOLZeQF8PnnHvbGMFg+6in+28t997nGBI/4PVZoJSlthfT5wWee+RLryGazptywkLVwsnjx4tj/v/Od7+jtt9/W+fPntX79ejkcDnm93lHnezweOZ2Ta/3gBTlWJMLzUgimqp5SHT8SDSy0kpS2Qvz8KMQyl5pM6ihnG/9ZLBZFviiN2+1Wd3d37LKBgQHdunVLbrc7V3cPlLzE8SO0iuBe7jVYGpgqaU0lDgaDGhoaUjAYVDgc1tDQkIaHh/XZZ5/pypUrCgQCCgQCOnnypK5du6Zly5ZJktauXavz58/r4sWLGhwcVEdHhxoaGiY1GBZAaqKtIr4Au88iOXuZVYFgWDNmuDRz5t0fV40j30VDiUqr5eTw4cM6ePBg7Peuri498cQT2rhxo/bu3ave3l6Vl5errq5Or776aix81NXVad++fWpvb9ft27e1aNEiHThwILuPBACQkXKbZdTYI0mqqijTpR0t8lpoQcHUSyucPP/883r++eeTXnb69OkJr9va2qrW1tZ07g4AMIXiB0tHxXf1EFIwVXI25gQAULjiu3qiBodD8tzx57FUKBWEEwDAGIldPXTzYCoRTgAA40rW1QPkGrsSAwAAoxBOAACAUQgnAADAKIQTAABgFMIJAAAwCuEEAAAYhanEQBGJX80zcRM3ACgUhBOgSLhqHKost+W7GAAwaYQToAhYLFJluW3Uxm2zXBV678UV+S0YAGSAcAIUkfjVPJ0VtKIAKEwMiAUAAEYhnAAAAKMQTgAAgFEIJwAAwCiEEwAAYBTCCQAAMArhBAAAGIV1ToACFl2inqXqARQTwglQoFiuHkCxIpwABShxuXqWqgdQTBhzAhSw6HL1vkAw30UBgKwhnAAAAKMQTgAAgFEIJwAAwCgMiAUApCxx2nokkp9yoLgRTgAA92QvsyoQDGvGDNeo44PDIXnu+PNUKhQrwgkA4J7KbRbZy6yx6euSVFVRpks7WuS10IKC7CKcAABSFp2+DuQSA2IBAIBRCCcAAMAodOsAACYl2Qye+GOMR0G6CCcAgIyMN4NnKBhSRdndTSmZ0YN0EU4AABlJNoMnugll9BgzepAJwglQIOKbyROb0YF8ip/B46ywjTkGpItwAhQAV41DleW2e58IAEWAcAIYzmKRKsttSZvOAaAYEU6AApGs6RwAihHrnAAAAKMQTgAAgFEIJwAAwChphZMTJ07oySef1MKFC/XDH/5w1GU9PT1at26dHnroIa1Zs0ZXrlwZdXlXV5daWlrU0NCgDRs2qK+vb/KlBwAARSetcDJr1ixt3bpV69atG3V8eHhYW7Zs0cqVK/X+++/rueee09atWzUwMCBJunnzprZv3669e/fq0qVLevDBB7Vt27bsPQoAAFA00gon3/zmN7Vy5UpNnz591PHLly9rcHBQGzdulN1u1+OPP64HHnhAZ8+elSR1dnZq+fLlWrp0qSorK9XW1qarV6+qt7c3e48EAAAUhaxMJb5x44bcbres1rtZZ8GCBbpx44akkS6f+vr62GU1NTWaPXu2enp6NHfu3Izvl1Uy74o+FzwnZsuknqhTFINsv475zDPfZOooK+HE5/PJ5Rq98VN1dbU8Ho8kye/3J73c5/NlfJ/Tpzszvm4xS9yAC2ainlBqcvWa571kvkzqKCvhxOl0yuv1jjrm8XjkdI4ECIfDMeHlmejv9ykUCmd8/WJjsYy8AD7/3MPmWgbLpJ6i1wEKWbY/m/jMM19iHdls1pQbFrISTubNm6cjR44oHA7HunauX7+u9evXS5Lcbre6u7tj5w8MDOjWrVtyu92Tul9ekGNFIjwvhYb6QinI1WcTn3nmy6SO0hoQGwwGNTQ0pGAwqHA4rKGhIQ0PD+vhhx+W3W7X0aNHFQgEdObMGX3yySdatWqVJGnt2rU6f/68Ll68qMHBQXV0dKihoWFS402AQhUIhjVjhkszZ478uGoc+S4SABglrZaTw4cP6+DBg7Hfu7q69MQTT+jHP/6xDh8+rPb2dnV0dKi2tlaHDh1STU2NJKmurk779u1Te3u7bt++rUWLFunAgQNZfSBAIbBYJHuZNbaJX1VFmS7taJHXwrc/AIhKK5w8//zzev7555NeNn/+fL355pvjXre1tVWtra3plQ4oUvGb+AEARmP5egAAYBTCCQAAMArhBAAAGIVwAgAAjEI4AQAARsnKImwAJidx7wmmFQMoZYQTII/sZdbYomzxBodD8tzx56lUAJBfhBMgj8ptllGLskliYTYAJY9wAhgg2aJsbAkPoFQRTgDDjNfVAwClgnACGCaxq2eWq0Lvvbgi38UCgCnDVGLAUNGuHl+APXgAlBbCCQAAMArhBAAAGIVwAgAAjEI4AQAARiGcAAAAoxBOAACAUQgnAADAKIQTAABgFFaIBQDkXOIeUZHI6GNscol4hBMAQM6Mt1fUUDCkijJb7PfB4ZA8d/xTXTwYinACAMiZxL2iJMX2i4oeq6oo06UdLfJaaEHBCMIJACDnontFSZKzwjbmGBCPAbEAAMAohBMAAGAUwgkAADAK4QQAABiFcAIAAIxCOAEAAEYhnAAAAKMQTgAAgFEIJwAAwCisEAtkGZuZAcDkEE6ALHLVOFRZzmZmADAZhBMgSywWqbLcxmZmADBJhBMgy9jMDAAmhwGxAADAKLScADkWP0A2/v8AgOQIJ0CO2MusCgTDmjHDle+iAAUhMbwzVqt0EU6AHCm3WWQvs8YGyErSLFeF3ntxRX4LBhhmvCDPbLfSRTgBcix+gKyzwnaPs4HSkyzIM9uttBFOAABGYKYboggnwDjo/waA/MjqVOKXXnpJCxcuVGNjY+zns88+i13e09OjdevW6aGHHtKaNWt05cqVbN49kDWuGodmznSN+nHVOPJdLAAoCVlvOfne976nF198cczx4eFhbdmyRU899ZROnDihd955R1u3btW7776radOmZbsYQMYSV3qVxu//ZpowMLWi7zPeb8VtyhZhu3z5sgYHB7Vx40bZ7XY9/vjjeuCBB3T27NmpKgKQlmj/93j94ImtK0wZBnIr/j0Xfb9VTaNFsxhlveXk5MmTOnnypO6//34988wz+va3vy1JunHjhtxut6zWu3lowYIFunHjRsb3RXK+i28T2TPRcxj/PCe2rjBNGMi+id5z0RZNn5UxYSaazN+lrIaTv/zLv9Tf/d3fadq0abpy5Yr+5m/+Ri6XS4899ph8Pp9crtHfLKurq+XxeDK6r+nTndkoctHh23tuJT6/TBMGcmui99x458AsmdRPVsPJ17/+9dj/H3nkET399NPq6urSY489JqfTKa/XO+p8j8cjpzOzkNHf71MoFJ5UeYuJxTLyAvj8cw/fICYp+lwmE31+JzoHQPb8/vd333P33Tfx+xJmSfy7ZLNZU25YyOlUYqvVqsgXr5h58+bpyJEjCofDsa6d69eva/369RnfPi/GsSIRnpdc4vkFpkZ01djxAkk83pdmy6R+sjog9u2335bX61U4HNaVK1d04sQJrVq1SpL08MMPy2636+jRowoEAjpz5ow++eST2OUAAETFrxq7cNev1Px//1++i4QplNWWkzfeeEP/8A//oFAopD/5kz9RW1ubVq9eLUkqLy/X4cOH1d7ero6ODtXW1urQoUOqqanJZhEAAEUkOsaEMV2lJevhZCLz58/Xm2++mc27BKYUs6IAIPdYvh5IwXi7pgIAso9wAnxholaRxF1TWdMEAHKHcAJoZOXJyvJ792nT/w0AuUc4QclLXHmSVhEAyK8p21sHMF20VcQXGLuPDgBg6hBOAACAUQgnAADAKIQTAABgFAbEAgAKWvz0f/bYKQ6EEwBAQUq2OOLgcEieO/48lgrZQDhBweNbE1CaEhdHrKoo06UdLfJZR38W8LlQeAgnKGiJi6fxrQkoPdFlAMbbZoLPhcJDOEHBSlw8LfqtyWvhmxJQihJbUiQlbU3h88F8hBMUvOi3ponQ9QOUjvjPBMalFCbCCYpO4sZ9VdPo+gFK1XjjUmhhNRvhBEVjvP5mSXwwASUulRZWmINwgqKRrL85uokfH0wAUDgIJyg68UHEWWG7x9kAANOwfD0AADAK4QQAABiFcAIAAIzCmBOUpPjpxolTjwEA+UU4QUGZbKiYaLoxAMAMhBMUjMR9dDIx0XRjAIAZCCcoCIn76EiTCxVMNwYAcxFOUFAIFQBQ/JitAwAAjEI4AQAARqFbBwBQchJn+7ERqFkIJwCAkjHecgKDwyF57vjzVCokIpwAAEpGsuUEqirKdGlHi7wWWlBMQTjBlEjWhBp/LPH36DEAyIX4mX8wD+EEOZds8bShYEgVZbZxf5doZgWAUkU4QU5NtHha9Fji7xLNrABQyggnmFBi10umki2eFj2W+DsA5ANdzeYgnGBcid0xdLMAKEbJZvDQ1ZxfhBMkldgdk69ulug3l0x2IAaAVCTO4KGrOf8IJ5hQvrpaxluLAAByha5mcxBOYKTxvskAAIof4QRpSWW9kmxK/CYDAPmUyuddts4pZYQTxMS/WRJDyHjdLImDxhgwBqAYJfsMTPZ5l8pEAiYb3BvhBJKSL5QWL9mSz4mDxqIDxnzWu98EGMgKoBgkfgYmGyA73kSCxM9EEyYbmI5wUiImmq8/0UJpiSZar4RBrACKXeIg2WQtzql8JjLYdmKEE8Nkqx8y/naqpo1tFUnWjJgseKRjotYVACgmqXwZ4zMxc1MaTv7whz/oRz/6kc6fP6+qqipt3rxZTz/99FQW4Z6ytSJgJreTaj9kslUM49cDSRZGks3XjzY1ZrvrZbIhBwBMl07wyNZnYimtWDul4WTPnj0KhUK6cOGCent79eyzz6qurk5LliyZymKMK9m4i8HhkLwDdwNCKiOvx2upmOh2Uu2HTCxj4oDUaIpPnIIb/+ag+wUAsiOXwWOyf1cyvX8TQs+UhRO/36+uri699dZbqqqq0te+9jU98cQT+uUvf5lROLHZrFktXzQcbD7+gf4YCEmSXJVl+slTDZo5M25J4+GQ/ugbHHXdLzkrVZHwokn3dqIvjPlfdsk/HJSjfKRqysutYwZSRW97urNc//h/GkfdV/RY9Hbuc9glSYvmTpd/eOQNdJ/DLnuZdcztJJ4Tf73xbifdc3J525zDOZyT33Pyff+FeM60ynINJ/myGAiGZC+b3N8VKflyD4lBKPFv2NBwSH7v6NvJRPR+yspG/o6l83fbEolMTUb6+OOPtW7dOl27di127K233tLrr7+ut956ayqKAAAACkB2mx8m4Pf75XQ6Rx2rrq6Wz+ebqiIAAIACMGXhxOFwjAkiHo9nTGABAAClbcrCyVe+8hVJ0s2bN2PHuru7NW/evKkqAgAAKABT2nLy2GOP6ac//am8Xq+6u7t16tQpPfnkk1NVBAAAUACmbECsNLLOSXt7uy5cuCCn06ktW7YYt84JAADIrykNJwAAAPcyZd06AAAAqSCcAAAAoxBOAACAUQgnAADAKIQTAABgFMIJAAAwCuGkyBw5ckTf+ta31NTUpG984xv6yU9+olAolO9iQSPr/LS1tamxsVGPPvqo3njjjXwXCXECgYB27typ5uZmNTY2avXq1ers7Mx3sTCB/v5+PfLII1q3bl2+i4IkfvWrX2nNmjVqaGjQn//5n+vs2bMpX7csh+VCHoTDYe3fv18LFizQ//zP/2jLli1yOp36/ve/n++ilbw9e/YoFArpwoUL6u3t1bPPPqu6ujotWbIk30WDpGAwqFmzZunYsWOaM2eOPvzwQ23atEm1tbVqbGzMd/GQxMsvvyy3262hoaF8FwUJLl68qP379+vAgQNqampSf3+//H5/yten5aTIfP/739ef/umfqry8XHPmzNG3vvUtffjhh/kuVsnz+/3q6urSCy+8oKqqKn3ta1/TE088oV/+8pf5Lhq+4HA41NbWptraWlmtVi1evFhNTU26evVqvouGJP7rv/5Lvb29+ou/+It8FwVJdHR06Ac/+IEWL14sq9WqGTNmqLa2NuXrE06K3Pvvv8/migb47W9/K0n66le/Gju2YMEC3bhxI08lwr34/X5du3aN94+BAoGA9u7dq127dsliseS7OEgQCoX061//Wv39/Vq1apWWLVumv//7v9fAwEDKt0E4KSChUEjBYDDpT7JxJcePH1dPT482bNiQh9Iint/vl9PpHHWsurpaPp8vTyXCRCKRiLZv3676+notW7Ys38VBgp/97GdatmyZ5s+fn++iIInbt29reHhY77zzjo4fP663335bv//977V///6Ub4MxJwXke9/7ni5fvpz0spkzZ+o//uM/Yr+fPn1aP/vZz3Ts2DFNnz59qoqIcTgcjjFBxOPxjAksyL9IJKJdu3apr69PR48e5Zu5YX7729/q9OnTOn36dL6LgnF86UtfkiQ9/fTTuv/++yVJmzdv1g9+8IOUb4NwUkCOHz+e0nlnzpzRK6+8on/6p39SXV1djkuFVHzlK1+RJN28eTNWJ93d3XQZGCYSiWj37t36+OOP9frrr8vhcOS7SEjw4Ycfqq+vT83NzZJGungCgYAeeeQRnTt3TlVVVXkuIaqrqzV79uxJBXu6dYrMv/3bv2nfvn36+c9/Lrfbne/i4AsOh0OPPfaYfvrTn8rr9aq7u1unTp3Sk08+me+iIc6ePXv00Ucf6bXXXuOPnKFaW1v17rvvxlpP2tra5Ha7dfr0aVoiDfLtb39bb7zxhv73f/9XXq9XP//5z2OBMhWWSCQSyWH5MMWam5vV19cnu90eO7Zo0SIdOXIkj6WCNLLOSXt7uy5cuCCn06ktW7bo6aefznex8IVPP/1Uzc3NstvtKiu726i8adMmbd68OY8lw0ROnTqlf/mXf9HJkyfzXRTECQaD+vGPf6zOzk7ZbDatWLFCO3fuTDn0E04AAIBR6NYBAABGIZwAAACjEE4AAIBRCCcAAMAohBMAAGAUwgkAADAK4QQAABiFcAIAAIxCOAEAAEYhnAAAAKMQTgAAgFEIJwAAwCj/H+8BgToxxWvyAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.hist(make_samples(10000).log(),bins=100)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 467,
   "id": "7f620d8f-62cd-49bc-abf7-004b30547f56",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(array([  1.,   0.,   0.,   0.,   1.,   0.,   1.,   0.,   0.,   0.,   0.,\n",
       "          0.,   0.,   0.,   1.,   0.,   0.,   0.,   0.,   0.,   0.,   0.,\n",
       "          1.,   0.,   0.,   0.,   1.,   0.,   0.,   1.,   1.,   2.,   4.,\n",
       "          0.,   1.,   0.,   2.,   0.,   5.,   4.,   4.,   6.,   7.,   8.,\n",
       "          3.,   8.,  13.,  15.,  11.,  11.,  11.,  24.,  14.,  29.,  39.,\n",
       "         32.,  45.,  49.,  46.,  58.,  57.,  80.,  85.,  71.,  92., 120.,\n",
       "        124., 158., 153., 199., 197., 210., 262., 257., 306., 378., 370.,\n",
       "        381., 434., 456., 445., 466., 504., 498., 492., 466., 434., 387.,\n",
       "        319., 298., 278., 193., 139.,  96.,  70.,  30.,  21.,   8.,   4.,\n",
       "          3.]),\n",
       " array([0.82106376, 0.84984374, 0.87862378, 0.90740377, 0.93618375,\n",
       "        0.96496379, 0.99374378, 1.02252376, 1.05130374, 1.08008373,\n",
       "        1.10886383, 1.13764381, 1.1664238 , 1.19520378, 1.22398376,\n",
       "        1.25276375, 1.28154385, 1.31032383, 1.33910382, 1.3678838 ,\n",
       "        1.39666378, 1.42544377, 1.45422387, 1.48300385, 1.51178384,\n",
       "        1.54056382, 1.56934381, 1.59812379, 1.62690377, 1.65568388,\n",
       "        1.68446386, 1.71324384, 1.74202383, 1.77080381, 1.79958379,\n",
       "        1.8283639 , 1.85714388, 1.88592386, 1.91470385, 1.94348383,\n",
       "        1.97226381, 2.0010438 , 2.02982378, 2.05860376, 2.08738399,\n",
       "        2.11616397, 2.14494395, 2.17372394, 2.20250392, 2.2312839 ,\n",
       "        2.26006389, 2.28884387, 2.31762385, 2.34640384, 2.37518382,\n",
       "        2.4039638 , 2.43274379, 2.46152401, 2.49030399, 2.51908398,\n",
       "        2.54786396, 2.57664394, 2.60542393, 2.63420391, 2.66298389,\n",
       "        2.69176388, 2.72054386, 2.74932384, 2.77810383, 2.80688405,\n",
       "        2.83566403, 2.86444402, 2.893224  , 2.92200398, 2.95078397,\n",
       "        2.97956395, 3.00834394, 3.03712392, 3.0659039 , 3.09468389,\n",
       "        3.12346387, 3.15224385, 3.18102407, 3.20980406, 3.23858404,\n",
       "        3.26736403, 3.29614401, 3.32492399, 3.35370398, 3.38248396,\n",
       "        3.41126394, 3.44004393, 3.46882391, 3.49760389, 3.52638412,\n",
       "        3.5551641 , 3.58394408, 3.61272407, 3.64150405, 3.67028403,\n",
       "        3.69906402]),\n",
       " <BarContainer object of 100 artists>)"
      ]
     },
     "execution_count": 467,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.hist(flow.sample((10000,)).log(), bins=100)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "598e2d6f-f01f-46e7-aae6-52d24f3a170e",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "colab": {
   "provenance": []
  },
  "interpreter": {
   "hash": "83b589253c6ae2165fd99d3b5e434b8a0ff74c98e791d87ced25152a201010fd"
  },
  "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.8.16"
  },
  "widgets": {
   "application/vnd.jupyter.widget-state+json": {
    "0072a65d9c8643a187c17d656586cefb": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "HTMLModel",
     "state": {
      "_dom_classes": [],
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "HTMLModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/controls",
      "_view_module_version": "1.5.0",
      "_view_name": "HTMLView",
      "description": "",
      "description_tooltip": null,
      "layout": "IPY_MODEL_73d1bec1ef2c45ccb5dafff1ed2d44af",
      "placeholder": "​",
      "style": "IPY_MODEL_f94489e61e9341558dc8f5f81bfd4fba",
      "value": "epoch: 1000, loss: 0.8142212629318237: 100%"
     }
    },
    "0b3159afb4844ccd940f9ba4d8692e98": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "DescriptionStyleModel",
     "state": {
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "DescriptionStyleModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "StyleView",
      "description_width": ""
     }
    },
    "2c684b979a4f4148b9113fd69d582a9d": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "HTMLModel",
     "state": {
      "_dom_classes": [],
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "HTMLModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/controls",
      "_view_module_version": "1.5.0",
      "_view_name": "HTMLView",
      "description": "",
      "description_tooltip": null,
      "layout": "IPY_MODEL_ebe29fab349d41d28f3d3f07b67c6f45",
      "placeholder": "​",
      "style": "IPY_MODEL_0b3159afb4844ccd940f9ba4d8692e98",
      "value": " 1001/1001 [00:06&lt;00:00, 187.18it/s]"
     }
    },
    "334cbb56667a4b90b44275363c9dee17": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "DescriptionStyleModel",
     "state": {
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "DescriptionStyleModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "StyleView",
      "description_width": ""
     }
    },
    "48a3467b5983445a839eca742537e195": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "FloatProgressModel",
     "state": {
      "_dom_classes": [],
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "FloatProgressModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/controls",
      "_view_module_version": "1.5.0",
      "_view_name": "ProgressView",
      "bar_style": "success",
      "description": "",
      "description_tooltip": null,
      "layout": "IPY_MODEL_cdfd007ea89a495f9042f73fb3dbc187",
      "max": 1001,
      "min": 0,
      "orientation": "horizontal",
      "style": "IPY_MODEL_726e4ff203f64fd7b1e18975265938ae",
      "value": 1001
     }
    },
    "58fee13b7c76483e802b7a0e0c9ca21a": {
     "model_module": "@jupyter-widgets/base",
     "model_module_version": "1.2.0",
     "model_name": "LayoutModel",
     "state": {
      "_model_module": "@jupyter-widgets/base",
      "_model_module_version": "1.2.0",
      "_model_name": "LayoutModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "LayoutView",
      "align_content": null,
      "align_items": null,
      "align_self": null,
      "border": null,
      "bottom": null,
      "display": null,
      "flex": null,
      "flex_flow": null,
      "grid_area": null,
      "grid_auto_columns": null,
      "grid_auto_flow": null,
      "grid_auto_rows": null,
      "grid_column": null,
      "grid_gap": null,
      "grid_row": null,
      "grid_template_areas": null,
      "grid_template_columns": null,
      "grid_template_rows": null,
      "height": null,
      "justify_content": null,
      "justify_items": null,
      "left": null,
      "margin": null,
      "max_height": null,
      "max_width": null,
      "min_height": null,
      "min_width": null,
      "object_fit": null,
      "object_position": null,
      "order": null,
      "overflow": null,
      "overflow_x": null,
      "overflow_y": null,
      "padding": null,
      "right": null,
      "top": null,
      "visibility": null,
      "width": null
     }
    },
    "726e4ff203f64fd7b1e18975265938ae": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "ProgressStyleModel",
     "state": {
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "ProgressStyleModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "StyleView",
      "bar_color": null,
      "description_width": ""
     }
    },
    "73d1bec1ef2c45ccb5dafff1ed2d44af": {
     "model_module": "@jupyter-widgets/base",
     "model_module_version": "1.2.0",
     "model_name": "LayoutModel",
     "state": {
      "_model_module": "@jupyter-widgets/base",
      "_model_module_version": "1.2.0",
      "_model_name": "LayoutModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "LayoutView",
      "align_content": null,
      "align_items": null,
      "align_self": null,
      "border": null,
      "bottom": null,
      "display": null,
      "flex": null,
      "flex_flow": null,
      "grid_area": null,
      "grid_auto_columns": null,
      "grid_auto_flow": null,
      "grid_auto_rows": null,
      "grid_column": null,
      "grid_gap": null,
      "grid_row": null,
      "grid_template_areas": null,
      "grid_template_columns": null,
      "grid_template_rows": null,
      "height": null,
      "justify_content": null,
      "justify_items": null,
      "left": null,
      "margin": null,
      "max_height": null,
      "max_width": null,
      "min_height": null,
      "min_width": null,
      "object_fit": null,
      "object_position": null,
      "order": null,
      "overflow": null,
      "overflow_x": null,
      "overflow_y": null,
      "padding": null,
      "right": null,
      "top": null,
      "visibility": null,
      "width": null
     }
    },
    "7c79b4f490834d9381a951011da270b0": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "FloatProgressModel",
     "state": {
      "_dom_classes": [],
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "FloatProgressModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/controls",
      "_view_module_version": "1.5.0",
      "_view_name": "ProgressView",
      "bar_style": "success",
      "description": "",
      "description_tooltip": null,
      "layout": "IPY_MODEL_b3522228b1b94327800ec8d7f7eb1cef",
      "max": 1001,
      "min": 0,
      "orientation": "horizontal",
      "style": "IPY_MODEL_bd39f72b67004725828ce5c49ada154d",
      "value": 1001
     }
    },
    "b3522228b1b94327800ec8d7f7eb1cef": {
     "model_module": "@jupyter-widgets/base",
     "model_module_version": "1.2.0",
     "model_name": "LayoutModel",
     "state": {
      "_model_module": "@jupyter-widgets/base",
      "_model_module_version": "1.2.0",
      "_model_name": "LayoutModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "LayoutView",
      "align_content": null,
      "align_items": null,
      "align_self": null,
      "border": null,
      "bottom": null,
      "display": null,
      "flex": null,
      "flex_flow": null,
      "grid_area": null,
      "grid_auto_columns": null,
      "grid_auto_flow": null,
      "grid_auto_rows": null,
      "grid_column": null,
      "grid_gap": null,
      "grid_row": null,
      "grid_template_areas": null,
      "grid_template_columns": null,
      "grid_template_rows": null,
      "height": null,
      "justify_content": null,
      "justify_items": null,
      "left": null,
      "margin": null,
      "max_height": null,
      "max_width": null,
      "min_height": null,
      "min_width": null,
      "object_fit": null,
      "object_position": null,
      "order": null,
      "overflow": null,
      "overflow_x": null,
      "overflow_y": null,
      "padding": null,
      "right": null,
      "top": null,
      "visibility": null,
      "width": null
     }
    },
    "b98ebd27b6834f74ace9468761148b60": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "HTMLModel",
     "state": {
      "_dom_classes": [],
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "HTMLModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/controls",
      "_view_module_version": "1.5.0",
      "_view_name": "HTMLView",
      "description": "",
      "description_tooltip": null,
      "layout": "IPY_MODEL_58fee13b7c76483e802b7a0e0c9ca21a",
      "placeholder": "​",
      "style": "IPY_MODEL_df4e853a026b40c999efed50489938dc",
      "value": "epoch: 1000, loss: 4.794800758361816: 100%"
     }
    },
    "bd39f72b67004725828ce5c49ada154d": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "ProgressStyleModel",
     "state": {
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "ProgressStyleModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "StyleView",
      "bar_color": null,
      "description_width": ""
     }
    },
    "cdfd007ea89a495f9042f73fb3dbc187": {
     "model_module": "@jupyter-widgets/base",
     "model_module_version": "1.2.0",
     "model_name": "LayoutModel",
     "state": {
      "_model_module": "@jupyter-widgets/base",
      "_model_module_version": "1.2.0",
      "_model_name": "LayoutModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "LayoutView",
      "align_content": null,
      "align_items": null,
      "align_self": null,
      "border": null,
      "bottom": null,
      "display": null,
      "flex": null,
      "flex_flow": null,
      "grid_area": null,
      "grid_auto_columns": null,
      "grid_auto_flow": null,
      "grid_auto_rows": null,
      "grid_column": null,
      "grid_gap": null,
      "grid_row": null,
      "grid_template_areas": null,
      "grid_template_columns": null,
      "grid_template_rows": null,
      "height": null,
      "justify_content": null,
      "justify_items": null,
      "left": null,
      "margin": null,
      "max_height": null,
      "max_width": null,
      "min_height": null,
      "min_width": null,
      "object_fit": null,
      "object_position": null,
      "order": null,
      "overflow": null,
      "overflow_x": null,
      "overflow_y": null,
      "padding": null,
      "right": null,
      "top": null,
      "visibility": null,
      "width": null
     }
    },
    "d3afeead45d04b71ad89545b7df8f10d": {
     "model_module": "@jupyter-widgets/base",
     "model_module_version": "1.2.0",
     "model_name": "LayoutModel",
     "state": {
      "_model_module": "@jupyter-widgets/base",
      "_model_module_version": "1.2.0",
      "_model_name": "LayoutModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "LayoutView",
      "align_content": null,
      "align_items": null,
      "align_self": null,
      "border": null,
      "bottom": null,
      "display": null,
      "flex": null,
      "flex_flow": null,
      "grid_area": null,
      "grid_auto_columns": null,
      "grid_auto_flow": null,
      "grid_auto_rows": null,
      "grid_column": null,
      "grid_gap": null,
      "grid_row": null,
      "grid_template_areas": null,
      "grid_template_columns": null,
      "grid_template_rows": null,
      "height": null,
      "justify_content": null,
      "justify_items": null,
      "left": null,
      "margin": null,
      "max_height": null,
      "max_width": null,
      "min_height": null,
      "min_width": null,
      "object_fit": null,
      "object_position": null,
      "order": null,
      "overflow": null,
      "overflow_x": null,
      "overflow_y": null,
      "padding": null,
      "right": null,
      "top": null,
      "visibility": null,
      "width": null
     }
    },
    "def2376828aa4a14a575702349e0c27b": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "HBoxModel",
     "state": {
      "_dom_classes": [],
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "HBoxModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/controls",
      "_view_module_version": "1.5.0",
      "_view_name": "HBoxView",
      "box_style": "",
      "children": [
       "IPY_MODEL_b98ebd27b6834f74ace9468761148b60",
       "IPY_MODEL_7c79b4f490834d9381a951011da270b0",
       "IPY_MODEL_ec9c533c50ca4a77ba6b1ff9e384fb93"
      ],
      "layout": "IPY_MODEL_ea08d940b991408cb58ea981e9803534"
     }
    },
    "df4e853a026b40c999efed50489938dc": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "DescriptionStyleModel",
     "state": {
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "DescriptionStyleModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "StyleView",
      "description_width": ""
     }
    },
    "ea08d940b991408cb58ea981e9803534": {
     "model_module": "@jupyter-widgets/base",
     "model_module_version": "1.2.0",
     "model_name": "LayoutModel",
     "state": {
      "_model_module": "@jupyter-widgets/base",
      "_model_module_version": "1.2.0",
      "_model_name": "LayoutModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "LayoutView",
      "align_content": null,
      "align_items": null,
      "align_self": null,
      "border": null,
      "bottom": null,
      "display": null,
      "flex": null,
      "flex_flow": null,
      "grid_area": null,
      "grid_auto_columns": null,
      "grid_auto_flow": null,
      "grid_auto_rows": null,
      "grid_column": null,
      "grid_gap": null,
      "grid_row": null,
      "grid_template_areas": null,
      "grid_template_columns": null,
      "grid_template_rows": null,
      "height": null,
      "justify_content": null,
      "justify_items": null,
      "left": null,
      "margin": null,
      "max_height": null,
      "max_width": null,
      "min_height": null,
      "min_width": null,
      "object_fit": null,
      "object_position": null,
      "order": null,
      "overflow": null,
      "overflow_x": null,
      "overflow_y": null,
      "padding": null,
      "right": null,
      "top": null,
      "visibility": null,
      "width": null
     }
    },
    "ebaf9556ed42453ca590126ec8f25dea": {
     "model_module": "@jupyter-widgets/base",
     "model_module_version": "1.2.0",
     "model_name": "LayoutModel",
     "state": {
      "_model_module": "@jupyter-widgets/base",
      "_model_module_version": "1.2.0",
      "_model_name": "LayoutModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "LayoutView",
      "align_content": null,
      "align_items": null,
      "align_self": null,
      "border": null,
      "bottom": null,
      "display": null,
      "flex": null,
      "flex_flow": null,
      "grid_area": null,
      "grid_auto_columns": null,
      "grid_auto_flow": null,
      "grid_auto_rows": null,
      "grid_column": null,
      "grid_gap": null,
      "grid_row": null,
      "grid_template_areas": null,
      "grid_template_columns": null,
      "grid_template_rows": null,
      "height": null,
      "justify_content": null,
      "justify_items": null,
      "left": null,
      "margin": null,
      "max_height": null,
      "max_width": null,
      "min_height": null,
      "min_width": null,
      "object_fit": null,
      "object_position": null,
      "order": null,
      "overflow": null,
      "overflow_x": null,
      "overflow_y": null,
      "padding": null,
      "right": null,
      "top": null,
      "visibility": null,
      "width": null
     }
    },
    "ebe29fab349d41d28f3d3f07b67c6f45": {
     "model_module": "@jupyter-widgets/base",
     "model_module_version": "1.2.0",
     "model_name": "LayoutModel",
     "state": {
      "_model_module": "@jupyter-widgets/base",
      "_model_module_version": "1.2.0",
      "_model_name": "LayoutModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "LayoutView",
      "align_content": null,
      "align_items": null,
      "align_self": null,
      "border": null,
      "bottom": null,
      "display": null,
      "flex": null,
      "flex_flow": null,
      "grid_area": null,
      "grid_auto_columns": null,
      "grid_auto_flow": null,
      "grid_auto_rows": null,
      "grid_column": null,
      "grid_gap": null,
      "grid_row": null,
      "grid_template_areas": null,
      "grid_template_columns": null,
      "grid_template_rows": null,
      "height": null,
      "justify_content": null,
      "justify_items": null,
      "left": null,
      "margin": null,
      "max_height": null,
      "max_width": null,
      "min_height": null,
      "min_width": null,
      "object_fit": null,
      "object_position": null,
      "order": null,
      "overflow": null,
      "overflow_x": null,
      "overflow_y": null,
      "padding": null,
      "right": null,
      "top": null,
      "visibility": null,
      "width": null
     }
    },
    "ec9c533c50ca4a77ba6b1ff9e384fb93": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "HTMLModel",
     "state": {
      "_dom_classes": [],
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "HTMLModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/controls",
      "_view_module_version": "1.5.0",
      "_view_name": "HTMLView",
      "description": "",
      "description_tooltip": null,
      "layout": "IPY_MODEL_d3afeead45d04b71ad89545b7df8f10d",
      "placeholder": "​",
      "style": "IPY_MODEL_334cbb56667a4b90b44275363c9dee17",
      "value": " 1001/1001 [00:11&lt;00:00, 142.66it/s]"
     }
    },
    "f81ce42cf96f44bb825475f791703948": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "HBoxModel",
     "state": {
      "_dom_classes": [],
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "HBoxModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/controls",
      "_view_module_version": "1.5.0",
      "_view_name": "HBoxView",
      "box_style": "",
      "children": [
       "IPY_MODEL_0072a65d9c8643a187c17d656586cefb",
       "IPY_MODEL_48a3467b5983445a839eca742537e195",
       "IPY_MODEL_2c684b979a4f4148b9113fd69d582a9d"
      ],
      "layout": "IPY_MODEL_ebaf9556ed42453ca590126ec8f25dea"
     }
    },
    "f94489e61e9341558dc8f5f81bfd4fba": {
     "model_module": "@jupyter-widgets/controls",
     "model_module_version": "1.5.0",
     "model_name": "DescriptionStyleModel",
     "state": {
      "_model_module": "@jupyter-widgets/controls",
      "_model_module_version": "1.5.0",
      "_model_name": "DescriptionStyleModel",
      "_view_count": null,
      "_view_module": "@jupyter-widgets/base",
      "_view_module_version": "1.2.0",
      "_view_name": "StyleView",
      "description_width": ""
     }
    }
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
