{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "9c696e2b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os \n",
    "import torch \n",
    "import numpy as np \n",
    "import networkx as nx \n",
    "\n",
    "from matplotlib import pyplot as plt \n",
    "\n",
    "from matplotlib.patches import FancyArrowPatch\n",
    "from networkx.drawing.nx_pylab import draw_networkx_nodes, draw_networkx_labels\n",
    "\n",
    "from dagma import utils \n",
    "from dagma.nonlinear import DagmaMLP, DagmaNonlinear"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "6b8679d7",
   "metadata": {},
   "outputs": [],
   "source": [
    "project_dir = \"../../\"\n",
    "\n",
    "os.chdir(project_dir)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "72724da0",
   "metadata": {},
   "outputs": [],
   "source": [
    "from data_generation.simulation_data import InterventionDataset\n",
    "\n",
    "from models.helper.functions import gumbelSoftMLP, gnet_z\n",
    "from models.nodags.resblock import iResBlock\n",
    "from models.dccd.implicitblock import imBlock\n",
    "from models.helper.layers.mlpLipschitz import linearLipschitz\n",
    "from models.dccd.trainer_dccd import DCCDTrainer\n",
    "\n",
    "from utils.plotting import draw_curved_edges\n",
    "from utils.graph_transformations import *\n",
    "from utils.error_metrics import *"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "419d0c5f",
   "metadata": {},
   "source": [
    "## Load data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "66208a26",
   "metadata": {},
   "outputs": [],
   "source": [
    "data_dir = \"datasets/sachs\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "d8cbc58c",
   "metadata": {},
   "outputs": [],
   "source": [
    "intervention_sets = [[None]]\n",
    "\n",
    "datasets = list()\n",
    "for i in range(len(intervention_sets)):\n",
    "    dataset = np.load(os.path.join(data_dir, f\"dataset-{i}.npy\"))\n",
    "    datasets.append(\n",
    "        dataset - dataset.mean(axis=0)\n",
    "    )\n",
    "\n",
    "datasets = [np.vstack(datasets)]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "a6ab61ca",
   "metadata": {},
   "outputs": [],
   "source": [
    "n_nodes = datasets[0].shape[1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "d9b65f87",
   "metadata": {},
   "outputs": [],
   "source": [
    "adj_mat = np.load(os.path.join(data_dir, \"adj-mat.npy\"))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "524d58ea",
   "metadata": {},
   "source": [
    "## Training the model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "d968379e",
   "metadata": {},
   "outputs": [],
   "source": [
    "g_x = gumbelSoftMLP(\n",
    "    n_nodes = n_nodes,\n",
    "    lip_constant = 0.9,\n",
    "    activation = \"tanh\"\n",
    ")\n",
    "\n",
    "g_z = gnet_z(\n",
    "    n_nodes = n_nodes\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "b35df0ec",
   "metadata": {},
   "outputs": [],
   "source": [
    "implicit_block = imBlock(\n",
    "    nnet_x = g_x, \n",
    "    nnet_z = g_z, \n",
    "    confounders = False\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "1ea85483",
   "metadata": {},
   "outputs": [],
   "source": [
    "im_trainer = DCCDTrainer(\n",
    "    implicit_block, \n",
    "    max_epochs=1000, \n",
    "    batch_size=1024, \n",
    "    lr=1e-2,\n",
    "    lambda_c=1e-2, \n",
    "    rho=250.0\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "6ce952e2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 1000/1000, Intervention: 1/1, log P(X): -211.56\n"
     ]
    }
   ],
   "source": [
    "im_trainer.train(\n",
    "    intervention_datasets=datasets,\n",
    "    intervention_targets=intervention_sets,\n",
    "    verbose=True\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "25f0b537",
   "metadata": {},
   "outputs": [],
   "source": [
    "with open(os.path.join(data_dir, \"names.txt\"), \"r\") as file:\n",
    "    names = file.readlines()\n",
    "    \n",
    "names = [name.strip() for name in names]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "4171c457",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, axs = plt.subplots(1, 2)\n",
    "\n",
    "plt.rcParams[\"font.family\"] = \"Helvetica\"\n",
    "\n",
    "d = len(names)\n",
    "\n",
    "axs[0].imshow(np.abs(adj_mat) > 0, cmap='bwr', vmin=-1, vmax=1)\n",
    "axs[0].set_xticks(range(d))\n",
    "axs[0].set_yticks(range(d))\n",
    "axs[0].set_xticklabels(names, fontsize=7, rotation=90)\n",
    "axs[0].set_yticklabels(names, fontsize=7)\n",
    "axs[0].set_title(\"True Graph (Directed)\", fontsize=10)\n",
    "\n",
    "axs[1].imshow(g_x.get_w_adj().detach().cpu().numpy() > 0.5, cmap='bwr', vmin=-1, vmax=1)\n",
    "axs[1].set_xticks(range(d))\n",
    "axs[1].set_yticks(range(d))\n",
    "axs[1].set_xticklabels(names, fontsize=7, rotation=90)\n",
    "_ = axs[1].set_yticklabels(names, fontsize=7)\n",
    "axs[1].set_title(\"DCCD-CONF: Estimated Graph (Directed)\", fontsize=10)\n",
    "\n",
    "fig.subplots_adjust(wspace=0.5)\n",
    "\n",
    "fig.savefig(\"../results/sachs-adj-mat.pdf\", bbox_inches=\"tight\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "590a191b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "SHD = 18\n"
     ]
    }
   ],
   "source": [
    "shd, _ = compute_shd(np.abs(adj_mat) > 0, g_x.get_w_adj().detach().cpu().numpy() > 0.5)\n",
    "\n",
    "print(f\"SHD = {int(shd)}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f4ab7513",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "causal-discovery",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.12.4"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
