{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "7a5ed717",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/wulab/.local/lib/python3.8/site-packages/pandas/core/computation/expressions.py:20: UserWarning: Pandas requires version '2.7.3' or newer of 'numexpr' (version '2.7.1' currently installed).\n",
      "  from pandas.core.computation.check import NUMEXPR_INSTALLED\n"
     ]
    }
   ],
   "source": [
    "# Import all necessary packages\n",
    "\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.integrate import solve_ivp\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "from sklearn.model_selection import train_test_split\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "import math\n",
    "import optuna\n",
    "from sklearn import preprocessing"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "4dd5d4ee",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'using cpu'"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Set up GPU accelerated training\n",
    "\n",
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "print('using GPU:', torch.cuda.get_device_name()) if torch.cuda.is_available() else 'using cpu'"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1804dcc6",
   "metadata": {},
   "source": [
    "# Test dataset generation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "ed431853",
   "metadata": {},
   "outputs": [],
   "source": [
    "# specifying constant parameters\n",
    "\n",
    "T_0 = 300\n",
    "V = 1\n",
    "k_0 = 8.46*(np.power(10,6))\n",
    "C_p = 0.231\n",
    "rho_L = 1000\n",
    "Q_s = 0.0\n",
    "T_s = 402\n",
    "F = 5\n",
    "E = 5*(np.power(10,4))\n",
    "delta_H = -1.15*(np.power(10,4))\n",
    "R = 8.314\n",
    "C_A0s = 4\n",
    "C_As = 1.95\n",
    "t_final = 0.01\n",
    "t_step = 1e-4\n",
    "P = np.array([[1060, 22], [22, 0.52]])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "8903fb7a",
   "metadata": {},
   "outputs": [],
   "source": [
    "# generating inputs and initial states for CSTR, all expressed in deviation form\n",
    "\n",
    "u1_list = np.linspace(-3.5, 3.5, 10, endpoint=True)\n",
    "u2_list = np.linspace(-5e5, 5e5, 10, endpoint=True)\n",
    "T_initial = np.linspace(300, 600, 40, endpoint=True) - T_s\n",
    "CA_initial = np.linspace(0, 6, 40, endpoint=True) - C_As"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "8f6f7972",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "number of initial conditions: 118\n",
      "shape of x_deviation is (118, 2)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_3737431/457810299.py:31: RuntimeWarning: invalid value encountered in sqrt\n",
      "  sqrt = np.sqrt(-2688000 * i**2 + 15772800000)\n"
     ]
    },
    {
     "data": {
      "image/png": "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AXxv6nJycgq9zQUXXMCAAQO83qc/qakTEZHikZoKiYngcnneU1OLrPQ///zDKaecQq9evWjUqBG9e/fGcxcsOOWUU0hLS2PAgAHs37+fVq1a0bt3b+Dfs3B79uzh9NNPp02bNjRv3pyffvrpuPsC2Lt3L+eddx4tW7akWbNmDB06lDfffJN169Zx6qmncuqppwJw6623kpycTNOmTXniiScA8l3vzz//pGPHjrRp04bLLruMPXv2ADBgwACaNGlCixYtuO+++47J0bx5c3bs2IG1lgoVKhw+g3XNNdfw119/8c8//3D++eeTnp7O+++/z2uvvUarVq0YN24cAGPHjqVTp07UqVPncCOXnp5Os2bNAE/TdMkll3D22WdTv359HnjggXyPx9E5J06cyPDhw7n//vtp1aoVy5cv58MPP6Rt27a0bNmSSy+99D9N7ciRI0lOTqZBgwaMGDHi8J/p+eeff8y+nnzySV5++WW+/fZb0tLS6N27N61ateKXX37hoosuOrzeX3/9xcUXX3zM9omJiTz44IO0adOGYcOGHffY//rrrzRq1IikpCTuvPPOw1mOPGuYnp7OaaedRosWLTj99NNZtWoV4Dn7eOeddx5zbP3OWhvSr6SkJCsiIkVvwYIFBV85JcVat9ta+PfldnuW+yAuLs5aa+3o0aNt6dKl7erVq21OTo7t0KGDHTdunLXW2u7du9tp06b9Z/2jtz948KDduXOntdbazZs327p169rc3Nx8t7HW2m+//dbeeOONh7/fsWOHtdbahIQEu3nz5sPLt27daq21Njs723bv3t3Onj37mPU2b95su3btavfs2WOttfb555+3Tz31lN2yZYtt0KDB4Rzbt28/JsfNN99sR4wYYefOnWuTk5MPZ6pXr57ds2ePHT16tD3vvPOstdY+8cQT9qWXXjq8bb9+/WyvXr1sTk6OnT9/vq1bt6611tqVK1fapk2bWmut/fTTT23t2rXtjh077P79+218fLxdtWrVfzIcL2e/fv3ssGHD/rPeIY888oh98803D6931lln2ZycHLtkyRJbo0YNu3///v9k//TTT+3tt99+zM9x5J9tbm6ubdiwod20aZO11tqrrrrKDh8+/JhjlpCQYF944YUTHvv9+/fbmjVr2hUrVlhrrb3yyivzzXL++efbzz77zFpr7ccff2wvvPDCEx7bo+X33xCQZr3seXSmTkRE/O+RR+Doy4379nmWF5F27dpRs2ZNXC4XrVq1Ij09vcDbWmt5+OGHadGiBWeccQZr165l48aNx12/efPm/PXXXzz44IOMGzeOMmXK5LveN998Q5s2bWjdujXz58/P93Lk5MmTWbBgAZ07d6ZVq1YMGTKEjIwMypQpQ2xsLDfccAPff/89brf7mG27du3K2LFjGTt2LLfeeitz585l7dq1lCtXjri4uJP+3BdddBEul4smTZoc9+c9/fTTD2dp0qQJGRn/fd58QXICzJs3j65du9K8eXNSU1OZP3/+4c8uv/xyXC4X9evXp06dOixatOik2Y9mjKFv376kpKSwY8cOJk2axDnnnJPvuldccQVw/GO/aNEi6tSpc/geclddlf9zEiZNmsTVV18NQN++fRk/fvzhzwpybIuamjoREfG/vMtSBV7uhZiYmMNfR0REkJ2dXeBtU1NT2bx5M9OnT2fWrFlUqVLlhPcQa9CgATNmzKB58+Y8+uijDBw48Jh1Vq5cycsvv8yoUaOYM2cO5513Xr41rbWceeaZzJo1i1mzZrFgwQI+/vhjIiMjmTp1Kr169WLEiBGcffbZx2zbrVs3xo0bx7hx4zjllFOoVKkS3377LV27di3Qz33kMbN5l6tPtE5+x7UgOcFzSfLtt99m7ty5PPHEE/85Fkff3sPb231cd911pKSk8NVXX3HZZZcRGZn/MxYONbzHO/ZFoSDHtqipqRMREf+Ljy/ccj+Jiori4MGDxyzfuXMnlStXJioqitGjRx9zNupo69atw+1206dPH+6//35mzJgBQKlSpdi9ezcAu3btIi4ujjJlyrBx40Z+++23w9sfuV6HDh2YMGECy5YtAzzj9ZYsWcKePXvYuXMn5557Lq+99hqzZ88+JketWrXYsmULS5cupU6dOnTp0oWXX36Zbt26HbPukfssSsfLefT+du/eTbVq1Th48CCpR42nHDZsGLm5uSxfvpwVK1bQsGHDAu376H1Ur16d6tWr88wzz3DddSd/PPzxjn3Dhg1ZsWLF4bO9Q4cOzXf7Tp068fXXXwOefxgUtJn2Fz0mTERE/G/QIOjf/7+XYN1uz/Ji1L9/f1q0aEGbNm3+01j07t2bnj170rx5c5KTk2nUqNEJ68ydO5f7778fl8tFVFQU77333uH6Z599NtWrV2f06NG0bt2aRo0aUatWLTp37vyfHEeu99lnn3HVVVdx4MABAJ555hlKlSrFhRdeSGZmJtZaXn311XyztG/f/vAszq5du/LQQw/RpUuXY9br2bMnvXr14qeffuKtt94q3IE7gd27d+eb88orr+Smm27izTff5Ntvv+Xpp5+mffv2VKpUifbt2/+nGYuPj6ddu3bs2rWL999/n9jY2ALt+9prr+WWW26hRIkSTJo0iRIlStC7d282b95M48aNT7p9pUqV8j32DRo04N133+Xss88mLi6Otm3b5rv9W2+9xXXXXcdLL71EpUqV+PTTTwuU219McZ0SdEpycrJNS0tzOoaISMhZuHBhgf7iPCw11TOGbtUqzxm6QYMgbxaqSFG54447aN26NTfccINPdfbs2UPJkiWx1nL77bdTv3597r777iJK6ZHff0PGmOnW2mRv6ulMnYiIFI/evdXEiV8lJSURFxfHK6+84nOtDz/8kCFDhpCVlUXr1q25+eabiyChf6mpExERkZAwffr0Iqt19913F/mZOX/TRAkREfFaqA/hEfEXf/y3o6ZORES8Ehsby9atW9XYiRSStZatW7cWeEJIQenyq4iIeKVmzZqsWbOGzZs3Ox1FJOjExsZSs2bNIq2ppk5ERLwSFRV1+I77IuI8XX4VERERCQFq6kRERERCgJo6ERERkRCgpk5EREQkBKipExEREQkBaupEREREQoCaOhEREZEQoKZOREREJASoqRMREREJAWrqREREREKAmjoRERGREKCmTkRERCQEqKkTERERCQFq6kRERERCgJo6ERERkRCgpk5EREQkBKipExEREQkBaupEREREQoCaOhEREZEQoKZOREREJASoqRMREREJAWrqREREREKAmjoRERGREKCmTkRERCQEqKkTERERCQFq6kRERERCgJo6ERERkRCgpk5EREQkBKipExEREQkBaupEREREQoCaOhEREZEQoKZOREREJASoqRMREREJAWrqREREREKAmjoRERGREKCmTkRERCQEqKkTERERCQFq6kRERERCQKTTAaRgrLWs2bqX+au3kbF5D+u372PbngNkZmWzZtteqpYtQVxsFO6YSCqUjKVS6VgqlSlBzQpx1K5cmjLuaKd/BBEREfEjNXUBbtueTEakrWLU3DVs2LEfgJioCKqVdVO+VAwVSkZzIDuH6uXj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",
      "text/plain": [
       "<Figure size 720x720 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# sieve out initial states that lie outside of stability region\n",
    "\n",
    "T_start = list()\n",
    "CA_start = list()\n",
    "\n",
    "for T in T_initial:\n",
    "    for CA in CA_initial:\n",
    "        x = np.array([CA, T])\n",
    "        if x @ P @ x < 372:\n",
    "            CA_start.append(CA)\n",
    "            T_start.append(T)\n",
    "print(\"number of initial conditions: {}\".format(len(CA_start)))\n",
    "\n",
    "# convert to np.arrays\n",
    "CA_start = np.array([CA_start])\n",
    "T_start = np.array([T_start])\n",
    "x_deviation = np.concatenate((CA_start.T, T_start.T), axis=1)  # every row is a pair of initial states within stability region\n",
    "print(\"shape of x_deviation is {}\".format(x_deviation.shape))\n",
    "\n",
    "\"\"\"\n",
    "    equation for the stability ellipse is 1060x^2 + 44xy + 0.52y^2 - 372 = 0\n",
    "\"\"\"\n",
    "# prepare x and y coordinates for plotting the stability region\n",
    "y = np.linspace(-100, 100, 100000, endpoint=True)\n",
    "\n",
    "x_upper = list()\n",
    "x_lower = list()\n",
    "y_plot = list()\n",
    "\n",
    "for i in y:\n",
    "    sqrt = np.sqrt(-2688000 * i**2 + 15772800000)\n",
    "    if sqrt >= 0:\n",
    "        y_plot.append(i)\n",
    "        x_upper.append((-4400 * i + sqrt) / 212000)\n",
    "        x_lower.append((-4400 * i - sqrt) / 212000)\n",
    "        pass\n",
    "    pass\n",
    "\n",
    "\n",
    "plt.figure(figsize=(10,10))\n",
    "# plot stability region        \n",
    "plt.plot(x_lower, y_plot, color='steelblue')\n",
    "plt.plot(x_upper, y_plot, color='steelblue')\n",
    "plt.ylim([-100, 100])\n",
    "plt.xlim([-2, 2])\n",
    "\n",
    "plt.xlabel(\"C_A - C_As\")\n",
    "plt.ylabel(\"T - T_s\")\n",
    "plt.scatter(CA_start, T_start, color='red', label='Initial states within stability region')\n",
    "plt.title(\"Stability region of CSTR 1\")\n",
    "plt.legend()\n",
    "plt.show() "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "aca52390",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Open-loop simulations of the first-principles model of CSTR\n",
    "\n",
    "def CSTR_simulation(F, V, C_A0, k_0, E, R, T_0, delta_H, rho_L, C_p, Q, t_final, t_step, C_A_initial, T_initial):\n",
    "    \"\"\"\n",
    "        simulating CSTR using forward Euler method\n",
    "    \"\"\"\n",
    "    \n",
    "    C_A_list = list()  # evolution of CA over time\n",
    "    T_list = list()  # evolution of T over time\n",
    "    \n",
    "    C_A = C_A_initial + C_As\n",
    "    T = T_initial + T_s\n",
    "    \n",
    "    for i in range(int(t_final / t_step)):\n",
    "        dCAdt = F / V * (C_A0 - C_A) - k_0 * np.exp(-E / (R * T)) * C_A**2\n",
    "        dTdt = F / V * (T_0 - T) - delta_H / (rho_L * C_p) * k_0 * np.exp(-E / (R * T)) * C_A**2 + Q / (rho_L * C_p * V)\n",
    "        \n",
    "        C_A += dCAdt * t_step\n",
    "        T += dTdt * t_step\n",
    "\n",
    "        if (i+1)% 10 == 0:\n",
    "            C_A_list.append(C_A - C_As)  # in deviation form\n",
    "            T_list.append(T - T_s)  # in deviation form \n",
    "    \n",
    "        \n",
    "    return C_A_list, T_list"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "78e2da64",
   "metadata": {},
   "outputs": [],
   "source": [
    "# get X and y data for training and testing\n",
    "\n",
    "CA_output = list()\n",
    "T_output = list()\n",
    "CA_input = list()\n",
    "T_input = list()\n",
    "CA0_input = list()\n",
    "Q_input = list()\n",
    "\n",
    "for u1 in u1_list:\n",
    "    C_A0 = u1 + C_A0s\n",
    "    \n",
    "    for u2 in u2_list:\n",
    "        Q = u2 + Q_s\n",
    "        \n",
    "        for C_A_initial, T_initial in x_deviation:\n",
    "            CA0_input.append(u1)\n",
    "            Q_input.append(u2)\n",
    "            CA_input.append(C_A_initial)\n",
    "            T_input.append(T_initial)\n",
    "            \n",
    "            C_A_list, T_list = \\\n",
    "                CSTR_simulation(F, V, C_A0, k_0, E, R, T_0, delta_H, rho_L, C_p, Q, t_final, t_step, C_A_initial, T_initial)\n",
    "            CA_output.append(C_A_list)\n",
    "            T_output.append(T_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "f917b532",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "RNN_input shape is (11800, 10, 4)\n",
      "[ 1.28076923e+00 -7.12307692e+01 -3.50000000e+00 -5.00000000e+05]\n",
      "[ 1.28076923e+00 -7.12307692e+01 -3.50000000e+00 -5.00000000e+05]\n",
      "RNN_output shape is (11800, 10, 2)\n"
     ]
    }
   ],
   "source": [
    "# collate input for RNN\n",
    "\n",
    "CA0_input = np.array(CA0_input)\n",
    "CA0_input = CA0_input.reshape(-1,1,1)\n",
    "\n",
    "Q_input = np.array(Q_input)\n",
    "Q_input = Q_input.reshape(-1,1,1)\n",
    "\n",
    "CA_input = np.array(CA_input)\n",
    "CA_input = CA_input.reshape(-1,1,1)\n",
    "\n",
    "T_input = np.array(T_input)\n",
    "T_input = T_input.reshape(-1,1,1)\n",
    "\n",
    "RNN_input = np.concatenate((CA_input, T_input, CA0_input, Q_input), axis=2)\n",
    "\n",
    "\"\"\"\n",
    "    the input to RNN is in the shape [number of samples x timestep x variables], and the input variables are same for every\n",
    "    time step, not sure if my treatment here is correct\n",
    "\"\"\"\n",
    "RNN_input = RNN_input.repeat(10, axis=1) # 10 time steps in this example\n",
    "print(\"RNN_input shape is {}\".format(RNN_input.shape))\n",
    "\n",
    "# checking the input is duplicated 10 times for each time step\n",
    "print(RNN_input[0, 0])\n",
    "print(RNN_input[0, 1])\n",
    "\n",
    "# collate output for RNN\n",
    "\n",
    "CA_output = np.array(CA_output)\n",
    "CA_output = CA_output.reshape(-1, 10, 1)\n",
    "\n",
    "T_output = np.array(T_output)\n",
    "T_output = T_output.reshape(-1, 10, 1)\n",
    "\n",
    "RNN_output = np.concatenate((CA_output, T_output), axis=2)\n",
    "print(\"RNN_output shape is {}\".format(RNN_output.shape))  # output shape: number of samples x timestep x variables\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "44191af9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[-0.01258149  0.86831812  0.          0.        ]\n",
      "[7.35427655e-01 1.53825756e+03 4.99074074e+00 1.01851852e+11]\n",
      "[-0.00638671  0.55490048]\n",
      "[7.21030616e-01 1.61082839e+03]\n",
      "X_total_test shape is: torch.Size([11800, 10, 4])\n",
      "y_toal_test shape is: torch.Size([11800, 10, 2])\n"
     ]
    }
   ],
   "source": [
    "import joblib\n",
    "\n",
    "# Normalization\n",
    "\n",
    "scaler_X = preprocessing.StandardScaler().fit(RNN_input.reshape(-1, 4))\n",
    "scaler_y = preprocessing.StandardScaler().fit(RNN_output.reshape(-1, 2))\n",
    "\n",
    "print(scaler_X.mean_)\n",
    "print(scaler_X.var_)\n",
    "print(scaler_y.mean_)\n",
    "print(scaler_y.var_)\n",
    "\n",
    "mean_y = torch.from_numpy(scaler_y.mean_).float()\n",
    "std_y = torch.from_numpy(np.sqrt(scaler_y.var_)).float()\n",
    "\n",
    "RNN_input = scaler_X.transform(RNN_input.reshape(-1, 4)).reshape(-1,10,4)\n",
    "RNN_output = scaler_y.transform(RNN_output.reshape(-1, 2)).reshape(-1,10,2)\n",
    "\n",
    "joblib.dump(scaler_X, 'scaler_X.pkl')\n",
    "joblib.dump(scaler_y, 'scaler_y.pkl')\n",
    "\n",
    "X_total_test = torch.from_numpy(RNN_input).float() \n",
    "y_total_test = torch.from_numpy(RNN_output).float() \n",
    "\n",
    "print(f'X_total_test shape is: {X_total_test.shape}')\n",
    "print(f'y_toal_test shape is: {y_total_test.shape}')\n",
    "\n",
    "dataset_total_test = TensorDataset(X_total_test,y_total_test)\n",
    "dataloader_total_test = DataLoader(dataset_total_test, batch_size=12000, shuffle=False)\n",
    "torch.save(dataloader_total_test,'dataloader_total_test.pt')\n",
    "torch.save(mean_y,'mean_y.pt')\n",
    "torch.save(std_y,'std_y.pt')\n",
    "np.save('mean_X.npy',scaler_X.mean_)\n",
    "np.save('std_X.npy',np.sqrt(scaler_X.var_))\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7377db75",
   "metadata": {},
   "source": [
    "# Data generation for datadriven"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "bede540b",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Randomly draw n samples from X_total_test for data driven points\n",
    "n = 50\n",
    "indices = np.random.choice(X_total_test.shape[0], n, replace=False)   \n",
    "X_data = X_total_test[indices]\n",
    "y_data = y_total_test[indices]  \n",
    "\n",
    "dataset_data = TensorDataset(X_data, y_data)\n",
    "dataloader_data = DataLoader(dataset_data, batch_size=256, shuffle=True)\n",
    "torch.save(dataloader_data, 'dataloader_data.pt')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "900856ff",
   "metadata": {},
   "source": [
    "# Data generation for collocation points"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "bd10ba78",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "number of initial conditions: 118\n",
      "shape of x_deviation is (118, 2)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_3737431/699231809.py:55: RuntimeWarning: invalid value encountered in sqrt\n",
      "  sqrt = np.sqrt(-2688000 * i**2 + 15772800000)\n"
     ]
    },
    {
     "data": {
      "image/png": "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v5QsEvsl72Ahc1ZtY1rmZmav2sbuA/kYoG2jGnRLq03HpjXZnrWIYUOHlNs8t+/N5cXvlzJh0UY6N6vJ/Rf0pFpcdLm1XxmMMced1AXl8Ku1duhvvW6MiQLOBXoE1MkD8nzP5xhjVgGtgdkVGKqIiIgch9hoL33b1qdv2/oUWcuKTXuYtWIrM1duY9y0lRRNdfsTP/t5Km0a1aB5vSSa16tOap3qpTqp4UDeQRb8vJOJizcydekmjDFcMqAlvx/YCq8n+L/rq0yh+pu6ocAya+364gJjTB1gp7W20BjTHGgFrA5WgCIiInJsPMbQpmEN2jSswe8HtSY7r4Cl63fx2ZR57Pd4+X7BBj7NXwO4RK9m9VjqJsdTNyme5GoxxEdHcSDvIAfyCqiXHE9OfiHb9+Wybvt+Nuw8QGGRJTEuijN6NmNEelrIX2alooRqUncR8PZhZQOBB40xxT84us5au/NXNUVERCSkJcRG0aNFHfati2Hw4BMospYtu3PI2rKXrK372Lw7m617cli+aQ/7cg6Sk19AYZH7uZjHQHxMFCmJsTSplUi/tvXpmlaL9o1TiIkK//u7/paQTOqstVeUUPY+8P6vpxYREZGqzGMMDVISaJCSQN+2Jf3k3h1mPVhQRHJCDEaXxCpRSCZ1IiIiIoGqxUZDbLCjCG2R9QtCERERkTClpE5EREQkDCipExEREQkDSupEREREwoCSOhEREZEwoKROREREJAwoqRMREREJA0rqRERERMKAkjoRERGRMKCkTkRERCQMKKkTERERCQNK6kRERETCgJI6ERERkTCgpE5EREQkDCipExEREQkDSupEREREwoCSOhEREZEwoKROREREJAwoqRMREREJA0rqRERERMKAkjoRERGRMKCkTkRERCQMKKkTERERCQNK6kRERETCgJI6ERERkTCgpE5EREQkDCipExEREQkDSupEREREwoCSOhEREZEwoKROREREJAwoqRMREREJA0rqRERERMKAkjoRERGRMKCkTkRERCQMKKkTERERCQNK6kRERETCgJI6ERERkTCgpE5EREQkDCipExEREQkDSupEREREwoCSOhEREZEwoKROREREJAwoqRMREREJA0rqRERERMKAkjoRERGRMKCkTkRERCQMKKkTERERCQNK6kRERETCgJI6ERERkTCgpE5EREQkDCipExER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",
      "text/plain": [
       "<Figure size 720x720 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Collocation points in each region:\n",
      "Region 1: 28 points\n",
      "Region 2: 35 points\n",
      "Region 3: 25 points\n",
      "Total collocation points shape: (88, 2)\n"
     ]
    }
   ],
   "source": [
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# generating inputs and initial states for CSTR, all expressed in deviation form\n",
    "\n",
    "u1_list = np.linspace(-3.5, 3.5, 4, endpoint=True)\n",
    "u2_list = np.linspace(-5e5, 5e5, 4, endpoint=True)\n",
    "T_initial = np.linspace(300, 600, 40, endpoint=True) - T_s\n",
    "CA_initial = np.linspace(0, 6, 40, endpoint=True) - C_As\n",
    "\n",
    "# 假设你已有的初始状态数据\n",
    "# sieve out initial states that lie outside of stability region\n",
    "T_start = list()\n",
    "CA_start = list()\n",
    "\n",
    "for T in T_initial:\n",
    "    for CA in CA_initial:\n",
    "        x = np.array([CA, T])\n",
    "        if x @ P @ x < 372:\n",
    "            CA_start.append(CA)\n",
    "            T_start.append(T)\n",
    "print(\"number of initial conditions: {}\".format(len(CA_start)))\n",
    "\n",
    "# convert to np.arrays \n",
    "CA_start = np.array(CA_start)\n",
    "T_start = np.array(T_start)\n",
    "x_deviation = np.concatenate((CA_start[:, None], T_start[:, None]), axis=1)\n",
    "print(\"shape of x_deviation is {}\".format(x_deviation.shape))\n",
    "\n",
    "points = np.vstack((CA_start, T_start)).T\n",
    "\n",
    "circle_centers = [(-1.2, 50), (0.0, 0),  (1.2, -55)]\n",
    "radius = 20  \n",
    "\n",
    "collocation_points = []\n",
    "\n",
    "for center in circle_centers:\n",
    "    x_c, y_c = center\n",
    "    distances = np.sqrt((CA_start - x_c)**2 + (T_start - y_c)**2)  # ✅ shape = (N,)\n",
    "    in_circle_mask = distances <= radius\n",
    "    selected_points = points[in_circle_mask]  # ✅ shape match\n",
    "    collocation_points.append(selected_points)\n",
    "\n",
    "\"\"\"\n",
    "    equation for the stability ellipse is 1060x^2 + 44xy + 0.52y^2 - 372 = 0\n",
    "\"\"\"\n",
    "# prepare x and y coordinates for plotting the stability region\n",
    "y = np.linspace(-100, 100, 100000, endpoint=True)\n",
    "\n",
    "x_upper = list()\n",
    "x_lower = list()\n",
    "y_plot = list()\n",
    "for i in y:\n",
    "    sqrt = np.sqrt(-2688000 * i**2 + 15772800000)\n",
    "    if sqrt >= 0:\n",
    "        y_plot.append(i)\n",
    "        x_upper.append((-4400 * i + sqrt) / 212000)\n",
    "        x_lower.append((-4400 * i - sqrt) / 212000)\n",
    "        pass\n",
    "    pass\n",
    "\n",
    "\n",
    "# Step 2: 画图展示选取结果\n",
    "plt.figure(figsize=(10, 10))\n",
    "plt.plot(x_lower, y_plot, color='steelblue')\n",
    "plt.plot(x_upper, y_plot, color='steelblue')\n",
    "plt.ylim([-100, 100])\n",
    "plt.xlim([-2, 2])\n",
    "\n",
    "# 标记圆形区域\n",
    "colors = ['blue', 'green', 'orange', 'purple', 'cyan']\n",
    "for i, points_in_circle in enumerate(collocation_points):\n",
    "    if len(points_in_circle) > 0:\n",
    "        plt.scatter(points_in_circle[:, 0], points_in_circle[:, 1], color=colors[i], label=f'Collocation Zone {i+1}')\n",
    "        # 画圆边界\n",
    "        circle = plt.Circle(circle_centers[i], radius, color=colors[i], fill=False, linestyle='--')\n",
    "        plt.gca().add_patch(circle)\n",
    "\n",
    "plt.xlabel(\"C_A - C_As\")\n",
    "plt.ylabel(\"T - T_s\")\n",
    "plt.legend()\n",
    "plt.title(\"Selected 5 Circular Collocation Regions\")\n",
    "plt.grid(True)\n",
    "plt.show()\n",
    "\n",
    "print(\"Collocation points in each region:\")\n",
    "for i, points_in_circle in enumerate(collocation_points):\n",
    "    print(f\"Region {i+1}: {len(points_in_circle)} points\")\n",
    "  \n",
    "x_deviation = np.vstack(collocation_points)\n",
    "print(\"Total collocation points shape:\", x_deviation.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "1413b4fd",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "RNN_input shape is (1408, 10, 4)\n",
      "[-1.18076923e+00  3.64615385e+01 -3.50000000e+00 -5.00000000e+05]\n",
      "[-1.18076923e+00  3.64615385e+01 -3.50000000e+00 -5.00000000e+05]\n"
     ]
    }
   ],
   "source": [
    "# get X and y data for training and testing\n",
    "\n",
    "CA_input = list()\n",
    "T_input = list()\n",
    "CA0_input = list()\n",
    "Q_input = list()\n",
    "\n",
    "for u1 in u1_list:\n",
    "    C_A0 = u1 + C_A0s\n",
    "    \n",
    "    for u2 in u2_list:\n",
    "        Q = u2 + Q_s\n",
    "        \n",
    "        for C_A_initial, T_initial in x_deviation:\n",
    "            CA0_input.append(u1)\n",
    "            Q_input.append(u2)\n",
    "            CA_input.append(C_A_initial)\n",
    "            T_input.append(T_initial)\n",
    "            \n",
    "# collate input for RNN\n",
    "\n",
    "CA0_input = np.array(CA0_input)\n",
    "CA0_input = CA0_input.reshape(-1,1,1)\n",
    "\n",
    "Q_input = np.array(Q_input)\n",
    "Q_input = Q_input.reshape(-1,1,1)\n",
    "\n",
    "CA_input = np.array(CA_input)\n",
    "CA_input = CA_input.reshape(-1,1,1)\n",
    "\n",
    "T_input = np.array(T_input)\n",
    "T_input = T_input.reshape(-1,1,1)\n",
    "\n",
    "RNN_input = np.concatenate((CA_input, T_input, CA0_input, Q_input), axis=2)\n",
    "\n",
    "\"\"\"\n",
    "    the input to RNN is in the shape [number of samples x timestep x variables], and the input variables are same for every\n",
    "    time step, not sure if my treatment here is correct\n",
    "\"\"\"\n",
    "RNN_input = RNN_input.repeat(10, axis=1) # 10 time steps in this example\n",
    "print(\"RNN_input shape is {}\".format(RNN_input.shape))\n",
    "\n",
    "# checking the input is duplicated 10 times for each time step\n",
    "print(RNN_input[0, 0])\n",
    "print(RNN_input[0, 1])            "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "bd68043e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "X_collocation shape is: torch.Size([1408, 10, 4])\n"
     ]
    }
   ],
   "source": [
    "mean_X = np.load('mean_X.npy')\n",
    "std_X = np.load('std_X.npy')\n",
    "\n",
    "RNN_input = (RNN_input - mean_X) / std_X\n",
    "RNN_input = torch.from_numpy(RNN_input).float()\n",
    "X_collocation = RNN_input\n",
    "print(f'X_collocation shape is: {X_collocation.shape}')\n",
    "torch.save(X_collocation, 'X_collocation.pt')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "7093aaea",
   "metadata": {},
   "source": [
    "# Neural network"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "98638552",
   "metadata": {},
   "outputs": [],
   "source": [
    "class RNN(nn.Module):\n",
    "    \"Defines a RNN network\"\n",
    "    \n",
    "    def __init__(self, N_INPUT, N_OUTPUT, N_HIDDEN, N_LAYERS):\n",
    "        super(RNN, self).__init__()\n",
    "        self.layers = N_LAYERS\n",
    "        \n",
    "        if isinstance(N_HIDDEN, list):\n",
    "            self.rnn = nn.LSTM(N_INPUT, \n",
    "                                N_HIDDEN[0], \n",
    "                                batch_first=True)\n",
    "            \n",
    "            self.rnn1 = nn.ModuleList(\n",
    "                [nn.LSTM(N_HIDDEN[i], \n",
    "                        N_HIDDEN[i+1],\n",
    "                       batch_first=True) for i in range(N_LAYERS - 1)]\n",
    "            )\n",
    "            \n",
    "            self.output_layer = nn.Linear(N_HIDDEN[-1], N_OUTPUT)\n",
    "            \n",
    "            self.list_flag = True\n",
    "            \n",
    "        else:\n",
    "            self.rnn = nn.LSTM(N_INPUT, \n",
    "                                N_HIDDEN,\n",
    "                                N_LAYERS,\n",
    "                                batch_first=True,\n",
    "                                dropout=0.1)\n",
    "            \n",
    "            self.output_layer = nn.Linear(N_HIDDEN, N_OUTPUT)\n",
    "            \n",
    "            self.list_flag = False                                             \n",
    "        \n",
    "    def forward(self, x):\n",
    "\n",
    "        x, _ = self.rnn(x)\n",
    "        \n",
    "        if self.list_flag:\n",
    "            for i in range(self.layers - 1):\n",
    "                x, _ = self.rnn1[i](x)\n",
    "                \n",
    "        x = self.output_layer(x)\n",
    "        return x"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1835d219",
   "metadata": {},
   "source": [
    "# Regional division"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "86c023ef",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "\n",
    "import torch\n",
    "import numpy as np\n",
    "from sklearn.cluster import KMeans\n",
    "\n",
    "mean_X = np.load('mean_X.npy')\n",
    "std_X = np.load('std_X.npy')\n",
    "mean_y = torch.load('mean_y.pt')\n",
    "std_y=torch.load('std_y.pt')\n",
    "\n",
    "X_collocation = torch.load('X_collocation.pt')\n",
    "print(f'X_collocation shape is {X_collocation.shape}')\n",
    "\n",
    "# X_collocation shape is torch.Size([1000, 10, 4]), save the first time step only\n",
    "X_collocation = X_collocation[:, 0, :2]  \n",
    "X_np = X_collocation.cpu().numpy() \n",
    "\n",
    "K = 3\n",
    "kmeans = KMeans(n_clusters=K, random_state=66)\n",
    "region_ids = kmeans.fit_predict(X_np)   # shape: (1000,)\n",
    "\n",
    "Ω_k = [np.where(region_ids == k)[0] for k in range(K)]  \n",
    "np.save('Ω_k.npy', Ω_k) \n",
    "\n",
    "\n",
    "X_region_1 = X_collocation[Ω_k[0]]  #\n",
    "X_region_2 = X_collocation[Ω_k[1]]  # \n",
    "X_region_3 = X_collocation[Ω_k[2]]  # \n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\"\"\"\n",
    "    equation for the stability ellipse is 1060x^2 + 44xy + 0.52y^2 - 372 = 0\n",
    "\"\"\"\n",
    "# prepare x and y coordinates for plotting the stability region\n",
    "y = np.linspace(-100, 100, 100000, endpoint=True)\n",
    "\n",
    "x_upper = list()\n",
    "x_lower = list()\n",
    "y_plot = list()\n",
    "for i in y:\n",
    "    sqrt = np.sqrt(-2688000 * i**2 + 15772800000)\n",
    "    if sqrt >= 0:\n",
    "        y_plot.append(i)\n",
    "        x_upper.append((-4400 * i + sqrt) / 212000)\n",
    "        x_lower.append((-4400 * i - sqrt) / 212000)\n",
    "        pass\n",
    "    pass\n",
    "\n",
    "\n",
    "plt.figure(figsize=(10,10))\n",
    "# plot stability region        \n",
    "plt.plot(x_lower, y_plot, color='steelblue')\n",
    "plt.plot(x_upper, y_plot, color='steelblue')\n",
    "plt.ylim([-100, 100])\n",
    "plt.xlim([-2, 2])\n",
    "\n",
    "plt.xlabel(\"C_A - C_As\")\n",
    "plt.ylabel(\"T - T_s\")\n",
    "plt.scatter(X_region_1[:, 0]* std_X[0] + mean_X[0] ,\n",
    "            X_region_1[:, 1]* std_X[1] + mean_X[1] ,\n",
    "            color='red', label='Region 1', s=10)\n",
    "plt.scatter(X_region_2[:, 0]* std_X[0] + mean_X[0] ,\n",
    "            X_region_2[:, 1]* std_X[1] + mean_X[1] ,\n",
    "            color='green', label='Region 2', s=10)\n",
    "plt.scatter(X_region_3[:, 0]* std_X[0] + mean_X[0] ,\n",
    "            X_region_3[:, 1]* std_X[1] + mean_X[1] ,\n",
    "            color='blue', label='Region 3', s=10)\n",
    "plt.title(\"Stability region of CSTR 1\")\n",
    "plt.legend()\n",
    "plt.show() "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "eea6fd1d",
   "metadata": {},
   "source": [
    "# Resample collocation"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a056497f",
   "metadata": {},
   "outputs": [],
   "source": [
    "def resample_collocation(X_collocation, Ω_k, alpha_k, N_total=400):\n",
    "    \"\"\"\n",
    "        Resample collocation points from X_collocation based on Ω_k.\n",
    "        X_collocation: torch.Tensor, shape (N, 10, 4)\n",
    "        Ω_k: list of numpy arrays, each containing indices of collocation points for each region\n",
    "        alpha_k: list of floats, weights for each region, \n",
    "                should sum to number of loss (loss_data + loss_region_1 + loss_region_2 + loss_region_3 )\n",
    "        N_total: int, total number of collocation points to sample\n",
    "    \"\"\"\n",
    "    sampled_indices = []\n",
    "    alpha_k = np.array(alpha_k[1:])/4 # \n",
    "    ratio_k = alpha_k / np.sum(alpha_k) \n",
    "    ratio_k = np.round(ratio_k, 2)  # \n",
    "    print(f'ratio_k: {ratio_k}')\n",
    "    for i, Ω in enumerate(Ω_k):\n",
    "        N_region = int(N_total * ratio_k[i])\n",
    "        print(f'Region {i+1}, N_region: {N_region}, Ω size: {Ω.shape[0]}')\n",
    "        sampled_indices_region = np.random.choice(Ω, size=N_region, replace=False)\n",
    "        sampled_indices.append(sampled_indices_region)\n",
    "        \n",
    "    sampled_indices = np.array(sampled_indices)\n",
    "    return sampled_indices\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4817b4a8",
   "metadata": {},
   "source": [
    "# Baseline"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "76fe0358",
   "metadata": {},
   "outputs": [],
   "source": [
    "mean_X = np.load('mean_X.npy')\n",
    "std_X = np.load('std_X.npy')\n",
    "mean_y = torch.load('mean_y.pt')\n",
    "std_y=torch.load('std_y.pt')\n",
    "\n",
    "dataloader_data = torch.load('dataloader_data.pt')\n",
    "X_collocation = torch.load('X_collocation.pt')\n",
    "dataloader_total_test = torch.load('dataloader_total_test.pt')\n",
    "\n",
    "Ω_k = np.load('Ω_k.npy', allow_pickle=True)  \n",
    "\n",
    "def train_model_with_moo(model, n_epochs, dataloader_data, X_collocation):\n",
    "\n",
    "    train_losses = []\n",
    "    valid_losses = []\n",
    "    avg_train_losses = []\n",
    "    avg_valid_losses = []\n",
    "    loss_list = []\n",
    "    para_list = []  # \n",
    "\n",
    "    F_para = torch.nn.Parameter(torch.tensor(0.625),requires_grad=True)\n",
    "    k_0_para = torch.nn.Parameter(torch.tensor(1.25),requires_grad=True) \n",
    "    optimizer = torch.optim.Adam([\n",
    "            {'params': model.parameters(), 'lr': 1e-3},  # \n",
    "            {'params': F_para, 'lr': 1e-2},  # \n",
    "            {'params': k_0_para, 'lr': 1e-2}  # \n",
    "        ])      \n",
    "\n",
    "    alpha_k = [1.0, 1.0, 1.0, 1.0]  # \n",
    "    print(f'alpha_k: {alpha_k}')\n",
    "    \n",
    "    '''\n",
    "    Set resample_collocation function to resample collocation points based on Ω_k and alpha_k.\n",
    "    '''\n",
    "    sampled_indices = resample_collocation(X_collocation, Ω_k, alpha_k)  # Resample collocation points\n",
    "\n",
    "    for epoch in range(1, n_epochs+1):\n",
    "        model.train()\n",
    "        # Training loop\n",
    "        for id_batch, (x_batch, y_batch) in enumerate(dataloader_data):\n",
    "            x_batch, y_batch = x_batch.to(device), y_batch.to(device)\n",
    "            optimizer.zero_grad()\n",
    "\n",
    "            NN_output_data = model(x_batch)  #\n",
    "\n",
    "            # Data driven loss term\n",
    "            loss1 = torch.mean((NN_output_data[:, :, :] - y_batch[:, :, :]) ** 2)\n",
    "            loss1 = 1e2 * loss1  # \n",
    "            print(f'Epoch {epoch}, batch {id_batch}, loss1: {loss1.item()}')\n",
    "            \n",
    "            X_collocation = X_collocation.to(device)\n",
    "            NN_output = model(X_collocation)\n",
    "\n",
    "            # Compute the physics-driven loss term\n",
    "            CA_NN_input = X_collocation[:, :, 0] * std_X[0] + mean_X[0] + C_As\n",
    "            T_NN_input = X_collocation[:, :, 1] * std_X[1] + mean_X[1] + T_s\n",
    "            C_A0 = X_collocation[:, :, 2] * std_X[2] + mean_X[2] + C_A0s\n",
    "            Q = X_collocation[:, :, 3] * std_X[3] + mean_X[3] + Q_s\n",
    "            NN_output = NN_output * std_y.to(device) + mean_y.to(device) + torch.from_numpy(np.array([C_As, T_s])).float().to(device)\n",
    "            \n",
    "            dCA_first = (NN_output[:, 1:2, 0] - CA_NN_input[:, 0:1 ]) / (2*t_step*10)\n",
    "            dT_first = (NN_output[:, 1:2, 1] - T_NN_input[:, 0:1]) / (2*t_step*10)\n",
    "\n",
    "            dCA_center = (NN_output[:, 2:, 0] - NN_output[:, :-2, 0]) / (2 * t_step*10)\n",
    "            dT_center = (NN_output[:, 2:, 1] - NN_output[:, :-2, 1]) / (2 * t_step*10)\n",
    "\n",
    "            dCA_last = (NN_output[:, -1:, 0] - NN_output[:, -2:-1, 0]) / (t_step*10)\n",
    "            dT_last = (NN_output[:, -1:, 1] - NN_output[:, -2:-1, 1]) / (t_step*10)\n",
    "\n",
    "            dCA = torch.cat((dCA_first, dCA_center, dCA_last), 1)\n",
    "            dT = torch.cat((dT_first, dT_center, dT_last), 1)\n",
    "\n",
    "            lossCA = dCA - F_para * F / V * (C_A0 - NN_output[:, :, 0]) + k_0_para * k_0 * torch.exp(-E / (R * NN_output[:, :, 1])) * NN_output[:, :, 0]**2\n",
    "\n",
    "            lossT = dT - F_para * F / V * (T_0 - NN_output[:, :, 1]) + delta_H / (rho_L * C_p) * k_0_para * k_0 * torch.exp(-E / (R * NN_output[:, :, 1])) * NN_output[:, :, 0]**2 - Q / (rho_L * C_p * V)\n",
    "\n",
    "            loss_region_1 = 2*1e-2 * torch.mean(lossCA[sampled_indices[0],:]**2) + 1e-5 *torch.mean(lossT[sampled_indices[0],:]**2)\n",
    "            loss_region_2 = 2*1e-2 *torch.mean(lossCA[sampled_indices[1],:]**2) + 1e-5 *torch.mean(lossT[sampled_indices[1],:]**2)\n",
    "            loss_region_3 = 2*1e-2 *torch.mean(lossCA[sampled_indices[2],:]**2) + 1e-5 *torch.mean(lossT[sampled_indices[2],:]**2)\n",
    "\n",
    "            # backpropagate joint loss\n",
    "            loss_physics = alpha_k[1] * loss_region_1 + alpha_k[2] * loss_region_2 + alpha_k[3] * loss_region_3 \n",
    "            print(f'Epoch {epoch}, batch {id_batch}, loss_physics: {loss_physics.item()}')\n",
    "\n",
    "            loss = alpha_k[0]  * loss1 + loss_physics  # add all loss terms together           \n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "\n",
    "            # record training loss\n",
    "            train_losses.append(loss.item())\n",
    "            loss_list.append([loss1.item(), loss_region_1.item(), loss_region_2.item(), loss_region_3.item()])\n",
    "            para_list.append([F_para.item(), k_0_para.item()])\n",
    "\n",
    "\n",
    "        model.eval() # prep model for evaluation   \n",
    "        for val_batch,(x_valbatch, y_valbatch) in enumerate(dataloader_total_test):\n",
    "            x_valbatch, y_valbatch = x_valbatch.to(device), y_valbatch.to(device)  # use valiadation data   \n",
    "            NN_output = model(x_valbatch)\n",
    "            # Data driven loss term\n",
    "            loss1 = torch.mean((NN_output[:, :, :] - y_valbatch[:, :, :])**2)  \n",
    "            loss =  loss1\n",
    "            valid_losses.append(loss.item())\n",
    "            print(f'valid_loss: {loss.item()}')\n",
    "            \n",
    "        train_loss = np.mean(train_losses)\n",
    "        valid_loss = np.mean(valid_losses)\n",
    "        avg_train_losses.append(train_loss)\n",
    "        avg_valid_losses.append(valid_loss)\n",
    "\n",
    "        print_msg = (f'[{epoch}/{n_epochs}] ' +\n",
    "                    f'train_loss: {train_loss:.5f} ' +\n",
    "                    f'valid_loss: {valid_loss:.5f}')\n",
    "        \n",
    "        print(print_msg)        \n",
    "        train_losses = []\n",
    "        valid_losses = []\n",
    "    \n",
    "    plt.figure(figsize=(16, 8))\n",
    "    plt.plot(range(1, len(avg_valid_losses) + 1), avg_valid_losses, label='Validation Loss')\n",
    "    plt.xlabel('epochs')\n",
    "    plt.ylabel('loss')\n",
    "    plt.yscale('log')\n",
    "    plt.grid(True)\n",
    "    plt.legend()\n",
    "    plt.show()\n",
    "\n",
    "\n",
    "    para_list = np.array(para_list)\n",
    "    plt.figure(figsize=(12, 10))\n",
    "    plt.plot(para_list[:,0], label='F', color='r')\n",
    "    plt.plot(para_list[:,1], label='k_0', color='g')\n",
    "    plt.axhline(y=1.0, color='b', linestyle='--', label='true para')\n",
    "    plt.xlabel(\"Epoch\")\n",
    "    plt.ylabel(\"Value\")\n",
    "    plt.title(\"Parameter Estimation\")\n",
    "    plt.yscale('log')\n",
    "    plt.grid(True)\n",
    "    plt.legend()\n",
    "    plt.show()\n",
    "\n",
    "\n",
    "    loss_list = np.array(loss_list)\n",
    "    plt.figure(figsize=(12, 10))\n",
    "    plt.plot(loss_list[:,0], label='data loss', color='r')\n",
    "    plt.plot(loss_list[:,1], label='CA loss', color='g')\n",
    "    plt.plot(loss_list[:,2], label='T loss', color='b')\n",
    "    plt.xlabel(\"Epoch\")\n",
    "    plt.ylabel(\"Loss\")\n",
    "    plt.yscale('log')\n",
    "    plt.title(\"Loss\")\n",
    "    plt.legend()\n",
    "    plt.show\n",
    "\n",
    "\n",
    "    return model, avg_train_losses, avg_valid_losses,loss_list , para_list\n",
    "\n",
    "# Train the PIRNN model\n",
    "model_PINN = RNN(4, 2, [128,64], 2)\n",
    "model_PINN.to(device)\n",
    "print(model_PINN)\n",
    "\n",
    "n_epochs = 2500\n",
    "model_PINN, train_loss, valid_loss, loss_list, para_list = train_model_with_moo(model_PINN, n_epochs, dataloader_data, X_collocation)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1f854452",
   "metadata": {},
   "outputs": [],
   "source": [
    "np.save('loss_list_baseline.npy',loss_list)\n",
    "np.save('valid_losses_baseline.npy',valid_loss)\n",
    "np.save('para_list_baseline.npy', para_list)\n",
    "torch.save(model_PINN,'baseline.pkl')"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4abe330a",
   "metadata": {},
   "source": [
    "# Training processs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d3a23ae0",
   "metadata": {},
   "outputs": [],
   "source": [
    "from pymoo.algorithms.moo.nsga2 import NSGA2\n",
    "from pymoo.operators.mutation.pm import PM\n",
    "from pymoo.operators.crossover.sbx import SBX\n",
    "from pymoo.algorithms.moo.nsga3 import NSGA3\n",
    "from pymoo.core.problem import Problem\n",
    "from pymoo.optimize import minimize\n",
    "import torch\n",
    "import numpy as np\n",
    "from pymoo.core.sampling import Sampling\n",
    "import copy\n",
    "class CustomSampling(Sampling):\n",
    "    def __init__(self, initial_population):\n",
    "        super().__init__()\n",
    "        self.initial_population = initial_population\n",
    "\n",
    "    def _do(self, problem, n_samples, **kwargs):\n",
    "        return np.tile(self.initial_population, (n_samples, 1))\n",
    "\n",
    "# 定义 MOO 问题\n",
    "class NeuralNetworkMOO(Problem):\n",
    "    def __init__(self, model, dataloader_data, X_collocation, device):\n",
    "        super().__init__(\n",
    "            n_var=sum(p.numel() for p in model.parameters()) + 2,  \n",
    "            n_obj=4, \n",
    "            n_constr=0,  \n",
    "            xl=np.hstack([[-10] * sum(p.numel() for p in model.parameters()), 0.5, 0.5]),  \n",
    "            xu=np.hstack([[10] * sum(p.numel() for p in model.parameters()), 2.0, 2.0]))  \n",
    "        \n",
    "        self.model = model\n",
    "        self.dataloader_data = dataloader_data\n",
    "        self.X_collocation = X_collocation\n",
    "        self.device = device\n",
    "\n",
    "    def _evaluate(self, X, out, *args, **kwargs):\n",
    "\n",
    "        # print('x shape:', X.shape)\n",
    "        Ω_k = np.load('Ω_k.npy', allow_pickle=True)  # \n",
    "\n",
    "        object_list = []\n",
    "        for i in range(X.shape[0]):\n",
    "            \n",
    "            start_idx = 0\n",
    "            for param in self.model.parameters():\n",
    "                num_params = param.numel()  # \n",
    "                # print('num_params:', num_params)\n",
    "                param_shape = param.shape   # \n",
    "                # print('param_shape:', param_shape)\n",
    "          \n",
    "                param.data = torch.tensor(\n",
    "                    X[i, start_idx:start_idx + num_params].reshape(param_shape),    # \n",
    "                    device=self.device, \n",
    "                    dtype=param.dtype\n",
    "                )\n",
    "                start_idx += num_params     \n",
    "            \n",
    "            F_para = X[i, start_idx]  # \n",
    "            k_0_para = X[i, start_idx + 1]  # \n",
    "\n",
    "            # Data-driven loss (L_data)\n",
    "            for id_batch, (x_batch, y_batch) in enumerate(self.dataloader_data):\n",
    "\n",
    "                x_batch, y_batch = x_batch.to(self.device), y_batch.to(self.device)\n",
    "                \n",
    "\n",
    "                NN_output = self.model(x_batch)\n",
    "                L_data = torch.mean((NN_output - y_batch) ** 2)\n",
    "                L_data =  1e2 * L_data               \n",
    "\n",
    "                # Compute the physics-driven loss term\n",
    "                X_collocation = self.X_collocation.to(device)\n",
    "                NN_output = self.model(X_collocation)\n",
    "\n",
    "                CA_NN_input = X_collocation[:, :, 0] * std_X[0] + mean_X[0] + C_As\n",
    "                T_NN_input = X_collocation[:, :, 1] * std_X[1] + mean_X[1] + T_s\n",
    "                C_A0 = X_collocation[:, :, 2] * std_X[2] + mean_X[2] + C_A0s\n",
    "                Q = X_collocation[:, :, 3] * std_X[3] + mean_X[3] + Q_s\n",
    "\n",
    "                NN_output = NN_output * std_y.to(device) + mean_y.to(device) + torch.from_numpy(np.array([C_As, T_s])).float().to(device)\n",
    "                \n",
    "                dCA_first = (NN_output[:, 1:2, 0] - CA_NN_input[:, 0:1 ]) / (2*t_step*10)\n",
    "                dT_first = (NN_output[:, 1:2, 1] - T_NN_input[:, 0:1]) / (2*t_step*10)\n",
    "\n",
    "                dCA_center = (NN_output[:, 2:, 0] - NN_output[:, :-2, 0]) / (2 * t_step*10)\n",
    "                dT_center = (NN_output[:, 2:, 1] - NN_output[:, :-2, 1]) / (2 * t_step*10)\n",
    "\n",
    "                dCA_last = (NN_output[:, -1:, 0] - NN_output[:, -2:-1, 0]) / (t_step*10)\n",
    "                dT_last = (NN_output[:, -1:, 1] - NN_output[:, -2:-1, 1]) / (t_step*10)\n",
    "\n",
    "                dCA = torch.cat((dCA_first, dCA_center, dCA_last), 1)\n",
    "                dT = torch.cat((dT_first, dT_center, dT_last), 1)\n",
    "\n",
    "                lossCA = dCA - F_para * F / V * (C_A0 - NN_output[:, :, 0]) + k_0_para* k_0 * torch.exp(-E / (R * NN_output[:, :, 1])) * NN_output[:, :, 0]**2\n",
    "                lossT = dT - F_para * F / V * (T_0 - NN_output[:, :, 1]) + delta_H / (rho_L * C_p) * k_0_para*k_0 * torch.exp(-E / (R * NN_output[:, :, 1])) * NN_output[:, :, 0]**2 - Q / (rho_L * C_p * V)\n",
    "\n",
    "\n",
    "                loss_region_1 = 2*1e-2 * torch.mean(lossCA[Ω_k[0],:]**2) + 1e-5 *torch.mean(lossT[Ω_k[0],:]**2)\n",
    "                loss_region_2 = 2*1e-2 *torch.mean(lossCA[Ω_k[1],:]**2) + 1e-5 *torch.mean(lossT[Ω_k[1],:]**2)\n",
    "                loss_region_3 = 2*1e-2 *torch.mean(lossCA[Ω_k[2],:]**2) + 1e-5 *torch.mean(lossT[Ω_k[2],:]**2)\n",
    "\n",
    "\n",
    "\n",
    "            object_list.append([L_data.item(),\n",
    "                                loss_region_1.item(),\n",
    "                                loss_region_2.item(),\n",
    "                                loss_region_3.item()])\n",
    "        \n",
    "        out[\"F\"] = np.array(object_list)  \n",
    "        # print(out[\"F\"].shape)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "4edcc0b2",
   "metadata": {},
   "source": [
    "# PaRS"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "aa8c8c33",
   "metadata": {},
   "outputs": [],
   "source": [
    "def calculate_weights_std(F, loss_list, temperature=1):   \n",
    "    \"\"\"\n",
    "        Calculate weights based on standard deviation and improvement ratio.\n",
    "        F: numpy array of shape (n_samples, n_objectives), current objective values\n",
    "        loss_list: list of lists, each inner list contains past loss values for each objective\n",
    "        temperature: float, temperature parameter for softmax\n",
    "    \"\"\"\n",
    "    \n",
    "    alpha = 10 \n",
    "    loss_list = np.array(loss_list)\n",
    "    prev_losses = np.min(loss_list[-alpha:], axis=0)\n",
    "\n",
    "    print(\"Previous losses:\", prev_losses)\n",
    "\n",
    "    min_F = np.min(F, axis=0)\n",
    "    print(\"Min:\", min_F)\n",
    "    max_F = np.max(F, axis=0)\n",
    "    print(\"Max:\", max_F)\n",
    "    F_normalized = (F - min_F) / (max_F - min_F)\n",
    "    means = np.mean(F, axis=0)\n",
    "    print (\"Means:\", means)\n",
    "    stds = np.std(F_normalized, axis=0)\n",
    "    print(\" Stds:\", stds)\n",
    "\n",
    "\n",
    "    improvement_ratio = means / prev_losses\n",
    "    print(\"Improvement ratio:\", improvement_ratio)\n",
    "\n",
    "    scores = stds / improvement_ratio\n",
    "    print(\"Scores:\", scores)\n",
    "\n",
    "    if prev_losses[0] > 1e-4: \n",
    "        temperature = np.array([0.2,0.8,0.8,0.8])\n",
    "    else: \n",
    "        temperature = 0.8\n",
    "    exp_scores = np.exp(scores /temperature)\n",
    "    weights = 4 * exp_scores / np.sum(exp_scores)\n",
    "\n",
    "\n",
    "    return weights"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "41423977",
   "metadata": {},
   "outputs": [],
   "source": [
    "mean_X = np.load('mean_X.npy')\n",
    "std_X = np.load('std_X.npy')\n",
    "mean_y = torch.load('mean_y.pt')\n",
    "std_y=torch.load('std_y.pt')\n",
    "\n",
    "dataloader_data = torch.load('dataloader_data.pt')\n",
    "X_collocation = torch.load('X_collocation.pt')\n",
    "dataloader_total_test = torch.load('dataloader_total_test.pt')\n",
    "\n",
    "Ω_k = np.load('Ω_k.npy', allow_pickle=True)  \n",
    "\n",
    "def train_model_with_moo(model, n_epochs, dataloader_data, X_collocation):\n",
    "  \n",
    "    train_losses = []\n",
    "    valid_losses = []\n",
    "    avg_train_losses = []\n",
    "    avg_valid_losses = []\n",
    "    loss_list = []\n",
    "    para_list = []  #\n",
    "    weight_list = []\n",
    "\n",
    "\n",
    "    # 定义 Adam 优化器\n",
    "    F_para = torch.nn.Parameter(torch.tensor(0.625),requires_grad=True)\n",
    "    k_0_para = torch.nn.Parameter(torch.tensor(1.25),requires_grad=True) \n",
    "    optimizer = torch.optim.Adam([\n",
    "            {'params': model.parameters(), 'lr': 1e-3},  # \n",
    "            {'params': F_para, 'lr': 1e-2},  # \n",
    "            {'params': k_0_para, 'lr': 1e-2}  # \n",
    "        ])      \n",
    "\n",
    "    alpha_k = [1.0,\n",
    "               1.0,\n",
    "               1.0,\n",
    "               1.0]  # \n",
    "    print(f'alpha_k: {alpha_k}')\n",
    "\n",
    "    '''\n",
    "    Set resample_collocation function to resample collocation points based on Ω_k and alpha_k.\n",
    "    '''\n",
    "    sampled_indices = resample_collocation(X_collocation, Ω_k, alpha_k)  # Resample collocation points\n",
    "\n",
    "    # Adam 优化阶段\n",
    "    for epoch in range(1, n_epochs+1):\n",
    "        model.train()\n",
    "        # Training loop\n",
    "        for id_batch, (x_batch, y_batch) in enumerate(dataloader_data):\n",
    "            x_batch, y_batch = x_batch.to(device), y_batch.to(device)\n",
    "            optimizer.zero_grad()\n",
    "\n",
    "            NN_output_data = model(x_batch)  # \n",
    "\n",
    "            # Data driven loss term\n",
    "            loss1 = torch.mean((NN_output_data[:, :, :] - y_batch[:, :, :]) ** 2)\n",
    "            loss1 =  1e2 * loss1  #\n",
    "            print(f'Epoch {epoch}, batch {id_batch}, loss1: {loss1.item()}')\n",
    "            \n",
    "\n",
    "            NN_output = model(X_collocation)\n",
    "            # Compute the physics-driven loss term\n",
    "            CA_NN_input = X_collocation[:, :, 0] * std_X[0] + mean_X[0] + C_As\n",
    "            T_NN_input = X_collocation[:, :, 1] * std_X[1] + mean_X[1] + T_s\n",
    "            C_A0 = X_collocation[:, :, 2] * std_X[2] + mean_X[2] + C_A0s\n",
    "            Q = X_collocation[:, :, 3] * std_X[3] + mean_X[3] + Q_s\n",
    "\n",
    "            NN_output = NN_output * std_y.to(device) + mean_y.to(device) + torch.from_numpy(np.array([C_As, T_s])).float().to(device)\n",
    "            \n",
    "            dCA_first = (NN_output[:, 1:2, 0] - CA_NN_input[:, 0:1 ]) / (2*t_step*10)\n",
    "            dT_first = (NN_output[:, 1:2, 1] - T_NN_input[:, 0:1]) / (2*t_step*10)\n",
    "\n",
    "            dCA_center = (NN_output[:, 2:, 0] - NN_output[:, :-2, 0]) / (2 * t_step*10)\n",
    "            dT_center = (NN_output[:, 2:, 1] - NN_output[:, :-2, 1]) / (2 * t_step*10)\n",
    "\n",
    "            dCA_last = (NN_output[:, -1:, 0] - NN_output[:, -2:-1, 0]) / (t_step*10)\n",
    "            dT_last = (NN_output[:, -1:, 1] - NN_output[:, -2:-1, 1]) / (t_step*10)\n",
    "\n",
    "            dCA = torch.cat((dCA_first, dCA_center, dCA_last), 1)\n",
    "            dT = torch.cat((dT_first, dT_center, dT_last), 1)\n",
    "\n",
    "            lossCA = dCA - F_para * F / V * (C_A0 - NN_output[:, :, 0]) + k_0_para * k_0 * torch.exp(-E / (R * NN_output[:, :, 1])) * NN_output[:, :, 0]**2\n",
    "\n",
    "            lossT = dT - F_para * F / V * (T_0 - NN_output[:, :, 1]) + delta_H / (rho_L * C_p) * k_0_para * k_0 * torch.exp(-E / (R * NN_output[:, :, 1])) * NN_output[:, :, 0]**2 - Q / (rho_L * C_p * V)\n",
    "\n",
    "            loss_region_1 = 2*1e-2 * torch.mean(lossCA[sampled_indices[0],:]**2) + 1e-5 *torch.mean(lossT[sampled_indices[0],:]**2)\n",
    "            loss_region_2 = 2*1e-2 *torch.mean(lossCA[sampled_indices[1],:]**2) + 1e-5 *torch.mean(lossT[sampled_indices[1],:]**2)\n",
    "            loss_region_3 = 2*1e-2 *torch.mean(lossCA[sampled_indices[2],:]**2) + 1e-5 *torch.mean(lossT[sampled_indices[2],:]**2)\n",
    "\n",
    "            # backpropagate joint loss\n",
    "            loss_physics = alpha_k[1] * loss_region_1 + alpha_k[2] * loss_region_2 + alpha_k[3] * loss_region_3 \n",
    "            print(f'Epoch {epoch}, batch {id_batch}, loss_physics: {loss_physics.item()}')\n",
    "\n",
    "            loss = alpha_k[0] *  loss1 + loss_physics  # add all loss terms together           \n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "\n",
    "\n",
    "            # record training loss\n",
    "            train_losses.append(loss.item())\n",
    "            loss_list.append([loss1.item(), \n",
    "                              loss_region_1.item(), \n",
    "                              loss_region_2.item(), \n",
    "                              loss_region_3.item()])\n",
    "            weight_list.append(alpha_k)\n",
    "            para_list.append([F_para.item(), k_0_para.item()])  # \n",
    "\n",
    "        ''' \n",
    "        Add PaRS optimization here\n",
    "        '''\n",
    "        if  epoch ==100 or epoch == 400 or epoch == 700 or epoch == 1000 \\\n",
    "            or epoch == 1300 or epoch == 1700 or epoch == 2000\\\n",
    "                  or epoch == 2300 or epoch == 2700 or epoch == 3000 or epoch == 3300 or epoch == 3700:\n",
    "            initial_population = []\n",
    "            for param in model.parameters():\n",
    "                initial_population.append(param.detach().cpu().numpy().flatten())\n",
    "\n",
    "            initial_population.append(np.array([F_para.item()]))\n",
    "            initial_population.append(np.array([k_0_para.item()]))\n",
    "            initial_population = np.concatenate(initial_population, axis=0)   \n",
    "\n",
    "            sampling = CustomSampling(initial_population)\n",
    "\n",
    "            model_moo = copy.deepcopy(model).to(device)\n",
    "\n",
    "            problem = NeuralNetworkMOO(model_moo, dataloader_data, X_collocation, device)\n",
    "\n",
    "            algorithm = NSGA2(\n",
    "                pop_size=50, \n",
    "                sampling=sampling,\n",
    "                mutation=PM(prob=0.2, eta=5),\n",
    "                crossover=SBX(prob=0.9, eta=10)\n",
    "                )\n",
    "\n",
    "            res = minimize(problem,\n",
    "                            algorithm,\n",
    "                            termination=('n_gen', 30),  # \n",
    "                            seed=42,\n",
    "                            verbose=False)\n",
    "            print(\"Pareto front objectives:\", res.F)\n",
    "                \n",
    "            # calcualte the weights\n",
    "            traget_weights = calculate_weights_std(res.F, loss_list)\n",
    "            print(\"traget_weights:\", traget_weights)\n",
    "            alpha_k = traget_weights\n",
    "\n",
    "            sampled_indices = resample_collocation(X_collocation, Ω_k, alpha_k)  # Resample collocation points\n",
    "\n",
    "    \n",
    "\n",
    "        model.eval() # prep model for evaluation   \n",
    "        for val_batch,(x_valbatch, y_valbatch) in enumerate(dataloader_total_test):\n",
    "            x_valbatch, y_valbatch = x_valbatch.to(device), y_valbatch.to(device)  # use valiadation data   \n",
    "            NN_output = model(x_valbatch)\n",
    "            # Data driven loss term\n",
    "            loss1 = torch.mean((NN_output[:, :, :] - y_valbatch[:, :, :])**2)  \n",
    "            loss =  loss1\n",
    "            valid_losses.append(loss.item())\n",
    "            print(f'valid_loss: {loss.item()}')\n",
    "            \n",
    "        train_loss = np.mean(train_losses)\n",
    "        valid_loss = np.mean(valid_losses)\n",
    "        avg_train_losses.append(train_loss)\n",
    "        avg_valid_losses.append(valid_loss)\n",
    "\n",
    "        print_msg = (f'[{epoch}/{n_epochs}] ' +\n",
    "                    f'train_loss: {train_loss:.5f} ' +\n",
    "                    f'valid_loss: {valid_loss:.5f}')\n",
    "        \n",
    "        print(print_msg)        \n",
    "        train_losses = []\n",
    "        valid_losses = []\n",
    "    \n",
    "    plt.figure(figsize=(16, 8))\n",
    "    plt.plot(range(1, len(avg_valid_losses) + 1), avg_valid_losses, label='Validation Loss')\n",
    "    plt.xlabel('epochs')\n",
    "    plt.ylabel('loss')\n",
    "    plt.yscale('log')\n",
    "    plt.grid(True)\n",
    "    plt.legend()\n",
    "    plt.show()\n",
    "\n",
    "    para_list = np.array(para_list)\n",
    "    plt.figure(figsize=(12, 10))\n",
    "    plt.plot(para_list[:,0], label='F', color='r')\n",
    "    plt.plot(para_list[:,1], label='k_0', color='g')\n",
    "    plt.axhline(y=1.0, color='b', linestyle='--', label='true para')\n",
    "    plt.xlabel(\"Epoch\")\n",
    "    plt.ylabel(\"Value\")\n",
    "    plt.title(\"Parameter Estimation\")\n",
    "    plt.yscale('log')\n",
    "    plt.grid(True)\n",
    "    plt.legend()\n",
    "    plt.show()\n",
    "\n",
    "\n",
    "    return model, avg_train_losses, avg_valid_losses,loss_list, weight_list, para_list\n",
    "\n",
    "# Train the PIRNN model\n",
    "model_PINN = RNN(4, 2, [128,64], 2)\n",
    "model_PINN.to(device)\n",
    "print(model_PINN)\n",
    "\n",
    "n_epochs = 2500\n",
    "model_PINN, train_loss, valid_loss, loss_list, weight_list, para_list = train_model_with_moo(model_PINN, n_epochs, dataloader_data, X_collocation)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f8a6a167",
   "metadata": {},
   "outputs": [],
   "source": [
    "np.save('loss_list_PaRS_2.npy',loss_list)\n",
    "np.save('valid_losses_PaRS_2.npy',valid_loss)\n",
    "np.save('weight_list_PaRS_2.npy',weight_list)\n",
    "np.save('para_list_PaRS_2.npy', para_list)\n",
    "torch.save(model_PINN,'PaRS_2.pkl')\n"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8ac71699",
   "metadata": {},
   "source": [
    "# R3 sampling"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e56246a2",
   "metadata": {},
   "outputs": [],
   "source": [
    "mean_X = np.load('mean_X.npy')\n",
    "std_X = np.load('std_X.npy')\n",
    "mean_y = torch.load('mean_y.pt')\n",
    "std_y=torch.load('std_y.pt')\n",
    "\n",
    "dataloader_data = torch.load('dataloader_data.pt')\n",
    "X_collocation = torch.load('X_collocation.pt')\n",
    "dataloader_total_test = torch.load('dataloader_total_test.pt')\n",
    "\n",
    "Ω_k = np.load('Ω_k.npy', allow_pickle=True)  # \n",
    "\n",
    "def train_model_with_moo(model, n_epochs, dataloader_data, X_collocation):\n",
    "\n",
    "    train_losses = []\n",
    "    valid_losses = []\n",
    "    avg_train_losses = []\n",
    "    avg_valid_losses = []\n",
    "    loss_list = []\n",
    "    para_list = []  # \n",
    "\n",
    "    # 定义 Adam 优化器\n",
    "    F_para = torch.nn.Parameter(torch.tensor(0.625),requires_grad=True)\n",
    "    k_0_para = torch.nn.Parameter(torch.tensor(1.25),requires_grad=True) \n",
    "    optimizer = torch.optim.Adam([\n",
    "            {'params': model.parameters(), 'lr': 1e-3},  #\n",
    "            {'params': F_para, 'lr': 1e-2},  # \n",
    "            {'params': k_0_para, 'lr': 1e-2}  # \n",
    "        ])      \n",
    "\n",
    "    # alpha_k = [1.0, 1.0, 1.0, 1.0]  # 权重系数，初始值为1.0\n",
    "    # print(f'alpha_k: {alpha_k}')\n",
    "    \n",
    "    # '''\n",
    "    # Set resample_collocation function to resample collocation points based on Ω_k and alpha_k.\n",
    "    # '''\n",
    "    # sampled_indices = resample_collocation(X_collocation, Ω_k, alpha_k)  # Resample collocation points\n",
    "\n",
    "    Nr = 400\n",
    "    def sample_uniform(n_resample):\n",
    "       \n",
    "        pool_size = X_collocation.shape[0]  # \n",
    "        indices = torch.randperm(pool_size)[:n_resample]  #\n",
    "        return X_collocation[indices].to(device)  #   \n",
    "\n",
    "    def compute_residual(X_collocation_Nr):\n",
    "   \n",
    "        NN_output = model(X_collocation_Nr)\n",
    "\n",
    "     \n",
    "        CA_NN_input = X_collocation_Nr[:, :, 0] * std_X[0] + mean_X[0] + C_As\n",
    "        T_NN_input = X_collocation_Nr[:, :, 1] * std_X[1] + mean_X[1] + T_s\n",
    "        C_A0 = X_collocation_Nr[:, :, 2] * std_X[2] + mean_X[2] + C_A0s\n",
    "        Q = X_collocation_Nr[:, :, 3] * std_X[3] + mean_X[3] + Q_s\n",
    "\n",
    "        NN_output = NN_output * std_y.to(device) + mean_y.to(device) + torch.from_numpy(np.array([C_As, T_s])).float().to(device)\n",
    "        \n",
    "        dCA_first = (NN_output[:, 1:2, 0] - CA_NN_input[:, 0:1 ]) / (2*t_step*10)\n",
    "        dT_first = (NN_output[:, 1:2, 1] - T_NN_input[:, 0:1]) / (2*t_step*10)\n",
    "\n",
    "        dCA_center = (NN_output[:, 2:, 0] - NN_output[:, :-2, 0]) / (2 * t_step*10)\n",
    "        dT_center = (NN_output[:, 2:, 1] - NN_output[:, :-2, 1]) / (2 * t_step*10)\n",
    "\n",
    "        dCA_last = (NN_output[:, -1:, 0] - NN_output[:, -2:-1, 0]) / (t_step*10)\n",
    "        dT_last = (NN_output[:, -1:, 1] - NN_output[:, -2:-1, 1]) / (t_step*10)\n",
    "\n",
    "\n",
    "        dCA = torch.cat((dCA_first, dCA_center, dCA_last), 1)\n",
    "        dT = torch.cat((dT_first, dT_center, dT_last), 1)\n",
    "\n",
    "        lossCA = dCA - F_para * F / V * (C_A0 - NN_output[:, :, 0]) + k_0_para * k_0 * torch.exp(-E / (R * NN_output[:, :, 1])) * NN_output[:, :, 0]**2\n",
    "\n",
    "        lossT = dT - F_para * F / V * (T_0 - NN_output[:, :, 1]) + delta_H / (rho_L * C_p) * k_0_para * k_0 * torch.exp(-E / (R * NN_output[:, :, 1])) * NN_output[:, :, 0]**2 - Q / (rho_L * C_p * V)\n",
    "\n",
    "     \n",
    "        lossCA = torch.mean(lossCA**2, dim=1, keepdim=True) \n",
    "\n",
    "        lossT = torch.mean(lossT**2, dim=1, keepdim=True)\n",
    "        # residuals = torch.cat((residualCA, residualT), dim=1)\n",
    "        residuals = 2 * 1e-2 * lossCA + 1e-5 * lossT\n",
    "        return residuals\n",
    "\n",
    "    X_collocation_Nr = sample_uniform(Nr)\n",
    "\n",
    "    for epoch in range(1, n_epochs+1):\n",
    "        model.train()\n",
    "        ###################\n",
    "        ###################\n",
    "        \n",
    "        if epoch % 400== 0 and epoch > 100:\n",
    "            # Step 1: \n",
    "            residuals = compute_residual(X_collocation_Nr)\n",
    "            print('residuals shape is', residuals.shape)\n",
    "            # Step 2: \n",
    "            # tau_CA = residuals[:, 0].mean().item()  \n",
    "            # tau_T = residuals[:, 1].mean().item()  \n",
    "            # print(f'Epoch {epoch}, tau_CA: {tau_CA}, tau_T: {tau_T}')\n",
    "            tau = residuals.mean().item()\n",
    "\n",
    "            # Step 3: \n",
    "            # retained_mask = (residuals[:, 0] > tau_CA) & (residuals[:, 1] > tau_T)\n",
    "            retained_mask = residuals[:, 0] > tau\n",
    "            retained_samples = X_collocation_Nr[retained_mask]\n",
    "            print(f'Number of retained samples: {len(retained_samples)}')\n",
    "        \n",
    "            # Step 4: \n",
    "            n_resample = Nr - len(retained_samples)\n",
    "            resampled_points = sample_uniform(n_resample)\n",
    "\n",
    "            # Step 5: \n",
    "            X_collocation_Nr = torch.cat((retained_samples, resampled_points), dim=0)\n",
    "            \n",
    "            # X_collocation_Nr = sample_uniform(Nr)\n",
    "\n",
    "        # Training loop\n",
    "        for id_batch, (x_batch, y_batch) in enumerate(dataloader_data):\n",
    "            x_batch, y_batch = x_batch.to(device), y_batch.to(device)\n",
    "            optimizer.zero_grad()\n",
    "\n",
    "            NN_output_data = model(x_batch)  # \n",
    "\n",
    "            # Data driven loss term\n",
    "            loss1 = torch.mean((NN_output_data[:, :, :] - y_batch[:, :, :]) ** 2)\n",
    "            loss1 = 1e2 * loss1  #\n",
    "            print(f'Epoch {epoch}, batch {id_batch}, loss1: {loss1.item()}')\n",
    "            \n",
    "            X_collocation_Nr = X_collocation_Nr.to(device)\n",
    "            NN_output = model(X_collocation_Nr)\n",
    "\n",
    "            # Compute the physics-driven loss term\n",
    "            CA_NN_input = X_collocation_Nr[:, :, 0] * std_X[0] + mean_X[0] + C_As\n",
    "            T_NN_input = X_collocation_Nr[:, :, 1] * std_X[1] + mean_X[1] + T_s\n",
    "            C_A0 = X_collocation_Nr[:, :, 2] * std_X[2] + mean_X[2] + C_A0s\n",
    "            Q = X_collocation_Nr[:, :, 3] * std_X[3] + mean_X[3] + Q_s\n",
    "            NN_output = NN_output * std_y.to(device) + mean_y.to(device) + torch.from_numpy(np.array([C_As, T_s])).float().to(device)\n",
    "            \n",
    "            dCA_first = (NN_output[:, 1:2, 0] - CA_NN_input[:, 0:1 ]) / (2*t_step*10)\n",
    "            dT_first = (NN_output[:, 1:2, 1] - T_NN_input[:, 0:1]) / (2*t_step*10)\n",
    "\n",
    "            dCA_center = (NN_output[:, 2:, 0] - NN_output[:, :-2, 0]) / (2 * t_step*10)\n",
    "            dT_center = (NN_output[:, 2:, 1] - NN_output[:, :-2, 1]) / (2 * t_step*10)\n",
    "\n",
    "            dCA_last = (NN_output[:, -1:, 0] - NN_output[:, -2:-1, 0]) / (t_step*10)\n",
    "            dT_last = (NN_output[:, -1:, 1] - NN_output[:, -2:-1, 1]) / (t_step*10)\n",
    "\n",
    "            dCA = torch.cat((dCA_first, dCA_center, dCA_last), 1)\n",
    "            dT = torch.cat((dT_first, dT_center, dT_last), 1)\n",
    "\n",
    "            lossCA = dCA - F_para * F / V * (C_A0 - NN_output[:, :, 0]) + k_0_para * k_0 * torch.exp(-E / (R * NN_output[:, :, 1])) * NN_output[:, :, 0]**2\n",
    "\n",
    "            lossT = dT - F_para * F / V * (T_0 - NN_output[:, :, 1]) + delta_H / (rho_L * C_p) * k_0_para * k_0 * torch.exp(-E / (R * NN_output[:, :, 1])) * NN_output[:, :, 0]**2 - Q / (rho_L * C_p * V)\n",
    "            lossCA = torch.mean(lossCA**2)\n",
    "            lossT = torch.mean(lossT**2)\n",
    "            \n",
    "            # backpropagate joint loss\n",
    "            loss_physics = 2*1e-2 * lossCA + 1e-5 * lossT  # \n",
    "            print(f'Epoch {epoch}, batch {id_batch}, loss_physics: {loss_physics.item()}')\n",
    "\n",
    "            loss =  loss1 + 3 * loss_physics  # add all loss terms together           \n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "\n",
    "            # record training loss\n",
    "            train_losses.append(loss.item())\n",
    "            loss_list.append([loss1.item(), \n",
    "                              loss_physics.item()])\n",
    "            para_list.append([F_para.item(), k_0_para.item()])\n",
    "\n",
    "\n",
    "        # 验证过程\n",
    "        model.eval() # prep model for evaluation   \n",
    "        for val_batch,(x_valbatch, y_valbatch) in enumerate(dataloader_total_test):\n",
    "            x_valbatch, y_valbatch = x_valbatch.to(device), y_valbatch.to(device)  # use valiadation data   \n",
    "            NN_output = model(x_valbatch)\n",
    "            # Data driven loss term\n",
    "            loss1 = torch.mean((NN_output[:, :, :] - y_valbatch[:, :, :])**2)  \n",
    "            loss =  loss1\n",
    "            valid_losses.append(loss.item())\n",
    "            print(f'valid_loss: {loss.item()}')\n",
    "            \n",
    "        # # \n",
    "        train_loss = np.mean(train_losses)\n",
    "        valid_loss = np.mean(valid_losses)\n",
    "        avg_train_losses.append(train_loss)\n",
    "        avg_valid_losses.append(valid_loss)\n",
    "\n",
    "        print_msg = (f'[{epoch}/{n_epochs}] ' +\n",
    "                    f'train_loss: {train_loss:.5f} ' +\n",
    "                    f'valid_loss: {valid_loss:.5f}')\n",
    "        \n",
    "        print(print_msg)        \n",
    "        # # \n",
    "        train_losses = []\n",
    "        valid_losses = []\n",
    "    \n",
    "    plt.figure(figsize=(16, 8))\n",
    "    plt.plot(range(1, len(avg_valid_losses) + 1), avg_valid_losses, label='Validation Loss')\n",
    "    plt.xlabel('epochs')\n",
    "    plt.ylabel('loss')\n",
    "    plt.yscale('log')\n",
    "    plt.grid(True)\n",
    "    plt.legend()\n",
    "    plt.show()\n",
    "\n",
    "\n",
    "    para_list = np.array(para_list)\n",
    "    plt.figure(figsize=(12, 10))\n",
    "    plt.plot(para_list[:,0], label='F', color='r')\n",
    "    plt.plot(para_list[:,1], label='k_0', color='g')\n",
    "    plt.axhline(y=1.0, color='b', linestyle='--', label='true para')\n",
    "    plt.xlabel(\"Epoch\")\n",
    "    plt.ylabel(\"Value\")\n",
    "    plt.title(\"Parameter Estimation\")\n",
    "    plt.yscale('log')\n",
    "    plt.grid(True)\n",
    "    plt.legend()\n",
    "    plt.show()\n",
    "\n",
    "\n",
    "\n",
    "    return model, avg_train_losses, avg_valid_losses,loss_list , para_list\n",
    "\n",
    "# Train the PIRNN model\n",
    "model_PINN = RNN(4, 2, [128,64], 2)\n",
    "model_PINN.to(device)\n",
    "print(model_PINN)\n",
    "\n",
    "n_epochs = 2500\n",
    "model_PINN, train_loss, valid_loss, loss_list, para_list = train_model_with_moo(model_PINN, n_epochs, dataloader_data, X_collocation)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "4641b253",
   "metadata": {},
   "outputs": [],
   "source": [
    "np.save('loss_list_R3_2.npy',loss_list)\n",
    "np.save('valid_losses_R3_2.npy',valid_loss)\n",
    "np.save('para_list_R3_2.npy', para_list)\n",
    "torch.save(model_PINN,'R3_2.pkl')"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.8.10"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
