{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import csv\n",
    "import dill\n",
    "import torch\n",
    "import numpy as np\n",
    "from tqdm import tqdm\n",
    "from tabulate import tabulate\n",
    "from torch.utils.data import DataLoader\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from datasets.MD17.MD17Dataset import MD17SingleDataset\n",
    "from datasets.RMD17.RMD17Dataset import RMD17SingleDataset\n",
    "from datasets.SMD17.SMD17Dataset import SMD17SingleDataset\n",
    "from scripts.Chemistry.losses import EnergyLoss, PosForceLoss\n",
    "from scripts.commom_util import *"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_configs(model_path):\n",
    "    configs = dict()\n",
    "    with open(f\"{model_path}/info.txt\") as fp:\n",
    "        for line in fp:\n",
    "            line = line.strip().split(\" \")\n",
    "            if len(line)>1:\n",
    "                configs[line[0]] = line[1]\n",
    "\n",
    "    return configs\n",
    "\n",
    "def predict(model_path, epoch, configs, device, root):\n",
    "    # initialize\n",
    "    if configs[\"dataset\"]==\"MD17SingleDataset\":\n",
    "        dataset = MD17SingleDataset(configs[\"style\"], configs[\"molecule\"], \"test\", configs[\"split\"], root)\n",
    "    elif configs[\"dataset\"]==\"SMD17SingleDataset\":\n",
    "        dataset = SMD17SingleDataset(configs[\"style\"], configs[\"molecule\"], \"test\", configs[\"split\"], root)\n",
    "    else:\n",
    "        dataset = RMD17SingleDataset(configs[\"style\"], configs[\"molecule\"], \"test\", configs[\"split\"], root)\n",
    "    identifier = dataset.identifier\n",
    "    test_dataloader = DataLoader(dataset, batch_size=1, shuffle=False, collate_fn=dataset.collate)\n",
    "    \n",
    "    path = glob.glob(f\"{model_path}/{epoch:03d}_*.pth\")\n",
    "    if not path: raise ValueError(\"model not found\")\n",
    "    model = torch.load(path[0], map_location=torch.device(torch.cuda.current_device()), pickle_module=dill)\n",
    "    print(f\"Using model {path[0]}\")\n",
    "    model.eval()\n",
    "    \n",
    "    # test\n",
    "    preds = []\n",
    "    losses = []\n",
    "    tq = tqdm(test_dataloader)\n",
    "    for data, label in tq:\n",
    "        data = {i:v.to(device) for i, v in data.items()}\n",
    "        label = {i:v.to(device) for i, v in label.items()}\n",
    "        pred = model(data)\n",
    "        preds.append( ((torch.cat((pred[\"E\"], pred[\"F\"].reshape(1, -1)), axis=1)).squeeze()).tolist() )\n",
    "     \n",
    "        loss_E = (EnergyLoss(pred, label).to(\"cpu\").item())\n",
    "        loss_F = PosForceLoss(pred, label)\n",
    "        loss_F = [ (l.to(\"cpu\").item()) for l in loss_F]\n",
    "        losses.append([loss_E]+loss_F)\n",
    "    \n",
    "    # save result\n",
    "    save_reult(model_path, identifier, preds, losses)\n",
    "\n",
    "def metric(model_path, configs, if_print=False):\n",
    "    # Every dataset has different unit, unify them to Energy:eV and Force: eV/A\n",
    "    toEv = {\"RMD17SingleDataset\":0.0433634, \"MD17SingleDataset\":0.0433634, \"SMD17SingleDataset\":0.0433634}\n",
    "    ds = configs[\"dataset\"]\n",
    "    scale = toEv[configs[\"dataset\"]]\n",
    "\n",
    "    # load and calculate\n",
    "    style, molecule, split = configs[\"style\"], configs[\"molecule\"], configs[\"split\"]\n",
    "    with open(f\"{model_path}/loss_{ds}_{style}_{molecule}{split}test.csv\", newline='') as fp:\n",
    "        cdata = list(csv.reader(fp, quoting=csv.QUOTE_NONNUMERIC))\n",
    "        cdata = [[c for c in row] for row in cdata]\n",
    "        loss_Emole , loss_Fmole= [], []\n",
    "        hit, total = 0, len(cdata)\n",
    "        failE, failF = 0, 0\n",
    "\n",
    "        #for row in tqdm(cdata):\n",
    "        for row in cdata:\n",
    "            loss_Emole.append(row[0]) \n",
    "            loss_Fmole.append(row[1:])\n",
    "            m = max(row[1:])\n",
    "            if row[0]<=0.02/scale and m<=0.03/scale: hit += 1\n",
    "            else:\n",
    "                if row[0]>0.02/scale: failE += 1\n",
    "                if m>0.03/scale: failF += 1\n",
    "        loss_Emole , loss_Fmole= np.array(loss_Emole), np.array(loss_Fmole)\n",
    "\n",
    "        EMAE = np.mean(loss_Emole)\n",
    "        FMAE = np.mean(loss_Fmole)\n",
    "        EFWT = hit/total\n",
    "        if if_print:\n",
    "            print(f\"Energy MAE: {EMAE:.3f},\\tForce MAE: {FMAE:.3f},\\tEFwT: {EFWT:.3f} (failE:{(failE/total):.3f}, failF:{(failF/total):.3f})\\n\")\n",
    "        return EMAE, FMAE, EFWT"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Energy Loss\n",
      "+----+---------+----------+-----------+----------+----------+----------+-----------+----------+\n",
      "|    |       a |        b |         e |        m |        n |        s |         t |        u |\n",
      "+====+=========+==========+===========+==========+==========+==========+===========+==========+\n",
      "| og | 816.046 | 1000.86  | 496.824   | 217.765  | 870.55   | 147.071  | 1903.14   | 456.211  |\n",
      "+----+---------+----------+-----------+----------+----------+----------+-----------+----------+\n",
      "| IR | 137.125 |   16.634 |   4.63941 |  30.5689 |  66.6814 |  78.2999 |   36.7364 |  81.0208 |\n",
      "+----+---------+----------+-----------+----------+----------+----------+-----------+----------+\n",
      "\n",
      "Force Loss\n",
      "+----+---------+----------+----------+----------+----------+---------+----------+---------+\n",
      "|    |       a |        b |        e |        m |        n |       s |        t |       u |\n",
      "+====+=========+==========+==========+==========+==========+=========+==========+=========+\n",
      "| og | 1.27187 | 0.356377 | 0.617618 | 0.738791 | 0.807172 | 1.01507 | 0.841261 | 1.12929 |\n",
      "+----+---------+----------+----------+----------+----------+---------+----------+---------+\n",
      "| IR | 7.13277 | 0.615958 | 2.96911  | 4.9974   | 2.38903  | 4.77678 | 2.91234  | 4.9343  |\n",
      "+----+---------+----------+----------+----------+----------+---------+----------+---------+\n"
     ]
    }
   ],
   "source": [
    "# Ablation test of IReLU (when denormalization, label rescaling applied)\n",
    "# Model: CGCNN\n",
    "# Dataset: MD17\n",
    "\n",
    "need_predict = False\n",
    "EMAE_table = [[\"a\", \"b\", \"e\", \"m\", \"n\", \"s\", \"t\", \"u\"]]\n",
    "FMAE_table = [[\"a\", \"b\", \"e\", \"m\", \"n\", \"s\", \"t\", \"u\"]]\n",
    "EFWT_table = [[\"a\", \"b\", \"e\", \"m\", \"n\", \"s\", \"t\", \"u\"]]\n",
    "\n",
    "for act in [\"og\", \"IR\"]:\n",
    "    EMAE_row = [act]\n",
    "    FMAE_row = [act]\n",
    "    EFWT_row = [act]\n",
    "    \n",
    "    for mole in [\"a\", \"b\", \"e\", \"m\", \"n\", \"s\", \"t\", \"u\"]:\n",
    "        #print(f\"act:{act}, molecule:{mole}\")\n",
    "        model_path = f\"./checkpoints/cgcnnnobn_smd17_{act}_{mole}\"\n",
    "        cfg = get_configs(model_path)\n",
    "\n",
    "        if need_predict:\n",
    "            root = \"../../datasets/SMD17/datas\"\n",
    "            predict(model_path, 49, cfg, \"cuda\", root)\n",
    "        EMAE, FMAE, EFWT = metric(model_path, cfg, if_print=False)\n",
    "        EMAE_row.append(EMAE)\n",
    "        FMAE_row.append(FMAE)\n",
    "        EFWT_row.append(EFWT)\n",
    "    \n",
    "    EMAE_table.append(EMAE_row)\n",
    "    FMAE_table.append(FMAE_row)\n",
    "    EFWT_table.append(EFWT_row)\n",
    "\n",
    "tableE = tabulate(EMAE_table, headers='firstrow', tablefmt='grid')\n",
    "tableF = tabulate(FMAE_table, headers='firstrow', tablefmt='grid')\n",
    "\n",
    "print(\"Energy Loss\")\n",
    "print(tableE)\n",
    "print(\"\")\n",
    "print(\"Force Loss\")\n",
    "print(tableF)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "ocp",
   "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.13"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
