{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "b6593c0d",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np \n",
    "import matplotlib.pyplot as plt \n",
    "import networkx as nx \n",
    "import os \n",
    "\n",
    "from matplotlib.patches import FancyArrowPatch\n",
    "from networkx.drawing.nx_pylab import draw_networkx_nodes, draw_networkx_labels\n",
    "\n",
    "import torch\n",
    "from torch.utils.data import DataLoader \n",
    "from sklearn.covariance import graphical_lasso "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "67e309e7",
   "metadata": {},
   "outputs": [],
   "source": [
    "project_dir = '../../'\n",
    "\n",
    "os.chdir(project_dir)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "6542bd0a",
   "metadata": {},
   "outputs": [],
   "source": [
    "from data_generation.graph import DirectedGraphGenerator \n",
    "from data_generation.structural_models import SEM\n",
    "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": "code",
   "execution_count": 5,
   "id": "a7b40b1b",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Experiment parameters - data\n",
    "\n",
    "n_nodes = 5\n",
    "exp_density = 2\n",
    "n_samples = 1000"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "bf487a71",
   "metadata": {},
   "outputs": [],
   "source": [
    "graph_gen = DirectedGraphGenerator(\n",
    "    nodes=n_nodes, \n",
    "    expected_density=exp_density, \n",
    "    enforce_dag=False\n",
    ")\n",
    "\n",
    "graph = graph_gen()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "1503df06",
   "metadata": {},
   "outputs": [],
   "source": [
    "sem = SEM(\n",
    "    graph=graph,\n",
    "    contractive=True,  \n",
    "    confounders=True, \n",
    "    off_diag_nonzeros=2,\n",
    "    noise_scale=0.5\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "90645954",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAABI0AAAF0CAYAAACubclCAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjAsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvlHJYcgAAAAlwSFlzAAAPYQAAD2EBqD+naQAAeqlJREFUeJzt3Xt8z/X///H7e2Zz2IjNEMPkEzZElM5ChBI1HZAkFKs+OXT40Dp9ItWnfHRiZTp9ciqE+kZNoaOiSAzlkDNjDjOHzbbX74/n772ZjZ3e7/frvff7dr1cdnm99n6/3q/X4/3e9n56Pzyej6fDsixLAAAAAAAAwBkC7A4AAAAAAAAA3oekEQAAAAAAAAogaQQAAAAAAIACSBoBAAAAAACgAJJGAAAAAAAAKICkEQAAAAAAAAogaQQAAAAAAIACSBoBAAAAAACgAJJGAAAAAAAAKICkEWyzdu1aDR48WBdddJEqV66sypUr6x//+IceeOABrVq1yra4GjVqpJtvvtm26wOAXdauXatBgwYpKipKlSpVUkhIiC699FK9/PLLOnTokFuvvXr1anXo0EHVq1eXw+HQpEmT3Hq90lq2bJkcDoeWLVvmkett2LBBAwYMUOPGjVWpUiWFh4fr0ksv1UMPPaS0tLTc4+699141atTIIzEBcD/ej4vm6fdjV7Pzfbtz584aNmxY7vfO13LOnDm2xOONPvvsM/Xs2VO1a9dWUFCQatasqc6dO2v69Ok6ffq0W655+PBhXXDBBZo/f75bzl9agXYHAP/09ttv66GHHlLTpk31yCOPKCYmRg6HQxs2bNDMmTN12WWXafPmzbrooovsDhUA/MLUqVMVFxenpk2b6rHHHlN0dLROnz6tVatWKSEhQT/99JM+/fRTt13/vvvu0/HjxzVr1izVqFGDBIjMB7err75azZs319NPP61GjRrp4MGD+v333zVr1iw9+uijqlatmiTpqaee0iOPPGJzxABcgfdj/2DX+/aCBQv0ww8/6MMPP/T4tcsDy7J033336f3331ePHj00ceJERUZG6ujRo1q6dKni4uJ08OBBt/zsatSooZEjR+qxxx5Tjx49FBQU5PJrlIoFeNj3339vBQQEWD179rQyMjIKPebjjz+2du/efc5zHD9+3F3hWQ0bNrRuuukmt50fALzNjz/+aFWoUMHq1q2bderUqQL3Z2RkWAsWLHBrDIGBgdbw4cPdeg1XWLp0qSXJWrp0qUvOd77x7J577rGqVq1qpaWlFXp/Tk6OS2IA4D14Py4+T74f+5LLL7/cuuuuu/Ld5nwtP/nkE5uiKtqJEyc8cp2XXnrJkmQ999xzhd6/d+9e67vvvnPb9fft22cFBgZa06dPd9s1SorpafC4F154QRUqVNDbb799zuzp7bffrgsvvFCSKd0MCQnRH3/8oa5duyo0NFSdO3eWJCUlJalXr16qX7++KlWqpCZNmuiBBx7QwYMH853v2WeflcPh0OrVq3XbbbepWrVqql69uu6++24dOHCg0BgWL16sSy+9VJUrV1azZs307rvvuvBVAADv8cILL8jhcOidd95RcHBwgfuDgoJ0yy235H6fk5Ojl19+Wc2aNVNwcLAiIiJ0zz33aNeuXfked/3116tFixZauXKlrr32WlWpUkWNGzfWiy++qJycHEnS+++/L4fDoaysLE2ZMkUOh0MOhyP3HOvWrVOvXr1Uo0YNVapUSa1bt9YHH3yQ7zrOc/z999/5bi9s6kJxYnLauHGjunXrpipVqig8PFzDhg3TsWPHCn0NlyxZos6dO6tatWqqUqWKrr76an399df5jnGORb/99pv69OmjGjVqnLeiNjU1VdWqVVNISEih95/5OhU2zeHIkSMaPHiwatasqZCQEN10003aunWrHA6Hnn322QJxrV27VrfffruqV6+umjVratSoUcrKytKmTZvUrVs3hYaGqlGjRnr55ZfzXefUqVMaPXq0WrdunfvYK6+8UgsWLDjncwNQON6PvfP92Pm8li5dquHDhys8PFxhYWG67bbbtGfPnnzHFvdnUtj79ieffKL27durevXqua/Hfffdl++YtLQ0Pfroo4qKilJQUJDq1aunESNG6Pjx4+eM32n16tX65ZdfNGDAgCKPdb5G69evV9++fVW9enXVrl1b9913n44ePZp7XJs2bXTttdcWeHx2drbq1aun2267Lfe2zMxMjRs3Lve1qVWrlgYNGlTg85izXci8efPUpk0bVapUSc8995zbX6PTp0/rpZdeUrNmzfTUU08VekydOnV0zTXX5H5/6NAhxcXFqV69egoKClLjxo315JNPKiMjo1SvUe3atdWlSxclJCScN1aPsjtrBf+SlZVlVa5c2bryyiuL/ZiBAwdaFStWtBo1amRNmDDB+vrrr60vv/zSsizLmjJlijVhwgRr4cKF1vLly60PPvjAuuSSS6ymTZtamZmZued45plnLElWw4YNrccee8z68ssvrYkTJ1pVq1a12rRpk+/Yhg0bWvXr17eio6OtDz/80Pryyy+t22+/3ZJkLV++3HUvBgB4gaysLKtKlSpW+/bti/2Y+++/35JkPfTQQ9bixYuthIQEq1atWlZkZKR14MCB3OM6dOhghYWFWf/4xz+shIQEKykpyYqLi7MkWR988IFlWZaVkpJi/fTTT5Ykq0+fPtZPP/1k/fTTT5ZlWdbGjRut0NBQ66KLLrI+/PBD6//+7/+svn37WpKsl156Kfc67733niXJ2rZtW744C/tf6OLEZFnmf/oiIiKsevXqWe+99571xRdfWP3797caNGhQ4Jz/+9//LIfDYfXu3duaN2+e9dlnn1k333yzVaFCBWvJkiW5x505Fj3xxBNWUlKSNX/+/HO+zuPGjbMkWX379rWWLVt23v9lHThwoNWwYcPc77Ozs61rrrnGqlSpkvXiiy9aX331lfXcc89Z//jHPyxJ1jPPPFMgrqZNm1rPP/+8lZSUZD3++OO5P+NmzZpZr7/+upWUlGQNGjTIkmTNnTs39/FHjhyx7r33Xut///uf9c0331iLFy+2Hn30USsgICDfawrg/Hg/9t73Y+fzaty4sfXwww9bX375pZWYmGjVqFHD6tixY6l+Jme/b//444+Ww+Gw7rrrLuuLL76wvvnmG+u9996zBgwYkHvM8ePHrdatW1vh4eHWxIkTrSVLllivvfaaVb16datTp05FVqD++9//tipUqGAdO3as0J/PmZVGZ44NTz/9tJWUlGRNnDjRCg4OtgYNGpR73GuvvWZJsv7888985/ziiy8sSdbChQstyzLjUrdu3ayqVatazz33nJWUlGQlJiZa9erVs6Kjo/ONcQ0bNrTq1q1rNW7c2Hr33XetpUuXWr/88ovbX6Mff/zRkmQ98cQT530dnU6ePGm1atXKqlq1qvXKK69YX331lfXUU09ZgYGBVo8ePUr8Gjm99NJLVkBAgHX48OFixeFuJI3gUfv27bMkFSiJtCwzUJ4+fTr3y/kHPXDgQEuS9e6775733Dk5Odbp06et7du3W5Lyle463/RGjhyZ7zHTp0+3JFkfffRR7m0NGza0KlWqZG3fvj33tpMnT1o1a9a0HnjggVI9bwDwVud7Xy7Mhg0bLElWXFxcvtt//vlnS5I1duzY3Ns6dOhgSbJ+/vnnfMdGR0dbN954Y77bJFkPPvhgvtvuuusuKzg42NqxY0e+27t3725VqVLFOnLkiGVZJf+QUpyYnnjiCcvhcFhr1qzJd1yXLl3ynfP48eNWzZo1rZ49e+Y7Ljs727rkkkusyy+/PPc251j09NNPW8Vx6tQpq3fv3pYkS5JVoUIFq02bNtaTTz5ppaSk5Dv27A8f//d//2dJsqZMmZLvuAkTJpwzafTqq6/mO7Z169aWJGvevHm5t50+fdqqVauWddttt50zbud4PnjwYKtNmzbFeq4AeD8+V0ze8H7sfF5nv9Yvv/yyJcnau3evZVkl+5mc/b79yiuvWJJyX8vCTJgwwQoICLBWrlyZ7/Y5c+ZYkqwvvvjivM+je/fuVrNmzQrcfr6k0csvv5zv2Li4OKtSpUq5n9UOHjxoBQUF5XtulmVZd9xxh1W7dm3r9OnTlmVZ1syZMwv8p4NlWdbKlSstSdbkyZNzb2vYsKFVoUIFa9OmTfmOdfdrNGvWLEuSlZCQcM5jzpSQkGBJsj7++ON8tzunuH311VeWZRX/NXJKSkqyJFmLFi0qVhzuxvQ0eI22bduqYsWKuV+vvvpqvvtjY2MLPCYlJUXDhg1TZGSkAgMDVbFiRTVs2FCSWXHmbP3798/3/R133KHAwEAtXbo03+2tW7dWgwYNcr+vVKmSLr74Ym3fvr3Uzw8AfIHz/fLee+/Nd/vll1+u5s2bF5gCUKdOHV1++eX5bmvVqlWx3k+/+eYbde7cWZGRkfluv/fee3XixAn99NNPpXgGxYtp6dKliomJ0SWXXJLvuH79+uX7/scff9ShQ4c0cOBAZWVl5X7l5OSoW7duWrlyZYFy+MLGs8IEBwfr008/VXJysv773//qrrvu0oEDBzR+/Hg1b95cmzZtOudjly9fLsmMc2fq27fvOR9z9sqhzZs3l8PhUPfu3XNvCwwMVJMmTQr8/D755BNdffXVCgkJyR2Pp02bVuhYDMA1eD/23Pux05lTA52xSsqNt6Q/kzNddtllksz79scff6zdu3cXOObzzz9XixYt1Lp163zP8cYbbyzWSnJ79uxRRETEeY85W2HP+dSpU0pJSZEkhYWFqWfPnvrggw9ypxUePnxYCxYs0D333KPAwMDc2C+44AL17NkzX+ytW7dWnTp1CsTeqlUrXXzxxflu88RrVBLffPONqlatqj59+uS73fnzd/68i/saOTl/RoU9PzuQNIJHhYeHq3LlyoUOTjNmzNDKlSu1cOHCAvdVqVIld4UYp5ycHHXt2lXz5s3T448/rq+//lq//PKLVqxYIUk6efJkgfPUqVMn3/eBgYEKCwtTampqvtvDwsIKPDY4OLjQcwJAeRYeHq4qVapo27ZtxTre+X5Zt27dAvddeOGFLn0/TU1NPed1zoylpIoTU2pqaoExQyo4juzfv1+S1KdPn3z/8VGxYkW99NJLsiyrwPLYhT2n82nevLlGjBihjz76SDt27NDEiROVmpp6zn4LzvgDAwNVs2bNfLfXrl37nI85+9igoCBVqVJFlSpVKnD7qVOncr+fN2+e7rjjDtWrV08fffSRfvrpJ61cuVL33XdfvuMAnB/vx4XH5E3vx2fH6+w75Yy3pD+TM1133XWaP3++srKydM8996h+/fpq0aKFZs6cme85rl27tsDzCw0NlWVZBfq6nu3kyZMF3tOLUtRzlsyKe7t371ZSUpIkaebMmcrIyMiXPNu/f7+OHDmioKCgAvHv27evQOyFvYbufo2cRQMl+RusU6dOvt5fkkn6BAYG5vt5F+c1cnL+jLzls2dg0YcArlOhQgV16tRJX331lfbu3ZvvzSA6OlqSCjTOk1TgD1Eyzfh+//13vf/++xo4cGDu7Zs3bz7n9fft26d69erlfp+VlaXU1NRCBywA8AcVKlRQ586dtWjRIu3atUv169c/7/HO98u9e/cWOHbPnj0KDw93WWxhYWHau3dvgdudTUed13L+4+rMppOSivzHc1HX3rdvX4Hbz77NGcMbb7yhK664otBznZ2oKWxMKy6Hw6GRI0fq3//+t9atW3fO48LCwpSVlaVDhw7lSwYV9pzK6qOPPlJUVJRmz56d77md/fMAcH68H5/72t74flyYsv5MevXqpV69eikjI0MrVqzQhAkT1K9fPzVq1EhXXnll7n/An2uBnqLOHx4eXiBx5go33nijLrzwQr333nu68cYb9d5776l9+/a5n++c1w4LC9PixYsLPUdoaGi+78/1s3Hna9SuXTvVrFlTCxYs0IQJE4r8/QgLC9PPP/8sy7LyHZuSkqKsrKx81yrOa+Tk/Bm58m+4LKg0gseNGTNG2dnZGjZsmE6fPl3q8zj/MM9eWeLtt98+52OmT5+e7/uPP/5YWVlZuv7660sdBwCUd2PGjJFlWRo6dKgyMzML3H/69Gl99tlnkqROnTpJMomCM61cuVIbNmzIXd3SFTp37qxvvvmmwMo0H374oapUqZL7ocC5+szatWvzHVdY5WpxdezYUevXr9fvv/+e7/YZM2bk+/7qq6/WBRdcoOTkZLVr167Qr3OtFFqUwj6gSeaDR1paWu7/8BemQ4cOkqTZs2fnu33WrFmliuV8HA6HgoKC8v2Ded++fayeBpQC78cFecP7cXG56mcSHBysDh066KWXXpJkVj2TzDTiLVu2KCwsrNDnd/ZqbGdr1qyZtm7dWsJnVbQKFSpowIABmj9/vr777jutWrWqwIpmN998s1JTU5WdnV1o7E2bNi3RNd3xGlWsWFFPPPGENm7cqOeff77QY1JSUvTDDz9IMn8X6enpmj9/fr5jPvzww9z7S/IaOTl/RoUllOxApRE87uqrr9Zbb72lhx9+WJdeeqnuv/9+xcTEKCAgQHv37tXcuXMlqcB0tLM1a9ZMF110kf71r3/JsizVrFlTn332WW7JX2HmzZunwMBAdenSRevXr9dTTz2lSy65pEDPBwDwJ1deeaWmTJmiuLg4tW3bVsOHD1dMTIxOnz6t1atX65133lGLFi3Us2dPNW3aVPfff7/eeOMNBQQEqHv37vr777/11FNPKTIyUiNHjnRZXM8884w+//xzdezYUU8//bRq1qyp6dOn6//+7//08ssvq3r16pJMj4OmTZvq0UcfVVZWlmrUqKFPP/1U33//famvPWLECL377ru66aabNG7cONWuXVvTp0/Xxo0b8x0XEhKiN954QwMHDtShQ4fUp08fRURE6MCBA/r999914MABTZkypVQx3H///Tpy5IhiY2PVokULVahQQRs3btR///tfBQQE6IknnjjnY7t166arr75ao0ePVlpamtq2bauffvop9x+yAQGu+39D57LIcXFx6tOnj3bu3Knnn39edevW1V9//eWy6wD+gPfjgrzh/bi4yvIzefrpp7Vr1y517txZ9evX15EjR/Taa6+pYsWKuf8RMGLECM2dO1fXXXedRo4cqVatWiknJ0c7duzQV199pdGjR6t9+/bnvMb111+vd999V3/++WeBfkFldd999+mll15Sv379VLlyZd1555357r/rrrs0ffp09ejRQ4888oguv/xyVaxYUbt27dLSpUvVq1cv3Xrrree9hideo8cee0wbNmzQM888o19++UX9+vVTZGSkjh49qm+//VbvvPOOnnvuOV199dW655579NZbb2ngwIH6+++/1bJlS33//fd64YUX1KNHD91www0leo2cVqxYobCwMLVs2bI4L7372deDG/5uzZo11qBBg6yoqCgrODjYqlSpktWkSRPrnnvusb7++uvc4wYOHGhVrVq10HMkJydbXbp0sUJDQ60aNWpYt99+u7Vjx45zrgzz66+/Wj179rRCQkKs0NBQq2/fvtb+/fvznbNhw4bWTTfdVOBaHTp0sDp06OCS5w4A3mjNmjXWwIEDrQYNGlhBQUFW1apVrTZt2lhPP/10vtW6srOzrZdeesm6+OKLrYoVK1rh4eHW3Xffbe3cuTPf+Tp06GDFxMQUuM7ZK8ZYVuGr9ViWZf3xxx9Wz549rerVq1tBQUHWJZdcYr333nsFjvvzzz+trl27WtWqVbNq1aplPfzww7kriJ29Wk9xY3KOMZUqVbJq1qxpDR482FqwYEGBc1qWZS1fvty66aabrJo1a1oVK1a06tWrZ910002FrkRz5pLL5/Pll19a9913nxUdHW1Vr17dCgwMtOrWrWvddtttuctgny/+Q4cOWYMGDbIuuOACq0qVKlaXLl2sFStWWJKs1157rci4zjX+FvYavvjii1ajRo2s4OBgq3nz5tbUqVNzzwug5Hg/zh+T3e/HztXTzl6Rq7BV4Yr7Mzn7eX7++edW9+7drXr16llBQUFWRESE1aNHD+u7777L97j09HQrPj7eatq0qRUUFGRVr17datmypTVy5Ehr3759530eR48etUJCQgqsiHa+1dPOfo3OtUKeZVnWVVddZUmy+vfvX+j1T58+bb3yyivWJZdcYlWqVMkKCQmxmjVrZj3wwAPWX3/9lXvcuT6PeeI1clqwYIF10003WbVq1bICAwOtGjVqWB07drQSEhKsjIyM3ONSU1OtYcOGWXXr1rUCAwOthg0bWmPGjLFOnTpV6HmLeo1ycnKshg0bWg8//HCx4vQEh2VZlmfTVIDnPfvss3ruued04MABr5kbCgCAp82YMUP9+/fXDz/8oKuuusrucAAAHvbwww/r66+/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",
      "text/plain": [
       "<Figure size 1500x400 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, axs = plt.subplots(1, 3) \n",
    "fig.set_figwidth(15)\n",
    "fig.set_figheight(4)\n",
    "\n",
    "pos = nx.circular_layout(graph)\n",
    "\n",
    "draw_networkx_nodes(graph, pos, ax=axs[0], node_size=300, node_color='white', edgecolors='black')\n",
    "draw_networkx_labels(graph, pos, ax=axs[0], font_size=8, font_color=\"black\")\n",
    "draw_curved_edges(graph, pos, axs[0], curve_scale=0.1, node_size=300)\n",
    "\n",
    "_ = axs[1].imshow(sem.confounder_sigma, cmap='bwr', vmin=-1, vmax=1)\n",
    "\n",
    "_ = axs[2].imshow(sem.confounder_inverse_cov, cmap='bwr', vmin=-5, vmax=5)\n",
    "\n",
    "axs[0].set_title(\"Graph\")\n",
    "axs[1].set_title(\"Confounder Sigma\")\n",
    "_ = axs[2].set_title(\"Confounder noise (Inverse Cov)\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "f2f9c619",
   "metadata": {},
   "outputs": [],
   "source": [
    "n_inter = 5\n",
    "intervention_sets = [[None]] + [[i] for i in range(n_inter)]\n",
    "\n",
    "datasets = list()\n",
    "intervention_torch_datasets = list()\n",
    "for targets in intervention_sets:\n",
    "    data = sem.generateData(n_samples=n_samples, intervention_set=targets)\n",
    "    datasets.append(data)\n",
    "\n",
    "    intervention_torch_datasets.append(\n",
    "        InterventionDataset(data, targets)\n",
    "    )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "7f056b3a",
   "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": 11,
   "id": "b6f418d6",
   "metadata": {},
   "outputs": [],
   "source": [
    "implicit_block = imBlock(\n",
    "    nnet_x=g_x, \n",
    "    nnet_z=g_z,\n",
    "    confounders=True\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "f751b600",
   "metadata": {},
   "outputs": [],
   "source": [
    "im_trainer = DCCDTrainer(\n",
    "    implicit_block, \n",
    "    max_epochs=200, \n",
    "    batch_size=512, \n",
    "    lr=1e-1,\n",
    "    lambda_c=1e-3, \n",
    "    rho=1e-2\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "a9f2ddaf",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 200/200, Intervention: 6/6, log P(X): 1.58\n"
     ]
    }
   ],
   "source": [
    "im_trainer.train(\n",
    "    intervention_datasets=datasets,\n",
    "    intervention_targets=intervention_sets,\n",
    "    verbose=True\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "8b34db6c",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0.5, 1.0, 'Estimated')"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAhYAAAEhCAYAAAAj0OlbAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjAsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvlHJYcgAAAAlwSFlzAAAPYQAAD2EBqD+naQAAH2pJREFUeJzt3Xlw1PX9x/HXkpCDkCwEDAIJENERkAaFoI1armBKjAj1KHhAOLQyRgaGzlQjrUAP4tWZWkUKHqBVRKlyFCoFEQJKtUGNoniMHRAsIAhDNgQJJvn8/vCXrWsS2E0+my/f3edj5vvHfvZ7vHfZ7zuv/R6LxxhjBAAAYEEbpwsAAACRg2ABAACsIVgAAABrCBYAAMAaggUAALCGYAEAAKwhWAAAAGsIFgAAwBqCBQAAsIZg0Uo++OADTZ06Vb1791ZiYqISExN1wQUX6I477tCOHTucLq9FPB6P5s6d2+Tzw4YNk8fjOeN0unUE48SJE5o7d662bNnS4Lm5c+fK4/Ho66+/btE2gLPV0qVLT7t/NbZfNGb//v2aO3euysvLGzxXvx85YdeuXZo7d6727Nljfd1Ovq5IFOt0AdFg0aJFuuuuu3ThhRdqxowZuuiii+TxePTxxx/rhRde0ODBg/X555+rd+/eTpcaFo8//rh8Pp//8bp16/T73/9eS5YsUZ8+ffzj6enpLdrOiRMnNG/ePEnfhRkgGv1wv6rXr1+/oJbfv3+/5s2bp169euniiy8OeO62227TqFGjbJQZsl27dmnevHkaNmyYevXq5UgNCA7BIszefPNN3XnnnSooKNDf/vY3xcXF+Z8bMWKEioqKtGLFCiUmJp52PSdOnFC7du3CXW5Y/LChffLJJ5Kk/v37Kzs7u8nl3PyaAaecab9qifT09BZ/AUDk41RImM2fP18xMTFatGhRQKj4vhtvvFHdunXzP540aZLat2+vnTt3Ki8vT8nJycrNzZUkHT16VHfeeae6d++uuLg4nXfeeZo9e7aqq6v9y+/Zs0cej0dLly5tsK0fnnKoPwT40Ucf6aabbpLX61WXLl00ZcoUVVRUBCzr8/l0++23q1OnTmrfvr1GjRqlzz77rAXvzv/U1/Huu+/qhhtuUMeOHf1HcIYNG9boEYhJkyb5v7ns2bNH55xzjiRp3rx5/sO/kyZNCljmq6++OuPrBCLZihUrdNlll8nr9apdu3Y677zzNGXKFEnSli1bNHjwYEnS5MmTG5ymbOyUQa9evXTNNddo7dq1uuSSS5SYmKi+fftq7dq1kr47RdO3b18lJSXp0ksvbXDqd8eOHRo/frx69eqlxMRE9erVSzfddJO++OIL/zxLly7VjTfeKEkaPny4v67v97jXXntNubm5SklJUbt27XTFFVdo06ZNDV7/unXrdPHFFys+Pl6ZmZl6+OGHW/aGogGOWIRRbW2tNm/erOzsbHXt2jWkZU+dOqVrr71Wd9xxh+655x7V1NTo5MmTGj58uP7zn/9o3rx5ysrK0rZt21RSUqLy8nKtW7eu2bVef/31GjdunKZOnaqdO3equLhYkvT0009LkowxGjt2rLZv36777rtPgwcP1ptvvqn8/Pxmb7Mx1113ncaPH69p06apqqoq6OW6du2q9evXa9SoUZo6dapuu+02SfKHjXpnep2A29XW1qqmpiZgzOPxKCYmRv/61780btw4jRs3TnPnzlVCQoK++OILvf7665KkgQMHasmSJZo8ebJ+/etfq6CgQNKZT1O+//77Ki4u1uzZs+X1ejVv3jxdd911Ki4u1qZNmzR//nx5PB7dfffduuaaa7R7927/Udo9e/bowgsv1Pjx45WamqoDBw5o4cKFGjx4sHbt2qXOnTuroKBA8+fP17333qsFCxZo4MCBkuT/8vHcc89p4sSJGjNmjJ555hm1bdtWixYt0k9/+lP985//9H8x27Rpk8aMGaOcnBwtX75ctbW1evDBB/XVV1/Z+weAZBA2Bw8eNJLM+PHjGzxXU1Njvv32W/9UV1fnf66wsNBIMk8//XTAMn/5y1+MJPPSSy8FjD/wwANGktmwYYMxxpjdu3cbSWbJkiUNtivJzJkzx/94zpw5RpJ58MEHA+a78847TUJCgr+uV1991UgyjzzySMB8f/jDHxqs80yWLFliJJmysrIGddx3330N5h86dKgZOnRog/HCwkLTs2dP/+PDhw83WUuwrxNwq/r9qrEpJibGGGPMww8/bCSZY8eONbmesrKyJvtH/X70fT179jSJiYnmyy+/9I+Vl5cbSaZr166mqqrKP75q1SojyaxZs6bJ7dfU1Jjjx4+bpKSkgH6zYsUKI8ls3rw5YP6qqiqTmppqRo8eHTBeW1trBgwYYC699FL/2GWXXWa6detmvvnm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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",
    "axs[0].imshow(np.abs(sem.weights) > 0, cmap='bwr', vmin=-1, vmax=1)\n",
    "axs[0].set_title(\"Ground Truth\")\n",
    "\n",
    "axs[1].imshow(g_x.get_w_adj().detach().cpu().numpy(), cmap='bwr', vmin=-1, vmax=1)\n",
    "axs[1].set_title(\"Estimated\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "cc4a7d83",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Text(0.5, 1.0, 'Estimated')"
      ]
     },
     "execution_count": 17,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAhYAAAEhCAYAAAAj0OlbAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjAsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvlHJYcgAAAAlwSFlzAAAPYQAAD2EBqD+naQAAH0ZJREFUeJzt3XtwVPXdx/HPkpBNCEkwIAgkkAgWEBosBG28cNWUELmMaPEGEbSVITAwOFONtAJ1ShTtTKkgBS9Qq4hSBSlWCuWuFBvQCIrVwQckFjAKQzZECST5PX/4ZB/XJLAbfpvD2bxfM/vHHs6e89mw+81nz9ndeIwxRgAAABa0cDoAAACIHBQLAABgDcUCAABYQ7EAAADWUCwAAIA1FAsAAGANxQIAAFhDsQAAANZQLAAAgDUUiya0d+9e3XvvverWrZvi4uIUFxenK664Qvfff792797tdLwL4vF4NGfOnHOuc+TIEc2ZM0fFxcVhyXCu7d9zzz1q3bp1WPYLXAyWL18uj8fT4GXr1q1Bbedcz6M5c+bI4/HYDR6k/fv3a86cOTp06JD1bTt5vyJRtNMBmoslS5Zo6tSp6tGjh6ZPn67evXvL4/Ho448/1ssvv6wBAwbowIED6tatm9NRw+bIkSOaO3eu0tLSdNVVV7lu+4AbLFu2TD179qyz/Morrwzq9ud6Ht13330aPny4jZgh279/v+bOnavBgwcrLS3NkQwIDsWiCbzzzjuaMmWKcnNz9de//lUxMTH+fxs6dKjy8/O1atUqxcXFnXM733zzjVq1ahXuuBeN5nZ/ARv69OmjzMzMsGw7JSVFKSkpYdk2IgenQprAvHnzFBUVpSVLlgSUiu+77bbb1KlTJ//12kP3+/btU3Z2thISEjRs2DBJ0okTJzRlyhR17txZMTExuvzyyzVr1ixVVlb6b3/o0CF5PB4tX768zr5+eNqi9jDgRx99pDvuuENJSUnq0KGDJk2apLKysoDb+nw+/eIXv1Dbtm3VunVrDR8+XJ9++ul5fwZbt27VgAEDJEkTJ070H56tzXGu+5uWlqZ77rmnzjYHDx6swYMHB7X9WgcOHNCIESPUunVrpaam6oEHHgj4uQGRbtWqVbrmmmuUlJSkVq1a6fLLL9ekSZMknf95VN8pg7S0NN18881at26dfvKTnyguLk69evXSunXrJH13iqZXr16Kj4/X1VdfXee07+7du3X77bcrLS1NcXFxSktL0x133KHPP//cv87y5ct12223SZKGDBniz/X9+fbPf/5Tw4YNU2Jiolq1aqXrrrtOmzZtqnP/33zzTV111VXyer1KT0/Xk08+eWE/UNTBEYswq66u1pYtW5SZmamOHTuGdNszZ85o1KhRuv/++/XQQw+pqqpKp0+f1pAhQ/TZZ59p7ty5ysjI0I4dO1RYWKji4mK9+eabjc46duxYjRs3Tvfee6/27dungoICSdLzzz8vSTLGaMyYMdq5c6ceeeQRDRgwQO+8845ycnLOu+1+/fpp2bJlmjhxon79618rNzdXkgJe/dR3f4MVzPbPnj2rUaNG6d5779UDDzyg7du369FHH1VSUpIeeeSRoPcFXMyqq6vrPHc8Ho+ioqL0r3/9S+PGjdO4ceM0Z84cxcbG6vPPP9fmzZslBfc8qs8HH3yggoICzZo1S0lJSZo7d65uueUWFRQUaNOmTZo3b548Ho8efPBB3XzzzTp48KD/CO2hQ4fUo0cP3X777UpOTtbRo0e1ePFiDRgwQPv371e7du2Um5urefPm6eGHH9aiRYvUr18/SfKfOn7xxRc1YcIEjR49Wn/+85/VsmVLLVmyRD/72c/0j3/8w/8iZdOmTRo9erSysrK0cuVKVVdXa/78+fryyy/t/QdAMgirY8eOGUnm9ttvr/NvVVVV5uzZs/5LTU2N/9/y8vKMJPP8888H3OZPf/qTkWReffXVgOWPP/64kWQ2bNhgjDHm4MGDRpJZtmxZnf1KMrNnz/Zfnz17tpFk5s+fH7DelClTTGxsrD/XW2+9ZSSZBQsWBKz3u9/9rs4261NUVNRgpoburzHGdO3a1eTl5dVZPmjQIDNo0KCQtv/Dn9uIESNMjx49zpkbcINly5YZSfVeoqKijDHGPPnkk0aSOXnyZIPbOdfzqHZWfF/Xrl1NXFyc+eKLL/zLiouLjSTTsWNHU1FR4V++Zs0aI8msXbu2wf1XVVWZU6dOmfj4+IBZs2rVKiPJbNmyJWD9iooKk5ycbEaOHBmwvLq62vTt29dcffXV/mXXXHON6dSpk/n222/9y3w+n0lOTq5zv9B4nApxUP/+/dWyZUv/5fe//32ddcaOHRtwffPmzYqPj9ett94asLz2VEF9h/6CNWrUqIDrGRkZOn36tEpLSyVJW7ZskSTdddddAevdeeedjd7nD/3w/trk8Xg0cuTIgGUZGRkBh1wBt3vhhRdUVFQUcHn33XclyX+a4+c//7leffVV/fe//7Wyz6uuukqdO3f2X+/Vq5ek705Xfv99UrXLv/+cO3XqlB588EF1795d0dHRio6OVuvWrVVRUaGPP/74vPveuXOnTpw4oby8PFVVVfkvNTU1Gj58uIqKilRRUaGKigoVFRXplltuUWxsrP/2CQkJdeYCLgynQsKsXbt2iouLq/eX14oVK/TNN9/o6NGjdX6pS1KrVq2UmJgYsOz48eO67LLL6pznbN++vaKjo3X8+PFGZ23btm3Ada/XK0n69ttv/fuOjo6us95ll13W6H1+X33316ZWrVoFDBTpu/t4+vTpsO0TaGq9evVq8M2bAwcO1Jo1a/THP/5REyZMUGVlpXr37q1Zs2bpjjvuaPQ+k5OTA67XvpesoeXff87deeed2rRpk37zm99owIABSkxMlMfj0YgRI/yz51xqT2P88MXW9504cUIej0c1NTX1zitbMwzfoViEWVRUlIYOHaoNGzbo6NGjAe+zqP34V0Ofy67vc9Vt27bVu+++K2NMwL+XlpaqqqpK7dq1kyT/L9AfvjHxQotHVVWVjh8/HlAujh071uhtfl9DnyOPjY2t9w2WX3/9tf/+AgjO6NGjNXr0aFVWVmrXrl0qLCzUnXfeqbS0NGVlZTVplrKyMq1bt06zZ8/WQw895F9eWVmpEydOBLWN2hnw1FNP6ac//Wm963To0EFnz56Vx+Opd17ZmmH4DqdCmkBBQYGqq6s1efJknT179oK2NWzYMJ06dUpr1qwJWP7CCy/4/1367okUGxurvXv3Bqz3xhtvNHrfQ4YMkSS99NJLActXrFgR1O1/eAQkWGlpaXXux6effqpPPvnEyvaB5sjr9WrQoEF6/PHHJUnvv/++f7nUNM8jj8cjY4x/n7WeffZZVVdX18lbX67rrrtObdq00f79+5WZmVnvJSYmxv+plNdffz3giEl5ebn+9re/hekeNk8csWgC1113nRYtWqRp06apX79++uUvf6nevXurRYsWOnr0qF577TVJCuo0wIQJE7Ro0SLl5eXp0KFD+vGPf6y3335b8+bN04gRI3TjjTdK+u4Je/fdd+v5559Xt27d1LdvX/373/8OugTUJzs7WwMHDtSvfvUr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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",
    "axs[0].imshow(sem.confounder_sigma, cmap='bwr', vmin=-1, vmax=1)\n",
    "axs[0].set_title(\"Ground truth\")\n",
    "\n",
    "axs[1].imshow(implicit_block.cov_mat.detach().cpu().numpy(), cmap='bwr', vmin=-1, vmax=1)\n",
    "axs[1].set_title(\"Estimated\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ac915a66",
   "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
}
