{
 "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": [
    "# Some summary variables for all models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "model_label = 'Transformer'\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]\n",
    "attnheads = [1]\n",
    "wdecays = [0.0]\n",
    "dropout = 0.0\n",
    "hidden_size = 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 = 200\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",
    "}"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Code/func to retrieve attention attention maps across layers (called blocks)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "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 model\n",
    "##### Luke: This is the primary code you need to load the torch model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "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 # final training step\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": 6,
   "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": "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.18"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
