{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "db86ceb7-d2df-40d9-9bbb-f437d6114259",
   "metadata": {},
   "source": [
    "# Overview\n",
    "\n",
    "This is the first part of a 2 part tutorial which runs the PaRoutes benchmark.\n",
    "In this half of the tutorial, we run search algorithms using a pre-trained reaction model.\n",
    "Because these algorithms require policies and value functions,\n",
    "these must be carefully chosen before running the algorithm.\n",
    "\n",
    "**NOTE:** the PaRoutes reaction model requires the following additional packages:\n",
    "- `pandas` (to load data table)\n",
    "- `rdchiral` (to apply reaction templates)\n",
    "- `keras` (their pre-trained model is a template-classification model using keras)\n",
    "\n",
    "The model also requires some data files, which can be downloaded with `paroutes_setup.sh`.",
    "Running the following commands should set up everything:\n",
    "\n",
    "```\n",
    "pip install -U tensorflow keras\n",
    "conda install -c conda-forge rdchiral_cpp  # faster C++ version\n",
    "conda install -c conda-forge pandas\n",
    "cd tutorials/search  # need to be in this directory for the setup script to run correctly\n",
    "bash paroutes_setup.sh\n",
    "```"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9dbc23a9-f0cd-4619-97c9-c24bdae7025e",
   "metadata": {},
   "source": [
    "## Imports"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "8160f287-de1e-46f4-a255-6e62aab0a460",
   "metadata": {},
   "outputs": [],
   "source": [
    "from __future__ import annotations\n",
    "import math\n",
    "import pprint"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "b4bd66f4-7c84-4ee7-870e-0b580443c61c",
   "metadata": {},
   "outputs": [],
   "source": [
    "from syntheseus.search.chem import Molecule, BackwardReaction"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "23fcbd26-c250-4d39-a92b-f0dd14e2f36f",
   "metadata": {},
   "source": [
    "## Step 0: make 2 example molecules for testing"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "20295725-3f0c-4feb-8863-63f8658a6f1a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(Molecule(smiles='c1ccccc1', identifier=None, metadata={}),\n",
       " Molecule(smiles='CCCCCCCCCC', identifier=None, metadata={}))"
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "benzene = Molecule(\"c1ccccc1\", make_rdkit_mol=False)\n",
    "decane = Molecule(\"C\"*10, make_rdkit_mol=False)\n",
    "benzene, decane"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e470b5bb-ae52-40c4-a342-10af36be9a74",
   "metadata": {},
   "source": [
    "## Step 1: load PaRoutes mol-inventory and reaction model\n",
    "\n",
    "PaRoutes has two tasks: `n=1` and `n=5`.\n",
    "This notebook uses the `n=5` task by default,\n",
    "but this can be changed with the variable `PAROUTES_N`.\n",
    "We use version 2.0 of this benchmark which was released in late 2022.\n",
    "\n",
    "The file `paroutes.py` which accompanies this notebook contains wrappers for the pre-trained reaction models and sets of purchasable molecules.\n",
    "We load these below and briefly explore their properties."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "5b060b2c-83d2-4a00-a303-ab7a5b77f21b",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": []
    }
   ],
   "source": [
    "import paroutes\n",
    "PAROUTES_N = 5"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "74baf3d7-1d3c-4c1e-9b24-b2de62cd072e",
   "metadata": {},
   "source": [
    "#### 1a: load PaRoutes standardized mol inventory"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "9e8d0b43-bd7a-40b0-b327-3edb00a63407",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Size of inventory: 13325\n",
      "First few molecules in inventory:\n",
      "[Molecule(smiles='Cc1cc(C(C)N)cnc1OCC(F)(F)F', identifier=None, metadata={}),\n",
      " Molecule(smiles='N#CC1Cc2ccc(Nc3ccncc3[N+](=O)[O-])cc2C1',\n",
      "          identifier=None,\n",
      "          metadata={}),\n",
      " Molecule(smiles='COC=CC1CCCOC1', identifier=None, metadata={}),\n",
      " Molecule(smiles='CC(O)c1cccc(NCC(=O)OC(C)(C)C)n1',\n",
      "          identifier=None,\n",
      "          metadata={}),\n",
      " Molecule(smiles='CC(C)C[C@@H](C(=O)O)N(C)C(=O)OC(C)(C)C',\n",
      "          identifier=None,\n",
      "          metadata={})]\n"
     ]
    }
   ],
   "source": [
    "# Load the inventory and look at the first few mols\n",
    "inventory = paroutes.PaRoutesInventory(n=PAROUTES_N)\n",
    "print(f\"Size of inventory: {len(inventory.purchasable_mols())}\")\n",
    "print(\"First few molecules in inventory:\")\n",
    "pprint.pprint(list(inventory.purchasable_mols())[:5])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "b1e8c7b7-ee82-452a-bfd6-2b23a1d30f6e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(Molecule(smiles='c1ccccc1', identifier=None, metadata={'is_purchasable': True}),\n",
       " Molecule(smiles='CCCCCCCCCC', identifier=None, metadata={'is_purchasable': False}))"
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Example: Use the inventory to label some molecules\n",
    "# We find out that benzene is purchasable and decane is not\n",
    "inventory.fill_metadata(benzene)\n",
    "inventory.fill_metadata(decane)\n",
    "benzene, decane"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1876bb42-3d95-4104-8b43-3caf1179d1de",
   "metadata": {},
   "source": [
    "#### 1b: load PaRoutes reaction model and look at outputs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "b101e2c0-be72-4d2a-b12e-d182aaa304fc",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Output type: <class 'list'>, len: 2\n"
     ]
    }
   ],
   "source": [
    "# Load the reaction model, which is a template classifier\n",
    "rxn_model = paroutes.PaRoutesModel()\n",
    "\n",
    "# Example: call it on benzene and decane.\n",
    "# Like all reaction models, it returns a list of reaction lists (1 per input molecule)\n",
    "rxn_model_output = rxn_model([benzene, decane])\n",
    "print(f\"Output type: {type(rxn_model_output)}, len: {len(rxn_model_output)}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "dbb7e430-7eab-4947-9446-c34a2e6ad486",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Reaction #1 for mol #1:\n",
      "BackwardReaction(reactants=frozenset({Molecule(smiles='Nc1ccccc1',\n",
      "                                               identifier=None,\n",
      "                                               metadata={})}),\n",
      "                 product=Molecule(smiles='c1ccccc1',\n",
      "                                  identifier=None,\n",
      "                                  metadata={'is_purchasable': True}),\n",
      "                 identifier=None,\n",
      "                 metadata={'softmax': 0.05530696362257004,\n",
      "                           'template': '[c:2]:[cH;D2;+0:1]:[c:3]>>N-[c;H0;D3;+0:1](:[c:2]):[c:3]',\n",
      "                           'template_idx': 15583,\n",
      "                           'template_library_occurence': 326,\n",
      "                           'template_rank': 0})\n",
      "\n",
      "Reaction #11 for mol #1:\n",
      "BackwardReaction(reactants=frozenset({Molecule(smiles='Brc1ccccc1',\n",
      "                                               identifier=None,\n",
      "                                               metadata={})}),\n",
      "                 product=Molecule(smiles='c1ccccc1',\n",
      "                                  identifier=None,\n",
      "                                  metadata={'is_purchasable': True}),\n",
      "                 identifier=None,\n",
      "                 metadata={'softmax': 0.0013848659582436085,\n",
      "                           'template': '[c:2]:[cH;D2;+0:1]:[c:3]>>Br-[c;H0;D3;+0:1](:[c:2]):[c:3]',\n",
      "                           'template_idx': 685,\n",
      "                           'template_library_occurence': 433,\n",
      "                           'template_rank': 34})\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# Examine some sample reactions\n",
    "# Notice that the model includes a lot of metadata in the reaction objects,\n",
    "# not just the products and reactants.\n",
    "# Here, the metadata is the template used, the rank of the template and its softmax value,\n",
    "# and the number of times the template occurred in the training data\n",
    "for (i, j) in [(0, 0), (0, 10)]:\n",
    "    print(f\"Reaction #{j+1} for mol #{i+1}:\")\n",
    "    pprint.pprint(rxn_model_output[i][j])\n",
    "    print()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "6b95208e-ce3d-430d-a3d1-42867c345414",
   "metadata": {},
   "source": [
    "#### 1c: load list of target molecules for search"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "a27056ca-79cf-4ce1-a618-1a7310e50926",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "['COc1ccc(F)c(-c2ccc(COc3cc(C(CC(=O)O)C4CC4)ccc3F)nc2CC(C)(C)C)c1',\n",
       " 'CCc1cc2nncc(N3CCc4[nH]nc(C(=O)NC5CC5)c4C3)c2cc1OC',\n",
       " 'Clc1ccc(-n2cncn2)c(Cc2nc3c(NCCC4CCCCN4)nccc3o2)c1',\n",
       " 'CCOCc1nc2c(N)nc3ccc(O)cc3c2n1CC(C)(C)NS(C)(=O)=O',\n",
       " 'O=C(O)CCNC(=O)c1nc(-c2ccncc2)c2c(cc(-c3ccccc3)c(=O)n2CC2CCCCC2)c1O']"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# List of target molecules to synthesize\n",
    "target_smiles_list = paroutes.get_target_smiles(n=PAROUTES_N)\n",
    "target_smiles_list[:5]"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3359f01d-ddd1-4ad9-857e-9b93c787c074",
   "metadata": {},
   "source": [
    "## Step 2: set up algorithms"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "f25c0a2a-9bbc-4692-a384-93eb219d3261",
   "metadata": {},
   "outputs": [],
   "source": [
    "RXN_MODEL_CALL_LIMIT = 100\n",
    "TIME_LIMIT_S = 300"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "df4f6506-0dcd-4296-b590-994ea69e9fa3",
   "metadata": {},
   "source": [
    "### 2a: Retro-star\n",
    "\n",
    "Retro-star is a best-first search algorithm for AND/OR trees\n",
    "which tries to find minimum-cost routes.\n",
    "It requires 3 \"value functions\" to be provided to the algorithm.\n",
    "\n",
    "1. A function to give the cost of each molecule (OrNode)\n",
    "2. A function to give the cost of each reaction (AndNode)\n",
    "3. A function to estimate the minimum cost of leaf OrNodes (called the \"reaction number\" in the paper). This is effectively a value function in RL (except lower values are better)\n",
    "\n",
    "We will implement these using subclasses of `BaseNodeEvaluator` class from `syntheseus.search.node_evaluation`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "29e52211-0eac-4467-92cd-b6bc3fe67fdf",
   "metadata": {},
   "outputs": [],
   "source": [
    "from syntheseus.search.algorithms.best_first import retro_star\n",
    "from syntheseus.search.graph.and_or import AndNode, OrNode, AndOrGraph\n",
    "from syntheseus.search.node_evaluation import BaseNodeEvaluator"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "c891cc66-016e-456b-ada5-5f9f980e07a2",
   "metadata": {},
   "outputs": [],
   "source": [
    "#1: OrNode cost function.\n",
    "# We will follow the paper and give molecules a cost of 0 if they are purchasable,\n",
    "# and a cost of infinity otherwise. This class is provided as a default in retro_star.\n",
    "# If purchasable molecules have non-zero costs then a different cost function could be used\n",
    "\n",
    "or_node_cost_fn = retro_star.MolIsPurchasableCost()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "b6e6d682-756a-4dcd-931b-097b27ca0ddd",
   "metadata": {},
   "outputs": [],
   "source": [
    "#2: AndNode cost function\n",
    "# We will follow the original paper and define the cost of the reaction as the -log(softmax)\n",
    "# of the reaction model output, thresholded at a minimum value\n",
    "# This cost function is very specific to reactions from this particular reaction model though.\n",
    "\n",
    "class NegLogSoftmaxRxnCost(BaseNodeEvaluator[AndNode]):\n",
    "    def __call__(self, nodes: list[AndNode], graph: AndOrGraph=None) -> list[float]:\n",
    "        softmax_vals = [\n",
    "            node.reaction.metadata[\"softmax\"]\n",
    "            for node in nodes\n",
    "        ]\n",
    "        return [-math.log(max(val, 1e-4)) for val in softmax_vals]\n",
    "\n",
    "and_node_cost_fn = NegLogSoftmaxRxnCost()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "ca316b02-b56f-41f8-b1dd-c814f8ad265b",
   "metadata": {},
   "outputs": [],
   "source": [
    "#3: value function\n",
    "# Here we just use a constant value function which is always 0,\n",
    "# corresponding to the \"retro*-0\" algorithm (the most optimistic)\n",
    "\n",
    "from syntheseus.search.node_evaluation.common import ConstantNodeEvaluator\n",
    "\n",
    "retro_star_value_function = ConstantNodeEvaluator(0.0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "578c26a3-340a-478e-9bfd-30528e28dfba",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Set up algorithm\n",
    "retro_star_alg = retro_star.RetroStarSearch(\n",
    "    reaction_model=rxn_model,\n",
    "    mol_inventory=inventory,\n",
    "    or_node_cost_fn=or_node_cost_fn,\n",
    "    and_node_cost_fn=and_node_cost_fn,\n",
    "    value_function=retro_star_value_function,\n",
    "    limit_reaction_model_calls=RXN_MODEL_CALL_LIMIT,\n",
    "    time_limit_s=TIME_LIMIT_S,\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "628e9336-3263-4dc3-991b-f62b2c1e6ac4",
   "metadata": {},
   "source": [
    "### 2b: MCTS\n",
    "\n",
    "MCTS is an RL algorithm that tries to find routes of maximum value.\n",
    "Standard MCTS works on an MDP where states are sets of molecules\n",
    "and actions are backward reactions.\n",
    "MCTS requires 2 input functions, with an optional third.\n",
    "\n",
    "1. A _reward function_ which defines the rewards for terminal states\n",
    "2. A _value function_ which estimates the rewards attainable from a state\n",
    "3. (optional) a policy which assigns higher numbers to desirable actions\n",
    "\n",
    "We will implement these using subclasses of `BaseNodeEvaluator` class from `syntheseus.search.node_evaluation`."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "af997a32-1f39-4788-b88a-463d92b7c9c6",
   "metadata": {},
   "outputs": [],
   "source": [
    "from syntheseus.search.graph.molset import MolSetNode, MolSetGraph"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "aa02eecf-71c9-4f17-bc21-5df514e7d4a7",
   "metadata": {},
   "outputs": [],
   "source": [
    "#1: reward function\n",
    "# Here we use the most basic reward: 1.0 if a state is solved (all molecules are purchasable),\n",
    "# and 0.0 otherwise.\n",
    "from syntheseus.search.node_evaluation.common import HasSolutionValueFunction\n",
    "\n",
    "mcts_reward_function = HasSolutionValueFunction()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "5bfd9f29-58af-4dc5-b760-20ee65979777",
   "metadata": {},
   "outputs": [],
   "source": [
    "#2: value function\n",
    "# while previous works have used rollouts,\n",
    "# for demonstration purposes we just use a constant value function of 0.5\n",
    "# (this number is chosen to be between the lowest and highest rewards of 0/1\n",
    "\n",
    "mcts_value_function = ConstantNodeEvaluator(0.5)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "9db78293-5fbc-4d8f-9c3c-dfb3e3b3d3b5",
   "metadata": {},
   "outputs": [],
   "source": [
    "#3: policy\n",
    "# this could be anything, and would often be another neural network in RL.\n",
    "# However, for this example we just use the softmax values from the reaction model,\n",
    "# normalized by the total number of reactions\n",
    "# (note: this implementation will always call the policy on all of a node's children simultaneously)\n",
    "\n",
    "class MCTS_Reaction_Softmax_Policy(BaseNodeEvaluator[MolSetNode]):\n",
    "    def __call__(self, nodes: list[MolSetNode], graph: MolSetGraph=None) -> list[float]:\n",
    "        \n",
    "        # This policy requires accessing the underlying graph to get the reaction\n",
    "        assert graph is not None\n",
    "        non_normalized_softmaxes: list[float] = []\n",
    "        for node in nodes:\n",
    "            parent_nodes = list(graph.predecessors(node))\n",
    "            assert len(parent_nodes) == 1\n",
    "            parent_rxn: BackwardReaction = graph._graph.edges[parent_nodes[0], node][\"reaction\"]\n",
    "            non_normalized_softmaxes.append(parent_rxn.metadata[\"softmax\"])\n",
    "        \n",
    "        # Normalize the results\n",
    "        normalization_constant = sum(non_normalized_softmaxes)\n",
    "        return [v / normalization_constant for v in non_normalized_softmaxes]\n",
    "\n",
    "mcts_policy = MCTS_Reaction_Softmax_Policy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "4321240d-a248-4acb-9b28-706ef5bace14",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Set up algorithm\n",
    "from syntheseus.search.algorithms.mcts.base import pucb_bound\n",
    "from syntheseus.search.algorithms.mcts.molset import MolSetMCTS\n",
    "mcts_alg = MolSetMCTS(\n",
    "    reaction_model=rxn_model,\n",
    "    mol_inventory=inventory,\n",
    "    reward_function=mcts_reward_function,\n",
    "    value_function=mcts_value_function,\n",
    "    policy=mcts_policy,\n",
    "    limit_reaction_model_calls=RXN_MODEL_CALL_LIMIT,\n",
    "    bound_constant=100.0,\n",
    "    bound_function=pucb_bound,\n",
    "    time_limit_s=TIME_LIMIT_S,\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "933ae569-8e5a-4109-876b-d4ebd18836eb",
   "metadata": {},
   "source": [
    "## Step 3: run algorithms and save results\n",
    "\n",
    "The main result needed for analysis is the output graph.\n",
    "We will save this as a pickle file and analyze this in the next notebook."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "656bb71b-fabe-4b34-99cc-b0d9c43e65e9",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<rdkit.Chem.rdchem.Mol at 0x7f3724621850>"
      ]
     },
     "execution_count": 21,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "target_mol = Molecule(target_smiles_list[100])\n",
    "target_mol.rdkit_mol"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "5af33f0f-daad-49ea-8d36-2d898fec07d6",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CPU times: user 14 s, sys: 83.4 ms, total: 14.1 s\n",
      "Wall time: 12.6 s\n"
     ]
    }
   ],
   "source": [
    "%%time\n",
    "# Run retro-star\n",
    "retro_star_alg.reset()\n",
    "retro_star_search_graph, _ = retro_star_alg.run_from_mol(target_mol)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "d6af0d40-c7fb-48d8-b63c-42762e46976d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CPU times: user 14.7 s, sys: 168 ms, total: 14.9 s\n",
      "Wall time: 13.4 s\n"
     ]
    }
   ],
   "source": [
    "%%time\n",
    "# Run MCTS\n",
    "mcts_alg.reset()\n",
    "mcts_search_graph, _ = mcts_alg.run_from_mol(target_mol)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "2907cca5-c96c-473b-b257-c5f333685b95",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "retro star graph size: 17108\n",
      "mcts graph size: 13380\n"
     ]
    }
   ],
   "source": [
    "# Save graphs\n",
    "import pickle\n",
    "for graph, alg_name in [(retro_star_search_graph, \"retro star\"), (mcts_search_graph, \"mcts\")]:\n",
    "    print(f\"{alg_name} graph size: {len(graph)}\")\n",
    "    with open(f\"search-results-{alg_name}.pkl\", \"wb\") as f:\n",
    "        pickle.dump(graph, f)"
   ]
  }
 ],
 "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.10.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
