{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "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": 3,
   "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",
    "curriculum = 'All'\n",
    "seeds = [2235, 6312, 6068, 9742, 8880, 2197, 669, 6256, 3309, 2541, 8643, 7785, 195, 6914, 29]\n",
    "# seeds = [2235, 6312, 6068, 9742, 8880, 2197, 669, 6256, 3309] #, 2541, 8643, 7785, 195, 6914, 29]\n",
    "# seeds = [2235, 6312, 6068, 9742]\n",
    "#seeds = [6914, 29]\n",
    "device = 'mps'\n",
    "\n",
    "# model params\n",
    "nblocks = [4] #, 5]\n",
    "attnheads = [1]# 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 = 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",
    "}\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": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 350x125 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure(figsize=(3.5, 1.25))\n",
    "head = 1\n",
    "pestr = '1d-fixed'\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",
    "plt.subplot(1,4,1)\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(f\"{pestr}\", fontsize=8, fontname='Arial')\n",
    "plt.xlabel('Epoch', fontsize=8, fontname='Arial')\n",
    "plt.ylabel('Accuracy', fontsize=8, fontname='Arial')\n",
    "plt.yticks(fontsize=6)\n",
    "plt.legend(loc=4,prop={'size': 5})\n",
    "plt.xticks(fontsize=6)\n",
    "plt.ylim([0.25, 1.0])\n",
    "sns.despine()\n",
    "\n",
    "pestr = 'relative'\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",
    "plt.subplot(1,4,2)\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(f\"{pestr}\", fontsize=8, fontname='Arial')\n",
    "plt.xlabel('Epoch', fontsize=8, fontname='Arial')\n",
    "ax.set_ylabel('')\n",
    "ax.set_yticks([])\n",
    "plt.legend(prop={'size': 5}).remove()\n",
    "plt.xticks(fontsize=6)\n",
    "plt.ylim([0.25, 1.0])\n",
    "sns.despine()\n",
    "plt.tight_layout()\n",
    "\n",
    "pestr = 'rope'\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",
    "plt.subplot(1,4,3)\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(f\"{pestr}\", fontsize=8, fontname='Arial')\n",
    "plt.xlabel('Epoch', fontsize=8, fontname='Arial')\n",
    "ax.set_ylabel('')\n",
    "ax.set_yticks([])\n",
    "plt.legend(prop={'size': 5}).remove()\n",
    "plt.xticks(fontsize=6)\n",
    "plt.ylim([0.25, 1.0])\n",
    "sns.despine()\n",
    "plt.tight_layout()\n",
    "\n",
    "pestr = '2d-fixed'\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",
    "plt.subplot(1,4,4)\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(f\"{pestr}\", fontsize=8, fontname='Arial')\n",
    "plt.xlabel('Epoch', fontsize=8, fontname='Arial')\n",
    "ax.set_ylabel('')\n",
    "ax.set_yticks([])\n",
    "plt.legend(prop={'size': 5}).remove()\n",
    "plt.xticks(fontsize=6)\n",
    "plt.ylim([0.25, 1.0])\n",
    "sns.despine()\n",
    "plt.tight_layout()\n",
    "\n",
    "outputdir = '../figures/training_traj/'\n",
    "if not os.path.exists(outputdir):\n",
    "    os.makedirs(outputdir)\n",
    "# plt.savefig(f'{outputdir}training_traj_predesigned_1head.pdf',transparent=True,dpi=300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 350x125 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "plt.figure(figsize=(3.5, 1.25))\n",
    "head = 1\n",
    "pestr = 'random'\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",
    "plt.subplot(1,4,1)\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(f\"{pestr}\", fontsize=8, fontname='Arial')\n",
    "plt.xlabel('Epoch', fontsize=8, fontname='Arial')\n",
    "plt.ylabel('Accuracy', fontsize=8, fontname='Arial')\n",
    "plt.yticks(fontsize=6)\n",
    "plt.legend(loc=1,prop={'size': 5}).remove()\n",
    "plt.xticks(fontsize=6)\n",
    "plt.ylim([0.25, 1.0])\n",
    "sns.despine()\n",
    "\n",
    "pestr = 'learn-0.2'\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",
    "plt.subplot(1,4,2)\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(f\"{pestr}\", fontsize=8, fontname='Arial')\n",
    "plt.xlabel('Epoch', fontsize=8, fontname='Arial')\n",
    "ax.set_ylabel('')\n",
    "ax.set_yticks([])\n",
    "plt.legend(prop={'size': 5}).remove()\n",
    "plt.xticks(fontsize=6)\n",
    "plt.ylim([0.25, 1.0])\n",
    "sns.despine()\n",
    "plt.tight_layout()\n",
    "\n",
    "pestr = 'c-nope'\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",
    "plt.subplot(1,4,3)\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(f\"{pestr}\", fontsize=8, fontname='Arial')\n",
    "plt.xlabel('Epoch', fontsize=8, fontname='Arial')\n",
    "ax.set_ylabel('')\n",
    "ax.set_yticks([])\n",
    "plt.legend(prop={'size': 5}).remove()\n",
    "plt.xticks(fontsize=6)\n",
    "plt.ylim([0.25, 1.0])\n",
    "sns.despine()\n",
    "plt.tight_layout()\n",
    "\n",
    "pestr = 'nope'\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",
    "plt.subplot(1,4,4)\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(f\"{pestr}\", fontsize=8, fontname='Arial')\n",
    "plt.xlabel('Epoch', fontsize=8, fontname='Arial')\n",
    "ax.set_ylabel('')\n",
    "ax.set_yticks([])\n",
    "plt.legend(prop={'size': 5}).remove()\n",
    "plt.xticks(fontsize=6)\n",
    "plt.ylim([0.25, 1.0])\n",
    "sns.despine()\n",
    "plt.tight_layout()\n",
    "\n",
    "\n",
    "outputdir = '../figures/training_traj/'\n",
    "if not os.path.exists(outputdir):\n",
    "    os.makedirs(outputdir)\n",
    "# plt.savefig(f'{outputdir}training_traj_learnables_1head.pdf',transparent=True,dpi=300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "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.002000</td>\n",
       "      <td>0.999463</td>\n",
       "      <td>4785.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>1.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.000831</td>\n",
       "      <td>0.999837</td>\n",
       "      <td>4785.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>1.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.927062</td>\n",
       "      <td>0.558719</td>\n",
       "      <td>4785.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>1.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>1.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.963873</td>\n",
       "      <td>0.508867</td>\n",
       "      <td>4785.0</td>\n",
       "      <td>4.0</td>\n",
       "      <td>1.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>1.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>1.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>1.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.002000  0.999463  4785.0    4.0    1.0     0.0\n",
       "2d-fixed   4000.0    0.0  0.000831  0.999837  4785.0    4.0    1.0     0.0\n",
       "c-nope     4000.0    0.0  0.927062  0.558719  4785.0    4.0    1.0     0.0\n",
       "learn-0.2  4000.0    0.0  0.000000  1.000000  4785.0    4.0    1.0     0.0\n",
       "nope       4000.0    0.0  0.963873  0.508867  4785.0    4.0    1.0     0.0\n",
       "random     4000.0    0.0  0.000000  1.000000  4785.0    4.0    1.0     0.0\n",
       "relative   4000.0    0.0  0.000000  1.000000  4785.0    4.0    1.0     0.0\n",
       "rope       4000.0    0.0  0.000000  1.000000  4785.0    4.0    1.0     0.0"
      ]
     },
     "execution_count": 5,
     "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": 6,
   "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",
    "head = 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\"Generalization performance\\nmean sorted\",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/validation_regularPEs/'\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": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "2d-fixed vs. learn-0.2 | t = 0.9615553366540599 | p = 0.3525875516376735\n",
      "learn-0.2 vs. relative | t = 2.5065549071720072 | p = 0.025146114183522875\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": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\\begin{tabular}{lrrrr}\n",
      "\\toprule\n",
      "PE & Validation acc & Validation SD & Training acc & Training SD \\\\\n",
      "\\midrule\n",
      "2d-fixed & 0.977 & 0.073 & 1.000 & 0.000 \\\\\n",
      "learn-0.2 & 0.956 & 0.039 & 1.000 & 0.000 \\\\\n",
      "relative & 0.920 & 0.042 & 1.000 & 0.000 \\\\\n",
      "random & 0.888 & 0.046 & 1.000 & 0.000 \\\\\n",
      "rope & 0.805 & 0.115 & 1.000 & 0.000 \\\\\n",
      "1d-fixed & 0.781 & 0.185 & 0.999 & 0.000 \\\\\n",
      "nope & 0.334 & 0.020 & 0.509 & 0.002 \\\\\n",
      "c-nope & 0.314 & 0.042 & 0.559 & 0.102 \\\\\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['PE'] = 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": "iVBORw0KGgoAAAANSUhEUgAAAL8AAACMCAYAAAAtBHgFAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8g+/7EAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAsKUlEQVR4nO2de1zO5//Hnx3vdFJZhExjjOYwswfZHBZ+Q84MzXl82SrMbMghbckXsXIsjDlk5izmMJPNKYsRYkJK7qTD6i6duztcvz/8un+lg/tORfo8H489pvu+Du/P5359rs91va/3dV1aQgiBhEQtRPtlGyAh8bKQxC9Ra5HEL1FrkcQvUWuRxC9Ra5HEL1FrkcQvUWuRxC9Ra5HEL1FrqdHiP378OEOGDGHQoEEMHDiQTZs2qb6bMmUK8fHxapUTHx/PlClTAHB1deXgwYNq23Dz5k0WLFgAwN69ezl69KjaeefNm0dMTIza6ZOTk3F2dlZd77Fjx0pN17NnTx49elTi87i4OPr27cugQYPYsGEDv/zyi9p1l8W4ceO4dOnSC5dTFo8ePaJnz57lplm7di1r167VuGzdihr1somPj2fFihUcPHgQc3NzMjIyGDduHM2aNaN37978+OOPapfVoEEDjdIXpW3btrRt2xaAkJAQOnXqpHbeS5cu4eLionb6NWvWYGtri6+vL//++y9Dhw6lc+fOvPHGG2rlv3z5Mra2tnh7e6td5+tMjRV/cnIyubm5ZGVlYW5ujpGREcuWLUMmkwFPW78dO3Zw+fJl/vzzT1JTU5HL5YwfP57Hjx8THBxM3bp12bx5M4mJiYwfP54//vijWB0+Pj789ddfpKamYmlpibe3N5aWlnTu3Jl27doRFxfH7Nmz+fHHH5k6dSp//PGHqtz58+fz+++/U7duXRITE3F0dOTUqVNoaWkB4OfnR0JCAlOnTsXf35+YmBg8PT3Jzs7G3NwcDw8PmjZtWsye7t2706ZNGwAsLS0xMzMjMTERXV1dZs+eTVxcHM2bNycnJ6fE/QoLC8Pb25vMzEwWLFiAlZWV6j5NnTqVI0eOIJPJGDx4MMuXL+fdd9/F09OTsLAw8vLyGDduHCNGjECpVLJgwQJu3rxJkyZNSE5OLlHXo0ePcHJyomXLlvzzzz/Y2trSuXNnDh48SEpKCuvWraNFixaEhoaWes23b99WvU1btWqlKlehUPDdd9+p3mrTpk177luhXEQNxsPDQ9ja2orhw4cLLy8vcfv2bdV39vb2Ijo6Whw4cEB0795dpKWliejoaNGyZUtx7tw5IYQQY8eOFadOnRLR0dHC3t5eCCHE3LlzxYEDB0RUVJRwdnYW+fn5QgghXF1dxZYtW4QQQrRs2VIEBQUJIYQIDg4WY8eOLZa3MP0vv/wihBBiy5YtYtWqVSXsL7RRqVQKe3t7ce3aNSGEEMePHxfDhg0r99qPHj0qevfuLXJzc8X3338vVq5cKYQQ4u+//xYtW7YU0dHRJfIcOHBAzJ07VwghxJo1a8SaNWtU/541a5ZwdXUVPj4+QgghvL29xdatW4UQQmRkZIghQ4aIsLAwsXnzZjFr1ixRUFAgoqKiRNu2bUVwcHCxegrv882bN0VeXp7o1auXyr7Vq1eLJUuWlHvNAwYMUP1Gvr6+qt/mm2++EadOnRJCCJGUlCR69+4tEhMTi12LJtToPr+bmxuBgYGMHDmSmJgYHB0dOXHiRIl0H3zwAcbGxlhbWwPQpUsXABo3bkxqamqpZTdt2pS5c+eyd+9eli5dytWrV8nMzFR936FDh3JtGz58OAEBAQAEBAQwbNiwMtNGRUVhYmLCe++9B0C/fv2Qy+WkpaWVmv7w4cMsXbqUtWvXoqury+XLl+nfv7/qWps0aVKubc/i5OREZGQkYWFhTJs2DYDz58+zZ88eBg8ezGeffUZqair37t3j8uXLODg4oKWlRdOmTcu8D5aWlrRp0wYdHR0aNmyouufW1takpqaWec0xMTHEx8fTrVs3AIYMGaIq8/z586xZs4bBgwfz+eefk5eXR2RkpEbXWpQa2+05c+YMmZmZODg4MHLkSEaOHMm+ffs4dOgQ/fr1K5ZWT0+v2N+6us+/7NDQUL755hsmT55M37590dHRQRSJ/q5Tp065+T/44ANSUlI4ffo0devWLVeQ+fn5qu5QIUII8vLySqTdtGkTu3fvZvv27TRv3hygRF4dHR0AFixYwK1btwDw9PQss/60tDSePHkCPO1a1K9fn4KCAlasWKHqZiUlJWFiYsLx48eL3Yey7qW+vn6pNj3vmg0NDYuVXzRfQUEB27dvx9zcHICEhAQsLCwIDg4u89rKo8a2/AYGBnh7exMbGws8vXG3b9/mnXfeqZTyr169SufOnXF0dMTGxoazZ8+Sn59fbh4dHZ1iaYYNG8bixYvLbPUL0zdr1oyUlBSuX78OPPViWVlZqX7kQg4ePMihQ4fYs2ePSvjw9E126NAh4OlDK5fLAViyZAmHDx/m8OHDqkF5aXz//fc4OjoyefJk3NzcALCzs+Pnn39GCIFCoWDo0KFERETQpUsXjhw5QkFBATExMYSEhJR7T8qivGtu3LgxgYGBAMW8Z4U2wdO35YABA1QPbUWosS2/nZ0dLi4uTJ06ldzcXIQQdO3alenTp1dK+Q4ODri4uPDJJ58gk8lo06YN0dHR5ebp2rUrK1aswMjICAcHB/r374+fnx99+vQpNX2vXr2YOnUqmzZtwsfHhyVLlpCVlYWJiQk+Pj4l0vv4+KClpcV//vMf1WceHh7MmDEDV1dX+vfvT7NmzTTq9pw4cYLIyEhWrlyJtrY2R44c4cCBA0ybNo3vv/+egQMHkpeXh5OTE61bt+btt98mPDycfv360bhxY1q2bKl2XUXR19cv85pXrFjBvHnzWLdunapbBLBw4ULc3d0ZOHAgQgiWLFlCvXr1KlQ/gJYQ0kququLAgQNcu3at3C6HxMujxrb8rzrOzs48evSIzZs3v2xTJMpAavklai01dsArIfGiSOKXqLVI4peotUjil6i11HjxCyFIT09HGrdLaEq1iz8wMJB58+aV+NzHx4cRI0Ywbtw41QylOmRkZNCxY0cyMjIq00yJWkC1in/lypWsXLmyRCt969YtwsLC2LdvH99++y0rVqyoTrMkainVOsnVtm1bunbtqop2LCQkJIQPP/wQgPbt23P79u0yy1AqlSiVStXf6enpANy+fbtY+MG9e/dISUnh7NmzPHz4kKZNm9KjRw/MzMxUU/LvvPOOKnBLovZRreLv06dPqUve0tPTadCggerv8vrvGzduZN26dSU+9/T05PLly2Xmi4qKIioqqthnRQPCJGofr0R4g7GxcbE+u7Z22b2xL774gs8//1z1d3p6Oj169GDhwoUVavklai+vhPjfe+89fH19GT9+PNevX+ftt98uM62+vn6JWHEAW1tbjdbPSki8VPEvW7aMoUOH0q5dO9555x1GjhyJlpYWS5cufZlmSdQSanxgW3p6Oh07duTq1asYGxu/bHMkahA1fpJLQqKiSOKXqLVI4peotbwS3p5XjYcPH5a5MLpu3bolNpOSqJlI4n+GpKQkunTpQkFBQanf6+joEBoaWu7CaenhqRlI3p7/4/r160RERACQmJio2qAqLi6O7du3M2HCBKysrDA0NFTtjdm8efNiuwvA04enbdu2L/TwSFQPkvh5urdk586dn7svz7Po6Ohw6dIlrK2tefToEQqFAoDY2FhVzJFcLmf58uXMnTuXN998E2NjYxo2bAiAhYWFahc5ieqnQt2evLw8tXY9q0k8u+GUunng6cPz0YcfkaMsuUFsIcuXLy/xmUxfRtDFoBIPgNRtqh40UnBkZCRz5swhMTGRPXv24OTkxMqVK2nWrFlV2VctWFtbc/HixQq33KGhoeUKvyxylDkoFIpi4q+MMYeEemjU7Zk4cSIzZszAw8ODgIAAjhw5wq5du9i9e3dV2lgulT3DW5E+uzotf2kUbflLG3M8O94AnjvmkFAfjVr+1NRU3n//fdXfgwYNYsuWLZVu1MukXr16/PXXX6puR79+/cjPz0dHR4cTJ05Qt27dEq2utbU1QReDUCgUPHnyhFGjRpX58Ghra7Nnzx7q1q2renM8evSIAQP6k5dXerdr+/btpX6uq6tDcPAladxQQTQSv5GREbGxsarddS9fvqw6DOJ1omifetCgQfz6668MHDiQdu3alZnH2tpaJcKiD8/Ro0fZv38/n376KQMGDCizz66rq1um+MvidRt3VTcadXvCwsKYP38+UVFRNGnShNTUVFatWvVSX72vemCbnZ0dUVFR2NjYlLuVdlFvUSHh4eG4uLiwfv16WrRoUSJPVXuLtm/fjp+fH05OTkyYMKHK6nlZaNR0tG7dmn379vHgwQPV1tqlxdZL/D9OTk4qAZVH0TdHobfn/PnzwNNDGQrFX53eHj8/P6KiovDz83stxa9Ryx8TE8POnTt58uRJsaWGLzP+/lVv+TXlVZoke91bfo3EP2rUKNq1a0fr1q2LnaoxdOjQKjFOHV438cP/t/wjRozgyZMn1K1bl3379gGatfwvS7xlzVO8cnMUmhzgNWjQII0P/apq0tLSRMuWLUVaWtrLNqXS2bZtm+jcubPYtm1bhfLb2tqKBg0aCFtb20q2rGwSExNFw4YNRYMGDUr816hRI5GYmFhttjwPjfr8tra23L59G1tb26p6FiWKMGHChAq12IUtb+GZXnl5eYSGhgLqtb4v0nIXdRU/O2AvzU38MtFI/Ldv32b48OHUq1cPmUyGEAItLS1Onz5dVfZJqEmht6i0eYYnT57wySefAKXPMxSlvBlmdccbzz4gLVq0KNdN/LLQSPy+vr5VZYfEC/Do0SO6fvQR2aUcPv0sBQUFjBgxAgADmYwLQcVji2pSy/2iqCX+P//8E3t7e/7+++9Sv2/cuHGlGiWhGQqFQi3hP0t2TsnYIqg5LfeLopb4b968ib29fam7rUHxg4Ilqh8LCwsMZDKNHwADmQwLC4sqsurVRy3xz5gxAyjuz09PTyc2NrbUmUeJ6sXa2poLQRWPLXrVqC5XqUZ9/r179xISEsKcOXMYMmQIRkZG9OnTh5kzZ1aaQRIVo7TYookTJ/L48WMaNWrEtm3bgFfQ1/4MlTHgVheNxL9jxw62bNnCsWPH6NWrFwsWLGDkyJGS+F8xCsX91VdfqSa5ntdnLyu2qOj/n6Uq3hzVOeDWSPz6+vo0aNCAc+fOMXr0aHR1dYttFy7xaqHuPMGjR4/o9lFXsnKyS/3excWl1M/ryAw4H3Sh0h+A6hpwayR+a2trpk+fTkREBB9++CGurq5Sn/81QKFQkJWTzWjLpjTQN1ArT7wym13/PizVW1RT0Ej8K1asICgoiEWLFiGTyejatatq8kSi5rPr34cv24RqReNuT1xcHAEBAeTm5tKpU6dy99KXqFlUpOUvRNMxw6vgadJI/F5eXjx8+JDhw4cjhODgwYM8evQINze3qrJPohqwsLCgjsxA45a/jswACwuLp2OGrl3JylZ/zFDHwIDzFyp/vKAJGok/KCiIgIAAVWv/8ccfM3DgwCoxTKL6sLa25nzQhQqvJAsNDSUrO5vFn9rzVn2z59b3ICEFt/1/vvTxgkbiz8/PJy8vT7V6Ky8vT7V3zfMoKChgwYIFPHjwAGNjY7y8vIrNLk6ePJmcnBy0tLR48803WbJkiSamSbwgRecJnkVdb8tb9c1o3eiNyjatytBI/IMHD2bs2LEMGDAALS0tfv31VwYMGKBW3lOnTiGTydi9ezfHjx9n06ZNuLq6qr5PSkoqcUqjhERVorb44+PjGTx4MK1bt+avv/4iODiY4cOHM2bMGLXyh4SE0LVrVwC6devGpk2bVN/FxMSQmprK5MmTUSqVzJ49+7UMpHrdefBvSqWmq2rUEn9ISAjTpk1jxYoVdOvWjW7durFmzRr8/Pxo164dbdu2fW4Z6enpqmWGRkZGxU5fLCgoYOLEiYwZM4bo6Gi++OILfvvtt2JLJQsp6xxeicqjMLbmWW/N80Ij3Pb9WS32VRZqif+HH35gw4YNxVrjGTNmYGdnh5eXF/7+/s8to+hxoxkZGZiYmKi+s7KyYsSIEejo6GBjY4OJiQnJycmlRhyWdQ6vROVQWmxNobfmebE1i0fY85al2XPrePBvSokH5WW4StUSf2ZmZqndkE6dOpGamqpWRe+99x5BQUH06tWLc+fO0aFDB9V3Z8+e5ddff2X16tXExcWRlZWFubl5qeWUdQ6vROVQNLbm2cC458XWvGVZsQHv08U4XcnWILzCQGbAhRcMrVBL/Hl5eRQUFJSY0MrPzyc3N1etij755BPOnTuHo6Mjenp6+Pj4qI4i7dmzp+o7LS0tPD09S+3yQNnn8EpUHhUJjIOnLkx1eDbd08U42XQzbo2ZrtFz86fkZXA+PeyFXaVqib9z586sW7dOFddfyLp169QemOro6JTY36eot8fDw0OtciSqD3UD4ywsLKhjYIDbfvX7/HUMDEp0a810jaina1JGjspHLfHPnDmTKVOmEBAQQKtWrZDJZNy6dQtLS0v8/Pyq2kaJVxxra2vOX9BskqzGhDcYGxuza9cugoODCQsLQ1tbmzFjxvDBBx9UtX0SNYTKmCRLyct4bhpN0j0Ptf38WlpadOnShS5dulRKxRISz3I+Paxa65P2uJZ4ZdB0wPuiSOKXeGV4JQe8EhLVwSvb55eQUBdNwyOe7jtkoFFXxkBW0lWqKdI5vBKVSnnnC5QXHqHpyTTVFt4gIaEuzx7oV5TywiMqw1WqKZL4JSqdV3lTrKJI4pd45ahoSLWmSOKXeKV4kZBqTZHEL/FKUdExQ0Wo8eIvdFZJK7peH+rVq1emyEv7nY2MjMoMgS+PGi/+wtVh0oKW2ktF3dw13s9fUFBAQkJCmU9/4Uqvs2fPVugGvcz8Ndn26sxfa1t+bW1trKysnpvO2Nj4hSbBXmb+mmz7q5C/LKSNNiVqLZL4JWotr7349fX1mTZtWoUXvb/M/DXZ9lch//Oo8QNeCYmK8tq3/BISZSGJX6LWIon/BVB3tzqJV5MaLf7ExER+++03Nm3aREWHLjExMQQGBiKXyzXKp1Ao2LFjB/v27atQvZVBYfBXWQdO1xYbKkqNFr9SqUQulyOXy3FxcdH4B1AqlVy8eBFfX1/mzZvH0aNH1cqnUCg4evQoBQUFau1QrQnqxigpFAp2796NQqFAW1u7UsSXlpamUfqqsKE6qfHenry8PHR1dVm4cCFDhgzReCOtiIgIfv75Z1q1asXIkSOfmz4lJYX9+/eTkJCAra0tgwcPRktLS2XHi5CcnMxPP/1EdnY28+bNK/OwP4VCwf79+zl16hRGRkasWLECS0vLUvdTVZeUlBS2bt2KTCbDycnpueECSUlJHDx4kN9//x0dHR38/PwwNzfXyIaYmBjCwsIIDw/niy++qPbDDWtcy5+ZmUlMTAwKhYKYmBgePnxIfHw8crkcLS0ttVuf3NxcQkND2blzJ82aNVNL+ImJiezfv59jx47xxx9/cO/ePcaNG0dMTMwLC1+hUHD48GHi4uKIi4tj9erVZdpw7Ngx0tLS8Pb2pnPnzixevJh///23wq2vQqHgwIEDREZGkpSUxFdffVVuOYmJiRw/fpz09HTmzZvHhx9+yJkzZ0hKSkJbW5v8/Pzn1hkVFcXGjRsJDw8nNjYWZ2fnCnddK0qNEr9SqWTevHn07t1bdayRu7s7hw4dolGjRnTs2FGt1iMvL4+bN29y9OhR3njjDXr37s3y5ctxd3dXLaLev39/sR8jPz+f06dPc//+fXJycjAwMGD8+PG4uLiwc+fOF3rlKxQK9u3bR2ZmJvb29gwcOJDo6Gg8PT2LlZufn8/JkyeJiYmhU6dO6OvrY2pqSrdu3Zg1axYJCQkat56FdWdnZzN9+nTVyZpnz54tNX1+fj6//fYb0dHRfPDBB1y6dIkzZ86gVCqZOHEiCQkJap3TFhQURPv27Rk7diweHh4YGxtz48YNjWx/UWqU+PX19Zk1axbt27enS5cu+Pv7s3jxYuDpa3vKlClkZ2dz584dPD09yywnPz+fU6dOYWlpSb9+/diwYQN16tShffv2eHp6snLlSo4cOUJiYqIqj46ODp988gnt27fHwcEBHx8f3NzcOHfuHFpaWsVEp+mDkJGRQXp6Ovb29iQlJREaGoqTkxOmpqbMnz+/mA39+vXD0dGRqKgo1q9fT0hICCNGjKBnz57s3btXYxsK6+7ZsyctW7ZELpejp6fH9evXOXfuXInWWEdHBwcHByZOnIihoSFBQUEAjBo1igkTJnD16lUA/v7771IPFRRCqBofQ0NDTExMKCgowM7OjhYtWlTvuEHUQCIjI8WECRPEmTNnxPr168XChQtFRkaGOHXqlJg8ebIYN26c2Lt3b7llJCcnCyGEWLp0qVizZo3qc0dHRzF9+nQRHR1dar6EhATx2WefidDQULFz504xevRocffuXREWFiYOHz4s4uPjhRBC5OXlaXRN2dnZIiUlRUycOFHExsYKIYRISkoS06dPFzk5OSXSOjo6il27donMzEwxc+ZMYWdnJy5evCjkcrlQKBRCCCEKCgrUqjs9PV0IIURERISYNGmSaNOmjZg0aZJYvHixmDRpUpnlbNmyRQQEBIiUlBQxcuRI0bdvX/HPP/+ImzdvigkTJpT7G1y8eFGMGDFC7Ny5U6xcuVJMnTpVpKSkqK67OqhRLX8hb731Fu7u7jx48IDQ0FDmz5+PoaEhZmZm3Llzh88++4wRI0aUW4aZmRlCCHJzc7G1tQVg/fr16OnpMXv27DK30bC0tGTRokWsX78egJUrV6Kvr8+dO3cQQuDq6qr2q78oMplMtTBHS0uL7Oxsdu7cSYcOHYiJiSnmzpXJZCxevJjTp0/j7+/PuXPn8PDw4O2332b69OmsW7eOlJQUtLS01Op/GxkZkZWVhY+PD7dv36ZPnz6sXbuWhQsX0rhx4zLduT179mTLli1cvnyZtm3bMmnSJPLy8vDy8mLYsGGq36C01rxLly7Mnj2bRo0a8T//8z84ODiwfPly4uPj2b9/Pz4+PlU+BqjR3p7MzEycnZ2ZPn06enp6+Pr60rdvX4YMGaLyOuTm5qKnp1dmGeHh4fzwww+kpqZiYmLCokWLkMlkvPHG0+N1CstJSkoqtrSuML2XlxdyuRyZTIanpyc//fQTMTEx/Pe//0VLSwshhEYLLQIDA9m8eTN6enp069YNc3Nzbty4QVxcHO+++y5fffWVqov14MEDZs6cydixY+nRowezZ8/mww8/pGHDhgQEBODl5aW6DnV48OABV65coV69evTs2ZOEhARWr17Np59+StOmTUvdIS0iIoL9+/czevRoYmNjWbduHZ999hkfffQRKSkpmJiYqO0Fun79OtHR0ezcuZNWrVqRkpKCj49P1XmBquX9UoWEh4eLyZMni06dOolff/1VXL9+XYSFhZVId+bMmTJf3w8fPhRubm7i4cOHQi6Xi0mTJonr168LIYS4cuWKOHv2rBg5cqSYPn26yM/PV+VTKpXim2++Eb/99pvIzMwUzs7Ows3NTZw6dUrExcWpuhPqdoEK7YuMjBSPHz8WYWFhwtXVVRw6dEjcv39feHt7i5kzZxbLk56eLm7cuCGGDh0qAgICVJ+PHz9eHDt2rETZzyM4OFg4ODiIAwcOiFWrVomNGzeKY8eOCTc3N+Hr61tqOfn5+SI8PFwMGTJEnDlzRuTl5YlLly6JrVu3ikmTJom4uDhVurL4999/xfbt28WiRYtEaGioEEKIWbNmFbuGyqZGt/yFPHr0iPDwcOzt7bly5QqrV69m5syZdOzYUTUoTE5OpnHjxqxZs6bUlrion/7ChQvs2rULX19fNmzYgJ+fH9999x06Ojqkp6czevRoVb579+6pPFBXr16la9euNGnShNDQUORyOa6urjRo0EDtaylsIeVyORs2bKCgoICFCxdibGzMzZs3CQwM5Ouvvy6WJyYmhhs3buDg4EBGRgb79+8nJSWFwYMHY2NjQ0ZGBkZGRmr74C9evMhvv/3G+++/z5MnT0hPTycpKQl46uZctWpViXIKffampqYEBgaSn5+Po6Mj4eHh7Nq1iw0bNmBsbFzmfEhsbCw+Pj6MGTOG9u3bo1AoGDt2LEuWLCl2eGFl8lqI/1nOnTtHdHQ0Y8aMYdiwYSgUCs6cOcMff/xBTk4O/fr1KzNvUYFERETg7OyMvb09pqamTJ06lXv37mFmZkajRo1UeQo9L40aNeL999/n4cOH3L17Fx0dHSwsLJg2bRq6urpqiy8iIoLVq1fTvHlzbt26xbRp07C0tOTgwYNkZWUxefJkIiMj6dixY7EHOTs7mx07dqCvr4+BgQEKhYLk5GTu37+Pl5cXlpaWHDhwgGHDhpXZFStq4+PHj/n555+xtLTE3t6epk2b8vXXXzNs2DC6detWav6AgABu3ryJo6Mj27dvp0ePHoSEhDBy5EgMDQ356aefGDt2LE2aNCmRNy0tDRMTE+Lj43FycqJ///706dOHffv2oaenh7Ozc6V2gWrkgLc8cnNz6d69O2PGjGH27Nm0bt2aOXPmsG3bNrp27Ur79u3LPUGyUBRXrlzBxcWFuXPn4urqypMnTzh9+jTNmzenUaNGxQZjNjY2uLq68vbbb3P//n3S0tKYOHEiHh4e9OrVix9++EEVAqDOANTExIRu3brh5OTEpEmTWLp0Kbt27SIqKgoLCwtWr15NQEAAM2bMKDaYzMnJQalUYmVlxePHjzE1NWXQoEH079+ftWvX4uPjw8GDB4sdAP4sheK6fv06e/bs4fbt2xgbG6t2StPX1y93HNG+fXsuXLhAXFwcDRs2ZPfu3dja2vL48WMGDRpERkYGTZo04cyZMyXyGhoaolQqmTRpEkOGDOHjjz/m22+/pXHjxiiVSubMmVOpg+DXsuUHGD9+PFZWVnh5efH7779z7do13n//fbp06YKxsXGZrbAQAqVSyahRo3BxcaFLly78+OOPGBoa0qFDB65fv46joyOmpqYl8qampvLLL7/Qo0cPWrVqRUREBLNmzcLU1BRLS0vmzp2rdheoaPcgPj6eBg0aEBoayp49e7h27RpHjx7F398fXV1dxowZo8qXk5ODTCbDy8uLfv360bZtW9LS0pgxYwb16tVj4cKFmJmZlRuOoVQq2bx5M4aGhlhYWLBnzx769+/P1atXSU9Px8XFhQsXLpCfn4+Li0uJ+/jgwQM2bdpEREQE//nPf+jZsydTpkyhbt262NraMnXqVK5cuYKZmRlvv/12ifofPnxI06ZNmTJlCkOHDsXBwQF4unObk5MTbdq0UesePo/XruUHiIuLw8bGBi8vL44fP861a9fo3r07VlZWrFmzBqVSqfLEPIuWlhYymYxdu3bRpUsX1q5di5GREb1796ZTp06cPXuWrKwsVfqiZZiamvL555/TqlUr7t27x6pVq3BwcMDf3x87OzvWr19PXl6eWtdQKEwhBJaWlsDT7pCNjQ1z587l22+/JS4uroQbUSaTkZ6ezj///ENcXBx5eXn8/PPP1KtXj2XLlmFmZkZGRga6urplvoX09fUZPXo0H3/8MYMGDWLatGkYGBjQrVs3xowZw4oVK8jJycHCwgJnZ+cS+d966y2cnZ0ZMmQIvXr1Yty4cbRv355Vq1aRkZFBYGAgbdq0KVX4AG+++SZKpRJTU1PeeecdAG7dukXTpk0xMjJSTaS9KK+l+K2srPDw8OD48eOEhITQvXt3OnXqRFpaGrGxsejr65OZmVmuC7LwFdyyZUvatm3Lm2++yb1790hPT+f06dPMmTOHtLS0EmXo6+ujVCrZsGEDzZs354svvgCeDiILCgo0jgEqOnv81ltvce7cOd59910aNmxISEgIffv2LZHH2NgYd3d3Lly4oBofzJ49m+XLlzNjxgxcXFyIj49HR0enzG6EmZkZNjY2wFOf/LBhwxg4cCDnz5+nUaNGyOVyxowZg7W1NfHx8SXyN2nShNGjR7NmzRosLS2ZOXMmO3bsQFdXl5iYGL7//ntycnLKvGZ9fX369+/PwoUL8fT05OTJk+Tm5nL79m02b97M8uXLX3g2+LXt9sDTKMlz587RuHFjkpKSuH79On/99Rc2NjaYmpqycOHC5y6OVigU+Pn5ERYWhrGxMXK5nMmTJ2NqakqnTp2oW7cuQIluVHJyMubm5gDMmzePlJQUfH190dLSQqlUVnhRdmBgIMeOHeP999/nk08+oUGDBmV24WJiYtiwYQNubm54eXmhUCjw9vbm8OHDnDx5Ei8vL432w1EoFCxZsgR3d3du3ryJu7s7zZo1Y+PGjWU2JNHR0ezatQsjIyNycnJwcHBAS0uLwMBAvvzyy2JvuKJlFP595coVEhISqFOnDikpKSgUCrKyshBCcOPGDX788ccKbVhVWMlrTWxsrPjhhx/EvHnzxDfffCP27Nkj4uLixOPHj4UQQmRlZQkhyvdBP3nyRKSmporjx4+rQiFu3LghAgIChJeXlyqkoWgZhf5wNzc3MWLECJGfny/WrFkjgoKCVGnOnz8vTpw4Iby9vcut/9kyk5KShFKpVOv6CwoKhFKpFF9++aXqM7lcLlxdXUVWVpaqTHXqF0KI06dPiyFDhohffvlFeHt7i7t37xazrTSSkpKEv7+/+Ouvv4QQQkyfPl0cPnxYZGdni7CwMJGWllZqvkKb8vPzRWBgoJg1a5bYt2+f6vvZs2e/UCjEa9ntKYqVlRUzZszgv//9L7m5ubz77rsYGBiwc+dO/Pz8+Oqrr4iPj0dbW7vMLoCpqSnGxsaEhobSpEkT/vnnH5YtW8b9+/dp2rQpc+bMITExsVjrW9gaffnll+zduxdtbW1atGiBv78/KSkpbNu2TRU8l5GRoZYno7BMCwsL9PT01PJ8FIZKpKamcvbsWWJjY7l58yb169fn1q1b7Nq1i9TUVLXDoXv27MmiRYswNTXFxcWFli1bFrOtNCwsLBg+fDh2dnbcuXOHOnXq0KlTJ7799luOHTvGiRMnVGmL2qCtrU12djZHjx7l0qVLyOVy1RjA398fuVyOsbGxxotwVPdGqHMHazhCCDIyMli2bBmff/45e/bsITg4mClTpmBsbMzWrVvx9fV9bhcgPj6exMREbt26RXx8PH///TdeXl4EBgbSokUL7OzsVPU9K4bCMIvo6Gi2bt2KQqFgxowZNGvWjKysLBYuXIibmxtmZmZVcg/Cw8NZvHgxDRo0wMLCgnr16qGjo8Pp06dp2LAhc+bMoUGDBhqHY2iKl5cXJ06cwN7eno4dO9KuXTvu3r1LcnIyPXr0oH79+sW6cenp6Wzbto1mzZphbm7OsmXL6NSpE0FBQbi5uXHp0iUSExNJT0/H29tbo3mAWiH+QlJSUtDT02PXrl00btyY3bt3M2/ePM6ePcvkyZNVMUD5+fnlBqZ5enrSuXNnbGxs8PDwwNTUlMmTJ/Pee++Rm5uLTCYrNV9BQQFJSUls2LCBTz/9lNatWwMQHBxMUFAQLi4uyGSyKhNfYmIiMpmMa9euERcXR2xsLKNGjeLkyZMEBwezatUqZDLZc6//RYiIiCA3N5dWrVqRlJSEu7s7FhYWNG3alIsXL7J06VLq169fLE/h5BfA3bt30dPTw9zcnG3btuHv78/Jkyc5ffo0UVFRuLq6qm3La9/tKUphJOeFCxeoX78+s2fP5rvvviMkJITMzEzCwsJ48uTJc3/4zz77jC1btnDnzh3s7Oxo2LAhBQUFuLm5MWfOHNWuDuvWrSvWNdHW1iYtLY2srCyV8O/cucP27dtp3rw5BgYGGq1G05Q33ngDQ0NDIiMjiYyM5NNPP8XExARtbW2MjIzYu3fvc71AL0rz5s1p1aoV8NQDVr9+fSIiIujXrx99+/bl4sWLJfIUCr+goIB33nmHZs2akZSURExMDJ6enixfvhxjY2NVlK661KqWv5DCyaexY8cSFxdHo0aNSEpKIjg4mNzcXPz9/fnjjz/49ddf8fHxKbOMS5cu0aFDBxISEggJCaF58+Zoa2sTHBxMamoqcXFxxRaYwNM5iG+//ZY+ffpgYGDAxYsX+eCDD4pNVL2IN0gdFAoFubm5GBoacuLECR48eMDHH3/M/fv3OXToEDt37sTAwKDK6i9k48aNWFlZ0apVK5YtW4aVlRXdu3cvN/ykkOjoaObPn8/GjRvZv38/Bw4cwNnZmT59+qhdf60UPzyNW8nKysLc3JwdO3ZgZGTElClT+PPPP9m2bRuxsbG4u7vz0UcfPbes+fPnY2dnx6BBgwDo3bs3bdq0YdWqVUDJMUBERAQ//fQTHTp04I033uDjjz8GnrZseXl5BAQEEBISgo6ODosXL66SkF4hBAcPHiQ0NJThw4fTrl07AgMDuXbtGt988021LCaPjIzE3d2dsWPHIpfLiYqKYv78+RgZGamVPzAwkK1bt9K7d2+aNGlChw4dNDu2qMJ+oteEyMhI8cUXX6j+PnTokOjQoYMqpPl5LsCsrCzx9ddfi/PnzwshhHB1dRXjx49XfV9W/sLPMzMzS6SLiYkR165dE6tXrxbOzs4VuCr1SExMFKGhoaKgoECcPHlSLF++XBw5cqTK6iuNu3fvCh8fHxEUFCSys7OFEOqHXxfmLwyZ1pRaL/6kpCQxatQoERISIk6cOCEGDBig8ker6/sODw8XX3/9tbC3txeTJk1SfV5e/vz8fJGTkyNcXV2FXC4vM913331XYhljZRMYGCjc3d3F0aNHq7Sesii63kET4RelIvlqbbenKA8ePMDDw4Pg4GC2bdtG586dNd4DJyoqih07drBo0SKg5IxvWaSnp5OcnMy1a9eQy+Xo6upSr149Hj9+jEKh4N9//8XX17fC16YOSUlJ/P3336WGSrzOSOL/P6Kjo0lPT6d169YvtPkTqC/8Qgr3rdHS0mLGjBnExMSQnZ1No0aN6NOnj2oCrir977URSfzP8LJEFhUVhaenJ+PGjStxsuSLPowSpSPd0Wd4Wa2rjY0N8+fPZ9OmTVy7dq3Yd5Lwqwap5X/FSEhIwNLSUuriVAOS+F9RpD5+1SO9T19RJOFXPZL4JWotkvglai2S+CVqLZL4JWotkvglai2S+CVqLZL4JWotkvglai2S+CVqLZL4JWot/wtNIiX/w5zlpwAAAABJRU5ErkJggg==",
      "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
}
