{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5f270e84",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os \n",
    "import torch \n",
    "import numpy as np \n",
    "import random\n",
    "import pandas as pd\n",
    "\n",
    "from matplotlib import pyplot as plt \n",
    "\n",
    "plt.rcParams[\"font.family\"] = \"Helvetica\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "3dce22df",
   "metadata": {},
   "outputs": [],
   "source": [
    "project_dir = \"../../\"\n",
    "\n",
    "os.chdir(project_dir)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "bf81ebf3",
   "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": "code",
   "execution_count": 4,
   "id": "512fb398",
   "metadata": {},
   "outputs": [],
   "source": [
    "data_dir = \"datasets/perturb-cite-seq\"\n",
    "\n",
    "condition = \"cocult\"\n",
    "\n",
    "condition_dir = os.path.join(data_dir, condition)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "0be2b4d3",
   "metadata": {},
   "outputs": [],
   "source": [
    "training_datasets = [\n",
    "    np.load(os.path.join(condition_dir + \"/training_data\", f\"dataset-{i}.npy\"))\n",
    "    for i in range(55)\n",
    "]\n",
    "\n",
    "validation_datasets = [\n",
    "    np.load(os.path.join(condition_dir + \"/validation_data\", f\"dataset-{i}.npy\")) for i in range(6)\n",
    "]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2f35f467",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "tot_dataset = training_datasets + validation_datasets\n",
    "\n",
    "# centering the data\n",
    "tot_dataset_centered = list()\n",
    "for dataset in tot_dataset:\n",
    "    dataset_centered = dataset - dataset.mean(axis=0)\n",
    "    tot_dataset_centered.append(dataset_centered)\n",
    "\n",
    "chosen_genes = list(range(61))\n",
    "total_datasets = [tot_dataset_centered[i][:, chosen_genes] for i in chosen_genes]\n",
    "total_interventions = [[i] for i in range(len(total_datasets))]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "d03681af",
   "metadata": {},
   "outputs": [],
   "source": [
    "combined = list(zip(total_interventions, total_datasets))\n",
    "random.shuffle(combined)\n",
    "\n",
    "shuffled_interventions, shuffled_datasets = zip(*combined)\n",
    "\n",
    "n_valid_data = 6\n",
    "\n",
    "training_intervention_sets = shuffled_interventions[:-n_valid_data]\n",
    "training_dataset_centered = shuffled_datasets[:-n_valid_data]\n",
    "\n",
    "validation_dataset_centered = shuffled_datasets[-n_valid_data:]\n",
    "validation_intervention_sets = shuffled_interventions[-n_valid_data:]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b4f5087c",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 20/20, Intervention: 55/55, log P(X): -79.218\n"
     ]
    }
   ],
   "source": [
    "causal_mech = gumbelSoftMLP(\n",
    "    n_nodes=61,\n",
    "    lip_constant=0.9,\n",
    "    activation=\"relu\"\n",
    ")\n",
    "\n",
    "g_z = gnet_z(\n",
    "    n_nodes=61\n",
    ")\n",
    "\n",
    "implicit_block = imBlock(\n",
    "    nnet_x=causal_mech, \n",
    "    nnet_z=g_z,\n",
    "    confounders=True\n",
    ")\n",
    "\n",
    "trainer = DCCDTrainer(\n",
    "    implicit_block, \n",
    "    max_epochs=100, \n",
    "    batch_size=1024, \n",
    "    lr=1e-2,\n",
    "    lambda_c=1e-3, \n",
    "    rho=1e-1,\n",
    "    glasso_iters=1\n",
    ")\n",
    "\n",
    "trainer.train(\n",
    "    intervention_datasets=training_dataset_centered,\n",
    "    intervention_targets=training_intervention_sets,\n",
    "    verbose=True,\n",
    "    sleep_time=False\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "331b861c",
   "metadata": {},
   "outputs": [],
   "source": [
    "genes = pd.read_csv(\"datasets/perturb-cite-seq/genes.csv\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "0c2d742d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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SCk/efKkfu7cufoTJ06o9lHLli0zuZ6DBw9WtnXj7X3YsGH63r1766dOnaofNWqUUr537179pEmT9NevX1fVExAQoPlZmNKvXz/90qVL9ZmZmaryQ4cO6ePi4pQfg/Pnzyv7oOPHjz+1/sWLFyvbkKl9xdMUyR6YFq1hB39/fy5cuMCAAQNUJxIXL17MrFmzyMjIYPLkycq4toWFBTt37iQpKYmPP/5Y6b0kJiZiZ2fH2LFj+fLLL5V6tMpr167N1KlTuXjxomoYTKvcWO3atVVDKqVLlzb5PK2ufpUqVbh27RqfffaZKtzx22+/YWFhoTzfcP4lMTGRzMxMIiIilHN4AD4+PqSmpmarv1ixYkydOjXbsMykSZN48cUXSU1NZe/evcpQwr59+3j77bdp06YNK1asoH///gC0bduWatWqsWDBAtVyDd+FloULF2Y7QjScb5sxY4bqseLFizN8+HCCgoJUR+haz7979y6zZs3Cw8ODhQsXqspnz56Nh4cH3377rVJ+584dYmJiaNWqlWp9Ll++zLVr13jppZdU5VrDtcblS5cuVcovXrxISEgIPj4+qp7WxYsXqV27Nt27d1eGxUH7fM306dNp3749nTp1Uq3PsGHDGDRoEDqdTtWLP3jwIN26daNevXp4e3vz5ptvAuDt7c3gwYO5du0aR48eVYYAz507x/z585VltW3bVqmrd+/eVKxYEVDfhffw4cOcPn2aL774gi+//FL5TozP8RgP8/7000/07duXBg0aqAIkWn8HrVu3pnXr1mT1559/smLFCkJDQ5Vzn4b6DefnJk6cSNOmTQGYNWsWn376Kfb29qptws7OjkmTJgHqIdD169czevRoRo0axVdffaWU79+/XzV8azhXN336dBYtWoS9vb3qtIXWsPhPP/2En58fAQEBqnIthnPihu3f8PloDf317NmTtm3bUr9+fQCl5z9v3jwWLFjAiRMnCAkJUU4TaO0nIiIiGDRoEPXq1cPX15devXo9dV2NFckGzHCCH1Cdi9AadlixYgVdu3Zl6dKlPHjwgBYtWgCPz8tYWVlhZWWlGkcGGDp0KJGRkSxZskTZwJs3b86BAwe4fv06Vlb//Wi1ytu3b8/Fixfp06ePashDqxy0wxdaY+daXf3IyEjOnz+Pn5+fKnH24osv0r1792zPT05O5v79+8yZM0fVKLm5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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, axs = plt.subplots(1)\n",
    "\n",
    "axs.imshow(implicit_block.nnet_x.get_w_adj().detach().cpu().numpy() > 0.5, cmap=\"bwr\", vmin=-1, vmax=1)\n",
    "\n",
    "axs.set_xticks(range(61))\n",
    "_ = axs.set_yticks(range(61))\n",
    "_ = axs.set_xticklabels(genes[\"Genes\"], rotation=90, fontsize=4.5)\n",
    "_ = axs.set_yticklabels(genes[\"Genes\"], fontsize=4.5)\n",
    "\n",
    "fig.savefig(\"../results/pcs-adj-mat.pdf\", bbox_inches=\"tight\")\n",
    "\n"
   ]
  }
 ],
 "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
}
