{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "cd89acf9-7787-4e95-9cf1-f30aba767cd2",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "8a120102",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "import argparse\n",
    "import numpy as np\n",
    "import os\n",
    "import pandas as pd\n",
    "import scipy as sp\n",
    "import sys\n",
    "import torch\n",
    "import torch.nn.functional as F\n",
    "import warnings\n",
    "import random\n",
    "import collections\n",
    "\n",
    "# CD-T Imports\n",
    "import tqdm\n",
    "import seaborn as sns\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.cm as cm\n",
    "import matplotlib.colors as mcolors\n",
    "import pickle\n",
    "import json\n",
    "\n",
    "from torch import nn\n",
    "\n",
    "warnings.filterwarnings(\"ignore\")\n",
    "\n",
    "base_dir = os.path.split(os.getcwd())[0]\n",
    "sys.path.append(base_dir)\n",
    "\n",
    "from argparse import Namespace\n",
    "from methods.bag_of_ngrams.processing import cleanReports, cleanSplit, stripChars\n",
    "from pyfunctions.general import extractListFromDic, readJson, combine_token_attn, compute_word_intervals\n",
    "from pyfunctions.pathology import extract_synoptic, fixLabelProstateGleason, fixProstateLabels, fixLabel, exclude_labels\n",
    "from pyfunctions.cdt_from_source_nodes import *\n",
    "from pyfunctions.cdt_source_to_target import *\n",
    "from sklearn import preprocessing\n",
    "from sklearn.model_selection import train_test_split\n",
    "from torch.utils.data import DataLoader, RandomSampler, SequentialSampler, TensorDataset\n",
    "from transformers import AutoTokenizer, AutoModel\n",
    "from transformers import BertTokenizer, BertForSequenceClassification"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "f6d6ada4-4781-4789-b3b1-1d044c11b3d3",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<torch.autograd.grad_mode.set_grad_enabled at 0x75347c2b6d80>"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "torch.autograd.set_grad_enabled(False)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e7651660-7f59-4b59-a574-afecc52dc306",
   "metadata": {
    "user_expressions": []
   },
   "source": [
    "## Model Arguments"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "c3183be1-3bf6-4f5a-8134-9bdd83db0a56",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "args = {\n",
    "    'model_type': 'bert', # bert, medical_bert, pubmed_bert, biobert, clinical_biobert\n",
    "    'task': 'path',\n",
    "    'field': 'PrimaryGleason'\n",
    "}\n",
    "\n",
    "device = 'cpu'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "8053ccd9",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "if args['model_type'] == 'bert':\n",
    "    bert_path = 'bert-base-uncased'\n",
    "    tokenizer = BertTokenizer.from_pretrained('bert-base-uncased')\n",
    "elif args['model_type'] == 'medical_bert':\n",
    "    bert_path = f\"{base_dir}/models/pretrained/bert_pretrain_output_all_notes_150000/\"\n",
    "    tokenizer = BertTokenizer.from_pretrained(bert_path, local_files_only=True)\n",
    "elif args['model_type'] == 'pubmed_bert':\n",
    "    bert_path = \"microsoft/BiomedNLP-PubMedBERT-base-uncased-abstract\"\n",
    "    tokenizer = AutoTokenizer.from_pretrained(\"microsoft/BiomedNLP-PubMedBERT-base-uncased-abstract\")\n",
    "elif args['model_type'] == 'pubmed_bert_full':\n",
    "    bert_path = \"microsoft/BiomedNLP-PubMedBERT-base-uncased-abstract-fulltext\"\n",
    "    tokenizer = AutoTokenizer.from_pretrained(\"microsoft/BiomedNLP-PubMedBERT-base-uncased-abstract-fulltext\")\n",
    "elif args['model_type'] == 'biobert':\n",
    "    bert_path = \"dmis-lab/biobert-v1.1\"\n",
    "    tokenizer = AutoTokenizer.from_pretrained(\"dmis-lab/biobert-v1.1\")\n",
    "elif args['model_type'] == 'clinical_biobert':\n",
    "    bert_path = \"emilyalsentzer/Bio_ClinicalBERT\"\n",
    "    tokenizer = AutoTokenizer.from_pretrained(\"emilyalsentzer/Bio_ClinicalBERT\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a836de3c-46aa-4c5f-a16a-7c1b8e832c71",
   "metadata": {
    "user_expressions": []
   },
   "source": [
    "## Load data"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "502827dc-85fc-43d1-8d70-0f0210ffcfe0",
   "metadata": {
    "user_expressions": []
   },
   "source": [
    "you can cutomize the code here to read in your own data."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "4ea84464",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Token indices sequence length is longer than the specified maximum sequence length for this model (1345 > 512). Running this sequence through the model will result in indexing errors\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "2066 517 324\n"
     ]
    }
   ],
   "source": [
    "# Read in data\n",
    "#field = 'PrimaryGleason' # out of PrimaryGleason, SecondaryGleason', 'MarginStatusNone', 'SeminalVesicleNone'\n",
    "path = f\"../data/prostate.json\"\n",
    "data = readJson(path)\n",
    "\n",
    "# Clean reports\n",
    "data = cleanSplit(data, stripChars)\n",
    "data['dev_test'] = cleanReports(data['dev_test'], stripChars)\n",
    "data = fixLabel(data)\n",
    "\n",
    "train_documents = [extract_synoptic(patient['document'].lower(), tokenizer) for patient in data['train']]\n",
    "val_documents = [extract_synoptic(patient['document'].lower(), tokenizer) for patient in data['val']]\n",
    "test_documents = [extract_synoptic(patient['document'].lower(), tokenizer) for patient in data['test']]\n",
    "print(len(train_documents), len(val_documents),len(test_documents))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "5df4aea9",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "# Create datasets\n",
    "train_labels = [patient['labels'][args['field']] for patient in data['train']]\n",
    "val_labels = [patient['labels'][args['field']] for patient in data['val']]\n",
    "test_labels = [patient['labels'][args['field']] for patient in data['test']]\n",
    "\n",
    "train_documents, train_labels = exclude_labels(train_documents, train_labels)\n",
    "val_documents, val_labels = exclude_labels(val_documents, val_labels)\n",
    "test_documents, test_labels = exclude_labels(test_documents, test_labels)\n",
    "\n",
    "le = preprocessing.LabelEncoder()\n",
    "le.fit(train_labels)\n",
    "\n",
    "# Map raw label to processed label\n",
    "le_dict = dict(zip(le.classes_, le.transform(le.classes_)))\n",
    "le_dict = {str(key):le_dict[key] for key in le_dict}\n",
    "\n",
    "for label in val_labels + test_labels:\n",
    "    if str(label) not in le_dict:\n",
    "        le_dict[str(label)] = len(le_dict)\n",
    "\n",
    "# Map processed label back to raw label\n",
    "inv_le_dict = {v: k for k, v in le_dict.items()}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "d23e85ca",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "documents_full = train_documents + val_documents + test_documents\n",
    "labels_full = train_labels + val_labels + test_labels"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "a6ffd4d3-5e8f-4587-bb2a-f06b61918c09",
   "metadata": {
    "user_expressions": []
   },
   "source": [
    "## Load Trained Models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "dd45f9a1",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "BertForSequenceClassification(\n",
       "  (bert): BertModel(\n",
       "    (embeddings): BertEmbeddings(\n",
       "      (word_embeddings): Embedding(30522, 768, padding_idx=0)\n",
       "      (position_embeddings): Embedding(512, 768)\n",
       "      (token_type_embeddings): Embedding(2, 768)\n",
       "      (LayerNorm): LayerNorm((768,), eps=1e-12, elementwise_affine=True)\n",
       "      (dropout): Dropout(p=0.1, inplace=False)\n",
       "    )\n",
       "    (encoder): BertEncoder(\n",
       "      (layer): ModuleList(\n",
       "        (0-11): 12 x BertLayer(\n",
       "          (attention): BertAttention(\n",
       "            (self): BertSdpaSelfAttention(\n",
       "              (query): Linear(in_features=768, out_features=768, bias=True)\n",
       "              (key): Linear(in_features=768, out_features=768, bias=True)\n",
       "              (value): Linear(in_features=768, out_features=768, bias=True)\n",
       "              (dropout): Dropout(p=0.1, inplace=False)\n",
       "            )\n",
       "            (output): BertSelfOutput(\n",
       "              (dense): Linear(in_features=768, out_features=768, bias=True)\n",
       "              (LayerNorm): LayerNorm((768,), eps=1e-12, elementwise_affine=True)\n",
       "              (dropout): Dropout(p=0.1, inplace=False)\n",
       "            )\n",
       "          )\n",
       "          (intermediate): BertIntermediate(\n",
       "            (dense): Linear(in_features=768, out_features=3072, bias=True)\n",
       "            (intermediate_act_fn): GELUActivation()\n",
       "          )\n",
       "          (output): BertOutput(\n",
       "            (dense): Linear(in_features=3072, out_features=768, bias=True)\n",
       "            (LayerNorm): LayerNorm((768,), eps=1e-12, elementwise_affine=True)\n",
       "            (dropout): Dropout(p=0.1, inplace=False)\n",
       "          )\n",
       "        )\n",
       "      )\n",
       "    )\n",
       "    (pooler): BertPooler(\n",
       "      (dense): Linear(in_features=768, out_features=768, bias=True)\n",
       "      (activation): Tanh()\n",
       "    )\n",
       "  )\n",
       "  (dropout): Dropout(p=0.1, inplace=False)\n",
       "  (classifier): Linear(in_features=768, out_features=3, bias=True)\n",
       ")"
      ]
     },
     "execution_count": 17,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "#load finetuned model\n",
    "model_path = f\"{base_dir}/PG_best_ckpts/bert\" # /{args['model_type']}_{args['field']}\"\n",
    "checkpoint_file = f\"{model_path}/save_output\"\n",
    "# config_file = f\"{model_path}/save_output/config.json\"\n",
    "\n",
    "model = BertForSequenceClassification.from_pretrained(checkpoint_file, num_labels=len(le_dict), output_hidden_states=True)\n",
    "\n",
    "model = model.eval()\n",
    "model.to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "3f772304",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "BertForSequenceClassification(\n",
       "  (bert): BertModel(\n",
       "    (embeddings): BertEmbeddings(\n",
       "      (word_embeddings): Embedding(30522, 768, padding_idx=0)\n",
       "      (position_embeddings): Embedding(512, 768)\n",
       "      (token_type_embeddings): Embedding(2, 768)\n",
       "      (LayerNorm): LayerNorm((768,), eps=1e-12, elementwise_affine=True)\n",
       "      (dropout): Dropout(p=0.1, inplace=False)\n",
       "    )\n",
       "    (encoder): BertEncoder(\n",
       "      (layer): ModuleList(\n",
       "        (0-11): 12 x BertLayer(\n",
       "          (attention): BertAttention(\n",
       "            (self): BertSdpaSelfAttention(\n",
       "              (query): Linear(in_features=768, out_features=768, bias=True)\n",
       "              (key): Linear(in_features=768, out_features=768, bias=True)\n",
       "              (value): Linear(in_features=768, out_features=768, bias=True)\n",
       "              (dropout): Dropout(p=0.1, inplace=False)\n",
       "            )\n",
       "            (output): BertSelfOutput(\n",
       "              (dense): Linear(in_features=768, out_features=768, bias=True)\n",
       "              (LayerNorm): LayerNorm((768,), eps=1e-12, elementwise_affine=True)\n",
       "              (dropout): Dropout(p=0.1, inplace=False)\n",
       "            )\n",
       "          )\n",
       "          (intermediate): BertIntermediate(\n",
       "            (dense): Linear(in_features=768, out_features=3072, bias=True)\n",
       "            (intermediate_act_fn): GELUActivation()\n",
       "          )\n",
       "          (output): BertOutput(\n",
       "            (dense): Linear(in_features=3072, out_features=768, bias=True)\n",
       "            (LayerNorm): LayerNorm((768,), eps=1e-12, elementwise_affine=True)\n",
       "            (dropout): Dropout(p=0.1, inplace=False)\n",
       "          )\n",
       "        )\n",
       "      )\n",
       "    )\n",
       "    (pooler): BertPooler(\n",
       "      (dense): Linear(in_features=768, out_features=768, bias=True)\n",
       "      (activation): Tanh()\n",
       "    )\n",
       "  )\n",
       "  (dropout): Dropout(p=0.1, inplace=False)\n",
       "  (classifier): Linear(in_features=768, out_features=3, bias=True)\n",
       ")"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model.to(device)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9c47f12e-74aa-4c25-a2a6-1230a32bfe60",
   "metadata": {
    "tags": [],
    "user_expressions": []
   },
   "source": [
    "# Test systematic bias and collect head activations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "fd4b17cf-c841-48e5-9ebf-86be83c9c66f",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 1/1 [00:04<00:00,  4.15s/it]\n"
     ]
    }
   ],
   "source": [
    "# variance of head output across samples\n",
    "# size of head output (context) -> torch.Size([1, 512, 768])\n",
    "\n",
    "with torch.no_grad():\n",
    "    NUM_SAMPLES = 1\n",
    "    POS = 0\n",
    "    index_lst = random.sample(range(0, len(documents_full)), NUM_SAMPLES)\n",
    "    docs = [documents_full[i] for i in index_lst]\n",
    "    \n",
    "    collect = []\n",
    "    for doc in tqdm.tqdm(docs):\n",
    "        encoding = get_encoding(doc, tokenizer, device)\n",
    "        _, _, _, context_lst = prop_classifier_model_patched(encoding, model, device, [], output_context=True)\n",
    "        context = np.stack(context_lst)\n",
    "        context_at_pos = context[:, :, POS, :, :].transpose(1,0,2,3).squeeze(0) #[layer, head, d]\n",
    "        collect.append(context_at_pos)\n",
    "    collect = np.stack(collect)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "797c5182",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "(200, 12, 1, 512, 12, 64)"
      ]
     },
     "execution_count": 33,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "collect.shape\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "02a71c6a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "(12, 512, 12, 64)\n"
     ]
    }
   ],
   "source": [
    "mean_acts = np.mean(collect.squeeze(2), axis=0)\n",
    "print(mean_acts.shape)\n",
    "path = f\"{base_dir}/output/\"\n",
    "\n",
    "with open(os.path.join(path, f\"{args['model_type']}_mean_acts_random_500.pkl\"), 'wb') as handle:\n",
    "    pickle.dump(mean_acts, handle)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 86,
   "id": "e5857269-5c7f-463c-9a5d-6c8e6034a155",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "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zUruAKh0evfujffffx+rV6/GG2+8gV27duHWrVuIiYnRnLNw4UJ8+OGHGDFiBGbMmAFnZ2dYWlrik08+0ZkEuLm5FavvyMhILFmyBBMmTEBwcDCcnJxgYWGBYcOG4enTp/nOr1q1qta/lUolAGjO/eeffwCg0ImEeWXucePGYdy4cTrPuX37doGPv3PnDlxdXfN9sFevXh1WVla4c+dOgY8tzKpVqyCEQO/evbUm6r322mvYsGEDLly4UODwUlHu3LmD7OxsfP755/j88891nvPicy7qtS5NDLreF+7u7prjUvbv4+ODgICAfPsdHR21LuW8desWhBAFXhFVp04dALlXcISEhODmzZv45JNP0KRJE9jb20OtVqNVq1aauO7duwe1Wg1XV9d8bena9yIPDw8AuclmQTIyMnD79m00b968yPZKysrKKt9rnxd3ad/bRGXB5MJE2dnZ4a233sJ///tfpKamYtWqVXBwcMCbb76pOWf9+vVo164dYmNjtR776NEjnW0W9xvV+vXrMXjwYMyePVtr/+3btzXf3ksi70qU5+dXvMjZ2RkAEBUVpZkj8aIGDRoU+PiqVaviyJEj+b45pqenIzs7W9N+SajVas1cj4JiWrVqVbEuZdXFyckJlpaWGDRoECIiInSe4+XlVaq2i6tq1apITU3Nt//mzZsAUKrXTQrOzs5QKBQ4cOCAJoF5Xt6+M2fO4OTJk1izZg3CwsI0xy9fvqx1vpOTExQKBdLS0vK1pWvfi/z9/eHk5IQdO3YgOjpa5+/Sjh07oFar0blzZ80+W1tbqFSqfOfevn27RK9tdnY27ty5o5Vg5MWdty+vIvNif0w+SB84LGLChg4dipycHMybNw+7du1Cv379UKFCBc1xhUKR74P31KlThU7gKw5d7f7www+4ceNGqdpr3bo1HB0dsWzZMp0T94DcxKF+/fo4efIkAgICdG4ODg4F9tGxY0c8fvwY27dv19q/bt06zfGS+umnn/D3338jIiIC+/bty7c1atQI69atQ3Z2NoDcP3i6vsEX9O2+QoUKaN++PZKSktC0aVOdz/nFb6vFUVAcunTs2BF79+7VJBN51q1bhwoVKqBVq1Yl7l8K3bt3hxACN27c0Pm6NGnSBMD/EuYX36/Lly/X+re9vT1efvllbN26FZmZmZr9jx49ws6dO4uMx8bGBuPHj8f5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oFQqMWPGDAQHB8POzg4AcPHiRdSoUaPIdpRKJZRKpda+J8+EXmImIiIqDRMqOkhCtuRi5syZSE1NRWhoKHJychAYGIj169drjisUCkRHc3iDiIjKATPLLmRLLipWrIi4uDhkZmYiOzsbFStW1DoeEhIiU2RERERUFrIvomVrayt3CERERHplSld6SEH25IKIiKi8M6UrPaTA5IKIiEjPzCy34I3LiIiISFqsXBAREembmZUumFwQERHpmblN6OSwCBEREUmKlQsiIiI949UiREREJCkzyy04LEJERETSKpeVixt3M+UOAe5O8q88mqOW/wZuCiOpBR64fFvuENDZx0XuEPA4M1vuEJD5LEfuEOBgWy4/+kpMLeT/jHCtLP9npUHI+FG4dOlSzJs3D6mpqWjUqBEWLVqEoKAgneempqbiww8/RGJiIpKTkzF69GgsWrSoxH2yckFERKRnCon+V1JxcXEYM2YMJk+ejKSkJAQFBaFLly5ISUnReb5KpUK1atUwefJk+Pr6lvr5MrkgIiIqpxYuXIihQ4di2LBh8PHxwaJFi1CrVi3ExsbqPL927dpYvHgxBg8eDEdHx1L3y9ogERGRnkk1QqxSqaBSqbT2KZVKKJXKfOdmZWUhMTEREydO1NofEhKChIQEaQIqACsXREREeqaQaIuOjoajo6PWFh0drbPP27dvIycnBy4u2vO9XFxckJaWJv2TfA4rF0RERPomUeUiKioKkZGRWvt0VS20un6hbCKE0PtkeyYXREREJqKgIRBdnJ2dYWlpma9KkZ6enq+aITUOixAREemZHFeL2NjYwN/fH/Hx8Vr74+Pj0bp1aymfXj6sXBAREemZXEv+REZGYtCgQQgICEBgYCBWrFiBlJQUjBgxAkDuMMuNGzewbt06zWNOnDgBAHj8+DH++ecfnDhxAjY2NmjYsGGx+2VyQUREVE717dsXd+7cwfTp05GamorGjRtj165d8PT0BJC7aNaLa140b95c89+JiYnYuHEjPD098ddffxW7X4UQRrBEm8SSbz2VOwSjWKEzO0f+H+1TI1iNEQCOXrsrdwhcofNfxrBCp52NpdwhwF4p/3e7rGy13CEYxUrCThX0/374M12av0t1q9tJ0o6+yf7uzsjIwMaNG5GQkIC0tDQoFAq4uLigTZs2eOutt2Bvby93iERERGVjHHdCMBhZ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",
      "text/plain": [
       "<Figure size 640x480 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "var_context = np.var(collect, axis=0) #collect = [N, layer, head, d]\n",
    "avg_var = np.mean(var_context, axis=-1)\n",
    "\n",
    "s = sns.heatmap(avg_var, xticklabels = range(12), yticklabels = range(12), cmap='Blues', cbar_kws={'label': ''})\n",
    "s.set(xlabel='Head', ylabel='Layer')\n",
    "plt.title(\"Variance of Attention Head Output\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "cc3b9435",
   "metadata": {
    "user_expressions": []
   },
   "source": [
    "## Head to logits direct influence"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "78272157",
   "metadata": {},
   "outputs": [],
   "source": [
    "doc = documents_full[0]\n",
    "label = labels_full[0]\n",
    "encoding = get_encoding(doc, tokenizer, device)\n",
    "# label_idx = le_dict[label]\n",
    "    \n",
    "source_nodes = [(0, 0, 0)]\n",
    "\n",
    "rel, irrel, _, _ = prop_classifier_model_patched(encoding, model, device, source_nodes,\n",
    "                                                             mean_acts=None,\n",
    "                                                             set_irrel_to_mean=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "d7e5d04d",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Linear(in_features=768, out_features=3, bias=True)"
      ]
     },
     "execution_count": 27,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "rel.shape\n",
    "irrel.shape\n",
    "model.classifier"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "7a2c23c9-991a-4033-aaf2-e0fc633ac176",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def patch_h_to_logits_at_pos(encoding, model, pos, label_idx, device, mean_act=None, mean_ablated=False):\n",
    "    # patch one node at a time\n",
    "    raw_logit = model(**encoding)[0][0][label_idx] # == raw_irrel\n",
    "    h_ctbn_lst = []\n",
    "    for level in range(1):\n",
    "        h_ctbn_lst.append([])\n",
    "        for h in range(1):\n",
    "            patched_entry = [(level, pos, h)]\n",
    "            rel, irrel, _, _ = prop_classifier_model_patched(encoding, model, device, patched_entry,\n",
    "                                                             mean_acts=None,\n",
    "                                                             set_irrel_to_mean=False)\n",
    "            rel = rel[0][label_idx] # head ctbn\n",
    "            h_ctbn_lst[level].append(rel / abs(raw_logit) * 100)\n",
    "\n",
    "        h_ctbn_lst[level] = torch.stack(h_ctbn_lst[level])\n",
    "    h_ctbn_tot = torch.stack(h_ctbn_lst).cpu().numpy()\n",
    "    return h_ctbn_tot"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "b23ede8c-20b9-411d-a67d-40acc74f9d4c",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "# load in the mean (pre-collected) to do mean ablation\n",
    "path = f\"{base_dir}/output/{args['task']}/{args['model_type']}_{args['field']}/h_to_logits\"\n",
    "os.makedirs(path, exist_ok=True)\n",
    "\n",
    "with open(os.path.join(path, f\"mean_head_out_res_500.pkl\"), 'rb') as handle:\n",
    "    back = pickle.load(handle)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "e2bd5a04",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "1it [00:05,  5.69s/it]\n"
     ]
    }
   ],
   "source": [
    "# zero-ablating single head at a time [this code also allows you to ablate multiple heads]\n",
    "# h -> logits\n",
    "\n",
    "with torch.no_grad():\n",
    "    NUM_SAMPLES = 1\n",
    "    POS = 0\n",
    "    index_lst = random.sample(range(0, len(documents_full)), NUM_SAMPLES)\n",
    "    docs = [documents_full[i] for i in index_lst]\n",
    "    labels = [labels_full[i] for i in index_lst]\n",
    "\n",
    "    h_ctbn = []\n",
    "    for doc, label in tqdm.tqdm(zip(docs, labels)):\n",
    "        encoding = get_encoding(doc, tokenizer, device)\n",
    "        label_idx = le_dict[label]\n",
    "        h_ctbn_tot = patch_h_to_logits_at_pos(encoding, model, POS, label_idx, device)\n",
    "        h_ctbn.append(h_ctbn_tot) #[12, 12] = [Layer, HEAD]\n",
    "\n",
    "    h_ctbn = np.stack(h_ctbn)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "a648abe1-02a1-4de7-abb1-0f7b81d35798",
   "metadata": {},
   "outputs": [],
   "source": [
    "# mean-ablating single head at a time [this code also allows you to ablate multiple heads]\n",
    "# h -> logits\n",
    "        \n",
    "with torch.no_grad():\n",
    "    NUM_SAMPLES = 200\n",
    "    POS = 0\n",
    "    index_lst = random.sample(range(0, len(documents_full)), NUM_SAMPLES)\n",
    "    docs = [documents_full[i] for i in index_lst]\n",
    "    labels = [labels_full[i] for i in index_lst]\n",
    "\n",
    "    h_ctbn = []\n",
    "    for doc, label in tqdm.tqdm(zip(docs, labels)):\n",
    "        encoding = get_encoding(doc, tokenizer, device)\n",
    "        label_idx = le_dict[label]\n",
    "        h_ctbn_tot = patch_h_to_logits_at_pos(encoding, model, POS, label_idx, device,\n",
    "                                              mean_act=back, mean_ablated=True)\n",
    "        h_ctbn.append(h_ctbn_tot) #[12, 12] = [Layer, HEAD]\n",
    "\n",
    "    h_ctbn = np.stack(h_ctbn)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 440,
   "id": "f7dd92ff-f9e0-4598-85c7-e268cbad7b87",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "avg_h_ctbn = np.mean(h_ctbn, axis=0) #avg over samples"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "3f6a115c-79cd-4e9e-928e-8a5ac3a54531",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "ename": "FileNotFoundError",
     "evalue": "[Errno 2] No such file or directory: '/Users/georgiazhou/research-machine1/CD_Circuit/output/path/bert_PrimaryGleason/h_to_logits/singleH_200_mean_ablated_nov15.pkl'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mFileNotFoundError\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[17], line 7\u001b[0m\n\u001b[1;32m      2\u001b[0m os\u001b[38;5;241m.\u001b[39mmakedirs(path, exist_ok\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mTrue\u001b[39;00m)\n\u001b[1;32m      4\u001b[0m \u001b[38;5;66;03m#with open(os.path.join(path, f\"singleH_{NUM_SAMPLES}_nov15.pkl\"), 'wb') as handle:\u001b[39;00m\n\u001b[1;32m      5\u001b[0m \u001b[38;5;66;03m#    pickle.dump(avg_h_ctbn, handle)\u001b[39;00m\n\u001b[0;32m----> 7\u001b[0m \u001b[38;5;28;01mwith\u001b[39;00m \u001b[38;5;28mopen\u001b[39m(os\u001b[38;5;241m.\u001b[39mpath\u001b[38;5;241m.\u001b[39mjoin(path, \u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124msingleH_200_mean_ablated_nov15.pkl\u001b[39m\u001b[38;5;124m\"\u001b[39m), \u001b[38;5;124m'\u001b[39m\u001b[38;5;124mrb\u001b[39m\u001b[38;5;124m'\u001b[39m) \u001b[38;5;28;01mas\u001b[39;00m handle:\n\u001b[1;32m      8\u001b[0m     back \u001b[38;5;241m=\u001b[39m pickle\u001b[38;5;241m.\u001b[39mload(handle)\n",
      "File \u001b[0;32m~/miniconda3/lib/python3.12/site-packages/IPython/core/interactiveshell.py:310\u001b[0m, in \u001b[0;36m_modified_open\u001b[0;34m(file, *args, **kwargs)\u001b[0m\n\u001b[1;32m    303\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m file \u001b[38;5;129;01min\u001b[39;00m {\u001b[38;5;241m0\u001b[39m, \u001b[38;5;241m1\u001b[39m, \u001b[38;5;241m2\u001b[39m}:\n\u001b[1;32m    304\u001b[0m     \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mValueError\u001b[39;00m(\n\u001b[1;32m    305\u001b[0m         \u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mIPython won\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mt let you open fd=\u001b[39m\u001b[38;5;132;01m{\u001b[39;00mfile\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m by default \u001b[39m\u001b[38;5;124m\"\u001b[39m\n\u001b[1;32m    306\u001b[0m         \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mas it is likely to crash IPython. If you know what you are doing, \u001b[39m\u001b[38;5;124m\"\u001b[39m\n\u001b[1;32m    307\u001b[0m         \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124myou can use builtins\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124m open.\u001b[39m\u001b[38;5;124m\"\u001b[39m\n\u001b[1;32m    308\u001b[0m     )\n\u001b[0;32m--> 310\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m io_open(file, \u001b[38;5;241m*\u001b[39margs, \u001b[38;5;241m*\u001b[39m\u001b[38;5;241m*\u001b[39mkwargs)\n",
      "\u001b[0;31mFileNotFoundError\u001b[0m: [Errno 2] No such file or directory: '/Users/georgiazhou/research-machine1/CD_Circuit/output/path/bert_PrimaryGleason/h_to_logits/singleH_200_mean_ablated_nov15.pkl'"
     ]
    }
   ],
   "source": [
    "path = f\"{base_dir}/output/{args['task']}/{args['model_type']}_{args['field']}/h_to_logits\"\n",
    "os.makedirs(path, exist_ok=True)\n",
    "\n",
    "#with open(os.path.join(path, f\"singleH_{NUM_SAMPLES}_nov15.pkl\"), 'wb') as handle:\n",
    "#    pickle.dump(avg_h_ctbn, handle)\n",
    "    \n",
    "with open(os.path.join(path, f\"singleH_200_mean_ablated_nov15.pkl\"), 'rb') as handle:\n",
    "    back = pickle.load(handle)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 441,
   "id": "f52aaea4-a392-4cf7-a977-82fe37e9ad61",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjIAAAHFCAYAAADosxNlAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjYuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8o6BhiAAAACXBIWXMAAA9hAAAPYQGoP6dpAABOVklEQVR4nO3deXhM59sH8O9kMpnsQUIWZLFG7BIl0TSWhqJa1QWxJE0pSm2vLVW1i6WUl6JqiyrSX5VGF0ttrVoahGqopUIosYsImSQzz/uHN/Mzsidz5szE93Nd57rMOWee557tuHM/zzlHIYQQICIiIrJAVnIHQERERFRWTGSIiIjIYjGRISIiIovFRIaIiIgsFhMZIiIislhMZIiIiMhiMZEhIiIii8VEhoiIiCwWExkiIiKyWM91IvPnn3/i3XffhZ+fH2xtbeHo6IgWLVpg7ty5uHv3rmT9Xrt2DVOmTMGJEydK9byoqCj4+voarFMoFBg2bJjxggOwdOlSrF27Nt/6S5cuQaFQFLjNnMTHx6Nhw4aws7ODQqEo9H3et28fFAoFvv322wK3Dxs2DAqFQsJIC7d27VooFApcunSp3G0V9nmWV9u2baFQKPSLnZ0dmjZtioULF0Kn0xm9v2flfX779u3TryvoN1IS5vidnzJlChQKBW7fvm3Sfn19fREVFaV/XNbjVVnpdDp89dVXePnll+Hm5gaVSoVq1arh1VdfxbZt2/TfrbzPJm9RqVRwdXVFy5YtMWrUKCQnJ5skXpLfc5vIfPnllwgMDERiYiLGjh2L7du3Y8uWLXj77bexfPlyvPfee5L1fe3aNUydOrXUB4ZJkyZhy5Yt0gT1lMIO6p6enjh06BC6du0qeQxldevWLfTr1w+1a9fG9u3bcejQIdSrV0/usGQlVSIDALVq1cKhQ4dw6NAhxMfHo3r16hg1ahRiYmIk6a84Zf2NWPJ33ti2bNmCSZMm6R+X9XhVFllZWejSpQsiIyNRrVo1LFu2DHv27MHy5cvh5eWFt99+G9u2bTN4zocffohDhw5h//79+Oqrr9C9e3ckJCSgadOmmDdvnuQxk/ys5Q5ADocOHcKQIUMQHh6OrVu3Qq1W67eFh4fjf/7nf7B9+3YZIzT06NEj2Nvbo3bt2rLGoVar0bp1a1ljKM65c+eQk5ODvn37IiwsTO5wKjw7OzuD70Tnzp3h7++PJUuWYMaMGVCpVPmeI4RAVlYW7OzsjB6PsX8jlvCdN7bmzZvL1vfo0aOxY8cOxMXFoX///gbbevTogbFjx+Lx48cG6729vQ0+oy5dumD06NHo0aMHxo0bh0aNGqFz584miZ/k8VxWZGbNmgWFQoEVK1YYJDF5bGxs8Nprr+kf63Q6zJ07F/7+/lCr1ahWrRr69++Pq1evGjyvbdu2aNSoERITExEaGgp7e3vUqlULs2fP1pdD9+3bh5YtWwIA3n33XX1ZdMqUKQCelMYdHR1x6tQpdOzYEU5OTujQoYN+W2Fl8y+++AL16tWDWq1GQEAANm3aZLA9r0z9rGeHMHx9fZGcnIz9+/frY8vrs7Ay+4EDB9ChQwc4OTnB3t4eISEh+PHHHwvsZ+/evRgyZAjc3Nzg6uqKHj164Nq1awW+pmclJCQgODgY9vb2cHJyQnh4OA4dOqTfHhUVhRdffBEA0LNnTygUCrRt27ZEbZdGfHw8goOD4eDgAEdHR3Tq1AlJSUkG+xw9ehS9evWCr68v7Ozs4Ovri969e+Py5cv52jt8+DDatGkDW1tbeHl5ISYmBjk5Ofn227NnD9q2bQtXV1fY2dnB29sbb775Jh49elRorEV9ngCQmpqKvn37olq1alCr1WjQoAHmz59f5qEhlUqFwMBAPHr0CLdu3QLw3+HP5cuXo0GDBlCr1YiLiwMAnD9/HhEREQb9f/755/na/fvvv/HKK6/A3t4ebm5uGDx4MDIyMvLtV9BvRKfTYfHixWjWrBns7OxQqVIltG7dGgkJCcW+R3J/50uiuN9Fnu+//x5NmjSBWq1GrVq1sGjRogKPC08PLRV3vLp48SJ69eoFLy8vqNVquLu7o0OHDmWq3qSlpWHlypXo1KlTviQmT926ddGkSZNi27Kzs8OqVaugUqlYlXkOPHeJjFarxZ49exAYGIiaNWuW6DlDhgzB+PHjER4ejoSEBEyfPh3bt29HSEhIvvHrtLQ09OnTB3379kVCQgI6d+6MmJgYrF+/HgDQokULrFmzBgDw8ccf68vyAwYM0LeRnZ2N1157De3bt8f333+PqVOnFhlfQkIC/vd//xfTpk3Dt99+Cx8fH/Tu3bvQuR9F2bJlC2rVqoXmzZvrYyuqVL9//360b98e6enpWLVqFTZu3AgnJyd069YN8fHx+fYfMGAAVCoVNmzYgLlz52Lfvn3o27dvsXFt2LABr7/+OpydnbFx40asWrUK9+7dQ9u2bXHgwAEAT4YV8v4TnDVrFg4dOoSlS5cW27ZOp0Nubm6+paAbw8+aNQu9e/dGQEAAvvnmG3z11VfIyMhAaGgoTp8+rd/v0qVLqF+/PhYuXIgdO3Zgzpw5uH79Olq2bGnwnTl9+jQ6dOiA+/fvY+3atVi+fDmSkpIwY8YMg34vXbqErl27wsbGBqtXr8b27dsxe/ZsODg4IDs7u9DXVtTneevWLYSEhGDnzp2YPn06EhIS8PLLL2PMmDHlmnf1zz//wNraGpUrV9av27p1K5YtW4ZPPvkEO3bs0L9fLVu2xF9//YX58+fjhx9+QNeuXTF8+HCD7/yNGzcQFhaGv/76C0uXLsVXX32Fhw8fljjGqKgojBgxAi1btkR8fDw2bdqE1157TZ+8m+t3viRK8rsAgO3bt6NHjx5wdXVFfHw85s6di40bN+oTysIUd7zq0qULjh07hrlz52LXrl1YtmwZmjdvjvv37+vbyEvoihve3Lt3L3JyctC9e/cyvRfP8vLyQmBgIA4ePIjc3FyjtElmSjxn0tLSBADRq1evEu1/5swZAUB88MEHBuuPHDkiAIiPPvpIvy4sLEwAEEeOHDHYNyAgQHTq1En/ODExUQAQa9asyddfZGSkACBWr15d4DYfHx+DdQCEnZ2dSEtL06/Lzc0V/v7+ok6dOvp1kydPFgV93GvWrBEAREpKin5dw4YNRVhYWL59U1JS8sXdunVrUa1aNZGRkWHQf6NGjUSNGjWETqcz6OfZ93Hu3LkCgLh+/Xq+/vJotVrh5eUlGjduLLRarX59RkaGqFatmggJCdGv27t3rwAg/vOf/xTa3rP7FrfkSU1NFdbW1uLDDz80aCcjI0N4eHiId955p9C+cnNzxcOHD4WDg4NYtGiRfn3Pnj0L/fye/ly+/fZbAUCcOHGi2Nf1rMI+zwkTJhT4fR0yZIhQKBTi7NmzRbYbFhYmGjZsKHJyckROTo64du2avs23335bvx8A4eLiIu7evWvw/E6dOokaNWqI9PR0g/XDhg0Ttra2+v3Hjx8vFApFvtceHh4uAIi9e/fq1z37G/n1118FADFx4sQiX4u5feeF+O9v9tatWwVuL83vomXLlqJmzZpCo9EY7Ofq6prvuODj4yMiIyP1jws7Xt2+fVsAEAsXLizydcTFxQmlUini4uKK3G/27NkCgNi+fXuR++XJ+2zmzZtX6D49e/YUAMSNGzdK1CZZpueuIlNae/fuBQCDWfwA8MILL6BBgwbYvXu3wXoPDw+88MILBuuaNGlS4JBCUd58880S79uhQwe4u7vrHyuVSvTs2RMXLlzIN/xlTJmZmThy5AjeeustODo6GvTfr18/XL16FWfPnjV4ztNDdgD0ZeKi3p+zZ8/i2rVr6NevH6ys/vuVdXR0xJtvvonDhw8XObxSnDlz5iAxMTHf8s477xjst2PHDuTm5qJ///4GlRtbW1uEhYUZnD3z8OFDjB8/HnXq1IG1tTWsra3h6OiIzMxMnDlzRr/f3r17C/38ntasWTPY2Njg/fffR1xcHC5evFjm15tnz549CAgIyPd9jYqKghACe/bsKbaN5ORkqFQqqFQqeHl5Yf78+ejTpw++/PJLg/3at29vUKHJysrC7t278cYbb8De3t7g/ezSpQuysrJw+PBhAE/eo4YNG6Jp06YGbUZERBQb388//wwAGDp0aLH7loSpvvMlUdLfRWZmJo4ePYru3bvDxsbGYL9u3bqVuf8qVaqgdu3amDdvHhYsWICkpKQChyTzfi+FDRdJSRRQVaWK57mb7Ovm5gZ7e3ukpKSUaP87d+4AeHL2wrO8vLzyHYxcXV3z7adWq/NNUCuKvb09nJ2dS7y/h4dHoevu3LmDGjVqlLit0rh37x6EEIW+N3n9P+3Z9ydvjlJR709xn4FOp8O9e/dgb29fuhfw/2rVqoWgoKB866tWrWrw+MaNGwCgnzPwrKf/M4mIiMDu3bsxadIktGzZEs7OzlAoFOjSpYvBa71z506Rn1+e2rVr45dffsHcuXMxdOhQZGZmolatWhg+fDhGjBhR8hf7lDt37hQ456qwz64gtWvXxqZNm6BQKGBraws/P78CP4dnP7s7d+4gNzcXixcvxuLFiwtsO28I7s6dO/Dz88u3vaD37Vm3bt2CUqks0b4lYarvfEmU9HchhIAQwiBZzlPQupJSKBTYvXs3pk2bhrlz5+J//ud/UKVKFfTp0wczZ86Ek5NTqdrz9vYGgBIfm0vi8uXLUKvVqFKlitHaJPPz3CUySqUSHTp0wM8//4yrV68W+5983kHo+vXr+fa9du0a3NzcjB5jaa9dkpaWVui6vPhtbW0BABqNxmCCc3muUVG5cmVYWVnh+vXr+bblTWY0xvvz9GdQUD9WVlYGf+1LJe+15M1DKkx6ejp++OEHTJ48GRMmTNCv12g0+a5P5OrqWuTn97TQ0FCEhoZCq9Xi6NGjWLx4MUaOHAl3d3f06tWr1K/H1dW13J+dra1tgUngs579TleuXFlfxSisWpKXvJTmPXpW1apVodVqkZaWVuB/+KVlqu98SZT0dyGEgEKh0CfiTyvJe1gUHx8frFq1CsCTMwa/+eYbTJkyBdnZ2Vi+fHmp2mrXrh1UKhW2bt2KwYMHlysuAPj3339x7NgxhIWFwdr6ufuv7rnyXA4txcTEQAiBgQMHFjhRMicnR3+tgvbt2wOAfrJunsTERJw5c0Z/RlFpGOsvsjy7d+82OEhptVrEx8ejdu3a+uQr7y/vP//80+C5z16TIS++ksTm4OCAVq1a4bvvvjPYX6fTYf369ahRo4ZRruFSv359VK9eHRs2bDAoFWdmZmLz5s36Mzak1qlTJ1hbW+Off/5BUFBQgQvw5D9tIUS+M+JWrlwJrVZrsK5du3aFfn6FUSqVaNWqlX5i8/Hjx4uMu7DPs0OHDjh9+nS+569btw4KhQLt2rUrst3ysLe3R7t27ZCUlIQmTZoU+F7m/Ufdrl07JCcn4+TJkwZtbNiwodh+8k67XbZsWZH7mdt3viRK+rtwcHBAUFAQtm7danC8e/jwIX744Ydi+ynp8apevXr4+OOP0bhx42K/kwXx8PDAgAEDsGPHDqxbt67Aff755598x7CCPH78GAMGDEBubi7GjRtX6ljIsjyXaWpwcDCWLVuGDz74AIGBgRgyZAgaNmyInJwcJCUlYcWKFWjUqBG6deuG+vXr4/3338fixYthZWWFzp0749KlS5g0aRJq1qyJUaNGlbr/2rVrw87ODl9//TUaNGgAR0dHeHl56UvTpeXm5ob27dtj0qRJcHBwwNKlS/H3338bnILdpUsXVKlSBe+99x6mTZsGa2trrF27FleuXMnXXuPGjbFp0ybEx8ejVq1asLW1RePGjQvsOzY2FuHh4WjXrh3GjBkDGxsbLF26FH/99Rc2btxolCvjWllZYe7cuejTpw9effVVDBo0CBqNBvPmzcP9+/cxe/bscvdREr6+vpg2bRomTpyIixcv4pVXXkHlypVx48YN/PHHH3BwcMDUqVPh7OyMl156CfPmzYObmxt8fX2xf/9+rFq1CpUqVTJo8+OPP0ZCQgLat2+PTz75BPb29vj888+RmZlpsN/y5cuxZ88edO3aFd7e3sjKysLq1asBAC+//HKRcRf2eY4aNQrr1q1D165dMW3aNPj4+ODHH3/E0qVLMWTIEMn/Q160aBFefPFFhIaGYsiQIfD19UVGRgYuXLiAbdu26efojBw5EqtXr0bXrl0xY8YMuLu74+uvv8bff/9dbB+hoaHo168fZsyYgRs3buDVV1+FWq1GUlIS7O3t8eGHHxb5HhXEFN/5p23btq3AYZq33nqrxL+LadOmoWvXrujUqRNGjBgBrVaLefPmwdHRsdirmBd2vLp9+zaGDRuGt99+G3Xr1oWNjQ327NmDP//806ASuW7dOkRHR2P16tXFzpNZsGABLl68iKioKOzYsQNvvPEG3N3dcfv2bezatQtr1qzBpk2bDE7BTk1NxeHDh6HT6ZCeno6kpCSsXr0aly9fxvz589GxY8eSvtVkqWSaZGwWTpw4ISIjI4W3t7ewsbERDg4Oonnz5uKTTz4RN2/e1O+n1WrFnDlzRL169YRKpRJubm6ib9++4sqVKwbt5Z3F8ayCzjbauHGj8Pf3FyqVSgAQkydP1u/r4OBQYLyFnbU0dOhQsXTpUlG7dm2hUqmEv7+/+Prrr/M9/48//hAhISHCwcFBVK9eXUyePFmsXLky31lLly5dEh07dhROTk4CgL7Pgs7gEEKI3377TbRv3144ODgIOzs70bp1a7Ft2zaDffLO4EhMTDRYn3fm0NNnnhRm69atolWrVsLW1lY4ODiIDh06iN9//73A9kpz1lJh+w4dOrTAM722bt0q2rVrJ5ydnYVarRY+Pj7irbfeEr/88ot+n6tXr4o333xTVK5cWTg5OYlXXnlF/PXXX/nOCBFCiN9//120bt1aqNVq4eHhIcaOHStWrFhh8LkcOnRIvPHGG8LHx0eo1Wrh6uoqwsLCREJCQrGvs7DPUwghLl++LCIiIoSrq6tQqVSifv36Yt68eQZnwRSmsO/7s/K+owVJSUkR0dHRonr16kKlUomqVauKkJAQMWPGDIP9Tp8+LcLDw4Wtra2oUqWKeO+998T3339f7FlLQjz5/X722WeiUaNGwsbGRri4uIjg4GCD76g5fufzzloqbMlTkt+FEEJs2bJFNG7cWNjY2Ahvb28xe/ZsMXz4cFG5cmWD/Qr6jhZ0vLpx44aIiooS/v7+wsHBQTg6OoomTZqIzz77TOTm5uZ7Hwo6S7Mgubm5Ii4uTrRv315UqVJFWFtbi6pVq4rOnTuLDRs26L+beZ9N3qJUKkXlypVFYGCgGDlypEhOTi5Rf2T5FEJwWjcR0fMmJycHzZo1Q/Xq1bFz5065wyEqs+dyaImI6Hnz3nvvITw8HJ6enkhLS8Py5ctx5swZLFq0SO7QiMqFiQwR0XMgIyMDY8aMwa1bt6BSqdCiRQv89NNPxc6xIjJ3HFoiIiIii/Vcnn5NREREFQMTGSIiIrJYTGSIiIjIYjGRISIiIotVIc9aWnb4ktwhwNXepvidJPYgK0fuEOBsq5I7BADAZz+cKX4niX3wSn25Q4BKadyrzloqlZX874PaWil3CNDkaovfSWKPcvLfMdvU+rWQ5sa6T7NpHm2UdrKTVhulnYqEFRkiIiKyWBWyIkNERGROFFbyV+AqKiYyREREEmMiIx0mMkRERBJjIiMdzpEhIiIii8WKDBERkcRYkZEOExkiIiKJKZRMZKTCoSUiIiKyWKzIEBERScyKQ0uSYSJDREQkMc6RkQ6HloiIiMhisSJDREQkMVZkpCNrInP16lUsW7YMBw8eRFpaGhQKBdzd3RESEoLBgwejZs2acoZHRERkFAorDoBIRbZ39sCBA2jQoAG2bNmCpk2bon///ujbty+aNm2KrVu3omHDhvj999/lCo+IiIgsgGwVmVGjRmHAgAH47LPPCt0+cuRIJCYmFtmORqOBRqMxWJeTrYHKRm20WImIiMqDQ0vSka0i89dff2Hw4MGFbh80aBD++uuvYtuJjY2Fi4uLwbJj3TJjhkpERFQuCiulURbKT7ZExtPTEwcPHix0+6FDh+Dp6VlsOzExMUhPTzdYOvUfYsxQiYiIyoWJjHRkG1oaM2YMBg8ejGPHjiE8PBzu7u5QKBRIS0vDrl27sHLlSixcuLDYdtRqNdRqw2Eklc1diaImIiIicyJbIvPBBx/A1dUVn332Gb744gtotVoAgFKpRGBgINatW4d33nlHrvCIiIiMhvdako6sp1/37NkTPXv2RE5ODm7fvg0AcHNzg0qlkjMsIiIio+KwkHTM4oJ4KpWqRPNhiIiIiJ5mFokMERFRRcaKjHSYyBAREUmMd7+WDq+ZTERERBaLFRkiIiKJcWhJOkxkiIiIJMZERjocWiIiIiKLxYoMERGRxFiRkQ4TGSIiIokxkZEOExkiIiKJMZGRToVMZDwc1cXvJDGVUv7pR7bW8sdgpVDIHQIAwK+Gs9whwNFG/gOZOXwvdULIHQJUVubxvZSb2lr+76S5HCPIclXIRIaIiMic8KaR0mEiQ0REJDEOLUlH/jozERERURmxIkNERCQxVmSkw0SGiIhIYkxkpMOhJSIiIrJYrMgQERFJzIqn/EuGiQwREZHEFExkJMOhJSIiIrJYrMgQERFJTMErGEuGiQwREZHEOEdGOkxkiIiIJMY5MtLhHBkiIiKyWGadyFy5cgXR0dFF7qPRaPDgwQODJSdbY6IIiYiIiqewUhhlofzMOpG5e/cu4uLiitwnNjYWLi4uBsvmlYtNFCEREVHxrBQKoyyUn6xzZBISEorcfvHixWLbiImJwejRow3Wbb9wt1xxERERkWWQNZHp3r07FAoFhBCF7lPcKWtqtRpqtdpgncom0yjxERERGQOHhaQj69CSp6cnNm/eDJ1OV+By/PhxOcMjIiIyCs6RkY6siUxgYGCRyUpx1RoiIiJ6vsmayIwdOxYhISGFbq9Tpw727t1rwoiIiIiMz8pKYZSltH799Vd069YNXl5eUCgU2Lp1q8F2IQSmTJkCLy8v2NnZoW3btkhOTjbSqzYNWROZ0NBQvPLKK4Vud3BwQFhYmAkjIiIiMj6FlXGW0srMzETTpk2xZMmSArfPnTsXCxYswJIlS5CYmAgPDw+Eh4cjIyOjnK/YdHhlXyIiogqqc+fO6Ny5c4HbhBBYuHAhJk6ciB49egAA4uLi4O7ujg0bNmDQoEGmDLXMzPo6MkRERBWBQqEwymJMKSkpSEtLQ8eOHfXr1Go1wsLCcPDgQaP2JSVWZIiIiCRmrJtGajQaaDSGV68v6DIkJZGWlgYAcHd3N1jv7u6Oy5cvlz1IE2NFhoiISGLGOv26oKvZx8bGli+2Zyo9QgijV3+kxIoMERGRhSjoavZlqcYAgIeHB4AnlRlPT0/9+ps3b+ar0pgzVmSIiIgkZqyKjFqthrOzs8FS1kTGz88PHh4e2LVrl35ddnY29u/fX+SlUcwNKzJEREQSk+uGjw8fPsSFCxf0j1NSUnDixAlUqVIF3t7eGDlyJGbNmoW6deuibt26mDVrFuzt7RERESFLvGXBRIaIiKiCOnr0KNq1a6d/nDcsFRkZibVr12LcuHF4/PgxPvjgA9y7dw+tWrXCzp074eTkJFfIpcZEhoiISGJy3Sepbdu2xd6YecqUKZgyZYrpgjKyCpnIqJTyT/2pYif/W5ujlf8+VXKVU5/l7+ksdwhmwUUt//dSZwb3T1Mp5f9emsPv0xxohfzHa1PgDR+l83x8g4iIiKhCkv/PMyIiogrOWBfEo/yYyBAREUnMki4wZ2k4tEREREQWixUZIiIiiSlYNpAMExkiIiKJcY6MdJjIEBERSYynX0uHxS4iIiKyWKzIEBERSYxnLUmHiQwREZHEOEdGOhxaIiIiIovFigwREZHEONlXOkxkiIiIJKZkIiMZ2YeWHj9+jAMHDuD06dP5tmVlZWHdunVFPl+j0eDBgwcGS062RqpwiYiIyIzImsicO3cODRo0wEsvvYTGjRujbdu2uH79un57eno63n333SLbiI2NhYuLi8HynxX/K3XoREREJaa0UhhlofxkTWTGjx+Pxo0b4+bNmzh79iycnZ3Rpk0bpKamlriNmJgYpKenGyxvvz9cwqiJiIhKh4mMdGSdI3Pw4EH88ssvcHNzg5ubGxISEjB06FCEhoZi7969cHBwKLYNtVoNtVptsE5l81iqkImIiMiMyJrIPH78GNbWhiF8/vnnsLKyQlhYGDZs2CBTZERERMbDaop0ZE1k/P39cfToUTRo0MBg/eLFiyGEwGuvvSZTZERERMbDREY6ss6ReeONN7Bx48YCty1ZsgS9e/eGEMLEURERERmXtZXCKAvlJ2siExMTg59++qnQ7UuXLoVOpzNhRERERGRJeEE8IiIiiXFoSTpMZIiIiCTGREY6sl/Zl4iIiKisWJEhIiKSmNKKdQOpMJEhIiKSGIeWpMMUkYiIiCwWKzJEREQSY0VGOkxkiIiIJMZERjocWiIiIiKLVSErMkozSHyd1PK/tVk58l8VWWUOHwYAf3cnuUNAyr1HcocAv8p2codgFlRmcAaJlrdfAQDkaJ+P90GpMI9jYUUk//+2REREFRyHlqTDRIaIiEhiTGSkI399lYiIiKiMWJEhIiKSmDUrMpJhIkNERCQxDi1Jh0NLREREZLFYkSEiIpIYKzLSYSJDREQkMSYy0uHQEhEREVksVmSIiIgkxoqMdJjIEBERSYyJjHQ4tEREREQWixUZIiIiibEiIx3ZE5kzZ87g8OHDCA4Ohr+/P/7++28sWrQIGo0Gffv2Rfv27Yt8vkajgUajMViXk62BykYtZdhEREQlxkRGOrIOLW3fvh3NmjXDmDFj0Lx5c2zfvh0vvfQSLly4gNTUVHTq1Al79uwpso3Y2Fi4uLgYLPEr/tdEr4CIiKh4SiuFURbKT9ZEZtq0aRg7dizu3LmDNWvWICIiAgMHDsSuXbvwyy+/YNy4cZg9e3aRbcTExCA9Pd1g6fn+cBO9AiIiIpKTrIlMcnIyoqKiAADvvPMOMjIy8Oabb+q39+7dG3/++WeRbajVajg7OxssHFYiIiJzwoqMdGSfI5PHysoKtra2qFSpkn6dk5MT0tPT5QuKiIjICJiESEfWioyvry8uXLigf3zo0CF4e3vrH1+5cgWenp5yhEZEREQWQNaKzJAhQ6DVavWPGzVqZLD9559/LvasJSIiInOnVLAiIxVZE5nBgwcXuX3mzJkmioSIiEg6VkxkJMMr+xIREZHFYiJDREQkMaXCOEtpTJkyBQqFwmDx8PCQ5gXKyGzOWiIiIqqorGQ6a6lhw4b45Zdf9I+VSqUscUiJiQwREVEFZW1tXSGrME9jIkNERCQxY521VND9BdVqNdTqgi8Ee/78eXh5eUGtVqNVq1aYNWsWatWqZZRYzAXnyBAREUnMSqEwylLQ/QVjY2ML7LNVq1ZYt24dduzYgS+//BJpaWkICQnBnTt3TPzqpaUQQgi5gzC2n/++IXcIqOFiK3cIyMrRyR0CVKWdnSaRc3ceyR0CrqY/ljsEtK/lKncIZkFlJf/fcNqKd+gtkxyt/O9DYM1KkvexIemqUdp5M6BqqSoyT8vMzETt2rUxbtw4jB492ijxmAMOLREREVmIkiYtBXFwcEDjxo1x/vx5I0clrwqZyOTo5M/wVWZwXw0rVcWbnV5WNZzlv5HozUxN8TtJzEbJSgQAmMO1yazNIAgz+CigtJb/fTAFuc5aeppGo8GZM2cQGhoqdyhGVSETGSIiInMix5V9x4wZg27dusHb2xs3b97EjBkz8ODBA0RGRpo8FikxkSEiIqqArl69it69e+P27duoWrUqWrdujcOHD8PHx0fu0IyKiQwREZHE5DjvYdOmTabvVAZMZIiIiCTGm0ZKR/6Zf0RERERlxIoMERGRxJRmcNZSRcVEhoiISGIcWpIOh5aIiIjIYrEiQ0REJDEzuVtLhcREhoiISGIcWpIOExkiIiKJcbKvdMxujkwFvBk3ERERScTsEhm1Wo0zZ87IHQYREZHRWCmMs1B+sg0tjR49usD1Wq0Ws2fPhqurKwBgwYIFpgyLiIjI6JScIyMZ2RKZhQsXomnTpqhUqZLBeiEEzpw5AwcHByhK8MFrNBpoNBqDdTnZGqhs1MYMl4iIiMyQbENLM2fORHp6OiZNmoS9e/fqF6VSibVr12Lv3r3Ys2dPse3ExsbCxcXFYPn2y8UmeAVEREQlY6VQGGWh/BRCxtm1iYmJ6Nu3L7p164bY2FioVCqoVCqcPHkSAQEBJWqjoIrMrov3ZK/I1HO1l7V/ANDq5I7AfKRrcuQOAcevP5A7BLT3c5U7BGjNYEI/y/xPmMFHYRYaejpL3sfBS3eM0k6Ir/y/YXMj62Tfli1b4tixY7h16xaCgoJw6tSpEg0nPU2tVsPZ2dlgkTuJISIiItOQ/Toyjo6OiIuLw6ZNmxAeHg6tVit3SEREREbFYSHpyJ7I5OnVqxdefPFFHDt2DD4+PnKHQ0REZDQczpSO2SQyAFCjRg3UqFFD7jCIiIiMihUZ6ZjdBfGIiIiISsqsKjJEREQVkZJlA8kwkSEiIpIYh5akwxyRiIiILBYrMkRERBJjQUY6TGSIiIgkZgVmMlLh0BIRERFZLFZkiIiIJMahJekwkSEiIpKYFRMZyVTIROZRjvz3azKLy1Fb8da2eexVSrlDgMpK/pFcc7iWhTncTk3F/1UAmMedyInKq0ImMkRERObEHP62raiYyBAREUmMZy1Jh4kMERGRxFiRkY4ZjJgTERERlQ0rMkRERBLj/HLpMJEhIiKSGPMY6XBoiYiIiCwWKzJEREQSs+JsX8kwkSEiIpIY8xjpcGiJiIiILBYrMkRERBJj1UA6TGSIiIgkpuDYkmSYJBIREZHFMquKzL179xAXF4fz58/D09MTkZGRqFmzptxhERERlQsviCedUldkcnJyUKtWLZw+fbrcnXt5eeHOnTsAgJSUFAQEBGDOnDk4f/48vvjiCzRu3Bh///13kW1oNBo8ePDAYMnJ1pQ7NiIiImNRKIyzUH6lTmRUKhU0Go1RxvvS0tKg1WoBAB999BH8/f3xzz//YOfOnbhw4QJCQ0MxadKkItuIjY2Fi4uLwfL9qiXljo2IiMhYrIy0UH5lel8+/PBDzJkzB7m5uUYL5MiRI5g0aRLs7e0BAGq1Gh9//DEOHz5c5PNiYmKQnp5usLz+3jCjxUVERETmq0xzZI4cOYLdu3dj586daNy4MRwcHAy2f/fddyVuK6+yo9Fo4O7ubrDN3d0dt27dKvL5arUaarXaYJ3KJqPE/RMREUmNZy1Jp0yJTKVKlfDmm28aJYAOHTrA2toaDx48wLlz59CwYUP9ttTUVLi5uRmlHyIiIrlwsq90ypTIrFmzxiidT5482eBx3rBSnm3btiE0NNQofREREVHFoxBCiLI8MTc3F/v27cM///yDiIgIODk54dq1a3B2doajo6Ox4yyVTSf/lbV/AAj0dJY7BGjL9tFWSFm5OrlDwJGr6XKHgLZ+leUOAdla+b+XtkpOmwR4jMjj7y798fpuxiOjtFPFyb74nZ4zZarIXL58Ga+88gpSU1Oh0WgQHh4OJycnzJ07F1lZWVi+fLmx4yQiIrJYHFqSTpn+LBkxYgSCgoJw79492NnZ6de/8cYb2L17t9GCIyIiIipKmSoyBw4cwO+//w4bGxuD9T4+Pvj3X/mHdYiIiMwJz1qSTpkSGZ1Op7+Q3dOuXr0KJyencgdFRERUkXBoSTplGloKDw/HwoUL9Y8VCgUePnyIyZMno0uXLsaKjYiIiCqo27dv48cff0RCQgKuX79e5nbKlMh89tln2L9/PwICApCVlYWIiAj4+vri33//xZw5c8ocDBERUUWkMNJSFkuXLoWfnx9sbW0RGBiI3377rTwvxSg2b96MOnXqYOrUqZg8eTJq165d5ku7lPn068ePH2Pjxo04fvw4dDodWrRogT59+hhM/pULT79+gqdW/hdPv36Cp18/wdOvn+Ax4glTnH796HGWUdqxt7Mt1f7x8fHo168fli5dijZt2uCLL77AypUrcfr0aXh7exslppJ4+PChwaVZmjRpgm+//Rb16tUDAPz4448YOHAgrl27Vuq2y/RrzszMhJ2dHaKjo7FkyRIsXboUAwYMMIskhoiIyNzIdffrBQsW4L333sOAAQPQoEEDLFy4EDVr1sSyZcuM/yKLEBgYiO+//17/2NraGjdv3tQ/vnHjRr4TiEqqTImMu7s7oqOjceDAgTJ1SkRERKWn0Wjw4MEDg0Wj0RS4b3Z2No4dO4aOHTsarO/YsSMOHjxoinD1duzYgS+++AJvvPEGrl27hkWLFqFnz57w8PCAm5sbJkyYgKVLl5ap7TKdtbRx40asXbsWHTp0gI+PD6Kjo9G/f394eXmVKQhjU5nB9HBzONPODEIwGzZmMJSgUsr/iViZwbdCJf9HAR3kH1Ixh+NUTq7cEZjH+2AKCiMN48XOno2pU6carJs8eTKmTJmSb9/bt29Dq9UWeEPmtLQ0o8RTUr6+vvjpp5+wYcMGhIWFYcSIEbhw4QIuXLgArVYLf39/2NqWbtgsT5kOKd26dcPmzZtx7do1DBkyBBs3boSPjw9effVVfPfdd8jNNYNfBxERkbkQOqMsMTExSE9PN1hiYmKK7PrZa9gIIWS7rk1ERAT++OMPJCUloW3bttDpdGjWrFmZkxigjIlMHldXV4waNQonT57EggUL8Msvv+Ctt96Cl5cXPvnkEzx6ZJx7SxARERGgVqvh7OxssKjV6gL3dXNzg1KpzFd9uXnzZr4qjSn8/PPPmD9/Po4dO4ZVq1Zhzpw5iIiIwNixY/H48eMyt1uuRCYtLQ1z585FgwYNMGHCBLz11lvYvXs3PvvsM2zZsgXdu3cvT/NEREQVgkLojLKUho2NDQIDA7Fr1y6D9bt27UJISIgxX16xxo0bh6ioKCQmJmLQoEGYPn062rZti6SkJKjVajRr1gw///xzmdou0+nX3333HdasWYMdO3YgICAAAwYMQN++fVGpUiX9PsnJyWjevDmys7PLFFh5bD5V+tO3jK2ph/xXOOaplf+llf/saxy+el/uEBDqLf/p1+bwvTSHOWzmMDckK1f+z8Ic3oe61aQ/XmseGufyC2pHl1Ltn3f69fLlyxEcHIwVK1bgyy+/RHJyMnx8fIwSU0m4ublhx44dCAwMxN27d9G6dWucO3dOvz05ORmDBg0q00lEZZrs++6776JXr174/fff0bJlywL3qVWrFiZOnFiW5omIiMgIevbsiTt37mDatGm4fv06GjVqhJ9++smkSQwA2NvbIyUlBYGBgbhy5Uq+OTENGzYs85nQZarIPHr0CPb29mXq0BRYkXnCHP7yNResyDzBiswTrMg8wYrMEyapyGTcN0o7aqdKRmnH1L7++msMHDgQlSpVwqNHjxAXF4fXX3/dKG2XqSLzdBLz+PFj5OTkGGx3dpb/qrZERERmo5TzWyqaPn364JVXXsHFixdRt25dg6ko5VWmRCYzMxPjx4/HN998gzt37uTbXtCdsYmIiOj55erqCldXV6O3W6azlsaNG4c9e/Zg6dKlUKvVWLlyJaZOnQovLy+sW7fO2DESERFZNDnOWnpelKkis23bNqxbtw5t27ZFdHQ0QkNDUadOHfj4+ODrr79Gnz59jB0nERGR5WISIpkyVWTu3r0LPz8/AE/mw9y9excA8OKLL+LXX381XnREREQVgZGu7Ev5lSmRqVWrFi5dugQACAgIwDfffAPgSaXGxaV057gTERERlVWZEpl3330XJ0+eBADExMTo58qMGjUK48aNM2qAREREFo8VGcmUaY7MqFGj9P9u164d/v77bxw9ehRVq1bFmjVrStxOUlISKlWqpB+mWr9+PZYtW4bU1FT4+Phg2LBh6NWrV5FtaDSafLcwz8nWQGVT8L0niIiITE7HJEQq5brXUh5vb2/06NEDzs7OiIuLK/Hz3nvvPf0Q1cqVK/H+++8jKCgIEydORMuWLTFw4ECsXr26yDZiY2Ph4uJisHy3ckl5Xg4RERFZiDJd2bcwJ0+eRIsWLUp8HRkHBwecOXMG3t7eaNGiBQYPHoz3339fv33Dhg2YOXMmkpOTC22joIrMT+fvyF6R4ZV9zQuv7PsEr+z7BK/s+wSv7PuEKa7sm3Mr1SjtqKp6G6WdiqRMQ0vGYmdnh1u3bsHb2xv//vsvWrVqZbC9VatWSElJKbINtVqd7xbmKpuHRo+ViIiozDi/RTJGGVoqq86dO2PZsmUAgLCwMHz77bcG27/55hvUqVNHjtCIiIjIApSqItOjR48it9+/f79Unc+ZMwdt2rRBWFgYgoKCMH/+fOzbtw8NGjTA2bNncfjwYWzZsqVUbRIREZkdMxhSrahKlcgUd40YFxcX9O/fv8TteXl5ISkpCbNnz8a2bdsghMAff/yBK1euoE2bNvj9998RFBRUmhCJiIjMD4eWJGPUyb7mYvOpa3KHwMm+ZoaTfZ/gZN8nONn3CU72fcIkk33T/jFKOyqP2kZppyKRdbIvERHR84A3fJQOExkiIiKpMZGRDBMZIiIiqTGRkYysp18TERERlQcrMkRERFJjRUYyTGSIiIgkxsm+0uHQEhEREVksVmSIiIikpmNFRipMZIiIiKRmBheCrKgqZCKTo5P/C2MGF6uEVv63AVYwgzcCAKzkfzNUVhzJBczjqrrm8H+K0gzeCKUZ/C6IyqtCJjJERERmhZN9JcNEhoiISGI8a0k6rHUTERGRxWJFhoiISGqsyEiGiQwREZHUmMhIhokMERGR1HRauSOosDhHhoiIiCwWKzJEREQSE7yyr2SYyBAREUmNQ0uS4dASERERWSxWZIiIiKTGioxkmMgQERFJTGiZyEiFQ0tERERksWRNZD788EP89ttv5WpDo9HgwYMHBktOtsZIERIRERmBTmechfKRNZH5/PPP0bZtW9SrVw9z5sxBWlpaqduIjY2Fi4uLwfL9qiUSREtERFRGOq1xFspH9qGlnTt3okuXLvj000/h7e2N119/HT/88AN0Jcw8Y2JikJ6ebrC8/t4wiaMmIiIicyB7ItO4cWMsXLgQ165dw/r166HRaNC9e3fUrFkTEydOxIULF4p8vlqthrOzs8GislGbKHoiIqLiCZ3WKAvlJ3sik0elUuGdd97B9u3bcfHiRQwcOBBff/016tevL3doRERE5cM5MpIxm0Tmad7e3pgyZQpSUlKwfft2ucMhIiIqF1ZkpCNrIuPj4wOlUlnodoVCgfDwcBNGRERERJZE1gvipaSkyNk9ERGRabCaIhle2ZeIiEhqnN8iGbOcI0NERERUEqzIEBERSYz3WpIOExkiIiKpcY6MZDi0RERERBaLFRkiIiKpsSIjGSYyREREEhM8a0kyHFoiIiIi+Pr6QqFQGCwTJkyQO6xisSJDREQkNQsZWpo2bRoGDhyof+zo6ChjNCVTIRMZnU7IHQKytfLHoDWDSqa1Uv73AQCEGYShUirkDgE5ZvDbsJL/bUBWrvw/Dhsz+D6YwzFCC/m/kyZhIYmMk5MTPDw85A6jVDi0REREJDGh0xllkdqcOXPg6uqKZs2aYebMmcjOzpa8z/KqkBUZIiKiikij0UCj0RisU6vVUKvV5W57xIgRaNGiBSpXrow//vgDMTExSElJwcqVK8vdtpRYkSEiIpKaTmuUJTY2Fi4uLgZLbGxsod1OmTIl3wTeZ5ejR48CAEaNGoWwsDA0adIEAwYMwPLly7Fq1SrcuXPHVO9SmbAiQ0REJDUjzZGJiYnB6NGjDdYVVY0ZNmwYevXqVWSbvr6+Ba5v3bo1AODChQtwdXUtXaAmxESGiIjIQpR2GMnNzQ1ubm5l6ispKQkA4OnpWabnmwoTGSIiIomZ+00jDx06hMOHD6Ndu3ZwcXFBYmIiRo0ahddeew3e3t5yh1ckJjJERERSM/Mr+6rVasTHx2Pq1KnQaDTw8fHBwIEDMW7cOLlDKxYTGSIioudcixYtcPjwYbnDKBMmMkRERFKzkAviWSImMkRERBITTGQkw+vIEBERkcViRYaIiEhipri9wPOKiQwREZHEhDncobOCkn1oafHixYiMjMQ333wDAPjqq68QEBAAf39/fPTRR8jNzS3y+RqNBg8ePDBYcrI1RT6HiIjIlIRWZ5SF8pM1kZk+fTomTpyIzMxMjBgxAnPmzMGoUaPQp08fREZGYuXKlZg+fXqRbRR034mE1Z+b6BUQERGRnBRCCCFX57Vr18a8efPQo0cPnDx5EoGBgYiLi0OfPn0AAFu2bMG4ceNw/vz5Qtso6E6gW87cgsqm/HcCLY8WXs6y9g8A5pC8WyvljuAJ+b7l/3UiLUPuENDE3UnuEGClkDsCICtX/h+Hs1r+H0dWrhn8MMxAgIf0x+v0VR8bpR2X92YYpZ2KRNY5MtevX0dQUBAAoGnTprCyskKzZs3021u0aIFr164V2UZB951Q2TwweqxERERlxWEh6cg6tOTh4YHTp08DAM6fPw+tVqt/DADJycmoVq2aXOERERGRmZO1IhMREYH+/fvj9ddfx+7duzF+/HiMGTMGd+7cgUKhwMyZM/HWW2/JGSIREVG5sSIjHVkTmalTp8LOzg6HDx/GoEGDMH78eDRp0gTjxo3Do0eP0K1bt2In+xIREZk7nZnf/dqSyZrIKJVKTJw40WBdr1690KtXL5kiIiIiIkvCC+IRERFJjFf2lQ4TGSIiIolxjox0ZL+yLxEREVFZsSJDREQkMVZkpMNEhoiISGKcIyMdJjJEREQS07EiIxnOkSEiIiKLxYoMERGRxDhHRjoVMpFJ1+TKHQIyNLyKIwBY5ZrBrY4B6Mzg9tc6nfwx3M/KkTsEqJXy3/X5ZqZG7hCgsrKTOwTcfpQtdwhQWz8fAwNMZKTzfHyDiIiIqEKqkBUZIiIic8KzlqTDRIaIiEhiHFqSDoeWiIiIyGKxIkNERCQxVmSkw0SGiIhIYjrOkZEMh5aIiIjIYrEiQ0REJDEOLUmHiQwREZHEhJYXSZUKExkiIiKJ8Toy0uEcGSIiIrJYrMgQERFJjHNkpCNrInP9+nUsW7YMBw4cwPXr16FUKuHn54fu3bsjKioKSjO4uRwREVF5MZGRjmxDS0ePHkWDBg2wbds2ZGVl4dy5c2jRogUcHBwwZswYhIaGIiMjQ67wiIiIyALIlsiMHDkSo0aNQlJSEg4ePIi4uDicO3cOmzZtwsWLF/H48WN8/PHHxbaj0Wjw4MEDgyUnW2OCV0BERFQyOq3OKAvlJ1sic/z4cfTr10//OCIiAsePH8eNGzdQuXJlzJ07F99++22x7cTGxsLFxcVg2bFumZShExERlYrQ6YyyUH6yJTLVqlXD9evX9Y9v3LiB3NxcODs7AwDq1q2Lu3fvFttOTEwM0tPTDZZO/YdIFjcRERGZD9km+3bv3h2DBw/GvHnzoFarMX36dISFhcHOzg4AcPbsWVSvXr3YdtRqNdRqtcE6lU3xCRAREZGpcLKvdGRLZGbMmIHr16+jW7du0Gq1CA4Oxvr16/XbFQoFYmNj5QqPiIjIaIRWyB1ChSVbIuPo6Ij4+HhkZWUhNzcXjo6OBts7duwoU2RERERkKWS/IJ6tra3cIRAREUmKZxxJR/ZEhoiIqKITOg4tSYWJDBERkcR0nCMjGd40koiIiCwWKzJEREQS4+nX0mEiQ0REJDGefi0dDi0RERGRxWJFhoiISGKc7CsdJjJEREQS4xwZ6XBoiYiIiCxWhazIJKXelzsEuNrbyB0CVFYKuUMwGzohf1n3blaO3CEA9+QOwDzcepQtdwhm4Wam/O+D2lr+v6db+0jfh44XxJNMhUxkiIiIzAnPWpKO/KkwERERyW7mzJkICQmBvb09KlWqVOA+qamp6NatGxwcHODm5obhw4cjO1veyh4rMkRERBKzhJtGZmdn4+2330ZwcDBWrVqVb7tWq0XXrl1RtWpVHDhwAHfu3EFkZCSEEFi8eLEMET/BRIaIiEhiljC0NHXqVADA2rVrC9y+c+dOnD59GleuXIGXlxcAYP78+YiKisLMmTPh7OxsqlANMJEhIiKSmLESGY1GA41GY7BOrVZDrVYbpf2iHDp0CI0aNdInMQDQqVMnaDQaHDt2DO3atZM8hoJwjgwREZGFiI2NhYuLi8ESGxtrkr7T0tLg7u5usK5y5cqwsbFBWlqaSWIoCBMZIiIiiem0OqMsMTExSE9PN1hiYmIK7XfKlClQKBRFLkePHi3x61Ao8l/WQwhR4HpT4dASERGRxISRriNT2mGkYcOGoVevXkXu4+vrW6K2PDw8cOTIEYN19+7dQ05OTr5KjSkxkSEiIqqg3Nzc4ObmZpS2goODMXPmTFy/fh2enp4AnkwAVqvVCAwMNEofZcFEhoiISGKWcNPI1NRU3L17F6mpqdBqtThx4gQAoE6dOnB0dETHjh0REBCAfv36Yd68ebh79y7GjBmDgQMHynbGEmAGiUxmZiY2bNiAgwcPIi0tDQqFAu7u7mjTpg169+4NBwcHuUMkIiIqF0u4aeQnn3yCuLg4/ePmzZsDAPbu3Yu2bdtCqVTixx9/xAcffIA2bdrAzs4OERER+PTTT+UKGQCgEEK+m9CcPn0a4eHhePToEcLCwuDu7g4hBG7evIn9+/fDwcEBO3fuREBAQKnaff+bE9IEXAov+1eTOwTea+kp5nCvpesPNcXvJLEqtiq5QzAL5nCvpXqu8v+RxnstPdGraXXJ+9jbvJVR2mmXdKT4nZ4zslZkhg4dipdeeglxcXGwsTG8yWJ2djaioqIwdOhQ7N27V6YIiYiIys8SLohnqWRNZI4cOYKjR4/mS2IAwMbGBh999BFeeOEFGSIjIiIyHkuYI2OpZE1kKleujPPnzxc6dHThwgVUrly5yDYKusqhNicbSlX+5IiIiIgqFlkHJwcOHIjIyEh8+umnOHnyJNLS0nDjxg2cPHkSn376KaKjozFo0KAi2yjoKocntq420SsgIiIqntDpjLJQfrJO9gWAOXPmYNGiRfozloAnVwn08PDAyJEjMW7cuCKfX1BFZtS2v2WvyHCyr3nhZN8nONn3CU72fYKTfZ8wxWTfHfVaGKWdTueOG6WdikT206/Hjx+P8ePHIyUlRX+vBg8PD/j5+ZXo+QVd5VDuJIaIiOhpnOwrHflT4f/n5+eH4OBgBAcH65OYK1euIDo6WubIiIiIyFyZTSJTkLt37xpcnIeIiMgSCa3OKAvlJ+vQUkJCQpHbL168aKJIiIiIpMPTr6UjayLTvXt3KBQKFDXfWM5bgxMREZF5k3VoydPTE5s3b4ZOpytwOX6cs7OJiMjyCa0wykL5yZrIBAYGFpmsFFetISIisgQ6IYyyUH6yDi2NHTsWmZmZhW6vU6cO77NEREREhZI1kQkNDS1yu4ODA8LCwkwUDRERkTS0rKZIRvYL4hEREVV0nN4iHbO+jgwRERFRUViRISIikhiHlqTDRIaIiEhiHFqSToVMZNbNWy53CPhPZXe5Q4Bvy9Zyh4BLiYflDgEAkHnritwhoNZLr8sdAh7evi13CMh6cEvuEJBx7R+5Q0C9Dt3lDgH//nlM7hBQ78UQuUMwyd2vWZGRDufIEBERkcWqkBUZIiIic8KhJekwkSEiIpIYh5akw6ElIiIislisyBAREUmMQ0vSYSJDREQkMSYy0uHQEhEREVksVmSIiIgkxsm+0mEiQ0REJDEOLUnHrIeWbty4gWnTpskdBhEREZkps05k0tLSMHXqVLnDICIiKhetEEZZKD9Zh5b+/PPPIrefPXvWRJEQERFJh0NL0pE1kWnWrBkUCgVEAVlm3nqFQiFDZERERMbDaop0ZE1kXF1dMWfOHHTo0KHA7cnJyejWrVuRbWg0Gmg0GoN1QqeFwkpptDiJiIjIPMmayAQGBuLatWvw8fEpcPv9+/cLrNY8LTY2Nt88GqVnIKyrBxktTiIiovLg0JJ0ZJ3sO2jQIPj6+ha63dvbG2vWrCmyjZiYGKSnpxssSs/mRo6UiIio7DjZVzqyVmTeeOONIrdXrlwZkZGRRe6jVquhVqsN1nFYiYiI6Plg1qdfX7lyBdHR0XKHQUREVC46Iy2Un1knMnfv3kVcXJzcYRAREZULh5akI+vQUkJCQpHbL168aKJIiIiIyBLJmsh079690OvI5OF1ZIiIyNLxrCXpyDq05Onpic2bN0On0xW4HD9+XM7wiIiIjIJDS9KRNZEJDAwsMlkprlpDREREzzdZh5bGjh2LzMzMQrfXqVMHe/fuNWFERERExsehJenImsiEhoYWud3BwQFhYWEmioaIiEgaHBaSjqyJDBER0fOAFRnpmPV1ZIiIiIiKwooMERGRxDi0JB0mMkRERBLj0JJ0OLRERERElktQPllZWWLy5MkiKyuLMTAGxsAYGIOZxmBOcZB8FEJw4O5ZDx48gIuLC9LT0+Hs7MwYGANjYAyMwQxjMKc4SD4cWiIiIiKLxUSGiIiILBYTGSIiIrJYTGQKoFarMXnyZKjVasbAGBgDY2AMZhqDOcVB8uFkXyIiIrJYrMgQERGRxWIiQ0RERBaLiQwRERFZLCYyREREZLGYyDxj6dKl8PPzg62tLQIDA/Hbb7+ZtP9ff/0V3bp1g5eXFxQKBbZu3WrS/gEgNjYWLVu2hJOTE6pVq4bu3bvj7NmzJo1h2bJlaNKkCZydneHs7Izg4GD8/PPPJo3habGxsVAoFBg5cqRJ+50yZQoUCoXB4uHhYdIYAODff/9F37594erqCnt7ezRr1gzHjh0zWf++vr753geFQoGhQ4eaLIbc3Fx8/PHH8PPzg52dHWrVqoVp06ZBp9OZLAYAyMjIwMiRI+Hj4wM7OzuEhIQgMTFRsv6KOyYJITBlyhR4eXnBzs4Obdu2RXJysklj+O6779CpUye4ublBoVDgxIkTRu2fzBsTmafEx8dj5MiRmDhxIpKSkhAaGorOnTsjNTXVZDFkZmaiadOmWLJkicn6fNb+/fsxdOhQHD58GLt27UJubi46duyIzMxMk8VQo0YNzJ49G0ePHsXRo0fRvn17vP7660Y/QJZEYmIiVqxYgSZNmpi8bwBo2LAhrl+/rl9OnTpl0v7v3buHNm3aQKVS4eeff8bp06cxf/58VKpUyWQxJCYmGrwHu3btAgC8/fbbJothzpw5WL58OZYsWYIzZ85g7ty5mDdvHhYvXmyyGABgwIAB2LVrF7766iucOnUKHTt2xMsvv4x///1Xkv6KOybNnTsXCxYswJIlS5CYmAgPDw+Eh4cjIyPDZDFkZmaiTZs2mD17ttH6JAsi652ezMwLL7wgBg8ebLDO399fTJgwQZZ4AIgtW7bI0vfTbt68KQCI/fv3yxpH5cqVxcqVK03aZ0ZGhqhbt67YtWuXCAsLEyNGjDBp/5MnTxZNmzY1aZ/PGj9+vHjxxRdljeFZI0aMELVr1xY6nc5kfXbt2lVER0cbrOvRo4fo27evyWJ49OiRUCqV4ocffjBY37RpUzFx4kTJ+3/2mKTT6YSHh4eYPXu2fl1WVpZwcXERy5cvN0kMT0tJSREARFJSkiR9k3liReb/ZWdn49ixY+jYsaPB+o4dO+LgwYMyRWUe0tPTAQBVqlSRpX+tVotNmzYhMzMTwcHBJu176NCh6Nq1K15++WWT9vu08+fPw8vLC35+fujVqxcuXrxo0v4TEhIQFBSEt99+G9WqVUPz5s3x5ZdfmjSGp2VnZ2P9+vWIjo6GQqEwWb8vvvgidu/ejXPnzgEATp48iQMHDqBLly4miyE3NxdarRa2trYG6+3s7HDgwAGTxZEnJSUFaWlpBsdNtVqNsLCw5/64SaZjLXcA5uL27dvQarVwd3c3WO/u7o60tDSZopKfEAKjR4/Giy++iEaNGpm071OnTiE4OBhZWVlwdHTEli1bEBAQYLL+N23ahOPHj0s6/6A4rVq1wrp161CvXj3cuHEDM2bMQEhICJKTk+Hq6mqSGC5evIhly5Zh9OjR+Oijj/DHH39g+PDhUKvV6N+/v0lieNrWrVtx//59REVFmbTf8ePHIz09Hf7+/lAqldBqtZg5cyZ69+5tshicnJwQHByM6dOno0GDBnB3d8fGjRtx5MgR1K1b12Rx5Mk7NhZ03Lx8+bLJ46HnExOZZzz7F54QwqR/9ZmbYcOG4c8//5Tlr7369evjxIkTuH//PjZv3ozIyEjs37/fJMnMlStXMGLECOzcuTPfX7+m1LlzZ/2/GzdujODgYNSuXRtxcXEYPXq0SWLQ6XQICgrCrFmzAADNmzdHcnIyli1bJksis2rVKnTu3BleXl4m7Tc+Ph7r16/Hhg0b0LBhQ5w4cQIjR46El5cXIiMjTRbHV199hejoaFSvXh1KpRItWrRAREQEjh8/brIYnsXjJsmJicz/c3Nzg1KpzFd9uXnzZr6/Np4XH374IRISEvDrr7+iRo0aJu/fxsYGderUAQAEBQUhMTERixYtwhdffCF538eOHcPNmzcRGBioX6fVavHrr79iyZIl0Gg0UCqVksfxLAcHBzRu3Bjnz583WZ+enp75kscGDRpg8+bNJoshz+XLl/HLL7/gu+++M3nfY8eOxYQJE9CrVy8ATxLLy5cvIzY21qSJTO3atbF//35kZmbiwYMH8PT0RM+ePeHn52eyGPLknUGXlpYGT09P/frn+bhJpsc5Mv/PxsYGgYGB+rMh8uzatQshISEyRSUPIQSGDRuG7777Dnv27JHlAFkQIQQ0Go1J+urQoQNOnTqFEydO6JegoCD06dMHJ06ckCWJAQCNRoMzZ84Y/KchtTZt2uQ7/f7cuXPw8fExWQx51qxZg2rVqqFr164m7/vRo0ewsjI8ZCqVSpOffp3HwcEBnp6euHfvHnbs2IHXX3/d5DH4+fnBw8PD4LiZnZ2N/fv3P3fHTZIPKzJPGT16NPr164egoCAEBwdjxYoVSE1NxeDBg00Ww8OHD3HhwgX945SUFJw4cQJVqlSBt7e3SWIYOnQoNmzYgO+//x5OTk76KpWLiwvs7OxMEsNHH32Ezp07o2bNmsjIyMCmTZuwb98+bN++3ST9Ozk55ZsT5ODgAFdXV5POFRozZgy6desGb29v3Lx5EzNmzMCDBw9MWgEYNWoUQkJCMGvWLLzzzjv4448/sGLFCqxYscJkMQBPhrjWrFmDyMhIWFub/tDVrVs3zJw5E97e3mjYsCGSkpKwYMECREdHmzSOHTt2QAiB+vXr48KFCxg7dizq16+Pd999V5L+ijsmjRw5ErNmzULdunVRt25dzJo1C/b29oiIiDBZDHfv3kVqaiquXbsGAPrE28PDQ5brLpGJyXnKlDn6/PPPhY+Pj7CxsREtWrQw+SnHe/fuFQDyLZGRkSaLoaD+AYg1a9aYLIbo6Gj951C1alXRoUMHsXPnTpP1XxA5Tr/u2bOn8PT0FCqVSnh5eYkePXqI5ORkk8YghBDbtm0TjRo1Emq1Wvj7+4sVK1aYPIYdO3YIAOLs2bMm71sIIR48eCBGjBghvL29ha2trahVq5aYOHGi0Gg0Jo0jPj5e1KpVS9jY2AgPDw8xdOhQcf/+fcn6K+6YpNPpxOTJk4WHh4dQq9XipZdeEqdOnTJpDGvWrClw++TJk40aB5knhRBCmChnIiIiIjIqzpEhIiIii8VEhoiIiCwWExkiIiKyWExkiIiIyGIxkSEiIiKLxUSGiIiILBYTGSIiIrJYTGSIqFj79u2DQqHA/fv35Q6FiMgAExkiCxEVFYXu3bvnW88kg4ieZ0xkiIiIyGIxkSGqYA4ePIiXXnoJdnZ2qFmzJoYPH47MzEz99vXr1yMoKAhOTk7w8PBAREQEbt68adDGTz/9hHr16sHOzg7t2rXDpUuXTPwqiIhKhokMUQVy6tQpdOrUCT169MCff/6J+Ph4HDhwAMOGDdPvk52djenTp+PkyZPYunUrUlJSEBUVpd9+5coV9OjRA126dMGJEycwYMAATJgwQYZXQ0RUPN40kshCREVFYf369bC1tTVYr9VqkZWVhXv37mH48OGws7PDF198od9+4MABhIWFITMzM99zASAxMREvvPACMjIy4OjoiI8++ghbt25FcnIyFAoFAGDChAmYM2cO7t27h0qVKkn6OomISsNa7gCIqOTatWuHZcuWGaw7cuQI+vbtCwA4duwYLly4gK+//lq/XQgBnU6HlJQUNGjQAElJSZgyZQpOnDiBu3fvQqfTAQBSU1MREBCAM2fOoHXr1vokBgCCg4NN8OqIiEqPiQyRBXFwcECdOnUM1l29elX/b51Oh0GDBmH48OH5nuvt7Y3MzEx07NgRHTt2xPr161G1alWkpqaiU6dOyM7OBvAk8SEishRMZIgqkBYtWiA5OTlfspPn1KlTuH37NmbPno2aNWsCAI4ePWqwT0BAALZu3Wqw7vDhw5LES0RUXpzsS1SBjB8/HocOHcLQoUNx4sQJnD9/HgkJCfjwww8BPKnK2NjYYPHixbh48SISEhIwffp0gzYGDx6Mf/75B6NHj8bZs2exYcMGrF27VoZXQ0RUPCYyRBVIkyZNsH//fpw/fx6hoaFo3rw5Jk2aBE9PTwBA1apVsXbtWvznP/9BQEAAZs+ejU8//dSgDW9vb2zevBnbtm1D06ZNsXz5csyaNUuOl0NEVCyetUREREQWixUZIiIislhMZIiIiMhiMZEhIiIii8VEhoiIiCwWExkiIiKyWExkiIiIyGIxkSEiIiKLxUSGiIiILBYTGSIiIrJYTGSIiIjIYjGRISIiIovFRIaIiIgs1v8Bk9ZUAfpvia4AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# if you want to visualize the effect of each head to logits in a heatmap...\n",
    "# this plot is from zero-ablation\n",
    "\n",
    "cm_max = max(avg_h_ctbn.flatten())\n",
    "cm_min = -cm_max\n",
    "\n",
    "s = sns.heatmap(avg_h_ctbn, xticklabels = range(12), yticklabels = range(12), cmap='RdBu', cbar_kws={'label': '%'},\n",
    "               vmin=cm_min, vmax=cm_max)\n",
    "s.set(xlabel='Head', ylabel='Layer')\n",
    "plt.title(f\"Contribution of Heads to Prediction Logits: CD\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "1c99a633-533e-4317-b48e-696778dff9e4",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# if you want to visualize the effect of each head to logits in a heatmap...\n",
    "# this plot is from mean-ablation\n",
    "\n",
    "s = sns.heatmap(back, xticklabels = range(12), yticklabels = range(12), cmap='Blues', cbar_kws={'label': '%'})\n",
    "s.set(xlabel='Head', ylabel='Layer')\n",
    "plt.title(f\"Contribution of Heads to Prediction Logits: CD\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "77bc2046-23b8-4cae-8c45-51f6eae9d9e3",
   "metadata": {
    "user_expressions": []
   },
   "source": [
    "## Extract top N heads and their attended words"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1dc01456-4ee4-4b17-8fe1-ef7f6c797797",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "# a is the calcuated effect of each head on the logits\n",
    "\n",
    "N = 6\n",
    "a = back\n",
    "top_idxs = np.argsort(a.ravel())[-N:]\n",
    "rows, cols = top_idxs//12, top_idxs%12\n",
    "\n",
    "positives = []\n",
    "for r, c in zip(rows, cols):\n",
    "    positives.append((r, c))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "47541c67-97b5-4f0a-b385-05c3facdd7a2",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def collect_attended_tokens(positives_heads, device, tokenizer, N=100, Z_thres=2, POS=0):\n",
    "    index_lst = random.sample(range(0, len(documents_full)), N)\n",
    "    docs = [documents_full[i] for i in index_lst]\n",
    "    \n",
    "    collect = collections.defaultdict(int)\n",
    "    for doc in docs:\n",
    "        encoding = get_encoding(doc, tokenizer, device)\n",
    "        raw_rel, raw_irrel, raw_att_probs_lst, _ = prop_classifier_model_patched(encoding, model, device, [], output_att_prob=True)\n",
    "        raw_att_probs = torch.stack(raw_att_probs_lst).cpu().numpy()\n",
    "\n",
    "        avg_att_m = np.zeros((512))\n",
    "        for level, h in positives:\n",
    "            att_m = raw_att_probs[level, h, POS, :]\n",
    "            avg_att_m += att_m\n",
    "\n",
    "        avg_att_m /= len(positives)\n",
    "        \n",
    "        # convert to word level\n",
    "        interval_dict, word_lst = compute_word_intervals(encoding, tokenizer)\n",
    "        word_att_m = combine_token_attn(interval_dict, avg_att_m)\n",
    "        \n",
    "        Z = (word_att_m - np.mean(word_att_m)) / np.std(word_att_m)\n",
    "\n",
    "        positive_words = np.where(Z > Z_thres)\n",
    "        \n",
    "        for w_idx in positive_words[0]:\n",
    "            w = word_lst[w_idx]\n",
    "            #collect[w] += 1\n",
    "            collect[w] += word_att_m[w_idx]\n",
    "            \n",
    "    return collect"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "f08e38c6-1bfd-4deb-b732-bea3e7dbb6ae",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/html": [
       "<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"><span style=\"color: #800000; text-decoration-color: #800000\">╭─────────────────────────────── </span><span style=\"color: #800000; text-decoration-color: #800000; font-weight: bold\">Traceback </span><span style=\"color: #bf7f7f; text-decoration-color: #bf7f7f; font-weight: bold\">(most recent call last)</span><span style=\"color: #800000; text-decoration-color: #800000\"> ────────────────────────────────╮</span>\n",
       "<span style=\"color: #800000; text-decoration-color: #800000\">│</span> <span style=\"color: #bfbf7f; text-decoration-color: #bfbf7f\">/tmp/ipykernel_825660/</span><span style=\"color: #808000; text-decoration-color: #808000; font-weight: bold\">1888346581.py</span>:<span style=\"color: #0000ff; text-decoration-color: #0000ff\">1</span> in <span style=\"color: #00ff00; text-decoration-color: #00ff00\">&lt;module&gt;</span>                                                <span style=\"color: #800000; text-decoration-color: #800000\">│</span>\n",
       "<span style=\"color: #800000; text-decoration-color: #800000\">│</span>                                                                                                  <span style=\"color: #800000; text-decoration-color: #800000\">│</span>\n",
       "<span style=\"color: #800000; text-decoration-color: #800000\">│</span> <span style=\"color: #800000; text-decoration-color: #800000; font-style: italic\">[Errno 2] No such file or directory: '/tmp/ipykernel_825660/1888346581.py'</span>                       <span style=\"color: #800000; text-decoration-color: #800000\">│</span>\n",
       "<span style=\"color: #800000; text-decoration-color: #800000\">│</span>                                                                                                  <span style=\"color: #800000; text-decoration-color: #800000\">│</span>\n",
       "<span style=\"color: #800000; text-decoration-color: #800000\">│</span> <span style=\"color: #bfbf7f; text-decoration-color: #bfbf7f\">/tmp/ipykernel_825660/</span><span style=\"color: #808000; text-decoration-color: #808000; font-weight: bold\">4109569869.py</span>:<span style=\"color: #0000ff; text-decoration-color: #0000ff\">12</span> in <span style=\"color: #00ff00; text-decoration-color: #00ff00\">collect_attended_tokens</span>                                <span style=\"color: #800000; text-decoration-color: #800000\">│</span>\n",
       "<span style=\"color: #800000; text-decoration-color: #800000\">│</span>                                                                                                  <span style=\"color: #800000; text-decoration-color: #800000\">│</span>\n",
       "<span style=\"color: #800000; text-decoration-color: #800000\">│</span> <span style=\"color: #800000; text-decoration-color: #800000; font-style: italic\">[Errno 2] No such file or directory: '/tmp/ipykernel_825660/4109569869.py'</span>                       <span style=\"color: #800000; text-decoration-color: #800000\">│</span>\n",
       "<span style=\"color: #800000; text-decoration-color: #800000\">╰──────────────────────────────────────────────────────────────────────────────────────────────────╯</span>\n",
       "<span style=\"color: #ff0000; text-decoration-color: #ff0000; font-weight: bold\">ValueError: </span>too many values to unpack <span style=\"font-weight: bold\">(</span>expected <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">2</span><span style=\"font-weight: bold\">)</span>\n",
       "</pre>\n"
      ],
      "text/plain": [
       "\u001b[31m╭─\u001b[0m\u001b[31m──────────────────────────────\u001b[0m\u001b[31m \u001b[0m\u001b[1;31mTraceback \u001b[0m\u001b[1;2;31m(most recent call last)\u001b[0m\u001b[31m \u001b[0m\u001b[31m───────────────────────────────\u001b[0m\u001b[31m─╮\u001b[0m\n",
       "\u001b[31m│\u001b[0m \u001b[2;33m/tmp/ipykernel_825660/\u001b[0m\u001b[1;33m1888346581.py\u001b[0m:\u001b[94m1\u001b[0m in \u001b[92m<module>\u001b[0m                                                \u001b[31m│\u001b[0m\n",
       "\u001b[31m│\u001b[0m                                                                                                  \u001b[31m│\u001b[0m\n",
       "\u001b[31m│\u001b[0m \u001b[3;31m[Errno 2] No such file or directory: '/tmp/ipykernel_825660/1888346581.py'\u001b[0m                       \u001b[31m│\u001b[0m\n",
       "\u001b[31m│\u001b[0m                                                                                                  \u001b[31m│\u001b[0m\n",
       "\u001b[31m│\u001b[0m \u001b[2;33m/tmp/ipykernel_825660/\u001b[0m\u001b[1;33m4109569869.py\u001b[0m:\u001b[94m12\u001b[0m in \u001b[92mcollect_attended_tokens\u001b[0m                                \u001b[31m│\u001b[0m\n",
       "\u001b[31m│\u001b[0m                                                                                                  \u001b[31m│\u001b[0m\n",
       "\u001b[31m│\u001b[0m \u001b[3;31m[Errno 2] No such file or directory: '/tmp/ipykernel_825660/4109569869.py'\u001b[0m                       \u001b[31m│\u001b[0m\n",
       "\u001b[31m╰──────────────────────────────────────────────────────────────────────────────────────────────────╯\u001b[0m\n",
       "\u001b[1;91mValueError: \u001b[0mtoo many values to unpack \u001b[1m(\u001b[0mexpected \u001b[1;36m2\u001b[0m\u001b[1m)\u001b[0m\n"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "positive_attended_token_freq = collect_attended_tokens(positives, device, tokenizer, N=500, Z_thres=2)\n",
    "positive_attended_token_freq = sorted(positive_attended_token_freq.items(), key=lambda k_v: k_v[1], reverse=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 109,
   "id": "83551f85-b6a3-489c-bf38-67ac3901825b",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "with open('result_top1.json', 'w') as fp:\n",
    "    json.dump(positive_attended_token_freq, fp)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "1dc3e012-2859-4ba9-9543-746727379c0f",
   "metadata": {
    "user_expressions": []
   },
   "source": [
    "## Extra: Ablate with means of different label groups"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0ef70143-71be-4ca1-b608-936a7bb4cae4",
   "metadata": {},
   "outputs": [],
   "source": [
    "label_idx_4_5 = [] #934\n",
    "label_idx_4 = [] #855\n",
    "label_idx_5 = [] #79\n",
    "label_idx_3 = [] #1934\n",
    "\n",
    "for i, x in enumerate(labels_full):\n",
    "    if x == '4':\n",
    "        label_idx_4_5.append(i)\n",
    "        label_idx_4.append(i)\n",
    "    elif x == '5':\n",
    "        label_idx_4_5.append(i)\n",
    "        label_idx_5.append(i)\n",
    "    elif x == '3':\n",
    "        label_idx_3.append(i)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "44ca1f3b-074b-40c8-aa30-383e70dc7c95",
   "metadata": {},
   "outputs": [],
   "source": [
    "path = f\"{base_dir}/output/{args['task']}/{args['model_type']}_{args['field']}/h_to_logits\"\n",
    "os.makedirs(path, exist_ok=True)\n",
    "\n",
    "#with open(os.path.join(path, f\"mean_head_out_res_{NUM_SAMPLES}.pkl\"), 'wb') as handle:\n",
    "#    pickle.dump(mean_act, handle)\n",
    "    \n",
    "with open(os.path.join(path, f\"mean_head_out_res_label4_5_500.pkl\"), 'rb') as handle:\n",
    "    back = pickle.load(handle)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "810affd9-85bc-4136-8b6f-77b0fd3ea2c4",
   "metadata": {},
   "outputs": [],
   "source": [
    "with torch.no_grad():\n",
    "    NUM_SAMPLES = 200\n",
    "    POS = 0\n",
    "    index_lst = random.sample(label_idx_3, NUM_SAMPLES)\n",
    "    docs = [documents_full[i] for i in index_lst]\n",
    "    labels = [labels_full[i] for i in index_lst]\n",
    "\n",
    "    h_ctbn = []\n",
    "    for doc, label in tqdm.tqdm(zip(docs, labels)):\n",
    "        encoding = get_encoding(doc, tokenizer, device)\n",
    "        label_idx = le_dict[label]\n",
    "        h_ctbn_tot = patch_h_to_logits_at_pos(encoding, model, POS, label_idx, device,\n",
    "                                              mean_act=back, mean_ablated=True)\n",
    "        h_ctbn.append(h_ctbn_tot) #[12, 12] = [Layer, HEAD]\n",
    "\n",
    "    h_ctbn = np.stack(h_ctbn)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "26f3e65e-d0ba-4899-85ce-881f9ed50261",
   "metadata": {
    "user_expressions": []
   },
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.12.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
