{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "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": [
    "# Experimental parameters"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "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",
    "# 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",
    "initializations = [0.1, 0.2, 0.3, 0.4, 0.5, 0.6, 0.7, 0.8, 0.9, \n",
    "                   1.0, 1.1, 1.2, 1.3, 1.4, 1.5, 1.6, 1.7, 1.8, 1.9, 2.0]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Define function retrieving attention weights for each model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "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 models and compute attention for each model"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Load model and get attention weights for 2d-fixed"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "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 model and get attention weights for all learnable PE models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "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",
    "initializations = [0.1, 0.2, 0.3, 0.4, 0.5, 0.6, 0.7, 0.8, 0.9, \n",
    "                   1.0, 1.1, 1.2, 1.3, 1.4, 1.5, 1.6, 1.7, 1.8, 1.9, 2.0]\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 init in initializations:\n",
    "    petype = 'learn'\n",
    "    pestr = petype + '-' + str(init)\n",
    "    resultdir = f\"../results/\"\n",
    "    modelname = f\"model-{model_label}_\" \\\n",
    "                f\"pe-learn-{init}_\" \\\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",
    "        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=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": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/p5/xfmq1vxd4sx5zq766ht140jh0000gn/T/ipykernel_43840/343643881.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_43840/343643881.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_43840/343643881.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_43840/343643881.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['Cosine'] = []\n",
    "corr_vals['JSD'] = []\n",
    "corr_vals['Layer'] = []\n",
    "corr_vals['Puzzle'] = []\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['Cosine'].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 validation performance across positional encodings"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "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=\"Cosine\",hue=\"PE\",palette='rocket',\n",
    "#                         )\n",
    "#     plt.xticks(rotation=-45,fontsize=7)\n",
    "#     plt.xlabel(\"Pos. enc.\",fontsize=10)\n",
    "#     plt.title(f\"Cosine with 2d-fixed, Layer{l}\",fontsize=10)\n",
    "#     plt.ylabel('Cosine', fontsize=10, fontname='Arial')\n",
    "#     plt.yticks(fontsize=6)\n",
    "#     sns.despine()\n",
    "#     plt.tight_layout()\n",
    "#     i += 1\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": 18,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAVQAAAClCAYAAAADKwW9AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8g+/7EAAAACXBIWXMAAA9hAAAPYQGoP6dpAAArKklEQVR4nO3deVxU1f8/8BcOMCADioSWSCJuaUlqrqUVYJYrKKmkYGnuChZuKEqKUggGJaR9yC2tXD4quNH3F2KJueGCW2J+Q1BEXECRGRhghjm/P/zOjYHZ7jAMMryfj4ePh8y973vPzDAvzpx75owFY4yBEEJInTVr6AYQQoi5oEAlhBAjoUAlhBAjoUAlhBAjoUAlhBAjoUAlhBAjoUAlhBAjoUAlhBAjoUAlhBAjoUBtACkpKfD19cXo0aMxatQoJCYmctumT5+OBw8e6HWcBw8eYPr06QCA0NBQ7N+/X+82XL16FWFhYQCAPXv24PDhw3rXLl26FPn5+Xrv/+TJE8yZM4e7v0eOHFG7n5eXF+7evVvr9vv37+ODDz7A6NGj8f3332Pnzp16n1uTwMBAnD17ttbtXbt2hY+PD/fPz88PGRkZAID4+Hi89dZbKtt9fHzwzz//6H3e6o/d+vXrcf78eYPan5eXh2XLlgFQfS5Jw7Js6AY0NQ8ePEBMTAz2798PR0dHlJaWIjAwEO7u7hgyZAh++OEHvY/Vpk0bXvtX16NHD/To0QMAcPHiRfTr10/v2rNnz2Lu3Ll6779+/Xp0794dGzZswKNHjzBmzBj0798fL7zwgl71GRkZ6N69O2JjY/U+Z10cOHCA+396ejpCQkLw559/AgD8/f0RFBRk8LGrP3bnzp1D//79DTrOvXv3kJeXB0D1uSQNiwLVxJ48eQKZTAapVApHR0fY2dkhKioKQqEQwLNe2vbt25GRkYHff/8dJSUluHPnDiZPnox79+7hzJkzaNGiBTZt2oTCwkJMnjwZx44dUzlHXFwcTp8+jZKSEjg7OyM2NhbOzs7o378/PDw8cP/+fSxatAg//PADZsyYgWPHjnHHXbZsGX777Te0aNEChYWF8Pf3R2pqKiwsLAAAGzduxMOHDzFjxgzs2LED+fn5WLNmDcrLy+Ho6IiIiAi0b99epT1vv/02XnvtNQCAs7MzWrZsicLCQlhaWmLRokW4f/8+OnbsiIqKilqPV1ZWFmJjY1FWVoawsDC8+OKL3OM0Y8YMHDx4EEKhED4+Pli7di1effVVrFmzBllZWZDL5QgMDMS4ceNQWVmJsLAwXL16Fa6urnjy5Ilez1e/fv3w6NEjvffX9hzs3buXe+x8fX1x7do1LF++HOvXr4dIJMLKlSvx+PFjWFtbY8mSJejduzdCQ0Nhb2+P69evo6CgAOPHj8esWbMQERGBe/fuITw8HCNGjEBCQgJ27NiB3NxchIeH48mTJ2jevDnCwsLg4eGh8TjVnT17Ft999x1sbW2RnZ2N9957D/b29jh69CgUCgUSExPRunVr/PTTTzh48CCkUimsrKwQExODjh07wsvLCx988AFOnz4NhUKByMhIvPbaa0hLS0NCQgJkMhlat26NmJgYODk58Xo8Gw1GTC4iIoJ1796d+fn5sejoaHb9+nVum6enJ8vLy2P79u1jb7/9NhOLxSwvL4916dKFpaenM8YYCwgIYKmpqSwvL495enoyxhhbsmQJ27dvH8vNzWVz5sxhVVVVjDHGQkND2ebNmxljjHXp0oWdPHmSMcbYmTNnWEBAgEqtcv+dO3cyxhjbvHkz++abb2q1X9nGyspK5unpyTIzMxljjKWkpLCxY8dqve+HDx9mQ4YMYTKZjK1atYqtW7eOMcbYuXPnWJcuXVheXl6tmn379rElS5Ywxhhbv349W79+Pff/kJAQFhoayuLi4hhjjMXGxrKtW7cyxhgrLS1lvr6+LCsri23atImFhIQwhULBcnNzWY8ePdiZM2dqnatLly4qP//8889s6NCh3PnefPNNNnr0aO7fxx9/XOsY2p4D5WPH2LPnUdmGiRMnsmvXrjHGGLt9+zbz8vJiMpmMLVmyhM2ePZtVVVWxBw8esNdff509ffpU5fmr/v8PP/yQpaSkMMYYy8zMZJ6enqyiokLjcao7c+YM69mzJ8vPz2elpaXs9ddf534XFi9ezLZt28bEYjELDAxkUqmUMcZYfHw8i4iI4O5bfHw8Y4yxtLQ0NmLECMYY454DxhhLTEzkfo/NEfVQG8CKFSswbdo0nDhxAqdOnYK/vz+ioqIwbNgwlf369OkDkUgEkUgEABg4cCAAwMXFBSUlJWqP3b59eyxZsgR79uxBTk4OLly4ABcXF257r169tLbNz88P69atg7+/P5KTk/Hdd99p3Dc3Nxf29vbo2bMnAGDYsGEIDw+HWCyGvb19rf0PHDiAmJgYbNq0CZaWlsjIyMC6deu4++rq6qq1bTXNnj0b48aNA2MMq1evBgCcOHECUqkUSUlJAACJRIKbN28iIyMD48ePh4WFBdq3b6/1cfDx8QEAyGQytG3bFt9++y23TZ+3/Lqeg5pKS0tx+fJlbkxUee6CggIAwODBg9GsWTO0bt0ajo6OEIvFGo+Tm5vL/R717NkTDg4OuHXrlsbjODg4qByja9euaNu2LQCgVatW3O9cu3btUFJSApFIhHXr1uHw4cPIzc3Fn3/+iW7dunH148ePB/DsHURoaCgeP36MIUOGYNasWfD29oaXlxfeeustrY9fY0aBamJ//PEHysrKMHz4cIwfPx7jx4/Hf//7XyQlJdUKVCsrK5WfLS11P11XrlzBggUL8Omnn+KDDz6AQCAAq7ZCo62trdb6Pn36oLi4GGlpaWjRooXWkKuqquKGApQYY5DL5bX2TUxMxK5du/Djjz+iY8eOAFCrViAQAADCwsJw7do1AMCaNWs0nl8sFuPp06cAgMePH6N169ZQKBSIiYnhhhiKiopgb2+PlJQUlcdB22NZfQzVELqeg5oUCgWsra1VzvvgwQM4OzsDADccBDx7zDQdS93t1Z8PfY5T83dO+Zwo3bt3DwEBAfj444/xzjvvwNnZGVlZWdz26o+rQqGAQCDA3LlzMWzYMBw/fhwxMTG4cuUKZs+erfY+NHZ0ld/EbGxsEBsby/U+GGO4fv06unbtapTjX7hwAf3794e/vz/c3Nxw/PhxVFVVaa0RCAQq+4wdOxarV6/G2LFjte7v7u6O4uJiXLp0CcCz2QsvvvgiHB0dVfbfv38/kpKSsHv3bi5MgWc9bmVP8sqVK7hz5w4AIDIyEgcOHMCBAwe0XmxZtWoV/P398emnn2LFihUAgAEDBuDnn38GYwyPHz/GmDFjkJ2djYEDB+LgwYNQKBTIz8/HxYsXtT4mdaHtOaj+WCv/b29vDzc3N+6xOH/+PMaOHav2D5OSQCCotV0kEsHV1RW//vorAODSpUt4+PAhunTpYrT7du3aNbi5ueHjjz9Gjx49cPToUZXfHeUMjtTUVLRv3x4tWrTAyJEjwRjDlClT8Mknn+D69etGa8/zhnqoJjZgwADMnTsXM2bMgEwmA2MMgwYNqtOV4+qGDx+OuXPnYujQoRAKhXjttde4q8GaDBo0CDExMbCzs8Pw4cMxYsQIbNy4Ee+//77a/b29vTFjxgwkJiYiLi4OkZGRkEqlsLe3R1xcXK394+LiYGFhgWnTpnG3RUREIDg4GKGhoRgxYgTc3d15veX/9ddfcevWLaxbtw7NmjXDwYMHsW/fPsybNw+rVq3CqFGjIJfLMXv2bHTr1g2dOnXC//7v/2LYsGFwcXExOGR27dqFo0ePqtw2b948vPfee9zP2p6D6o/du+++i/DwcHz11VeIiYnBypUrsWXLFggEAnz77bewtrbW2I5OnTpBIpEgJCQEEyZM4G5XHmfDhg2wsrLC+vXrtR6Hr7feegs7d+6El5cXhEIh+vbti5s3b3LbMzMzsW/fPgiFQkRHRwMAFixYgM8++wxWVlawsbHBypUrjdae540F0/ZehDRJ+/btQ2Zmpta324TUpJyh0q5du4ZuSoOhHipRMWfOHNy9exebNm1q6KYQ0uhQD5UQQoyELkoRQoiRUKASQoiRUKASQoiRUKASQoiRNPpAZYxBIpFo/SQKIYSYQqMP1NLSUrzxxhsoLS1t6KYQQpq4Rh+ohBDyvKBAJYQQIzHrQN2yZQt69eqFLVu2NHRTCCFNgFkHanx8PHJychAfH9/QTSGENAFmHahBQUHo0KGD0VZyIoQQbRr9Z/klEgneeOMNXLhwgVvZnhBCGoJZrjaVm5uL4uLiWre3bNkSbm5uRq8jhBDADAO1qKgIvXv3hkKhqLVNIBDg5s2bar9x0dA6QghRMrtAdXJywsWLF1FcXIybN29yq6N36dIFLVu21BiKhtYRQoiS2QUqgFpvz7t06cJ9M2d91BFCCGDmV/kJIcSUzKaHevfuXZSXl6vcpvzysOpfIgY8e3uv/EK4vLw8FBUV8a4jhJCazGbaVHFxMSoqKvSqEQqFOH/+PACgT5++qKgo11GhrLPB+fPnKFQJIWqZTQ+1srISAohgAYHW/RiqUFEh4XqlFRXlaG7ZAs0stD8UCiZHWcVTFBUVUaASQtQym0CtizL50zrV0/xVQghgZoFaBYlBdSIrJ1g2s9K6j1whg0RWVOt2mr9KCFEyq0DV9y1/zeC10OPYmvah+auEECWzClS+nJycIBTaQFxRu+epjlBoozYgaf4qIQQwo0C1trZGRYV+b/mFQiE3Ber8+XNqp01V72kq0bQpQog2ZhOox48fVzsPVVcwurq6agxJbT1NPvNXa56TEGKeTBaoCoUCYWFhyMnJgUgkQnR0NFq1asVtX7NmDa5cuQIrKyusWbMGHTp04HX8du3aaVy+z9hvwfPy8tDnjT6oqFQ/73XGjBm1bhNaC3H+wnmVUKXZAYSYF5MFampqKoRCIXbt2oWUlBQkJiYiNDQUAHDjxg38/fff2LNnD06dOoWEhAR8/fXXpmoaRxlwNXuaNQOuqKhIY5hqUlFZoTKHlWYHEGJ+TBaoFy9exKBBgwAAgwcPRmJiIrfNxcUF1tbWkMlkKC0thZ2dncbjVFZWorKykvtZIqk9bqpvMFanLuCUPU1NAddK6AxLC+3TrQBAzmR4XPFI5TaaHUCI+TFZoEokEu4tuZ2dHUpLS7ltjDFUVlbigw8+gEQiwYYNGzQe5z//+Q8SEhI0bjckGAHVgKupZsA5OTnBRmhTKyS1sVEzQ4BmBxBiXkwWqCKRiAvR0tJS2Nvbc9uSk5Ph6uqKH3/8EYWFhfjkk0+QlJQEoVBY6zgzZ87ElClTuJ8lEgneeecd7mc+wViTvuOWrq6uOMdjdoCyXa6urnQxixAzZrJA7dmzJ06ePAlvb2+kp6ejV69e3DYHBwfY29ujWbNmaNGiBeRyOeRyudpAtba2hrW1tdZz1fWCzpYtWxAfH4+goCBMnTpV7T6GzA7Iy8tD3z59Ua5hMRZ1F7NshDY4RwuyENIomCxQhw4divT0dPj7+8PKygpxcXGIiorCmDFjMHLkSJw7dw7+/v5QKBSYM2eO1nHU+lb966c1BaohioqKUF5RDjdRB9gKbHTuL60qR64khxZkIaSRMChQ5XI5LC35lQoEAnz11Vcqtymv8gNAZGSkIU2pF0FBQVwPVR/8L4Lpu2Ki+v1ouhUhzydeqXjr1i0sXrwYhYWF2L17N2bPno1169bB3d29vtrXIKZOnap3z9SQi2C5klyD20bTrQh5fvEK1IiICCxbtgwRERFo06YNPvnkEyxbtgy7du2qr/Y996pfBJs4cSLu3buHtm3b4pdfftF4Eewl27YQCrSPAwNARVUlCqT3NJ6PplsR8nzhFaglJSXo3bs39/Po0aOxefNmozeqsVG+zV64cCE3VKDuopRyulXNkNSGplsR0njwClQ7OzsUFBTAwuLZYnYZGRlqr8Q3VbqGCuoy3coY6jr2qs/sB0KaMl6BumzZMsyZMwe5ubkYPXo0SkpK8M0339RT08xTQy3GYoyxV76zH+jiGWlqeAVqt27d8N///hc5OTmoqqqCu7u7zjmhpO6MMX+1LmOvymD09fXF7t274evri0uXLvH+KK8SXTwj5opXoObn5+Onn37C06dPUf3LUmtOhyLGpZy/6m7XXu/5q7dKb9eav2rI2Ku6YIyLi0NcXJzeH+Wli2ekqeAVqCEhIfDw8ECfPn24cVRiOL7zV2+V3jZl8wDULRjp4hlpangFanl5OcLCwuqrLU2KIfNX+fZQjaWhgpEugpHGhlegdu/eHdevX0f37t3rqz1NhiGrW/EJSeV0q8a8GEtkZCSKiooQGRlZ74Gq7gIaXTwjfPEK1OvXr8PPz+//vtxOCMYYLCwskJaWVl/tM2v1vboVgCa5GAvfnq2mC2j6XDyjICbV8QpUbeuUkvplyHSrS5cuobyiHP1bvgoHS92LzZTIS3G2+C/uYlbN3q2perbKkJo8eTJ2796NCRMm6JxVUB3fnq3y3UJGRgavMeK6BDExT3oF6u+//w5PT0+cO3dO7XYXFxejNoo0PG1TtXT1bOsSxIbOKqgrNzc3rqep7xixoUEMUM/WXOkVqFevXoWnpyfOnj2rdruvr68x20SM7GzxX7xrlFO1+jp0gr2gudZ9xVVlOFfyDxeihgYxYJzpVmFhYbxWC6sLQ4KYerbmS69ADQ4OBqA631QikaCgoACdO3eun5YRjfhOt3pN5A47S1udxy2VS3FNckvltnMl//BqmzKIR7fqiRcs7bXuWygX4+DjS0aZL6uUm5uL3r17Y+vWrQDAa6jAVDT90ejXrx+FaSPHawx1z549uHjxIhYvXgxfX1/Y2dnh/fffx2effVZPzSM18ZlupZwdUDMktVHODlD2Nr0cPeBopf7ruZWeyCQ49uSK6o1Mj3nK+uzDA9+eX0ONEQPq/2g8T6FPDMMrULdv347NmzfjyJEj8Pb2RlhYGMaPH0+BakJ8plvVZTEWrkafzFOzz8EnmXoUqqprwPEZ08zLy0O/vn0hLddvaMLWxgYZ5+o+RlxXph57pfUY+OEVqNbW1mjTpg3S09MxceJEWFpaqnylMzENPr/Ihi7GouzdHnt8Re32mmr2bN926IKWAu0zC4qrSpFe8iyE8vLy0K9PX0j1HHu1FdogQ830Ln3HNIuKiiAtL8dHL3RCayvtwyEPZVLsLPx3jNjQIK4rU4+90noM/PEK1Hbt2iEoKAjZ2dl48803ERoaSmOoZkpd71bfZQZthDZcUOpSPYilFeV4374rWum4CPa4qgz/T/y3UaZ37SzkP0YsLS/HlHYd8aKN9iC+Xy7F1rvZGtupra3aeuCmGnul9Rj44xWoMTExOHnyJMLDwyEUCjFo0CAMHTq0vtpGGpim3q22np+hQazc//+J/+bVRkPfuivNf6kH2gm1jxHfrZDg24KrKrdtvZtttHaqa6umnq2hY6+GDhXQegz88H7Lf//+fSQnJ0Mmk6Ffv35o1qxZfbWNGBn/LxM0jCFBrMSnhwr822P0deiCFyy11xXKy5BccrNGL9GwQeKlnV7Dy7bahzTuSEvx1T/XVNo5o4s7XrLT3rMtKJUi8eYto/VsaZqW6fAK1OjoaNy+fRt+fn5gjGH//v24e/cuVqxYUV/tI0ZiyGIspuTk5ARboY3ePVTbGmO2yXoOMaicz8YG3xboN0Zsa6N6PkMl3tR/xgVgnJ5tQ0zTaqoXs3gF6smTJ5GcnMz1St99912MGjWqXhpGjIvP7ABj4dMjdnV1RUYdhgr49FC5850zbIzY1saG63nqUjOI+fRQgX97tqsHv44OLbQPTeQ8lWDFicu15vUCpp2m1ZQvZvEK1KqqKsjlcm6VfrlcDoFAUC8NI8Zn6AvIkKECQ3rEhgwVKHua+vZQlQFn6PnqGsT69lCN0SOu61CBoYzx7RA1NZaeLa9A9fHxQUBAAEaOHAkLCwscOnQII0eOrK+2keeAoUMFpuoR1yXg6nJOYwSxtrbW7IGvOHGZVxuNdRHMUMb6dgilxtKz1TtQHzx4AB8fH3Tr1g2nT5/GmTNn4Ofnh0mTJtVn+0gDq0swmqpHXJeLYNXP2VAX6wDdbeXzlh/4d6ggclh/uLdy0Fp363EJwn49a7SLYIYyh56tXoF68eJFzJs3DzExMRg8eDAGDx6M9evXY+PGjfDw8ECPHj3qu52kAZnyF7KuF88MCcaa53yeLtbVlXsrB3Rr46j3/nl5eejXry+kUj17trY2yMignq2SXoH69ddf4/vvv4eHhwd3W3BwMAYMGIDo6Gjs2LGj3hpImpa69IgNDUZN53weJq8rx4j1fctfc+w153GJzprq+xQVFUEqLUfUeC+4O7fUWnfrUTFC9xxr1D1bY9MrUMvKylTCVKlfv34oKdH9hBHCh6E94roEo6kveKgb1tA4+6EOY6/LflW/5KYu7s4t0d3FWe/9G2PPtj7oFahyuRwKhaLWJP6qqirIZLJ6aRghhnBzczPpl/sZY4gBeBY4fGc/ALqD48th/dFBxxhqzuOSWsF769ETrTU191H2bKOnjIL7Sy9orysoxOKth9RO72rs9ArU/v37IyEhgVsXVSkhIUFtz5WQhhQfH4+cnBzEx8fXa6A2hiEGZuA+oXt+N+yEBq5OZi70CtTPPvsM06dPR3JyMl555RUIhUJcu3YNzs7O2LhxY323kRBegoKCTLJivzIYf/zxR+67r3x8fJ6LIQbl2GuYnm/5a469Ro33hLuz9otZtx49qRW8i7cc4t3Whhp7rQ96BapIJMIvv/yCM2fOICsrC82aNcOkSZPQp0+f+m4fIbxNnTq13t/qK7m5uSE5ORn37t1DcnIyvvjiC5OcV5e6jr26OzvyGkNVCh41GO10XMy6+6gY6w+dAPAsTPv364cyqVTtvjXHXpvb2uJsRsZzG6p6z0O1sLDAwIEDMXDgwPpsDyGNjql6xID+F7OAuo293npUrLMt1fdxcnKCra0NF5S62Nr+2yMuk0qxbt5EdHRpo7UmO/8BFib88lyPvfL6pBQhpDZT9Yj5XsyqTt8gVgZj6J5jerVJGYyurq7IyDC8R9zRpQ1edW+n1zmfZxSohDQShl7M4hPEhgajstbQHnF2/gON2zTtw2fstWZblYz9CSsKVEIaEUNe5HyDuC7BaKiFCb/w2p/v2CtQe/y1Pj5hZbJAVSgUCAsLQ05ODkQiEaKjo9GqVStu+969e7F3715UVlbio48+wrhx40zVNELM3vO+UhOfMVQA3Nhr7JJZ6PSyi87j/3MnHyFrv1cZf62PT1iZLFBTU1MhFAqxa9cupKSkIDExEaGhoQCA27dvIykpCTt27IBCoUBiYqKpmkUIqQem+naITi+74LXOhh/P2J+wMlmgXrx4EYMGDQIADB48WCU0T58+DXd3d4SEhKC4uBgLFizQeJzKykqVb1qVSCT112hCmrj6XgvXyckJzW1t9X7L39zWVmW+bPade3rV6btfXZksUCUSCUSiZ8uP2dnZobS0lNv2+PFjXL9+HT/99BOKioowffp0pKSkwMKi9kcq/vOf/yAhIcFUzSakyTLFWriurq44m5Fh8OyAz9c+Xx8sMlmgikQiLkRLS0thb2/PbWvZsiX69esHW1tbtGvXDiKRCEVFRXjhhdqfCZ45cyamTJnC/SyRSPDOO+/U/x0gpIkx1Vq4dbkIFrdkNjq+3FbnObLv3OPC1xizAzQxWaD27NkTJ0+ehLe3N9LT09GrVy9uW+/evbF3717IZDKUlpaipKQEjo7qP/ZmbW3NfQULIaR+Pe8Xs5heqxX8u9+DBw8wfNgw3rMD7hUU6HUekwXq0KFDkZ6eDn9/f1hZWSEuLg5RUVEYM2YMXnnlFQwfPhwTJkwAACxdupS+q4oQopFy7DVk7fd61zS3ffbliGVSKb6NWIJOHV7WWfNPzh3MD1+r9zlMFqgCgQBfffWVym3Kq/wAMG3aNEybNs1UzSGENGJ8x14BqFzM6tThZfR4pbPR20UT+wkhjZIhY6/KQP0n945e59B3PyUKVELIc8NU81fnr9D/bTwfFKiEkOdCXb6gkW8Qf7t6CTq56TGGmnuHV/hSoBJCnguGTtMyKIj1mxyg/37/hwKVEPLcMObiL0DtIFbODuBz5V45O0AfFKiEkEZP3yA2dHaAvihQCSFNSn0uT9hM9y6EEEL0QT1UQkiTZexpWhSohJAmqS7TtDShQCWENEl1WU1LEwpUQkiTZezVtBp9oDL2bOYtrdxPCKlPdnZ2ahe9r67RB6py0WpaZJoQUp8uXLjAfeuIJhZM2cVrpBQKBR4+fKj2r4dyNf/jx4/rfCCojupMXdeY2kp1TaSH2qxZM7z44ota9xGJRLx/0amO6kxV1xDnpDrj1inRxH5CCDESClRCCDESsw5Ua2trzJs3j/eX+lEd1ZmiriHOSXXGraup0V+UIoSQ54VZ91AJIcSUKFAJIcRIKFAJIcRIKFAJIcRIGv3Efm0UCgV+++03XLp0CRKJBCKRCL1798Z7772n9RMPVNe469RJTU3FlStXsGDBAqqjulpWrVqFWbNmoU2bNgCAsrIyHDp0CGPGjOF15d+sA3X58uVo3rw53nrrLdjZ2UEsFuPkyZP4448/8OWXX1KdmdYBQEZGhsrPLVq0QFpaGl5//XUMGTKE6qhOxe+//46zZ89iw4YNcHNzg4WFBe7fv4/Fixfjm2++0VhXk1kHak5ODnbu3Klym7e3Nz766COqM+M6AEhKSqp12+uvv460tDStLyyqa5p1L7/8MhYuXIjg4GBs27YNrVq1wvz58xEQEKCxRh2zDlRra2scOXIE77zzDuzs7FBaWor09HTY29tTnRnXAcBXX32lcx+qo7rqPDw8EBISgunTp2Pt2rUQCASQy+W8jmHWE/sfP36MDRs2IDMzExKJBM2bN8eAAQMwc+ZMtGzZkurMtA4A1q5diyVLlmjdh+qoTumzzz7j3tpfvHgR33zzDSorK7Fs2TJ4eHjofyBmxqKioqiuCdYxxtjNmzepjur0VpffterMOlAby5NJdcatU7py5Qrbvn0727BhA9u+fTu7du1anY6nzQ8//MDKysq4nysqKtiJEyd01slkMnb8+HGV2zIzM/U6Z0hICEtKSuLTTKZQKFhqaiqTyWQsLy+P5efn612blZXFfvnlF7Zhwwa2a9cudvv2bV7nru7QoUNat8vlcpaUlMS2b9/OSkpKmFgsZjdu3NB53PLycvbPP/8wxhg7ceIE++GHH9ixY8d01tX1d03JrN/yG2rTpk2YNGkSbG1tAQCVlZXIyMjAoEGDtNbJ5XKcOnUKb7/9NnfbpUuX0LNnT53nXLBgAQYPHgxfX1+928kYQ1paGt59913cv38fzZo1Q9u2bfWqvXHjBjIzM1FcXIxWrVph4MCBePnll/U+d3WHDx/GyJEjNW6vqqrCoUOHIBaL4evrCwsLC+Tn56Nr165aj1tRUYG7d++iY8eO+PPPP3Hjxg107NgRnp6eOtsUFxeHmzdvqswQOHXqFLp27YrPP/9c7/t27Ngx3Lt3T+fFiYEDB6J169bYsmULnJycIJFIEBERAZFIhPDwcI11S5cuhZOTExYuXAjg2VSx4OBgtG3bFsuWLdN6zoCAAHh5eSEjIwOLFy+Gu7u7zvvzxRdf4OnTp4iKisLevXuxdetWODg4oFWrVti8ebPGui1btuDIkSMYMGAAMjIy0K1bN+Tk5MDX1xd+fn4a6zZv3oz27dujb9++aNGiBXd7YGAgduzYobFu+fLlsLe3h1gsxp07d2BtbQ2ZTIbOnTtj+fLlGuvmzZsHb29v5Ofn4/r163j77bdx6tQpvPDCC1qfBwC4evVqrSl6r776qtaaWowSy41EWloa27Fjh879BgwYwEaPHs0KCwsZY4yJxWK2aNEitmrVKq11oaGhLCYmhvu5qqqKzZ07l0VGRuo856RJk9jmzZvZzJkzWXZ2ts79GWMsPDyczZ8/n0mlUrZjxw7m5eXFfH192dSpU7XWbd68mY0dO5ZFR0ezDz/8kK1YsYIFBASwvXv3aq3btGkTS01NZcXFxSq3BwQEaK0LCwtjUVFRLCwsjAUGBrJPP/2UTZ48ma1evVpr3dy5c9n+/ftZfHw8mz17Ntu5cycLCgrS+Twwxtj48ePV3j5u3DitdXfv3lX5d/v2bTZy5Eh2+fJlrXWBgYHs2LFjbMKECSo9VX9/f611EydO5P4fHh7O/X/ChAla6xj793G/f/8+CwkJYbGxsay8vFxrTfXjKhQK9umnn6ocS5OxY8cyuVzOGHvW+w4KCmJyuVxnO3ft2sViYmLY2LFj2axZs9iFCxcYY88eL22qP27e3t5MJpMxxjQ/r0rK9gQEBHDtZYyxDz/8UGtdbGwsmzVrFtuxYwfbv38/+/HHH9nMmTNZbGys1rqazPoqf35+vsrPnTp1QlxcHDw8PLQONHfu3BlTpkzB3LlzsXXrVohEIkRHR+ucrnPnzh3uKuMXX3yBVatWISEhAf7+/jrbamFhgalTp2LEiBGIjo5Gu3btMGfOHAiFQo01f//9N3bt2gUAmDRpEv744w9s2rQJgYGBWs915MgR7NmzBwKBAJWVlVi4cCG2bduGSZMmae1tiEQiXLp0CRs3bkTr1q0xffp09O7dW+ck++zsbG7605AhQ/A///M/sLS0xIQJE7TWFRYWYsyYMQgMDMS2bdsgEAjg7++PcePGaa0Dnj2eNd8dXLlyBTY2Nlrrli5dCsaYyn1ydHTEt99+q7UHxxiDp6cnCgsLMW/ePHz99dcAoPMqsUKhgFQqRXl5OVJSUtCrVy+4uLjAyspKa93kyZNx48YN7rm2sLBAeno6jhw5gqNHj2qsEwgEuH//Pl588UXcuXMHMpmMq9dFLBajZcuWePToEYqLi1FVVaXz/lV/jv/66y989913cHBwQFlZmdY6xhgqKiogFoshkUjw+++/QyQS6ZxkLxKJcOPGDfTq1Qt//fUXPDw8kJOTAwcHB611Z86cwe7du1Vumzx5MsaPH8/rHY1ZByq9ODQz5xcHAERFRWHt2rXIzs6GQqEAAHTv3h2RkZFa67Zv367z2OooH/Nx48bBzs4OEydOhFwuR2hoqNa6adOmYcyYMWCMYf369UhLS8Pp06cRFRWltW7t2rX4/PPPsXbtWl7tXLx4MaZPnw4bGxuIxWJER0cDANq3b6+1bv78+fD394eDgwMkEgmioqKQmJio15xgpVdffRUbNmzA0aNHcezYMa37Tpo0CaNHj4alpSU2btyI5ORkSCQSrFmzRmvdl19+iXXr1qGgoADbtm3DSy+9hA4dOmDlypVa6wz9A1zrOIzRGGpNkydP5l5YKSkpSEhI4F4cXl5eGuvS0tIQExMDxhhWrlyJtLQ0iMViBAcHw8XFRWNdQUEBPv/8c8TGxtbapm1M9PLly1i+fLnKi8PDwwMrVqzA6tWrNdalp6fjyy+/VHlxpKen46WXXtLaQ1Xn6NGjWLZsWa1PqFR36NAhJCQkwNLSEmvWrOFeHMHBwVpfyA8fPuReHJcvX+ZeHGFhYXB1deXVzvpW8w83H5WVlVAoFLxfvFKplBvn5+vJkydwdHTkVcMYw5MnT9CqVSuDzsmXVCqFQCCo86LP+sjNzVX7B3jRokW8ftcoUNWgF4fxmfLFQUhDoUAlhBAjoeX7CCHESChQCSHESChQCSHESChQCSHESChQCSHESChQCSHESChQCSHESChQyXPp6tWrCAsL02u7WCzG3Llz9ao7e/Ys9xFfXftqwud8pGkx68/yk8arR48e6NGjh17bnz59iqysLL3q+JxDE0PPR8wf9VCJyT18+BDBwcHw8fHBG2+8ga5du6Jr167IzMzk9lH2JM+ePYtPPvkEQUFBGD58OKZMmYLi4mKVnmZERAQePnyIWbNmcbfL5XIsX74cEyZMwJAhQzBnzhxIpVKVdij33b59O3x8fODj44Nhw4aha9euyM3N1XgMdedTSkxMxPDhwzFq1ChERUWhqqpK430g5ocClZhccHAw3njjDRw4cABHjx6FSCTCr7/+il69eqnd//Lly1i6dClSUlJgY2ODQ4cOqWwPDw9H69at8f3333O3ZWZmQiAQYPfu3UhNTeW+4E+dyZMn48CBAzhw4AC6d++O2bNnw83NTeMx1J0PeLbozG+//YZ9+/YhKSkJt2/f5pZX1HUfiHmgt/zEpC5dugSxWIyPP/4YwLPlFF1cXPDkyRONNZ07d+ZW3erWrRuePn2q8zx9+/aFvb09fv75Z2RnZ+PWrVsoKyvT+uV+GzduRGlpKebPn6/1GJqcPn0aI0eO5Ba58fPzQ3JyMjp16mTQfSCND/VQiUllZWWhW7du3M9isRgFBQV45ZVXNNZUX2TbwsIC+qznc/ToUSxatAi2trbw8/ND3759tdalpqbi8OHDWLduHbfSGN9jKBSKWquUKdeYNeQ+kMaHApWYlKOjI/7++29UVlZCJpMhIiICEyZMgJ2dncHHtLS0rLU49pkzZzB8+HCMHTsWDg4OyMjIQFVVldr6GzduIDIyEgkJCRCJRDqPoe58ADBgwAAcOnQIUqkUcrkc+/btQ9++fQ2+X6Txobf8xKS8vLxw/PhxjBo1CnZ2dvD29sasWbPqdExnZ2e4uLhg4sSJ3Nv1cePGISQkBAcOHEDz5s3Ru3dv3L17V+0XEUZHR0Mul2PBggVc6IaEhGg8hrrzAYCnpyeysrLw4YcfQi6X480330RgYKDKxTZi3mg9VEIIMRJ6y08IIUZCgUoIIUZCgUoIIUZCgUoIIUZCgUoIIUZCgUoIIUZCgUoIIUZCgUoIIUZCgUoIIUZCgUoIIUZCgUoIIUby/wH/q7VimYD+bQAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 350x175 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 350x175 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "tmpdf = corr_vals.groupby(['PE','Puzzle']).mean(numeric_only=True)\n",
    "plt.figure(figsize=(3.5,1.75))\n",
    "ax = sns.boxplot(data=tmpdf,x=\"PE\",y=\"Cosine\",hue=\"PE\",fliersize=1,palette='rocket')\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(np.arange(0,len(initializations)), initializations, rotation=-90,fontsize=7)\n",
    "plt.xlabel(r\"$\\sigma$ initialization\",fontsize=8)\n",
    "plt.title(f\"Similarity to 2d-fixed PE attention maps\",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/validationXattnsimilarity_learnablePE/'\n",
    "if not os.path.exists(outputdir):\n",
    "    os.makedirs(outputdir)\n",
    "# plt.savefig(f'{outputdir}attn_similarity_learnablePEs.pdf',transparent=True,dpi=300)\n",
    "\n",
    "\n",
    "plt.figure(figsize=(3.5,1.75))\n",
    "ax = sns.boxplot(data=tmpdf,x=\"PE\",y=\"JSD\",hue=\"PE\",fliersize=1,palette='rocket')\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(np.arange(0,len(initializations)), initializations, rotation=-90,fontsize=7)\n",
    "plt.xlabel(r\"$\\sigma$ initialization\",fontsize=8)\n",
    "plt.title(f\"Similarity to 2d-fixed PE attention maps\",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_learnablePEs_JSD.pdf',transparent=True,dpi=300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "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></th>\n",
       "      <th>Cosine</th>\n",
       "      <th>Layer</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>PE</th>\n",
       "      <th>Puzzle</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th rowspan=\"5\" valign=\"top\">learn-0.1</th>\n",
       "      <th>0</th>\n",
       "      <td>0.844850</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>0.849153</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>0.850401</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>0.813054</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>0.842470</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>...</th>\n",
       "      <th>...</th>\n",
       "      <td>...</td>\n",
       "      <td>...</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th rowspan=\"5\" valign=\"top\">learn-2.0</th>\n",
       "      <th>103</th>\n",
       "      <td>0.583553</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>104</th>\n",
       "      <td>0.579322</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>105</th>\n",
       "      <td>0.566698</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>106</th>\n",
       "      <td>0.615292</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>107</th>\n",
       "      <td>0.598791</td>\n",
       "      <td>2.5</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>2160 rows × 2 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "                    Cosine  Layer\n",
       "PE        Puzzle                 \n",
       "learn-0.1 0       0.844850    2.5\n",
       "          1       0.849153    2.5\n",
       "          2       0.850401    2.5\n",
       "          3       0.813054    2.5\n",
       "          4       0.842470    2.5\n",
       "...                    ...    ...\n",
       "learn-2.0 103     0.583553    2.5\n",
       "          104     0.579322    2.5\n",
       "          105     0.566698    2.5\n",
       "          106     0.615292    2.5\n",
       "          107     0.598791    2.5\n",
       "\n",
       "[2160 rows x 2 columns]"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "tmpdf"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\\begin{tabular}{rrr}\n",
      "\\toprule\n",
      "PE & Cosine & SD \\\\\n",
      "\\midrule\n",
      "0.100 & 0.836 & 0.018 \\\\\n",
      "0.200 & 0.838 & 0.016 \\\\\n",
      "0.300 & 0.808 & 0.013 \\\\\n",
      "0.400 & 0.779 & 0.013 \\\\\n",
      "0.500 & 0.748 & 0.014 \\\\\n",
      "0.600 & 0.722 & 0.016 \\\\\n",
      "0.700 & 0.715 & 0.016 \\\\\n",
      "0.800 & 0.710 & 0.016 \\\\\n",
      "0.900 & 0.694 & 0.015 \\\\\n",
      "1.000 & 0.685 & 0.015 \\\\\n",
      "1.100 & 0.697 & 0.015 \\\\\n",
      "1.200 & 0.692 & 0.017 \\\\\n",
      "1.300 & 0.685 & 0.016 \\\\\n",
      "1.400 & 0.672 & 0.016 \\\\\n",
      "1.500 & 0.657 & 0.017 \\\\\n",
      "1.600 & 0.647 & 0.022 \\\\\n",
      "1.700 & 0.642 & 0.016 \\\\\n",
      "1.800 & 0.624 & 0.021 \\\\\n",
      "1.900 & 0.606 & 0.023 \\\\\n",
      "2.000 & 0.577 & 0.019 \\\\\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'] = initializations\n",
    "\n",
    "df_to_latex['Cosine'] = meandf.Cosine.values\n",
    "df_to_latex['SD'] = sddf.Cosine.values\n",
    "\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": [
    "# Correlation with similarity to 2d absolute PE with generalization performance"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Load generalization performances"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "df = pd.DataFrame()\n",
    "for init in initializations:\n",
    "      pestr = 'learn'\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-learn-{init}_\" \\\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",
    "\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 + '-' + str(init)\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",
    "models = df_ds.model.unique()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 175x175 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "df_similarity = corr_vals.groupby(['PE']).mean(numeric_only=True).reset_index()\n",
    "\n",
    "head = 1\n",
    "df_performance = 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_performance = df_performance.groupby('pe').mean(numeric_only=True).reset_index()\n",
    "\n",
    "\n",
    "rho, p = stats.spearmanr(df_similarity.Cosine.values,df_performance.accuracy.values)\n",
    "plt.figure(figsize=(1.75,1.75))\n",
    "ax = sns.regplot(x=df_similarity.Cosine.values,y=df_performance.accuracy.values,scatter=True,\n",
    "                 color='k',scatter_kws={'s':3})\n",
    "plt.text(x=.7,y=0.4,s=r\"$\\rho =$ \" + (str(round(rho,3))),fontsize=8)\n",
    "sns.despine()\n",
    "plt.xlabel('Cosine similarity to\\nground truth attn',fontsize=8)\n",
    "plt.ylabel('Accuracy',fontsize=8)\n",
    "plt.title('Generalization vs. \\nground truth similarity',fontsize=8)\n",
    "# plt.title('Validation performance\\nand cosine similarity to ground truth model')\n",
    "plt.xticks(fontsize=7)\n",
    "plt.yticks(fontsize=7)\n",
    "plt.ylim([0.25,1.1])\n",
    "plt.tight_layout()\n",
    "outputdir = '../figures/manuscript_figures/validationXattnsimilarity_learnablePE/'\n",
    "if not os.path.exists(outputdir):\n",
    "    os.makedirs(outputdir)\n",
    "plt.savefig(f'{outputdir}performanceXattnsimilarity.pdf',transparent=True,dpi=300)"
   ]
  }
 ],
 "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
}
