{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "099b95a6",
   "metadata": {},
   "outputs": [],
   "source": [
    "import csv\n",
    "import os\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import seaborn\n",
    "import torch\n",
    "from torch.utils.data import DataLoader, random_split\n",
    "from dcdfg_extend_pc import vaeGraph, VAEGraphShared\n",
    "from make_dataset import PerturbDataset\n",
    "import networkx as nx\n",
    "from sklearn.preprocessing import StandardScaler\n",
    "import sys\n",
    "sys.path.insert(1, '../')\n",
    "from data.simulations import DatasetLowRankGenerator\n",
    "from tqdm import tqdm\n",
    "import matplotlib.pyplot as plt\n",
    "from torch.utils.data import Dataset\n",
    "#from dcdfg.metrics import fdr, shd_metric\n",
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "figure_path = '/XXXX-11/XXXX-12/XXXX-10-XXXX-9/XXXX-6/figures/VBFG/simulation'\n",
    "!mkdir -p {figure_path}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "c3acc9c3",
   "metadata": {},
   "outputs": [],
   "source": [
    "class SimulationDataset(Dataset):\n",
    "    \"\"\"\n",
    "    A generic class for simulation data loading and extraction, as well as pre-filtering of interventions\n",
    "    NOTE: the 0-th regime should always be the observational one\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(\n",
    "        self,\n",
    "        file_path,\n",
    "        i_dataset,\n",
    "        intervention=True,\n",
    "        fraction_regimes_to_ignore=None,\n",
    "        regimes_to_ignore=None,\n",
    "        load_ignored=False,\n",
    "    ) -> None:\n",
    "        \"\"\"\n",
    "        :param str file_path: Path to the data and the DAG\n",
    "        :param int i_dataset: Exemplar to use (usually in [1,10])\n",
    "        :param boolean intervention: If True, use interventional data with interventional targets\n",
    "        :param list regimes_to_ignore: Regimes that are ignored during training\n",
    "        \"\"\"\n",
    "        super(SimulationDataset, self).__init__()\n",
    "        self.file_path = file_path\n",
    "        self.i_dataset = i_dataset\n",
    "        self.intervention = intervention\n",
    "        # load data\n",
    "        all_data, all_masks, all_regimes = self.load_data()\n",
    "        # index of all regimes, even if not used in the regimes_to_ignore case\n",
    "        self.all_regimes_list = np.unique(all_regimes)\n",
    "\n",
    "        if fraction_regimes_to_ignore is not None or regimes_to_ignore is not None:\n",
    "            if fraction_regimes_to_ignore is not None and regimes_to_ignore is not None:\n",
    "                raise ValueError(\"either fraction or list, not both\")\n",
    "            if fraction_regimes_to_ignore is not None:\n",
    "                # select fraction to ignore\n",
    "                np.random.seed(0)\n",
    "                sampling_list = self.all_regimes_list\n",
    "                self.regimes_to_ignore = np.random.choice(\n",
    "                    sampling_list,\n",
    "                    int(fraction_regimes_to_ignore * len(sampling_list)),\n",
    "                )\n",
    "            else:\n",
    "                self.regimes_to_ignore = regimes_to_ignore\n",
    "\n",
    "            to_keep = np.array(\n",
    "                [\n",
    "                    regime not in self.regimes_to_ignore\n",
    "                    for regime in np.array(all_regimes)\n",
    "                ]\n",
    "            )\n",
    "            if not load_ignored:\n",
    "                data = all_data[to_keep]\n",
    "                masks = [mask for i, mask in enumerate(all_masks) if to_keep[i]]\n",
    "                regimes = np.array(\n",
    "                    [regime for i, regime in enumerate(all_regimes) if to_keep[i]]\n",
    "                )\n",
    "            else:\n",
    "                data = all_data[~to_keep]\n",
    "                masks = [mask for i, mask in enumerate(all_masks) if ~to_keep[i]]\n",
    "                regimes = np.array(\n",
    "                    [regime for i, regime in enumerate(all_regimes) if ~to_keep[i]]\n",
    "                )\n",
    "        else:\n",
    "            data = all_data\n",
    "            masks = all_masks\n",
    "            regimes = all_regimes\n",
    "\n",
    "        self.data = data\n",
    "        self.regimes = regimes\n",
    "        self.masks = np.array(masks, dtype=object)\n",
    "\n",
    "        self.num_regimes = np.unique(self.regimes).shape[0]\n",
    "        self.num_samples = self.data.shape[0]\n",
    "        self.dim = self.data.shape[1]\n",
    "\n",
    "    def __getitem__(self, idx):\n",
    "        if self.intervention:\n",
    "            # binarize mask from list\n",
    "            masks_list = self.masks[idx]\n",
    "            masks = np.ones((self.dim,))\n",
    "            for j in masks_list:\n",
    "                masks[j] = 0\n",
    "            return (\n",
    "                self.data[idx].astype(np.float32),\n",
    "                masks.astype(np.float32),\n",
    "                self.regimes[idx],\n",
    "            )\n",
    "        else:\n",
    "            # put full ones mask\n",
    "            return (\n",
    "                self.data[idx].astype(np.float32),\n",
    "                np.ones((self.dim,)).astype(np.float32),\n",
    "                self.regimes[idx],\n",
    "            )\n",
    "\n",
    "    def __len__(self):\n",
    "        return self.data.shape[0]\n",
    "\n",
    "    def load_data(self):\n",
    "        \"\"\"\n",
    "        Load the mask, regimes, and data\n",
    "        \"\"\"\n",
    "        if self.intervention:\n",
    "            name_data = f\"data_interv{self.i_dataset}.npy\"\n",
    "        else:\n",
    "            name_data = f\"data{self.i_dataset}.npy\"\n",
    "\n",
    "        # Load data\n",
    "        self.data_path = os.path.join(self.file_path, name_data)\n",
    "        data = np.load(self.data_path)\n",
    "\n",
    "        # Load intervention masks and regimes\n",
    "        masks = []\n",
    "        if self.intervention:\n",
    "            name_data = f\"data_interv{self.i_dataset}.npy\"\n",
    "            interv_path = os.path.join(\n",
    "                self.file_path, f\"intervention{self.i_dataset}.csv\"\n",
    "            )\n",
    "            regimes = np.genfromtxt(\n",
    "                os.path.join(self.file_path, f\"regime{self.i_dataset}.csv\"),\n",
    "                delimiter=\",\",\n",
    "            )\n",
    "            regimes = regimes.astype(int)\n",
    "\n",
    "            # read masks\n",
    "            with open(interv_path, \"r\") as f:\n",
    "                interventions_csv = csv.reader(f)\n",
    "                for row in interventions_csv:\n",
    "                    mask = [int(x) for x in row]\n",
    "                    masks.append(mask)\n",
    "        else:\n",
    "            regimes = np.array([0] * data.shape[0])\n",
    "\n",
    "        return data, masks, regimes\n",
    "\n",
    "    def convert_masks(self, idxs):\n",
    "        \"\"\"\n",
    "        Convert mask index to mask vectors\n",
    "        :param np.ndarray idxs: indices of mask to convert\n",
    "        :return: masks\n",
    "        Example:\n",
    "            if self.masks[i] = [1,4]\n",
    "                self.dim = 10 then\n",
    "            masks[i] = [1,0,1,1,0,1,1,1,1,1]\n",
    "        \"\"\"\n",
    "        masks_list = [self.masks[i] for i in idxs]\n",
    "\n",
    "        masks = torch.ones((idxs.shape[0], self.dim))\n",
    "        for i, m in enumerate(masks_list):\n",
    "            for j in m:\n",
    "                masks[i, j] = 0\n",
    "\n",
    "        return masks"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "07d56208",
   "metadata": {},
   "source": [
    "## 1.1 Load Training and Validation Datasets"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "b000b1bb",
   "metadata": {},
   "outputs": [],
   "source": [
    "# path='/XXXX-13/XXXX-14/XXXX-15/shared_data/dcdfg_pc/data_p100_m10_n50000_linear_uniform'\n",
    "data_path='/XXXX-11/XXXX-12/XXXX-10-XXXX-9/XXXX-6/data/VBFG/simulation'\n",
    "graph = np.load(f'{data_path}/DAG0.npy')\n",
    "U0 = np.load(f'{data_path}/U0.npy')\n",
    "V0 = np.load(f'{data_path}/V0.npy')\n",
    "module_causal_order = np.load(f'{data_path}/module_order0.npy')\n",
    "# intervention = pd.read_csv(f'{data_path}/intervention1.csv')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "6a1d8a94",
   "metadata": {},
   "outputs": [],
   "source": [
    "n_samples = 100000\n",
    "np.random.seed(42)\n",
    "\n",
    "\n",
    "train_dataset = SimulationDataset(\n",
    "    data_path, 1, fraction_regimes_to_ignore=None, intervention=True\n",
    ")\n",
    "\"\"\"\n",
    "regimes_to_ignore = train_dataset.regimes_to_ignore\n",
    "test_dataset = SimulationDataset(\n",
    "    data_path, 1, regimes_to_ignore=regimes_to_ignore, load_ignored=True, intervention=True\n",
    ")\n",
    "\"\"\"\n",
    "\n",
    "train_size = int(0.8 * len(train_dataset))\n",
    "val_size = len(train_dataset) - train_size\n",
    "train_dataset, val_dataset = random_split(train_dataset, [train_size, val_size])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "9c0efd79",
   "metadata": {},
   "outputs": [],
   "source": [
    "#test part\n",
    "seed=42\n",
    "batch_size=128"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "2bcc08f2",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_loader = torch.utils.data.DataLoader(train_dataset, batch_size=batch_size, shuffle=True)\n",
    "val_loader = torch.utils.data.DataLoader(val_dataset, batch_size=batch_size, shuffle=True)\n",
    "#test_loader = torch.utils.data.DataLoader(test_dataset, batch_size=batch_size, shuffle=True)\n",
    "test_loader = val_loader\n",
    "n_train, n_val, n_test = len(train_loader), len(val_loader), len(test_loader)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "cd0bedb3",
   "metadata": {},
   "outputs": [],
   "source": [
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "b790582f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 500x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(figsize=(5, 5))\n",
    "ax.imshow(graph)\n",
    "plt.savefig(f'{figure_path}/graph_rank1.png')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "804fe847",
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset = SimulationDataset(\n",
    "    data_path, 1, fraction_regimes_to_ignore=None, intervention=True\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "id": "11507b3a",
   "metadata": {},
   "outputs": [],
   "source": [
    "from sklearn.decomposition import PCA\n",
    "pca = PCA(n_components=2)\n",
    "x_pca = pca.fit_transform(dataset.data)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bd1780b8",
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.figure()\n",
    "plt.scatter(x_pca[:, 0], x_pca[:, 1], s=5)\n",
    "plt.xlabel('PCA 1')\n",
    "plt.ylabel('PCA 2')\n",
    "plt.tight_layout()\n",
    "plt.savefig(f'{figure_path}/pca.png')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "bdbad9c8",
   "metadata": {},
   "source": [
    "## 1.2 Define hyperparameters and train the model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "6c56234f",
   "metadata": {},
   "outputs": [],
   "source": [
    "#test part genetic\n",
    "num_vars=graph.shape[0]\n",
    "num_layers=2\n",
    "num_dec_layers=1\n",
    "num_interventions=0\n",
    "num_modules=1\n",
    "hid_dim=50\n",
    "seed=42\n",
    "train_num_epochs = 200\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "d4534b81",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "# leaves:  2\n"
     ]
    }
   ],
   "source": [
    "model = vaeGraph(num_vars,\n",
    "                 num_interventions,\n",
    "                 num_modules,\n",
    "                 num_layers,\n",
    "                 num_dec_layers,\n",
    "                 hid_dim,\n",
    "                 batch_size,\n",
    "                 seed,\n",
    "                 device=device,\n",
    "                 spn_target='factor',\n",
    "                 z_prior={\n",
    "                     'mean': 0.0,\n",
    "                     'std': 1.0\n",
    "                 },\n",
    "                 noise_level=0.01,\n",
    "                 max_copies=8,\n",
    "                 p_conn=0.05,\n",
    "                 sparsity_temp=0.0,\n",
    "                 nonlin='linear').to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "29fc1e94",
   "metadata": {},
   "outputs": [],
   "source": [
    "nn_params = list(model.weights_n2m) + list(model.biases_n2m)\\\n",
    "    + list(model.weights_m2n) + list(model.biases_m2n)\\\n",
    "    + [model.weight_mu, model.weight_std, model.bias_mu, model.bias_std]\\\n",
    "    + [model.log_stds]\\\n",
    "    + [m.weight for m in model.bns_enc] + [m.weight for m in model.bns_dec]\\\n",
    "    + [m.bias for m in model.bns_enc] + [m.bias for m in model.bns_dec]\n",
    "optimizer = torch.optim.Adam(nn_params, lr=1e-4, weight_decay=1e-2)\n",
    "optimizer_fg = torch.optim.Adam(list(model.sample_uv.parameters()), lr=1e-2)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "8aab9aa3",
   "metadata": {},
   "outputs": [],
   "source": [
    "def pred_score(pred, truth):\n",
    "    tp = np.sum((pred > 0) & (truth > 0))\n",
    "    tn = np.sum((pred == 0) & (truth == 0))\n",
    "    fp = np.sum((pred > 0) & (truth == 0))\n",
    "    fn = np.sum((pred == 0) & (truth > 0))\n",
    "    # print(tp, tn, fp, fn)\n",
    "\n",
    "    tpr = tp / (tp + fn)\n",
    "    fpr = fp / (fp + tn)\n",
    "\n",
    "    precision = tp / (tp + fp)\n",
    "    recall = tpr\n",
    "    f1 = 2 * precision * recall / (precision + recall)\n",
    "\n",
    "    return precision, recall, f1\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "d6fa8b91",
   "metadata": {},
   "source": [
    "# Regular Training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "01fbc268",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_2090819/4213278239.py:11: RuntimeWarning: invalid value encountered in long_scalars\n",
      "  precision = tp / (tp + fp)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch1\tAverage Loss: 11868222.68, 2126519.04, Partition Entropy: 0.431\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch2\tAverage Loss: 1963622.19, 1399023.27, Partition Entropy: 0.353\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch3\tAverage Loss: 1240858.26, 1036310.92, Partition Entropy: 0.311\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch4\tAverage Loss: 981127.63, 846403.84, Partition Entropy: 0.281\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch5\tAverage Loss: 833194.63, 742022.46, Partition Entropy: 0.259\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch6\tAverage Loss: 737859.92, 666736.42, Partition Entropy: 0.242\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch7\tAverage Loss: 660155.02, 604808.71, Partition Entropy: 0.226\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch8\tAverage Loss: 586338.69, 552346.85, Partition Entropy: 0.215\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch9\tAverage Loss: 533054.23, 502935.54, Partition Entropy: 0.203\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch10\tAverage Loss: 482363.86, 457873.88, Partition Entropy: 0.195\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch11\tAverage Loss: 435287.93, 414751.63, Partition Entropy: 0.189\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch12\tAverage Loss: 395691.83, 375342.73, Partition Entropy: 0.183\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch13\tAverage Loss: 353239.75, 340509.26, Partition Entropy: 0.181\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch14\tAverage Loss: 316953.97, 308127.51, Partition Entropy: 0.181\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch15\tAverage Loss: 287139.24, 277217.16, Partition Entropy: 0.179\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch16\tAverage Loss: 259724.89, 249337.75, Partition Entropy: 0.180\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch17\tAverage Loss: 231359.03, 224171.63, Partition Entropy: 0.186\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch18\tAverage Loss: 207373.52, 202484.82, Partition Entropy: 0.195\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch19\tAverage Loss: 186560.45, 180794.38, Partition Entropy: 0.204\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch20\tAverage Loss: 166392.84, 161303.20, Partition Entropy: 0.212\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch21\tAverage Loss: 148523.15, 143748.67, Partition Entropy: 0.219\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch22\tAverage Loss: 132617.21, 128042.00, Partition Entropy: 0.226\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch23\tAverage Loss: 117491.13, 113566.45, Partition Entropy: 0.238\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch24\tAverage Loss: 105132.36, 100606.63, Partition Entropy: 0.252\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch25\tAverage Loss: 91890.21, 88714.85, Partition Entropy: 0.271\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch26\tAverage Loss: 80951.33, 78625.68, Partition Entropy: 0.292\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch27\tAverage Loss: 71233.84, 68155.96, Partition Entropy: 0.299\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch28\tAverage Loss: 62154.78, 59217.79, Partition Entropy: 0.306\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch29\tAverage Loss: 53418.94, 51561.61, Partition Entropy: 0.323\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch30\tAverage Loss: 46909.88, 43910.47, Partition Entropy: 0.332\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch31\tAverage Loss: 39184.48, 37637.45, Partition Entropy: 0.344\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch32\tAverage Loss: 32966.94, 31254.59, Partition Entropy: 0.358\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch33\tAverage Loss: 27869.39, 26125.98, Partition Entropy: 0.367\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch34\tAverage Loss: 23032.03, 20965.18, Partition Entropy: 0.375\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch35\tAverage Loss: 18479.02, 16376.58, Partition Entropy: 0.380\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch36\tAverage Loss: 13913.57, 12381.05, Partition Entropy: 0.383\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch37\tAverage Loss: 10538.14, 8701.92, Partition Entropy: 0.387\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch38\tAverage Loss: 6574.43, 5416.40, Partition Entropy: 0.393\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch39\tAverage Loss: 3446.92, 2472.47, Partition Entropy: 0.397\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch40\tAverage Loss: 682.55, -283.77, Partition Entropy: 0.401\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch41\tAverage Loss: -778.77, -2629.08, Partition Entropy: 0.404\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch42\tAverage Loss: -3876.26, -4844.10, Partition Entropy: 0.406\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch43\tAverage Loss: -6033.87, -6889.06, Partition Entropy: 0.407\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch44\tAverage Loss: -7841.83, -8541.42, Partition Entropy: 0.406\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch45\tAverage Loss: -9657.61, -10261.53, Partition Entropy: 0.405\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch46\tAverage Loss: -10504.48, -11603.75, Partition Entropy: 0.406\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch47\tAverage Loss: -12511.24, -13155.32, Partition Entropy: 0.406\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch48\tAverage Loss: -13651.04, -14427.52, Partition Entropy: 0.406\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch49\tAverage Loss: -14967.64, -15395.52, Partition Entropy: 0.409\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch50\tAverage Loss: -15713.12, -16321.44, Partition Entropy: 0.413\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch51\tAverage Loss: -16725.70, -17201.75, Partition Entropy: 0.418\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch52\tAverage Loss: -17484.10, -17995.27, Partition Entropy: 0.422\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch53\tAverage Loss: -18034.15, -18597.41, Partition Entropy: 0.423\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch54\tAverage Loss: -18711.25, -19287.39, Partition Entropy: 0.432\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch55\tAverage Loss: -19419.22, -19868.38, Partition Entropy: 0.429\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch56\tAverage Loss: -20113.77, -20416.20, Partition Entropy: 0.433\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch57\tAverage Loss: -20604.95, -20850.88, Partition Entropy: 0.433\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch58\tAverage Loss: -20881.40, -21267.21, Partition Entropy: 0.441\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch59\tAverage Loss: -21386.12, -21701.72, Partition Entropy: 0.451\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch60\tAverage Loss: -21773.67, -22047.95, Partition Entropy: 0.455\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch61\tAverage Loss: -22044.75, -22428.90, Partition Entropy: 0.463\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch62\tAverage Loss: -22425.38, -22718.67, Partition Entropy: 0.466\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch63\tAverage Loss: -22654.42, -22947.95, Partition Entropy: 0.470\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch64\tAverage Loss: -22873.86, -23256.10, Partition Entropy: 0.485\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch65\tAverage Loss: -23140.58, -23503.13, Partition Entropy: 0.493\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch66\tAverage Loss: -23419.92, -23741.60, Partition Entropy: 0.501\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch67\tAverage Loss: -23609.12, -23908.40, Partition Entropy: 0.502\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch68\tAverage Loss: -23777.06, -24020.03, Partition Entropy: 0.502\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch69\tAverage Loss: -23909.55, -24131.13, Partition Entropy: 0.503\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch70\tAverage Loss: -24058.95, -24210.01, Partition Entropy: 0.505\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch71\tAverage Loss: -24138.42, -24273.07, Partition Entropy: 0.505\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch72\tAverage Loss: -24191.95, -24339.63, Partition Entropy: 0.510\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch73\tAverage Loss: -24316.72, -24405.89, Partition Entropy: 0.512\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch74\tAverage Loss: -24390.83, -24450.03, Partition Entropy: 0.512\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch75\tAverage Loss: -24446.27, -24487.66, Partition Entropy: 0.515\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch76\tAverage Loss: -24498.49, -24547.80, Partition Entropy: 0.519\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch77\tAverage Loss: -24581.20, -24587.68, Partition Entropy: 0.520\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch78\tAverage Loss: -24657.31, -24631.41, Partition Entropy: 0.522\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch79\tAverage Loss: -24689.59, -24705.81, Partition Entropy: 0.532\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch80\tAverage Loss: -24777.79, -24776.79, Partition Entropy: 0.536\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch81\tAverage Loss: -24827.18, -24908.15, Partition Entropy: 0.539\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch82\tAverage Loss: -24956.32, -25163.29, Partition Entropy: 0.558\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch83\tAverage Loss: -25320.45, -25567.57, Partition Entropy: 0.581\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch84\tAverage Loss: -25864.48, -26097.53, Partition Entropy: 0.612\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch85\tAverage Loss: -26441.95, -26578.12, Partition Entropy: 0.620\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch86\tAverage Loss: -26730.75, -26830.65, Partition Entropy: 0.629\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch87\tAverage Loss: -27013.85, -27235.15, Partition Entropy: 0.647\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch88\tAverage Loss: -27297.57, -27484.30, Partition Entropy: 0.649\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch89\tAverage Loss: -27487.30, -27684.52, Partition Entropy: 0.657\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch90\tAverage Loss: -27868.24, -27926.37, Partition Entropy: 0.664\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch91\tAverage Loss: -27624.07, -27955.02, Partition Entropy: 0.662\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch92\tAverage Loss: -28016.55, -28030.95, Partition Entropy: 0.667\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch93\tAverage Loss: -28098.12, -28170.77, Partition Entropy: 0.672\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch94\tAverage Loss: -28195.16, -28249.04, Partition Entropy: 0.680\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch95\tAverage Loss: -28234.97, -28312.60, Partition Entropy: 0.683\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch96\tAverage Loss: -28247.04, -28421.85, Partition Entropy: 0.683\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch97\tAverage Loss: -28533.68, -28520.97, Partition Entropy: 0.685\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch98\tAverage Loss: -28486.66, -28532.85, Partition Entropy: 0.685\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch99\tAverage Loss: -28615.06, -28650.61, Partition Entropy: 0.692\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch100\tAverage Loss: -28668.53, -28658.29, Partition Entropy: 0.691\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch101\tAverage Loss: -28706.50, -28677.56, Partition Entropy: 0.691\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch102\tAverage Loss: -28683.17, -28704.34, Partition Entropy: 0.692\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch103\tAverage Loss: -28758.31, -28719.51, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch104\tAverage Loss: -28523.67, -28428.65, Partition Entropy: 0.681\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch105\tAverage Loss: -28624.50, -28611.00, Partition Entropy: 0.691\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch106\tAverage Loss: -28794.42, -28694.68, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch107\tAverage Loss: -28708.65, -28707.41, Partition Entropy: 0.692\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch108\tAverage Loss: -28779.70, -28717.72, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch109\tAverage Loss: -28788.89, -28725.83, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch110\tAverage Loss: -28406.15, -28603.81, Partition Entropy: 0.685\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch111\tAverage Loss: -28666.27, -28536.78, Partition Entropy: 0.690\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch112\tAverage Loss: -28746.64, -28671.15, Partition Entropy: 0.692\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch113\tAverage Loss: -28786.52, -28712.53, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch114\tAverage Loss: -28788.64, -28688.97, Partition Entropy: 0.691\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch115\tAverage Loss: -28825.58, -28725.32, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch116\tAverage Loss: -28775.61, -28736.11, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch117\tAverage Loss: -28748.99, -28734.06, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch118\tAverage Loss: -28804.00, -28732.34, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch119\tAverage Loss: -28821.66, -28737.00, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch120\tAverage Loss: -28833.38, -28747.89, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch121\tAverage Loss: -28800.77, -28750.69, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch122\tAverage Loss: -28782.75, -28754.18, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch123\tAverage Loss: -28821.07, -28756.05, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch124\tAverage Loss: -28764.55, -28649.67, Partition Entropy: 0.687\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch125\tAverage Loss: -28824.03, -28723.33, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch126\tAverage Loss: -28847.32, -28755.88, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch127\tAverage Loss: -28851.09, -28756.86, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch128\tAverage Loss: -28757.74, -28755.91, Partition Entropy: 0.692\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch129\tAverage Loss: -28840.18, -28767.79, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch130\tAverage Loss: -28868.37, -28770.25, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch131\tAverage Loss: -28822.85, -28768.49, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch132\tAverage Loss: -28873.83, -28777.90, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch133\tAverage Loss: -28879.38, -28785.37, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch134\tAverage Loss: -28885.27, -28788.96, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch135\tAverage Loss: -28874.37, -28785.79, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch136\tAverage Loss: -28280.58, -28742.11, Partition Entropy: 0.690\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch137\tAverage Loss: -28844.41, -28764.47, Partition Entropy: 0.692\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch138\tAverage Loss: -28890.21, -28789.91, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch139\tAverage Loss: -28899.57, -28801.32, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch140\tAverage Loss: -28907.89, -28807.18, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch141\tAverage Loss: -28887.73, -28809.72, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch142\tAverage Loss: -28911.56, -28813.01, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch143\tAverage Loss: -28921.24, -28818.18, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch144\tAverage Loss: -28919.83, -28817.79, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch145\tAverage Loss: -28925.49, -28822.88, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch146\tAverage Loss: -28932.66, -28827.72, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch147\tAverage Loss: -28937.03, -28832.17, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch148\tAverage Loss: -28940.31, -28838.33, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch149\tAverage Loss: -28937.06, -28833.65, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch150\tAverage Loss: -28905.78, -28787.87, Partition Entropy: 0.691\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch151\tAverage Loss: -28938.61, -28832.51, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch152\tAverage Loss: -28955.76, -28848.67, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch153\tAverage Loss: -28968.56, -28851.16, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch154\tAverage Loss: -28963.34, -28859.45, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch155\tAverage Loss: -28973.39, -28864.23, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch156\tAverage Loss: -28986.06, -28873.11, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch157\tAverage Loss: -28987.68, -28879.49, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch158\tAverage Loss: -28997.33, -28886.39, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch159\tAverage Loss: -29010.35, -28896.30, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch160\tAverage Loss: -29018.58, -28903.85, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch161\tAverage Loss: -29037.73, -28919.71, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch162\tAverage Loss: -29052.40, -28934.17, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch163\tAverage Loss: -29069.85, -28952.95, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch164\tAverage Loss: -29094.50, -28973.18, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch165\tAverage Loss: -29119.30, -28997.58, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch166\tAverage Loss: -29154.70, -29031.66, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch167\tAverage Loss: -29196.48, -29072.03, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch168\tAverage Loss: -29246.28, -29121.47, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch169\tAverage Loss: -29305.89, -29184.25, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch170\tAverage Loss: -29391.04, -29281.75, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch171\tAverage Loss: -29531.03, -29406.72, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch172\tAverage Loss: -29599.66, -29450.91, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch173\tAverage Loss: -29605.95, -29463.91, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch174\tAverage Loss: -29607.49, -29467.68, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch175\tAverage Loss: -29608.06, -29468.51, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch176\tAverage Loss: -29608.37, -29469.30, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch177\tAverage Loss: -29608.54, -29468.78, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch178\tAverage Loss: -29608.53, -29469.47, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch179\tAverage Loss: -29608.71, -29469.21, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch180\tAverage Loss: -29609.00, -29469.87, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n"
     ]
    }
   ],
   "source": [
    "train_loss_list=[]\n",
    "eval_loss_list = []\n",
    "loss_comp = {\n",
    "    're': [],\n",
    "    'uv_kl': [],\n",
    "    'z_kl': [],\n",
    "    'l1_reg': [],\n",
    "    'precision': [],\n",
    "    'recall': [],\n",
    "    'f1': []\n",
    "}\n",
    "\n",
    "stop_counter = 0\n",
    "patience = 5\n",
    "stop_thred = 1.0\n",
    "prev_loss = 0\n",
    "memory = 0.25\n",
    "\n",
    "for epoch in range(train_num_epochs):\n",
    "    # kl_coeff = min(1.0, 0.01 * epoch)\n",
    "    kl_coeff = 1.0\n",
    "    model.train()\n",
    "    \n",
    "    train_loss = 0\n",
    "    eval_loss = 0\n",
    "    re_loss, uv_kl, z_kl, l1_reg = 0, 0, 0, 0\n",
    "    for inputs in train_loader:\n",
    "        data, masks, regimes=inputs\n",
    "        batch_size = data.shape[0]\n",
    "        data=data.to(device)\n",
    "        masks=masks.to(device)\n",
    "        regimes=regimes.to(device)\n",
    "        \n",
    "        _re_loss, _uv_kl, _z_kl, _l1_reg = model.losses(data, masks)\n",
    "        loss = _re_loss + kl_coeff * _uv_kl + kl_coeff * _z_kl \n",
    "\n",
    "        train_loss += loss.detach().cpu().item() * (batch_size / n_train)\n",
    "        re_loss += _re_loss.detach().cpu().item() * (batch_size / n_train)\n",
    "        uv_kl += _uv_kl.detach().cpu().item() * (batch_size / n_train)\n",
    "        z_kl += _z_kl.detach().cpu().item() * (batch_size / n_train)\n",
    "        l1_reg += _l1_reg.detach().cpu().item() * (batch_size / n_train)\n",
    "\n",
    "\n",
    "        optimizer.zero_grad()\n",
    "        optimizer_fg.zero_grad()\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        optimizer_fg.step()\n",
    "\n",
    "    train_loss_list.append(train_loss)\n",
    "    loss_comp['re'].append(re_loss)\n",
    "    loss_comp['uv_kl'].append(uv_kl)\n",
    "    loss_comp['z_kl'].append(z_kl)\n",
    "    loss_comp['l1_reg'].append(l1_reg)\n",
    "        \n",
    "    # validation\n",
    "    model.eval()\n",
    "    with torch.no_grad():\n",
    "        for inputs in val_loader:\n",
    "            data,masks,regimes=inputs\n",
    "            batch_size = data.shape[0]\n",
    "            data = data.to(device)\n",
    "            masks = masks.to(device)\n",
    "            regimes = regimes.to(device)\n",
    "            \n",
    "            _re_loss, _uv_kl, _z_kl, _l1_reg = model.losses(data, masks)\n",
    "            loss = _re_loss + kl_coeff * _uv_kl + kl_coeff * _z_kl \n",
    "            eval_loss += loss.detach().cpu().item() * (batch_size / n_val)\n",
    "\n",
    "    eval_loss_list.append(eval_loss)\n",
    "    # Smoothing\n",
    "    if epoch > 0:\n",
    "        eval_loss = prev_loss * memory + eval_loss * (1 - memory)\n",
    "    \n",
    "    # Graph evaluation\n",
    "    with torch.no_grad():\n",
    "        n2m, m2n = model.sample_uv.fg.sample_deterministic(hard=True)\n",
    "        n2m = n2m.squeeze().detach().cpu().numpy()\n",
    "        m2n = m2n.squeeze().detach().cpu().numpy()\n",
    "        graph_pred = (np.matmul(n2m, m2n.T) > 0).astype(int)\n",
    "    precision, recall, f1 = pred_score(graph_pred, graph)\n",
    "    loss_comp['precision'].append(precision)\n",
    "    loss_comp['recall'].append(recall)\n",
    "    loss_comp['f1'].append(f1)\n",
    "    \n",
    "    print(f\"Epoch{epoch + 1}\\tAverage Loss: {train_loss:.2f}, {eval_loss:.2f}, Partition Entropy: {model.sample_uv.fg.entropy_y():.3f}\")\n",
    "    print(f\"\\tGraph prediction: precision={precision:.3f}, recall={recall:.3f}, f1 score={f1:.3f}\\n\")\n",
    "    # Update early stopping variables\n",
    "    if prev_loss - eval_loss < stop_thred:\n",
    "        stop_counter += 1\n",
    "    else:\n",
    "        stop_counter = 0\n",
    "    prev_loss = eval_loss\n",
    "    \n",
    "    if stop_counter > patience:\n",
    "        break"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "5ed329e7",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.9326995490074173\n"
     ]
    }
   ],
   "source": [
    "mse = 0\n",
    "with torch.no_grad():\n",
    "    for inputs in train_loader:\n",
    "        data,masks,regimes=inputs\n",
    "        batch_size = data.shape[0]\n",
    "        data = data.to(device)\n",
    "        masks = masks.to(device)\n",
    "        regimes = regimes.to(device)\n",
    "        _mse = model.mse(data, masks)\n",
    "        mse += _mse.detach().cpu().item() * (batch_size/n_train)\n",
    "print(mse)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "7be0723c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.9192088588028187\n"
     ]
    }
   ],
   "source": [
    "mse = 0\n",
    "with torch.no_grad():\n",
    "    for inputs in val_loader:\n",
    "        data,masks,regimes=inputs\n",
    "        batch_size = data.shape[0]\n",
    "        data = data.to(device)\n",
    "        masks = masks.to(device)\n",
    "        regimes = regimes.to(device)\n",
    "        _mse = model.mse(data, masks)\n",
    "        mse += _mse.detach().cpu().item() * (batch_size/n_val)\n",
    "print(mse)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "cb80d34c",
   "metadata": {},
   "source": [
    "# Train with the true factor graph"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "080e31fc",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_2516475/4213278239.py:11: RuntimeWarning: invalid value encountered in long_scalars\n",
      "  precision = tp / (tp + fp)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch1\tAverage Loss: 86448206.60, 19908492.32, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch2\tAverage Loss: 30014641.22, 11146702.93, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch3\tAverage Loss: 19983659.02, 6376597.99, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch4\tAverage Loss: 14437191.81, 3955546.20, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch5\tAverage Loss: 10681483.42, 2636524.27, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch6\tAverage Loss: 7979481.87, 1856966.51, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch7\tAverage Loss: 5918483.41, 1368402.66, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch8\tAverage Loss: 4410841.47, 1049353.47, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch9\tAverage Loss: 3270904.55, 830195.29, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch10\tAverage Loss: 2467418.46, 680477.45, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch11\tAverage Loss: 1850789.59, 572594.38, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch12\tAverage Loss: 1362600.74, 488950.74, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch13\tAverage Loss: 1111158.46, 426576.04, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch14\tAverage Loss: 775461.75, 378752.58, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch15\tAverage Loss: 663727.37, 339311.57, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch16\tAverage Loss: 487582.62, 305312.17, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch17\tAverage Loss: 408307.75, 276402.27, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch18\tAverage Loss: 332082.82, 250391.48, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch19\tAverage Loss: 280858.76, 226870.06, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch20\tAverage Loss: 243420.81, 204774.81, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch21\tAverage Loss: 211727.25, 184511.91, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch22\tAverage Loss: 193860.72, 165563.55, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch23\tAverage Loss: 163798.25, 148232.22, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch24\tAverage Loss: 148143.82, 132240.12, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch25\tAverage Loss: 132917.09, 117216.68, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch26\tAverage Loss: 112222.83, 103555.03, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch27\tAverage Loss: 99277.40, 91115.42, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch28\tAverage Loss: 85982.25, 79671.25, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch29\tAverage Loss: 74293.09, 69325.07, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch30\tAverage Loss: 64246.34, 59871.49, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch31\tAverage Loss: 55067.80, 51311.53, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch32\tAverage Loss: 46750.14, 43477.73, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch33\tAverage Loss: 38886.81, 36397.45, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch34\tAverage Loss: 32161.33, 29965.33, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch35\tAverage Loss: 25719.11, 24141.27, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch36\tAverage Loss: 20297.13, 18824.56, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch37\tAverage Loss: 15498.84, 14056.52, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch38\tAverage Loss: 10473.69, 9696.71, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch39\tAverage Loss: 7270.18, 5757.52, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch40\tAverage Loss: 3016.01, 2233.49, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch41\tAverage Loss: -188.43, -969.05, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch42\tAverage Loss: -3317.31, -3864.86, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch43\tAverage Loss: -6102.82, -6480.49, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch44\tAverage Loss: -8412.88, -8853.29, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch45\tAverage Loss: -10562.69, -10973.97, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch46\tAverage Loss: -12404.08, -12903.53, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch47\tAverage Loss: -14479.09, -14630.78, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch48\tAverage Loss: -16068.93, -16179.56, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch49\tAverage Loss: -17495.48, -17586.18, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch50\tAverage Loss: -18712.77, -18857.33, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch51\tAverage Loss: -19812.36, -19996.39, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch52\tAverage Loss: -20871.90, -21016.57, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch53\tAverage Loss: -21913.72, -21923.74, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch54\tAverage Loss: -22754.33, -22735.12, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch55\tAverage Loss: -23489.37, -23471.35, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch56\tAverage Loss: -24177.38, -24138.44, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch57\tAverage Loss: -24739.76, -24735.90, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch58\tAverage Loss: -25309.52, -25268.95, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch59\tAverage Loss: -25776.36, -25741.85, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch60\tAverage Loss: -26229.86, -26167.62, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch61\tAverage Loss: -26608.67, -26556.13, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch62\tAverage Loss: -26977.33, -26901.68, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch63\tAverage Loss: -27288.50, -27215.56, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch64\tAverage Loss: -27576.27, -27501.44, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch65\tAverage Loss: -27836.96, -27749.93, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch66\tAverage Loss: -28038.21, -27965.61, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch67\tAverage Loss: -28239.44, -28148.25, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch68\tAverage Loss: -28371.82, -28298.78, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch69\tAverage Loss: -28530.12, -28431.97, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch70\tAverage Loss: -28649.27, -28546.67, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch71\tAverage Loss: -28756.68, -28648.79, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch72\tAverage Loss: -28854.21, -28740.76, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch73\tAverage Loss: -28950.48, -28828.48, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch74\tAverage Loss: -29030.24, -28911.11, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch75\tAverage Loss: -29103.53, -28981.94, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch76\tAverage Loss: -29171.44, -29043.02, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch77\tAverage Loss: -29234.02, -29096.72, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch78\tAverage Loss: -29272.04, -29139.65, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch79\tAverage Loss: -29317.18, -29174.71, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch80\tAverage Loss: -29348.85, -29203.07, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch81\tAverage Loss: -29375.75, -29224.21, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch82\tAverage Loss: -29394.41, -29245.28, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch83\tAverage Loss: -29416.16, -29263.17, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch84\tAverage Loss: -29429.59, -29278.43, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch85\tAverage Loss: -29444.08, -29286.11, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch86\tAverage Loss: -29457.07, -29300.10, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch87\tAverage Loss: -29472.82, -29310.91, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch88\tAverage Loss: -29484.90, -29317.04, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch89\tAverage Loss: -29492.12, -29323.44, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch90\tAverage Loss: -29497.65, -29329.35, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch91\tAverage Loss: -29504.60, -29334.20, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch92\tAverage Loss: -29509.42, -29343.50, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch93\tAverage Loss: -29518.63, -29347.03, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch94\tAverage Loss: -29525.17, -29353.72, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch95\tAverage Loss: -29531.89, -29357.05, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch96\tAverage Loss: -29536.59, -29358.01, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch97\tAverage Loss: -29539.65, -29367.09, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch98\tAverage Loss: -29546.71, -29365.04, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch99\tAverage Loss: -29550.65, -29374.31, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch100\tAverage Loss: -29558.36, -29381.03, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch101\tAverage Loss: -29560.31, -29385.46, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch102\tAverage Loss: -29568.66, -29384.92, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch103\tAverage Loss: -29571.89, -29391.96, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch104\tAverage Loss: -29576.88, -29389.44, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch105\tAverage Loss: -29580.89, -29402.29, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch106\tAverage Loss: -29588.05, -29407.36, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch107\tAverage Loss: -29593.03, -29411.28, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch108\tAverage Loss: -29595.33, -29415.80, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch109\tAverage Loss: -29602.93, -29409.00, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch110\tAverage Loss: -29605.49, -29423.38, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch111\tAverage Loss: -29611.05, -29431.03, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch112\tAverage Loss: -29614.37, -29434.98, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch113\tAverage Loss: -29621.93, -29436.67, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch114\tAverage Loss: -29623.42, -29439.96, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch115\tAverage Loss: -29632.76, -29444.70, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch116\tAverage Loss: -29636.78, -29452.73, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch117\tAverage Loss: -29637.06, -29454.94, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch118\tAverage Loss: -29644.11, -29460.28, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch119\tAverage Loss: -29651.16, -29466.62, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch120\tAverage Loss: -29654.35, -29472.62, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch121\tAverage Loss: -29661.65, -29476.97, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch122\tAverage Loss: -29668.00, -29478.85, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch123\tAverage Loss: -29671.38, -29489.38, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch124\tAverage Loss: -29679.82, -29495.24, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch125\tAverage Loss: -29683.86, -29492.77, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch126\tAverage Loss: -29690.95, -29501.29, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch127\tAverage Loss: -29699.29, -29506.96, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch128\tAverage Loss: -29700.61, -29506.87, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch129\tAverage Loss: -29708.59, -29516.38, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch130\tAverage Loss: -29714.68, -29523.54, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch131\tAverage Loss: -29718.21, -29525.29, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch132\tAverage Loss: -29723.68, -29534.16, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch133\tAverage Loss: -29729.10, -29540.64, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch134\tAverage Loss: -29731.62, -29547.44, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch135\tAverage Loss: -29740.72, -29548.64, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch136\tAverage Loss: -29738.84, -29547.23, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch137\tAverage Loss: -29744.68, -29548.71, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch138\tAverage Loss: -29746.30, -29555.64, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch139\tAverage Loss: -29749.16, -29560.54, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch140\tAverage Loss: -29750.34, -29559.96, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch141\tAverage Loss: -29753.31, -29550.56, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch142\tAverage Loss: -29753.66, -29559.35, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch143\tAverage Loss: -29751.76, -29560.44, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch144\tAverage Loss: -29754.03, -29560.27, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch145\tAverage Loss: -29753.10, -29563.27, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch146\tAverage Loss: -29753.23, -29563.74, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch147\tAverage Loss: -29754.89, -29562.55, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch148\tAverage Loss: -29759.35, -29563.09, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch149\tAverage Loss: -29755.96, -29563.76, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch150\tAverage Loss: -29754.90, -29565.41, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch151\tAverage Loss: -29758.24, -29563.85, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch152\tAverage Loss: -29754.09, -29556.04, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch153\tAverage Loss: -29754.78, -29562.22, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch154\tAverage Loss: -29759.41, -29565.34, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch155\tAverage Loss: -29755.41, -29566.40, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch156\tAverage Loss: -29758.71, -29551.57, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch157\tAverage Loss: -29757.83, -29558.23, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch158\tAverage Loss: -29755.20, -29564.57, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch159\tAverage Loss: -29756.20, -29566.16, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch160\tAverage Loss: -29755.24, -29566.00, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch161\tAverage Loss: -29758.14, -29563.64, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch162\tAverage Loss: -29758.35, -29562.72, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch163\tAverage Loss: -29759.24, -29563.91, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch164\tAverage Loss: -29758.20, -29561.56, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch165\tAverage Loss: -29756.01, -29561.70, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch166\tAverage Loss: -29761.75, -29564.03, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch167\tAverage Loss: -29759.84, -29557.70, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch168\tAverage Loss: -29759.01, -29563.00, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch169\tAverage Loss: -29756.86, -29561.82, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch170\tAverage Loss: -29758.55, -29566.12, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch171\tAverage Loss: -29755.72, -29562.69, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch172\tAverage Loss: -29758.81, -29563.97, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch173\tAverage Loss: -29758.13, -29562.06, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch174\tAverage Loss: -29758.27, -29564.94, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch175\tAverage Loss: -29757.36, -29560.09, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch176\tAverage Loss: -29757.16, -29563.20, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch177\tAverage Loss: -29759.68, -29565.31, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch178\tAverage Loss: -29759.81, -29561.90, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch179\tAverage Loss: -29755.91, -29563.52, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch180\tAverage Loss: -29758.88, -29564.77, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch181\tAverage Loss: -29759.43, -29562.65, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch182\tAverage Loss: -29758.06, -29561.94, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch183\tAverage Loss: -29759.46, -29561.79, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch184\tAverage Loss: -29758.53, -29561.88, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch185\tAverage Loss: -29759.49, -29565.26, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch186\tAverage Loss: -29757.85, -29562.43, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch187\tAverage Loss: -29757.94, -29563.85, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch188\tAverage Loss: -29757.21, -29567.03, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch189\tAverage Loss: -29757.81, -29564.29, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch190\tAverage Loss: -29761.29, -29558.98, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch191\tAverage Loss: -29758.24, -29560.38, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch192\tAverage Loss: -29761.86, -29553.75, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch193\tAverage Loss: -29762.56, -29559.50, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch194\tAverage Loss: -29757.73, -29562.45, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch195\tAverage Loss: -29759.72, -29562.28, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch196\tAverage Loss: -29758.12, -29553.80, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch197\tAverage Loss: -29761.61, -29560.40, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch198\tAverage Loss: -29760.07, -29562.96, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch199\tAverage Loss: -29757.49, -29560.17, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch200\tAverage Loss: -29759.88, -29560.81, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch201\tAverage Loss: -29760.59, -29556.85, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch202\tAverage Loss: -29757.43, -29559.64, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch203\tAverage Loss: -29757.82, -29555.31, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch204\tAverage Loss: -29755.86, -29558.13, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch205\tAverage Loss: -29760.07, -29561.94, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch206\tAverage Loss: -29756.90, -29561.75, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch207\tAverage Loss: -29759.31, -29555.13, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch208\tAverage Loss: -29756.61, -29555.64, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch209\tAverage Loss: -29759.47, -29561.64, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch210\tAverage Loss: -29760.24, -29556.40, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch211\tAverage Loss: -29758.47, -29559.56, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch212\tAverage Loss: -29759.94, -29549.14, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch213\tAverage Loss: -29759.38, -29559.85, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch214\tAverage Loss: -29758.34, -29559.36, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch215\tAverage Loss: -29756.63, -29562.60, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch216\tAverage Loss: -29759.48, -29557.55, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch217\tAverage Loss: -29760.46, -29555.12, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch218\tAverage Loss: -29760.69, -29554.62, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch219\tAverage Loss: -29761.36, -29552.97, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch220\tAverage Loss: -29759.87, -29561.12, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch221\tAverage Loss: -29759.57, -29561.87, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch222\tAverage Loss: -29757.61, -29561.57, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch223\tAverage Loss: -29760.69, -29553.22, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch224\tAverage Loss: -29761.54, -29556.06, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch225\tAverage Loss: -29759.62, -29559.97, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch226\tAverage Loss: -29761.34, -29560.88, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch227\tAverage Loss: -29760.42, -29551.22, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch228\tAverage Loss: -29760.94, -29559.72, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch229\tAverage Loss: -29760.09, -29562.04, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch230\tAverage Loss: -29761.18, -29562.08, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch231\tAverage Loss: -29759.78, -29562.56, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch232\tAverage Loss: -29760.84, -29564.92, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch233\tAverage Loss: -29759.18, -29561.74, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch234\tAverage Loss: -29759.07, -29563.92, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch235\tAverage Loss: -29760.21, -29561.85, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch236\tAverage Loss: -29759.74, -29552.61, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch237\tAverage Loss: -29760.42, -29558.55, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch238\tAverage Loss: -29762.60, -29557.96, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch239\tAverage Loss: -29759.24, -29560.86, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch240\tAverage Loss: -29759.93, -29558.18, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch241\tAverage Loss: -29760.83, -29549.27, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch242\tAverage Loss: -29760.25, -29559.16, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch243\tAverage Loss: -29760.42, -29558.77, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch244\tAverage Loss: -29761.72, -29549.26, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch245\tAverage Loss: -29760.13, -29558.94, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch246\tAverage Loss: -29762.43, -29561.35, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch247\tAverage Loss: -29759.28, -29563.42, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch248\tAverage Loss: -29759.53, -29562.59, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch249\tAverage Loss: -29763.04, -29561.16, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch250\tAverage Loss: -29758.54, -29549.88, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch251\tAverage Loss: -29758.98, -29556.33, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch252\tAverage Loss: -29761.10, -29555.93, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch253\tAverage Loss: -29759.11, -29561.93, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch254\tAverage Loss: -29759.86, -29562.58, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch255\tAverage Loss: -29760.51, -29563.46, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch256\tAverage Loss: -29757.66, -29564.10, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch257\tAverage Loss: -29760.15, -29558.38, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch258\tAverage Loss: -29762.03, -29562.50, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch259\tAverage Loss: -29759.23, -29561.32, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch260\tAverage Loss: -29759.32, -29555.47, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch261\tAverage Loss: -29761.33, -29560.97, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch262\tAverage Loss: -29760.05, -29555.30, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch263\tAverage Loss: -29760.87, -29556.62, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch264\tAverage Loss: -29760.60, -29561.65, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch265\tAverage Loss: -29758.70, -29561.28, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch266\tAverage Loss: -29759.14, -29553.89, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch267\tAverage Loss: -29760.84, -29560.87, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch268\tAverage Loss: -29759.85, -29560.84, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch269\tAverage Loss: -29761.66, -29560.30, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch270\tAverage Loss: -29761.73, -29558.00, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch271\tAverage Loss: -29756.95, -29559.55, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch272\tAverage Loss: -29759.95, -29555.37, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch273\tAverage Loss: -29759.19, -29560.75, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch274\tAverage Loss: -29760.61, -29561.75, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch275\tAverage Loss: -29762.68, -29547.90, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch276\tAverage Loss: -29762.59, -29557.17, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch277\tAverage Loss: -29759.16, -29555.50, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch278\tAverage Loss: -29760.26, -29558.99, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch279\tAverage Loss: -29758.49, -29557.91, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch280\tAverage Loss: -29760.67, -29560.69, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch281\tAverage Loss: -29761.05, -29555.87, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch282\tAverage Loss: -29762.31, -29556.74, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch283\tAverage Loss: -29758.66, -29562.40, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch284\tAverage Loss: -29760.64, -29562.66, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch285\tAverage Loss: -29762.95, -29562.78, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch286\tAverage Loss: -29758.31, -29552.71, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch287\tAverage Loss: -29762.65, -29559.81, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch288\tAverage Loss: -29761.64, -29560.55, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch289\tAverage Loss: -29759.13, -29560.95, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch290\tAverage Loss: -29758.00, -29560.95, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch291\tAverage Loss: -29760.50, -29561.94, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch292\tAverage Loss: -29762.75, -29559.34, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch293\tAverage Loss: -29758.16, -29561.92, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch294\tAverage Loss: -29758.23, -29558.98, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch295\tAverage Loss: -29760.85, -29560.08, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch296\tAverage Loss: -29760.20, -29561.48, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch297\tAverage Loss: -29761.97, -29562.44, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch298\tAverage Loss: -29760.19, -29560.32, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch299\tAverage Loss: -29758.12, -29563.07, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch300\tAverage Loss: -29761.14, -29563.52, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n"
     ]
    }
   ],
   "source": [
    "train_loss_list=[]\n",
    "eval_loss_list = []\n",
    "loss_comp = {\n",
    "    're': [],\n",
    "    'uv_kl': [],\n",
    "    'z_kl': [],\n",
    "    'l1_reg': [],\n",
    "    'precision': [],\n",
    "    'recall': [],\n",
    "    'f1': []\n",
    "}\n",
    "\n",
    "stop_counter = 0\n",
    "patience = 5\n",
    "stop_thred = 1.0\n",
    "prev_loss = 0\n",
    "memory = 0.25\n",
    "\n",
    "for epoch in range(train_num_epochs):\n",
    "    # kl_coeff = min(1.0, 0.01 * epoch)\n",
    "    kl_coeff = 1.0\n",
    "    model.train()\n",
    "    \n",
    "    train_loss = 0\n",
    "    eval_loss = 0\n",
    "    re_loss, uv_kl, z_kl, l1_reg = 0, 0, 0, 0\n",
    "    for inputs in train_loader:\n",
    "        data, masks, regimes=inputs\n",
    "        batch_size = data.shape[0]\n",
    "        data=data.to(device)\n",
    "        masks=masks.to(device)\n",
    "        regimes=regimes.to(device)\n",
    "        \n",
    "        n2m = torch.tensor(np.tile(U0, (batch_size, 1, num_modules))).float().to(device)\n",
    "        m2n = torch.tensor(np.tile(V0.T, (batch_size, 1, num_modules))).float().to(device)\n",
    "        _re_loss, _uv_kl, _z_kl, _l1_reg = model.losses(data, masks, n2m, m2n)\n",
    "        loss = _re_loss + kl_coeff * _uv_kl + kl_coeff * _z_kl \n",
    "\n",
    "        train_loss += loss.detach().cpu().item() * (batch_size / n_train)\n",
    "        re_loss += _re_loss.detach().cpu().item() * (batch_size / n_train)\n",
    "        uv_kl += _uv_kl.detach().cpu().item() * (batch_size / n_train)\n",
    "        z_kl += _z_kl.detach().cpu().item() * (batch_size / n_train)\n",
    "        l1_reg += _l1_reg.detach().cpu().item() * (batch_size / n_train)\n",
    "\n",
    "\n",
    "        optimizer.zero_grad()\n",
    "        #optimizer_fg.zero_grad()\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        #optimizer_fg.step()\n",
    "\n",
    "    train_loss_list.append(train_loss)\n",
    "    loss_comp['re'].append(re_loss)\n",
    "    loss_comp['uv_kl'].append(uv_kl)\n",
    "    loss_comp['z_kl'].append(z_kl)\n",
    "    loss_comp['l1_reg'].append(l1_reg)\n",
    "        \n",
    "    # validation\n",
    "    model.eval()\n",
    "    with torch.no_grad():\n",
    "        for inputs in val_loader:\n",
    "            data,masks,regimes=inputs\n",
    "            batch_size = data.shape[0]\n",
    "            data = data.to(device)\n",
    "            masks = masks.to(device)\n",
    "            regimes = regimes.to(device)\n",
    "            n2m = torch.tensor(np.tile(U0, (batch_size, 1, num_modules))).float().to(device)\n",
    "            m2n = torch.tensor(np.tile(V0.T, (batch_size, 1, num_modules))).float().to(device)\n",
    "            _re_loss, _uv_kl, _z_kl, _l1_reg = model.losses(data, masks, n2m, m2n)\n",
    "            loss = _re_loss + kl_coeff * _uv_kl + kl_coeff * _z_kl \n",
    "            eval_loss += loss.detach().cpu().item() * (batch_size / n_val)\n",
    "\n",
    "    eval_loss_list.append(eval_loss)\n",
    "    # Smoothing\n",
    "    if epoch > 0:\n",
    "        eval_loss = prev_loss * memory + eval_loss * (1 - memory)\n",
    "    \n",
    "    # Graph evaluation\n",
    "    with torch.no_grad():\n",
    "        n2m, m2n = model.sample_uv.fg.sample_deterministic()\n",
    "        n2m = n2m.squeeze().detach().cpu().numpy()\n",
    "        m2n = m2n.squeeze().detach().cpu().numpy()\n",
    "        graph_pred = (np.matmul(n2m, m2n.T) > 0).astype(int)\n",
    "    precision, recall, f1 = pred_score(graph_pred, graph)\n",
    "    loss_comp['precision'].append(precision)\n",
    "    loss_comp['recall'].append(recall)\n",
    "    loss_comp['f1'].append(f1)\n",
    "    \n",
    "    print(f\"Epoch{epoch + 1}\\tAverage Loss: {train_loss:.2f}, {eval_loss:.2f}, Partition Entropy: {model.sample_uv.fg.entropy_y():.3f}\")\n",
    "    print(f\"\\tGraph prediction: precision={precision:.3f}, recall={recall:.3f}, f1 score={f1:.3f}\\n\")\n",
    "    # Update early stopping variables\n",
    "    if prev_loss - eval_loss < stop_thred:\n",
    "        stop_counter += 1\n",
    "    else:\n",
    "        stop_counter = 0\n",
    "    prev_loss = eval_loss\n",
    "    \n",
    "    if stop_counter > patience:\n",
    "        break"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "ef3c41ea",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.5500776659965514\n"
     ]
    }
   ],
   "source": [
    "mse = 0\n",
    "with torch.no_grad():\n",
    "    for inputs in train_loader:\n",
    "        data,masks,regimes=inputs\n",
    "        batch_size = data.shape[0]\n",
    "        data = data.to(device)\n",
    "        masks = masks.to(device)\n",
    "        regimes = regimes.to(device)\n",
    "        n2m = torch.tensor(np.tile(U0, (batch_size, 1, 1))).float().to(device)\n",
    "        m2n = torch.tensor(np.tile(V0.T, (batch_size, 1, 1))).float().to(device)\n",
    "        _mse = model.mse(data, masks, n2m, m2n)\n",
    "        mse += _mse.detach().cpu().item() * (batch_size/n_train)\n",
    "print(mse)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "8b264f81",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.5605488901685\n"
     ]
    }
   ],
   "source": [
    "mse = 0\n",
    "with torch.no_grad():\n",
    "    for inputs in val_loader:\n",
    "        data,masks,regimes=inputs\n",
    "        batch_size = data.shape[0]\n",
    "        data = data.to(device)\n",
    "        masks = masks.to(device)\n",
    "        regimes = regimes.to(device)\n",
    "        n2m = torch.tensor(np.tile(U0, (batch_size, 1, 1))).float().to(device)\n",
    "        m2n = torch.tensor(np.tile(V0.T, (batch_size, 1, 1))).float().to(device)\n",
    "        _mse = model.mse(data, masks, n2m, m2n)\n",
    "        mse += _mse.detach().cpu().item() * (batch_size/n_val)\n",
    "print(mse)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "488969b1",
   "metadata": {},
   "source": [
    "# Train with a null graph"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "f54f59cc",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_2516475/4213278239.py:11: RuntimeWarning: invalid value encountered in long_scalars\n",
      "  precision = tp / (tp + fp)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch1\tAverage Loss: -29436.62, -29288.59, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch2\tAverage Loss: -29469.38, -29305.77, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch3\tAverage Loss: -29489.28, -29323.73, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch4\tAverage Loss: -29510.55, -29348.50, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch5\tAverage Loss: -29540.15, -29375.00, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch6\tAverage Loss: -29553.58, -29384.29, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch7\tAverage Loss: -29555.80, -29388.09, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch8\tAverage Loss: -29556.88, -29389.72, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch9\tAverage Loss: -29557.66, -29389.81, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch10\tAverage Loss: -29558.02, -29390.89, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch11\tAverage Loss: -29558.27, -29391.35, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch12\tAverage Loss: -29558.40, -29391.60, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch13\tAverage Loss: -29558.41, -29391.03, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch14\tAverage Loss: -29558.50, -29391.04, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch15\tAverage Loss: -29558.56, -29391.51, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch16\tAverage Loss: -29558.66, -29391.28, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n"
     ]
    }
   ],
   "source": [
    "train_loss_list=[]\n",
    "eval_loss_list = []\n",
    "loss_comp = {\n",
    "    're': [],\n",
    "    'uv_kl': [],\n",
    "    'z_kl': [],\n",
    "    'l1_reg': [],\n",
    "    'precision': [],\n",
    "    'recall': [],\n",
    "    'f1': []\n",
    "}\n",
    "\n",
    "stop_counter = 0\n",
    "patience = 5\n",
    "stop_thred = 1.0\n",
    "prev_loss = 0\n",
    "memory = 0.25\n",
    "\n",
    "for epoch in range(train_num_epochs):\n",
    "    # kl_coeff = min(1.0, 0.01 * epoch)\n",
    "    kl_coeff = 1.0\n",
    "    model.train()\n",
    "    \n",
    "    train_loss = 0\n",
    "    eval_loss = 0\n",
    "    re_loss, uv_kl, z_kl, l1_reg = 0, 0, 0, 0\n",
    "    for inputs in train_loader:\n",
    "        data, masks, regimes=inputs\n",
    "        batch_size = data.shape[0]\n",
    "        data=data.to(device)\n",
    "        masks=masks.to(device)\n",
    "        regimes=regimes.to(device)\n",
    "        \n",
    "        n2m = torch.zeros(batch_size, num_vars, num_modules).float().to(device)\n",
    "        m2n = torch.zeros(batch_size, num_vars, num_modules).float().to(device)\n",
    "        _re_loss, _uv_kl, _z_kl, _l1_reg = model.losses(data, masks, n2m, m2n)\n",
    "        loss = _re_loss + kl_coeff * _uv_kl + kl_coeff * _z_kl \n",
    "\n",
    "        train_loss += loss.detach().cpu().item() * (batch_size / n_train)\n",
    "        re_loss += _re_loss.detach().cpu().item() * (batch_size / n_train)\n",
    "        uv_kl += _uv_kl.detach().cpu().item() * (batch_size / n_train)\n",
    "        z_kl += _z_kl.detach().cpu().item() * (batch_size / n_train)\n",
    "        l1_reg += _l1_reg.detach().cpu().item() * (batch_size / n_train)\n",
    "\n",
    "\n",
    "        optimizer.zero_grad()\n",
    "        #optimizer_fg.zero_grad()\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        #optimizer_fg.step()\n",
    "\n",
    "    train_loss_list.append(train_loss)\n",
    "    loss_comp['re'].append(re_loss)\n",
    "    loss_comp['uv_kl'].append(uv_kl)\n",
    "    loss_comp['z_kl'].append(z_kl)\n",
    "    loss_comp['l1_reg'].append(l1_reg)\n",
    "        \n",
    "    # validation\n",
    "    model.eval()\n",
    "    with torch.no_grad():\n",
    "        for inputs in val_loader:\n",
    "            data,masks,regimes=inputs\n",
    "            batch_size = data.shape[0]\n",
    "            data = data.to(device)\n",
    "            masks = masks.to(device)\n",
    "            regimes = regimes.to(device)\n",
    "            n2m = torch.zeros(batch_size, num_vars, num_modules).float().to(device)\n",
    "            m2n = torch.zeros(batch_size, num_vars, num_modules).float().to(device)\n",
    "            _re_loss, _uv_kl, _z_kl, _l1_reg = model.losses(data, masks, n2m, m2n)\n",
    "            loss = _re_loss + kl_coeff * _uv_kl + kl_coeff * _z_kl \n",
    "            eval_loss += loss.detach().cpu().item() * (batch_size / n_val)\n",
    "\n",
    "    eval_loss_list.append(eval_loss)\n",
    "    # Smoothing\n",
    "    if epoch > 0:\n",
    "        eval_loss = prev_loss * memory + eval_loss * (1 - memory)\n",
    "    \n",
    "    # Graph evaluation\n",
    "    with torch.no_grad():\n",
    "        n2m, m2n = model.sample_uv.fg.sample_deterministic()\n",
    "        n2m = n2m.squeeze().detach().cpu().numpy()\n",
    "        m2n = m2n.squeeze().detach().cpu().numpy()\n",
    "        graph_pred = (np.matmul(n2m, m2n.T) > 0).astype(int)\n",
    "    precision, recall, f1 = pred_score(graph_pred, graph)\n",
    "    loss_comp['precision'].append(precision)\n",
    "    loss_comp['recall'].append(recall)\n",
    "    loss_comp['f1'].append(f1)\n",
    "    \n",
    "    print(f\"Epoch{epoch + 1}\\tAverage Loss: {train_loss:.2f}, {eval_loss:.2f}, Partition Entropy: {model.sample_uv.fg.entropy_y():.3f}\")\n",
    "    print(f\"\\tGraph prediction: precision={precision:.3f}, recall={recall:.3f}, f1 score={f1:.3f}\\n\")\n",
    "    # Update early stopping variables\n",
    "    if prev_loss - eval_loss < stop_thred:\n",
    "        stop_counter += 1\n",
    "    else:\n",
    "        stop_counter = 0\n",
    "    prev_loss = eval_loss\n",
    "    \n",
    "    if stop_counter > patience:\n",
    "        break"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "7c3627e5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.9283872709274303\n"
     ]
    }
   ],
   "source": [
    "mse = 0\n",
    "with torch.no_grad():\n",
    "    for inputs in train_loader:\n",
    "        data,masks,regimes=inputs\n",
    "        batch_size = data.shape[0]\n",
    "        data = data.to(device)\n",
    "        masks = masks.to(device)\n",
    "        regimes = regimes.to(device)\n",
    "        n2m = torch.zeros(batch_size, num_vars, num_modules).float().to(device)\n",
    "        m2n = torch.zeros(batch_size, num_vars, num_modules).float().to(device)\n",
    "        _mse = model.mse(data, masks, n2m, m2n)\n",
    "        mse += _mse.detach().cpu().item() * (batch_size/n_train)\n",
    "print(mse)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "a3072367",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.9364705567906617\n"
     ]
    }
   ],
   "source": [
    "mse = 0\n",
    "with torch.no_grad():\n",
    "    for inputs in val_loader:\n",
    "        data,masks,regimes=inputs\n",
    "        batch_size = data.shape[0]\n",
    "        data = data.to(device)\n",
    "        masks = masks.to(device)\n",
    "        regimes = regimes.to(device)\n",
    "        n2m = torch.zeros(batch_size, num_vars, num_modules).float().to(device)\n",
    "        m2n = torch.zeros(batch_size, num_vars, num_modules).float().to(device)\n",
    "        _mse = model.mse(data, masks, n2m, m2n)\n",
    "        mse += _mse.detach().cpu().item() * (batch_size/n_val)\n",
    "print(mse)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c6e48cd2",
   "metadata": {},
   "source": [
    "# Train with randomly sampled graphs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "b908e8fe",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_2516475/4213278239.py:11: RuntimeWarning: invalid value encountered in long_scalars\n",
      "  precision = tp / (tp + fp)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch1\tAverage Loss: -29558.61, -29391.12, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch2\tAverage Loss: -29558.61, -29391.39, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch3\tAverage Loss: -29558.64, -29391.27, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch4\tAverage Loss: -29558.69, -29391.24, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch5\tAverage Loss: -29558.70, -29391.40, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch6\tAverage Loss: -29558.83, -29391.34, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n",
      "Epoch7\tAverage Loss: -29558.86, -29391.40, Partition Entropy: 0.693\n",
      "\tGraph prediction: precision=nan, recall=0.000, f1 score=nan\n",
      "\n"
     ]
    }
   ],
   "source": [
    "train_loss_list=[]\n",
    "eval_loss_list = []\n",
    "loss_comp = {\n",
    "    're': [],\n",
    "    'uv_kl': [],\n",
    "    'z_kl': [],\n",
    "    'l1_reg': [],\n",
    "    'precision': [],\n",
    "    'recall': [],\n",
    "    'f1': []\n",
    "}\n",
    "\n",
    "stop_counter = 0\n",
    "patience = 5\n",
    "stop_thred = 1.0\n",
    "prev_loss = 0\n",
    "memory = 0.25\n",
    "\n",
    "for epoch in range(train_num_epochs):\n",
    "    # kl_coeff = min(1.0, 0.01 * epoch)\n",
    "    kl_coeff = 1.0\n",
    "    model.train()\n",
    "    \n",
    "    train_loss = 0\n",
    "    eval_loss = 0\n",
    "    re_loss, uv_kl, z_kl, l1_reg = 0, 0, 0, 0\n",
    "    for inputs in train_loader:\n",
    "        data, masks, regimes=inputs\n",
    "        batch_size = data.shape[0]\n",
    "        data=data.to(device)\n",
    "        masks=masks.to(device)\n",
    "        regimes=regimes.to(device)\n",
    "        \n",
    "        n2m = torch.bernoulli(torch.ones(batch_size, num_vars, num_modules).float()).float().to(device)\n",
    "        m2n = 1 - n2m\n",
    "        _re_loss, _uv_kl, _z_kl, _l1_reg = model.losses(data, masks, n2m, m2n)\n",
    "        loss = _re_loss + kl_coeff * _uv_kl + kl_coeff * _z_kl \n",
    "\n",
    "        train_loss += loss.detach().cpu().item() * (batch_size / n_train)\n",
    "        re_loss += _re_loss.detach().cpu().item() * (batch_size / n_train)\n",
    "        uv_kl += _uv_kl.detach().cpu().item() * (batch_size / n_train)\n",
    "        z_kl += _z_kl.detach().cpu().item() * (batch_size / n_train)\n",
    "        l1_reg += _l1_reg.detach().cpu().item() * (batch_size / n_train)\n",
    "\n",
    "\n",
    "        optimizer.zero_grad()\n",
    "        #optimizer_fg.zero_grad()\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "        #optimizer_fg.step()\n",
    "\n",
    "    train_loss_list.append(train_loss)\n",
    "    loss_comp['re'].append(re_loss)\n",
    "    loss_comp['uv_kl'].append(uv_kl)\n",
    "    loss_comp['z_kl'].append(z_kl)\n",
    "    loss_comp['l1_reg'].append(l1_reg)\n",
    "        \n",
    "    # validation\n",
    "    model.eval()\n",
    "    with torch.no_grad():\n",
    "        for inputs in val_loader:\n",
    "            data, masks, regimes=inputs\n",
    "            batch_size = data.shape[0]\n",
    "            data = data.to(device)\n",
    "            masks = masks.to(device)\n",
    "            regimes = regimes.to(device)\n",
    "            n2m = torch.bernoulli(torch.ones(batch_size, num_vars, num_modules).float()).float().to(device)\n",
    "            m2n = 1 - n2m\n",
    "            _re_loss, _uv_kl, _z_kl, _l1_reg = model.losses(data, masks, n2m, m2n)\n",
    "            loss = _re_loss + kl_coeff * _uv_kl + kl_coeff * _z_kl \n",
    "            eval_loss += loss.detach().cpu().item() * (batch_size / n_val)\n",
    "\n",
    "    eval_loss_list.append(eval_loss)\n",
    "    # Smoothing\n",
    "    if epoch > 0:\n",
    "        eval_loss = prev_loss * memory + eval_loss * (1 - memory)\n",
    "    \n",
    "    # Graph evaluation\n",
    "    with torch.no_grad():\n",
    "        n2m, m2n = model.sample_uv.fg.sample_deterministic()\n",
    "        n2m = n2m.squeeze().detach().cpu().numpy()\n",
    "        m2n = m2n.squeeze().detach().cpu().numpy()\n",
    "        graph_pred = (np.matmul(n2m, m2n.T) > 0).astype(int)\n",
    "    precision, recall, f1 = pred_score(graph_pred, graph)\n",
    "    loss_comp['precision'].append(precision)\n",
    "    loss_comp['recall'].append(recall)\n",
    "    loss_comp['f1'].append(f1)\n",
    "    \n",
    "    print(f\"Epoch{epoch + 1}\\tAverage Loss: {train_loss:.2f}, {eval_loss:.2f}, Partition Entropy: {model.sample_uv.fg.entropy_y():.3f}\")\n",
    "    print(f\"\\tGraph prediction: precision={precision:.3f}, recall={recall:.3f}, f1 score={f1:.3f}\\n\")\n",
    "    # Update early stopping variables\n",
    "    if prev_loss - eval_loss < stop_thred:\n",
    "        stop_counter += 1\n",
    "    else:\n",
    "        stop_counter = 0\n",
    "    prev_loss = eval_loss\n",
    "    \n",
    "    if stop_counter > patience:\n",
    "        break"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "ae8a8dcd",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.9283488970756544\n"
     ]
    }
   ],
   "source": [
    "mse = 0\n",
    "with torch.no_grad():\n",
    "    for inputs in train_loader:\n",
    "        data,masks,regimes=inputs\n",
    "        batch_size = data.shape[0]\n",
    "        data = data.to(device)\n",
    "        masks = masks.to(device)\n",
    "        regimes = regimes.to(device)\n",
    "        n2m = torch.bernoulli(torch.ones(batch_size, num_vars, num_modules).float()).float().to(device)\n",
    "        m2n = 1 - n2m\n",
    "        _mse = model.mse(data, masks, n2m, m2n)\n",
    "        mse += _mse.detach().cpu().item() * (batch_size/n_train)\n",
    "print(mse)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "b2f5a079",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1.9364170574458537\n"
     ]
    }
   ],
   "source": [
    "mse = 0\n",
    "with torch.no_grad():\n",
    "    for inputs in val_loader:\n",
    "        data,masks,regimes=inputs\n",
    "        batch_size = data.shape[0]\n",
    "        data = data.to(device)\n",
    "        masks = masks.to(device)\n",
    "        regimes = regimes.to(device)\n",
    "        n2m = torch.bernoulli(torch.ones(batch_size, num_vars, num_modules).float()).float().to(device)\n",
    "        m2n = 1 - n2m\n",
    "        _mse = model.mse(data, masks, n2m, m2n)\n",
    "        mse += _mse.detach().cpu().item() * (batch_size/n_val)\n",
    "print(mse)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6c3827d5",
   "metadata": {},
   "source": [
    "# Inspect latent space"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "e05b4cf9",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_z_posterior(data_loader, mask_type=None):\n",
    "    mu, std = [], []\n",
    "    with torch.no_grad():\n",
    "        for inputs in data_loader:\n",
    "            data, masks, regimes=inputs\n",
    "            batch_size = data.shape[0]\n",
    "            data = data.to(device)\n",
    "            masks = masks.to(device)\n",
    "            regimes = regimes.to(device)\n",
    "            if mask_type is None:\n",
    "                n2m, m2n = None, None\n",
    "            elif mask_type == 'random':\n",
    "                n2m = torch.bernoulli(torch.ones(batch_size, num_vars, num_modules).float()).float().to(device)\n",
    "                m2n = 1 - n2m\n",
    "                model.mse(data, masks, n2m, m2n)\n",
    "            elif mask_type == 'true':\n",
    "                n2m = torch.tensor(np.tile(U0, (batch_size, 1, num_modules))).float().to(device)\n",
    "                m2n = torch.tensor(np.tile(V0.T, (batch_size, 1, num_modules))).float().to(device)\n",
    "                model.mse(data, masks, n2m, m2n)\n",
    "            elif mask_type == 'null':\n",
    "                n2m = torch.zeros(batch_size, num_vars, num_modules).float().to(device)\n",
    "                m2n = torch.zeros(batch_size, num_vars, num_modules).float().to(device)\n",
    "                model.mse(data, masks, n2m, m2n)\n",
    "            else:\n",
    "                model.mse(data, masks)\n",
    "            mu.append(model.mu_q.detach().cpu().numpy())\n",
    "            std.append(model.std_q.detach().cpu().numpy())\n",
    "    return np.concatenate(mu, 0), np.concatenate(std, 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 43,
   "id": "61c8363c",
   "metadata": {},
   "outputs": [],
   "source": [
    "mu_z_true, std_z_true = get_z_posterior(train_loader, 'true')\n",
    "mu_z_null, std_z_null = get_z_posterior(train_loader, 'null')\n",
    "mu_z_rand, std_z_rand = get_z_posterior(train_loader, 'random')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 44,
   "id": "43e4f254",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_2717121/1271579223.py:4: UserWarning: Dataset has 0 variance; skipping density estimate. Pass `warn_singular=False` to disable this warning.\n",
      "  seaborn.kdeplot(df, ax=ax[0])\n",
      "/tmp/ipykernel_2717121/1271579223.py:9: UserWarning: Dataset has 0 variance; skipping density estimate. Pass `warn_singular=False` to disable this warning.\n",
      "  seaborn.kdeplot(df, ax=ax[1])\n"
     ]
    },
    {
     "data": {
      "image/png": "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E1q8n2AwGDVFry63262MjIyoGwDr8EcqpWJdfc8etQ4AdEbxGlcB4erjWEkppu8BAKyoTZVSH330kR577LEGy/v166eSkpJW7augoEDTpk3TuHHjNH78eC1cuFBVVVWRq/FNnTpV/fr1U2FhoaTQdLxx48Zp8ODB8ng8eu211/THP/5RixcvbstbAWCyQLDx6XvhRBVJKQCdWTzHVejawtVQzqhDaqD2OZJSAADraVNSyu12x2wY/uWXX6p3796t2tell16q3bt3a+7cuSopKdHo0aO1YsWKSPPz4uJi2e2HjqxVVVW67rrr9O233yolJUXDhg3Tn/70J1166aVteSsATOarHSPHSkqFp/T56SkFoBOL57gKXVt4+p69bvWxzZCMQxcWAQDASto0fe+HP/yh5s+fL5/PJ0my2WwqLi7WL37xC1188cWt3t+sWbP0zTffyOPxaO3atcrPz488t3LlSi1fvjzy+Ne//rW++uorHTx4UPv27dMHH3xAQgrowMJj5FhJKZeD6XsAOr94jKsWL16skSNHRloVTJgwQa+//nrk+ZqaGs2cOVM9e/ZUenq6Lr744gY9PdHxhY+X0Ve0DS2jpxQAwIralJR6+OGHVVlZqd69e+vgwYM6/fTTNWTIEHXr1k333HNPvGME0ImFLwbkijV9z+mQJPkZRwPoxOIxrjriiCN03333ad26dfr444911llnafLkyfrss88kSTfffLNeeeUVPf/881q1apV27Nihiy66qD3fFkwQa/qezRZOSvnNCAkAgCa1afpeZmam3nrrLb3//vv65JNPVFlZqeOPP14TJ06Md3wAOrlDlVKOBs+Fe0ox4wBAZxaPcdUFF1wQ9fiee+7R4sWLtWbNGh1xxBF68skn9cwzz+iss86SJC1btkzDhw/XmjVrdOKJJ8b1/cA8kUqpqBM9oYMolVIAACtqdVIqGAxq+fLleuGFF7Rt2zbZbDYNGjRIOTk5MgxDNlvDK2gBQGPCCSe3s2FSylWbqArQmxVAJ9Ue46pAIKDnn39eVVVVmjBhgtatWyefzxeV5Bo2bJj69++v1atXN5mU8ng8UVc4jtX7Ctbhj5zoqbPQxvQ9AIB1tWr6nmEY+uEPf6irr75a3333nY477jgdc8wx+uabbzR9+nRdeOGF7RUngE6qqZ5S4WVM3wPQGcV7XLVx40alp6fL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f6yXX35Zq1evlmEYOv/88yNnxWbOnCmPx6N3331XGzdu1P3336/09HTl5eXp73//uyRp06ZN2rlzpx599FFJUmFhoZ5++mktWbJEn332mW6++Wb95Cc/0apVq6LiuO2223Tffffp888/18iRIxvE+eijj+rhhx/WQw89pE8//VSTJk3SD3/4w6iBXXg/N954oz7//HNNmjSp0fednZ2tSZMm6amnnpIkVVdX67nnntOVV17ZYN1LLrlEu3bt0uuvv65169bp+OOP19lnn619+/ZJkiorK3X++eerqKhI//73v3XuuefqggsuUHFxcdR+Hn74YY0bN07//ve/dd111+naa6/Vpk2bGv9yAABAh9PVxlY5OTny+/168cUXZRhGg+ebei8///nPtWrVKv3jH//Qm2++qZUrV2r9+vWt+rwBJJABAIdh2rRpxuTJkw3DMIxgMGi89dZbhtvtNmbPnm18+eWXhiTj/fffj6y/Z88eIyUlxfjrX/9qGIZhHHfcccZdd90Vc9/vvPOOIcnYv39/ZFlNTY2RmppqfPDBB1HrXnXVVcZll10Wtd1LL70Utc68efOMUaNGRR737dvXuOeee6LWOeGEE4zrrrvOMAzD2Lp1qyHJWLhwYbOfw4ABA4xHHnnEeOmll4zBgwcbwWDQeOqpp4wxY8YYhmEYmZmZxrJlywzDMIz/+7//MzIyMoyampqofQwePNh47LHHGn2NY445xvjtb38b9Zo/+clPIo+DwaDRp08fY/Hixc3GCwAArImxVcjtt99uOJ1Oo0ePHsa5555rPPDAA0ZJSUmT7+XAgQOGy+WKfBaGYRh79+41UlJSjBtvvLHZ1wSQeM5Gs1UA0EL//Oc/lZ6eLp/Pp2AwqP/5n//RXXfdpaKiIjmdTuXn50fW7dmzp44++mh9/vnnkqQbbrhB1157rd58801NnDhRF198ccwzb2GbN29WdXW1zjnnnKjlXq9XY8aMiVo2bty4RvdTUVGhHTt26OSTT45afvLJJ0dNo2tuP/V9//vf189+9jO9++67Wrp0acwqqU8++USVlZUNejAcPHhQW7ZskRSqlLrrrrv06quvaufOnfL7/Tp48GCDSqm6n5XNZlNOTk6kASgAAOiYGFtJ99xzjwoKCvT2229r7dq1WrJkie699169++67Ou6442Jus2XLFnm93qjPp0ePHjr66KObfT0A5iApBeCwnXnmmVq8eLFcLpf69u3boH9SU66++mpNmjRJr776qt58800VFhbq4Ycf1vXXXx9z/crKSknSq6++qn79+kU953a7ox7Hq7l4a/bjdDp1xRVXaN68eVq7dq1efPHFButUVlYqNzdXK1eubPBcVlaWpNDVbN566y099NBDGjJkiFJSUvSjH/0oUroflpSUFPXYZrMpGAy2OF4AAGA9jK1CevbsqUsuuUSXXHKJ7r33Xo0ZM0YPPfRQpFUCgI6PnlIADltaWpqGDBmi/v37Rw2ahg8fLr/fr7Vr10aW7d27V5s2bdKIESMiy/Ly8vS///u/euGFF3TLLbfoiSeekCS5XC5JUiAQiKw7YsQIud1uFRcXa8iQIVG3vLy8FseckZGhvn376v33349a/v7770fF1hZXXnmlVq1apcmTJ6t79+4Nnj/++ONVUlIip9PZ4D306tUrEsf06dN14YUX6rjjjlNOTo62bdt2WHEBAICOgbFVQy6XS4MHD45cfS/Wexk8eLCSkpKiPp/9+/fryy+/POzXB9A+qJQC0G6GDh2qyZMn65prrtFjjz2mbt266bbbblO/fv00efJkSdJNN92k8847T0cddZT279+vd955R8OHD5ckDRgwQDabTf/85z91/vnnKyUlRd26ddPs2bN18803KxgM6pRTTlF5ebnef/99ZWRkaNq0aS2O7+c//7nmzZunwYMHa/To0Vq2bJk2bNjQ5BX2WmL48OHas2ePUlNTYz4/ceJETZgwQVOmTNEDDzygo446Sjt27NCrr76qCy+8UOPGjdPQoUP1wgsv6IILLpDNZtOdd95JBRQAAF1cVxlb/fOf/9Szzz6rH//4xzrqqKNkGIZeeeUVvfbaa1q2bFmj7yU9PV1XXXWVfv7zn6tnz57q06ePfvnLX8puj67FmDNnjr777js9/fTTrYoLQPyRlALQrpYtW6Ybb7xRP/jBD+T1enXaaafptddei0w7CwQCmjlzpr799ltlZGTo3HPP1SOPPCJJ6tevn+6++27ddtttmjFjhqZOnarly5frV7/6lXr37q3CwkJ9/fXXysrK0vHHH6/bb7+9VbHdcMMNKi8v1y233KJdu3ZpxIgRevnllzV06NDDft/1+0XVZbPZ9Nprr+mXv/ylZsyYod27dysnJ0ennXaasrOzJUkLFizQlVdeqZNOOkm9evXSL37xC1VUVBx2XAAAoGPrCmOrESNGKDU1Vbfccou2b98ut9utoUOH6g9/+IOuuOKKJt/Lgw8+qMrKSl1wwQXq1q2bbrnlFpWXl0ftf+fOnQ36dAIwh80wYlxjEwAAAAAAAGhH9JQCAAAAAABAwpGUAgAAAAAAQMKRlAIAAAAAAEDCkZQCAAAAAABAwpGUAgAAAAAAQMKRlAIAAAAAAEDCkZQCAAAAAABAwpGUAgAAAAAAQMKRlAIAAAAAAEDCkZQCAAAAAABAwpGUAgAAAAAAQMKRlAIAAAAAAEDC/X8acRxvAtylxwAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 1200x400 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(1, 2, figsize=(12, 4))\n",
    "\n",
    "df = pd.DataFrame(np.concatenate([mu_z_true, mu_z_null, mu_z_rand], 1), columns=['true', 'null', 'rand'])\n",
    "seaborn.kdeplot(df, ax=ax[0])\n",
    "ymin, ymax = ax[0].get_ylim()\n",
    "ax[0].plot([mu_z_null.mean(), mu_z_null.mean()], [ymin, ymax], color='orange')\n",
    "ax[0].set_xlabel('Posterior Mean')\n",
    "df = pd.DataFrame(np.concatenate([std_z_true, std_z_null, std_z_rand], 1), columns=['true', 'null', 'rand'])\n",
    "seaborn.kdeplot(df, ax=ax[1])\n",
    "ymin, ymax = ax[1].get_ylim()\n",
    "ax[1].plot([std_z_null.mean(), std_z_null.mean()], [ymin, ymax], color='orange')\n",
    "ax[1].set_xlabel('Posterior Std.')\n",
    "plt.tight_layout()\n",
    "plt.savefig(f'{figure_path}/posterior.png')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "50517b0f",
   "metadata": {},
   "source": [
    "## 1.3 Training sanity check"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "c2071d82",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x145b12b47e80>"
      ]
     },
     "execution_count": 14,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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qj13OBzy+4//gzAxMVGff/65Fi9erJSUFE2dOlUZGRkqKCgI+fs4nc5Txp2cPJYjXN+npk7+3oGfVeB715XL5dKHH36oFStWqHPnznr55ZfVsWNH5ebmSpLeeustrV+/XpdffrmWLl2qDh06aMOGDWE59unYJlh46FgAQK1deuml+uabb9S6dWu1a9eu2qNevXpq37694uLitHr16tO+/+KLL9aXX34ZHDAoSZ988omcTqc6duxY4zrcbrcGDx6smTNn6quvvtKuXbuCYzxiY2NrvDx9kyZNtG/fvmrbTr519FzfJzCm4mzHa9u2rWJjY6uNlaioqNCmTZvUuXPnGtVZExdffLHWr19fLSh98sknSkxMVMuWLSUdDyv9+vXTE088oS+++EKxsbFatmxZcP8ePXpo8uTJ+vTTT9W1a1ctWrQobPX9km2ChZcxFgBQa1lZWTp8+LBuu+02bdq0STt37tSqVas0ZswYVVZWyuv1atKkSXrkkUe0cOFC7dy5Uxs2bNAbb7whSRo5cqS8Xq9GjRqlr7/+WmvWrNH999+v22+/PXgZ5FyWL1+u2bNnKycnR7t379bChQvl9/uDwaR169bauHGjdu3apUOHDp31X/wDBw7U5s2btXDhQm3fvl3Tpk3T119/HXz9XN+nadOmiouL08qVK3XgwAEVFhaecox69erp3nvv1cMPP6yVK1fq22+/1bhx43TkyBGNHTu2xuf+XO677z7l5eXp/vvv1/fff6/3339f06ZN08SJE+V0OrVx40Y988wz2rx5s/bs2aO//e1vOnjwoC6++GLl5uZq8uTJWr9+vXbv3q0PPvhA27dvj+g4C9uMsQh0LMoq6FgAQKhSU1P1ySefaNKkSbr22mvl8/mUnp6uIUOGBBemmjJlitxut6ZOnaq9e/cqJSVF99xzjyQpPj5eq1at0oMPPqjLLrtM8fHxuummm/Rf//VfNa6hfv36+tvf/qbp06errKxM7du31+LFi9WlSxdJ0kMPPaRRo0apc+fOOnr0aLDVfzqZmZmaMmWKHnnkEZWVlenOO+/UHXfcERzPca7v43a7NXv2bD355JOaOnWqrrzySmVnZ59ynGeffVZ+v1+33367iouL1atXL61atarGd6bURIsWLfSPf/xDDz/8sDIyMtSwYUONHTtWjz/+uCQpKSlJ69at00svvaSioiKl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AQAA2wvpUkhsbKx69uyp1atXB7f5/X6tXr1affv2DXtxAADgwhLypZCJEydq1KhR6tWrl3r37q2XXnpJpaWlGjNmTCTqAwAAF5CQg8Wtt96qgwcPaurUqdq/f7+6d++ulStXnjKgEwAARB+m9AYAAOdU07/fLJsOAADChmABAADChmABAADChmABAADChmABAADChmABAADChmABAADChmABAADChmABAADCJuLLpv9SYKLPoqIisw8NAABqKfB3+1wTdpseLIqLiyVJaWlpZh8aAADUUXFxsZKTk8/4uulrhfj9fu3du1eJiYlyOBxh+9yioiKlpaUpLy+PNUgswPm3FuffWpx/a3H+zWEYhoqLi5Wamiqn88wjKUzvWDidTrVs2TJin5+UlMQvloU4/9bi/FuL828tzn/kna1TEcDgTQAAEDYECwAAEDa2CRYej0fTpk2Tx+OxupSoxPm3FuffWpx/a3H+zy+mD94EAAD2ZZuOBQAAsB7BAgAAhA3BAgAAhA3BAgAAhI1tgsXcuXPVunVreb1e9enTR5999pnVJdnSunXrNHz4cKWmpsrhcOi9996r9rphGJo6dapSUlIUFxenwYMHa/v27dYUazMzZszQZZddpsTERDVt2lQjRozQtm3bqu1TVlamrKwsNWrUSAkJCbrpppt04MABiyq2l/nz56tbt27BSZj69u2rFStWBF/n3Jvr2WeflcPh0IQJE4Lb+BmcH2wRLJYuXaqJEydq2rRp+vzzz5WRkaHMzEzl5+dbXZrtlJaWKiMjQ3Pnzj3t6zNnztTs2bP1yiuvaOPGjapXr54yMzNVVlZmcqX2s3btWmVlZWnDhg368MMPVVFRoWuvvValpaXBfX73u9/p73//u9555x2tXbtWe/fu1Y033mhh1fbRsmVLPfvss9qyZYs2b96sgQMH6te//rW++eYbSZx7M23atEmvvvqqunXrVm07P4PzhGEDvXv3NrKysoLPKysrjdTUVGPGjBkWVmV/koxly5YFn/v9fqN58+bG888/H9xWUFBgeDweY/HixRZUaG/5+fmGJGPt2rWGYRw/1zExMcY777wT3Oe7774zJBnr16+3qkxba9CggfH6669z7k1UXFxstG/f3vjwww+Nq6++2njwwQcNw+D3/3xywXcsysvLtWXLFg0ePDi4zel0avDgwVq/fr2FlUWf3Nxc7d+/v9rPIjk5WX369OFnEQGFhYWSpIYNG0qStmzZooqKimrnv1OnTmrVqhXnP8wqKyu1ZMkSlZaWqm/fvpx7E2VlZWnYsGHVzrXE7//5xPRFyMLt0KFDqqysVLNmzaptb9asmb7//nuLqopO+/fvl6TT/iwCryE8/H6/JkyYoH79+qlr166Sjp//2NhY1a9fv9q+nP/w2bp1q/r27auysjIlJCRo2bJl6ty5s3Jycjj3JliyZIk+//xzbdq06ZTX+P0/f1zwwQKIRllZWfr666/18ccfW11KVOnYsaNycnJUWFiod999V6NGjdLatWutLisq5OXl6cEHH9SHH34or9drdTk4iwv+Ukjjxo3lcrlOGfl74MABNW/e3KKqolPgfPOziKzx48dr+fLlWrNmjVq2bBnc3rx5c5WXl6ugoKDa/pz/8ImNjVW7du3Us2dPzZgxQxkZGZo1axbn3gRbtmxRfn6+Lr30Urndbrndbq1du1azZ8+W2+1Ws2bN+BmcJy74YBEbG6uePXtq9erVwW1+v1+rV69W3759Laws+rRp00bNmzev9rMoKirSxo0b+VmEgWEYGj9+vJYtW6Z//etfatOmTbXXe/bsqZiYmGrnf9u2bdqzZw/nP0L8fr98Ph/n3gSDBg3S1q1blZOTE3z06tVLI0eODP43P4Pzgy0uhUycOFGjRo1Sr1691Lt3b7300ksqLS3VmDFjrC7NdkpKSrRjx47g89zcXOXk5Khhw4Zq1aqVJkyYoKeeekrt27dXmzZtNGXKFKWmpmrEiBHWFW0TWVlZWrRokd5//30lJiYGrxsnJycrLi5OycnJGjt2rCZOnKiGDRsqKSlJ999/v/r27atf/epXFld/4Zs8ebKGDh2qVq1aqbi4WIsWLVJ2drZWrVrFuTdBYmJicDxRQL169dSoUaPgdn4G5wmrb0sJl5dfftlo1aqVERsba/Tu3dvYsGGD1SXZ0po1awxJpzxGjRplGMbxW06nTJliNGvWzPB4PMagQYOMbdu2WVu0TZzuvEsy3nrrreA+R48eNe677z6jQYMGRnx8vHHDDTcY+/bts65oG7nzzjuN9PR0IzY21mjSpIkxaNAg44MPPgi+zrk338m3mxoGP4PzBcumAwCAsLngx1gAAIDzB8ECAACEDcECAACEDcECAACEDcECAACEDcECAACEDcECAACEDcECAACEDcECAACEDcECAACEDcECAACEDcECAACEzf8H35zRclYFWGMAAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure()\n",
    "plt.plot(loss_comp['re'], label='reconstruction loss')\n",
    "plt.plot(loss_comp['uv_kl'], label='KL Graph')\n",
    "plt.plot(loss_comp['z_kl'], label='KL Z')\n",
    "#plt.xlim(20, 300)\n",
    "#plt.ylim(0, 5e4)\n",
    "plt.legend()\n",
    "#plt.savefig(f'{figure_path}/loss_spnfactor.png')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "e221b2a2",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x151c306cfb50>"
      ]
     },
     "execution_count": 24,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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t8mTJ2Pp0qXW75m/7mlqakJRURG8vb35uutvUs+glUpKSopQqVRiy5Yt4tSpU+KRRx4R7u7uQqfTSR3agKLX60VOTo7IyckRAMTGjRtFTk6OKCkpEUIIsX79euHu7i4+//xzkZubK+bNmycCAwNFS0uLxJFLb8WKFUKtVov9+/eLiooK69bc3Gwd8+ijj4qAgACxd+9ecfz4cRETEyNiYmIkjHpgWLt2rUhPTxfnz58Xubm5Yu3atUImk4lvv/1WCMG89dTVd9MIwfxdy1NPPSX2798vzp8/L77//nsRFxcnvLy8RFVVlRCCeetPQ7YYEUKI119/XQQEBAilUimioqLEkSNHpA5pwNm3b58A0GlLSEgQQlhu733mmWeERqMRKpVKzJ49WxQWFkob9ADRVd4AiPfff986pqWlRTz22GPCw8NDODk5iQULFoiKigrpgh4gli9fLkaOHCmUSqUYPny4mD17trUQEYJ566mfFyPMX9cWLVokvL29hVKpFL6+vmLRokXi3Llz1uPMW/+RCSGENNdkiIiIiIbonBEiIiIaOFiMEBERkaRYjBAREZGkWIwQERGRpFiMEBERkaRYjBAREZGkWIwQERGRpFiMEBERkaRYjBAREZGkWIwQERGRpFiMEBERkaRYjBAREZGk/h8wd1FTDerwEgAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure()\n",
    "plt.plot(loss_comp['precision'], label='precision')\n",
    "plt.plot(loss_comp['recall'], label='recall')\n",
    "plt.plot(loss_comp['f1'], label='f1')\n",
    "\n",
    "plt.legend()\n",
    "#plt.savefig(f'{figure_path}/graph_eval_spnfactor.png')\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "37ed770e",
   "metadata": {},
   "source": [
    "## 1.4 Evaluate the factor graph"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 70,
   "id": "defeee7c",
   "metadata": {},
   "outputs": [],
   "source": [
    "# pred_adj = model.weight_mask.detach().cpu().numpy()\n",
    "with torch.no_grad():\n",
    "    n2m, m2n = model.sample_uv.fg.sample_deterministic(hard=True)\n",
    "    n2m = n2m.detach().cpu().numpy()\n",
    "    m2n = m2n.detach().cpu().numpy()\n",
    "graph_pred = np.matmul(n2m, m2n.T)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "id": "6071e9d5",
   "metadata": {},
   "outputs": [],
   "source": [
    "def plot_bar(df, num_modules, partition = False):\n",
    "    column_sums = np.sum(df,axis=0)\n",
    "    if partition==True:\n",
    "        plt.bar(range(num_modules+1),column_sums)\n",
    "        plt.xlabel('Partition')\n",
    "    else:\n",
    "        plt.bar(range(num_modules),column_sums)\n",
    "        plt.xlabel('Factor')\n",
    "    plt.ylabel('Edge counts')\n",
    "    #plt.savefig('ori_y_sampled.png',bbox_inches='tight')\n",
    "    plt.show()\n",
    "    return True"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8a02141f",
   "metadata": {},
   "outputs": [],
   "source": [
    "plot_bar(U0[:, module_causal_order],10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "101133dc",
   "metadata": {},
   "outputs": [],
   "source": [
    "plot_bar(n2m,10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ed2517c1",
   "metadata": {},
   "outputs": [],
   "source": [
    "plot_bar(V0[module_causal_order].T,10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ba129db1",
   "metadata": {},
   "outputs": [],
   "source": [
    "plot_bar(m2n,10)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "e3771529",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "array([0.        , 0.        , 0.        , 0.02093462, 0.02290758,\n",
       "       0.95317316, 0.39094242, 0.02032896, 0.5153729 , 0.47325373],\n",
       "      dtype=float32)"
      ]
     },
     "execution_count": 30,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "n2m[3]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 65,
   "id": "7dd94599",
   "metadata": {},
   "outputs": [],
   "source": [
    "def boolean_prod(n2m, m2n, thred):\n",
    "    m = n2m.shape[1]\n",
    "    b = (n2m[:, [0]] @ m2n[:, [0]].T) > thred\n",
    "    for i in range(1, m):\n",
    "        b = b | ((n2m[:, [i]] @ m2n[:, [i]].T) > thred)\n",
    "    return b"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "id": "41ee66c1",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_3009657/4213278239.py:13: RuntimeWarning: invalid value encountered in double_scalars\n",
      "  f1 = 2 * precision * recall / (precision + recall)\n",
      "/tmp/ipykernel_3009657/4213278239.py:11: RuntimeWarning: invalid value encountered in long_scalars\n",
      "  precision = tp / (tp + fp)\n"
     ]
    }
   ],
   "source": [
    "threds = np.arange(0, 1.0, 0.001)\n",
    "precision_list, recall_list, f1_list = [], [], []\n",
    "for i in range(len(threds)):\n",
    "    graph_pred = boolean_prod(n2m, m2n, threds[i]).astype(int)\n",
    "    p, r, f = pred_score(graph_pred, graph)\n",
    "    precision_list.append(p)\n",
    "    recall_list.append(r)\n",
    "    f1_list.append(f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "id": "30f1e32e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from sklearn.metrics import auc\n",
    "plt.figure()\n",
    "plt.plot(precision_list, recall_list)\n",
    "plt.xlabel('precision')\n",
    "plt.ylabel('recall')\n",
    "idx = np.argsort(precision_list)\n",
    "plt.title(f'AUC = {auc(np.array(precision_list)[idx], np.array(recall_list)[idx]):.2f}')\n",
    "plt.savefig(f'{figure_path}/auc_spnfactor.png')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 69,
   "id": "58eeebeb",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(1660, 96, 81)"
      ]
     },
     "execution_count": 69,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "graph_pred = boolean_prod(n2m, m2n, 0.5).astype(int)\n",
    "np.sum(graph != graph_pred), np.sum(n2m != U0[:, module_causal_order]), np.sum(m2n != V0[module_causal_order].T)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 74,
   "id": "3df96dc4",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "7"
      ]
     },
     "execution_count": 74,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "n2m.sum()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "id": "b55a3100",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAzYAAAGiCAYAAAA1J1M9AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8qNh9FAAAACXBIWXMAAA9hAAAPYQGoP6dpAAA0M0lEQVR4nO3deXhTdb7H8U9KugFNKBW6SIGKICooCoIsd8Cx2gFUUJRBcYZFx63I5oojqCgWHUcrKKDOXNQriOIdZORRuVIFryMCgnVAZNOqjLUFZySpLAXa3/2jlwxpG2hKs/zo+/U8eR5zcnJ+3/NL0q8fzsmJwxhjBAAAAAAWi4l0AQAAAABwogg2AAAAAKxHsAEAAABgPYINAAAAAOsRbAAAAABYj2ADAAAAwHoEGwAAAADWI9gAAAAAsB7BBgAAAID1CDZAiDkcDj344IORLuOYRo8erebNm0e6DAA4KbVv316jR4/23V+5cqUcDodWrlwZsZqqq16jrQYMGKAuXbpEugxECMEGUaGoqEjjxo1Tp06d1LRpUzVt2lRnnXWWcnNz9fe//z3S5YXUgAED5HA4jns70XC0b98+Pfjgg1HVSAEg1F588UW/v6UJCQnq1KmTxo0bp9LS0kiXF5S33347Kv6hrLy8XLNnz1a/fv2UnJysuLg4ZWRk6IorrtCrr76qioqKSJeIRsoZ6QKAZcuW6de//rWcTqdGjhypc889VzExMdqyZYv+8pe/aO7cuSoqKlK7du0iXWpI/P73v9eNN97ou79u3TrNmjVL9913n84880zf8nPOOeeExtm3b58eeughSVVhCgAak+nTpysrK0sHDhzQRx99pLlz5+rtt9/Wpk2b1LRp07DW8otf/EL79+9XXFxcUM97++239eyzz0Y03OzevVsDBw7U+vXrlZOTo/vvv18tW7ZUSUmJVqxYoeuuu047duzQ1KlTI1YjGi+CDSLqq6++0ogRI9SuXTsVFBQoPT3d7/HHHntMc+bMUUzMsQ8u7t27V82aNQtlqSFzySWX+N1PSEjQrFmzdMkllxwzgNi8zwAQbgMHDlSPHj0kSTfeeKNSUlL05JNPaunSpbr22mtrfU6o/s7GxMQoISGhwbcbDr/5zW/02Wef6b//+7911VVX+T02ZcoUffrpp9q6desxt3HgwAHFxcUdt7cDweIdhYh6/PHHtXfvXs2fP79GqJEkp9Op8ePHKzMz07fsyPdBvvrqKw0aNEhJSUkaOXKkpKomdMcddygzM1Px8fE644wz9MQTT8gY43v+N998I4fDoRdffLHGeNVP+XrwwQflcDi0Y8cOjR49Wi1atJDb7daYMWO0b98+v+eWl5dr0qRJatWqlZKSknTFFVfoH//4xwnOkH8dmzdv1nXXXafk5GT169dPUtXRl9oC0OjRo9W+fXvfPrdq1UqS9NBDDwU8ve3777/X0KFD1bx5c7Vq1Up33nknpxQAOCn98pe/lFR1KrR07N5SWVmp/Px8nX322UpISFBqaqpuvvlm/fTTT37bNMbokUceUZs2bdS0aVNddNFF+uKLL2qMHeg7NmvWrNGgQYOUnJysZs2a6ZxzztHTTz/tq+/ZZ5+VJL9T645o6Bprs3r1ai1fvlw33XRTjVBzRI8ePXzzdvS+Llq0SPfff79OPfVUNW3aVF6vV//617905513qmvXrmrevLlcLpcGDhyozz//vNb5eu2113TfffcpLS1NzZo10xVXXKGdO3fWWsfmzZt10UUXqWnTpjr11FP1+OOP12kfYTeO2CCili1bptNPP129evUK6nmHDx9WTk6O+vXrpyeeeEJNmzaVMUZXXHGFPvjgA91www3q1q2bli9frrvuukvff/+9nnrqqXrXOXz4cGVlZSkvL08bNmzQn/70J7Vu3VqPPfaYb50bb7xRr7zyiq677jr16dNH77//vgYPHlzvMWtzzTXXqGPHjnr00Uf9wtrxtGrVSnPnztWtt96qK6+80teQjj69raKiQjk5OerVq5eeeOIJrVixQn/84x/VoUMH3XrrrQ26HwAQaV999ZUkKSUlxbestt4iSTfffLNefPFFjRkzRuPHj1dRUZGeeeYZffbZZ/rb3/6m2NhYSdK0adP0yCOPaNCgQRo0aJA2bNigSy+9VAcPHjxuPe+9954uu+wypaena8KECUpLS9OXX36pZcuWacKECbr55ptVXFys9957T//1X/9V4/nhqPGtt96SJF1//fXHXbe6hx9+WHFxcbrzzjtVXl6uuLg4bd68WW+++aauueYaZWVlqbS0VM8995z69++vzZs3KyMjw28bM2bMkMPh0D333KNdu3YpPz9f2dnZKiwsVGJiom+9n376Sb/61a901VVXafjw4XrjjTd0zz33qGvXrho4cGDQtcMiBogQj8djJJmhQ4fWeOynn34yu3fv9t327dvne2zUqFFGkrn33nv9nvPmm28aSeaRRx7xW3711Vcbh8NhduzYYYwxpqioyEgy8+fPrzGuJPPAAw/47j/wwANGkhk7dqzfeldeeaVJSUnx3S8sLDSSzG233ea33nXXXVdjm8ezePFiI8l88MEHNeq49tpra6zfv39/079//xrLR40aZdq1a+e7v3v37oC1HJnT6dOn+y0/77zzTPfu3etcOwBEm/nz5xtJZsWKFWb37t1m586dZtGiRSYlJcUkJiaaf/zjH8aYwL3lf//3f40ks2DBAr/l7777rt/yXbt2mbi4ODN48GBTWVnpW+++++4zksyoUaN8yz744AO/v/OHDx82WVlZpl27duann37yG+fobeXm5pra/tctFDXW5sorrzSSzJ49e/yW79+/369nH70PR/b1tNNO8+vlxhhz4MABU1FR4besqKjIxMfH+/WjI9s49dRTjdfr9S1//fXXjSTz9NNP+5b179/fSDIvv/yyb1l5eblJS0szw4YNO+b+wX6cioaI8Xq9klTrZYYHDBigVq1a+W5HDr8frfpRhLfffltNmjTR+PHj/ZbfcccdMsbonXfeqXett9xyi9/9//iP/9A///lP3z68/fbbklRj7IkTJ9Z7zLrU0dBq28+vv/46pGMCQDhkZ2erVatWyszM1IgRI9S8eXMtWbJEp556qt961XvL4sWL5Xa7dckll+jHH3/03bp3767mzZvrgw8+kCStWLFCBw8e1O233+53ilhd+sBnn32moqIiTZw4US1atPB77OhtBRKOGqXAfXvevHl+PfvIqdJHGzVqlN9RFUmKj4/3fc+moqJC//znP9W8eXOdccYZ2rBhQ41t/Pa3v1VSUpLv/tVXX6309HRfDz6iefPmfkeV4uLi1LNnT/pZI8CpaIiYI3+cfv755xqPPffccyorK1NpaWmth7ydTqfatGnjt+zbb79VRkaG3x89Sb4ri3377bf1rrVt27Z+95OTkyVVHe52uVz69ttvFRMTow4dOvitd8YZZ9R7zNpkZWU16PaOlpCQ4PsezhHJyck1zs8GABs9++yz6tSpk5xOp1JTU3XGGWfU+PJ6bb1l+/bt8ng8at26da3b3bVrl6R/95iOHTv6Pd6qVStfzwjkyGlx9f39lXDUKPn3bbfb7Vs+bNgwX+133HFHrd/NrK1/VVZW6umnn9acOXNUVFTk97yjTxE8onrdDodDp59+ur755hu/5W3atKkRCJOTk0/6n48AwQYR5Ha7lZ6erk2bNtV47Mh3bqr/sTri6H/lCVagf/061pfkmzRpUutyE8T3XBpC9X/tkqr2p7Y6gv3Sf6B9BICTQc+ePX1XRQuktt5SWVmp1q1ba8GCBbU+p/o/CEVCuGrs3LmzJGnTpk3q27evb3lmZqbvIj/Jycn68ccfazy3tv716KOPaurUqRo7dqwefvhhtWzZUjExMZo4caIqKyvrXWe09GyEH8EGETV48GD96U9/0tq1a9WzZ88T2la7du20YsUKlZWV+R212bJli+9x6d9HW/bs2eP3/BM5otOuXTtVVlbqq6++8jtKc7xLXjaE5OTkWg+vV9+fupzOAADw16FDB61YsUJ9+/at9X/OjzjSY7Zv367TTjvNt3z37t3HPfJ95Gj/pk2blJ2dHXC9QH/Hw1GjJF122WWaOXOmFixY4Bds6uuNN97QRRddpD//+c9+y/fs2aNTTjmlxvrbt2/3u2+M0Y4dO074d95w8uA7Noiou+++W02bNtXYsWNr/QXoYP51ZdCgQaqoqNAzzzzjt/ypp56Sw+HwXQnF5XLplFNO0Ycffui33pw5c+qxB1WObHvWrFl+y/Pz8+u9zbrq0KGDtmzZot27d/uWff755/rb3/7mt96Rq/tUD3QAgMCGDx+uiooKPfzwwzUeO3z4sO9vanZ2tmJjYzV79my/3lWXPnD++ecrKytL+fn5Nf5GH72tI7+pU32dcNQoSX379tUll1yi559/XkuXLq11nWD6dpMmTWqsv3jxYn3//fe1rv/yyy+rrKzMd/+NN97QDz/8wJXO4MMRG0RUx44dtXDhQl177bU644wzNHLkSJ177rkyxqioqEgLFy5UTExMjXOea3P55Zfroosu0u9//3t98803Ovfcc/U///M/Wrp0qSZOnOj3/Zcbb7xRM2fO1I033qgePXroww8/1LZt2+q9H926ddO1116rOXPmyOPxqE+fPiooKNCOHTvqvc26Gjt2rJ588knl5OTohhtu0K5duzRv3jydffbZvi96SlWnAZx11ll67bXX1KlTJ7Vs2VJdunSp9zndANAY9O/fXzfffLPy8vJUWFioSy+9VLGxsdq+fbsWL16s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CQmRLieqVFZWqkePHnr00UclSeedd542bdqkefPmadSoURGuLnJef/11LViwQAsXLtTZZ5+twsJCTZw4URkZGY16XoATRW/6N3pT7ehLgdGbQiPqTkU75ZRT1KRJkxpXCiktLVVaWlqEqoqccePGadmyZfrggw/Upk0b3/K0tDQdPHhQe/bs8Vv/ZJ+n9evXa9euXTr//PPldDrldDq1atUqzZo1S06nU6mpqY1yXiQpPT1dZ511lt+yM888U999950k+fa/sX227rrrLt17770aMWKEunbtqt/85jeaNGmS8vLyJDXeeUHd0Zdqojf5ozfVjr4UGL0pNKIu2MTFxal79+4qKCjwLausrFRBQYF69+4dwcrCyxijcePGacmSJXr//feVlZXl93j37t0VGxvrN09bt27Vd999d1LP08UXX6yNGzeqsLDQd+vRo4dGjhzp++/GOC+S1Ldv3xqXXd22bZvatWsnScrKylJaWprf3Hi9Xq1Zs+aknpt9+/YpJsb/T12TJk1UWVkpqfHOC+qOvvRv9Kba0ZtqR18KjN4UIpG+ekFtFi1aZOLj482LL75oNm/ebG666SbTokULU1JSEunSwubWW281brfbrFy50vzwww++2759+3zr3HLLLaZt27bm/fffN59++qnp3bu36d27dwSrjoyjrzxjTOOdl7Vr1xqn02lmzJhhtm/fbhYsWGCaNm1qXnnlFd86M2fONC1atDBLly41f//7382QIUNO+ktHjho1ypx66qm+S2r+5S9/Maeccoq5++67fes0xnlBcOhLVehNdUdvoi8dC70pNKIy2BhjzOzZs03btm1NXFyc6dmzp/nkk08iXVJYSar1Nn/+fN86+/fvN7fddptJTk42TZs2NVdeeaX54YcfIld0hFRvHo15Xt566y3TpUsXEx8fbzp37myef/55v8crKyvN1KlTTWpqqomPjzcXX3yx2bp1a4SqDQ+v12smTJhg2rZtaxISEsxpp51mfv/735vy8nLfOo1xXhC8xt6XjKE3BYPeVIW+VDt6U2g4jDnqJ04BAAAAwEJR9x0bAAAAAAgWwQYAAACA9Qg2AAAAAKxHsAEAAABgPYINAAAAAOsRbAAAAABYj2ADAAAAwHoEGwAAAADWI9gAAAAAsB7BBgAAAID1CDYAAAAArEewAQAAAGC9/wOlO4goMdPQeQAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 1000x500 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(1, 2, figsize=(10, 5))\n",
    "ax[0].imshow(graph)\n",
    "ax[1].imshow((graph_pred > 0.99).astype(int))\n",
    "ax[0].set_title('Ground Truth')\n",
    "ax[1].set_title('Predicted Graph')\n",
    "plt.savefig(f'{figure_path}/graph.png')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "bd9e2058",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "4\n"
     ]
    }
   ],
   "source": [
    "up_nodes_pred = np.where(n2m.squeeze() > 0)[0]\n",
    "up_nodes = np.where(U0.squeeze() > 0)[0]\n",
    "print(len(np.intersect1d(up_nodes, up_nodes_pred)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "b5ac901f",
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'figure_path' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[35], line 11\u001b[0m\n\u001b[1;32m      9\u001b[0m ax[\u001b[38;5;241m1\u001b[39m]\u001b[38;5;241m.\u001b[39mset_title(\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mPredicted U\u001b[39m\u001b[38;5;124m'\u001b[39m)\n\u001b[1;32m     10\u001b[0m plt\u001b[38;5;241m.\u001b[39mtight_layout()\n\u001b[0;32m---> 11\u001b[0m plt\u001b[38;5;241m.\u001b[39msavefig(\u001b[38;5;124mf\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;132;01m{\u001b[39;00mfigure_path\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m/U.png\u001b[39m\u001b[38;5;124m'\u001b[39m)\n",
      "\u001b[0;31mNameError\u001b[0m: name 'figure_path' is not defined"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYYAAAHqCAYAAAAAtunEAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8qNh9FAAAACXBIWXMAAA9hAAAPYQGoP6dpAAArbElEQVR4nO3deXRV5b3G8eckJCchJGFOQKbIIFTgMgkEuIA1EkW9aHHAYgVqBRWVoRWIq2AZNBdKLUIp1FoZXFiUXpHSi7owCgoNiMEBkEkJlwgkyHQiUEKG9/7B5VxfE5AkZ8o+389aZy2zz94775Hfm+fs/e53b5cxxggAgP8TEewGAABCC8EAALAQDAAAC8EAALAQDAAAC8EAALAQDAAAC8EAALAQDAAAC8EAICS1atVKI0eO9P68YcMGuVwubdiwIWht+r7vt9EpCIYAcblcV/UKdtG3atVKt99+e4Xvffzxx3K5XFq6dGlgG4WAW7p0qVWXMTExateunR5//HEVFBQEu3mVsm7dOv3mN78J2u8/ePCgXC6X5s6dW+H7c+fOlcvl0sGDBwPbsCuoFewGhItXXnnF+nn58uVav359ueUdOnQIZLOAK5oxY4ZSUlJ0/vx5bdq0SYsWLdK6deu0c+dO1a5dO6Bt6d+/v/71r38pOjq6UtutW7dOCxcuDGo41DQEQ4A88MAD1s9btmzR+vXryy3/vnPnzgW8AwKX3HrrrerRo4ck6Re/+IUaNGig559/XmvWrNH9999f4TZnz55VXFycz9sSERGhmJgYn+8X5XEqKYQMHDhQHTt2VE5Ojvr376/atWvr6aeflnTxVFRF33gqOsd5+vRpjR8/Xs2bN5fb7VabNm00e/ZslZWVBeBTwMl+/OMfS5Jyc3MlSSNHjlSdOnX01VdfafDgwYqPj9fw4cMlSWVlZZo3b56uv/56xcTEKCkpSWPGjNGpU6esfRpjNGvWLDVr1ky1a9fWjTfeqF27dpX73ZcbY9i6dasGDx6sevXqKS4uTp07d9YLL7zgbd/ChQsl2adzL/F1G52CI4YQc+LECd16660aNmyYHnjgASUlJVVq+3PnzmnAgAE6fPiwxowZoxYtWuif//ynMjIydPToUc2bN88/DUdY+OqrryRJDRo08C4rKSlRenq6+vXrp7lz53qPcMeMGaOlS5dq1KhRevLJJ5Wbm6s//OEP+uSTT7R582ZFRUVJkqZNm6ZZs2Zp8ODBGjx4sLZv365BgwbpwoULP9ie9evX6/bbb1eTJk00btw4JScna/fu3frHP/6hcePGacyYMTpy5EiFp20D1cYaySAoxo4da77/v3/AgAFGklm8eHG59SWZZ555ptzyli1bmhEjRnh/njlzpomLizP79u2z1psyZYqJjIw0hw4dumK7WrZsaW677bYK39u2bZuRZJYsWXLFfaDmW7JkiZFk3n33XfPNN9+YvLw8s3LlStOgQQMTGxtrvv76a2OMMSNGjDCSzJQpU6ztP/zwQyPJrFixwlr+9ttvW8uPHTtmoqOjzW233WbKysq86z399NNGklXb77//vpFk3n//fWOMMSUlJSYlJcW0bNnSnDp1yvo9391XRX3NX22sSG5urpFkfvvb31b4/m9/+1sjyeTm5l5xP4HEqaQQ43a7NWrUqCpvv2rVKv37v/+76tWrp+PHj3tfaWlpKi0t1QcffODD1sLp0tLS1KhRIzVv3lzDhg1TnTp1tHr1al1zzTXWeo8++qj186pVq5SYmKibb77ZqsPu3burTp06ev/99yVJ7777ri5cuKAnnnjCOsUzfvz4H2zbJ598otzcXI0fP15169a13vvuvi4nEG2sqTiVFGKuueaaSl918V379+/X559/rkaNGlX4/rFjx6q870uuptPBGRYuXKh27dqpVq1aSkpK0nXXXaeICPv7ZK1atdSsWTNr2f79++XxeNS4ceMK93upDv/nf/5HktS2bVvr/UaNGqlevXpXbNul01odO3a8+g8U4DZWRij1K4IhxMTGxlZq/dLSUuvnsrIy3XzzzZo0aVKF67dr1+6K+4uJidG//vWvCt87d+6cdx2Eh549e3qvSroct9tdLizKysrUuHFjrVixosJtLvfFJZAC1cZL/aUm9SuCoYaoV6+eTp8+bS27cOGCjh49ai1r3bq1zpw5o7S0tCr9npYtW+qLL76o8L29e/d61wGupHXr1nr33XfVt2/fK37ZuVRL+/fv17XXXutd/s0335S7Mqii3yFJO3fuvGK9X+6beCDaKF0MmNq1a3v7z/ft3btXtWvXVsOGDX9wX4HCGEMN0bp163LjAy+++GK5I4Z7771X2dnZeuedd8rt4/Tp0yopKbni7xk8eLC+/vprvfnmm9byoqIivfTSS2rcuLG6detWtQ+BsHHvvfeqtLRUM2fOLPdeSUmJ90tOWlqaoqKitGDBAhljvOtczdVz3bp1U0pKiubNm1fuS9N393VpTsX31wlEGyUpMjJSgwYN0tq1a3Xo0CHrvUOHDmnt2rUaNGiQIiMjr2p/gcARQw3xi1/8Qo888oiGDh2qm2++WZ999pneeeedct8ynnrqKf3973/X7bffrpEjR6p79+46e/asduzYob/97W86ePDgFb+ZjB49Wi+//LLuuece/fznP1fXrl114sQJvfbaa9q5c6eWL19erTEQhIcBAwZozJgxyszM1KeffqpBgwYpKipK+/fv16pVq/TCCy/o7rvvVqNGjfSrX/1KmZmZuv322zV48GB98skneuutt37wG3RERIQWLVqkO+64Q126dNGoUaPUpEkT7dmzR7t27fJ+Oerevbsk6cknn1R6eroiIyM1bNiwgLTxkueee069e/dWt27dNHr0aLVq1UoHDx7Uiy++KJfLpeeee656/8N9LchXRYWty12uev3111e4fmlpqZk8ebJp2LChqV27tklPTzdffvlluctVjTHm22+/NRkZGaZNmzYmOjraNGzY0PTp08fMnTvXXLhw4QfbdurUKTNhwgSTkpJioqKiTEJCgrnxxhvNW2+9VeXPi5rl0uWq27Ztu+J6I0aMMHFxcZd9/8UXXzTdu3c3sbGxJj4+3nTq1MlMmjTJHDlyxLtOaWmpmT59umnSpImJjY01AwcONDt37ixX29+/XPWSTZs2mZtvvtnEx8ebuLg407lzZ7NgwQLv+yUlJeaJJ54wjRo1Mi6Xq1y/82Ubr2T37t3mvvvuM40bNza1atUyjRs3NsOGDTO7d+++qu0DyWXMd46NAABhjzEGAICFYAAAWAgGAICFYAAAWAgGAIClRgTDwoUL1apVK8XExKhXr1766KOPgt0kIKDoAwikkL9c9bXXXtODDz6oxYsXq1evXpo3b55WrVqlvXv3XvbmV99VVlamI0eOKD4+PqRuUgXfMMbo22+/VdOmTcvdr8cpqtMHqH9n81v9B3MSxdXo2bOnGTt2rPfn0tJS07RpU5OZmXlV2+fl5RlJvBz+ysvL81cJBl11+gD1Hx4vX9d/SN8S48KFC8rJyVFGRoZ3WUREhNLS0pSdnV3hNkVFRSoqKvL+bP7vgKifBquWovzbYARciYq1SesUHx8f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\n",
      "text/plain": [
       "<Figure size 400x500 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(1, 2, figsize=(4, 5))\n",
    "ax[0].imshow(U0[:, module_causal_order], aspect='auto')\n",
    "ax[1].imshow(n2m, aspect='auto')\n",
    "\n",
    "for i in range(2):\n",
    "    ax[i].set_xlabel('Factor')\n",
    "    ax[i].set_ylabel('Node')\n",
    "ax[0].set_title('True U')\n",
    "ax[1].set_title('Predicted U')\n",
    "plt.tight_layout()\n",
    "plt.savefig(f'{figure_path}/U.png')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "016dc696",
   "metadata": {},
   "outputs": [
    {
     "ename": "NameError",
     "evalue": "name 'figure_path' is not defined",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mNameError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[36], line 11\u001b[0m\n\u001b[1;32m      9\u001b[0m ax[\u001b[38;5;241m1\u001b[39m]\u001b[38;5;241m.\u001b[39mset_title(\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mPredicted V\u001b[39m\u001b[38;5;124m'\u001b[39m)\n\u001b[1;32m     10\u001b[0m plt\u001b[38;5;241m.\u001b[39mtight_layout()\n\u001b[0;32m---> 11\u001b[0m plt\u001b[38;5;241m.\u001b[39msavefig(\u001b[38;5;124mf\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;132;01m{\u001b[39;00mfigure_path\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m/V.png\u001b[39m\u001b[38;5;124m'\u001b[39m)\n",
      "\u001b[0;31mNameError\u001b[0m: name 'figure_path' is not defined"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYYAAAHqCAYAAAAAtunEAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8qNh9FAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAsF0lEQVR4nO3da3QUZbr28atDkk4ISTiFBOQUgYhK2CIIBhhgxmAUdBBhFAZGYKug4gHZoxC3hwF1snGpgyCCjiOCm4jiiOiIuDAiDhgQA8j5oMQhAgki0OEwQJJ+3g9u+uUxAUnS6Wo6/99avRaprqrcTe6nr66qriqXMcYIAID/E+Z0AQCA4EIwAAAsBAMAwEIwAAAsBAMAwEIwAAAsBAMAwEIwAAAsBAMAwEIwAAharVu31siRI30/f/bZZ3K5XPrss88cq+nnfl5jKCAYAsjlcp3Xw6mm379/v8LDwzV8+PCzznPkyBFFR0fr5ptvDmBlcMLrr79u9WVUVJRSUlJ07733qqioyOnyKmXx4sX605/+5Mjvfvfdd+VyufTqq6+edZ6lS5fK5XJp2rRpAazs7MKdLqA2eeONN6yf586dq6VLl5abfumllwayLJ8mTZqob9++WrRokY4fP666deuWm+fdd9/ViRMnzhkeCC2TJ09WcnKyTpw4oRUrVmjmzJlavHixNm3aVGGP1KRevXrp3//+tyIjIyu13OLFizVjxgxHwqF///6Kj49Xdna27rjjjgrnyc7OVp06dTRkyJAAV1cxgiGAfv5mumrVKi1duvQX32TP9iZdE4YNG6YlS5bo/fffr7BJs7OzFR8fr/79+wekHjjv+uuvV5cuXSRJd9xxhxo1aqTnn39eixYt0tChQytc5tixY4qJifF7LWFhYYqKivL7emuS2+3W4MGDNXv2bO3du1fNmjWznj9x4oQWLlyovn37qkmTJg5VaWNXUpDp06ePOnTooLy8PPXq1Ut169bVI488IumnXVEVfeKpaB/n4cOHNW7cOLVo0UJut1tt27bVlClT5PV6z/n7Bw4cqJiYGGVnZ5d7bv/+/crJydHgwYPldrur/BpxYfvNb34jScrPz5ckjRw5UvXq1dO3336rfv36KTY2VsOGDZMkeb1eTZ06VZdffrmioqKUmJioMWPG6NChQ9Y6jTF66qmn1Lx5c9WtW1e//vWvtXnz5nK/+2zHGFavXq1+/fqpQYMGiomJUceOHfXCCy/46psxY4Yke3fuaf6usSLDhw+X1+vV/Pnzyz334YcfyuPx+P7PggFbDEHoxx9/1PXXX68hQ4Zo+PDhSkxMrNTyx48fV+/evbVnzx6NGTNGLVu21BdffKHMzEzt27dPU6dOPeuyMTExGjBggN555x0dPHhQDRs29D331ltvqaysLKgaGIH37bffSpIaNWrkm1ZaWqqMjAz17NlTzz77rG8Ld8yYMXr99dc1atQo3X///crPz9eLL76odevWaeXKlYqIiJAkPf7443rqqafUr18/9evXT2vXrtW1116rU6dO/WI9S5cu1Q033KCmTZvqgQceUFJSkrZu3ap//OMfeuCBBzRmzBjt3bu3wt22gaqxV69eat68ubKzszV+/HjruezsbNWtW1c33XTTL64nYAwcM3bsWPPzP0Hv3r2NJDNr1qxy80syTzzxRLnprVq1MiNGjPD9/OSTT5qYmBizY8cOa76JEyeaOnXqmN27d5+zrg8//NBIMi+//LI1/eqrrzYXXXSRKSsr+4VXhlAwe/ZsI8l88skn5ocffjAFBQVm/vz5plGjRiY6Otp8//33xhhjRowYYSSZiRMnWsv/85//NJLMvHnzrOlLliyxpu/fv99ERkaa/v37G6/X65vvkUceMZKs3l62bJmRZJYtW2aMMaa0tNQkJyebVq1amUOHDlm/58x1VTTWaqrGs3nooYeMJLN9+3bfNI/HY6KioszQoUN/cflAYldSEHK73Ro1alSVl1+wYIF+9atfqUGDBjpw4IDvkZ6errKyMn3++efnXP7aa69VQkKCtTspPz9fq1at0tChQxUWRtvUJunp6UpISFCLFi00ZMgQ1atXTwsXLtRFF11kzXf33XdbPy9YsEDx8fHq27ev1YedO3dWvXr1tGzZMknSJ598olOnTum+++6zdvGMGzfuF2tbt26d8vPzNW7cONWvX9967sx1nU0gajzt9LHEM8fV3//+d504cSLotsLZlRSELrrookp/6+JMO3fu1IYNG5SQkFDh8/v37z/n8uHh4br11lv10ksvac+ePbrooot8zRxsDYyaN2PGDKWkpCg8PFyJiYm65JJLyn04CA8PV/Pmza1pO3fulMfjOesB1dN9+K9//UuS1K5dO+v5hIQENWjQ4Jy1nd6t1aFDh/N/QQGu8bSOHTuqQ4cOevPNN33HCrOzs9W4cWNlZGRUqf6aQjAEoejo6ErNX1ZWZv3s9XrVt29fPfzwwxXOn5KS8ovrHD58uF588UW9+eab+uMf/6g333xTl112ma644opK1YYLX9euXX3fSjobt9tdLiy8Xq+aNGmiefPmVbjM2T64BFKgaxw+fLgmTpyor776Ss2bN9eyZcs0ZswYhYcH11txcFWDc2rQoIEOHz5sTTt16pT27dtnTWvTpo2OHj2q9PT0Kv+ubt26qU2bNsrOzlbfvn21efNmPf3001VeH2qfNm3a6JNPPlGPHj3O+WGnVatWkn769H7xxRf7pv/www/lvhlU0e+QpE2bNp2z38+2WykQNZ5p6NChyszMVHZ2tlq1ahW0X+ZgZ/EFpE2bNuWOD7zyyivlthhuueUW5ebm6uOPPy63jsOHD6u0tPS8ft+wYcO0bt06PfHEE3K5XPr9739f9eJR69xyyy0qKyvTk08+We650tJS34ec9PR0RUREaPr06TLG+OY517fnTrvyyiuVnJysqVOnlvvQdOa6Tp9T8fN5AlHjmVq2bKlf/epXeuutt/S///u/Sk5OVvfu3Su1jkBgi+ECcscdd+iuu+7SoEGD1LdvX3399df6+OOP1bhxY2u+hx56SO+//75uuOEGjRw5Up07d9axY8e0ceNGvfPOO/ruu+/KLVOR4cOHa/LkyVq0aJF69Oih1q1b19ArQyjq3bu3xowZo6ysLK1fv17XXnutIiIitHPnTi1YsEAvvPCCBg8erISEBP3xj39UVlaWbrjhBvXr10/r1q3TRx999It9GhYWppkzZ+rGG2/UFVdcoVGjRqlp06batm2bNm/e7Ptw1LlzZ0nS/fffr4yMDN9ZxoGo8eeGDx+u0aNHa+/evfrv//7vqv3n1jSHvxVVq53t66qXX355hfOXlZWZCRMmmMaNG5u6deuajIwM880335T7uqoxxhw5csRkZmaatm3bmsjISNO4cWPTvXt38+yzz5pTp06dd41XXXWVkWReeumlSr8+XNhOf111zZo155xvxIgRJiYm5qzPv/LKK6Zz584mOjraxMbGmtTUVPPwww+bvXv3+uYpKyszkyZNMk2bNjXR0dGmT58+ZtOmTeV6++dfVz1txYoVpm/fviY2NtbExMSYjh07munTp/ueLy0tNffdd59JSEgwLper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\n",
      "text/plain": [
       "<Figure size 400x500 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(1, 2, figsize=(4, 5))\n",
    "ax[0].imshow(V0[module_causal_order].T, aspect='auto')\n",
    "ax[1].imshow(m2n, aspect='auto')\n",
    "\n",
    "for i in range(2):\n",
    "    ax[i].set_xlabel('Factor')\n",
    "    ax[i].set_ylabel('Node')\n",
    "ax[0].set_title('True V')\n",
    "ax[1].set_title('Predicted V')\n",
    "plt.tight_layout()\n",
    "plt.savefig(f'{figure_path}/V.png')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "02765154",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
  "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.9.0"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
