{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "7a232b18-9e7e-4536-a69d-1e1d95c5fc73",
   "metadata": {},
   "source": [
    "## Motivational Shapley Value Experiment"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "16468b5d-4449-4487-9be9-b72936c54e39",
   "metadata": {},
   "source": [
    "Run this notebook to recreate the example in Appendix A."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "7f6a2087-0426-439e-b62b-9633acf0db35",
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "eef667fd-7ff3-45c9-9b91-c44a668e926e",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "os.chdir(\"..\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "9a96922e-acac-4049-a550-70a2d2553b07",
   "metadata": {},
   "source": [
    "Training commands: Train models with the following command:\n",
    "```\n",
    ".scripts/train_models_linear.sh\n",
    "```"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "8cc340d2-2850-4d12-9011-bb42ab2b04c5",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "from slalom_explanations.transformer_models import DistilBert, GPT2, Bert\n",
    "from transformers import AutoTokenizer"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "c0dce511-f216-4e57-8684-7b4f49594bbc",
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_logits_for_single_token(model, tokenizer, n_max=10, use_cls=True, input_str=\"perfect\", use_device=\"cuda:2\"):\n",
    "    logit_results = []\n",
    "    cls_tok = torch.tensor([101])\n",
    "    input_tok = torch.tensor(tokenizer.convert_tokens_to_ids(input_str)).reshape(1)\n",
    "    sep_tok = torch.tensor([102])\n",
    "    with torch.no_grad():\n",
    "        x = input_tok\n",
    "        for i in range(n_max):\n",
    "            #print(x)\n",
    "            if use_cls:\n",
    "                input = torch.cat((cls_tok, x, sep_tok))\n",
    "            else:\n",
    "                input =x\n",
    "            logit_score = model(input.reshape(1,-1).to(use_device))[\"logits\"]\n",
    "            logit_results.append(logit_score.detach().cpu())\n",
    "            x = torch.cat((x, input_tok), dim=0)\n",
    "        \n",
    "    return torch.stack(logit_results)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "8d464284-fe51-4993-b7cb-8f39e141ca22",
   "metadata": {},
   "source": [
    "### 2 layers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "7d34567a-1808-4a5d-a042-4aa171ef2197",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_3552432/1516351806.py:6: FutureWarning: You are using `torch.load` with `weights_only=False` (the current default value), which uses the default pickle module implicitly. It is possible to construct malicious pickle data which will execute arbitrary code during unpickling (See https://github.com/pytorch/pytorch/blob/main/SECURITY.md#untrusted-models for more details). In a future release, the default value for `weights_only` will be flipped to `True`. This limits the functions that could be executed during unpickling. Arbitrary objects will no longer be allowed to be loaded via this mode unless they are explicitly allowlisted by the user via `torch.serialization.add_safe_globals`. We recommend you start setting `weights_only=True` for any use case where you don't have full control of the loaded file. Please open an issue on GitHub for any issues related to this experimental feature.\n",
      "  s_dict = torch.load(f\"{prefix_path}/{run}\")\n",
      "Some weights of DistilBertForSequenceClassification were not initialized from the model checkpoint at distilbert/distilbert-base-uncased and are newly initialized: ['classifier.bias', 'classifier.weight', 'pre_classifier.bias', 'pre_classifier.weight']\n",
      "You should probably TRAIN this model on a down-stream task to be able to use it for predictions and inference.\n",
      "Some weights of GPT2ForSequenceClassification were not initialized from the model checkpoint at gpt2 and are newly initialized: ['score.weight']\n",
      "You should probably TRAIN this model on a down-stream task to be able to use it for predictions and inference.\n"
     ]
    }
   ],
   "source": [
    "prefix_path= \"models_lin_bin_rerun\"\n",
    "run_ids = [\"sparsebow_bert_2_r0.pt\", \"sparsebow_gpt2_2_r0.pt\"] ## insert run_ids here.\n",
    "\n",
    "model_list, tok_list = [], []\n",
    "for run in run_ids:\n",
    "    s_dict = torch.load(f\"{prefix_path}/{run}\")\n",
    "    # extract num_layers, num_heads\n",
    "    parts = run.split(\"_\")\n",
    "    n_heads = 12\n",
    "    n_layers = int(parts[-2])\n",
    "    use_device = \"cuda:2\"\n",
    "    if \"gpt2\" in run:\n",
    "        model_obj = GPT2(n_layer=n_layers, n_head=n_heads)\n",
    "        tokenizer = AutoTokenizer.from_pretrained('distilgpt2', use_fast=True)\n",
    "    else:\n",
    "        if \"distilbert\" in run:\n",
    "            model_obj = DistilBert(n_layers=n_layers, n_heads=n_heads)\n",
    "        else:\n",
    "            model_obj = Bert(n_layers=n_layers, n_heads=n_heads)\n",
    "        tokenizer = AutoTokenizer.from_pretrained('bert-base-uncased', use_fast=True)\n",
    "    model_obj.model.load_state_dict(s_dict)\n",
    "    for p in model_obj.model.parameters():\n",
    "        p.requires_grad_(False)\n",
    "        \n",
    "    model_trained = model_obj.model.to(use_device).eval()\n",
    "    model_list.append(model_trained)\n",
    "    tok_list.append(tokenizer)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "id": "707466fd-8ab0-4879-823c-db8baebaba9a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([0.5247], device='cuda:2')\n",
      "tensor([0.5176], device='cuda:2')\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7fc9144ca0e0>"
      ]
     },
     "execution_count": 16,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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hvbt2xMREcGgQYNYt25dtdsvW7YMwzAqLZmZmQ0qXERERFqXgCZPO3nyJKNHj+aiiy7iww8/JDExkZ07d5KQkFDrvtu3b/cbo5yUlBR4tSIiItJqBRRKHnvsMdLT05k/f75vXbdu3eq0b1JSEvHx8QEVJyIiIm1HQJdv/vWvf3HmmWdy5ZVXkpSUxNChQ3nxxRfrtO8ZZ5xBamoqP/7xj/nyyy9r3NbtdpOTk+O3iIiISOsWUCjZs2cPzz33HL169WLx4sXccsst3Hbbbbz88svV7pOamsrzzz/PO++8wzvvvEN6ejoXXnghGzZsqHafOXPmEBcX51vS09MDKVNERERaoIDufRMWFsaZZ57JypUrfetuu+021q5dy6pVq+p80AsuuIDOnTvz97//vcrn3W43brfb9zgnJ4f09HTd+0ZERKQFadJ736SmptK/f3+/df369SMjIyOgIkeMGMGuXbuqfd7lchEbG+u3iIiISOsWUCgZPXo027dv91u3Y8cOunTpEtBBN23aRGpqakD7iIiISOsW0OibO++8k3POOYdHHnmEq666iq+++ooXXniBF154wbfNjBkzOHjwIK+88goATz31FN26dWPAgAEUFhby0ksv8emnn/Lxxx837jsRERGRFi2gUHLWWWexaNEiZsyYwcMPP0y3bt146qmnmDx5sm+bw4cP+13OKSoq4u677+bgwYNERkYyePBgPvnkEy666KLGexciIiLS4gXU0TVYAu0oIyIiIsHXpB1dRURERJqKQomIiIiEBIUSERERCQkKJSIiIhISFEpEREQkJCiUiIiISEhQKBEREZGQoFAiIiIiIUGhREREREKCQomIiIiEBIUSERERCQkKJSIiIhISFEpEREQkJCiUiIiISEhQKBEREZGQoFAiIiIiIUGhREREREKCQomIiIiEBIUSERERCQkKJSIiIhISFEpEREQkJCiUiIiISEhQKBEREZGQoFAiIiIiIUGhREREREKCQomIiIiEBIUSERERCQkKJSIiIhISFEpEREQkJCiUiIiISEhQKBEREZGQoFAiIiIiIUGhREREREKCQomIiIiEBIUSERERCQkKJSIiIhISFEpEREQkJCiUiIiISEhQKBEREZGQoFAiIiIiIUGhREREREKCQomIiIiEBIUSERERCQkKJSIiIhISAg4lBw8e5JprrqF9+/ZE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fOtTUlIoKioiKyvLb3t91yzVfW5VGTlyJECb/76FhYXRs2dPhg8fzpw5cxgyZAh//vOfG/W71iZDSVhYGMOHD2fp0qW+dV6vl6VLlzJq1KggVtay5OXlsXv3blJTU4NdSovRrVs3UlJS/L57OTk5rFmzRt+9ABw4cIDjx4+36e+eaZrceuutLFq0iE8//ZRu3br5PT98+HCcTqffd2379u1kZGS06e9abZ9bVTZt2gTQpr9vVfF6vbjd7sb9rjVuX9yWY+HChabL5TIXLFhgfvvtt+ZNN91kxsfHm5mZmcEuLWTdfffd5rJly8y9e/eaX375pTl27FizQ4cO5tGjR4NdWkjJzc01N27caG7cuNEEzCeeeMLcuHGjuW/fPtM0TfPRRx814+PjzX/+85/mN998Y15xxRVmt27dzIKCgiBXHjw1fWa5ubnmPffcY65atcrcu3ev+cknn5jDhg0ze/XqZRYWFga79KC55ZZbzLi4OHPZsmXm4cOHfUt+fr5vm5tvvtns3Lmz+emnn5rr1q0zR40aZY4aNSqIVQdfbZ/brl27zIcffthct26duXfvXvOf//yn2b17d/P8888PcuXBdf/995vLly839+7da37zzTfm/fffbxqGYX788cemaTbed63NhhLTNM3/+7//Mzt37myGhYWZI0aMMFevXh3skkLaL37xCzM1NdUMCwszO3bsaP7iF78wd+3aFeyyQs5nn31mApWWa6+91jRNa1jw7373OzM5Odl0uVzmmDFjzO3btwe36CCr6TPLz883L774YjMxMdF0Op1mly5dzBtvvLHN/wFR1ecFmPPnz/dtU1BQYP761782ExISzMjISHPixInm4cOHg1d0CKjtc8vIyDDPP/98s127dqbL5TJ79uxp3nvvvWZ2dnZwCw+y66+/3uzSpYsZFhZmJiYmmmPGjPEFEtNsvO+aYZqmWc+WGxEREZFG0yb7lIiIiEjoUSgRERGRkKBQIiIiIiFBoURERERCgkKJiIiIhASFEhEREQkJCiUiIiISEhRKREREJCQolIiIiEhIUCgRERGRkKBQIiIiIiFBoURERERCwv8DPOUPH0zQCDQAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "input_str=\"perfect\",\n",
    "for r, run in enumerate(run_ids):\n",
    "    if \"bert\" in run:\n",
    "        cls_tok = torch.tensor([101, 102])\n",
    "        base_logit = model_obj.model(cls_tok.reshape(1,-1).to(use_device))[\"logits\"]\n",
    "        base_logit = base_logit[:,1]-base_logit[:,0]\n",
    "        print(base_logit)\n",
    "        w_cls = compute_logits_for_single_token(model_list[r], tok_list[r], n_max=30, use_cls=True, use_device=use_device, input_str=input_str)\n",
    "        w_cls = w_cls[:,:,1]-w_cls[:,:,0]\n",
    "        plt.plot(w_cls, \"-x\")\n",
    "    else: # GPT\n",
    "        neutral_tok = torch.tensor(tok_list[r].convert_tokens_to_ids(\"the\"))\n",
    "        #print(neutral_tok)\n",
    "        base_logit = model_obj.model(neutral_tok.reshape(1,-1).to(use_device))[\"logits\"]\n",
    "        base_logit = base_logit[:,1]-base_logit[:,0]\n",
    "        print(base_logit)\n",
    "        wo_cls = compute_logits_for_single_token(model_list[r], tok_list[r], n_max=30, use_cls=False, input_str=input_str)\n",
    "        wo_cls = wo_cls[:,:,1]-wo_cls[:,:,0]\n",
    "        lin = torch.arange(30)*(wo_cls[0]-base_logit.cpu())+base_logit.cpu()\n",
    "plt.plot(wo_cls, \"-x\", label=\"bert\")\n",
    "plt.plot(w_cls, \"-x\", label=\"gpt\")\n",
    "#plt.plot(torch.arange(30)-1,lin, \"--x\", c=\"k\", label=\"linear\")\n",
    "plt.legend()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "5e098ec0-301b-49d3-8236-eb7c3ea17e64",
   "metadata": {},
   "source": [
    "The models fail to capture the linear relationship."
   ]
  },
  {
   "cell_type": "markdown",
   "id": "70d38a1e-a270-4dea-aaa2-db6caf13b7d9",
   "metadata": {},
   "source": [
    "## Train a fixed context fully connected model on the sparse BOW"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 193,
   "id": "3fb1bcf3-38e9-46c4-9df3-0cb69f36b5cf",
   "metadata": {},
   "outputs": [],
   "source": [
    "## Load the same dataset as in the main code an train model.\n",
    "from bow_dataset import SparseBoWDataset"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 194,
   "id": "291c8d44-b7b7-435c-8454-1126af86bdcb",
   "metadata": {},
   "outputs": [],
   "source": [
    "words_dict = {\"the\": (0.2, 0.0), \"we\": (0.2, 0.0), \"movie\": (0.2, 0.0), \"watch\": (0.2, 0.0),\n",
    "      \"best\": (0.06, 1.0), \"perfect\": (0.06, 1.5), \"good\": (0.08, 0.7),\n",
    "      \"worst\": (0.06, -1.5), \"poor\": (0.06, -1.0),\"ok\": (0.08, -0.7)}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 195,
   "id": "5d9e232d-e201-4cfc-b2b5-0df2d596ae4b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch.nn as nn\n",
    "class TwoLayerFixedContext(nn.Module):\n",
    "    def __init__(self, input_dim: int, hidden_dim: int):\n",
    "        super().__init__()\n",
    "        self.layer1 = nn.Linear(input_dim, hidden_dim)\n",
    "        self.layer2 = nn.Linear(hidden_dim, 2)\n",
    "\n",
    "    def forward(self, x):\n",
    "        x = self.layer1(x)\n",
    "        x = torch.relu(x)\n",
    "        return self.layer2(x)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 196,
   "id": "c86ac6ad-6721-4ec5-b1d8-8cb99a15ff14",
   "metadata": {},
   "outputs": [],
   "source": [
    "## Train the linear model using BERT embeddings\n",
    "## Create wrapper\n",
    "from torch.utils.data import DataLoader\n",
    "from wrappers import DistilBertWrapper\n",
    "run, i = \"2024-01-15-19:08:16_distilgpt2_2_12\", 5\n",
    "s_dict = torch.load(f\"{prefix_path}AttentionMatricesRaw/{run}/epoch_{str(i).zfill(3)}/model_trained.pt\")\n",
    "model_obj = DistilBert(n_layers=n_layers, n_heads=n_heads)\n",
    "wrapper = DistilBertWrapper(model_obj.model, use_device)\n",
    "distil_tok = AutoTokenizer.from_pretrained('bert-base-uncased', use_fast=True)\n",
    "\n",
    "myds = SparseBoWDataset(distil_tok, all_words_dict =  words_dict)\n",
    "mydl = DataLoader(myds, batch_size=32)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "edc9c5e1-6675-401f-94c8-69e21022d4ce",
   "metadata": {},
   "outputs": [],
   "source": [
    "from torch.optim import Adam\n",
    "def fit_sgd(my_dl, model, num_eps=10, lr=5e-4, batch_size=32):\n",
    "    model = model.to(wrapper.use_device)\n",
    "    my_optim = Adam(model. parameters(), lr = lr)\n",
    "    losses = []\n",
    "    iters = 0\n",
    "    loss_fn = nn.CrossEntropyLoss()\n",
    "    for ep in range(num_eps):\n",
    "        losses = []\n",
    "        for batch in my_dl:\n",
    "            my_optim.zero_grad()\n",
    "            #output = example_model.forward(batch[\"input_ids\"])\n",
    "            #loss = myloss(output, batch[\"label\"])\n",
    "            input_embs = wrapper.compute_embeddings(batch[\"input_ids\"].to(wrapper.use_device), 1)\n",
    "            #print(input_embs.shape)\n",
    "            output = model.forward(input_embs.reshape(len(input_embs), -1))\n",
    "            loss = loss_fn(output, batch[\"label\"].to(wrapper.use_device))\n",
    "            loss.backward()\n",
    "            my_optim.step()\n",
    "            losses.append(loss.item())\n",
    "            iters += 1\n",
    "        print(torch.tensor([losses]).mean())\n",
    "    return model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "f51d0504-c7d9-49e9-9979-7cf69b0a4e39",
   "metadata": {},
   "outputs": [],
   "source": [
    "fc_model = TwoLayerFixedContext(32*768, 400)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 83,
   "id": "7781f743-d49d-4a1b-91c7-c07146d6b1e2",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor(0.3792)\n",
      "tensor(0.3582)\n",
      "tensor(0.3722)\n",
      "tensor(0.3583)\n",
      "tensor(0.3694)\n",
      "tensor(0.3566)\n",
      "tensor(0.3578)\n",
      "tensor(0.3669)\n",
      "tensor(0.3615)\n",
      "tensor(0.3521)\n",
      "tensor(0.3454)\n",
      "tensor(0.3586)\n",
      "tensor(0.3497)\n",
      "tensor(0.3416)\n",
      "tensor(0.3808)\n",
      "tensor(0.3573)\n",
      "tensor(0.3577)\n",
      "tensor(0.3577)\n",
      "tensor(0.3522)\n",
      "tensor(0.3629)\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "TwoLayerFixedContext(\n",
       "  (layer1): Linear(in_features=24576, out_features=400, bias=True)\n",
       "  (layer2): Linear(in_features=400, out_features=2, bias=True)\n",
       ")"
      ]
     },
     "execution_count": 83,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "fit_sgd(mydl, fc_model, num_eps=20)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "4aa132e4-aa6f-4a04-998c-042a3ab64d57",
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_logits_for_single_token_fc(model, n_max=10, use_cls=True):\n",
    "    logit_results = []\n",
    "    cls_tok = torch.tensor([101])\n",
    "    input_tok = torch.tensor([3819])\n",
    "    pad_tok = torch.tensor([1996])\n",
    "    inputs_pad = torch.ones(n_max+1, 32, dtype=torch.long)*pad_tok\n",
    "    if use_cls:\n",
    "        x = cls_tok.reshape(1)\n",
    "    else:\n",
    "        x = torch.empty(0)\n",
    "    \n",
    "    for i in range(n_max+1):\n",
    "        inputs_pad[i, :len(x)] = x\n",
    "        x = torch.cat((x, input_tok), dim=0)\n",
    "    inputs_pad[:,-1] = 102\n",
    "    #print(inputs_pad)\n",
    "    with torch.no_grad():\n",
    "        input_embs = wrapper.compute_embeddings(inputs_pad.to(wrapper.use_device), 1)\n",
    "        logit_score = model.forward(input_embs.reshape(len(input_embs), -1))\n",
    "        score = logit_score[:,1]-logit_score[:,0]\n",
    "    return score"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "8bf4b6c6-7887-46fd-810a-afe4a2094c84",
   "metadata": {},
   "outputs": [],
   "source": [
    "res = compute_logits_for_single_token_fc(fc_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "a935ac55-dd0d-4a0e-87e6-4f14246e8222",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([ 0.2260,  1.4419,  2.9185,  4.2735,  5.1394,  5.8389,  6.8366,  7.7847,\n",
       "         8.6427,  9.5605, 10.4050], device='cuda:2')"
      ]
     },
     "execution_count": 15,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "res"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 88,
   "id": "453379ce-aea4-483b-a159-a7199a4096cc",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[<matplotlib.lines.Line2D at 0x7f766c5c4d30>]"
      ]
     },
     "execution_count": 88,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAVIAAAFPCAYAAAAfnUaqAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAABZR0lEQVR4nO2dd1hUZ9q4b1ARFRXRIIolQkSCWMAWFQ0qNgRFpUj6biLuJtmYqsnuZlP3SzQxyW7KF001RaUIqIiKWIIKsTA2NDZGRVBUGECRJsz5/eGP+WJnhjPMOW/OfV17bWbmlOf2GZ55z3vO+752kiRJaGhoaGhYjL2tA9DQ0NBQO1oh1dDQ0GgkWiHV0NDQaCRaIdXQ0NBoJM2tfYK+ffvi6elp9n4nTpzgvvvuM3u/goIC3N3dFb+f6H4gvqPmJ+/5GvNda2pHnU5Hfn7+/70hWZnQ0FCL9vP392/S8zX1fqL7SZL4jpqfvOdrzHetqR07d+583WvhLu2jo6NVsZ+liO7XmHOqxVHzk3e/xu5rCTe1fi0qx2Zg6S9FSEiIzJEoC9H9JEl8R81P/VjqeGNdU2yLtE+fPrYOwaqI7gfiO2p+6kcuR8UW0kGDBtk6BKsiuh+I76j5qR+5HBVbSC29e6cWRPcD8R01P/Ujl6NiC6mvr6+tQ7AqovuB+I6an/qRy1GxhXTDhg22DsGqiO4H4jtqfupHLkfFFlINDQ0NtaDYQjps2DBbh2BVRPcD8R01P/Ujl6PVC2lBQQFTp05l+fLlZu1XXl5upYiUgeh+IL6j5qd+zHVcvnw5U6dOpaCg4Lr3rV5I3d3dWb16tdkjDw4dOmSliJSB6H4gvqPmp37MdYyOjmb16tU33e1X7KW9hoaGhlqwkyTrLjUydepUVq9ebfZ+V69epUWLFlaISBmI7gfiO2p+6iQ/Px93d3fs7Owsdryxrim2RZqWlmbrEKyK6H4gvqPmpz6ys7Pp168fX375JSCfo2IL6aVLl2wdglUR3Q/Ed9T81MWePXsICgrC29ubhx56CJDPUbGFtHPnzrYOwaqI7gfiO2p+6mH37t0EBQVx//33s2HDBtq3bw/I53jbQqrT6W45oF+n07Fw4UIWLlxIREQEpaWlsgRyI4MHD7bKcZWC6H4gvqPmpx4+/vhj+vbty/r162nXrp3pfbkcb1lIExISgGtF80bS09OZN28e8+bNY8iQIYwbN06WQG5k7dq1VjmuUhDdD8R31PyUT01NDQDffvvtTUUU5HO8ZSENDw/H39//pvd1Oh3vvffeddvpdDr0er0swWhoaGjIRVZWFl5eXuTk5ODo6Ejbtm2tdi6z+kj9/f356quvTK/rL+tdXFxkDQrEnwtRdD8Q31HzUy6ZmZlMnDiRHj16cO+99952O5vNRxoeHm7679jYWIKCgnB2dpYlmN9z9epV2Y+pJET3A/EdNT9lsmPHDiZOnIifnx+pqak4OTnddlu5HO/4QL6dnR23+7i0tJRBgwaRnZ19x0Lq4+NDq1atAOjSpQs+Pj74+fnh6urK0KFDSUlJAa61do1GI/v27QOgurqarl27UlxcjIuLC6NHjyY5ORmA/v3706JFC7KzswGYMmUKe/bs4fz587Rr144JEyaY+nn79u2Lk5MTO3fuBGDixInk5ORQUFBAmzZtCAkJITY2Fri27ECnTp3YsWMHAEFBQRw7doy8vDxatmzJjBkziI2NxWg04unpibu7OxkZGQAEBgaSl5eHXq+nWbNmREZGsnLlSmpqaujZsyeenp5s3rwZgICAAH744Qc8PDyAa8POkpOTqayspFu3bvj4+Jiebxs+fDhlZWUcPnwYgIiICNavX8/ly5dxc3PD39+f1NRUAIYMGUJVVRUHDx4EYPr06WzdupWSkhI6derE8OHDWbNmDQB+fn4A7N27F4DQ0FCysrIoKiqiQ4cOBAYGkpSUBEC/fv1wdHRk9+7dAAQHB6PT6SgsLKRt27ZMmjSJ+Ph4U77bt29PVlYWmZmZvPnmmxw+fJj8/HxatWpFWFiYad4FLy8vXF1d2b59OwBjx44lNzeX06dP4+DgwMyZM4mLi6Ourg4PDw969OjB1q1bARg9ejQFBQXk5uZib29PVFQUiYmJVFdX06NHD7y8vEhPTwdg5MiRFBUVcfToUQCioqJISUnhypUruLu74+vra5pObdiwYZSXl5uGDoaHh5OWlsalS5fo3LkzgwcPNvWrnTt3jokTJ3LgwAEAwsLCyMjIwGAw0LFjRwICAli1ahUAAwcOxN7e3nTfISQkhF27dnHhwgXat2/PuHHjSExMBK7Nkdm6dWt27doFwOTJk9m/fz9nz57FycmJ4OBg4uLiAPD29sbFxYXMzEwAxo8fz5EjRzhz5gyOjo5Mnz6dFStWIEkSvXv3xs3NjW3btgEwZswYTp06xcmTJ2nevDkRERHEx8dTW1tLr1692Lt3r+lve9SoURQWFnL8+HHs7OyYNWsWSUlJVFVV0b17d7y9vdm4cSMAI0aMwGAwcOTIEQAiIyNJTU2lvLycrl27MmDAANatWwfA0KFDqaioICcnB4AZM2awadMmysrK7lojpk2bxvbt26+rEfHx8bzwwgt4enqyZMkSfvvttzvWiMzMTGbPnn3XGjF37lwyMzNxcnLCwcEBg8Fgqj/AnRe/u9PHMTExUm5urtmLRDWUZcuWWbSfWhDdT5LEd9T8lMn+/ful8vLyBm1rqeONdc2iFunChQsJDw/Hw8PD1E96u1appUNEKysrTS1ZERHdD8R31PyUwy+//MLixYv5/vvvcXBwaPB+ljqaPUT0xudEExIS8Pf3NxXRuLg4q/SR1l8yi4rofiC+o+anDLZu3UpwcDDnz5+ntrbWrH3lcmx+qzfT09NN/R3vvfceQ4YMITw8HL1eT0RExHXbOjs7ExMTI0swv8dgMMh+TCUhuh+I76j52Z4tW7YQEhLCiBEjWLVqFa1btzZrf7kcb1lIg4KCCAoKYsGCBde97+HhcdubT3LTsWPHJjmPrRDdD8R31Pxsy7Fjx5gyZYrppp4ll+hyOSp2rH1AQICtQ7AqovuB+I6an23p3bs3n3zyicVFFORzVGwhrX9sRFRE9wPxHTU/25Cenk5iYiJ2dnbExMQ06oaYXI6KLaQaGhoaN5KWlkZoaChLly5tsm7GhqDYQjpw4EBbh2BVRPcD8R01v6Zlw4YNTJ06lbFjxxIXF4ednV2jjymXo2ILqb29YkOTBdH9QHxHza/p2LRpE9OmTSMoKIjExERatmwpy3HlclTscsy3msJPJET3A/EdNb+mw8vLi6eeeoqVK1fKVkTBfMfbLcd8y8ef5KR+OWYNDQ0Nc9m8eTP9+/ene/fufPbZZ7YOh+joaKKjo5k6dep17yun7X4DISEhtg7BqojuB+I7an7WJSUlhcmTJ7Nw4UKrnUMuR8UW0vqZb0RFdD8Q31Hzsx5r1qxhxowZTJkyhXfffddq55HLUbGF9MKFC7YOwaqI7gfiO2p+1mH16tXMnDmTqVOnEhsba9YkJOYil6NiC2n9Kn+iIrofiO+o+VmHoqIipk+fzvLly2nRooVVzyWX4x2n0ZMDS6fRq66ulvXunNIQ3Q/Ed9T85OXo0aP06dMHAEmSZHlO9G5Y6mj2NHq2on62cFER3Q/Ed9T85D2Xr6+vaVWGpiii9eeVA8UWUg0NjT8GK1euJDIykvDwcEJDQ20djkUotpD6+vraOgSrIrofiO+o+TWe+Ph4oqKiiIqK4scff6R5c6s/2n4dcjkqtpCaO0Gr2hDdD8R31PwahyRJLFmyhFmzZrF06dImL6Ign6NiC6n2jJ76Ed1R87OcK1euYGdnx6pVq2xWREFFz5FaOtZeQ0NDTJYvX46Xlxd5eXm0bt2aZs2a2TqkBnO7sfaKffyptLTUKovqKQXR/UB8R83PfJYtW8ajjz7Ko48+yjfffGPzImqpo2oef9q/f7+tQ7AqovuB+I6an3n89NNPPProozz++OOKKKIgn6NiC+nZs2dtHYJVEd0PxHfU/BpOcXExTz/9NE888QRff/21IoooyOdomx7eBuDk5GTrEKyK6H4gvqPm1zAkSaJjx478+uuveHt7K2rCaLkcFdtHWldXp5hfLWsguh+I76j53Z3vvvuOTZs28f3339vszvydsNRRNX2kcXFxtg7BqojuB+I7an535ttvv+XJJ5/EyclJUa3Q3yNXDpVpp6GhoWq++eYbnnzySebMmcMXX3yh2EIqF4q18/b2tnUIVkV0PxDfUfO7Ndu2beOpp57ir3/9K59//rmii6hcOVRep8X/x8XFxdYhWBXR/UB8R83v1owcOZJly5Yxa9asJpvFyVLkyqFifyoyMzNtHYJVEd0PxHfU/K5nyZIlrF+/Hnt7e6KjoxVfREG+HGpDRDU0NBrNF198wZw5c0hPT7d1KFbldkNEkaxMaGioRftdvHhR5kiUheh+kiS+o+Z3jc8++0wCpLlz50pGo9HKUcmD0WiUVmafkfLPnbdo/xvrmmIv7Y8cOWLrEKyK6H4gvqPmd+050WeffZYXXniBjz/+WBWX84VlVTz+3W5ejNtP7LZDshxTsYX0zJkztg7BqojuB+I7an4wevRo3n33XRYtWqSKIrp6/1kmfpLBkXOX+P5PQ+hSWyjLcRVbSB0dHW0dglUR3Q/Ed/wj+y1fvpyysjI8PT35xz/+ofgiWlpRw9+W7+W55XsJ6N2JDc+PJrCPq2w5VOwQUQ0NDWXy8ccf8+KLL/L555/z9NNP2zqcu/LLsYvMS9hPZU0d74T5Mm2ge6OPqZohoitWrLB1CFZFdD8Q3/GP6PfRRx/x4osv8uqrr/LXv/7VBlE1nIqaWl5PzuHxb3fh1bktaS88eFMRlSuHt30gX6fTMXv2bLKzs697X6/Xk5CQgIeHB3q9npiYGKtMbmvlhrLNEd0PxHf8o/l9+OGHvPLKK7z22mv8+9//VvTlvC6vhJfi9nOurJK3p/Xl0Qd63jJeuXJ4y0JaXyh1Ot1Nn0VERJiKq16vZ/bs2cTHx8sSzO/p3bu37MdUEqL7gfiOfzS/Fi1a8I9//IN33nlHsUW0ptbIp5uP8/mWE/Tr5sw3jw/G457bT5UnVw5vWUjDw8NvubFer7/utYeHh9UewHVzc7PKcZWC6H4gvuMfxS87O5tBgwYxd+5cG0d0Z46fv8wLcfs4cu4yzwd58XSgJ82b3bn3Uq4cmtVHmp6eftPYVBcXl1u2XBvLtm3bZD+mkhDdD8R3/CP4vf/++wwePJhff/3V1uHcFqNR4utteqZ8up2qq0aSnh7Jc+N637WIgnw5NGvSktLS0lu+bzAY5IhFQ0NDQSQnJxMXF8ebb77JAw88YOtwbkl+SQUvx+/nV72BP4/sxbxJfXBs0fSTbcsy+9PtCizAiRMnGDRoEABdunTBx8cHPz8/XF1dGTp0KCkpKQD4+/tjNBrZt28fAMOGDSMtLY3i4mJcXFwYPXo0ycnJAPTv358WLVqY+mqnTJnCnj17OH/+PO3atWPChAkkJCQA0LdvX5ycnNi5cycAEydOJCcnh4KCAtq0aUNISAixsbEA9OnTh06dOrFjxw4AgoKCOHbsGHl5ebRs2ZIZM2YQGxuL0WjE09MTd3d3MjIyAAgMDCQvLw+9Xk+zZs2IjIxk5cqV1NTU0LNnTzw9Pdm8eTMAAQEBuLi4mOYfiI6OJjk5mcrKSrp164aPjw9paWkADB8+nLKyMg4fPgxc66Nev349ly9fxs3NDX9/f1JTUwEYMmQIVVVVHDx4EIDp06ezdetWSkpK6NSpE8OHD2fNmjUA+Pn5AbB3714AQkNDycrKoqioiA4dOhAYGEhSUhIA/fr1w9HRkd27dwMQHByMTqejsLCQtm3bMmnSJFM/uY+PD+3btycrK4vS0lKKi4s5fPgw+fn5tGrVirCwMJO3l5cXrq6ubN++HYCxY8eSm5vL6dOncXBwYObMmcTFxVFXV4eHhwc9evRg69atwLUHwQsKCsjNzcXe3p6oqCgSExOprq6mR48eeHl5mbqdRo4cSVFREUePHgUgKiqKlJQUrly5gru7O76+vmzYsMH0vSsvL+fQoWsjXsLDw0lLS+PSpUt07tyZwYMHs3btWgB69OjBoUOHOHDgAABhYWFkZGRgMBjo2LEjAQEBrFq1CoCBAwdib29vunoLCQlh165dXLhwgfbt2zNu3DgSExMB8PX1pXXr1qY11ydPnsz+/fs5e/YsTk5OBAcHmyYk9vb2xsXFxTT5xvjx4zly5AhnzpzB0dGR6dOns2LFCiRJonfv3ri5uZlaYWPGjOHUqVOcPHmS5s2bExERQXx8PLW1tfzyyy/ExcURHh6Ol5cX+fn5FBYWcvz4cezs7Jg1axZJSUlUVVXRvXt3vL292bhxIwAjRozAYDCYRkZFRkaSmppKeXk5Xbt2ZcCAAaxbtw6AoUOHUlFRQU5ODgAzZsxg06ZNlJWV3bFGSBLQayhvpxyhpb2RFwc2Z3ZQL5IS4syqEaWlpRw4cOCuNWLu3LlkZmbi5OSEg4PDzY3HO40nvfHjxYsXS/7+/te95+zsLG3cuLHBY1IbSlZWlkX7qQXR/SRJfEdR/aqqqqRhw4ZJs2fPtnUot6TocpUU88Nuqef8FOmF2L1SWWWNxceyNIeNGmsfFBR0y/cHDx5szmEaxMmTJ2U/ppIQ3Q/EdxTRz2Aw0LJlS3755RfGjBlj63BuIv3weSZ+ksHuUyV8+Yg/H0UOpJ1jC4uPJ1cO71pIf3/Z7uHhcd1ner2ewYMHW+U5UiUulCUnovuB+I4i+UmSxBtvvMGAAQMoLS2lZcuWivIrr65lfsIBnvphDwO6ObP++VFM8u3S6OPK5nirZuvGjRulefPmSYA0b948KT4+3vRZbm6u6b158+ZJJSUlZjWBNTQ0lIXRaJRef/11CZDee+89W4dzEzv1xdLI9zdJPq+vk1bsOq2IqfpurGuKnY80Li5O5kiUheh+kiS+owh+RqNR+uc//ykB0oIFC677zNZ+VVdrpf9Ze1i699UUKfx/d0ini67Ifg5LHW+sa8ppu99AbW2trUOwKqL7gfiOIvidOHGCDz74gA8++ICXX375us9s6Xf47CVeiN3HyaIrvDrJm6dGedDMXv7RVHI5KraQ9urVy9YhWBXR/UB8RzX7SdeuRunduzdHjx6lZ8+eN21jC786o8TijFw+3ngMz3ucWPXsSO7v0s5q55PLUbGF9N5777V1CFZFdD8Q31GtfpIk8eqrr1JSUsLixYtvWUSh6f1OF1/hxbj97M0rYc6Dnjwf1JuWza37cL1cjoqdRm/Lli22DsGqiO4H4juq0U+SJObPn8/ChQvx8fG54+QjTem3Mjufyf/ZxsXL1cTNGc78Sd5WL6Ign6NiW6QaGhryIkkSr7zyCosWLeI///kPzz33nK1DorKmjn+tyiE+O5/wQd14a2pf2rRUYVlq5E2vu+Lv7y+FhoZKy5YtM2u/M2fOWCkiZSC6nySJ76g2v59//lkCpE8//bRB21vb71jhJWn8R1sl73+uk+L32Obf0lzHZcuWSaGhoTeN8LT6pb27uzurV68mOjrarP0KC+VZlEqpiO4H4juqzS8yMpL169fz7LPPNmh7a/qtzM5n6mc7kCRY/exIwgd1s9q57oS5jtHR0axevRp39+tn2ldsH+nx48dtHYJVEd0PxHdUg5/0//tEf/nlF5o3b87EiRMbvK81/Cpr6nglfj8vxe8npH8XVj07kt6d28p+noYil6NiOyOUOgO3XIjuB+I7Kt1PkiTmzp3Lp59+iqenJw8++KBZ+8vtd/z8ZZ5ZpuOMoZIPIwbYrBX6e+Ry1FYR1dAQEEmSeO655/jss8/48ssvmTNnjk3jWZmdzz+Tc+jWoRVfPOxv01aoHKhmFdH6uTBFRXQ/EN9RyX6vv/46n332GYsXL7a4iMrhp7RL+RuRK4eKvbSvqqqydQhWRXQ/EN9RyX4PPfQQXl5ePPbYYxYfo7F+SryUvxG5cqjYFmn37t1tHYJVEd0PxHdUmp/RaOSTTz7hypUr+Pj4NKqIQuP8lHJX/m7IlUPFtki9vb1tHYJVEd0PxHdUkp/RaOQvf/kLX3/9Nb1792bKlCmNPqYlfr9/wD5iUDfemtaX1g6KLTOy5VCxLdL69V9ERXQ/EN9RKX5Go5E5c+bw9ddf891338lSRMF8v+PnLzPt8+2kHDjHhxED+CBigKKLKMiXQ2Vbamho3BFJkoiJieHbb7/l+++/b/TlvKX8/q78aoXdUGoKFFtIR4wYYesQrIrofiC+oxL87Ozs8PDwYOnSpTz66KOyHrshfmq7lL8RuXJo9Uv7goICpk6dalqCt6HctNypYIjuB+I72tKvrq6OX375BYC///3vshdRuLvf7y/lF6nkUv5GzM3h8uXLmTp1KgUFBde9r9ix9vVrYouK6H4gvqOt/Orq6njyyScJCgqy6kqmd/K78a78TIXelb8b5ubwdmPt1fXzoaHxB6euro4//elP/Pzzz/z8889NPou92i/lrYVih4jW1dXRrJn1J3a1FaL7gfiOTe1XV1fH448/zooVK/j555+Jioqy+vl+7/f7B+zfDfNVbSv091iaQ9UMEU1NTbV1CFZFdD8Q37Gp/S5fvsyRI0dYvny51YsoXO8nyqX8jciVQ8W2ycvLy20dglUR3Q/Ed2wqv9raWoqLi+ncuTM7d+5sslZweXm58JfycuVQsf8iXbt2tXUIVkV0PxDfsSn8amtreeSRR9i/fz8HDhygRYsWVj9nPVJbN6Z9vp0zhkoWRQwQphX6e+TKoWIL6YABA2wdglUR3Q/Ed7S2X21tLQ8//DCJiYnExcU1WRGVJIm4PWd4e2cV3Tu0FvoBe7lyqNg+0nXr1tk6BKsiuh+I72hNv6tXr/LQQw+RmJhIfHw806dPt9q5fk9xeTVzfsxm/sqD+LatUty0d3IjVw4V2yLV0Pgjs2fPHtauXUtCQgLTpk1rknNuOXqBV+IPYJQkFj86CMOBLUL1h1oTxf4rDR061NYhWBXR/UB8R2v41dbW0qxZM4YPH87JkydxdXWV/Rw3UllTx/+k/saPv54msM89LAzvj2tbR3Idxc4fyJdDxQ4RraiosFJEykB0PxDfUW6/mpoaIiIimD9/PkCTFNGD+WVM+XQb8dlneGdaX757YgiubR0B8fMH5juqbohoTk6OlSJSBqL7gfiOcvrV1NQQGRlJamqq2YvUWUKdUeKzzceZ/sUO2jg0J+Vvo3h0+L3XLQYnev7AfEdtiKiGhkKprq4mIiKCDRs2kJSURHBwsFXPd8ZQwQux+9DllfB04H08N643Ds0Ve99ZFSh2iGh1dTUtW7a0QkTKQHQ/EN9RLr8FCxbwxhtvkJyczKRJk2SI7NZIkkRCdj5vrj6Ei5MDH0cOZPC9LrfdXvT8geWOqhkiumnTJluHYFVE9wPxHeXye+GFF8jIyLBqES25UsNff9LxSsIBJvfrQupzo+5YREH8/IF8jootpGVlZbYOwaqI7gfiOzbGr6qqiujoaHbt2oWDg4NVn3D45dhFJn6Swa8ni/nfh/35MGIAbR3v/nC/6PkD+RwVW0ib4o6lLRHdD8R3tNSvqqqK6dOnk5ycbNViVXW1jjdXH+Lxb3fh3aUdG54fzeR+XRq8v+j5A/kcLbrZpNfrSU9Px8XFBb1eT3h4OB4eHrIEVI/2DKL6Ed3REr+qqirCwsL45ZdfWLNmDUFBQVaIDHIKyng+dh9nDBW8NbUvjw3ved0d+YYgev7Axs+RJiQkEBMTQ3h4OPPmzWPBggWyBPN7UlJSZD+mkhDdD8R3tMTvT3/6ExkZGaSkpFiliNYZJb7YeoLpX+zAoZk9KX8L4PER95pdREH8/IF8jhYV0tjYWFlOrqHxR2PevHmkpKQwbtw42Y99xlBB9JJf+WDDUZ4a5UHyM2KPk1cSFhVSFxcXBg0aZLrEHz9+vNxx4e/vL/sxlYTofiC+Y0P9Kioq+Ne//kVVVRV+fn6MHTtW1jgkSSJRl8/k/2yjoLSSFbMfYP4k70Y/Gyp6/kA+R4v+pePj4wHw9PQkPj6e8PBwWYL5PUajUfZjKgnR/UB8x4b4VVRUEBoayqJFi/jtt99kj6G0ooZnl+3lxbj9TPDpzLrnRzHMo6MsxxY9fyCfo0U3m9LT01mwYAF6vZ45c+YAsHjx4ltue+LECQYNGgRAly5d8PHxwc/PD1dXV4YOHWrqo/D398doNLJv3z7g2oOyZ86cobi4GBcXF0aPHk1ycjIA/fv3p0WLFmRnZwMwZcoU9uzZw/nz52nXrh0TJkwgISEBgL59++Lk5MTOnTsBmDhxIjk5ORQUFNCmTRtCQkJMXRV9+vShU6dO7NixA4CgoCCOHTtGXl4eLVu2ZMaMGcTGxmI0GvH09MTd3Z2MjAwAAgMDycvLQ6/X06xZMyIjI1m5ciU1NTX07NkTT09PNm/eDEBAQACJiYmmG3TR0dEkJydTWVlJt27d8PHxIS0tDYDhw4dTVlbG4cOHAYiIiGD9+vVcvnwZNzc3/P39TcslDBkyhKqqKg4ePAjA9OnT2bp1KyUlJXTq1Inhw4ezZs0aAPz8/ADYu3cvAKGhoWRlZVFUVESHDh0IDAwkKSkJgH79+uHo6Mju3bsBCA4ORqfTUVhYSNu2bZk0aZLpx9XHx4f27duTlZVFZmYmb775JocPHyY/P59WrVoRFhZmmnfBy8sLV1dXtm/fDsDYsWPJzc3l9OnTODg4MHPmTOLi4qirq8PDw4MePXqwdetWAEaPHk1BQQG5ubnY29sTFRVFYmIi1dXV9OjRAy8vL9LT0wEYOXIkRUVFHD16FICoqChSUlK4cuUK7u7u+Pr6smHDBgCGDRtGeXk5hw4dAiA8PJy0tDQuXbpE586dGTx4MGvXrgXg3LlzGI1GDhw4AEBYWBgZGRkYDAY6duyIn58fgYGB5Obm8tVXX9G6dWuTe0hICLt27eLChQu0b9+ecePGkZiYCICvry+tW7dm165dAEyePJn9+/dz9uxZnJycCA4OJi4ujhOXm5F0ri1XjXZE96ygv/E3asq7sX3PEc6cOYOjoyPTp09nxYoVSJJE7969cXNzY9u2bQCMGTOGU6dOcfLkSZo3b05ERATx8fHU1tbSq1cv9u7da/p7HDVqFIWFhRw/fhw7OztmzZpFUlISVVVVdO/eHW9vbzZu3AhcWyveYDCYVuisH/paXl5O165dGTBggGn6uqFDh1JRUWEaqjljxgw2bdpEWVnZXWvEtGnT2L59e6NqRGZmJrNnz75rjZg7dy6ZmZk4OTnh4OBw8zLOkpnk5uZK8+bNu+61s7OzlJube8vtQ0NDzT2FJEmStGzZMov2Uwui+0mS+I538quqqpICAwOlNm3aSBkZGbKet7KmVnpr9SGp5/wU6eGvfpXOlVbKevx6RM+fJFnueGNdM7tFqtPpGDJkiOm1h4cHr732GqWlpeYe6o401RyMtkJ0PxDf8U5+Dg4OjB49mnfeeYeAgADZznn47CWej93LqeIK/hXiwxMj7sXe3vw78g1B9PyBfI5m95H6+/ubLvHqKS4ulr1juv5yT1RE9wPxHW/lV15eTlpaGnZ2drz11luyFdE6o8TiX3KZ9vl2mtnbs+bZAP4c0MtqRRTEzx/I52h2i9TDw4Px48ezcOFCnJ2dAUz9pHJSXFws+zGVhOh+IL7jjX6XL18mODiYQ4cOodfrTX8fjT5PeTVzV+xjR24RMaM8eHGCFy2bW38lUdHzB/I5WnSzKSgoyGojMupxcbnzhApqR3Q/EN/x936XL19m8uTJHDx4kA0bNshWRHV5JTzzs46rdUZ+enIYI+/rJMtxG4Lo+QP5HBU7H+no0aNtHYJVEd0PxHes97t06RKTJ08mJyeHtLQ0hg0b1uhjS5LED1mneXftYfp3c+bzh/xxa+/Y6OOag+j5A/kcFTtpSf1jDKIiuh+I71jvV1FRQV1dHRs3bpSliFbU1PJ87D7eWH2IRx7oyYqYB5q8iIL4+QP5HBXbItXQUDoVFRWcP38eNzc3srKyLBrPfiO5F8v560/Z5JdU8mm0H6EDusoQqYa1UWyLtH///rYOwaqI7gdiO5aVlfHf//6XmTNnIkmSLEV03cFzTPtsB3VGiVXPjLR5ERU5f/XI5ajYFmmLFnefeFbNiO4H4jqWlpYyceJETp8+zaZNmxpdRK/WGVm4/ghfbTvJlP5dWDCzP04tbf+nKWr+fo9cjopdjrl+aJeoiO4HYjqWlpYyYcIEjh8/zrx580zDny3lwqUqHv5qJ9/tOMXrIT58Fu2niCIKYubvRsx1vN1yzFbPWP1yzBoaIrB9+3ZOnTrFpk2bTGPJLWWnvphnl+/FDlge8wBD7rKGkobtiY6OJjo6mqlTp173vmJXEb106RLt2rWzQkTKQHQ/EMuxoqKCVq1aYWdnZ/Ky1E+SJL7edpL31x9hyL0d+DTan3vaKm+1TpHydzssdVTNKqJ79uyxdQhWRXQ/EMfRYDAQEBBgWgmi/g/PEr/LVVd5+mcd/079jadG9eKnJ4cpsoiCOPm7E3I5KqMz5hacP3/e1iFYFdH9QAzH4uJigoKCyM/PJzg4+LrPzPU7dv4yf/kpm4uXqvnykUFM8nWTM1TZESF/d0MuR8UWUtEvKUT3A/U7FhUVERQUxNmzZ9myZQu+vr7XfW6O36p9Bby68iA9XFqz+m8B9OrURu5wZUft+WsIcjkqtpBOmDDB1iFYFdH9QP2Ob7/9NufOnWPLli307dv3ps8b4ldTa+Tfaw+zNOs00/3c+fd0X1o7KPbP7jrUnr+GIJejYvtI62e4FxXR/UD9jgsWLGDHjh23LKJwd79zZZVELcli2a483gnz5aPIAaopoqD+/DUEuRwVW0g1NGzBhQsXGDt2LAcOHKBVq1bcd999Fh1nx4kipvx3O+fLqoj/ywgefcD8deU11INifx5v1woQBdH9QH2O9UW0qKioQSNebuVnNEr87y+5LEo7ysj7OvGfWX64tHGwRrhWR235swS5HBVbSJ2cnGwdglUR3Q/U5Xj+/HnGjh2LwWBg69ateHt733WfG/3KKq/yUtw+0n+7wHNj72NukBfNrDiDvbVRU/4sRS5HxV7a16/oJyqi+4F6HCVJIiwsjJKSkgYXUbje79DZMkI/3c7uUyV8+8RgXpzQR9VFFNSTv8Ygl6Nix9praDQVdnZ2fPTRR2zZsoU+ffqYvX/8njPM+CKTto7NSflbAGO9O1shSg0lcLux9oodImowGIRe6kB0P1C+47lz5/jwww95//33LZoF6NyFIv67/SzLd51h1pDuvDm1L44trL+WUlOh9PzJgaWOqhkimpOTY+sQrIrofqBsx7NnzxIYGEhsbCyFhYVm73/GUEH017tZqStg4cz+vD+zv1BFFJSdP7mQy1GxN5tubDqLhuh+oFzHs2fPMmbMGCoqKti6dSvdu3c3a//ME0U8vUyHXW0tiX8dja97eytFaluUmj85kctRsYW0TRvlD6FrDKL7gTIdDQYDgYGBVFVVsXXrVjw9PRu8b/2CdG+nHGaEZ0eCnc8LW0RBmfmTG7kcFdtHajQasbdXbM9DoxHdD5TpKEkSb775Jo8//jgeHh4N3q+6to5/JR8ids8ZngzoxWuTvbG3Q3F+cqLE/MmNpY6q6SONjY21dQhWRXQ/UJZjfn4+69evx87OjrfeesusInrxcjUPfbWTpL0FfBgxgNdDfGjezF5RftZAdD+Qz1Gxl/YaGnKRl5fHmDFjaN68OTk5OWbdoT+YX0bMj3uoM0rEznkAvx4drBiphlpRbCG15Hk+NSG6HyjD8fTp04wZMwZJkkhLSzOriK7aV8C8hAN4d2nHkkcH0bnd9WvLK8HPmojuB/I5KraQdurUydYhWBXR/cD2jqdPnyYwMBA7Ozu2bt1Kz549G7RfnVHigw1H+fKXXGb4ufM/M/rd8tEmW/tZG9H9QD5HxfaR7tixw9YhWBXR/UAZjp6enmYV0UtVV3lq6W6WZOTyzyn3syhywG2fD1WCnzUR3Q/kc9SGiGoIx+nTpzEYDPTs2ZP09HR69OjRoP1yL5YT9vkOsk+X8N2fhvLUKA9t6juN61DdENGLFy9yzz33WCEiZSC6H9jG8eTJkwQGBjJs2DDi4uIavN+Woxd4bvleOrdz5KvHBjdoKRDRcyi6H1juqJrHn44dO2brEKyK6H7Q9I56vZ4HH3yQli1b8vHHHzdoH0mSWPxLLn/+fjdD73Uh6ekRDV5PSfQciu4H8jkqtpDm5eXZOgSrIrofNK1jbm4uDz74IK1atWLLli24u7vfdZ+qq3W8ELuP99Yd4a8PerLkscG0dWz4XX3Rcyi6H8jnqNi79i1bKnOtb7kQ3Q+a1nHHjh04OTmxadMmunbtetftz5VVMufHbI6dv8yn0X6EDrj7Pjcieg5F9wP5HBXbR6qh0RBKS0txdnYGoKqqCkdHxzvvAGSfNjDnRx0OzexY8thgocfLa1gH1fSRij48TXQ/sL7jsWPH6Nu3L19//TVAg4po3O4zRC/ZSa9OrVn1bECjiqjoORTdDxQwRDQ9PR29Xm8asxwUFCRLQPUYjUZZj6c0RPcD6zoePXqUMWPG4OzsTEhIyF23r60z8u7a3/g+8xTRQ3vw1tS+ODRvXDtC9ByK7gfyOVpUSNPT04mPj2fx4sXo9XrGjx9Pbm6uLAHVY870ZmpEdD+wnuORI0cYO3YsHTp0YPPmzXTufOelPUqu1PDMMh27Thp4J8yXR4b1kOX5UNFzKLofyOdoUSGdM2cO2dnZAHh4eLBx40ZZgvk9DbnrqmZE9wPrOb7++uu4uLiwefNmXF1d77jt0cLLzP5hD+XVtfz01DAe8OgoWxyi51B0P5DP0exrG71ej8FgwNnZGZ1OR2lpqVlTkjWUjIwM2Y+pJET3A/kd6++Lfvvtt2zZsuWuRXTDoUJmfLGD1g7NWPXMSFmLKIifQ9H9QD5HswupTqfDxcWFhIQEPDw8WLJkCQkJCbIEo6FxOw4dOoS/vz/Hjx+nbdu2dxyNYjRK/Cf9OHN+zObBPveQ+PQIuru0bsJoNf5omP3405IlS5gzZw4lJSU4OztTWlpKhw4duN1hfHx8aNWqFQBdunTBx8cHPz8/XF1dGTp0KCkpKQD4+/tjNBrZt28fAEOHDiU3N5fi4mJcXFwYPXo0ycnJAPTv358WLVqYuhemTJnCnj17OH/+PO3atWPChAmm4t63b1+cnJxM61dPnDiRnJwcCgoKaNOmDSEhIaY7d3369KFTp06miQyCgoI4duwYeXl5tGzZkhkzZhAbG4vRaMTT0xN3d3fTL1pgYCB5eXno9XqaNWtGZGQkK1eupKamhp49e+Lp6cnmzZsBCAgI4PDhwxgMBgCio6NJTk6msrKSbt264ePjQ1paGgDDhw+nrKyMw4cPAxAREcH69eu5fPkybm5u+Pv7k5qaCsCQIUOoqqri4MGDAEyfPp2tW7dSUlJCp06dGD58OGvWrAHAz88PgL179wIQGhpKVlYWRUVFdOjQgcDAQJKSkgDo168fjo6O7N69G4Dg4GB0Oh2FhYW0bduWSZMmER8fb8p3+/btycrKoqSkhKioKA4fPkx+fj6tWrUiLCzMNO+Cl5cXrq6ubN++HYCxY8eSm5vL6dOncXBwYObMmcTFxXHq1Cnef/99unTpwnPPPUe7du0YPXo0BQUF5ObmYm9vT1RUFImJiVyqqGZtcUd2n7tKkFsVY11rCAgYSVFREUePHgUgKiqKlJQUrly5gru7O76+vmzYsAGAYcOGUV5ezqFDhwAIDw8nLS2NS5cu0blzZwYPHszatWsB6N69Ox06dODAgQMAhIWFkZGRgcFgoGPHjgQEBLBq1SoABg4ciL29PTqdDoCQkBB27drFhQsXaN++PePGjSMxMREAX19fWrduza5duwCYPHky+/fv5+zZszg5OREcHGwa/urt7Y2LiwuZmZkAjB8/niNHjnDmzBkcHR2ZPn06K1asQJIkevfujZubG9u2bQNgzJgxnDp1ipMnT9K8eXMiIiKIj4+ntraWXr160apVK9P3btSoURQWFnL8+HHs7OyYNWsWSUlJVFVV0b17d7y9vU1dfCNGjMBgMHDkyBEAIiMjSU1Npby8nK5duzJgwADWrVtn+juvqKgwLUI3Y8YMNm3aRFlZ2V1rxLRp09i+fXujakRJSQkBAQF3rRFz584lMzMTJycnHBwcMBgMpmMDIJnJxo0bJWdn5+veA6Ts7Oxbbh8aGmruKSRJkqRff/3Vov3Uguh+kiSP44EDB6R77rlHGjBggHTx4sU7bptXfEWa+PEvks/r66QNOecafe67IXoORfeTJMsdb6xrZl/aW6M/9Fbo9fomOY+tEN0PGu9YU1NDSEgI7u7ubNq06Y5zR2blFjP1s+1U1NSR+PRIJvR1a9S5G4LoORTdD+RzNPuuvYeHB4MHDzaNKKl/ltTf31+WgOpp1kysNcJvRHQ/aLyjg4MDP//8M/fffz8dO976RtGFy1Us2nCMuOwzjPDsyGfR/nRo49Co8zYU0XMouh/I52jRENHS0lLmz5/PoEGDyM7OZv78+bdtqWpDRDXMZd++fXz77bd8/PHHt/2iV12t47sdp/h8ywmaN7PjhSAvHh7Wg+bNFDtYT0MgZBki6uzszOLFi4mJiWHx4sVWudxfuXKl7MdUEqL7gWWOe/fuZdy4cWRmZnLlypWbPpckiXUHzzH+419YlHaUiMHd2PpyII+PuLfJi6joORTdD+RzVOzsTzU1NbYOwaqI7gfmO9YXUU9PT9LS0mjXrt11n+cUlPFOymF2njQwps89fPfEUO5zdZIzZLMQPYei+4F8jootpA1dY0etiO4H5jnq9XrGjRvHfffdR1pammlGJ7i2rvyHG44Sl30Gz3uc+P5PQwjsc+eH8ZsC0XMouh/I56jYQir6OF/R/cA8x549e/LSSy/xzDPP/N+0eDf0g74Z2peHhvWghUL6QUXPoeh+IJ+jMr6Rt6D+4XVREd0PGua4e/du0tPTadasGf/4xz9wdnZGkiTW51zrB/0w7Sjhg/6vH1QpRRTEz6HofiCfo2JbpBris2vXLiZMmMCgQYMYN24cdnZ2t+gHHcJ9rm1tHaqGxh1R7HLMAQEBVopIGYjuB3d23LlzJ+PHj6dv374kJSVRVF7DqysPEPrZdoqv1PD9n4bw3Z+GKrqIip5D0f3AfMfbLcds9ULq7u7O6tWriY6ONmu/CxcuWCkiZSC6H9zecefOnUyYMIF+/fqRvGYtP+suMubDraw/VMiboX1ZN3eUIm4m3Q3Rcyi6H5jvGB0dzerVq2+afk85HU43IPpSsKL7we0dO3TowMSJE3lx0XfM+Fqn6H7QOyF6DkX3A/kctT5SjSZDp9PRu3dvrjp1xj7oBV5MOqb1g2oIgbaKqEaTsGPHDiZOmsT9gWEU+c7C8x4n/jnlflVcwmto3IhqVhGtn1dQVET3g/9z3Lz1F8aNn4jU0YOqARGq6ge9E6LnUHQ/kM9RsZf2lZWVtg7BqojuB1BRUcmHS5OZHxONQxcv5i78mnkhA3Bu3TSzM1kb0XMouh/I56jYQtqtWzdbh2BVRPc7dLaMH/M7sHNDIp09fdmQmkK/e++82qfaED2HovuBfI6KLaQ+Pj62DsGqiOp3tc7If9KP89+U3Xj26EriF+8S4OlC8+aK/apZjKg5rEd0P5DPUbF9pPVrFomKiH76i+WE/28mH/+QyPmv5zDw3FoC+7gKWURBzBz+HtH9QD5HMb/hGk2KJEms2H2Gt9ccpuXFwxgS3yFw9CgGDuhv69A0NJoExRbS4cOH2zoEqyKKn+FKDfNXHmDj4fMMdzzLqh/+yYMPXlvNsbCw0NbhWRVRcng7RPcD+RwVO9a+rKzMShEpAxH8fjl2kYmfZLDnlIHFj/hTsPlHHnzwQZKTk3F0dBTC8U5ofurHXEfVjbWvX09bVNTsV3W1jjdXH+Lxb3dxf5d2rH12BBN9u7B69WqSkpJwdHQE1O3YEDQ/9WOu4+3G2iv20l5Dmfx27hLPr9jHyeIrvBHqg9ulo4wbOYT09PSbvlwaGn8UFDtEtLa2Vti7vaA+P6NR4tsdJ1m4/ige97ThP7P8OLlvB2FhYYwfP56EhARatmx53T5qczQXzU/9WOqomiGi69evt3UIVkVNfucvVfHYt7t4d+1vPDa8J8nPjCRXt41p06YxceLEWxZRUJejJWh+6kcuR8X+3Fy+fNnWIVgVtfitzznHq4kHadncnp+eHEZA704YDAZmzZrF5MmTiYuLw8Hh1kM+1eJoKZqf+pHLUbGF1M3NzdYhWBWl+12pruWtNYeI25PPpL5uvDejHx3aXCuYLi4upKWl4e/vf9siCsp3bCyan/qRy1GxhdTf39/WIVgVJfvtzSvh+dh9XLxczcKZ/YkY3A07OzvWrFlDeno6n3zyCQ888MBdj6NkRznQ/NSPXI6K7SNNTU21dQhWRYl+tf9/nHz4l1l0aO1A6nOjiBzSHTs7O1atWsXMmTMpKCigtra2QcdToqOcaH7qRy5HxbZINZqWvOIKXojbx968Ep4d25u/jb3PtORHcnIyERERhIWFsWzZMlq0aGHjaDU0lIViC+mQIUNsHYJVUYqfJEkk6gp4Y/UhOrRpQfxfhjOop4vp86ysLCIiIpg+fTo///yzWUVUKY7WQvNTP3I5KnaIaFVVlZUiUgZK8CuruMqzy/fyUvx+JvTtTOpzo64rogCDBw9m0aJFFrVEleBoTTQ/9WOuo+qGiB48eNBKESkDW/tl5hYx6T8ZbDt2kc8e8uOjyIG0dfy/QpmYmMiOHTto0aIFzz33nEUPLdva0dpofurHXEfVLcesYR2qa+t4L/U3Hv56J/d2bMP650cT0r/rddvEx8cTGRnJ0qVLbRSlhoa6UOwQ0aqqKtPkFyJiC78TFy7z3PJ9HL9wmZcn9GH2KA/s7e2u2yYuLo6HHnqIqKgoli5d2qghgloO1Y3ofmC5o2qGiG7dutXWIViVpvSTJIkfsk4x5b/bqakzkvT0SOY86HlTEU1ISOChhx4iOjqaH374odHjrLUcqhvR/UA+R8XetS8pKbF1CFalqfwkSeLdtb/xzfaTPDa8J69Nvp9WDs1uua2XlxfPPPMMH330Ec2a3Xobc9ByqG5E9wP5HBXbIu3UqZOtQ7AqTeW3JEPPN9tP8va0vrw9zfeWRXTjxo1UVlbSv39//vOf/8hSREHLodoR3Q/kc2x0IZ0/fz6lpaUyhHI9oi9z0BR+ibp83lt3hL+NvY/Hht97y21++uknJk2axJdffin7+bUcqhvR/UAhS43odDoWLlwoSyA3smbNGqscVylY22/L0QvMSzjArCHdeXG81y23+fHHH3n88cd54oknmDt3ruwxaDlUN6L7gXyOjSqker0eDw8PWQLRkI+9eSU8/ZOOwD6uvBvmi52d3U3bLF26lMcff5w///nPfPXVV9jbK7aXR0ND8Vj815OQkEB4eLicsVyHn5+f1Y6tBKzll3uxnD9/vxufru34NNqP5s1uneKTJ0/y1FNPsXjxYqsVUS2H6kZ0P5DP0aK79qWlpTg7O8sSgIZ8nL9UxWPf7KKTU0u+eXzwLW8s1V9FvPHGGwC3bK1qaGiYh0WFNC4ujpiYmAZte+LECQYNGgRAly5d8PHxwc/PD1dXV4YOHUpKSgpwbV5Ao9HIvn37AKiuriYvL4/i4mJcXFwYPfraWukA/fv3p0WLFmRnZwMwZcoU9uzZw/nz52nXrh0TJkwgISEBgL59++Lk5MTOnTsBmDhxIjk5ORQUFNCmTRtCQkKIjY0FoE+fPnTq1IkdO3YAEBQUxLFjx8jLy6Nly5bMmDGD2NhYjEYjnp6euLu7k5GRAUBgYCB5eXno9XqaNWtGZGQkK1eupKamhp49e+Lp6cnmzZsBCAgIYOXKlaZukejoaJKTk6msrKRbt274+PiQlpYGXOsMLysrM612GBERwfr167l8+TJubm74+/uTmppKVR38ePYeqmtqeaRLKetWrWT69Ols3bqVkpISOnXqxLFjx3juuef417/+RVRUFAB79+4FIDQ0lKysLIqKiujQoQOBgYEkJSUB0K9fPxwdHdm9ezcAwcHB6HQ6CgsLadu2LZMmTSI+Ph4AHx8f2rdvT1ZWFpmZmbz55pscPnyY/Px8WrVqRVhYmGneBS8vL1xdXdm+fTsAY8eOJTc3l9OnT+Pg4MDMmTOJi4ujrq4ODw8PevToYXrub/To0RQUFJCbm4u9vT1RUVEkJiZSXV1Njx498PLyIj09HYCRI0dSVFTE0aNHAYiKiiIlJYUrV67g7u6Or68vGzZsAGDYsGGUl5dz6NAhAMLDw0lLS+PSpUt07tyZwYMHs3btWgDOnTtHXV0dBw4cACAsLIyMjAwMBgMdO3YkICCAVatWATBw4EDs7e3R6XQAhISEsGvXLi5cuED79u0ZN24ciYmJAPj6+tK6dWt27doFwOTJk9m/fz9nz57FycmJ4OBg4uLiAPD29sbFxYXMzEwAxo8fz5EjRzhz5gyOjo5Mnz6dFStWIEkSvXv3xs3NjW3btgEwZswYTp06xcmTJ2nevDkRERHEx8dTW1tLr1692Lt3r+n7MWrUKAoLCzl+/Dh2dnbMmjWLpKQkqqqq6N69O97e3mzcuBGAESNGYDAYOHLkCACRkZGkpqZSXl5O165dGTBgAOvWrQNg6NChVFRUkJOTA8CMGTPYtGkTZWVld60R06ZNY/v27Y2qEZmZmcyePfuuNWLu3LlkZmbi5OSEg4MDBoOB65DMZOPGjVJJSYnptYeHx3WvbyQ0NNTcU0iSJEnLli2zaD+1IKdfZU2tFPllptT/zQ3SscJLt9xmyZIlEiA9/fTTktFolO3cd0LLoboR3U+SLHe8sa5Z3CKtR6/X89577xEVFSXrjNqhoaGyHUuJyOVXZ5R4fsU+9p0pZdnsYfTu3PambZYsWcKcOXN45pln+PTTT5vscl7LoboR3Q9kdGxsRQek3NzcBlfuhpKWlmZpSKpADj+j0Sj9I+mA1OvVFCntUOEtt6murpb69esnPfvss03WEq1Hy6G6Ed1Pkix3lKVFCtduOC1ZsgSABQsWMGfOHFlbpEVFRbIdS4nI4ffp5hP89Gse78/ox3ifzjd9Xj8hw7Zt22jXrl2T31jScqhuRPcD+Rwtfu7F2dmZefPmIUkSixcvln2hrA4dOsh6PKXRWL/lu/L4aOMxXp7gxayhPW76/PPPP2fgwIGUlpbSvn17m9yd13KobkT3A/kcFfsUdmBgoK1DsCqN8Us7VMg/kg7y2PCePDPmvps+/+yzz3j22WcJDg6mffv2jYiycWg5VDei+4F8jootpPWP3oiKpX67Txn42/K9TPJ1443Qvje1NP/73//yt7/9jZdeeolFixbZ9DlRLYfqRnQ/kM9RsYVU42aOFl7mye9349fDmY8iB9LshvlET5w4wYsvvsgrr7zCBx98oD1sr6HRRCh2PtJ+/frZOgSrYq5fQWklj3+7C/cOrVny2GAcW9w8aum+++5jz549DBgwQBFFVMuhuhHdD+RzVGyLVPQlDszxK62o4fFvd9G8mR1L/zSEdo7Xr+b50UcfmW78DRw4UBFFFLQcqh3R/UA+R8Uux1w/HFFUGupXWVPHn7/fjeFKDT/8eSiu7a5P/IcffshLL71k9lLJTYGWQ3Ujuh+Y73i75ZitfmlfvxyzhvnU1hl5dpmOI4WXWT77ATzucbru84ULFzJ//nz+8Y9/8M477yimJaqhISrR0dFER0czderU6z9o/NgA80YANJTS0lKZI1EWd/MzGo3SK/H7JM/X1kpbj1646fP4+HgJkP75z382+YilhvJHz6HaEd1Pkix3vLGuKbaPtH6WHFG5m9+HaUeJ25PPBxH9edDrnps+Dw0N5ccff+Ttt99WbEv0j55DtSO6H8jnqNhCWlhYaOsQrMqd/L7fcZLPt+Ty92Bvpvt1u+6zRYsWodPpaNmyJY888ohiiyj8sXMoAqL7gXyOii2kbdvePIuRSNzOL+XAWd5KOcxTAb2IGe153WfvvPMOL7/8smmeTaXzR82hKIjuBzI6ytHPYE5fQkO5evWqzJEoi1v57Th+Uer991Rp7nKdVFd3fb/nW2+9JQHS22+/3VQhNpo/Yg5FQnQ/SbLcUTV9pPUzrovKjX45BWXE/JjNMA8XFoYPwP53o5YWLFjAG2+8wbvvvsvrr7/e1KFazB8th6Ihuh/I56jYkU1/JPKKK3jiu9143NOGLx8ZhEPz63/fxowZY3peVENDQ3kotkXq4+Nj6xCsSr1fUXk1j327E6eWzfj2iSG0aXntt02SJJYtW0ZNTQ1Dhw5VZRH9o+RQVET3A/kcFVtIbTn9W1PQvn17yqtr+fP3uymvruOHPw+jk1NL4FoR/de//sXDDz9sWvhLjfwRcigyovuBfI6KLaRZWVm2DsGqbNuRxV9/ykZ/8Qrf/2kIPTq2Bq4V0X/+85+8++67fPDBB8yYMcPGkVqO6DnU/NSPXI6KHWsvMkajRMKZVuzUG1jy6CB83a/9KkqSxN///nf+53/+hw8//JCXX37ZxpFqaGj8ntuNtbeTJEmy5omnTp1q0Vj74uJiOnbsaIWIbM/7646w+JdcPnvInyn9u5jelySJl19+mW7duvHCCy/YMEJ5EDmHoPmJgKWON9Y1xV7aHz582NYhWIWV2fl8+Usu0fe3NBVRSZI4cOAAdnZ2LFq0SIgiCuLmsB7NT/3I5ajYQpqfn2/rEGQn+3QJryUeJHJwN3xbXASuFdF58+YxaNAg9Hq9jSOUFxFz+Hs0P/Ujl6NiC2mrVq1sHYKsFJRWMufHPQzo3p53wnxp3bqV6VL+ww8/5OOPP8bDw8PWYcqKaDm8Ec1P/cjlqNg+UpG4Ul1L+JdZXKq8yqpnR9LJqSWSJPHiiy/yySef8Nlnn/HMM8/YOkwNDY0Gopo+UlHu8huNEi/F7Sev+ArfPDHY9Kzo119/zdq1a/niiy+ELaKi5PB2aH7qRy5HbYiolfkk/RgbDhey5NHBeLu1Q5IkLl++jJOTE/v37/9DXD5paIiOYlukXl5etg6h0azZf5b/bj7BKxP7MN6nM5Ik8be//Y3Ro0fTq1cv4YuoCDm8E5qf+pHLUbEtUldXV1uH0Cj2nynl5fj9hA3syl8f9ESSJJ599lm++OILlixZgru7u61DtDpqz+Hd0PzUj1yOim2Rbt++3dYhWExhWRWzf9jD/V3a8f7M/kiSxDPPPMP//u//8vXXXzN79mxV+zUU0R01P/Ujl6M2RFRmqq7WEfPjHprZ27HksUE4tmhGVlYWS5Ys4euvv+bJJ5+0dYgaGhoWorohoufPn6dz585WiMh6SJLEcyv2sfFwIQl/GUHfru1MayqdOHGC++67z7StGv3MRXRHzU/9WOqomsefcnNzbR2C2Xy+5QRr9p9lUcRAfLq0JSYmhn//+98A1xVRUKefuYjuqPmpH7kcFVtIT58+besQzGJ9zjk+TDvG80G9mezbmdmzZ/PNN9/QvXv3W26vNj9LEN1R81M/cjkq9q69g4ODrUNoMIfOlvFC7H6m9OvCMw968OSTT7J06VKWLl3Ko48+est91ORnKaI7an7qRy5HxfaRqoWLl6uZ9tl2XJwciJ8zgi8+/YR58+axdOlSHnnkEVuHp6GhYQVU00caFxdn6xDuSnVtHXN+3MNVo8RXjw2mlUMz/vKXv5CamnrXIqoGv8YiuqPmp37kcrTo0l6n05Geng7A7t27+eqrr3B2dpYloHrq6upkPZ7cSJLEa4kHyTl7iWVPDuH9f71KTEwMffv2ZeLEiXfdX+l+ciC6o+anfuRytKhFmp6ezrx585g3bx5Dhgxh3LhxsgTze5Q+pdxX2/Qk6gp4b5oPH/9zLp9//jlHjhxp8P5K95MD0R01P/Ujl6PZhVSn0/Hee++ZXoeHh6PT6WSflLhHjx6yHk9ONv12nvfWHWHOqJ6s/Pg1YmNjWb58OTNnzmzwMZTsJxeiO2p+6kcuR7MLqb+/P1999ZXpdWlpKQAuLi6yBFTP1q1bZT2eXBwtvMxzy/cSdH9nTiR+TFxcHMuXLyciIsKs4yjVT05Ed9T81I9cjhb1kYaHh5v+OzY2lqCgINn7SJWI4UoNT/2wm+4urfkkaiDZ9z5OcHCwWS1RDQ0N8WjU40+lpaUMGjSI7Ozs2xZSHx8f03RxXbp0wcfHBz8/P1xdXRk6dCgpKSnAtZau0Whk3759AAwZMgS9Xk9xcTEuLi6MHj2a5ORkAPr370+LFi3Izs4GYMqUKezZs4fz58/Trl07JkyYQEJCAgB9+/bFycmJnTt3AjBx4kRycnIoKCigTZs2hISEEBsbC0CfPn3o1KkTO3bsACAoKIhjx46Rl5dHsxYtWVnchYOnCvHNX83zf51Nz549ycjIACAwMJC8vDz0ej3NmjUjMjKSlStXUlNTQ8+ePfH09GTz5s0ABAQEkJOTY2rNR0dHk5ycTGVlJd26dcPHx4e0tDQAhg8fTllZmWmRroiICNavX8/ly5dxc3PD39+f1NRU079ZVVUVBw8eBGD69Ols3bqVkpISOnXqxPDhw1mzZg0Afn5+AOzduxeA0NBQsrKyKCoqokOHDgQGBpKUlARAv379cHR0ZPfu3QAEBwej0+koLCykbdu2TJo0ifj4eFO+27dvT1ZWFgaDgVmzZnH48GHy8/Np1aoVYWFhpnkXvLy8cHV1NU0cMXbsWHJzczl9+jQODg7MnDmTuLg46urq8PDwoEePHqYWxOjRoykoKCA3Nxd7e3uioqJITEykurqaHj164OXlZbohOnLkSIqKijh69CgAUVFRpKSkcOXKFdzd3fH19WXDhg0ADBs2jPLycg4dOgRcazSkpaVx6dIlOnfuzODBg1m7di0A3bp1w8XFhQMHDgAQFhZGRkYGBoOBjh07EhAQwKpVqwAYOHAg9vb26HQ6AEJCQti1axcXLlygffv2jBs3jsTERAB8fX1p3bo1u3btAmDy5Mns37+fs2fP4uTkRHBwsOlus7e3Ny4uLmRmZgIwfvx4jhw5wpkzZ3B0dGT69OmsWLECSZLo3bs3bm5ubNu2DYAxY8Zw6tQpTp48SfPmzYmIiCA+Pp7a2lp69epFy5YtTf3+o0aNorCwkOPHj2NnZ8esWbNISkqiqqqK7t274+3tzcaNGwEYMWIEBoPBtG9kZCSpqamUl5fTtWtXBgwYwLp16wAYOnQoFRUV5OTkADBjxgw2bdpEWVnZXWvEtGnT2L59e6NqhMFgYNSoUXetEXPnziUzMxMnJyccHBwwGAymYwMgNYKYmBgpNzf3jtuEhoZadOydO3datJ81MBqN0qsr90ue81dJYydPlZo3by79+uuvjTqmkvysheiOmp/6sdTxxrpm8XOkCxcuZP78+Xh4eFBaWmpqXcmFksb5fp95imVZJ+m0+0u2pa8jISGBYcOGNeqYSvKzFqI7an7qx6Zj7RMSEvD39zcV0bi4ONn7SO3tlTFWIOPYRd5edQCnzM/QbUsnISGBadOmNfq4SvGzJqI7an7qRy5Hs/tI9Xo9np6e173n7OxMSUnJLbdX8xDR3IvlhH2+A//u7XHS/cikSZMIDQ21dVgaGho2ptFDRD08PJAk6br/3a6INob6jndbUVZxlT9/k4Vj6Uk+fXgQn3/+uaxF1NZ+TYHojpqf+pHLUbFt9+rqapudu7bOyF9+2Mm+7//F0W/nU1dZLvs5bOnXVIjuqPmpH7kcFTuNni1HVbyRtI/Vi16iNm8/q1Yl06FDB9nPoY0aUT+an/qRy1GxhdRWS8F+l3GMj1/9C7Vn9rNm9eoGTUBiCdpSt+pH81M/cjkq9tK+/mHqpiQzt4i34n6ldeVFUtassVoRBdv4NTWiO2p+6kcuR8W2SJuaowXFzPlmGwED+/D1oqM4thR/dnANDQ15sHohrV+OOTo6mujo6AbvN3LkSCtGdT0XSy8zYtxk6q7WsO3AriYpok3pZytEd9T81I+5jsuXL2f58uU3Lcds9Ut7d3d3Vq9ebVYRBSgqKrJSRNdTcqmcASPHU5J7gP8ueBfnNk3TEm0qP1siuqPmp37MdYyOjmb16tW4u7tf975i+0jrJ5iwJiWXyuk7fByFx/bx3+9W8Fh4iNXPWU9T+Nka0R01P/Ujl+Mfto+0sqaOsFc/p/D4AT79PpZnHm78sE8NDY0/JopdRdRoNFptrG95ZQ0xP+nYm1fKB1O6E/JAX6uc505Y008piO6o+akfSx1Vs4po/RyEcnOxpAxPvxFsTvyB7/80xCZFFKznpyREd9T81I9cjootpFeuXJH9mBdLyvB9YCwX9Yd464kpDPPoKPs5Goo1/JSG6I6an/qRy1GxfaQ33hVrLOeLS/EdPgbD6aMs/nklsyOCZT2+ucjtp0REd9T81I9cjootpL6+vrId60p1LQERMRhOH+Or5Yn8ecYk2Y5tKXL6KRXRHTU/9SOXo2Iv7evX0Gks5dW1PPHdLpoPmcUPiWsVUURBPj8lI7qj5qd+5HJUbItUDs4VlTB0YjitRjzCilfC8O8h/yxOGhoaGoptkTZ2TaSCC8X4DnuQs4d+5fXx3RVXRBvrpwZEd9T81I9cjlYvpPVj7euX4G0o5eWWT6ZccKGY/g8EUlqgZ2n8ah6dGmTxsaxFY/zUguiOmp/6Mddx+fLlTJ06VT1j7evXFTeXssoaBo6aQOnZk/yQsJpHQsdZdBxrY6mfmhDdUfNTP+Y6qm6svSVcqrrK49/uxmnIDH5auYaHQ8baOiQNDY0/AIodInr16lVatGjR4O1Pn7vI5L+8TvOBYSyLGY6ve3uzz9mUmOunRkR31PzUj6WOqhkimpaW1uBtT5+7yMAHHuRY2s8snNRF8UUUzPNTK6I7an7qRy5HxRbSS5cuNWi70wUXGPDAaC5fzGd58lomDR9g5cjkoaF+akZ0R81P/cjlqNjnSDt37nzXbU6fu8iA4aMpLzpLbHIqMycENEFk8tAQP7UjuqPmp37kclRsi3Tw4MF3/Ly0ooa/rMihhZsXcavWqaqIwt39REB0R81P/cjlqNhCunbt2tt+pj9TyOS/f8P58loyVv3EjPHqW1vmTn6iILqj5qd+5HJU7KX97TiRdw7/EQ9SfeUSe/Yf4v4u7WwdkoaGxh8cxRbSQYMG3fTe8dNnGTTiQSrLLpKYsoF+PWw3n2hjuZWfaIjuqPmpH7kcFTtE9OrVq9e9PnaqAP8Ro6ksKyJ5bRqhgeoeB3yjn4iI7qj5qR9zHVU3RPTAgQOm/y4qr+bPi7dQZ5RYnZrGlAeHyh1mk/N7P1ER3VHzUz/mOt5uiKhiL+3rOXIyn6djD1Lt5Ma+vXvxctP6RDU0NJSFYoeIVlZWoj97kWEjH8Thnp7s2pTCfa5OVojQNlRWVtKqVStbh2FVRHfU/NSPpY6qGSL6Y3wyw0Y+SE3FJWKXfCRUEQXIyMiwdQhWR3RHzU/9yOWoyEv7nBOnmPvSPOqqK1mXls64B/xsHZLsGAwGW4dgdUR31PzUj1yOiiuklTV1hL/2KXXVlazfuImxw9Qxdt5cOnZU76NbDUV0R81P/cjlaFEfqV6vJyEhAQ8PD/R6PTExMTg7O99yW0v6SH/MOkXvtrU84HufuaGphoqKClq3bm3rMKyK6I6an/qx1FGWPtKIiAjmzZtHeHg44eHhzJ4925LD3JZHh9/LyYO7ZT2m0li1apWtQ7A6ojtqfupHLkezC6ler7/utYeHB+np6bIEo6GhoaFGzC6k6enpuLi4XPeei4sLOp1OtqAABg4cKOvxlIbofiC+o+anfuRyNLuQlpaW3vJ9ue/w2dsr9sksWRDdD8R31PzUj1yOst21v12BPXHihGligC5duuDj44Ofnx+urq4MHTqUlJQUAPz9/TEajezbtw+A6upqTp8+TXFxMS4uLowePZrk5GQA+vfvT4sWLcjOzgZgypQp7Nmzh/Pnz9OuXTsmTJhAQkICAH379sXJyYmdO3cCMHHiRHJycigoKKBNmzaEhIQQGxsLQJ8+fejUqRM7duwAICgoiGPHjpGXl0fLli2ZMWMGsbGxGI1GPD09cXd3Nz2HFhgYSF5eHnq9nmbNmhEZGcnKlSupqamhZ8+eeHp6snnzZgACAgJMN+vg2rCz5ORkKisr6datGz4+PqYlEIYPH05ZWRmHDx8GrvVPr1+/nsuXL+Pm5oa/vz+pqakADBkyhKqqKg4ePAjA9OnT2bp1KyUlJXTq1Inhw4ezZs0aAPz8rj1StnfvXgBCQ0PJysqiqKiIDh06EBgYSFJSEgD9+vXD0dGR3buv9VsHBwej0+koLCykbdu2TJo0ifj4eAB8fHxo3749WVlZZGZm8uabb3L48GHy8/Np1aoVYWFhpnkXvLy8cHV1Zfv27QCMHTuW3NxcTp8+jYODAzNnziQuLo66ujo8PDzo0aMHW7duBWD06NEUFBSQm5uLvb09UVFRJCYmUl1dTY8ePfDy8jJ1OY0cOZKioiKOHj0KQFRUFCkpKVy5cgV3d3d8fX3ZsGEDcG2d8/LyctPqkuHh4aSlpXHp0iU6d+7M4MGDTVOvnTt3jtraWtMww7CwMDIyMjAYDHTs2JGAgABTH9zAgQOxt7c3XbmFhISwa9cuLly4QPv27Rk3bhyJiYkA+Pr60rp1a3bt2gXA5MmT2b9/P2fPnsXJyYng4GDi4uIA8Pb2xsXFhczMTADGjx/PkSNHOHPmDI6OjkyfPp0VK1YgSRK9e/fGzc2Nbdu2ATBmzBhOnTrFyZMnad68OREREcTHx1NbW0uvXr3Yu3evKd5Ro0ZRWFjI8ePHsbOzY9asWSQlJVFVVUX37t3x9vZm48aNAIwYMQKDwcCRI0cAiIyMJDU1lfLycrp27cqAAQNYt24dAEOHDqWiooKcnBwAZsyYwaZNmygrK7trjZg2bRrbt29vVI3IzMxk9uzZd60Rc+fOJTMzEycnJxwcHG5uOEpmsnjxYsnf3/+695ydnaWNGzfecvvQ0FBzTyFJkiQtW7bMov3Uguh+kiS+o+anfix1vLGumd2uDQoKuuX7cs+mHRISIuvxlIbofiC+o+anfuRyNLuQ1l+O1qPX6xk8ePBtnyO1lPrLGlER3Q/Ed9T81I9cjhb1kcbHxzN//nyGDBnC7t27Tf1jcnLhwgXZj6kkRPcD8R01P/Ujl6NFt6w8PDxYsGAB4eHhLFiwQPbWKPzfTRBREd0PxHfU/NSPXI6Kfb6h/i61qIjuB+I7an7qRy5HxRbSc+fOWbSfuUua2Go/0f1AfEfNT97zNea71tSOTb7USFOjli+ppYju15hzqsVR85N3v8buawk3FlKrz5Dft29fPD09zd6voKDgpnVRtP3Us58tzqnt98fczxbn1Ol05Ofnm15bvZBqaGhoiI5wl/YaGhoaTY1WSDU0NDQaiaKWGjFn5n21otPpTJNp7N69m6+++ko4x3rmz5/Pa6+9Jpxfeno6er3eNMrvdsOm1YperzdNl6nX6wkPD79pRKPa0Ol0zJ492zSJST2y1ZzGDvqXk99PhpKbmyuFh4fbMBrrsGDBguv++8YJYEQhOztbAqSSkhJbhyIrGzdulGJiYiRJuvYd9fDwsHFE8vP776gkSSZftRIfH2/6Pt6IXDVHMYU0Nzf3lrNKiUR2dvZ1Trm5uRIg5ebm2jAq6xAfHy95eHgIV0hvdBIxdzf+Haq9kNZzYyGVs+Yopo+0qWbetyX+/v589dVXptf1c7je6K12EhISCA8Pt3UYsqPX6zEYDDg7O6PT6SgtLVX9Je+tcHFxYdCgQaZL/PHjx9s6JKsgZ81RTCFtqpn3bc3vC0xsbCxBQUFC9SGWlpYK5fN7dDodLi4upj61JUuWmCYQF4n6SYg8PT2Jj48X8kcR5K05irrZdCtuJ6t2SktLSUhIuKnzW+3ExcURExNj6zCsgsFgQK/Xm378YmJi6NChA5Jgj2Knp6ezYMEC9Ho9c+bMAWDx4sU2jqrpsKTmKKZF6uzsfNMvQf1llIjMnz+fjRs3CuWXnp5OZGSkrcOwGh4eHjg7O5tyVv//InU/6fV6du/eTVBQEDExMeTm5hIXF3fT6sEiIGfNUUwhbaqZ95XAwoULmT9/Ph4eHpSWlgrV6o6Li2PJkiUsWbIEvV7Pe++9J0yhEbE/9EZ0Oh1Dhgwxvfbw8OC1114T6jtaj6w1x9I7YNbgxkcRgoKCbBiNdYiPjzetb1VSUiItXrzYxhFZDwR8IiEoKMh0117Ex59yc3OlefPmXffeja/VCrd4HE+umqOosfZ6vZ7FixebZt4X7WFuvV5/0wQuzs7OlJSU2Cgi61BaWsqSJUuYP38+MTExzJkzB39/f1uHJQulpaXMnz+fQYMGkZ2dbbqyEIn09HR0Op3pby8oKEjVjunp6WzcuJGFCxcyb948hgwZYrqBJlfNUVQh1dDQ0FAjiukj1dDQ0FArWiHV0NDQaCRaIdXQ0NBoJFoh1dDQ0GgkWiHV0NDQaCRaIdXQ0NBoJFoh1dDQ0Ggk/w9IbGGPHPQ6tgAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 400x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "res = res.cpu()\n",
    "plt.plot(res)\n",
    "plt.plot(torch.arange(len(res))*(res[1]-res[0])+res[0], \"--k\")"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "55809d8c-d091-4d70-a970-9662972b8d91",
   "metadata": {},
   "source": [
    "## Calculate shapley values for this case."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 146,
   "id": "54b08adf-5e6a-48f5-b46c-776c57f12881",
   "metadata": {},
   "outputs": [],
   "source": [
    "from ground_truth_models import ShapleyValues"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 204,
   "id": "8129160d-30a1-4fbe-9361-b25f7adc8338",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0\n",
      "1\n",
      "2\n",
      "3\n",
      "4\n",
      "5\n",
      "6\n",
      "7\n",
      "8\n",
      "9\n",
      "10\n",
      "11\n",
      "12\n",
      "13\n",
      "14\n",
      "15\n",
      "16\n",
      "17\n",
      "18\n",
      "19\n",
      "0\n",
      "1\n",
      "2\n",
      "3\n",
      "4\n",
      "5\n",
      "6\n",
      "7\n",
      "8\n",
      "9\n",
      "10\n",
      "11\n",
      "12\n",
      "13\n",
      "14\n",
      "15\n",
      "16\n",
      "17\n",
      "18\n",
      "19\n"
     ]
    }
   ],
   "source": [
    "shap_lists_runs = []\n",
    "for k, r in enumerate(run_ids):\n",
    "    myShap = ShapleyValues(model_list[k], tok_list[k], method=\"kernel\", num_samples = 500, impute_token=\"the\")\n",
    "    use_cls = (\"bert\" in r)\n",
    "    relevant_token = \"perfect\"\n",
    "    shap_list = []\n",
    "    cls_tok = torch.tensor([101])\n",
    "    sep_tok = torch.tensor([102])\n",
    "    input_tok = torch.tensor(tok_list[k].convert_tokens_to_ids(relevant_token)).reshape(1)\n",
    "    if use_cls:\n",
    "        x = torch.cat((cls_tok, input_tok), dim=0)\n",
    "    else:\n",
    "        x = input_tok\n",
    "        \n",
    "    for i in range(20):\n",
    "        print(i)\n",
    "        if use_cls:\n",
    "            x_sep =  torch.cat((x, sep_tok), dim=0)\n",
    "        else:\n",
    "            x_sep = x\n",
    "        #print(x_sep)\n",
    "        shapley = myShap.get_signed_importance_for_tokens([i.item() for i in x_sep.flatten()])\n",
    "        #print(shapley)\n",
    "        shap_list.append(torch.tensor(shapley))\n",
    "        x = torch.cat((x, input_tok), dim=0)\n",
    "    shap_lists_runs.append(shap_list)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "707fa225-6acc-4356-b16b-a9c92fb066e3",
   "metadata": {},
   "source": [
    "## Compute SHAP values for the linear model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 84,
   "id": "a1e55ea1-cdcf-434f-87a7-47f77bcdd265",
   "metadata": {},
   "outputs": [],
   "source": [
    "## Wrapper for linear model to compute shap values\n",
    "class LinWrapper:\n",
    "    def __init__(self, mymodel):\n",
    "        self.model = mymodel\n",
    "        self.device = wrapper.use_device\n",
    "        \n",
    "    def __call__(self, inputs):\n",
    "        # pad to full length\n",
    "        input_pad = torch.ones(len(inputs), 32, dtype=torch.long)*1996\n",
    "        input_pad[:, 1:inputs.shape[1]+1] = inputs\n",
    "        input_pad[:, 0] = 101\n",
    "        input_pad[:, -1] = 102\n",
    "        #print(input_pad)\n",
    "        input_embs = wrapper.compute_embeddings(input_pad.to(wrapper.use_device), 1)\n",
    "        logit_score = self.model.forward(input_embs.reshape(len(input_embs), -1))\n",
    "        return {\"logits\": logit_score}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 85,
   "id": "f3b59a7a-57b2-439e-bb7b-e40b631efb2b",
   "metadata": {},
   "outputs": [],
   "source": [
    "lin_model_wrapper = LinWrapper(fc_model)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 86,
   "id": "1f1c3aeb-387d-4ca2-921f-d2d61b537db5",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0\n",
      "1\n",
      "2\n",
      "3\n",
      "4\n",
      "5\n",
      "6\n",
      "7\n",
      "8\n",
      "9\n",
      "10\n",
      "11\n",
      "12\n",
      "13\n",
      "14\n",
      "15\n",
      "16\n",
      "17\n",
      "18\n",
      "19\n"
     ]
    }
   ],
   "source": [
    "myShaplin = ShapleyValues(lin_model_wrapper, tokenizer, method=\"kernel\", num_samples = 500, impute_token=\"the\")\n",
    "use_cls=False\n",
    "shap_list_lin = []\n",
    "cls_tok = torch.tensor([101])\n",
    "input_tok = torch.tensor([3819])\n",
    "if use_cls:\n",
    "    x = torch.cat((cls_tok, input_tok), dim=0)\n",
    "else:\n",
    "    x = input_tok\n",
    "for i in range(20):\n",
    "    print(i)\n",
    "    #logit_results = []\n",
    "    shapley = myShaplin.get_signed_importance_for_tokens([i.item() for i in x.flatten()])\n",
    "    #print(shapley)\n",
    "    shap_list_lin.append(torch.tensor(shapley))\n",
    "    x = torch.cat((x, input_tok), dim=0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 205,
   "id": "05a3f07d-c699-4cde-be44-6f847039bce6",
   "metadata": {},
   "outputs": [],
   "source": [
    "shap_list_bert, shap_list_gpt = shap_lists_runs[0], shap_lists_runs[1]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 229,
   "id": "350b5662-d79d-47dc-9573-47d3afa89087",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "tensor([ 0.4813,  0.4511,  0.4536,  0.4323,  0.4473,  0.3881, -0.0037],\n",
       "       dtype=torch.float64)"
      ]
     },
     "execution_count": 229,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "shap_list_bert[4]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 207,
   "id": "a5070280-bf85-46fc-9e13-9d33175d692b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0\n",
      "1\n",
      "2\n",
      "3\n",
      "4\n",
      "5\n",
      "6\n",
      "7\n",
      "8\n",
      "9\n"
     ]
    }
   ],
   "source": [
    "my_matrix = torch.ones(20, 20)*float(\"nan\")\n",
    "my_matrix_lin = torch.zeros(20,20)*float(\"nan\")\n",
    "my_matrix_exp = torch.zeros(20, 20)*float(\"nan\")\n",
    "my_matrix_gpt = torch.zeros(20, 20)*float(\"nan\")\n",
    "\n",
    "for i in range(10):\n",
    "    print(i)\n",
    "    my_matrix[i, :len(shap_list_bert[i])-2] = shap_list_bert[i][1:-1]\n",
    "    my_matrix_lin[i, :len(shap_list_lin[i])] = shap_list_lin[i]\n",
    "    my_matrix_gpt[i, :len(shap_list_lin[i])] = shap_list_gpt[i]\n",
    "    my_matrix_exp[i, :i+1] = 1.5"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 208,
   "id": "ba0331ee-ac3a-4bf7-94ba-190c349b68bf",
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save((my_matrix, my_matrix_lin, my_matrix_gpt, my_matrix_exp), \"figures/mat_motivation.pt\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 209,
   "id": "cddac2b8-0aae-4fc3-9dbe-61b14aed0f08",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "my_matrix, my_matrix_lin, my_matrix_gpt, my_matrix_exp = torch.load(\"figures/mat_motivation.pt\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 210,
   "id": "d7a1524a-bf94-4ec2-8f00-1594afc76047",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 227,
   "id": "39af3c9e-dd75-4a48-b221-dc897dcc156b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 328x300 with 5 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.style.use('./figures/matplot.mplstyle')\n",
    "plt.rc('axes', titlesize=10) \n",
    "f, ax = plt.subplots(2, 2)\n",
    "ax = [ax[0][0], ax[0][1], ax[1][0], ax[1][1]]\n",
    "sub_sz = 10\n",
    "vmax = 7 #None #5\n",
    "vmin = 0.3 #None # 0.5\n",
    "ax[0].matshow(my_matrix_exp[:sub_sz, :sub_sz], cmap = \"jet\", vmax=vmax, vmin=vmin)\n",
    "ax[0].set_title(\"expected\")\n",
    "ax[1].matshow(my_matrix_lin[:sub_sz, :sub_sz], cmap = \"jet\", vmax=vmax, vmin=vmin)\n",
    "ax[1].set_title(\"fully connected\")\n",
    "ax[2].matshow(my_matrix[:sub_sz, :sub_sz], cmap = \"jet\", vmax=vmax, vmin=vmin)\n",
    "ax[2].set_title(\"BERT\")\n",
    "pcs = ax[3].matshow(my_matrix_gpt[:sub_sz, :sub_sz], cmap = \"jet\", vmax=vmax, vmin=vmin)\n",
    "ax[3].set_title(\"GPT-2\")\n",
    "for a in ax:\n",
    "    a.set_xlabel(\"token \\#\")\n",
    "    a.grid([])\n",
    "ax[2].set_xticks([])\n",
    "ax[3].set_xticks([])\n",
    "ax[0].set_ylabel(\"seq. length\")\n",
    "ax[2].set_ylabel(\"seq. length\")\n",
    "f.set_size_inches(3.28,3)\n",
    "plt.tight_layout()\n",
    "cbar = f.colorbar(pcs, ax=[ax[1], ax[3]], shrink=0.8, pad=0.15)\n",
    "cbar.ax.get_yaxis().labelpad = 10\n",
    "cbar.ax.get_yaxis().set_ticks([1.0,3.0, 5])\n",
    "cbar.set_label('SHAP values', rotation=270)\n",
    "f.savefig(\"figures/figure_motivation.pdf\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "transformer",
   "language": "python",
   "name": "transformer"
  },
  "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.10.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
