{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "2ad058da",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "2.8.0+cu128\n",
      "cuda\n"
     ]
    }
   ],
   "source": [
    "import math\n",
    "import scipy.io\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "import copy\n",
    "import datetime\n",
    "import numpy as np\n",
    "import random\n",
    "import json\n",
    "import os\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "from sklearn.model_selection import train_test_split\n",
    "from torch.nn import functional as F\n",
    "\n",
    "print(torch.__version__)\n",
    "\n",
    "if torch.cuda.is_available():\n",
    "    device = torch.device('cuda')\n",
    "else : \n",
    "    device = torch.device('cpu')\n",
    "print(device)\n",
    "\n",
    "# self defined package\n",
    "import dataset.aug_index as ai # data augmentation and some index for reconstruction\n",
    "import NSC_module.modules_encoder as enc\n",
    "import NSC_module.modules_decoder as dec\n",
    "import NSC_module.custom_loss as c_loss # not used in single ablation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "c04aa8c2",
   "metadata": {},
   "outputs": [],
   "source": [
    "from lightning.pytorch import Trainer, seed_everything # used for seed setting"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "b39a4c76",
   "metadata": {},
   "outputs": [],
   "source": [
    "# X = np.load('./dataset/D1N2_wave.npy')\n",
    "# y = np.load('./dataset/D1N2_neo.npy')\n",
    "# ds = \"D1N2\"\n",
    "\n",
    "# X = np.load('./dataset/D2N2_wave.npy')\n",
    "# y = np.load('./dataset/D2N2_neo.npy')\n",
    "# ds = \"D2N2\"\n",
    "\n",
    "X = np.load('./dataset/NP_T1_wave.npy')\n",
    "y = np.load('./dataset/NP_T1_neo.npy')\n",
    "ds = \"NPT1\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "e46e0de2",
   "metadata": {},
   "outputs": [],
   "source": [
    "class NEAC(nn.Module):\n",
    "    def __init__(self, latent_dim=4, num_windows=3, require_lr_window=True, require_quant=True):\n",
    "        super().__init__()\n",
    "        # Abalation\n",
    "        self.encoder = enc.NeuralSignalCodec(latent_dim=latent_dim, num_windows=num_windows, required_quant=require_quant, required_lr_window=require_lr_window)\n",
    "        self.decoder = dec.Base_decoder(latent_dim=latent_dim)\n",
    "\n",
    "    def forward(self, x, x_neo, quant_force=None):\n",
    "        z = self.encoder(x, x_neo, quant_force)\n",
    "        x_recon = self.decoder(z.float())\n",
    "        return z, x_recon"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "e6e4575d",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Seed set to 1\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "seed  1\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "💡 Tip: For seamless cloud uploads and versioning, try installing [litmodels](https://pypi.org/project/litmodels/) to enable LitModelCheckpoint, which syncs automatically with the Lightning model registry.\n",
      "GPU available: True (cuda), used: True\n",
      "TPU available: False, using: 0 TPU cores\n",
      "HPU available: False, using: 0 HPUs\n",
      "Seed set to 2\n",
      "💡 Tip: For seamless cloud uploads and versioning, try installing [litmodels](https://pypi.org/project/litmodels/) to enable LitModelCheckpoint, which syncs automatically with the Lightning model registry.\n",
      "GPU available: True (cuda), used: True\n",
      "TPU available: False, using: 0 TPU cores\n",
      "HPU available: False, using: 0 HPUs\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "seed  2\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Seed set to 3\n",
      "💡 Tip: For seamless cloud uploads and versioning, try installing [litmodels](https://pypi.org/project/litmodels/) to enable LitModelCheckpoint, which syncs automatically with the Lightning model registry.\n",
      "GPU available: True (cuda), used: True\n",
      "TPU available: False, using: 0 TPU cores\n",
      "HPU available: False, using: 0 HPUs\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "seed  3\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Seed set to 4\n",
      "💡 Tip: For seamless cloud uploads and versioning, try installing [litmodels](https://pypi.org/project/litmodels/) to enable LitModelCheckpoint, which syncs automatically with the Lightning model registry.\n",
      "GPU available: True (cuda), used: True\n",
      "TPU available: False, using: 0 TPU cores\n",
      "HPU available: False, using: 0 HPUs\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "seed  4\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Seed set to 5\n",
      "💡 Tip: For seamless cloud uploads and versioning, try installing [litmodels](https://pypi.org/project/litmodels/) to enable LitModelCheckpoint, which syncs automatically with the Lightning model registry.\n",
      "GPU available: True (cuda), used: True\n",
      "TPU available: False, using: 0 TPU cores\n",
      "HPU available: False, using: 0 HPUs\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "seed  5\n",
      "PSNR :  17.654028 / 17.759264\n",
      "R2 :  0.05201149 / 0.5122776\n",
      "NRMSE :  0.14540799 / 0.14097025\n",
      "SNDR :  2.4737382 / 4.489254\n"
     ]
    }
   ],
   "source": [
    "# training part\n",
    "\n",
    "cr = 32 \n",
    "seed_list = [1, 2, 3, 4, 5]\n",
    "\n",
    "results = {}\n",
    "\n",
    "train_epochs = 100\n",
    "\n",
    "for cho in range(99, 100):\n",
    "    choice = cho\n",
    "    results[f'{choice}'] = {}\n",
    "\n",
    "    psnr_valt = []\n",
    "    r2_valt = []\n",
    "    sndr_valt = []\n",
    "    nrmse_valt = []\n",
    "    psnr_mid_valt = []\n",
    "    r2_mid_valt = []\n",
    "    sndr_mid_valt = []\n",
    "    nrmse_mid_valt = []\n",
    "\n",
    "    for se in range (len(seed_list)):\n",
    "        psnr_val = []\n",
    "        r2_val = []\n",
    "        sndr_val = []\n",
    "        nrmse_val = []\n",
    "        psnr_mid_val = []\n",
    "        r2_mid_val = []\n",
    "        sndr_mid_val = []\n",
    "        nrmse_mid_val = []\n",
    "        \n",
    "        seed = seed_list[se]\n",
    "        print(\"seed \", seed)\n",
    "        seed_everything(seed, workers=True)\n",
    "\n",
    "        model = NEAC(latent_dim=int(128/cr), num_windows = 3, require_lr_window = True, require_quant = False).to(device)\n",
    "\n",
    "        trainer = Trainer(deterministic=True)\n",
    "        \n",
    "        optimizer = optim.AdamW([\n",
    "            {'params': [model.encoder.alpha, model.encoder.beta, model.encoder.gamma], 'lr': 1e-3},  \n",
    "            {'params': model.encoder.boundary_params, 'lr': 1e-3, 'weight_decay': 1e-4},\n",
    "            {'params': model.decoder.parameters(), 'lr': 1e-3, 'weight_decay': 1e-4},\n",
    "        ])\n",
    "\n",
    "        # Datasets\n",
    "        bs = 64\n",
    "\n",
    "        # class not involve in training, here we hold for valid incase the other demand\n",
    "        X_tmp, X_test, y_tmp, y_test = train_test_split(X, y, test_size=0.2)\n",
    "        X_train, X_valid, y_train, y_valid = train_test_split(X_tmp, y_tmp, test_size=0.125)\n",
    "        \n",
    "        X_train = np.array(X_train)\n",
    "\n",
    "        X_train = torch.tensor(X_train, dtype=torch.float32)\n",
    "        X_valid = torch.tensor(X_valid, dtype=torch.float32)\n",
    "        X_test  = torch.tensor(X_test , dtype=torch.float32)\n",
    "\n",
    "        y_train = torch.tensor(y_train, dtype=torch.float32)\n",
    "        y_valid = torch.tensor(y_valid, dtype=torch.float32)\n",
    "        y_test  = torch.tensor(y_test , dtype=torch.float32)\n",
    "\n",
    "        train_dataset = TensorDataset(X_train, y_train)\n",
    "        valid_dataset = TensorDataset(X_valid, y_valid)\n",
    "        test_dataset  = TensorDataset(X_test, y_test)\n",
    "\n",
    "        train_loader = DataLoader(train_dataset, batch_size=bs, shuffle=True)\n",
    "        valid_loader = DataLoader(valid_dataset, batch_size=bs, shuffle=False)\n",
    "        test_loader = DataLoader(test_dataset, batch_size=bs, shuffle=False)\n",
    "\n",
    "        best_psnr = -np.inf\n",
    "\n",
    "        sr = 20000\n",
    "\n",
    "        for epoch in range(train_epochs):\n",
    "            model.train()\n",
    "            running_loss = 0.0\n",
    "            for (inputs, inputs_neo) in train_loader:\n",
    "                inputs = inputs.to(device)\n",
    "                inputs_neo = inputs_neo.to(device)\n",
    "\n",
    "                optimizer.zero_grad()\n",
    "                _, outputs = model(inputs, inputs_neo)\n",
    "\n",
    "                loss = c_loss.window_energy_dist_loss(outputs, inputs, inputs_neo, 'gaussian', 'mse', sampling_rate=sr)\n",
    "                \n",
    "                loss.backward()\n",
    "                clip_norm = 10\n",
    "                total_norm = torch.nn.utils.clip_grad_norm_(model.parameters(), max_norm=clip_norm)\n",
    "                #print(f\"Gradient norm before clipping: {total_norm.item()}\")\n",
    "                optimizer.step()\n",
    "                running_loss += loss.item()\n",
    "\n",
    "            #print(running_loss/len(train_loader))\n",
    "\n",
    "            model.eval()\n",
    "            all_recons = []\n",
    "            all_targets = []\n",
    "\n",
    "            with torch.no_grad():\n",
    "                for (inputs, inputs_neo) in valid_loader:\n",
    "                    inputs = inputs.to(device)\n",
    "                    inputs_neo = inputs_neo.to(device)\n",
    "\n",
    "                    _, outputs = model(inputs, inputs_neo)\n",
    "                    for i in inputs:\n",
    "                        all_targets.append(i.cpu().numpy())\n",
    "                    for o in outputs:\n",
    "                        all_recons.append(o.cpu().numpy())\n",
    "\n",
    "            tmp_psnr_mid = 0\n",
    "            for i in range (len(all_recons)):\n",
    "                tmp_psnr_mid += ai.psnr(all_recons[i][32:96], all_targets[i][32:96])\n",
    "            \n",
    "            psnr_mid_valv = tmp_psnr_mid / len(all_recons)\n",
    "            if psnr_mid_valv > best_psnr:\n",
    "                best_psnr = psnr_mid_valv\n",
    "                best_model_state = copy.deepcopy(model.state_dict())\n",
    "\n",
    "        save_name = \"./ablation_model/\" + ds + 'ablation_choice_' + str(choice) + '_' + str(seed) + '.pt'\n",
    "        torch.save(best_model_state, save_name)\n",
    "\n",
    "        o_list = []\n",
    "        i_list = []\n",
    "        z_list = []\n",
    "\n",
    "        best_model_state = torch.load(save_name)\n",
    "        model.load_state_dict(best_model_state)\n",
    "        model.eval()\n",
    "\n",
    "        if choice == 3:  # PTQ\n",
    "            with torch.no_grad():\n",
    "                boundaries, alpha_q, beta_q, gamma_q = model.encoder.get_quantized_params_for_ptq()\n",
    "                model.encoder.alpha.copy_(alpha_q)\n",
    "                model.encoder.beta.copy_(beta_q)\n",
    "                model.encoder.gamma.copy_(gamma_q)\n",
    "\n",
    "        with torch.no_grad():\n",
    "            for (inputs, inputs_neo) in test_loader:\n",
    "                inputs = inputs.to(device)\n",
    "                inputs_neo = inputs_neo.to(device)\n",
    "\n",
    "                z, outputs = model(inputs, inputs_neo)\n",
    "                for i in inputs:\n",
    "                    i_list.append(i.cpu().numpy())\n",
    "                for o in outputs:\n",
    "                    o_list.append(o.cpu().numpy())\n",
    "                for z in z:\n",
    "                    z_list.append(z.cpu().numpy())\n",
    "\n",
    "        np.save(\"./ablation_value/\" + ds + \"/\" + str(choice) + str(se) + \"abl_target.npy\", np.array(i_list))\n",
    "        np.save(\"./ablation_value/\" + ds + \"/\" + str(choice) + str(se) + \"abl_recon.npy\", np.array(o_list))\n",
    "        np.save(\"./ablation_value/\" + ds + \"/\" + str(choice) + str(se) + \"abl_latent.npy\", np.array(z_list))\n",
    "\n",
    "        for i in range(len(i_list)):\n",
    "            psnr_val.append(ai.psnr(o_list[i], i_list[i]))\n",
    "            r2_val.append(ai.r2_score(o_list[i], i_list[i]))\n",
    "            sndr_val.append(ai.sndr(o_list[i], i_list[i]))\n",
    "            nrmse_val.append(ai.nrmse(o_list[i], i_list[i]))\n",
    "            \n",
    "            psnr_mid_val.append(ai.psnr(o_list[i][32:96], i_list[i][32:96]))\n",
    "            r2_mid_val.append(ai.r2_score(o_list[i][32:96], i_list[i][32:96]))\n",
    "            sndr_mid_val.append(ai.sndr(o_list[i][32:96], i_list[i][32:96]))\n",
    "            nrmse_mid_val.append(ai.nrmse(o_list[i][32:96], i_list[i][32:96]))\n",
    "        \n",
    "        psnr_val = np.array(psnr_val)\n",
    "        r2_val = np.array(r2_val)\n",
    "        sndr_val = np.array(sndr_val)\n",
    "        nrmse_val = np.array(nrmse_val)\n",
    "        psnr_mid_val = np.array(psnr_mid_val)\n",
    "        r2_mid_val = np.array(r2_mid_val)\n",
    "        sndr_mid_val = np.array(sndr_mid_val)\n",
    "        nrmse_mid_val = np.array(nrmse_mid_val)\n",
    "\n",
    "        psnr_valt.append(psnr_val.mean())\n",
    "        r2_valt.append(r2_val.mean())\n",
    "        sndr_valt.append(sndr_val.mean())\n",
    "        nrmse_valt.append(nrmse_val.mean())\n",
    "        psnr_mid_valt.append(psnr_mid_val.mean())\n",
    "        r2_mid_valt.append(r2_mid_val.mean())\n",
    "        sndr_mid_valt.append(sndr_mid_val.mean())\n",
    "        nrmse_mid_valt.append(nrmse_mid_val.mean())\n",
    "\n",
    "        # print(psnr_val / len(i_list))\n",
    "        # print(r2_val / len(i_list))\n",
    "        # print(sndr_val/ len(i_list))\n",
    "        # print(nrmse_val/ len(i_list))\n",
    "        # print(psnr_mid_val/ len(i_list))\n",
    "        # print(r2_mid_val/ len(i_list))\n",
    "        # print(sndr_mid_val/ len(i_list))\n",
    "        # print(nrmse_mid_val/ len(i_list))\n",
    "\n",
    "    psnr_valt = np.array(psnr_valt)\n",
    "    r2_valt = np.array(r2_valt)\n",
    "    sndr_valt = np.array(sndr_valt)\n",
    "    nrmse_valt = np.array(nrmse_valt)\n",
    "    psnr_mid_valt = np.array(psnr_mid_valt)\n",
    "    r2_mid_valt = np.array(r2_mid_valt)\n",
    "    sndr_mid_valt = np.array(sndr_mid_valt)\n",
    "    nrmse_mid_valt = np.array(nrmse_mid_valt)\n",
    "\n",
    "    results[f'{choice}']['psnr_mean'] = psnr_valt.mean()\n",
    "    results[f'{choice}']['r2_mean'] = r2_valt.mean()\n",
    "    results[f'{choice}']['sndr_mean'] = sndr_valt.mean()\n",
    "    results[f'{choice}']['nrmse_mean'] = nrmse_valt.mean()\n",
    "    results[f'{choice}']['psnr_mid_mean'] = psnr_mid_valt.mean()\n",
    "    results[f'{choice}']['r2_mid_mean'] = r2_mid_valt.mean()\n",
    "    results[f'{choice}']['sndr_mid_mean'] = sndr_mid_valt.mean()\n",
    "    results[f'{choice}']['nrmse_mid_mean'] = nrmse_mid_valt.mean()\n",
    "    \n",
    "    results[f'{choice}']['psnr_std'] = psnr_valt.std()\n",
    "    results[f'{choice}']['r2_std'] = r2_valt.std()\n",
    "    results[f'{choice}']['sndr_std'] = sndr_valt.std()\n",
    "    results[f'{choice}']['nrmse_std'] = nrmse_valt.std()\n",
    "    results[f'{choice}']['psnr_mid_std'] = psnr_mid_valt.std()\n",
    "    results[f'{choice}']['r2_mid_std'] = r2_mid_valt.std()\n",
    "    results[f'{choice}']['sndr_mid_std'] = sndr_mid_valt.std()\n",
    "    results[f'{choice}']['nrmse_mid_std'] = nrmse_mid_valt.std()\n",
    "\n",
    "    results[f'{choice}']['psnr_best'] = psnr_valt.max()\n",
    "    results[f'{choice}']['r2_best'] = r2_valt.max()\n",
    "    results[f'{choice}']['sndr_best'] = sndr_valt.max()\n",
    "    results[f'{choice}']['nrmse_best'] = nrmse_valt.min()\n",
    "    results[f'{choice}']['psnr_mid_best'] = psnr_mid_valt.max()\n",
    "    results[f'{choice}']['r2_mid_best'] = r2_mid_valt.max()\n",
    "    results[f'{choice}']['sndr_mid_best'] = sndr_mid_valt.max()\n",
    "    results[f'{choice}']['nrmse_mid_best'] = nrmse_mid_valt.min()\n",
    "\n",
    "    print(\"PSNR : \", psnr_valt.mean(), '/', psnr_mid_valt.mean())\n",
    "    print(\"R2 : \", r2_valt.mean(), '/', r2_mid_valt.mean())\n",
    "    print(\"NRMSE : \", nrmse_valt.mean(), '/', nrmse_mid_valt.mean())\n",
    "    print(\"SNDR : \", sndr_valt.mean(), '/', sndr_mid_valt.mean())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "5e0798f9-0f5f-4f4d-a713-a1e6de54a52d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'99': {'psnr_mean': np.float32(17.654028), 'r2_mean': np.float32(0.05201149), 'sndr_mean': np.float32(2.4737382), 'nrmse_mean': np.float32(0.14540799), 'psnr_mid_mean': np.float32(17.759264), 'r2_mid_mean': np.float32(0.5122776), 'sndr_mid_mean': np.float32(4.489254), 'nrmse_mid_mean': np.float32(0.14097025), 'psnr_std': np.float32(0.19961308), 'r2_std': np.float32(0.14106344), 'sndr_std': np.float32(0.16199486), 'nrmse_std': np.float32(0.0044385637), 'psnr_mid_std': np.float32(0.17092647), 'r2_mid_std': np.float32(0.010041113), 'sndr_mid_std': np.float32(0.15773384), 'nrmse_mid_std': np.float32(0.0019357194), 'psnr_best': np.float32(17.916206), 'r2_best': np.float32(0.2523715), 'sndr_best': np.float32(2.7258723), 'nrmse_best': np.float32(0.13842702), 'psnr_mid_best': np.float32(17.991646), 'r2_mid_best': np.float32(0.53089225), 'sndr_mid_best': np.float32(4.715881), 'nrmse_mid_best': np.float32(0.13803056)}}\n"
     ]
    }
   ],
   "source": [
    "print(results)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "da359930-8a88-4eeb-9bff-a3d2739bf071",
   "metadata": {},
   "outputs": [],
   "source": [
    "with open(\"./ablation_result/abl_append_\" + ds + \".txt\", \"w\") as f:\n",
    "    f.write(str(results))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "2a7f70cd-d6d2-43ff-87f3-b36fa67e1e86",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.12.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
