{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "9742f0d5",
   "metadata": {},
   "outputs": [],
   "source": [
    "from utils import dataset\n",
    "from models.clam import CLAM\n",
    "from torch.utils.data import DataLoader\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "import torch.nn.functional as F\n",
    "import pandas as pd\n",
    "import os\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "1a3624ad",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "47945\n",
      "65450\n",
      "65450\n",
      "17504\n",
      "17504\n",
      "17504\n"
     ]
    }
   ],
   "source": [
    "#Real Dataset\n",
    "\n",
    "\n",
    "\n",
    "real_df_1 = \"real_songs_2.csv\"\n",
    "real_df_1 = pd.read_csv(real_df_1)\n",
    "real_df_2 = \"real_yt_covers.csv\"\n",
    "real_df_2 = pd.read_csv(real_df_2)\n",
    "\n",
    "\n",
    "real_mert_1 = \"real_songs_mert\"\n",
    "real_mert_2 = \"yt_covers_mert\"\n",
    "\n",
    "real_wav2vec2_1 = \"real_songs_wav2vec2\"\n",
    "real_wav2vec2_2 = \"yt_covers_wav2vec2\"\n",
    "\n",
    "print(len(real_df_1))\n",
    "print(len(os.listdir(real_mert_1)))\n",
    "print(len(os.listdir(real_wav2vec2_1)))\n",
    "\n",
    "print(len(real_df_2))\n",
    "print(len(os.listdir(real_mert_2)))\n",
    "print(len(os.listdir(real_wav2vec2_2)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "7baef53b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "fake_df = \"ai_generated_music_metadata.csv\"\n",
    "fake_df = pd.read_csv(fake_df)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "836f41c7",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>filename</th>\n",
       "      <th>model_name</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>ai_covers_0.mp3</td>\n",
       "      <td>AI_COVERS</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>ai_covers_1.mp3</td>\n",
       "      <td>AI_COVERS</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>ai_covers_2.mp3</td>\n",
       "      <td>AI_COVERS</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>ai_covers_3.mp3</td>\n",
       "      <td>AI_COVERS</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>ai_covers_4.mp3</td>\n",
       "      <td>AI_COVERS</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "          filename model_name\n",
       "0  ai_covers_0.mp3  AI_COVERS\n",
       "1  ai_covers_1.mp3  AI_COVERS\n",
       "2  ai_covers_2.mp3  AI_COVERS\n",
       "3  ai_covers_3.mp3  AI_COVERS\n",
       "4  ai_covers_4.mp3  AI_COVERS"
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "fake_df.head()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "e23ec37b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "64972\n",
      "64958\n",
      "64958\n"
     ]
    }
   ],
   "source": [
    "#Fake Dataset\n",
    "\n",
    "\n",
    "fake_df = \"ai_generated_music_metadata.csv\"\n",
    "fake_df = pd.read_csv(fake_df)\n",
    "fake_mert = \"ai_generated_music_mert\"\n",
    "fake_wav2vec2 = \"ai_generated_music_wav2vec2\"\n",
    "\n",
    "\n",
    "print(len(fake_df))\n",
    "print(len(os.listdir(fake_mert)))\n",
    "print(len(os.listdir(fake_wav2vec2)))\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "8c40f0b5",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 47945/47945 [00:13<00:00, 3654.89it/s]\n",
      "100%|██████████| 17504/17504 [00:01<00:00, 10381.10it/s]\n",
      "100%|██████████| 64972/64972 [00:17<00:00, 3678.32it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "47945\n",
      "17504\n",
      "64958\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "from tqdm import tqdm\n",
    "\n",
    "def remove_filename(df, folder):\n",
    "    filenames = os.listdir(folder)\n",
    "    #make copy of df\n",
    "    df_copy = df.copy()\n",
    "    for index, row in tqdm(df_copy.iterrows(), total=df_copy.shape[0]):\n",
    "        filename = row['filename']\n",
    "        if filename.endswith('.mp3'):\n",
    "            filename = filename[:-4]\n",
    "        if filename.endswith('.wav'):\n",
    "            filename = filename[:-4]\n",
    "\n",
    "        if filename + '.pt' not in filenames:\n",
    "            df_copy.drop(index, inplace=True)\n",
    "    return df_copy\n",
    "\n",
    "\n",
    "real_df_1 = remove_filename(real_df_1, real_mert_1)\n",
    "real_df_2 = remove_filename(real_df_2, real_mert_2)\n",
    "fake_df = remove_filename(fake_df, fake_mert)\n",
    "print(len(real_df_1))\n",
    "print(len(real_df_2))\n",
    "print(len(fake_df))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "6e654958",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "47945\n",
      "17504\n"
     ]
    }
   ],
   "source": [
    "print(len(real_df_1))\n",
    "print(len(real_df_2))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "b9facf9d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "65449\n"
     ]
    }
   ],
   "source": [
    "real_df = pd.concat([real_df_1, real_df_2])\n",
    "print(len(real_df))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "2b17865a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "model_name\n",
       "suno_3.5     23695\n",
       "Udio_1.5     19500\n",
       "riffusion     7043\n",
       "Yue           5278\n",
       "Diffrythm     4606\n",
       "suno_3        3512\n",
       "AI_COVERS     1166\n",
       "suno_2         110\n",
       "suno_4          48\n",
       "Name: count, dtype: int64"
      ]
     },
     "execution_count": 10,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "fake_df[\"model_name\"].value_counts()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "901d4ef6",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "17047\n",
      "47911\n",
      "0.7375688906678162\n",
      "48273\n",
      "17176\n"
     ]
    }
   ],
   "source": [
    "from sklearn.model_selection import train_test_split\n",
    "\n",
    "\n",
    "fake_df_test = fake_df[fake_df[\"model_name\"].isin([\"riffusion\", \"suno_3\", \"AI_COVERS\"  , 'suno_4' , 'Yue'])]\n",
    "print(len(fake_df_test))\n",
    "fake_df_train = fake_df[~fake_df[\"model_name\"].isin([\"riffusion\", \"suno_3\", \"AI_COVERS\" , \"suno_4\" , \"Yue\"] )]\n",
    "print(len(fake_df_train))\n",
    "\n",
    "ratio = len(fake_df_train) / ( len(fake_df_test) + len(fake_df_train)  )\n",
    "print(ratio)\n",
    "\n",
    "real_df = pd.concat([real_df_1, real_df_2])\n",
    "\n",
    "real_df_train, real_df_test = train_test_split(real_df, test_size= 1 - ratio, random_state=42)\n",
    "print(len(real_df_train))\n",
    "print(len(real_df_test))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "af6715f6",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Preloading embeddings for train split...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  24%|██▍       | 11662/48273 [00:19<01:01, 591.88it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for yt_covers_8382.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  33%|███▎      | 15893/48273 [00:26<00:55, 579.63it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for yt_covers_5219.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  58%|█████▊    | 28041/48273 [00:47<00:35, 572.47it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for yt_covers_11880.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  60%|██████    | 28966/48273 [00:49<00:31, 611.51it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for yt_covers_9557.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  67%|██████▋   | 32401/48273 [00:54<00:27, 586.45it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "PytorchStreamReader failed locating file data/0: file not found\n",
      "\n",
      "Warning: Error loading embeddings for yt_covers_11542.pt: PytorchStreamReader failed locating file data/0: file not found. Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU: 100%|██████████| 48273/48273 [01:21<00:00, 591.42it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Preloading embeddings for train split...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:   1%|▏         | 631/47911 [00:01<01:23, 566.74it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for diffrythm_566.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  10%|█         | 4920/47911 [00:08<01:16, 560.51it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for suno_3_5_122.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  11%|█         | 5095/47911 [00:09<01:14, 571.07it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for suno_3_5_278.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  14%|█▍        | 6935/47911 [00:12<01:09, 586.79it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for suno_3_5_2113.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  35%|███▍      | 16598/47911 [00:28<00:54, 576.70it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for suno_3_5_11812.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  49%|████▉     | 23434/47911 [00:40<00:42, 569.51it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for suno_3_5_18613.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  59%|█████▉    | 28474/47911 [00:49<00:31, 618.53it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for suno_3_5_23683.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  61%|██████    | 29095/47911 [00:50<00:31, 592.89it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for udio_1_5_607.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  75%|███████▍  | 35726/47911 [01:01<00:19, 628.59it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for udio_1_5_7188.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU:  98%|█████████▊| 46738/47911 [01:19<00:01, 617.46it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for udio_1_5_18205.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading train data to CPU: 100%|██████████| 47911/47911 [01:20<00:00, 591.50it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Preloading embeddings for test split...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading test data to CPU:   5%|▌         | 939/17176 [00:01<00:26, 620.21it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for yt_covers_13185.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading test data to CPU:   7%|▋         | 1255/17176 [00:02<00:25, 625.07it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for yt_covers_9011.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading test data to CPU:  44%|████▍     | 7547/17176 [00:11<00:15, 635.72it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for yt_covers_10928.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading test data to CPU:  61%|██████▏   | 10541/17176 [00:16<00:10, 639.09it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for yt_covers_7006.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading test data to CPU:  78%|███████▊  | 13328/17176 [00:21<00:05, 644.25it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for yt_covers_5289.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading test data to CPU:  80%|████████  | 13779/17176 [00:21<00:05, 636.08it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "PytorchStreamReader failed locating file data/0: file not found\n",
      "\n",
      "Warning: Error loading embeddings for yt_covers_7797.pt: PytorchStreamReader failed locating file data/0: file not found. Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading test data to CPU:  98%|█████████▊| 16860/17176 [00:26<00:00, 649.13it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for yt_covers_15133.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading test data to CPU: 100%|██████████| 17176/17176 [00:27<00:00, 630.64it/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Preloading embeddings for test split...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading test data to CPU:  47%|████▋     | 7941/17047 [00:13<00:14, 611.73it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for suno_3_1802.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading test data to CPU:  49%|████▉     | 8315/17047 [00:13<00:14, 618.67it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "Warning: Error loading embeddings for suno_3_2149.pt: . Skipping sample.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading test data to CPU: 100%|██████████| 17047/17047 [00:27<00:00, 620.44it/s]\n"
     ]
    }
   ],
   "source": [
    "from utils import dataset\n",
    "\n",
    "train_real_dataset = dataset.SongsDataset(real_df_train, real_mert_1, real_wav2vec2_1, label = 0)\n",
    "train_fake_dataset = dataset.SongsDataset(fake_df_train, fake_mert, fake_wav2vec2, label = 1)\n",
    "\n",
    "test_real_dataset = dataset.SongsDataset(real_df_test, real_mert_1, real_wav2vec2_1, label = 0 , split = \"test\")\n",
    "test_fake_dataset = dataset.SongsDataset(fake_df_test, fake_mert, fake_wav2vec2, label = 1 , split = \"test\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "296bb56e",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "48268\n",
      "47901\n",
      "17169\n",
      "17045\n"
     ]
    }
   ],
   "source": [
    "print(len(train_real_dataset))\n",
    "print(len(train_fake_dataset))\n",
    "print(len(test_real_dataset))\n",
    "print(len(test_fake_dataset))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "738fc507",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "96169\n",
      "34214\n"
     ]
    }
   ],
   "source": [
    "train_dataset = torch.utils.data.ConcatDataset([train_real_dataset, train_fake_dataset])\n",
    "test_dataset = torch.utils.data.ConcatDataset([test_real_dataset, test_fake_dataset])\n",
    "print(len(train_dataset))\n",
    "print(len(test_dataset))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "359ca8a5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Starting Training...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 1/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 162.86batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 1/50\n",
      "  Train Total Loss: 0.2780 | CLS Loss: 0.2548 | Align Loss (raw): 0.0465\n",
      "  Train Acc: 0.8843 | F1: 0.8794 | Recall: 0.8483 | Precision: 0.9129\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 1/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 698.60batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.1281 | CLS Loss: 0.1053 | Align Loss (raw): 0.0456\n",
      "  Val Acc: 0.9604 | F1: 0.9597 | Recall: 0.9434 | Precision: 0.9767\n",
      "  New Best F1: 0.9597\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 2/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 166.74batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 2/50\n",
      "  Train Total Loss: 0.1086 | CLS Loss: 0.0912 | Align Loss (raw): 0.0350\n",
      "  Train Acc: 0.9690 | F1: 0.9687 | Recall: 0.9647 | Precision: 0.9727\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 2/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 695.73batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0874 | CLS Loss: 0.0744 | Align Loss (raw): 0.0260\n",
      "  Val Acc: 0.9754 | F1: 0.9754 | Recall: 0.9763 | Precision: 0.9745\n",
      "  New Best F1: 0.9754\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 3/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 167.19batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 3/50\n",
      "  Train Total Loss: 0.0824 | CLS Loss: 0.0702 | Align Loss (raw): 0.0244\n",
      "  Train Acc: 0.9762 | F1: 0.9760 | Recall: 0.9736 | Precision: 0.9784\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 3/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 707.18batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0692 | CLS Loss: 0.0579 | Align Loss (raw): 0.0227\n",
      "  Val Acc: 0.9812 | F1: 0.9814 | Recall: 0.9912 | Precision: 0.9719\n",
      "  New Best F1: 0.9814\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 4/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 166.61batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 4/50\n",
      "  Train Total Loss: 0.0692 | CLS Loss: 0.0593 | Align Loss (raw): 0.0197\n",
      "  Train Acc: 0.9799 | F1: 0.9797 | Recall: 0.9776 | Precision: 0.9819\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 4/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 699.44batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0646 | CLS Loss: 0.0559 | Align Loss (raw): 0.0175\n",
      "  Val Acc: 0.9809 | F1: 0.9807 | Recall: 0.9715 | Precision: 0.9900\n",
      "  Model not saved. Best F1 so far: 0.9814\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 5/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 169.38batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 5/50\n",
      "  Train Total Loss: 0.0626 | CLS Loss: 0.0542 | Align Loss (raw): 0.0169\n",
      "  Train Acc: 0.9818 | F1: 0.9816 | Recall: 0.9794 | Precision: 0.9839\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 5/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 689.00batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0732 | CLS Loss: 0.0646 | Align Loss (raw): 0.0173\n",
      "  Val Acc: 0.9792 | F1: 0.9795 | Recall: 0.9940 | Precision: 0.9655\n",
      "  Model not saved. Best F1 so far: 0.9814\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 6/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 163.95batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 6/50\n",
      "  Train Total Loss: 0.0551 | CLS Loss: 0.0478 | Align Loss (raw): 0.0145\n",
      "  Train Acc: 0.9845 | F1: 0.9844 | Recall: 0.9828 | Precision: 0.9860\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 6/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 676.52batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0473 | CLS Loss: 0.0409 | Align Loss (raw): 0.0128\n",
      "  Val Acc: 0.9857 | F1: 0.9856 | Recall: 0.9807 | Precision: 0.9907\n",
      "  New Best F1: 0.9856\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 7/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 168.53batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 7/50\n",
      "  Train Total Loss: 0.0520 | CLS Loss: 0.0453 | Align Loss (raw): 0.0133\n",
      "  Train Acc: 0.9850 | F1: 0.9849 | Recall: 0.9835 | Precision: 0.9863\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 7/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 685.77batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0454 | CLS Loss: 0.0399 | Align Loss (raw): 0.0111\n",
      "  Val Acc: 0.9877 | F1: 0.9877 | Recall: 0.9866 | Precision: 0.9888\n",
      "  New Best F1: 0.9877\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 8/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 169.52batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 8/50\n",
      "  Train Total Loss: 0.0476 | CLS Loss: 0.0415 | Align Loss (raw): 0.0122\n",
      "  Train Acc: 0.9856 | F1: 0.9855 | Recall: 0.9839 | Precision: 0.9871\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 8/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 663.92batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0450 | CLS Loss: 0.0389 | Align Loss (raw): 0.0121\n",
      "  Val Acc: 0.9881 | F1: 0.9881 | Recall: 0.9869 | Precision: 0.9893\n",
      "  New Best F1: 0.9881\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 9/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 166.08batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 9/50\n",
      "  Train Total Loss: 0.0454 | CLS Loss: 0.0397 | Align Loss (raw): 0.0115\n",
      "  Train Acc: 0.9863 | F1: 0.9862 | Recall: 0.9850 | Precision: 0.9874\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 9/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 699.54batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0444 | CLS Loss: 0.0391 | Align Loss (raw): 0.0106\n",
      "  Val Acc: 0.9872 | F1: 0.9871 | Recall: 0.9813 | Precision: 0.9931\n",
      "  Model not saved. Best F1 so far: 0.9881\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 10/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 168.45batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 10/50\n",
      "  Train Total Loss: 0.0416 | CLS Loss: 0.0362 | Align Loss (raw): 0.0109\n",
      "  Train Acc: 0.9879 | F1: 0.9878 | Recall: 0.9864 | Precision: 0.9892\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 10/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 686.37batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0365 | CLS Loss: 0.0310 | Align Loss (raw): 0.0111\n",
      "  Val Acc: 0.9900 | F1: 0.9900 | Recall: 0.9895 | Precision: 0.9905\n",
      "  New Best F1: 0.9900\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 11/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 169.75batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 11/50\n",
      "  Train Total Loss: 0.0408 | CLS Loss: 0.0356 | Align Loss (raw): 0.0103\n",
      "  Train Acc: 0.9878 | F1: 0.9877 | Recall: 0.9861 | Precision: 0.9892\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 11/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 737.43batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0356 | CLS Loss: 0.0311 | Align Loss (raw): 0.0090\n",
      "  Val Acc: 0.9898 | F1: 0.9898 | Recall: 0.9883 | Precision: 0.9912\n",
      "  Model not saved. Best F1 so far: 0.9900\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 12/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 168.20batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 12/50\n",
      "  Train Total Loss: 0.0375 | CLS Loss: 0.0326 | Align Loss (raw): 0.0098\n",
      "  Train Acc: 0.9888 | F1: 0.9888 | Recall: 0.9874 | Precision: 0.9902\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 12/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 665.56batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0373 | CLS Loss: 0.0326 | Align Loss (raw): 0.0094\n",
      "  Val Acc: 0.9900 | F1: 0.9900 | Recall: 0.9896 | Precision: 0.9904\n",
      "  New Best F1: 0.9900\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 13/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 166.16batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 13/50\n",
      "  Train Total Loss: 0.0359 | CLS Loss: 0.0312 | Align Loss (raw): 0.0094\n",
      "  Train Acc: 0.9894 | F1: 0.9893 | Recall: 0.9883 | Precision: 0.9903\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 13/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 726.25batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0423 | CLS Loss: 0.0376 | Align Loss (raw): 0.0095\n",
      "  Val Acc: 0.9873 | F1: 0.9872 | Recall: 0.9794 | Precision: 0.9950\n",
      "  Model not saved. Best F1 so far: 0.9900\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 14/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 167.02batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 14/50\n",
      "  Train Total Loss: 0.0346 | CLS Loss: 0.0300 | Align Loss (raw): 0.0091\n",
      "  Train Acc: 0.9898 | F1: 0.9898 | Recall: 0.9888 | Precision: 0.9907\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 14/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 682.37batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0446 | CLS Loss: 0.0400 | Align Loss (raw): 0.0091\n",
      "  Val Acc: 0.9867 | F1: 0.9866 | Recall: 0.9771 | Precision: 0.9963\n",
      "  Model not saved. Best F1 so far: 0.9900\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 15/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 166.31batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 15/50\n",
      "  Train Total Loss: 0.0340 | CLS Loss: 0.0295 | Align Loss (raw): 0.0089\n",
      "  Train Acc: 0.9901 | F1: 0.9900 | Recall: 0.9891 | Precision: 0.9910\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 15/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 716.69batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0437 | CLS Loss: 0.0396 | Align Loss (raw): 0.0083\n",
      "  Val Acc: 0.9873 | F1: 0.9872 | Recall: 0.9794 | Precision: 0.9951\n",
      "  Model not saved. Best F1 so far: 0.9900\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 16/50 Training: 100%|██████████| 4809/4809 [00:26<00:00, 178.93batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 16/50\n",
      "  Train Total Loss: 0.0316 | CLS Loss: 0.0272 | Align Loss (raw): 0.0087\n",
      "  Train Acc: 0.9907 | F1: 0.9907 | Recall: 0.9900 | Precision: 0.9913\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 16/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 692.28batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0387 | CLS Loss: 0.0350 | Align Loss (raw): 0.0073\n",
      "  Val Acc: 0.9881 | F1: 0.9880 | Recall: 0.9802 | Precision: 0.9960\n",
      "  Model not saved. Best F1 so far: 0.9900\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 17/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 166.26batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 17/50\n",
      "  Train Total Loss: 0.0308 | CLS Loss: 0.0266 | Align Loss (raw): 0.0084\n",
      "  Train Acc: 0.9912 | F1: 0.9911 | Recall: 0.9902 | Precision: 0.9921\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 17/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 665.89batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0335 | CLS Loss: 0.0294 | Align Loss (raw): 0.0082\n",
      "  Val Acc: 0.9897 | F1: 0.9897 | Recall: 0.9857 | Precision: 0.9937\n",
      "  Model not saved. Best F1 so far: 0.9900\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 18/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 164.55batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 18/50\n",
      "  Train Total Loss: 0.0303 | CLS Loss: 0.0262 | Align Loss (raw): 0.0082\n",
      "  Train Acc: 0.9912 | F1: 0.9912 | Recall: 0.9905 | Precision: 0.9918\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 18/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 706.25batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0448 | CLS Loss: 0.0405 | Align Loss (raw): 0.0086\n",
      "  Val Acc: 0.9864 | F1: 0.9865 | Recall: 0.9962 | Precision: 0.9771\n",
      "  Model not saved. Best F1 so far: 0.9900\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 19/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 166.96batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 19/50\n",
      "  Train Total Loss: 0.0296 | CLS Loss: 0.0256 | Align Loss (raw): 0.0080\n",
      "  Train Acc: 0.9911 | F1: 0.9910 | Recall: 0.9904 | Precision: 0.9917\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 19/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 648.55batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0331 | CLS Loss: 0.0290 | Align Loss (raw): 0.0081\n",
      "  Val Acc: 0.9913 | F1: 0.9912 | Recall: 0.9873 | Precision: 0.9952\n",
      "  New Best F1: 0.9912\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 20/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 164.46batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 20/50\n",
      "  Train Total Loss: 0.0287 | CLS Loss: 0.0248 | Align Loss (raw): 0.0077\n",
      "  Train Acc: 0.9916 | F1: 0.9916 | Recall: 0.9909 | Precision: 0.9923\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 20/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 664.22batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0300 | CLS Loss: 0.0263 | Align Loss (raw): 0.0074\n",
      "  Val Acc: 0.9916 | F1: 0.9916 | Recall: 0.9892 | Precision: 0.9940\n",
      "  New Best F1: 0.9916\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 21/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 162.10batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 21/50\n",
      "  Train Total Loss: 0.0272 | CLS Loss: 0.0233 | Align Loss (raw): 0.0077\n",
      "  Train Acc: 0.9920 | F1: 0.9920 | Recall: 0.9915 | Precision: 0.9925\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 21/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 692.70batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0310 | CLS Loss: 0.0265 | Align Loss (raw): 0.0090\n",
      "  Val Acc: 0.9917 | F1: 0.9917 | Recall: 0.9879 | Precision: 0.9954\n",
      "  New Best F1: 0.9917\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 22/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 163.03batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 22/50\n",
      "  Train Total Loss: 0.0255 | CLS Loss: 0.0217 | Align Loss (raw): 0.0076\n",
      "  Train Acc: 0.9926 | F1: 0.9925 | Recall: 0.9920 | Precision: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 22/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 709.68batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.1067 | CLS Loss: 0.0982 | Align Loss (raw): 0.0170\n",
      "  Val Acc: 0.9689 | F1: 0.9698 | Recall: 0.9989 | Precision: 0.9424\n",
      "  Model not saved. Best F1 so far: 0.9917\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 23/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 164.40batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 23/50\n",
      "  Train Total Loss: 0.0261 | CLS Loss: 0.0224 | Align Loss (raw): 0.0075\n",
      "  Train Acc: 0.9924 | F1: 0.9923 | Recall: 0.9915 | Precision: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 23/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 662.28batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0422 | CLS Loss: 0.0387 | Align Loss (raw): 0.0070\n",
      "  Val Acc: 0.9872 | F1: 0.9871 | Recall: 0.9815 | Precision: 0.9927\n",
      "  Model not saved. Best F1 so far: 0.9917\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 24/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 168.13batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 24/50\n",
      "  Train Total Loss: 0.0240 | CLS Loss: 0.0206 | Align Loss (raw): 0.0069\n",
      "  Train Acc: 0.9927 | F1: 0.9927 | Recall: 0.9919 | Precision: 0.9935\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 24/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 650.73batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0302 | CLS Loss: 0.0267 | Align Loss (raw): 0.0070\n",
      "  Val Acc: 0.9931 | F1: 0.9931 | Recall: 0.9924 | Precision: 0.9938\n",
      "  New Best F1: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 25/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 171.33batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 25/50\n",
      "  Train Total Loss: 0.0236 | CLS Loss: 0.0200 | Align Loss (raw): 0.0072\n",
      "  Train Acc: 0.9932 | F1: 0.9932 | Recall: 0.9925 | Precision: 0.9939\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 25/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 722.12batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0319 | CLS Loss: 0.0283 | Align Loss (raw): 0.0072\n",
      "  Val Acc: 0.9915 | F1: 0.9915 | Recall: 0.9913 | Precision: 0.9918\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 26/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 163.96batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 26/50\n",
      "  Train Total Loss: 0.0234 | CLS Loss: 0.0199 | Align Loss (raw): 0.0069\n",
      "  Train Acc: 0.9932 | F1: 0.9932 | Recall: 0.9923 | Precision: 0.9941\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 26/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 662.10batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0311 | CLS Loss: 0.0281 | Align Loss (raw): 0.0060\n",
      "  Val Acc: 0.9910 | F1: 0.9909 | Recall: 0.9859 | Precision: 0.9960\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 27/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 170.46batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 27/50\n",
      "  Train Total Loss: 0.0227 | CLS Loss: 0.0192 | Align Loss (raw): 0.0070\n",
      "  Train Acc: 0.9937 | F1: 0.9936 | Recall: 0.9928 | Precision: 0.9944\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 27/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 692.31batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0336 | CLS Loss: 0.0298 | Align Loss (raw): 0.0075\n",
      "  Val Acc: 0.9906 | F1: 0.9907 | Recall: 0.9913 | Precision: 0.9900\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 28/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 164.38batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 28/50\n",
      "  Train Total Loss: 0.0226 | CLS Loss: 0.0191 | Align Loss (raw): 0.0069\n",
      "  Train Acc: 0.9935 | F1: 0.9934 | Recall: 0.9929 | Precision: 0.9939\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 28/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 685.37batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0297 | CLS Loss: 0.0261 | Align Loss (raw): 0.0072\n",
      "  Val Acc: 0.9921 | F1: 0.9921 | Recall: 0.9930 | Precision: 0.9912\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 29/50 Training: 100%|██████████| 4809/4809 [00:31<00:00, 150.54batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 29/50\n",
      "  Train Total Loss: 0.0219 | CLS Loss: 0.0185 | Align Loss (raw): 0.0069\n",
      "  Train Acc: 0.9937 | F1: 0.9936 | Recall: 0.9930 | Precision: 0.9942\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 29/50 Validation: 100%|██████████| 1203/1203 [00:02<00:00, 586.94batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0264 | CLS Loss: 0.0233 | Align Loss (raw): 0.0061\n",
      "  Val Acc: 0.9919 | F1: 0.9919 | Recall: 0.9887 | Precision: 0.9952\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 30/50 Training: 100%|██████████| 4809/4809 [00:42<00:00, 113.94batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 30/50\n",
      "  Train Total Loss: 0.0210 | CLS Loss: 0.0175 | Align Loss (raw): 0.0069\n",
      "  Train Acc: 0.9939 | F1: 0.9939 | Recall: 0.9933 | Precision: 0.9945\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 30/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 657.01batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0279 | CLS Loss: 0.0247 | Align Loss (raw): 0.0065\n",
      "  Val Acc: 0.9928 | F1: 0.9928 | Recall: 0.9916 | Precision: 0.9941\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 31/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 163.83batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 31/50\n",
      "  Train Total Loss: 0.0206 | CLS Loss: 0.0173 | Align Loss (raw): 0.0066\n",
      "  Train Acc: 0.9940 | F1: 0.9939 | Recall: 0.9934 | Precision: 0.9945\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 31/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 685.74batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0381 | CLS Loss: 0.0345 | Align Loss (raw): 0.0071\n",
      "  Val Acc: 0.9892 | F1: 0.9892 | Recall: 0.9939 | Precision: 0.9847\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 32/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 165.42batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 32/50\n",
      "  Train Total Loss: 0.0202 | CLS Loss: 0.0169 | Align Loss (raw): 0.0067\n",
      "  Train Acc: 0.9946 | F1: 0.9946 | Recall: 0.9943 | Precision: 0.9949\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 32/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 690.12batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0323 | CLS Loss: 0.0296 | Align Loss (raw): 0.0053\n",
      "  Val Acc: 0.9908 | F1: 0.9908 | Recall: 0.9850 | Precision: 0.9965\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 33/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 165.67batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 33/50\n",
      "  Train Total Loss: 0.0197 | CLS Loss: 0.0164 | Align Loss (raw): 0.0066\n",
      "  Train Acc: 0.9942 | F1: 0.9941 | Recall: 0.9936 | Precision: 0.9947\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 33/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 673.81batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0271 | CLS Loss: 0.0239 | Align Loss (raw): 0.0064\n",
      "  Val Acc: 0.9930 | F1: 0.9930 | Recall: 0.9935 | Precision: 0.9926\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 34/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 165.72batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 34/50\n",
      "  Train Total Loss: 0.0184 | CLS Loss: 0.0152 | Align Loss (raw): 0.0064\n",
      "  Train Acc: 0.9945 | F1: 0.9945 | Recall: 0.9943 | Precision: 0.9947\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 34/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 677.96batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0295 | CLS Loss: 0.0266 | Align Loss (raw): 0.0059\n",
      "  Val Acc: 0.9917 | F1: 0.9917 | Recall: 0.9897 | Precision: 0.9937\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 35/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 162.78batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 35/50\n",
      "  Train Total Loss: 0.0185 | CLS Loss: 0.0154 | Align Loss (raw): 0.0062\n",
      "  Train Acc: 0.9948 | F1: 0.9947 | Recall: 0.9942 | Precision: 0.9953\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 35/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 669.84batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0307 | CLS Loss: 0.0277 | Align Loss (raw): 0.0060\n",
      "  Val Acc: 0.9919 | F1: 0.9919 | Recall: 0.9886 | Precision: 0.9953\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 36/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 165.45batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 36/50\n",
      "  Train Total Loss: 0.0178 | CLS Loss: 0.0147 | Align Loss (raw): 0.0062\n",
      "  Train Acc: 0.9949 | F1: 0.9948 | Recall: 0.9944 | Precision: 0.9953\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 36/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 657.55batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0296 | CLS Loss: 0.0262 | Align Loss (raw): 0.0068\n",
      "  Val Acc: 0.9915 | F1: 0.9915 | Recall: 0.9928 | Precision: 0.9902\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 37/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 163.99batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 37/50\n",
      "  Train Total Loss: 0.0170 | CLS Loss: 0.0139 | Align Loss (raw): 0.0063\n",
      "  Train Acc: 0.9951 | F1: 0.9951 | Recall: 0.9947 | Precision: 0.9955\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 37/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 665.15batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0335 | CLS Loss: 0.0306 | Align Loss (raw): 0.0058\n",
      "  Val Acc: 0.9921 | F1: 0.9921 | Recall: 0.9921 | Precision: 0.9921\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 38/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 164.62batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 38/50\n",
      "  Train Total Loss: 0.0172 | CLS Loss: 0.0140 | Align Loss (raw): 0.0063\n",
      "  Train Acc: 0.9950 | F1: 0.9949 | Recall: 0.9947 | Precision: 0.9952\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 38/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 686.80batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0307 | CLS Loss: 0.0277 | Align Loss (raw): 0.0058\n",
      "  Val Acc: 0.9912 | F1: 0.9911 | Recall: 0.9856 | Precision: 0.9967\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 39/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 167.28batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 39/50\n",
      "  Train Total Loss: 0.0181 | CLS Loss: 0.0151 | Align Loss (raw): 0.0061\n",
      "  Train Acc: 0.9947 | F1: 0.9946 | Recall: 0.9941 | Precision: 0.9951\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 39/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 691.30batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0276 | CLS Loss: 0.0244 | Align Loss (raw): 0.0063\n",
      "  Val Acc: 0.9930 | F1: 0.9930 | Recall: 0.9920 | Precision: 0.9941\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 40/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 166.12batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 40/50\n",
      "  Train Total Loss: 0.0175 | CLS Loss: 0.0145 | Align Loss (raw): 0.0059\n",
      "  Train Acc: 0.9951 | F1: 0.9950 | Recall: 0.9946 | Precision: 0.9955\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 40/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 668.95batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0257 | CLS Loss: 0.0230 | Align Loss (raw): 0.0053\n",
      "  Val Acc: 0.9928 | F1: 0.9928 | Recall: 0.9923 | Precision: 0.9933\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 41/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 166.24batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 41/50\n",
      "  Train Total Loss: 0.0165 | CLS Loss: 0.0135 | Align Loss (raw): 0.0060\n",
      "  Train Acc: 0.9954 | F1: 0.9954 | Recall: 0.9952 | Precision: 0.9956\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 41/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 709.24batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0303 | CLS Loss: 0.0274 | Align Loss (raw): 0.0058\n",
      "  Val Acc: 0.9918 | F1: 0.9918 | Recall: 0.9864 | Precision: 0.9973\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 42/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 170.23batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 42/50\n",
      "  Train Total Loss: 0.0165 | CLS Loss: 0.0134 | Align Loss (raw): 0.0060\n",
      "  Train Acc: 0.9951 | F1: 0.9951 | Recall: 0.9947 | Precision: 0.9954\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 42/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 717.85batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0322 | CLS Loss: 0.0288 | Align Loss (raw): 0.0067\n",
      "  Val Acc: 0.9919 | F1: 0.9919 | Recall: 0.9879 | Precision: 0.9959\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 43/50 Training: 100%|██████████| 4809/4809 [00:30<00:00, 155.60batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 43/50\n",
      "  Train Total Loss: 0.0157 | CLS Loss: 0.0128 | Align Loss (raw): 0.0059\n",
      "  Train Acc: 0.9957 | F1: 0.9957 | Recall: 0.9954 | Precision: 0.9960\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 43/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 655.59batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0349 | CLS Loss: 0.0321 | Align Loss (raw): 0.0055\n",
      "  Val Acc: 0.9921 | F1: 0.9922 | Recall: 0.9963 | Precision: 0.9881\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 44/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 165.69batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 44/50\n",
      "  Train Total Loss: 0.0148 | CLS Loss: 0.0118 | Align Loss (raw): 0.0059\n",
      "  Train Acc: 0.9958 | F1: 0.9957 | Recall: 0.9951 | Precision: 0.9964\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 44/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 646.51batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0343 | CLS Loss: 0.0312 | Align Loss (raw): 0.0063\n",
      "  Val Acc: 0.9916 | F1: 0.9916 | Recall: 0.9915 | Precision: 0.9917\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 45/50 Training: 100%|██████████| 4809/4809 [00:30<00:00, 157.09batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 45/50\n",
      "  Train Total Loss: 0.0152 | CLS Loss: 0.0121 | Align Loss (raw): 0.0062\n",
      "  Train Acc: 0.9958 | F1: 0.9958 | Recall: 0.9953 | Precision: 0.9964\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 45/50 Validation: 100%|██████████| 1203/1203 [00:02<00:00, 562.25batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0268 | CLS Loss: 0.0240 | Align Loss (raw): 0.0057\n",
      "  Val Acc: 0.9923 | F1: 0.9923 | Recall: 0.9908 | Precision: 0.9939\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 46/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 163.28batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 46/50\n",
      "  Train Total Loss: 0.0153 | CLS Loss: 0.0123 | Align Loss (raw): 0.0061\n",
      "  Train Acc: 0.9957 | F1: 0.9957 | Recall: 0.9953 | Precision: 0.9961\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 46/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 693.39batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0280 | CLS Loss: 0.0254 | Align Loss (raw): 0.0053\n",
      "  Val Acc: 0.9925 | F1: 0.9924 | Recall: 0.9888 | Precision: 0.9961\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 47/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 165.61batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 47/50\n",
      "  Train Total Loss: 0.0150 | CLS Loss: 0.0122 | Align Loss (raw): 0.0056\n",
      "  Train Acc: 0.9958 | F1: 0.9958 | Recall: 0.9956 | Precision: 0.9960\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 47/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 669.61batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0362 | CLS Loss: 0.0332 | Align Loss (raw): 0.0060\n",
      "  Val Acc: 0.9904 | F1: 0.9904 | Recall: 0.9902 | Precision: 0.9906\n",
      "  Model not saved. Best F1 so far: 0.9931\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 48/50 Training: 100%|██████████| 4809/4809 [00:28<00:00, 166.83batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 48/50\n",
      "  Train Total Loss: 0.0152 | CLS Loss: 0.0123 | Align Loss (raw): 0.0058\n",
      "  Train Acc: 0.9956 | F1: 0.9955 | Recall: 0.9951 | Precision: 0.9959\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 48/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 688.77batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0252 | CLS Loss: 0.0226 | Align Loss (raw): 0.0051\n",
      "  Val Acc: 0.9940 | F1: 0.9940 | Recall: 0.9948 | Precision: 0.9932\n",
      "  New Best F1: 0.9940\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 49/50 Training: 100%|██████████| 4809/4809 [00:29<00:00, 163.41batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 49/50\n",
      "  Train Total Loss: 0.0144 | CLS Loss: 0.0116 | Align Loss (raw): 0.0058\n",
      "  Train Acc: 0.9960 | F1: 0.9960 | Recall: 0.9958 | Precision: 0.9962\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 49/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 650.25batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0327 | CLS Loss: 0.0298 | Align Loss (raw): 0.0058\n",
      "  Val Acc: 0.9920 | F1: 0.9920 | Recall: 0.9951 | Precision: 0.9890\n",
      "  Model not saved. Best F1 so far: 0.9940\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 50/50 Training: 100%|██████████| 4809/4809 [00:30<00:00, 159.46batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch 50/50\n",
      "  Train Total Loss: 0.0140 | CLS Loss: 0.0111 | Align Loss (raw): 0.0056\n",
      "  Train Acc: 0.9960 | F1: 0.9960 | Recall: 0.9956 | Precision: 0.9965\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Epoch 50/50 Validation: 100%|██████████| 1203/1203 [00:01<00:00, 689.43batch/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "  Val Total Loss: 0.0339 | CLS Loss: 0.0309 | Align Loss (raw): 0.0061\n",
      "  Val Acc: 0.9911 | F1: 0.9912 | Recall: 0.9957 | Precision: 0.9866\n",
      "  Model not saved. Best F1 so far: 0.9940\n",
      "Training Finished.\n",
      "Best Validation Accuracy achieved: 0.0000\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "\n",
    "\n",
    "from sklearn.metrics import accuracy_score, f1_score, precision_score, recall_score\n",
    "from torch.optim import Adam , AdamW\n",
    "from torch.utils.data import DataLoader\n",
    "import torch.optim as optim\n",
    "from tqdm import tqdm\n",
    "import warnings\n",
    "from sklearn.metrics import roc_curve\n",
    "import numpy as np\n",
    "\n",
    "def compute_eer(y_true, y_scores):\n",
    "    fpr, tpr, thresholds = roc_curve(y_true, y_scores)\n",
    "    fnr = 1 - tpr\n",
    "    # Find the point where FPR = FNRbest_model_l1_loss.pth\n",
    "    eer_threshold_index = np.nanargmin(np.absolute((fnr - fpr)))\n",
    "    eer = (fpr[eer_threshold_index] + fnr[eer_threshold_index]) / 2\n",
    "    return eer\n",
    "\n",
    "\n",
    "# train_loader = DataLoader(train_dataset, batch_size=16, shuffle=True)\n",
    "# test_loader = DataLoader(test_dataset, batch_size=16, shuffle=True)\n",
    "\n",
    "val_split = 0.2\n",
    "train_size = int((1 - val_split) * len(train_dataset))\n",
    "val_size = len(train_dataset) - train_size\n",
    "train_dataset, val_dataset = torch.utils.data.random_split(train_dataset, [train_size, val_size] , generator=torch.Generator().manual_seed(42))\n",
    "train_loader = DataLoader(train_dataset, batch_size=16, shuffle=True)\n",
    "val_loader = DataLoader(val_dataset, batch_size=16, shuffle=True)\n",
    "test_loader = DataLoader(test_dataset, batch_size=16, shuffle=True)\n",
    "warnings.filterwarnings(\"ignore\")\n",
    "\n",
    "device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
    "in_channel1 = 13\n",
    "in_channel2 = 13\n",
    "embed_dim1 = 768\n",
    "embed_dim2 = 768\n",
    "best_acc = 0\n",
    "best_f1 = 0\n",
    "\n",
    "model = CLAM(in_channel1 , in_channel2 , embed_dim1 , embed_dim2).to(device)\n",
    "\n",
    "optimizer = AdamW(model.parameters(), lr=1e-4)\n",
    "\n",
    "classification_criterion = nn.BCEWithLogitsLoss() # For fake/real classification\n",
    "alignment_criterion = nn.L1Loss() # For embedding alignment\n",
    "save_name = \"best_model_l1_alignment.pt\"\n",
    "save_folder = \"model_wts\"\n",
    "# --- Hyperparameters ---\n",
    "epochs = 50\n",
    "alignment_loss_weight = 0.5 # Weight factor for the alignment loss \n",
    "print(\"Starting Training...\")\n",
    "\n",
    "\n",
    "for epoch in range(epochs):\n",
    "\n",
    "    model.train()\n",
    "    train_loss_total = 0\n",
    "    train_loss_cls = 0\n",
    "    train_loss_align = 0\n",
    "    train_preds_all = []\n",
    "    train_labels_all = []\n",
    "\n",
    "    for batch in tqdm(train_loader, desc=f\"Epoch {epoch+1}/{epochs} Training\", unit=\"batch\"):\n",
    "        optimizer.zero_grad()\n",
    "\n",
    "        embed1, embed2, labels = batch\n",
    "        embed1 = embed1.to(device)\n",
    "        embed2 = embed2.to(device)\n",
    "        labels = labels.float().to(device).view(-1)\n",
    "\n",
    "        # --- Forward pass ---\n",
    "        outputs , real_emb, fake_emb  = model.forward_training(embed1, embed2)\n",
    "        outputs = outputs.view(-1) \n",
    "\n",
    "        # --- Calculate Classification Loss (for all samples) ---\n",
    "        classification_loss = classification_criterion(outputs, labels)\n",
    "\n",
    " \n",
    "        real_indices = (labels == 0).nonzero(as_tuple=True)[0]\n",
    "\n",
    "        alignment_loss = torch.tensor(0.0).to(device) # Initialize alignment loss for this batch\n",
    "        if real_indices.nelement() > 0: # Check if there are any real samples in the batch\n",
    "            # 2. Select the embeddings corresponding to real samples\n",
    "            real_emb_filtered = real_emb[real_indices]\n",
    "            fake_emb_filtered = fake_emb[real_indices]\n",
    "\n",
    "            # 3. Calculate alignment loss (e.g., MSE between real and fake embeddings for real samples)\n",
    "            alignment_loss = alignment_criterion(real_emb_filtered, fake_emb_filtered)\n",
    "\n",
    "        # --- Combine Losses ---\n",
    "        total_loss = classification_loss + alignment_loss_weight * alignment_loss\n",
    "\n",
    "        # --- Backward pass and optimization ---\n",
    "        total_loss.backward()\n",
    "        optimizer.step()\n",
    "\n",
    "        # --- Accumulate metrics and losses for reporting ---\n",
    "        train_loss_total += total_loss.item()\n",
    "        train_loss_cls += classification_loss.item()\n",
    "        train_loss_align += alignment_loss.item() # Note: this is the raw alignment loss before weighting\n",
    "\n",
    "        preds = torch.sigmoid(outputs).detach().round() # Get predictions (0 or 1)\n",
    "        train_preds_all.extend(preds.cpu().numpy())\n",
    "        train_labels_all.extend(labels.cpu().numpy())\n",
    "\n",
    "    # --- Calculate Epoch Metrics (Training) ---\n",
    "    num_batches = len(train_loader)\n",
    "    avg_train_loss_total = train_loss_total / num_batches\n",
    "    avg_train_loss_cls = train_loss_cls / num_batches\n",
    "    avg_train_loss_align = train_loss_align / num_batches # Average raw alignment loss across batches where it was calculated\n",
    "\n",
    "    train_acc = accuracy_score(train_labels_all, train_preds_all)\n",
    "    train_f1 = f1_score(train_labels_all, train_preds_all, average='binary', zero_division=0)\n",
    "    train_recall = recall_score(train_labels_all, train_preds_all, average='binary', zero_division=0)\n",
    "    train_precision = precision_score(train_labels_all, train_preds_all, average='binary', zero_division=0)\n",
    "\n",
    "    print(f\"Epoch {epoch+1}/{epochs}\")\n",
    "    print(f\"  Train Total Loss: {avg_train_loss_total:.4f} | CLS Loss: {avg_train_loss_cls:.4f} | Align Loss (raw): {avg_train_loss_align:.4f}\")\n",
    "    print(f\"  Train Acc: {train_acc:.4f} | F1: {train_f1:.4f} | Recall: {train_recall:.4f} | Precision: {train_precision:.4f}\")\n",
    "\n",
    "\n",
    "    # --- Validation ---\n",
    "    model.eval()\n",
    "    val_loss_total = 0\n",
    "    val_loss_cls = 0\n",
    "    val_loss_align = 0 \n",
    "    val_preds_all = []\n",
    "    val_labels_all = []\n",
    "\n",
    "    with torch.no_grad():\n",
    "        for batch in tqdm(val_loader, desc=f\"Epoch {epoch+1}/{epochs} Validation\", unit=\"batch\"):\n",
    "            embed1, embed2, labels = batch\n",
    "            embed1 = embed1.to(device)\n",
    "            embed2 = embed2.to(device)\n",
    "            labels = labels.float().to(device).view(-1)\n",
    "\n",
    "            # --- Forward pass (validation) ---\n",
    "            outputs , real_emb, fake_emb  = model.forward_training(embed1, embed2)\n",
    "            outputs = outputs.view(-1)\n",
    "\n",
    "            # --- Calculate Classification Loss (Validation) ---\n",
    "            classification_loss = classification_criterion(outputs, labels)\n",
    "\n",
    "            real_indices = (labels == 0).nonzero(as_tuple=True)[0]\n",
    "            alignment_loss = torch.tensor(0.0).to(device)\n",
    "            if real_indices.nelement() > 0:\n",
    "                real_emb_filtered = real_emb[real_indices]\n",
    "                fake_emb_filtered = fake_emb[real_indices]\n",
    "                alignment_loss = alignment_criterion(real_emb_filtered, fake_emb_filtered)\n",
    "\n",
    "            total_loss = classification_loss + alignment_loss_weight * alignment_loss\n",
    "\n",
    "            val_loss_total += total_loss.item()\n",
    "            val_loss_cls += classification_loss.item()\n",
    "            val_loss_align += alignment_loss.item()\n",
    "\n",
    "            preds = torch.sigmoid(outputs).detach().round()\n",
    "            val_preds_all.extend(preds.cpu().numpy())\n",
    "            val_labels_all.extend(labels.cpu().numpy())\n",
    "\n",
    "    num_batches_val = len(val_loader)\n",
    "    avg_val_loss_total = val_loss_total / num_batches_val\n",
    "    avg_val_loss_cls = val_loss_cls / num_batches_val\n",
    "    avg_val_loss_align = val_loss_align / num_batches_val\n",
    "\n",
    "    val_acc = accuracy_score(val_labels_all, val_preds_all)\n",
    "    val_f1 = f1_score(val_labels_all, val_preds_all, average='binary', zero_division=0)\n",
    "    val_recall = recall_score(val_labels_all, val_preds_all, average='binary', zero_division=0)\n",
    "    val_precision = precision_score(val_labels_all, val_preds_all, average='binary', zero_division=0)\n",
    "\n",
    "    print(f\"  Val Total Loss: {avg_val_loss_total:.4f} | CLS Loss: {avg_val_loss_cls:.4f} | Align Loss (raw): {avg_val_loss_align:.4f}\")\n",
    "    print(f\"  Val Acc: {val_acc:.4f} | F1: {val_f1:.4f} | Recall: {val_recall:.4f} | Precision: {val_precision:.4f}\")\n",
    "\n",
    "    current_val_metric = val_acc \n",
    "\n",
    "    #Best F1\n",
    "    if val_f1 > best_f1:\n",
    "        best_f1 = val_f1\n",
    "        print(f\"  New Best F1: {best_f1:.4f}\")\n",
    "        torch.save(model.state_dict(), os.path.join(save_folder, save_name))\n",
    "    else:\n",
    "        print(f\"  Model not saved. Best F1 so far: {best_f1:.4f}\")\n",
    "\n",
    "\n",
    "\n",
    "print(\"Training Finished.\")\n",
    "print(f\"Best Validation Accuracy achieved: {best_acc:.4f}\")\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "2426b781",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Testing: 100%|██████████| 2139/2139 [00:05<00:00, 367.26batch/s]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Test Total Loss: 0.3365 | CLS Loss: 0.3158 | Align Loss (raw): 0.0414\n",
      "Test Acc: 0.9290 | F1: 0.9238 | Recall: 0.8639 | Precision: 0.9925\n",
      "EER: 0.0603\n"
     ]
    }
   ],
   "source": [
    "#On test set\n",
    "model.load_state_dict(torch.load(os.path.join(save_folder, f\"{save_name}\")))\n",
    "model.eval()\n",
    "test_loss_total = 0\n",
    "test_loss_cls = 0\n",
    "test_loss_align = 0\n",
    "test_preds_all = []\n",
    "test_labels_all = []\n",
    "test_scores_all = []    \n",
    "\n",
    "\n",
    "with torch.no_grad():\n",
    "    for batch in tqdm(test_loader, desc=\"Testing\", unit=\"batch\"):\n",
    "        embed1, embed2, labels = batch\n",
    "        embed1 = embed1.to(device)\n",
    "        embed2 = embed2.to(device)\n",
    "        labels = labels.float().to(device).view(-1)\n",
    "\n",
    "        # --- Forward pass (testing) ---\n",
    "        outputs , real_emb, fake_emb  = model.forward_training(embed1, embed2)\n",
    "        outputs = outputs.view(-1)\n",
    "\n",
    "        # --- Calculate Classification Loss (Testing) ---\n",
    "        classification_loss = classification_criterion(outputs, labels)\n",
    "\n",
    "        real_indices = (labels == 1).nonzero(as_tuple=True)[0]\n",
    "        alignment_loss = torch.tensor(0.0).to(device)\n",
    "        if real_indices.nelement() > 0:\n",
    "            real_emb_filtered = real_emb[real_indices]\n",
    "            fake_emb_filtered = fake_emb[real_indices]\n",
    "            alignment_loss = alignment_criterion(real_emb_filtered, fake_emb_filtered)\n",
    "\n",
    "        total_loss = classification_loss + alignment_loss_weight * alignment_loss\n",
    "\n",
    "        test_loss_total += total_loss.item()\n",
    "        test_loss_cls += classification_loss.item()\n",
    "        test_loss_align += alignment_loss.item()\n",
    "\n",
    "        preds = torch.sigmoid(outputs).detach().round()\n",
    "        test_preds_all.extend(preds.cpu().numpy())\n",
    "        test_labels_all.extend(labels.cpu().numpy())\n",
    "        test_scores_all.extend(torch.sigmoid(outputs).cpu().numpy())\n",
    "\n",
    "    num_batches_test = len(test_loader)\n",
    "    avg_test_loss_total = test_loss_total / num_batches_test\n",
    "    avg_test_loss_cls = test_loss_cls / num_batches_test    \n",
    "    avg_test_loss_align = test_loss_align / num_batches_test\n",
    "    test_acc = accuracy_score(test_labels_all, test_preds_all)  \n",
    "    test_f1 = f1_score(test_labels_all, test_preds_all, average='binary', zero_division=0)\n",
    "    test_recall = recall_score(test_labels_all, test_preds_all, average='binary', zero_division=0)\n",
    "    test_precision = precision_score(test_labels_all, test_preds_all, average='binary', zero_division=0)\n",
    "    eer = compute_eer(test_labels_all, test_scores_all)  # Calculate EER using the raw scores\n",
    "    print(f\"Test Total Loss: {avg_test_loss_total:.4f} | CLS Loss: {avg_test_loss_cls:.4f} | Align Loss (raw): {avg_test_loss_align:.4f}\")\n",
    "    print(f\"Test Acc: {test_acc:.4f} | F1: {test_f1:.4f} | Recall: {test_recall:.4f} | Precision: {test_precision:.4f}\")\n",
    "    print(f\"EER: {eer:.4f}\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.13.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
