{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "ename": "ModuleNotFoundError",
     "evalue": "No module named 'map_dataset'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mModuleNotFoundError\u001b[0m                       Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[1], line 21\u001b[0m\n\u001b[1;32m     19\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01mclf\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m classifier\n\u001b[1;32m     20\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01mmex\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m mex\n\u001b[0;32m---> 21\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01mmap_dataset\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m dna_dataset\n\u001b[1;32m     22\u001b[0m \u001b[38;5;28;01mfrom\u001b[39;00m \u001b[38;5;21;01mtorch\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mutils\u001b[39;00m\u001b[38;5;21;01m.\u001b[39;00m\u001b[38;5;21;01mdata\u001b[39;00m \u001b[38;5;28;01mimport\u001b[39;00m DataLoader\n\u001b[1;32m     24\u001b[0m \u001b[38;5;66;03m# Specific Imports from Third-Party Libraries\u001b[39;00m\n",
      "\u001b[0;31mModuleNotFoundError\u001b[0m: No module named 'map_dataset'"
     ]
    }
   ],
   "source": [
    "# Standard Library Imports\n",
    "import os\n",
    "import sys\n",
    "\n",
    "# Third-Party Library Imports\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "import torch\n",
    "import logomaker\n",
    "\n",
    "# Local Imports\n",
    "parent_dir = \"../\"\n",
    "root_dir = \"../..\"\n",
    "sys.path.append(root_dir)\n",
    "sys.path.append(parent_dir)\n",
    "import utils\n",
    "import sim_dataset\n",
    "from clf import classifier\n",
    "from mex import mex\n",
    "from map_dataset import dna_dataset\n",
    "from torch.utils.data import DataLoader\n",
    "\n",
    "# Specific Imports from Third-Party Libraries\n",
    "from tqdm import tqdm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "clf loaded\n"
     ]
    }
   ],
   "source": [
    "# Set experiment name\n",
    "exp_name = \"coo\"\n",
    "\n",
    "# Set torch device\n",
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "\n",
    "# Load classifier\n",
    "model_dir = os.path.join(parent_dir, exp_name, \"models\")\n",
    "clf = classifier().to(device)\n",
    "clf.load_state_dict(torch.load(os.path.join(model_dir, \"clf.pt\")))\n",
    "clf.eval()\n",
    "print(\"clf loaded\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Warning: Some weights of classifier not frozen. Froze all weights.\n",
      "Warning: Some weights of encoder not frozen. Froze all weights.\n",
      "explainer loaded\n"
     ]
    }
   ],
   "source": [
    "# Distance matrix\n",
    "indices = torch.arange(500)\n",
    "D = torch.abs(indices.unsqueeze(1) - indices.unsqueeze(0))\n",
    "D = D.unsqueeze(0).to(device)\n",
    "\n",
    "# Background samples\n",
    "bkgd_seqs = utils.generate_background(1000)\n",
    "\n",
    "# Load explainer (using classifier encoder)\n",
    "mex_type = \"suff\"\n",
    "explainer_dir = os.path.join(parent_dir, exp_name, \"results\", mex_type)\n",
    "explainer = mex(clf, D, bkgd_seqs, init_mask='ones', freeze=True).to(device)\n",
    "explainer.load_state_dict(torch.load(os.path.join(explainer_dir, \"explainer.pt\")))\n",
    "explainer.eval()\n",
    "print(\"explainer loaded\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Path to data configs\n",
    "sim_pos_config_path = os.path.join(parent_dir, exp_name, \"data\", \"spi1_ctcf_exp_config.json\")\n",
    "sim_neg_config_path = os.path.join(parent_dir, exp_name, \"data\", \"spi1_ctcf_exp_neg_config.json\")\n",
    "\n",
    "# Dataloaders\n",
    "batch_size = 10\n",
    "num_batches = 50\n",
    "mex_test_loader = sim_dataset.create_data_loader(\n",
    "    sim_pos_config_path, neg_motif_config_path=sim_neg_config_path, \n",
    "    batch_size=batch_size, num_batches=num_batches\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 50/50 [00:05<00:00,  9.80it/s]\n"
     ]
    }
   ],
   "source": [
    "filtered_data = utils.filter_dataset(mex_test_loader, clf)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "dataloader = DataLoader(filtered_data, batch_size=batch_size, shuffle=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 87,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 50/50 [00:03<00:00, 13.97it/s]\n"
     ]
    }
   ],
   "source": [
    "# Testing filtering correct predictions\n",
    "correct_x = []\n",
    "correct_y = []\n",
    "incorrect = 0\n",
    "for batch in tqdm(mex_test_loader):\n",
    "    x, y = batch\n",
    "    x = x.float().to(device)\n",
    "    probs = clf(x).squeeze(-1).detach().cpu()\n",
    "    y_hat = (probs >= 0.5)*1.0\n",
    "    correct_idx = (y_hat == y)\n",
    "    x = x.detach().cpu()\n",
    "    correct_x.append(x[correct_idx])\n",
    "    correct_y.append(y[correct_idx])\n",
    "correct_x = torch.cat(correct_x, dim = 0)\n",
    "correct_y = torch.cat(correct_y, dim = 0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 89,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "torch.Size([473, 500, 4])"
      ]
     },
     "execution_count": 89,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "correct_x.shape"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 90,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.946"
      ]
     },
     "execution_count": 90,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 68,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([1., 1., 1., 0., 0., 1., 1., 1., 0., 0.], dtype=torch.float64)"
      ]
     },
     "execution_count": 68,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "y"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Iterator over test set\n",
    "it = iter(mex_test_loader)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "f(x) =  0.9999228715896606\n",
      "f(x) =  0.9999228715896606\n"
     ]
    }
   ],
   "source": [
    "# Explain true positive predictions in test set\n",
    "x, y = next(it)\n",
    "x = x[0].unsqueeze(0).float().to(device)\n",
    "\n",
    "print('f(x) = ', clf(x).item())\n",
    "print('f(x) = ', explainer.clf(x).item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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3gbgYHPkkUfq1ltv1TPPPKNnnnlG73rXu3TBBRe4k4G8rwE4mD399NP60Y9+pFNOOSWwnvc2AIcl6yBw5plnWgsWLAisO/HEE60vfOEL++mIAKA4kqwHH3zQ/d00TWvixInWrbfe6q7r6emxGhoarDvvvNOyLMvat2+fVVFRYd17773umK1bt1qJRMJauXLlsB07AMRpaWmxJFmPPvqoZVm8twE4dIwePdq66667eF8DcFBrb2+3pk2bZq1atcp6xzveYV199dWWZfGZDcDh64DPBOnr69PatWs1b968wPp58+ZpzZo1++moAGBgNm7cqObm5sB7WlVVld7xjne472lr165Vf39/YMzkyZM1c+ZM3vcAHBBaW1slSWPGjJHEexuAg18mk9G9996rzs5OzZkzh/c1AAe1K664Qu9///v1nve8J7Ce9zYAh6vU/j6AQnbt2qVMJqPGxsbA+sbGRjU3N++nowKAgXHet6Le0zZt2uSOqays1OjRo0NjeN8DsL9ZlqXFixfrbW97m2bOnCmJ9zYAB6/nn39ec+bMUU9Pj0aMGKEHH3xQb3rTm9yJPt7XABxs7r33Xj377LN6+umnQ4/xmQ3A4eqAD4I4DMMI/G5ZVmgdABwsBvKexvsegAPBpz71Kf31r3/V448/HnqM9zYAB5sTTjhB69ev1759+3T//ffrIx/5iB599FH3cd7XABxMNm/erKuvvlqPPPKIqqurY8fx3gbgcHPAl8MaN26ckslkKNrc0tISilwDwIFu4sSJkpT3PW3ixInq6+vT3r17Y8cAwP5w5ZVX6qGHHtIf//hHHXnkke563tsAHKwqKyt1/PHHa/bs2VqyZIlOPfVUffe73+V9DcBBae3atWppadGsWbOUSqWUSqX06KOP6nvf+55SqZT73sR7G4DDzQEfBKmsrNSsWbO0atWqwPpVq1Zp7ty5++moAGBgpk6dqokTJwbe0/r6+vToo4+672mzZs1SRUVFYMz27dv1wgsv8L4HYL+wLEuf+tSn9MADD+gPf/iDpk6dGnic9zYAhwrLstTb28v7GoCD0rvf/W49//zzWr9+vftn9uzZ+vCHP6z169fr2GOP5b0NwGHpoCiHtXjxYl1yySWaPXu25syZox/96EdqamrSggUL9vehAUBIR0eHXnvtNff3jRs3av369RozZoyOOuooLVq0SLfccoumTZumadOm6ZZbblFtba0uvvhiSVJDQ4Pmz5+vT3/60xo7dqzGjBmjz3zmMzr55JNDje0AYDhcccUV+tnPfqZf//rXGjlypHv3YENDg2pqamQYBu9tAA461113nc477zxNmTJF7e3tuvfee/WnP/1JK1eu5H0NwEFp5MiRbs82R11dncaOHeuu570NwOHooAiCXHTRRdq9e7duvvlmbd++XTNnztSKFSt09NFH7+9DA4CQZ555Ru985zvd3xcvXixJ+shHPqJ77rlHn/vc59Td3a2FCxdq7969estb3qJHHnlEI0eOdLf5zne+o1QqpQsvvFDd3d1697vfrXvuuUfJZHLYXw8ALFu2TJJ0zjnnBNb/+Mc/1mWXXSZJvLcBOOjs2LFDl1xyibZv366GhgadcsopWrlypc4991xJvK8BODTx3gbgcGRYlmXt74MAAAAAAAAAAAAotwO+JwgAAAAAAAAAAMBAEAQBAAAAAAAAAACHJIIgAAAAAAAAAADgkEQQBAAAAAAAAAAAHJIIggAAAAAAAAAAgEMSQRAAAAAAAAAAAHBIIggCAAAAAAAAAAAOSQRBAAAAAAAAAADAIYkgCAAAAAAAAAAAOCQRBAEAAAAAAAAAAIckgiAAAAAAAAAAAOCQRBAEAAAAAAAAAAAckv5/sc8Ah7ucDFkAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 2000x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Visualize explainer\n",
    "center_slice = slice(1,500)\n",
    "w, sigma = explainer(x)\n",
    "explanation = (((explainer._get_mask(w, sigma, D))))\n",
    "sample = x.squeeze(0).cpu()\n",
    "mask = (explanation.detach().cpu().view(500,-1))\n",
    "scores_to_plot = (mask.repeat(1,4).cpu() * sample.numpy())[center_slice]\n",
    "fig, ax = plt.subplots(figsize=(20, 3))\n",
    "plot_df = pd.DataFrame(scores_to_plot, columns=[\"A\", \"C\", \"G\", \"T\"])\n",
    "logomaker.Logo(plot_df, ax=ax)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "suff =  0.0057544708251953125\n",
      "necc =  0.594709575176239\n"
     ]
    }
   ],
   "source": [
    "# Sanity check\n",
    "masks = mask.unsqueeze(0).to(device)\n",
    "num_samples = 500\n",
    "indices = np.random.choice(len(bkgd_seqs), num_samples, replace=False)\n",
    "b_samples = torch.tensor(bkgd_seqs[indices], device=device)\n",
    "x_S, x_Sc = utils.sample_masked_X(x, masks, b_samples)\n",
    "x_S = x_S.float()\n",
    "x_Sc = x_Sc.float()\n",
    "\n",
    "# Evaluate f_x, f_xS, and f_xSc\n",
    "f_x = clf(x).squeeze(-1)\n",
    "f_xS = clf(x_S.view(-1, 500, 4)).view(-1, num_samples).mean(-1)\n",
    "f_xSc = clf(x_Sc.view(-1, 500, 4)).view(-1, num_samples).mean(-1)\n",
    "\n",
    "# Evaluate losses\n",
    "suff = torch.abs(f_x - f_xS)\n",
    "print(\"suff = \", suff.item())\n",
    "\n",
    "# Evaluate losses\n",
    "necc = torch.abs(f_x - f_xSc)\n",
    "print(\"necc = \", necc.item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "cuda121",
   "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.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
