{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Gradient-Based Explanations of Causal Proxy Model\n",
    "\n",
    "This notebook generates gradient-based visualizations to show how different representations within a CPM model capture the different aspects that they are trained to represent."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## General Setup\n",
    "\n",
    "Loads modules, CEBaB data, and Static Proxy Model."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "from typing import List\n",
    "from libs import *\n",
    "from modelings.modelings_bert import *\n",
    "from modelings.modelings_roberta import *\n",
    "from modelings.modelings_gpt2 import *\n",
    "from modelings.modelings_lstm import *\n",
    "from IPython.display import display, HTML\n",
    "\"\"\"\n",
    "For evaluate, we use a single random seed, as\n",
    "the models are trained with 5 different seeds\n",
    "already.\n",
    "\"\"\"\n",
    "_ = random.seed(123)\n",
    "_ = np.random.seed(123)\n",
    "_ = torch.manual_seed(123)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Load CEBaB data and model weights."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {},
   "outputs": [],
   "source": [
    "\"\"\"\n",
    "The following blocks will run CEBaB benchmark in\n",
    "all the combinations of the following conditions.\n",
    "\"\"\"\n",
    "grid = {\n",
    "    \"eval_split\": [\"test\"],\n",
    "    # dev,test\n",
    "    \"control\": [\"ablation\"],\n",
    "    # baseline-random,baseline-blackbox,hdims,layers,ks,approximate,ablation\n",
    "    \"seed\": [42],\n",
    "    # 42, 66, 77\n",
    "    \"h_dim\": [192],\n",
    "    # 1,16,64,128,192\n",
    "    # 1,16,64,75\n",
    "    \"interchange_layer\" : [10],\n",
    "    # 0,1; 2,4,6,8,10,12\n",
    "    \"class_num\": [5],\n",
    "    \"k\" : [19684], \n",
    "    # 0;10,100,500,1000,3000,6000,9848,19684\n",
    "    \"alpha\" : [1.0],\n",
    "    # 0.0,1.0\n",
    "    \"beta\" : [1.0],\n",
    "    # 0.0,1.0\n",
    "    \"gemma\" : [0.0],\n",
    "    # 0.0,3.0\n",
    "    \"model_arch\" : [\"bert-base-uncased\"],\n",
    "    # lstm, bert-base-uncased, roberta-base, gpt2\n",
    "    \"lr\" : [\"8e-05\"],\n",
    "    # 8e-05; 0.001\n",
    "    \"counterfactual_type\" : [\"true\"]\n",
    "    # approximate,true\n",
    "}\n",
    "\n",
    "keys, values = zip(*grid.items())\n",
    "permutations_dicts = [dict(zip(keys, v)) for v in itertools.product(*values)]\n",
    "\n",
    "device = 'cuda:2'\n",
    "batch_size = 32\n",
    "\n",
    "if grid[\"control\"][0] == \"hdims\" or grid[\"control\"][0] == \"layers\":\n",
    "    assert grid[\"eval_split\"][0] == \"dev\"\n",
    "else:\n",
    "    assert grid[\"eval_split\"][0] == \"test\"\n",
    "    \n",
    "aspect_label_encode = {\n",
    "    \"Negative\":0,\n",
    "    \"Positive\":1,\n",
    "    \"unknown\":2,\n",
    "    \"no majority\": 2,\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Running for this setting:  (('eval_split', 'test'), ('control', 'ablation'), ('seed', 42), ('h_dim', 192), ('interchange_layer', 10), ('class_num', 5), ('k', 19684), ('alpha', 1.0), ('beta', 1.0), ('gemma', 0.0), ('model_arch', 'bert-base-uncased'), ('lr', '8e-05'), ('counterfactual_type', 'true'))\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Using custom data configuration CEBaB--CEBaB-ccd674d249652bd4\n",
      "Reusing dataset parquet (../train_cache/CEBaB___parquet/CEBaB--CEBaB-ccd674d249652bd4/0.0.0/7328ef7ee03eaf3f86ae40594d46a1cec86161704e02dd19f232d81eee72ade8)\n"
     ]
    },
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "080b2113a7154d4497ca684c470b07c3",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/5 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dropping no majority reviews: 16.6382% of train_exclusive dataset.\n",
      "Dropping no majority reviews: 16.03% of train_inclusive dataset.\n",
      "intervention_h_dim=192\n"
     ]
    }
   ],
   "source": [
    "i = 0\n",
    "eval_split=permutations_dicts[i][\"eval_split\"]\n",
    "seed=permutations_dicts[i][\"seed\"]\n",
    "class_num=permutations_dicts[i][\"class_num\"]\n",
    "alpha=permutations_dicts[i][\"alpha\"]\n",
    "beta=permutations_dicts[i][\"beta\"]\n",
    "gemma=permutations_dicts[i][\"gemma\"]\n",
    "h_dim=permutations_dicts[i][\"h_dim\"]\n",
    "dataset_type = f'{class_num}-way'\n",
    "control=permutations_dicts[i][\"control\"]\n",
    "model_arch=permutations_dicts[i][\"model_arch\"]\n",
    "k=permutations_dicts[i][\"k\"]\n",
    "interchange_layer=permutations_dicts[i][\"interchange_layer\"]\n",
    "lr=permutations_dicts[i][\"lr\"]\n",
    "counterfactual_type=permutations_dicts[i][\"counterfactual_type\"]\n",
    "\n",
    "if model_arch == \"bert-base-uncased\":\n",
    "    model_path = \"BERT\"\n",
    "    model_module = BERTForCEBaB\n",
    "    explainer_module = CausalProxyModelForBERT\n",
    "elif model_arch == \"roberta-base\":\n",
    "    model_path = \"RoBERTa\" \n",
    "    model_module = RoBERTaForCEBaB\n",
    "    explainer_module = CausalProxyModelForRoBERTa\n",
    "elif model_arch == \"gpt2\":\n",
    "    model_path = \"gpt2\"\n",
    "    model_module = GPT2ForCEBaB\n",
    "    explainer_module = CausalProxyModelForGPT2\n",
    "elif model_arch == \"lstm\":\n",
    "    model_path = \"lstm\"\n",
    "    model_module = LSTMForCEBaB\n",
    "    explainer_module = CausalProxyModelForLSTM\n",
    "model_path += f\"-{control}\"\n",
    "grid_conditions=(\n",
    "    (\"eval_split\", eval_split),\n",
    "    (\"control\", control),\n",
    "    (\"seed\", seed),\n",
    "    (\"h_dim\", h_dim),\n",
    "    (\"interchange_layer\", interchange_layer),\n",
    "    (\"class_num\", class_num),\n",
    "    (\"k\", k),\n",
    "    (\"alpha\", alpha),\n",
    "    (\"beta\", beta),\n",
    "    (\"gemma\", gemma),\n",
    "    (\"model_arch\", model_arch),\n",
    "    (\"lr\", lr),\n",
    "    (\"counterfactual_type\", counterfactual_type)\n",
    ")\n",
    "print(\"Running for this setting: \", grid_conditions)\n",
    "\n",
    "blackbox_model_path = f'CEBaB/{model_arch}.CEBaB.sa.'\\\n",
    "                      f'{class_num}-class.exclusive.seed_{seed}'\n",
    "cpm_model_path = f'../proxy_training_results/{model_path}/'\\\n",
    "                 f'cebab.alpha.{alpha}.beta.{beta}.gemma.{gemma}.'\\\n",
    "                 f'lr.{lr}.dim.{h_dim}.hightype.{model_arch}.'\\\n",
    "                 f'CEBaB.cls.dropout.0.1.enc.dropout.0.1.counter.type.'\\\n",
    "                 f'{counterfactual_type}.k.{k}.int.layer.{interchange_layer}.'\\\n",
    "                 f'seed_{seed}/'\n",
    "\n",
    "# load data from HF\n",
    "cebab = datasets.load_dataset(\n",
    "    'CEBaB/CEBaB', use_auth_token=True,\n",
    "    cache_dir=\"../train_cache/\"\n",
    ")\n",
    "\n",
    "train, dev, test = preprocess_hf_dataset_inclusive(\n",
    "    cebab, verbose=1, dataset_type=dataset_type\n",
    ")\n",
    "\n",
    "eval_dataset = dev if eval_split == 'dev' else test\n",
    "\n",
    "tf_model = model_module(\n",
    "    blackbox_model_path, \n",
    "    device=device, \n",
    "    batch_size=batch_size\n",
    ")\n",
    "explainer = explainer_module(\n",
    "    blackbox_model_path,\n",
    "    cpm_model_path, \n",
    "    device=device, \n",
    "    batch_size=batch_size,\n",
    "    intervention_h_dim=h_dim,\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Computing Saliency Map\n",
    "\n",
    "The code below sets up a pipeline for computing and visualizing saliency maps for the causal proxy model, and the control blackbox model."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### General Utility Functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def softmax(x):\n",
    "    e_x = np.exp(x - x.max())\n",
    "    return e_x / np.sum(e_x)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def tokenize_sentence(sentence, explanator):\n",
    "    \"\"\"\n",
    "    Tokenize the sentence using the `explanator` tokenizer\n",
    "    \"\"\"\n",
    "    explanator.blackbox_model.eval()\n",
    "    explanator.cpm_model.model.eval()\n",
    "    is_split_into_words = isinstance(sentence, List)\n",
    "    # Original ratings.\n",
    "    x = explanator.tokenizer([sentence], padding=True, truncation=True, return_tensors='pt', is_split_into_words=is_split_into_words)\n",
    "    x_batch = {k: v.to(explanator.device) for k, v in x.items()}\n",
    "    return x_batch"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_max(\n",
    "    x, \n",
    "    key_fn=lambda x: x\n",
    "):\n",
    "    \"\"\"\n",
    "    Returns maximum of n-dimensional array `x` at last dimension.\n",
    "\n",
    "    Parameters \n",
    "    -----------\n",
    "    x : numpy narray\n",
    "    key_fn : single-variable function, default is identity\n",
    "        Function by which to sort input in order to choose maximum value. By default,\n",
    "        no key function is used.\n",
    "    \"\"\"\n",
    "    index = np.argmax(key_fn(x), axis=-1)\n",
    "    shp = np.array(x.shape)\n",
    "    dim_index = [np.arange(i) for i in shp[:-1]]\n",
    "    dim_index.append(index)\n",
    "    return x[tuple(dim_index)]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "def highlight_word(word, weight, is_significant, is_pos, pos_color='224,108,120', neg_color='88,117,220', max_weight=0.8):\n",
    "    \"\"\"\n",
    "    Highlights a word using HTML based on its weight\n",
    "    \"\"\"\n",
    "    if not is_significant:\n",
    "        return word\n",
    "    color = pos_color if is_pos else neg_color\n",
    "    return f\"\"\"<span style=\"background-color:rgba({color},{str(abs(weight) / max_weight)})\">{word}</span>\"\"\"\n",
    "\n",
    "def display_colored_text(plot_df, n_color=10):\n",
    "    \"\"\"\n",
    "    Displays sentence as colored words \n",
    "    NOTE: needs some calibration\n",
    "    \"\"\"\n",
    "    result = \"\"\n",
    "    for model in plot_df['model'].unique():\n",
    "        model_df = plot_df[plot_df['model'] == model].copy()\n",
    "        most_significant_words = model_df.sort_values(by='weight', key=np.abs, ascending=False).iloc[:n_color]['word']\n",
    "        model_df['is_pos'] = model_df['weight'] > 0\n",
    "        model_df['is_significant'] = model_df['word'].isin(most_significant_words)\n",
    "        to_plot = model_df.apply(\n",
    "            lambda r: highlight_word(r['word'], r['weight'], r['is_significant'], r['is_pos'], max_weight=model_df['weight'].abs().max()),\n",
    "            axis=1\n",
    "        )\n",
    "        colored_text = ' '.join(list(to_plot.values))\n",
    "        title = f'Saliency for {model}:'\n",
    "        result += f'<p>{title}</p><p>{colored_text}</p>'\n",
    "    result = f\"\"\"<div style=\"background-color:#FFFFFF;color:black\">{result}</div>\"\"\"\n",
    "    display(HTML(result))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "def display_bar_plot(plot_df):\n",
    "    \"\"\"\n",
    "    Helper function to plot saliency map for `explain_interventions_with_gradients`.\n",
    "    \"\"\"\n",
    "    g = sns.catplot(\n",
    "        data=plot_df,\n",
    "        x='word',\n",
    "        y='weight',\n",
    "        col='model',\n",
    "        kind='bar'\n",
    "    )\n",
    "    axes = g.axes[0]\n",
    "    for ax in axes:\n",
    "        _ = ax.set_xticklabels(ax.get_xticklabels(), rotation=90)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Gradient Functions\n",
    "\n",
    "The functions below each compute the gradients across the lowest layer of the provided model, which corresponds to the inputted token embeddings. These gradients are then used to compute and visualize the saliency of the model with respect to the specified aspect.\n",
    "\n",
    "The functions are as follows:\n",
    " - `compute_saliency_of_logits`: takes the gradient of the input layer with respect to the logit of the specified aspect; if no aspect is specified (for control), then the gradient is taken with respect to the overall sentiment logit. NOTE: this function will not work with the blackbox model, since it does not compute aspect-specific logits.\n",
    " - `compute_saliency_of_final_layer`: takes the gradient of the input layer with respect to the slice of the final layer that represents the specified aspect; if no aspect is specified (for control), then the gradient is taken with respect to the entire final layer.\n",
    " - `compute_weighted_saliency`: takes the gradient of the input layer with respect to the slice of the final layer that represents the specified aspect, but weighted by the gradient from the final logit to the final layer; if no aspect is specified, then the gradient is taken with respect to the entire final layer.\n",
    " - `compute_guided_saliency`: takes the gradient of the input layer with respect to logit for overall sentiment, but only guided through the slice of the final layer that represents the specified aspect; if no aspect is specified (for control), then the gradient is guided through the entire final layer (should be the same as taking the gradient of the input with respect to the final overall logit)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_saliency_of_logits(model, x, explanator, aspect=None):\n",
    "    \"\"\"\n",
    "    Applies a saliency map on the logits of the output of the `explanator` \n",
    "    model when run on a single input sentence. It uses the tokens produced by\n",
    "    the `explanator` tokenizer to determine which parts of the input had the\n",
    "    most influence on the chosen logit.\n",
    "\n",
    "    aspect : int\n",
    "        Chooses which output to focus on, from the following:\n",
    "        0 - ambiance\n",
    "        1 - food\n",
    "        2 - noise\n",
    "        3 - service\n",
    "        None - overall review\n",
    "    \"\"\"\n",
    "    logit_index = aspect + 1 if aspect is not None else 0\n",
    "\n",
    "    model.eval()\n",
    "    model.zero_grad()\n",
    "\n",
    "    outputs = model(\n",
    "        **x,\n",
    "        output_hidden_states=True,\n",
    "    )\n",
    "\n",
    "    outputs.hidden_states[0].retain_grad()\n",
    "\n",
    "    classification = outputs.logits[logit_index].argmax()\n",
    "    output_max = outputs.logits[logit_index][0, classification]\n",
    "    output_max.backward()\n",
    "\n",
    "    return outputs.hidden_states[0].grad.data.clone().detach().cpu().numpy()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_saliency_of_final_layer(model, x, explanator, aspect=None):\n",
    "    \"\"\"\n",
    "    Computes gradient-based saliency of how the input features affect \n",
    "    the computation of the final layer of the CPM model, which is used to\n",
    "    predict the overall sentiment of a sentence.\n",
    "\n",
    "    model\n",
    "        model to explain (either cpm_model.model or blackbox_model)\n",
    "    x\n",
    "        tokenized sentence to explain\n",
    "    explanator\n",
    "        CPM model\n",
    "    aspect : int\n",
    "        Different aspects correspond to different slices of the \n",
    "        final model layer. The options are:\n",
    "        0 - ambiance\n",
    "        1 - food\n",
    "        2 - noise\n",
    "        3 - service\n",
    "        None - all of the above (for control)\n",
    "    \"\"\"\n",
    "    # zero gradients\n",
    "    model.eval()\n",
    "    model.zero_grad()\n",
    "\n",
    "    # compute output\n",
    "    outputs = model(\n",
    "        **x,\n",
    "        output_hidden_states=True,\n",
    "    )\n",
    "    outputs.hidden_states[0].retain_grad()\n",
    "\n",
    "    # get slice of final layer that corresponds to aspect provided\n",
    "    if aspect is not None:\n",
    "        start_idx = aspect*explanator.intervention_h_dim\n",
    "        end_idx = (aspect+1)*explanator.intervention_h_dim\n",
    "    else:\n",
    "        start_idx = 0\n",
    "        end_idx = outputs.hidden_states[-1].size(2)\n",
    "    \n",
    "    # run gradient backwards for the final layer,\n",
    "    # only computing the gradients for the range of the specified aspect\n",
    "    output_grad = outputs.hidden_states[-1]\n",
    "    mask = torch.zeros_like(output_grad)\n",
    "    mask[:, 0, start_idx:end_idx] = 1\n",
    "    output_grad.backward(gradient=mask)\n",
    "\n",
    "    # return gradient at input layer\n",
    "    return outputs.hidden_states[0].grad.data.clone().detach().cpu().numpy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_weighted_saliency(model, x, explanator, aspect=None):\n",
    "    \"\"\"\n",
    "    Computes the gradient-based saliency of the output logits,\n",
    "    through the specified slice of the final layer of the CPM model.\n",
    "    Corresponds to\n",
    "    dy/d(aspect representation) * d(aspect representation)/dx\n",
    "\n",
    "    model\n",
    "        model to explain (either cpm_model.model or blackbox_model)\n",
    "    x\n",
    "        tokenized sentence to explain\n",
    "    explanator\n",
    "        CPM model\n",
    "    aspect : int\n",
    "        Different aspects correspond to different slices of the \n",
    "        final model layer. The options are:\n",
    "        0 - ambiance\n",
    "        1 - food\n",
    "        2 - noise\n",
    "        3 - service\n",
    "        None - all of the above (for control)\n",
    "    \"\"\"\n",
    "    # zero gradients\n",
    "    model.eval()\n",
    "    model.zero_grad()\n",
    "\n",
    "    # compute output\n",
    "    outputs = model(\n",
    "        **x,\n",
    "        output_hidden_states=True,\n",
    "    )\n",
    "\n",
    "    # get aspect representation\n",
    "    if aspect is not None:\n",
    "        start_idx = aspect*explanator.intervention_h_dim\n",
    "        end_idx = (aspect+1)*explanator.intervention_h_dim\n",
    "    else:\n",
    "        start_idx = 0\n",
    "        end_idx = outputs.hidden_states[-1].size(2)\n",
    "\n",
    "    first_layer = outputs.hidden_states[0]\n",
    "    last_layer = outputs.hidden_states[-1]\n",
    "\n",
    "    first_layer.retain_grad()\n",
    "\n",
    "    # compute backwards gradient from final layer to input layer\n",
    "    mask = torch.zeros_like(last_layer)\n",
    "    mask[:, 0, start_idx:end_idx] = 1\n",
    "    last_layer.backward(gradient=mask, retain_graph=True)\n",
    "\n",
    "    first_layer_gradient = first_layer.grad.data.clone().detach()\n",
    "\n",
    "\n",
    "    # reset gradients\n",
    "    model.zero_grad()\n",
    "    \n",
    "    last_layer.retain_grad()\n",
    "\n",
    "    # compute backwards gradient from output logits to final layer\n",
    "    logits = outputs.logits[0] if isinstance(outputs.logits, tuple) else outputs.logits\n",
    "    classification = logits.argmax()\n",
    "    output_max = logits[0, classification]\n",
    "    output_max.backward(retain_graph=True)\n",
    "\n",
    "    last_layer_gradient = last_layer.grad.data.clone().detach()\n",
    "\n",
    "    # multiply gradient at input level with gradient of CLS vector at final layer, \n",
    "    # since only this slice is used in computing the output logits\n",
    "    weighted_gradient = first_layer_gradient * last_layer_gradient[:, 0, :]\n",
    "\n",
    "    return weighted_gradient.clone().detach().cpu().numpy()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [],
   "source": [
    "def compute_guided_saliency(model, x, explanator, aspect=None):\n",
    "    \"\"\"\n",
    "    Computes the gradient-based saliency of the output logits,\n",
    "    through the specified slice of the final layer of the CPM model.\n",
    "    Corresponds to\n",
    "    dy/d(aspect representation) * d(aspect representation)/dx\n",
    "\n",
    "    model\n",
    "        model to explain (either cpm_model.model or blackbox_model)\n",
    "    x\n",
    "        tokenized sentence to explain\n",
    "    explanator\n",
    "        CPM model\n",
    "    aspect : int\n",
    "        Different aspects correspond to different slices of the \n",
    "        final model layer. The options are:\n",
    "        0 - ambiance\n",
    "        1 - food\n",
    "        2 - noise\n",
    "        3 - service\n",
    "        None - all of the above (for control)\n",
    "    \"\"\"\n",
    "    # zero gradients\n",
    "    model.eval()\n",
    "    model.zero_grad()\n",
    "\n",
    "    # compute output\n",
    "    outputs = model(\n",
    "        **x,\n",
    "        output_hidden_states=True,\n",
    "    )\n",
    "\n",
    "    first_layer = outputs.hidden_states[0]\n",
    "    last_layer = outputs.hidden_states[explanator.interchange_hidden_layer]\n",
    "\n",
    "    # get aspect representation\n",
    "    if aspect is not None:\n",
    "        start_idx = aspect*explanator.intervention_h_dim\n",
    "        end_idx = (aspect+1)*explanator.intervention_h_dim\n",
    "    else:\n",
    "        start_idx = 0\n",
    "        end_idx = outputs.hidden_states[-1].size(2)\n",
    "    \n",
    "    # compute backwards gradient from output logits to final layer\n",
    "    last_layer.retain_grad()\n",
    "\n",
    "    logits = outputs.logits[0] if isinstance(outputs.logits, tuple) else outputs.logits\n",
    "    classification = logits.argmax()\n",
    "    output_max = logits[0, classification]\n",
    "    output_max.backward(retain_graph=True)\n",
    "\n",
    "    last_layer_gradient = last_layer.grad.data.clone().detach()\n",
    "\n",
    "    # reset gradients\n",
    "    model.zero_grad()\n",
    "\n",
    "    # compute backwards gradient from final layer to input layer,\n",
    "    # guided by the gradient from the logits to the final layer\n",
    "    first_layer.retain_grad()\n",
    "\n",
    "    \n",
    "    mask = torch.zeros_like(last_layer)\n",
    "    mask[:, 0, start_idx:end_idx] = last_layer_gradient[:, 0, start_idx:end_idx]\n",
    "    last_layer.backward(gradient=mask)\n",
    "\n",
    "    first_layer_gradient = first_layer.grad.data.clone().detach().cpu().numpy()\n",
    "\n",
    "    return first_layer_gradient"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Visualization Function\n",
    "\n",
    "This function uses one of the defined gradient-computation functions to visualize the aspect-specific saliency across an input sentence.\n",
    "This function allows the user to specify whether to show the control blackbox model, whether to normalize the output, and whether to select the maximum gradient to display by applying a key (such as `np.abs`), as well as other display flags."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [],
   "source": [
    "def explain_interventions_with_gradients(\n",
    "        sentence, \n",
    "        explanator, \n",
    "        aspect, \n",
    "        show_control=True,\n",
    "        normalize=True, \n",
    "        softmax_norm=False, \n",
    "        control_for_gradient=False, \n",
    "        saliency_fn=compute_guided_saliency,\n",
    "        grad_apply_fn=lambda x: x,\n",
    "        max_key_fn=lambda x: x,\n",
    "        display_fn=display_bar_plot\n",
    "    ):\n",
    "    \"\"\"\n",
    "    Provides gradient-based saliency map to display which tokens \n",
    "    had the most effect on the intermediate representations\n",
    "    encoded by the `explanator.cpm_model` in the final layer.\n",
    "\n",
    "    sentence : str\n",
    "        Single input example\n",
    "    explanator : StaticCausalProxyModelForBERT \n",
    "    aspect : int\n",
    "        Chooses which embedded aspect to focus on.\n",
    "        0 - ambiance\n",
    "        1 - food\n",
    "        2 - noise\n",
    "        3 - service\n",
    "    show_control : bool, default True\n",
    "        Whether to display a second saliency plot, with \n",
    "    normalize : bool, default True\n",
    "        Whether to normalize output gradients\n",
    "    softmax_norm : bool, default False\n",
    "        Whether to apply softmax function in normalization (by default,\n",
    "        divide the gradients by their sum)\n",
    "    control_for_gradient : bool, default False\n",
    "        Whether to control for the change in gradients by computing \n",
    "        saliency both for the particular aspect representation and for \n",
    "        the whole final layer (our control), and plotting the difference\n",
    "        between these\n",
    "    saliency_fn\n",
    "        The function to use in order to compute gradient-based saliency\n",
    "        (right now, either `compute_saliency_of_final_layer`, `compute_weighted_saliency`, `compute_guided_saliency` or `compute_saliency_of_logits`)\n",
    "        NOTE: for `compute_saliency_of_logits`, set `show_control` to False, since the blackbox model does not have aspect-specific logits.\n",
    "    \"\"\"\n",
    "    explanator.blackbox_model.eval()\n",
    "    explanator.cpm_model.model.eval()\n",
    "\n",
    "    models = {'CPM': explanator.cpm_model.model}\n",
    "    if show_control:\n",
    "        models['control'] = explanator.blackbox_model\n",
    "\n",
    "    x = tokenize_sentence(sentence, explanator)\n",
    "    tokens = explanator.tokenizer.convert_ids_to_tokens(x['input_ids'][0])\n",
    "\n",
    "    plot_dfs = []\n",
    "    for model_name in models:\n",
    "        gradients = saliency_fn(models[model_name], x, explanator, aspect=aspect)\n",
    "        gradients = grad_apply_fn(gradients)\n",
    "        saliency = get_max(gradients, key_fn=max_key_fn)[0]\n",
    "\n",
    "        if control_for_gradient:\n",
    "            gradient_control = saliency_fn(models[model_name], x, explanator, aspect=None)\n",
    "            gradient_control = grad_apply_fn(gradient_control)\n",
    "            gradient_control = get_max(gradient_control, key_fn=max_key_fn)[0]\n",
    "            saliency = np.abs(saliency - gradient_control)\n",
    "\n",
    "        plot_df = pd.DataFrame({\n",
    "            'word': tokens,\n",
    "            'weight': saliency,\n",
    "            'model': model_name\n",
    "        })\n",
    "\n",
    "        plot_dfs.append(plot_df)\n",
    "\n",
    "    if normalize:\n",
    "        for plot_df in plot_dfs:\n",
    "            if softmax_norm:\n",
    "                plot_df['weight'] = softmax(plot_df['weight'])\n",
    "            else:\n",
    "                plot_df['weight'] = plot_df['weight'] / plot_df['weight'].sum()\n",
    "\n",
    "    plot_data = pd.concat(plot_dfs)\n",
    "    \n",
    "    display_fn(plot_data)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Gradient-Based Visualizations\n",
    "\n",
    "Graphs visualizations of which parts of the input the CPM model focuses on in representing a certain concept, such as ambiance, food, noise, or service."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [],
   "source": [
    "example_sentence = 'The music was too loud, and the decorations were tasteless, but they had friendly waiters and delicious pasta'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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Laq6dOmCwYMHY+bMmZDJZOjXrx+6deumcxkRERERWSaZcP8VUg+QkZGBd999F+np6ZDJZACqk8Nly5bB0dERixYtQnBwMIYMGWLyBpsKx2NZHnd3BwzaEa5X2b3DYjlG9RHj7u5g0H1UOUZVE2MiEZlLvceoent747vvvkNmZiZu3boFd3d3eHh4iMvj4uLq30oiIiIionvoNUa1hqenJ3x9fcUkdcaMGSZpFBERERFRnWdUIyMjMW/ePNja2qJfv361lguCALVabdLGEREREdGjq85EtXv37rCxsQEAODg4YN68eRrLBUHA4sWLTds6IiIiInpk1Zmo3nth1JIlS9C+fftaZZYsWWKaVhERERHRI0+vi6nat2+P1NRUxMfHo7KyEvPmzUN8fDxeffVVU7ePiIiIiB5Rel1M9c033+C9996DQqHA2bNnYWtri7y8PMTGxpq6fURERET0iNIrUY2Pj0d8fDwiIiJgb28PKysrTJ8+HWfOnDF1+4iIiIjoEaVXoiqTydC0aVPxdY3KykrTtIqIiIiIHnl6jVFt06YN3n//fbz44osoLy/HqVOnEB8fj8cff9zU7SMiIiKiR5ReZ1TffvttWFtb44033kBaWhpeffVVlJeX17plFRERERFRQ9HrjOobb7yBZcuWYcGCBcjPz4eLi4vGEAAiIiIiooamV6Iql8tx+PBhbNiwAY6OjggODkZgYCBsbW1N3T4iIiIiekTplaiuX78eKpUKAFBUVIT4+Hj0798fPj4++Oyzz0zaQCIiIiJqHC7N7GClsNJZ7m7FXeQVljb49vVKVFUqFc6cOYP9+/cjMTERp0+fxpNPPgl/f/8GbxARERERSYOVwgrZK4/rLKea0c0k29crUe3ZsycKCgrw3HPPISwsDMHBwXBycjJJg4iIiIiIAD0T1a+++gqJiYk4deoUjh07Brlcjp49e8Le3t7U7SMiIiKiR5ReiWrr1q0xbtw4AMDt27exbds2DBgwAI8//jg2bdpkwuYRERER0aNKr0R17NixmD17Nn777TckJibiypUrCAgIwLPPPmvi5hERERHRo0qvRPX48eOYMWMGevXqhXfeeQc9evSAjY2NqdtGRERERI8wvRLV4cOHY/HixaZuCxERERGRSK9HqDJJJSIiIiJz0ytRJSIiIiIyN726/omIiIjqy8HJFjZy/VKPsso7KC64beIWkdQxUSUiIiKzsJFbY8iW7/Uqu/ull1Fs4vaQ9JktUc3KykJ0dDTc3NyQk5ODiIgIeHt7a5QRBAFLly6FWq1GSUkJ+vXrhxEjRgAAXn/9dVy4cEEsW1paiqlTp+LNN9/EqlWr8PXXX8PKqvpZtF27dsXq1avNtWtEREREZAJmS1SjoqIwcuRIhISEIDExEZGRkbUeFrBv3z5cvXoVa9asQXl5OQYOHIiAgAB4eXnh8ccfxxdffCGWnT59OgYNGiROb9myBV5eXubaHXoEOTjZwEYu11murLISxQVlZmgRERHRw80siWp+fj4OHDiAVatWAQACAwMxbdo0ZGdnQ6VSieXi4+PRp08fAIBSqURAQAD27NmDSZMmITIyUiyXlZUFmUwGDw8Pcd7nn38OGxsbVFZW4o033tBYRtQQbORyDNm6UWe53S+ORzGYqBIREdWXWRLVrKws2NnZQalUAgAUCgUcHR2RmZmpkahmZmbC1dVVnHZ1dcX169drre+bb77BK6+8Ik77+/ujRYsWaNu2LU6cOIGwsDDs2bMHtra2erfR1dXemF0jC+Lu7vBQbosaBv/PNDEmkhTwc2lZTPH/ZXEXU1VUVCA1NRWzZs0S5/Xo0UN8/fTTT0Mul+PEiRPo2bOn3utVq0tQVSU0aFvJtAz9QOTm1m9YviHbq++2qP7M/f4wlNS/gBkTyRSk/rk0NyenppDL9btTaGVlFQoK/jZxi2ozx3eftm2YJVH18PBAaWkpysvLoVQqUVFRgaKiInh6emqU8/T0hFqtFqfVajXatGmjUeaHH37AgAEDNOZdvnwZbdu2FaflcjnKytj1SkRERNIllzfBL//O1atsv1fdTdwaaTLLDf+dnZ0RFBSE5ORkAMChQ4fg5+cHlUqFhIQEFBdXZ+ChoaFISkoCAJSXl+PYsWMYPHiwxrri4+MxbNgwjXnvv/8+KisrAQA3b95ETk4OfH19Tb1bRERERBbBpZkd3N0d9Prn0syusZsrMlvX//z58xETE4OkpCTk5ORg4cKFAICVK1ciKioK/v7+GDhwINLT0zF37lwUFxdjypQpGrewOn36NFq3bg17e82xUwEBAZg5cyY8PDxw7do1LF26FO7uj+YvDyIiIqL7WSmscHPpVb3KtpjT2sSt0Z/ZElUvLy+sXbu21vzdu3eLr2UyGcLDw+tcR6dOndCpU6da8+8dr0pEREREDwezdP0TERERERmKiSoRERERSRITVSIiIiKSJCaqRERERCRJTFSJiIiISJIs7slURERERFJiCU+YslRMVImIiIjqQS5vgi+26feEqddH8D7vhmCiSmRCDk62sJHr9zErq7yD4oLbJm5R4+JZByIiMgQTVSITspFbY8iW7/Uqu/ull1Fs4vY0Nj7XmoiIDMGLqYiIiIhIkpioEhEREZEkMVElIiIiIkniGFUiIqKHgIOTEjZyhV5lyyorUFxQbuIWEdUfE1UiIqKHgI1cgUE7wvUqu3dYLIrBRJWkj4kqEQpdzbsAACAASURBVBERGcTByQY2crleZcsqK1FcUGbiFtHDionqPVya2cBKofuDd7eiEnmF/NAREdGjyUYux5CtG/Uqu/vF8SgGvzPJOExU72GlkCN37Vc6y7lPDgP4oSMiIiIyKSaqRET3cGlmByuFlV5l71bcRV5hqYlbRET06GKiSnVq5iSHQm6js1xFZRkKCyrN0CIi07NSWCF75XG9yqpmdDNxa4iIHm1MVKlOCrkNPt/cX2e5CWN/AsBElYiIiBoWb/hPRERERJLERJWIiIiIJImJKhERERFJEhNVIiIiIpIkJqpEREREJElmu+o/KysL0dHRcHNzQ05ODiIiIuDt7a1RRhAELF26FGq1GiUlJejXrx9GjBgBANi2bRtiY2Mh/+8j21q0aIGtW7fqrEdERERElslsiWpUVBRGjhyJkJAQJCYmIjIyEps2bdIos2/fPly9ehVr1qxBeXk5Bg4ciICAAHh5eQEAVq1ahe7du9dat656RERkGCcHpV6PlAb4WGkiMh2zJKr5+fk4cOAAVq1aBQAIDAzEtGnTkJ2dDZVKJZaLj49Hnz59AABKpRIBAQHYs2cPJk2aBADYsmULEhMTUVZWhtGjR6NDhw561SMiIsPo+0hpgI+VJiLTMUuimpWVBTs7OyiVSgCAQqGAo6MjMjMzNRLVzMxMuLq6itOurq64fv06AKB9+/Zo164dfH19ce3aNYwePRrbt2+HSqXSWk9frq72BpV3d3cwqPzDzhKOhznbaOy2LOE4mhP/zywHj4Pl4eer8VjC8TBmW6Zon8U8mapz587i61atWsHHxweJiYkYNWpUg6xfrS4xKFnNzS1ukO1KmSFvuMY4HoZ+IOrbRmOOh7nbKHXmPB7GbstcbXzYvoAf9veuJeDnq/FYwvEwRxtNEQ/NctW/h4cHSktLUV5eDgCoqKhAUVERPD09Ncp5enpCrVaL02q1Wixz+fJljbJyuRxlZWU66xERERGRZTJLours7IygoCAkJycDAA4dOgQ/Pz+oVCokJCSguLg6Aw8NDUVSUhIAoLy8HMeOHcPgwYMBANHR0SgsLAQAlJaW4tSpUwgICNBZj4iIqL6aOcnh7u6g179mTvpdhEZEupmt63/+/PmIiYlBUlIScnJysHDhQgDAypUrERUVBX9/fwwcOBDp6emYO3cuiouLMWXKFPEWVr1798bcuXPRunVrZGRkYM6cOejYsSMAaK1HRERUXwq5DT7f3F+vshPG/gSg0rQNInpEmC1R9fLywtq1a2vN3717t/haJpMhPDz8gfXHjh2LsWPHPnCZtnpEREREZJn4ZCoiIiIikiQmqkREREQkSUxUiYiIiEiSmKgSERERkSQxUSUiIiIiSWKiSkRERESSxESViIiIiCSJiSoRERERSZLZbvhPREQkBU7NFJArlDrLVVaUo6CwwgwtIqK6MFElIqJHilyhxI7/G6iz3LA3fgDARJWoMbHrn4iIiIgkiYkqEREREUkSu/6JyChOTk0hl+v+rVtZWYWCgr/N0CIiInrYMFElIqPI5U3wxbZcneVeH+FuhtaQJXNupoC1Hhc3AcCdinLk8wInMiFHJzso5VY6y5VX3kVRQakZWvRoY6JKRESNylqhxPnVL+hV9olp8eAFTmRKSrkV3t6eobPcJ8O9zdAa4hhVIiIiIpIkJqpEREREJElMVImIiIhIkpioEhGRyaSmpmDBggikpqY0dlOIyALxYqpGwqtciehR8P33/8bly5dQVnYbfn7+jd0cIrIwTFQbCa9yJaJHwe3bZRp/iejh59LMFlYK/VLMuxV3tC5nokqSlJqagl27tmPo0OE8C0NERGRBrBTWyFn9o15lm097XutyJqokSewuJCIiIl5MRZLE7kIiIiLiGVUiImowLs2UsFIoxGkrK5n4193dQaPs3YoK5BWWm7V9RGRZzJaoZmVlITo6Gm5ubsjJyUFERAS8vTUfPyYIApYuXQq1Wo2SkhL069cPI0aMAACsX78eFy5cgIuLCy5duoQxY8YgODgYALBt2zbExsZCLpcDAFq0aIGtW7eaa9eIiOi/rBQKZK9dJE7fLcwT/947HwBUk+cBYKJK2jk42cJGrl+6UlZ5B8UFt03cosbl3KwprBX6dYjfqahCfuHfJm6RaZktUY2KisLIkSMREhKCxMREREZGYtOmTRpl9u3bh6tXr2LNmjUoLy/HwIEDERAQAC8vLyQnJ2Pjxo2wtrbG+fPnMXLkSBw5cgRKZfUtnlatWoXu3buba3eIiIjIDGzk1nhhi34X5sS/9DyKTdyexmataIJTn2brVbbzJJWJW2N6ZklU8/PzceDAAaxatQoAEBgYiGnTpiE7Oxsq1f8OYnx8PPr06QMAUCqVCAgIwJ49ezBp0iR88cUXaNKk+heEl5cXSktLUVxcLCaqW7ZsQWJiIsrKyjB69Gh06NDBHLtGFsjByQY2/z37rktZZSWKCzhOloiIqDGYJVHNysqCnZ2dmFQqFAo4OjoiMzNTI1HNzMyEq6urOO3q6orr168DgJikAkBiYiKee+45uLm5AQDat2+Pdu3awdfXF9euXcPo0aOxfft2jXU/ypyaKSDX8+EClRXlKHjIHy5gI5dj8LaVepXdM2IGisFElchYNtbWGn+JiAxhcZEjKysL3333HZYvXy7O69y5s/i6VatW8PHxQWJiIkaNGqX3el1d7Q1qx/0XBZhafbe34/8G6lVu2Bs/wN1dv6T2Xg19PLRdgGEsY9djTD1zbssSWMLxsIQ2Slldx+HlTo9hz19XMbh9a4PqGbMtU3iY3xcP63GsqVNxtwoKK/3GchpS9kHbknI9S9iWNmZJVD08PFBaWory8nIolUpUVFSgqKgInp6eGuU8PT2hVqvFabVajTZt2ojTmZmZWLRoEZYtWwZnZ2dx/uXLl9G2bVtxWi6Xo6zMsLNganWJQclqbm79RsEY+p9Zn+0Zuy1D6tX3eNzv7l1B/FvXus2xX8bWq++2LIHUj4e53x+GsoRExhB1Hb+uLd3QtaWb1nrm/pxI/b1rrIf181Wfbb249bhedba+2K1RYr25j4ep6xm7LW3Mch9VZ2dnBAUFITk5GQBw6NAh+Pn5QaVSISEhAcXF1TsWGhqKpKQkAEB5eTmOHTuGwYMHAwCuXbuGxYsXIyYmBq6urti7dy9SU1MBANHR0SgsLAQAlJaW4tSpUwgICDDHrj1QamoKFiyIQGpqSqO1QUp4PIiIqjEeEhnGbF3/8+fPR0xMDJKSkpCTk4OFCxcCAFauXImoqCj4+/tj4MCBSE9Px9y5c1FcXIwpU6aIt7B68803kZ+fLyauZWVl+Ne//gUA6N27N+bOnYvWrVsjIyMDc+bMQceOHc21a7XwqUqaeDyIiKoxHhIZxmyJqpeXF9auXVtr/u7du8XXMpkM4eHhD6z/008/1bnusWPHYuzYsfVvZAPhU5U06XM8HJ0UUMr/NzZW2xjV8spyFBU83Bd8GcPByQ42ciu9ypZV3kVxQamJW0RE9zP2+yE1NQW7dm3H0KHDmeDSI8XiLqaih5NSrsT47QPE6eySyv/+zdSYDwAbh+8DwET1fjZyK4PGY1nOaFgiy9XMSQ6F3Eac1vYjvKKyDIUFlQ9cjynPxOp7yz7ero8aAxNVIiIiE1HIbRD3zfPidFH5nf/+zdSYDwBzR/8I4MGJqil76vS9ZR9v10eNgYkqERFZJOdmCljreY/oOxXlyJfAPaIff7oJrqQLaNNF1thNIbIITFSJiMgiWSuUSP5siF5lg9/aDSkMGXJv1QTurRq7FUSWg4kqkQQ5ONnCRq7fx7Os8g6KC26buEVEZC73X1wK1D22lReX0sOOiWo9uTRTwkqh0JhXV0C5W1GBvMJys7bP3O6/cACo+3hou3DgUWcjt8YLW37Uq2z8S8/zwigJcGlmCyuF7pB6t+IO8gr5w4Lqdv/FpUDdF5jy4lJ62DFRrScrhQLZaxdpzLtbmCf+vXeZavI8AA93onr/hQMAkF98579/NS8e0HbhAJGlsVJYI2e17h8Xzac9r7MMERFVM8uTqYiIiIiIDMVElYiIiIgkiYkqSZJMofmXiOhRxphIjyqOUSVJcupuhaKTVXDsyt9SpuboZAelno9eLa+8iyI+epXI7BgT6VHFRJUkybZNE9i2YUA2B6XcCm9vz9Cr7CfDvU3cmgdzbtYU1gr93g93KqqQX/i3iVtEZF6MifSoYqJqQSzxKSwAYKXQ/EtkKGtFE5z6NFuvsp0nqUzcGiIiMhcmqiZgY22t8behWOJTWAA+MpCIiIiMw0TVBF7u9Bj2/HUVg9u3buymSAIfGUhERETGYKJqAl1buqFrS7fGbgYRERGRRePIbCIiIiKSJCaqRERERCRJTFSJiIiISJKYqBIRERGRJDFRJSIiIiJJYqJKRERERJLERJWIiIiIJImJKhERERFJEhNVIiIiIpIksz2ZKisrC9HR0XBzc0NOTg4iIiLg7e2tUUYQBCxduhRqtRolJSXo168fRowYUa9lRERERGSZzJaoRkVFYeTIkQgJCUFiYiIiIyOxadMmjTL79u3D1atXsWbNGpSXl2PgwIEICAiAl5eX0cuIiIiIyDKZpes/Pz8fBw4cQHBwMAAgMDAQKSkpyM7O1igXHx+PXr16AQCUSiUCAgKwZ8+eei0jIiIiIstkljOqWVlZsLOzg1KpBAAoFAo4OjoiMzMTKpVKLJeZmQlXV1dx2tXVFdevX6/XMn01aSKr/uvQ1KDy1XWaGbwdALB2aG5wPaW94XUAwM7IevZNVVpKPriOo5517q/namd4veZ2zkZtq7mdo5H17I2oY2fktmwMrudupzBqWy52VkbVa2qn32/de+vYNNX/9/G99eT2htezcjRuv5o4Gnccmzjo9392bx0p0zceAvWPicbEQ8D0MdGYeHh/PWNiojHxEDB9TDQmHtauZ3hMNCYeAqaPicbEw/vrGRMTjYmHgOljojHxUBeZIAhCg6xJi9OnT+P1119HSkqKOC8wMBCrV6+Gn5+fOG/o0KF45513EBISAgBYunQpioqKsHDhQqOXEREREZFlMkvXv4eHB0pLS1FeXg4AqKioQFFRETw9PTXKeXp6Qq1Wi9NqtVosY+wyIiIiIrJMZklUnZ2dERQUhOTkZADAoUOH4OfnB5VKhYSEBBQXFwMAQkNDkZSUBAAoLy/HsWPHMHjw4HotIyIiIiLLZJaufwC4fv06YmJi4O7ujpycHLz//vto3bo1hgwZgqioKPj7+0MQBMTFxSE/Px/FxcXo27cvXnrpJQAwehkRERERWSazJapERERERIbgk6mIiIiISJKYqBIRERGRJDFRJSIiIiJJYqJKRERERJLERJWIiIiIJImJKhERERFJEhNVMquEhASUlJQ0djMsUnZ2NsrKynSW06dMQ7l586bZtkX0sGE8rB/GxEcDE9UGVFJSgvPnz0MQBPFxsbpcvHgRiYmJ4nRSUhIMubVtVVWV2QPd+vXrja67Y8cOjBo1CsePH9e7zrlz57Qu37VrF/7zn/8AAObPn48RI0bg4MGDeq+/sLBQ77LGKi8vx99//w0AuHr1Kn799VfcuXNHa51FixZpTF+6dAmzZ8/Wua1p06YhPz/f+MbeQ9exnzFjBs6fP2/weqdNm4bvv//eqPakpqYCAPbu3YvY2FhkZWVprVOf90dVVRWys7ORlZWFrKwsvP/++wa3+VHVGPEQsKyYaIp4CNTvPW+OeAhYZkzU59gbExONjYc1bTJXTGyseGhtlq1YEB8fH8hkMq1lfH198c0332jM279/P+bNm4fWrVtj48aNePPNNzFp0iT07NlT67oWL16Mfv36idM3btxAXFwcwsPD66wTFxcHZ2dnjBs3Di+++CJyc3MxceJEvPHGG1q3lZ+fjwULFoiPsu3Vqxc+/PBDODs71yrbt2/fOo9DYWEhJk6cqHVbdVm9ejUqKipw6NAhjfk7duyos87OnTvxf//3f3UuT0pKwvz583H48GGkp6dj8eLFWL9+PYKCgrS2JT09HTNmzICrqys2b96Mt956C++//z46depUZ50rV67ggw8+gCAI+OyzzzBz5kxERkbCy8tL67beffddBAYGYsCAARg1ahT8/f3x66+/Ijo6ulbZmiBTXFyMGzduiF/UurZRw8rKCmvWrMGdO3cwfPhw+Pr6ai1fn2P/2GOP4ddff8Wnn36Kvn37on///rC21h1WMjIy8OKLL+osd781a9Zg3LhxuHDhAhYtWoSJEydiyZIlWLlyZZ11jH1/JCQkIDo6GiUlJXB0dERBQQEUCoXWOvoE7latWmHy5Mk6y0mBJcRDwLiYaEg8BEwTE00RDwHj3vPGxEPg4YuJ9T32xsREY+MhYL6Y2JjxkInqfUaNGoUFCxZoLXP/rzoA2LNnD3766ScsWbIESqUSX3zxBebPn68zMD/xxBN45ZVXNLb/oPXfq6ysDBMnTsTu3bvh7e2NXbt2ISIiQmsdoPpLoEePHpg6dSoAIDU1FYsXL8aSJUtqlfXz88PMmTOxY8cONG/eHP7+/gCAlJQUo86g3UuhUODZZ5/VmLdu3Tp07doV+fn5uHDhghhI0tLS4OnpqXV9Hh4esLe3x48//oiwsDD4+PhApVLpbMcXX3yBTZs24fPPP4etrS02bNiAjz76SOvxX7VqFaZOnYqdO3fCzs4O0dHRWLFihc7/s1atWuHVV1/F5s2bMWTIEHzwwQeIiYl5YNmwsDAA1V9+R48eFefb2Nhg8ODBOvfrk08+gVKpRHFxMbZt24aPP/4YgwYNwtChQ6FUKmuVr8+xr9nvqqoq/PTTTxg2bBiee+45jB49Wuv/QUBAAPLz8+Hq6irOW7FiBWbOnKl1e4899hiefvppfPLJJxg7dizGjh2LGzduaK1j7PsjISEBCQkJWLJkCebNm4fbt2/jk08+0VonMzMTw4cP11rm/qREyiwhHgLGxURD4iFgupjY0PEQMO49b0w8BB6+mFjfY29MTDQ2HgLmi4mNGQ+ZqN7nueee01mmT58+tea1bNkSTZs2FaebNGkCW1tbnet6UDdLUVGR1jo16/3hhx8watQoAICjo6PObbm5uYnlgeovhUuXLj2wbExMDJRKJW7evCkGcgBo27YtPvroI53bMtSMGTMwYMAALFiwAKtWrYJcLgcAVFZWYuHChVrrXr58GZ9//jl+/vlnzJ49G8XFxbh+/brObXp5eaFVq1bitI2Njc7j6OXlhR49emDfvn0AAHd3dzRr1kzntm7fvg1BELBz505ERUUBQJ1dmr/++isAYPPmzRg7dqzOdd8vJSUF3bp1w88//4z4+HiUlpZCrVYjKioKXbt2xciRIzXK1+fY79mzB926dcO3336L7777Dk888QTatWuHTZs2wcHBAVOmTHlgvTNnzuD555/H448/DoVCAUEQcO3aNZ2BOTMzE3/88Qd27NiBr776CgBQUFCgtY6x74/mzZvD2tpa7I60tbVFaWmp1joBAQE6A3Nubq7ObUuFJcRDwLiYaEg8BMwbE+vzmQSMe88bEw9r6j1MMbG+x96YmGhsPATMFxMbMx4yUb2Prl/8ANCjR49a83JycpCamoqqqirk5eXhwIEDeg2q9vHxwYsvvoiuXbsCqP5VP2zYMK11cnNzMWnSJFy4cAE9e/bE4cOHcfHiRZ3bysnJwZ07d8RuiMrKyjrfJDW/Mi9evAi1Wi3+0rt165Ze43QMNWDAAADV3XE1gQEA5HK5zvFmM2bMwL///W8sWrQI9vb2WL16NXr37q1zm9nZ2cjOzha781JSUnDt2jWtdXJyclBWVibWycrKwpUrV3Ruy8nJCf7+/ujSpQs6d+6MDRs26AwMDwrIP//8s87k4aOPPkJJSQm6du2K2bNnIzAwUFw2Z86cWolqfY79okWLIAgChg4dii+//BJt2rQBAAwZMkTr2LHbt2/jX//6lzgtCAK+/PJLrdsCgOeffx4REREYMWIEPDw8sGjRIp0J0IPeH7169dK5rfPnz+PcuXNQKpWIjo6Gk5MTTp8+rbXOtGnTdK7X2GEzjcES4iFgXEw0JB4C5o2J9flMAsbFRGPiIfDwxcT6HntjYqKx8RAwX0xszHgoEwwdqf6QO3r0KM6cOYORI0eiadOm2LVrFzZs2ICWLVvigw8+qHNMTFZWFubMmYMTJ05AJpPBz88PcXFxeo2hOXTokMY4qQcF/nuVlZUhOTkZnTt3RsuWLZGYmAgXFxd06dJFa729e/ciLi4OHTt2BFA9CPu9997DoEGD6qyzb98+fPjhh2K3QHZ2NqKjo9G/f3+d+2WM6dOnQ6VSISAgAABw7Ngx5OTk6OxiuF9SUpLOD965c+fw9ttvi8HZzc0Nq1evho+PT511jhw5gnnz5qG8vBxOTk5Qq9VYuXIlnnnmGZ1tKiwshKOjI2QyGXJzc6FUKrWesXjQ+J709HTs2bNH63bGjh2LuLg4tGzZUmN+RkYGli9fjhUrVjywnjHHfsqUKVi+fDlsbGxqbWvz5s11dr9mZGTA29tbY15+fn6d4wMb2l9//YX27dtrLXPhwgVYW1vD1dUVS5cuRWFhISZPnqz1/ZGTk4OYmBhcvnwZAQEBmD17dq1jY0ksIR4CxsVEY+IhYN6Y2FDxENAdE42Jh8DDGxONPfbGxMTGjoeA7pjYmPGQiep9JkyYgP79++PFF19EXl4enn/+efHDkZycjFWrVmmtX3MVY9OmTVFUVKRX18n9vv/+e7z88ssG1dm1axeGDh2qs9zFixdx+PBhyGQy9OjRA4899pjOOmq1GmlpaQCAp556Ci4uLga1zRAlJSVYs2aNOA6pe/fumDp1Kuzt7eusk5GRgS1btuDWrVuoqqoCoF/wOnfuHJycnFBcXAygugtPnwuBCgoK8PvvvwOoPh5OTk4665w7dw6lpaXw8/PD3r17kZ6ejrFjx8LDw6POOuPHj0doaCgA4M6dOzh79iwEQcD8+fO1bqvmTMCNGzfw+OOPo6Ki4oFjU+9nzLF/EH3ev7dv38Znn32GoqIizJ49G1988QXGjx+vc3C+vuO27vWgiyP0uSji1KlT6Ny5szj9yy+/QKlUaj3LOGnSJHh7e6Ndu3Y4fPgw2rZta3B7pcRS4yGgX0w0Jh4C5ouJxn4mjYmJxsZD4OGMiQ0VDwHd72Fj4yFgvpjYmPGQXf/3ad68uThu6YcffkBQUJDYLZCenq617pUrV3Dr1i1xnM3mzZvrDOQrVqzA+PHj8fbbb2vMrxmbou1NnZubizVr1uDq1au4e/cugOrbe+iTqDo7O6N58+YAoFcwAQBXV1f07dtXnDb2i0Mf9vb2mDt3rsa89PR0rWdGpk2bhmeffRZdu3aFlZUVAP3uZffqq6/in//85wPH2NXl66+/xmuvvSZe/LBt2zacOXMGH3zwgdZ6q1evxvjx4w26MnPRokW1zgAsW7ZMZxvT09Mxb948tGrVCps2bdL7iusHHftr1649MDDXNU5Mn/cvUH0hi4uLC3Jzc2FjY4N27dohLi4OkZGRDyy/ZcsWBAcH4+effxYD3aBBg7B3716t2wGqbx1Uc0FEZWUl/vzzT70uitixY4dGYO7WrZvOC4Ls7e3F98Lo0aP1unWOlFlCPASMj4nGxEPAfDHRmHgIGBcTjYmHwMMbEw2Jh0D9YqKh8RAwf0xszHjIRPU+947tSEpKEn+9AdB6yjo8PBxHjhyBp6cnmjSpvj2ttvE9KpUKcrkcjo6OGm9wfcamLFmyBP369UNeXh7CwsKQlZUldpVpc+jQIbz33nto0aIFBEHARx99hCVLlmiM17nfgz58V69eNVmiWlRUhH379mmcCUhKSsJ3331XZx13d/dav9L06XZ65plnagXl1NRU+Pn51Vnn8uXLGtMjRowQ72GnTbt27Qy+MlMQBPG2LFVVVcjNzcXJkyd1bsvYK66B6oB+5coV8djX9Su7adOmGD9+/APbrM/YKgcHB8yYMUM8ExISEoITJ07UWf7atWuYMmUKbty4gblz56JTp05635szMjJSo/tYEAStX4arV68GUH0sal4D1f8Hur7sHRwcxNcymUxjes2aNRoX4VgCS4iHgHEx0Zh4CJg3JhoTDwHjYqIx8RB4uGOivvEQqF9MNDQeAuaLiVKIh0xU76NWq5GdnY2MjAykpqbi448/BlB9al7bQO/z58/jt99+07jPXs1VkA/y6quvAqh+w9x/W4iawdd1admyJQYMGIATJ06I42f++usvrXUAYOvWrdi7d6/Y/VZYWIioqCitgfneL46arhZT3hB64sSJ8PT0RKtWrcQzAbpGpwQEBGD79u3o2rWr2FWydu1anVdo/r//9/+wZMkSBAUFifU2b978wMBccw/FwsJC/Pbbb+L8qqoqPPHEEzr3y5grM4cOHSqe5akZM/bmm2/q3JaxV1yvWrUKp06dQmZmJp588klkZWWJ3YD3+/DDD2ud2ajRunVrnduqCag1nxddQW/GjBmYNWsWQkJCMHjwYJw5cwb5+fno27cvOnbsiDVr1tRZ9/4xjlVVVXp9Xu5/39nY2ODDDz/UWufQoUN49913xelTp06J06dPn7a4RNUS4iFgXEw0Jh4C5o2JxsRDwLiYaEg8BB7+mGhIPATqFxMNjYeA+WNiY8ZDJqr3GT9+PF577TWUlJQgIiIC9vb2OHv2rM5fX76+vvj77781ugX0CSgXL15Eeno6nnvuOWzYsAFpaWmYMmWK1nua1VyZWlJSgrNnz6JZs2Z6/YJt2bKlxhixZs2aoUWLFlrrLF26VOMDHRgYiLi4OJ3bMpZSqcQ///lPjXkhISFa66xZs0bj/nNA9ZkIXYnq5s2b0aFDB/zxxx/ivLrO+nz55ZcQBAGrV6/G9OnTNdrr5uamdTtA9ZWkERERGD58uN5XZk6f/YloYQAAH7FJREFUPh1BQUE4cuQIZDIZunfvrtcXgLFXXBcWFuLTTz/FokWLMG/ePACo8//6+vXrdSYq2rp4a7Ru3Rrjx49HQUEBFixYgKNHj2L06NF1lu/bty+efPJJ3LlzB82bN0dgYCB27tyJPXv2iGMFtdW99wuguLgYr732Wp3la65U7dWrl84u1vs5ODigbdu24vS9r/W5glpqLCEeAsbFRGPiIWDemGhMPASMi4mGxEPg4Y+JhsRDoH4x0dB4CJgvJkohHjJRvc9TTz2FhIQEjXkdO3ass6vl3l/W/fr1Q9u2bTXugzZw4ECt2/vuu+8wc+ZMpKen49tvv0V4eDg+/fRTrV2Tjz32GH788UeMGjUKYWFhuH37tl5jP27cuIFff/1V/IWcmpqq88N66tQp8bUhXS3G6t+/P44fP46nnnpKvDVIYmKi1isLhw0bVuum5Js2bdK5rTFjxmgEWKDup5LUjN+JjY0V51VVVem8j1yNkJAQjS+YmsCnjbW1NcaPHy+OC1q3bh0mT56sNckCqoN5zRXXW7ZsEa+41qXmLErNBTAA6uyKmzVrFtq2bYuKigqcP39e/LI4f/68xr0Y6zJmzBg8/vjjSEpKAlC7K+p+v/zyC06ePInJkydj2bJluHDhAgoKCrBy5UqdZ8ACAgLE/2eZTAYXFxe9rjwtKSkRb31zb9JUc5X4g0ydOlVj7OK97h3fZSksIR4CxsVEY+IhYN6YaEw8BIyLiYbEQ+Dhj4mGxEOgfjHR0HgImD8mNmo8FEgv4eHhwiuvvFJr/j/+8Q/h6NGjtf7Fx8cL06ZN07ne5cuXC4IgCLGxscKXX34pCIIgxMXFaa0TGhoq7NmzRyguLhYqKiqE4uJivfbh+vXrwujRo4UOHToIHTp0EPr37y9cuHBBa53g4GAhLCxMCAsLE8aMGSPMmjVLOHnypF7bM0ZN2+795+Pjo1fd/Px8IT8/36DtFRcXC3/99ZdQVVUllJWV6SwfFxcnfPrpp0J5ebkwZMgQoXv37sLnn3+us96tW7eE9957T3jnnXeEv//+W4iMjBQKCgq01gkLCxNu374tTpeWlgphYWG6d+q/SkpKhJKSEkEQ/n979x7U1JXHAfwLVFcrjJWZyrajdac+8F1UGo0IuogOoHaoj4JdseqOgK4KiihSFrEKWNxqZ32Cu6tWUVpbUYH6wFVARAUto6ItrqKArfjgWRYBgbN/sLklIclNbkhubvx9ZpwhNzncE7x+Oeaec36M1dTU8L4+MDCQnT17lu3bt4/5+Pgwf39/tmjRIrWvPXLkCGOMsdjYWKX3UVNTw6Kjo3nPlZOTo/Q4JSWFfffdd7ztPD09ua89PDzY0aNHWWhoqNY2ra2tjDH9r4/g4GD28OFDduPGDebh4cHOnTvHgoODdW5vycwpDxkTlolC8pAx02aiIXnImP7XvL55yJjlZqI+eciYYZkoNA8ZM10mipmHNFDVw2effdbhWHl5OWOMsZiYGKXjubm5LDAwkPd7hoSEsFOnTjEXFxdWUVHBWlpa2KpVq7S22bNnD8vNzWVxcXEsMjKSff311+z58+e851L0UfGP9dKlS+wvf/mL1jYpKSm837czLV26tMMxvp9HWVkZmz17Nhfic+bMYaWlpbznyszMZOPHj2d+fn6soaGBzZs3j128eFFrG0XgpKamsiVLljDGGIuIiOA915o1a9g333zDVq9ezRhj7Pbt2+zTTz/V2uavf/1rh2PqrkF1Hjx4wPLz81leXh7Ly8vTaZDQ3NzMWlpaGGOMpaWlsUOHDrGvvvpKaxt1gbhmzRrec23cuLHDsbCwMN52165d477++OOPeV/PmPDrQ8igyc/Pj3355Zcdjm/cuJF5e3vr1F+pMJc8ZExYJgrJQ8ZMm4lC8pAxYde8kDxkzHIzUUgeMiYsE4XmIWOmy0Qx85Bu/euh/cIAhZaWFm6S9ePHj7l5WH369FGqaqGJv78/9u7di+DgYNjb2+Pzzz/HgAEDtLYJDAwEAAwbNgznz5/H7t27ERMTo3FeimKVpGofVTcYVsfHxwf/+c9/kJOTAwBwdXXl7Z8hdu7c2WGlZVVVldY227ZtQ3BwMHcL7/r169i2bRu2bt2qtZ2QlaBCy9c6ODhgzpw53G3DoUOHKq2AbC8/P5/7vtu3b+feV0FBAV6+fMl7Ln1XXCtkZ2cjISEBFRUVaG1tBWMMtbW18Pf319imqqoK+/fvx/vvvw+gbVPs58+fa3y9Yg/OmzdvKm3e3draqlMf218LSUlJvK8HhF8fpaWlOH36NFJTU3Hy5Em0trbiyZMnWtsMGjQIwcHBWLduHe7fvw9PT08sWrQIkZGROpVflBJzyUNAv0w0JA8B02aikDwEhF3zQlfGW2omCslDQL9MNDQPFedTMGYmipmHNFBV8d1332HWrFk6v37evHkA2iZeKzYGBtpWxE2bNo23/ejRo7F7927u8dq1a/H06VOtbQ4ePIhz586hqKgILi4uCA4O1lpxxJA+pqam4ssvv8SwYcMAtP1DCAkJwfTp03nfmxA7duzArVu3dF5pCQC///3vlcLU1dUVly5d4j2XkJWgQsvXKlazKn6519fXo6ysTO1rFXOdFL9A8/LyuOd0CS99V1wrKPbte+edd2BtbQ3GGBITE7W2iY2NRUxMDHcNjxs3DrGxsRpfr1iRXV5ezn0NtC3AGDNmjMZ2isotW7du5ea1hYeHK82P00To9aFu0MS3cEPxM4+Li8OqVauwaNGiDs9JiRTyENAvEw3toykzUUgeAsKueaG7hVhqJgrJQ0C/TBSah4DpM1HMPKSBqoqEhATk5uaqfU5dXdvz588DaFvVp2nDX20U/1Nsj2+FYGlpKerq6rB8+XJ4eXnxVkUxpI8XL17E2bNnua1RXr58ifDwcKMNVKurq/VaaQm0fULSvtxcZWUl7358gLCVoBs3buRKNdrY2KCxsVGnesZyuRzTp09HU1MTAgICUFhYqHFD7CVLlnDb9ahKTk7mPZfQFdcDBw6Ei4uL0rGgoCCtbRwcHDqUFNT2i+PDDz8E0DYgUd2y5e7duxpXd4eFhaGmpgbV1dU4duwYhg4dyruyVUHo9aFu0KTPSlUpDkxVSSEPAf0y0dA+mjITheQhIOyaF7pbiKVmopA8BPTLRKF5CJg+E8XMQxqoqlDdTqE9bX8pQgIPAEJDQ7kLtLm5GcXFxRrPr6CoFVxYWIhDhw7h0aNHGDJkiNrNhg3t45tvvskFMgB06dIFb775pt7fR1f6rrQEgI8++gjTpk3jtmOprKzssKWLOjNnzsQXX3yB69ev4+jRoxgzZgzvL4Fu3bphypQp3ONJkyYhNTWVd9sOLy8vDBo0CFeuXAHQ9j9fTeUaNQUyAK1blghdca1Y2fvWW29h3bp1Snsv8pXVa2lpwZUrV5QqEGlrc+/ePQwYMAAFBQUdVkpra5ecnIza2lp4e3ujuLgYaWlpePToEYKCgiCXy/HJJ59o7KPQ6wPQb8NvQHnfwIKCAqU9BNUN7MydFPIQEJaJQvtoykwUkoeAsGteSB4ClpeJhuQhoF8mCs1DQJxMFCsPaaCqwtTby0RERMDT05N73NjYiISEBK1tDhw4gClTpuCHH37A9evXddq8XKjnz5/j4MGDSlu46DJHSqji4mJkZGRg4MCBmDlzJmxtbXnntsnlcqSlpeHGjRuwsrLSudZ0TEwMAgICkJCQACsrK6XbXpoYUr62f//+6N+/P/e4s8suaqqMUl5ejoyMDI3t9uzZg1GjRnGP2wcm3xykwMBA1NXVKW1Irq1NVFQUvvjiC6USfrqcKygoCC4uLrC1teW2HfL09MSyZcs0fuKnIPT60HfDb0B5YKc6wJLiPqpSyEPAcjNRSB4Cwq55IXkIWF4mGpKHgH6ZKDQPAdNnoqh5qPOyq1dYc3OzSc/Ht6rzvffeYxMnTmQbNmxgly5dYi9fvjRaXyorK1loaCiTyWRs3LhxLCwsjP3yyy9GO19zczP381astNR1+632tmzZwvsadSuFnz17prXN6tWr2alTp9jy5cvZ1atXWUpKita/L39/f7V/5s2bx9zc3PR+X9oIXXF96tQpjc9duHBB6zn9/Pw6HMvNzdXahjHGzpw5o9MxhRs3brCdO3eyYcOGMTc3N7ZgwQI2duxYdunSJdbY2Mh7PlW6XB+Klbjtf56bN2/W2ubf//63xuf4fpZSYW55yJjlZmJn5SFj/Ne8kDxkzPIy0ZA8ZExYJuqbh4yZPhPFzEP6RFVFUlIS9uzZg+joaEyePBlA28fz69evx65du3jng+pLta7106dP4eTkxNtm1apVndoPVe3nivn6+uKjjz7iHsfGxvLOGROq/S01voUN/v7+iI+Pxx//+Eel+S+MMVhZWfFu+K3v7gmA/qUaDan/rC+hK67bf4KlatKkSVrPKZfLUVpaqrSh9cOHD3k3q546dWqHY9rqVI8cORIjR45Eamoq0tPTcf/+fSxcuBDHjx/H+vXr1X46Yuj1IeS2q7pPH2tra2FnZ8f7szRHUshDRTtLzER98hAw7JoXkoeA5WWiIXkICMtEffMQMH0mipmHNFBVceHCBRw4cEBproyjoyNCQkLwt7/9TeuKZiHa141W1C7mm3Rs7EBWnKP9KkugrX/s/3N7zMGnn34KBwcH/PnPf0ZYWJjSc1u2bOFtr+/uCYD+pRq11X/WpYKTPgxdzSzE8ePHkZiYiF69enFzv2prazF37lyt7YTeLvT19YW1tTUGDhwIOzs7xMfHa3ytodeHkNuuSUlJyMnJQWxsLHr16oXo6GgkJyfDzs4Of//733kH8OZGCnkIUCYqGHLNC8lDgDJRlZBMNGT6hKkyUdQ81Pmz11eEtg2H+TYjFqKyspLl5uay48ePs5SUFJaSksIWLlzY6efRV1JSksbnFBU4zEVZWZnS4/z8fPbjjz/yttu0aRObOXMmO3ToEKuoqNDpXPv372enT59mBQUFbPTo0WzIkCFsx44dvO0UmyWbwoEDB0x2rrlz57JHjx5xf8rKytjatWt52+l7u1Ad1b93Xel667q5uZndvXuXffXVV2z79u063XZdsGABV93oxx9/ZO+//z578OABKy4uZgEBAYL6KybKw99YeiYKyUPGKBNVCcnEzshDxoybiWLmIX2iqoJp2bLCGNvNhIeH49dff8U777zDfX9dJmwbm9BVlmLYv3+/0rYmDg4OiI+P570VJ2Sl8M8//8ytprxy5Qqys7ORkpKi8fXffvstXF1dkZGRgZUrVwIAvL298f333+v8/vQldDWzEP/85z877LUYFxfH207f24Xq9OnTR6fXRUZG4tmzZ9yinNDQUMyYMYO7la1JcnIydu/ezS0aUmxMrq2ueN++fbnFIRkZGfD09MQf/vAHAND4KZI5ozz8jaVnotDdZCgTlQnJxM7IQ8C4mShmHtJAVUVDQwPq6uqU9lsD2m5rNDQ0dPr5amtrceTIEaVjWVlZnX4eS6TYRqS4uJj7Gmibl6TYTFobfVYKa6pmM2jQIK2/sEtLS7F06VI8fvwYa9euxbBhw3jnHklJfX09oqOj0dDQgLi4OGzevBmhoaHo2bOn1nb63i40RHNzs9LK8W3btiEyMpJ3oHr69GmcO3cO3bp1AwC8ePECAQEBWoO5/bVw/vx57hcxoDzfUCooD6XFkEzUd+cEykT1hGSiKfMQEJaJYuYhDVRVzJ07F/Pnz8f8+fO5/wncv38fhw4dMkoJRJlM1mHitbnMdzJ3ijlH5eXlSvOPunbtioCAAN7227Ztw759++Du7o7AwEDIZDK89pr6fxJC5zqFhIRg1apV8PDwwLRp03Dnzh1UVVXB3d0dQ4YMwc6dO3V6r+YqPj4ezs7OyMvLw+uvvw4/Pz9s2bIFmzZt0tru3XffxZkzZ+Dr64t58+bhxYsXvIubhFJd8GNlZaWxVGN7gwYN4kIZaCsVqdhi5cmTJ2o3466vr8fJkyfx6NEjPH/+nNsw/NatWygpKTHkbYiC8lBaDMlEffIQoEzUREgmmjIPAWGZKGYe0kBVhbOzMyIiIrB9+3bcvHkTAODk5ITIyEgMGTKk0883YsQI+Pj4oEePHkoTr/nqCZPfbqdkZWVh4sSJerfXZ6Ww0Go27u7uGDFiBJqbm9G7d2+MHz8eJ0+eRHp6us5VRMyZPvW620tLS0NgYCAGDBiAK1euoLGxscOndp2loqJC0L6XTU1NHeqK29nZIT8/H/v27cOuXbs6tFm7di22bt2Kuro67Nq1CzY2Nvjpp59w+PBhzJw5s3PfmAlQHkqLIZmo784JlInqCclEU+YhICwTxcxDK6ZtEhIxOi8vL0RFRaFv377cCtIdO3boVK+XaLZjxw6dyvgZW0tLCwoKCrBkyRK89957uHfvHqqrq/HJJ59g/PjxGDt2rNhdNEhUVBQ+++wzREdHIzo6GvX19VizZg127NihtV1CQgJGjhyJrKws/Pe//8WIESMwefJkrlJKZ6qqqkJMTAwuXrwIKysruLm5Yd26dVz5QE3c3Nw6lDVUKCkpQXZ2dqf39VVHeWg8lImmISQTTZmHgLBMFDUPdV529YrIycnhfc3ly5c77XwrVqzocKykpKTTvv+r4Nq1a8zLy4sNHTqUDR48mDk6OrLBgweL3S0lnp6e3NceHh7s6NGjLDQ0VMQedY7vv/+eTZs2jU2ZMoUtXryYyeVylp6ernP7mpoalpKSwqZOncpGjhxpxJ4qKyoq4n2NtpXcmlaAmzo/jI3yUJooE8VjSCaKlYeM8WeimHlIt/5VZGRkcPMoNLlw4QLGjRvXKed7++23BdUTJr85cuQIDh06hD179iAiIgK//PILDh8+LHa3lLSfn9S7d2/Mnj0bs2fPFrFHnUOfet3tCd2zUR/37t1D//79ceLEiQ7P6fJvTNtKbk0rwE2dH8ZGeShNlIniEZKJpshDwLBMFDMPaaCqIjk5GV9//bXW16jW5DVEWloaJkyYoHc9YfKbt99+G/b29mhtbeUeG2NFsiHaz/9JSkoSsSedT7Ve9969e7F48WKtbUpLS1FXV4fly5fDy8ur0yscAYbV0RbK1PlhbJSH0kSZKC59M9EUeQiYPhM7Kz9ooKrip59+Mun5li5d2qFiRWZmpkn7IHUPHjxAeXk5WlpasH//frzxxhtG3dpDH2VlZejbty+2bt0KDw8PAG3/w5b6nLs1a9YgLCwMvr6+HUrx1dbW8g5Uhe7ZqA/FJ0ghISEdShSePXu2087Tnqnzw9goD6WJMtH0DMlEU+QhYPpM7Kz8oIGqyNSVVZNiTXAxzZs3D0+fPkVQUBAiIiJQU1Nj1K099BEWFoaamhpUV1fj2LFjGDp0qEWsbPX29kbPnj0hk8mwfPly7jj7/+IXPvru2WgIdXW0a2pqjHY+IhzlYeegTDQ9QzLRlHkISC8TadU/kby4uDg4OTnBy8tL7K6oVVtbC29vb/j4+ODOnTvIz8+Hi4sL5HI5V9FFqhITEzFixAilms2NjY343e9+p7Wdk5MT3njjDbi7u8PDw4N3z0YhNG2Zw/5fm502kieWijJRPEIy0RR5CEg3E+kTVSJ5mZmZWLp0qdjdUCsoKAguLi6wtbXlPtHw9PTEsmXLkJubK3LvDHf48GHs379f6RjfIBXQf89GIXr06KH21hljDAcPHjTquQkRE2WieIRkoinyEJBuJtJAlUjemDFjlCpmAG21rhcsWCBOh9pZunQpcnJy8OjRI0ycOBHvvvsuqqurUVtbaxb9M9SYMWOUqggBQEpKCj788EOt7UwRylFRURrrSav2mRBLQpkoHiGZaIo8BKSbiXTrn0jeihUrcPv2bTg5OXFb2ty8eRPp6eki9+w3Xl5eSE9Px/3797Fw4UKMHz8eBQUFyMjIELtrBomKisLdu3cxduxYdOnSBQCQnZ2Nb775RuSe/aaiogLx8fFobGxEbGysTrW3CZEyykTxUCZ2PmuxO0CIoYqLi7Fs2TJMmDABMpkMMpkMvXv3FrtbSnx9fWFtbY2BAwfCzs4O8fHxkg7k6Oho1NXVITc3FxMmTOAC2Rwpam936dJFqfY2IZaKMtH0KBONh279E8nbuHEjRo0apXTMyclJpN6o1/6W1t69e8XrSCfp2rUrbG1twRjrUJbR2tq8/v8rpPY2IVJGmWh6lInGQwNVInmjRo1CdnY2V2vYzc3NKFU9OkufPn3E7oLBHjx4gG+//Rb9+vXD8ePHlZ67du2aSL1Sr7q6GgC4vQ3r6+tRWloqZpcIMSrKRNOjTDQeGqgSyfvXv/6Fs2fPcp8g7Nq1i5v3RIwjMDAQJ06cQHl5Oa5evar0nLlVEpLL5Zg+fTqampoQEBCAwsJCREZGit0tQoyGMtH0KBONhwaqRPKKi4uRnJysdGzdunUi9ebV4OzsDGdnZ5w9e9ZkVZ+E8vLygqOjIy5fvgygrQpOfn6+yL0ixHgoE02PMtF4aKBKJM/W1rbDsR49eojQk1ePugon6o6JQdPm1mfOnEFJSQl8fX1N3CNCTIMyUTyUiZ2PBqpE8hoaGrB582aMHj0aAPDDDz+gpaVF5F4RsSk2t87MzETXrl3h7OwMALh+/Tr69esncu8IMR7KRKKOVDOR9lElktfQ0IDdu3fj4sWLsLKygqurK4KCgjpseE1eLU+ePIGDgwPWr1+PDRs2KD23adMms56TRYghKBOJOlLNRPpElUhet27dsHLlSqxcuRIA0NLSAhsbG5F7RcTm4OAAoG2+XlNTE7fxeVNTE4qKisTsGiFGRZlI1JFqJtJAlUheZGQknj17hoSEBABAaGgoZsyYgcmTJ4vcM2IOpk6dCnd3dwwfPhwAUFhYiCVLlojcK0KMhzKRaCO1TKRb/0TywsPDsXnzZu4xYwyRkZGIiYkRsVfEnBQVFSEvLw8AIJPJ4OjoKHKPCDEeykTCR0qZSJ+oEsmzt7dXemxlZWXWVTaI6Tk6Opp1EBPSmSgTCR8pZSINVInkVVRU4ODBg0orXKuqqkTuFSGEiIMykVgSuvVPJK+qqgoxMTHcClc3NzesW7cOvXr1ErtrhBBicpSJxJLQQJUQQgghhJgla7E7QIihKioqsHbtWoSEhODFixeIiopCTU2N2N0ihBBRUCYSS0IDVSJ58fHxcHZ2RpcuXdC9e3f4+flhy5YtYneLEEJEQZlILAkNVInkOTg4YM6cOXj99dcBAEOHDqUVroSQVxZlIrEkNFAlklddXQ2gbQsWAKivr0dpaamYXSKEENFQJhJLQttTEcmTy+WYPn06mpqaEBAQgMLCQrOtWUwIIcZGmUgsCa36JxahuLgYly9fBtAW0vn5+fD19RW5V4QQIg7KRGIpaKBKJGv+/PkanyspKUFWVpYJe0MIIeKiTCSWiG79E8nq0aMHFi5ciMzMTHTt2hXOzs4AgOvXr6Nfv34i944QQkyLMpFYIhqoEsmKjo6Gg4MD0tPTsWbNGu74hAkTsGnTJhF7RgghpkeZSCwRrfonkuXg4ACgbS5WU1MTd7ypqQlFRUVidYsQQkRBmUgsEX2iSiRv6tSpcHd3x/DhwwEAhYWFWLJkici9IoQQcVAmEktCi6mIRSgqKkJeXh4AQCaTwdHRUeQeEUKIeCgTiaWggSohhBBCCDFLNEeVEEIIIYSYJRqoEkIIIYQQs0QDVUJM5OjRo3B3d0d4eLjYXSGEEFFRHhJd0ap/Qkxkzpw5KC8vx88//yx2VwghRFSUh0RX9IkqIYQQQggxSzRQJa+88PBwODo64k9/+hNaWlrw8OFDTJgwAYwxPHnyBB988AE8PT3x+PFjREREYO7cufD19UViYiIYY2hqaoK/vz8cHR2RlJSEgIAAjBo1ClevXkVlZSUWL16M2bNnY8WKFaisrBT77RJCiEaUh8Tc0K1/8srbvHkzLl++jNWrV8PGxgaZmZmorKzE7du3MXz4cMyYMQNyuRzbt29Ha2srjhw5goaGBsyZMwe9e/eGj48PDh48CEdHR1RVVSExMRFpaWmwtbXFhg0b8NZbb2Hv3r2oq6vDrFmzMGrUKLHfMiGEqEV5SMwNfaJKCICJEyciMzMTAHDr1i1MmTIFWVlZAIA7d+5gyJAhSE1NxaxZswAA3bp1g7e3N44dO6b0fTw8PAAA06dPx+DBg3Hu3Dl88MEHAABbW1u4urqa6B0RQogwlIfEnNBAlRAAkyZNwoULF1BfX4/u3bvD3d0dWVlZqKmpgZ2dHaqqqtDU1AR7e3uujb29PZ48eaL0fWxtbbmvKysr0dzcjF69enHHevbsafw3QwghBqA8JOaEBqqEAJDL5SgpKcGxY8cwfvx4uLm54c6dOzhx4gTc3Nxgb2+Prl27Ks2pqqyshIODg8bvaW9vj9dee02pTXV1tVHfByGEGIrykJgTGqgSAqB79+6QyWTYtWsXXF1d0atXLwwfPhyJiYmQy+WwtraGj48PUlJSAAANDQ04deoUZs6cqfF72tjYYMqUKThx4gQAoK6ujrudRggh5orykJgTm+jo6GixO0GIOfj1119RVVUFPz8/AMDz58/R2NiI2bNnAwDGjh2LnJwc/OMf/8CxY8fg6emJjz/+GFZWVli0aBHKyspw48YN9OnTB3379gUAyGQyHD9+HAcOHEBubi4GDhyI7OxsvHjxAjKZTLT3Sggh2lAeEnNhxRhjYneCEEIIIYQQVXTrnxBCCCGEmCUaqBJCCCGEELNEA1VCCCGEEGKWaKBKCCGEEELMEg1UCSGEEEKIWaKBKiGEEEIIMUs0UCWEEEIIIWaJBqqEEEIIIcQs0UCVEEIIIYSYpf8BNR5rd9HuynkAAAAASUVORK5CYII=\n",
      "text/plain": [
       "<Figure size 720x360 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# which tokens influence the representation of ambiance?\n",
    "explain_interventions_with_gradients(\n",
    "    example_sentence, explainer, \n",
    "    0, normalize=True, softmax_norm=False, \n",
    "    control_for_gradient=False, \n",
    "    saliency_fn=compute_guided_saliency, grad_apply_fn=np.abs\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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HByMgIAAxMXFISwsDGvWrNFrk5KSglOnTqFVq1al+qenp2PDhg1YuHAhACAzMxM5OTmWCJ2IiIiIqoBFhv6zsrJw8OBB+Pv7AwB8fX2RmJiItLQ0vXYtWrTA8OHDYWNTun7++eef0aRJE0RHRyMyMhK//PILmjVrZonwiYiIiKgKWOSMqkqlgr29PRQKBQBALpfDyckJqampUCqVoraRkpKCCxcuYMuWLbCzs8Po0aNRVFSEoUOHio7D1dXBlPCJiHTc3R2rOgSzYH4kIimqNlf93717F507d0adOnUAAD179sS6desqVKiq1fkoLhbMFCERVUcVLTwzMvIssh9LY34koqpiKD9aZOjfw8MDBQUF0Gg0AACtVovc3Fx4enqK3kbDhg1hZfVPuLa2trrtEREREVHNY5FC1cXFBX5+foiPjwcAJCQkwNvbG0qlErGxscjLM36G4s0338SxY8cgCA8/8ScmJsLPz8+scRMRERFR1bGOiIiIsMSOvL29sWLFChw9ehRHjhxBeHg4nJ2d8fHHH+OFF16Ah4cHiouLMWvWLCQmJuL27du4ceMGfH19AQBeXl64e/cufvrpJxw6dAharRYTJ06Era2t6Bju3dNC4MgWET2ibl0F7h5WiWrr8LInCgq0Ju9HypgfiaiqGMqPMkGoPamJc7CI6HHu7o4V+mYqzlElIqpcVT5HlYiIiIioolioEhEREZEksVAlIiIiIklioUpEREREksRClYiIiIgkiYUqEREREUkSC1UiIiIikiQWqkREREQkSSxUiYiIiEiSWKgSERERkSSxUCUiIiIiSWKhSkRERESSxEKViIiIiCSJhSoRERERSRILVSIiIiKSJBaqRERERCRJLFSJiIiISJJYqBIRERGRJLFQJSIiIiJJYqFKRERERJLEQpWIiIiIJImFKhERERFJksUKVZVKhdGjRyM8PBwfffQRbty4UWa748ePo3fv3liwYEGZ6/Py8tClSxd888035gyXiIiIiKqYxQrViIgIBAUFYebMmRg4cCDCwsJKtUlJScGpU6fQqlWrcrezYMECNGrUyJyhEhEREZEEWKRQzcrKwsGDB+Hv7w8A8PX1RWJiItLS0vTatWjRAsOHD4eNjU2Z29m3bx+aNm2KZs2amTtkIiIiIqpiZVeElUylUsHe3h4KhQIAIJfL4eTkhNTUVCiVSlHb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06YNevbsiU8++QSffPIJunXrhg4dOojePxERERFVLybPUR0yZAj27t0run3jxo2xbNmyUst37dql+93Kygrh4eHlbmP48OEVC5KIiIiIqi2DhWrr1q0hk8kAAM8++2yp9b169TJPVERERERU6xksVH///XcIgoCJEyciOjpab13dunXh7Oxs1uCIiIiIqPYyWKiWXFX/3XffwcGh9D32jhw5go4dO5onMiIiIiKq1UTNUXVwcEBSUhJu3ryJoqIi3fKVK1di9+7dZguOiIiIiGovUYXqtGnT8Mcff6B58+awtbXVLb9z547ZAiMiIsuqX88O1nJb4w0BPNAWITOn0MwREVFtJ6pQPXPmDOLj4yGXy/WWL1261CxBERGR5VnLbZGxbL2otu6jQgCwUCUi8xJ1H9Vnn30WVlalm7Zq1arSAyIiIiIiAgycUV2yZInud0dHRwQHB6Njx46oW7eubvnWrVsREBBg3giJiIiIqFYqt1D96aef4O/vr3vcqlUrZGdnIzs7W7dMo9GYNzoiIiIiqrXKLVTfffddjBkzxmDnNWvWVHY8REREREQADMxRfbRI3bhxY5lthg4dWukBEREREREBIq/6X7ZsGdLT0yEIgt5ymUwGV1dXdOrUCU2bNjVLgERERERUO4m66v+5557D6tWrcebMGaSmpuL06dNYt24dLl26hD/++APBwcHYs2ePuWMlIiIiolpE1BlVOzs7xMTEQKlU6palpaVh8eLFWLhwIW7duoVx48YhMDDQbIESERERUe0i6ozqjRs39IpUAFAqlbh8+TIAoFGjRrC3t6/86IiIiIio1hJVqFpbW+Pbb79FWloatFotbt++jRUrVui+BODy5csoKCgwa6BEREREVLuIGvr/8ssvMXHiRHz11VeQyWQAgDZt2mDBggXIycnBihUreAcAIiIiIgmqX88e1nJrUW0faB8gM0c6Jx9FFapeXl7YsGEDUlNTcefOHbi7u8PDw0O3PioqymwBEhEREZHprOXWuD3/mqi2DSdL6y5Ooob+S3h6eqJt27a6InXChAlmCYqIiIiIqNwzqmFhYZg+fTrq1KmDbt26lVovCALUarVZgyMiIiLDHJ0VsLOVi2pbWKRFXja//pyqj3IL1Y4dO8LOzg4A4OjoiOnTp+utFwQBc+fONW90REREZJCdrRxvbpsmqu3uvl8iDyxU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\n",
      "text/plain": [
       "<Figure size 720x360 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# which tokens influence the representation of food?\n",
    "explain_interventions_with_gradients(\n",
    "    example_sentence, explainer, 1, \n",
    "    normalize=True, softmax_norm=False, \n",
    "    control_for_gradient=False, \n",
    "    saliency_fn=compute_guided_saliency, grad_apply_fn=np.abs\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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SNqYqFKDMGLECOzevRtDhw6FTCbD7t27xeGpAQMGYP78+cjJyYG9vT327t1bqe7evXvRv39/mJmZYefOnbh06RLCwsK0tmvo0H9ERATGjRuHoKAg8cr/5ORk7N+/H++//77e+wOAmTNnIi0tDUql0qD6RFR/1KeY5+zsjICAAOzYsUOcozpt2jRs2LABffr0QVRUFE6cOIHu3bvj1KlTyMvLQ9++fXXad9OmTVFcXIyjR4/izz//rDSH/8nXXN3xImkzU6lUqrruBNHT9uzZg02bNuHq1auwtLTEgQMHkJCQgAsXLqBz586IiorC1atXkZKSgpdffhl+fn64cuUK1qxZg+3bt6Nz586YMmUKzMzM0KZNG5SVlWHJkiVISEjAs88+iyNHjuD48eMYMmQIvL298eeff+LTTz/F7t27kZubK15NOmPGDGRkZOD48eN49dVX0aRJ7dzRrXnz5ggICMCqVavw9ddfY9euXbh69SpiYmLg5OSE+/fvY/z48UhLS8PFixdRWlqK559/Xqxfsf3ChQv4888/8corr8DR0RHPPPMMAGDTpk349ttvkZ6ejosXL2LAgAG10m8iMo6GHvN69uyJHTt24Msvv8SOHTvw7rvvomvXrlAoFHjxxRexYsUKbN++HUePHkVMTAxat26NI0eOYMWKFbh58yaKi4tRWFiIDRs24OrVq2jSpAm6deuG8vJyrFu3DqdOncLYsWMRExODlJQUnD17FtbW1uLDC7y8vKo8Xo8ePcLYsWORlpaGlJQU9OrVS22qAdU9mSAIQl13goiIiIjoabzhPxERERFJEhNVIiIiIpIkJqpEREREJElMVImIiIhIkpioEhEREZEk8T6q/5OdXYjyct4AgYhMw8nJRnuhOsSYSESmoike8owqEREREUkSE1UiIiIikiQmqkREREQkSUxUiYiIiEiSmKgSERERkSQxUSUiIiIiSWKiSkRERESSxESViIiIiCSJiSoRERERSZLJnkyVnp6OyMhIODo6IjMzE+Hh4XB3d69U7tSpU1CpVOjduzdmzZolrg8LC0NycrK4XFJSgqFDh2LBggXYsWMHoqOjIZfLAQAtW7bE9u3bjf+iiIiIiMhoTJaoqlQqBAcHIygoCImJiYiIiMDmzZvVyly5cgW///47OnbsWKl+06ZNcejQIXE5KioKAwcOFJdXr14NPz8/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\n",
      "text/plain": [
       "<Figure size 720x360 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# which tokens influence the representation of noise?\n",
    "explain_interventions_with_gradients(\n",
    "    example_sentence, explainer, \n",
    "    2, normalize=True, softmax_norm=False, \n",
    "    control_for_gradient=False, \n",
    "    saliency_fn=compute_guided_saliency, grad_apply_fn=np.abs\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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SiUAQKFQwNnZGcnJyVCpVKK2kZGRge3bt2PJkiWYOnWqWXG4uTma1Y+Iqh4PDyer9pM65kcikqJKc3mqefPmYeLEiahRw/yDwBpNLgoLhXKMioikoqwFZFpazlP1KyupF7jMj0RUUYzlR6sUqp6ensjLy4NWq4VSqYROp0N2dja8vLxE9ddoNLh9+za+//57AMCpU6fw999/Iy0tDWFhYaVOFSAiIiKiyssqFZ6rqysCAgIQFxeHwMBAxMfHw8fHByqVCjExMfDz84OTU+nVtJubG9auXat/PGXKFHh5eWHMmDFWiJ6IiIiIKoLVzvqfNm0atm7dirCwMGzcuBEzZ84EACxevBgXL14EABQWFmLGjBk4c+YM4uPjS7xE1cKFC3HmzBkcOHAAq1evtlb4RERERGRlMkEQqs2kJM7BIqq6PDycynQL1cfnqKYsPi6qn2pcG85RJSIqZ8byI+9MRURERESSxEKViIiIiCSJhSoRERERSRILVSIiIiKSJBaqRERERCRJLFSJiIiISJJYqBIRERGRJLFQJSIiIiJJYqFKRERERJLEQpWIiIiIJImFKhERERFJEgtVIiIiIpIkFqpEREREJEksVImIiIhIklioEhEREZEksVAlIiIiIklioUpEREREksRClYiIiIgkiYUqEREREUkSC1UiIiIikiQWqkREREQkSSxUiYiIiEiSbK21I7VajVmzZsHd3R2pqakICQmBt7d3sXYnTpxAeHg4OnTogEmTJumXb926FfHx8ahTpw6uXr2KoKAg9O7d21rhExEREZGVWa1QDQ8PR//+/REYGIjY2FiEhoZi7dq1Bm2SkpJw+vRpNGvWrFj/ffv2YeHChXB0dERmZiY6dOiANm3awMvLy0rPgIiIiIisySpD/xkZGTh06BDatWsHAPD390dCQgJSUlIM2jVp0gQfffQRbG2L188rVqyAo6MjAMDFxQX29va4e/eu5YMnIiIiogphlSOqarUa9vb2UCqVAACFQgFnZ2ckJydDpVKJ2kaNGv/U1H/++Sfq1auHFi1alCkONzfHMrUnoqrLw8PJqv2kjvmRiKTIakP/5SU7OxuLFi3CkiVLIJPJytRXo8lFYaFgociIqCKVtYBMS8t5qn5lJfUCl/mRiCqKsfxolaF/T09P5OXlQavVAgB0Oh2ys7PLPL80KysLISEhCAsLK/FE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\n",
      "text/plain": [
       "<Figure size 720x360 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# which tokens influence the representation of service?\n",
    "explain_interventions_with_gradients(\n",
    "    example_sentence, explainer, \n",
    "    3, normalize=True, softmax_norm=False, \n",
    "    control_for_gradient=False, \n",
    "    saliency_fn=compute_guided_saliency, grad_apply_fn=np.abs\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  warnings.warn(\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAqoAAAGCCAYAAAAokuGsAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAABSKklEQVR4nO3dd1hT59sH8G/YCjhAxG3rQqt1UIuKow60aNW6UYtUbNW66lbEbQUV62jFun9aR2vrQBSsVquoda8WV7VuXIBsVPZ5//DNKTEkOURCTuD7uS4vycnz5NzEeHNzzjMUgiAIICIiIiKSGTNjB0BERERElBcWqkREREQkSyxUiYiIiEiWWKgSERERkSyxUCUiIiIiWWKhSkRERESyxEKVirTo6Gj069cPLi4uOts+efJEclt9/PPPP/jqq6/g7e2Nzz77DN7e3ti2bRtSU1MRHx+PQYMGwcXFBd27d4e3tze6du2KWbNmIS0tDVFRUeLzM2fOzPP1Bw0ahA8++AAjRowwSPxEZFrklP/kKCMjQ8yrjx49MnY4pIlAVMRFRUUJderUKfC2+XHjxg2hZcuWwt9//y0eu3LliuDq6iocOnRIPFanTh3h5MmTgiAIQmpqqtCpUydh6dKl4vMNGzYUGjZsKCQkJKi8/s2bN4VGjRoJ/fv3L/DYich0ySH/GUq7du2EM2fOvPXr1KlTR4iKiiqAiMgQeEWVqBAEBASgV69eaNiwoXisQYMGGDhwoMY+tra2aNeuHU6cOCEee//991GhQgXs2LFDpe1PP/2Ezp07F3zgRERERmRh7ACIAOCXX37BmjVr0KhRI9jb2+PSpUuoW7cuRo8ejaVLl+LGjRsYPHgwPvvsMwBAZmYmli5disuXLwMAmjRpggkTJsDS0hIAsHr1auzbtw8VKlTARx99pHKu3H0VCgVatmyJUaNGQaFQGOR7i4+Px7lz5zB69Gi158aMGaO1b1ZWFiws/vtvqlAo4O3tjY0bN2LIkCEwNzdHUlISXr16hSpVquD+/fsFHT4RGVhRzn8AEBcXh7lz5yIuLg5ZWVlo1KgRJkyYABsbG0RGRiIoKAiCIEChUGDKlClo2LAh/vjjDyxevBjlypVDo0aNcP78eZiZmWHlypVwdHTEtGnTEBsbi8DAQJQqVQpTp07F6tWrcezYMYwdOxYXLlzAuXPnMGPGDHTr1k3r+0XyxkKVZMHLywsxMTHYuXMnwsLCYG1tjbZt28LGxgbfffcdrl+/Dm9vb3h5ecHCwgJr167FjRs3sG3bNgDAsGHDsHbtWowaNQrHjh3D1q1bERYWhjJlymDx4sUq51q/fj2uXbuGbdu2IScnB4MGDULVqlXx6aef6oxz9+7dCAkJ0fj8li1b1I5FRUUBAJydndWes7Ky0vha0dHROHjwID7//HOV4z179sTy5cvxxx9/oFOnTti5cyf69OmDM2fO6IyfiOSnKOc/AJg8eTJcXV0xevRoZGRkwMvLC8+fP0fp0qUxdOhQfP/992jWrBkuXLiAoUOH4tChQ+jQoQOSkpIwb948LFiwAJMnT8bQoUOxc+dODB8+HAsWLMDZs2fh7++PZs2aAQCCg4PRvn173LlzB6tXr8aZM2eQlpam9f0i+WOhSrLSsGFD2NvbAwCqV68OFxcXKBQKuLi44OXLl4iLi4OzszNCQ0MxcuRImJubAwA+/fRTfPfddxg1ahQOHDiANm3aoEyZMgCALl26YP369eI5QkJCMGLECJibm8Pc3Byenp7Yu3evpETdq1cv9OrVq+C/8VwWLlyIUqVKISMjA5999hl8fX1Vnrezs0PPnj2xZcsWeHh44PLly/jiiy9YqBKZuKKY/6Kjo3Hy5EnMnz8fwOtfzgMDA+Hg4IDDhw/Dzs5OLDSbNm2K0qVL48iRI+jRowcA4N1330XVqlUBQPKkpw4dOgAAmjdvDgAIDAzU+H6R/LFQJVmxtbUVv7awsBAfK29/Z2ZmAgCePXuGsmXLim0dHBwQHR0NAIiJiUHdunXF50qXLq1yjmfPnmHjxo3YvXs3AODFixcoVaqUAb6b15RJNjo6Gu+8847O9n5+fnB3d9faZtCgQfD09MQPP/yAtm3bFkCURGRsRTH/PXv2TIxRqV69euJzuY8r2yn7AK9/MVeytrYW3wNtlMV+7hg0vV8kfyxUySRVrFgRCQkJ4uP4+Hjx1nr58uURHx8vPpeYmKjWd8SIEeLko5ycHCQnJ0s6rz63vhwcHNCiRQucPHlSvHKgtGzZMjRp0iTfxWb16tXRunVrbNu2DREREfnqS0SmzZTyX4UKFcQYK1WqBOD1cKhSpUqhYsWKKrEq2yn7FBRt7xfJHwtVMkk9e/bE3r170a1bNygUCuzdu1e8JeXp6Ql/f38kJCSgbNmyCAsLU+sbFhaGTp06wdzcHCEhIfj333/h5+en87z63vqfOXMmfH194eHhIc78P3HiBA4cOIChQ4fm+/UAYMKECXj8+DGsra316k9EpsmU8p+zszNatWqF3bt3i2NUx44di7Vr16Jdu3YICAjA+fPn8eGHH+LSpUtISkpC+/btJb22ra0t0tLScObMGdy8eVNtPH/u71nT+0XypxAEQTB2EET79u3DsmXLkJ6ejlGjRiE+Ph6bNm1CuXLlsGDBAmzYsAGHDh1Co0aNsHbtWtja2mL58uW4ePEigLxnve7duxfOzs5o3rw5li5dCjc3N2zYsAEKhQLfffcdzp07B2tra1SoUAHz5s1DQkICxo0bh7///htubm7YtGmTOKapINy6dQvffvstUlNTYW5ujrJly2Ly5MmoWrUq4uPjMXbsWJw7dw4uLi7o3bu3StJVPn/jxg189NFHWLJkicprb9y4EVu2bEFSUhLatGmDZcuWFVjcRGRYRT3/5Z71n52djcGDB8PT0xMAcPXqVSxatAg5OTkqs/5Pnz6NuXPn4vnz5/Dx8UH9+vUREBCA9PR0fPnll/D19cXWrVvx008/wc7ODgEBAVi3bh1+++031KhRA19++SW6desG4PWQibzeL0EQ8MUXX+DcuXNo1KgRVqxYwSutMsRClYiIiIhkiQv+ExEREZEssVAlIiIiIllioUpEREREssRClYiIiIhkiYUqEREREclSsVxHNS4uFTk5XOyAiAqXk5O97kZGxNxIRMagLTfyiioRERERyRILVSIiIiKSJRaqRERERCRLLFSJiIiISJZYqBIRERGRLLFQJSIiIiJZklWhmpWVhf/9739o3Lgx7ty5k2cbQRAQFBSEqVOnYtSoUdi9e3chR0lEREREhUFW66ju2rULjRs3xqtXrzS2OXDgAB48eICVK1ciPT0dnTt3hpubG6pUqVKIkRIRERGRocnqiqqXlxdcXV21tgkNDUWbNm0AANbW1nBzc0N4eHhhhEdEREREhUhWV1SlePz4MRwdHcXHjo6OePToUb5ew9HRrqDDIiIyecyNRCQ3Jleo5kWhUOSrPbcJJCJj4BaqRETqtOVGkytUK1eujLi4OPFxXFwc3nnnHeMFRERalSljC0tLaaOMMjNzkJj4wsARERGRqTCJQvXw4cNo1qwZ7O3t0b17d4SHh8PLywvp6ek4d+4cRo0aZewQiUgDS0sz/PFTrKS2HQY6GTgaIiIyJbKaTHX58mXMmzcPALBq1SocPnwYALB8+XLcvHkTANC5c2dUrVoVU6ZMwfjx4zFy5EhUrVrVaDETERERkWEoBEEodgOSOA6LqPA4Odnn64pqbGyKgSMyHo5RJSJSpy03yuqKKhERERGREgtVIiIiIpIlFqpEREREJEssVImIiIhIllioEhEREZEssVAlIiIiIllioUpEREREssRClYiIiIhkiYUqEREREckSC1UiIiIikiUWqkREREQkSyxUiYiIiEiWWKgSERERkSyxUCUiIiIiWWKhSkRERESyxEKViIiIiGTJwtgBEBEVJIfSJWFuZS6pbXZGNuKTXho4IiIi0hcLVSIqUsytzPFs8QNJbStMrm7gaIiI6G3w1j8RERERyRILVSIiIiKSJRaqRERERCRLLFSJiIiISJZYqBIRERGRLLFQJSIiIiJZKrbLUzmUtoG5laWkttkZmYhPStP7XGVLW8HCylpS26yMdCQkZeh9LiIiIqKiotgWquZWlohdtVVSW6cR3gD0L1QtrKzxb/CnktrWHh0KgIUqEREREW/9ExEREZEssVAlIiIiIllioUpEREREssRClYiIiIhkiYUqEREREckSC1UiIiIikiUWqkREREQkSyxUiYiIiEiWWKgSERERkSyxUCUiIiIiWWKhSkRERESyZGHsAIiITJVD6RIwt5KWRrMzsgwcDRFR0cNClYhIT+ZWFogJPiipbfnRHxs4GiKiooe3/omIiIhIllioEhEREZEssVAlIiIiIllioUpEREREssRClYiIiIhkibP+i6Aypa1gaWUtqW1mRjoSkzIMHBERERFR/smqUH3y5Anmz5+PcuXKISYmBtOnT0fVqlVV2uzevRsLFy6EpaUlAKBChQrYtWuXMcKVLUsra+z5X2dJbXsM+Q0AC1UiIiKSH1kVqnPmzEG/fv3g4eGBiIgIzJw5E5s2bVJrt2LFCjRr1qzwAyQiIiKiQiObMaoJCQn4888/0bp1awCAu7s7Lly4gOjoaLW2O3fuxKJFizB37lzcvHmzsEMlIiIiokIgmyuqT548QcmSJWFt/XpspZWVFUqVKoXHjx/D2dlZbFenTh3UrFkTjRo1wsOHD9G/f3+EhISotNHF0dEu3/E5Odnnu4++CvNcxjgfkTb8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3L9/H/v378eLFy/w5ZdfAgB2796NKlWqoGbNmnB2dsb48eMxYcIElCpVCk5OTixUiYiIiIoohSAIxe4+D29vqXNysseGzZ0ktf3C53fe+ifSA2/9ExGpM+lb/0RERERUPLFQJSIiIiJZYqFKRERERLLEQpWIiIiIZImFKhERERHJEgtVIiIiIpIlFqpEREREJEssVImIiIhIllioEhEREZEssVAlIiIiIllioUpEREREssRClYiIiIhkiYUqEREREckSC1UiIiIikiUWqkREREQkSyxUiYiIiEiWWKgSERERkSyxUCUiIiIiWWKhSkRERESyxEKViIiIiGSJhSoRERERyZKFsQMgecjITMMXPr9LbktERERkaCxUCQCQlJgJINPYYRARERGJeOufiIiIiGSJhSoRERERyRILVSIiIiKSJRaqRERERCRLLFSJiIiISJZYqBIRERGRLLFQJSIiIiJZYqFKRERERLLEQpWIiIiIZKlY7kxlZqYwdghERLLD3EhEcqMQBEEwdhBERERERG/irX8iIiIikiUWqkREREQkSyxUiYiIiEiWWKgSERERkSyxUCUiIiIiWWKhSkRERESyxEKViIiIiGSJhSoRERERyRILVSIiIiKSJRaqRERERCRLLFTJoP744w+kpqYaOwyTFR0djbS0NJ3tpLQpKM+ePSu0cxEVVcyNb4e5sfhgoaqn1NRU/PvvvxAEAenp6ZL63LlzBxEREeLj48ePQxAEyefMyckxSmJbu3at3n1DQkLg5eWF8+fPS+7zzz//aH1+3759+PnnnwEAs2fPRq9evXDy5EnJr5+UlCS57dtIT0/HixcvAAAPHjzAkSNHkJWVpbVPYGCgyuO7d+9i0qRJOs81evRoJCQk6B9sLrre/3HjxuHWrVv5ft3Ro0djx44desVz6dIlAMD+/fuxcOFCPHnyRGc/fT8nOTk5iI6OxpMnT/DkyRNMmzYt3zEXZ8bIjYBx8iNzo36YG1UxN2pnUShnkam6detCoVBobdOoUSNs375d5dixY8fg7++P6tWrY+PGjfjyyy8xfPhwtGrVSutrLViwAB06dBAfP336FIsWLYKfn5/GPosWLULZsmUxePBg9O7dG7GxsRg2bBiGDBmi9VwJCQmYO3cuTpw4AQBo06YNZs2ahbJly+bZvn379hrfi6SkJAwbNkzr+TQJDg5GRkYGTp06pXJ8z549Gvvs3bsX//vf/zQ+f/z4ccyePRunT59GZGQkFixYgLVr16Jly5ZaY4mMjMS4cePg6OiIzZs3Y+jQoZg2bRrq16+vtd/9+/cxY8YMCIKAdevWYfz48Zg5cyaqVKmitd/EiRPh7u4OT09PeHl5oWnTpjhy5Ajmz5+v1laZXFJSUvD06VPxh7SucyiZm5tj5cqVyMrKQs+ePdGoUSOt7d/m/a9RowaOHj2KdevWoV27dujUqRMsLHSnkqioKPTu3VtnuzetXLkSgwcPxu3btxEYGIhhw4Zh0aJF+O6777T20+dzcvjwYcyfPx+pqakoVaoUEhMTYWVlpfU8UpJ1tWrVMGLECJ3t5MIUciOgX35kblTH3Pgf5sa8GTM3FutC1cvLC3PnztXa5s3f4gAgPDwcv//+O4KCgmBtbY0ff/wRs2fP1pmMa9eujQEDBqicP6/Xzy0tLQ3Dhg1DWFgYqlatin379mH69Ola+wCvE3+LFi0watQoCIKAy5cvY8GCBQgKCsqzvaurK8aPH489e/agfPnyaNq0KQDgwoULePTokc7zaWNlZYW2bduqHFu9ejWaNGmChIQE3L59W0wef//9N9555x2tr1epUiXY2dnh4MGD8Pb2Rt26deHs7Kwzjh9//BGbNm3Chg0bUKJECaxfvx7z5s3T+W+wYsUKjBo1Cnv37kXJkiUREBCApUuX6uxXrVo1DBw4EJs3b0bXrl0xY8YMBAQE5NnW29sbwOsffGfPnhWP29jY4JNPPtH5vX3//fewtrZGSkoKdu/eje+++w5dunRBt27dYG1trdb+bd5/5fedk5ODgwcPokePHujYsSP69++v9d/Bzc0NCQkJcHR0FI8tW7YM48eP13q+GjVq4IMPPsD3338PHx8f+Pj44OnTp1r7APp9Tg4fPozDhw8jKCgI/v7+ePXqFb7//nutfR4/foyePXtqbfNmMSJ3ppAbAf3yI3OjOubG/zA35s2YubFYF6odO3bU2aZdu3ZqxypWrAhbW1vxsZmZGUqUKKHztfK6rZKcnKy1j/J1f/vtN3h5eQEASpUqpfNc5cqVE9sDQJ06dXD37l2N7QMCAmBtbY1nz55h1KhR4vF3330X8+bN03m+/Bo3bhw8PT0xd+5crFixApaWlgCAzMxMLFy4UGvfe/fuYcOGDTh06BAmTZqElJQUST8wqlSpgmrVqomPbWxsJL2XVapUQYsWLXDgwAEAr9/b0qVL6+z36tUrCIKAvXv3Ys6cOQCg8XbmkSNHAACbN2+Gj4+Pztd+04ULF/Dhhx/i0KFDCA0NxcuXLxEXF4c5c+agSZMm6Nevn0r7t3n/w8PD8eGHH+KXX37Br7/+itq1a6NmzZrYtGkT7O3tMXLkyDz7Xb9+HR9//DFq1aoFKysrCIKAhw8f6kzGjx8/xpUrV7Bnzx5s3boVAJCYmKjzPdHnc1K+fHlYWFiItyFLlCiBly9fau3j5uamMxnHxsbqjFdOTCE3AvrlR+ZGdcyN/2FuzJsxc2OxLlR1/ZYPAC1atFA7FhMTg0uXLiEnJwfx8fH4888/JQ2irlu3Lnr37o0mTZoAAC5duoQePXpo7RMbG4vhw4fjzp07aNWqFU6fPo07d+7oPFdMTAyysrLE2w6ZmZlaPxDK3yzv3LmDuLg48Te758+f6xyXow9PT08Ar2/DKRMBAFhaWuocTzRu3Dj89NNPCAwMhJ2dHYKDg/HRRx/pPGd0dDSio6PF23gXLlzAw4cPdfaLiYlBWlqa2O/Jkye4f/++zn5lypRB06ZN0bBhQzRo0ADr16/XmQzySsSHDh3SWTjMmzcPqampaNKkCSZNmgR3d3fxucmTJ6sl47d5/wMDAyEIArp164YtW7aIVxm6du2qdczYq1ev8MMPP4iPBUHAli1btJ4LAD7++GNMnz4dvXr1QqVKlRAYGCip+Mnrc9KmTRutff7991/8888/sLa2xvz581GmTBlcu3ZNa5/Ro0frjEXf28PGYgq5EdAvPzI3qmNu/A9zY96MmRsVQn5HrBchZ8+exfXr19GvXz/Y2tpi3759WL9+PSpWrIgZM2ZoHAPz5MkTTJ48GRcvXoRCoYCrqysWLVokaczMqVOnVMZG5ZXsc0tLS8OJEyfQoEEDVKxYEREREXBwcEDDhg219tu/fz8WLVqEevXqQaFQ4MaNG5gyZQq6dOmitd+BAwcwa9YsODs7Q6FQ4NmzZ5g/fz46deqk83vTx5gxY+Ds7Aw3NzcoFAqcPXsWMTExOm8pvOn48eM6/6P9888/+Prrr8WEXK5cOQQHB6Nu3bpa+505cwb+/v5IT09HmTJlEBcXh+XLl6N58+Y640pKSkKpUqWgUCgQGxsLa2trrVcq8hrTExkZifDwcK3n8fHxQVBQECpUqKByPCoqCkuXLsWyZcvy7KfP+z9y5EgsXboUNjY2aufavHmzxluvUVFRqFq1qsqxhIQEjWMDDeHWrVuoU6eOxudv374NCwsLODo6YvHixUhKSsKIESO0fkZiYmIQEBCAe/fuwc3NDZMmTVJ7b0yNKeRGQL/8yNyojrlRHXOjKmPmxmJdqH7xxRfo1KkTevfujfj4eHz88cfif4YTJ05gxYoVWvsrZy3a2toiOTlZ0q2SN+3YsQN9+/bNV599+/ahW7duOtvduXMHp0+fBgC4u7ujRo0akl4/Li4Of//9NwRBQJMmTeDg4JCv+PIjNTUVK1euxNmzZyEIApo3b45Ro0bBzs5OY5+oqCjs3LkTz58/R05ODgBpCeuff/5BmTJlkJKSAkEQUKNGDUkD3YHXt1P++usv8T0pU6aMzj7//PMPXr58CVdXV+zfvx+RkZHw8fFBpUqVNPbx9fVF9+7dAQBZWVm4ceMGrK2tMXXqVK3nUv72//TpU9SqVQsZGRl5jr96kz7vf16kfI5fvXqFdevWITk5GZMmTcKPP/4IX19fnQPypYzVyktekyJ0TYa4evUqGjRoID7+448/YG1trfUK4/Dhw1G1alXUrFkTp0+fxrvvvqtXvHJiqrkRkJYfmRtVMTeqY25UZczcWKxv/ZcvX14cq/Tbb7+hZcuW4m2AyMhIrX3v37+P58+fi+NqNm/erDF5L1u2DL6+vvj6669VjivHoWj7EMfGxmLlypV48OABsrOzAbxezkNKoVq2bFmUL18eCoVCUvJQcnR0RPv27cXH+v7AkMLOzk4t0URGRmq9IjJ69Gi0bdsWTZo0gbm5OQBp69cNHDgQS5YsyXNsnTbbtm3DZ599Jk562L17N65fv44ZM2Zo7RccHAxfX998zcgMDAxExYoVVY59++23OmOMjIyEv78/qlWrhk2bNkmebZ3X+//w4cM8k7Gm8WFSPsfA60ksDg4OiI2NhY2NDWrWrIlFixZh5syZebbftWsXWrVqhUOHDonJrUuXLti/f7/W8yitXbtWnAiRmZmJmzdv6pwMsWfPHpVk/OGHH+qcDGRnZyd+Fvr37y9pyRy5M4XcCOifH5kbVTE3qmNuVGXM3FisC9XcYzmOHz8u/rYGQOvlaT8/P5w5cwaVK1eGmdnrpWi1jedxdnaGpaUlSpUqpfKBljIOJSgoCB06dEB8fDy8vb3x5MkT8faYNqdOncKUKVNQoUIF5OTkICYmBkFBQSrjc/KS13+4Bw8eGCwZJycn48CBAypXAI4fP45ff/1VY59y5cqp/VYm5VZT8+bN1RLxpUuX4OrqqrXfvXv3VB736tVLXLtOm5o1a+Z7RqYgCOJyLDk5OYiNjcXly5d1nkvf2dbA60R+//598f3X9Ju1ra0tfH1984xZyngqe3t7jBs3DrNnzwYAeHh44OLFixrb379/H9u2bcPTp08xdepU1K9fX/K6nAAwc+ZMldvHgiBoLJiCg4MBvH4vlF8Dr/8NdP2gt7e3F79WKBQqj1euXKkyAcdUmEJuBPTLj8yN6pgb88bcKI/cWKwL1bi4OERHRyMqKgqXLl0Sf5t79eqV1oHd//77L44ePaqytp5y1mNeBg4cCOD1h+PNJSB0/RZTsWJFeHp64uLFi3BzcwMASQsK79q1C/v37xdvuSUlJWHOnDk6k3HuHxjK2yv5+Q+QX8OGDUPlypVRrVo18QqArtEozZo1Q0hICJo0aSLeGlm1ahW++eYbrf3ee+89BAUFoWXLlmK/zZs3a0zGyvUTk5KScPToUfF4Tk4OateurfN702dGZrdu3cQrPMqxYl9++aXOc+k723rFihW4evUqHj9+jPfffx9PnjxBSkpKnm1nzZqldkVDqXr16jrPpfwcKf/f6Ep0EyZMwMSJE+Hh4YFPPvkE169fR0JCAtq3b4969eph5cqVWs/35hjHnJwcnZNf3vzs2djYYNasWVr7nDp1ChMnThQfX716VXx87do1kyxUTSE3AvrlR+ZGdcyN6pgbVRkzNxbrQtXX1xefffYZUlNTMX36dNjZ2eHGjRs6f9tq1KgRXrx4oXILQMpQ3zt37iAyMhIdO3bE+vXr8ffff2PkyJFa1y9TzkZNTU3FjRs3ULp0aUm/sVasWFFlXFjp0qXVBpPnZfHixSr/id3d3bFo0SKd/fRlbW2NJUuWqBzz8PDQ2mflypUq680Br68+6ErGmzdvhouLC65cuSIe03a1Z8uWLRAEAcHBwRgzZoxKzOXKldN6LuD17NHp06ejZ8+ekmdkjhkzBi1btsSZM2dgZmYGNzc3SYlf39nWSUlJWLNmDQIDA+Hv7w8AGv+9Hz16pLFI0XZ7V6l69erw9fVFYmIi5s6di7Nnz6J///4a27dr1w4NGzZEVlYWypcvD3d3d+zduxfh4eH4+++/dX5vuRdqz8nJQUpKCj777LM82ypnp7Zp00bnRMU32dvb49133xUf5/5aysxpOTKF3Ajolx+ZG9UxN6pjbnxNDrmxWBeqjRs3xuHDh1WO1atXT+Otldy/TXfo0AHvvvuuyppnnTt31nq+X3/9FePHj0dkZCR++eUX+Pn5Yc2aNVi+fLnGPjVq1MDBgwfh5eUFb29vvHr1StI4j6dPn+LIkSNwdXWFQqHAxYsXJf3nvHr1qvh1fm6v6KtTp044f/48GjduLC4FEhERoXUmYY8ePdQWI9+0aZPOcw0aNEglqQLadyGpXLkyAKisnZeTk6Nz7TglDw8PlR8symSnjYWFBXx9fdGgQQMIgoBVq1ZhxIgRGpOI0pgxY8TZ1jt37hRnW+uivHqinPwCQOMtuAkTJuDdd99FRkYG/v33X/GHxL///ivp6tegQYNQq1YtHD9+HID67ac3HT58GH/99RdGjBiBb7/9Frdv30ZiYiKWL1+u8+oX8HoNP+W/t0KhgIODg84Zp6mpqeKSN7kLpnr16mnsM2rUKJVxi7nlHtNlSkwhNwL65UfmRnXMjeqYG1UZNTcKpMbPz08YMGCA2vGvvvpKOHv2rNqf0NBQYfTo0Tpfd+nSpYIgCMLChQuFLVu2CIIgCIsWLdLap3v37kJ4eLiQkpIiZGRkCCkpKZK+h0ePHgn9+/cXXFxcBBcXF6FTp07C7du3dfZr3bq14O3tLXh7ewuDBg0SJkyYIFy+fFnSOfWhjC/3n7p160rqm5CQICQmJubrfCkpKcKtW7eEnJwcIS0tTVKfRYsWCWvWrBHS09OFrl27Cs2aNRM2bNigs9/z58+FKVOmCGPHjhVevHghzJw5U2e83t7ewqtXr8THL1++FLy9vSXFKQiCkJqaKqSmpgqCIAhJSUk62w8fPlz4/fffhY0bNwo9evQQBg0aJAwZMiTPtj///LMgCIIQGBio8n0kJSUJgYGBOs/1559/qjwOCQkRdu3apbOfp6en+LWHh4ewY8cOYeLEiTr75eTkCIKQv8/J2LFjhfv37wt///234OHhIRw+fFgYO3aspL7FgZxyoyDolx+ZG/PG3KiKuVGVMXMjC1UN5s2bp3bs2bNngiAIQkBAgMrxU6dOCcOHD9f5muPGjRN+++03oVWrVkJcXJyQnZ0tTJgwQWuf1atXC6dOnRIWLFggzJgxQ/jll1+E58+f6zyXMkblf86TJ08Ko0aN0tkvJCREZ5uCNHLkSLVjut6TqKgooU+fPmLi7tu3r/Dw4UOd54qIiBDc3d2F/v37C2lpaYK3t7dw4sQJnf3mzJkjCIIg7Nu3TxgxYoQgCILg7++vs9+UKVOEX3/9VZg0aZIgCIJw7do1Yfr06Vr7zJw5U+1YXp/FvNy7d084f/68cO7cOeHcuXOSCoSsrCwhOztbEARBCAsLE7Zu3Sps3rxZa5+8EuGUKVN0nuubb75ROzZ58mSd/S5cuCB+PXDgQJ3tlfT5nOhTMPXv319Yvny52vFvvvlG6NKli+R4TYVccqMg6JcfmRvVMTeqY25UZczcWKxv/WuTezKAUnZ2tjig+unTp+LYqypVqqjsYKHJoEGDsG7dOnz99ddwcHDAokWLUKtWLa19hg8fDgCoX78+jhw5glWrViEgIEDjOBTlrMg3Y3xzMWFNevTogX///Rd//vknFAoFWrVqpTPGt7Fy5Uq1mZW6dv9YtmwZxo4dKw70v3jxIpYtW4alS5dq7afv7E99t7F1dnZG3759xVuG7733nsqsx9zOnz8vvu6KFSvE7+3y5cvi7Glt8jvbWun48eNYs2YN4uLikJOTA0EQkJycjEGDBmnsk5CQgE2bNuHDDz8UF8J+/vy5xvbK9TcjIyNVFu3OycmRFGPuz8O2bdt0tlfS53Py8OFDHDhwAGFhYQgNDUVOTg6io6O1nqdOnToYO3Yspk2bhjt37sDT0xNDhgzBjBkzdI4NNEVyyY1A/vIjc6NmzI3qmBtVGTM3FutCddeuXejdu7fk9t7e3gBeD7I+e/aseNzGxgaffPKJzv6urq5YtWqV+Hjq1KmIiYnR2mfLli04fPgwbt68iZYtW2Ls2LFadxl52xj37duH5cuXo379+gCArVu3Yty4cejatavOvvoIDg7GlStXJM2sVKpQoYJKAm3dujVOnjyp81z6zv7Udxtb5SxW5Q/2ly9fIioqKs+2yjFOyh+e586dE597+PChxh1NlPI721pJuVZftWrVYGZmBkEQsHbtWq19AgMDERAQIH6WmzdvjsDAQI3tlbOxnz17Jn4NvJ548cEHH2jsp9ytZenSpeJ4Nj8/P537bSvp8znJq2DSNWFD+Z4vWLAAEyZMwJAhQ9SeMzWmkBuB/OVH5kbNmBvVMTeqMmZuLNaF6po1a3Dq1Kk8n8trD9sjR44AeD2LT9MCv9oofzPMTdeMwIcPHyI1NRVjxoxB586dde6E8rYxnjhxAgcPHlTZB9vPz89gyTgxMVHyzEqlJ0+eqGwvFx8fr3MNPkD/2Z/ffPONuE2jubk50tPTJe1h3KJFC3Tt2hUZGRkYNmwYrl69qnEh7BEjRohL9bxp+/btOs+l72zr2rVro2XLlirHvvrqK619nJ2d1bYR1Pbbf8+ePQG8LkbeXKrl1q1bGmd2T5o0CcnJyUhMTMTu3bvx3nvvSZrRqqTP5ySvgik/s1NNtTB9kynkRiB/+ZG5UTPmRnXMjaqMmRuLdaH65tIJuWn7B9AnyQHAxIkTxQ9jVlYW7t69q3PrPuVvi1evXsXWrVvx6NEj1KtXL8/FhQsiRicnJ5Wt8ywtLeHk5KTXa0mRn5mVSv369cMnn3wCR0dHKBQKxMXFqS3jkpdevXphyZIluHjxInbs2IEPPvhA0uxPGxsbdOzYUXzctm1b7Nu3T+dSHZ07d4aLiwtOnz4NQRDg5+en8d9bUyIGoHWZEn1nWytn9FasWBHTpk1TWXdR11Z62dnZOHPmjMruQ9r63L59G7Vq1cLly5fVZklr6/fLL78gOTkZXbp0wd27dxEWFoZHjx7hq6++QosWLfD5559rjBHQ/3MidZFvpdxrBV6+fFll3cC8ijpTYAq5EdAvPzI3qmNu/A9zo2bGyo3FulAt7GVl/P394enpKT5OT0/Hxo0btfb58ccf0bFjR1y6dAkXL16UtNj/23j+/Dm2bNmisnSLrnFRb+Pu3bs4dOgQateujV69esHOzk7nmLYWLVogLCxM/A1S6v7SAQEBGDZsGNasWQOFQqFyq0ubt9nGtkaNGioJuKC3XNS0I8qzZ89w6NAhjf1Wr16NJk2aiI9zJ0ld446GDx+O1NRUlYXItfWZNWsWlixZorJtn5RzffXVV2jVqhXs7OzEJYc8PT0xevRojVf7ctPnc5KfRb6Vchd1bxZ3prqOqinkRqBw8yNzozrmRlXMjaoKLDdKnnZVTGRlZRXq+XTN4mzUqJHw0UcfCXPnzhVOnjwpZGZmGjSe+Ph4YeLEiYKbm5vQvHlzYfLkycKTJ08Mdr6srCzxPVfOrJS6BFduixcv1tkmrxnCsbGxOvtNmjRJ+O2334QxY8YIZ8+eFUJCQrT+uw0aNCjPP97e3kKbNm3y9X3pou9s699++03jc0ePHtV6zv79+6sdO3XqlNY+giAIBw8elHRM6a+//hJWrlwp1K9fX2jTpo0wePBgoVmzZsLJkyeF9PR0nefLi67PiXL2be73c+HChVr7/PHHHxqf0/VemhK55UZBKNz8yNyojrlRFXOjqoLKjcX6iuq2bduwevVqzJkzBx06dADw+lL87Nmz8cMPP+gcD5pfb+5lHRMTg8aNG+vsM2HChAKNIy+5x4h5eXmhX79+4uPAwECdY8X0pfytE4DOCQ2DBg1CUFAQ2rVrpzLeRRAEKBQKnQt953cFBaX8btP4tvs+54e+s61zX716U9u2bbWes0WLFnj48CGqVasmHrt//77WBaqB1wuYv0nbFpSNGjVCo0aNsG/fPoSHh+POnTvw9fXFnj17MHv2bI1XRd7mc6LP7da8rjwmJyfD3t5e53spV6aQG5X9DJ0fmRs1Y25UxdyoqqByY7EuVI8ePYoff/xR5faDi4sLxo8fj2+//VbrbD195N4rWrlXsa4BxoVRpCrPk3tmJfA6RuH/x/PIwfTp0+Hs7IwvvvgCkydPVnlu8eLFOvvndwUFpfxu06ht3+fcCawgvO1MZn3s2bMHa9euRdmyZcUxX8nJyRgwYIDWfvreJvTy8oKZmRlq164Ne3t7BAUFaW3/Np8TfW63btu2DX/++ScCAwNRtmxZzJkzB9u3b4e9vT2+//57nT+k5MgUciNQOPmRuVEz5kZVzI2qCiw3Sr72WgRpW2BY1+LD+oiPjxdOnTol7NmzRwgJCRFCQkIEX1/fAj+PPrZt26bxOeWuG3IRFRWl8vj8+fPCjRs3dPabP3++0KtXL2Hr1q1CXFyc5PNt2rRJOHDggHD58mXB1dVVqFevnhAcHKyzn3KB5MLw448/Ftq5BgwYIDx69Ej8ExUVJUydOlVnv/zeJszLm//2+SHl1nVWVpZw69YtYfPmzcLKlSsl3W4dPHiwuLPRjRs3hA8//FC4d++ecPfuXWHYsGF6x2tMzI3/YW7UjLlRFXOjqoLKjcX6iqqgZYkKXfve6sPPzw8pKSmoVq2aeLVA1+DswqLvzEpj2LRpk8pSJs7OzggKCtJ5C07fFRQeP34szqI8c+YMjh8/jpCQEI3td+3ahVatWuHQoUMYP348AKBLly7Yv3+/pO9PH/rOZNbHhg0b1NZYXLBggc5++b1NmJcqVapIbjtjxgzExsZizZo1AF7PLO/WrZt4Kzsv27dvx6pVq8T9xJUzU7XtJ161alXUrFkTAHDo0CF4enqK+3trunokd8yN/2Fu1Iy5URVzo6qCyo3FulBNS0tDamqqyvpqwOvbGLnHYRSU5ORk/PzzzyrHjh07VuDnKaqUy4bcvXtX/Bp4PRZJuYC0NvmdIaxpJ5s6depovS15//59bNu2DU+fPsXUqVNRv359reONTM3Lly8xZ84cpKWlYcGCBVi4cCEmTpyI0qVLa+2X39uEbysrK0tMxMDr3VhmzJihNRkfPHgQhw8fFouxV69eYdiwYVqTce7PwpEjR8QfwIDqOENTwtxoWpgb5YG5UVVB5cZiXagOGDAAPj4+8PHxEav+O3fuYOvWrQbZ+tDNzU1toLVcxjiZAuU4o2fPnqmMObKyssKwYcN09l+2bBk2btyI9u3bY/jw4XBzc1NZF/FN+o5xmjBhAiZOnAgPDw988sknuH79OhISEtC+fXvUq1cPK1eu1BmrnAUFBaFp06Y4d+4cSpYsif79+2Px4sWYP3++1n41atTAwYMH4eXlBW9vb7x69UrnJI+38eaEH4VCoXGbRqXatWurXDEsUaIE6tWrB+D1Fb68FuB++fIl9u7di0ePHuH58+fiIuFXrlzBgwcP3vbbMArmRtPC3CgPzI2qCio3FutCtWnTpvD398eKFSsQGRkJAGjcuDFmzJgh/gMUpPfffx89evSAra2tykBrbXsH03+Ut1COHTuGjz76KN/98ztDWN+dbNq1a4eGDRsiKysL5cuXh7u7O/bu3Yvw8PB87R4iV/nZpzu3sLAwDB8+HLVq1cKZM2eQnp6udsWuIMXFxeV73UuFQqG2n7ilpSXOnz+PjRs34ocfflDrM3XqVCxduhSpqan44YcfYG5ujn/++Qc//fQTevXqZZDvzdCYG00Lc6M8MDeqKqjcqBC0DUaiAtW5c2fMmjULVatWFWeNBgcHS96flzQLDg6WtHVfYcjKysJff/2FESNGoFGjRrh9+zYSExPx+eefw93dHc2aNTN2iG9l1qxZmDdvHubMmYM5c+bg5cuXmDJlCoKDg7X2W7NmDRo2bIhjx47hxYsXeP/999GhQwc4OjoaJM6EhAQEBATgxIkTUCgUaNOmDaZNmyZuG5iXNm3aqG1lqPTgwQMcP37cILEWd8yNhsPcWHiYGw1E8rSrIujPP//U2eb06dMFdr6vv/5a7diDBw8K7PWLiwsXLgidO3cW3nvvPaFu3bqCi4uLULduXWOHpcbT01P82sPDQ9ixY4cwceJEI0ZUMPbv3y988sknQseOHYWhQ4cKLVq0EMLDwyX3T0pKEkJCQoROnToJDRs2NGCk6m7evKn1eW2zuDXN/i7sPFIYmBtNE3OjcTE3qiqoPFKsb/0fOnRIHDOhydGjR9G8efMCOV+lSpXyvXcwqfv555+xdetWrF69Gv7+/njy5Al++uknY4elJve4pPLly6NPnz7o06ePESMqGPnZpzs3fddqzK/bt2+jZs2aCA0NVXtO1/83bbO4Nc3+Luw8UhiYG00Tc6NxMTeqKqg8UqwL1e3bt+OXX37R2ubN/XffRlhYGFq1apWvvYNJXaVKleDg4ICcnBzxcVpampGjUpd7zM+2bduMGEnBe3Of7nXr1mHo0KFa+zx8+BCpqakYM2YMOnfuXOC7Gynpu3+2vgo7jxQG5kbTxNxofMyN/ymoPFKsC9V//vmnUM83cuRItR0qIiIiCjWGouDevXt49uwZsrOzsWnTJpQpU8agS3nkV1RUFKpWrYqlS5fCw8MDwOt1Ik19vN2UKVMwefJkeHl5qW2/l5ycrDMZ67tWY34pryCNGzdObWvC33//vUDPBRR+HikMzI2mibnROJgb81ZQeaRYF6qFLa9t1Ex1L3Bj8vb2RkxMDL766iv4+/sjKSnJoEt55NekSZOQnJyMxMRE7N69G++9916RmNHapUsXlC5dGm5ubhgzZox4XPj/iS+65HetxreV1/7ZSUlJBj0n6Ye5sWAwNxoHc6NhcdY/mZwFCxagcePG6Ny5s7FD0Sg5ORldunRBjx49cP36dZw/fx4tW7ZEixYtxJ1cTNXatWvx/vvvq+zTnJ6eDmtra639GjdujDJlyqB9+/bw8PDQuVajvjQtlyP8/97sXEieiirmRuNibjQMXlElkxMREYGRI0caOwyNvvrqK7Rq1Qp2dnbi1QxPT0+MHj0ap06dMnJ0b++nn37Cpk2bVI7pSsRA/tdq1JetrW2et8wEQcCWLVsMfn4iY2FuNC7mRsNgoUom54MPPlDbb3zTpk0YPHiwcQJ6w4gRI3Dy5Ek8evQIH330EWrUqIHExEQkJyfLJsa38cEHH6jsIAQAISEh6Nmzp9Z+hZGIgdcTBjTtI/1m3ERFCXOjcTE3GgZv/ZPJ+frrr3Ht2jU0btxYXMomMjIS4eHhRo5MVefOnREeHo47d+7A19cX7u7uuHz5Mg4dOmTs0N7KrFmzcOvWLTRr1gyWlpYAgOPHj+PXX381cmSq4uLiEBQUhPT0dAQGBkred5vIVDE3Ghdzo2GYGTsAovy6e/cuRo8ejVatWsHNzQ1ubm4oX768scNS4+XlBTMzM9SuXRv29vYICgoy6UQ8Z84cpKam4tSpU2jVqpWYiOVKue+2paWlyr7bREUVc6NxMDcaFm/9k8n55ptv0KRJE5VjjRs3Nk4wWuS+lbVu3TrjBVJArKysYGdnB0EQ1LZkNDOT3++8+u67TWSqmBuNg7nRsFiokslp0qQJjh8/Lu4t3KZNG4Ps4lGQqlSpYuwQ3tq9e/ewc+dOVK9eHXv27FF57sKFC8YJSovExEQAENc1fPnyJR4+fGjEiIgMi7nROJgbDYuFKpmc//3vf/j999/FKwc//PCDONaJDGf48OEIDQ3Fs2fPcPbsWZXn5LiLUIsWLdC1a1dkZGRg2LBhuHr1KmbMmGHssIgMhrnROJgbDYuFKpmcu3fvYvv27SrHpk2bZqRoio+mTZuiadOm+P333wtlV5O31blzZ9StW1dc9sbPzw/nz583clREhsPcaBzMjYbFQpVMjp2dndoxW1tbI0RSPOW1q0lex4xF06LWBw8exIMHD+Dl5VXIEREVDuZG42JuNAwWqmRy0tLSsHDhQri6ukKhUODixYvIzs42dlgkE8pFrSMiImBlZYWmTZsCAC5evAgXFxcjR0dkOMyNpI2p5kauo0omJy0tDatWrcKJEycAvJ4w8NVXX6ktdE3FU3R0NJydnTF79mzMnTtX5bn58+fLeiwW0dtgbiRtTDU38ooqmRwbGxuMHz8e48ePBwBkZ2fD3NzcyFGRXDg7OwN4PV4vIyNDXPg8IyMDN2/eNGZoRAbF3EjamGpuZKFKJmfGjBmIjY3FmjVrAAATJ05Et27d0KFDByNHRnLSqVMntG/fHg0aNIBCocCVK1cwYsQIY4dFZDDMjSSFqeVG3vonk+Pn54eFCxeKjwVBwIwZMxAQEGDEqEiObt26hbNnz0IQBDRr1kzW47CI3hZzI0llSrmRV1TJ5Dg4OKg8VigUst5Vg4ynTp06qFOnjrHDICoUzI0klSnlRhaqZHLi4uKwZcsWlZmtCQkJxg6LiMiomBupKOKtfzI5CQkJCAgIwIkTJ6BQKNCmTRtMmzYNZcuWNXZoRERGw9xIRRELVSIiIiKSJTNjB0CUX3FxcZg6dSrGjRuHV69eYdasWUhKSjJ2WERERsXcSEURC1UyOUFBQWjatCksLS1RokQJ9O/fH4sXLzZ2WERERsXcSEURC1UyOc7Ozujbty9KliwJAHjvvfc4s5WIij3mRiqKWKiSyUlMTATweukVAHj58iUePnxoxIiIiIyPuZGKIi5PRSanRYsW6Nq1KzIyMjBs2DBcvXpVtnsUExEVFuZGKoo4659M0r1793Dq1CkAr5Pz+fPn4eXlZeSoiIiMi7mRihoWqmQyfHx8ND734MEDHDt2rBCjISKSB+ZGKsp4659Mhq2tLXx9fREREQErKys0bdoUAHDx4kVZ71NMRGRIzI1UlLFQJZMxZ84cODs7Izw8HFOmTBGPt2rVCvPnzzdiZERExsPcSEUZZ/2TyXB2dgYA3L17FxkZGeLxjIwM3Lx501hhEREZFXMjFWW8okomp1OnTmjfvj0aNGgAhUKBK1euYMSIEcYOi4jIqJgbqSjiZCoySbdu3cLZs2chCAKaNWvGcVhERGBupKKHhSoRERERyRLHqBIRERGRLLFQJSIiIiJZYqFKZCA7duxA+/bt4efnZ+xQiIhkg7mR8oOz/okMpG/fvnj27BkeP35s7FCIiGSDuZHyg1dUiYiIiEiWWKhSsePn5wcXFxd89tlnyM7Oxv3799GqVSsIgoDo6Gh0794dnp6eePr0Kfz9/TFgwAB4eXlh3bp1EAQBGRkZGDRoEFxcXLBt2zZ8+eWXeP/993H27FnEx8dj6NCh6NOnD77++mvEx8cb+9slIpKEuZHkiLf+qdhZuHAhzpw5g0mTJsHc3BwRERGIj4/HtWvX0KBBA3Tr1g0tWrTAihUrkJOTg59//hlpaWno27cvnJyc0KNHD2zZsgUuLi5ISEjA+vXrsW/fPtjZ2WHu3LmoWLEi1q1bh9TUVPTu3RtNmjQx9rdMRKQTcyPJEa+oUrHUpk0bREREAACuXLmCjh074tixYwCA69evo169eti3bx969+4NALCxsUGXLl2we/duldfx8PAAAHTr1g1169bF4cOH0b17dwCAnZ0dWrduXUjfERHR22NuJLlhoUrFUtu2bXH06FG8fPkSJUuWRPv27XHs2DEkJSXB3t4eCQkJyMjIgIODg9jHwcEB0dHRKq9jZ2cnfh0fH4+srCyULVtWPFa6dGnDfzNERAWEuZHkhoUqFUvu7u548OABdu/ejRYtWqBNmza4fv06QkND0aZNGzg4OMDKykplHFV8fDycnZ01vqaDgwMsLCxU+iQmJhry2yAiKlDMjSQ3LFSpWLKxsYGbmxt++OEHtG7dGmXLlkWDBg2wdu1atGjRAmZmZujRo4d4OystLQ2//fYbevXqpfE1zc3N0bFjR4SGhgIAUlNTxVtoRESmgLmR5IaFKhVbbdu2Ra1atWBvby8+dnFxga2tLYDXM2DNzMwwYMAA+Pj4oGvXrvj0008BAEOGDAEATJgwAadPnxZfc9asWYiOjkbv3r0xefJkNG/eHCdOnEBwcHAhf3dERPphbiQ5UQiCIBg7CCIiIiKiN/GKKhERERHJEgtVIiIiIpIlFqpEREREJEssVImIiIhIllioEhEREZEssVAlIiIiIllioUpEREREssRClYiIiIhkiYUqEREREcnS/wHOSHqc8QgmtwAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 720x360 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# which tokens influence the representation of noise?\n",
    "explain_interventions_with_gradients(\n",
    "    example_sentence, cpm_explainer, 2, normalize=True, softmax_norm=False, control_for_gradient=False, saliency_fn=compute_guided_saliency, max_key_fn=np.abs\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  warnings.warn(\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(224,108,120,0.5791871049766436)\">[CLS]</span> the <span style=\"background-color:rgba(224,108,120,0.5834737695719576)\">music</span> was <span style=\"background-color:rgba(224,108,120,0.4327806125995057)\">too</span> <span style=\"background-color:rgba(88,117,220,1.0)\">loud</span> , and the <span style=\"background-color:rgba(88,117,220,0.6236991533178343)\">decorations</span> were taste ##less , but they had <span style=\"background-color:rgba(88,117,220,0.48892623629506404)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.7956085918122262)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.6396777279517661)\">delicious</span> <span style=\"background-color:rgba(88,117,220,0.6682189668347428)\">pasta</span> <span style=\"background-color:rgba(224,108,120,0.4829758588356244)\">[SEP]</span></p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.7614021517839192)\">[CLS]</span> the music was <span style=\"background-color:rgba(224,108,120,0.5092843935731525)\">too</span> <span style=\"background-color:rgba(224,108,120,0.6342833317445231)\">loud</span> , and the <span style=\"background-color:rgba(88,117,220,0.6465867990633256)\">decorations</span> were <span style=\"background-color:rgba(88,117,220,0.9222189292425717)\">taste</span> <span style=\"background-color:rgba(88,117,220,1.0)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.748178079043467)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.8448035404669969)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.9892006129346772)\">delicious</span> <span style=\"background-color:rgba(88,117,220,0.7890479809682556)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# which tokens influence the representation of noise?\n",
    "explain_interventions_with_gradients(\n",
    "    example_sentence, cpm_explainer, 2, normalize=True, softmax_norm=False, control_for_gradient=False, saliency_fn=compute_guided_saliency, max_key_fn=np.abs, display_fn=display_colored_text\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  warnings.warn(\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(224,108,120,0.5249118518050994)\">[CLS]</span> the music was too <span style=\"background-color:rgba(224,108,120,0.33869018300126646)\">loud</span> , and the <span style=\"background-color:rgba(224,108,120,0.6662489386852669)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.42523596313142475)\">taste</span> <span style=\"background-color:rgba(224,108,120,0.4105173316620831)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.6224246724236416)\">friendly</span> <span style=\"background-color:rgba(224,108,120,1.0)\">waiter</span> <span style=\"background-color:rgba(224,108,120,0.36462679622661975)\">##s</span> and <span style=\"background-color:rgba(224,108,120,0.8303134640837139)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.7331719807201494)\">pasta</span> [SEP]</p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.7711151040203997)\">[CLS]</span> <span style=\"background-color:rgba(224,108,120,0.24203210336070727)\">the</span> music was too <span style=\"background-color:rgba(224,108,120,0.5227846791288765)\">loud</span> , and <span style=\"background-color:rgba(224,108,120,0.49804589951780415)\">the</span> <span style=\"background-color:rgba(224,108,120,0.7650143142246931)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.9770078968827943)\">taste</span> <span style=\"background-color:rgba(224,108,120,1.0)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.6979441212129568)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.806793641420771)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.9659819287781044)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.7078216031949561)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# which tokens influence the representation of service?\n",
    "explain_interventions_with_gradients(\n",
    "    example_sentence, cpm_explainer, 3, normalize=True, softmax_norm=False, control_for_gradient=False, saliency_fn=compute_guided_saliency, grad_apply_fn=np.abs, display_fn=display_colored_text\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "ambiance\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p>[CLS] the music was too <span style=\"background-color:rgba(224,108,120,0.5801911132338835)\">loud</span> , <span style=\"background-color:rgba(224,108,120,0.4626121574886977)\">and</span> the <span style=\"background-color:rgba(224,108,120,0.9880765217607794)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.49723047118185876)\">taste</span> <span style=\"background-color:rgba(224,108,120,0.5590052088110992)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.40294578873794845)\">friendly</span> <span style=\"background-color:rgba(224,108,120,1.0)\">waiter</span> <span style=\"background-color:rgba(224,108,120,0.5971234170654398)\">##s</span> <span style=\"background-color:rgba(224,108,120,0.27065276410498)\">and</span> <span style=\"background-color:rgba(224,108,120,0.9621578369496453)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.6691737345748489)\">pasta</span> [SEP]</p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.7120055027636593)\">[CLS]</span> <span style=\"background-color:rgba(224,108,120,0.2818841123788514)\">the</span> music was too <span style=\"background-color:rgba(224,108,120,0.6241514290068051)\">loud</span> , and <span style=\"background-color:rgba(224,108,120,0.4761196314790065)\">the</span> <span style=\"background-color:rgba(224,108,120,0.6945836303329963)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.9308752324841411)\">taste</span> <span style=\"background-color:rgba(224,108,120,1.0)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.6808726396006268)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.8617444370681464)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.9688381497289617)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.7664740959412996)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "food\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(224,108,120,0.36812205260933)\">[CLS]</span> the music was too <span style=\"background-color:rgba(224,108,120,0.23913721537684657)\">loud</span> , and the <span style=\"background-color:rgba(224,108,120,1.0)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.4936254837956224)\">taste</span> <span style=\"background-color:rgba(224,108,120,0.4256389885402326)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.2533826371035261)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.3575708412081197)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.7207138134033387)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.6844144407544662)\">pasta</span> <span style=\"background-color:rgba(224,108,120,0.4619949033889133)\">[SEP]</span></p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.8883231792612594)\">[CLS]</span> the music was <span style=\"background-color:rgba(224,108,120,0.5354460409036612)\">too</span> <span style=\"background-color:rgba(224,108,120,0.5886290772003703)\">loud</span> , and the <span style=\"background-color:rgba(224,108,120,0.6760708704431887)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,1.0)\">taste</span> <span style=\"background-color:rgba(224,108,120,0.9564088223894066)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.8037744242527081)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.8700426037353125)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.9163026344747734)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.7826921885758732)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "noise\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(224,108,120,0.5791871387384263)\">[CLS]</span> the <span style=\"background-color:rgba(224,108,120,0.5834737598608505)\">music</span> was <span style=\"background-color:rgba(224,108,120,0.4327805990475418)\">too</span> <span style=\"background-color:rgba(224,108,120,1.0)\">loud</span> , and the <span style=\"background-color:rgba(224,108,120,0.6236991087832529)\">decorations</span> were taste ##less , but they had <span style=\"background-color:rgba(224,108,120,0.4889262404187908)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.7956085833093308)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.6396777413933407)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.6682189786941312)\">pasta</span> <span style=\"background-color:rgba(224,108,120,0.4829758585686282)\">[SEP]</span></p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.7614021794709854)\">[CLS]</span> the music was <span style=\"background-color:rgba(224,108,120,0.5092844281543248)\">too</span> <span style=\"background-color:rgba(224,108,120,0.634283328689132)\">loud</span> , and the <span style=\"background-color:rgba(224,108,120,0.6465868655938486)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.9222189810038711)\">taste</span> <span style=\"background-color:rgba(224,108,120,1.0)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.7481781295018566)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.844803573855815)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.9892006674955198)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.7890480390573562)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "service\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(224,108,120,0.5249118518050994)\">[CLS]</span> the music was too <span style=\"background-color:rgba(224,108,120,0.33869018300126646)\">loud</span> , and the <span style=\"background-color:rgba(224,108,120,0.6662489386852669)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.42523596313142475)\">taste</span> <span style=\"background-color:rgba(224,108,120,0.4105173316620831)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.6224246724236416)\">friendly</span> <span style=\"background-color:rgba(224,108,120,1.0)\">waiter</span> <span style=\"background-color:rgba(224,108,120,0.36462679622661975)\">##s</span> and <span style=\"background-color:rgba(224,108,120,0.8303134640837139)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.7331719807201494)\">pasta</span> [SEP]</p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.7711151040203997)\">[CLS]</span> <span style=\"background-color:rgba(224,108,120,0.24203210336070727)\">the</span> music was too <span style=\"background-color:rgba(224,108,120,0.5227846791288765)\">loud</span> , and <span style=\"background-color:rgba(224,108,120,0.49804589951780415)\">the</span> <span style=\"background-color:rgba(224,108,120,0.7650143142246931)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.9770078968827943)\">taste</span> <span style=\"background-color:rgba(224,108,120,1.0)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.6979441212129568)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.806793641420771)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.9659819287781044)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.7078216031949561)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import warnings\n",
    "warnings.filterwarnings('ignore')\n",
    "\n",
    "example_sentence = 'The music was too loud, and the decorations were tasteless, but they had friendly waiters and delicious pasta'\n",
    "aspects = ['ambiance', 'food', 'noise', 'service']\n",
    "\n",
    "for i, a in enumerate(aspects):\n",
    "    print(a)\n",
    "    explain_interventions_with_gradients(\n",
    "        example_sentence, \n",
    "        cpm_explainer, \n",
    "        i, \n",
    "        show_control=True,\n",
    "        normalize=True, \n",
    "        softmax_norm=False, \n",
    "        control_for_gradient=False, \n",
    "        saliency_fn=compute_guided_saliency,\n",
    "        grad_apply_fn=np.abs,\n",
    "        max_key_fn=lambda x: x,\n",
    "        display_fn=display_colored_text\n",
    "    )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "ambiance\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p>[CLS] the music was too <span style=\"background-color:rgba(88,117,220,0.5801911205982717)\">loud</span> , <span style=\"background-color:rgba(88,117,220,0.46261213316177874)\">and</span> the <span style=\"background-color:rgba(224,108,120,0.98807646470439)\">decorations</span> were <span style=\"background-color:rgba(88,117,220,0.4972304541611052)\">taste</span> <span style=\"background-color:rgba(88,117,220,0.5590052187808517)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.40294579261926733)\">friendly</span> <span style=\"background-color:rgba(224,108,120,1.0)\">waiter</span> <span style=\"background-color:rgba(88,117,220,0.5971233986354347)\">##s</span> <span style=\"background-color:rgba(88,117,220,0.2706527635664146)\">and</span> <span style=\"background-color:rgba(224,108,120,0.9621578269038996)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.6691737822176688)\">pasta</span> [SEP]</p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.7120055065386721)\">[CLS]</span> <span style=\"background-color:rgba(224,108,120,0.2818841186804858)\">the</span> music was too <span style=\"background-color:rgba(224,108,120,0.6241514714632593)\">loud</span> , and <span style=\"background-color:rgba(224,108,120,0.4761196425408062)\">the</span> <span style=\"background-color:rgba(88,117,220,0.6945837070738443)\">decorations</span> were <span style=\"background-color:rgba(88,117,220,0.9308752796117912)\">taste</span> <span style=\"background-color:rgba(88,117,220,1.0)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.6808726422493947)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.8617445309666762)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.968838158975821)\">delicious</span> <span style=\"background-color:rgba(88,117,220,0.7664741643294612)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "food\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(224,108,120,0.3681220387325813)\">[CLS]</span> the music was too <span style=\"background-color:rgba(224,108,120,0.23913720254576074)\">loud</span> , and the <span style=\"background-color:rgba(224,108,120,1.0)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.49362546367587823)\">taste</span> <span style=\"background-color:rgba(88,117,220,0.4256389662208804)\">##less</span> , but they had <span style=\"background-color:rgba(88,117,220,0.2533826302654236)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.3575708162971167)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.7207137839744514)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.684414452274572)\">pasta</span> <span style=\"background-color:rgba(88,117,220,0.46199488723857784)\">[SEP]</span></p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.8883231446526141)\">[CLS]</span> the music was <span style=\"background-color:rgba(224,108,120,0.5354460378816024)\">too</span> <span style=\"background-color:rgba(224,108,120,0.588629052587729)\">loud</span> , and the <span style=\"background-color:rgba(224,108,120,0.676070886122906)\">decorations</span> were <span style=\"background-color:rgba(88,117,220,1.0)\">taste</span> <span style=\"background-color:rgba(88,117,220,0.9564087928589273)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.8037744624052939)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.8700426092990022)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.91630263900188)\">delicious</span> <span style=\"background-color:rgba(88,117,220,0.7826921991189395)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "noise\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(224,108,120,0.5791871049766436)\">[CLS]</span> the <span style=\"background-color:rgba(224,108,120,0.5834737695719576)\">music</span> was <span style=\"background-color:rgba(224,108,120,0.4327806125995057)\">too</span> <span style=\"background-color:rgba(88,117,220,1.0)\">loud</span> , and the <span style=\"background-color:rgba(88,117,220,0.6236991533178343)\">decorations</span> were taste ##less , but they had <span style=\"background-color:rgba(88,117,220,0.48892623629506404)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.7956085918122262)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.6396777279517661)\">delicious</span> <span style=\"background-color:rgba(88,117,220,0.6682189668347428)\">pasta</span> <span style=\"background-color:rgba(224,108,120,0.4829758588356244)\">[SEP]</span></p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.7614021517839192)\">[CLS]</span> the music was <span style=\"background-color:rgba(224,108,120,0.5092843935731525)\">too</span> <span style=\"background-color:rgba(224,108,120,0.6342833317445231)\">loud</span> , and the <span style=\"background-color:rgba(88,117,220,0.6465867990633256)\">decorations</span> were <span style=\"background-color:rgba(88,117,220,0.9222189292425717)\">taste</span> <span style=\"background-color:rgba(88,117,220,1.0)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.748178079043467)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.8448035404669969)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.9892006129346772)\">delicious</span> <span style=\"background-color:rgba(88,117,220,0.7890479809682556)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "service\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(88,117,220,0.5249118706486894)\">[CLS]</span> the music was too <span style=\"background-color:rgba(224,108,120,0.3386901929127272)\">loud</span> , and the <span style=\"background-color:rgba(224,108,120,0.6662489625073295)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.42523596359720434)\">taste</span> <span style=\"background-color:rgba(88,117,220,0.41051732872189384)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.6224246765432179)\">friendly</span> <span style=\"background-color:rgba(88,117,220,1.0)\">waiter</span> <span style=\"background-color:rgba(224,108,120,0.36462681034462535)\">##s</span> and <span style=\"background-color:rgba(88,117,220,0.8303134829560641)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.733171954608218)\">pasta</span> [SEP]</p><p>Saliency for control:</p><p><span style=\"background-color:rgba(88,117,220,0.7711150914828917)\">[CLS]</span> <span style=\"background-color:rgba(224,108,120,0.24203209713775245)\">the</span> music was too <span style=\"background-color:rgba(88,117,220,0.5227846449394798)\">loud</span> , and <span style=\"background-color:rgba(224,108,120,0.49804589343818945)\">the</span> <span style=\"background-color:rgba(88,117,220,0.7650143021874122)\">decorations</span> were <span style=\"background-color:rgba(88,117,220,0.9770078296024668)\">taste</span> <span style=\"background-color:rgba(224,108,120,1.0)\">##less</span> , but they had <span style=\"background-color:rgba(224,108,120,0.6979440838507994)\">friendly</span> <span style=\"background-color:rgba(224,108,120,0.8067935656394006)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.9659819291476077)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.7078215902605643)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "example_sentence = 'The music was too loud, and the decorations were tasteless, but they had friendly waiters and delicious pasta'\n",
    "aspects = ['ambiance', 'food', 'noise', 'service']\n",
    "\n",
    "for i, a in enumerate(aspects):\n",
    "    print(a)\n",
    "    explain_interventions_with_gradients(\n",
    "        example_sentence, \n",
    "        cpm_explainer, \n",
    "        i, \n",
    "        show_control=True,\n",
    "        normalize=True, \n",
    "        softmax_norm=False, \n",
    "        control_for_gradient=False, \n",
    "        saliency_fn=compute_guided_saliency,\n",
    "        grad_apply_fn=lambda x: x,\n",
    "        max_key_fn=np.abs,\n",
    "        display_fn=display_colored_text\n",
    "    )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "ambiance\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(88,117,220,0.5298616412566498)\">[CLS]</span> the <span style=\"background-color:rgba(88,117,220,0.46540892782544724)\">music</span> was <span style=\"background-color:rgba(224,108,120,0.8530173162733783)\">soothing</span> , and the <span style=\"background-color:rgba(224,108,120,1.0)\">decorations</span> were <span style=\"background-color:rgba(88,117,220,0.4661374768019608)\">pleasant</span> . i <span style=\"background-color:rgba(224,108,120,0.49020633777823786)\">especially</span> <span style=\"background-color:rgba(88,117,220,0.4563918742629308)\">liked</span> the friendly <span style=\"background-color:rgba(224,108,120,0.8505353686917045)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.591674191395838)\">delicious</span> <span style=\"background-color:rgba(88,117,220,0.5686423971430367)\">pasta</span> [SEP]</p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.8556546030259364)\">[CLS]</span> the <span style=\"background-color:rgba(224,108,120,0.7270105188389)\">music</span> was <span style=\"background-color:rgba(224,108,120,0.939276390180519)\">soothing</span> , and the <span style=\"background-color:rgba(224,108,120,0.8599747834540777)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.9754041728302886)\">pleasant</span> <span style=\"background-color:rgba(88,117,220,0.7777830547462159)\">.</span> i <span style=\"background-color:rgba(224,108,120,1.0)\">especially</span> <span style=\"background-color:rgba(224,108,120,0.7518467548730349)\">liked</span> the friendly waiter ##s and <span style=\"background-color:rgba(224,108,120,0.8477371868456712)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.6455564612306037)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "food\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(88,117,220,0.7160974324871323)\">[CLS]</span> the <span style=\"background-color:rgba(88,117,220,0.6571405172551411)\">music</span> was <span style=\"background-color:rgba(224,108,120,1.0)\">soothing</span> , and the <span style=\"background-color:rgba(224,108,120,0.9301030709604845)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.6439158327080876)\">pleasant</span> . i <span style=\"background-color:rgba(224,108,120,0.7420248098075847)\">especially</span> <span style=\"background-color:rgba(224,108,120,0.7518271352186122)\">liked</span> the friendly <span style=\"background-color:rgba(224,108,120,0.7181462009954801)\">waiter</span> ##s and <span style=\"background-color:rgba(88,117,220,0.6583414044880795)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.8785234171370315)\">pasta</span> [SEP]</p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.9409127687711645)\">[CLS]</span> the <span style=\"background-color:rgba(224,108,120,0.8375393978868315)\">music</span> was <span style=\"background-color:rgba(224,108,120,0.9131634395397576)\">soothing</span> , and the <span style=\"background-color:rgba(88,117,220,0.7502826978530754)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.9660323455014997)\">pleasant</span> <span style=\"background-color:rgba(88,117,220,0.8471502596277615)\">.</span> i <span style=\"background-color:rgba(224,108,120,1.0)\">especially</span> <span style=\"background-color:rgba(224,108,120,0.7456416941126489)\">liked</span> the friendly waiter ##s and <span style=\"background-color:rgba(224,108,120,0.8752135714005276)\">delicious</span> <span style=\"background-color:rgba(88,117,220,0.8856834236145804)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "noise\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(224,108,120,0.6595896490445521)\">[CLS]</span> the <span style=\"background-color:rgba(88,117,220,0.5803655835835241)\">music</span> was <span style=\"background-color:rgba(224,108,120,1.0)\">soothing</span> , and the <span style=\"background-color:rgba(224,108,120,0.81924816937086)\">decorations</span> were <span style=\"background-color:rgba(88,117,220,0.5886871319290518)\">pleasant</span> . i <span style=\"background-color:rgba(224,108,120,0.6067880560230654)\">especially</span> <span style=\"background-color:rgba(224,108,120,0.6168661163750839)\">liked</span> the friendly <span style=\"background-color:rgba(88,117,220,0.6883042297631832)\">waiter</span> ##s and <span style=\"background-color:rgba(224,108,120,0.8152292582656445)\">delicious</span> <span style=\"background-color:rgba(88,117,220,0.9140644991153382)\">pasta</span> [SEP]</p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.821791248854653)\">[CLS]</span> the music was <span style=\"background-color:rgba(224,108,120,0.7902753545114674)\">soothing</span> , <span style=\"background-color:rgba(88,117,220,0.44761000093246317)\">and</span> the <span style=\"background-color:rgba(88,117,220,0.6506654220455779)\">decorations</span> were <span style=\"background-color:rgba(224,108,120,0.7419461179524369)\">pleasant</span> <span style=\"background-color:rgba(88,117,220,0.9035016641817692)\">.</span> i <span style=\"background-color:rgba(224,108,120,1.0)\">especially</span> <span style=\"background-color:rgba(224,108,120,0.7007532325747915)\">liked</span> the friendly waiter ##s <span style=\"background-color:rgba(88,117,220,0.5929238220562345)\">and</span> <span style=\"background-color:rgba(224,108,120,0.6321240150231132)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.5852180022972111)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "service\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(224,108,120,0.582931911250477)\">[CLS]</span> the music was <span style=\"background-color:rgba(88,117,220,0.690836834772501)\">soothing</span> , and the <span style=\"background-color:rgba(224,108,120,0.818478208676408)\">decorations</span> were <span style=\"background-color:rgba(88,117,220,0.6241494908522716)\">pleasant</span> . i <span style=\"background-color:rgba(224,108,120,0.39364530139646153)\">especially</span> <span style=\"background-color:rgba(88,117,220,0.4662245599533055)\">liked</span> the <span style=\"background-color:rgba(88,117,220,0.4856010909622499)\">friendly</span> <span style=\"background-color:rgba(88,117,220,1.0)\">waiter</span> ##s and <span style=\"background-color:rgba(88,117,220,0.4031515620159326)\">delicious</span> <span style=\"background-color:rgba(224,108,120,0.5495322542035943)\">pasta</span> [SEP]</p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.7407533708546994)\">[CLS]</span> <span style=\"background-color:rgba(224,108,120,0.21266369062022933)\">the</span> music was <span style=\"background-color:rgba(224,108,120,0.41305882152929224)\">soothing</span> <span style=\"background-color:rgba(224,108,120,0.4213665168343563)\">,</span> and <span style=\"background-color:rgba(88,117,220,0.14868635508396144)\">the</span> <span style=\"background-color:rgba(88,117,220,0.47811904111292375)\">decorations</span> were <span style=\"background-color:rgba(88,117,220,0.5016825421408027)\">pleasant</span> <span style=\"background-color:rgba(224,108,120,1.0)\">.</span> i <span style=\"background-color:rgba(224,108,120,0.6156711889368806)\">especially</span> <span style=\"background-color:rgba(88,117,220,0.5158427440781242)\">liked</span> <span style=\"background-color:rgba(88,117,220,0.46549290164155355)\">the</span> friendly waiter ##s and delicious <span style=\"background-color:rgba(224,108,120,0.5026008141625163)\">pasta</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "example_sentence = 'The music was soothing, and the decorations were pleasant. I especially liked the friendly waiters and delicious pasta'\n",
    "aspects = ['ambiance', 'food', 'noise', 'service']\n",
    "\n",
    "for i, a in enumerate(aspects):\n",
    "    print(a)\n",
    "    explain_interventions_with_gradients(\n",
    "        example_sentence, \n",
    "        cpm_explainer, \n",
    "        i, \n",
    "        show_control=True,\n",
    "        normalize=False, \n",
    "        softmax_norm=False, \n",
    "        control_for_gradient=False, \n",
    "        saliency_fn=compute_guided_saliency,\n",
    "        grad_apply_fn=lambda x: x,\n",
    "        max_key_fn=np.abs,\n",
    "        display_fn=display_colored_text\n",
    "    )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "ambiance\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(224,108,120,0.5455735215422927)\">[CLS]</span> the <span style=\"background-color:rgba(88,117,220,0.6153982994833748)\">music</span> was <span style=\"background-color:rgba(88,117,220,0.937661559189108)\">bother</span> <span style=\"background-color:rgba(224,108,120,0.5365651387686922)\">##some</span> , and there were <span style=\"background-color:rgba(224,108,120,0.5045729186014838)\">no</span> <span style=\"background-color:rgba(224,108,120,0.9273014888656889)\">decorations</span> whatsoever . not to mention , the <span style=\"background-color:rgba(224,108,120,0.4753857672060733)\">waiter</span> ##s were <span style=\"background-color:rgba(88,117,220,0.5616499755922628)\">rude</span> and the <span style=\"background-color:rgba(88,117,220,1.0)\">pasta</span> was <span style=\"background-color:rgba(88,117,220,0.5041930194248414)\">cold</span> [SEP]</p><p>Saliency for control:</p><p><span style=\"background-color:rgba(88,117,220,0.4096787696856597)\">[CLS]</span> the <span style=\"background-color:rgba(224,108,120,0.35140867716615254)\">music</span> was <span style=\"background-color:rgba(224,108,120,1.0)\">bother</span> <span style=\"background-color:rgba(88,117,220,0.6055982956405254)\">##some</span> , and there were <span style=\"background-color:rgba(224,108,120,0.5392544527961648)\">no</span> <span style=\"background-color:rgba(88,117,220,0.5823799101059602)\">decorations</span> <span style=\"background-color:rgba(88,117,220,0.3313651441830264)\">whatsoever</span> . not to mention , the <span style=\"background-color:rgba(224,108,120,0.2536051277445861)\">waiter</span> ##s were <span style=\"background-color:rgba(224,108,120,0.297665430208242)\">rude</span> and the pasta was <span style=\"background-color:rgba(224,108,120,0.3110813932040846)\">cold</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "food\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(224,108,120,0.5106922510261772)\">[CLS]</span> the <span style=\"background-color:rgba(224,108,120,0.45202975911100873)\">music</span> was <span style=\"background-color:rgba(88,117,220,1.0)\">bother</span> <span style=\"background-color:rgba(224,108,120,0.5892057694882415)\">##some</span> , and there were no <span style=\"background-color:rgba(224,108,120,0.4645609158192154)\">decorations</span> whatsoever . not to mention , the <span style=\"background-color:rgba(88,117,220,0.6348926483395468)\">waiter</span> ##s were <span style=\"background-color:rgba(224,108,120,0.46979859860225265)\">rude</span> and the <span style=\"background-color:rgba(88,117,220,0.751558023127441)\">pasta</span> was <span style=\"background-color:rgba(224,108,120,0.4039441155191452)\">cold</span> <span style=\"background-color:rgba(224,108,120,0.3721923729556596)\">[SEP]</span></p><p>Saliency for control:</p><p><span style=\"background-color:rgba(224,108,120,0.4108297258123924)\">[CLS]</span> the <span style=\"background-color:rgba(224,108,120,0.3302336070273922)\">music</span> was <span style=\"background-color:rgba(224,108,120,1.0)\">bother</span> <span style=\"background-color:rgba(88,117,220,0.5877081202507262)\">##some</span> , and there were <span style=\"background-color:rgba(224,108,120,0.547844976974118)\">no</span> <span style=\"background-color:rgba(88,117,220,0.6190529326594265)\">decorations</span> <span style=\"background-color:rgba(224,108,120,0.35026690196845023)\">whatsoever</span> . not to mention , the waiter ##s were <span style=\"background-color:rgba(224,108,120,0.2875213261734627)\">rude</span> and the <span style=\"background-color:rgba(88,117,220,0.2696888012948763)\">pasta</span> was <span style=\"background-color:rgba(224,108,120,0.26237937506614545)\">cold</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "noise\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(224,108,120,0.49066414945125186)\">[CLS]</span> the <span style=\"background-color:rgba(88,117,220,0.5063366110482839)\">music</span> was <span style=\"background-color:rgba(88,117,220,0.8178433004715414)\">bother</span> <span style=\"background-color:rgba(88,117,220,0.5899537150667844)\">##some</span> , and there were no <span style=\"background-color:rgba(224,108,120,0.7798334006097791)\">decorations</span> whatsoever . not to <span style=\"background-color:rgba(224,108,120,0.38212067972141023)\">mention</span> , the <span style=\"background-color:rgba(224,108,120,0.5712962363299766)\">waiter</span> ##s were <span style=\"background-color:rgba(224,108,120,0.43890736134129194)\">rude</span> and the <span style=\"background-color:rgba(88,117,220,1.0)\">pasta</span> was <span style=\"background-color:rgba(88,117,220,0.6280438217180087)\">cold</span> [SEP]</p><p>Saliency for control:</p><p><span style=\"background-color:rgba(88,117,220,0.46193489073253274)\">[CLS]</span> the <span style=\"background-color:rgba(224,108,120,0.37660601798031074)\">music</span> was <span style=\"background-color:rgba(224,108,120,1.0)\">bother</span> <span style=\"background-color:rgba(88,117,220,0.6396205669795669)\">##some</span> , and there were <span style=\"background-color:rgba(224,108,120,0.5946535829557)\">no</span> <span style=\"background-color:rgba(88,117,220,0.6395508422622053)\">decorations</span> <span style=\"background-color:rgba(88,117,220,0.36966283737843614)\">whatsoever</span> . not to mention , the waiter ##s were <span style=\"background-color:rgba(224,108,120,0.3215033729543055)\">rude</span> and the <span style=\"background-color:rgba(88,117,220,0.2612413802185068)\">pasta</span> was <span style=\"background-color:rgba(224,108,120,0.311583414003537)\">cold</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "service\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div style=\"background-color:#FFFFFF;color:black\"><p>Saliency for CPM:</p><p><span style=\"background-color:rgba(224,108,120,0.44808938634112994)\">[CLS]</span> the <span style=\"background-color:rgba(224,108,120,0.3621656205838094)\">music</span> was <span style=\"background-color:rgba(88,117,220,0.3884872619882505)\">bother</span> ##some , and there were <span style=\"background-color:rgba(88,117,220,0.25029135845106903)\">no</span> <span style=\"background-color:rgba(224,108,120,0.4321294981269185)\">decorations</span> whatsoever . not to mention , the <span style=\"background-color:rgba(224,108,120,0.7399681417550933)\">waiter</span> ##s were <span style=\"background-color:rgba(88,117,220,0.2985443178494275)\">rude</span> and the <span style=\"background-color:rgba(224,108,120,1.0)\">pasta</span> was <span style=\"background-color:rgba(224,108,120,0.515049108920661)\">cold</span> <span style=\"background-color:rgba(88,117,220,0.5455386122633098)\">[SEP]</span></p><p>Saliency for control:</p><p><span style=\"background-color:rgba(88,117,220,0.4655118149189366)\">[CLS]</span> the <span style=\"background-color:rgba(224,108,120,0.343240397037684)\">music</span> was <span style=\"background-color:rgba(224,108,120,1.0)\">bother</span> <span style=\"background-color:rgba(88,117,220,0.6437851979729606)\">##some</span> , and there were <span style=\"background-color:rgba(224,108,120,0.5866899348581618)\">no</span> <span style=\"background-color:rgba(88,117,220,0.6416157746056007)\">decorations</span> <span style=\"background-color:rgba(224,108,120,0.3885491441035719)\">whatsoever</span> . not to mention , the <span style=\"background-color:rgba(224,108,120,0.26895234569754534)\">waiter</span> ##s were <span style=\"background-color:rgba(224,108,120,0.31154641355969215)\">rude</span> and the pasta was <span style=\"background-color:rgba(224,108,120,0.2778966680768601)\">cold</span> [SEP]</p></div>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "example_sentence = 'The music was bothersome, and there were no decorations whatsoever. Not to mention, the waiters were rude and the pasta was cold'\n",
    "aspects = ['ambiance', 'food', 'noise', 'service']\n",
    "\n",
    "for i, a in enumerate(aspects):\n",
    "    print(a)\n",
    "    explain_interventions_with_gradients(\n",
    "        example_sentence, \n",
    "        cpm_explainer, \n",
    "        i, \n",
    "        show_control=True,\n",
    "        normalize=True, \n",
    "        softmax_norm=False, \n",
    "        control_for_gradient=False, \n",
    "        saliency_fn=compute_guided_saliency,\n",
    "        grad_apply_fn=lambda x: x,\n",
    "        max_key_fn=np.abs,\n",
    "        display_fn=display_colored_text\n",
    "    )"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Integrated Gradients\n",
    "\n",
    "Running gradient-based visualizations for the CPM model, using `captum.ai`'s implementation of integrated gradients."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "from captum.attr import visualization as viz\n",
    "import torch.nn.functional as F"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Slightly modifying `LayerIntegratedGradients` to give us the opportunity of running gradients through the hidden layer of the CPM.\n",
    "This requires modifying the `_run_forward()` function so that it can return the hidden layers, and modifying the `gradient_func` used in the integrated gradients computation so that it computes the gradient through the aspect representations that we specify. \n",
    "Note that, at least with this implementation, computing the convergence delta won't work (but I don't think it should take too many changes to fix this)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "#!/usr/bin/env python3\n",
    "import functools\n",
    "import warnings\n",
    "from typing import Any, Callable, List, overload, Tuple, Union\n",
    "\n",
    "from captum._utils.common import (\n",
    "    _extract_device,\n",
    "    _format_additional_forward_args,\n",
    "    _format_outputs,\n",
    ")\n",
    "from captum._utils.gradient import _forward_layer_eval #, _run_forward\n",
    "#################### IMPORT MODULES ###########################\n",
    "from inspect import signature\n",
    "from captum._utils.common import _select_targets, _format_input\n",
    "###############################################################\n",
    "from captum._utils.typing import BaselineType, Literal, ModuleOrModuleList, TargetType\n",
    "from captum.attr._core.integrated_gradients import IntegratedGradients\n",
    "from captum.attr._utils.attribution import GradientAttribution, LayerAttribution\n",
    "from captum.attr._utils.common import (\n",
    "    _format_input_baseline,\n",
    "    _tensorize_baseline,\n",
    "    _validate_input,\n",
    ")\n",
    "from captum.log import log_usage\n",
    "from torch import Tensor\n",
    "from torch.nn.parallel.scatter_gather import scatter\n",
    "\n",
    "############### A FEW CONSTANTS (SHOULD LATER INTEGRATE AS ARGUMENTS) ####################\n",
    "INTERVENTION_H_DIM = 192\n",
    "MODEL_H_DIM = 768\n",
    "INTERCHANGE_LAYER = interchange_layer\n",
    "CONTROL = False\n",
    "\n",
    "####################### OVERRIDE RUN FORWARD FUNCTION ####################################\n",
    "def _run_forward(\n",
    "    forward_func: Callable,\n",
    "    inputs: Union[Tensor, Tuple[Tensor, ...]],\n",
    "    target: TargetType = None,\n",
    "    additional_forward_args: Any = None,\n",
    ") -> Tensor:\n",
    "    forward_func_args = signature(forward_func).parameters\n",
    "    if len(forward_func_args) == 0:\n",
    "        output = forward_func()\n",
    "        return output if target is None else _select_targets(output, target)\n",
    "\n",
    "    # make everything a tuple so that it is easy to unpack without\n",
    "    # using if-statements\n",
    "    inputs = _format_input(inputs)\n",
    "    additional_forward_args = _format_additional_forward_args(additional_forward_args)\n",
    "\n",
    "    output = forward_func(\n",
    "        *(*inputs, *additional_forward_args)\n",
    "        if additional_forward_args is not None\n",
    "        else inputs,\n",
    "    )\n",
    "    if CONTROL:\n",
    "        logits = output.logits # take first logit for overall sentiment\n",
    "    else:\n",
    "        logits = output.logits[0] # take first logit for overall sentiment\n",
    "    final_activation = output.hidden_states[INTERCHANGE_LAYER]\n",
    "    return _select_targets(logits, target), final_activation\n",
    "#########################################################################################\n",
    "\n",
    "\n",
    "class LayerIntegratedGradients(LayerAttribution, GradientAttribution):\n",
    "    r\"\"\"\n",
    "    Layer Integrated Gradients is a variant of Integrated Gradients that assigns\n",
    "    an importance score to layer inputs or outputs, depending on whether we\n",
    "    attribute to the former or to the latter one.\n",
    "\n",
    "    Integrated Gradients is an axiomatic model interpretability algorithm that\n",
    "    attributes / assigns an importance score to each input feature by approximating\n",
    "    the integral of gradients of the model's output with respect to the inputs\n",
    "    along the path (straight line) from given baselines / references to inputs.\n",
    "\n",
    "    Baselines can be provided as input arguments to attribute method.\n",
    "    To approximate the integral we can choose to use either a variant of\n",
    "    Riemann sum or Gauss-Legendre quadrature rule.\n",
    "\n",
    "    More details regarding the integrated gradients method can be found in the\n",
    "    original paper:\n",
    "    https://arxiv.org/abs/1703.01365\n",
    "\n",
    "    \"\"\"\n",
    "\n",
    "    def __init__(\n",
    "        self,\n",
    "        forward_func: Callable,\n",
    "        layer: ModuleOrModuleList,\n",
    "        device_ids: Union[None, List[int]] = None,\n",
    "        multiply_by_inputs: bool = True,\n",
    "    ) -> None:\n",
    "        r\"\"\"\n",
    "        Args:\n",
    "            forward_func (callable):  The forward function of the model or any\n",
    "                        modification of it\n",
    "            layer (ModuleOrModuleList):\n",
    "                        Layer or list of layers for which attributions are computed.\n",
    "                        For each layer the output size of the attribute matches\n",
    "                        this layer's input or output dimensions, depending on\n",
    "                        whether we attribute to the inputs or outputs of the\n",
    "                        layer, corresponding to the attribution of each neuron\n",
    "                        in the input or output of this layer.\n",
    "\n",
    "                        Please note that layers to attribute on cannot be\n",
    "                        dependent on each other. That is, a subset of layers in\n",
    "                        `layer` cannot produce the inputs for another layer.\n",
    "\n",
    "                        For example, if your model is of a simple linked-list\n",
    "                        based graph structure (think nn.Sequence), e.g. x -> l1\n",
    "                        -> l2 -> l3 -> output. If you pass in any one of those\n",
    "                        layers, you cannot pass in another due to the\n",
    "                        dependence, e.g.  if you pass in l2 you cannot pass in\n",
    "                        l1 or l3.\n",
    "\n",
    "            device_ids (list(int)): Device ID list, necessary only if forward_func\n",
    "                        applies a DataParallel model. This allows reconstruction of\n",
    "                        intermediate outputs from batched results across devices.\n",
    "                        If forward_func is given as the DataParallel model itself,\n",
    "                        then it is not necessary to provide this argument.\n",
    "            multiply_by_inputs (bool, optional): Indicates whether to factor\n",
    "                        model inputs' multiplier in the final attribution scores.\n",
    "                        In the literature this is also known as local vs global\n",
    "                        attribution. If inputs' multiplier isn't factored in,\n",
    "                        then this type of attribution method is also called local\n",
    "                        attribution. If it is, then that type of attribution\n",
    "                        method is called global.\n",
    "                        More detailed can be found here:\n",
    "                        https://arxiv.org/abs/1711.06104\n",
    "\n",
    "                        In case of layer integrated gradients, if `multiply_by_inputs`\n",
    "                        is set to True, final sensitivity scores are being multiplied by\n",
    "                        layer activations for inputs - layer activations for baselines.\n",
    "\n",
    "        \"\"\"\n",
    "        LayerAttribution.__init__(self, forward_func, layer, device_ids=device_ids)\n",
    "        GradientAttribution.__init__(self, forward_func)\n",
    "        self.ig = IntegratedGradients(forward_func, multiply_by_inputs)\n",
    "\n",
    "        if isinstance(layer, list) and len(layer) > 1:\n",
    "            warnings.warn(\n",
    "                \"Multiple layers provided. Please ensure that each layer is\"\n",
    "                \"**not** solely solely dependent on the outputs of\"\n",
    "                \"another layer. Please refer to the documentation for more\"\n",
    "                \"detail.\"\n",
    "            )\n",
    "\n",
    "    @overload\n",
    "    def attribute(\n",
    "        self,\n",
    "        inputs: Union[Tensor, Tuple[Tensor, ...]],\n",
    "        baselines: BaselineType,\n",
    "        target: TargetType,\n",
    "        additional_forward_args: Any,\n",
    "        n_steps: int,\n",
    "        method: str,\n",
    "        internal_batch_size: Union[None, int],\n",
    "        return_convergence_delta: Literal[False],\n",
    "        attribute_to_layer_input: bool,\n",
    "    ) -> Union[Tensor, Tuple[Tensor, ...], List[Union[Tensor, Tuple[Tensor, ...]]]]:\n",
    "        ...\n",
    "\n",
    "    @overload\n",
    "    def attribute(\n",
    "        self,\n",
    "        inputs: Union[Tensor, Tuple[Tensor, ...]],\n",
    "        baselines: BaselineType,\n",
    "        target: TargetType,\n",
    "        additional_forward_args: Any,\n",
    "        n_steps: int,\n",
    "        method: str,\n",
    "        internal_batch_size: Union[None, int],\n",
    "        return_convergence_delta: Literal[True],\n",
    "        attribute_to_layer_input: bool,\n",
    "    ) -> Tuple[\n",
    "        Union[Tensor, Tuple[Tensor, ...], List[Union[Tensor, Tuple[Tensor, ...]]]],\n",
    "        Tensor,\n",
    "    ]:\n",
    "        ...\n",
    "\n",
    "    @overload\n",
    "    def attribute(\n",
    "        self,\n",
    "        inputs: Union[Tensor, Tuple[Tensor, ...]],\n",
    "        baselines: BaselineType = None,\n",
    "        target: TargetType = None,\n",
    "        additional_forward_args: Any = None,\n",
    "        n_steps: int = 50,\n",
    "        method: str = \"gausslegendre\",\n",
    "        internal_batch_size: Union[None, int] = None,\n",
    "        return_convergence_delta: bool = False,\n",
    "        attribute_to_layer_input: bool = False,\n",
    "    ) -> Union[\n",
    "        Union[Tensor, Tuple[Tensor, ...], List[Union[Tensor, Tuple[Tensor, ...]]]],\n",
    "        Tuple[\n",
    "            Union[Tensor, Tuple[Tensor, ...], List[Union[Tensor, Tuple[Tensor, ...]]]],\n",
    "            Tensor,\n",
    "        ],\n",
    "    ]:\n",
    "        ...\n",
    "\n",
    "    @log_usage()\n",
    "    def attribute(\n",
    "        self,\n",
    "        inputs: Union[Tensor, Tuple[Tensor, ...]],\n",
    "        baselines: BaselineType = None,\n",
    "        target: TargetType = None,\n",
    "        additional_forward_args: Any = None,\n",
    "        n_steps: int = 50,\n",
    "        method: str = \"gausslegendre\",\n",
    "        internal_batch_size: Union[None, int] = None,\n",
    "        return_convergence_delta: bool = False,\n",
    "        attribute_to_layer_input: bool = False,\n",
    "        ############## NEW ARGUMENT: ASPECT OF INTEREST ############################\n",
    "        aspect: Union[int, None] = None\n",
    "        ############################################################################\n",
    "    ) -> Union[\n",
    "        Union[Tensor, Tuple[Tensor, ...], List[Union[Tensor, Tuple[Tensor, ...]]]],\n",
    "        Tuple[\n",
    "            Union[Tensor, Tuple[Tensor, ...], List[Union[Tensor, Tuple[Tensor, ...]]]],\n",
    "            Tensor,\n",
    "        ],\n",
    "    ]:\n",
    "        r\"\"\"\n",
    "        This method attributes the output of the model with given target index\n",
    "        (in case it is provided, otherwise it assumes that output is a\n",
    "        scalar) to layer inputs or outputs of the model, depending on whether\n",
    "        `attribute_to_layer_input` is set to True or False, using the approach\n",
    "        described above.\n",
    "\n",
    "        In addition to that it also returns, if `return_convergence_delta` is\n",
    "        set to True, integral approximation delta based on the completeness\n",
    "        property of integrated gradients.\n",
    "\n",
    "        Args:\n",
    "\n",
    "            inputs (tensor or tuple of tensors):  Input for which layer integrated\n",
    "                        gradients are computed. If forward_func takes a single\n",
    "                        tensor as input, a single input tensor should be provided.\n",
    "                        If forward_func takes multiple tensors as input, a tuple\n",
    "                        of the input tensors should be provided. It is assumed\n",
    "                        that for all given input tensors, dimension 0 corresponds\n",
    "                        to the number of examples, and if multiple input tensors\n",
    "                        are provided, the examples must be aligned appropriately.\n",
    "            baselines (scalar, tensor, tuple of scalars or tensors, optional):\n",
    "                        Baselines define the starting point from which integral\n",
    "                        is computed and can be provided as:\n",
    "\n",
    "                        - a single tensor, if inputs is a single tensor, with\n",
    "                          exactly the same dimensions as inputs or the first\n",
    "                          dimension is one and the remaining dimensions match\n",
    "                          with inputs.\n",
    "\n",
    "                        - a single scalar, if inputs is a single tensor, which will\n",
    "                          be broadcasted for each input value in input tensor.\n",
    "\n",
    "                        - a tuple of tensors or scalars, the baseline corresponding\n",
    "                          to each tensor in the inputs' tuple can be:\n",
    "                            - either a tensor with matching dimensions to\n",
    "                              corresponding tensor in the inputs' tuple\n",
    "                              or the first dimension is one and the remaining\n",
    "                              dimensions match with the corresponding\n",
    "                              input tensor.\n",
    "                            - or a scalar, corresponding to a tensor in the\n",
    "                              inputs' tuple. This scalar value is broadcasted\n",
    "                              for corresponding input tensor.\n",
    "\n",
    "                        In the cases when `baselines` is not provided, we internally\n",
    "                        use zero scalar corresponding to each input tensor.\n",
    "\n",
    "                        Default: None\n",
    "            target (int, tuple, tensor or list, optional):  Output indices for\n",
    "                        which gradients are computed (for classification cases,\n",
    "                        this is usually the target class).\n",
    "                        If the network returns a scalar value per example,\n",
    "                        no target index is necessary.\n",
    "                        For general 2D outputs, targets can be either:\n",
    "\n",
    "                        - a single integer or a tensor containing a single\n",
    "                          integer, which is applied to all input examples\n",
    "\n",
    "                        - a list of integers or a 1D tensor, with length matching\n",
    "                          the number of examples in inputs (dim 0). Each integer\n",
    "                          is applied as the target for the corresponding example.\n",
    "\n",
    "                        For outputs with > 2 dimensions, targets can be either:\n",
    "\n",
    "                        - A single tuple, which contains #output_dims - 1\n",
    "                          elements. This target index is applied to all examples.\n",
    "\n",
    "                        - A list of tuples with length equal to the number of\n",
    "                          examples in inputs (dim 0), and each tuple containing\n",
    "                          #output_dims - 1 elements. Each tuple is applied as the\n",
    "                          target for the corresponding example.\n",
    "\n",
    "                        Default: None\n",
    "            additional_forward_args (any, optional): If the forward function\n",
    "                        requires additional arguments other than the inputs for\n",
    "                        which attributions should not be computed, this argument\n",
    "                        can be provided. It must be either a single additional\n",
    "                        argument of a Tensor or arbitrary (non-tuple) type or a\n",
    "                        tuple containing multiple additional arguments including\n",
    "                        tensors or any arbitrary python types. These arguments\n",
    "                        are provided to forward_func in order following the\n",
    "                        arguments in inputs.\n",
    "                        For a tensor, the first dimension of the tensor must\n",
    "                        correspond to the number of examples. It will be\n",
    "                        repeated for each of `n_steps` along the integrated\n",
    "                        path. For all other types, the given argument is used\n",
    "                        for all forward evaluations.\n",
    "                        Note that attributions are not computed with respect\n",
    "                        to these arguments.\n",
    "                        Default: None\n",
    "            n_steps (int, optional): The number of steps used by the approximation\n",
    "                        method. Default: 50.\n",
    "            method (string, optional): Method for approximating the integral,\n",
    "                        one of `riemann_right`, `riemann_left`, `riemann_middle`,\n",
    "                        `riemann_trapezoid` or `gausslegendre`.\n",
    "                        Default: `gausslegendre` if no method is provided.\n",
    "            internal_batch_size (int, optional): Divides total #steps * #examples\n",
    "                        data points into chunks of size at most internal_batch_size,\n",
    "                        which are computed (forward / backward passes)\n",
    "                        sequentially. internal_batch_size must be at least equal to\n",
    "                        #examples.\n",
    "                        For DataParallel models, each batch is split among the\n",
    "                        available devices, so evaluations on each available\n",
    "                        device contain internal_batch_size / num_devices examples.\n",
    "                        If internal_batch_size is None, then all evaluations are\n",
    "                        processed in one batch.\n",
    "                        Default: None\n",
    "            return_convergence_delta (bool, optional): Indicates whether to return\n",
    "                        convergence delta or not. If `return_convergence_delta`\n",
    "                        is set to True convergence delta will be returned in\n",
    "                        a tuple following attributions.\n",
    "                        Default: False\n",
    "            attribute_to_layer_input (bool, optional): Indicates whether to\n",
    "                        compute the attribution with respect to the layer input\n",
    "                        or output. If `attribute_to_layer_input` is set to True\n",
    "                        then the attributions will be computed with respect to\n",
    "                        layer input, otherwise it will be computed with respect\n",
    "                        to layer output.\n",
    "                        Note that currently it is assumed that either the input\n",
    "                        or the output of internal layer, depending on whether we\n",
    "                        attribute to the input or output, is a single tensor.\n",
    "                        Support for multiple tensors will be added later.\n",
    "                        Default: False\n",
    "            Returns:\n",
    "                **attributions** or 2-element tuple of **attributions**, **delta**:\n",
    "                - **attributions** (*tensor*, tuple of *tensors* or tuple of *tensors*):\n",
    "                        Integrated gradients with respect to `layer`'s inputs or\n",
    "                        outputs. Attributions will always be the same size and\n",
    "                        dimensionality as the input or output of the given layer,\n",
    "                        depending on whether we attribute to the inputs or outputs\n",
    "                        of the layer which is decided by the input flag\n",
    "                        `attribute_to_layer_input`.\n",
    "\n",
    "                        For a single layer, attributions are returned in a tuple if\n",
    "                        the layer inputs / outputs contain multiple tensors,\n",
    "                        otherwise a single tensor is returned.\n",
    "\n",
    "                        For multiple layers, attributions will always be\n",
    "                        returned as a list. Each element in this list will be\n",
    "                        equivalent to that of a single layer output, i.e. in the\n",
    "                        case that one layer, in the given layers, inputs / outputs\n",
    "                        multiple tensors: the corresponding output element will be\n",
    "                        a tuple of tensors. The ordering of the outputs will be\n",
    "                        the same order as the layers given in the constructor.\n",
    "                - **delta** (*tensor*, returned if return_convergence_delta=True):\n",
    "                        The difference between the total approximated and true\n",
    "                        integrated gradients. This is computed using the property\n",
    "                        that the total sum of forward_func(inputs) -\n",
    "                        forward_func(baselines) must equal the total sum of the\n",
    "                        integrated gradient.\n",
    "                        Delta is calculated per example, meaning that the number of\n",
    "                        elements in returned delta tensor is equal to the number of\n",
    "                        of examples in inputs.\n",
    "\n",
    "            Examples::\n",
    "\n",
    "                >>> # ImageClassifier takes a single input tensor of images Nx3x32x32,\n",
    "                >>> # and returns an Nx10 tensor of class probabilities.\n",
    "                >>> # It contains an attribute conv1, which is an instance of nn.conv2d,\n",
    "                >>> # and the output of this layer has dimensions Nx12x32x32.\n",
    "                >>> net = ImageClassifier()\n",
    "                >>> lig = LayerIntegratedGradients(net, net.conv1)\n",
    "                >>> input = torch.randn(2, 3, 32, 32, requires_grad=True)\n",
    "                >>> # Computes layer integrated gradients for class 3.\n",
    "                >>> # attribution size matches layer output, Nx12x32x32\n",
    "                >>> attribution = lig.attribute(input, target=3)\n",
    "        \"\"\"\n",
    "        inps, baselines = _format_input_baseline(inputs, baselines)\n",
    "        _validate_input(inps, baselines, n_steps, method)\n",
    "\n",
    "        baselines = _tensorize_baseline(inps, baselines)\n",
    "        additional_forward_args = _format_additional_forward_args(\n",
    "            additional_forward_args\n",
    "        )\n",
    "\n",
    "        def flatten_tuple(tup):\n",
    "            return tuple(\n",
    "                sum((list(x) if isinstance(x, (tuple, list)) else [x] for x in tup), [])\n",
    "            )\n",
    "\n",
    "        if self.device_ids is None:\n",
    "            self.device_ids = getattr(self.forward_func, \"device_ids\", None)\n",
    "\n",
    "        inputs_layer = _forward_layer_eval(\n",
    "            self.forward_func,\n",
    "            inps,\n",
    "            self.layer,\n",
    "            device_ids=self.device_ids,\n",
    "            additional_forward_args=additional_forward_args,\n",
    "            attribute_to_layer_input=attribute_to_layer_input,\n",
    "        )\n",
    "\n",
    "        # if we have one output\n",
    "        if not isinstance(self.layer, list):\n",
    "            inputs_layer = (inputs_layer,)\n",
    "\n",
    "        num_outputs = [1 if isinstance(x, Tensor) else len(x) for x in inputs_layer]\n",
    "        num_outputs_cumsum = torch.cumsum(\n",
    "            torch.IntTensor([0] + num_outputs), dim=0  # type: ignore\n",
    "        )\n",
    "        inputs_layer = flatten_tuple(inputs_layer)\n",
    "\n",
    "        baselines_layer = _forward_layer_eval(\n",
    "            self.forward_func,\n",
    "            baselines,\n",
    "            self.layer,\n",
    "            device_ids=self.device_ids,\n",
    "            additional_forward_args=additional_forward_args,\n",
    "            attribute_to_layer_input=attribute_to_layer_input,\n",
    "        )\n",
    "        baselines_layer = flatten_tuple(baselines_layer)\n",
    "\n",
    "        # inputs -> these inputs are scaled\n",
    "        def gradient_func(\n",
    "            forward_fn: Callable,\n",
    "            inputs: Union[Tensor, Tuple[Tensor, ...]],\n",
    "            target_ind: TargetType = None,\n",
    "            additional_forward_args: Any = None,\n",
    "        ) -> Tuple[Tensor, ...]:\n",
    "            if self.device_ids is None or len(self.device_ids) == 0:\n",
    "                scattered_inputs = (inputs,)\n",
    "            else:\n",
    "                # scatter method does not have a precise enough return type in its\n",
    "                # stub, so suppress the type warning.\n",
    "                scattered_inputs = scatter(  # type:ignore\n",
    "                    inputs, target_gpus=self.device_ids\n",
    "                )\n",
    "\n",
    "            scattered_inputs_dict = {\n",
    "                scattered_input[0].device: scattered_input\n",
    "                for scattered_input in scattered_inputs\n",
    "            }\n",
    "\n",
    "            with torch.autograd.set_grad_enabled(True):\n",
    "\n",
    "                def layer_forward_hook(\n",
    "                    module, hook_inputs, hook_outputs=None, layer_idx=0\n",
    "                ):\n",
    "                    device = _extract_device(module, hook_inputs, hook_outputs)\n",
    "                    is_layer_tuple = (\n",
    "                        isinstance(hook_outputs, tuple)\n",
    "                        # hook_outputs is None if attribute_to_layer_input == True\n",
    "                        if hook_outputs is not None\n",
    "                        else isinstance(hook_inputs, tuple)\n",
    "                    )\n",
    "\n",
    "                    if is_layer_tuple:\n",
    "                        return scattered_inputs_dict[device][\n",
    "                            num_outputs_cumsum[layer_idx] : num_outputs_cumsum[\n",
    "                                layer_idx + 1\n",
    "                            ]\n",
    "                        ]\n",
    "\n",
    "                    return scattered_inputs_dict[device][num_outputs_cumsum[layer_idx]]\n",
    "\n",
    "                hooks = []\n",
    "                try:\n",
    "\n",
    "                    layers = self.layer\n",
    "                    if not isinstance(layers, list):\n",
    "                        layers = [self.layer]\n",
    "\n",
    "                    for layer_idx, layer in enumerate(layers):\n",
    "                        hook = None\n",
    "                        # TODO:\n",
    "                        # Allow multiple attribute_to_layer_input flags for\n",
    "                        # each layer, i.e. attribute_to_layer_input[layer_idx]\n",
    "                        if attribute_to_layer_input:\n",
    "                            hook = layer.register_forward_pre_hook(\n",
    "                                functools.partial(\n",
    "                                    layer_forward_hook, layer_idx=layer_idx\n",
    "                                )\n",
    "                            )\n",
    "                        else:\n",
    "                            hook = layer.register_forward_hook(\n",
    "                                functools.partial(\n",
    "                                    layer_forward_hook, layer_idx=layer_idx\n",
    "                                )\n",
    "                            )\n",
    "\n",
    "                        hooks.append(hook)\n",
    "\n",
    "                    output = _run_forward(\n",
    "                        self.forward_func, tuple(), target_ind, additional_forward_args\n",
    "                    )\n",
    "                finally:\n",
    "                    for hook in hooks:\n",
    "                        if hook is not None:\n",
    "                            hook.remove()\n",
    "\n",
    "                # assert output[0].numel() == 1, (\n",
    "                #     \"Target not provided when necessary, cannot\"\n",
    "                #     \" take gradient with respect to multiple outputs.\"\n",
    "                # )\n",
    "                # torch.unbind(forward_out) is a list of scalar tensor tuples and\n",
    "                # contains batch_size * #steps elements\n",
    "                # grads = torch.autograd.grad(torch.unbind(output), inputs)\n",
    "\n",
    "                ##################### TAKE GRADIENT THROUGH ACTIVATION #########################\n",
    "                model_output, final_layer_activation = output\n",
    "\n",
    "                # get aspect representation\n",
    "                if aspect is not None:\n",
    "                    start_idx = aspect * INTERVENTION_H_DIM\n",
    "                    end_idx = (aspect+1) * INTERVENTION_H_DIM\n",
    "                else:\n",
    "                    start_idx = 0\n",
    "                    # end_idx = model_output.hidden_states[-1].size(2)\n",
    "                    end_idx = MODEL_H_DIM\n",
    "\n",
    "                # compute backwards gradient from output logits to final layer\n",
    "                final_layer_gradient = torch.autograd.grad(\n",
    "                    outputs=torch.unbind(model_output),\n",
    "                    inputs=final_layer_activation,\n",
    "                    retain_graph=True\n",
    "                )\n",
    "\n",
    "                final_layer_gradient = torch.stack(final_layer_gradient).squeeze(0).clone().detach()\n",
    "\n",
    "                # reset gradients ??\n",
    "\n",
    "                # compute backwards gradient from final layer to input layer,\n",
    "                # guided by the gradient from the logits to the final layer\n",
    "                mask = torch.zeros_like(final_layer_activation)\n",
    "                mask[:, 0, start_idx:end_idx] = final_layer_gradient[:, 0, start_idx:end_idx]\n",
    "\n",
    "                grads = torch.autograd.grad(\n",
    "                    outputs=torch.unbind(final_layer_activation),\n",
    "                    inputs=inputs,\n",
    "                    grad_outputs=torch.unbind(mask)\n",
    "                )\n",
    "                ###############################################################################\n",
    "\n",
    "            return grads\n",
    "\n",
    "        self.ig.gradient_func = gradient_func\n",
    "        all_inputs = (\n",
    "            (inps + additional_forward_args)\n",
    "            if additional_forward_args is not None\n",
    "            else inps\n",
    "        )\n",
    "\n",
    "        attributions = self.ig.attribute.__wrapped__(  # type: ignore\n",
    "            self.ig,  # self\n",
    "            inputs_layer,\n",
    "            baselines=baselines_layer,\n",
    "            target=target,\n",
    "            additional_forward_args=all_inputs,\n",
    "            n_steps=n_steps,\n",
    "            method=method,\n",
    "            internal_batch_size=internal_batch_size,\n",
    "            return_convergence_delta=False,\n",
    "        )\n",
    "\n",
    "        # handle multiple outputs\n",
    "        output: List[Tuple[Tensor, ...]] = [\n",
    "            tuple(\n",
    "                attributions[\n",
    "                    int(num_outputs_cumsum[i]) : int(num_outputs_cumsum[i + 1])\n",
    "                ]\n",
    "            )\n",
    "            for i in range(len(num_outputs))\n",
    "        ]\n",
    "\n",
    "        if return_convergence_delta:\n",
    "            start_point, end_point = baselines, inps\n",
    "            # computes approximation error based on the completeness axiom\n",
    "            delta = self.compute_convergence_delta(\n",
    "                attributions,\n",
    "                start_point,\n",
    "                end_point,\n",
    "                additional_forward_args=additional_forward_args,\n",
    "                target=target,\n",
    "            )\n",
    "            return _format_outputs(isinstance(self.layer, list), output), delta\n",
    "        return _format_outputs(isinstance(self.layer, list), output)\n",
    "\n",
    "\n",
    "    def has_convergence_delta(self) -> bool:\n",
    "        return True\n",
    "\n",
    "    @property\n",
    "    def multiplies_by_inputs(self):\n",
    "        return self.ig.multiplies_by_inputs"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Code taken from CS 224u that uses `LayerIntegratedGradients` to compute integrated gradients for a BERT model.\n",
    "Note that we will need to modify this code slightly for other models."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "def hf_predict_one_proba(text, tokenizer, model):\n",
    "    input_ids = tokenizer.encode(\n",
    "        text, add_special_tokens=True, return_tensors='pt').to(model.device)\n",
    "    model.eval()\n",
    "    with torch.no_grad():\n",
    "        logits = model(input_ids).logits\n",
    "        # case when cpm returns multiple predictions, take the first\n",
    "        # (corresponding to overall sentiment)\n",
    "        if isinstance(logits, tuple):\n",
    "            logits = logits[0]\n",
    "        preds = F.softmax(logits, dim=1)\n",
    "    model.train()\n",
    "    return preds.squeeze(0)\n",
    "\n",
    "\n",
    "def hf_ig_encodings(text, tokenizer):\n",
    "    pad_id = tokenizer.pad_token_id\n",
    "    cls_id = tokenizer.cls_token_id\n",
    "    sep_id = tokenizer.sep_token_id\n",
    "    input_ids = tokenizer.encode(text, add_special_tokens=False)\n",
    "    base_ids = [pad_id] * len(input_ids)\n",
    "    input_ids = [cls_id] + input_ids + [sep_id]\n",
    "    base_ids = [cls_id] + base_ids + [sep_id]\n",
    "    return torch.LongTensor([input_ids]), torch.LongTensor([base_ids])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "ASPECTS = ['ambiance', 'food', 'noise', 'service']\n",
    "asp_to_ind = {\n",
    "    'overall': None,\n",
    "    **({a: i for i, a in enumerate(ASPECTS)})\n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "def hf_ig_analyses(inputs, targets, tokenizer, model):\n",
    "    data = []\n",
    "    for text, true_class in zip(inputs, targets):\n",
    "        # create visualization for each aspect \n",
    "        for aspect, _ in enumerate(ASPECTS):\n",
    "            score_vis = hf_ig_analysis_one(text, true_class, tokenizer, model, aspect=aspect)\n",
    "            data.append(score_vis)\n",
    "    html = viz.visualize_text(data)\n",
    "\n",
    "def hf_ig_analyses_new(inputs, targets, tokenizer, model):\n",
    "    data = []\n",
    "    for text in inputs:\n",
    "        # create visualization for each aspect \n",
    "        for aspect, _ in enumerate(ASPECTS):\n",
    "            score_vis = hf_ig_analyses_multiple(text, targets, tokenizer, model, aspect=aspect)\n",
    "            data.append(score_vis)\n",
    "    html = viz.visualize_text(data)\n",
    "    \n",
    "def hf_ig_analyses_multiple(text, true_classes, tokenizer, model, device=0, aspect=None):\n",
    "    \n",
    "    scores_for_true_classes = []\n",
    "    for true_class in true_classes:\n",
    "        # Option to look at different layers:\n",
    "        # layer = model.roberta.encoder.layer[0]\n",
    "        # layer = model.roberta.embeddings.word_embeddings\n",
    "        layer = model.bert.embeddings\n",
    "\n",
    "        def ig_forward(inputs):\n",
    "            # logits at first index, cpm returns multiple logits\n",
    "            # logits = model(inputs).logits\n",
    "            # return logits[pred_ind] if isinstance(logits, tuple) else logits\n",
    "            return model(inputs, output_hidden_states=True)\n",
    "\n",
    "        ig = LayerIntegratedGradients(ig_forward, layer)\n",
    "\n",
    "        input_ids, base_ids = hf_ig_encodings(text, tokenizer)\n",
    "\n",
    "        attrs = ig.attribute(\n",
    "            input_ids.to(model.device),\n",
    "            base_ids.to(model.device),\n",
    "            target=true_class,\n",
    "            return_convergence_delta=False,\n",
    "            attribute_to_layer_input=False,\n",
    "            aspect=aspect\n",
    "        )\n",
    "\n",
    "        # Summarize and z-score normalize the attributions\n",
    "        # for each representation in `layer`:\n",
    "        scores = attrs.sum(dim=-1).squeeze(0)\n",
    "        scores_for_true_classes += [scores]\n",
    "    scores_for_true_classes = torch.stack(scores_for_true_classes, dim=0).sum(dim=0)\n",
    "    scores = scores_for_true_classes\n",
    "    scores = (scores - scores.mean()) / scores.norm()\n",
    "\n",
    "    # Intuitive tokens to help with analysis:\n",
    "    raw_input = tokenizer.convert_ids_to_tokens(input_ids.tolist()[0])\n",
    "    # RoBERTa-specific clean-up:\n",
    "    # raw_input = [x.strip(\"Ġ\") for x in raw_input]\n",
    "\n",
    "    # Predictions for comparisons:\n",
    "    pred_probs = hf_predict_one_proba(text, tokenizer, model)\n",
    "    pred_class = pred_probs.argmax()\n",
    "\n",
    "    score_vis = viz.VisualizationDataRecord(\n",
    "        word_attributions=scores,\n",
    "        pred_prob=pred_probs.max(),\n",
    "        pred_class=pred_class,\n",
    "        true_class=true_class,\n",
    "        # override attribution label with aspect label, to make the visualization clearer\n",
    "        attr_class=ASPECTS[aspect] if aspect is not None else None,\n",
    "        attr_score=attrs.sum(),\n",
    "        raw_input_ids=raw_input,\n",
    "        convergence_score=None)\n",
    "    \n",
    "    max_score = max([abs(score) for score in scores.tolist()])\n",
    "    mag = 1.0/max_score\n",
    "    word_idx = 0\n",
    "    latex_str = \"\"\n",
    "    for score in scores.tolist():\n",
    "        sign = \"green\" if score > 0 else \"red\"\n",
    "        color_mag = int(abs(score*mag)*100)\n",
    "        word = raw_input[word_idx]\n",
    "        latex_str += \"\\\\colorbox{\"+sign+\"!\"+str(color_mag)+\"}{\\\\strut \"+word+\"}\"\n",
    "        word_idx += 1\n",
    "        \n",
    "    print(latex_str)\n",
    "    print()\n",
    "    \n",
    "    return score_vis\n",
    "    \n",
    "def hf_ig_analysis_one(text, true_class, tokenizer, model, device=0, aspect=None):\n",
    "    # Option to look at different layers:\n",
    "    # layer = model.roberta.encoder.layer[0]\n",
    "    # layer = model.roberta.embeddings.word_embeddings\n",
    "    layer = model.bert.embeddings\n",
    "\n",
    "    def ig_forward(inputs):\n",
    "        # logits at first index, cpm returns multiple logits\n",
    "        # logits = model(inputs).logits\n",
    "        # return logits[pred_ind] if isinstance(logits, tuple) else logits\n",
    "        return model(inputs, output_hidden_states=True)\n",
    "\n",
    "    ig = LayerIntegratedGradients(ig_forward, layer)\n",
    "\n",
    "    input_ids, base_ids = hf_ig_encodings(text, tokenizer)\n",
    "\n",
    "    attrs = ig.attribute(\n",
    "        input_ids.to(model.device),\n",
    "        base_ids.to(model.device),\n",
    "        target=true_class,\n",
    "        return_convergence_delta=False,\n",
    "        attribute_to_layer_input=False,\n",
    "        aspect=aspect\n",
    "    )\n",
    "\n",
    "    # Summarize and z-score normalize the attributions\n",
    "    # for each representation in `layer`:\n",
    "    scores = attrs.sum(dim=-1).squeeze(0)\n",
    "    scores = (scores - scores.mean()) / scores.norm()\n",
    "    \n",
    "    # Intuitive tokens to help with analysis:\n",
    "    raw_input = tokenizer.convert_ids_to_tokens(input_ids.tolist()[0])\n",
    "    # RoBERTa-specific clean-up:\n",
    "    # raw_input = [x.strip(\"Ġ\") for x in raw_input]\n",
    "\n",
    "    # Predictions for comparisons:\n",
    "    pred_probs = hf_predict_one_proba(text, tokenizer, model)\n",
    "    pred_class = pred_probs.argmax()\n",
    "\n",
    "    score_vis = viz.VisualizationDataRecord(\n",
    "        word_attributions=scores,\n",
    "        pred_prob=pred_probs.max(),\n",
    "        pred_class=pred_class,\n",
    "        true_class=true_class,\n",
    "        # override attribution label with aspect label, to make the visualization clearer\n",
    "        attr_class=ASPECTS[aspect] if aspect is not None else None,\n",
    "        attr_score=attrs.sum(),\n",
    "        raw_input_ids=raw_input,\n",
    "        convergence_score=None)\n",
    "    \n",
    "    max_score = max([abs(score) for score in scores.tolist()])\n",
    "    mag = 1.0/max_score\n",
    "    word_idx = 0\n",
    "    latex_str = \"\"\n",
    "    for score in scores.tolist():\n",
    "        sign = \"green\" if score > 0 else \"red\"\n",
    "        color_mag = int(abs(score*mag)*100)\n",
    "        word = raw_input[word_idx]\n",
    "        latex_str += \"\\\\colorbox{\"+sign+\"!\"+str(color_mag)+\"}{\\\\strut \"+word+\"}\"\n",
    "        word_idx += 1\n",
    "        \n",
    "    print(latex_str)\n",
    "    print()\n",
    "    \n",
    "    return score_vis"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Create integrated gradients visualizations!"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "# example_sentences = [\n",
    "#     # all negative\n",
    "#     \"Horrible experience! Distasteful decorations, slow service, bland food, and all the while with music blasting from the walls\",\n",
    "#     # all neutral (I tried?)\n",
    "#     \"The decorations were okay, the servers were meh, the pasta so-so, and the music not my taste, but also not bad\",\n",
    "#     # all positive\n",
    "#     \"Wonderful place all around! Kind waiters, good vibes, and pleasant music - not to mention the best sushi I ever had\",\n",
    "#     # some negative, some positive\n",
    "#     \"Although the decorations were tasteless and the music was far too loud, the food arrived quickly, and the dessert was incredible\",\n",
    "#     # some negative, some unmentioned\n",
    "#     \"Tacky decorations, deafening music :(\",\n",
    "#     # some positive, some unmentioned\n",
    "#     \"Kind waiters, awesome fish tacos!\"\n",
    "# ]\n",
    "\n",
    "# labels = [0, 2, 4, 3, 0, 4]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "metadata": {},
   "outputs": [],
   "source": [
    "example_sentences = [\n",
    "    'The music was too loud, and the decorations were tasteless, but they had friendly waiters and delicious pasta'\n",
    "]\n",
    "\n",
    "labels = [3, 4]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 37,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\\colorbox{red!13}{\\strut [CLS]}\\colorbox{red!11}{\\strut the}\\colorbox{red!12}{\\strut music}\\colorbox{red!14}{\\strut was}\\colorbox{green!61}{\\strut too}\\colorbox{green!42}{\\strut loud}\\colorbox{red!27}{\\strut ,}\\colorbox{red!6}{\\strut and}\\colorbox{red!3}{\\strut the}\\colorbox{red!12}{\\strut decorations}\\colorbox{red!4}{\\strut were}\\colorbox{red!36}{\\strut taste}\\colorbox{green!39}{\\strut ##less}\\colorbox{red!23}{\\strut ,}\\colorbox{red!0}{\\strut but}\\colorbox{red!23}{\\strut they}\\colorbox{red!33}{\\strut had}\\colorbox{green!41}{\\strut friendly}\\colorbox{red!2}{\\strut waiter}\\colorbox{red!17}{\\strut ##s}\\colorbox{red!3}{\\strut and}\\colorbox{green!100}{\\strut delicious}\\colorbox{red!22}{\\strut pasta}\\colorbox{red!13}{\\strut [SEP]}\n",
      "\n",
      "\\colorbox{red!29}{\\strut [CLS]}\\colorbox{red!9}{\\strut the}\\colorbox{green!3}{\\strut music}\\colorbox{red!1}{\\strut was}\\colorbox{green!49}{\\strut too}\\colorbox{green!37}{\\strut loud}\\colorbox{red!25}{\\strut ,}\\colorbox{red!8}{\\strut and}\\colorbox{red!15}{\\strut the}\\colorbox{red!21}{\\strut decorations}\\colorbox{red!5}{\\strut were}\\colorbox{red!28}{\\strut taste}\\colorbox{green!21}{\\strut ##less}\\colorbox{red!13}{\\strut ,}\\colorbox{red!1}{\\strut but}\\colorbox{red!22}{\\strut they}\\colorbox{red!19}{\\strut had}\\colorbox{green!18}{\\strut friendly}\\colorbox{red!13}{\\strut waiter}\\colorbox{red!7}{\\strut ##s}\\colorbox{green!15}{\\strut and}\\colorbox{green!100}{\\strut delicious}\\colorbox{red!4}{\\strut pasta}\\colorbox{red!16}{\\strut [SEP]}\n",
      "\n",
      "\\colorbox{red!15}{\\strut [CLS]}\\colorbox{red!4}{\\strut the}\\colorbox{red!9}{\\strut music}\\colorbox{red!2}{\\strut was}\\colorbox{green!55}{\\strut too}\\colorbox{green!37}{\\strut loud}\\colorbox{red!28}{\\strut ,}\\colorbox{red!7}{\\strut and}\\colorbox{red!6}{\\strut the}\\colorbox{red!15}{\\strut decorations}\\colorbox{green!3}{\\strut were}\\colorbox{red!26}{\\strut taste}\\colorbox{green!25}{\\strut ##less}\\colorbox{red!19}{\\strut ,}\\colorbox{red!4}{\\strut but}\\colorbox{red!20}{\\strut they}\\colorbox{red!31}{\\strut had}\\colorbox{green!11}{\\strut friendly}\\colorbox{red!13}{\\strut waiter}\\colorbox{red!10}{\\strut ##s}\\colorbox{red!6}{\\strut and}\\colorbox{green!100}{\\strut delicious}\\colorbox{green!7}{\\strut pasta}\\colorbox{red!16}{\\strut [SEP]}\n",
      "\n",
      "\\colorbox{red!17}{\\strut [CLS]}\\colorbox{green!5}{\\strut the}\\colorbox{green!1}{\\strut music}\\colorbox{red!9}{\\strut was}\\colorbox{green!52}{\\strut too}\\colorbox{green!32}{\\strut loud}\\colorbox{red!23}{\\strut ,}\\colorbox{red!4}{\\strut and}\\colorbox{red!7}{\\strut the}\\colorbox{red!7}{\\strut decorations}\\colorbox{red!9}{\\strut were}\\colorbox{red!21}{\\strut taste}\\colorbox{green!29}{\\strut ##less}\\colorbox{red!21}{\\strut ,}\\colorbox{green!11}{\\strut but}\\colorbox{red!19}{\\strut they}\\colorbox{red!27}{\\strut had}\\colorbox{green!8}{\\strut friendly}\\colorbox{red!16}{\\strut waiter}\\colorbox{red!20}{\\strut ##s}\\colorbox{red!2}{\\strut and}\\colorbox{green!100}{\\strut delicious}\\colorbox{red!18}{\\strut pasta}\\colorbox{red!11}{\\strut [SEP]}\n",
      "\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<table width: 100%><div style=\"border-top: 1px solid; margin-top: 5px;             padding-top: 5px; display: inline-block\"><b>Legend: </b><span style=\"display: inline-block; width: 10px; height: 10px;                 border: 1px solid; background-color:                 hsl(0, 75%, 60%)\"></span> Negative  <span style=\"display: inline-block; width: 10px; height: 10px;                 border: 1px solid; background-color:                 hsl(0, 75%, 100%)\"></span> Neutral  <span style=\"display: inline-block; width: 10px; height: 10px;                 border: 1px solid; background-color:                 hsl(120, 75%, 50%)\"></span> Positive  </div><tr><th>True Label</th><th>Predicted Label</th><th>Attribution Label</th><th>Attribution Score</th><th>Word Importance</th><tr><td><text style=\"padding-right:2em\"><b>4</b></text></td><td><text style=\"padding-right:2em\"><b>2 (0.75)</b></text></td><td><text style=\"padding-right:2em\"><b>ambiance</b></text></td><td><text style=\"padding-right:2em\"><b>0.25</b></text></td><td><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [CLS]                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> music                    </font></mark><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> was                    </font></mark><mark style=\"background-color: hsl(120, 75%, 83%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> too                    </font></mark><mark style=\"background-color: hsl(120, 75%, 88%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> loud                    </font></mark><mark style=\"background-color: hsl(0, 75%, 94%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> decorations                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> were                    </font></mark><mark style=\"background-color: hsl(0, 75%, 92%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> taste                    </font></mark><mark style=\"background-color: hsl(120, 75%, 89%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##less                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> but                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> they                    </font></mark><mark style=\"background-color: hsl(0, 75%, 93%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> had                    </font></mark><mark style=\"background-color: hsl(120, 75%, 88%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> friendly                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> waiter                    </font></mark><mark style=\"background-color: hsl(0, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##s                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(120, 75%, 72%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> delicious                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> pasta                    </font></mark><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [SEP]                    </font></mark></td><tr><tr><td><text style=\"padding-right:2em\"><b>4</b></text></td><td><text style=\"padding-right:2em\"><b>2 (0.75)</b></text></td><td><text style=\"padding-right:2em\"><b>food</b></text></td><td><text style=\"padding-right:2em\"><b>0.23</b></text></td><td><mark style=\"background-color: hsl(0, 75%, 93%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [CLS]                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(120, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> music                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> was                    </font></mark><mark style=\"background-color: hsl(120, 75%, 85%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> too                    </font></mark><mark style=\"background-color: hsl(120, 75%, 89%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> loud                    </font></mark><mark style=\"background-color: hsl(0, 75%, 94%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> decorations                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> were                    </font></mark><mark style=\"background-color: hsl(0, 75%, 93%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> taste                    </font></mark><mark style=\"background-color: hsl(120, 75%, 94%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##less                    </font></mark><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> but                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> they                    </font></mark><mark style=\"background-color: hsl(0, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> had                    </font></mark><mark style=\"background-color: hsl(120, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> friendly                    </font></mark><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> waiter                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##s                    </font></mark><mark style=\"background-color: hsl(120, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(120, 75%, 69%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> delicious                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> pasta                    </font></mark><mark style=\"background-color: hsl(0, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [SEP]                    </font></mark></td><tr><tr><td><text style=\"padding-right:2em\"><b>4</b></text></td><td><text style=\"padding-right:2em\"><b>2 (0.75)</b></text></td><td><text style=\"padding-right:2em\"><b>noise</b></text></td><td><text style=\"padding-right:2em\"><b>0.31</b></text></td><td><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [CLS]                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> music                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> was                    </font></mark><mark style=\"background-color: hsl(120, 75%, 83%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> too                    </font></mark><mark style=\"background-color: hsl(120, 75%, 89%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> loud                    </font></mark><mark style=\"background-color: hsl(0, 75%, 93%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> decorations                    </font></mark><mark style=\"background-color: hsl(120, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> were                    </font></mark><mark style=\"background-color: hsl(0, 75%, 94%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> taste                    </font></mark><mark style=\"background-color: hsl(120, 75%, 92%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##less                    </font></mark><mark style=\"background-color: hsl(0, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> but                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> they                    </font></mark><mark style=\"background-color: hsl(0, 75%, 92%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> had                    </font></mark><mark style=\"background-color: hsl(120, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> friendly                    </font></mark><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> waiter                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##s                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(120, 75%, 69%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> delicious                    </font></mark><mark style=\"background-color: hsl(120, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> pasta                    </font></mark><mark style=\"background-color: hsl(0, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [SEP]                    </font></mark></td><tr><tr><td><text style=\"padding-right:2em\"><b>4</b></text></td><td><text style=\"padding-right:2em\"><b>2 (0.75)</b></text></td><td><text style=\"padding-right:2em\"><b>service</b></text></td><td><text style=\"padding-right:2em\"><b>0.16</b></text></td><td><mark style=\"background-color: hsl(0, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [CLS]                    </font></mark><mark style=\"background-color: hsl(120, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(120, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> music                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> was                    </font></mark><mark style=\"background-color: hsl(120, 75%, 83%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> too                    </font></mark><mark style=\"background-color: hsl(120, 75%, 90%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> loud                    </font></mark><mark style=\"background-color: hsl(0, 75%, 94%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> decorations                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> were                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> taste                    </font></mark><mark style=\"background-color: hsl(120, 75%, 91%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##less                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(120, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> but                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> they                    </font></mark><mark style=\"background-color: hsl(0, 75%, 93%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> had                    </font></mark><mark style=\"background-color: hsl(120, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> friendly                    </font></mark><mark style=\"background-color: hsl(0, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> waiter                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##s                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(120, 75%, 67%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> delicious                    </font></mark><mark style=\"background-color: hsl(0, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> pasta                    </font></mark><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [SEP]                    </font></mark></td><tr></table>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "CONTROL = False\n",
    "_ = random.seed(seed)\n",
    "_ = np.random.seed(seed)\n",
    "_ = torch.manual_seed(seed)\n",
    "hf_ig_analyses_new(\n",
    "    example_sentences, labels, explainer.tokenizer, \n",
    "    explainer.cpm_model.model,\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 180,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\\colorbox{red!15}{\\strut [CLS]}\\colorbox{green!12}{\\strut the}\\colorbox{green!27}{\\strut music}\\colorbox{green!31}{\\strut was}\\colorbox{red!100}{\\strut too}\\colorbox{red!28}{\\strut loud}\\colorbox{red!52}{\\strut ,}\\colorbox{red!12}{\\strut and}\\colorbox{green!35}{\\strut the}\\colorbox{red!15}{\\strut decorations}\\colorbox{red!22}{\\strut were}\\colorbox{green!19}{\\strut taste}\\colorbox{green!15}{\\strut ##less}\\colorbox{red!30}{\\strut ,}\\colorbox{green!66}{\\strut but}\\colorbox{green!21}{\\strut they}\\colorbox{green!82}{\\strut had}\\colorbox{green!41}{\\strut friendly}\\colorbox{red!31}{\\strut waiter}\\colorbox{red!47}{\\strut ##s}\\colorbox{red!4}{\\strut and}\\colorbox{green!22}{\\strut delicious}\\colorbox{red!6}{\\strut pasta}\\colorbox{red!7}{\\strut [SEP]}\n",
      "\n",
      "\\colorbox{red!9}{\\strut [CLS]}\\colorbox{red!74}{\\strut the}\\colorbox{green!63}{\\strut music}\\colorbox{red!34}{\\strut was}\\colorbox{red!44}{\\strut too}\\colorbox{green!9}{\\strut loud}\\colorbox{red!30}{\\strut ,}\\colorbox{red!13}{\\strut and}\\colorbox{green!19}{\\strut the}\\colorbox{red!11}{\\strut decorations}\\colorbox{red!11}{\\strut were}\\colorbox{green!24}{\\strut taste}\\colorbox{red!11}{\\strut ##less}\\colorbox{green!0}{\\strut ,}\\colorbox{green!11}{\\strut but}\\colorbox{green!33}{\\strut they}\\colorbox{green!17}{\\strut had}\\colorbox{green!100}{\\strut friendly}\\colorbox{green!3}{\\strut waiter}\\colorbox{green!12}{\\strut ##s}\\colorbox{green!12}{\\strut and}\\colorbox{red!38}{\\strut delicious}\\colorbox{red!12}{\\strut pasta}\\colorbox{red!14}{\\strut [SEP]}\n",
      "\n",
      "\\colorbox{red!25}{\\strut [CLS]}\\colorbox{red!8}{\\strut the}\\colorbox{green!2}{\\strut music}\\colorbox{green!6}{\\strut was}\\colorbox{red!12}{\\strut too}\\colorbox{red!37}{\\strut loud}\\colorbox{green!12}{\\strut ,}\\colorbox{green!0}{\\strut and}\\colorbox{green!4}{\\strut the}\\colorbox{red!29}{\\strut decorations}\\colorbox{red!18}{\\strut were}\\colorbox{red!2}{\\strut taste}\\colorbox{red!0}{\\strut ##less}\\colorbox{red!7}{\\strut ,}\\colorbox{green!9}{\\strut but}\\colorbox{green!2}{\\strut they}\\colorbox{red!12}{\\strut had}\\colorbox{green!100}{\\strut friendly}\\colorbox{red!8}{\\strut waiter}\\colorbox{red!8}{\\strut ##s}\\colorbox{green!13}{\\strut and}\\colorbox{green!25}{\\strut delicious}\\colorbox{green!1}{\\strut pasta}\\colorbox{red!6}{\\strut [SEP]}\n",
      "\n",
      "\\colorbox{red!31}{\\strut [CLS]}\\colorbox{green!97}{\\strut the}\\colorbox{green!20}{\\strut music}\\colorbox{red!49}{\\strut was}\\colorbox{red!0}{\\strut too}\\colorbox{red!99}{\\strut loud}\\colorbox{green!38}{\\strut ,}\\colorbox{red!29}{\\strut and}\\colorbox{green!20}{\\strut the}\\colorbox{red!43}{\\strut decorations}\\colorbox{red!11}{\\strut were}\\colorbox{green!2}{\\strut taste}\\colorbox{green!26}{\\strut ##less}\\colorbox{red!7}{\\strut ,}\\colorbox{green!37}{\\strut but}\\colorbox{green!12}{\\strut they}\\colorbox{green!26}{\\strut had}\\colorbox{green!84}{\\strut friendly}\\colorbox{red!32}{\\strut waiter}\\colorbox{green!10}{\\strut ##s}\\colorbox{green!18}{\\strut and}\\colorbox{red!28}{\\strut delicious}\\colorbox{red!32}{\\strut pasta}\\colorbox{red!28}{\\strut [SEP]}\n",
      "\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<table width: 100%><div style=\"border-top: 1px solid; margin-top: 5px;             padding-top: 5px; display: inline-block\"><b>Legend: </b><span style=\"display: inline-block; width: 10px; height: 10px;                 border: 1px solid; background-color:                 hsl(0, 75%, 60%)\"></span> Negative  <span style=\"display: inline-block; width: 10px; height: 10px;                 border: 1px solid; background-color:                 hsl(0, 75%, 100%)\"></span> Neutral  <span style=\"display: inline-block; width: 10px; height: 10px;                 border: 1px solid; background-color:                 hsl(120, 75%, 50%)\"></span> Positive  </div><tr><th>True Label</th><th>Predicted Label</th><th>Attribution Label</th><th>Attribution Score</th><th>Word Importance</th><tr><td><text style=\"padding-right:2em\"><b>4</b></text></td><td><text style=\"padding-right:2em\"><b>2 (0.98)</b></text></td><td><text style=\"padding-right:2em\"><b>ambiance</b></text></td><td><text style=\"padding-right:2em\"><b>-0.01</b></text></td><td><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [CLS]                    </font></mark><mark style=\"background-color: hsl(120, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(120, 75%, 93%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> music                    </font></mark><mark style=\"background-color: hsl(120, 75%, 92%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> was                    </font></mark><mark style=\"background-color: hsl(0, 75%, 80%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> too                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> loud                    </font></mark><mark style=\"background-color: hsl(0, 75%, 89%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(120, 75%, 91%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> decorations                    </font></mark><mark style=\"background-color: hsl(0, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> were                    </font></mark><mark style=\"background-color: hsl(120, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> taste                    </font></mark><mark style=\"background-color: hsl(120, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##less                    </font></mark><mark style=\"background-color: hsl(0, 75%, 94%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(120, 75%, 83%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> but                    </font></mark><mark style=\"background-color: hsl(120, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> they                    </font></mark><mark style=\"background-color: hsl(120, 75%, 79%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> had                    </font></mark><mark style=\"background-color: hsl(120, 75%, 90%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> friendly                    </font></mark><mark style=\"background-color: hsl(0, 75%, 94%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> waiter                    </font></mark><mark style=\"background-color: hsl(0, 75%, 91%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##s                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(120, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> delicious                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> pasta                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [SEP]                    </font></mark></td><tr><tr><td><text style=\"padding-right:2em\"><b>4</b></text></td><td><text style=\"padding-right:2em\"><b>2 (0.98)</b></text></td><td><text style=\"padding-right:2em\"><b>food</b></text></td><td><text style=\"padding-right:2em\"><b>0.18</b></text></td><td><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [CLS]                    </font></mark><mark style=\"background-color: hsl(0, 75%, 84%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(120, 75%, 83%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> music                    </font></mark><mark style=\"background-color: hsl(0, 75%, 93%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> was                    </font></mark><mark style=\"background-color: hsl(0, 75%, 91%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> too                    </font></mark><mark style=\"background-color: hsl(120, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> loud                    </font></mark><mark style=\"background-color: hsl(0, 75%, 94%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(120, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> decorations                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> were                    </font></mark><mark style=\"background-color: hsl(120, 75%, 94%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> taste                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##less                    </font></mark><mark style=\"background-color: hsl(120, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(120, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> but                    </font></mark><mark style=\"background-color: hsl(120, 75%, 91%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> they                    </font></mark><mark style=\"background-color: hsl(120, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> had                    </font></mark><mark style=\"background-color: hsl(120, 75%, 73%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> friendly                    </font></mark><mark style=\"background-color: hsl(120, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> waiter                    </font></mark><mark style=\"background-color: hsl(120, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##s                    </font></mark><mark style=\"background-color: hsl(120, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(0, 75%, 92%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> delicious                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> pasta                    </font></mark><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [SEP]                    </font></mark></td><tr><tr><td><text style=\"padding-right:2em\"><b>4</b></text></td><td><text style=\"padding-right:2em\"><b>2 (0.98)</b></text></td><td><text style=\"padding-right:2em\"><b>noise</b></text></td><td><text style=\"padding-right:2em\"><b>0.19</b></text></td><td><mark style=\"background-color: hsl(0, 75%, 93%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [CLS]                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(120, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> music                    </font></mark><mark style=\"background-color: hsl(120, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> was                    </font></mark><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> too                    </font></mark><mark style=\"background-color: hsl(0, 75%, 89%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> loud                    </font></mark><mark style=\"background-color: hsl(120, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(120, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(120, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 92%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> decorations                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> were                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> taste                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##less                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(120, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> but                    </font></mark><mark style=\"background-color: hsl(120, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> they                    </font></mark><mark style=\"background-color: hsl(0, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> had                    </font></mark><mark style=\"background-color: hsl(120, 75%, 62%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> friendly                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> waiter                    </font></mark><mark style=\"background-color: hsl(0, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##s                    </font></mark><mark style=\"background-color: hsl(120, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(120, 75%, 91%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> delicious                    </font></mark><mark style=\"background-color: hsl(120, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> pasta                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [SEP]                    </font></mark></td><tr><tr><td><text style=\"padding-right:2em\"><b>4</b></text></td><td><text style=\"padding-right:2em\"><b>2 (0.98)</b></text></td><td><text style=\"padding-right:2em\"><b>service</b></text></td><td><text style=\"padding-right:2em\"><b>0.23</b></text></td><td><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [CLS]                    </font></mark><mark style=\"background-color: hsl(120, 75%, 81%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(120, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> music                    </font></mark><mark style=\"background-color: hsl(0, 75%, 93%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> was                    </font></mark><mark style=\"background-color: hsl(0, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> too                    </font></mark><mark style=\"background-color: hsl(0, 75%, 84%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> loud                    </font></mark><mark style=\"background-color: hsl(120, 75%, 93%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(0, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(120, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> the                    </font></mark><mark style=\"background-color: hsl(0, 75%, 93%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> decorations                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> were                    </font></mark><mark style=\"background-color: hsl(120, 75%, 100%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> taste                    </font></mark><mark style=\"background-color: hsl(120, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##less                    </font></mark><mark style=\"background-color: hsl(0, 75%, 99%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ,                    </font></mark><mark style=\"background-color: hsl(120, 75%, 93%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> but                    </font></mark><mark style=\"background-color: hsl(120, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> they                    </font></mark><mark style=\"background-color: hsl(120, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> had                    </font></mark><mark style=\"background-color: hsl(120, 75%, 83%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> friendly                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> waiter                    </font></mark><mark style=\"background-color: hsl(120, 75%, 98%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> ##s                    </font></mark><mark style=\"background-color: hsl(120, 75%, 97%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> and                    </font></mark><mark style=\"background-color: hsl(0, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> delicious                    </font></mark><mark style=\"background-color: hsl(0, 75%, 95%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> pasta                    </font></mark><mark style=\"background-color: hsl(0, 75%, 96%); opacity:1.0;                     line-height:1.75\"><font color=\"black\"> [SEP]                    </font></mark></td><tr></table>"
      ],
      "text/plain": [
       "<IPython.core.display.HTML object>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "CONTROL = True\n",
    "_ = random.seed(seed)\n",
    "_ = np.random.seed(seed)\n",
    "_ = torch.manual_seed(seed)\n",
    "hf_ig_analyses(\n",
    "    example_sentences, labels, explainer.tokenizer, \n",
    "    explainer.blackbox_model\n",
    ")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.7.6"
  },
  "vscode": {
   "interpreter": {
    "hash": "4eadbeb4763107051c423ffe9ee9ffcb0adb376f6d3c57fb486fb39aa2d1a822"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
