{
 "cells": [
  {
   "cell_type": "code",
   "id": "f84f0be0",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-10-20T15:12:28.907442Z",
     "start_time": "2024-10-20T15:12:27.460470Z"
    }
   },
   "source": [
    "import networkx as nx\n",
    "import numpy as np\n",
    "import torch\n",
    "from model import CaT\n",
    "from CaT.datasets import reorder_dag, get_full_ordering\n",
    "from utils.inference import CausalInference\n",
    "import matplotlib.pyplot as plt\n",
    "from datasets import get_full_ordering, reorder_dag\n",
    "\n",
    "shuffling = 0\n",
    "seed = 1\n",
    "standardize = 0\n",
    "sample_size = 100000\n",
    "batch_size = 100\n",
    "max_iters = 20000\n",
    "eval_interval = 100\n",
    "eval_iters = 100\n",
    "validation_fraction = 0.3\n",
    "np.random.seed(seed=seed)\n",
    "torch.manual_seed(seed)\n",
    "device = 'cuda'\n",
    "dropout_rate = 0.0\n",
    "learning_rate = 5e-3\n",
    "ff_n_embed = 30\n",
    "num_heads = 2\n",
    "n_layers = 2\n",
    "embed_dim = 30\n",
    "head_size = 30\n"
   ],
   "outputs": [],
   "execution_count": 1
  },
  {
   "cell_type": "markdown",
   "id": "37367d6c",
   "metadata": {},
   "source": [
    "We create data from a 'true' DAG (which is DAGnx3 below) but also provide two other DAGs which are incorrect in different ways. DAGnx1 is fully exogenous, and DAGnx2 has one missing edge and another directed edge reversed.\n",
    "\n",
    "We imagine we are interested in the 'total' effect of X -> Y.  As these simulations are linear, this effect can be calculated easily by hand.\n",
    "\n",
    "In the case where X1 -> Y and X1 -> X2 -> Y, the total effect is the sum of the effect from both paths. e.g. if X1 -> Y has a coefficient of  0.8,  X1 -> X2 of 0.8, and X2 -> Y of 0.4, then the total effect is 0.8 + (0.8 x 0.4) = 1.12"
   ]
  },
  {
   "cell_type": "code",
   "id": "5de6b09c-fb7a-4a95-ab23-954550e69e24",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-10-20T15:12:29.039284Z",
     "start_time": "2024-10-20T15:12:28.908931Z"
    }
   },
   "source": [
    "def generate_data(N, d=1):\n",
    "    DAGnx1 = nx.DiGraph()\n",
    "    DAGnx2 = nx.DiGraph()\n",
    "    DAGnx3 = nx.DiGraph()  # correct graph\n",
    "    \n",
    "    Ux1 = np.random.randn(N,d)\n",
    "    X1 =  Ux1\n",
    "\n",
    "    Ux2 = np.random.randn(N,d)\n",
    "    X2 =  0.8 * X1 + Ux2\n",
    "    \n",
    "    X20 = Ux2\n",
    "    X21 = 0.8 + Ux2\n",
    "\n",
    "    Uy = np.random.randn(N,d)\n",
    "    Y =  0.8 * X1 + 0.4 * X2 + Uy  # 0.4*0.8   + 0.8 = 1.12  Total effect\n",
    "    \n",
    "    Ux3 = np.random.randn(N,d)\n",
    "    X3 = 0.7 * Y + 0.6 * X1 + Ux3\n",
    "\n",
    "    Y0 = 0.4 * X20 + Uy \n",
    "    Y1 = 0.8 + 0.4 * X21 + Uy\n",
    "\n",
    "    all_data_dict = {'X1': X1, 'X2': X2, 'X3': X3, 'Y': Y}\n",
    "\n",
    "    # types can be 'cat' (categorical) 'cont' (continuous) or 'bin' (binary)\n",
    "    var_types = {'X1': 'cont', 'X2': 'cont', 'X3': 'cont', 'Y': 'cont'}\n",
    "\n",
    "    DAGnx1.add_edges_from([('X1', 'Y'), ('X2', 'Y'), ('X3', 'Y')])\n",
    "    DAGnx1 = reorder_dag(dag=DAGnx1)  # topologically sorted dag\n",
    "    var_names1 = list(DAGnx1.nodes())  # topologically ordered list of variables\n",
    "    all_data1 = np.stack([all_data_dict[key] for key in var_names1], axis=1)\n",
    "    causal_ordering1 = get_full_ordering(DAGnx1)\n",
    "    ex_dag_stuff = (all_data1, DAGnx1, var_names1, causal_ordering1, var_types)\n",
    "    \n",
    "    DAGnx2.add_edges_from([('X1', 'Y'), ('X1', 'X2'), ('X2', 'Y'), ('X3', 'Y')])\n",
    "    DAGnx2 = reorder_dag(dag=DAGnx2)  # topologically sorted dag\n",
    "    var_names2 = list(DAGnx2.nodes())  # topologically ordered list of variables\n",
    "    all_data2 = np.stack([all_data_dict[key] for key in var_names2], axis=1)\n",
    "    causal_ordering2 = get_full_ordering(DAGnx2)\n",
    "    mediated_dag_stuff = (all_data2, DAGnx2, var_names2, causal_ordering2, var_types)\n",
    "    \n",
    "    DAGnx3.add_edges_from([('X1', 'Y'), ('X1', 'X2'), ('X1', 'X3'), ('X2', 'Y'), ('Y', 'X3')])\n",
    "    DAGnx3 = reorder_dag(dag=DAGnx3)  # topologically sorted dag\n",
    "    var_names3 = list(DAGnx3.nodes())  # topologically ordered list of variables\n",
    "    all_data3 = np.stack([all_data_dict[key] for key in var_names3], axis=1)\n",
    "    causal_ordering3 = get_full_ordering(DAGnx3)\n",
    "    correct_dag_stuff =  (all_data3, DAGnx3, var_names3, causal_ordering3, var_types)\n",
    "\n",
    "    return ex_dag_stuff, mediated_dag_stuff, correct_dag_stuff, Y0, Y1\n",
    "\n",
    "d=1\n",
    "_, _, _, Y0, Y1 = generate_data(N=1000000, d=d)\n",
    "ATE = (Y1 - Y0).mean(0)  \n",
    "ex_dag_stuff, mediated_dag_stuff, correct_dag_stuff, Y0, Y1 = generate_data(N=sample_size, d=d)\n",
    "\n",
    "all_data1, DAGnx1, var_names1, causal_ordering1, var_types1 = ex_dag_stuff\n",
    "all_data2, DAGnx2, var_names2, causal_ordering2, var_types2 = mediated_dag_stuff\n",
    "all_data3, DAGnx3, var_names3, causal_ordering3, var_types3 = correct_dag_stuff\n",
    "\n",
    "print(var_names1, 'ATE of X1 on Y:', ATE)\n",
    "print(all_data1.shape)"
   ],
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "['X1', 'X2', 'X3', 'Y'] ATE of X1 on Y: [1.12]\n",
      "(100000, 4, 1)\n"
     ]
    }
   ],
   "execution_count": 2
  },
  {
   "cell_type": "code",
   "id": "7277cb8c-d1a1-44dc-a567-f0a7d2803887",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-10-20T15:12:29.041957Z",
     "start_time": "2024-10-20T15:12:29.040300Z"
    }
   },
   "source": [],
   "outputs": [],
   "execution_count": 2
  },
  {
   "cell_type": "code",
   "id": "7865de21-8b33-4b80-b3df-41ddedac0119",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-10-20T15:12:29.250345Z",
     "start_time": "2024-10-20T15:12:29.043572Z"
    }
   },
   "source": [
    "print('Incorrect DAG 1:')\n",
    "nx.draw_planar(\n",
    "    DAGnx1,\n",
    "    with_labels=True,\n",
    "    node_size=1000,\n",
    "    node_color=\"#ffff8f\",\n",
    "    width=0.5,\n",
    "    font_size=10,\n",
    ")\n",
    "plt.show()\n",
    "\n",
    "print('Incorrect DAG 2:')\n",
    "nx.draw_planar(\n",
    "    DAGnx2,\n",
    "    with_labels=True,\n",
    "    node_size=1000,\n",
    "    node_color=\"#ffff8f\",\n",
    "    width=0.5,\n",
    "    font_size=10,\n",
    ")\n",
    "plt.show()\n",
    "\n",
    "print('Correct DAG:')\n",
    "nx.draw_planar(\n",
    "    DAGnx3,\n",
    "    with_labels=True,\n",
    "    node_size=1000,\n",
    "    node_color=\"#ffff8f\",\n",
    "    width=0.5,\n",
    "    font_size=10,\n",
    ")\n",
    "plt.show()"
   ],
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Incorrect DAG 1:\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ],
      "image/png": "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"
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Incorrect DAG 2:\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ],
      "image/png": "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"
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Correct DAG:\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ],
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAApQAAAHzCAYAAACe1o1DAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjkuMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy80BEi2AAAACXBIWXMAAA9hAAAPYQGoP6dpAAAouklEQVR4nO3de7CdZWHv8d/OhZAgHMTITaIpCNRWoccSWy7WcBPwUpTxlDIdKuIchVbKwdsR8YC2csSpp1aoSntEkalTwWmol6IjOIK2OBJQCOIUCtIjAgE3tFxKCIS9zh8rCTubXPbe71rrfZ/3/XxmMraw915PJJJfnu9aa4/1er1eAABglubUfQAAAMpmUAIAUIlBCQBAJQYlAACVGJQAAFRiUAIAUIlBCQBAJQYlAACVGJQAAFRiUAIAUIlBCQBAJQYlAACVGJQAAFRiUAIAUIlBCQBAJQYlAACVGJQAAFRiUAIAUIlBCQBAJQYlAACVGJQAAFRiUAIAUIlBCQBAJQYlAACVGJQAAFRiUAIAUIlBCQBAJfPqPgBAM/WSrEnydJKJ9P/8PT/JwiRjNZ4LoHkMSoAk/fF4T5IHJv14ejMfNz/JbpN+LEl/ZAJ011iv1+vVfQiAevSS3J9kVZI78uxN5MQ0PnfDx81Jsl+SA5PsHreXQBcZlEBH3ZXk+iQPpT8Cq/yrcMPnL05ycJJ9Kp8OoCQGJdAxa5Jcm+T2IT7G/kmWRwoHusKgBDrkziTXJFmbajeS2zKWZEGSo5K8dIiPA9AMBiXQAb0kK9NP3KN2SJJl8dxKoM0MSqDlekn+OcmNNZ5hWfrD0qgE2skbmwMttzL1jskNZ1hZ8xkAhsegBFrsztSTuTfn+vRfWQ7QPgYl0FJr0n8BTpNcnf65ANrFoARa6tr0X83dJGvTPxdAuxiUQAvdlf77TDbtNYe99M8lfQPtYlACLdNLc543uSU/SPPGLsDsGZRAy9yf/rdTbLLxJKvrPgTAwBiUQMusSvPf73EsyS11HwJgYAxKoEXWJLkjzc/JvfTP6RXfQDsYlECL3JNkou5DTNNE+ucFKJ9BCbTIA5npv9aeeWYihxzyqZxwwuc3+euPPLImS5Z8JOec84956KH/zLHH/nX23PO8LFjw3ixZ8pG8611/n0cffbLCWeckebDC5wM0h0EJtMjqzPSGcu7cObn00pPyrW/9S770pZs2/vUzzliRXXZZlPPOOyZz5ozl+ONfnq997e25444P5tJLT8o119yR0077SoWzTsQLc4C2mFf3AQAGo5fZ3vjtt9+uueCCN+SMM1bkiCNemhtu+Hm+/OUfZ+XKs7LddvOy3Xbzcvrph278+Je8ZJf80R8dmj//8+9WPPOD68/d9BcRAWydQQm0xJokT8/6s88449W58spbc/LJX8qtt96fc899bQ488EWb/dj77nskK1asymtes8+sH6/vqfTPvaji1wGol+QNtMTsx2SSjI2N5bOffUu+851/zW677ZgPfODI53zMSSddlkWL3p8XvejD2Wmn7fO5z51Y6TH71g3gawDUy6AEWqL6q7s///kfZtGi7XL33Q/nF7945Dl//5OffFN+9KP35KtffXvuums87373Vys/ZvLMAL4GQL3Ger1e09+wDWAaHknyhVl/9vXX353XvOav8u1vn5aPfvTqJMk115yesbHNP7/xn/7pZ3n1qy/Kffd9OHvs8V9m/bjJqUl2qvD5APVzQwm0xPxZf+YTTzyVU075u5x++qE5/PB9c8klv58bbvh5Lr54y98TfGKi/2fxtWur3jB6KjtQPv8mA1piYfqjcubPpTz77G+k1+vlggvekCRZunSXfOITv5v3vvdrOe64l+WnP12dBx54LMuWvTjPe96C3Hbb/Xnf+76eQw/9lSxdukuFM2+3/twAZZO8gRb5SpJ7Z/QZ1113Z4488rO59to/zmGH7b3J3zvmmIuzbt1EPvSho3POOVflpz9dnbVrn8mSJTvnhBNekQ984KjsvHOVQbhXkrdU+HyAZjAogRb5fpIfp4xvvzgnySuTHFb3QQAq8xxKoEV2SxljMumfc9e6DwEwEAYl0CJLUs6/1uakf16A8pXyb16AaViYZL80/1sZjqV/Ti/IAdrBoARa5oD0vz92k/WSHFj3IQAGxqAEWmaPJIvrPsQ2LE6ye92HABgYgxJombEkB9d9iG04OM3P8gDTZ1ACLbRPkv3TvNE2lv659qn7IAADZVACLbU8yYK6DzHFgvTPBdAuBiXQUguTHFX3IaY4Ol7ZDbSRQQm02EuTHFL3IdY7JFI30FYGJdByy9b/6PoZAIbH9/IGOqCXZGWS62t47ENjTAJt54YS6ICx9HrL8q1vzc8jj6zNsF/9/cwzE1mzJun13hBjEugCgxJo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"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "execution_count": 3
  },
  {
   "cell_type": "markdown",
   "id": "4e6dea20",
   "metadata": {},
   "source": [
    "We run the model with the same data each time (although the column ordering of the data is different in each case, according to the assumed underlying DAG), and provide fit statistics as well as estimated causal links."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8541753a",
   "metadata": {},
   "source": [
    "### Incorrect DAG 1 (fully exogenous)"
   ]
  },
  {
   "cell_type": "code",
   "id": "8ab383e3",
   "metadata": {
    "scrolled": true,
    "ExecuteTime": {
     "end_time": "2024-10-20T15:16:28.221490Z",
     "start_time": "2024-10-20T15:12:29.251474Z"
    }
   },
   "source": [
    "indices = np.arange(0, len(all_data1))\n",
    "np.random.shuffle(indices)\n",
    "\n",
    "val_inds = indices[:int(validation_fraction*len(indices))]\n",
    "train_inds = indices[int(validation_fraction*len(indices)):]\n",
    "train_data = all_data1[train_inds]\n",
    "val_data = all_data1[val_inds]\n",
    "train_data, val_data = torch.from_numpy(train_data).float(),  torch.from_numpy(val_data).float()\n",
    "\n",
    "input_dim = all_data1.shape[2]\n",
    "\n",
    "model1 = CaT(input_dim=input_dim,\n",
    "                    dropout_rate=dropout_rate,\n",
    "                    head_size=head_size,\n",
    "                    num_heads=num_heads,\n",
    "                    ff_n_embed=ff_n_embed,\n",
    "                    embed_dim= embed_dim,\n",
    "                    dag=DAGnx1,\n",
    "                    causal_ordering=causal_ordering1,\n",
    "                    n_layers=n_layers,\n",
    "                    device=device,\n",
    "                    var_types=var_types1, activation_function='Swish'\n",
    "                    ).to(device)\n",
    "\n",
    "optimizer = torch.optim.AdamW(model1.parameters(), lr=learning_rate)\n",
    "\n",
    "def get_batch(train_data, val_data, split, device, batch_size):\n",
    "    data = train_data if split == 'train' else val_data\n",
    "    ix = torch.randint(0, len(data), (batch_size,))\n",
    "    x = data[ix]\n",
    "    return x.to(device)\n",
    "\n",
    "all_var_losses = {}\n",
    "for iter_ in range(0, max_iters):\n",
    "    # train and update the model\n",
    "    model1.train()\n",
    "\n",
    "    xb = get_batch(train_data=train_data, val_data=val_data, split='train', device=device, batch_size=batch_size)\n",
    "    xb_mod = torch.clone(xb.detach())\n",
    "    X, loss, loss_dict = model1(X=xb, targets=xb_mod\n",
    "                                )\n",
    "\n",
    "    optimizer.zero_grad(set_to_none=True)\n",
    "    loss.backward()\n",
    "    optimizer.step()\n",
    "\n",
    "\n",
    "    if iter_ % eval_interval == 0:  # evaluate the loss (no gradients)\n",
    "        for key in loss_dict.keys():\n",
    "            if key not in all_var_losses.keys():\n",
    "                all_var_losses[key] = []\n",
    "            all_var_losses[key].append(loss_dict[key])\n",
    "\n",
    "        model1.eval()\n",
    "        eval_loss = {}\n",
    "        for split in ['train', 'val']:\n",
    "            losses = torch.zeros(eval_iters)\n",
    "            for k in range(eval_iters):\n",
    "\n",
    "                xb = get_batch(train_data=train_data, val_data=val_data, split=split, device=device,\n",
    "                               batch_size=batch_size)\n",
    "                xb_mod = torch.clone(xb.detach())\n",
    "                X, loss, loss_dict = model1(X=xb, targets=xb_mod)\n",
    "                losses[k] = loss.item()\n",
    "            eval_loss[split] = losses.mean()\n",
    "        model1.train()\n",
    "        print(f\"step {iter_} of {max_iters}: train_loss {eval_loss['train']:.4f}, val loss {eval_loss['val']:.4f}\")\n",
    " "
   ],
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "step 0 of 20000: train_loss 3.4367, val loss 3.5552\n",
      "step 100 of 20000: train_loss 0.6900, val loss 0.6938\n",
      "step 200 of 20000: train_loss 0.6825, val loss 0.6586\n",
      "step 300 of 20000: train_loss 0.6647, val loss 0.6666\n",
      "step 400 of 20000: train_loss 0.7025, val loss 0.7252\n",
      "step 500 of 20000: train_loss 0.6961, val loss 0.7068\n",
      "step 600 of 20000: train_loss 0.7143, val loss 0.7138\n",
      "step 700 of 20000: train_loss 0.6729, val loss 0.6609\n",
      "step 800 of 20000: train_loss 0.6658, val loss 0.6913\n",
      "step 900 of 20000: train_loss 0.6702, val loss 0.6895\n",
      "step 1000 of 20000: train_loss 0.6773, val loss 0.6946\n",
      "step 1100 of 20000: train_loss 0.6691, val loss 0.6538\n",
      "step 1200 of 20000: train_loss 0.6885, val loss 0.6975\n",
      "step 1300 of 20000: train_loss 0.6547, val loss 0.6786\n",
      "step 1400 of 20000: train_loss 0.6761, val loss 0.6891\n",
      "step 1500 of 20000: train_loss 0.6579, val loss 0.6931\n",
      "step 1600 of 20000: train_loss 0.6612, val loss 0.6784\n",
      "step 1700 of 20000: train_loss 0.6618, val loss 0.6709\n",
      "step 1800 of 20000: train_loss 0.6770, val loss 0.6871\n",
      "step 1900 of 20000: train_loss 0.6651, val loss 0.6610\n",
      "step 2000 of 20000: train_loss 0.6739, val loss 0.6805\n",
      "step 2100 of 20000: train_loss 0.7002, val loss 0.7014\n",
      "step 2200 of 20000: train_loss 0.6710, val loss 0.6944\n",
      "step 2300 of 20000: train_loss 0.6790, val loss 0.6841\n",
      "step 2400 of 20000: train_loss 0.6890, val loss 0.6897\n",
      "step 2500 of 20000: train_loss 0.6876, val loss 0.6972\n",
      "step 2600 of 20000: train_loss 0.6610, val loss 0.6693\n",
      "step 2700 of 20000: train_loss 0.6753, val loss 0.6785\n",
      "step 2800 of 20000: train_loss 0.6634, val loss 0.6656\n",
      "step 2900 of 20000: train_loss 0.6719, val loss 0.6680\n",
      "step 3000 of 20000: train_loss 0.6882, val loss 0.6674\n",
      "step 3100 of 20000: train_loss 0.6739, val loss 0.6806\n",
      "step 3200 of 20000: train_loss 0.6747, val loss 0.7005\n",
      "step 3300 of 20000: train_loss 0.6769, val loss 0.6805\n",
      "step 3400 of 20000: train_loss 0.6665, val loss 0.6891\n",
      "step 3500 of 20000: train_loss 0.6735, val loss 0.6845\n",
      "step 3600 of 20000: train_loss 0.6694, val loss 0.6977\n",
      "step 3700 of 20000: train_loss 0.6772, val loss 0.6528\n",
      "step 3800 of 20000: train_loss 0.6590, val loss 0.6829\n",
      "step 3900 of 20000: train_loss 0.6828, val loss 0.6747\n",
      "step 4000 of 20000: train_loss 0.6831, val loss 0.6808\n",
      "step 4100 of 20000: train_loss 0.6718, val loss 0.6851\n",
      "step 4200 of 20000: train_loss 0.6632, val loss 0.6808\n",
      "step 4300 of 20000: train_loss 0.6959, val loss 0.6912\n",
      "step 4400 of 20000: train_loss 0.7016, val loss 0.7350\n",
      "step 4500 of 20000: train_loss 0.6909, val loss 0.6969\n",
      "step 4600 of 20000: train_loss 0.6586, val loss 0.6915\n",
      "step 4700 of 20000: train_loss 0.6687, val loss 0.6661\n",
      "step 4800 of 20000: train_loss 0.6552, val loss 0.6823\n",
      "step 4900 of 20000: train_loss 0.6686, val loss 0.6669\n",
      "step 5000 of 20000: train_loss 0.6529, val loss 0.6916\n",
      "step 5100 of 20000: train_loss 0.6882, val loss 0.7219\n",
      "step 5200 of 20000: train_loss 0.6757, val loss 0.6623\n",
      "step 5300 of 20000: train_loss 0.6631, val loss 0.6735\n",
      "step 5400 of 20000: train_loss 0.6642, val loss 0.6665\n",
      "step 5500 of 20000: train_loss 0.6757, val loss 0.6694\n",
      "step 5600 of 20000: train_loss 0.6731, val loss 0.6761\n",
      "step 5700 of 20000: train_loss 0.6568, val loss 0.6738\n",
      "step 5800 of 20000: train_loss 0.6671, val loss 0.6765\n",
      "step 5900 of 20000: train_loss 0.6660, val loss 0.6914\n",
      "step 6000 of 20000: train_loss 0.6518, val loss 0.6609\n",
      "step 6100 of 20000: train_loss 0.6922, val loss 0.6886\n",
      "step 6200 of 20000: train_loss 0.6976, val loss 0.6804\n",
      "step 6300 of 20000: train_loss 0.6579, val loss 0.6639\n",
      "step 6400 of 20000: train_loss 0.6519, val loss 0.6703\n",
      "step 6500 of 20000: train_loss 0.6711, val loss 0.6734\n",
      "step 6600 of 20000: train_loss 0.6760, val loss 0.6789\n",
      "step 6700 of 20000: train_loss 0.6947, val loss 0.6947\n",
      "step 6800 of 20000: train_loss 0.6672, val loss 0.6658\n",
      "step 6900 of 20000: train_loss 0.6718, val loss 0.6862\n",
      "step 7000 of 20000: train_loss 0.6920, val loss 0.7067\n",
      "step 7100 of 20000: train_loss 0.6813, val loss 0.7012\n",
      "step 7200 of 20000: train_loss 0.6710, val loss 0.6945\n",
      "step 7300 of 20000: train_loss 0.6823, val loss 0.6769\n",
      "step 7400 of 20000: train_loss 0.6658, val loss 0.6786\n",
      "step 7500 of 20000: train_loss 0.6771, val loss 0.6779\n",
      "step 7600 of 20000: train_loss 0.6666, val loss 0.6813\n",
      "step 7700 of 20000: train_loss 0.6784, val loss 0.6875\n",
      "step 7800 of 20000: train_loss 0.6454, val loss 0.6909\n",
      "step 7900 of 20000: train_loss 0.6667, val loss 0.6679\n",
      "step 8000 of 20000: train_loss 0.6637, val loss 0.6806\n",
      "step 8100 of 20000: train_loss 0.6736, val loss 0.6780\n",
      "step 8200 of 20000: train_loss 0.6601, val loss 0.6476\n",
      "step 8300 of 20000: train_loss 0.6666, val loss 0.6877\n",
      "step 8400 of 20000: train_loss 0.6634, val loss 0.6760\n",
      "step 8500 of 20000: train_loss 0.6773, val loss 0.6880\n",
      "step 8600 of 20000: train_loss 0.6744, val loss 0.6957\n",
      "step 8700 of 20000: train_loss 0.6696, val loss 0.6570\n",
      "step 8800 of 20000: train_loss 0.6536, val loss 0.6766\n",
      "step 8900 of 20000: train_loss 0.6679, val loss 0.6757\n",
      "step 9000 of 20000: train_loss 0.6645, val loss 0.6719\n",
      "step 9100 of 20000: train_loss 0.6800, val loss 0.6785\n",
      "step 9200 of 20000: train_loss 0.6693, val loss 0.6765\n",
      "step 9300 of 20000: train_loss 0.6743, val loss 0.7016\n",
      "step 9400 of 20000: train_loss 0.6763, val loss 0.6866\n",
      "step 9500 of 20000: train_loss 0.6615, val loss 0.6770\n",
      "step 9600 of 20000: train_loss 0.6814, val loss 0.6719\n",
      "step 9700 of 20000: train_loss 0.6733, val loss 0.6853\n",
      "step 9800 of 20000: train_loss 0.6531, val loss 0.6741\n",
      "step 9900 of 20000: train_loss 0.6820, val loss 0.6781\n",
      "step 10000 of 20000: train_loss 0.6580, val loss 0.6848\n",
      "step 10100 of 20000: train_loss 0.6637, val loss 0.6830\n",
      "step 10200 of 20000: train_loss 0.6898, val loss 0.6761\n",
      "step 10300 of 20000: train_loss 0.6921, val loss 0.7003\n",
      "step 10400 of 20000: train_loss 0.6508, val loss 0.6805\n",
      "step 10500 of 20000: train_loss 0.6728, val loss 0.6878\n",
      "step 10600 of 20000: train_loss 0.6732, val loss 0.6893\n",
      "step 10700 of 20000: train_loss 0.6398, val loss 0.6724\n",
      "step 10800 of 20000: train_loss 0.6906, val loss 0.7145\n",
      "step 10900 of 20000: train_loss 0.6713, val loss 0.6970\n",
      "step 11000 of 20000: train_loss 0.6480, val loss 0.6740\n",
      "step 11100 of 20000: train_loss 0.6625, val loss 0.6881\n",
      "step 11200 of 20000: train_loss 0.6735, val loss 0.6855\n",
      "step 11300 of 20000: train_loss 0.6733, val loss 0.6769\n",
      "step 11400 of 20000: train_loss 0.6537, val loss 0.6693\n",
      "step 11500 of 20000: train_loss 0.6588, val loss 0.6780\n",
      "step 11600 of 20000: train_loss 0.6498, val loss 0.6723\n",
      "step 11700 of 20000: train_loss 0.6634, val loss 0.6860\n",
      "step 11800 of 20000: train_loss 0.6656, val loss 0.6677\n",
      "step 11900 of 20000: train_loss 0.6587, val loss 0.6814\n",
      "step 12000 of 20000: train_loss 0.6706, val loss 0.6701\n",
      "step 12100 of 20000: train_loss 0.6856, val loss 0.6881\n",
      "step 12200 of 20000: train_loss 0.6747, val loss 0.6762\n",
      "step 12300 of 20000: train_loss 0.6751, val loss 0.6694\n",
      "step 12400 of 20000: train_loss 0.6598, val loss 0.6825\n",
      "step 12500 of 20000: train_loss 0.6665, val loss 0.7059\n",
      "step 12600 of 20000: train_loss 0.6686, val loss 0.6775\n",
      "step 12700 of 20000: train_loss 0.6740, val loss 0.6772\n",
      "step 12800 of 20000: train_loss 0.6695, val loss 0.6639\n",
      "step 12900 of 20000: train_loss 0.6785, val loss 0.6890\n",
      "step 13000 of 20000: train_loss 0.6618, val loss 0.6618\n",
      "step 13100 of 20000: train_loss 0.6553, val loss 0.6733\n",
      "step 13200 of 20000: train_loss 0.6651, val loss 0.6877\n",
      "step 13300 of 20000: train_loss 0.6633, val loss 0.6889\n",
      "step 13400 of 20000: train_loss 0.6838, val loss 0.7004\n",
      "step 13500 of 20000: train_loss 0.6874, val loss 0.6671\n",
      "step 13600 of 20000: train_loss 0.6818, val loss 0.6945\n",
      "step 13700 of 20000: train_loss 0.6811, val loss 0.6888\n",
      "step 13800 of 20000: train_loss 0.6804, val loss 0.6921\n",
      "step 13900 of 20000: train_loss 0.6668, val loss 0.6765\n",
      "step 14000 of 20000: train_loss 0.6804, val loss 0.6538\n",
      "step 14100 of 20000: train_loss 0.6665, val loss 0.6724\n",
      "step 14200 of 20000: train_loss 0.6698, val loss 0.6724\n",
      "step 14300 of 20000: train_loss 0.6895, val loss 0.6881\n",
      "step 14400 of 20000: train_loss 0.6686, val loss 0.6813\n",
      "step 14500 of 20000: train_loss 0.6757, val loss 0.6829\n",
      "step 14600 of 20000: train_loss 0.6855, val loss 0.6933\n",
      "step 14700 of 20000: train_loss 0.6700, val loss 0.6745\n",
      "step 14800 of 20000: train_loss 0.6929, val loss 0.7111\n",
      "step 14900 of 20000: train_loss 0.6501, val loss 0.6601\n",
      "step 15000 of 20000: train_loss 0.6707, val loss 0.6784\n",
      "step 15100 of 20000: train_loss 0.6794, val loss 0.6736\n",
      "step 15200 of 20000: train_loss 0.6515, val loss 0.6843\n",
      "step 15300 of 20000: train_loss 0.6636, val loss 0.6847\n",
      "step 15400 of 20000: train_loss 0.6787, val loss 0.6729\n",
      "step 15500 of 20000: train_loss 0.6616, val loss 0.6737\n",
      "step 15600 of 20000: train_loss 0.6656, val loss 0.6889\n",
      "step 15700 of 20000: train_loss 0.6599, val loss 0.6782\n",
      "step 15800 of 20000: train_loss 0.6640, val loss 0.6728\n",
      "step 15900 of 20000: train_loss 0.6578, val loss 0.6791\n",
      "step 16000 of 20000: train_loss 0.6760, val loss 0.6994\n",
      "step 16100 of 20000: train_loss 0.6783, val loss 0.6929\n",
      "step 16200 of 20000: train_loss 0.6612, val loss 0.6805\n",
      "step 16300 of 20000: train_loss 0.6535, val loss 0.6684\n",
      "step 16400 of 20000: train_loss 0.6780, val loss 0.6717\n",
      "step 16500 of 20000: train_loss 0.6664, val loss 0.6938\n",
      "step 16600 of 20000: train_loss 0.6778, val loss 0.6775\n",
      "step 16700 of 20000: train_loss 0.6723, val loss 0.6702\n",
      "step 16800 of 20000: train_loss 0.6650, val loss 0.6650\n",
      "step 16900 of 20000: train_loss 0.6485, val loss 0.6737\n",
      "step 17000 of 20000: train_loss 0.6792, val loss 0.6933\n",
      "step 17100 of 20000: train_loss 0.6553, val loss 0.6858\n",
      "step 17200 of 20000: train_loss 0.6801, val loss 0.7025\n",
      "step 17300 of 20000: train_loss 0.6589, val loss 0.6728\n",
      "step 17400 of 20000: train_loss 0.6685, val loss 0.6808\n",
      "step 17500 of 20000: train_loss 0.6537, val loss 0.6736\n",
      "step 17600 of 20000: train_loss 0.6680, val loss 0.6874\n",
      "step 17700 of 20000: train_loss 0.6848, val loss 0.6856\n",
      "step 17800 of 20000: train_loss 0.6826, val loss 0.6605\n",
      "step 17900 of 20000: train_loss 0.6858, val loss 0.6794\n",
      "step 18000 of 20000: train_loss 0.6733, val loss 0.6805\n",
      "step 18100 of 20000: train_loss 0.6656, val loss 0.6618\n",
      "step 18200 of 20000: train_loss 0.6768, val loss 0.6936\n",
      "step 18300 of 20000: train_loss 0.6824, val loss 0.6746\n",
      "step 18400 of 20000: train_loss 0.6690, val loss 0.6737\n",
      "step 18500 of 20000: train_loss 0.6848, val loss 0.6816\n",
      "step 18600 of 20000: train_loss 0.6582, val loss 0.6762\n",
      "step 18700 of 20000: train_loss 0.6759, val loss 0.6892\n",
      "step 18800 of 20000: train_loss 0.6768, val loss 0.6968\n",
      "step 18900 of 20000: train_loss 0.6677, val loss 0.6949\n",
      "step 19000 of 20000: train_loss 0.6565, val loss 0.6904\n",
      "step 19100 of 20000: train_loss 0.6584, val loss 0.6698\n",
      "step 19200 of 20000: train_loss 0.6741, val loss 0.6839\n",
      "step 19300 of 20000: train_loss 0.6700, val loss 0.6795\n",
      "step 19400 of 20000: train_loss 0.6808, val loss 0.6752\n",
      "step 19500 of 20000: train_loss 0.6747, val loss 0.6833\n",
      "step 19600 of 20000: train_loss 0.6670, val loss 0.6876\n",
      "step 19700 of 20000: train_loss 0.6516, val loss 0.6754\n",
      "step 19800 of 20000: train_loss 0.6642, val loss 0.6707\n",
      "step 19900 of 20000: train_loss 0.6631, val loss 0.6743\n"
     ]
    }
   ],
   "execution_count": 4
  },
  {
   "cell_type": "code",
   "id": "545ff6c9",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-10-20T15:16:28.604656Z",
     "start_time": "2024-10-20T15:16:28.222569Z"
    }
   },
   "source": [
    "   \n",
    "model1.eval()\n",
    "\n",
    "\n",
    "int_nodes_vals0 = {'X1':np.array([0.0,])}\n",
    "int_nodes_vals1 = {'X1':np.array([1.0,])}\n",
    "effect_var = 'Y'\n",
    "effect_index = var_names1.index(effect_var)\n",
    "\n",
    "inf = CausalInference(dag=DAGnx1)\n",
    "preds0 = inf.forward(all_data1, model=model1, intervention_nodes_vals=int_nodes_vals0)\n",
    "preds1 = inf.forward(all_data1, model=model1, intervention_nodes_vals=int_nodes_vals1)\n",
    "ATE_pred = (preds1[:,effect_index,:] - preds0[:,effect_index,:]).mean(0)\n",
    "eATE = np.abs(ATE_pred - ATE)\n",
    "print('ATE:', ATE, 'est ATE:', ATE_pred, 'error:', eATE)\n",
    "\n",
    "preds = model1(train_data.to(device))\n",
    "plt.scatter(train_data[:,effect_index,-1].detach().cpu().numpy(), preds[:, effect_index, -1].detach().cpu().numpy())\n",
    "print('Mean Squared Error Across All Vars:', ((train_data - preds.detach().cpu())**2).mean())\n",
    "print('Mean Squared Error Across Outcome:', ((train_data[:,effect_index,:] - preds[:,effect_index,:].detach().cpu())**2).mean())"
   ],
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/matthewvowels/GitHub/Causal_Transformer/utils/inference.py:18: UserWarning: No mask has been specified. If padding has been used, the absence of a mask may lead to incorrect results.\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "ATE: [1.12] est ATE: [0.27077529] error: [0.84922471]\n",
      "Mean Squared Error Across All Vars: tensor(1.7041)\n",
      "Mean Squared Error Across Outcome: tensor(0.6653)\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ],
      "image/png": "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ix07dmDVqlXweDxIS0szcIXGcTgcAfciABg8eDDefPNNg1YUHgMRBT179kTPnj0VrxsxYgSsVis+++wzjBs3DgDQ1taGEydOoF+/fnov0xBqvze//e1v8dRTT3V8/tVXX+H222/HG2+8gTFjxui5RMOo/d4A7Tsh48eP79hFs1hSa6MyIyMDI0aMwPvvv99R7u7z+fD+++9j3rx5xi7OYKIo4j//8z/x9ttv469//SsKCwuNXpJp3HbbbThw4EDAYzNnzsT111+PRYsWpWwQAgClpaWdyrwPHz5s2nsRA5EYyc3NxezZs1FWVoa+ffuiX79+WLlyJQDgvvvuM3h1xrrmmmsCPu/atSsAYODAgSn/7q6+vh633nor+vXrh+eeew5nzpzp+LNU2g1YuHAh/u3f/g0jR47E6NGj8cILL6C5uRkzZ840emmGmjt3LjZu3IhNmzahW7duHTkzNpsNWVlZBq/OWN26deuUK5OTk4OCgoKUz6FZsGABbr75Zjz99NP4wQ9+gN27d2PdunWm3W1lIBJDK1euRJcuXfDQQw/h0qVLGDNmDCorK81bu02G27ZtG44ePYqjR492CspEUTRoVfH3wx/+EGfOnMHPfvYzOJ1ODBs2DBUVFZ0SWFPN6tWrAQC33nprwOPr169XPPqj1DVq1Ci8/fbbWLJkCX7+85+jsLAQL7zwAh588EGjlxaSIKbSbzsiIiIyldQ6jCYiIiJTYSBCREREhmEgQkRERIZhIEJERESGYSBCREREhmEgQkRERIZhIEJERESGYSBCREREhmEgQkRERIZhIEJERESGYSBCREREhvn/AfJqtciQY9HGAAAAAElFTkSuQmCC"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "execution_count": 5
  },
  {
   "cell_type": "markdown",
   "id": "442f6d34",
   "metadata": {},
   "source": [
    "## Incorrect DAG 2 (missing and reverse edges)"
   ]
  },
  {
   "cell_type": "code",
   "id": "e280e550",
   "metadata": {
    "scrolled": true,
    "ExecuteTime": {
     "end_time": "2024-10-20T15:20:22.064925Z",
     "start_time": "2024-10-20T15:16:28.605782Z"
    }
   },
   "source": [
    "indices = np.arange(0, len(all_data2))\n",
    "np.random.shuffle(indices)\n",
    "\n",
    "val_inds = indices[:int(validation_fraction*len(indices))]\n",
    "train_inds = indices[int(validation_fraction*len(indices)):]\n",
    "train_data = all_data2[train_inds]\n",
    "val_data = all_data2[val_inds]\n",
    "train_data, val_data = torch.from_numpy(train_data).float(),  torch.from_numpy(val_data).float()\n",
    "\n",
    "input_dim = all_data2.shape[2]\n",
    "\n",
    "model2 = CaT(input_dim=input_dim,\n",
    "                    dropout_rate=dropout_rate,\n",
    "                    head_size=head_size,\n",
    "                    num_heads=num_heads,\n",
    "                    ff_n_embed=ff_n_embed,\n",
    "                    embed_dim= embed_dim,\n",
    "                    dag=DAGnx2,\n",
    "                    causal_ordering=causal_ordering2,\n",
    "                    n_layers=n_layers,\n",
    "                    device=device,\n",
    "                    var_types=var_types2, activation_function='Swish'\n",
    "                    ).to(device)\n",
    "\n",
    "optimizer = torch.optim.AdamW(model2.parameters(), lr=learning_rate)\n",
    "\n",
    "def get_batch(train_data, val_data, split, device, batch_size):\n",
    "    data = train_data if split == 'train' else val_data\n",
    "    ix = torch.randint(0, len(data), (batch_size,))\n",
    "    x = data[ix]\n",
    "    return x.to(device)\n",
    "\n",
    "all_var_losses = {}\n",
    "for iter_ in range(0, max_iters):\n",
    "    # train and update the model\n",
    "    model2.train()\n",
    "\n",
    "    xb = get_batch(train_data=train_data, val_data=val_data, split='train', device=device, batch_size=batch_size)\n",
    "    xb_mod = torch.clone(xb.detach())\n",
    "    X, loss, loss_dict = model2(X=xb, targets=xb_mod)\n",
    "\n",
    "    optimizer.zero_grad(set_to_none=True)\n",
    "    loss.backward()\n",
    "    optimizer.step()\n",
    "\n",
    "\n",
    "    if iter_ % eval_interval == 0:  # evaluate the loss (no gradients)\n",
    "        for key in loss_dict.keys():\n",
    "            if key not in all_var_losses.keys():\n",
    "                all_var_losses[key] = []\n",
    "            all_var_losses[key].append(loss_dict[key])\n",
    "\n",
    "        model2.eval()\n",
    "        eval_loss = {}\n",
    "        for split in ['train', 'val']:\n",
    "            losses = torch.zeros(eval_iters)\n",
    "            for k in range(eval_iters):\n",
    "\n",
    "                xb = get_batch(train_data=train_data, val_data=val_data, split=split, device=device,\n",
    "                               batch_size=batch_size)\n",
    "                xb_mod = torch.clone(xb.detach())\n",
    "                X, loss, loss_dict = model2(X=xb, targets=xb_mod)\n",
    "                losses[k] = loss.item()\n",
    "            eval_loss[split] = losses.mean()\n",
    "        model2.train()\n",
    "        print(f\"step {iter_} of {max_iters}: train_loss {eval_loss['train']:.4f}, val loss {eval_loss['val']:.4f}\")\n",
    " "
   ],
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "step 0 of 20000: train_loss 8.4819, val loss 8.4404\n",
      "step 100 of 20000: train_loss 1.6971, val loss 1.7047\n",
      "step 200 of 20000: train_loss 1.7025, val loss 1.7093\n",
      "step 300 of 20000: train_loss 1.6768, val loss 1.7105\n",
      "step 400 of 20000: train_loss 1.6871, val loss 1.6687\n",
      "step 500 of 20000: train_loss 1.6673, val loss 1.6861\n",
      "step 600 of 20000: train_loss 1.6631, val loss 1.6875\n",
      "step 700 of 20000: train_loss 1.6743, val loss 1.6678\n",
      "step 800 of 20000: train_loss 1.6462, val loss 1.6647\n",
      "step 900 of 20000: train_loss 1.7067, val loss 1.7050\n",
      "step 1000 of 20000: train_loss 1.6843, val loss 1.6258\n",
      "step 1100 of 20000: train_loss 1.6590, val loss 1.6578\n",
      "step 1200 of 20000: train_loss 1.6760, val loss 1.6767\n",
      "step 1300 of 20000: train_loss 1.6580, val loss 1.6705\n",
      "step 1400 of 20000: train_loss 1.6821, val loss 1.7108\n",
      "step 1500 of 20000: train_loss 1.7000, val loss 1.6792\n",
      "step 1600 of 20000: train_loss 1.6661, val loss 1.6854\n",
      "step 1700 of 20000: train_loss 1.6718, val loss 1.7006\n",
      "step 1800 of 20000: train_loss 1.6936, val loss 1.6983\n",
      "step 1900 of 20000: train_loss 1.6716, val loss 1.6765\n",
      "step 2000 of 20000: train_loss 1.6636, val loss 1.6861\n",
      "step 2100 of 20000: train_loss 1.6800, val loss 1.6673\n",
      "step 2200 of 20000: train_loss 1.6759, val loss 1.6720\n",
      "step 2300 of 20000: train_loss 1.6716, val loss 1.6827\n",
      "step 2400 of 20000: train_loss 1.6901, val loss 1.7098\n",
      "step 2500 of 20000: train_loss 1.6809, val loss 1.6861\n",
      "step 2600 of 20000: train_loss 1.6723, val loss 1.6838\n",
      "step 2700 of 20000: train_loss 1.6667, val loss 1.6700\n",
      "step 2800 of 20000: train_loss 1.6449, val loss 1.6980\n",
      "step 2900 of 20000: train_loss 1.7064, val loss 1.6768\n",
      "step 3000 of 20000: train_loss 1.6579, val loss 1.6860\n",
      "step 3100 of 20000: train_loss 1.6870, val loss 1.6748\n",
      "step 3200 of 20000: train_loss 1.6872, val loss 1.6607\n",
      "step 3300 of 20000: train_loss 1.7186, val loss 1.7007\n",
      "step 3400 of 20000: train_loss 1.6214, val loss 1.6579\n",
      "step 3500 of 20000: train_loss 1.6706, val loss 1.6915\n",
      "step 3600 of 20000: train_loss 1.6924, val loss 1.7018\n",
      "step 3700 of 20000: train_loss 1.6931, val loss 1.6867\n",
      "step 3800 of 20000: train_loss 1.6846, val loss 1.6649\n",
      "step 3900 of 20000: train_loss 1.6454, val loss 1.6717\n",
      "step 4000 of 20000: train_loss 1.6544, val loss 1.6898\n",
      "step 4100 of 20000: train_loss 1.6748, val loss 1.6625\n",
      "step 4200 of 20000: train_loss 1.6788, val loss 1.6674\n",
      "step 4300 of 20000: train_loss 1.6700, val loss 1.6688\n",
      "step 4400 of 20000: train_loss 1.6665, val loss 1.6535\n",
      "step 4500 of 20000: train_loss 1.6989, val loss 1.6857\n",
      "step 4600 of 20000: train_loss 1.6461, val loss 1.6740\n",
      "step 4700 of 20000: train_loss 1.6895, val loss 1.6663\n",
      "step 4800 of 20000: train_loss 1.6577, val loss 1.7322\n",
      "step 4900 of 20000: train_loss 1.6579, val loss 1.6626\n",
      "step 5000 of 20000: train_loss 1.6986, val loss 1.6859\n",
      "step 5100 of 20000: train_loss 1.6800, val loss 1.6484\n",
      "step 5200 of 20000: train_loss 1.6660, val loss 1.6647\n",
      "step 5300 of 20000: train_loss 1.6698, val loss 1.6755\n",
      "step 5400 of 20000: train_loss 1.6729, val loss 1.6337\n",
      "step 5500 of 20000: train_loss 1.6712, val loss 1.6743\n",
      "step 5600 of 20000: train_loss 1.7212, val loss 1.6788\n",
      "step 5700 of 20000: train_loss 1.6518, val loss 1.6581\n",
      "step 5800 of 20000: train_loss 1.6580, val loss 1.6726\n",
      "step 5900 of 20000: train_loss 1.6702, val loss 1.6826\n",
      "step 6000 of 20000: train_loss 1.6959, val loss 1.6526\n",
      "step 6100 of 20000: train_loss 1.6717, val loss 1.6485\n",
      "step 6200 of 20000: train_loss 1.6568, val loss 1.6708\n",
      "step 6300 of 20000: train_loss 1.6643, val loss 1.6880\n",
      "step 6400 of 20000: train_loss 1.6769, val loss 1.6593\n",
      "step 6500 of 20000: train_loss 1.6790, val loss 1.6699\n",
      "step 6600 of 20000: train_loss 1.6586, val loss 1.6869\n",
      "step 6700 of 20000: train_loss 1.6851, val loss 1.7052\n",
      "step 6800 of 20000: train_loss 1.6486, val loss 1.6531\n",
      "step 6900 of 20000: train_loss 1.6992, val loss 1.7115\n",
      "step 7000 of 20000: train_loss 1.7004, val loss 1.6812\n",
      "step 7100 of 20000: train_loss 1.6648, val loss 1.6818\n",
      "step 7200 of 20000: train_loss 1.6800, val loss 1.6743\n",
      "step 7300 of 20000: train_loss 1.6882, val loss 1.6985\n",
      "step 7400 of 20000: train_loss 1.6609, val loss 1.6668\n",
      "step 7500 of 20000: train_loss 1.6741, val loss 1.6569\n",
      "step 7600 of 20000: train_loss 1.6734, val loss 1.6497\n",
      "step 7700 of 20000: train_loss 1.6796, val loss 1.6873\n",
      "step 7800 of 20000: train_loss 1.6898, val loss 1.7100\n",
      "step 7900 of 20000: train_loss 1.6810, val loss 1.6996\n",
      "step 8000 of 20000: train_loss 1.6877, val loss 1.7017\n",
      "step 8100 of 20000: train_loss 1.6941, val loss 1.6445\n",
      "step 8200 of 20000: train_loss 1.6717, val loss 1.6390\n",
      "step 8300 of 20000: train_loss 1.6625, val loss 1.6553\n",
      "step 8400 of 20000: train_loss 1.6910, val loss 1.6934\n",
      "step 8500 of 20000: train_loss 1.6647, val loss 1.6696\n",
      "step 8600 of 20000: train_loss 1.6599, val loss 1.6609\n",
      "step 8700 of 20000: train_loss 1.7244, val loss 1.6877\n",
      "step 8800 of 20000: train_loss 1.6623, val loss 1.6599\n",
      "step 8900 of 20000: train_loss 1.6881, val loss 1.6889\n",
      "step 9000 of 20000: train_loss 1.6818, val loss 1.6626\n",
      "step 9100 of 20000: train_loss 1.6579, val loss 1.6710\n",
      "step 9200 of 20000: train_loss 1.6505, val loss 1.6853\n",
      "step 9300 of 20000: train_loss 1.6807, val loss 1.6592\n",
      "step 9400 of 20000: train_loss 1.6678, val loss 1.6653\n",
      "step 9500 of 20000: train_loss 1.6860, val loss 1.6507\n",
      "step 9600 of 20000: train_loss 1.6609, val loss 1.6726\n",
      "step 9700 of 20000: train_loss 1.6996, val loss 1.7181\n",
      "step 9800 of 20000: train_loss 1.6840, val loss 1.6351\n",
      "step 9900 of 20000: train_loss 1.6999, val loss 1.6488\n",
      "step 10000 of 20000: train_loss 1.6585, val loss 1.6556\n",
      "step 10100 of 20000: train_loss 1.6766, val loss 1.7013\n",
      "step 10200 of 20000: train_loss 1.6491, val loss 1.6746\n",
      "step 10300 of 20000: train_loss 1.6931, val loss 1.6522\n",
      "step 10400 of 20000: train_loss 1.7122, val loss 1.6746\n",
      "step 10500 of 20000: train_loss 1.6621, val loss 1.6431\n",
      "step 10600 of 20000: train_loss 1.6764, val loss 1.6874\n",
      "step 10700 of 20000: train_loss 1.6750, val loss 1.6665\n",
      "step 10800 of 20000: train_loss 1.6899, val loss 1.6460\n",
      "step 10900 of 20000: train_loss 1.6374, val loss 1.6718\n",
      "step 11000 of 20000: train_loss 1.6750, val loss 1.6322\n",
      "step 11100 of 20000: train_loss 1.6879, val loss 1.6818\n",
      "step 11200 of 20000: train_loss 1.6471, val loss 1.6403\n",
      "step 11300 of 20000: train_loss 1.6806, val loss 1.6841\n",
      "step 11400 of 20000: train_loss 1.6876, val loss 1.6851\n",
      "step 11500 of 20000: train_loss 1.6925, val loss 1.7024\n",
      "step 11600 of 20000: train_loss 1.7118, val loss 1.6697\n",
      "step 11700 of 20000: train_loss 1.6494, val loss 1.6694\n",
      "step 11800 of 20000: train_loss 1.6687, val loss 1.6655\n",
      "step 11900 of 20000: train_loss 1.6815, val loss 1.6672\n",
      "step 12000 of 20000: train_loss 1.6790, val loss 1.6942\n",
      "step 12100 of 20000: train_loss 1.6821, val loss 1.6661\n",
      "step 12200 of 20000: train_loss 1.6920, val loss 1.6789\n",
      "step 12300 of 20000: train_loss 1.7023, val loss 1.6961\n",
      "step 12400 of 20000: train_loss 1.6996, val loss 1.7168\n",
      "step 12500 of 20000: train_loss 1.6826, val loss 1.6659\n",
      "step 12600 of 20000: train_loss 1.6664, val loss 1.6659\n",
      "step 12700 of 20000: train_loss 1.6802, val loss 1.6810\n",
      "step 12800 of 20000: train_loss 1.6462, val loss 1.6772\n",
      "step 12900 of 20000: train_loss 1.6631, val loss 1.6485\n",
      "step 13000 of 20000: train_loss 1.6878, val loss 1.7400\n",
      "step 13100 of 20000: train_loss 1.6556, val loss 1.6946\n",
      "step 13200 of 20000: train_loss 1.6449, val loss 1.6801\n",
      "step 13300 of 20000: train_loss 1.6654, val loss 1.6770\n",
      "step 13400 of 20000: train_loss 1.6536, val loss 1.6962\n",
      "step 13500 of 20000: train_loss 1.6676, val loss 1.6924\n",
      "step 13600 of 20000: train_loss 1.6706, val loss 1.6557\n",
      "step 13700 of 20000: train_loss 1.6589, val loss 1.6710\n",
      "step 13800 of 20000: train_loss 1.6590, val loss 1.6846\n",
      "step 13900 of 20000: train_loss 1.6604, val loss 1.7000\n",
      "step 14000 of 20000: train_loss 1.6476, val loss 1.6422\n",
      "step 14100 of 20000: train_loss 1.6629, val loss 1.6792\n",
      "step 14200 of 20000: train_loss 1.6376, val loss 1.6742\n",
      "step 14300 of 20000: train_loss 1.6875, val loss 1.6806\n",
      "step 14400 of 20000: train_loss 1.6760, val loss 1.6587\n",
      "step 14500 of 20000: train_loss 1.7030, val loss 1.6878\n",
      "step 14600 of 20000: train_loss 1.6936, val loss 1.6902\n",
      "step 14700 of 20000: train_loss 1.6755, val loss 1.6443\n",
      "step 14800 of 20000: train_loss 1.6378, val loss 1.6984\n",
      "step 14900 of 20000: train_loss 1.6893, val loss 1.6491\n",
      "step 15000 of 20000: train_loss 1.6687, val loss 1.6403\n",
      "step 15100 of 20000: train_loss 1.6924, val loss 1.6854\n",
      "step 15200 of 20000: train_loss 1.6875, val loss 1.6560\n",
      "step 15300 of 20000: train_loss 1.6942, val loss 1.6863\n",
      "step 15400 of 20000: train_loss 1.6824, val loss 1.6598\n",
      "step 15500 of 20000: train_loss 1.6686, val loss 1.6308\n",
      "step 15600 of 20000: train_loss 1.6919, val loss 1.6525\n",
      "step 15700 of 20000: train_loss 1.6617, val loss 1.6711\n",
      "step 15800 of 20000: train_loss 1.6822, val loss 1.6562\n",
      "step 15900 of 20000: train_loss 1.6556, val loss 1.6568\n",
      "step 16000 of 20000: train_loss 1.6549, val loss 1.6996\n",
      "step 16100 of 20000: train_loss 1.6786, val loss 1.6891\n",
      "step 16200 of 20000: train_loss 1.6227, val loss 1.6386\n",
      "step 16300 of 20000: train_loss 1.6845, val loss 1.6392\n",
      "step 16400 of 20000: train_loss 1.6602, val loss 1.6498\n",
      "step 16500 of 20000: train_loss 1.6605, val loss 1.6695\n",
      "step 16600 of 20000: train_loss 1.6862, val loss 1.6750\n",
      "step 16700 of 20000: train_loss 1.6603, val loss 1.6517\n",
      "step 16800 of 20000: train_loss 1.6718, val loss 1.6650\n",
      "step 16900 of 20000: train_loss 1.6671, val loss 1.6624\n",
      "step 17000 of 20000: train_loss 1.6669, val loss 1.6635\n",
      "step 17100 of 20000: train_loss 1.6697, val loss 1.6432\n",
      "step 17200 of 20000: train_loss 1.6770, val loss 1.7086\n",
      "step 17300 of 20000: train_loss 1.6609, val loss 1.6942\n",
      "step 17400 of 20000: train_loss 1.6659, val loss 1.6577\n",
      "step 17500 of 20000: train_loss 1.6663, val loss 1.6683\n",
      "step 17600 of 20000: train_loss 1.6545, val loss 1.6775\n",
      "step 17700 of 20000: train_loss 1.6468, val loss 1.6973\n",
      "step 17800 of 20000: train_loss 1.6482, val loss 1.6518\n",
      "step 17900 of 20000: train_loss 1.7170, val loss 1.6778\n",
      "step 18000 of 20000: train_loss 1.6743, val loss 1.6643\n",
      "step 18100 of 20000: train_loss 1.6404, val loss 1.6787\n",
      "step 18200 of 20000: train_loss 1.6751, val loss 1.6322\n",
      "step 18300 of 20000: train_loss 1.6845, val loss 1.7106\n",
      "step 18400 of 20000: train_loss 1.6462, val loss 1.6611\n",
      "step 18500 of 20000: train_loss 1.6758, val loss 1.6819\n",
      "step 18600 of 20000: train_loss 1.6684, val loss 1.6579\n",
      "step 18700 of 20000: train_loss 1.6675, val loss 1.6741\n",
      "step 18800 of 20000: train_loss 1.6496, val loss 1.6979\n",
      "step 18900 of 20000: train_loss 1.6563, val loss 1.6606\n",
      "step 19000 of 20000: train_loss 1.6595, val loss 1.6885\n",
      "step 19100 of 20000: train_loss 1.6425, val loss 1.6592\n",
      "step 19200 of 20000: train_loss 1.6893, val loss 1.6515\n",
      "step 19300 of 20000: train_loss 1.6507, val loss 1.6395\n",
      "step 19400 of 20000: train_loss 1.6722, val loss 1.6675\n",
      "step 19500 of 20000: train_loss 1.6554, val loss 1.6711\n",
      "step 19600 of 20000: train_loss 1.6507, val loss 1.6405\n",
      "step 19700 of 20000: train_loss 1.6599, val loss 1.6423\n",
      "step 19800 of 20000: train_loss 1.6869, val loss 1.6646\n",
      "step 19900 of 20000: train_loss 1.6664, val loss 1.6753\n"
     ]
    }
   ],
   "execution_count": 6
  },
  {
   "cell_type": "code",
   "id": "87f7acec",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-10-20T15:20:22.477142Z",
     "start_time": "2024-10-20T15:20:22.065988Z"
    }
   },
   "source": [
    "   \n",
    "model2.eval()\n",
    "\n",
    "\n",
    "int_nodes_vals0 = {'X1':np.array([0.0,])}\n",
    "int_nodes_vals1 = {'X1':np.array([1.0,])}\n",
    "effect_var = 'Y'\n",
    "effect_index = var_names2.index(effect_var)\n",
    "\n",
    "inf = CausalInference(dag=DAGnx2)\n",
    "preds0 = inf.forward(all_data2, model=model2, intervention_nodes_vals=int_nodes_vals0)\n",
    "preds1 = inf.forward(all_data2, model=model2, intervention_nodes_vals=int_nodes_vals1)\n",
    "ATE_pred = (preds1[:,effect_index,:] - preds0[:,effect_index,:]).mean(0)\n",
    "eATE = np.abs(ATE_pred - ATE)\n",
    "print('ATE:', ATE, 'est ATE:', ATE_pred, 'error:', eATE)\n",
    "\n",
    "preds = model2(train_data.to(device))\n",
    "plt.scatter(train_data[:,effect_index,-1].detach().cpu().numpy(), preds[:, effect_index, -1].detach().cpu().numpy())\n",
    "print('Mean Squared Error Across All Vars:', ((train_data - preds.detach().cpu())**2).mean())\n",
    "print('Mean Squared Error Across Outcome:', ((train_data[:,effect_index,:] - preds[:,effect_index,:].detach().cpu())**2).mean())"
   ],
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "ATE: [1.12] est ATE: [0.48470021] error: [0.63529979]\n",
      "Mean Squared Error Across All Vars: tensor(1.5491)\n",
      "Mean Squared Error Across Outcome: tensor(0.6720)\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ],
      "image/png": "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"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "execution_count": 7
  },
  {
   "cell_type": "markdown",
   "id": "a17b0cde",
   "metadata": {},
   "source": [
    "## Correct DAG"
   ]
  },
  {
   "cell_type": "code",
   "id": "6c8a281b",
   "metadata": {
    "scrolled": true,
    "ExecuteTime": {
     "end_time": "2024-10-20T15:24:52.346115Z",
     "start_time": "2024-10-20T15:20:22.478294Z"
    }
   },
   "source": [
    "indices = np.arange(0, len(all_data3))\n",
    "np.random.shuffle(indices)\n",
    "\n",
    "val_inds = indices[:int(validation_fraction*len(indices))]\n",
    "train_inds = indices[int(validation_fraction*len(indices)):]\n",
    "train_data = all_data3[train_inds]\n",
    "val_data = all_data3[val_inds]\n",
    "train_data, val_data = torch.from_numpy(train_data).float(),  torch.from_numpy(val_data).float()\n",
    "\n",
    "input_dim = all_data3.shape[2]\n",
    "\n",
    "model3 = CaT(input_dim=input_dim,\n",
    "                    dropout_rate=dropout_rate,\n",
    "                    head_size=head_size,\n",
    "                    num_heads=num_heads,\n",
    "                    ff_n_embed=ff_n_embed,\n",
    "                    embed_dim= embed_dim,\n",
    "                    dag=DAGnx3,\n",
    "                    causal_ordering=causal_ordering3,\n",
    "                    n_layers=n_layers,\n",
    "                    device=device,\n",
    "                    var_types=var_types3, activation_function='Swish'\n",
    "                    ).to(device)\n",
    "\n",
    "optimizer = torch.optim.AdamW(model3.parameters(), lr=learning_rate)\n",
    "\n",
    "def get_batch(train_data, val_data, split, device, batch_size):\n",
    "    data = train_data if split == 'train' else val_data\n",
    "    ix = torch.randint(0, len(data), (batch_size,))\n",
    "    x = data[ix]\n",
    "    return x.to(device)\n",
    "\n",
    "all_var_losses = {}\n",
    "for iter_ in range(0, max_iters):\n",
    "    # train and update the model\n",
    "    model3.train()\n",
    "\n",
    "    xb = get_batch(train_data=train_data, val_data=val_data, split='train', device=device, batch_size=batch_size)\n",
    "    xb_mod = torch.clone(xb.detach())\n",
    "    X, loss, loss_dict = model3(X=xb, targets=xb_mod)\n",
    "\n",
    "    optimizer.zero_grad(set_to_none=True)\n",
    "    loss.backward()\n",
    "    optimizer.step()\n",
    "\n",
    "\n",
    "    if iter_ % eval_interval == 0:  # evaluate the loss (no gradients)\n",
    "        for key in loss_dict.keys():\n",
    "            if key not in all_var_losses.keys():\n",
    "                all_var_losses[key] = []\n",
    "            all_var_losses[key].append(loss_dict[key])\n",
    "\n",
    "        model3.eval()\n",
    "        eval_loss = {}\n",
    "        for split in ['train', 'val']:\n",
    "            losses = torch.zeros(eval_iters)\n",
    "            for k in range(eval_iters):\n",
    "\n",
    "                xb = get_batch(train_data=train_data, val_data=val_data, split=split, device=device,\n",
    "                               batch_size=batch_size)\n",
    "                xb_mod = torch.clone(xb.detach())\n",
    "                X, loss, loss_dict = model3(X=xb, targets=xb_mod)\n",
    "                losses[k] = loss.item()\n",
    "            eval_loss[split] = losses.mean()\n",
    "        model3.train()\n",
    "        print(f\"step {iter_} of {max_iters}: train_loss {eval_loss['train']:.4f}, val loss {eval_loss['val']:.4f}\")\n",
    " "
   ],
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "step 0 of 20000: train_loss 5.5690, val loss 5.5025\n",
      "step 100 of 20000: train_loss 3.0368, val loss 2.9659\n",
      "step 200 of 20000: train_loss 2.9939, val loss 3.0165\n",
      "step 300 of 20000: train_loss 2.9999, val loss 2.9951\n",
      "step 400 of 20000: train_loss 2.9939, val loss 2.9238\n",
      "step 500 of 20000: train_loss 3.0671, val loss 3.0140\n",
      "step 600 of 20000: train_loss 3.0295, val loss 3.0206\n",
      "step 700 of 20000: train_loss 2.9638, val loss 3.0059\n",
      "step 800 of 20000: train_loss 2.9903, val loss 2.9991\n",
      "step 900 of 20000: train_loss 3.1656, val loss 3.0584\n",
      "step 1000 of 20000: train_loss 3.0151, val loss 2.9791\n",
      "step 1100 of 20000: train_loss 3.0465, val loss 2.9636\n",
      "step 1200 of 20000: train_loss 3.0063, val loss 2.9292\n",
      "step 1300 of 20000: train_loss 3.0021, val loss 2.9554\n",
      "step 1400 of 20000: train_loss 3.0043, val loss 2.9968\n",
      "step 1500 of 20000: train_loss 2.9842, val loss 2.9619\n",
      "step 1600 of 20000: train_loss 3.0420, val loss 2.9448\n",
      "step 1700 of 20000: train_loss 2.9997, val loss 3.0247\n",
      "step 1800 of 20000: train_loss 3.0210, val loss 2.9576\n",
      "step 1900 of 20000: train_loss 3.0347, val loss 2.9608\n",
      "step 2000 of 20000: train_loss 2.9772, val loss 3.0055\n",
      "step 2100 of 20000: train_loss 3.0295, val loss 2.9863\n",
      "step 2200 of 20000: train_loss 3.0154, val loss 2.9856\n",
      "step 2300 of 20000: train_loss 2.9746, val loss 3.0050\n",
      "step 2400 of 20000: train_loss 3.0026, val loss 3.0082\n",
      "step 2500 of 20000: train_loss 3.0174, val loss 2.9970\n",
      "step 2600 of 20000: train_loss 2.9708, val loss 3.0219\n",
      "step 2700 of 20000: train_loss 3.0098, val loss 2.9804\n",
      "step 2800 of 20000: train_loss 3.0112, val loss 2.9736\n",
      "step 2900 of 20000: train_loss 3.0308, val loss 2.9912\n",
      "step 3000 of 20000: train_loss 3.0011, val loss 3.0106\n",
      "step 3100 of 20000: train_loss 2.9804, val loss 3.0174\n",
      "step 3200 of 20000: train_loss 3.0317, val loss 3.0062\n",
      "step 3300 of 20000: train_loss 3.0489, val loss 3.0078\n",
      "step 3400 of 20000: train_loss 3.0185, val loss 2.9780\n",
      "step 3500 of 20000: train_loss 3.0188, val loss 2.9701\n",
      "step 3600 of 20000: train_loss 2.9922, val loss 2.9880\n",
      "step 3700 of 20000: train_loss 3.0362, val loss 2.9548\n",
      "step 3800 of 20000: train_loss 3.0395, val loss 2.9750\n",
      "step 3900 of 20000: train_loss 3.0533, val loss 2.9804\n",
      "step 4000 of 20000: train_loss 3.0052, val loss 2.9796\n",
      "step 4100 of 20000: train_loss 2.9844, val loss 2.9620\n",
      "step 4200 of 20000: train_loss 3.0122, val loss 3.0278\n",
      "step 4300 of 20000: train_loss 3.0182, val loss 2.9797\n",
      "step 4400 of 20000: train_loss 3.0342, val loss 2.9587\n",
      "step 4500 of 20000: train_loss 3.0049, val loss 3.0247\n",
      "step 4600 of 20000: train_loss 3.0037, val loss 2.9938\n",
      "step 4700 of 20000: train_loss 3.0012, val loss 3.0285\n",
      "step 4800 of 20000: train_loss 3.0022, val loss 2.9529\n",
      "step 4900 of 20000: train_loss 3.0392, val loss 3.0286\n",
      "step 5000 of 20000: train_loss 3.0561, val loss 3.0091\n",
      "step 5100 of 20000: train_loss 2.9849, val loss 2.9654\n",
      "step 5200 of 20000: train_loss 3.0332, val loss 2.9801\n",
      "step 5300 of 20000: train_loss 2.9902, val loss 2.9294\n",
      "step 5400 of 20000: train_loss 2.9908, val loss 3.0349\n",
      "step 5500 of 20000: train_loss 2.9985, val loss 3.0097\n",
      "step 5600 of 20000: train_loss 2.9878, val loss 2.9548\n",
      "step 5700 of 20000: train_loss 3.0184, val loss 3.0072\n",
      "step 5800 of 20000: train_loss 2.9529, val loss 2.9691\n",
      "step 5900 of 20000: train_loss 3.0767, val loss 3.0393\n",
      "step 6000 of 20000: train_loss 2.9850, val loss 2.9871\n",
      "step 6100 of 20000: train_loss 3.0024, val loss 2.9809\n",
      "step 6200 of 20000: train_loss 2.9857, val loss 2.9493\n",
      "step 6300 of 20000: train_loss 3.0500, val loss 3.0101\n",
      "step 6400 of 20000: train_loss 3.0443, val loss 2.9854\n",
      "step 6500 of 20000: train_loss 2.9441, val loss 2.9939\n",
      "step 6600 of 20000: train_loss 3.0377, val loss 2.9816\n",
      "step 6700 of 20000: train_loss 3.0224, val loss 2.9642\n",
      "step 6800 of 20000: train_loss 2.9523, val loss 2.9615\n",
      "step 6900 of 20000: train_loss 3.0518, val loss 2.9464\n",
      "step 7000 of 20000: train_loss 2.9982, val loss 2.9458\n",
      "step 7100 of 20000: train_loss 3.0567, val loss 3.0282\n",
      "step 7200 of 20000: train_loss 3.0207, val loss 3.0370\n",
      "step 7300 of 20000: train_loss 2.9630, val loss 2.9898\n",
      "step 7400 of 20000: train_loss 2.9798, val loss 3.0265\n",
      "step 7500 of 20000: train_loss 2.9865, val loss 2.9761\n",
      "step 7600 of 20000: train_loss 3.0191, val loss 2.9946\n",
      "step 7700 of 20000: train_loss 3.0100, val loss 2.9523\n",
      "step 7800 of 20000: train_loss 3.0543, val loss 3.0057\n",
      "step 7900 of 20000: train_loss 3.0056, val loss 2.9981\n",
      "step 8000 of 20000: train_loss 2.9861, val loss 2.9678\n",
      "step 8100 of 20000: train_loss 2.9885, val loss 2.9746\n",
      "step 8200 of 20000: train_loss 2.9735, val loss 2.9708\n",
      "step 8300 of 20000: train_loss 3.0082, val loss 3.0166\n",
      "step 8400 of 20000: train_loss 2.9990, val loss 2.9640\n",
      "step 8500 of 20000: train_loss 3.0409, val loss 2.9836\n",
      "step 8600 of 20000: train_loss 3.0038, val loss 2.9591\n",
      "step 8700 of 20000: train_loss 3.0346, val loss 2.9748\n",
      "step 8800 of 20000: train_loss 3.0290, val loss 3.0135\n",
      "step 8900 of 20000: train_loss 3.0129, val loss 2.9368\n",
      "step 9000 of 20000: train_loss 2.9681, val loss 2.9605\n",
      "step 9100 of 20000: train_loss 3.0185, val loss 2.9709\n",
      "step 9200 of 20000: train_loss 3.0470, val loss 2.9937\n",
      "step 9300 of 20000: train_loss 2.9871, val loss 2.9361\n",
      "step 9400 of 20000: train_loss 3.0417, val loss 2.9552\n",
      "step 9500 of 20000: train_loss 3.0188, val loss 2.9917\n",
      "step 9600 of 20000: train_loss 3.0230, val loss 2.9735\n",
      "step 9700 of 20000: train_loss 3.0298, val loss 2.9752\n",
      "step 9800 of 20000: train_loss 2.9916, val loss 2.9967\n",
      "step 9900 of 20000: train_loss 3.0130, val loss 2.9548\n",
      "step 10000 of 20000: train_loss 3.0178, val loss 3.0018\n",
      "step 10100 of 20000: train_loss 2.9808, val loss 3.0275\n",
      "step 10200 of 20000: train_loss 2.9369, val loss 2.9897\n",
      "step 10300 of 20000: train_loss 2.9406, val loss 2.9966\n",
      "step 10400 of 20000: train_loss 3.0211, val loss 2.9908\n",
      "step 10500 of 20000: train_loss 3.0544, val loss 2.9985\n",
      "step 10600 of 20000: train_loss 2.9787, val loss 2.9652\n",
      "step 10700 of 20000: train_loss 3.0311, val loss 2.9455\n",
      "step 10800 of 20000: train_loss 3.0280, val loss 2.9611\n",
      "step 10900 of 20000: train_loss 3.0150, val loss 2.9893\n",
      "step 11000 of 20000: train_loss 3.0404, val loss 2.9864\n",
      "step 11100 of 20000: train_loss 3.0238, val loss 3.0192\n",
      "step 11200 of 20000: train_loss 3.0094, val loss 2.9329\n",
      "step 11300 of 20000: train_loss 3.0656, val loss 2.9604\n",
      "step 11400 of 20000: train_loss 2.9881, val loss 2.9902\n",
      "step 11500 of 20000: train_loss 3.0217, val loss 2.9912\n",
      "step 11600 of 20000: train_loss 2.9900, val loss 2.9617\n",
      "step 11700 of 20000: train_loss 3.0110, val loss 2.9436\n",
      "step 11800 of 20000: train_loss 3.0070, val loss 3.0005\n",
      "step 11900 of 20000: train_loss 3.0467, val loss 2.9528\n",
      "step 12000 of 20000: train_loss 2.9957, val loss 2.9969\n",
      "step 12100 of 20000: train_loss 2.9632, val loss 2.9883\n",
      "step 12200 of 20000: train_loss 3.0380, val loss 2.9441\n",
      "step 12300 of 20000: train_loss 3.0316, val loss 3.0348\n",
      "step 12400 of 20000: train_loss 3.0460, val loss 2.9624\n",
      "step 12500 of 20000: train_loss 3.0495, val loss 2.9811\n",
      "step 12600 of 20000: train_loss 2.9921, val loss 2.9748\n",
      "step 12700 of 20000: train_loss 3.0058, val loss 2.9616\n",
      "step 12800 of 20000: train_loss 3.0062, val loss 2.9711\n",
      "step 12900 of 20000: train_loss 3.0278, val loss 2.9447\n",
      "step 13000 of 20000: train_loss 3.0165, val loss 2.9630\n",
      "step 13100 of 20000: train_loss 3.0317, val loss 2.9673\n",
      "step 13200 of 20000: train_loss 3.0182, val loss 2.9853\n",
      "step 13300 of 20000: train_loss 3.0114, val loss 2.9734\n",
      "step 13400 of 20000: train_loss 2.9978, val loss 2.9709\n",
      "step 13500 of 20000: train_loss 3.0022, val loss 2.9949\n",
      "step 13600 of 20000: train_loss 3.0084, val loss 2.9270\n",
      "step 13700 of 20000: train_loss 2.9934, val loss 2.9845\n",
      "step 13800 of 20000: train_loss 2.9947, val loss 2.9892\n",
      "step 13900 of 20000: train_loss 2.9775, val loss 2.9158\n",
      "step 14000 of 20000: train_loss 3.0232, val loss 2.9575\n",
      "step 14100 of 20000: train_loss 3.0402, val loss 3.0265\n",
      "step 14200 of 20000: train_loss 2.9847, val loss 3.0153\n",
      "step 14300 of 20000: train_loss 2.9698, val loss 2.9814\n",
      "step 14400 of 20000: train_loss 3.0012, val loss 2.9515\n",
      "step 14500 of 20000: train_loss 2.9728, val loss 2.9829\n",
      "step 14600 of 20000: train_loss 3.0628, val loss 2.9613\n",
      "step 14700 of 20000: train_loss 2.9960, val loss 2.9899\n",
      "step 14800 of 20000: train_loss 3.0289, val loss 2.9923\n",
      "step 14900 of 20000: train_loss 3.0141, val loss 2.9355\n",
      "step 15000 of 20000: train_loss 3.0250, val loss 2.9740\n",
      "step 15100 of 20000: train_loss 2.9949, val loss 2.9414\n",
      "step 15200 of 20000: train_loss 3.0642, val loss 3.0194\n",
      "step 15300 of 20000: train_loss 2.9848, val loss 2.9634\n",
      "step 15400 of 20000: train_loss 3.0092, val loss 2.9621\n",
      "step 15500 of 20000: train_loss 3.0066, val loss 2.9617\n",
      "step 15600 of 20000: train_loss 3.0072, val loss 2.9354\n",
      "step 15700 of 20000: train_loss 2.9957, val loss 2.9326\n",
      "step 15800 of 20000: train_loss 3.0553, val loss 2.9592\n",
      "step 15900 of 20000: train_loss 3.0155, val loss 2.9614\n",
      "step 16000 of 20000: train_loss 3.0244, val loss 2.9784\n",
      "step 16100 of 20000: train_loss 2.9744, val loss 2.9857\n",
      "step 16200 of 20000: train_loss 2.9761, val loss 2.9544\n",
      "step 16300 of 20000: train_loss 3.0723, val loss 2.9770\n",
      "step 16400 of 20000: train_loss 3.0447, val loss 3.0042\n",
      "step 16500 of 20000: train_loss 3.0081, val loss 2.9989\n",
      "step 16600 of 20000: train_loss 2.9719, val loss 3.0217\n",
      "step 16700 of 20000: train_loss 3.0549, val loss 2.9452\n",
      "step 16800 of 20000: train_loss 3.0392, val loss 2.9988\n",
      "step 16900 of 20000: train_loss 3.0225, val loss 3.0073\n",
      "step 17000 of 20000: train_loss 3.0563, val loss 2.9929\n",
      "step 17100 of 20000: train_loss 3.0251, val loss 2.9945\n",
      "step 17200 of 20000: train_loss 2.9717, val loss 2.9972\n",
      "step 17300 of 20000: train_loss 2.9743, val loss 2.9869\n",
      "step 17400 of 20000: train_loss 2.9458, val loss 2.9329\n",
      "step 17500 of 20000: train_loss 3.0448, val loss 3.0273\n",
      "step 17600 of 20000: train_loss 3.0548, val loss 2.9167\n",
      "step 17700 of 20000: train_loss 2.9814, val loss 2.9701\n",
      "step 17800 of 20000: train_loss 2.9995, val loss 2.9646\n",
      "step 17900 of 20000: train_loss 3.0148, val loss 2.9571\n",
      "step 18000 of 20000: train_loss 3.0130, val loss 3.0043\n",
      "step 18100 of 20000: train_loss 2.9848, val loss 2.9713\n",
      "step 18200 of 20000: train_loss 2.9973, val loss 3.0081\n",
      "step 18300 of 20000: train_loss 2.9845, val loss 2.9690\n",
      "step 18400 of 20000: train_loss 3.0179, val loss 2.9646\n",
      "step 18500 of 20000: train_loss 3.0021, val loss 2.9819\n",
      "step 18600 of 20000: train_loss 3.0019, val loss 2.9923\n",
      "step 18700 of 20000: train_loss 3.0274, val loss 2.9564\n",
      "step 18800 of 20000: train_loss 3.0035, val loss 2.9780\n",
      "step 18900 of 20000: train_loss 3.0310, val loss 2.9493\n",
      "step 19000 of 20000: train_loss 2.9955, val loss 2.9640\n",
      "step 19100 of 20000: train_loss 3.0345, val loss 3.0066\n",
      "step 19200 of 20000: train_loss 3.0079, val loss 2.9546\n",
      "step 19300 of 20000: train_loss 2.9790, val loss 3.0206\n",
      "step 19400 of 20000: train_loss 3.0495, val loss 2.9888\n",
      "step 19500 of 20000: train_loss 3.0209, val loss 2.9669\n",
      "step 19600 of 20000: train_loss 2.9987, val loss 2.9831\n",
      "step 19700 of 20000: train_loss 2.9714, val loss 2.9590\n",
      "step 19800 of 20000: train_loss 3.0187, val loss 3.0211\n",
      "step 19900 of 20000: train_loss 2.9887, val loss 2.9437\n"
     ]
    }
   ],
   "execution_count": 8
  },
  {
   "cell_type": "code",
   "id": "6057349c",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-10-20T15:24:53.032684Z",
     "start_time": "2024-10-20T15:24:52.347572Z"
    }
   },
   "source": [
    "   \n",
    "model3.eval()\n",
    "\n",
    "int_nodes_vals0 = {'X1':np.array([0.0,])}\n",
    "int_nodes_vals1 = {'X1':np.array([1.0,])}\n",
    "effect_var = 'Y'\n",
    "effect_index = var_names3.index(effect_var)\n",
    "\n",
    "inf = CausalInference(dag=DAGnx3)\n",
    "preds0 = inf.forward(all_data3, model=model3, intervention_nodes_vals=int_nodes_vals0)\n",
    "preds1 = inf.forward(all_data3, model=model3, intervention_nodes_vals=int_nodes_vals1)\n",
    "ATE_pred = (preds1[:,effect_index,:] - preds0[:,effect_index,:]).mean(0)\n",
    "eATE = np.abs(ATE_pred - ATE)\n",
    "print('ATE:', ATE, 'est ATE:', ATE_pred, 'error:', eATE)\n",
    "\n",
    "preds = model3(train_data.to(device))\n",
    "plt.scatter(train_data[:,effect_index,-1].detach().cpu().numpy(), preds[:, effect_index, -1].detach().cpu().numpy())\n",
    "print('Mean Squared Error Across All Vars:', ((train_data - preds.detach().cpu())**2).mean())\n",
    "print('Mean Squared Error Across Outcome:', ((train_data[:,effect_index,:] - preds[:,effect_index,:].detach().cpu())**2).mean())"
   ],
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "ATE: [1.12] est ATE: [1.05465286] error: [0.06534714]\n",
      "Mean Squared Error Across All Vars: tensor(1.0201)\n",
      "Mean Squared Error Across Outcome: tensor(1.0037)\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ],
      "image/png": "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"
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "execution_count": 9
  },
  {
   "cell_type": "code",
   "id": "1565ba79",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-10-20T15:24:53.035249Z",
     "start_time": "2024-10-20T15:24:53.033437Z"
    }
   },
   "source": [],
   "outputs": [],
   "execution_count": 9
  },
  {
   "cell_type": "code",
   "id": "8ae93b5a",
   "metadata": {
    "ExecuteTime": {
     "end_time": "2024-10-20T15:24:53.037732Z",
     "start_time": "2024-10-20T15:24:53.036091Z"
    }
   },
   "source": [],
   "outputs": [],
   "execution_count": 9
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.9.19"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
