{
  "cells": [
    {
      "cell_type": "code",
      "execution_count": 2,
      "metadata": {
        "id": "3zRbozB_OEdf"
      },
      "outputs": [],
      "source": [
        "import numpy as np\n",
        "import pandas as pd\n",
        "import matplotlib as mpl\n",
        "import matplotlib.pyplot as plt\n",
        "import networkx as nx\n",
        "import scipy.stats\n",
        "from datetime import datetime, timedelta\n",
        "\n",
        "import os\n",
        "import arrow\n",
        "import itertools\n",
        "import random\n",
        "import pickle\n",
        "from abc import ABC, abstractmethod\n",
        "from functools import partial\n",
        "from tqdm import tqdm\n",
        "tqdm = partial(tqdm, position=0, leave=True)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 44,
      "metadata": {
        "id": "tXZmiUMvNV7_"
      },
      "outputs": [],
      "source": [
        "import torch\n",
        "import torch.nn.functional as F\n",
        "import torch.optim as optim\n",
        "from torch.utils.data import DataLoader\n",
        "from torch_geometric.nn import GCNConv, SAGEConv, ChebConv\n",
        "from torch_geometric.data import Data\n",
        "from torch_geometric.utils import get_laplacian, to_dense_adj, to_networkx, degree\n",
        "from torch_geometric.transforms import LaplacianLambdaMax\n",
        "\n",
        "rootpath = \".\""
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "LUGblW_pyq4b"
      },
      "source": [
        "# Utils"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 32,
      "metadata": {
        "id": "Dt5t8N3VysV6"
      },
      "outputs": [],
      "source": [
        "def lap2mat(edge_index, edge_weight):\n",
        "    n_node = edge_index.max() + 1\n",
        "    n_edge = edge_index.shape[1]\n",
        "\n",
        "    lap_mat = torch.zeros((n_node, n_node))\n",
        "    for i in range(n_edge):\n",
        "        lap_mat[edge_index[0, i], edge_index[1, i]] = edge_weight[i]\n",
        "    \n",
        "    return lap_mat\n",
        "\n",
        "def eval_points_generate(ts, ns):\n",
        "    \"\"\"\n",
        "    Generate points for intensity evaluation\n",
        "\n",
        "    Args:\n",
        "    - ts: numpy array, eval times\n",
        "    - ns: numpy array, eval nodes\n",
        "    \"\"\"\n",
        "    return np.array(list(itertools.product(ts, ns)))"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "lMPytFTAh50r"
      },
      "source": [
        "# Proposed Model"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "CVONI0d3iGit"
      },
      "source": [
        "## Kernel Basis"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 33,
      "metadata": {
        "id": "qZq4NrIuh6xA"
      },
      "outputs": [],
      "source": [
        "class DeepNetworkBasis(torch.nn.Module):\n",
        "    \"\"\"\n",
        "    Deep Neural Network Basis Kernel\n",
        "\n",
        "    This class directly models the kernel-induced feature mapping by a deep \n",
        "    neural network.\n",
        "    \"\"\"\n",
        "    def __init__(self, data_dim, basis_dim, \n",
        "                 init_gain=5e-1, init_bias=1e-3, nn_width=5):\n",
        "        \"\"\"\n",
        "        Args:\n",
        "        - data_dim:  dimension of input data point\n",
        "        - basis_dim: dimension of basis function\n",
        "        - nn_width:  the width of each layer in NN\n",
        "        \"\"\"\n",
        "        super(DeepNetworkBasis, self).__init__()\n",
        "        # configurations\n",
        "        self.data_dim  = data_dim\n",
        "        self.basis_dim = basis_dim\n",
        "        # init parameters for net\n",
        "        self.init_gain   = init_gain\n",
        "        self.init_bias   = init_bias\n",
        "        # network for basis function\n",
        "        self.net = torch.nn.Sequential(\n",
        "            # torch.nn.Linear(data_dim, nn_width),  # [ data_dim, n_hidden_nodes ]\n",
        "            # torch.nn.ReLU(), \n",
        "            torch.nn.Linear(data_dim, nn_width),  # [ data_dim, n_hidden_nodes ]\n",
        "            torch.nn.Softplus(beta=100), \n",
        "            # torch.nn.Linear(nn_width, nn_width),  # [ n_hidden_nodes, n_hidden_nodes ]\n",
        "            # torch.nn.ReLU(), \n",
        "            torch.nn.Linear(nn_width, nn_width),  # [ n_hidden_nodes, n_hidden_nodes ]\n",
        "            torch.nn.Softplus(beta=100),                  \n",
        "            # torch.nn.Linear(nn_width, nn_width),  # [ n_hidden_nodes, n_hidden_nodes ]\n",
        "            # torch.nn.Softplus(), \n",
        "            torch.nn.Linear(nn_width, basis_dim), # [ n_hidden_nodes, basis_dim ]\n",
        "            # torch.nn.Softplus(beta=1)\n",
        "            # torch.nn.Sigmoid()\n",
        "        )\n",
        "        self.net.apply(self.init_weights)\n",
        "\n",
        "    def init_weights(self, m):\n",
        "        \"\"\"\n",
        "        initialize weight matrices in network\n",
        "        \"\"\"\n",
        "        if type(m) == torch.nn.Linear:\n",
        "            torch.nn.init.xavier_uniform_(m.weight, gain=self.init_gain)\n",
        "            m.bias.data.fill_(self.init_bias)\n",
        "\n",
        "    def forward(self, x):\n",
        "        \"\"\"\n",
        "        customized forward function returning basis function evaluated at x\n",
        "        with size [ batch_size, data_dim ]\n",
        "        \"\"\"\n",
        "        return self.net(x)                                                      # [ batch_size, basis_dim ]\n",
        "\n",
        "\n",
        "class GraphLocalFilterBasis(torch.nn.Module):\n",
        "    \"\"\"\n",
        "    Basis on graph implemented by graph local filter\n",
        "    \"\"\"\n",
        "    def __init__(self, B, learnable=False):\n",
        "        \"\"\"\n",
        "        Args:\n",
        "        - B: Graph local filter, can be learnable or not.\n",
        "        \"\"\"\n",
        "        super(GraphLocalFilterBasis, self).__init__()\n",
        "        # configurations\n",
        "        self.B               = torch.nn.Parameter(B, requires_grad=learnable)\n",
        "        self.mask            = torch.nn.Parameter((B != 0.).float(), requires_grad=False)\n",
        "\n",
        "    def forward(self, x, y):\n",
        "        \"\"\"\n",
        "        Args:\n",
        "        - x:  [ batch_size, 1 ]\n",
        "        - y:  [ batch_size, 1 ]\n",
        "        \"\"\"\n",
        "        return (self.B * self.mask)[x, y]                                       # [ batch_size, 1 ]"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "SYGdRckYiacb"
      },
      "source": [
        "## Kernel"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 34,
      "metadata": {
        "id": "yVPwbhoSwcRg"
      },
      "outputs": [],
      "source": [
        "class TemporalDeepBasisL3netLocalFilterOnGraphKernel(torch.nn.Module):\n",
        "    \"\"\"\n",
        "    Temporal Basis Kernel with Chebnet local filter on graph.\n",
        "    \"\"\"\n",
        "    def __init__(self, device, tau_max, T, G,\n",
        "                 n_basis_time, loc_order_list, data_dim, basis_dim, \n",
        "                 init_gain=5e-1, init_bias=1e-1, init_std=1e1,\n",
        "                 nn_width_basis_time=5):\n",
        "        \"\"\"\n",
        "        Arg:\n",
        "        - G: torch_geometric object\n",
        "        \"\"\"\n",
        "        super(TemporalDeepBasisL3netLocalFilterOnGraphKernel, self).__init__()\n",
        "        # configurations\n",
        "        self.device          = device\n",
        "        self.n_basis_time    = n_basis_time\n",
        "        self.n_basis_loc     = len(loc_order_list)\n",
        "        self.data_dim        = data_dim\n",
        "        self.basis_dim       = basis_dim\n",
        "        self.tau_max         = tau_max\n",
        "        self.T               = T\n",
        "        self.n_node          = G.x.shape[0]\n",
        "        self.init_std        = init_std\n",
        "\n",
        "        self.xbasiss_time    = torch.nn.ModuleList([])\n",
        "        self.ybasiss_time    = torch.nn.ModuleList([])\n",
        "        self.Bbasiss_loc     = torch.nn.ModuleList([])\n",
        "        self.weights         = torch.nn.ParameterList([])\n",
        "        # self.weights_mark   = torch.nn.ParameterList([])\n",
        "\n",
        "        for i in range(n_basis_time):\n",
        "            self.xbasiss_time.append(DeepNetworkBasis(1, basis_dim=1, \n",
        "                                                 init_gain=init_gain, init_bias=init_bias,\n",
        "                                                 nn_width=nn_width_basis_time))\n",
        "            self.ybasiss_time.append(DeepNetworkBasis(1, basis_dim=1, \n",
        "                                                 init_gain=init_gain, init_bias=init_bias,\n",
        "                                                 nn_width=nn_width_basis_time))\n",
        "        \n",
        "        self.A_ = to_dense_adj(G.edge_index)[0]\n",
        "        filter_list = []\n",
        "        for order in loc_order_list:\n",
        "            if order == 0:\n",
        "                filter_list.append(self.init_local_filter(torch.eye(self.A_.shape[1]).float()))\n",
        "            else:\n",
        "                A_total = torch.zeros_like(self.A_)\n",
        "                for i in range(1, order + 1):\n",
        "                    A_total += self.A_.matrix_power(i)\n",
        "                filter_list.append(self.init_local_filter((A_total != 0).float()))\n",
        "        for i in range(self.n_basis_loc):\n",
        "            self.Bbasiss_loc.append(GraphLocalFilterBasis(filter_list[i], learnable=True))\n",
        "\n",
        "        self.weights.append(torch.nn.Parameter(torch.empty(self.n_basis_time, self.n_basis_loc).uniform_(-init_std,to=init_std), requires_grad=True))\n",
        "\n",
        "    \n",
        "    def init_local_filter(self, mask):\n",
        "        \"\"\"\n",
        "        Initialize local filter with normal distribution on k-hop neighborhood.\n",
        "        \"\"\"\n",
        "        in_size = self.n_basis_loc ** 2 * self.A_.sum(1).mean()\n",
        "        std_ = np.sqrt(1. / in_size)\n",
        "        return torch.randn((mask.shape[0], mask.shape[0])) * std_ * mask\n",
        "\n",
        "\n",
        "    def forward(self, x, y):\n",
        "        \"\"\"\n",
        "        customized forward function returning kernel evaluation at x and y with \n",
        "        size [ batch_size, data_dim ], where\n",
        "        - x: the first input with size  [ batch_size, data_dim ]\n",
        "        - y: the second input with size [ batch_size, data_dim ], history\n",
        "        \"\"\"\n",
        "\n",
        "        K_t = []\n",
        "        K_l = []\n",
        "        mask_t = (x[:, 0]-y[:, 0]).unsqueeze(-1) <= self.tau_max                 # [ batch_size ]\n",
        "\n",
        "        for xbasis_t, ybasis_t in zip(self.xbasiss_time, self.ybasiss_time):\n",
        "            xbasis_func_time = xbasis_t((x[:, 0]-y[:, 0]).unsqueeze(-1) / (self.T[1] - self.T[0])) * mask_t    # [ batch_size, basis_dim ]\n",
        "            ybasis_func_time = ybasis_t((y[:, 0].unsqueeze(-1) - self.T[0]) / (self.T[1] - self.T[0]))                       # [ batch_size, basis_dim ]\n",
        "            ki_t          = (xbasis_func_time * ybasis_func_time).sum(1)    # [ batch_size ]\n",
        "            K_t.append(ki_t)\n",
        "        K_t = torch.stack(K_t, 0)                                               # [ n_basis_time, batch_size ]\n",
        "\n",
        "        mask = mask_t.bool()\n",
        "        for Bbasis in self.Bbasiss_loc:\n",
        "            Bbasis_func = torch.zeros_like(xbasis_func_time)\n",
        "            Bbasis_func[mask] = Bbasis(y[:, [1]][mask].long(), x[:, [1]][mask].long())\n",
        "            K_l.append(Bbasis_func.squeeze(-1))                                 # [ batch_size ]\n",
        "        K_l = torch.stack(K_l, 0)                                               # [ n_basis_loc, batch_size ]\n",
        "\n",
        "        # weight_soft      = torch.nn.functional.softplus(self.weights[0], beta=100)  # [ n_basis_time, n_basis_loc ]\n",
        "        weight_soft      = self.weights[0]                                      # [ n_basis_time, n_basis_loc ]\n",
        "        K = (torch.permute(torch.einsum('il,jl->ijl', K_t, K_l), (-1, -2, -3)) * weight_soft.T).sum((-1, -2))\n",
        "\n",
        "        return K      # [ batch_size ]"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "sRPassrHHNdq"
      },
      "source": [
        "### Kernel with parametric temporal basis"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 35,
      "metadata": {
        "id": "WHJ9Y-esHVDE"
      },
      "outputs": [],
      "source": [
        "class TemporalParametricKernelL3netLocalFilterOnGraph(torch.nn.Module):\n",
        "    \"\"\"\n",
        "    Temporal Basis Kernel with Chebnet local filter on graph.\n",
        "    \"\"\"\n",
        "    def __init__(self, device, tau_max, T, G,\n",
        "                 time_basis_list, loc_filter_list,\n",
        "                 data_dim, basis_dim,\n",
        "                 basis_weight=None, init_std=1.):\n",
        "        \"\"\"\n",
        "        Arg:\n",
        "        - G: torch_geometric object\n",
        "        \"\"\"\n",
        "        super(TemporalParametricKernelL3netLocalFilterOnGraph, self).__init__()\n",
        "        # configurations\n",
        "        self.device          = device\n",
        "        self.n_basis_time    = len(time_basis_list)\n",
        "        self.n_basis_loc     = len(loc_filter_list)\n",
        "        self.data_dim        = data_dim\n",
        "        self.basis_dim       = basis_dim\n",
        "        self.tau_max         = tau_max\n",
        "        self.T               = T\n",
        "        self.n_node          = G.x.shape[0]\n",
        "\n",
        "        self.time_basiss     = torch.nn.ModuleList([])\n",
        "        self.Bbasiss_loc     = torch.nn.ModuleList([])\n",
        "        self.weights         = torch.nn.ParameterList([])\n",
        "\n",
        "        for time_basis in time_basis_list:\n",
        "            self.time_basiss.append(time_basis)\n",
        "        \n",
        "        for filter in loc_filter_list:\n",
        "            self.Bbasiss_loc.append(GraphLocalFilterBasis(filter))\n",
        "        \n",
        "        if basis_weight is not None:\n",
        "            self.weights.append(torch.nn.Parameter(torch.Tensor(basis_weight), requires_grad=False))\n",
        "        else: \n",
        "            self.weights.append(torch.nn.Parameter(torch.empty(self.n_basis_time, self.n_basis_loc).uniform_(-init_std,to=init_std), requires_grad=True))\n",
        "\n",
        "\n",
        "    def forward(self, x, y):\n",
        "        \"\"\"\n",
        "        customized forward function returning kernel evaluation at x and y with \n",
        "        size [ batch_size, data_dim ], where\n",
        "        - x: the first input with size  [ batch_size, data_dim ]\n",
        "        - y: the second input with size [ batch_size, data_dim ], history\n",
        "        \"\"\"\n",
        "\n",
        "        K_t = []\n",
        "        K_l = []\n",
        "        mask_t = (x[:, 0]-y[:, 0]) <= self.tau_max                              # [ batch_size ]\n",
        "\n",
        "        for basis in self.time_basiss:\n",
        "            K_t.append(basis(x[:, 0], y[:, 0]))\n",
        "        K_t = torch.stack(K_t, 0)                                               # [ n_basis_time, batch_size ]\n",
        "\n",
        "        mask = mask_t.bool()\n",
        "        for Bbasis in self.Bbasiss_loc:\n",
        "            Bbasis_func = torch.zeros_like(K_t[0])\n",
        "            Bbasis_func[mask] = Bbasis(y[:, 1][mask].long(), x[:, 1][mask].long())\n",
        "            K_l.append(Bbasis_func)                                             # [ batch_size ]\n",
        "        K_l = torch.stack(K_l, 0)                                               # [ n_basis_loc, batch_size ]\n",
        "\n",
        "        # weight_soft      = torch.nn.functional.softplus(self.weights[0], beta=100)  # [ n_basis_time, n_basis_loc ]\n",
        "        weight_soft      = self.weights[0]                                      # [ n_basis_time, n_basis_loc ]\n",
        "        K = (torch.permute(torch.einsum('il,jl->ijl', K_t, K_l), (-1, -2, -3)) * weight_soft.T).sum((-1, -2))\n",
        "\n",
        "        return K      # [ batch_size ]"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 36,
      "metadata": {
        "id": "fuCN7gnYLILb"
      },
      "outputs": [],
      "source": [
        "class ExpBasis(torch.nn.Module):\n",
        "    \"\"\"\n",
        "    Exponential Kernel function\n",
        "    \"\"\"\n",
        "    def __init__(self, beta=1.):\n",
        "        super(ExpBasis, self).__init__()\n",
        "        self.beta    = beta\n",
        "\n",
        "    def forward(self, t, his_t):\n",
        "        delta_t = t - his_t\n",
        "        return self.beta * torch.exp(- self.beta * delta_t)\n",
        "\n",
        "\n",
        "class BaseExponentialCosineBasis(torch.nn.Module):\n",
        "    \"\"\"\n",
        "    Exponential Cosine Kernel without touching 0 in cosine.\n",
        "    \"\"\"\n",
        "    def __init__(self, alpha, beta, freq):\n",
        "        \"\"\"\n",
        "        Arg:\n",
        "        - alpha: kernel magnitude\n",
        "        - beta: decaying rate\n",
        "        \"\"\"\n",
        "        super(BaseExponentialCosineBasis, self).__init__()\n",
        "        self._alpha = alpha\n",
        "        self._beta  = beta\n",
        "        self._freq  = freq\n",
        "    \n",
        "    def forward(self, t, his_t):\n",
        "        return self._alpha * torch.exp(- self._beta * torch.abs(t - his_t)) * \\\n",
        "                (torch.cos(self._freq * torch.clamp(his_t, min=0., max=None)) * 0.5 + 0.5)"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "gwzuikAxki13"
      },
      "source": [
        "## Point process"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 37,
      "metadata": {
        "id": "wAqx0sMTkkA0"
      },
      "outputs": [],
      "source": [
        "def count_parameters(model):\n",
        "    return sum(p.numel() for p in model.parameters() if p.requires_grad)\n",
        "\n",
        "def tensor_linspace(start, stop, num):\n",
        "    \"\"\"\n",
        "    Creates a tensor of shape [num, *start.shape] whose values are evenly spaced from start to end, inclusive.\n",
        "    Replicates but the multi-dimensional bahaviour of numpy.linspace in PyTorch.\n",
        "    \"\"\"\n",
        "    # create a tensor of 'num' steps from 0 to 1\n",
        "    steps = torch.arange(num, dtype=torch.float32, device=start.device) / (num - 1)\n",
        "    for i in range(start.ndim):\n",
        "        steps = steps.unsqueeze(-1)\n",
        "    \n",
        "    # the output starts at 'start' and increments until 'stop' in each dimension\n",
        "    out = start[None] + steps*(stop - start)[None]\n",
        "    \n",
        "    return out"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 38,
      "metadata": {
        "id": "5LMGap9nkr8c"
      },
      "outputs": [],
      "source": [
        "class BasePointProcess(torch.nn.Module):\n",
        "    \"\"\"\n",
        "    Point Process Base Class\n",
        "    \"\"\"\n",
        "    @abstractmethod\n",
        "    def __init__(self, T, S, data_dim, device, numerical_int=True, int_res=100, eval_res=50, eval_points=None):\n",
        "        \"\"\"\n",
        "        Args:\n",
        "        - T:             time horizon. e.g. (0, 1)\n",
        "        - S:             bounded space for marks. e.g. a two dimensional box region [(0, 1), (0, 1)]\n",
        "        - data_dim:      dimension of input data\n",
        "        - numerical_int: numerical integral flag, use simpson integration in 1D case.\n",
        "        - int_res:       numerical integral resolution\n",
        "        \"\"\"\n",
        "        super(BasePointProcess, self).__init__()\n",
        "        # configuration\n",
        "        self.data_dim      = data_dim\n",
        "        self.T             = T # time horizon. e.g. (0, 1)\n",
        "        self.S             = S # bounded space for marks. e.g. a two dimensional box region [(0, 1), (0, 1)]\n",
        "        self.numerical_int = numerical_int\n",
        "        self.int_res       = int_res\n",
        "        self.eval_res      = eval_res \n",
        "        self.device        = device\n",
        "\n",
        "        # numerical likelihood integral preparation\n",
        "        self.tt       = np.linspace(self.T[0], self.T[1], eval_res)  # [ eval_res ]\n",
        "        self.ss       = [ np.linspace(S_k[0], S_k[1], eval_res) for S_k in self.S ]     # [ data_dim - 1, eval_res ]\n",
        "        # spatio-temporal coordinates that need to be evaluated\n",
        "        self.t_coords = torch.ones((eval_res ** (len(S)), 1), device=self.device)                     # [ eval_res^(data_dim - 1), 1 ]\n",
        "        self.s_coords = torch.FloatTensor(np.array(list(itertools.product(*self.ss)))).to(self.device) # [ eval_res^(data_dim - 1), data_dim - 1 ]\n",
        "        # unit volumn\n",
        "        self.unit_vol = np.prod([ S_k[1] - S_k[0] for S_k in self.S ] + [ self.T[1] - self.T[0] ]) / (self.eval_res) ** (len(S)+1)\n",
        "\n",
        "        if eval_points is not None:\n",
        "            self.t_eval = torch.ones((eval_points.shape[0], 1), device=self.device)\n",
        "            self.s_eval = eval_points.to(self.device)\n",
        "\n",
        "    def eval(self, X):\n",
        "        \"\"\"\n",
        "        return conditional intensity evaluation at grid points, the numerical \n",
        "        integral can be further calculated by summing up these evaluations and \n",
        "        scaling by the unit volumn.\n",
        "\n",
        "        -- X: all the data points.\n",
        "        \"\"\"\n",
        "        batch_size, seq_len, _ = X.shape\n",
        "        n_eval_points = self.s_eval.shape[0]\n",
        "        integral = []\n",
        "        for t in self.tt:\n",
        "            # all possible points at time t (x_t) \n",
        "            t_coord = self.t_eval * t\n",
        "            xt      = torch.cat([t_coord, self.s_eval], 1) # [ n_eval_points, data_dim ] \n",
        "            xt      = xt\\\n",
        "                .unsqueeze_(0)\\\n",
        "                .repeat(batch_size, 1, 1)\\\n",
        "                .reshape(-1, self.data_dim)                  # [ batch_size * n_eval_points, data_dim ]\n",
        "            # history points before time t (H_t)\n",
        "            mask = ((X[:, :, 0].clone() < t) * (X[:, :, 0].clone() > 0))\\\n",
        "                .unsqueeze_(-1)\\\n",
        "                .repeat(1, 1, self.data_dim)                 # [ batch_size, seq_len, data_dim ]\n",
        "            ht   = X * mask                                  # [ batch_size, seq_len, data_dim ]\n",
        "            ht   = ht\\\n",
        "                .unsqueeze_(1)\\\n",
        "                .repeat(1, n_eval_points, 1, 1)\\\n",
        "                .reshape(-1, seq_len, self.data_dim)         # [ batch_size * n_eval_points, seq_len, data_dim ]\n",
        "            # lambda and integral \n",
        "            lams = self.cond_lambda(xt, ht).reshape(batch_size, -1)                     # [ batch_size, int_res^(data_dim - 1) ]\n",
        "            integral.append(lams)                            \n",
        "        # NOTE: second dimension is time, third dimension is mark space\n",
        "        integral = torch.stack(integral, 1)                  # [ batch_size, int_res, int_res^(data_dim - 1) ]\n",
        "        return integral\n",
        "    \n",
        "    def numerical_sample(self, X):\n",
        "        \"\"\"\n",
        "        return conditional intensity evaluation at grid points, the numerical \n",
        "        integral can be further calculated by summing up these evaluations and \n",
        "        scaling by the unit volumn.\n",
        "\n",
        "        -- X: all the data points.\n",
        "        \"\"\"\n",
        "        batch_size, seq_len, _ = X.shape\n",
        "        integral = []\n",
        "        for t in self.tt:\n",
        "            # all possible points at time t (x_t) \n",
        "            t_coord = self.t_coords * t\n",
        "            xt      = torch.cat([t_coord, self.s_coords], 1) # [ int_res^(data_dim - 1), data_dim ] \n",
        "            xt      = xt\\\n",
        "                .unsqueeze_(0)\\\n",
        "                .repeat(batch_size, 1, 1)\\\n",
        "                .reshape(-1, self.data_dim)                  # [ batch_size * int_res^(data_dim - 1), data_dim ]\n",
        "            # history points before time t (H_t)\n",
        "            mask = ((X[:, :, 0].clone() < t) * (X[:, :, 0].clone() > 0))\\\n",
        "                .unsqueeze_(-1)\\\n",
        "                .repeat(1, 1, self.data_dim)                 # [ batch_size, seq_len, data_dim ]\n",
        "            ht   = X * mask                                  # [ batch_size, seq_len, data_dim ]\n",
        "            ht   = ht\\\n",
        "                .unsqueeze_(1)\\\n",
        "                .repeat(1, self.eval_res ** (self.data_dim - 1), 1, 1)\\\n",
        "                .reshape(-1, seq_len, self.data_dim)         # [ batch_size * int_res^(data_dim - 1), seq_len, data_dim ]\n",
        "            # lambda and integral \n",
        "            lams = self.cond_lambda(xt, ht).reshape(batch_size, -1)                     # [ batch_size, int_res^(data_dim - 1) ]\n",
        "            integral.append(lams)                            \n",
        "        # NOTE: second dimension is time, third dimension is mark space\n",
        "        integral = torch.stack(integral, 1)                  # [ batch_size, int_res, int_res^(data_dim - 1) ]\n",
        "        return integral\n",
        "\n",
        "    def numerical_integral(self, X):\n",
        "        \"\"\"\n",
        "        return conditional intensity integral over time, using Simpson Quadrature\n",
        "        numerical integration in torchquad.\n",
        "        \n",
        "        Now only work in 1D case (data_dim == 1). Higher dimension case remain to implement.\n",
        "\n",
        "        -- X: all the data points.\n",
        "        \"\"\"\n",
        "\n",
        "        return self.numerical_sample(X).sum() * self.unit_vol\n",
        "    \n",
        "    def cond_lambda(self, xi, hti):\n",
        "        \"\"\"\n",
        "        return conditional intensity given x\n",
        "        Args:\n",
        "        - xi:   current i-th point       [ batch_size, data_dim ]\n",
        "        - hti:  history points before ti [ batch_size, seq_len, data_dim ]\n",
        "        Return:\n",
        "        - lami: i-th lambda              [ batch_size ]\n",
        "        \"\"\"\n",
        "        # if length of the history is zero\n",
        "        if hti.size()[0] == 0:\n",
        "            return self.mu(xi[:, 1]) * torch.ones(xi.shape[0], device = xi.device)\n",
        "        # otherwise treat zero in the time (the first) dimension as invalid points\n",
        "        batch_size, seq_len, _ = hti.shape\n",
        "        mask = hti[:, :, 0].clone() > 0                                          # [ batch_size, seq_len ]\n",
        "        xii  = xi.unsqueeze(1).repeat(1, seq_len, 1).reshape(-1, self.data_dim) # [ batch_size * seq_len, data_dim ]\n",
        "        hti  = hti.reshape(-1, self.data_dim)                                    # [ batch_size * seq_len, data_dim ]\n",
        "        K    = self.kernel(xii, hti).reshape(batch_size, seq_len)                # [ batch_size, seq_len ]\n",
        "        K    = K * mask                                                          # [ batch_size, seq_len ]\n",
        "        lami = K.sum(1) + self.mu(xi[:, 1])                                      # [ batch_size ]\n",
        "        return lami\n",
        "\n",
        "    def log_likelihood(self, X, n_sampled_fouriers=200):\n",
        "        \"\"\"\n",
        "        return log-likelihood given sequence X\n",
        "        Args:\n",
        "        - X:      input points sequence [ batch_size, seq_len, data_dim ]\n",
        "        Return:\n",
        "        - lams:   sequence of lambda    [ batch_size, seq_len ]\n",
        "        - loglik: log-likelihood        [ batch_size ]\n",
        "        \"\"\"\n",
        "        batch_size, seq_len, _ = X.shape\n",
        "        lams     = [\n",
        "            self.cond_lambda(X[:, i, :].clone(), X[:, :i, :].clone())\n",
        "            for i in range(seq_len) ]\n",
        "        lams     = torch.stack(lams, dim=1)                                   # [ batch_size, seq_len ]\n",
        "        # log-likelihood\n",
        "        mask     = X[:, :, 0] > 0                                             # [ batch_size, seq_len ]\n",
        "        # print((lams*mask).min())\n",
        "        sumlog   = torch.log(torch.clamp(lams, min=1e-5)) * mask                       # [ batch_size, seq_len ]\n",
        "        if self.numerical_int:\n",
        "            integral = self.numerical_integral(X)                             # scalar\n",
        "            loglik = sumlog.sum() - integral                                  # scalar\n",
        "        else: \n",
        "            # TODO: integral in analytical form\n",
        "            pass\n",
        "        return loglik\n",
        "\n",
        "    @abstractmethod\n",
        "    def mu(self):\n",
        "        \"\"\"\n",
        "        return base intensity\n",
        "        \"\"\"\n",
        "        raise NotImplementedError()\n",
        "\n",
        "    @abstractmethod\n",
        "    def forward(self, X):\n",
        "        \"\"\"\n",
        "        custom forward function returning conditional intensities and corresponding log-likelihood\n",
        "        \"\"\"\n",
        "        # return conditional intensities and corresponding log-likelihood\n",
        "        return self.log_likelihood(X)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 39,
      "metadata": {
        "id": "a-2JVzeRkxAp"
      },
      "outputs": [],
      "source": [
        "class TemporalPointProcessOnGraph(BasePointProcess):\n",
        "    \"\"\"\n",
        "    Point Process on graph with deep temporal basis and graph local filter.\n",
        "    \"\"\"\n",
        "    def __init__(self, device,\n",
        "                 T, G, mu, tau_max, kernel,\n",
        "                 data_dim=2, numerical_int=True,\n",
        "                 int_res=100, int_res_loc=200, pen_res_time=50, eval_res=100):\n",
        "        \"\"\"\n",
        "        Args:\n",
        "        - T:             time horizon. e.g. (0, 1)\n",
        "        - S:             bounded space for marks. e.g. a two dimensional box region [(0, 1), (0, 1)]\n",
        "        - n_basis:       number of basis functions\n",
        "        - n_mark_cate:   number of unique mark categories\n",
        "        - basis_dim:     dimension of basis function\n",
        "        - data_dim:      dimension of input data\n",
        "        - numerical_int: numerical integral flag\n",
        "        - int_res:       numerical integral resolution\n",
        "        - nn_width:      the width of each layer in kernel basis NN\n",
        "        \"\"\"\n",
        "        super(TemporalPointProcessOnGraph, self).__init__(T, (), data_dim, device, numerical_int, int_res, eval_res)\n",
        "        # configuration\n",
        "        self.n_node         = G.x.shape[0]\n",
        "        self._mu            = torch.nn.Parameter(torch.tensor(mu), requires_grad=False)\n",
        "        self.int_res_loc    = int_res_loc\n",
        "        self.pen_res_time   = pen_res_time\n",
        "        # self.tgrids         = torch.linspace(self.T[0], self.T[1], self.pen_res_time, device=self.device)    # [ pen_res ]\n",
        "        # mm                  = [ torch.linspace(s[0], s[1], self.pen_res) for s in self.S ]\n",
        "        # self.mgrids         = torch.tensor(list(itertools.product(*mm)), device=self.device)              # [ int_res^(data_dim - 1), data_dim - 1 ]\n",
        "        self.tgrids         = torch.linspace(self.T[0], self.T[1], self.pen_res_time, device=self.device)    # [ pen_res ]\n",
        "        self.lgrids         = torch.arange(G.x.shape[0], device=self.device).unsqueeze(-1)              # [ n_node, 1 ]\n",
        "        # deep nn basis kernel on graph\n",
        "        self.kernel         = kernel\n",
        "        self.n_basis_time   = kernel.n_basis_time\n",
        "        self.n_basis_loc    = kernel.n_basis_loc\n",
        "        self.data_dim       = kernel.data_dim\n",
        "        self.basis_dim      = kernel.basis_dim\n",
        "        \n",
        "        # self.eval_res decide the spatio-temporal grids that feature functions evaluated on, which are used for all the approximation later.\n",
        "        self.tt             = torch.linspace(0, self.kernel.tau_max, self.eval_res).to(self.device).reshape(-1, 1)\n",
        "        \n",
        "    \n",
        "    def mu(self, X):\n",
        "        \"\"\"\n",
        "        return base intensity\n",
        "        \"\"\"\n",
        "        return self._mu[X.long()]\n",
        "\n",
        "\n",
        "    def forward(self, X, n_sampled_fouriers=200):\n",
        "        \"\"\"\n",
        "        custom forward function returning conditional intensities and corresponding log-likelihood\n",
        "        \"\"\"\n",
        "        # return conditional intensities and corresponding log-likelihood\n",
        "        self.nn_evaluation(X)\n",
        "        return self.log_likelihood(X)\n",
        "\n",
        "\n",
        "    def nn_evaluation(self, X):\n",
        "        \"\"\"\n",
        "        implement nn evaluation required for SGD, including ybasis on events and\n",
        "        xbasis on uniform grid with \"self.eval_res\".\n",
        "\n",
        "        -- X: data points: [ bath_size, seq_len, data_dim ]\n",
        "        \"\"\"\n",
        "\n",
        "        batch_size, seq_len, _ = X.shape\n",
        "        ts   = X[:, :, 0].clone()                                               # [ batch_size, seq_len ]\n",
        "        ls   = X[:, :, 1].clone()                                               # [ batch_size, seq_len ]\n",
        "        mask_all = ts > 0\n",
        "\n",
        "        # mask out entries with zero kernel values\n",
        "        taus  = ts[:, None, :] - ts[:, :, None]                                 # [ batch_size, seq_len, seq_len ]\n",
        "        self.mask = ((taus > 0) * (taus <= self.kernel.tau_max) * \\\n",
        "                     mask_all[:, None, :]).bool()                               # [ batch_size, seq_len, seq_len ]\n",
        "        \n",
        "        # penalized grids on location\n",
        "        if self.lgrids.device != X.device:\n",
        "            lgrids = self.lgrids.to(X.device)\n",
        "        else:\n",
        "            lgrids = self.lgrids.clone()                                        # [ pen_res_loc, 1 ]\n",
        "\n",
        "        xbasis_time_grid  = []\n",
        "        ybasis_time_event = []\n",
        "        Bbasis_loc_filter = []\n",
        "        weights           = []\n",
        "\n",
        "        for xbasis_t, ybasis_t in zip(self.kernel.xbasiss_time, self.kernel.ybasiss_time):\n",
        "            ybasis_time_event.append(ybasis_t((ts.unsqueeze(-1) - self.T[0]) / (self.T[1] - self.T[0])).squeeze(-1) * mask_all)  # [ batch_size, seq_len ]\n",
        "            xbasis_time_grid.append(xbasis_t(self.tt / (self.T[1] - self.T[0])).squeeze(-1))  # [ eval_res ]\n",
        "\n",
        "        self.xbasis_time_grids  = torch.stack(xbasis_time_grid, 0)              # [ n_basis_time, eval_res ]\n",
        "        self.ybasis_time_events = torch.stack(ybasis_time_event, 0)             # [ n_basis_time, batch_size, seq_len ]\n",
        "        \n",
        "        for Bbasis in self.kernel.Bbasiss_loc:\n",
        "            Bbasis_loc_filter.append(Bbasis.B * Bbasis.mask)\n",
        "        self.Bbasis_loc_filters   = torch.stack(Bbasis_loc_filter, 0)           # [ n_basis_loc, n_node, n_node ]\n",
        "            \n",
        "        self.weights_soft       = self.kernel.weights[0].clone()+0              # [ n_basis_time, n_basis_loc ]\n",
        "        # if len(self.weights.shape) == 1:\n",
        "        #     self.weights = self.weights.unsqueeze(0)\n",
        "        self.h = self.kernel.tau_max / (self.eval_res - 1)\n",
        "\n",
        "    \n",
        "    def log_likelihood(self, X):\n",
        "        \"\"\"\n",
        "        return log-likelihood given sequence X\n",
        "\n",
        "        Args:\n",
        "        - X:      input points sequence [ batch_size, seq_len, data_dim ]\n",
        "        Return:\n",
        "        - loglik: log-likelihood        [ batch_size ]\n",
        "        \"\"\"\n",
        "\n",
        "        batch_size, seq_len, _ = X.shape\n",
        "        ts   = X[:, :, 0].clone()\n",
        "        ls   = X[:, :, 1].clone()\n",
        "        mask_all = ts > 0\n",
        "\n",
        "        taus  = ts[:, None, :] - ts[:, :, None]                                 # [ batch_size, seq_len, seq_len ]\n",
        "        ybasis_time_val = self.ybasis_time_events.clone()                       # [ n_basis_time, batch_size, seq_len ]\n",
        "        xbasis_time_val = torch.zeros_like(ybasis_time_val).unsqueeze(-2).repeat(1, 1, seq_len, 1)   \n",
        "                                                                                # [ n_basis_time, batch_size, seq_len, seq_len ]\n",
        "\n",
        "        ttaus = torch.clamp(taus[self.mask], max=self.kernel.tau_max-0.01)      # [ mask.sum() ]\n",
        "        int_start_idx = (torch.div(ttaus, self.h, rounding_mode=\"floor\")).long()  # [ mask.sum() ]\n",
        "        int_end_idx   = int_start_idx + 1\n",
        "        int_start     = self.xbasis_time_grids[:, int_start_idx]                # [ n_basis_time, mask.sum() ]\n",
        "        int_end       = self.xbasis_time_grids[:, int_end_idx]                  # [ n_basis_time, mask.sum() ]\n",
        "        int_prop      = (torch.remainder(ttaus, self.h) / self.h).unsqueeze(0).repeat(self.n_basis_time, 1)  # [ n_basis_time, mask.sum() ]\n",
        "\n",
        "        xbasis_time_val[:, self.mask] = torch.lerp(int_start, int_end, int_prop) # [ n_basis_time, mask.sum() ]\n",
        "        kernel_time = xbasis_time_val * ybasis_time_val[:, :, :, None]          # [ n_basis_time, batch_size, seq_len, seq_len ]\n",
        "\n",
        "        ls_pairs = torch.stack((ls[:, None, :].repeat(1, seq_len, 1),\n",
        "                               ls[:, :, None].repeat(1, 1, seq_len)), axis=-1)  # [ batch_size, seq_len, seq_len, 2 ]\n",
        "        kernel_loc  = []\n",
        "        for filter in self.Bbasis_loc_filters:\n",
        "            kernel_loc.append(filter[ls_pairs[..., 1].long(), ls_pairs[..., 0].long()])\n",
        "        kernel_loc  = torch.stack(kernel_loc, axis=0)                           # [ n_basis_loc, batch_size, seq_len, seq_len ]\n",
        "\n",
        "\n",
        "        kernel_val = torch.einsum('ibmn,jbmn->ijbmn', kernel_time, kernel_loc)\n",
        "        \n",
        "        lams = (kernel_val.T * self.weights_soft.T).sum((-1, -2)).T.sum(1) + self.mu(X[:, :, 1]) # [ batch_size, seq_len ]\n",
        "\n",
        "        integral, reg = self.numerical_integral(X)\n",
        "\n",
        "        ## sometime needs normalization\n",
        "        loglik = (torch.log(torch.clamp(lams, min=1e-5)) * mask_all).sum() - integral\n",
        "\n",
        "        return loglik, reg\n",
        "\n",
        "\n",
        "    def numerical_integral(self, X):\n",
        "        \"\"\"\n",
        "        return efficient computation of conditional intensity function integral\n",
        "        in log-likelihood of multiple sequences.\n",
        "\n",
        "        -- X: all data points.\n",
        "        \"\"\"\n",
        "\n",
        "        # if self.data_dim != 1:\n",
        "        #         raise NotImplementedError(\"Can only do quadrature integration on 1D data!\")\n",
        "\n",
        "        batch_size, seq_len, _ = X.shape\n",
        "        ts   = X[:, :, 0].clone()\n",
        "        ls   = X[:, :, 1].clone()\n",
        "        mask_all = ts > 0\n",
        "        n_event = mask_all.sum()\n",
        "        baserate = self._mu.sum() * (self.T[1] - self.T[0]) * batch_size\n",
        "\n",
        "        ## integration of time kernel\n",
        "        # the resolution of temporal integration is int_res\n",
        "\n",
        "        grids = torch.linspace(0.0,\n",
        "                               self.kernel.tau_max-0.01,\n",
        "                               self.int_res, device=X.device)                   # [ int_res ]\n",
        "        h   = self.kernel.tau_max / (self.int_res - 1)\n",
        "        hs  = torch.ones(self.int_res, device=X.device) * h / 3                 # [ int_res ]\n",
        "        hs[1] = h / 2\n",
        "\n",
        "        ybasis_time_val = self.ybasis_time_events.clone()                       # [ n_basis_time, batch_size, seq_len ]\n",
        "\n",
        "        int_start_idx = (torch.div(grids, self.h, rounding_mode=\"floor\")).long()  # [ int_res ]\n",
        "        int_end_idx   = int_start_idx + 1\n",
        "        int_start     = self.xbasis_time_grids[:, int_start_idx]                # [ n_basis_time, int_res ]\n",
        "        int_end       = self.xbasis_time_grids[:, int_end_idx]                  # [ n_basis_time, int_res ]\n",
        "        int_prop      = (torch.remainder(grids, self.h) / self.h).unsqueeze(0).repeat(self.n_basis_time, 1)  # [ n_basis_time, int_res ]\n",
        "\n",
        "        xbasis_time_val = torch.lerp(int_start, int_end, int_prop)              # [ n_basis_time, int_res ]\n",
        "\n",
        "        # compute integration from start to each grid\n",
        "        odd_idx  = np.arange(1, self.int_res-1, 2)\n",
        "        simp_g        = torch.zeros_like(xbasis_time_val)                       # [ n_basis_time, int_res ]\n",
        "        xbasis_time_val2   = xbasis_time_val.clone()\n",
        "        xbasis_time_val2[:, odd_idx] = xbasis_time_val2[:, odd_idx] * 3\n",
        "        simp_g[:, 1:] = xbasis_time_val[:, 1:] + xbasis_time_val2[:, :-1]       # [ n_basis_time, int_res ]\n",
        "        simp_v        = (torch.cumsum(simp_g, dim=1) * hs)                      # [ n_basis_time, int_res ]\n",
        "\n",
        "        int_t = torch.clamp((self.T[1] - ts), max=self.kernel.tau_max-1e-1)     # [ batch_size, seq_len ]\n",
        "        int_start_idx = (torch.div(int_t, h, rounding_mode=\"floor\") * mask_all).long()\n",
        "        int_end_idx   = int_start_idx + 1\n",
        "        int_start     = simp_v[:, int_start_idx]                                # [ n_basis_time, batch_size, seq_len ]\n",
        "        int_end       = simp_v[:, int_end_idx]                                  # [ n_basis_time, batch_size, seq_len ]\n",
        "        int_prop      = (torch.remainder(int_t, h) / h).unsqueeze(0).repeat(self.n_basis_time, 1, 1) \n",
        "                                                                                # [ n_basis_time, batch_size, seq_len ]\n",
        "\n",
        "        int_xbasis_time  = torch.lerp(int_start, int_end, int_prop)             # [ n_basis_time, batch_size, seq_len ]\n",
        "        integral_time = int_xbasis_time * ybasis_time_val * mask_all            # [ n_basis_time, batch_size, seq_len ]\n",
        "\n",
        "        ## integration of location kernel\n",
        "        integral_loc  = []\n",
        "        for filter in self.Bbasis_loc_filters:\n",
        "            integral_loc.append(filter[ls.long(), :].sum(-1))\n",
        "        integral_loc  = torch.stack(integral_loc, axis=0)                       # [ n_basis_loc, batch_size, seq_len ]\n",
        "\n",
        "\n",
        "        integral = torch.einsum('ibl,jbl->ijbl', integral_time, integral_loc).T * self.weights_soft.T\n",
        "        \n",
        "        # reg = None\n",
        "        w = ((grids / self.kernel.tau_max) * ((grids / self.kernel.tau_max) > 0.5)) ** 10\n",
        "        reg = (xbasis_time_val ** 2 * w).sum() * h\n",
        "\n",
        "        return integral.sum() + baserate, reg\n",
        "\n",
        "\n",
        "    def penalty_term(self, X):\n",
        "        \"\"\"\n",
        "        return the penalty of lambda function, guaranting lambda function to be nonnegative\n",
        "        using log barrier\n",
        "\n",
        "        -- X: all the data points.\n",
        "        \"\"\" \n",
        "\n",
        "        batch_size, seq_len, _ = X.shape\n",
        "        ts   = X[:, :, 0].clone()\n",
        "        ls   = X[:, :, 1].clone()\n",
        "        mask_all = ts > 0\n",
        "\n",
        "        if self.tgrids.device != X.device:\n",
        "            tgrids = self.tgrids.to(X.device)\n",
        "            lgrids = self.lgrids.to(X.device)\n",
        "        else:\n",
        "            tgrids = self.tgrids.clone()\n",
        "            lgrids = self.lgrids.clone()\n",
        "\n",
        "        lams = []\n",
        "\n",
        "        # compute time kernel\n",
        "        tts  = ts.unsqueeze(0).repeat(self.pen_res_time, 1, 1)                  # [ pen_res_time, batch_size, seq_len ]\n",
        "        taus = (tgrids - tts.T).T                                               # [ pen_res_time, batch_size, seq_len ]\n",
        "        mask = ((taus > 0) * (taus <= self.kernel.tau_max)).bool()              # [ pen_res_time, batch_size, seq_len ]\n",
        "\n",
        "        ybasis_time_val = self.ybasis_time_events.clone()                       # [ n_basis_time, batch_size, seq_len ]\n",
        "        xbasis_time_val = torch.zeros_like(ybasis_time_val).unsqueeze(1).repeat(1, self.pen_res_time, 1, 1)  # [ n_basis_time, pen_res_time, batch_size, seq_len ]\n",
        "\n",
        "        ttaus = torch.clamp(taus[mask], max=self.kernel.tau_max-0.01)           # [ mask.sum() ]\n",
        "        int_start_idx = (torch.div(ttaus, self.h, rounding_mode=\"floor\")).long()  # [ mask.sum() ]\n",
        "        int_end_idx   = int_start_idx + 1\n",
        "        int_start     = self.xbasis_time_grids[:, int_start_idx]                # [ n_basis_time, mask.sum() ]\n",
        "        int_end       = self.xbasis_time_grids[:, int_end_idx]                  # [ n_basis_time, mask.sum() ]\n",
        "        int_prop      = (torch.remainder(ttaus, self.h) / self.h).unsqueeze(0).repeat(self.n_basis_time, 1)  # [ n_basis_time, mask.sum() ]\n",
        "\n",
        "        xbasis_time_val[:, mask] = torch.lerp(int_start, int_end, int_prop)     # [ n_basis_time, pen_res_time, batch_size, seq_len ]\n",
        "        pen_time = xbasis_time_val * ybasis_time_val[:, None, :, :]             # [ n_basis_time, pen_res_time, batch_size, seq_len ]\n",
        "\n",
        "        # compute location kernel\n",
        "        ls_pairs = torch.stack((lgrids[:, None].repeat(1, batch_size, seq_len),\n",
        "                               ls[None, :, :].repeat(self.n_node, 1, 1)), axis=-1)  \n",
        "                                                                                # [ n_node, batch_size, seq_len, 2 ]\n",
        "        pen_loc = []\n",
        "        for filter in self.Bbasis_loc_filters:\n",
        "            pen_loc.append(filter[ls_pairs[..., 1].long(), ls_pairs[..., 0].long()])\n",
        "        pen_loc  = torch.stack(pen_loc, axis=0)                                 # [ n_basis_loc, n_node, batch_size, seq_len ]\n",
        "\n",
        "        F_pen = torch.einsum('ihmn,jlmn->ijhlmn', pen_time, pen_loc)            # [ n_basis_time, n_basis_loc,\n",
        "                                                                                #   pen_res_time, pen_res_loc, batch_size, seq_len ]\n",
        "        \n",
        "        F_pen = (F_pen.T * self.weights_soft.T).sum((-1, -2)).T.sum(-1) + self.mu(lgrids[None, :, :])        \n",
        "                                                                                # [ pen_res_time, pen_res_loc, batch_size ]\n",
        "\n",
        "        return F_pen\n",
        "\n",
        "\n",
        "    def sample_intensity(self, points, seq, device):\n",
        "        \"\"\"\n",
        "        return conditional intensity evaluation at grid points,\n",
        "\n",
        "        - points: shape [ len(points), data_dim ]\n",
        "        - seq: history, can be numpy array. [ batch_size=1, seq_len, data_dim ]\n",
        "        \"\"\"\n",
        "\n",
        "        if not torch.is_tensor(seq):\n",
        "            X = torch.tensor(seq, dtype=torch.float32).to(device)\n",
        "        else:\n",
        "            X = seq.to(device)\n",
        "        if not torch.is_tensor(points):\n",
        "            points = torch.tensor(points, dtype=torch.float32).to(device)\n",
        "        else:\n",
        "            points = points.to(device)\n",
        "\n",
        "        batch_size, seq_len, _ = X.shape\n",
        "        ts   = X[:, :, 0].clone()\n",
        "        ls   = X[:, :, 1].clone()\n",
        "        mask_all = ts > 0\n",
        "\n",
        "        with torch.no_grad():\n",
        "\n",
        "            self.nn_evaluation(X)\n",
        "\n",
        "            ## compute time kernel\n",
        "            tts   = ts.unsqueeze(0).repeat(len(points), 1, 1)                   # [ len(points), batch_size, seq_len ]\n",
        "            taus = (points[:, 0] - tts.T).T                                     # [ len(points), batch_size, seq_len ]\n",
        "            mask = ((taus > 0) * (taus <= self.kernel.tau_max)).bool()          # [ len(points), batch_size, seq_len ]\n",
        "\n",
        "\n",
        "            ybasis_time_val = self.ybasis_time_events.clone()                   # [ n_basis_time, batch_size, seq_len ]\n",
        "            xbasis_time_val = torch.zeros_like(ybasis_time_val).unsqueeze(1).repeat(1, len(points), 1, 1)   \n",
        "                                                                                # [ n_basis_time, len(points), batch_size, seq_len ]\n",
        "\n",
        "            ttaus = torch.clamp(taus[mask], max=self.kernel.tau_max-0.01)       # [ mask.sum() ]\n",
        "            int_start_idx = (torch.div(ttaus, self.h, rounding_mode=\"floor\")).long()  # [ mask.sum() ]\n",
        "            int_end_idx   = int_start_idx + 1\n",
        "            int_start     = self.xbasis_time_grids[:, int_start_idx]            # [ n_basis_time, mask.sum() ]\n",
        "            int_end       = self.xbasis_time_grids[:, int_end_idx]              # [ n_basis_time, mask.sum() ]\n",
        "            int_prop      = (torch.remainder(ttaus, self.h) / self.h).unsqueeze(0).repeat(self.n_basis_time, 1)  \n",
        "                                                                                # [ n_basis_time, mask.sum() ]\n",
        "            xbasis_time_val[:, mask] = torch.lerp(int_start, int_end, int_prop) # [ n_basis_time, mask.sum() ]\n",
        "\n",
        "            kernel_time = xbasis_time_val * ybasis_time_val[:, None, :, :]      # [ n_basis_time, len(points), batch_size, seq_len ]\n",
        "\n",
        "            ## compute location kernel\n",
        "            lgrids = points[:, 1]                                               # [ len(points) ]\n",
        "            ls_pairs = torch.stack((lgrids[:, None, None].repeat(1, batch_size, seq_len),\n",
        "                               ls[None, :, :].repeat(len(points), 1, 1)), axis=-1)  \n",
        "                                                                                # [ len(points), batch_size, seq_len, 2 ]\n",
        "            kernel_loc = []\n",
        "            for filter in self.Bbasis_loc_filters:\n",
        "                kernel_loc.append(filter[ls_pairs[..., 1].long(), ls_pairs[..., 0].long()])\n",
        "            kernel_loc  = torch.stack(kernel_loc, axis=0)                       # [ n_basis_loc, len(points), batch_size, seq_len ]\n",
        "\n",
        "            kernel_val = torch.einsum('ilbs,jlbs->ijlbs', kernel_time, kernel_loc)\n",
        "            lams = (((kernel_val.T * self.weights_soft.T).sum((-1, -2)).T) * mask_all).sum(-1) + self.mu(lgrids[:, None]) \n",
        "                                                                                # [ len(points), batch_size ]\n",
        "\n",
        "        return lams.cpu().numpy().T\n",
        "\n",
        "\n",
        "    def sample_intensity_parametric(self, points, seq, device):\n",
        "        \"\"\"\n",
        "        return conditional intensity evaluation at grid points, for parametric model\n",
        "\n",
        "        - points: shape [ len(points), data_dim ]\n",
        "        - seq: history, can be numpy array. [ batch_size=1, seq_len, data_dim ]\n",
        "        \"\"\"\n",
        "        if not torch.is_tensor(seq):\n",
        "            X = torch.tensor(seq, dtype=torch.float32).to(device)\n",
        "        else:\n",
        "            X = seq.to(device)\n",
        "        if not torch.is_tensor(points):\n",
        "            points = torch.tensor(points, dtype=torch.float32).to(device)\n",
        "        else:\n",
        "            points = points.to(device)\n",
        "\n",
        "        batch_size, seq_len, _ = X.shape\n",
        "        lams = []\n",
        "        with torch.no_grad():\n",
        "            for p in points:\n",
        "                t = p[0]\n",
        "                # expand point dim\n",
        "                pt   = p.unsqueeze(0).repeat(batch_size, 1)                         # [ batch_size, data_dim ]\n",
        "                # history points before time t (H_t)\n",
        "                mask = ((X[:, :, 0] < t) * (X[:, :, 0] > self.T[0]))[:, :, None]    # [ batch_size, seq_len, data_dim ]\n",
        "                ht   = X * mask                                                     # [ batch_size, seq_len, data_dim ]\n",
        "                lam  = self.cond_lambda(pt, ht)                                     # [ batch_size ]\n",
        "                lams.append(lam)\n",
        "        lams = torch.stack(lams, 0)                                             # [ len(points), batch_size ]\n",
        "        return lams.cpu().numpy().T"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "f_8r8JAqBRV_"
      },
      "source": [
        "# Visualization"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 40,
      "metadata": {
        "id": "XeFCNpGD-gPw"
      },
      "outputs": [],
      "source": [
        "def calc_temporal_kernel(kernel, ngrid, node=0):\n",
        "    tt = torch.linspace(kernel.T[0], kernel.T[1], ngrid)                        # [ ngrid ]\n",
        "    ttau = torch.linspace(0, kernel.tau_max, ngrid)                             # [ ngrid ]\n",
        "    t_tau_pair = torch.FloatTensor(list(itertools.product(ttau.numpy(), tt.numpy()))) # [ ngrid^2, 2 ]\n",
        "    fixed_s = node\n",
        "    s  = torch.ones_like(t_tau_pair[:, 0]) * fixed_s\n",
        "    xx = torch.stack([t_tau_pair[:, 0] + t_tau_pair[:, 1], s], dim=0).T         # [ ngrid^2, 2 ]\n",
        "    yy = torch.stack([t_tau_pair[:, 1], s], dim=0).T                            # [ ngrid^2, 2 ]\n",
        "    with torch.no_grad():\n",
        "        vals = kernel(xx, yy)                                                   # [ ngrid^2 ]\n",
        "        vals = vals.reshape(ngrid, ngrid).numpy().T                             # [ ngrid, ngrid ]\n",
        "    return vals\n",
        "\n",
        "\n",
        "def calc_graph_kernel(kernel):\n",
        "    ss = torch.arange(kernel.n_node)                                            # [ n_node ]\n",
        "    ssp = torch.arange(kernel.n_node)                                           # [ n_node ]\n",
        "    s_sp_pair = torch.FloatTensor(list(itertools.product(ss.numpy(), ssp.numpy())))  # [ n_node^2, 2 ]\n",
        "    fixed_t = 1.\n",
        "    t  = torch.ones_like(s_sp_pair[:, 0]) * fixed_t\n",
        "    xx = torch.stack([t, s_sp_pair[:, 0]], dim=0).T                             # [ n_node^2, 2 ]\n",
        "    yy = torch.stack([t, s_sp_pair[:, 1]], dim=0).T                             # [ n_node^2, 2 ]\n",
        "    with torch.no_grad():\n",
        "        vals = kernel(xx, yy)                                                   # [ n_node^2 ]\n",
        "        vals = vals.reshape(kernel.n_node, kernel.n_node).numpy().T             # [ n_node, n_node ]\n",
        "    return vals\n",
        "\n",
        "\n",
        "def calc_graph_pointprocess(model, points, T_plot, ngrid):\n",
        "    ts      = torch.linspace(T_plot[0], T_plot[1], ngrid)\n",
        "    lamvals = []\n",
        "    with torch.no_grad():\n",
        "        for t in ts:\n",
        "            _t     = t.unsqueeze(0).repeat(model.n_node, 1)                 # [ n_node, 1 ]\n",
        "            ss     = torch.arange(model.n_node).unsqueeze(-1)               # [ n_node, 1 ]\n",
        "            x      = torch.cat([_t, ss], 1)                                 # [ n_node, 2 ]\n",
        "            ind    = np.where((points[:, 0] < t) * (points[:, 0] > model.T[0]))[0]\n",
        "            his_x  = points[ind, :]                                         # [ seq_len, 2 ]\n",
        "            his_x  = his_x.unsqueeze(0).repeat(model.n_node, 1, 1)          # [ n_node, seq_len, 2 ]\n",
        "            lamval = model.cond_lambda(x, his_x)                            # [ n_node ]\n",
        "            lamval = lamval.numpy()\n",
        "            lamvals.append(lamval)\n",
        "    lamvals = np.stack(lamvals, 0).T                                        # [ n_node, ngrid ]\n",
        "    return lamvals  \n",
        "\n",
        "\n",
        "def plot_fitted_temporal_graph_model(model,\n",
        "                                     points,\n",
        "                                     T_plot,\n",
        "                                     ngrid=1000,\n",
        "                                     annotation=False,\n",
        "                                     plot_events=False,\n",
        "                                     time_kernel_ylim=None, node=0,\n",
        "                                     graph_kernel_ylim=None,\n",
        "                                     lam_ylim=None,\n",
        "                                     filename=\"Epoch 0\",\n",
        "                                     savefig=False,\n",
        "                                     savepath=\"aa\"):\n",
        "    \"\"\"\n",
        "    visualize the fitted model, including both the kernels and intensity.\n",
        "\n",
        "    - points: event sequence, [ seq_len, 2 ]\n",
        "    \"\"\"          \n",
        "\n",
        "    fig = plt.figure(figsize=(12, 3.5))\n",
        "\n",
        "    ker_vals = calc_temporal_kernel(model.kernel, ngrid, node)\n",
        "    ax1 = fig.add_subplot(131)\n",
        "    if time_kernel_ylim:\n",
        "        im = ax1.imshow(ker_vals, vmin=time_kernel_ylim[0], vmax=time_kernel_ylim[1])\n",
        "    else:\n",
        "        im = ax1.imshow(ker_vals)\n",
        "    fig.colorbar(im, ax=ax1, shrink=0.7)\n",
        "    ax1.set_xlabel(r\"$\\tau$\", labelpad=-3, fontsize=12)\n",
        "    ax1.set_ylabel(r\"$t^\\prime$\", labelpad=-3, fontsize=12)\n",
        "    ax1.set_xticks([0, ngrid-1])\n",
        "    ax1.set_xticklabels([0, model.kernel.tau_max])\n",
        "    ax1.set_yticks([0, ngrid-1])\n",
        "    ax1.set_yticklabels([model.T[0], model.T[1]])\n",
        "    ax1.set_title(r\"Temporal $k(\\cdot, \\cdot, %d, %d)$\" % (node, node), fontsize=15)\n",
        "\n",
        "    ker_vals = calc_graph_kernel(model.kernel)\n",
        "    ax2 = fig.add_subplot(132)\n",
        "    if graph_kernel_ylim:\n",
        "        im = ax2.imshow(ker_vals, vmin=graph_kernel_ylim[0], vmax=graph_kernel_ylim[1])\n",
        "    else:\n",
        "        im = ax2.imshow(ker_vals)\n",
        "    fig.colorbar(im, ax=ax2, shrink=0.7)\n",
        "    ax2.set_xlabel(\"node\", labelpad=-1, fontsize=12)\n",
        "    ax2.set_ylabel(\"node\", labelpad=-1, fontsize=12)\n",
        "    ax2.set_xticks([0, model.kernel.n_node-1])\n",
        "    ax2.set_xticklabels([0, model.kernel.n_node-1])\n",
        "    ax2.set_yticks([0, model.kernel.n_node-1])\n",
        "    ax2.set_yticklabels([0, model.kernel.n_node-1])\n",
        "    ax2.set_title(r\"Graph $k(1, 1, \\cdot, \\cdot)$\", fontsize=15)\n",
        "\n",
        "    if annotation:\n",
        "        for i in range(model.kernel.n_node):\n",
        "            for j in range(model.kernel.n_node):\n",
        "                text = ax2.text(j, i, np.round(ker_vals[i, j], 2),\n",
        "                       ha=\"center\", va=\"center\", color=\"r\")\n",
        "\n",
        "    lamvals = calc_graph_pointprocess(model, points, T_plot, ngrid)\n",
        "    ax3 = fig.add_subplot(133)\n",
        "    axw, axh = ax3.figure.get_size_inches()\n",
        "    ax3.figure.set_size_inches(axw, axh / 1.2)\n",
        "    lam_scale = (np.min(lamvals), np.max(lamvals))\n",
        "    if lam_ylim:\n",
        "        im = ax3.imshow(lamvals, vmin=lam_ylim[0], vmax=lam_ylim[1], aspect=\"auto\")\n",
        "    else:\n",
        "        im = ax3.imshow(lamvals, vmin=lam_scale[0], vmax=lam_scale[1], aspect=\"auto\")\n",
        "    ax3.set_xlabel(\"$t$\", labelpad=-1, fontsize=12)\n",
        "    ax3.set_ylabel(\"node\", labelpad=-1, fontsize=12)\n",
        "    ax3.set_xticks([0, ngrid - 1])\n",
        "    ax3.set_xticklabels(np.round_([T_plot[0], T_plot[1]], 1))\n",
        "    ax3.set_yticks([0, model.kernel.n_node-1])\n",
        "    ax3.set_yticklabels([0, model.kernel.n_node-1])\n",
        "    fig.colorbar(im, ax=ax3, shrink=.8)\n",
        "    # ax.title.set_text('lambda at t=%.1f' % lam_ts[j])\n",
        "    ax3.set_title(filename + \" lambda evolution\", fontsize=15)\n",
        "\n",
        "    if plot_events:\n",
        "        ps = points[(points[:, 0] > T_plot[0]) * (points[:, 0] <= T_plot[1])]\n",
        "        ax3.scatter((ps[:, 0] - T_plot[0]) / (T_plot[1] - T_plot[0]) * (ngrid-1), ps[:, 1], marker=\"x\", c=\"red\", s=20, label=\"event\")\n",
        "        ax3.legend()\n",
        "\n",
        "    plt.show()\n",
        "    if savefig: fig.savefig(\"%s/%s.pdf\" % (savepath, filename+\" Intensity evolution\"))"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "vd1iTBCw_Zuy"
      },
      "source": [
        "# Training Functions"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 41,
      "metadata": {
        "id": "2PQ_rOXo_ctr"
      },
      "outputs": [],
      "source": [
        "def config_generate(lr=1e-2,\n",
        "                    epoch=500,\n",
        "                    batch_size=64,\n",
        "                    lam_reg=100,\n",
        "                    penalty=False,\n",
        "                    reg=False,\n",
        "                    mae_eval=False,\n",
        "                    t_init=1e1,\n",
        "                    t_upp=1e6,\n",
        "                    t_mul=1.3,\n",
        "                    b_bar=10.,\n",
        "                    b_upp=-5.,\n",
        "                    device=\"cpu\"):\n",
        "    \n",
        "    if not penalty:\n",
        "        print(\"no penalty!\")\n",
        "    if not reg:\n",
        "        print(\"no reg!\")\n",
        "    if not mae_eval:\n",
        "        print(\"no mae_eval!\")\n",
        "\n",
        "    config = {\n",
        "        'lr': lr,\n",
        "        'epoch': epoch,\n",
        "        'batch_size': batch_size,\n",
        "        'lam_reg': lam_reg,\n",
        "        'penalty': penalty,\n",
        "        'reg': reg,\n",
        "        'mae_eval': mae_eval,\n",
        "        't_init': t_init,\n",
        "        't_upp': t_upp,\n",
        "        't_mul': t_mul,\n",
        "        'b_bar': b_bar,\n",
        "        'b_upp': b_upp,\n",
        "        'device': device\n",
        "    }\n",
        "\n",
        "    return config"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 42,
      "metadata": {
        "id": "8UsBJRIhA54P"
      },
      "outputs": [],
      "source": [
        "def train(model,\n",
        "            trg_model,\n",
        "            config,\n",
        "            train_data,\n",
        "            test_data,\n",
        "            eval_points,\n",
        "            plot_points,\n",
        "            plot_ngrid,\n",
        "            modelname=\"pp\",\n",
        "            save_model=True,\n",
        "            save_path=None,\n",
        "            load_iter=10):\n",
        "    \"\"\"training procedure\"\"\"\n",
        "\n",
        "    if save_model:\n",
        "        if os.path.exists(save_path): \n",
        "            print(\"Duplicated folder!\")\n",
        "            return None\n",
        "        else:\n",
        "            print(\"Create folder!\")\n",
        "            os.makedirs(save_path)\n",
        "\n",
        "    if trg_model is not None:\n",
        "        plot_fitted_temporal_graph_model(trg_model,\n",
        "                                     plot_points,\n",
        "                                     T_plot=model.T,\n",
        "                                     ngrid=500,\n",
        "                                     annotation=False,\n",
        "                                     lam_ylim=None,\n",
        "                                     filename=\"True\",\n",
        "                                     savefig=save_model,\n",
        "                                     savepath=save_path)\n",
        "\n",
        "    n_events_train = (train_data[:, :, 0] > 0).sum()\n",
        "    n_events_test = (test_data[:, :, 0] > 0).sum()\n",
        "    print(\"[%s] #Training sequences: %d\" % (arrow.now(), train_data.shape[0]))\n",
        "    print(\"[%s] #Testing sequences: %d\" % (arrow.now(), test_data.shape[0]))\n",
        "    print(\"[%s] #Training events: %d\" % (arrow.now(), n_events_train))\n",
        "    print(\"[%s] #Testing events: %d\" % (arrow.now(), n_events_test))\n",
        "    train_loader = DataLoader(torch.utils.data.TensorDataset(train_data), \n",
        "                              shuffle=True, batch_size=config['batch_size'], drop_last=False)\n",
        "    test_loader  = DataLoader(torch.utils.data.TensorDataset(test_data), \n",
        "                              shuffle=False, batch_size=config['batch_size'], drop_last=False)\n",
        "    \n",
        "    # Evaluate synthetic data with True model\n",
        "    if trg_model is not None:\n",
        "        print(\"[%s] Compute true intensity...\" % arrow.now())\n",
        "        true_lams = []\n",
        "        for batch in test_loader:\n",
        "            true_lams.append(trg_model.sample_intensity_parametric(eval_points, batch[0], device=\"cpu\"))\n",
        "        true_lams = np.concatenate(true_lams, axis=0)                               # [ n_test_seq, len(eval_points) ]\n",
        "\n",
        "\n",
        "    train_llks = []\n",
        "    test_llks  = []\n",
        "    test_maes  = []\n",
        "    test_mres  = []\n",
        "    wall_time  = []\n",
        "    lam_mins   = []\n",
        "    losses     = []\n",
        "    ts         = []\n",
        "    bs         = []\n",
        "    \n",
        "    if config[\"penalty\"]:\n",
        "        b = -config[\"b_bar\"]\n",
        "\n",
        "    i = 0\n",
        "    fea = 0\n",
        "    bar_flag = 0\n",
        "    t = config[\"t_upp\"]\n",
        "\n",
        "    model.to(config[\"device\"])\n",
        "    optimizer = optim.Adadelta(model.parameters(), lr=config[\"lr\"])\n",
        "    \n",
        "    # Data fitting\n",
        "    print(\"[%s] Start Model Learning...\" % arrow.now())\n",
        "    t0 = arrow.now()\n",
        "    while i < config[\"epoch\"]:\n",
        "        try:\n",
        "            epoch_llk      = 0\n",
        "            epoch_loss     = 0\n",
        "            epoch_llk_loss = 0\n",
        "            epoch_bar_loss = 0\n",
        "            epoch_reg_loss = 0\n",
        "            num_overshot   = 0\n",
        "            lam_min        = np.inf\n",
        "\n",
        "            for batch in train_loader:\n",
        "                X         = batch[0].to(config[\"device\"])\n",
        "                optimizer.zero_grad()\n",
        "                loglik, reg    = model(X)\n",
        "                llk       = - loglik / X.shape[0]\n",
        "                loss      = llk\n",
        "\n",
        "                if config[\"reg\"]:\n",
        "                    loss = loss + reg * config[\"lam_reg\"]\n",
        "\n",
        "                if config[\"penalty\"]:\n",
        "                    lams      = model.penalty_term(X)\n",
        "                    b_temp    = min(lams.min().item() - config[\"b_bar\"], config[\"b_upp\"])\n",
        "                    if b_temp < b:\n",
        "                        num_overshot += 1\n",
        "                    else: b_temp = b\n",
        "                    bar       = (- torch.log(torch.clamp(lams - b, min=1e-5))).mean()\n",
        "                    lam_min   = min(lam_min, lams.min().item())\n",
        "\n",
        "                    loss      = loss + bar / t\n",
        "                \n",
        "                loss.backward()\n",
        "                optimizer.step()\n",
        "\n",
        "                epoch_llk += loglik\n",
        "                epoch_loss += loss\n",
        "                epoch_llk_loss += llk\n",
        "                if config[\"reg\"]:\n",
        "                    epoch_reg_loss += config[\"lam_reg\"] * reg\n",
        "                if config[\"penalty\"]:\n",
        "                    epoch_bar_loss += bar / t\n",
        "                    b = min(lam_min - config[\"b_bar\"], config[\"b_upp\"])\n",
        "\n",
        "            with torch.no_grad():\n",
        "                test_llk = 0\n",
        "                for batch in test_loader:\n",
        "                    X_test      = batch[0].to(config[\"device\"])\n",
        "                    te_llk, reg = model(X_test)\n",
        "                    test_llk    += te_llk\n",
        "\n",
        "                train_e_llk = (epoch_llk / n_events_train).item()\n",
        "                test_e_llk  = (test_llk / n_events_test).item()\n",
        "                print(\"[%s] Epoch : %d,\\tTraining llk : %.8f\" % (arrow.now(), i, train_e_llk))\n",
        "                print(\"[%s] Epoch : %d,\\tTesting llk : %.8f\" % (arrow.now(), i, test_e_llk))\n",
        "                train_llks.append(train_e_llk)\n",
        "                test_llks.append(test_e_llk)\n",
        "\n",
        "                if config[\"mae_eval\"]:\n",
        "                    lams_test = []\n",
        "                    for batch in test_loader:\n",
        "                        lams_test.append(model.sample_intensity(eval_points, batch[0], device=config[\"device\"]))\n",
        "                    lams_test = np.concatenate(lams_test, axis=0)\n",
        "                    test_mae   = np.mean(np.abs(lams_test - true_lams))\n",
        "                    test_mre   = np.mean(np.abs(lams_test - true_lams) / true_lams)\n",
        "                    print(\"[%s] Epoch : %d,\\tTesting lams MAE : %.8f\" % (arrow.now(), i, test_mae))\n",
        "                    print(\"[%s] Epoch : %d,\\tTesting lams MRE : %.8f\" % (arrow.now(), i, test_mre))\n",
        "                    test_maes.append(test_mae)\n",
        "                    test_mres.append(test_mre)\n",
        "\n",
        "            t_e = arrow.now()\n",
        "            wall_time.append((t_e - t0).total_seconds())\n",
        "            \n",
        "            logout = \"[%s] Epoch : %d, \\ttotal_loss : %.8f, \\tllk_loss : %.8f\" % (arrow.now(), i, epoch_loss, epoch_llk_loss)\n",
        "            ret    = [epoch_loss.item(), epoch_llk_loss.item()]\n",
        "            if config[\"reg\"]:\n",
        "                logout = logout + \"\\treg_loss : %.8f\" % (epoch_reg_loss)\n",
        "                ret.append(epoch_reg_loss.item())\n",
        "            if config[\"penalty\"]:\n",
        "                logout = logout + \"\\tbarrier_loss : %.8f, min_lam : %.8f, num_oveshot : %d, t : %.5f, b : %.5f\" % (epoch_bar_loss, lam_min, num_overshot, t, b)\n",
        "                ret.append(epoch_bar_loss.item())\n",
        "\n",
        "            print(logout)\n",
        "            losses.append(ret)\n",
        "\n",
        "            if (i+1) % load_iter == 0:\n",
        "                model.cpu()\n",
        "                plot_fitted_temporal_graph_model(model,\n",
        "                                                plot_points,\n",
        "                                                T_plot=model.T,\n",
        "                                                ngrid=500,\n",
        "                                                annotation=False,\n",
        "                                                lam_ylim=None,\n",
        "                                                filename=\"Epoch %d\" % i,\n",
        "                                                savefig=save_model,\n",
        "                                                savepath=save_path)\n",
        "\n",
        "                if save_model:\n",
        "                    torch.save(model.state_dict(), \"%s/%s-%d.pth\" % (save_path, modelname, i))\n",
        "\n",
        "                model.to(config[\"device\"])\n",
        "\n",
        "            if config[\"penalty\"]:\n",
        "                lam_mins.append(lam_min)\n",
        "                ts.append(t)\n",
        "                bs.append(b)\n",
        "                if lam_min < config[\"b_bar\"] + config[\"b_upp\"]:\n",
        "                    t = config[\"t_init\"]\n",
        "                    b = lam_min - config[\"b_bar\"]\n",
        "                    bar_flag = 1\n",
        "                else:\n",
        "                    t = min(t*config[\"t_mul\"], config[\"t_upp\"])\n",
        "                    b = config[\"b_upp\"]\n",
        "\n",
        "            num_overshot = 0\n",
        "            i += 1\n",
        "\n",
        "        except KeyboardInterrupt:\n",
        "            break\n",
        "    \n",
        "    return train_llks, test_llks, test_maes, test_mres, losses, wall_time, lam_mins, ts, bs"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "LRVU8kceEXt5"
      },
      "source": [
        "# Experiments"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "IYkhoRobwWAN"
      },
      "outputs": [],
      "source": [
        "seed = 100\n",
        "torch.manual_seed(seed)\n",
        "np.random.seed(seed)\n",
        "\n",
        "## model configuration\n",
        "T = [0., 50.]\n",
        "tau_max  = 5.\n",
        "mu_max   = .3\n",
        "mu       = mu_max * np.ones(3)\n",
        "data_dim = 2\n",
        "n_basis_time = 1\n",
        "n_basis_loc  = 2\n",
        "data_name    = \"data-basecos_exp-neg-graph-V3\"\n",
        "model_name   = \"Model_for_%s_nbasistime%d_nbasisloc%d\" % (data_name, n_basis_time, n_basis_loc)\n",
        "save_path    = rootpath + \"/saved_models/%s\" % model_name\n",
        "G = torch.load(rootpath + \"/data/%s/graph.pt\" % data_name)\n",
        "\n",
        "device     = torch.device(\"cuda:0\" if torch.cuda.is_available() else \"cpu\")\n",
        "\n",
        "## true model\n",
        "timebasis = BaseExponentialCosineBasis(alpha=1.5, beta=2., freq=.2)\n",
        "basis_weight = np.array([[.5, .2]])\n",
        "filter_1 = torch.Tensor([[0.5, 0.0, 0.0],\n",
        "                         [0.0, 0.7, 0.0],\n",
        "                         [0.0, 0.0, 0.5]])\n",
        "filter_2 = torch.Tensor([[0.0, 0.0, 0.0],\n",
        "                         [-0.2, 0.0, 0.4],\n",
        "                         [0.0, 0.0, 0.0]])\n",
        "kernel = TemporalParametricKernelL3netLocalFilterOnGraph(device=\"cpu\", T=T, tau_max=tau_max, G=G,\n",
        "                                                         time_basis_list=[timebasis],\n",
        "                                                         loc_filter_list=[filter_1, filter_2],\n",
        "                                                         data_dim=data_dim, basis_weight=basis_weight,\n",
        "                                                         basis_dim=1, init_std=1e0)\n",
        "trg_model = TemporalPointProcessOnGraph(device=\"cpu\", T=T, G=G, mu=mu, tau_max=tau_max,\n",
        "                                 kernel=kernel, data_dim=data_dim)\n",
        "\n",
        "## initialize model\n",
        "order_list = [0, 1]\n",
        "kernel = TemporalDeepBasisL3netLocalFilterOnGraphKernel(T=T, G=G, tau_max=tau_max, device=device,\n",
        "                                        n_basis_time=n_basis_time, loc_order_list=order_list,\n",
        "                                        data_dim=data_dim, basis_dim=1, nn_width_basis_time=32,\n",
        "                                        init_gain=5e-1, init_bias=1e-2, init_std=1e-2,)\n",
        "init_model = TemporalPointProcessOnGraph(device=device, T=T, G=G, mu=mu, tau_max=tau_max,\n",
        "                                         kernel=kernel, data_dim=data_dim, numerical_int=True,\n",
        "                                         int_res=100, int_res_loc=200, pen_res_time=50, eval_res=100)\n",
        "\n",
        "\n",
        "## initialize training configuration\n",
        "config = config_generate(lr=1e-2, epoch=1000, batch_size=32,\n",
        "                        penalty=True, mae_eval=True,\n",
        "                        t_init=1e0, t_upp=1e6, t_mul=1.3, b_bar=mu_max/4.,\n",
        "                        b_upp=-mu_max/4., device=device)\n",
        "ts = np.linspace(T[0], T[1], 50)\n",
        "ns = np.arange(G.x.shape[0])\n",
        "eval_points = eval_points_generate(ts, ns)\n",
        "\n",
        "## data loading\n",
        "data          = np.load(rootpath + \"/data/%s/%s.npy\" % (data_name, data_name))\n",
        "data          = torch.FloatTensor(data)\n",
        "train_data    = data[:800]\n",
        "test_data     = data[800:]\n",
        "plot_points   = test_data[0]\n",
        "\n",
        "# training\n",
        "save_model = True\n",
        "train_llks, test_llks, test_maes, test_mres, losses, wall_time, lam_mins, ts, bs = train(init_model, trg_model, config, train_data, test_data,\n",
        "      eval_points=eval_points, plot_points=plot_points, plot_ngrid=100,\n",
        "      modelname=model_name, save_model=save_model, save_path=save_path, load_iter=50)\n",
        "if save_model:\n",
        "    np.save(rootpath + \"/saved_models/%s/losses.npy\" % model_name, losses)\n",
        "    np.save(rootpath + \"/saved_models/%s/metrics.npy\" % model_name, [train_llks, test_llks, test_maes, test_mres])\n",
        "    if config[\"penalty\"]:\n",
        "        np.save(rootpath + \"/saved_models/%s/lam_mins.npy\" % model_name, [lam_mins, ts, bs])"
      ]
    },
    {
      "attachments": {},
      "cell_type": "markdown",
      "metadata": {
        "id": "iiFyKvnuIAAl"
      },
      "source": [
        "# Evaluation"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 49,
      "metadata": {
        "id": "O0JKIHujiNg6"
      },
      "outputs": [],
      "source": [
        "seed = 100\n",
        "torch.manual_seed(seed)\n",
        "np.random.seed(seed)\n",
        "\n",
        "## model configuration\n",
        "T = [0., 50.]\n",
        "n_node = 3\n",
        "tau_max  = 5.\n",
        "mu_max   = 0.3\n",
        "mu       = mu_max * np.zeros(n_node)\n",
        "data_dim = 2\n",
        "n_basis_time = 1\n",
        "n_basis_loc  = 2\n",
        "data_name    = \"data-basecos_exp-neg-graph-V3\"\n",
        "model_name   = \"Model_for_%s_nbasistime%d_nbasisloc%d\" % (data_name, n_basis_time, n_basis_loc)\n",
        "save_path    = rootpath + \"/saved_models/%s\" % model_name\n",
        "G = torch.load(rootpath + \"/data/%s/graph.pt\" % data_name)\n",
        "\n",
        "device     = torch.device(\"cuda:0\" if torch.cuda.is_available() else \"cpu\")\n",
        "\n",
        "## true model\n",
        "timebasis = BaseExponentialCosineBasis(alpha=1.5, beta=2., freq=.2)\n",
        "basis_weight = np.array([[.5, .2]])\n",
        "filter_1 = torch.Tensor([[0.5, 0.0, 0.0],\n",
        "                         [0.0, 0.7, 0.0],\n",
        "                         [0.0, 0.0, 0.5]])\n",
        "filter_2 = torch.Tensor([[0.0, 0.0, 0.0],\n",
        "                         [-0.2, 0.0, 0.4],\n",
        "                         [0.0, 0.0, 0.0]])\n",
        "kernel = TemporalParametricKernelL3netLocalFilterOnGraph(device=\"cpu\", T=T, tau_max=tau_max, G=G,\n",
        "                                                         time_basis_list=[timebasis],\n",
        "                                                         loc_filter_list=[filter_1, filter_2],\n",
        "                                                         data_dim=data_dim, basis_weight=basis_weight,\n",
        "                                                         basis_dim=1, init_std=1e0)\n",
        "trg_model = TemporalPointProcessOnGraph(device=\"cpu\", T=T, G=G, mu=mu, tau_max=tau_max,\n",
        "                                 kernel=kernel, data_dim=data_dim)\n",
        "\n",
        "## initialize model\n",
        "order_list = [0, 1]\n",
        "kernel = TemporalDeepBasisL3netLocalFilterOnGraphKernel(T=T, G=G, tau_max=tau_max, device=device,\n",
        "                                        n_basis_time=n_basis_time, loc_order_list=order_list,\n",
        "                                        data_dim=data_dim, basis_dim=1, nn_width_basis_time=32,\n",
        "                                        init_gain=5e-1, init_bias=1e-2, init_std=1e-2,)\n",
        "init_model = TemporalPointProcessOnGraph(device=device, T=T, G=G, mu=mu, tau_max=tau_max,\n",
        "                                         kernel=kernel, data_dim=data_dim, numerical_int=True,\n",
        "                                         int_res=100, int_res_loc=200, pen_res_time=50, eval_res=100)\n",
        "init_model.load_state_dict(torch.load(rootpath + \"/saved_models/saved_model_dict.pth\"))\n",
        "\n",
        "## initialize training configuration\n",
        "config = config_generate(lr=1e-2, epoch=1000, batch_size=32,\n",
        "                        penalty=True, mae_eval=True,\n",
        "                        t_init=1e0, t_upp=1e6, t_mul=1.3, b_bar=mu/4.,\n",
        "                        b_upp=-mu/4., device=device)\n",
        "ts = np.linspace(T[0], T[1], 50)\n",
        "ns = np.arange(G.x.shape[0])\n",
        "eval_points = eval_points_generate(ts, ns)\n",
        "\n",
        "## data loading\n",
        "data          = np.load(rootpath + \"/data/%s/%s.npy\" % (data_name, data_name))\n",
        "data          = torch.FloatTensor(data)\n",
        "train_data    = data[:800]\n",
        "test_data     = data[800:]\n",
        "plot_points   = test_data[189]"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 40,
      "metadata": {
        "id": "Gad4qNsIhY8G"
      },
      "outputs": [],
      "source": []
    },
    {
      "cell_type": "code",
      "execution_count": 41,
      "metadata": {
        "id": "ai7S2wuOhZCm"
      },
      "outputs": [],
      "source": [
        "graph_kernel_vals = calc_graph_kernel(init_model.kernel).T\n",
        "\n",
        "events = plot_points[plot_points[:, 0] > 0]\n",
        "T_plot = [10., 50.]\n",
        "ngrid = 200\n",
        "lam_vals = calc_graph_pointprocess(init_model, events, T_plot=T_plot, ngrid=ngrid)\n",
        "\n",
        "select_event_idx = np.arange(0, len(events), 1)\n",
        "n_dep = len(select_event_idx)\n",
        "idx_pair = np.array(list(itertools.product(select_event_idx, select_event_idx)))\n",
        "event_dep_1 = events[idx_pair[:, 0]]\n",
        "event_dep_2 = events[idx_pair[:, 1]]\n",
        "with torch.no_grad():\n",
        "    dep_vals    = init_model.kernel(event_dep_1, event_dep_2).numpy()\n",
        "    dep_vals    = dep_vals.reshape(n_dep, n_dep)\n",
        "    upp_idx     = np.bool_(np.triu(np.ones((n_dep, n_dep))))\n",
        "    dep_vals[upp_idx] = 0."
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 42,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 358
        },
        "id": "yfZ0cNmahZCn",
        "outputId": "27a69f84-8d10-4295-c213-16223e318747"
      },
      "outputs": [
        {
          "data": {
            "text/plain": [
              "Text(0, 0.5, 'node')"
            ]
          },
          "execution_count": 42,
          "metadata": {},
          "output_type": "execute_result"
        },
        {
          "data": {
            "image/png": "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",
            "text/plain": [
              "<Figure size 1200x400 with 3 Axes>"
            ]
          },
          "metadata": {},
          "output_type": "display_data"
        }
      ],
      "source": [
        "fig = plt.figure(figsize=(12, 4))\n",
        "\n",
        "ax1 = fig.add_subplot(1, 3, 1)\n",
        "ax1.imshow(graph_kernel_vals, cmap=\"Blues\")\n",
        "ax1.set_xticks([])\n",
        "ax1.set_yticks([])\n",
        "ax1.set_xlabel(\"node\", fontsize=18)\n",
        "ax1.set_ylabel(\"node\", fontsize=18)\n",
        "\n",
        "ax2 = fig.add_subplot(1, 3, 2)\n",
        "ax2.imshow(dep_vals, cmap=\"Greys\")\n",
        "ax2.set_xticks([])\n",
        "ax2.set_yticks([])\n",
        "ax2.set_xlabel(\"event\", fontsize=18)\n",
        "ax2.set_ylabel(\"event\", fontsize=18)\n",
        "\n",
        "ax3 = fig.add_subplot(1, 3, 3)\n",
        "ax3.imshow(lam_vals, cmap=\"Reds\", aspect=ngrid/n_node)\n",
        "plot_events = events[(events[:, 0] >= T_plot[0]) * (events[:, 0] <= T_plot[1])]\n",
        "ax3.scatter((plot_events[:, 0] - T_plot[0]) / (T_plot[1] - T_plot[0]) * (ngrid-1), plot_events[:, 1], marker=\"x\", c=\"black\", alpha=0.4, s=30, linewidth=1.0)\n",
        "ax3.set_xticks([])\n",
        "ax3.set_yticks([])\n",
        "ax3.set_xlabel(\"time\", fontsize=18)\n",
        "ax3.set_ylabel(\"node\", fontsize=18)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "x0eBJyW1F3OK"
      },
      "outputs": [],
      "source": []
    }
  ],
  "metadata": {
    "accelerator": "GPU",
    "colab": {
      "gpuType": "T4",
      "provenance": [],
      "toc_visible": true
    },
    "gpuClass": "standard",
    "kernelspec": {
      "display_name": "Python 3",
      "name": "python3"
    },
    "language_info": {
      "name": "python"
    }
  },
  "nbformat": 4,
  "nbformat_minor": 0
}
