{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Latin Square Task NN\n",
    "Somewhat useful tutorial?\n",
    "https://cs230.stanford.edu/blog/namedentity/"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "device = torch.device('cpu')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "from dataset import puzzles_to_tensor\n",
    "train_puzzles, train_solutions = puzzles_to_tensor(['generated_puzzle_data_binary.csv'])\n",
    "test_puzzles, test_solutions = puzzles_to_tensor('puzzle_data_original.csv')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from dataset import PuzzleDataset\n",
    "import torch\n",
    "\n",
    "# create the datasets\n",
    "train_dataset = PuzzleDataset(train_puzzles,train_solutions)\n",
    "train_dataloader = torch.utils.data.DataLoader(train_dataset, batch_size=300, shuffle=True)\n",
    "\n",
    "test_dataset = PuzzleDataset(test_puzzles, test_solutions)\n",
    "test_dataloader = torch.utils.data.DataLoader(test_dataset, batch_size=108, shuffle=True)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "LSTM_row_col(\n",
      "  (lstm_row): LSTM(5, 160, batch_first=True, bidirectional=True)\n",
      "  (lstm_col): LSTM(5, 160, batch_first=True, bidirectional=True)\n",
      "  (final_layer): Sequential(\n",
      "    (0): Linear(in_features=320, out_features=4, bias=True)\n",
      "    (1): Softmax(dim=-1)\n",
      "  )\n",
      ")\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "'''\n",
    "Based on the approach in https://cs230.stanford.edu/files_winter_2018/projects/6939771.pdf\n",
    "where different LSTM layers compute row related or column related information.\n",
    "\n",
    "'''\n",
    "# https://blog.floydhub.com/long-short-term-memory-from-zero-to-hero-with-pytorch/\n",
    "class LSTM_row_col(nn.Module):\n",
    "    def __init__(self, input_dim=5, hidden_dim=512, n_layers=1, output_dim=4, bidirectional=True, dropout=0, device='cpu'):\n",
    "        super(LSTM_row_col, self).__init__()\n",
    "        self.n_layers = n_layers\n",
    "        self.bidirectional = bidirectional\n",
    "        self.device = device\n",
    "\n",
    "        self.lstm_row = nn.LSTM(\n",
    "            input_dim, hidden_dim, n_layers, batch_first=True, bidirectional=bidirectional,dropout=dropout)\n",
    "        self.lstm_col = nn.LSTM(\n",
    "            input_dim, hidden_dim, n_layers, batch_first=True, bidirectional=bidirectional,dropout=dropout)\n",
    "\n",
    "        if self.bidirectional:\n",
    "            self.hidden_dim = hidden_dim*2\n",
    "        else:\n",
    "            self.hidden_dim = hidden_dim\n",
    "        self.final_layer = nn.Sequential(torch.nn.Linear(self.hidden_dim, output_dim),\n",
    "                                          torch.nn.Softmax(dim = -1))\n",
    "        \n",
    "\n",
    "    def forward(self, x, hidden):\n",
    "        # see https://github.com/charlesakin/sudoku/blob/master/3LayerLSTMwithConfusionMatrix.ipynb\n",
    "        batch_size = x.size(0)\n",
    "\n",
    "        x_row = x.flatten(start_dim=1, end_dim=2)\n",
    "        x_col = x.transpose(1,2).flatten(start_dim=1, end_dim=2)\n",
    "\n",
    "        lstm_row_out, hidden_row = self.lstm_row(x_row, hidden[0])\n",
    "        lstm_col_out, hidden_col = self.lstm_row(x_col, hidden[1])\n",
    "\n",
    "        combined = lstm_row_out + lstm_col_out\n",
    "        out = self.final_layer(combined[:, -1, :])\n",
    "\n",
    "        # detach things\n",
    "        hidden_row = (hidden_row[0].detach(), hidden_row[1].detach())\n",
    "        hidden_col = (hidden_col[0].detach(), hidden_col[1].detach())\n",
    "        return out, (hidden_row, hidden_col)\n",
    "\n",
    "    def init_hidden(self, batch_size):\n",
    "        weight = next(self.parameters()).data\n",
    "        if self.bidirectional:\n",
    "            hidden = ((weight.new(self.n_layers*2, batch_size, int(self.hidden_dim/2)).zero_().to(device),\n",
    "                      weight.new(self.n_layers*2, batch_size, int(self.hidden_dim/2)).zero_().to(device)),\n",
    "                      (weight.new(self.n_layers*2, batch_size, int(self.hidden_dim/2)).zero_().to(device),\n",
    "                      weight.new(self.n_layers*2, batch_size, int(self.hidden_dim/2)).zero_().to(device)))\n",
    "        else:\n",
    "            hidden = ((weight.new(self.n_layers, batch_size, self.hidden_dim).zero_().to(device),\n",
    "                      weight.new(self.n_layers, batch_size, self.hidden_dim).zero_().to(device)),\n",
    "                      (weight.new(self.n_layers, batch_size, self.hidden_dim).zero_().to(device),\n",
    "                      weight.new(self.n_layers, batch_size, self.hidden_dim).zero_().to(device)))\n",
    "        return hidden\n",
    "\n",
    "model = LSTM_row_col(hidden_dim=160, n_layers=1, bidirectional=True, device=device)\n",
    "print(model)\n",
    "hidden = model.init_hidden(batch_size=24)\n",
    "pred, hidden = model(train_dataset[0:24][0].to(device), hidden)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def train_loop(dataloader, model, loss_fn, optimizer):\n",
    "    size = len(dataloader.dataset)\n",
    "    num_batches = len(dataloader)\n",
    "    h = model.init_hidden(batch_size=int(size/num_batches))\n",
    "    index_vec = np.zeros((int(size/num_batches), num_batches))\n",
    "    correct_vec = index_vec.copy()\n",
    "\n",
    "    c = 0\n",
    "    for batch, (X, y, index) in enumerate(dataloader):\n",
    "\n",
    "        # Compute prediction and loss\n",
    "        pred, h = model(X.to(device), h)\n",
    "        loss = loss_fn(pred, y.to(device))\n",
    "\n",
    "        # Backpropagation\n",
    "        optimizer.zero_grad()\n",
    "        loss.backward()\n",
    "        optimizer.step()\n",
    "\n",
    "        # save\n",
    "        index_vec[:, c] = index\n",
    "        correct_vec[:, c] = (pred.argmax(1) == y.to(device).argmax(1)).cpu()\n",
    "        c += 1\n",
    "    return correct_vec.reshape(-1), index_vec.reshape(-1)\n",
    "\n",
    "def test_loop(dataloader, model, loss_fn):\n",
    "    size = len(dataloader.dataset)\n",
    "    num_batches = len(dataloader)\n",
    "    h = model.init_hidden(batch_size=int(size/num_batches))\n",
    "    test_loss, correct = 0, 0\n",
    "    index_vec = np.zeros((int(size/num_batches), num_batches))\n",
    "    correct_vec = index_vec.copy()\n",
    "\n",
    "    with torch.no_grad():  # turn off learning\n",
    "        c = 0\n",
    "        for X, y, index in dataloader:\n",
    "\n",
    "            # make prediction\n",
    "            pred, _ = model(X.to(device), h)\n",
    "\n",
    "            # calculate loss across batches\n",
    "            test_loss += loss_fn(pred, y.to(device)).item()\n",
    "\n",
    "            # calculate correct across batches\n",
    "            correct += (pred.argmax(1) == y.to(device).argmax(1)\n",
    "                        ).type(torch.float).sum().item()\n",
    "\n",
    "            # save indices and accuracy to vector (used to investigate conditions)\n",
    "            index_vec[:, c] = index\n",
    "            correct_vec[:, c] = (pred.argmax(1) == y.to(device).argmax(1)).cpu()\n",
    "            c += 1\n",
    "    test_loss /= num_batches\n",
    "    correct /= size\n",
    "    return correct_vec.reshape(-1), index_vec.reshape(-1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 100/100 [09:51<00:00,  5.91s/it]\n"
     ]
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "from torch import nn\n",
    "from tqdm import tqdm\n",
    "import pandas as pd\n",
    "\n",
    "learning_rate = 0.01\n",
    "loss_fn = nn.CrossEntropyLoss()\n",
    "optimizer = torch.optim.Adam(model.parameters(), lr=learning_rate)\n",
    "\n",
    "epochs = 100\n",
    "results = pd.DataFrame(\n",
    "  columns=['epoch', 'accuracy', 'condition'])\n",
    "training_results = results.copy()\n",
    "details_df = pd.read_csv('puzzle_data_original.csv')\n",
    "\n",
    "def match_condition(item, key_df):\n",
    "    return key_df[key_df.ID == item].condition.values[0]\n",
    "\n",
    "for t in tqdm(range(epochs)):\n",
    "    # train\n",
    "    correct, index = train_loop(train_dataloader, model, loss_fn, optimizer)\n",
    "\n",
    "    row = pd.DataFrame({'epoch': [t],\n",
    "                        'accuracy': [np.mean(correct)]})\n",
    "    training_results = pd.concat([training_results, row], ignore_index=True)\n",
    "\n",
    "    # test\n",
    "    correct, index = test_loop(test_dataloader, model, loss_fn)\n",
    "\n",
    "    # get the condition\n",
    "    row = pd.DataFrame({'accuracy': correct,\n",
    "                        'item': index})\n",
    "    row['condition'] = row.item.apply(match_condition, key_df=details_df)\n",
    "\n",
    "    # add to results\n",
    "    row = row.groupby(['condition'], as_index=False)['accuracy'].mean()\n",
    "    row['epoch'] = t\n",
    "    results = pd.concat([results, row], ignore_index=True)\n",
    "\n",
    "    if t % 100 == 0:\n",
    "        training_results.to_csv('results/training_results.csv', index=False)\n",
    "        results.to_csv('results/results.csv', index=False)\n",
    "\n",
    "    \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 3 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# plot\n",
    "sns.set_theme()\n",
    "fig, axs = plt.subplots(1, 3)\n",
    "\n",
    "sns.lineplot(data=training_results, x='epoch', y='accuracy', ax=axs[0])\n",
    "axs[0].set_title('Training accuracy')\n",
    "axs[0].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', ax=axs[1])\n",
    "axs[1].set_title('Test accuracy')\n",
    "axs[1].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', hue='condition', ax=axs[2])\n",
    "axs[2].set_title('Test accuracy by condition')\n",
    "axs[2].set_ylim([0,1])\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  9%|▉         | 9/100 [02:16<22:59, 15.16s/it]\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "\u001b[1;32m/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb Cell 9\u001b[0m in \u001b[0;36m<cell line: 9>\u001b[0;34m()\u001b[0m\n\u001b[1;32m      <a href='vscode-notebook-cell:/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb#X11sZmlsZQ%3D%3D?line=5'>6</a>\u001b[0m train_dataloader \u001b[39m=\u001b[39m torch\u001b[39m.\u001b[39mutils\u001b[39m.\u001b[39mdata\u001b[39m.\u001b[39mDataLoader(\n\u001b[1;32m      <a href='vscode-notebook-cell:/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb#X11sZmlsZQ%3D%3D?line=6'>7</a>\u001b[0m     train_dataset, batch_size\u001b[39m=\u001b[39m\u001b[39m300\u001b[39m, shuffle\u001b[39m=\u001b[39m\u001b[39mTrue\u001b[39;00m)\n\u001b[1;32m      <a href='vscode-notebook-cell:/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb#X11sZmlsZQ%3D%3D?line=8'>9</a>\u001b[0m \u001b[39mfor\u001b[39;00m t \u001b[39min\u001b[39;00m tqdm(\u001b[39mrange\u001b[39m(epochs)):\n\u001b[1;32m     <a href='vscode-notebook-cell:/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb#X11sZmlsZQ%3D%3D?line=9'>10</a>\u001b[0m     \u001b[39m# train\u001b[39;00m\n\u001b[0;32m---> <a href='vscode-notebook-cell:/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb#X11sZmlsZQ%3D%3D?line=10'>11</a>\u001b[0m     correct, index \u001b[39m=\u001b[39m train_loop(train_dataloader, model, loss_fn, optimizer)\n\u001b[1;32m     <a href='vscode-notebook-cell:/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb#X11sZmlsZQ%3D%3D?line=12'>13</a>\u001b[0m     row \u001b[39m=\u001b[39m pd\u001b[39m.\u001b[39mDataFrame({\u001b[39m'\u001b[39m\u001b[39mepoch\u001b[39m\u001b[39m'\u001b[39m: [t\u001b[39m+\u001b[39mepochs],\n\u001b[1;32m     <a href='vscode-notebook-cell:/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb#X11sZmlsZQ%3D%3D?line=13'>14</a>\u001b[0m                         \u001b[39m'\u001b[39m\u001b[39maccuracy\u001b[39m\u001b[39m'\u001b[39m: [np\u001b[39m.\u001b[39mmean(correct)]})\n\u001b[1;32m     <a href='vscode-notebook-cell:/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb#X11sZmlsZQ%3D%3D?line=14'>15</a>\u001b[0m     training_results \u001b[39m=\u001b[39m pd\u001b[39m.\u001b[39mconcat([training_results, row], ignore_index\u001b[39m=\u001b[39m\u001b[39mTrue\u001b[39;00m)\n",
      "\u001b[1;32m/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb Cell 9\u001b[0m in \u001b[0;36mtrain_loop\u001b[0;34m(dataloader, model, loss_fn, optimizer)\u001b[0m\n\u001b[1;32m     <a href='vscode-notebook-cell:/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb#X11sZmlsZQ%3D%3D?line=14'>15</a>\u001b[0m \u001b[39m# Backpropagation\u001b[39;00m\n\u001b[1;32m     <a href='vscode-notebook-cell:/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb#X11sZmlsZQ%3D%3D?line=15'>16</a>\u001b[0m optimizer\u001b[39m.\u001b[39mzero_grad()\n\u001b[0;32m---> <a href='vscode-notebook-cell:/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb#X11sZmlsZQ%3D%3D?line=16'>17</a>\u001b[0m loss\u001b[39m.\u001b[39;49mbackward()\n\u001b[1;32m     <a href='vscode-notebook-cell:/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb#X11sZmlsZQ%3D%3D?line=17'>18</a>\u001b[0m optimizer\u001b[39m.\u001b[39mstep()\n\u001b[1;32m     <a href='vscode-notebook-cell:/Users/lukehearne/Documents/projects/LSTNN/code/LTSM_scratch.ipynb#X11sZmlsZQ%3D%3D?line=19'>20</a>\u001b[0m \u001b[39m# save\u001b[39;00m\n",
      "File \u001b[0;32m/opt/homebrew/Caskroom/miniforge/base/envs/LSTANN/lib/python3.9/site-packages/torch/_tensor.py:396\u001b[0m, in \u001b[0;36mTensor.backward\u001b[0;34m(self, gradient, retain_graph, create_graph, inputs)\u001b[0m\n\u001b[1;32m    387\u001b[0m \u001b[39mif\u001b[39;00m has_torch_function_unary(\u001b[39mself\u001b[39m):\n\u001b[1;32m    388\u001b[0m     \u001b[39mreturn\u001b[39;00m handle_torch_function(\n\u001b[1;32m    389\u001b[0m         Tensor\u001b[39m.\u001b[39mbackward,\n\u001b[1;32m    390\u001b[0m         (\u001b[39mself\u001b[39m,),\n\u001b[0;32m   (...)\u001b[0m\n\u001b[1;32m    394\u001b[0m         create_graph\u001b[39m=\u001b[39mcreate_graph,\n\u001b[1;32m    395\u001b[0m         inputs\u001b[39m=\u001b[39minputs)\n\u001b[0;32m--> 396\u001b[0m torch\u001b[39m.\u001b[39;49mautograd\u001b[39m.\u001b[39;49mbackward(\u001b[39mself\u001b[39;49m, gradient, retain_graph, create_graph, inputs\u001b[39m=\u001b[39;49minputs)\n",
      "File \u001b[0;32m/opt/homebrew/Caskroom/miniforge/base/envs/LSTANN/lib/python3.9/site-packages/torch/autograd/__init__.py:173\u001b[0m, in \u001b[0;36mbackward\u001b[0;34m(tensors, grad_tensors, retain_graph, create_graph, grad_variables, inputs)\u001b[0m\n\u001b[1;32m    168\u001b[0m     retain_graph \u001b[39m=\u001b[39m create_graph\n\u001b[1;32m    170\u001b[0m \u001b[39m# The reason we repeat same the comment below is that\u001b[39;00m\n\u001b[1;32m    171\u001b[0m \u001b[39m# some Python versions print out the first line of a multi-line function\u001b[39;00m\n\u001b[1;32m    172\u001b[0m \u001b[39m# calls in the traceback and some print out the last line\u001b[39;00m\n\u001b[0;32m--> 173\u001b[0m Variable\u001b[39m.\u001b[39;49m_execution_engine\u001b[39m.\u001b[39;49mrun_backward(  \u001b[39m# Calls into the C++ engine to run the backward pass\u001b[39;49;00m\n\u001b[1;32m    174\u001b[0m     tensors, grad_tensors_, retain_graph, create_graph, inputs,\n\u001b[1;32m    175\u001b[0m     allow_unreachable\u001b[39m=\u001b[39;49m\u001b[39mTrue\u001b[39;49;00m, accumulate_grad\u001b[39m=\u001b[39;49m\u001b[39mTrue\u001b[39;49;00m)\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "# get next training set\n",
    "# create the datasets\n",
    "train_puzzles, train_solutions = puzzles_to_tensor(['generated_puzzle_data_binary.csv','generated_puzzle_data_ternary.csv'])\n",
    "# create the datasets\n",
    "train_dataset = PuzzleDataset(train_puzzles, train_solutions)\n",
    "train_dataloader = torch.utils.data.DataLoader(\n",
    "    train_dataset, batch_size=300, shuffle=True)\n",
    "\n",
    "for t in tqdm(range(epochs)):\n",
    "    # train\n",
    "    correct, index = train_loop(train_dataloader, model, loss_fn, optimizer)\n",
    "\n",
    "    row = pd.DataFrame({'epoch': [t+epochs],\n",
    "                        'accuracy': [np.mean(correct)]})\n",
    "    training_results = pd.concat([training_results, row], ignore_index=True)\n",
    "\n",
    "    # test\n",
    "    correct, index = test_loop(test_dataloader, model, loss_fn)\n",
    "\n",
    "    # get the condition\n",
    "    row = pd.DataFrame({'accuracy': correct,\n",
    "                        'item': index})\n",
    "    row['condition'] = row.item.apply(match_condition, key_df=details_df)\n",
    "\n",
    "    # add to results\n",
    "    row = row.groupby(['condition'], as_index=False)['accuracy'].mean()\n",
    "    row['epoch'] = t+epochs\n",
    "    results = pd.concat([results, row], ignore_index=True)\n",
    "\n",
    "    if t % 100 == 0:\n",
    "        training_results.to_csv('results/training_results.csv', index=False)\n",
    "        results.to_csv('results/results.csv', index=False)\n",
    "\n",
    "fig, axs = plt.subplots(1, 3)\n",
    "\n",
    "sns.lineplot(data=training_results, x='epoch', y='accuracy', ax=axs[0])\n",
    "axs[0].set_title('Training accuracy')\n",
    "axs[0].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', ax=axs[1])\n",
    "axs[1].set_title('Test accuracy')\n",
    "axs[1].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', hue='condition', ax=axs[2])\n",
    "axs[2].set_title('Test accuracy by condition')\n",
    "axs[2].set_ylim([0,1])\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# get next training set\n",
    "# create the datasets\n",
    "train_puzzles, train_solutions = puzzles_to_tensor(['generated_puzzle_data_binary.csv','generated_puzzle_data_ternary.csv', 'generated_puzzle_data_quaternary.csv'])\n",
    "# create the datasets\n",
    "train_dataset = PuzzleDataset(train_puzzles, train_solutions)\n",
    "train_dataloader = torch.utils.data.DataLoader(\n",
    "    train_dataset, batch_size=300, shuffle=True)\n",
    "\n",
    "for t in tqdm(range(epochs)):\n",
    "    # train\n",
    "    correct, index = train_loop(train_dataloader, model, loss_fn, optimizer)\n",
    "\n",
    "    row = pd.DataFrame({'epoch': [t+epochs*2],\n",
    "                        'accuracy': [np.mean(correct)]})\n",
    "    training_results = pd.concat([training_results, row], ignore_index=True)\n",
    "\n",
    "    # test\n",
    "    correct, index = test_loop(test_dataloader, model, loss_fn)\n",
    "\n",
    "    # get the condition\n",
    "    row = pd.DataFrame({'accuracy': correct,\n",
    "                        'item': index})\n",
    "    row['condition'] = row.item.apply(match_condition, key_df=details_df)\n",
    "\n",
    "    # add to results\n",
    "    row = row.groupby(['condition'], as_index=False)['accuracy'].mean()\n",
    "    row['epoch'] = t+epochs*2\n",
    "    results = pd.concat([results, row], ignore_index=True)\n",
    "\n",
    "    if t % 100 == 0:\n",
    "        training_results.to_csv('results/training_results.csv', index=False)\n",
    "        results.to_csv('results/results.csv', index=False)\n",
    "\n",
    "fig, axs = plt.subplots(1, 3, figsize=(15,5))\n",
    "\n",
    "sns.lineplot(data=training_results, x='epoch', y='accuracy', ax=axs[0])\n",
    "axs[0].set_title('Training accuracy')\n",
    "axs[0].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', ax=axs[1])\n",
    "axs[1].set_title('Test accuracy')\n",
    "axs[1].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', hue='condition', ax=axs[2])\n",
    "axs[2].set_title('Test accuracy by condition')\n",
    "axs[2].set_ylim([0,1])\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# get next training set\n",
    "# create the datasets\n",
    "train_puzzles, train_solutions = puzzles_to_tensor(['generated_puzzle_data_ternary.csv'])\n",
    "# create the datasets\n",
    "train_dataset = PuzzleDataset(train_puzzles, train_solutions)\n",
    "train_dataloader = torch.utils.data.DataLoader(\n",
    "    train_dataset, batch_size=300, shuffle=True)\n",
    "\n",
    "for t in tqdm(range(epochs)):\n",
    "    # train\n",
    "    correct, index = train_loop(train_dataloader, model, loss_fn, optimizer)\n",
    "\n",
    "    row = pd.DataFrame({'epoch': [t+epochs*3],\n",
    "                        'accuracy': [np.mean(correct)]})\n",
    "    training_results = pd.concat([training_results, row], ignore_index=True)\n",
    "\n",
    "    # test\n",
    "    correct, index = test_loop(test_dataloader, model, loss_fn)\n",
    "\n",
    "    # get the condition\n",
    "    row = pd.DataFrame({'accuracy': correct,\n",
    "                        'item': index})\n",
    "    row['condition'] = row.item.apply(match_condition, key_df=details_df)\n",
    "\n",
    "    # add to results\n",
    "    row = row.groupby(['condition'], as_index=False)['accuracy'].mean()\n",
    "    row['epoch'] = t+epochs*3\n",
    "    results = pd.concat([results, row], ignore_index=True)\n",
    "\n",
    "    if t % 100 == 0:\n",
    "        training_results.to_csv('results/training_results.csv', index=False)\n",
    "        results.to_csv('results/results.csv', index=False)\n",
    "\n",
    "fig, axs = plt.subplots(1, 3)\n",
    "\n",
    "sns.lineplot(data=training_results, x='epoch', y='accuracy', ax=axs[0])\n",
    "axs[0].set_title('Training accuracy')\n",
    "axs[0].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', ax=axs[1])\n",
    "axs[1].set_title('Test accuracy')\n",
    "axs[1].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', hue='condition', ax=axs[2])\n",
    "axs[2].set_title('Test accuracy by condition')\n",
    "axs[2].set_ylim([0,1])\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# get next training set\n",
    "# create the datasets\n",
    "train_puzzles, train_solutions = puzzles_to_tensor(['generated_puzzle_data_quaternary.csv'])\n",
    "# create the datasets\n",
    "train_dataset = PuzzleDataset(train_puzzles, train_solutions)\n",
    "train_dataloader = torch.utils.data.DataLoader(\n",
    "    train_dataset, batch_size=300, shuffle=True)\n",
    "\n",
    "for t in tqdm(range(epochs)):\n",
    "    # train\n",
    "    correct, index = train_loop(train_dataloader, model, loss_fn, optimizer)\n",
    "\n",
    "    row = pd.DataFrame({'epoch': [t+epochs*4],\n",
    "                        'accuracy': [np.mean(correct)]})\n",
    "    training_results = pd.concat([training_results, row], ignore_index=True)\n",
    "\n",
    "    # test\n",
    "    correct, index = test_loop(test_dataloader, model, loss_fn)\n",
    "\n",
    "    # get the condition\n",
    "    row = pd.DataFrame({'accuracy': correct,\n",
    "                        'item': index})\n",
    "    row['condition'] = row.item.apply(match_condition, key_df=details_df)\n",
    "\n",
    "    # add to results\n",
    "    row = row.groupby(['condition'], as_index=False)['accuracy'].mean()\n",
    "    row['epoch'] = t+epochs*4\n",
    "    results = pd.concat([results, row], ignore_index=True)\n",
    "\n",
    "    if t % 100 == 0:\n",
    "        training_results.to_csv('results/training_results.csv', index=False)\n",
    "        results.to_csv('results/results.csv', index=False)\n",
    "\n",
    "fig, axs = plt.subplots(1, 3, figsize=(15,5))\n",
    "\n",
    "sns.lineplot(data=training_results, x='epoch', y='accuracy', ax=axs[0])\n",
    "axs[0].set_title('Training accuracy')\n",
    "axs[0].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', ax=axs[1])\n",
    "axs[1].set_title('Test accuracy')\n",
    "axs[1].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', hue='condition', ax=axs[2])\n",
    "axs[2].set_title('Test accuracy by condition')\n",
    "axs[2].set_ylim([0,1])\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# get next training set\n",
    "# create the datasets\n",
    "train_puzzles, train_solutions = puzzles_to_tensor(['generated_puzzle_data_binary.csv'])\n",
    "# create the datasets\n",
    "train_dataset = PuzzleDataset(train_puzzles, train_solutions)\n",
    "train_dataloader = torch.utils.data.DataLoader(\n",
    "    train_dataset, batch_size=300, shuffle=True)\n",
    "\n",
    "for t in tqdm(range(epochs)):\n",
    "    # train\n",
    "    correct, index = train_loop(train_dataloader, model, loss_fn, optimizer)\n",
    "\n",
    "    row = pd.DataFrame({'epoch': [t+epochs*5],\n",
    "                        'accuracy': [np.mean(correct)]})\n",
    "    training_results = pd.concat([training_results, row], ignore_index=True)\n",
    "\n",
    "    # test\n",
    "    correct, index = test_loop(test_dataloader, model, loss_fn)\n",
    "\n",
    "    # get the condition\n",
    "    row = pd.DataFrame({'accuracy': correct,\n",
    "                        'item': index})\n",
    "    row['condition'] = row.item.apply(match_condition, key_df=details_df)\n",
    "\n",
    "    # add to results\n",
    "    row = row.groupby(['condition'], as_index=False)['accuracy'].mean()\n",
    "    row['epoch'] = t+epochs*5\n",
    "    results = pd.concat([results, row], ignore_index=True)\n",
    "\n",
    "    if t % 100 == 0:\n",
    "        training_results.to_csv('results/training_results.csv', index=False)\n",
    "        results.to_csv('results/results.csv', index=False)\n",
    "\n",
    "fig, axs = plt.subplots(1, 3)\n",
    "\n",
    "sns.lineplot(data=training_results, x='epoch', y='accuracy', ax=axs[0])\n",
    "axs[0].set_title('Training accuracy')\n",
    "axs[0].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', ax=axs[1])\n",
    "axs[1].set_title('Test accuracy')\n",
    "axs[1].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', hue='condition', ax=axs[2])\n",
    "axs[2].set_title('Test accuracy by condition')\n",
    "axs[2].set_ylim([0,1])\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# get next training set\n",
    "# create the datasets\n",
    "train_puzzles, train_solutions = puzzles_to_tensor(['generated_puzzle_data_quaternary.csv'])\n",
    "# create the datasets\n",
    "train_dataset = PuzzleDataset(train_puzzles, train_solutions)\n",
    "train_dataloader = torch.utils.data.DataLoader(\n",
    "    train_dataset, batch_size=300, shuffle=True)\n",
    "\n",
    "for t in tqdm(range(epochs)):\n",
    "    # train\n",
    "    correct, index = train_loop(train_dataloader, model, loss_fn, optimizer)\n",
    "\n",
    "    row = pd.DataFrame({'epoch': [t+epochs*6],\n",
    "                        'accuracy': [np.mean(correct)]})\n",
    "    training_results = pd.concat([training_results, row], ignore_index=True)\n",
    "\n",
    "    # test\n",
    "    correct, index = test_loop(test_dataloader, model, loss_fn)\n",
    "\n",
    "    # get the condition\n",
    "    row = pd.DataFrame({'accuracy': correct,\n",
    "                        'item': index})\n",
    "    row['condition'] = row.item.apply(match_condition, key_df=details_df)\n",
    "\n",
    "    # add to results\n",
    "    row = row.groupby(['condition'], as_index=False)['accuracy'].mean()\n",
    "    row['epoch'] = t+epochs*6\n",
    "    results = pd.concat([results, row], ignore_index=True)\n",
    "\n",
    "    if t % 100 == 0:\n",
    "        training_results.to_csv('results/training_results.csv', index=False)\n",
    "        results.to_csv('results/results.csv', index=False)\n",
    "\n",
    "fig, axs = plt.subplots(1, 3, figsize=(15,5))\n",
    "\n",
    "sns.lineplot(data=training_results, x='epoch', y='accuracy', ax=axs[0])\n",
    "axs[0].set_title('Training accuracy')\n",
    "axs[0].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', ax=axs[1])\n",
    "axs[1].set_title('Test accuracy')\n",
    "axs[1].set_ylim([0,1])\n",
    "\n",
    "sns.lineplot(data=results, x='epoch', y='accuracy', hue='condition', ax=axs[2])\n",
    "axs[2].set_title('Test accuracy by condition')\n",
    "axs[2].set_ylim([0,1])\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3.9.13 ('LSTANN')",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.13"
  },
  "orig_nbformat": 4,
  "vscode": {
   "interpreter": {
    "hash": "254965b9f167056e74e837d1730ed8e27293664f71af1499bd499b7f558825b1"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
