{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import torch\n",
    "import lstnn.transformer_main as transformer_main\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "import scipy.stats as stats\n",
    "plt.rcParams['font.sans-serif'] = \"Arial\"\n",
    "sns.set_style(\"ticks\")\n",
    "from lstnn.dataset import get_dataset\n",
    "import numpy as np\n",
    "import os\n",
    "%matplotlib inline\n",
    "%load_ext autoreload\n",
    "%autoreload 2\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Behavioral comparison of Transformer models with Human Behavior"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1d-fixed\n",
      "2d-fixed\n",
      "relative\n",
      "rope\n",
      "learn-0.2\n",
      "random\n",
      "c-nope\n",
      "nope\n"
     ]
    }
   ],
   "source": [
    "model_label = 'Transformer'\n",
    "# model_label = 'BERT'\n",
    "curriculum = 'All'\n",
    "seeds = [2235, 6312, 6068, 9742, 8880, 2197, 669, 6256, 3309, 2541, 8643, 7785, 195, 6914, 29]\n",
    "device = 'mps'\n",
    "\n",
    "# model params\n",
    "nblocks = [4] #, 5]\n",
    "head = 2\n",
    "attnheads = [head]# 4, 8]\n",
    "# wdecays = [0.0, 0.01, 0.05, 0.1, 0.2] #wdecay = 0.0\n",
    "wdecays = [0.0] #, 0.1] #wdecay = 0.0\n",
    "dropout = 0.0\n",
    "hidden_size = 160 #160\n",
    "learning_rate = 0.0001\n",
    "training_acc_cutoff = 0.0\n",
    "cutoff_length = 0  # how many epochs must the model sustain the accuracy cutoff?\n",
    "last_epoch = 4000\n",
    "checkpoint_freq = 500\n",
    "\n",
    "positional_encodings = {\n",
    "      'pe-1dpe_':'absolute',\n",
    "      'pe-2dpe_':'absolute2d',\n",
    "      'pe-shaw_':'relative',\n",
    "      'pe-rope2_':'rope2',\n",
    "      'pe-learn-0.2_':'learn',\n",
    "      'pe-rndpe_':'rndpe',\n",
    "      'pe-cnope_':'cnope',\n",
    "      'pe-nope_':'nope'\n",
    "}\n",
    "pe_labels = {\n",
    "      'pe-1dpe_':'1d-fixed',\n",
    "      'pe-2dpe_':'2d-fixed',\n",
    "      'pe-shaw_':'relative',\n",
    "      'pe-rope2_':'rope',\n",
    "      'pe-learn-0.2_':'learn-0.2',\n",
    "      'pe-rndpe_':'random',\n",
    "      'pe-cnope_':'c-nope',\n",
    "      'pe-nope_':'nope'\n",
    "}\n",
    "\n",
    "# pe_string = '' #'pe-rndpe_' #'pe-nope_' #'pe-2dpe_' #''\n",
    "# pe_param = 'absolute1d' #'rndpe' #'absolute2d'\n",
    "\n",
    "# load results and model\n",
    "\n",
    "df = pd.DataFrame()\n",
    "for pe in positional_encodings:\n",
    "      pestr = pe_labels[pe]\n",
    "      print(pestr)\n",
    "      for epoch in range(0,last_epoch+1,checkpoint_freq):\n",
    "            for layer in nblocks:\n",
    "                  for attnhead in attnheads:\n",
    "                        for wdecay in wdecays:\n",
    "                              resultdir = f\"../results/\"\n",
    "                              modelname = f\"model-{model_label}_\" \\\n",
    "                                          f\"{pe}\" \\\n",
    "                                          f\"nl-{layer}_\" \\\n",
    "                                          f\"do-{dropout}_\" \\\n",
    "                                          f\"wd-{wdecay}_\" \\\n",
    "                                          f\"at-{attnhead}_\" \\\n",
    "                                          f\"hs-{hidden_size}_\" \\\n",
    "                                          f\"curr-{curriculum}_\" \\\n",
    "                                          f\"lr-{learning_rate}_\" \\\n",
    "                                          f\"co-{training_acc_cutoff}_\" \\\n",
    "                                          f\"col-{cutoff_length}/\"\n",
    "                              for seed in seeds:\n",
    "                                    checkpoint = f\"s-{seed}_\" \\\n",
    "                                                f\"e-{epoch}\" \n",
    "                                    # print('\\t',checkpoint)\n",
    "\n",
    "\n",
    "                                    try:\n",
    "                                          _df = pd.read_csv(resultdir + modelname + checkpoint +'.csv')\n",
    "                                          _df['seed'] = seed\n",
    "                                          _df['model'] = modelname\n",
    "                                          _df['layer'] = layer\n",
    "                                          _df['heads'] = attnhead\n",
    "                                          _df['wdecay'] = wdecay\n",
    "                                          _df['pe'] = pestr\n",
    "                                          _df = _df.loc[_df.epoch==epoch]\n",
    "\n",
    "                                          df = pd.concat([df, _df])\n",
    "                                    except:\n",
    "                                          # _df = pd.read_csv(resultdir + modelname + checkpoint +'.csv')\n",
    "\n",
    "                                          continue\n",
    "\n",
    "df_ds = df.copy()\n",
    "# df_ds = pd.DataFrame()\n",
    "# # get max epoch\n",
    "# for i in df.epoch.unique():\n",
    "#     data = df.loc[df.epoch == i]\n",
    "#     df_ds = pd.concat([df_ds, data])\n",
    "\n",
    "models = df_ds.model.unique()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Plot training and validation performances"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 70x100 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 70x100 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAADwAAABZCAYAAABixUebAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8g+/7EAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAJGklEQVR4nO2bXWwTVxbHfzOO7dhjkwQTSJYtULqChHS7oumm0CKCItSnEjUI2Ic2VamqhlZZpQLUtFKllH6oH5FaVdCX7kOLqmjVSpAidRUkWlSUfSBtQGIlsBrxJSUhH9hxQuzY8dfZB2M3gRBmnLglxj/J0thz59zz971zzz137igiItxHqH+0A783OcHZTk5wtpMTnO3kBGc7GRN89OhR3njjjVnLfPvtt3z//fcAfPbZZ/z444+ZcidFXsZrmIWzZ89SVVUFQFNT0+9SZ1qCu7q6+PjjjxERSktLKSwsxO12E41Gqa+vZ+fOndPKd3R08NVXXxEKhYhEIrzzzjsEg0FOnjzJ6dOncblcdHR0UFVVxa+//orL5eLll18GoLm5mY0bN7J582befvtt+vr6AGhsbKSmpsa485IGp0+flvXr18vo6Kh88skn8uWXX4qISCAQkGeeeUbcbrccOXJEmpubJRaLSX19vXi9XhEROXr0qDQ0NIiISHNzsxw5cmTa8YULF6S2tlZEREKhkDz55JPi9/tl3759cuLECRER8Xq9snXrVvF4PIZ9T7tLr169moKCAjo7OwkGg7S3twPg9/vp6elJlVNVlUOHDnHy5EmuXr3Kzz//jMlkuqPd8vJy4vE4ly5d4uLFi2zcuBFN0+js7KSnp4eDBw8CEI1GuXz5Mi6Xy5DfaQu22WwAxONxWltbefjhhwHwer04nc7UYBQIBNixYwd1dXVUVVVRVlZGW1vbrLZra2vp6Oigp6cndXvE43EOHz5MUVERAMPDwyxevNiw33MepTds2EBbWxsiwsjICHV1dVy6dCl1/urVq6iqSkNDA48//jgnTpwgFosBYDKZUsdTqa2t5fjx47jdbp544olp9SRtPv3004yNjRn2d86jdGNjIwcOHGDbtm1Eo1FeeeUVysvLcbvdAJSVlbFu3TpqamrIz89n06ZNdHd3IyJs2rSJ1tZWNE2bZnPZsmUUFxezZs2aVPd/6623aGlpYdu2bYgI77//vuHuDKDkVjyynJzgbCcnONvRJVhE8Pv9ZMOArisOBwIBKisrOXPmDA6HI/W75+I1/tP8L6KTkYw5qIcH/r6Wp1rqdZWd08RjfHCEWDiKY2khqzf/dS6m0kYxqaysKtNdfm6Ch3zYlSAPFC9i/YaSRJeXOIjATL1fmeE3IVGe5Ef5raAy5YKZbidFAZMJ08olun2ek+BgXz+VzsvYx/oJ/vviFJF3u9enKp/7uJBXUYntHw36ys6lothAH0gcJc+EUnBzXqvO1IwzobfcXayoKuqKv+guPyfBUe8wALJ0Bfbde6d4cYuYW79PPzlFu0KqxYXfjmf9cwTFYtXp8RwERycjqIFRAKzLilGL9N9HfySGBY+Pj+P3+/EP+Zi0Cj7Fhm1xojtHo1GuX79+2zWlpaUAeDweIpHpIaywsBCbzUYgEODGjRvTzlksFlwuF/F4nKGhodvsLl26FJPJRCQSwWw26/LfsOAzZ85w6tSpxJdVfwagYjTADuDGjRt88cUXt13T0tICwLFjx1KLcEnq6up45JFHOH/+PB0dHdPOPfTQQzz33HNEIpEZ7e7fvx9N0/B4PKk/9W7oyof9fn9q4pGcdV3p/B+m44cxW0ys+Od+llT8LTtb2Ol04nQ6GfT+F0coiNVio2jFqoSxvLxZ/+klS+58n2uadtvKRxJVVWe1q1cszCF5CA8NAILkO1Ft9nTN/O6kLTjmSYQk1bkITH/oAwxDpCVYRGDMCwjm4mKUWePsvUVagidGxrHEgigo2Jb/ab59yihpCfYP+cgnhGpSsZSUzLdPGSUtweMDHqxKGFOeimX5ivn2KaOkJTjY2w8IitkMrqXz7FJmSUtweOAaiBDP11DzF05IgjQFR6/fnPVoiyBPf9C/F0gvLI16QATzYteCCkmQhuBIcJK8kB+A/OX6Juz3EsbTwyEfViWEqqrk68xQ7iUMt7B/wIuVcCIG30waFhKGBQd6+1GIo5jzUIvvgxYO9/eBCGLVFlSWlMT4oDU8mDiwOxdUlpTEsOC4LxGS1IIiFHXhPYsz5HE8GkP1jwJgK11YSUMSQ4IDI+NYJYSiKNhWLM+UTxnFkGD/0AhWJlHzVKwPrMqQS5nFkOBg3zUU4qh5eZhK7oMWDvUnsqSYxY5qn3mF8V7HkOCo52aWZHcsuCwpia5AmlyrN9vihCSOdXERgYmJjDqWDpqm3TV70/XkYXBwkOrq6nlzLFPcuiVjJnQJjsfjDA8PIyJs2bKFU6dO3dWwUfx+P9XV1XOyraeFdXVpVVUpKSnB70/kwQ6HY94FJ8mkbcjt08p+DAm2WCw0NjZisVjm3ZFM2p5Kbr90tpMTnO3cd4J1L0qFw2Gamprw+Xxs376dXbt2GapoYGCA119/nUgkknrDpb29HYfDQXV1NS+99BKffvopXV1drF27lgMHDuDz+WhqaiIcDrNnzx62bNliVN/t6H2Frb29Xdra2iQWi8nu3bslFAoZegXuww8/lK6uLhERef755+XNN9+Ua9eupc739/fLa6+9JiIiH330kZw9e1YOHTokP/30k4RCIXnhhRcM1XcndHfp8+fPU1lZiaqqrFmzZtrLWHpoaGigsrISSMzNe3t7effdd3nxxRfp7e3lwoULPProowA89thjnDt3LlWn1WrF4XCk9WLWreju0oFAALs9sQ5ts9mYMJgeFhYWAvDNN9+wbt06SkpK2LVrF729vbS2tlJTUzPNfiAQIBAIpLYyJessKCgwVO+t6BZst9sJBoMABIPBtCb4x44d44cffuDzzz8nHA6jaRplZWX4fD40TUttPkvat9vtTExMoGkawWDwjvu4jKC7S1dUVPDLL78gIrjdbh588EFDFZ07d47vvvuOgwcPYjabqa+vJxKJcOXKFVwuF+Xl5XR3dwPQ3d1NRUUFFRUVdHd3Mzk5yejoKIsWLTKmbgZ0Ty0nJyfZu3cvg4ODbN++nWeffdZQRXv27KG/vz/VtXfu3MnXX39Nfn4+7733HitXrkyN0qtWreKDDz7A5/Oxb98+xsbGePXVV9m6dathgbeSm0tnOznB2U5OcLaTE5zt5ARnO/ed4P8DxHPBKjbGbeQAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 70x100 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAADwAAABZCAYAAABixUebAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8g+/7EAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAIiElEQVR4nO2ba2hU6RnHf+fMJXNLMhujSaxZb4u3dBfWxOqKbGwi7SpSbOgqIoq6t7DuElGwuxWpQcGoH/zg5UtBWkSogreiCFXsht2lSYyCbDUtxCrNJk6ukzEzmfs8/TDJaLobPXPTZpzfl8nMmfO8/3+ed973ec95jyIiwiuE+rIFvGiyhjOdrOFMJ2s408kaznSyhjMdfaoDNjc3c+jQIUSEkpISLBYLbW1tKIrCBx98wJo1azh//jw3btzA5XLhcDiorKxk9+7dKIrC5cuXOXnyJOFwmDfeeIP6+npsNlvqBEqKaWpqkrffflsGBwfl0KFDUl9fLyIi/f39UlVVJW1tbXLu3DlZunSp9Pb2it/vl3Xr1snVq1elvb1d1q9fLz6fT0RETpw4IQ0NDSnVl/IMA8yaNYv8/HyamprYv38/AAUFBVRXV9PS0oLNZqO6uprCwkIAVq1aRUtLC319fTx8+JC1a9cCEAqFKC0tTam2tBg2m80ARCIRFEWJfS4ihEIhAHQ6XezzSCSCqqqEw2FWrlzJnj17ABgeHiYQCKRUW1oHrSVLlnD27FkABgYGuH79OhUVFQB8/fXXDA0N4ff7uXLlCsuWLWPx4sVcu3aN3t5eAA4cOMCJEydSqiktGR5l27Zt7N27l9WrVxMOh/noo4946623aG9vp6CggI8//piBgQFWr17N8uXLAfj888/ZsmULIsLs2bP54osvUqpJEXnxVzzOnz9PS0sLDQ0NL7rpV28efikZfpm8chnOGs50NBkWEdxuN5nwc9c0D3s8HsrLy7l169aYQr6vvYsrv/0DIX8wbQK1ULpoLr/4/UZN302q8BhyDBAOhLBNsTPr3TeTCZUwik5l+s/maf5+coa7nYgI9tIpLNr8y2RCvTCSGrTcPYMAWApSuF5NM0ll2N3tBBGsOUKkzwGKAoo68qrAswa5p4+LjPw9+p4nf48f4MlLjgnVlq9Jc9K/4ZkmB8Xff4P3z/8Y+VRGVIy+joeM/YckMQPoF71LzjsrtH030UbCgRDqwCNKc/ox+0NIvy/RUEkhoRCRB/+CdBkeGhrC7Xbj7hkkbArgNFjIsRcypeIdQpEIfV7/03JAhGJr9IJAv9dPMJbIaPbzTUbMeh3eYIjH/mC0q48cNur1FJhNRCRCj9s7Eu/J6ZMtZtSgn8iMuZr1x2341q1bNDY2Rt/MtnMXO/MNOn5dNA2X28OfWr75wTm/W/MeAFcb/06n0zXm2K8WvslPfzKVf/77P/z1u/tjjs2cXMD6JQsJhsL88du//SBu3Xs/x5pjoNcfwqJRv6bVktvtjhUeo1XXg2/v4v7LGQotPqauqKZo1W8IhUL09vc/HR5QKCkuAqCvf4BgcKQHjDRrz8/HbLHg8fp4PPQ4msGRAcyg1zPpNTuRiNDT1/ck+yNMKSxEp9MRFDBatc0UcWc4NzeX3NxcOj3fURwYosAU4bWSqSgWGwZgap593HMnTxs/DzazFVvBpB89pj4nrlGb9FishBhyDGBSA+h0OtSS1xMN88JJ2HDQ8QgFQTEaUCdNSaWmtJKQYREh3NcNCLp8O6oxJ8Wy0kdChn0uD4bAEACGwslgiOdX9HJJyPBQtxMzPnQ6BUNREYrekGpdaSMhw+5uJ2bFh6pX0RcWpVpTWkncsOpHp1NRJ09Ntaa0kpBhT1c3eiWMqtehlkxLtaa0kpDh4KOuaDVksqBac1OtKa0kZDjS3x09Oc8+oUZoSKC0DPmDqG4nIBiLimACjdCQQIbdPYOY8KGoCsaSkjH3fycC8RsemZJ0ehVj0cQaoSERw45+TIofnV6HfmpqtyO8COI27O3oiK5ZDQaYYEUHJGA46OgCBDHlohpNaZCUXuI2LP09ACi5+RNuSoI4DYsI4uoDiU5Jijrxbj7Gpdg76MYUGQYFzKUTq6QcJS7D7tiyUCWndHq6NKWVuAx7Oh2oRBcNhtdnpktTWonLcOD7DhAhYjCj2vLSpSmtxGfY4QAgYsmbkCM0aFw8jF6rjwRdDIdDGHLz8Hi9aRWWCFar9bm1vaY7D6N7mv/f+d8tGT+GJsORSISenh5EhOXLl9PY2JjaTdtEb+dUVlYmFVtLhjV1aVVVKS4uxu12A2Cz2VJueJR0xobsPq3MJy7DRqORzz77DKMx9VNSOmM/TXY3baaTNZzpvHKGNV+IDwQC1NXV4XQ6qampiT1MpZVHjx6xa9cugsEgVVVVmEwmLly4gM1mo7Kykg8//JAjR47Q3NzM3Llzqa+vx+l0UldXRyAQoLa2NvbkS1JofYTtwoULcvr0aQmHw7Jly5bY43JaaWhokObmZhER2bRpk3z55ZfS1dUVO97Z2Snbt28XEZGDBw/K7du35dixY/LVV1+Jz+eTzZs3x9XeeGju0nfv3qW8vBxVVZkzZw73799//klP8cknn1BeXg5Ea/OOjg727dvH1q1b6ejo4N69eyxcuBCAiooK7ty5E2szJycHm82Gy+V6VhOa0NylPR4PFkt025HZbGZ4eDiuhux2OwBnzpxhwYIFFBcXs3btWjo6Ojh8+DBVVVVj4ns8HjweD1ardUyb+fnaNpGOh2bDFosF78ga2Ov1JlTgX7p0ievXr3P8+HECgQBWq5V58+bhdDqxWq10d3ePiW+xWBgeHsZqteL1emPmk0Fzly4rK+PmzZuICG1tbcycGd81rTt37nDx4kWOHj2KwWBg48aNBINBHjx4wKRJk5g/fz6tra0AtLa2UlZWRllZGa2trfj9fgYHB8nLS/6ykubS0u/3s2PHDhwOBzU1NWzYsCGuhmpra+ns7Ix17ffff59Tp05hMpnYv38/06dPj43SM2bM4MCBAzidTnbu3InL5eLTTz9lxQptO2afRbaWznSyhjOdrOFMJ2s408kaznReOcP/BcDfnY+BebyjAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 70x100 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 70x100 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAADwAAABZCAYAAABixUebAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8g+/7EAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAJcElEQVR4nO2bXWwU5RqAn5nZn3Zny/aP0tb2IBgpWIHYKicmhJYGezhGNHAUYxQDjQmEYNBqIF4RkQT/EhKrN9JwY7ggEZGLEy9UYr2gVFuSapALjylhKZRly7bb2Z/u33sutt1aKLLTbSts97nZbOf7eZ/vm37zzrcziogI8wj17w5grskJZzs54WwnJ5zt5ISznb9VuK2tjba2tjntc97NsMVsha6uLj788ENEBF3Xsdvt+P1+vF4vmzZt4s033+Srr77ixx9/xDAM3G43NTU1fPzxx9hsNtrb2zlx4gTFxcUsWLCAVatWAdDR0cGRI0eIx+NUV1dz8OBBSktLaWpq4umnn+bs2bPE43FaW1s5duwYfX197Nu3j2eeecacgJjk3Llz8thjj8nQ0JC0t7fLl19+KSIihmFIXV2dDA4OysmTJ2XdunXi9/slFovJ5s2b5fvvv5dffvlFmpubZWRkRILBoDz77LPyySefiNfrlbVr18rly5dFROTo0aPy+uuvi4jI+vXr5dixYyIisn//fnnppZckGo1KZ2enPPfcc2bDF9MzDLB06VJcLhctLS2cPXuW9vZ2fv/9dyKRCKFQCIC6ujoKCgoAePjhhxkeHqavr4/GxkacTicAzc3NJBIJfv31V1auXEl1dTUAL774Ip9//nmqv8bGRgAeeOABFi1ahMVioaqqCr/fbzr2aQnn5+cDcPjwYfr7+9m0aRNPPfUUnZ2djN9t2u32VHlFURCR1Geqc4uFSCRCPB5HUZTU30WEaDSa+m61WifVyYSMFq3Ozk5aWlrYuHEjly5dwuPxkEgk7lj+ySef5MyZM/j9fkZHR/n2228BWL16Nb29vbjdbgBOnDjBmjVrMgntjmQ0XDt37qS1tRVd16moqODRRx9NBT0VK1asYMeOHTz//PO4XC4qKysBKC0t5eDBg+zZs4dYLEZFRQWHDh3KJLQ7ouR2PLKcnHC2kxPOdtISFhEMwyAbFvS0rsOBQID6+np6enpSaSGA939X+e/+o8RGo39Re/apfqKG5gPb0iqbUeIxMnCTeDSGs6yQpetWZtLUtFE0lcVrlqddPiNhwzOEJARXVSlPbP9XJk3NGRktWobHB4CjeMGMBDMXZCg8DICjyHmXkvcOMzLDelnhTMQyJ0xbOBGLY3iGACiqLpupeGYd04vWyMgIhmEQGPQTssaRAo24rgEQi8W4cePGbXUqKioA8Hq9k27sAQoLC8nPzycQCNy2g2Gz2SgpKSGRSHD9+vXb2i0rK0PTNKLR6KRNgr/CtHBPTw8dHR3JL8sUQOFcz0/8Z0k1fr9/0tbMOAcOHADg9OnTXLlyZdKxzZs3s2rVKi5cuMA333wz6dhDDz3EK6+8QjQanbLdt99+G13X8Xq9qUG9G2ndDxuGkUo8xrOuyz9d5PzxMxT/o4wNuzdS5HQSjUbx3LwJkoCEgKKAAhWlpYDCoN9PNDa2nTO2pVPo1Mmz2zGCAfwjxqR+7VYrxS4Xoih4bvrG2lMAAYGFRUVoqkJM1bAvLE9L2PQMFxQUUFBQwLXARexBYWWsD/vJdoIigFAEgDKpTpjkmOqMBcuEsAAhQFOUsbrcUhcQwTXFscjYp/Wf62Hhv9OKf9qJh+Hx4bIEKYzdQIbzUNL8H5ppJBYj0X8p7fIZCA+hq2FUTUEpWoi1fm1y1lQ1NXu3hJaczvFPSYyVUyZO1fETIJEYKyepEwL4U7vKxPdwEHXpHKSWxoCPSi2MRVPRKqtRF5ZPyPzFzmUqUBgTEibkBBQ16aOoY4PH2ED9uTwT9RYUolhsacc9LeFIMEzYH8RpCaPZ8lHLq7DUrE7OGkzM0O2mt/9JJFlPBEkkUFR1TFYBVbul3PjgMHmgrLMsbHiGkHgMZ34UTdNRq5aiZLhBDlMOx4wzrUzL8AyRLyE0i4Jis6MuTO8aeC8wbWGHEkLTVNTCYlR73kzHNWtMW1jXxlbokjKwzQNhpxZOznDpIhRNu3ule4TpCQ/cxKmFUS0K6qLqmY5pVjEtLCKMejxYlDgWmxWtaslsxDVrmBYO+QzssREQwVJcjKrfP7sdMA1hw+NDJ4imqVhK768FC6YlPISDEKpFRSurnJGEYy4xLTxy3YeuJa/BSpr3oPcSpoUDA17y1QiqRcVyny1YMA3h2NUryYTdno9aOg9mWAYHAFBcxWCz36X0vYcp4XgkhsUYBBFsFZUoJm7L7hVMCRveYRyEUFQFe/U/ZiumWcWUcMDjQ1eS12Db4vtvwQKTwiH3FRQSKBYLlup5IBztvwwixOwFqI77K6UcJ600aXyv3rh6GWs8jth1jNEISuTv/eX/VnRdn/TM5lSk9cvDwMAADQ0NMxbYbHHrIxlTkZZwIpHA4/EgIjQ2NtLR0XHXhs1iGAYNDQ0ZtZ3ODKd1SquqSnl5OYaR/O3H6XTOuPA4s9k25J7Tyn5MCdtsNvbs2YPNNvMp5Wy2/Wdyz0tnOznhbGfeCae95RiJRNi7dy8+n48tW7awdetWUx1du3aNffv2EY1GaWpqIi8vj1OnTuF0OmloaOC1117jyJEjdHV1UVNTw7vvvovP52Pv3r1EIhF27dqVemErI9J9he3UqVNy/PhxicfjsmPHDgmHw6ZegXv//felq6tLREReffVVeeedd+Tq1aup4/39/fLGG2+IiMgHH3wg58+fl08//VR++OEHCYfDsn37dlP93Ym0T+kLFy5QX1+PqqosW7aMP/74w9TA7ty5k/r6eiCZm7vdbt577z1aWlpwu9389ttv1NXVAfD444/T29ub6tNut+N0OhkeHjbV51SkfUoHAgEcDgeQfBUvGAya6qiwsBBIvnX2yCOPUF5eztatW3G73Xz00Uc0NTVNaj8QCBAIBNB1fVKfLtdUDzClT9rCDocj9SJlKBSaVoJ/+vRpvvvuOz777DMikQi6rrN8+XJ8Ph+6rqceLxxv3+FwEAwG0XWdUCiUks+EtE/p2tpafv75Z0SEixcvsmSJuS2e3t5evv76a9ra2rBarWzbto1oNEpfXx8lJSWsWLGC7u5uALq7u6mtraW2tpbu7m5GR0cZGhpiwYLMn8tOO7UcHR2ltbWVgYEBtmzZwssvv2yqo127dtHf3586tV944QW++OIL8vLyOHToEIsXL06t0g8++CCHDx/G5/Px1ltvMTw8zO7du9mwYYNpwVvJ5dLZTk4428kJZzs54WwnJ5ztzDvh/wPANd9Uq0GKUAAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 70x100 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 70x100 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 70x100 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "for pe in pe_labels:\n",
    "    pestr = pe_labels[pe]\n",
    "    plt.figure(figsize=(0.7, 1))\n",
    "    #\n",
    "    df_ds_mod = df_ds.loc[(df_ds.pe==pestr) & \\\n",
    "                            (df_ds.condition=='average') & \\\n",
    "                            (df_ds.heads==head) & \\\n",
    "                            (df_ds.phase!='test')]\n",
    "    # plot_df = df_ds_mod.loc[(df_ds_mod['phase'] == 'train') & (df_ds_mod['condition'] != 'average')].reset_index()\n",
    "    ax = sns.lineplot(data=df_ds_mod,\n",
    "                x='epoch', y='accuracy', hue='phase', \n",
    "                legend=True, palette='rocket', alpha=0.75,\n",
    "                hue_order=['train', 'validation'])\n",
    "    plt.plot(df_ds_mod.epoch.unique(),np.repeat(0.827,len(df_ds_mod.epoch.unique())),\n",
    "                color='grey', linestyle='dashed',linewidth=1, markersize=2)\n",
    "    plt.title(pestr, fontsize=8, fontname='Arial')\n",
    "    plt.xlabel('', fontsize=8, fontname='Arial')\n",
    "    plt.ylabel('', fontsize=8, fontname='Arial')\n",
    "    plt.yticks([],fontsize=6)\n",
    "    plt.legend(loc=4,prop={'size': 5})\n",
    "    plt.xticks(fontsize=6)\n",
    "    plt.legend().remove()\n",
    "    plt.ylim([0.25, 1.0])\n",
    "    sns.despine()\n",
    "    plt.tight_layout()\n",
    "\n",
    "\n",
    "    outputdir = '../figures/manuscript_figures_v3/appendix/multihead_commonPEs/common_pes/'\n",
    "    if not os.path.exists(outputdir):\n",
    "        os.makedirs(outputdir)\n",
    "    plt.savefig(f'{outputdir}training_traj_{head}head_{pestr}.pdf',transparent=True,dpi=300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 86,
   "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>epoch</th>\n",
       "      <th>block</th>\n",
       "      <th>loss</th>\n",
       "      <th>accuracy</th>\n",
       "      <th>seed</th>\n",
       "      <th>layer</th>\n",
       "      <th>heads</th>\n",
       "      <th>wdecay</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>pe</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>1d-fixed</th>\n",
       "      <td>4000.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.001328</td>\n",
       "      <td>0.999658</td>\n",
       "      <td>4785.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2d-fixed</th>\n",
       "      <td>4000.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.000806</td>\n",
       "      <td>0.999748</td>\n",
       "      <td>4785.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>c-nope</th>\n",
       "      <td>4000.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.048244</td>\n",
       "      <td>0.980831</td>\n",
       "      <td>4785.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>learn-0.2</th>\n",
       "      <td>4000.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.000000</td>\n",
       "      <td>1.000000</td>\n",
       "      <td>4785.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>nope</th>\n",
       "      <td>4000.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.962999</td>\n",
       "      <td>0.510033</td>\n",
       "      <td>4785.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>random</th>\n",
       "      <td>4000.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.000000</td>\n",
       "      <td>1.000000</td>\n",
       "      <td>4785.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>relative</th>\n",
       "      <td>4000.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.000000</td>\n",
       "      <td>1.000000</td>\n",
       "      <td>4785.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>rope</th>\n",
       "      <td>4000.0</td>\n",
       "      <td>0.0</td>\n",
       "      <td>0.000000</td>\n",
       "      <td>1.000000</td>\n",
       "      <td>4785.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>0.0</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "            epoch  block      loss  accuracy    seed  layer  heads  wdecay\n",
       "pe                                                                        \n",
       "1d-fixed   4000.0    0.0  0.001328  0.999658  4785.0    4.0    4.0     0.0\n",
       "2d-fixed   4000.0    0.0  0.000806  0.999748  4785.0    4.0    4.0     0.0\n",
       "c-nope     4000.0    0.0  0.048244  0.980831  4785.0    4.0    4.0     0.0\n",
       "learn-0.2  4000.0    0.0  0.000000  1.000000  4785.0    4.0    4.0     0.0\n",
       "nope       4000.0    0.0  0.962999  0.510033  4785.0    4.0    4.0     0.0\n",
       "random     4000.0    0.0  0.000000  1.000000  4785.0    4.0    4.0     0.0\n",
       "relative   4000.0    0.0  0.000000  1.000000  4785.0    4.0    4.0     0.0\n",
       "rope       4000.0    0.0  0.000000  1.000000  4785.0    4.0    4.0     0.0"
      ]
     },
     "execution_count": 86,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "tmpdf = df_ds.loc[(df_ds.epoch==4000) & (df_ds.condition=='average') & (df_ds.phase=='train')]\n",
    "tmpdf.groupby('pe').mean(numeric_only=True)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Plot validation performance across positional encodings"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 87,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 210x155 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "i = 1\n",
    "tmpdf = df_ds.loc[(df_ds.epoch==df_ds.epoch.max()) & \\\n",
    "                (df_ds.phase=='validation') & \\\n",
    "                (df_ds.wdecay==0.0) & \\\n",
    "                (df_ds.heads==head) & \\\n",
    "                (df_ds.condition=='average') & \\\n",
    "                (df_ds.pe!='learn0')]\n",
    "tmpdf2 = tmpdf.groupby('pe').mean(numeric_only=True).reset_index()\n",
    "df_validation_order = tmpdf2.sort_values('accuracy',ascending=False)\n",
    "plt.figure(figsize=(2.1,1.55))\n",
    "ax = sns.boxplot(data=tmpdf,x=\"pe\",y=\"accuracy\",hue=\"pe\",fliersize=1,palette='rocket',\n",
    "                    order=df_validation_order.pe.values)\n",
    "plt.plot(range(len(tmpdf.pe.unique())),np.repeat(0.827,len(tmpdf.pe.unique())),\n",
    "            color='grey', linestyle='dashed',linewidth=1, markersize=2)\n",
    "plt.xticks(rotation=-45,fontsize=7)\n",
    "plt.xlabel(\"\",fontsize=10)\n",
    "plt.title(f\"Multihead attention ({head} heads),\\nmean sorted performance\",fontsize=8)\n",
    "plt.ylabel('Accuracy', fontsize=8, fontname='Arial')\n",
    "plt.yticks(fontsize=7)\n",
    "sns.despine()\n",
    "plt.tight_layout()\n",
    "i += 1\n",
    "outputdir = '../figures/manuscript_figures_v3/appendix/multihead_commonPEs/'\n",
    "if not os.path.exists(outputdir):\n",
    "    os.makedirs(outputdir)\n",
    "plt.savefig(f'{outputdir}validation_performance_all_{head}head.pdf',transparent=True,dpi=300)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### paired t-tests between 2d-fixed X learnable and learnable X relative"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 77,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "2d-fixed vs. learn-0.2 | t = 4.778663537001958 | p = 0.000294133305821761\n",
      "learn-0.2 vs. relative | t = -0.13250859756292568 | p = 0.8964676854824805\n"
     ]
    }
   ],
   "source": [
    "t, p = stats.ttest_rel(tmpdf.loc[tmpdf.pe=='2d-fixed'].accuracy.values,tmpdf.loc[tmpdf.pe=='learn-0.2'].accuracy.values)\n",
    "print('2d-fixed vs. learn-0.2 | t =', t, '| p =', p)\n",
    "t, p = stats.ttest_rel(tmpdf.loc[tmpdf.pe=='learn-0.2'].accuracy.values,tmpdf.loc[tmpdf.pe=='relative'].accuracy.values)\n",
    "print('learn-0.2 vs. relative | t =', t, '| p =', p)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### convert to latex table"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 78,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\\begin{tabular}{lrrrr}\n",
      "\\toprule\n",
      "PE (2 heads) & Validation acc & Validation SD & Training acc & Training SD \\\\\n",
      "\\midrule\n",
      "2d-fixed & 0.988 & 0.021 & 1.000 & 0.000 \\\\\n",
      "relative & 0.928 & 0.025 & 1.000 & 0.000 \\\\\n",
      "learn-0.2 & 0.926 & 0.038 & 1.000 & 0.000 \\\\\n",
      "rope & 0.872 & 0.077 & 1.000 & 0.000 \\\\\n",
      "random & 0.766 & 0.078 & 1.000 & 0.000 \\\\\n",
      "1d-fixed & 0.758 & 0.086 & 1.000 & 0.000 \\\\\n",
      "c-nope & 0.360 & 0.045 & 0.917 & 0.149 \\\\\n",
      "nope & 0.331 & 0.020 & 0.510 & 0.002 \\\\\n",
      "\\bottomrule\n",
      "\\end{tabular}\n",
      "\n"
     ]
    }
   ],
   "source": [
    "tmpdf = df_ds.loc[(df_ds.epoch==df_ds.epoch.max()) & \\\n",
    "                (df_ds.phase=='validation') & \\\n",
    "                (df_ds.wdecay==0.0) & \\\n",
    "                (df_ds.heads==head) & \\\n",
    "                (df_ds.condition=='average')]\n",
    "\n",
    "# Get mean and SD\n",
    "meandf = tmpdf.groupby(['pe','phase']).mean(numeric_only=True).reset_index()\n",
    "sddf = tmpdf.groupby(['pe','phase']).std(numeric_only=True).reset_index()\n",
    "\n",
    "df_validation_order = meandf.sort_values('accuracy',ascending=False)\n",
    "\n",
    "df_to_latex = pd.DataFrame()\n",
    "df_to_latex[f'PE ({head} heads)'] = df_validation_order.pe.unique()\n",
    "# df_to_latex['Phase'] = meandf.phase.values\n",
    "df_to_latex['Validation acc'] = df_validation_order.accuracy.values\n",
    "df_to_latex['Validation SD'] = sddf.iloc[df_validation_order.index].accuracy.values\n",
    "\n",
    "\n",
    "# training data\n",
    "tmpdf = df_ds.loc[(df_ds.epoch==df_ds.epoch.max()) & \\\n",
    "                (df_ds.phase=='train') & \\\n",
    "                (df_ds.wdecay==0.0) & \\\n",
    "                (df_ds.heads==head) & \\\n",
    "                (df_ds.condition=='average')]\n",
    "\n",
    "# Get mean and SD\n",
    "meandf = tmpdf.groupby(['pe','phase']).mean(numeric_only=True).reset_index()\n",
    "sddf = tmpdf.groupby(['pe','phase']).std(numeric_only=True).reset_index()\n",
    "\n",
    "\n",
    "df_to_latex['Training acc'] = meandf.iloc[df_validation_order.index].accuracy.values\n",
    "df_to_latex['Training SD'] = sddf.iloc[df_validation_order.index].accuracy.values\n",
    "# df_to_latex = df_to_latex.sort_values('Accuracy',ascending=False)\n",
    "\n",
    "# df_to_latex = df_to_latex.loc[df_to_latex.PE!='2d-fixed']\n",
    "\n",
    "print(df_to_latex.to_latex(index=False,header=True,float_format=\"{:.3f}\".format))\n",
    "# print(df_to_latex.T.to_latex(index=True,header=False,float_format=\"{:.2f}\".format))\n",
    "# df_latex = pd.concat([meandf[['PE','Similarity']],sddf['PE','Similarity']])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Compare attention maps"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "def retrieve_attn_weights(model,batch):\n",
    "    \n",
    "    attn_weights = []\n",
    "    embedding = model.w_embed(batch)\n",
    "\n",
    "    for block in model.blocks:\n",
    "        # transformer block\n",
    "        if block.positional_encoding in ['relative', 'rope2','cnope']:\n",
    "            attn_outputs, attn_weights_layer = block.selfattention.forward_attn(embedding)\n",
    "        elif block.positional_encoding in ['nope']:\n",
    "            attn_outputs, attn_weights_layer = block.selfattention(embedding, embedding, embedding, need_weights=True)\n",
    "        elif block.positional_encoding in ['rope']:\n",
    "            raise NotImplementedError()\n",
    "            attn_outputs = block.selfattention(query=embedding,key=embedding,value=embedding)\n",
    "        elif block.positional_encoding in ['scnope']:\n",
    "            raise NotImplementedError()\n",
    "            #attn_outputs = block.selfattention(embedding,embedding,embedding)\n",
    "            attn_outputs, l1_reg = block.selfattention(embedding)\n",
    "            #l1_reg = torch.mean(torch.abs(att))\n",
    "        elif block.positional_encoding in ['clearn']:\n",
    "            raise NotImplementedError()\n",
    "            embedding = block.pe(embedding) # positional encoding\n",
    "            embedding = block.dropout_embed(embedding)\n",
    "            attn_outputs, att = block.selfattention(embedding)\n",
    "        else:\n",
    "            embedding = block.pe(embedding) # positional encoding\n",
    "            embedding = block.dropout_embed(embedding)\n",
    "            attn_outputs, attn_weights_layer = block.selfattention(embedding, embedding, embedding, need_weights=True)\n",
    "        #attn_outputs = block.layernorm0(attn_outputs)\n",
    "        attn_outputs = block.layernorm0(attn_outputs+embedding) # w resid connection\n",
    "        transformer_out = block.mlp(attn_outputs)\n",
    "        #transformer_out = block.layernorm1(transformer_out)\n",
    "        embedding = block.layernorm1(transformer_out+attn_outputs) # w resid connection\n",
    "        attn_weights.append(attn_weights_layer)\n",
    "    \n",
    "    return attn_weights\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Load 2d fixed encoding"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Load 2d absolute attention\n",
    "validation_file = '../data/nn/puzzle_data_original.csv'\n",
    "dataloader = torch.utils.data.DataLoader(\n",
    "    get_dataset(validation_file), batch_size=108, shuffle=False)\n",
    "\n",
    "dropout = 0.0\n",
    "epoch = 4000\n",
    "layer = 4\n",
    "attnhead = 1\n",
    "wdecay = 0.0\n",
    "# for epoch in np.arange(0, last_epoch+1, checkpoint_freq):\n",
    "pe = 'pe-2dpe_'\n",
    "petype = 'absolute2d'\n",
    "pestr = '2dpe'\n",
    "resultdir = f\"../results/\"\n",
    "modelname = f\"model-{model_label}_\" \\\n",
    "            f\"{pe}\" \\\n",
    "            f\"nl-{layer}_\" \\\n",
    "            f\"do-{dropout}_\" \\\n",
    "            f\"wd-{wdecay}_\" \\\n",
    "            f\"at-{attnhead}_\" \\\n",
    "            f\"hs-{hidden_size}_\" \\\n",
    "            f\"curr-{curriculum}_\" \\\n",
    "            f\"lr-{learning_rate}_\" \\\n",
    "            f\"co-{training_acc_cutoff}_\" \\\n",
    "            f\"col-{cutoff_length}/\"\n",
    "model_str = pestr + '-' + str(attnhead) + 'H'\n",
    "# _df = df_good_models.loc[(df_good_models.epoch==4000) & (df_good_models.pe==pestr) & (df_good_models.heads==attnhead)]\n",
    "# if len(_df)<1: continue\n",
    "attn_weights_all = []\n",
    "for seed in seeds:\n",
    "    try: \n",
    "        checkpoint = f\"s-{seed}_\" \\\n",
    "                    f\"e-{epoch}\" \n",
    "        torch.manual_seed(seed)\n",
    "        model = transformer_main.Transformer(\n",
    "                    nblocks=layer,\n",
    "                    nhead=attnhead,\n",
    "                    dropout=dropout,\n",
    "                    embedding_dim=hidden_size,\n",
    "                    positional_encoding=petype)\n",
    "        model = model.to(device=torch.device('mps'))\n",
    "        model.load_state_dict(torch.load(resultdir + modelname + checkpoint +'.pt',map_location=torch.device('mps') ))\n",
    "    except:\n",
    "        continue\n",
    "\n",
    "    with torch.no_grad():\n",
    "        for i, batch in enumerate(dataloader):\n",
    "\n",
    "            # get features\n",
    "            test_features, test_labels, index = batch[0], batch[1], batch[2]\n",
    "\n",
    "            # flatten to accommodate transformer\n",
    "            test_features = torch.flatten(test_features,start_dim=1,end_dim=2)\n",
    "            test_features = test_features.to(device)\n",
    "\n",
    "            attn_weights_all.append(torch.stack(retrieve_attn_weights(model,test_features)))\n",
    "    \n",
    "attn_weight_avg2d = torch.mean(torch.stack(attn_weights_all),dim=0)\n",
    "attn_weight_avg2d = attn_weight_avg2d.reshape(layer,108,-1)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load attention maps of other models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Compare attention weights for each model x layer to 2d absolute model\n",
    "validation_file = '../data/nn/puzzle_data_original.csv'\n",
    "dataloader = torch.utils.data.DataLoader(\n",
    "    get_dataset(validation_file), batch_size=108, shuffle=False)\n",
    "\n",
    "dropout = 0.0\n",
    "epoch = 4000\n",
    "layer = 4\n",
    "attnhead = 1\n",
    "wdecay = 0.0\n",
    "# for epoch in np.arange(0, last_epoch+1, checkpoint_freq):\n",
    "attn_weight_avg = {}\n",
    "for pe in positional_encodings:\n",
    "    petype = positional_encodings[pe]\n",
    "    pestr = pe_labels[pe]\n",
    "    if pe_labels[pe] == 'learn-0.2':\n",
    "        pe_init = 0.2\n",
    "    else:\n",
    "        pe_init = 1.0\n",
    "    resultdir = f\"../results/\"\n",
    "    modelname = f\"model-{model_label}_\" \\\n",
    "                f\"{pe}\" \\\n",
    "                f\"nl-{layer}_\" \\\n",
    "                f\"do-{dropout}_\" \\\n",
    "                f\"wd-{wdecay}_\" \\\n",
    "                f\"at-{attnhead}_\" \\\n",
    "                f\"hs-{hidden_size}_\" \\\n",
    "                f\"curr-{curriculum}_\" \\\n",
    "                f\"lr-{learning_rate}_\" \\\n",
    "                f\"co-{training_acc_cutoff}_\" \\\n",
    "                f\"col-{cutoff_length}/\"\n",
    "    # _df = df_good_models.loc[(df_good_models.epoch==4000) & (df_good_models.pe==pestr) & (df_good_models.heads==attnhead)]\n",
    "    # if len(_df)<1: continue\n",
    "    attn_weights_all = []\n",
    "    for seed in seeds:\n",
    "        \n",
    "        try: \n",
    "            checkpoint = f\"s-{seed}_\" \\\n",
    "                        f\"e-{epoch}\" \n",
    "            torch.manual_seed(seed)\n",
    "            model = transformer_main.Transformer(\n",
    "                        nblocks=layer,\n",
    "                        nhead=attnhead,\n",
    "                        dropout=dropout,\n",
    "                        embedding_dim=hidden_size,\n",
    "                        positional_encoding=petype,\n",
    "                        pe_init=pe_init)\n",
    "            model = model.to(device=torch.device('mps'))\n",
    "            model.load_state_dict(torch.load(resultdir + modelname + checkpoint +'.pt',map_location=torch.device('mps') ))\n",
    "        except:\n",
    "            continue\n",
    "\n",
    "        with torch.no_grad():\n",
    "            for i, batch in enumerate(dataloader):\n",
    "\n",
    "                # get features\n",
    "                test_features, test_labels, index = batch[0], batch[1], batch[2]\n",
    "\n",
    "                # flatten to accommodate transformer\n",
    "                test_features = torch.flatten(test_features,start_dim=1,end_dim=2)\n",
    "                test_features = test_features.to(device)\n",
    "\n",
    "                attn_weights_all.append(torch.stack(retrieve_attn_weights(model,test_features)))\n",
    "\n",
    "    attn_weight_avg[pestr] = torch.mean(torch.stack(attn_weights_all),dim=0)\n",
    "    attn_weight_avg[pestr] = attn_weight_avg[pestr].reshape(layer,108,-1)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Calculate similarity of attention weights with 2d-fixed maps"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/p5/xfmq1vxd4sx5zq766ht140jh0000gn/T/ipykernel_43844/349795300.py:4: UserWarning: Implicit dimension choice for softmax has been deprecated. Change the call to include dim=X as an argument.\n",
      "  net_1_probs =  torch.nn.functional.softmax(net_1_logits)\n",
      "/var/folders/p5/xfmq1vxd4sx5zq766ht140jh0000gn/T/ipykernel_43844/349795300.py:5: UserWarning: Implicit dimension choice for softmax has been deprecated. Change the call to include dim=X as an argument.\n",
      "  net_2_probs=  torch.nn.functional.softmax(net_2_logits)\n",
      "/var/folders/p5/xfmq1vxd4sx5zq766ht140jh0000gn/T/ipykernel_43844/349795300.py:9: UserWarning: Implicit dimension choice for log_softmax has been deprecated. Change the call to include dim=X as an argument.\n",
      "  loss += torch.nn.functional.kl_div(torch.nn.functional.log_softmax(net_1_logits), m)\n",
      "/Users/tito/mambaforge/envs/lstnn/lib/python3.9/site-packages/torch/nn/functional.py:2949: UserWarning: reduction: 'mean' divides the total loss by both the batch size and the support size.'batchmean' divides only by the batch size, and aligns with the KL div math definition.'mean' will be changed to behave the same as 'batchmean' in the next major release.\n",
      "  warnings.warn(\n",
      "/var/folders/p5/xfmq1vxd4sx5zq766ht140jh0000gn/T/ipykernel_43844/349795300.py:10: UserWarning: Implicit dimension choice for log_softmax has been deprecated. Change the call to include dim=X as an argument.\n",
      "  loss += torch.nn.functional.kl_div(torch.nn.functional.log_softmax(net_2_logits), m)\n"
     ]
    }
   ],
   "source": [
    "def jsd(net_1_logits, net_2_logits):\n",
    "    \"\"\"Jensen Shannon Divergence\"\"\"\n",
    "\n",
    "    net_1_probs =  torch.nn.functional.softmax(net_1_logits)\n",
    "    net_2_probs=  torch.nn.functional.softmax(net_2_logits)\n",
    "\n",
    "    m = 0.5 * (net_1_probs + net_2_probs)\n",
    "    loss = 0.0\n",
    "    loss += torch.nn.functional.kl_div(torch.nn.functional.log_softmax(net_1_logits), m) \n",
    "    loss += torch.nn.functional.kl_div(torch.nn.functional.log_softmax(net_2_logits), m) \n",
    "    \n",
    "    return (0.5 * loss)\n",
    "\n",
    "corr_vals = {}\n",
    "corr_vals['PE'] = []\n",
    "corr_vals['Similarity'] = []\n",
    "corr_vals['Layer'] = []\n",
    "corr_vals['Puzzle'] = []\n",
    "corr_vals['JSD'] = []\n",
    "for pestr in attn_weight_avg:\n",
    "    for l in range(layer):\n",
    "        for p in range(108):\n",
    "            # r, p = stats.pearsonr(attn_weight_avg[pestr][l,p].cpu(),attn_weight_avg2d[l,p].cpu())\n",
    "            # r, p = stats.spearmanr(attn_weight_avg[pestr][l,p].cpu(),attn_weight_avg2d[l,p].cpu())\n",
    "            # r = torch.norm(attn_weight_avg[pestr][l,p].cpu() - attn_weight_avg2d[l,p].cpu()).item()\n",
    "            r = torch.nn.functional.cosine_similarity(attn_weight_avg[pestr][l,p], attn_weight_avg2d[l,p], dim=0).cpu().item()\n",
    "            j = jsd(attn_weight_avg[pestr][l,p], attn_weight_avg2d[l,p]).cpu().item()\n",
    "            corr_vals['PE'].append(pestr)\n",
    "            corr_vals['Similarity'].append(r)\n",
    "            corr_vals['JSD'].append(j)\n",
    "            corr_vals['Layer'].append(l+1)\n",
    "            corr_vals['Puzzle'].append(p)\n",
    "corr_vals = pd.DataFrame(corr_vals)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Plot attention similarity across layers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "# for l in range(layer):\n",
    "#     tmpdf = corr_vals.loc[corr_vals.Layer==l+1]\n",
    "#     plt.figure(figsize=(2,1.75))\n",
    "#     # ax = sns.boxplot(data=tmpdf,x=\"PE\",y=\"Correlation\",hue=\"PE\",fliersize=1,palette='rocket',\n",
    "#                         # order=['1d-abs','2d-abs','rel','rope2','learn0','learn','random','c-nope','nope'])\n",
    "#     ax = sns.barplot(data=tmpdf,x=\"PE\",y=\"Similarity\",hue=\"PE\",palette='rocket',\n",
    "#                         )\n",
    "#     plt.xticks(rotation=-45,fontsize=7)\n",
    "#     plt.xlabel(\"Pos. enc.\",fontsize=10)\n",
    "#     plt.title(f\"Similarity with 2d PE, Layer{l}\",fontsize=10)\n",
    "#     plt.ylabel('Similarity', fontsize=10, fontname='Arial')\n",
    "#     plt.yticks(fontsize=6)\n",
    "#     sns.despine()\n",
    "#     plt.tight_layout()\n",
    "#     i += 1i\n",
    "#     outputdir = '../figures/training_traj/'\n",
    "#     if not os.path.exists(outputdir):\n",
    "#         os.makedirs(outputdir)\n",
    "#     # plt.savefig(f'{outputdir}validation_performance_all_{head}head.pdf',transparent=True,dpi=300)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Take the max correlation attention map across layers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 200x150 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 200x175 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "tmpdf = corr_vals.groupby(['PE','Puzzle']).mean(numeric_only=True).reset_index()\n",
    "orderdf = tmpdf.groupby('PE').mean(numeric_only=True).reset_index()\n",
    "order = orderdf.sort_values('Similarity',ascending=False)\n",
    "plt.figure(figsize=(2.,1.5))\n",
    "ax = sns.boxplot(data=tmpdf,x=\"PE\",y=\"Similarity\",hue=\"PE\",fliersize=1,palette='rocket',order=order.PE.values)\n",
    "# ax = sns.barplot(data=tmpdf,x=\"PE\",y=\"Correlation\",hue=\"PE\",palette='rocket',\n",
    "#                     order=['1d-abs','2d-abs','rel','rope2','learn0','learn','random','c-nope','nope'])\n",
    "plt.xticks(rotation=-45,fontsize=7)\n",
    "plt.xlabel(\"\",fontsize=8)\n",
    "plt.title(f\"Similarity to 2d-fixed model\",fontsize=8)\n",
    "plt.ylabel('Cosine', fontsize=8, fontname='Arial')\n",
    "plt.yticks(fontsize=7)\n",
    "sns.despine()\n",
    "plt.tight_layout()\n",
    "i += 1\n",
    "outputdir = '../figures/manuscript_figures/validation_regularPEs/'\n",
    "if not os.path.exists(outputdir):\n",
    "    os.makedirs(outputdir)\n",
    "# plt.savefig(f'{outputdir}attn_similarity_{head}head.pdf',transparent=True,dpi=300)\n",
    "\n",
    "\n",
    "tmpdf = corr_vals.groupby(['PE','Puzzle']).mean(numeric_only=True).reset_index()\n",
    "orderdf = tmpdf.groupby('PE').mean(numeric_only=True).reset_index()\n",
    "order = orderdf.sort_values('JSD',ascending=False)\n",
    "plt.figure(figsize=(2.,1.75))\n",
    "ax = sns.boxplot(data=tmpdf,x=\"PE\",y=\"JSD\",hue=\"PE\",fliersize=1,palette='rocket',order=order.PE.values)\n",
    "# ax = sns.barplot(data=tmpdf,x=\"PE\",y=\"Correlation\",hue=\"PE\",palette='rocket',\n",
    "#                     order=['1d-abs','2d-abs','rel','rope2','learn0','learn','random','c-nope','nope'])\n",
    "plt.xticks(rotation=-45,fontsize=7)\n",
    "plt.xlabel(\"\",fontsize=8)\n",
    "plt.title(f\"Similarity to 2d-fixed model\",fontsize=8)\n",
    "plt.ylabel('JSD', fontsize=8, fontname='Arial')\n",
    "plt.yticks(fontsize=7)\n",
    "sns.despine()\n",
    "plt.tight_layout()\n",
    "i += 1\n",
    "outputdir = '../figures/manuscript_figures/appendix/regularPE_attnJSD/'\n",
    "if not os.path.exists(outputdir):\n",
    "    os.makedirs(outputdir)\n",
    "# plt.savefig(f'{outputdir}attn_similarity_{head}head_JSD.pdf',transparent=True,dpi=300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "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>PE</th>\n",
       "      <th>Puzzle</th>\n",
       "      <th>Similarity</th>\n",
       "      <th>Layer</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>0</th>\n",
       "      <td>1d-fixed</td>\n",
       "      <td>0</td>\n",
       "      <td>0.858683</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>1d-fixed</td>\n",
       "      <td>1</td>\n",
       "      <td>0.881694</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>1d-fixed</td>\n",
       "      <td>2</td>\n",
       "      <td>0.862001</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>1d-fixed</td>\n",
       "      <td>3</td>\n",
       "      <td>0.876732</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>1d-fixed</td>\n",
       "      <td>4</td>\n",
       "      <td>0.854091</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>859</th>\n",
       "      <td>rope</td>\n",
       "      <td>103</td>\n",
       "      <td>0.814432</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>860</th>\n",
       "      <td>rope</td>\n",
       "      <td>104</td>\n",
       "      <td>0.800873</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>861</th>\n",
       "      <td>rope</td>\n",
       "      <td>105</td>\n",
       "      <td>0.800141</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>862</th>\n",
       "      <td>rope</td>\n",
       "      <td>106</td>\n",
       "      <td>0.801951</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>863</th>\n",
       "      <td>rope</td>\n",
       "      <td>107</td>\n",
       "      <td>0.821683</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>864 rows × 4 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "           PE  Puzzle  Similarity  Layer\n",
       "0    1d-fixed       0    0.858683    2.5\n",
       "1    1d-fixed       1    0.881694    2.5\n",
       "2    1d-fixed       2    0.862001    2.5\n",
       "3    1d-fixed       3    0.876732    2.5\n",
       "4    1d-fixed       4    0.854091    2.5\n",
       "..        ...     ...         ...    ...\n",
       "859      rope     103    0.814432    2.5\n",
       "860      rope     104    0.800873    2.5\n",
       "861      rope     105    0.800141    2.5\n",
       "862      rope     106    0.801951    2.5\n",
       "863      rope     107    0.821683    2.5\n",
       "\n",
       "[864 rows x 4 columns]"
      ]
     },
     "execution_count": 20,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "tmpdf.reset_index()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Print stats to latex"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\\begin{tabular}{lrr}\n",
      "\\toprule\n",
      "PE & Cosine & SD \\\\\n",
      "\\midrule\n",
      "2d-fixed & 1.000 & 0.000 \\\\\n",
      "1d-fixed & 0.855 & 0.015 \\\\\n",
      "learn-0.2 & 0.838 & 0.016 \\\\\n",
      "rope & 0.804 & 0.018 \\\\\n",
      "random & 0.687 & 0.015 \\\\\n",
      "nope & 0.601 & 0.046 \\\\\n",
      "relative & 0.516 & 0.090 \\\\\n",
      "c-nope & 0.431 & 0.083 \\\\\n",
      "\\bottomrule\n",
      "\\end{tabular}\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# Get mean and SD\n",
    "meandf = tmpdf.groupby('PE').mean(numeric_only=True).reset_index()\n",
    "sddf = tmpdf.groupby('PE').std(numeric_only=True).reset_index()\n",
    "\n",
    "df_to_latex = pd.DataFrame()\n",
    "df_to_latex['PE'] = meandf.PE.values\n",
    "df_to_latex['Cosine'] = meandf.Similarity.values\n",
    "df_to_latex['SD'] = sddf.Similarity.values\n",
    "# df_to_latex['JSD'] = meandf.JSD.values\n",
    "# df_to_latex['JSD-SD'] = sddf.JSD.values\n",
    "\n",
    "df_to_latex = df_to_latex.sort_values('Cosine',ascending=False)\n",
    "# df_to_latex = df_to_latex.loc[df_to_latex.PE!='2d-fixed']\n",
    "\n",
    "print(df_to_latex.to_latex(index=False,float_format=\"{:.3f}\".format))\n",
    "# df_latex = pd.concat([meandf[['PE','Similarity']],sddf['PE','Similarity']])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Compare baseline similarity of PEs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 77,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "model1d = transformer_main.Transformer(\n",
    "            nblocks=layer,\n",
    "            nhead=attnhead,\n",
    "            dropout=dropout,\n",
    "            embedding_dim=hidden_size,\n",
    "            positional_encoding='absolute',\n",
    "            pe_init=1)\n",
    "model2d = transformer_main.Transformer(\n",
    "            nblocks=layer,\n",
    "            nhead=attnhead,\n",
    "            dropout=dropout,\n",
    "            embedding_dim=hidden_size,\n",
    "            positional_encoding='absolute2d',\n",
    "            pe_init=1)\n",
    "model_learn02 = transformer_main.Transformer(\n",
    "            nblocks=layer,\n",
    "            nhead=attnhead,\n",
    "            dropout=dropout,\n",
    "            embedding_dim=hidden_size,\n",
    "            positional_encoding='learn',\n",
    "            pe_init=1)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 111,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.7245560884475708"
      ]
     },
     "execution_count": 111,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.nn.functional.cosine_similarity(model1d.blocks[0].pe.pe.reshape(-1),model2d.blocks[0].pe.pe.reshape(-1),dim=0).cpu().item()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 110,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Similarity of learned PE with 2d-fixed PE: 0.0002961885261659821\n"
     ]
    }
   ],
   "source": [
    "dropout = 0.0\n",
    "epoch = 4000\n",
    "layer = 4\n",
    "attnhead = 1\n",
    "wdecay = 0.0\n",
    "resultdir = f\"../results/\"\n",
    "modelname = f\"model-{model_label}_\" \\\n",
    "        f\"pe-learn-0.2_\" \\\n",
    "        f\"nl-{layer}_\" \\\n",
    "        f\"do-{dropout}_\" \\\n",
    "        f\"wd-{wdecay}_\" \\\n",
    "        f\"at-{attnhead}_\" \\\n",
    "        f\"hs-{hidden_size}_\" \\\n",
    "        f\"curr-{curriculum}_\" \\\n",
    "        f\"lr-{learning_rate}_\" \\\n",
    "        f\"co-{training_acc_cutoff}_\" \\\n",
    "        f\"col-{cutoff_length}/\"\n",
    "# _df = df_good_models.loc[(df_good_models.epoch==4000) & (df_good_models.pe==pestr) & (df_good_models.heads==attnhead)]\n",
    "# if len(_df)<1: continue\n",
    "attn_weights_all = []\n",
    "learned_similarities = []\n",
    "for seed in seeds:\n",
    "\n",
    "    checkpoint = f\"s-{seed}_\" \\\n",
    "                f\"e-{epoch}\" \n",
    "    torch.manual_seed(seed)\n",
    "    model = transformer_main.Transformer(\n",
    "                nblocks=layer,\n",
    "                nhead=attnhead,\n",
    "                dropout=dropout,\n",
    "                embedding_dim=hidden_size,\n",
    "                positional_encoding='learn',\n",
    "                pe_init=0.2)\n",
    "    # model = model.to(device=torch.device('mps'))\n",
    "    model.load_state_dict(torch.load(resultdir + modelname + checkpoint +'.pt',map_location=torch.device('cpu') ))\n",
    "    for block in model.blocks:\n",
    "        sim = torch.nn.functional.cosine_similarity(block.pe.positional_encoding.reshape(-1),model2d.blocks[0].pe.pe.reshape(-1),dim=0).cpu().item()\n",
    "        learned_similarities.append(sim)\n",
    "print('Similarity of learned PE with 2d-fixed PE:', np.mean(learned_similarities))"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "lstnn",
   "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.18"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
