{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import lmdb\n",
    "import os\n",
    "import pickle5 as pickle\n",
    "from functools import lru_cache\n",
    "class LMDBDataset:\n",
    "    def __init__(self, db_path):\n",
    "        self.db_path = db_path\n",
    "        assert os.path.isfile(self.db_path), \"{} not found\".format(self.db_path)\n",
    "        env = self.connect_db(self.db_path)\n",
    "        with env.begin() as txn:\n",
    "            self._keys = list(txn.cursor().iternext(values=False))\n",
    "\n",
    "    def connect_db(self, lmdb_path, save_to_self=False):\n",
    "        env = lmdb.open(\n",
    "            lmdb_path,\n",
    "            subdir=False,\n",
    "            readonly=True,\n",
    "            lock=False,\n",
    "            readahead=False,\n",
    "            meminit=False,\n",
    "            max_readers=256,\n",
    "        )\n",
    "        if not save_to_self:\n",
    "            return env\n",
    "        else:\n",
    "            self.env = env\n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self._keys)\n",
    "\n",
    "    @lru_cache(maxsize=16)\n",
    "    def __getitem__(self, idx):\n",
    "        if not hasattr(self, \"env\"):\n",
    "            self.connect_db(self.db_path, save_to_self=True)\n",
    "        datapoint_pickled = self.env.begin().get(f\"{idx}\".encode(\"ascii\"))\n",
    "        data = pickle.loads(datapoint_pickled)\n",
    "        return data\n",
    "LMDBdataset=LMDBDataset(\"data/ligands/train.lmdb\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "LMDBdataset.connect_db(LMDBdataset.db_path, save_to_self=True)\n",
    "datapoint_pickled = LMDBdataset.env.begin().get(f\"{0}\".encode(\"ascii\"))\n",
    "data = pickle.loads(datapoint_pickled)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'COc1c(Cl)cc([N+]2C(=O)N[C@]3(CCSC3)C2=O)cc1Cl'"
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "data[\"smi\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import csv\n",
    "import math\n",
    "import time\n",
    "import random\n",
    "import networkx as nx\n",
    "import numpy as np\n",
    "from copy import deepcopy\n",
    "\n",
    "import lmdb\n",
    "import pickle5 as pickle\n",
    "from functools import lru_cache\n",
    "\n",
    "import torch\n",
    "import torch.nn.functional as F\n",
    "# from torch.utils.data import Dataset, DataLoader\n",
    "from torch.utils.data.sampler import SubsetRandomSampler\n",
    "import torchvision.transforms as transforms\n",
    "\n",
    "from torch_scatter import scatter\n",
    "from torch_geometric.data import Data, Dataset, DataLoader\n",
    "\n",
    "import rdkit\n",
    "from rdkit import Chem\n",
    "from rdkit.Chem.rdchem import HybridizationType\n",
    "from rdkit.Chem.rdchem import BondType as BT\n",
    "from rdkit.Chem import AllChem\n",
    "\n",
    "class MoleculeDataset(Dataset):\n",
    "    def __init__(self, db_path):\n",
    "        self.db_path = db_path\n",
    "        assert os.path.isfile(self.db_path), \"{} not found\".format(self.db_path)\n",
    "        env = self.connect_db(self.db_path)\n",
    "        with env.begin() as txn:\n",
    "            self._keys = list(txn.cursor().iternext(values=False))\n",
    "\n",
    "    @lru_cache(maxsize=16)\n",
    "    def __getitem__(self, idx):\n",
    "        if not hasattr(self, \"env\"):\n",
    "            self.connect_db(self.db_path, save_to_self=True)\n",
    "        datapoint_pickled = self.env.begin().get(f\"{idx}\".encode(\"ascii\"))\n",
    "        data = pickle.loads(datapoint_pickled)\n",
    "\n",
    "        mol = Chem.MolFromSmiles(data[\"smi\"])\n",
    "        # mol = Chem.AddHs(mol)\n",
    "\n",
    "        N = mol.GetNumAtoms()\n",
    "        M = mol.GetNumBonds()\n",
    "\n",
    "        type_idx = []\n",
    "        chirality_idx = []\n",
    "        atomic_number = []\n",
    "        atoms = mol.GetAtoms()\n",
    "        bonds = mol.GetBonds()\n",
    "        # Sample 2 different centers to start for i and j\n",
    "        start_i, start_j = random.sample(list(range(N)), 2)\n",
    "\n",
    "        # Construct the original molecular graph from edges (bonds)\n",
    "        edges = []\n",
    "        for bond in bonds:\n",
    "            edges.append([bond.GetBeginAtomIdx(), bond.GetEndAtomIdx()])\n",
    "        molGraph = nx.Graph(edges)\n",
    "        \n",
    "        # Get the graph for i and j after removing subgraphs\n",
    "        # G_i, removed_i = removeSubgraph(molGraph, start_i)\n",
    "        # G_j, removed_j = removeSubgraph(molGraph, start_j)\n",
    "\n",
    "        # percent_i, percent_j = random.uniform(0, 0.25), random.uniform(0, 0.25)\n",
    "        percent_i, percent_j = 0.25, 0.25\n",
    "        # percent_i, percent_j = 0.2, 0.2\n",
    "        G_i, removed_i = removeSubgraph(molGraph, start_i, percent_i)\n",
    "        G_j, removed_j = removeSubgraph(molGraph, start_j, percent_j)\n",
    "        \n",
    "        for atom in atoms:\n",
    "            type_idx.append(ATOM_LIST.index(atom.GetAtomicNum()))\n",
    "            chirality_idx.append(CHIRALITY_LIST.index(atom.GetChiralTag()))\n",
    "            atomic_number.append(atom.GetAtomicNum())\n",
    "\n",
    "        x1 = torch.tensor(type_idx, dtype=torch.long).view(-1,1)\n",
    "        x2 = torch.tensor(chirality_idx, dtype=torch.long).view(-1,1)\n",
    "        x = torch.cat([x1, x2], dim=-1)\n",
    "        # x shape (N, 2) [type, chirality]\n",
    "\n",
    "        # Mask the atoms in the removed list\n",
    "        x_i = deepcopy(x)\n",
    "        for atom_idx in removed_i:\n",
    "            # Change atom type to 118, and chirality to 0\n",
    "            x_i[atom_idx,:] = torch.tensor([len(ATOM_LIST), 0])\n",
    "        x_j = deepcopy(x)\n",
    "        for atom_idx in removed_j:\n",
    "            # Change atom type to 118, and chirality to 0\n",
    "            x_j[atom_idx,:] = torch.tensor([len(ATOM_LIST), 0])\n",
    "\n",
    "        # Only consider bond still exist after removing subgraph\n",
    "        row_i, col_i, row_j, col_j = [], [], [], []\n",
    "        edge_feat_i, edge_feat_j = [], []\n",
    "        G_i_edges = list(G_i.edges)\n",
    "        G_j_edges = list(G_j.edges)\n",
    "        for bond in mol.GetBonds():\n",
    "            start, end = bond.GetBeginAtomIdx(), bond.GetEndAtomIdx()\n",
    "            feature = [\n",
    "                BOND_LIST.index(bond.GetBondType()),\n",
    "                BONDDIR_LIST.index(bond.GetBondDir())\n",
    "            ]\n",
    "            if (start, end) in G_i_edges:\n",
    "                row_i += [start, end]\n",
    "                col_i += [end, start]\n",
    "                edge_feat_i.append(feature)\n",
    "                edge_feat_i.append(feature)\n",
    "            if (start, end) in G_j_edges:\n",
    "                row_j += [start, end]\n",
    "                col_j += [end, start]\n",
    "                edge_feat_j.append(feature)\n",
    "                edge_feat_j.append(feature)\n",
    "\n",
    "        edge_index_i = torch.tensor([row_i, col_i], dtype=torch.long)\n",
    "        edge_attr_i = torch.tensor(np.array(edge_feat_i), dtype=torch.long)\n",
    "        edge_index_j = torch.tensor([row_j, col_j], dtype=torch.long)\n",
    "        edge_attr_j = torch.tensor(np.array(edge_feat_j), dtype=torch.long)\n",
    "        \n",
    "        data_i = Data(x=x_i, edge_index=edge_index_i, edge_attr=edge_attr_i)\n",
    "        data_j = Data(x=x_j, edge_index=edge_index_j, edge_attr=edge_attr_j)\n",
    "        \n",
    "        return data, data_i, data_j\n",
    "\n",
    "\n",
    "    def connect_db(self, lmdb_path, save_to_self=False):\n",
    "        env = lmdb.open(\n",
    "            lmdb_path,\n",
    "            subdir=False,\n",
    "            readonly=True,\n",
    "            lock=False,\n",
    "            readahead=False,\n",
    "            meminit=False,\n",
    "            max_readers=256,\n",
    "        )\n",
    "        if not save_to_self:\n",
    "            return env\n",
    "        else:\n",
    "            self.env = env\n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self._keys)\n",
    "    \n",
    "Dataset=MoleculeDataset(\"data/ligands/train.lmdb\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Data(edge_attr=[44, 2], edge_index=[2, 44], x=[31, 2])"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "def removeSubgraph(Graph, center, percent=0.2):\n",
    "    assert percent <= 1\n",
    "    G = Graph.copy()\n",
    "    num = int(np.floor(len(G.nodes)*percent))\n",
    "    removed = []\n",
    "    temp = [center]\n",
    "    \n",
    "    while len(removed) < num:\n",
    "        neighbors = []\n",
    "        for n in temp:\n",
    "            neighbors.extend([i for i in G.neighbors(n) if i not in temp])      \n",
    "        for n in temp:\n",
    "            if len(removed) < num:\n",
    "                G.remove_node(n)\n",
    "                removed.append(n)\n",
    "            else:\n",
    "                break\n",
    "        temp = list(set(neighbors))\n",
    "    return G, removed\n",
    "\n",
    "ATOM_LIST = list(range(1,119))\n",
    "CHIRALITY_LIST = [\n",
    "    Chem.rdchem.ChiralType.CHI_UNSPECIFIED,\n",
    "    Chem.rdchem.ChiralType.CHI_TETRAHEDRAL_CW,\n",
    "    Chem.rdchem.ChiralType.CHI_TETRAHEDRAL_CCW,\n",
    "    Chem.rdchem.ChiralType.CHI_OTHER\n",
    "]\n",
    "BOND_LIST = [\n",
    "    BT.SINGLE, \n",
    "    BT.DOUBLE, \n",
    "    BT.TRIPLE, \n",
    "    BT.AROMATIC\n",
    "]\n",
    "BONDDIR_LIST = [\n",
    "    Chem.rdchem.BondDir.NONE,\n",
    "    Chem.rdchem.BondDir.ENDUPRIGHT,\n",
    "    Chem.rdchem.BondDir.ENDDOWNRIGHT\n",
    "]\n",
    "\n",
    "Dataset.__getitem__(5)[1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import networkx as nx\n",
    "import matplotlib.pyplot as plt\n",
    "data=Dataset.__getitem__(5)[1]\n",
    "\n",
    "# Convert torch_geometric data to a NetworkX graph\n",
    "G = nx.Graph()\n",
    "G.add_nodes_from(range(data.num_nodes))\n",
    "G.add_edges_from(data.edge_index.t().tolist())\n",
    "\n",
    "# Visualize the graph using Matplotlib\n",
    "pos = nx.spring_layout(G)\n",
    "nx.draw(G, pos, with_labels=True, font_weight='bold', node_size=70, font_size=8, font_color='black', edge_color='gray', linewidths=0.5)\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "molclr",
   "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.7.16"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
