{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "99d811ee-0b10-45a2-b0b4-9eb77d03cc12",
   "metadata": {},
   "source": [
    "# Compare thinning\n",
    "\n",
    "Compare thinning algorithms of HoTPP and EasyTPP"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "5d15cc90-4432-414b-be82-2b68989faad8",
   "metadata": {},
   "outputs": [],
   "source": [
    "import time\n",
    "import math\n",
    "import numpy as np\n",
    "import torch\n",
    "import hotpp\n",
    "from hotpp.data import PaddedBatch\n",
    "from matplotlib import pyplot as plt\n",
    "from sklearn.neighbors import KernelDensity\n",
    "\n",
    "DEVICE = \"cuda\" if torch.cuda.is_available() else \"cpu\""
   ]
  },
  {
   "cell_type": "markdown",
   "id": "17f21a8e-d879-44c1-ac33-fce6c7d5d3d0",
   "metadata": {},
   "source": [
    "# Create synthetic dataset\n",
    "\n",
    "Synthetic dataset with a predefined distribution of time delta."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "a6c94e50-838b-4ccd-8c09-9a9f4711ae9a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "class DTDistribution:\n",
    "    def __init__(self, label=0):\n",
    "        if label == 0:\n",
    "            self.means = torch.tensor([1, 3]).float()\n",
    "            self.stds = torch.tensor([0.2, 1])\n",
    "            self.probs = torch.tensor([0.7, 0.3])\n",
    "        elif label == 1:\n",
    "            self.means = torch.tensor([2]).float()\n",
    "            self.stds = torch.tensor([0.5])\n",
    "            self.probs = torch.tensor([1.0])\n",
    "        else:\n",
    "            raise NotImplementedError(label)\n",
    "\n",
    "    @property\n",
    "    def entropy(self):\n",
    "        n = 10000\n",
    "        s = self.sample(10000)\n",
    "        return -self.log_pdf(s).mean()\n",
    "\n",
    "    @property\n",
    "    def expectation(self):\n",
    "        return (self.means * self.probs).sum()\n",
    "\n",
    "    def log_pdf(self, dt):\n",
    "        nc = len(self.means)\n",
    "        shape = [nc] + [1] * dt.ndim\n",
    "        means = self.means.reshape(*shape)\n",
    "        stds = self.stds.reshape(*shape)\n",
    "        probs = self.probs.reshape(*shape)\n",
    "        log_pdfs = -0.5 * (2 * math.pi * stds ** 2).log() - (means - dt) ** 2 / 2 / stds ** 2\n",
    "        return torch.logsumexp(log_pdfs + probs.log(), 0)\n",
    "\n",
    "    def sample(self, size):\n",
    "        components = torch.multinomial(self.probs, size, replacement=True)  # S.\n",
    "        sample = (torch.randn(size, 1) * self.stds + self.means).take_along_dim(components[:, None], 1).squeeze(1)\n",
    "        return sample.clip(min=0)\n",
    "\n",
    "    def plot(self, sample_size=20, ax=None):\n",
    "        if ax is None:\n",
    "            ax = plt.gca()\n",
    "        x_min = 0\n",
    "        x_max = self.means.max().item() + self.stds.max().item() * 2\n",
    "        ax.set_xlim(x_min, x_max)\n",
    "        if sample_size:\n",
    "            sample = self.sample(sample_size)\n",
    "            for x in sample:\n",
    "                ax.axvline(x, 0, 1, linewidth=0.5)\n",
    "\n",
    "        xs = torch.linspace(x_min, x_max, 1000)\n",
    "        pdfs = self.log_pdf(xs).exp()\n",
    "        # Fix filling.\n",
    "        xs[-1] = xs[-2]\n",
    "        pdfs[-1] = 0\n",
    "        ax.fill(xs, pdfs, alpha=0.5)\n",
    "\n",
    "DTDistribution().plot()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "abde24c9-2c03-4a25-b2d9-e9758e0c65ff",
   "metadata": {},
   "source": [
    "# Fit the model to revert intensity function of the distribution"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "0b2e9b7d-db89-47ec-86e7-bdc5b836acfe",
   "metadata": {},
   "source": [
    "**Create batch**\n",
    "\n",
    "Each sequences is composed of two timestamps: 0 and sampled. All labels are identical."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "3bb36262-6d3b-4d3e-87ad-143727d5ac24",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([10000, 2])\n"
     ]
    }
   ],
   "source": [
    "sample_size = 10000\n",
    "num_labels = 1\n",
    "\n",
    "time_deltas = torch.cat([DTDistribution(label).sample(sample_size).reshape(sample_size)\n",
    "                         for label in range(num_labels)])  # (B).\n",
    "time_deltas = torch.stack([torch.zeros_like(time_deltas), time_deltas], 1)  # (B, 2).\n",
    "\n",
    "labels = torch.cat([torch.full([sample_size], label, dtype=torch.long)\n",
    "                    for label in range(num_labels)])  # (B).\n",
    "labels = labels[:, None].repeat(1, 2)  # (B, 2).\n",
    "\n",
    "batch = PaddedBatch(\n",
    "    {\n",
    "        \"timestamps\": time_deltas.cumsum(1),\n",
    "        \"labels\": labels\n",
    "    },\n",
    "    torch.full([len(time_deltas)], 2, dtype=torch.long)\n",
    ")\n",
    "print(batch.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "d31f3965-9ebe-4a96-8eda-ef5cf26335c9",
   "metadata": {},
   "outputs": [],
   "source": [
    "module = hotpp.modules.NextItemModule(\n",
    "    seq_encoder=hotpp.nn.RnnEncoder(\n",
    "        embeddings={\"labels\": {\"in\": num_labels, \"out\": 16}},\n",
    "        rnn_partial=lambda dim: hotpp.nn.ODEGRU(dim, 16, num_diff_layers=3),\n",
    "    ),\n",
    "    head_partial=lambda idim, odim: hotpp.nn.Head(idim, odim, hidden_dims=[16]),\n",
    "    loss=hotpp.losses.NHPLoss(num_labels, thinning_params={\"max_delta\": 5, \"max_steps\": 10000}),\n",
    "    optimizer_partial=lambda p: torch.optim.Adam(p, lr=0.001)\n",
    ").to(DEVICE).eval()\n",
    "\n",
    "batch = batch.to(DEVICE)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "cf01141a-998f-4ef7-b46f-d40bfa43b8fc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Mean entropy (minimum expected loss): 0.801\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "You are trying to `self.log()` but the `self.trainer` reference is not registered on the model yet. This is most likely because the model hasn't been passed to the `Trainer`\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Step 0, Loss 1.47, Grad 0.28\n",
      "Step 200, Loss 0.82, Grad 0.07\n",
      "Step 400, Loss 0.78, Grad 0.18\n",
      "Step 600, Loss 0.83, Grad 0.08\n",
      "Step 800, Loss 0.81, Grad 0.08\n"
     ]
    }
   ],
   "source": [
    "entropy = np.mean([DTDistribution(label).entropy.item() for label in range(num_labels)])\n",
    "print(f\"Mean entropy (minimum expected loss): {entropy:.3f}\")\n",
    "\n",
    "module.train()\n",
    "optimizer = module.configure_optimizers()\n",
    "for step in range(1000):\n",
    "    loss = module.training_step((batch, None), None)\n",
    "    optimizer.zero_grad()\n",
    "    loss.backward()\n",
    "    grad_norm = torch.nn.utils.clip_grad_norm_(module.parameters(), 1)\n",
    "    if step % 200 == 0:\n",
    "        print(f\"Step {step}, Loss {loss.item():.2f}, Grad {grad_norm.item():.2f}\")\n",
    "    optimizer.step()\n",
    "module.eval()\n",
    "pass"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e680c6c9-4ac4-4540-8a32-ddf8f49b8292",
   "metadata": {},
   "source": [
    "# Visualize results\n",
    "\n",
    "We can see, that intensity is correct."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "aacc01cb-a14f-4cd4-8db9-2f87431c48ef",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Label 0\n",
      "Expected time delta: 1.600000023841858\n",
      "Predicted time delta: 1.5521025657653809\n"
     ]
    }
   ],
   "source": [
    "label_states = []\n",
    "for label in range(num_labels):\n",
    "    i = label * sample_size\n",
    "    prefix = PaddedBatch({\n",
    "        \"timestamps\": batch.payload[\"timestamps\"][i:i + 1, :1],\n",
    "        \"labels\": batch.payload[\"labels\"][i:i + 1, :1]\n",
    "    }, torch.ones_like(batch.seq_lens[:1]))\n",
    "    assert prefix.payload[\"labels\"].item() == label\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        outputs = module(prefix)\n",
    "        label_states.append(module.encode(prefix)[1])\n",
    "    print(f\"Label {label}\")\n",
    "    print(\"Expected time delta:\", DTDistribution(label).expectation.item())\n",
    "    print(\"Predicted time delta:\", outputs.payload[\"timestamps\"].item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "efa2e149-8c2b-41c1-b1ff-0752809210f7",
   "metadata": {},
   "outputs": [],
   "source": [
    "class Density:\n",
    "    \"\"\"Simple intensity-to-density mapper.\"\"\"\n",
    "    def __init__(self, intensity, sample_size=100):\n",
    "        self.i = intensity\n",
    "        self.s = torch.linspace(0, 1, sample_size)\n",
    "\n",
    "    def __call__(self, deltas):\n",
    "        means = self.i(deltas[..., None] * self.s.to(deltas.device)).mean(-1)  # (...).\n",
    "        integrals = means * deltas\n",
    "        log_values = self.i(deltas).log() - integrals\n",
    "        values = log_values.exp()\n",
    "        return values\n",
    "\n",
    "class IntensityFN:\n",
    "    \"\"\"Intensity function wrapper.\"\"\"\n",
    "    def __init__(self, deltas, intensities):\n",
    "        self.deltas = deltas.cpu().numpy()\n",
    "        self.intensities = intensities.cpu().numpy()\n",
    "\n",
    "    def __call__(self, deltas):\n",
    "        values = np.interp(deltas.cpu().numpy(), self.deltas, self.intensities)\n",
    "        values = torch.tensor(values, device=deltas.device, dtype=deltas.dtype)\n",
    "        return values\n",
    "\n",
    "def get_density(deltas, intensities):\n",
    "    \"\"\"Convert intensity to density.\"\"\"\n",
    "    return Density(IntensityFN(deltas, intensities))(deltas)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "5f6ebac7-fa86-4cf7-9354-3c1376269c27",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAigAAAGzCAYAAAAFROyYAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjkuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/TGe4hAAAACXBIWXMAAA9hAAAPYQGoP6dpAABl+UlEQVR4nO3dd3iUVf7+8fdk0istDQhNmo0uiBWVIiiKrmXVFXAVd1f0q8u6ruxPUbeIrhV7h1UXu4ALgiAKLIIgVUBAQDoJoabXmef3x2FSqJlkkmfK/bquuWYyTJ75xCC5c87nnOOwLMtCRERExI+E2V2AiIiIyNEUUERERMTvKKCIiIiI31FAEREREb+jgCIiIiJ+RwFFRERE/I4CioiIiPgdBRQRERHxOwooIiIi4ncUUESkXmzbtg2Hw8HTTz/ts2vOmzcPh8PBvHnzfHZNEfFPCigiUmHSpEk4HA6WLVtmdyl1VlJSwl/+8heaN29OTEwMffr0Yc6cOXaXJSI1pIAiIkFp5MiRPPvss9xyyy1MmDABp9PJkCFDWLhwod2liUgNhNtdgIiIry1dupQPP/yQp556ivvvvx+A4cOHc9ZZZ/HAAw+waNEimysUkVPRCIqIeKW0tJRx48bRs2dPkpKSiIuL48ILL+Tbb7894ec899xztG7dmpiYGC6++GLWrl17zGs2bNjAddddR5MmTYiOjqZXr1588cUXtarx008/xel0cuedd1Y8Fx0dze23387ixYvZuXNnra4rIg1HIygi4pXc3FzeeustbrrpJkaNGkVeXh5vv/02gwYNYunSpXTr1q3a6999913y8vIYPXo0xcXFTJgwgUsvvZQ1a9aQmpoKwLp16zj//PNp0aIFDz74IHFxcXz88ccMGzaMzz77jGuuucarGleuXEnHjh1JTEys9nzv3r0BWLVqFRkZGbX/jyAi9U4BRUS80rhxY7Zt20ZkZGTFc6NGjaJz5868+OKLvP3229Vev3nzZjZt2kSLFi0AuPzyy+nTpw9PPvkkzz77LAD33nsvrVq14ocffiAqKgqAu+66iwsuuIC//OUvXgeUzMxM0tPTj3ne89yePXu8up6INDxN8YiIV5xOZ0U4cbvdHDx4kPLycnr16sWKFSuOef2wYcMqwgmYUYw+ffrw5ZdfAnDw4EG++eYbbrjhBvLy8ti/fz/79+/nwIEDDBo0iE2bNrF7926vaiwqKqoIOlVFR0dX/LmI+DcFFBHx2r///W+6dOlCdHQ0TZs2JTk5mRkzZpCTk3PMazt06HDMcx07dmTbtm2AGWGxLIuHH36Y5OTkardHHnkEgOzsbK/qi4mJoaSk5Jjni4uLK/5cRPybpnhExCvvv/8+I0eOZNiwYfz5z38mJSUFp9PJ+PHj2bJli9fXc7vdANx///0MGjTouK9p3769V9dMT08/7qhLZmYmAM2bN/eyShFpaAooIuKVTz/9lHbt2vH555/jcDgqnveMdhxt06ZNxzz3888/06ZNGwDatWsHQEREBP379/dJjd26dePbb78lNze3WqPskiVLKv5cRPybpnhExCtOpxMAy7IqnluyZAmLFy8+7uunTp1abTRj6dKlLFmyhMGDBwOQkpJCv379eP311ytGOKrat2+f1zVed911uFwu3njjjYrnSkpKmDhxIn369NEKHpEAoBEUETnGO++8w6xZs455/t577+XKK6/k888/55prruGKK65g69atvPbaa5xxxhnk5+cf8znt27fnggsu4A9/+AMlJSU8//zzNG3alAceeKDiNS+//DIXXHABZ599NqNGjaJdu3bs3buXxYsXs2vXLlavXu1V/X369OH6669n7NixZGdn0759e/7973+zbdu2Y1YZiYh/UkARkWO8+uqrx31+5MiRjBw5kqysLF5//XW++uorzjjjDN5//30++eST4x7iN3z4cMLCwnj++efJzs6md+/evPTSS9WWAZ9xxhksW7aMxx57jEmTJnHgwAFSUlLo3r0748aNq9XX8O677/Lwww/z3nvvcejQIbp06cL06dO56KKLanU9EWlYDqvqOK2IiIiIH1APioiIiPgdBRQRERHxOwooIiIi4ncUUERERMTvKKCIiIiI31FAEREREb8TEPuguN1u9uzZQ0JCQrWttUVERMR/WZZFXl4ezZs3JyzMuzGRgAgoe/bs0dbUIiIiAWrnzp20bNnSq88JiICSkJAAmC+w6sFfIiIi4r9yc3PJyMio+DnujYAIKJ5pncTERAUUERGRAFOb9gw1yYqIiIjfUUARERERv6OAIiIiIn4nIHpQRMRwuVyUlZXZXYbUgNPpJDw8XFsjiNSSAopIgMjPz2fXrl1YlmV3KVJDsbGxpKenExkZaXcpIgFHAUUkALhcLnbt2kVsbCzJycn6rdzPWZZFaWkp+/btY+vWrXTo0MHrTapEQp0CikgAKCsrw7IskpOTiYmJsbscqYGYmBgiIiLYvn07paWlREdH212SSEBRpBcJIBo5CSwaNRGpPf3fIyIiIn7Hq4Ayfvx4zjnnHBISEkhJSWHYsGFs3LjxpJ8zadIkHA5HtZuGOkVERORkvAoo8+fPZ/To0Xz//ffMmTOHsrIyBg4cSEFBwUk/LzExkczMzIrb9u3b61S0iIivbNu2DYfDwapVq+wuRUSq8KpJdtasWdU+njRpEikpKSxfvpyLLrrohJ/ncDhIS0urXYUiErBGjhzJ4cOHmTp1ao1e73A4mDJlCsOGDavXuqrKyMggMzOTZs2aATBv3jwuueQSDh06RKNGjRqsDhGprk49KDk5OQA0adLkpK/Lz8+ndevWZGRkcPXVV7Nu3bqTvr6kpITc3NxqNwkwlgXvvgszZthdichJOZ1O0tLSCA/XokYRf1LrgOJ2u7nvvvs4//zzOeuss074uk6dOvHOO+8wbdo03n//fdxuN+eddx67du064eeMHz+epKSkiltGRkZtyxS7vPYajBgBw4ZBVpbd1QQfy4KCAntutdworl+/fvzf//0fDzzwAE2aNCEtLY1HH3204s/btGkDwDXXXIPD4aj4GGDatGn06NGD6Oho2rVrx2OPPUZ5eXnFnzscDt566y2uueYaYmNj6dChA1988UXFnx86dIhbbrmlYpl2hw4dmDhxIlB9imfbtm1ccsklADRu3BiHw8HIkSN59913adq0KSUlJdW+pmHDhnHrrbfW6r+HiJyCVUu///3vrdatW1s7d+706vNKS0ut0047zXrooYdO+Jri4mIrJyen4rZz504LsHJycmpbrjSk//3PssLDLcv8KLOsJ56wu6KAV1RUZP30009WUVGReSI/v/K/b0Pf8vNrXPeIESOsq6++2rIsy7r44outxMRE69FHH7V+/vln69///rflcDis2bNnW5ZlWdnZ2RZgTZw40crMzLSys7Mty7KsBQsWWImJidakSZOsLVu2WLNnz7batGljPfrooxXvA1gtW7a0Jk+ebG3atMn6v//7Pys+Pt46cOCAZVmWNXr0aKtbt27WDz/8YG3dutWaM2eO9cUXX1iWZVlbt261AGvlypVWeXm59dlnn1mAtXHjRiszM9M6fPiwVVhYaCUlJVkff/xxxXvu3bvXCg8Pt7755puaf99EQkxOTk6tf37XagTl7rvvZvr06Xz77be0bNnSq8+NiIige/fubN68+YSviYqKIjExsdpNAsTu3XDddVBeDp7fgN96q9a/dUtw6dKlC4888ggdOnRg+PDh9OrVi7lz5wKQnJwMQKNGjUhLS6v4+LHHHuPBBx9kxIgRtGvXjgEDBvD3v/+d119/vdq1R44cyU033UT79u15/PHHyc/PZ+nSpQDs2LGD7t2706tXL9q0aUP//v0ZOnToMfU5nc6KKeuUlBTS0tJISkoiJiaGm2++uWLUBeD999+nVatW9OvXz+f/nUTEyyZZy7K45557mDJlCvPmzaNt27Zev6HL5WLNmjUMGTLE688VP1dSYsLJ3r1w9tnw9dfQvj1s3gzz5sGRoXPxgdhYyM+3771rqUuXLtU+Tk9PJzs7+6Sfs3r1ar777jv++c9/VjzncrkoLi6msLCQ2CP1VL12XFwciYmJFdf+wx/+wK9+9StWrFjBwIEDGTZsGOedd55XtY8aNYpzzjmH3bt306JFCyZNmsTIkSO1eZ5IPfEqoIwePZrJkyczbdo0EhISyDrSW+D5DQNg+PDhtGjRgvHjxwPwt7/9jXPPPZf27dtz+PBhnnrqKbZv384dd9zh4y9FbHfPPfD999CoEUyZAikpcPPN8PrrZhRFAcV3HA6Ii7O7Cq9FRERU+9jhcOB2u0/6Ofn5+Tz22GNce+21x/xZ1T2VTnbtwYMHs337dr788kvmzJnDZZddxujRo3n66adrXHv37t3p2rUr7777LgMHDmTdunXMUBO4SL3xKqC8+uqrAMcMaU6cOJGRI0cCZii16vbOhw4dYtSoUWRlZdG4cWN69uzJokWLOOOMM+pWufiXN96AN980Pzg/+ABOO808P2qUCSiffQYvvginWPEloS0iIgKXy1XtuR49erBx40bat29fp2snJyczYsQIRowYwYUXXsif//zn4wYUz8nDR9cBcMcdd/D888+ze/du+vfvrwZ+kXrk9RTPqcybN6/ax8899xzPPfecV0VJgMnONqMnAI8/DpdfXvlnPXpAt26wahW89x7ce68dFUqAaNOmDXPnzuX8888nKiqKxo0bM27cOK688kpatWrFddddR1hYGKtXr2bt2rX84x//qNF1x40bR8+ePTnzzDMpKSlh+vTp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zKy4LVy40Otr+PU+KH369OGcc87hpZdeAsDtdpORkcE999zDgw8+aHN1ocvhcDBlyhSGDRtmdykC7Nu3j5SUFObPn89FF11kdzkhr0mTJjz11FPcfvvtdpcSkvLz8+nRowevvPIK//jHP+jWrRvPP/+83WWFlEcffZSpU6eyatWqOl3Hb0dQSktLWb58Of379694LiwsjP79+7N48WIbKxPxLzk5OQC1Oi1UfMflcvHhhx9SUFBA37597S4nZI0ePZorrrii2s8OaXibNm2iefPmtGvXjltuuYUdO3Z4fQ2/Pc14//79uFwuUlNTqz2fmprKhg0bbKpKxL+43W7uu+8+zj//fM466yy7ywlJa9asoW/fvhQXFxMfH8+UKVM444wz7C4rJH344YesWLGCH374we5SQlqfPn2YNGkSnTp1IjMzk8cee4wLL7yQtWvXkpCQUOPr+G1AEZFTGz16NGvXrq3V/K74RqdOnVi1ahU5OTl8+umnjBgxgvnz5yukNLCdO3dy7733MmfOHKKjo+0uJ6QNHjy44nGXLl3o06cPrVu35uOPP/Zq6tNvA0qzZs1wOp3s3bu32vN79+4lLS3NpqpE/Mfdd9/N9OnTWbBgAS1btrS7nJAVGRlJ+/btAejZsyc//PADEyZM4PXXX7e5stCyfPlysrOz6dGjR8VzLpeLBQsW8NJLL1FSUoLT6bSxwtDVqFEjOnbsyObNm736PL/tQYmMjKRnz57MnTu34jm3283cuXM1vyshzbIs7r77bqZMmcI333xD27Zt7S5JqnC73ZSUlNhdRsi57LLLWLNmDatWraq49erVi1tuuYVVq1YpnNgoPz+fLVu2kJ6e7tXn+e0ICsCYMWMYMWIEvXr1onfv3jz//PMUFBRw22232V1ayMnPz6+Wfrdu3cqqVato0qQJrVq1srGy0DN69GgmT57MtGnTSEhIICsrC4CkpCRiYmJsri60jB07lsGDB9OqVSvy8vKYPHky8+bN46uvvrK7tJCTkJBwTB9WXFwcTZs2VX9WA7v//vsZOnQorVu3Zs+ePTzyyCM4nU5uuukmr67j1wHlxhtvZN++fYwbN46srCy6devGrFmzjmmclfq3bNkyLrnkkoqPx4wZA8CIESOYNGmSTVWFpldffRWAfv36VXt+4sSJjBw5suELCmHZ2dkMHz6czMxMkpKS6NKlC1999RUDBgywuzQR2+zatYubbrqJAwcOkJyczAUXXMD3339PcnKyV9fx631QREREJDT5bQ+KiIiIhC4FFBEREfE7CigiIiLidxRQRERExO8ooIiIiIjfUUARERERv6OAIiIiIn5HAUVERET8jgKKiIiI+B0FFBEREfE7CigiIiLid/4/jrBLJAleVzkAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "for label, state in enumerate(label_states):\n",
    "    inter_times = torch.linspace(0, 5, 100)\n",
    "    inter_times = PaddedBatch(inter_times[None, None], torch.tensor([1])).to(DEVICE)\n",
    "    with torch.no_grad():\n",
    "        inter = module._loss.interpolator(state, inter_times).payload  # (B, L, S, D).\n",
    "        intensity = module._loss.intensity(inter)[0, 0, :, label]\n",
    "        density = get_density(inter_times.payload[0, 0], intensity)\n",
    "\n",
    "    plt.title(f\"Label {label}\")\n",
    "    DTDistribution(label).plot(sample_size=None)\n",
    "    plt.axhline(0, color=\"k\", linewidth=0.5)\n",
    "    plt.plot(inter_times.payload[0, 0].cpu(), intensity.cpu(), label=\"Intensity\", c=\"red\")\n",
    "    plt.plot(inter_times.payload[0, 0].cpu(), density.cpu(), label=\"Density\", c=\"blue\")\n",
    "    plt.legend()\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "55bcb94d-cf82-4a8b-8fe8-d7312ab67b46",
   "metadata": {},
   "source": [
    "# Compare thinning"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "700efdc2-d533-4a95-92ba-fdd3d3b97ce0",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Ground truth KLD 0.05226421356201172\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "d = DTDistribution()\n",
    "\n",
    "def eval(sample):\n",
    "    assert sample.numel() == 10000\n",
    "    density = KernelDensity(bandwidth=\"scott\")\n",
    "    density.fit(sample[:, None])\n",
    "    true_sample = density.sample\n",
    "    s = d.sample(10000)\n",
    "    log_probs = -density.score_samples(s[:, None]).mean()\n",
    "    kld = log_probs - d.entropy\n",
    "    xs = torch.linspace(0, 5, 1000)\n",
    "    ys = torch.tensor(density.score_samples(xs[:, None])).exp()\n",
    "    plt.plot(xs, ys)\n",
    "    return kld.item()\n",
    "\n",
    "plt.title(\"Ground truth\")\n",
    "print(\"Ground truth KLD\", eval(d.sample(10000)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "0259b937-c575-4517-9ee3-73eeaced2b15",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Time 0.0036414098739624024\n",
      "HoTPP KLD 0.024885594844818115\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from hotpp.losses.tpp import thinning_sample\n",
    "\n",
    "def intensity_fn(dt):\n",
    "    dt = PaddedBatch(dt, torch.full([len(dt)], dt.shape[1], dtype=torch.long, device=dt.device))\n",
    "    inter = module._loss.interpolator(state, dt).payload  # (B, L, S, D).\n",
    "    intensity = module._loss.intensity(inter)[..., 0]\n",
    "    return intensity\n",
    "with torch.no_grad():\n",
    "    torch.cuda.synchronize()\n",
    "    start = time.time()\n",
    "    sample, mask = thinning_sample(100, 100, intensity_fn, max_steps=100, max_delta=5,\n",
    "                                   bound_samples=10, bound_factor=2, device=module.device)\n",
    "    print(\"Time\", (time.time() - start) / 100)\n",
    "    torch.cuda.synchronize()\n",
    "plt.title(\"HoTPP\")\n",
    "print(\"HoTPP KLD\", eval(sample.flatten().cpu()))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "c0368dbc-f566-424e-a6b3-a0f7f704c89d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Time 0.02116910457611084\n",
      "EasyTPP KLD 0.6821500658988953\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAiwAAAGzCAYAAAAMr0ziAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjkuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/TGe4hAAAACXBIWXMAAA9hAAAPYQGoP6dpAABQSUlEQVR4nO3deXhU5fnG8e+Zyb5CCCQEAmHfQyBABMGlRJbiLhYQRdNWKyKVX1wKtoJbBRUpVRAsLuDC4m5FBWkqIMqOkX0PsiYhLFnJNjO/PwKjqSyZkORMJvfnus5Vcjhz5pm0dm7f87zvazgcDgciIiIibsxidgEiIiIil6LAIiIiIm5PgUVERETcngKLiIiIuD0FFhEREXF7CiwiIiLi9hRYRERExO0psIiIiIjbU2ARERERt6fAIiIiIm5PgUVELmru3LkYhnHBY82aNTVeU0xMzEVrOnfMnTsXoNw5i8VCVFQUAwYMYPny5Re9b6NGjejXrx+ffPJJjX9GESnPy+wCRKR2ePrpp2nRosWvzrdu3brGa5k+fTp5eXnOn7/88ksWLFjAP/7xD8LDw53n+/Tp4/zzddddx6hRo3A4HKSlpfHqq6/ym9/8hi+++ILBgwc7r4uLi+Phhx8G4OjRo7z22mvceuutzJo1i/vvv78GPp2InI+hzQ9F5GLmzp1LUlIS69evp0ePHmaXc15Tp07l0UcfJS0tjZiYmF/9vWEYjBkzhhkzZjjPbdmyhdjYWAYMGMDSpUuBshGWzp07s3jxYud16enptG7dmiZNmrBr165q/ywicn56JCQiVWLq1Kn06dOHBg0a4O/vT3x8PB9++OGvrlu2bBl9+/alXr16BAUF0a5dOx5//HEA8vLyCAwM5KGHHvrV6w4fPozVamXy5MlVUm+XLl0IDw8nLS3totdFRkbSoUOHS14nItVLj4REpEKys7PJysoqd84wDBo0aADAP//5T2688UZGjhxJcXExCxcu5Pbbb2fx4sUMGTIEgG3btnH99dcTGxvL008/ja+vL3v37uW7774DICgoiFtuuYVFixYxbdo0rFar870WLFiAw+Fg5MiRVfJ5Tp06xalTpy75SKukpIRDhw45P6eImEOBRUQqJDEx8VfnfH19KSwsBGD37t34+/s7/+7BBx+ke/fuTJs2zRlYli1bRnFxMV999VW5XpNfGjVqFO+99x7Lli1j0KBBzvPvvvsuV111Fc2aNatU/YWFhWRlZTl7WB5//HFsNhu33357uetKSkqcwezo0aNMnjyZjIwMxo4dW6n3FZGqocAiIhUyc+ZM2rZtW+7cL0dAfhlWTp06hc1mo1+/fixYsMB5vl69egB89tlnJCUlYbH8+ql0YmIiUVFRvPfee87AsnXrVjZv3sycOXMqXf8bb7zBG2+84fzZz8+P5ORkxo0bV+66r7/+moYNG5b7jHfddRfPP/98pd9bRC6fAouIVEivXr0u2nS7ePFinn32WVJTUykqKnKeNwzD+edhw4bx+uuv88c//pHx48fTv39/br31VoYOHeoMLxaLhZEjRzJr1iwKCgoICAjgvffew8/P71ejIa646aabePDBBzEMg+DgYDp16kRgYOCvrktISODZZ5/FMAwCAgLo0KGDM2iJiHnUdCsil+3bb7/lxhtvxM/Pj1dffZUvv/ySZcuWcccdd/DLiYj+/v6sXLmS//znP9x1111s3ryZYcOGcd1112Gz2ZzXjRo1iry8PD799FMcDgfz58/n+uuvJzQ0tNI1Nm3alMTERPr370+vXr3OG1YAwsPDndf17t1bYUXETWiERUQu20cffYSfnx9Lly7F19fXef6tt9761bUWi4X+/fvTv39/pk2bxnPPPcdf//pXvvnmG2efTOfOnenWrRvvvfceTZs25eDBg7zyyis19nlExP0osIjIZbNarRiGUW6U5MCBA3z66aflrjt58iRhYWHlzsXFxQGUe4wEcNddd/HYY4/h6+tLgwYNyi3uJiJ1jwKLiFTIV199xc6dO391vk+fPgwZMoRp06YxaNAg7rjjDjIzM5k5cyatW7dm8+bNzmuffvppVq5cyZAhQ2jevDmZmZm8+uqrNG3alL59+5a77x133MFjjz3GJ598wujRo/H29q72zygi7kuBRUQqZOLEiec9/9Zbb3HPPffwxhtvMGXKFMaNG0eLFi14/vnnOXDgQLnAcuONN3LgwAHefPNNsrKyCA8P5+qrr+app576VX9KREQEAwYM4Msvv+Suu+6q1s8mIu5PS/OLiNu65ZZb2LJlC3v37jW7FBExmWYJiYhbOnbsGF988YVGV0QE0CMhEXEzaWlpfPfdd7z++ut4e3vzpz/9yeySRMQNaIRFRNzKihUruOuuu0hLS2PevHlERkaaXZKI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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from easy_tpp.model.torch_model.torch_thinning import EventSampler\n",
    "\n",
    "def intensity_fn2(time_seq, time_delta_seq, event_seq, dt, max_steps, compute_last_step_only):\n",
    "    dt = PaddedBatch(dt, torch.full([len(dt)], dt.shape[1], dtype=torch.long, device=dt.device))\n",
    "    inter = module._loss.interpolator(state, dt).payload  # (B, L, S, D).\n",
    "    intensity = module._loss.intensity(inter)[..., 0]\n",
    "    return intensity.unsqueeze(-1)\n",
    "\n",
    "sampler = EventSampler(num_sample=1, num_exp=100, num_samples_boundary=10, over_sample_rate=2, dtime_max=5, patience_counter=None, device=module.device)\n",
    "time_seq = torch.zeros(100, 100).to(module.device)\n",
    "time_delta_seq = time_seq\n",
    "event_seq = None\n",
    "dtime_boundary = time_seq + 5\n",
    "n_trials = 100\n",
    "with torch.no_grad():\n",
    "    torch.cuda.synchronize()\n",
    "    start = time.time()\n",
    "    for _ in range(n_trials):\n",
    "        accepted_dtimes, weights = sampler.draw_next_time_one_step(time_seq,\n",
    "                                                                   time_delta_seq,\n",
    "                                                                   event_seq,\n",
    "                                                                   dtime_boundary,\n",
    "                                                                   intensity_fn2)\n",
    "    print(\"Time\", (time.time() - start) / n_trials)\n",
    "    torch.cuda.synchronize()\n",
    "plt.title(\"EasyTPP\")\n",
    "print(\"EasyTPP KLD\", eval(accepted_dtimes.flatten().cpu()))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e9afe28a-61ef-4730-818c-1edf07fe6705",
   "metadata": {},
   "source": [
    "# Conclusion\n",
    "* The HoTPP thinning algorithm provides better sample than EasyTPP"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e3c2f93e-645d-45d8-b643-aa4105800632",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.12.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
