{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "ae641287",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_data_path = \"beach_profile_data/processed_data/beachdata_train.xlsx\"\n",
    "test_data_path = \"beach_profile_data/processed_data/beachdata_test.xlsx\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "id": "37f9be27",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "from sklearn.preprocessing import StandardScaler, LabelEncoder\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "\n",
    "\n",
    "def load_data(train_path: str, test_path: str):\n",
    "    \"\"\"Loads the training and test datasets from Excel files and returns them as pandas DataFrames.\"\"\"\n",
    "    # Load training data\n",
    "    train_df = pd.read_excel(train_path)\n",
    "    # Load test data\n",
    "    test_df = pd.read_excel(test_path)\n",
    "    return train_df, test_df\n",
    "\n",
    "\n",
    "def preprocess_data(train_df: pd.DataFrame, test_df: pd.DataFrame):\n",
    "    \"\"\"Preprocesses data by encoding categorical features and scaling numerical features.\"\"\"\n",
    "    # Identify categorical columns for encoding\n",
    "    categorical_features = ['Dominant Wave Direction']\n",
    "    # Encode categorical features using label encoding\n",
    "    for feature in categorical_features:\n",
    "        label_encoder = LabelEncoder()\n",
    "        train_df[feature] = label_encoder.fit_transform(train_df[feature])\n",
    "        test_df[feature] = label_encoder.transform(test_df[feature])\n",
    "    \n",
    "    # Select columns for scaling (excluding 'x' and 'y' columns)\n",
    "    numerical_features = train_df.drop(columns=['x', 'y'] + categorical_features).columns\n",
    "    scaler = StandardScaler()\n",
    "    # Scale the numerical columns\n",
    "    train_df[numerical_features] = scaler.fit_transform(train_df[numerical_features])\n",
    "    test_df[numerical_features] = scaler.transform(test_df[numerical_features])\n",
    "    \n",
    "    feature_names = list(train_df.columns)\n",
    "    \n",
    "    return train_df, test_df, feature_names\n",
    "\n",
    "\n",
    "def feature_selection(train_df: pd.DataFrame, feature_names: list):\n",
    "    \"\"\"Performs feature selection based on correlation with the target 'y'.\"\"\"\n",
    "    # Calculate correlation matrix\n",
    "    correlation_matrix = train_df.corr()\n",
    "    # Select features with correlation coefficient > 0.2 with 'y'\n",
    "    threshold = 0.2\n",
    "    selected_features = correlation_matrix.index[abs(correlation_matrix['y']) > threshold]\n",
    "    filtered_df = train_df[selected_features]\n",
    "    \n",
    "    selected_feature_names = list(selected_features.drop(['y']))\n",
    "    return filtered_df, selected_feature_names\n",
    "\n",
    "\n",
    "def bruun_dean_model_parameters(df: pd.DataFrame):\n",
    "    \"\"\"Computes additional parameters based on Bruun and Dean models and adds them to the DataFrame.\"\"\"\n",
    "    # Calculate horizontal retreat using Bruun model formula\n",
    "    df['horizontal_retreat'] = df['Mean Grain Size'] * df['Intertidal Slope'] / (df['Profile Length'] + 1)\n",
    "    # Include any other attributes from the Dean model you want to consider\n",
    "    # Currently, only the horizontal_retreat is added as a sample attributable to Bruun model\n",
    "    \n",
    "    return df\n",
    "\n",
    "\n",
    "def train_deep_learning_model(train_df: pd.DataFrame):\n",
    "    \"\"\"Defines and trains a PyTorch-based deep learning model using the processed dataset.\"\"\"\n",
    "    # Define a simple neural network architecture\n",
    "    class SimpleNN(nn.Module):\n",
    "        def __init__(self, input_size):\n",
    "            super(SimpleNN, self).__init__()\n",
    "            self.fc1 = nn.Linear(input_size, 64)  # Input layer\n",
    "            self.fc2 = nn.Linear(64, 32)  # Hidden layer\n",
    "            self.fc3 = nn.Linear(32, 1)  # Output layer\n",
    "        \n",
    "        def forward(self, x):\n",
    "            x = torch.relu(self.fc1(x))\n",
    "            x = torch.relu(self.fc2(x))\n",
    "            x = self.fc3(x)\n",
    "            return x\n",
    "    \n",
    "    # Prepare features and targets\n",
    "    X_train = train_df.drop(columns=['y']).values\n",
    "    y_train = train_df['y'].values\n",
    "\n",
    "    # Convert to PyTorch tensors\n",
    "    X_train_tensor = torch.tensor(X_train, dtype=torch.float32)\n",
    "    y_train_tensor = torch.tensor(y_train, dtype=torch.float32).view(-1, 1)\n",
    "    \n",
    "    # Initialize the Network, Loss Function, and Optimizer\n",
    "    model = SimpleNN(input_size=X_train_tensor.shape[1])\n",
    "    criterion = nn.MSELoss()  # Mean Squared Error Loss\n",
    "    optimizer = optim.Adam(model.parameters(), lr=0.001)\n",
    "    \n",
    "    # Training loop\n",
    "    epochs = 1000\n",
    "    for epoch in range(epochs):\n",
    "        model.train()\n",
    "        optimizer.zero_grad()  # Zero the gradients\n",
    "        outputs = model(X_train_tensor)  # Forward pass\n",
    "        loss = criterion(outputs, y_train_tensor)  # Calculate Loss\n",
    "        loss.backward()  # Backward pass\n",
    "        optimizer.step()  # Update weights\n",
    "    \n",
    "    return model\n",
    "\n",
    "\n",
    "def evaluate_model(model, test_df: pd.DataFrame):\n",
    "    \"\"\"Evaluates the trained deep learning model on the test dataset.\"\"\"\n",
    "    # Prepare test features\n",
    "    X_test = test_df.drop(columns=['y']).values\n",
    "    y_test = test_df['y'].values\n",
    "\n",
    "    # Convert to PyTorch tensors\n",
    "    X_test_tensor = torch.tensor(X_test, dtype=torch.float32)\n",
    "    y_test_tensor = torch.tensor(y_test, dtype=torch.float32).view(-1, 1)\n",
    "    \n",
    "    # Evaluate model\n",
    "    model.eval()\n",
    "    with torch.no_grad():\n",
    "        predictions = model(X_test_tensor)  # Get predictions\n",
    "        criterion = nn.MSELoss()\n",
    "        mse = criterion(predictions, y_test_tensor).item()  # Calculate Mean Squared Error\n",
    "    \n",
    "    return mse\n",
    "\n",
    "\n",
    "def main():\n",
    "    \"\"\"Main function to execute the pipeline.\"\"\"\n",
    "    # Define file paths\n",
    "    train_path = \"beach_profile_data/processed_data/beachdata_train.xlsx\"\n",
    "    test_path = \"beach_profile_data/processed_data/beachdata_test.xlsx\"\n",
    "    \n",
    "    # Load the data\n",
    "    train_df, test_df = load_data(train_path, test_path)\n",
    "    \n",
    "    # Preprocess the data\n",
    "    train_df, test_df, feature_names = preprocess_data(train_df, test_df)\n",
    "    \n",
    "    # Perform feature selection\n",
    "    # train_df, selected_feature_names = feature_selection(train_df, feature_names)\n",
    "    \n",
    "    # Enhance dataset with Bruun-Dean models parameters\n",
    "    train_df = bruun_dean_model_parameters(train_df)\n",
    "    test_df = bruun_dean_model_parameters(test_df)\n",
    "    \n",
    "    # Train the deep learning model\n",
    "    model = train_deep_learning_model(train_df)\n",
    "    \n",
    "    # Evaluate the model\n",
    "    mse = evaluate_model(model, test_df)\n",
    "    \n",
    "    return mse"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "id": "78ddcc0f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1.8388117551803589"
      ]
     },
     "execution_count": 49,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "main()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "114f0839",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1/100, Loss: nan\n",
      "Epoch 2/100, Loss: nan\n",
      "Epoch 3/100, Loss: nan\n",
      "Epoch 4/100, Loss: nan\n",
      "Epoch 5/100, Loss: nan\n",
      "Epoch 6/100, Loss: nan\n",
      "Epoch 7/100, Loss: nan\n",
      "Epoch 8/100, Loss: nan\n",
      "Epoch 9/100, Loss: nan\n",
      "Epoch 10/100, Loss: nan\n",
      "Epoch 11/100, Loss: nan\n",
      "Epoch 12/100, Loss: nan\n",
      "Epoch 13/100, Loss: nan\n",
      "Epoch 14/100, Loss: nan\n",
      "Epoch 15/100, Loss: nan\n",
      "Epoch 16/100, Loss: nan\n",
      "Epoch 17/100, Loss: nan\n",
      "Epoch 18/100, Loss: nan\n",
      "Epoch 19/100, Loss: nan\n",
      "Epoch 20/100, Loss: nan\n",
      "Epoch 21/100, Loss: nan\n",
      "Epoch 22/100, Loss: nan\n",
      "Epoch 23/100, Loss: nan\n",
      "Epoch 24/100, Loss: nan\n",
      "Epoch 25/100, Loss: nan\n",
      "Epoch 26/100, Loss: nan\n",
      "Epoch 27/100, Loss: nan\n",
      "Epoch 28/100, Loss: nan\n",
      "Epoch 29/100, Loss: nan\n",
      "Epoch 30/100, Loss: nan\n",
      "Epoch 31/100, Loss: nan\n",
      "Epoch 32/100, Loss: nan\n",
      "Epoch 33/100, Loss: nan\n",
      "Epoch 34/100, Loss: nan\n",
      "Epoch 35/100, Loss: nan\n",
      "Epoch 36/100, Loss: nan\n",
      "Epoch 37/100, Loss: nan\n",
      "Epoch 38/100, Loss: nan\n",
      "Epoch 39/100, Loss: nan\n",
      "Epoch 40/100, Loss: nan\n",
      "Epoch 41/100, Loss: nan\n",
      "Epoch 42/100, Loss: nan\n",
      "Epoch 43/100, Loss: nan\n",
      "Epoch 44/100, Loss: nan\n",
      "Epoch 45/100, Loss: nan\n",
      "Epoch 46/100, Loss: nan\n",
      "Epoch 47/100, Loss: nan\n",
      "Epoch 48/100, Loss: nan\n",
      "Epoch 49/100, Loss: nan\n",
      "Epoch 50/100, Loss: nan\n",
      "Epoch 51/100, Loss: nan\n",
      "Epoch 52/100, Loss: nan\n",
      "Epoch 53/100, Loss: nan\n",
      "Epoch 54/100, Loss: nan\n",
      "Epoch 55/100, Loss: nan\n",
      "Epoch 56/100, Loss: nan\n",
      "Epoch 57/100, Loss: nan\n",
      "Epoch 58/100, Loss: nan\n",
      "Epoch 59/100, Loss: nan\n",
      "Epoch 60/100, Loss: nan\n",
      "Epoch 61/100, Loss: nan\n",
      "Epoch 62/100, Loss: nan\n",
      "Epoch 63/100, Loss: nan\n",
      "Epoch 64/100, Loss: nan\n",
      "Epoch 65/100, Loss: nan\n",
      "Epoch 66/100, Loss: nan\n",
      "Epoch 67/100, Loss: nan\n",
      "Epoch 68/100, Loss: nan\n",
      "Epoch 69/100, Loss: nan\n",
      "Epoch 70/100, Loss: nan\n",
      "Epoch 71/100, Loss: nan\n",
      "Epoch 72/100, Loss: nan\n",
      "Epoch 73/100, Loss: nan\n",
      "Epoch 74/100, Loss: nan\n",
      "Epoch 75/100, Loss: nan\n",
      "Epoch 76/100, Loss: nan\n",
      "Epoch 77/100, Loss: nan\n",
      "Epoch 78/100, Loss: nan\n",
      "Epoch 79/100, Loss: nan\n",
      "Epoch 80/100, Loss: nan\n",
      "Epoch 81/100, Loss: nan\n",
      "Epoch 82/100, Loss: nan\n",
      "Epoch 83/100, Loss: nan\n",
      "Epoch 84/100, Loss: nan\n",
      "Epoch 85/100, Loss: nan\n",
      "Epoch 86/100, Loss: nan\n",
      "Epoch 87/100, Loss: nan\n",
      "Epoch 88/100, Loss: nan\n",
      "Epoch 89/100, Loss: nan\n",
      "Epoch 90/100, Loss: nan\n",
      "Epoch 91/100, Loss: nan\n",
      "Epoch 92/100, Loss: nan\n",
      "Epoch 93/100, Loss: nan\n",
      "Epoch 94/100, Loss: nan\n",
      "Epoch 95/100, Loss: nan\n",
      "Epoch 96/100, Loss: nan\n",
      "Epoch 97/100, Loss: nan\n",
      "Epoch 98/100, Loss: nan\n",
      "Epoch 99/100, Loss: nan\n",
      "Epoch 100/100, Loss: nan\n",
      "Test Loss: nan\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "from torch.utils.data import Dataset, DataLoader\n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "from sklearn.preprocessing import StandardScaler, OneHotEncoder\n",
    "from sklearn.compose import ColumnTransformer\n",
    "from sklearn.pipeline import Pipeline\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "\n",
    "# Bruun/Dean inspired model comment\n",
    "# Bruun: Δy = S * (ΔSL) / (L + ΔSL/S)\n",
    "# Dean: y = A * x^(2/3) (empirical beach profile)\n",
    "# We'll provide A * x^(2/3) as an additional feature.\n",
    "\n",
    "def dean_feature(x):\n",
    "    return np.power(x, 2 / 3)\n",
    "\n",
    "\n",
    "# Custom Dataset\n",
    "class BeachProfileDataset(Dataset):\n",
    "    def __init__(self, dataframe, preprocessor=None, fit_preprocessor=False):\n",
    "        self.df = dataframe.copy()\n",
    "        self.x_raw = self.df['x'].values.reshape(-1, 1)\n",
    "        self.y = self.df['y'].values.astype(np.float32)\n",
    "        self.df['dean_feature'] = dean_feature(self.df['x'])\n",
    "\n",
    "        self.feature_cols = [col for col in self.df.columns if col != 'y']\n",
    "        self.X = self.df[self.feature_cols]\n",
    "\n",
    "        if preprocessor is None:\n",
    "            categorical_cols = self.X.select_dtypes(include='object').columns.tolist()\n",
    "            numerical_cols = self.X.select_dtypes(include=['int64', 'float64']).columns.tolist()\n",
    "\n",
    "            self.preprocessor = ColumnTransformer([\n",
    "                ('num', StandardScaler(), numerical_cols),\n",
    "                ('cat', OneHotEncoder(sparse_output=False, handle_unknown='ignore'), categorical_cols)\n",
    "            ])\n",
    "        else:\n",
    "            self.preprocessor = preprocessor\n",
    "\n",
    "        if fit_preprocessor:\n",
    "            self.X = self.preprocessor.fit_transform(self.X)\n",
    "        else:\n",
    "            self.X = self.preprocessor.transform(self.X)\n",
    "\n",
    "        self.X = torch.tensor(self.X, dtype=torch.float32)\n",
    "        self.y = torch.tensor(self.y, dtype=torch.float32).view(-1, 1)\n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self.y)\n",
    "\n",
    "    def __getitem__(self, idx):\n",
    "        return self.X[idx], self.y[idx]\n",
    "\n",
    "\n",
    "# Neural Network Model\n",
    "class BeachNet(nn.Module):\n",
    "    def __init__(self, input_dim):\n",
    "        super(BeachNet, self).__init__()\n",
    "        self.model = nn.Sequential(\n",
    "            nn.Linear(input_dim, 64),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(64, 32),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(32, 1)\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        return self.model(x)\n",
    "\n",
    "\n",
    "# Load Data\n",
    "train_df = pd.read_excel(\"beach_profile_data/processed_data/beachdata_train.xlsx\")\n",
    "test_df = pd.read_excel(\"beach_profile_data/processed_data/beachdata_test.xlsx\")\n",
    "train_df = train_df.dropna()\n",
    "test_df = test_df.dropna()\n",
    "\n",
    "# Initialize Datasets\n",
    "train_dataset = BeachProfileDataset(train_df, fit_preprocessor=True)\n",
    "test_dataset = BeachProfileDataset(test_df, preprocessor=train_dataset.preprocessor)\n",
    "\n",
    "train_loader = DataLoader(train_dataset, batch_size=32, shuffle=True)\n",
    "test_loader = DataLoader(test_dataset, batch_size=32, shuffle=False)\n",
    "\n",
    "# Model Initialization\n",
    "input_dim = train_dataset.X.shape[1]\n",
    "model = BeachNet(input_dim)\n",
    "\n",
    "# Training Components\n",
    "criterion = nn.MSELoss()\n",
    "optimizer = torch.optim.Adam(model.parameters(), lr=0.001)\n",
    "\n",
    "# Training Loop\n",
    "def train_model(model, loader, optimizer, criterion, epochs=100):\n",
    "    model.train()\n",
    "    for epoch in range(epochs):\n",
    "        epoch_loss = 0.0\n",
    "        for inputs, targets in loader:\n",
    "            optimizer.zero_grad()\n",
    "            outputs = model(inputs)\n",
    "            loss = criterion(outputs, targets)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            epoch_loss += loss.item()\n",
    "        print(f\"Epoch {epoch + 1}/{epochs}, Loss: {epoch_loss / len(loader):.4f}\")\n",
    "\n",
    "# Evaluation Loop\n",
    "def evaluate_model(model, loader, criterion):\n",
    "    model.eval()\n",
    "    total_loss = 0.0\n",
    "    with torch.no_grad():\n",
    "        for inputs, targets in loader:\n",
    "            outputs = model(inputs)\n",
    "            loss = criterion(outputs, targets)\n",
    "            total_loss += loss.item()\n",
    "    print(f\"Test Loss: {total_loss / len(loader):.4f}\")\n",
    "\n",
    "\n",
    "# Train and Evaluate\n",
    "train_model(model, train_loader, optimizer, criterion, epochs=100)\n",
    "evaluate_model(model, test_loader, criterion)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 60,
   "id": "f4263a6b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Plot saved as cumulative_runnable_code.png\n"
     ]
    },
    {
     "data": {
      "image/png": "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NlEHKkafz8jIUHm+fHHXvsrCzs5OvuoIAMyfP1+thFrdaSWVLad59eqVSveGxcfHF3sNrAzvWVU+WQfyViJR7PC//PJLob8QxQRVlcSlKnwsm09xzlZJN9tIpVJcvHhRrfbzd5YrieKNHq1atVLrHPkU719QN05NderUSemmG39/f62NVij+ta/qzUaK5d58PZ2dnZXeCN/c3bAwly5dKvGjWMUbfsr79S8tdfp/UlISbt68WWKbZfHRumIfuH//vtJSoUVR/B04ODhoNF89n4uLi/zG0qysLOzatQsA8OjRI/mSa2ZmZiUugVYcOzs7fPbZZ9i3b5/82J07d/DkyRMNItc+dX/Purw+KVKnzwNQWja2KGXR552dnVGrVi35Y1WuffHx8fjnn3/kj0v7XlIRfPHFF/JlgO/du6f0x3BJ1MlpsrOzlV6zykAURZVyC8W8wt7evsA1sDK8Z1WLZB3I6/D5H31kZmbixx9/LFBGcb5xaGhoiUmJLue/a5vimrwlfbS0d+9etf/6jIiIKHHZuPj4eKUllEpa6rEoAwYMkH+/evXqcp3nJwgCZs2aJX985coVrFixQu12cnNzCywxpThqf/369RITxvT0dGzfvr3Q+vkUX+MtW7aUGNfmzZtLLKP4+gcEBCiNaFRU6vT/devWqXSDmeJScNq6Ia1JkybyaSVSqVSlN+7169fLvy/t/6nCKN44mh+HYjwDBw5Ua45xUTp06KC0SsSLFy80blOb1P096/L6pEidPh8SEqLSvh9l0ecB5X67cePGEstv3LhRPkji5OSExo0bay2W8lajRg18+eWX8seLFi1S+bVVzGlKWm5z//79leJa/SZ137cKuwZWhvesapOs16xZE5988on88f/+978Cq400adJEntBHR0crrSv7pkOHDhW4saoyq1evnvz7/fv3F1kuLi4On3/+eanOMXPmzGJXmpk5c6b8P4mbm1upN6/68MMP5SMR165dw6JFi1SuGx8fr/F6z6NHj1a6a3/27NlKG9mUJDw8HD179iywpJu7u7vSMnnTpk0r9qI9b948+ZJnlpaWhd7op7gG986dO4sdPbt48WKhGyu9afDgwWjQoAGAvP9HH3/8scoJSWpqqlrzUrVF1f7/8OFDlfuT4lJvkZGRpQ9OgSAImDx5svzx4sWLi217//79Stepjz76SCtxAHn7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uhT5nLOzc7HPl7XU1FQ8evRI/jgsLAw3btyAtbU1XF1dMWfOHERGRmLz5s0AgI8++gi//PILZs2ahfHjx+PUqVPYuXMnDh06pKsfgYiIiIhIicZLNy5YsAAA4O3trXEwmrh69Sq6desmf5w/XWXs2LHYuHEjoqOj5ctEAkDdunVx6NAhfP7551i5ciVq166NdevWcdlGIiIiIqowtJas61rXrl2LnTdf2O6kXbt2xfXr18swKiIiIiKi0qt869cQEREREVUTTNaJiIiIiCoolabBLF68WOnx/Pnzi3xOE4rtEhERERFVdyol6wsXLlRa7kYxqX7zOU0wWSciIiIiek3lG0zzb94sagdTTVXWtS+JiIiIiMqKSsl6cSu+VJTVYIiIiIiIqhom60REREREFRRXgyEiIiIiqqCYrBMRERERVVBM1omIiIiIKigm60REREREFZRKN5ieO3eurOMAAHTu3LlczkNEREREVBmolKx37dq1zNdBFwQBubm5ZXoOIiIiIqLKRO1NkYiIiIiIqHyolKx37tyZO4wSEREREZUzlZL1M2fOlHEYRERERET0Jq4GQ0RERERUQTFZJyIiIiKqoJisExERERFVUEzWiYiIiIgqKCbrREREREQVlEqrwejr68u/f3PzIsXnNMFNkYiIiIiqNqlMRHBYAkLiBdiEJcC3QS3o63F58OKolKyLoghBEArdGKm454iIiIiIAODo7WgsOnAX0UmZAPSx+eFVOFoZY4GfB/o2c9R1eBWWytNgikvGmagTERERUVGO3o7GlK3X/k3UX4tJysSUrddw9Ha0jiKr+FRK1mUymfxLKpUW+ZwmX2+2S0RERESVn1QmYtGBuyhsaDf/2KIDdyGVcfC3MCpNgyEiIiIiKk56di6iEjMRnZSB6MRMRP37793opAIj6opEANFJmbgclgDf+jblF3AlwWSdiIiIiIqVnSvDi+RMRCVmICop442kPO/7xPQcjc4Rm1J0Ql+dMVknIiIiqsakMhFxKVnykfDopAxEJr7+PiopE/GpWVDlFkVziQEcrYzhWMMETlbGcLQyQUZ2Ltace1Ji3VoWxlr4aaoeJutEREREVZQoikhIy0Z0Ut6oeHTS6+kp+Y9fJGciV4X54kYGevIE3LGGMZwU/nWqkfe9pbFhgXpSmYh9oVGIScosdN66AMDByhjeda01/4Gr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KZzHxMQgJiamBaIiIiIiItKckeoiqs2cOROJiYkoKirCoEGD8M9//hM///xzq5/QT0RERESkT1qPrMtkMoVzURTx4Ycf4sMPP9SoHUEQUFlZqW04RERERERthtbJuiiKEARB+q8gCAqvERERERFR02idrAP/S8qZnBMRERER6Y7ONkUiIiIiIiLd0skDpkREREREpHtM1omIiIiIDBSTdSIiIiIiA8VknYiIiIjIQOlkNZiGlJWVobi4GHK5XO06Hh4ezRUOEREREVGro7Nkvbq6Gtu2bcPnn3+OkydPoqCgQKP63BSJiIiIiEiRTpL17OxshIaG4ty5cwC43joRERERkS5onazfv38fzz77LLKyshSuW1hYwN7eHiYmJtregoiIiIjosaR1sr5u3TpkZWVBEAQYGxvjzTffxKuvvgofHx9dxEdERERE9NjSOlnfvXu3dLxt2za8+OKL2jZJRERERETQwdKNV65cgSAI8PPzY6JORERERKRDWifr5eXlAABfX19tmyIiIiIiojq0TtY7deoEABqtp05ERERERKppnawHBgZCFEVp2UYiIiIiItINrZP1mTNnwsjICBkZGTh16pQuYiIiIiIiIuggWR8wYADeeecdiKKIv/71r7h9+7Yu4iIiIiIieuxpnawDwNKlSxEdHY3MzEz07dsX69atw61bt3TRNBERERHRY0vrdda7du0qHZuYmCA/Px+RkZGIjIyEnZ0d7OzsIAiCynYEQcDVq1e1DYeIiIiIqM3QOlnPzs5WSMZrj0VRRGFhIYqKilS2IYqiWgk9EREREdHjROtkHahJtpvyGhERERERNU7rZD0rK0sXcRARERER0SO0TtY9PT11EQcRERERET1CJ6vBEBERERGR7jFZJyIiIiIyUEzWiYiIiIgMFJN1IiIiIiIDpdNNkbTBTZGIiIiIiBTpfFMkddRde10QBG6KRERERETUgGbfFKkxtUk6N00iIiIiImpYi22KVF1djaKiIpw7dw47d+7Evn37YG5ujvj4eDzzzDPahkFERERE1Oa0+KZIvr6+mDJlCvbt24eJEydi5syZ2LlzJ55//nltQyEiIiIialP0thrM2LFjkZCQALlcjmnTpuHmzZv6CoWIiIiIyCDpdenGKVOmwMPDA8XFxdiwYYNO2oyPj0fnzp1hbm6OgIAAnDhxotGymzZtwtCh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",
      "text/plain": [
       "<Figure size 800x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "\n",
    "# Data from the previous analysis\n",
    "iterations = [1, 2, 3, 4, 5]\n",
    "cumulative_runnable_code = [10/30, 16/30, 24/30, 27/30, 28/30]\n",
    "\n",
    "# Create the plot\n",
    "plt.figure(figsize=(8, 5)) # Adjust figure size for better readability\n",
    "plt.plot(iterations, cumulative_runnable_code, marker='o', linestyle='-')\n",
    "\n",
    "# Add title and labels\n",
    "plt.title('Runnable Code ratio vs. Iteration Number', fontsize=24, pad=25)\n",
    "plt.xlabel('Iteration Number', fontsize=24)\n",
    "plt.ylabel('Runnable Code probability', fontsize=24)\n",
    "\n",
    "# Add grid for better readability\n",
    "plt.grid(True)\n",
    "\n",
    "# Set x-axis ticks to be integers\n",
    "plt.xticks(iterations)\n",
    "# Set y-axis ticks to start from 0 and have reasonable intervals\n",
    "plt.yticks(np.arange(0.2, 1.1, 0.1)) \n",
    "\n",
    "print(\"Plot saved as cumulative_runnable_code.png\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "72bf7646",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np # Often useful, though not strictly necessary for this basic plot\n",
    "\n",
    "# --- 1. Data (Replace with your actual data) ---\n",
    "iterations = [1, 2, 3, 4, 5]\n",
    "human_eval_data = [64.6, 82.9, 88.4, 90.2, 90.9]\n",
    "human_eval_et_data = [52.4, 76.2, 84.1, 87.8, 89.6]\n",
    "mbpp_data = [63.9, 85.5, 91.1, 93.7, 93.0]\n",
    "mbpp_et_data = [52.2, 78.5, 86.4, 89.0, 90.4]\n",
    "\n",
    "# --- 2. Create the Plot ---\n",
    "plt.figure(figsize=(10, 6)) # Adjust figure size as needed\n",
    "\n",
    "# Plot each data series\n",
    "plt.plot(iterations, human_eval_data, label='HumanEval', color='blue', linestyle='-', marker='o', linewidth=2, markersize=8)\n",
    "plt.plot(iterations, human_eval_et_data, label='HumanEval-ET', color='red', linestyle='--', marker='s', linewidth=2, markersize=8)\n",
    "plt.plot(iterations, mbpp_data, label='MBPP', color='green', linestyle='-.', marker='^', linewidth=2, markersize=8)\n",
    "plt.plot(iterations, mbpp_et_data, label='MBPP-ET', color='magenta', linestyle=':', marker='d', linewidth=2, markersize=8)\n",
    "\n",
    "# --- 3. Customize the Plot ---\n",
    "# Title and Labels\n",
    "plt.title('Pass@1 of LCP with Different Number of Iterations on GPT-3.5-turbo', fontsize=18)\n",
    "plt.xlabel('Iterations', fontsize=16)\n",
    "plt.ylabel('Pass@1 (%)', fontsize=16)\n",
    "\n",
    "# X and Y axis ticks and limits\n",
    "plt.xticks(iterations, fontsize=14) # Ensure ticks are at 1, 2, 3, 4, 5\n",
    "plt.yticks(np.arange(45, 101, 5), fontsize=14) # Y-axis ticks from 55 to 100 with a step of 5\n",
    "plt.ylim(45, 100) # Set Y-axis limits\n",
    "\n",
    "# Legend\n",
    "plt.legend(fontsize=16)\n",
    "\n",
    "# Grid\n",
    "plt.grid(True, linestyle='--', alpha=0.7) # Light gray dashed grid\n",
    "\n",
    "# Improve layout\n",
    "plt.tight_layout()\n",
    "\n",
    "# --- 4. Show or Save the Plot ---\n",
    "plt.show()\n",
    "# To save the figure:\n",
    "# plt.savefig('pass1_lcp_graph.png', dpi=300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "02f33886",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import numpy as np # Often useful, though not strictly necessary for this basic plot\n",
    "\n",
    "# --- 1. Data (Replace with your actual data) ---\n",
    "iterations = [1, 2, 3, 4, 5]\n",
    "human_eval_data = [64.6, 82.9, 88.4, 90.2, 90.9]\n",
    "human_eval_et_data = [52.4, 76.2, 84.1, 87.8, 89.6]\n",
    "mbpp_data = [63.9, 85.5, 91.1, 93.7, 93.0]\n",
    "mbpp_et_data = [52.2, 78.5, 86.4, 89.0, 90.4]\n",
    "\n",
    "# --- 2. Create the Plot ---\n",
    "plt.figure(figsize=(10, 6)) # Adjust figure size as needed\n",
    "\n",
    "# Plot each data series\n",
    "plt.plot(iterations, human_eval_data, label='HumanEval', color='blue', linestyle='-', marker='o', linewidth=2, markersize=8)\n",
    "plt.plot(iterations, human_eval_et_data, label='HumanEval-ET', color='red', linestyle='--', marker='s', linewidth=2, markersize=8)\n",
    "plt.plot(iterations, mbpp_data, label='MBPP', color='green', linestyle='-.', marker='^', linewidth=2, markersize=8)\n",
    "plt.plot(iterations, mbpp_et_data, label='MBPP-ET', color='magenta', linestyle=':', marker='d', linewidth=2, markersize=8)\n",
    "\n",
    "# --- 3. Customize the Plot ---\n",
    "# Title and Labels\n",
    "# Increase fontsize for the title (e.g., from 18 to 22)\n",
    "plt.title('Pass@1 of LCP with Different Number of Iterations on GPT-3.5-turbo', fontsize=20)\n",
    "# Increase fontsize for the x-axis label (e.g., from 16 to 20)\n",
    "plt.xlabel('Iterations', fontsize=30)\n",
    "# Increase fontsize for the y-axis label (e.g., from 16 to 20)\n",
    "plt.ylabel('Pass@1 (%)', fontsize=30)\n",
    "\n",
    "# X and Y axis ticks and limits\n",
    "# You can also increase tick label font sizes if needed\n",
    "plt.xticks(iterations, fontsize=16) # Increased from 14 to 16\n",
    "plt.yticks(np.arange(45, 101, 5), fontsize=16) # Increased from 14 to 16\n",
    "plt.ylim(45, 100) # Set Y-axis limits\n",
    "\n",
    "# Legend\n",
    "# Increase fontsize for the legend (e.g., from 16 to 20)\n",
    "plt.legend(fontsize=30)\n",
    "\n",
    "# Grid\n",
    "plt.grid(True, linestyle='--', alpha=0.7) # Light gray dashed grid\n",
    "\n",
    "# Improve layout\n",
    "plt.tight_layout()\n",
    "\n",
    "# --- 4. Show or Save the Plot ---\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e70d1aa7",
   "metadata": {},
   "source": [
    "### baseline test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "id": "871a5fb3",
   "metadata": {},
   "outputs": [],
   "source": [
    "import re\n",
    "import traceback\n",
    "\n",
    "def extract_python_code_simplified(llm_output):\n",
    "    # Try to find ```python ... ``` block\n",
    "    python_block_match = re.search(r\"```python\\n(.*?)\\n```\", llm_output, re.DOTALL)\n",
    "    if python_block_match:\n",
    "        return python_block_match.group(1).strip()\n",
    "    \n",
    "    # Try to find generic ``` ... ``` block\n",
    "    generic_block_match = re.search(r\"```\\n(.*?)\\n```\", llm_output, re.DOTALL)\n",
    "    if generic_block_match:\n",
    "        return generic_block_match.group(1).strip()\n",
    "        \n",
    "    # If no blocks found, assume the entire output is code\n",
    "    return llm_output.strip()\n",
    "\n",
    "def check_code_runnability_simplified(python_code):\n",
    "    if not python_code:\n",
    "        return False, \"No Python code provided to check.\"\n",
    "\n",
    "    execution_scope = {}\n",
    "    \n",
    "    try:\n",
    "        # Compile and execute the script's top-level code\n",
    "        # This will define functions, including main(), in execution_scope\n",
    "        compiled_code = compile(python_code, '<string>', 'exec')\n",
    "        exec(compiled_code, execution_scope)\n",
    "    except SyntaxError as e:\n",
    "        return False, f\"Syntax Error: {e.msg} (line {e.lineno}, offset {e.offset})\"\n",
    "    except Exception as e:\n",
    "        # Errors during the initial exec (e.g., import errors at top level, issues in global code)\n",
    "        return False, f\"Error during script execution (outside main): {type(e).__name__}: {e}\\n{traceback.format_exc()}\"\n",
    "\n",
    "    # Check if main() was defined\n",
    "    if \"main\" not in execution_scope:\n",
    "        return False, \"Code does not define a main() function.\"\n",
    "    \n",
    "    main_func = execution_scope[\"main\"]\n",
    "    \n",
    "    if not callable(main_func):\n",
    "        return False, \"'main' is defined but is not a callable function.\"\n",
    "\n",
    "    # Call the main function\n",
    "    try:\n",
    "        main_func()\n",
    "    except Exception as e:\n",
    "        return False, f\"Error during main() execution: {type(e).__name__}: {e}\\n{traceback.format_exc()}\"\n",
    "    \n",
    "    return True, \"Code is syntactically correct and main() executed successfully.\"\n",
    "\n",
    "def extract_and_test_llm_code_simplified(llm_output):\n",
    "    extracted_code = extract_python_code_simplified(llm_output)\n",
    "    \n",
    "    # Since extract_python_code_simplified always returns a string,\n",
    "    # we don't need to check if extracted_code is None (unless it's empty).\n",
    "    if not extracted_code: # Handles case where llm_output itself was empty or only whitespace\n",
    "        return False, \"No Python code found or extracted.\", \"\"\n",
    "    \n",
    "    is_runnable, message = check_code_runnability_simplified(extracted_code)\n",
    "    return is_runnable, message, extracted_code"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "3f628cee",
   "metadata": {},
   "outputs": [],
   "source": [
    "# All other relevant columns you might select as features are expected to be in numerical (integer or float) or categorical (string) format.\n",
    "\n",
    "base_prompt = \"\"\"Please refer to the Bruun model and the Dean model to build a mathematical model to discuss the change of sea level height with respect to the distance from the starting point, and build a deep learning model based on our data. The following is our data path and structure:\n",
    "\n",
    "Data File Paths:\n",
    "\n",
    "Training Data: \"beach_profile_data/processed_data/beachdata_train.xlsx\"\n",
    "Test Data: \"beach_profile_data/processed_data/beachdata_test.xlsx\"\n",
    "Data Structure Insights:\n",
    "The training and testing datasets contain several columns. Key columns include:\n",
    "\n",
    "x: A numerical column representing the distance from a profile's origin; this is a primary input for predicting y.\n",
    "y: A numerical column representing the elevation; this is the target variable for prediction.\n",
    "All other relevant columns you might select as features are expected to be in numerical (integer or float) or categorical (string) format.\n",
    "\n",
    "create the code directly and The script should not run the main training or evaluation logic directly when the script file is executed. Instead, it should define all necessary functions with a main function main() that can be run WITHOUT ANY input. DO NOT USE tensorflow. Use provided data path in your code.\"\"\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "3520674f",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test Loss: 0.8933709357395645\n",
      "Test Loss: 0.6456502918154001\n",
      "Epoch 0, Loss: 2.3373\n",
      "Epoch 1, Loss: 1.5680\n",
      "Epoch 2, Loss: 1.0930\n",
      "Epoch 3, Loss: 1.4445\n",
      "Epoch 4, Loss: 0.8948\n",
      "Epoch 5, Loss: 0.9418\n",
      "Epoch 6, Loss: 0.9090\n",
      "Epoch 7, Loss: 0.7271\n",
      "Epoch 8, Loss: 0.8081\n",
      "Epoch 9, Loss: 0.7366\n",
      "Epoch 10, Loss: 0.7620\n",
      "Epoch 11, Loss: 0.8191\n",
      "Epoch 12, Loss: 0.7962\n",
      "Epoch 13, Loss: 0.7093\n",
      "Epoch 14, Loss: 0.8877\n",
      "Epoch 15, Loss: 0.8819\n",
      "Epoch 16, Loss: 0.8429\n",
      "Epoch 17, Loss: 0.7975\n",
      "Epoch 18, Loss: 0.6852\n",
      "Epoch 19, Loss: 0.7418\n",
      "Epoch 20, Loss: 0.8705\n",
      "Epoch 21, Loss: 0.7138\n",
      "Epoch 22, Loss: 0.6889\n",
      "Epoch 23, Loss: 0.6921\n",
      "Epoch 24, Loss: 0.7269\n",
      "Epoch 25, Loss: 0.6661\n",
      "Epoch 26, Loss: 0.7310\n",
      "Epoch 27, Loss: 0.6847\n",
      "Epoch 28, Loss: 0.6812\n",
      "Epoch 29, Loss: 0.7103\n",
      "Epoch 30, Loss: 0.6840\n",
      "Epoch 31, Loss: 0.6819\n",
      "Epoch 32, Loss: 0.7459\n",
      "Epoch 33, Loss: 0.6929\n",
      "Epoch 34, Loss: 0.7093\n",
      "Epoch 35, Loss: 0.6916\n",
      "Epoch 36, Loss: 0.7129\n",
      "Epoch 37, Loss: 0.7082\n",
      "Epoch 38, Loss: 0.7124\n",
      "Epoch 39, Loss: 0.6713\n",
      "Epoch 40, Loss: 0.6819\n",
      "Epoch 41, Loss: 0.6715\n",
      "Epoch 42, Loss: 0.6749\n",
      "Epoch 43, Loss: 0.6833\n",
      "Epoch 44, Loss: 0.6719\n",
      "Epoch 45, Loss: 0.6722\n",
      "Epoch 46, Loss: 0.6750\n",
      "Epoch 47, Loss: 0.6854\n",
      "Epoch 48, Loss: 0.6883\n",
      "Epoch 49, Loss: 0.6689\n",
      "Epoch 50, Loss: 0.6707\n",
      "Epoch 51, Loss: 0.6620\n",
      "Epoch 52, Loss: 0.6748\n",
      "Epoch 53, Loss: 0.6729\n",
      "Epoch 54, Loss: 0.6575\n",
      "Epoch 55, Loss: 0.6669\n",
      "Epoch 56, Loss: 0.6624\n",
      "Epoch 57, Loss: 0.6593\n",
      "Epoch 58, Loss: 0.6644\n",
      "Epoch 59, Loss: 0.6580\n",
      "Epoch 60, Loss: 0.6840\n",
      "Epoch 61, Loss: 0.6595\n",
      "Epoch 62, Loss: 0.6572\n",
      "Epoch 63, Loss: 0.6568\n",
      "Epoch 64, Loss: 0.6734\n",
      "Epoch 65, Loss: 0.6600\n",
      "Epoch 66, Loss: 0.6729\n",
      "Epoch 67, Loss: 0.6526\n",
      "Epoch 68, Loss: 0.6497\n",
      "Epoch 69, Loss: 0.6630\n",
      "Epoch 70, Loss: 0.6512\n",
      "Epoch 71, Loss: 0.6576\n",
      "Epoch 72, Loss: 0.6546\n",
      "Epoch 73, Loss: 0.6680\n",
      "Epoch 74, Loss: 0.6502\n",
      "Epoch 75, Loss: 0.6508\n",
      "Epoch 76, Loss: 0.6625\n",
      "Epoch 77, Loss: 0.6469\n",
      "Epoch 78, Loss: 0.6535\n",
      "Epoch 79, Loss: 0.6696\n",
      "Epoch 80, Loss: 0.6672\n",
      "Epoch 81, Loss: 0.6520\n",
      "Epoch 82, Loss: 0.6628\n",
      "Epoch 83, Loss: 0.6626\n",
      "Epoch 84, Loss: 0.6527\n",
      "Epoch 85, Loss: 0.6506\n",
      "Epoch 86, Loss: 0.6583\n",
      "Epoch 87, Loss: 0.6588\n",
      "Epoch 88, Loss: 0.6506\n",
      "Epoch 89, Loss: 0.6545\n",
      "Epoch 90, Loss: 0.6595\n",
      "Epoch 91, Loss: 0.6641\n",
      "Epoch 92, Loss: 0.6679\n",
      "Epoch 93, Loss: 0.6605\n",
      "Epoch 94, Loss: 0.6650\n",
      "Epoch 95, Loss: 0.6556\n",
      "Epoch 96, Loss: 0.6457\n",
      "Epoch 97, Loss: 0.6728\n",
      "Epoch 98, Loss: 0.6610\n",
      "Epoch 99, Loss: 0.6890\n",
      "Predictions: [array([1.6198033], dtype=float32), array([1.4871467], dtype=float32), array([1.3640963], dtype=float32), array([1.2471838], dtype=float32), array([1.152139], dtype=float32), array([1.0360154], dtype=float32), array([0.9035622], dtype=float32), array([0.768125], dtype=float32), array([0.6581423], dtype=float32), array([0.56027144], dtype=float32), array([0.46938512], dtype=float32), array([0.3913764], dtype=float32), array([0.34131193], dtype=float32), array([0.32737568], dtype=float32), array([0.30890185], dtype=float32), array([1.6198033], dtype=float32), array([1.5434114], dtype=float32), array([1.480285], dtype=float32), array([1.4107547], dtype=float32), array([1.3425965], dtype=float32), array([1.2874246], dtype=float32), array([1.2264885], dtype=float32), array([1.1678519], dtype=float32), array([1.1072992], dtype=float32), array([1.0375484], dtype=float32), array([0.97277987], dtype=float32), array([0.903173], dtype=float32), array([0.8300058], dtype=float32), array([0.761898], dtype=float32), array([0.6901959], dtype=float32), array([0.63521767], dtype=float32), array([0.55711305], dtype=float32), array([0.50033283], dtype=float32), array([0.4363913], dtype=float32), array([0.36400935], dtype=float32), array([0.32038593], dtype=float32), array([0.25383806], dtype=float32), array([0.16214882], dtype=float32), array([0.10789411], dtype=float32), array([1.6198033], dtype=float32), array([1.5200821], dtype=float32), array([1.3234495], dtype=float32), array([1.2349198], dtype=float32), array([0.5036578], dtype=float32), array([0.5008443], dtype=float32), array([0.46785048], dtype=float32), array([0.42974123], dtype=float32), array([0.4115819], dtype=float32), array([0.37321696], dtype=float32), array([0.33058324], dtype=float32), array([0.11567609], dtype=float32), array([0.04849366], dtype=float32), array([-0.02624603], dtype=float32), array([1.6198033], dtype=float32), array([1.5141352], dtype=float32), array([1.4167016], dtype=float32), array([1.3261323], dtype=float32), array([1.2238058], dtype=float32), array([1.1145809], dtype=float32), array([0.9638862], dtype=float32), array([0.7833033], dtype=float32), array([0.61346865], dtype=float32), array([0.44790074], dtype=float32), array([0.2836838], dtype=float32), array([0.16664271], dtype=float32), array([0.06565927], dtype=float32), array([-0.06098338], dtype=float32), array([-0.23580058], dtype=float32), array([-0.36009905], dtype=float32), array([-0.45400319], dtype=float32), array([-0.43057796], dtype=float32), array([-0.3976209], dtype=float32), array([-0.35805163], dtype=float32), array([-0.29762807], dtype=float32), array([-0.2342601], dtype=float32), array([-0.17571737], dtype=float32), array([-0.13949086], dtype=float32), array([-0.12119426], dtype=float32), array([1.6198033], dtype=float32), array([1.5438688], dtype=float32), array([1.4761682], dtype=float32), array([1.3851382], dtype=float32), array([1.2809092], dtype=float32), array([1.1617199], dtype=float32), array([1.0942688], dtype=float32), array([0.98427725], dtype=float32), array([0.8537462], dtype=float32), array([0.68283045], dtype=float32), array([0.56027144], dtype=float32), array([0.46145633], dtype=float32), array([0.37500724], dtype=float32), array([0.2939406], dtype=float32), array([0.25049138], dtype=float32), array([1.6198033], dtype=float32), array([1.5626236], dtype=float32), array([1.5008698], dtype=float32), array([1.4386584], dtype=float32), array([1.3782768], dtype=float32), array([1.3200004], dtype=float32), array([1.2697952], dtype=float32), array([1.210009], dtype=float32), array([1.1379588], dtype=float32), array([1.0793222], dtype=float32), array([0.9877264], dtype=float32), array([0.8541353], dtype=float32), array([0.74166024], dtype=float32), array([0.5670314], dtype=float32), array([0.4550622], dtype=float32), array([0.32295695], dtype=float32), array([0.24942045], dtype=float32), array([0.17753302], dtype=float32), array([0.1311305], dtype=float32), array([0.08202718], dtype=float32), array([0.0243933], dtype=float32), array([-0.03542729], dtype=float32), array([-0.10327192], dtype=float32), array([-0.16565622], dtype=float32), array([-0.20742054], dtype=float32), array([-0.30584964], dtype=float32), array([-0.36043188], dtype=float32), array([1.6196204], dtype=float32), array([1.4700384], dtype=float32), array([1.3332651], dtype=float32), array([1.2213529], dtype=float32), array([1.0837679], dtype=float32), array([0.9860403], dtype=float32), array([0.7099805], dtype=float32), array([0.5293879], dtype=float32), array([0.3266071], dtype=float32), array([0.20947395], dtype=float32), array([0.1125633], dtype=float32), array([0.00654749], dtype=float32), array([-0.05718895], dtype=float32), array([-0.25057963], dtype=float32), array([-0.418084], dtype=float32), array([-0.44926116], dtype=float32), array([-0.41855463], dtype=float32), array([-0.34651634], dtype=float32), array([-0.274704], dtype=float32), array([-0.17892654], dtype=float32), array([-0.12360294], dtype=float32), array([-0.11273099], dtype=float32), array([-0.10998489], dtype=float32), array([1.6198033], dtype=float32), array([1.5338051], dtype=float32), array([1.3042873], dtype=float32), array([1.0444468], dtype=float32), array([0.59554034], dtype=float32), array([0.3123299], dtype=float32), array([0.17739902], dtype=float32), array([0.05853562], dtype=float32), array([-0.16216137], dtype=float32), array([-0.44716296], dtype=float32), array([-0.37793717], dtype=float32), array([-0.25944236], dtype=float32), array([-0.14448257], dtype=float32), array([-0.11932166], dtype=float32), array([-0.1025386], dtype=float32), array([1.6198033], dtype=float32), array([1.0505787], dtype=float32), array([0.545092], dtype=float32), array([0.10975735], dtype=float32), array([-0.350113], dtype=float32), array([-0.39767143], dtype=float32), array([-0.24383838], dtype=float32), array([-0.12514831], dtype=float32), array([-0.10256805], dtype=float32), array([-0.23066457], dtype=float32), array([-0.5636012], dtype=float32), array([-0.8136689], dtype=float32), array([-1.1177211], dtype=float32), array([-1.4642154], dtype=float32), array([-1.7676431], dtype=float32), array([-1.9472557], dtype=float32), array([-1.9557985], dtype=float32), array([-2.0289352], dtype=float32), array([1.6198033], dtype=float32), array([1.5100185], dtype=float32), array([1.4377435], dtype=float32), array([1.2544653], dtype=float32), array([0.6763646], dtype=float32), array([0.12849997], dtype=float32), array([-0.24993034], dtype=float32), array([-0.4222177], dtype=float32), array([-0.20157291], dtype=float32), array([-0.10825141], dtype=float32), array([-0.50493926], dtype=float32), array([-0.8810802], dtype=float32), array([-1.5201122], dtype=float32), array([-1.7095135], dtype=float32), array([1.6198033], dtype=float32), array([1.549358], dtype=float32), array([1.5086462], dtype=float32), array([1.4496368], dtype=float32), array([1.393372], dtype=float32), array([1.329331], dtype=float32), array([1.269412], dtype=float32), array([1.2000445], dtype=float32), array([1.124162], dtype=float32), array([1.0578604], dtype=float32), array([1.0042062], dtype=float32), array([0.9483188], dtype=float32), array([0.8934432], dtype=float32), array([0.82922745], dtype=float32), array([0.7700709], dtype=float32), array([0.6640204], dtype=float32), array([0.10329066], dtype=float32), array([-0.03614451], dtype=float32), array([1.6198033], dtype=float32), array([1.5489005], dtype=float32), array([1.452839], dtype=float32), array([1.3554049], dtype=float32), array([1.2510161], dtype=float32), array([1.1467733], dtype=float32), array([1.1076823], dtype=float32), array([0.9762292], dtype=float32), array([0.8444057], dtype=float32), array([0.69601583], dtype=float32), array([0.57085216], dtype=float32), array([0.44790074], dtype=float32), array([0.32647154], dtype=float32), array([0.22104026], dtype=float32), array([0.12959604], dtype=float32), array([0.02539758], dtype=float32), array([-0.13961907], dtype=float32), array([-0.42769763], dtype=float32), array([-0.41828558], dtype=float32), array([-0.24428226], dtype=float32), array([-0.14277305], dtype=float32), array([1.6198033], dtype=float32), array([1.5196247], dtype=float32), array([1.4148718], dtype=float32), array([1.3042873], dtype=float32), array([1.2149911], dtype=float32), array([1.1195629], dtype=float32), array([1.0298835], dtype=float32), array([0.72259], dtype=float32), array([0.21434681], dtype=float32), array([-0.21056594], dtype=float32), array([-0.42328033], dtype=float32), array([-0.29128507], dtype=float32), array([-0.15239336], dtype=float32)]\n",
      "Actuals: [array([2.5328531], dtype=float32), array([2.1941197], dtype=float32), array([1.9249394], dtype=float32), array([1.6349837], dtype=float32), array([1.3161227], dtype=float32), array([0.9584201], dtype=float32), array([0.62329984], dtype=float32), array([0.28185654], dtype=float32), array([-0.01171239], dtype=float32), array([-0.2772794], dtype=float32), array([-0.47961614], dtype=float32), array([-0.7298272], dtype=float32), array([-0.8824832], dtype=float32), array([-0.9574562], dtype=float32), array([-1.0369456], dtype=float32), array([2.66654], dtype=float32), array([2.4560735], dtype=float32), array([2.276319], dtype=float32), array([2.070369], dtype=float32), array([1.8906145], dtype=float32), array([1.6873745], dtype=float32), array([1.5049101], dtype=float32), array([1.3107028], dtype=float32), array([1.0948168], dtype=float32), array([0.8689945], dtype=float32), array([0.65852815], dtype=float32), array([0.43993214], dtype=float32), array([0.21139999], dtype=float32), array([0.00093363], dtype=float32), array([-0.19959654], dtype=float32), array([-0.4136761], dtype=float32), array([-0.55729896], dtype=float32), array([-0.7686686], dtype=float32), array([-1.0044272], dtype=float32), array([-1.3016093], dtype=float32), array([-1.4551684], dtype=float32), array([-1.7749327], dtype=float32), array([-1.825517], dtype=float32), array([-1.9492677], dtype=float32), array([2.588857], dtype=float32), array([2.3973598], dtype=float32), array([2.107404], dtype=float32), array([1.9411987], dtype=float32), array([0.44896504], dtype=float32), array([0.401994], dtype=float32), array([0.2954059], dtype=float32), array([0.19152765], dtype=float32), array([0.12287764], dtype=float32), array([-0.01442232], dtype=float32), array([-0.19417681], dtype=float32), array([-0.9845548], dtype=float32), array([-1.1137252], dtype=float32), array([-1.2311528], dtype=float32), array([2.6195688], dtype=float32), array([2.3675513], dtype=float32), array([2.1715374], dtype=float32), array([2.0152683], dtype=float32), array([1.798479], dtype=float32), array([1.6087883], dtype=float32), array([1.4335502], dtype=float32), array([1.267345], dtype=float32), array([1.1887589], dtype=float32), array([1.0713313], dtype=float32), array([0.89880306], dtype=float32), array([0.69917613], dtype=float32), array([0.4191565], dtype=float32), array([0.25566116], dtype=float32), array([0.11745797], dtype=float32), array([0.01628951], dtype=float32), array([-0.0912018], dtype=float32), array([-0.22940506], dtype=float32), array([-0.38657746], dtype=float32), array([-0.5464595], dtype=float32), array([-0.71898776], dtype=float32), array([-0.8490614], dtype=float32), array([-0.96377915], dtype=float32), array([-1.0450752], dtype=float32), array([-1.1218548], dtype=float32), array([2.8002267], dtype=float32), array([2.6051161], dtype=float32), array([2.4443307], dtype=float32), array([2.263673], dtype=float32), array([2.1372125], dtype=float32), array([1.9935895], dtype=float32), array([1.8951309], dtype=float32), array([1.7524112], dtype=float32), array([1.5979488], dtype=float32), array([1.3658035], dtype=float32), array([1.1607568], dtype=float32), array([0.9864221], dtype=float32), array([0.8427991], dtype=float32), array([0.6467854], dtype=float32), array([0.5167117], dtype=float32), array([2.7162209], dtype=float32), array([2.535563], dtype=float32), array([2.3621316], dtype=float32), array([2.1661177], dtype=float32), array([1.9728137], dtype=float32), array([1.798479], dtype=float32), array([1.6566626], dtype=float32), array([1.5130397], dtype=float32), array([1.3450278], dtype=float32), array([1.2221805], dtype=float32), array([1.1056563], dtype=float32), array([0.9620333], dtype=float32), array([0.811184], dtype=float32), array([0.54742354], dtype=float32), array([0.36134598], dtype=float32), array([0.11294149], dtype=float32), array([-0.0830722], dtype=float32), array([-0.30076495], dtype=float32), array([-0.42270896], dtype=float32), array([-0.5708484], dtype=float32), array([-0.7180845], dtype=float32), array([-0.8490614], dtype=float32), array([-0.98816794], dtype=float32), array([-1.1209514], dtype=float32), array([-1.181472], dtype=float32), array([-1.3540002], dtype=float32), array([-1.4642013], dtype=float32), array([2.8047433], dtype=float32), array([2.3747776], dtype=float32), array([2.032431], dtype=float32), array([1.7406685], dtype=float32), array([1.3847727], dtype=float32), array([1.3043798], dtype=float32), array([1.2339233], dtype=float32), array([1.1128826], dtype=float32), array([0.98100233], dtype=float32), array([0.8157004], dtype=float32), array([0.65672153], dtype=float32), array([0.5103887], dtype=float32), array([0.42367294], dtype=float32), array([0.19514082], dtype=float32), array([-0.05507024], dtype=float32), array([-0.27908602], dtype=float32), array([-0.4967787], dtype=float32), array([-0.83009225], dtype=float32), array([-1.0432687], dtype=float32), array([-1.3269013], dtype=float32), array([-1.746931], dtype=float32), array([-1.9872059], dtype=float32), array([-2.0513391], dtype=float32), array([2.6457644], dtype=float32), array([2.395553], dtype=float32), array([2.226638], dtype=float32), array([2.1019843], dtype=float32), array([1.9366822], dtype=float32), array([1.723506], dtype=float32), array([1.592529], dtype=float32), array([1.558204], dtype=float32), array([1.3947088], dtype=float32), array([1.130045], dtype=float32), array([0.68382025], dtype=float32), array([0.25927433], dtype=float32), array([-0.23482485], dtype=float32), array([-0.5834945], dtype=float32), array([-0.9935877], dtype=float32), array([2.6024063], dtype=float32), array([2.388327], dtype=float32), array([2.216702], dtype=float32), array([1.907777], dtype=float32), array([1.5464613], dtype=float32), array([1.2158575], dtype=float32), array([0.8915767], dtype=float32), array([0.5275512], dtype=float32), array([0.16262238], dtype=float32), array([-0.20591958], dtype=float32), array([-0.5518792], dtype=float32), array([-0.8373187], dtype=float32), array([-1.1019825], dtype=float32), array([-1.3133521], dtype=float32), array([-1.5581435], dtype=float32), array([-1.688217], dtype=float32), array([-1.7586738], dtype=float32), array([-1.8788112], dtype=float32), array([2.7460294], dtype=float32), array([2.567178], dtype=float32), array([2.410909], dtype=float32), array([2.1877966], dtype=float32), array([1.9375855], dtype=float32), array([1.5003936], dtype=float32), array([1.0812675], dtype=float32), array([0.77956885], dtype=float32), array([0.3450868], dtype=float32), array([0.00635342], dtype=float32), array([-0.6521444], dtype=float32), array([-0.93758374], dtype=float32), array([-1.3567101], dtype=float32), array([-1.4109074], dtype=float32), array([2.6629267], dtype=float32), array([2.438911], dtype=float32), array([2.244704], dtype=float32), array([2.085725], dtype=float32), array([1.9303592], dtype=float32), array([1.70815], dtype=float32), array([1.5103298], dtype=float32), array([1.2619252], dtype=float32), array([1.0189404], dtype=float32), array([0.7434373], dtype=float32), array([0.57452226], dtype=float32), array([0.42367294], dtype=float32), array([0.24211182], dtype=float32), array([0.06055073], dtype=float32), array([-0.10565449], dtype=float32), array([-0.4317418], dtype=float32), array([-1.7586738], dtype=float32), array([-1.8788112], dtype=float32), array([2.5635648], dtype=float32), array([2.410909], dtype=float32), array([2.229348], dtype=float32), array([2.0595295], dtype=float32), array([1.9068736], dtype=float32), array([1.7713803], dtype=float32), array([1.714473], dtype=float32), array([1.5302022], dtype=float32), array([1.3775463], dtype=float32), array([1.2248904], dtype=float32), array([1.0776542], dtype=float32), array([0.9466773], dtype=float32), array([0.8030544], dtype=float32), array([0.5952979], dtype=float32), array([0.3938644], dtype=float32), array([0.1590092], dtype=float32), array([-0.04965051], dtype=float32), array([-0.23843797], dtype=float32), array([-0.31341097], dtype=float32), array([-0.63227195], dtype=float32), array([-0.7180845], dtype=float32), array([2.456977], dtype=float32), array([2.1850867], dtype=float32), array([1.9050671], dtype=float32), array([1.6783415], dtype=float32), array([1.4308404], dtype=float32), array([1.2158575], dtype=float32), array([1.0397161], dtype=float32), array([1.145401], dtype=float32), array([0.9629366], dtype=float32), array([0.8202169], dtype=float32), array([0.5790387], dtype=float32), array([0.35140982], dtype=float32), array([0.21501316], dtype=float32)]\n",
      "Epoch [1/100], Loss: 0.4927\n",
      "Epoch [2/100], Loss: 0.2880\n",
      "Epoch [3/100], Loss: 0.0888\n",
      "Epoch [4/100], Loss: 0.0520\n",
      "Epoch [5/100], Loss: 0.0419\n",
      "Epoch [6/100], Loss: 0.0403\n",
      "Epoch [7/100], Loss: 0.0295\n",
      "Epoch [8/100], Loss: 0.0405\n",
      "Epoch [9/100], Loss: 0.0267\n",
      "Epoch [10/100], Loss: 0.0388\n",
      "Epoch [11/100], Loss: 0.0412\n",
      "Epoch [12/100], Loss: 0.0545\n",
      "Epoch [13/100], Loss: 0.0252\n",
      "Epoch [14/100], Loss: 0.0269\n",
      "Epoch [15/100], Loss: 0.0407\n",
      "Epoch [16/100], Loss: 0.0323\n",
      "Epoch [17/100], Loss: 0.0245\n",
      "Epoch [18/100], Loss: 0.0203\n",
      "Epoch [19/100], Loss: 0.0306\n",
      "Epoch [20/100], Loss: 0.0249\n",
      "Epoch [21/100], Loss: 0.0322\n",
      "Epoch [22/100], Loss: 0.0249\n",
      "Epoch [23/100], Loss: 0.0274\n",
      "Epoch [24/100], Loss: 0.0219\n",
      "Epoch [25/100], Loss: 0.0243\n",
      "Epoch [26/100], Loss: 0.0411\n",
      "Epoch [27/100], Loss: 0.0302\n",
      "Epoch [28/100], Loss: 0.0269\n",
      "Epoch [29/100], Loss: 0.0176\n",
      "Epoch [30/100], Loss: 0.0335\n",
      "Epoch [31/100], Loss: 0.0373\n",
      "Epoch [32/100], Loss: 0.0335\n",
      "Epoch [33/100], Loss: 0.0218\n",
      "Epoch [34/100], Loss: 0.0283\n",
      "Epoch [35/100], Loss: 0.0205\n",
      "Epoch [36/100], Loss: 0.0274\n",
      "Epoch [37/100], Loss: 0.0153\n",
      "Epoch [38/100], Loss: 0.0213\n",
      "Epoch [39/100], Loss: 0.0148\n",
      "Epoch [40/100], Loss: 0.0332\n",
      "Epoch [41/100], Loss: 0.0255\n",
      "Epoch [42/100], Loss: 0.0269\n",
      "Epoch [43/100], Loss: 0.0256\n",
      "Epoch [44/100], Loss: 0.0268\n",
      "Epoch [45/100], Loss: 0.0243\n",
      "Epoch [46/100], Loss: 0.0243\n",
      "Epoch [47/100], Loss: 0.0219\n",
      "Epoch [48/100], Loss: 0.0238\n",
      "Epoch [49/100], Loss: 0.0161\n",
      "Epoch [50/100], Loss: 0.0277\n",
      "Epoch [51/100], Loss: 0.0194\n",
      "Epoch [52/100], Loss: 0.0189\n",
      "Epoch [53/100], Loss: 0.0253\n",
      "Epoch [54/100], Loss: 0.0135\n",
      "Epoch [55/100], Loss: 0.0155\n",
      "Epoch [56/100], Loss: 0.0197\n",
      "Epoch [57/100], Loss: 0.0095\n",
      "Epoch [58/100], Loss: 0.0152\n",
      "Epoch [59/100], Loss: 0.0221\n",
      "Epoch [60/100], Loss: 0.0257\n",
      "Epoch [61/100], Loss: 0.0124\n",
      "Epoch [62/100], Loss: 0.0154\n",
      "Epoch [63/100], Loss: 0.0216\n",
      "Epoch [64/100], Loss: 0.0226\n",
      "Epoch [65/100], Loss: 0.0224\n",
      "Epoch [66/100], Loss: 0.0240\n",
      "Epoch [67/100], Loss: 0.0217\n",
      "Epoch [68/100], Loss: 0.0129\n",
      "Epoch [69/100], Loss: 0.0193\n",
      "Epoch [70/100], Loss: 0.0216\n",
      "Epoch [71/100], Loss: 0.0243\n",
      "Epoch [72/100], Loss: 0.0332\n",
      "Epoch [73/100], Loss: 0.0135\n",
      "Epoch [74/100], Loss: 0.0270\n",
      "Epoch [75/100], Loss: 0.0194\n",
      "Epoch [76/100], Loss: 0.0215\n",
      "Epoch [77/100], Loss: 0.0184\n",
      "Epoch [78/100], Loss: 0.0211\n",
      "Epoch [79/100], Loss: 0.0151\n",
      "Epoch [80/100], Loss: 0.0244\n",
      "Epoch [81/100], Loss: 0.0351\n",
      "Epoch [82/100], Loss: 0.0147\n",
      "Epoch [83/100], Loss: 0.0288\n",
      "Epoch [84/100], Loss: 0.0238\n",
      "Epoch [85/100], Loss: 0.0241\n",
      "Epoch [86/100], Loss: 0.0066\n",
      "Epoch [87/100], Loss: 0.0209\n",
      "Epoch [88/100], Loss: 0.0102\n",
      "Epoch [89/100], Loss: 0.0202\n",
      "Epoch [90/100], Loss: 0.0214\n",
      "Epoch [91/100], Loss: 0.0238\n",
      "Epoch [92/100], Loss: 0.0350\n",
      "Epoch [93/100], Loss: 0.0073\n",
      "Epoch [94/100], Loss: 0.0218\n",
      "Epoch [95/100], Loss: 0.0121\n",
      "Epoch [96/100], Loss: 0.0324\n",
      "Epoch [97/100], Loss: 0.0133\n",
      "Epoch [98/100], Loss: 0.0087\n",
      "Epoch [99/100], Loss: 0.0185\n",
      "Epoch [100/100], Loss: 0.0224\n",
      "Predictions: [[0.91325355]\n",
      " [0.8953078 ]\n",
      " [0.8786616 ]\n",
      " [0.86077774]\n",
      " [0.845431  ]\n",
      " [0.8266809 ]\n",
      " [0.8054553 ]\n",
      " [0.7839204 ]\n",
      " [0.7644278 ]\n",
      " [0.74382114]\n",
      " [0.721853  ]\n",
      " [0.7029791 ]\n",
      " [0.69041705]\n",
      " [0.68670416]\n",
      " [0.68026847]\n",
      " [0.91325355]\n",
      " [0.9029192 ]\n",
      " [0.8943795 ]\n",
      " [0.88497347]\n",
      " [0.87575316]\n",
      " [0.86727524]\n",
      " [0.8574362 ]\n",
      " [0.8479681 ]\n",
      " [0.83819085]\n",
      " [0.8269283 ]\n",
      " [0.8164704 ]\n",
      " [0.80539346]\n",
      " [0.7937597 ]\n",
      " [0.7829304 ]\n",
      " [0.77104914]\n",
      " [0.75960094]\n",
      " [0.74307853]\n",
      " [0.7293407 ]\n",
      " [0.71387035]\n",
      " [0.6963577 ]\n",
      " [0.6844145 ]\n",
      " [0.6558252 ]\n",
      " [0.61281735]\n",
      " [0.5821858 ]\n",
      " [0.91325355]\n",
      " [0.8997633 ]\n",
      " [0.8730922 ]\n",
      " [0.8587975 ]\n",
      " [0.7301452 ]\n",
      " [0.72946453]\n",
      " [0.7214817 ]\n",
      " [0.7122614 ]\n",
      " [0.70786774]\n",
      " [0.6985855 ]\n",
      " [0.6875706 ]\n",
      " [0.58657944]\n",
      " [0.55044043]\n",
      " [0.52490544]\n",
      " [0.91325355]\n",
      " [0.89895874]\n",
      " [0.885778  ]\n",
      " [0.8735254 ]\n",
      " [0.857003  ]\n",
      " [0.8393667 ]\n",
      " [0.815047  ]\n",
      " [0.7863339 ]\n",
      " [0.7550217 ]\n",
      " [0.716655  ]\n",
      " [0.66943914]\n",
      " [0.6153544 ]\n",
      " [0.5584851 ]\n",
      " [0.51945674]\n",
      " [0.49643812]\n",
      " [0.48032275]\n",
      " [0.46071365]\n",
      " [0.4483875 ]\n",
      " [0.44544277]\n",
      " [0.44387162]\n",
      " [0.44192967]\n",
      " [0.44020167]\n",
      " [0.44018298]\n",
      " [0.44016555]\n",
      " [0.44014534]\n",
      " [0.91325355]\n",
      " [0.90298104]\n",
      " [0.89382267]\n",
      " [0.8815081 ]\n",
      " [0.86622334]\n",
      " [0.84697807]\n",
      " [0.83608687]\n",
      " [0.81832683]\n",
      " [0.7975345 ]\n",
      " [0.7696258 ]\n",
      " [0.74382114]\n",
      " [0.7199347 ]\n",
      " [0.6990187 ]\n",
      " [0.67408025]\n",
      " [0.65427804]\n",
      " [0.91325355]\n",
      " [0.90551823]\n",
      " [0.8971642 ]\n",
      " [0.8887483 ]\n",
      " [0.88057995]\n",
      " [0.8725353 ]\n",
      " [0.86442876]\n",
      " [0.85477525]\n",
      " [0.8431415 ]\n",
      " [0.8336735 ]\n",
      " [0.8188838 ]\n",
      " [0.79759634]\n",
      " [0.77971256]\n",
      " [0.7452443 ]\n",
      " [0.71838766]\n",
      " [0.6853427 ]\n",
      " [0.6537831 ]\n",
      " [0.62055254]\n",
      " [0.5953047 ]\n",
      " [0.56758165]\n",
      " [0.54004425]\n",
      " [0.5234278 ]\n",
      " [0.5138694 ]\n",
      " [0.50562704]\n",
      " [0.5001091 ]\n",
      " [0.48780325]\n",
      " [0.4802766 ]\n",
      " [0.9132288 ]\n",
      " [0.8929934 ]\n",
      " [0.8744907 ]\n",
      " [0.8566069 ]\n",
      " [0.83439136]\n",
      " [0.8186115 ]\n",
      " [0.77467537]\n",
      " [0.7363705 ]\n",
      " [0.6864938 ]\n",
      " [0.63531756]\n",
      " [0.58482194]\n",
      " [0.5341567 ]\n",
      " [0.51997846]\n",
      " [0.4946511 ]\n",
      " [0.47227898]\n",
      " [0.45531455]\n",
      " [0.44633588]\n",
      " [0.44348004]\n",
      " [0.4412054 ]\n",
      " [0.44018406]\n",
      " [0.44014868]\n",
      " [0.44013107]\n",
      " [0.4401264 ]\n",
      " [0.91325355]\n",
      " [0.9016197 ]\n",
      " [0.86999816]\n",
      " [0.82804227]\n",
      " [0.7512469 ]\n",
      " [0.6815061 ]\n",
      " [0.6204907 ]\n",
      " [0.5547722 ]\n",
      " [0.50608885]\n",
      " [0.46527812]\n",
      " [0.44466123]\n",
      " [0.4407232 ]\n",
      " [0.44017085]\n",
      " [0.44014224]\n",
      " [0.440106  ]\n",
      " [0.91325355]\n",
      " [0.82903236]\n",
      " [0.7401701 ]\n",
      " [0.58323777]\n",
      " [0.48170808]\n",
      " [0.44544482]\n",
      " [0.44023022]\n",
      " [0.4401503 ]\n",
      " [0.4400904 ]\n",
      " [0.44768837]\n",
      " [0.4221053 ]\n",
      " [0.38615403]\n",
      " [0.34277055]\n",
      " [0.30298844]\n",
      " [0.27322668]\n",
      " [0.26581052]\n",
      " [0.26552203]\n",
      " [0.2630531 ]\n",
      " [0.91325355]\n",
      " [0.8984018 ]\n",
      " [0.88862455]\n",
      " [0.8619535 ]\n",
      " [0.7682644 ]\n",
      " [0.59381956]\n",
      " [0.4947296 ]\n",
      " [0.44654307]\n",
      " [0.4401913 ]\n",
      " [0.44012347]\n",
      " [0.43052274]\n",
      " [0.37642616]\n",
      " [0.29742554]\n",
      " [0.27840635]\n",
      " [0.91325355]\n",
      " [0.90372366]\n",
      " [0.89821625]\n",
      " [0.89023346]\n",
      " [0.882622  ]\n",
      " [0.8739585 ]\n",
      " [0.86436677]\n",
      " [0.8531662 ]\n",
      " [0.84091365]\n",
      " [0.8302082 ]\n",
      " [0.82154465]\n",
      " [0.81257176]\n",
      " [0.8038465 ]\n",
      " [0.79363596]\n",
      " [0.78422993]\n",
      " [0.76566535]\n",
      " [0.57958674]\n",
      " [0.52331233]\n",
      " [0.91325355]\n",
      " [0.9036618 ]\n",
      " [0.8906666 ]\n",
      " [0.87748575]\n",
      " [0.8613965 ]\n",
      " [0.8445646 ]\n",
      " [0.8382528 ]\n",
      " [0.8170272 ]\n",
      " [0.79604936]\n",
      " [0.772163  ]\n",
      " [0.74604887]\n",
      " [0.716655  ]\n",
      " [0.6864566 ]\n",
      " [0.6406641 ]\n",
      " [0.5944384 ]\n",
      " [0.54047745]\n",
      " [0.5090672 ]\n",
      " [0.47092602]\n",
      " [0.4463206 ]\n",
      " [0.44024426]\n",
      " [0.44016904]\n",
      " [0.91325355]\n",
      " [0.89970136]\n",
      " [0.8855304 ]\n",
      " [0.86999816]\n",
      " [0.8555796 ]\n",
      " [0.84017104]\n",
      " [0.8256906 ]\n",
      " [0.7766803 ]\n",
      " [0.63757   ]\n",
      " [0.49969354]\n",
      " [0.44660315]\n",
      " [0.44172922]\n",
      " [0.44017538]], Actuals: [[0.94401044]\n",
      " [0.87425596]\n",
      " [0.8188244 ]\n",
      " [0.75911456]\n",
      " [0.69345236]\n",
      " [0.6197917 ]\n",
      " [0.55078125]\n",
      " [0.48046875]\n",
      " [0.4200149 ]\n",
      " [0.3653274 ]\n",
      " [0.3236607 ]\n",
      " [0.2721354 ]\n",
      " [0.24069941]\n",
      " [0.22526042]\n",
      " [0.20889136]\n",
      " [0.97154015]\n",
      " [0.9281994 ]\n",
      " [0.891183  ]\n",
      " [0.84877235]\n",
      " [0.81175596]\n",
      " [0.7699033 ]\n",
      " [0.7323289 ]\n",
      " [0.6923363 ]\n",
      " [0.6478795 ]\n",
      " [0.6013765 ]\n",
      " [0.55803573]\n",
      " [0.5130208 ]\n",
      " [0.46595982]\n",
      " [0.42261904]\n",
      " [0.3813244 ]\n",
      " [0.3372396 ]\n",
      " [0.30766368]\n",
      " [0.2641369 ]\n",
      " [0.2155878 ]\n",
      " [0.15438989]\n",
      " [0.12276786]\n",
      " [0.05691964]\n",
      " [0.04650298]\n",
      " [0.02101935]\n",
      " [0.95554316]\n",
      " [0.9161086 ]\n",
      " [0.8563988 ]\n",
      " [0.82217264]\n",
      " [0.51488096]\n",
      " [0.5052083 ]\n",
      " [0.48325893]\n",
      " [0.46186757]\n",
      " [0.44773066]\n",
      " [0.41945684]\n",
      " [0.38244048]\n",
      " [0.21968006]\n",
      " [0.19308035]\n",
      " [0.1688988 ]\n",
      " [0.9618676 ]\n",
      " [0.9099702 ]\n",
      " [0.86960566]\n",
      " [0.8374256 ]\n",
      " [0.7927827 ]\n",
      " [0.7537202 ]\n",
      " [0.7176339 ]\n",
      " [0.6834077 ]\n",
      " [0.6672247 ]\n",
      " [0.64304316]\n",
      " [0.60751486]\n",
      " [0.56640625]\n",
      " [0.5087426 ]\n",
      " [0.4750744 ]\n",
      " [0.4466146 ]\n",
      " [0.42578125]\n",
      " [0.40364584]\n",
      " [0.37518603]\n",
      " [0.34281993]\n",
      " [0.30989584]\n",
      " [0.27436757]\n",
      " [0.24758184]\n",
      " [0.22395833]\n",
      " [0.20721726]\n",
      " [0.19140625]\n",
      " [0.9990699 ]\n",
      " [0.9588914 ]\n",
      " [0.92578125]\n",
      " [0.8885789 ]\n",
      " [0.8625372 ]\n",
      " [0.8329613 ]\n",
      " [0.812686  ]\n",
      " [0.7832961 ]\n",
      " [0.7514881 ]\n",
      " [0.703683  ]\n",
      " [0.6614583 ]\n",
      " [0.625558  ]\n",
      " [0.59598213]\n",
      " [0.5556176 ]\n",
      " [0.52883184]\n",
      " [0.9817708 ]\n",
      " [0.94456846]\n",
      " [0.9088542 ]\n",
      " [0.86848956]\n",
      " [0.828683  ]\n",
      " [0.7927827 ]\n",
      " [0.7635789 ]\n",
      " [0.73400295]\n",
      " [0.6994048 ]\n",
      " [0.67410713]\n",
      " [0.6501116 ]\n",
      " [0.62053573]\n",
      " [0.5894717 ]\n",
      " [0.53515625]\n",
      " [0.4968378 ]\n",
      " [0.44568452]\n",
      " [0.40531993]\n",
      " [0.36049107]\n",
      " [0.33537945]\n",
      " [0.30487353]\n",
      " [0.27455357]\n",
      " [0.24758184]\n",
      " [0.21893601]\n",
      " [0.19159226]\n",
      " [0.17912947]\n",
      " [0.1436012 ]\n",
      " [0.12090774]\n",
      " [1.        ]\n",
      " [0.9114583 ]\n",
      " [0.84095985]\n",
      " [0.78087795]\n",
      " [0.70758927]\n",
      " [0.6910342 ]\n",
      " [0.6765253 ]\n",
      " [0.6515997 ]\n",
      " [0.624442  ]\n",
      " [0.59040177]\n",
      " [0.5576637 ]\n",
      " [0.5275298 ]\n",
      " [0.50967264]\n",
      " [0.46261162]\n",
      " [0.41108632]\n",
      " [0.36495537]\n",
      " [0.32012647]\n",
      " [0.2514881 ]\n",
      " [0.20758928]\n",
      " [0.14918154]\n",
      " [0.06268601]\n",
      " [0.01320685]\n",
      " [0.        ]\n",
      " [0.9672619 ]\n",
      " [0.9157366 ]\n",
      " [0.88095236]\n",
      " [0.8552827 ]\n",
      " [0.8212426 ]\n",
      " [0.77734375]\n",
      " [0.75037205]\n",
      " [0.7433036 ]\n",
      " [0.70963544]\n",
      " [0.6551339 ]\n",
      " [0.56324404]\n",
      " [0.47581846]\n",
      " [0.37406993]\n",
      " [0.30226934]\n",
      " [0.21781994]\n",
      " [0.9583333 ]\n",
      " [0.9142485 ]\n",
      " [0.87890625]\n",
      " [0.81529015]\n",
      " [0.74088544]\n",
      " [0.6728051 ]\n",
      " [0.60602677]\n",
      " [0.531064  ]\n",
      " [0.45591518]\n",
      " [0.38002232]\n",
      " [0.30877978]\n",
      " [0.25      ]\n",
      " [0.19549851]\n",
      " [0.15197173]\n",
      " [0.1015625 ]\n",
      " [0.07477678]\n",
      " [0.06026786]\n",
      " [0.03552827]\n",
      " [0.9879092 ]\n",
      " [0.9510789 ]\n",
      " [0.9188988 ]\n",
      " [0.8729539 ]\n",
      " [0.8214286 ]\n",
      " [0.7313988 ]\n",
      " [0.64508927]\n",
      " [0.5829613 ]\n",
      " [0.4934896 ]\n",
      " [0.4237351 ]\n",
      " [0.28813243]\n",
      " [0.22935268]\n",
      " [0.14304316]\n",
      " [0.13188244]\n",
      " [0.9707961 ]\n",
      " [0.92466515]\n",
      " [0.88467264]\n",
      " [0.85193455]\n",
      " [0.81994045]\n",
      " [0.77418154]\n",
      " [0.7334449 ]\n",
      " [0.6822917 ]\n",
      " [0.6322545 ]\n",
      " [0.5755208 ]\n",
      " [0.5407366 ]\n",
      " [0.50967264]\n",
      " [0.47228423]\n",
      " [0.43489584]\n",
      " [0.40066963]\n",
      " [0.33351934]\n",
      " [0.06026786]\n",
      " [0.03552827]\n",
      " [0.95033485]\n",
      " [0.9188988 ]\n",
      " [0.88151044]\n",
      " [0.84654015]\n",
      " [0.8151042 ]\n",
      " [0.78720236]\n",
      " [0.7754836 ]\n",
      " [0.7375372 ]\n",
      " [0.7061012 ]\n",
      " [0.67466515]\n",
      " [0.6443452 ]\n",
      " [0.6173735 ]\n",
      " [0.58779764]\n",
      " [0.54501486]\n",
      " [0.5035342 ]\n",
      " [0.45517114]\n",
      " [0.4122024 ]\n",
      " [0.37332588]\n",
      " [0.3578869 ]\n",
      " [0.2922247 ]\n",
      " [0.27455357]\n",
      " [0.92838544]\n",
      " [0.8723958 ]\n",
      " [0.81473213]\n",
      " [0.76804316]\n",
      " [0.7170759 ]\n",
      " [0.6728051 ]\n",
      " [0.6365327 ]\n",
      " [0.6582961 ]\n",
      " [0.6207217 ]\n",
      " [0.59133184]\n",
      " [0.5416667 ]\n",
      " [0.49479166]\n",
      " [0.46670386]]\n",
      "Epoch 1, Loss: 3.7043889901217293\n",
      "Epoch 2, Loss: 1.410523838856641\n",
      "Epoch 3, Loss: 1.1487928646452286\n",
      "Epoch 4, Loss: 0.9188877817462472\n",
      "Epoch 5, Loss: 1.6977183170178358\n",
      "Epoch 6, Loss: 1.0241828083115465\n",
      "Epoch 7, Loss: 0.7451395543620867\n",
      "Epoch 8, Loss: 1.0437169075012207\n",
      "Epoch 9, Loss: 0.7602501380092958\n",
      "Epoch 10, Loss: 0.9433818538399303\n",
      "Epoch 11, Loss: 1.2507526147014953\n",
      "Epoch 12, Loss: 0.9937785092522117\n",
      "Epoch 13, Loss: 0.7053799488965202\n",
      "Epoch 14, Loss: 0.9530435217654004\n",
      "Epoch 15, Loss: 2.419162459233228\n",
      "Epoch 16, Loss: 0.9721927200170124\n",
      "Epoch 17, Loss: 1.1001348166781313\n",
      "Epoch 18, Loss: 0.7016444074756959\n",
      "Epoch 19, Loss: 0.7740493758636362\n",
      "Epoch 20, Loss: 0.7624277437434477\n",
      "Epoch 21, Loss: 0.773796937044929\n",
      "Epoch 22, Loss: 0.6945864531923743\n",
      "Epoch 23, Loss: 0.8072178894106079\n",
      "Epoch 24, Loss: 0.7061951979994774\n",
      "Epoch 25, Loss: 0.7993042433963102\n",
      "Epoch 26, Loss: 0.6789901383659419\n",
      "Epoch 27, Loss: 0.7403324653997141\n",
      "Epoch 28, Loss: 0.8973507263204631\n",
      "Epoch 29, Loss: 0.9245107353610151\n",
      "Epoch 30, Loss: 0.6725148357012692\n",
      "Epoch 31, Loss: 0.7615772985360202\n",
      "Epoch 32, Loss: 0.9464802645585116\n",
      "Epoch 33, Loss: 0.9111652181429022\n",
      "Epoch 34, Loss: 0.7076561617500642\n",
      "Epoch 35, Loss: 0.7761778125868124\n",
      "Epoch 36, Loss: 0.6674819303786054\n",
      "Epoch 37, Loss: 0.7337790140334297\n",
      "Epoch 38, Loss: 0.665539364604389\n",
      "Epoch 39, Loss: 0.6731428977321176\n",
      "Epoch 40, Loss: 0.7295427304856917\n",
      "Epoch 41, Loss: 0.6822792098802679\n",
      "Epoch 42, Loss: 0.6926130761995035\n",
      "Epoch 43, Loss: 0.7230483251897728\n",
      "Epoch 44, Loss: 1.0136774944908478\n",
      "Epoch 45, Loss: 0.7570897643180454\n",
      "Epoch 46, Loss: 0.7072847117395962\n",
      "Epoch 47, Loss: 0.7919925315415158\n",
      "Epoch 48, Loss: 0.7176552423659492\n",
      "Epoch 49, Loss: 0.7097018694176394\n",
      "Epoch 50, Loss: 0.9814616187530405\n",
      "Epoch 51, Loss: 0.9912201411583844\n",
      "Epoch 52, Loss: 0.9482942933545393\n",
      "Epoch 53, Loss: 0.8034820096457705\n",
      "Epoch 54, Loss: 0.7154957617030424\n",
      "Epoch 55, Loss: 0.6914603094405988\n",
      "Epoch 56, Loss: 0.8134758744169684\n",
      "Epoch 57, Loss: 0.7884387380498297\n",
      "Epoch 58, Loss: 0.9531327039003372\n",
      "Epoch 59, Loss: 0.7280776526998071\n",
      "Epoch 60, Loss: 0.7201496332007296\n",
      "Epoch 61, Loss: 0.8636572343461654\n",
      "Epoch 62, Loss: 1.177676399841028\n",
      "Epoch 63, Loss: 1.0081577975960339\n",
      "Epoch 64, Loss: 0.9039953438674703\n",
      "Epoch 65, Loss: 0.6761494103599998\n",
      "Epoch 66, Loss: 0.7080347235588467\n",
      "Epoch 67, Loss: 0.7136972380035064\n",
      "Epoch 68, Loss: 0.6817317973164951\n",
      "Epoch 69, Loss: 0.7070973287610447\n",
      "Epoch 70, Loss: 0.690859343199169\n",
      "Epoch 71, Loss: 0.6775968828622032\n",
      "Epoch 72, Loss: 0.660523430827786\n",
      "Epoch 73, Loss: 0.7314560527310652\n",
      "Epoch 74, Loss: 0.8164368867874146\n",
      "Epoch 75, Loss: 0.7367480186855092\n",
      "Epoch 76, Loss: 0.7215046821271672\n",
      "Epoch 77, Loss: 0.6698912042467033\n",
      "Epoch 78, Loss: 0.7665590956149733\n",
      "Epoch 79, Loss: 0.6678016527610666\n",
      "Epoch 80, Loss: 0.6899584303883946\n",
      "Epoch 81, Loss: 0.6806909012443879\n",
      "Epoch 82, Loss: 0.8265124826746828\n",
      "Epoch 83, Loss: 0.7329323129618869\n",
      "Epoch 84, Loss: 0.6879712246796664\n",
      "Epoch 85, Loss: 0.6724221872932771\n",
      "Epoch 86, Loss: 0.6754232119111454\n",
      "Epoch 87, Loss: 0.9002556590472951\n",
      "Epoch 88, Loss: 0.7435011118650436\n",
      "Epoch 89, Loss: 0.6795137962874245\n",
      "Epoch 90, Loss: 0.6902259307749131\n",
      "Epoch 91, Loss: 0.6815033444915625\n",
      "Epoch 92, Loss: 0.6876482192207786\n",
      "Epoch 93, Loss: 0.7228574875523063\n",
      "Epoch 94, Loss: 0.8857291003360468\n",
      "Epoch 95, Loss: 0.8726726327748859\n",
      "Epoch 96, Loss: 0.6967408968245282\n",
      "Epoch 97, Loss: 0.7449890447451788\n",
      "Epoch 98, Loss: 0.7517672143876553\n",
      "Epoch 99, Loss: 0.6716302146806437\n",
      "Epoch 100, Loss: 0.6616629909943131\n",
      "Test Loss: 0.6945888088084757\n",
      "Test Loss: 0.7094122460111976\n",
      "Test Loss: 0.6545696510002017\n"
     ]
    }
   ],
   "source": [
    "from LLM_call import LLMModel\n",
    "from tqdm import tqdm\n",
    "\n",
    "llm = LLMModel(api_key=\"sk-rifpc-2Gg7xjJ4qrwzWY7hUhZKT3BlbkFJBkz9CHkx9LkVsSciz9Tg\", model=\"gpt-4o\")\n",
    "result_dict = {}\n",
    "\n",
    "for i in range(10):\n",
    "    output = llm.LLM_response(base_prompt, model=\"gpt-4o\")\n",
    "    is_runnable, message, extracted_code = extract_and_test_llm_code_simplified(output)\n",
    "    result_dict[i] = {\n",
    "        \"output\": output,\n",
    "        \"is_runnable\": is_runnable,\n",
    "        \"message\": message,\n",
    "        \"extracted_code\": extracted_code\n",
    "    }\n",
    "# save the result_dict to a file\n",
    "import json\n",
    "with open(\"result_dict_baseline.json\", \"w\") as f:\n",
    "    json.dump(result_dict, f, indent=4)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "bb1c487d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "9\n"
     ]
    }
   ],
   "source": [
    "# total runnable code\n",
    "total_runnable_code = sum(1 for result in result_dict.values() if result[\"is_runnable\"])\n",
    "print(total_runnable_code)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 51,
   "id": "eb0c2505",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "import pandas as pd\n",
      "import numpy as np\n",
      "import torch\n",
      "import torch.nn as nn\n",
      "import torch.optim as optim\n",
      "from torch.utils.data import Dataset, DataLoader\n",
      "\n",
      "# Define mathematical model functions\n",
      "def bruun_model(x, sea_level_rise):\n",
      "    # Bruun's rule is a simplistic model assuming a constant slope\n",
      "    # and relates the horizontal retreat distance (x) with sea-level rise.\n",
      "    return x - sea_level_rise * (x / initial_width)\n",
      "\n",
      "def dean_model(x, sea_level_rise):\n",
      "    # Dean model assumes profile convexity and allows calculation of new profile\n",
      "    return x - np.sqrt(sea_level_rise * x / initial_width)\n",
      "\n",
      "# Define a PyTorch Dataset\n",
      "class BeachProfileDataset(Dataset):\n",
      "    def __init__(self, file_path):\n",
      "        self.data = pd.read_excel(file_path)\n",
      "    \n",
      "    def __len__(self):\n",
      "        return len(self.data)\n",
      "    \n",
      "    def __getitem__(self, idx):\n",
      "        x = self.data.iloc[idx]['x']\n",
      "        y = self.data.iloc[idx]['y']\n",
      "        return torch.tensor([x], dtype=torch.float32), torch.tensor([y], dtype=torch.float32)\n",
      "\n",
      "# Define a PyTorch Model\n",
      "class BeachProfileModel(nn.Module):\n",
      "    def __init__(self):\n",
      "        super(BeachProfileModel, self).__init__()\n",
      "        self.linear1 = nn.Linear(1, 128)\n",
      "        self.relu = nn.ReLU()\n",
      "        self.linear2 = nn.Linear(128, 64)\n",
      "        self.linear3 = nn.Linear(64, 1)\n",
      "    \n",
      "    def forward(self, x):\n",
      "        x = self.linear1(x)\n",
      "        x = self.relu(x)\n",
      "        x = self.linear2(x)\n",
      "        x = self.relu(x)\n",
      "        x = self.linear3(x)\n",
      "        return x\n",
      "\n",
      "# Function to train the model\n",
      "def train_model(train_loader, model, criterion, optimizer, epochs):\n",
      "    for epoch in range(epochs):\n",
      "        for x, y in train_loader:\n",
      "            optimizer.zero_grad()\n",
      "            outputs = model(x)\n",
      "            loss = criterion(outputs, y)\n",
      "            loss.backward()\n",
      "            optimizer.step()\n",
      "\n",
      "# Function to evaluate the model\n",
      "def evaluate_model(test_loader, model, criterion):\n",
      "    total_loss = 0\n",
      "    with torch.no_grad():\n",
      "        for x, y in test_loader:\n",
      "            outputs = model(x)\n",
      "            loss = criterion(outputs, y)\n",
      "            total_loss += loss.item()\n",
      "    return total_loss / len(test_loader)\n",
      "\n",
      "# Main function\n",
      "def main():\n",
      "    # Constants\n",
      "    global initial_width\n",
      "    initial_width = 1000  # assumed constant width of the beach for the models\n",
      "    \n",
      "    # Load Dataset\n",
      "    train_data_path = \"beach_profile_data/processed_data/beachdata_train.xlsx\"\n",
      "    test_data_path = \"beach_profile_data/processed_data/beachdata_test.xlsx\"\n",
      "    \n",
      "    train_dataset = BeachProfileDataset(train_data_path)\n",
      "    test_dataset = BeachProfileDataset(test_data_path)\n",
      "    \n",
      "    train_loader = DataLoader(train_dataset, batch_size=32, shuffle=True)\n",
      "    test_loader = DataLoader(test_dataset, batch_size=32, shuffle=False)\n",
      "    \n",
      "    # Initialize Model, Loss Function and Optimizer\n",
      "    model = BeachProfileModel()\n",
      "    criterion = nn.MSELoss()\n",
      "    optimizer = optim.Adam(model.parameters(), lr=0.001)\n",
      "    epochs = 50\n",
      "\n",
      "    # Train the model\n",
      "    train_model(train_loader, model, criterion, optimizer, epochs)\n",
      "\n",
      "    # Evaluate the model\n",
      "    test_loss = evaluate_model(test_loader, model, criterion)\n",
      "    print(f\"Test Loss: {test_loss}\")\n",
      "\n",
      "# Ensure the script is not executed on import\n",
      "if __name__ == \"__main__\":\n",
      "    main()\n"
     ]
    }
   ],
   "source": [
    "code_sample = result_dict[1][\"extracted_code\"]\n",
    "print(code_sample)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 55,
   "id": "07e42b9b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "from torch.utils.data import Dataset, DataLoader\n",
    "\n",
    "# Define mathematical model functions\n",
    "def bruun_model(x, sea_level_rise):\n",
    "    # Bruun's rule is a simplistic model assuming a constant slope\n",
    "    # and relates the horizontal retreat distance (x) with sea-level rise.\n",
    "    return x - sea_level_rise * (x / initial_width)\n",
    "\n",
    "def dean_model(x, sea_level_rise):\n",
    "    # Dean model assumes profile convexity and allows calculation of new profile\n",
    "    return x - np.sqrt(sea_level_rise * x / initial_width)\n",
    "\n",
    "# Define a PyTorch Dataset\n",
    "class BeachProfileDataset(Dataset):\n",
    "    def __init__(self, file_path):\n",
    "        self.data = pd.read_excel(file_path)\n",
    "    \n",
    "    def __len__(self):\n",
    "        return len(self.data)\n",
    "    \n",
    "    def __getitem__(self, idx):\n",
    "        x = self.data.iloc[idx]['x']\n",
    "        y = self.data.iloc[idx]['y']\n",
    "        return torch.tensor([x], dtype=torch.float32), torch.tensor([y], dtype=torch.float32)\n",
    "\n",
    "# Define a PyTorch Model\n",
    "class BeachProfileModel(nn.Module):\n",
    "    def __init__(self):\n",
    "        super(BeachProfileModel, self).__init__()\n",
    "        self.linear1 = nn.Linear(1, 128)\n",
    "        self.relu = nn.ReLU()\n",
    "        self.linear2 = nn.Linear(128, 64)\n",
    "        self.linear3 = nn.Linear(64, 1)\n",
    "    \n",
    "    def forward(self, x):\n",
    "        x = self.linear1(x)\n",
    "        x = self.relu(x)\n",
    "        x = self.linear2(x)\n",
    "        x = self.relu(x)\n",
    "        x = self.linear3(x)\n",
    "        return x\n",
    "\n",
    "# Function to train the model\n",
    "def train_model(train_loader, model, criterion, optimizer, epochs):\n",
    "    for epoch in range(epochs):\n",
    "        for x, y in train_loader:\n",
    "            optimizer.zero_grad()\n",
    "            outputs = model(x)\n",
    "            loss = criterion(outputs, y)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "\n",
    "# Function to evaluate the model\n",
    "def evaluate_model(test_loader, model, criterion):\n",
    "    total_loss = 0\n",
    "    output_list = []\n",
    "    with torch.no_grad():\n",
    "        for x, y in test_loader:\n",
    "            outputs = model(x)\n",
    "            # append outputs and y to output_list for further analysis\n",
    "            output_list.append((outputs, y))\n",
    "            # loss = criterion(outputs, y)\n",
    "            # total_loss += loss.item()\n",
    "    return output_list\n",
    "\n",
    "# Main function\n",
    "def main():\n",
    "    # Constants\n",
    "    global initial_width\n",
    "    initial_width = 1000  # assumed constant width of the beach for the models\n",
    "    \n",
    "    # Load Dataset\n",
    "    train_data_path = \"beach_profile_data/processed_data/beachdata_train.xlsx\"\n",
    "    test_data_path = \"beach_profile_data/processed_data/beachdata_test.xlsx\"\n",
    "    \n",
    "    train_dataset = BeachProfileDataset(train_data_path)\n",
    "    test_dataset = BeachProfileDataset(test_data_path)\n",
    "    \n",
    "    train_loader = DataLoader(train_dataset, batch_size=32, shuffle=True)\n",
    "    test_loader = DataLoader(test_dataset, batch_size=32, shuffle=False)\n",
    "    \n",
    "    # Initialize Model, Loss Function and Optimizer\n",
    "    model = BeachProfileModel()\n",
    "    criterion = nn.MSELoss()\n",
    "    optimizer = optim.Adam(model.parameters(), lr=0.001)\n",
    "    epochs = 50\n",
    "\n",
    "    # Train the model\n",
    "    train_model(train_loader, model, criterion, optimizer, epochs)\n",
    "\n",
    "    # Evaluate the model\n",
    "    test_loss = evaluate_model(test_loader, model, criterion)\n",
    "    # print(f\"Test Loss: {test_loss}\")\n",
    "\n",
    "# Ensure the script is not executed on import\n",
    "if __name__ == \"__main__\":\n",
    "    main()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 56,
   "id": "8afd0982",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[(tensor([[2.6853],\n",
      "        [2.4010],\n",
      "        [2.2390],\n",
      "        [2.0052],\n",
      "        [1.8089],\n",
      "        [1.5889],\n",
      "        [1.3951],\n",
      "        [1.1997],\n",
      "        [1.0285],\n",
      "        [0.8891],\n",
      "        [0.7433],\n",
      "        [0.6293],\n",
      "        [0.5619],\n",
      "        [0.5420],\n",
      "        [0.5147],\n",
      "        [2.6853],\n",
      "        [2.4326],\n",
      "        [2.3921],\n",
      "        [2.2996],\n",
      "        [2.2004],\n",
      "        [2.0830],\n",
      "        [1.9639],\n",
      "        [1.8432],\n",
      "        [1.7102],\n",
      "        [1.5912],\n",
      "        [1.4959],\n",
      "        [1.3945],\n",
      "        [1.2883],\n",
      "        [1.1907],\n",
      "        [1.0837],\n",
      "        [0.9955],\n",
      "        [0.8841]]), tensor([[ 2.5080],\n",
      "        [ 2.1330],\n",
      "        [ 1.8350],\n",
      "        [ 1.5140],\n",
      "        [ 1.1610],\n",
      "        [ 0.7650],\n",
      "        [ 0.3940],\n",
      "        [ 0.0160],\n",
      "        [-0.3090],\n",
      "        [-0.6030],\n",
      "        [-0.8270],\n",
      "        [-1.1040],\n",
      "        [-1.2730],\n",
      "        [-1.3560],\n",
      "        [-1.4440],\n",
      "        [ 2.6560],\n",
      "        [ 2.4230],\n",
      "        [ 2.2240],\n",
      "        [ 1.9960],\n",
      "        [ 1.7970],\n",
      "        [ 1.5720],\n",
      "        [ 1.3700],\n",
      "        [ 1.1550],\n",
      "        [ 0.9160],\n",
      "        [ 0.6660],\n",
      "        [ 0.4330],\n",
      "        [ 0.1910],\n",
      "        [-0.0620],\n",
      "        [-0.2950],\n",
      "        [-0.5170],\n",
      "        [-0.7540],\n",
      "        [-0.9130]])), (tensor([[0.7915],\n",
      "        [0.6944],\n",
      "        [0.5938],\n",
      "        [0.5297],\n",
      "        [0.4688],\n",
      "        [0.4469],\n",
      "        [0.4314],\n",
      "        [2.6853],\n",
      "        [2.4206],\n",
      "        [2.1613],\n",
      "        [1.9807],\n",
      "        [0.7969],\n",
      "        [0.7924],\n",
      "        [0.7410],\n",
      "        [0.6845],\n",
      "        [0.6577],\n",
      "        [0.6057],\n",
      "        [0.5466],\n",
      "        [0.4336],\n",
      "        [0.4152],\n",
      "        [0.4178],\n",
      "        [2.6853],\n",
      "        [2.4186],\n",
      "        [2.3062],\n",
      "        [2.1678],\n",
      "        [1.9585],\n",
      "        [1.7262],\n",
      "        [1.4829],\n",
      "        [1.2214],\n",
      "        [0.9646],\n",
      "        [0.7114],\n",
      "        [0.4862]]), tensor([[-1.1470],\n",
      "        [-1.4080],\n",
      "        [-1.7370],\n",
      "        [-1.9070],\n",
      "        [-2.2610],\n",
      "        [-2.3170],\n",
      "        [-2.4540],\n",
      "        [ 2.5700],\n",
      "        [ 2.3580],\n",
      "        [ 2.0370],\n",
      "        [ 1.8530],\n",
      "        [ 0.2010],\n",
      "        [ 0.1490],\n",
      "        [ 0.0310],\n",
      "        [-0.0840],\n",
      "        [-0.1600],\n",
      "        [-0.3120],\n",
      "        [-0.5110],\n",
      "        [-1.3860],\n",
      "        [-1.5290],\n",
      "        [-1.6590],\n",
      "        [ 2.6040],\n",
      "        [ 2.3250],\n",
      "        [ 2.1080],\n",
      "        [ 1.9350],\n",
      "        [ 1.6950],\n",
      "        [ 1.4850],\n",
      "        [ 1.2910],\n",
      "        [ 1.1070],\n",
      "        [ 1.0200],\n",
      "        [ 0.8900],\n",
      "        [ 0.6990]])), (tensor([[0.4482],\n",
      "        [0.4193],\n",
      "        [0.4193],\n",
      "        [0.4257],\n",
      "        [0.4303],\n",
      "        [0.4359],\n",
      "        [0.4409],\n",
      "        [0.4460],\n",
      "        [0.4510],\n",
      "        [0.4573],\n",
      "        [0.4638],\n",
      "        [0.4704],\n",
      "        [0.4767],\n",
      "        [0.4838],\n",
      "        [2.6853],\n",
      "        [2.4327],\n",
      "        [2.3867],\n",
      "        [2.2715],\n",
      "        [2.0709],\n",
      "        [1.8298],\n",
      "        [1.6814],\n",
      "        [1.5128],\n",
      "        [1.3223],\n",
      "        [1.0709],\n",
      "        [0.8891],\n",
      "        [0.7314],\n",
      "        [0.6081],\n",
      "        [0.4969],\n",
      "        [0.4680],\n",
      "        [2.6853],\n",
      "        [2.4467],\n",
      "        [2.4144]]), tensor([[ 0.4780],\n",
      "        [ 0.1680],\n",
      "        [-0.0130],\n",
      "        [-0.1660],\n",
      "        [-0.2780],\n",
      "        [-0.3970],\n",
      "        [-0.5500],\n",
      "        [-0.7240],\n",
      "        [-0.9010],\n",
      "        [-1.0920],\n",
      "        [-1.2360],\n",
      "        [-1.3630],\n",
      "        [-1.4530],\n",
      "        [-1.5380],\n",
      "        [ 2.8040],\n",
      "        [ 2.5880],\n",
      "        [ 2.4100],\n",
      "        [ 2.2100],\n",
      "        [ 2.0700],\n",
      "        [ 1.9110],\n",
      "        [ 1.8020],\n",
      "        [ 1.6440],\n",
      "        [ 1.4730],\n",
      "        [ 1.2160],\n",
      "        [ 0.9890],\n",
      "        [ 0.7960],\n",
      "        [ 0.6370],\n",
      "        [ 0.4200],\n",
      "        [ 0.2760],\n",
      "        [ 2.7110],\n",
      "        [ 2.5110],\n",
      "        [ 2.3190]])), (tensor([[2.3341],\n",
      "        [2.2640],\n",
      "        [2.1530],\n",
      "        [2.0498],\n",
      "        [1.9310],\n",
      "        [1.7778],\n",
      "        [1.6527],\n",
      "        [1.5179],\n",
      "        [1.3228],\n",
      "        [1.1618],\n",
      "        [0.8987],\n",
      "        [0.7219],\n",
      "        [0.5347],\n",
      "        [0.4678],\n",
      "        [0.4509],\n",
      "        [0.4380],\n",
      "        [0.4240],\n",
      "        [0.4155],\n",
      "        [0.4182],\n",
      "        [0.4209],\n",
      "        [0.4232],\n",
      "        [0.4247],\n",
      "        [0.4282],\n",
      "        [0.4303],\n",
      "        [2.6841],\n",
      "        [2.3787],\n",
      "        [2.1823],\n",
      "        [1.9536],\n",
      "        [1.6592],\n",
      "        [1.5154],\n",
      "        [1.1164],\n",
      "        [0.8389]]), tensor([[ 2.1020],\n",
      "        [ 1.8880],\n",
      "        [ 1.6950],\n",
      "        [ 1.5380],\n",
      "        [ 1.3790],\n",
      "        [ 1.1930],\n",
      "        [ 1.0570],\n",
      "        [ 0.9280],\n",
      "        [ 0.7690],\n",
      "        [ 0.6020],\n",
      "        [ 0.3100],\n",
      "        [ 0.1040],\n",
      "        [-0.1710],\n",
      "        [-0.3880],\n",
      "        [-0.6290],\n",
      "        [-0.7640],\n",
      "        [-0.9280],\n",
      "        [-1.0910],\n",
      "        [-1.2360],\n",
      "        [-1.3900],\n",
      "        [-1.5370],\n",
      "        [-1.6040],\n",
      "        [-1.7950],\n",
      "        [-1.9170],\n",
      "        [ 2.8090],\n",
      "        [ 2.3330],\n",
      "        [ 1.9540],\n",
      "        [ 1.6310],\n",
      "        [ 1.2370],\n",
      "        [ 1.1480],\n",
      "        [ 1.0700],\n",
      "        [ 0.9360]])), (tensor([[0.5408],\n",
      "        [0.4584],\n",
      "        [0.4327],\n",
      "        [0.4163],\n",
      "        [0.4192],\n",
      "        [0.4262],\n",
      "        [0.4325],\n",
      "        [0.4375],\n",
      "        [0.4431],\n",
      "        [0.4523],\n",
      "        [0.4596],\n",
      "        [0.4701],\n",
      "        [0.4827],\n",
      "        [0.4889],\n",
      "        [0.4906],\n",
      "        [2.6853],\n",
      "        [2.4283],\n",
      "        [2.1143],\n",
      "        [1.6013],\n",
      "        [0.9392],\n",
      "        [0.5183],\n",
      "        [0.4508],\n",
      "        [0.4174],\n",
      "        [0.4230],\n",
      "        [0.4346],\n",
      "        [0.4485],\n",
      "        [0.4612],\n",
      "        [0.4748],\n",
      "        [0.4850],\n",
      "        [0.4979],\n",
      "        [2.6853],\n",
      "        [1.6104]]), tensor([[ 0.7900],\n",
      "        [ 0.6070],\n",
      "        [ 0.4310],\n",
      "        [ 0.2690],\n",
      "        [ 0.1730],\n",
      "        [-0.0800],\n",
      "        [-0.3570],\n",
      "        [-0.6050],\n",
      "        [-0.8460],\n",
      "        [-1.2150],\n",
      "        [-1.4510],\n",
      "        [-1.7650],\n",
      "        [-2.2300],\n",
      "        [-2.4960],\n",
      "        [-2.5670],\n",
      "        [ 2.6330],\n",
      "        [ 2.3560],\n",
      "        [ 2.1690],\n",
      "        [ 2.0310],\n",
      "        [ 1.8480],\n",
      "        [ 1.6120],\n",
      "        [ 1.4670],\n",
      "        [ 1.4290],\n",
      "        [ 1.2480],\n",
      "        [ 0.9550],\n",
      "        [ 0.4610],\n",
      "        [-0.0090],\n",
      "        [-0.5560],\n",
      "        [-0.9420],\n",
      "        [-1.3960],\n",
      "        [ 2.5850],\n",
      "        [ 2.3480]])), (tensor([[0.8645],\n",
      "        [0.4319],\n",
      "        [0.4299],\n",
      "        [0.4460],\n",
      "        [0.4628],\n",
      "        [0.4821],\n",
      "        [0.5034],\n",
      "        [0.5213],\n",
      "        [0.5410],\n",
      "        [0.5561],\n",
      "        [0.5744],\n",
      "        [0.5967],\n",
      "        [0.6250],\n",
      "        [0.6421],\n",
      "        [0.6429],\n",
      "        [0.6499],\n",
      "        [2.6853],\n",
      "        [2.4173],\n",
      "        [2.3327],\n",
      "        [2.0197],\n",
      "        [1.0586],\n",
      "        [0.4373],\n",
      "        [0.4262],\n",
      "        [0.4424],\n",
      "        [0.4675],\n",
      "        [0.4916],\n",
      "        [0.5375],\n",
      "        [0.5602],\n",
      "        [0.6018],\n",
      "        [0.6194],\n",
      "        [2.6853],\n",
      "        [2.4359]]), tensor([[ 2.1580],\n",
      "        [ 1.8160],\n",
      "        [ 1.4160],\n",
      "        [ 1.0500],\n",
      "        [ 0.6910],\n",
      "        [ 0.2880],\n",
      "        [-0.1160],\n",
      "        [-0.5240],\n",
      "        [-0.9070],\n",
      "        [-1.2230],\n",
      "        [-1.5160],\n",
      "        [-1.7500],\n",
      "        [-2.0210],\n",
      "        [-2.1650],\n",
      "        [-2.2430],\n",
      "        [-2.3760],\n",
      "        [ 2.7440],\n",
      "        [ 2.5460],\n",
      "        [ 2.3730],\n",
      "        [ 2.1260],\n",
      "        [ 1.8490],\n",
      "        [ 1.3650],\n",
      "        [ 0.9010],\n",
      "        [ 0.5670],\n",
      "        [ 0.0860],\n",
      "        [-0.2890],\n",
      "        [-1.0180],\n",
      "        [-1.3340],\n",
      "        [-1.7980],\n",
      "        [-1.8580],\n",
      "        [ 2.6520],\n",
      "        [ 2.4040]])), (tensor([[2.4168],\n",
      "        [2.3510],\n",
      "        [2.2806],\n",
      "        [2.1743],\n",
      "        [2.0490],\n",
      "        [1.9111],\n",
      "        [1.7473],\n",
      "        [1.6211],\n",
      "        [1.5421],\n",
      "        [1.4603],\n",
      "        [1.3803],\n",
      "        [1.2872],\n",
      "        [1.2024],\n",
      "        [1.0382],\n",
      "        [0.4301],\n",
      "        [0.4182],\n",
      "        [2.6853],\n",
      "        [2.4356],\n",
      "        [2.3559],\n",
      "        [2.2234],\n",
      "        [2.0128],\n",
      "        [1.7972],\n",
      "        [1.7110],\n",
      "        [1.5009],\n",
      "        [1.3089],\n",
      "        [1.0938],\n",
      "        [0.9041],\n",
      "        [0.7114],\n",
      "        [0.5406],\n",
      "        [0.4611],\n",
      "        [0.4376],\n",
      "        [0.4154]]), tensor([[ 2.1890],\n",
      "        [ 2.0130],\n",
      "        [ 1.8410],\n",
      "        [ 1.5950],\n",
      "        [ 1.3760],\n",
      "        [ 1.1010],\n",
      "        [ 0.8320],\n",
      "        [ 0.5270],\n",
      "        [ 0.3400],\n",
      "        [ 0.1730],\n",
      "        [-0.0280],\n",
      "        [-0.2290],\n",
      "        [-0.4130],\n",
      "        [-0.7740],\n",
      "        [-2.2430],\n",
      "        [-2.3760],\n",
      "        [ 2.5420],\n",
      "        [ 2.3730],\n",
      "        [ 2.1720],\n",
      "        [ 1.9840],\n",
      "        [ 1.8150],\n",
      "        [ 1.6650],\n",
      "        [ 1.6020],\n",
      "        [ 1.3980],\n",
      "        [ 1.2290],\n",
      "        [ 1.0600],\n",
      "        [ 0.8970],\n",
      "        [ 0.7520],\n",
      "        [ 0.5930],\n",
      "        [ 0.3630],\n",
      "        [ 0.1400],\n",
      "        [-0.1200]])), (tensor([[0.4222],\n",
      "        [0.4329],\n",
      "        [0.4431],\n",
      "        [0.4628],\n",
      "        [0.4754],\n",
      "        [2.6853],\n",
      "        [2.4204],\n",
      "        [2.3042],\n",
      "        [2.1143],\n",
      "        [1.9409],\n",
      "        [1.7372],\n",
      "        [1.5799],\n",
      "        [1.1345],\n",
      "        [0.4595],\n",
      "        [0.4248],\n",
      "        [0.4422],\n",
      "        [0.4580],\n",
      "        [0.4732]]), tensor([[-0.3510],\n",
      "        [-0.5600],\n",
      "        [-0.6430],\n",
      "        [-0.9960],\n",
      "        [-1.0910],\n",
      "        [ 2.4240],\n",
      "        [ 2.1230],\n",
      "        [ 1.8130],\n",
      "        [ 1.5620],\n",
      "        [ 1.2880],\n",
      "        [ 1.0500],\n",
      "        [ 0.8550],\n",
      "        [ 0.9720],\n",
      "        [ 0.7700],\n",
      "        [ 0.6120],\n",
      "        [ 0.3450],\n",
      "        [ 0.0930],\n",
      "        [-0.0580]]))]\n"
     ]
    }
   ],
   "source": [
    "# Load Dataset\n",
    "train_data_path = \"beach_profile_data/processed_data/beachdata_train.xlsx\"\n",
    "test_data_path = \"beach_profile_data/processed_data/beachdata_test.xlsx\"\n",
    "\n",
    "train_dataset = BeachProfileDataset(train_data_path)\n",
    "test_dataset = BeachProfileDataset(test_data_path)\n",
    "\n",
    "train_loader = DataLoader(train_dataset, batch_size=32, shuffle=True)\n",
    "test_loader = DataLoader(test_dataset, batch_size=32, shuffle=False)\n",
    "\n",
    "# Initialize Model, Loss Function and Optimizer\n",
    "model = BeachProfileModel()\n",
    "criterion = nn.MSELoss()\n",
    "optimizer = optim.Adam(model.parameters(), lr=0.001)\n",
    "epochs = 50\n",
    "\n",
    "# Train the model\n",
    "train_model(train_loader, model, criterion, optimizer, epochs)\n",
    "\n",
    "# Evaluate the model\n",
    "test_loss = evaluate_model(test_loader, model, criterion)\n",
    "print(test_loss)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "id": "fd693194",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "import pandas as pd\n",
      "import numpy as np\n",
      "from sklearn.preprocessing import StandardScaler, OneHotEncoder\n",
      "from sklearn.compose import ColumnTransformer\n",
      "from sklearn.pipeline import Pipeline\n",
      "from torch import nn\n",
      "import torch\n",
      "from torch.utils.data import DataLoader, TensorDataset\n",
      "\n",
      "def load_and_preprocess_data():\n",
      "    # Load the datasets using pandas\n",
      "    train_data = pd.read_excel(\"beach_profile_data/processed_data/beachdata_train.xlsx\")\n",
      "    test_data = pd.read_excel(\"beach_profile_data/processed_data/beachdata_test.xlsx\")\n",
      "    \n",
      "    # Define relevant columns, including features and target\n",
      "    features = ['Annual Mean Period', 'Annual Mean Spring Tidal Range', \n",
      "                'Annual Mean Tidal Range', 'Breaker Wave Height Hb', \n",
      "                'Deep Water Wave Height Hd', 'Dimensionless Settling Velocity', \n",
      "                'High Tide Sediment Settling Velocity', 'Intertidal Slope', \n",
      "                'Kurtosis', 'Latitude', 'Longitude', 'Mean Grain Size', \n",
      "                'Mean Grain Size (Mz)', 'Mean Wave Height', 'Period', \n",
      "                'Profile Azimuth', 'Profile Length', 'Relative Tidal Range', \n",
      "                'Skewness', 'Sorting Coefficient', 'x']\n",
      "    \n",
      "    # Separate features and target variable\n",
      "    X_train = train_data[features]\n",
      "    y_train = train_data['y']\n",
      "    X_test = test_data[features]\n",
      "    y_test = test_data['y']\n",
      "    \n",
      "    # Preprocess the 'Dominant Wave Direction' feature (one-hot encoding)\n",
      "    preprocessor = ColumnTransformer(\n",
      "        transformers=[\n",
      "            ('num', StandardScaler(), features),\n",
      "            ('cat', OneHotEncoder(), ['Dominant Wave Direction'])\n",
      "        ])\n",
      "    \n",
      "    # Create the pipeline for preprocessing\n",
      "    X_train = preprocessor.fit_transform(train_data)\n",
      "    X_test = preprocessor.transform(test_data)\n",
      "    \n",
      "    return X_train, y_train, X_test, y_test\n",
      "\n",
      "def apply_bruun_dean_models(dataframe):\n",
      "    # Bruun rule is generally linear, Dean model is power law (x^(2/3) relationship)\n",
      "    # Create initial predictions using the equations\n",
      "    dataframe['Bruun_Prediction'] = dataframe['x']  # Placeholder, usually needs a coefficient\n",
      "    dataframe['Dean_Prediction'] = dataframe['x'] ** (2 / 3)\n",
      "    return dataframe\n",
      "\n",
      "def build_deep_learning_model(inputshape):\n",
      "    class SimpleMLP(nn.Module):\n",
      "        def __init__(self):\n",
      "            super(SimpleMLP, self).__init__()\n",
      "            self.fc1 = nn.Linear(inputshape, 64)\n",
      "            self.fc2 = nn.Linear(64, 32)\n",
      "            self.fc3 = nn.Linear(32, 1)\n",
      "            self.relu = nn.ReLU()\n",
      "        \n",
      "        def forward(self, x):\n",
      "            x = self.relu(self.fc1(x))\n",
      "            x = self.relu(self.fc2(x))\n",
      "            x = self.fc3(x)\n",
      "            return x\n",
      "    \n",
      "    model = SimpleMLP()\n",
      "    return model\n",
      "\n",
      "def train_deep_learning_model(model, X_train, y_train):\n",
      "    # Prepare data loader for training\n",
      "    train_dataset = TensorDataset(torch.tensor(X_train, dtype=torch.float32), torch.tensor(y_train.values, dtype=torch.float32))\n",
      "    train_loader = DataLoader(train_dataset, batch_size=32, shuffle=True)\n",
      "    \n",
      "    # Initialize optimizer and loss function\n",
      "    criterion = nn.MSELoss()\n",
      "    optimizer = torch.optim.Adam(model.parameters(), lr=0.001)\n",
      "    \n",
      "    # Training loop\n",
      "    for epoch in range(100):  # Simple fixed number of epochs for illustration\n",
      "        for X_batch, y_batch in train_loader:\n",
      "            optimizer.zero_grad()\n",
      "            outputs = model(X_batch)\n",
      "            loss = criterion(outputs.flatten(), y_batch)\n",
      "            loss.backward()\n",
      "            optimizer.step()\n",
      "    \n",
      "    return model\n",
      "\n",
      "def evaluate_deep_learning_model(model, X_test, y_test):\n",
      "    # Prepare data loader for testing\n",
      "    test_dataset = TensorDataset(torch.tensor(X_test, dtype=torch.float32), torch.tensor(y_test.values, dtype=torch.float32))\n",
      "    test_loader = DataLoader(test_dataset, batch_size=32, shuffle=False)\n",
      "    \n",
      "    # Set the model to evaluation mode\n",
      "    model.eval()\n",
      "    \n",
      "    # Initialize loss and performance metrics\n",
      "    criterion = nn.MSELoss()\n",
      "    total_loss = 0\n",
      "    \n",
      "    with torch.no_grad():\n",
      "        for X_batch, y_batch in test_loader:\n",
      "            outputs = model(X_batch)\n",
      "            loss = criterion(outputs.flatten(), y_batch)\n",
      "            total_loss += loss.item()\n",
      "    \n",
      "    average_loss = total_loss / len(test_loader)\n",
      "    results = {\n",
      "        \"Mean Squared Error\": average_loss,\n",
      "    }\n",
      "    return results\n",
      "\n",
      "def main():\n",
      "    # Load and preprocess the data\n",
      "    X_train, y_train, X_test, y_test = load_and_preprocess_data()\n",
      "\n",
      "    # Load dataframe and apply the Bruun and Dean models\n",
      "    df_train = pd.DataFrame(X_train, columns=['x'] + list(range(X_train.shape[1] - 1)))\n",
      "    df_train = apply_bruun_dean_models(df_train)\n",
      "\n",
      "    # Build the deep learning model\n",
      "    model = build_deep_learning_model(X_train.shape[1])\n",
      "\n",
      "    # Train the model\n",
      "    trained_model = train_deep_learning_model(model, X_train, y_train)\n",
      "\n",
      "    # Evaluate the model\n",
      "    evaluation_results = evaluate_deep_learning_model(trained_model, X_test, y_test)\n",
      "\n",
      "    return evaluation_results\n"
     ]
    }
   ],
   "source": [
    "print(\"import pandas as pd\\nimport numpy as np\\nfrom sklearn.preprocessing import StandardScaler, OneHotEncoder\\nfrom sklearn.compose import ColumnTransformer\\nfrom sklearn.pipeline import Pipeline\\nfrom torch import nn\\nimport torch\\nfrom torch.utils.data import DataLoader, TensorDataset\\n\\ndef load_and_preprocess_data():\\n    # Load the datasets using pandas\\n    train_data = pd.read_excel(\\\"beach_profile_data/processed_data/beachdata_train.xlsx\\\")\\n    test_data = pd.read_excel(\\\"beach_profile_data/processed_data/beachdata_test.xlsx\\\")\\n    \\n    # Define relevant columns, including features and target\\n    features = ['Annual Mean Period', 'Annual Mean Spring Tidal Range', \\n                'Annual Mean Tidal Range', 'Breaker Wave Height Hb', \\n                'Deep Water Wave Height Hd', 'Dimensionless Settling Velocity', \\n                'High Tide Sediment Settling Velocity', 'Intertidal Slope', \\n                'Kurtosis', 'Latitude', 'Longitude', 'Mean Grain Size', \\n                'Mean Grain Size (Mz)', 'Mean Wave Height', 'Period', \\n                'Profile Azimuth', 'Profile Length', 'Relative Tidal Range', \\n                'Skewness', 'Sorting Coefficient', 'x']\\n    \\n    # Separate features and target variable\\n    X_train = train_data[features]\\n    y_train = train_data['y']\\n    X_test = test_data[features]\\n    y_test = test_data['y']\\n    \\n    # Preprocess the 'Dominant Wave Direction' feature (one-hot encoding)\\n    preprocessor = ColumnTransformer(\\n        transformers=[\\n            ('num', StandardScaler(), features),\\n            ('cat', OneHotEncoder(), ['Dominant Wave Direction'])\\n        ])\\n    \\n    # Create the pipeline for preprocessing\\n    X_train = preprocessor.fit_transform(train_data)\\n    X_test = preprocessor.transform(test_data)\\n    \\n    return X_train, y_train, X_test, y_test\\n\\ndef apply_bruun_dean_models(dataframe):\\n    # Bruun rule is generally linear, Dean model is power law (x^(2/3) relationship)\\n    # Create initial predictions using the equations\\n    dataframe['Bruun_Prediction'] = dataframe['x']  # Placeholder, usually needs a coefficient\\n    dataframe['Dean_Prediction'] = dataframe['x'] ** (2 / 3)\\n    return dataframe\\n\\ndef build_deep_learning_model(inputshape):\\n    class SimpleMLP(nn.Module):\\n        def __init__(self):\\n            super(SimpleMLP, self).__init__()\\n            self.fc1 = nn.Linear(inputshape, 64)\\n            self.fc2 = nn.Linear(64, 32)\\n            self.fc3 = nn.Linear(32, 1)\\n            self.relu = nn.ReLU()\\n        \\n        def forward(self, x):\\n            x = self.relu(self.fc1(x))\\n            x = self.relu(self.fc2(x))\\n            x = self.fc3(x)\\n            return x\\n    \\n    model = SimpleMLP()\\n    return model\\n\\ndef train_deep_learning_model(model, X_train, y_train):\\n    # Prepare data loader for training\\n    train_dataset = TensorDataset(torch.tensor(X_train, dtype=torch.float32), torch.tensor(y_train.values, dtype=torch.float32))\\n    train_loader = DataLoader(train_dataset, batch_size=32, shuffle=True)\\n    \\n    # Initialize optimizer and loss function\\n    criterion = nn.MSELoss()\\n    optimizer = torch.optim.Adam(model.parameters(), lr=0.001)\\n    \\n    # Training loop\\n    for epoch in range(100):  # Simple fixed number of epochs for illustration\\n        for X_batch, y_batch in train_loader:\\n            optimizer.zero_grad()\\n            outputs = model(X_batch)\\n            loss = criterion(outputs.flatten(), y_batch)\\n            loss.backward()\\n            optimizer.step()\\n    \\n    return model\\n\\ndef evaluate_deep_learning_model(model, X_test, y_test):\\n    # Prepare data loader for testing\\n    test_dataset = TensorDataset(torch.tensor(X_test, dtype=torch.float32), torch.tensor(y_test.values, dtype=torch.float32))\\n    test_loader = DataLoader(test_dataset, batch_size=32, shuffle=False)\\n    \\n    # Set the model to evaluation mode\\n    model.eval()\\n    \\n    # Initialize loss and performance metrics\\n    criterion = nn.MSELoss()\\n    total_loss = 0\\n    \\n    with torch.no_grad():\\n        for X_batch, y_batch in test_loader:\\n            outputs = model(X_batch)\\n            loss = criterion(outputs.flatten(), y_batch)\\n            total_loss += loss.item()\\n    \\n    average_loss = total_loss / len(test_loader)\\n    results = {\\n        \\\"Mean Squared Error\\\": average_loss,\\n    }\\n    return results\\n\\ndef main():\\n    # Load and preprocess the data\\n    X_train, y_train, X_test, y_test = load_and_preprocess_data()\\n\\n    # Load dataframe and apply the Bruun and Dean models\\n    df_train = pd.DataFrame(X_train, columns=['x'] + list(range(X_train.shape[1] - 1)))\\n    df_train = apply_bruun_dean_models(df_train)\\n\\n    # Build the deep learning model\\n    model = build_deep_learning_model(X_train.shape[1])\\n\\n    # Train the model\\n    trained_model = train_deep_learning_model(model, X_train, y_train)\\n\\n    # Evaluate the model\\n    evaluation_results = evaluate_deep_learning_model(trained_model, X_test, y_test)\\n\\n    return evaluation_results\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "id": "f39236fc",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n",
    "from sklearn.preprocessing import StandardScaler, OneHotEncoder\n",
    "from sklearn.compose import ColumnTransformer\n",
    "from sklearn.pipeline import Pipeline\n",
    "from torch import nn\n",
    "import torch\n",
    "from torch.utils.data import DataLoader, TensorDataset\n",
    "\n",
    "def load_and_preprocess_data():\n",
    "    # Load the datasets using pandas\n",
    "    train_data = pd.read_excel(\"beach_profile_data/processed_data/beachdata_train.xlsx\")\n",
    "    test_data = pd.read_excel(\"beach_profile_data/processed_data/beachdata_test.xlsx\")\n",
    "    \n",
    "    # Define relevant columns, including features and target\n",
    "    features = ['Annual Mean Period', 'Annual Mean Spring Tidal Range', \n",
    "                'Annual Mean Tidal Range', 'Breaker Wave Height Hb', \n",
    "                'Deep Water Wave Height Hd', 'Dimensionless Settling Velocity', \n",
    "                'High Tide Sediment Settling Velocity', 'Intertidal Slope', \n",
    "                'Kurtosis', 'Latitude', 'Longitude', 'Mean Grain Size', \n",
    "                'Mean Grain Size (Mz)', 'Mean Wave Height', 'Period', \n",
    "                'Profile Azimuth', 'Profile Length', 'Relative Tidal Range', \n",
    "                'Skewness', 'Sorting Coefficient', 'x']\n",
    "    \n",
    "    # Separate features and target variable\n",
    "    X_train = train_data[features]\n",
    "    y_train = train_data['y']\n",
    "    X_test = test_data[features]\n",
    "    y_test = test_data['y']\n",
    "    \n",
    "    # Preprocess the 'Dominant Wave Direction' feature (one-hot encoding)\n",
    "    preprocessor = ColumnTransformer(\n",
    "        transformers=[\n",
    "            ('num', StandardScaler(), features),\n",
    "            ('cat', OneHotEncoder(), ['Dominant Wave Direction'])\n",
    "        ])\n",
    "    \n",
    "    # Create the pipeline for preprocessing\n",
    "    X_train = preprocessor.fit_transform(train_data)\n",
    "    X_test = preprocessor.transform(test_data)\n",
    "    \n",
    "    return X_train, y_train, X_test, y_test\n",
    "\n",
    "def apply_bruun_dean_models(dataframe):\n",
    "    # Bruun rule is generally linear, Dean model is power law (x^(2/3) relationship)\n",
    "    # Create initial predictions using the equations\n",
    "    dataframe['Bruun_Prediction'] = dataframe['x']  # Placeholder, usually needs a coefficient\n",
    "    dataframe['Dean_Prediction'] = dataframe['x'] ** (2 / 3)\n",
    "    return dataframe\n",
    "\n",
    "def build_deep_learning_model(inputshape):\n",
    "    class SimpleMLP(nn.Module):\n",
    "        def __init__(self):\n",
    "            super(SimpleMLP, self).__init__()\n",
    "            self.fc1 = nn.Linear(inputshape, 64)\n",
    "            self.fc2 = nn.Linear(64, 32)\n",
    "            self.fc3 = nn.Linear(32, 1)\n",
    "            self.relu = nn.ReLU()\n",
    "        \n",
    "        def forward(self, x):\n",
    "            x = self.relu(self.fc1(x))\n",
    "            x = self.relu(self.fc2(x))\n",
    "            x = self.fc3(x)\n",
    "            return x\n",
    "    \n",
    "    model = SimpleMLP()\n",
    "    return model\n",
    "\n",
    "def train_deep_learning_model(model, X_train, y_train):\n",
    "    # Prepare data loader for training\n",
    "    train_dataset = TensorDataset(torch.tensor(X_train, dtype=torch.float32), torch.tensor(y_train.values, dtype=torch.float32))\n",
    "    train_loader = DataLoader(train_dataset, batch_size=32, shuffle=True)\n",
    "    \n",
    "    # Initialize optimizer and loss function\n",
    "    criterion = nn.MSELoss()\n",
    "    optimizer = torch.optim.Adam(model.parameters(), lr=0.001)\n",
    "    \n",
    "    # Training loop\n",
    "    for epoch in range(100):  # Simple fixed number of epochs for illustration\n",
    "        for X_batch, y_batch in train_loader:\n",
    "            optimizer.zero_grad()\n",
    "            outputs = model(X_batch)\n",
    "            loss = criterion(outputs.flatten(), y_batch)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "    \n",
    "    return model\n",
    "\n",
    "def evaluate_deep_learning_model(model, X_test, y_test):\n",
    "    # Prepare data loader for testing\n",
    "    test_dataset = TensorDataset(torch.tensor(X_test, dtype=torch.float32), torch.tensor(y_test.values, dtype=torch.float32))\n",
    "    test_loader = DataLoader(test_dataset, batch_size=32, shuffle=False)\n",
    "    \n",
    "    # Set the model to evaluation mode\n",
    "    model.eval()\n",
    "    \n",
    "    # Initialize loss and performance metrics\n",
    "    criterion = nn.MSELoss()\n",
    "    total_loss = 0\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        for X_batch, y_batch in test_loader:\n",
    "            outputs = model(X_batch)\n",
    "            loss = criterion(outputs.flatten(), y_batch)\n",
    "            total_loss += loss.item()\n",
    "    \n",
    "    average_loss = total_loss / len(test_loader)\n",
    "    results = {\n",
    "        \"Mean Squared Error\": average_loss,\n",
    "    }\n",
    "    return results\n",
    "\n",
    "def main():\n",
    "    # Load and preprocess the data\n",
    "    X_train, y_train, X_test, y_test = load_and_preprocess_data()\n",
    "\n",
    "    # Load dataframe and apply the Bruun and Dean models\n",
    "    df_train = pd.DataFrame(X_train, columns=['x'] + list(range(X_train.shape[1] - 1)))\n",
    "    df_train = apply_bruun_dean_models(df_train)\n",
    "\n",
    "    # Build the deep learning model\n",
    "    model = build_deep_learning_model(X_train.shape[1])\n",
    "\n",
    "    # Train the model\n",
    "    trained_model = train_deep_learning_model(model, X_train, y_train)\n",
    "\n",
    "    # Evaluate the model\n",
    "    evaluation_results = evaluate_deep_learning_model(trained_model, X_test, y_test)\n",
    "\n",
    "    return evaluation_results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "id": "e381564a",
   "metadata": {},
   "outputs": [],
   "source": [
    "result_temp = main()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 49,
   "id": "6e3e1fd1",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'Mean Squared Error': 1.0713630467653275}\n"
     ]
    }
   ],
   "source": [
    "print(result_temp)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "ea198e8f",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "from sklearn.preprocessing import LabelEncoder, StandardScaler\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "# Bruun model: y = y0 - (S / (L + x))\n",
    "def bruun_model(x, y0, S, L):\n",
    "    \"\"\"\n",
    "    Bruun model for sea level change with respect to distance.\n",
    "    x: distance from origin\n",
    "    y0: initial elevation\n",
    "    S: sea level rise\n",
    "    L: width of active profile\n",
    "    Returns: elevation at distance x\n",
    "    \"\"\"\n",
    "    return y0 - (S / (L + x))\n",
    "\n",
    "# Dean model: y = A * x^(2/3)\n",
    "def dean_model(x, A):\n",
    "    \"\"\"\n",
    "    Dean model for equilibrium beach profile.\n",
    "    x: distance from origin\n",
    "    A: dimensional profile scale parameter\n",
    "    Returns: elevation at distance x\n",
    "    \"\"\"\n",
    "    return A * np.power(x, 2/3)\n",
    "\n",
    "class BeachProfileNet(nn.Module):\n",
    "    def __init__(self, input_dim):\n",
    "        super(BeachProfileNet, self).__init__()\n",
    "        self.model = nn.Sequential(\n",
    "            nn.Linear(input_dim, 64),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(64, 32),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(32, 1)\n",
    "        )\n",
    "    def forward(self, x):\n",
    "        return self.model(x)\n",
    "\n",
    "def load_and_preprocess_data(train_path, test_path):\n",
    "    # Load data\n",
    "    train_df = pd.read_excel(train_path)\n",
    "    test_df = pd.read_excel(test_path)\n",
    "\n",
    "    # Concatenate for consistent encoding\n",
    "    full_df = pd.concat([train_df, test_df], axis=0, ignore_index=True)\n",
    "\n",
    "    # Encode 'Dominant Wave Direction'\n",
    "    if 'Dominant Wave Direction' in full_df.columns:\n",
    "        le = LabelEncoder()\n",
    "        full_df['Dominant Wave Direction'] = le.fit_transform(full_df['Dominant Wave Direction'].astype(str))\n",
    "\n",
    "    # Select features\n",
    "    feature_cols = [col for col in full_df.columns if col not in ['y']]\n",
    "    X = full_df[feature_cols].values\n",
    "    y = full_df['y'].values.reshape(-1, 1)\n",
    "\n",
    "    # Standardize features\n",
    "    scaler_X = StandardScaler()\n",
    "    scaler_y = StandardScaler()\n",
    "    X_scaled = scaler_X.fit_transform(X)\n",
    "    y_scaled = scaler_y.fit_transform(y)\n",
    "\n",
    "    # Split back to train/test\n",
    "    X_train = X_scaled[:len(train_df)]\n",
    "    y_train = y_scaled[:len(train_df)]\n",
    "    X_test = X_scaled[len(train_df):]\n",
    "    y_test = y_scaled[len(train_df):]\n",
    "\n",
    "    return X_train, y_train, X_test, y_test, scaler_X, scaler_y, feature_cols\n",
    "\n",
    "def train_model(X_train, y_train, input_dim, epochs=100, lr=0.001):\n",
    "    model = BeachProfileNet(input_dim)\n",
    "    criterion = nn.MSELoss()\n",
    "    optimizer = optim.Adam(model.parameters(), lr=lr)\n",
    "    X_train_tensor = torch.tensor(X_train, dtype=torch.float32)\n",
    "    y_train_tensor = torch.tensor(y_train, dtype=torch.float32)\n",
    "    for epoch in range(epochs):\n",
    "        model.train()\n",
    "        optimizer.zero_grad()\n",
    "        outputs = model(X_train_tensor)\n",
    "        loss = criterion(outputs, y_train_tensor)\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "    return model\n",
    "\n",
    "def evaluate_model(model, X_test, y_test, scaler_y):\n",
    "    model.eval()\n",
    "    X_test_tensor = torch.tensor(X_test, dtype=torch.float32)\n",
    "    with torch.no_grad():\n",
    "        preds = model(X_test_tensor).numpy()\n",
    "    preds_rescaled = scaler_y.inverse_transform(preds)\n",
    "    y_test_rescaled = scaler_y.inverse_transform(y_test)\n",
    "    mse = np.mean((preds_rescaled - y_test_rescaled) ** 2)\n",
    "    return mse, preds_rescaled, y_test_rescaled\n",
    "\n",
    "def main():\n",
    "    train_path = \"beach_profile_data/processed_data/beachdata_train.xlsx\"\n",
    "    test_path = \"beach_profile_data/processed_data/beachdata_test.xlsx\"\n",
    "    X_train, y_train, X_test, y_test, scaler_X, scaler_y, feature_cols = load_and_preprocess_data(train_path, test_path)\n",
    "    input_dim = X_train.shape[1]\n",
    "    model = train_model(X_train, y_train, input_dim)\n",
    "    mse, preds, y_true = evaluate_model(model, X_test, y_test, scaler_y)\n",
    "    print(f\"Test MSE: {mse:.4f}\")\n",
    "    # Example usage of Bruun and Dean models\n",
    "    x_example = np.linspace(0, 100, 100)\n",
    "    y_bruun = bruun_model(x_example, y0=5, S=1, L=50)\n",
    "    y_dean = dean_model(x_example, A=0.1)\n",
    "    print(\"Bruun model example output (first 5):\", y_bruun[:5])\n",
    "    print(\"Dean model example output (first 5):\", y_dean[:5])\n",
    "\n",
    "# The script does not execute anything unless main() is called. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "538231a3",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test MSE: 0.8063\n",
      "Bruun model example output (first 5): [4.98       4.98039604 4.9807767  4.98114286 4.98149533]\n",
      "Dean model example output (first 5): [0.         0.10067227 0.15980727 0.20940676 0.25367823]\n"
     ]
    }
   ],
   "source": [
    "main()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "d11de23b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "from torch.utils.data import Dataset, DataLoader\n",
    "from sklearn.preprocessing import LabelEncoder, StandardScaler\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "# -----------------------------\n",
    "# 1. Mathematical Models\n",
    "# -----------------------------\n",
    "\n",
    "def bruun_model(x, S, L, h, B):\n",
    "    \"\"\"\n",
    "    Bruun Rule: y(x) = S - (S / L) * (x - (h * L / S) + B)\n",
    "    S: Sea level rise\n",
    "    L: Width of active profile\n",
    "    h: Depth of closure\n",
    "    B: Berm height\n",
    "    x: Distance from origin (numpy array or float)\n",
    "    Returns predicted elevation y.\n",
    "    \"\"\"\n",
    "    return S - (S / L) * (x - (h * L / S) + B)\n",
    "\n",
    "def dean_model(x, A):\n",
    "    \"\"\"\n",
    "    Dean Profile: y(x) = A * x^(2/3)\n",
    "    A: Dean parameter (related to sediment size)\n",
    "    x: Distance from origin (numpy array or float)\n",
    "    Returns predicted elevation y.\n",
    "    \"\"\"\n",
    "    return A * np.power(x, 2/3)\n",
    "\n",
    "# -----------------------------\n",
    "# 2. Dataset and Preprocessing\n",
    "# -----------------------------\n",
    "\n",
    "class BeachProfileDataset(Dataset):\n",
    "    def __init__(self, df, feature_cols, target_col, scaler=None, label_encoders=None, fit_scaler=False):\n",
    "        self.X = df[feature_cols].copy()\n",
    "        self.y = df[target_col].values.astype(np.float32)\n",
    "        self.label_encoders = label_encoders or {}\n",
    "        # Encode categorical features\n",
    "        for col in self.X.select_dtypes(include=['object', 'category']).columns:\n",
    "            if col not in self.label_encoders:\n",
    "                self.label_encoders[col] = LabelEncoder()\n",
    "                self.X[col] = self.label_encoders[col].fit_transform(self.X[col])\n",
    "            else:\n",
    "                self.X[col] = self.label_encoders[col].transform(self.X[col])\n",
    "        # Scale features\n",
    "        if scaler is None:\n",
    "            scaler = StandardScaler()\n",
    "            if fit_scaler:\n",
    "                self.X = scaler.fit_transform(self.X)\n",
    "            else:\n",
    "                self.X = scaler.transform(self.X)\n",
    "        else:\n",
    "            if fit_scaler:\n",
    "                self.X = scaler.fit_transform(self.X)\n",
    "            else:\n",
    "                self.X = scaler.transform(self.X)\n",
    "        self.X = self.X.astype(np.float32)\n",
    "        self.scaler = scaler\n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self.y)\n",
    "\n",
    "    def __getitem__(self, idx):\n",
    "        return self.X[idx], self.y[idx]\n",
    "\n",
    "# -----------------------------\n",
    "# 3. Neural Network Model\n",
    "# -----------------------------\n",
    "\n",
    "class BeachProfileNet(nn.Module):\n",
    "    def __init__(self, input_dim):\n",
    "        super().__init__()\n",
    "        self.model = nn.Sequential(\n",
    "            nn.Linear(input_dim, 64),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(64, 32),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(32, 1)\n",
    "        )\n",
    "    def forward(self, x):\n",
    "        return self.model(x).squeeze(-1)\n",
    "\n",
    "# -----------------------------\n",
    "# 4. Training and Evaluation\n",
    "# -----------------------------\n",
    "\n",
    "def train_model(model, dataloader, criterion, optimizer, device, epochs=50):\n",
    "    model.train()\n",
    "    for epoch in range(epochs):\n",
    "        epoch_loss = 0.0\n",
    "        for X_batch, y_batch in dataloader:\n",
    "            X_batch = X_batch.to(device)\n",
    "            y_batch = y_batch.to(device)\n",
    "            optimizer.zero_grad()\n",
    "            outputs = model(X_batch)\n",
    "            loss = criterion(outputs, y_batch)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            epoch_loss += loss.item() * len(y_batch)\n",
    "        avg_loss = epoch_loss / len(dataloader.dataset)\n",
    "        if epoch % 10 == 0 or epoch == epochs - 1:\n",
    "            print(f\"Epoch {epoch+1}/{epochs}, Loss: {avg_loss:.4f}\")\n",
    "    return model\n",
    "\n",
    "def evaluate_model(model, dataloader, criterion, device):\n",
    "    model.eval()\n",
    "    total_loss = 0.0\n",
    "    preds = []\n",
    "    trues = []\n",
    "    with torch.no_grad():\n",
    "        for X_batch, y_batch in dataloader:\n",
    "            X_batch = X_batch.to(device)\n",
    "            y_batch = y_batch.to(device)\n",
    "            outputs = model(X_batch)\n",
    "            loss = criterion(outputs, y_batch)\n",
    "            total_loss += loss.item() * len(y_batch)\n",
    "            preds.append(outputs.cpu().numpy())\n",
    "            trues.append(y_batch.cpu().numpy())\n",
    "    avg_loss = total_loss / len(dataloader.dataset)\n",
    "    preds = np.concatenate(preds)\n",
    "    trues = np.concatenate(trues)\n",
    "    return avg_loss, preds, trues\n",
    "\n",
    "# -----------------------------\n",
    "# 5. Main Function\n",
    "# -----------------------------\n",
    "\n",
    "def main():\n",
    "    # Data paths\n",
    "    train_path = \"beach_profile_data/processed_data/beachdata_train.xlsx\"\n",
    "    test_path = \"beach_profile_data/processed_data/beachdata_test.xlsx\"\n",
    "\n",
    "    # Load data\n",
    "    train_df = pd.read_excel(train_path)\n",
    "    test_df = pd.read_excel(test_path)\n",
    "\n",
    "    # Feature selection\n",
    "    # Use all numeric columns and 'Dominant Wave Direction' as features, except 'y' (target)\n",
    "    feature_cols = [col for col in train_df.columns if col not in ['y']]\n",
    "    target_col = 'y'\n",
    "\n",
    "    # Find categorical columns\n",
    "    cat_cols = train_df.select_dtypes(include=['object', 'category']).columns.tolist()\n",
    "    # Ensure 'Dominant Wave Direction' is encoded\n",
    "    if 'Dominant Wave Direction' not in cat_cols:\n",
    "        cat_cols.append('Dominant Wave Direction')\n",
    "\n",
    "    # Fit label encoders on train, apply to test\n",
    "    label_encoders = {}\n",
    "    for col in cat_cols:\n",
    "        le = LabelEncoder()\n",
    "        train_df[col] = le.fit_transform(train_df[col])\n",
    "        test_df[col] = le.transform(test_df[col])\n",
    "        label_encoders[col] = le\n",
    "\n",
    "    # Standardize features\n",
    "    scaler = StandardScaler()\n",
    "    train_X = scaler.fit_transform(train_df[feature_cols])\n",
    "    test_X = scaler.transform(test_df[feature_cols])\n",
    "\n",
    "    # Dataset and DataLoader\n",
    "    train_dataset = BeachProfileDataset(train_df, feature_cols, target_col, scaler=scaler, label_encoders=label_encoders, fit_scaler=False)\n",
    "    test_dataset = BeachProfileDataset(test_df, feature_cols, target_col, scaler=scaler, label_encoders=label_encoders, fit_scaler=False)\n",
    "\n",
    "    train_loader = DataLoader(train_dataset, batch_size=32, shuffle=True)\n",
    "    test_loader = DataLoader(test_dataset, batch_size=32, shuffle=False)\n",
    "\n",
    "    # Model\n",
    "    device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "    model = BeachProfileNet(input_dim=len(feature_cols)).to(device)\n",
    "    criterion = nn.MSELoss()\n",
    "    optimizer = torch.optim.Adam(model.parameters(), lr=0.001)\n",
    "\n",
    "    # Train\n",
    "    model = train_model(model, train_loader, criterion, optimizer, device, epochs=50)\n",
    "    \n",
    "    # Evaluate\n",
    "    test_loss, preds, trues = evaluate_model(model, test_loader, criterion, device)\n",
    "    print(f\"Test MSE Loss: {test_loss:.4f}\")\n",
    "\n",
    "    # Example usage of mathematical models\n",
    "    # Bruun model: y(x) = S - (S / L) * (x - (h * L / S) + B)\n",
    "    x = np.linspace(0, 100, 101)\n",
    "    y_bruun = bruun_model(x, S=1.0, L=100, h=10, B=2)\n",
    "    y_dean = dean_model(x, A=0.1)\n",
    "    print(\"Bruun model example output:\", y_bruun[:5])\n",
    "    print(\"Dean model example output:\", y_dean[:5])\n",
    "\n",
    "# Note: The script does NOT execute anything on import. Call main() to run the workflow.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "3c7218ee",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1/50, Loss: 2.0188\n",
      "Epoch 11/50, Loss: 0.1938\n",
      "Epoch 21/50, Loss: 0.0484\n",
      "Epoch 31/50, Loss: 0.0227\n",
      "Epoch 41/50, Loss: 0.0135\n",
      "Epoch 50/50, Loss: 0.0130\n",
      "Test MSE Loss: 0.6765\n",
      "Bruun model example output: [10.98 10.97 10.96 10.95 10.94]\n",
      "Dean model example output: [0.         0.1        0.15874011 0.20800838 0.25198421]\n"
     ]
    }
   ],
   "source": [
    "main()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "fc1af093",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "from torch.utils.data import Dataset, DataLoader\n",
    "from sklearn.preprocessing import StandardScaler, LabelEncoder\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "# --- Bruun Model ---\n",
    "def bruun_model(x, A, B):\n",
    "    \"\"\"\n",
    "    Bruun Rule: y = A - B * x\n",
    "    x: distance from starting point\n",
    "    A, B: empirical parameters\n",
    "    \"\"\"\n",
    "    return A - B * x\n",
    "\n",
    "# --- Dean Model ---\n",
    "def dean_model(x, h, A):\n",
    "    \"\"\"\n",
    "    Dean Profile: y = h - A * x^{2/3}\n",
    "    x: distance from starting point\n",
    "    h, A: empirical parameters\n",
    "    \"\"\"\n",
    "    return h - A * np.power(x, 2/3)\n",
    "\n",
    "# --- Custom Dataset ---\n",
    "class BeachProfileDataset(Dataset):\n",
    "    def __init__(self, df, feature_cols, target_col, scaler=None, label_encoders=None, fit_scaler=False):\n",
    "        self.X = df[feature_cols].copy()\n",
    "        self.y = df[target_col].values.astype(np.float32)\n",
    "        self.scaler = scaler\n",
    "        self.label_encoders = label_encoders or {}\n",
    "        # Encode categorical columns\n",
    "        for col in self.X.select_dtypes(include=['object', 'category']).columns:\n",
    "            if col not in self.label_encoders:\n",
    "                self.label_encoders[col] = LabelEncoder()\n",
    "                self.X[col] = self.label_encoders[col].fit_transform(self.X[col])\n",
    "            else:\n",
    "                self.X[col] = self.label_encoders[col].transform(self.X[col])\n",
    "        # Standardize numerical columns\n",
    "        num_cols = self.X.select_dtypes(include=[np.number]).columns\n",
    "        if fit_scaler:\n",
    "            self.scaler = StandardScaler()\n",
    "            self.X[num_cols] = self.scaler.fit_transform(self.X[num_cols])\n",
    "        else:\n",
    "            self.X[num_cols] = self.scaler.transform(self.X[num_cols])\n",
    "        self.X = self.X.values.astype(np.float32)\n",
    "    def __len__(self):\n",
    "        return len(self.y)\n",
    "    def __getitem__(self, idx):\n",
    "        return self.X[idx], self.y[idx]\n",
    "\n",
    "# --- Neural Network Model ---\n",
    "class BeachProfileNet(nn.Module):\n",
    "    def __init__(self, input_dim):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.Linear(input_dim, 64),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(64, 32),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(32, 1)\n",
    "        )\n",
    "    def forward(self, x):\n",
    "        return self.net(x).squeeze(-1)\n",
    "\n",
    "# --- Data Loading and Preprocessing ---\n",
    "def load_data(train_path, test_path):\n",
    "    train_df = pd.read_excel(train_path)\n",
    "    test_df = pd.read_excel(test_path)\n",
    "    # Select features\n",
    "    feature_cols = [col for col in train_df.columns if col not in ['y']]\n",
    "    target_col = 'y'\n",
    "    # Handle categorical encoding and scaling\n",
    "    scaler = StandardScaler()\n",
    "    label_encoders = {}\n",
    "    train_dataset = BeachProfileDataset(train_df, feature_cols, target_col, scaler=None, label_encoders=label_encoders, fit_scaler=True)\n",
    "    test_dataset = BeachProfileDataset(test_df, feature_cols, target_col, scaler=train_dataset.scaler, label_encoders=train_dataset.label_encoders, fit_scaler=False)\n",
    "    return train_dataset, test_dataset, feature_cols, label_encoders\n",
    "\n",
    "# --- Training Function ---\n",
    "def train_model(model, train_loader, num_epochs=50, lr=0.001):\n",
    "    criterion = nn.MSELoss()\n",
    "    optimizer = optim.Adam(model.parameters(), lr=lr)\n",
    "    for epoch in range(num_epochs):\n",
    "        model.train()\n",
    "        running_loss = 0.0\n",
    "        for X_batch, y_batch in train_loader:\n",
    "            optimizer.zero_grad()\n",
    "            outputs = model(X_batch)\n",
    "            loss = criterion(outputs, y_batch)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "            running_loss += loss.item() * X_batch.size(0)\n",
    "        epoch_loss = running_loss / len(train_loader.dataset)\n",
    "        # Optionally print or log epoch_loss\n",
    "    return model\n",
    "\n",
    "# --- Evaluation Function ---\n",
    "def evaluate_model(model, test_loader):\n",
    "    model.eval()\n",
    "    predictions = []\n",
    "    targets = []\n",
    "    with torch.no_grad():\n",
    "        for X_batch, y_batch in test_loader:\n",
    "            outputs = model(X_batch)\n",
    "            predictions.append(outputs.numpy())\n",
    "            targets.append(y_batch.numpy())\n",
    "    predictions = np.concatenate(predictions)\n",
    "    targets = np.concatenate(targets)\n",
    "    mse = np.mean((predictions - targets) ** 2)\n",
    "    return mse, predictions, targets\n",
    "\n",
    "# --- Main Function ---\n",
    "def main():\n",
    "    train_path = \"beach_profile_data/processed_data/beachdata_train.xlsx\"\n",
    "    test_path = \"beach_profile_data/processed_data/beachdata_test.xlsx\"\n",
    "    train_dataset, test_dataset, feature_cols, label_encoders = load_data(train_path, test_path)\n",
    "    train_loader = DataLoader(train_dataset, batch_size=32, shuffle=True)\n",
    "    test_loader = DataLoader(test_dataset, batch_size=32, shuffle=False)\n",
    "    model = BeachProfileNet(input_dim=len(feature_cols))\n",
    "    model = train_model(model, train_loader, num_epochs=100, lr=0.001)\n",
    "    mse, predictions, targets = evaluate_model(model, test_loader)\n",
    "    print(f\"Test MSE: {mse:.4f}\")\n",
    "\n",
    "    # scaler = train_dataset.scaler\n",
    "    # Example usage of Bruun and Dean models\n",
    "    # x = np.linspace(0, 100, 100)\n",
    "    # y_bruun = bruun_model(x, A=5, B=0.01)\n",
    "    # y_dean = dean_model(x, h=5, A=0.05)\n",
    "    return model, feature_cols, label_encoders\n",
    "\n",
    "# --- Predict on New Data Function ---\n",
    "def predict_on_new_data(model, data_path, feature_cols, label_encoders, target_col='y'):\n",
    "    \"\"\"\n",
    "    Use the trained model to predict y for a new data file.\n",
    "    Returns:\n",
    "        x_list: list of x values (distance)\n",
    "        true_y_list: list of true y values (elevation)\n",
    "        predict_y_list: list of predicted y values\n",
    "    \"\"\"\n",
    "    df = pd.read_excel(data_path)\n",
    "    dataset = BeachProfileDataset(df, feature_cols, target_col, scaler=None, label_encoders=label_encoders, fit_scaler=True)\n",
    "    loader = DataLoader(dataset, batch_size=32, shuffle=False)\n",
    "    model.eval()\n",
    "    preds = []\n",
    "    with torch.no_grad():\n",
    "        for X_batch, _ in loader:\n",
    "            outputs = model(X_batch)\n",
    "            preds.append(outputs.numpy())\n",
    "    predict_y_list = np.concatenate(preds)\n",
    "    x_list = df['x'].tolist() if 'x' in df.columns else []\n",
    "    true_y_list = df[target_col].tolist() if target_col in df.columns else []\n",
    "    return x_list, true_y_list, predict_y_list.tolist()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "c8a8ab3d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test MSE: 3.2944\n"
     ]
    }
   ],
   "source": [
    "model, feature_cols, label_encoders = main()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "c2de6655",
   "metadata": {},
   "outputs": [],
   "source": [
    "x, y_ture, y_predict_windsurf = predict_on_new_data(model, \"beach_profile_data/processed_data/beachdata_single.xlsx\", feature_cols, label_encoders)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "cc5252db",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2.4715735912323, 2.36767578125, 2.227688789367676, 2.086095094680786, 1.9133492708206177, 1.7348555326461792, 1.6676647663116455, 1.4444841146469116, 1.228782057762146, 0.9605395197868347, 0.660384476184845, 0.3696815073490143, 0.11881380528211594, -0.1538306027650833, -0.40887412428855896, -0.6236438751220703, -0.789034366607666, -1.0666532516479492, -1.326110601425171, -1.806679129600525, -2.1190149784088135]\n"
     ]
    }
   ],
   "source": [
    "print(y_predict_windsurf)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "3f212bfd",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 1000x600 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# x_plot, y_true_plot, y_base, y_new = base_plot_output\n",
    "x_plot = [\n",
    "    0.0, 1.55, 3.65, 5.78, 8.38, 11.1, 12.12, 15.55, 18.94, 22.8,\n",
    "    27.02, 31.77, 36.65, 44.05, 51.52, 60.24, 70.7, 87.22, 103.04, 133.36,\n",
    "    152.91\n",
    "]\n",
    "\n",
    "y_true_plot = [\n",
    "    2.542, 2.373, 2.172, 1.984, 1.815, 1.665, 1.602, 1.398, 1.229, 1.06,\n",
    "    0.897, 0.752, 0.593, 0.363, 0.14, -0.12, -0.351, -0.56, -0.643, -0.996,\n",
    "    -1.091\n",
    "]\n",
    "\n",
    "y_base = [\n",
    "    2.6769042, 2.3924885, 2.261987, 2.0599403, 1.8234198, 1.5171967,\n",
    "    1.4008808, 1.0747753, 0.8327296, 0.56248415, 0.26728857, -0.06498109,\n",
    "    -0.37953135, -0.734211, -0.93088466, -1.1578621, -1.4238784, -1.8433229,\n",
    "    -2.244999, -3.014831, -3.5112088\n",
    "]\n",
    "\n",
    "y_new = [\n",
    "    2.58881326, 2.34806979, 2.15122727, 2.05923818, 1.80919691, 1.49883877,\n",
    "    1.48476992, 1.23896293, 0.95764242, 0.79837616, 0.59810174, 0.37667743,\n",
    "    0.03520706, -0.19719144, -0.38670532, -0.70037132, -0.85904528, -1.20291644,\n",
    "    -1.46837316, -1.99559494, -2.33455231\n",
    "]\n",
    "\n",
    "y_windsurf = [2.4715735912323, 2.36767578125, 2.227688789367676, 2.086095094680786, 1.9133492708206177, 1.7348555326461792, 1.6676647663116455, 1.4444841146469116, 1.228782057762146, 0.9605395197868347, 0.660384476184845, 0.3696815073490143, 0.11881380528211594, -0.1538306027650833, -0.40887412428855896, -0.6236438751220703, -0.789034366607666, -1.0666532516479492, -1.326110601425171, -1.806679129600525, -2.1190149784088135]\n",
    "\n",
    "plt.figure(figsize=(10, 6))\n",
    "plt.plot(x_plot,    y_true_plot,        label='Ground Truth',   marker='o', linestyle='-',  color='blue')\n",
    "plt.plot(x_plot,    y_base,        label='Baseline',      marker='x', linestyle='--', color='red')\n",
    "plt.plot(x_plot,    y_new,   label='Our framework',   marker='.', linestyle=':',  color='green')\n",
    "plt.plot(x_plot,    y_windsurf,   label='Windsurf',   marker='.', linestyle=':',  color='yellow')\n",
    "\n",
    "plt.xlabel(\"X (Distance along profile)\", fontsize=24)\n",
    "plt.ylabel(\"Y (Elevation/Profile Value)\", fontsize=24)\n",
    "# plt.title(\"Beach Profile: Ground Truth vs. Baseline vs. Our framework\", fontsize=24, pad=25)\n",
    "plt.legend(fontsize=24)\n",
    "plt.grid(True)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e7dd1047",
   "metadata": {},
   "outputs": [],
   "source": [
    "x_list, true_y_list, predict_y_list = predict_on_new_data(\n",
    "    model, feature_cols, train_dataset.scaler, train_dataset.label_encoders, \"your_new_file.xlsx\"\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 47,
   "id": "40d8dc4d",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "from torch.utils.data import Dataset, DataLoader\n",
    "from sklearn.preprocessing import StandardScaler, OneHotEncoder\n",
    "from sklearn.compose import ColumnTransformer\n",
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "# --- Bruun and Dean Models ---\n",
    "def bruun_model(x, S, L, h_star):\n",
    "    \"\"\"\n",
    "    Bruun Rule: y(x) = S * (1 - x / L) for x <= L, else 0\n",
    "    S: Sea level rise (scalar)\n",
    "    L: Width of active profile (scalar)\n",
    "    h_star: Depth of closure (scalar)\n",
    "    \"\"\"\n",
    "    x = np.array(x)\n",
    "    y = np.where(x <= L, S * (1 - x / L), 0)\n",
    "    return y\n",
    "\n",
    "def dean_model(x, A):\n",
    "    \"\"\"\n",
    "    Dean Profile: y(x) = A * x^(2/3)\n",
    "    A: Profile scale parameter (scalar)\n",
    "    \"\"\"\n",
    "    x = np.array(x)\n",
    "    y = A * np.power(x, 2/3)\n",
    "    return y\n",
    "\n",
    "# --- PyTorch Dataset ---\n",
    "class BeachProfileDataset(Dataset):\n",
    "    def __init__(self, df, feature_cols, target_col, transformer=None):\n",
    "        self.X = df[feature_cols]\n",
    "        self.y = df[target_col].values.astype(np.float32)\n",
    "        if transformer:\n",
    "            self.X = transformer.transform(self.X)\n",
    "        else:\n",
    "            self.X = self.X.values.astype(np.float32)\n",
    "        self.X = torch.tensor(self.X, dtype=torch.float32)\n",
    "        self.y = torch.tensor(self.y, dtype=torch.float32)\n",
    "\n",
    "    def __len__(self):\n",
    "        return len(self.y)\n",
    "\n",
    "    def __getitem__(self, idx):\n",
    "        return self.X[idx], self.y[idx]\n",
    "\n",
    "# --- Neural Network Model ---\n",
    "class BeachProfileNet(nn.Module):\n",
    "    def __init__(self, input_dim):\n",
    "        super().__init__()\n",
    "        self.net = nn.Sequential(\n",
    "            nn.Linear(input_dim, 64),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(64, 32),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(32, 1)\n",
    "        )\n",
    "    def forward(self, x):\n",
    "        return self.net(x).squeeze(-1)\n",
    "\n",
    "# --- Data Loading and Preprocessing ---\n",
    "def load_and_preprocess(train_path, test_path):\n",
    "    train_df = pd.read_excel(train_path)\n",
    "    test_df = pd.read_excel(test_path)\n",
    "    # Identify features\n",
    "    feature_cols = [col for col in train_df.columns if col not in ['y']]\n",
    "    target_col = 'y'\n",
    "    # Identify categorical and numerical columns\n",
    "    cat_cols = [col for col in feature_cols if train_df[col].dtype == 'object']\n",
    "    num_cols = [col for col in feature_cols if col not in cat_cols]\n",
    "    # Preprocessing pipeline\n",
    "    transformer = ColumnTransformer([\n",
    "        ('num', StandardScaler(), num_cols),\n",
    "        ('cat', OneHotEncoder(sparse=False, handle_unknown='ignore'), cat_cols)\n",
    "    ])\n",
    "    transformer.fit(train_df[feature_cols])\n",
    "    # Return processed data and transformer\n",
    "    return train_df, test_df, feature_cols, target_col, transformer\n",
    "\n",
    "# --- Training Function ---\n",
    "def train_model(model, dataloader, epochs=50, lr=1e-3):\n",
    "    device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "    model = model.to(device)\n",
    "    optimizer = torch.optim.Adam(model.parameters(), lr=lr)\n",
    "    criterion = nn.MSELoss()\n",
    "    model.train()\n",
    "    for epoch in range(epochs):\n",
    "        for X_batch, y_batch in dataloader:\n",
    "            X_batch, y_batch = X_batch.to(device), y_batch.to(device)\n",
    "            optimizer.zero_grad()\n",
    "            y_pred = model(X_batch)\n",
    "            loss = criterion(y_pred, y_batch)\n",
    "            loss.backward()\n",
    "            optimizer.step()\n",
    "    return model\n",
    "\n",
    "# --- Evaluation Function ---\n",
    "def evaluate_model(model, dataloader):\n",
    "    device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "    model = model.to(device)\n",
    "    model.eval()\n",
    "    preds, targets = [], []\n",
    "    with torch.no_grad():\n",
    "        for X_batch, y_batch in dataloader:\n",
    "            X_batch = X_batch.to(device)\n",
    "            y_pred = model(X_batch).cpu().numpy()\n",
    "            preds.append(y_pred)\n",
    "            targets.append(y_batch.numpy())\n",
    "    preds = np.concatenate(preds)\n",
    "    targets = np.concatenate(targets)\n",
    "    mse = np.mean((preds - targets) ** 2)\n",
    "    return mse, preds, targets\n",
    "\n",
    "# --- Main Function ---\n",
    "def main():\n",
    "    train_path = 'beach_profile_data/processed_data/beachdata_train.xlsx'\n",
    "    test_path = 'beach_profile_data/processed_data/beachdata_test.xlsx'\n",
    "    train_df, test_df, feature_cols, target_col, transformer = load_and_preprocess(train_path, test_path)\n",
    "    # Prepare datasets\n",
    "    train_dataset = BeachProfileDataset(train_df, feature_cols, target_col, transformer)\n",
    "    test_dataset = BeachProfileDataset(test_df, feature_cols, target_col, transformer)\n",
    "    train_loader = DataLoader(train_dataset, batch_size=32, shuffle=True)\n",
    "    test_loader = DataLoader(test_dataset, batch_size=32, shuffle=False)\n",
    "    # Model\n",
    "    input_dim = train_dataset.X.shape[1]\n",
    "    model = BeachProfileNet(input_dim)\n",
    "    # Train\n",
    "    model = train_model(model, train_loader, epochs=100, lr=1e-3)\n",
    "    # Evaluate\n",
    "    mse, preds, targets = evaluate_model(model, test_loader)\n",
    "    print(f'Test MSE: {mse:.4f}')\n",
    "    # Example: Bruun and Dean model usage\n",
    "    # x = np.linspace(0, 100, 100)\n",
    "    # y_bruun = bruun_model(x, S=1, L=100, h_star=10)\n",
    "    # y_dean = dean_model(x, A=0.1)\n",
    "    # print(y_bruun, y_dean)\n",
    "    return model, feature_cols, target_col, transformer\n",
    "\n",
    "# def predict_on_new_data(model, new_data_path, feature_cols, target_col, transformer):\n",
    "#     \"\"\"\n",
    "#     Use the trained model to predict on new data.\n",
    "#     Returns:\n",
    "#         x_list: list of x values (distance from origin)\n",
    "#         true_y_list: list of true y values (elevation)\n",
    "#         predict_y_list: list of predicted y values\n",
    "#     \"\"\"\n",
    "#     # Load new data\n",
    "#     new_df = pd.read_excel(new_data_path)\n",
    "#     # Process features\n",
    "#     X_new = transformer.transform(new_df[feature_cols])\n",
    "#     X_new_tensor = torch.tensor(X_new, dtype=torch.float32)\n",
    "#     # Predict\n",
    "#     model.eval()\n",
    "#     device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "#     model = model.to(device)\n",
    "#     with torch.no_grad():\n",
    "#         preds = model(X_new_tensor.to(device)).cpu().numpy()\n",
    "#     # Extract x and y\n",
    "#     x_list = new_df['x'].tolist()\n",
    "#     true_y_list = new_df[target_col].tolist()\n",
    "#     predict_y_list = preds.tolist()\n",
    "#     return x_list, true_y_list, predict_y_list\n",
    "\n",
    "def predict_on_new_data(model, new_data_path):\n",
    "    new_df = pd.read_excel(new_data_path)\n",
    "    feature_cols = [col for col in new_df.columns if col not in ['y']]\n",
    "    target_col = 'y'\n",
    "    # Identify categorical and numerical columns\n",
    "    cat_cols = [col for col in feature_cols if new_df[col].dtype == 'object']\n",
    "    num_cols = [col for col in feature_cols if col not in cat_cols]\n",
    "    # Preprocessing pipeline\n",
    "    transformer = ColumnTransformer([\n",
    "        ('num', StandardScaler(), num_cols),\n",
    "        ('cat', OneHotEncoder(sparse=False, handle_unknown='ignore'), cat_cols)\n",
    "    ])\n",
    "    transformer.fit(new_df[feature_cols])\n",
    "\n",
    "    new_dataset = BeachProfileDataset(new_df, feature_cols, target_col, transformer)\n",
    "    new_loader = DataLoader(new_dataset, batch_size=32, shuffle=False)\n",
    "    # Predict\n",
    "    mse, preds, targets = evaluate_model(model, new_loader)\n",
    "    print(f'New Data MSE: {mse:.4f}')\n",
    "    return preds, targets"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 46,
   "id": "ea979364",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "d:\\anaconda3\\envs\\llm\\lib\\site-packages\\sklearn\\preprocessing\\_encoders.py:975: FutureWarning: `sparse` was renamed to `sparse_output` in version 1.2 and will be removed in 1.4. `sparse_output` is ignored unless you leave `sparse` to its default value.\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test MSE: 1.2741\n"
     ]
    }
   ],
   "source": [
    "model, feature_cols, target_col, transformer = main()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 48,
   "id": "3808b692",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "d:\\anaconda3\\envs\\llm\\lib\\site-packages\\sklearn\\preprocessing\\_encoders.py:975: FutureWarning: `sparse` was renamed to `sparse_output` in version 1.2 and will be removed in 1.4. `sparse_output` is ignored unless you leave `sparse` to its default value.\n",
      "  warnings.warn(\n"
     ]
    },
    {
     "ename": "RuntimeError",
     "evalue": "mat1 and mat2 shapes cannot be multiplied (21x25 and 28x64)",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[1;31mRuntimeError\u001b[0m                              Traceback (most recent call last)",
      "Cell \u001b[1;32mIn[48], line 1\u001b[0m\n\u001b[1;32m----> 1\u001b[0m y_cursor_pred, y_cursor_true \u001b[38;5;241m=\u001b[39m \u001b[43mpredict_on_new_data\u001b[49m\u001b[43m(\u001b[49m\u001b[43mmodel\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[38;5;124;43mbeach_profile_data/processed_data/beachdata_single.xlsx\u001b[39;49m\u001b[38;5;124;43m\"\u001b[39;49m\u001b[43m)\u001b[49m\n",
      "Cell \u001b[1;32mIn[47], line 183\u001b[0m, in \u001b[0;36mpredict_on_new_data\u001b[1;34m(model, new_data_path)\u001b[0m\n\u001b[0;32m    181\u001b[0m new_loader \u001b[38;5;241m=\u001b[39m DataLoader(new_dataset, batch_size\u001b[38;5;241m=\u001b[39m\u001b[38;5;241m32\u001b[39m, shuffle\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mFalse\u001b[39;00m)\n\u001b[0;32m    182\u001b[0m \u001b[38;5;66;03m# Predict\u001b[39;00m\n\u001b[1;32m--> 183\u001b[0m mse, preds, targets \u001b[38;5;241m=\u001b[39m \u001b[43mevaluate_model\u001b[49m\u001b[43m(\u001b[49m\u001b[43mmodel\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mnew_loader\u001b[49m\u001b[43m)\u001b[49m\n\u001b[0;32m    184\u001b[0m \u001b[38;5;28mprint\u001b[39m(\u001b[38;5;124mf\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mNew Data MSE: \u001b[39m\u001b[38;5;132;01m{\u001b[39;00mmse\u001b[38;5;132;01m:\u001b[39;00m\u001b[38;5;124m.4f\u001b[39m\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m'\u001b[39m)\n\u001b[0;32m    185\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m preds, targets\n",
      "Cell \u001b[1;32mIn[47], line 108\u001b[0m, in \u001b[0;36mevaluate_model\u001b[1;34m(model, dataloader)\u001b[0m\n\u001b[0;32m    106\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m X_batch, y_batch \u001b[38;5;129;01min\u001b[39;00m dataloader:\n\u001b[0;32m    107\u001b[0m     X_batch \u001b[38;5;241m=\u001b[39m X_batch\u001b[38;5;241m.\u001b[39mto(device)\n\u001b[1;32m--> 108\u001b[0m     y_pred \u001b[38;5;241m=\u001b[39m \u001b[43mmodel\u001b[49m\u001b[43m(\u001b[49m\u001b[43mX_batch\u001b[49m\u001b[43m)\u001b[49m\u001b[38;5;241m.\u001b[39mcpu()\u001b[38;5;241m.\u001b[39mnumpy()\n\u001b[0;32m    109\u001b[0m     preds\u001b[38;5;241m.\u001b[39mappend(y_pred)\n\u001b[0;32m    110\u001b[0m     targets\u001b[38;5;241m.\u001b[39mappend(y_batch\u001b[38;5;241m.\u001b[39mnumpy())\n",
      "File \u001b[1;32md:\\anaconda3\\envs\\llm\\lib\\site-packages\\torch\\nn\\modules\\module.py:1532\u001b[0m, in \u001b[0;36mModule._wrapped_call_impl\u001b[1;34m(self, *args, **kwargs)\u001b[0m\n\u001b[0;32m   1530\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_compiled_call_impl(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)  \u001b[38;5;66;03m# type: ignore[misc]\u001b[39;00m\n\u001b[0;32m   1531\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m-> 1532\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_call_impl(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)\n",
      "File \u001b[1;32md:\\anaconda3\\envs\\llm\\lib\\site-packages\\torch\\nn\\modules\\module.py:1541\u001b[0m, in \u001b[0;36mModule._call_impl\u001b[1;34m(self, *args, **kwargs)\u001b[0m\n\u001b[0;32m   1536\u001b[0m \u001b[38;5;66;03m# If we don't have any hooks, we want to skip the rest of the logic in\u001b[39;00m\n\u001b[0;32m   1537\u001b[0m \u001b[38;5;66;03m# this function, and just call forward.\u001b[39;00m\n\u001b[0;32m   1538\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m (\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_pre_hooks\n\u001b[0;32m   1539\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_backward_hooks\n\u001b[0;32m   1540\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_forward_pre_hooks):\n\u001b[1;32m-> 1541\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m forward_call(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)\n\u001b[0;32m   1543\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[0;32m   1544\u001b[0m     result \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m\n",
      "Cell \u001b[1;32mIn[44], line 61\u001b[0m, in \u001b[0;36mBeachProfileNet.forward\u001b[1;34m(self, x)\u001b[0m\n\u001b[0;32m     60\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mforward\u001b[39m(\u001b[38;5;28mself\u001b[39m, x):\n\u001b[1;32m---> 61\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mnet\u001b[49m\u001b[43m(\u001b[49m\u001b[43mx\u001b[49m\u001b[43m)\u001b[49m\u001b[38;5;241m.\u001b[39msqueeze(\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m)\n",
      "File \u001b[1;32md:\\anaconda3\\envs\\llm\\lib\\site-packages\\torch\\nn\\modules\\module.py:1532\u001b[0m, in \u001b[0;36mModule._wrapped_call_impl\u001b[1;34m(self, *args, **kwargs)\u001b[0m\n\u001b[0;32m   1530\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_compiled_call_impl(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)  \u001b[38;5;66;03m# type: ignore[misc]\u001b[39;00m\n\u001b[0;32m   1531\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m-> 1532\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_call_impl(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)\n",
      "File \u001b[1;32md:\\anaconda3\\envs\\llm\\lib\\site-packages\\torch\\nn\\modules\\module.py:1541\u001b[0m, in \u001b[0;36mModule._call_impl\u001b[1;34m(self, *args, **kwargs)\u001b[0m\n\u001b[0;32m   1536\u001b[0m \u001b[38;5;66;03m# If we don't have any hooks, we want to skip the rest of the logic in\u001b[39;00m\n\u001b[0;32m   1537\u001b[0m \u001b[38;5;66;03m# this function, and just call forward.\u001b[39;00m\n\u001b[0;32m   1538\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m (\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_pre_hooks\n\u001b[0;32m   1539\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_backward_hooks\n\u001b[0;32m   1540\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_forward_pre_hooks):\n\u001b[1;32m-> 1541\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m forward_call(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)\n\u001b[0;32m   1543\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[0;32m   1544\u001b[0m     result \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m\n",
      "File \u001b[1;32md:\\anaconda3\\envs\\llm\\lib\\site-packages\\torch\\nn\\modules\\container.py:217\u001b[0m, in \u001b[0;36mSequential.forward\u001b[1;34m(self, input)\u001b[0m\n\u001b[0;32m    215\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mforward\u001b[39m(\u001b[38;5;28mself\u001b[39m, \u001b[38;5;28minput\u001b[39m):\n\u001b[0;32m    216\u001b[0m     \u001b[38;5;28;01mfor\u001b[39;00m module \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mself\u001b[39m:\n\u001b[1;32m--> 217\u001b[0m         \u001b[38;5;28minput\u001b[39m \u001b[38;5;241m=\u001b[39m \u001b[43mmodule\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43minput\u001b[39;49m\u001b[43m)\u001b[49m\n\u001b[0;32m    218\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28minput\u001b[39m\n",
      "File \u001b[1;32md:\\anaconda3\\envs\\llm\\lib\\site-packages\\torch\\nn\\modules\\module.py:1532\u001b[0m, in \u001b[0;36mModule._wrapped_call_impl\u001b[1;34m(self, *args, **kwargs)\u001b[0m\n\u001b[0;32m   1530\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_compiled_call_impl(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)  \u001b[38;5;66;03m# type: ignore[misc]\u001b[39;00m\n\u001b[0;32m   1531\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m-> 1532\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_call_impl(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)\n",
      "File \u001b[1;32md:\\anaconda3\\envs\\llm\\lib\\site-packages\\torch\\nn\\modules\\module.py:1541\u001b[0m, in \u001b[0;36mModule._call_impl\u001b[1;34m(self, *args, **kwargs)\u001b[0m\n\u001b[0;32m   1536\u001b[0m \u001b[38;5;66;03m# If we don't have any hooks, we want to skip the rest of the logic in\u001b[39;00m\n\u001b[0;32m   1537\u001b[0m \u001b[38;5;66;03m# this function, and just call forward.\u001b[39;00m\n\u001b[0;32m   1538\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;129;01mnot\u001b[39;00m (\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_forward_pre_hooks\n\u001b[0;32m   1539\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_backward_pre_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_backward_hooks\n\u001b[0;32m   1540\u001b[0m         \u001b[38;5;129;01mor\u001b[39;00m _global_forward_hooks \u001b[38;5;129;01mor\u001b[39;00m _global_forward_pre_hooks):\n\u001b[1;32m-> 1541\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m forward_call(\u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)\n\u001b[0;32m   1543\u001b[0m \u001b[38;5;28;01mtry\u001b[39;00m:\n\u001b[0;32m   1544\u001b[0m     result \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;01mNone\u001b[39;00m\n",
      "File \u001b[1;32md:\\anaconda3\\envs\\llm\\lib\\site-packages\\torch\\nn\\modules\\linear.py:116\u001b[0m, in \u001b[0;36mLinear.forward\u001b[1;34m(self, input)\u001b[0m\n\u001b[0;32m    115\u001b[0m \u001b[38;5;28;01mdef\u001b[39;00m \u001b[38;5;21mforward\u001b[39m(\u001b[38;5;28mself\u001b[39m, \u001b[38;5;28minput\u001b[39m: Tensor) \u001b[38;5;241m-\u001b[39m\u001b[38;5;241m>\u001b[39m Tensor:\n\u001b[1;32m--> 116\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[43mF\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mlinear\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;28;43minput\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mweight\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mbias\u001b[49m\u001b[43m)\u001b[49m\n",
      "\u001b[1;31mRuntimeError\u001b[0m: mat1 and mat2 shapes cannot be multiplied (21x25 and 28x64)"
     ]
    }
   ],
   "source": [
    "y_cursor_pred, y_cursor_true = predict_on_new_data(model, \"beach_profile_data/processed_data/beachdata_single.xlsx\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 42,
   "id": "d0a7a47a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2.542, 2.373, 2.172, 1.984, 1.815, 1.665, 1.602, 1.398, 1.229, 1.06, 0.897000000000001, 0.752000000000001, 0.593000000000001, 0.363, 0.140000000000001, -0.119999999999999, -0.350999999999998, -0.56, -0.642999999999998, -0.996, -1.091]\n"
     ]
    }
   ],
   "source": [
    "print(true_y_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "158dd403",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "llm",
   "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.19"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
