{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from datasets import load_dataset\n",
    "from datasets import Dataset\n",
    "from tqdm.auto import tqdm\n",
    "import os\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "from transformers import AutoTokenizer, AutoModelForCausalLM\n",
    "from transformers import AutoModelForSequenceClassification\n",
    "from transformers import pipeline\n",
    "import copy\n",
    "from sklearn.manifold import TSNE\n",
    "import matplotlib.pyplot as plt\n",
    "from rich import print as pprint\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import random\n",
    "import re\n",
    "# from parlai.utils.safety import OffensiveLanguageClassifier\n",
    "# from rouge_score import rouge_scorer\n",
    "# import evaluate"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "ds = load_dataset(\"google/boolq\", cache_dir='/data/takis/datasets/hf_hub')['train']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 9427/9427 [00:00<00:00, 67970.69it/s]\n"
     ]
    }
   ],
   "source": [
    "prompts = []\n",
    "\n",
    "for i, row in tqdm(enumerate(ds), total=len(ds)):\n",
    "    prompts.append('passage: %s, question: %s, answer: %s' %(row['passage'], row['question'], row['answer']))\n",
    "prompts = np.array(prompts)\n",
    "\n",
    "keep_idx = random.sample(list(range(len(prompts))), 600)\n",
    "prompts = prompts[keep_idx]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1\n"
     ]
    }
   ],
   "source": [
    "os.environ[\"CUDA_DEVICE_ORDER\"]=\"PCI_BUS_ID\"\n",
    "os.environ[\"CUDA_VISIBLE_DEVICES\"] = '0'\n",
    "print (torch.cuda.device_count())\n",
    "\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Loading checkpoint shards: 100%|██████████| 4/4 [00:00<00:00,  5.23it/s]\n"
     ]
    }
   ],
   "source": [
    "tokenizer = AutoTokenizer.from_pretrained(\"meta-llama/Meta-Llama-3.1-8B-Instruct\", cache_dir='/data/takis/models')\n",
    "model = AutoModelForCausalLM.from_pretrained(\"meta-llama/Meta-Llama-3.1-8B-Instruct\", cache_dir='/data/takis/models', torch_dtype=torch.bfloat16).to(device)\n",
    "model = model.eval()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Hidden states"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "activations = {}\n",
    "def get_activation(name):\n",
    "    def hook(model, input, output):\n",
    "        if (name in activations):\n",
    "            activations[name].append(output[0])\n",
    "        else:\n",
    "            activations[name] = [output[0]]\n",
    "    return hook\n",
    "\n",
    "\n",
    "hooks = []\n",
    "for i, layer in enumerate(model.model.layers):\n",
    "    hooks.append(layer.self_attn.register_forward_hook(get_activation(i)))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "cur prompt length (chars): 622:   0%|          | 0/600 [00:00<?, ?it/s]Starting from v4.46, the `logits` model output will have the same type as the model (except at train time, where it will always be FP32)\n",
      "cur prompt length (chars): 356: 100%|██████████| 600/600 [00:32<00:00, 18.67it/s] \n"
     ]
    }
   ],
   "source": [
    "hidden_states = []\n",
    "activations = {}\n",
    "\n",
    "pbar = tqdm(enumerate(prompts), total=len(prompts))\n",
    "for i, p in pbar:\n",
    "\n",
    "    pbar.set_description('cur prompt length (chars): %d' %(len(p)))\n",
    "    \n",
    "    with torch.no_grad():\n",
    "        inputs = tokenizer(p, return_tensors=\"pt\").input_ids\n",
    "        outputs = model.forward(inputs.to(device))\n",
    "        \n",
    "    for k in activations.keys():\n",
    "        activations[k] = torch.cat(activations[k], dim=1).detach().cpu()\n",
    "        activations[k] =  torch.mean(activations[k], dim=-2)\n",
    "\n",
    "\n",
    "    hidden_states.append(copy.deepcopy(activations))\n",
    "    activations = {}"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Attractors"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "target_layers = [i for i in range(len(model.model.layers))]\n",
    "all_prompts_attractors = hidden_states"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Different prompts t-sne"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "ic_prompts = [\"Write a short factual passage about a real-world topic, followed by a true-or-false question directly based on a detail in the passage. Provide the answer as 'true' or 'false'.\",\n",
    "           \"Generate a passage that implies something not explicitly stated. Then create a true-or-false question that requires inference from the passage, and provide the answer.\",\n",
    "           \"Invent a passage about a historical or scientific topic, then write a question that contains a subtle factual error. Answer with 'true' or 'false' accordingly.\",\n",
    "           \"Create a short informative passage, then generate a question that paraphrases part of it but introduces a slight inaccuracy. Provide the true/false answer.\",\n",
    "           \"Write a passage involving contrasting facts or events. Then ask a true-or-false question using negation (e.g., 'did not', 'never'), and answer it.\",\n",
    "           \"Generate a passage involving cause-and-effect or events in time. Then write a true-or-false question that tests understanding of that structure, and give the answer.\",\n",
    "           \"Write a passage that sounds plausible. Then ask a question that seems likely but is actually false given the passage. Provide the correct answer.\",\n",
    "           \"Produce a passage that could be misinterpreted, then pose a true-or-false question that tests precise understanding. Provide the correct answer.\",\n",
    "           \"Create a passage that blends general world knowledge with specific details. Then write a true-or-false question grounded only in the passage, and answer it.\",\n",
    "           \"Write a passage on a narrow topic. Then ask a question that seems related but is not supported by the passage. Provide 'false' as the answer.\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "cur prompt length (chars): 753: 100%|██████████| 10/10 [00:05<00:00,  1.97it/s]\n"
     ]
    }
   ],
   "source": [
    "hidden_states = []\n",
    "activations = {}\n",
    "\n",
    "pbar = tqdm(enumerate(prompts[:10]), total=len(prompts[:10]))\n",
    "for i, p in pbar:\n",
    "\n",
    "    for icp in ic_prompts:\n",
    "\n",
    "        pbar.set_description('cur prompt length (chars): %d' %(len(p)))\n",
    "        \n",
    "        with torch.no_grad():\n",
    "            inputs = tokenizer(p + '\\n' + icp, return_tensors=\"pt\").input_ids\n",
    "            outputs = model.forward(inputs.to(device))\n",
    "            \n",
    "        for k in activations.keys():\n",
    "            activations[k] = torch.cat(activations[k], dim=1).detach().cpu()\n",
    "            activations[k] =  torch.mean(activations[k], dim=-2)\n",
    "\n",
    "\n",
    "        hidden_states.append(copy.deepcopy(activations))\n",
    "        activations = {}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [],
   "source": [
    "target_layer = 16\n",
    "\n",
    "cur_attractors = []\n",
    "for i in range(len(hidden_states)):\n",
    "    cur_attractors.append(hidden_states[i][target_layer])\n",
    "cur_attractors = torch.stack(cur_attractors, dim=0).squeeze()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "cur_attractors_emb = TSNE(n_components=2, metric='cosine', learning_rate='auto', init='random', perplexity=8, random_state=0).fit_transform(cur_attractors.detach().to(float).cpu().numpy())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 500x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from matplotlib import rcParams\n",
    "\n",
    "\n",
    "rcParams['font.family'] = 'serif'\n",
    "rcParams['text.usetex'] = True\n",
    "\n",
    "rcParams['axes.grid'] = False\n",
    "rcParams['grid.linestyle'] = '--'\n",
    "rcParams['grid.alpha'] = 0.9\n",
    "\n",
    "rcParams['axes.facecolor'] = 'white'\n",
    "# rcParams['axes.facecolor'] = '#f8f9fa'\n",
    "# rcParams['axes.linewidth'] = 2\n",
    "# rcParams['axes.axisbelow'] = True\n",
    "rcParams['axes.labelsize'] = 35\n",
    "rcParams['axes.titlesize'] = 30\n",
    "\n",
    "rcParams['axes.spines.top'] = False\n",
    "rcParams['axes.spines.right'] = False\n",
    "rcParams['axes.spines.bottom'] = False\n",
    "rcParams['axes.spines.left'] = False\n",
    "\n",
    "rcParams['lines.linewidth'] = 4\n",
    "rcParams['lines.markersize'] = 15\n",
    "\n",
    "rcParams['legend.fontsize'] = 25\n",
    "rcParams['xtick.labelsize'] = 25\n",
    "rcParams['ytick.labelsize'] = 25\n",
    "rcParams['figure.figsize'] = (5, 3)\n",
    "\n",
    "plt.scatter(cur_attractors_emb[:, 0], cur_attractors_emb[:, 1], c=10*['maroon'] + \\\n",
    "                                                                  10*['darkgreen'] + \\\n",
    "                                                                  10*['navy'] + \\\n",
    "                                                                  10*['darkorange'] + \\\n",
    "                                                                  10*['lightcoral'] + \\\n",
    "                                                                  10*['dimgrey'] + \\\n",
    "                                                                  10*['darkolivegreen'] + \\\n",
    "                                                                  10*['turquoise'] + \\\n",
    "                                                                  10*['darkviolet'] + \\\n",
    "                                                                  10*['magenta'], edgecolor='black')\n",
    "plt.xticks([])\n",
    "plt.yticks([])\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Behavior steering"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [],
   "source": [
    "for hook in hooks:\n",
    "    hook.remove()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [],
   "source": [
    "target_layer = 16\n",
    "\n",
    "\n",
    "def add_noise(target_layer):\n",
    "    def hook(model, input, output):\n",
    "        if (abs(strength) > 1e-5):\n",
    "            output = (output[0] + strength*noise.to(device), *output[1:])\n",
    "        return output\n",
    "    return hook\n",
    "\n",
    "\n",
    "hook = model.model.layers[target_layer].self_attn.register_forward_hook(add_noise(target_layer))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [],
   "source": [
    "target_prompt = 4\n",
    "\n",
    "prompt = '''\\n Now generate 3 different passages, questions, and answers similar to the example above. Please make sure each question you generate has a boolean answer that can be answered by the passage. Make sure each passage and question is different and sufficiently rephrased. Please make sure you generate passages, questions and both true and false answers.'''"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\">passage: Casting for series one began in early <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">2009</span> in Brisbane, Melbourne and Sydney. All cast members had to be \n",
       "skilled in drama and dancing and had to cope with Australia's best choreographers. Filming began on <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">13</span> July <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">2009</span> \n",
       "and wrapped up in early November. The series premiere was originally planned for a mid-<span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">2010</span> premiere on ABC3, \n",
       "however, like Dead Gorgeous, the premiere was pushed to ABC1 on <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">31</span> May <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">2010</span> and ABC3 on <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">6</span> June <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">2010</span>. The first \n",
       "series premiered on Germany's ZDF on <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">26</span> September <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">2010</span>., question: are the dancers on dance academy real dancers, \n",
       "answer: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>\n",
       "</pre>\n"
      ],
      "text/plain": [
       "passage: Casting for series one began in early \u001b[1;36m2009\u001b[0m in Brisbane, Melbourne and Sydney. All cast members had to be \n",
       "skilled in drama and dancing and had to cope with Australia's best choreographers. Filming began on \u001b[1;36m13\u001b[0m July \u001b[1;36m2009\u001b[0m \n",
       "and wrapped up in early November. The series premiere was originally planned for a mid-\u001b[1;36m2010\u001b[0m premiere on ABC3, \n",
       "however, like Dead Gorgeous, the premiere was pushed to ABC1 on \u001b[1;36m31\u001b[0m May \u001b[1;36m2010\u001b[0m and ABC3 on \u001b[1;36m6\u001b[0m June \u001b[1;36m2010\u001b[0m. The first \n",
       "series premiered on Germany's ZDF on \u001b[1;36m26\u001b[0m September \u001b[1;36m2010\u001b[0m., question: are the dancers on dance academy real dancers, \n",
       "answer: \u001b[3;92mTrue\u001b[0m\n"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "pprint (prompts[target_prompt])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## No steering"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "No steering...\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"> \n",
       "\n",
       "passage: In computer science, a stack is a linear data structure that follows the LIFO <span style=\"font-weight: bold\">(</span>last in, first out<span style=\"font-weight: bold\">)</span> \n",
       "principle. It can be implemented using arrays, linked lists, or other data structures. Stacks are used in many \n",
       "applications, including parsing expressions, evaluating postfix notation, and implementing recursive algorithms. \n",
       "They are also used in recursive data structures such as trees and graphs., question: are stacks implemented using \n",
       "linked lists?, answer: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>\n",
       "\n",
       "passage: In computer networks, a router is a device that connects multiple networks together and routes traffic \n",
       "between them. It uses IP addresses to forward packets of data to their destination. Routers can be used to connect \n",
       "local area networks <span style=\"font-weight: bold\">(</span>LANs<span style=\"font-weight: bold\">)</span> to wide area networks <span style=\"font-weight: bold\">(</span>WANs<span style=\"font-weight: bold\">)</span> or to the internet. They are typically used in large \n",
       "networks, such as those used in businesses and governments., question: do routers connect local area networks to \n",
       "the internet?, answer: <span style=\"color: #ff0000; text-decoration-color: #ff0000; font-style: italic\">False</span>\n",
       "\n",
       "passage: In programming, a while loop is a control structure that allows a program to repeat a set of instructions \n",
       "as long as a certain condition is true. The loop continues to execute until the condition is false. While loops are\n",
       "commonly used in algorithms and are a fundamental part of many programming languages. They are often used in \n",
       "conjunction with conditional statements to control the flow of a program., question: do while loops repeat \n",
       "instructions until a condition is false?, answer: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>\n",
       "\n",
       "passage: In computer science, a binary search tree is a data structure that consists of nodes with keys and values.\n",
       "Each node has a value and a key, and it has at most two children, a left child and a right child. The keys in the \n",
       "left subtree are less than the key in the node, and the keys in the right subtree are greater than the key in the \n",
       "node. This property makes binary search trees efficient for searching and inserting data., question: do binary \n",
       "search trees have at least one child?, answer: <span style=\"color: #ff0000; text-decoration-color: #ff0000; font-style: italic\">False</span>\n",
       "\n",
       "passage: In software development, an interface is a contract that specifies a set of methods that must be \n",
       "implemented by any class that implements it. Interfaces are used to define a common set of methods that can be used\n",
       "by different classes, making it easier to write code that can work with different classes. They are often used in \n",
       "object-oriented programming languages such as Java and C#. Interfaces can also be used to define a set of methods \n",
       "that can be used by multiple classes., question: can interfaces be used to define a set of methods that can\n",
       "</pre>\n"
      ],
      "text/plain": [
       " \n",
       "\n",
       "passage: In computer science, a stack is a linear data structure that follows the LIFO \u001b[1m(\u001b[0mlast in, first out\u001b[1m)\u001b[0m \n",
       "principle. It can be implemented using arrays, linked lists, or other data structures. Stacks are used in many \n",
       "applications, including parsing expressions, evaluating postfix notation, and implementing recursive algorithms. \n",
       "They are also used in recursive data structures such as trees and graphs., question: are stacks implemented using \n",
       "linked lists?, answer: \u001b[3;92mTrue\u001b[0m\n",
       "\n",
       "passage: In computer networks, a router is a device that connects multiple networks together and routes traffic \n",
       "between them. It uses IP addresses to forward packets of data to their destination. Routers can be used to connect \n",
       "local area networks \u001b[1m(\u001b[0mLANs\u001b[1m)\u001b[0m to wide area networks \u001b[1m(\u001b[0mWANs\u001b[1m)\u001b[0m or to the internet. They are typically used in large \n",
       "networks, such as those used in businesses and governments., question: do routers connect local area networks to \n",
       "the internet?, answer: \u001b[3;91mFalse\u001b[0m\n",
       "\n",
       "passage: In programming, a while loop is a control structure that allows a program to repeat a set of instructions \n",
       "as long as a certain condition is true. The loop continues to execute until the condition is false. While loops are\n",
       "commonly used in algorithms and are a fundamental part of many programming languages. They are often used in \n",
       "conjunction with conditional statements to control the flow of a program., question: do while loops repeat \n",
       "instructions until a condition is false?, answer: \u001b[3;92mTrue\u001b[0m\n",
       "\n",
       "passage: In computer science, a binary search tree is a data structure that consists of nodes with keys and values.\n",
       "Each node has a value and a key, and it has at most two children, a left child and a right child. The keys in the \n",
       "left subtree are less than the key in the node, and the keys in the right subtree are greater than the key in the \n",
       "node. This property makes binary search trees efficient for searching and inserting data., question: do binary \n",
       "search trees have at least one child?, answer: \u001b[3;91mFalse\u001b[0m\n",
       "\n",
       "passage: In software development, an interface is a contract that specifies a set of methods that must be \n",
       "implemented by any class that implements it. Interfaces are used to define a common set of methods that can be used\n",
       "by different classes, making it easier to write code that can work with different classes. They are often used in \n",
       "object-oriented programming languages such as Java and C#. Interfaces can also be used to define a set of methods \n",
       "that can be used by multiple classes., question: can interfaces be used to define a set of methods that can\n"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "\n",
      "------------------------------------------------\n",
      "\n",
      "\n",
      "Steering...\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<pre style=\"white-space:pre;overflow-x:auto;line-height:normal;font-family:Menlo,'DejaVu Sans Mono',consolas,'Courier New',monospace\"> \n",
       "\n",
       "Passage <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">1</span>:\n",
       "In computing, a database is a collection of organized data that is stored in a way that facilitates efficient \n",
       "retrieval and manipulation. A database is a structured collection of data that is stored in a way that allows for \n",
       "efficient searching, sorting, and retrieval of data. A database is a collection of data that is organized in a way \n",
       "that allows for efficient querying and retrieval of data.\n",
       "\n",
       "Question <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">1</span>:\n",
       "Is a database a collection of organized data?\n",
       "Answer: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>\n",
       "\n",
       "Passage <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">2</span>:\n",
       "A stack is a linear data structure that follows the Last-In-First-Out <span style=\"font-weight: bold\">(</span>LIFO<span style=\"font-weight: bold\">)</span> principle. A stack is a data structure\n",
       "that allows elements to be added and removed from the top of the stack. A stack is a linear data structure that \n",
       "follows the LIFO principle, where the last element added to the stack is the first one to be removed.\n",
       "\n",
       "Question <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">2</span>:\n",
       "Is a stack a linear data structure?\n",
       "Answer: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>\n",
       "\n",
       "Passage <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">3</span>:\n",
       "A binary search tree is a data structure in which each node has a comparable value. A binary search tree is a data \n",
       "structure that is used to store a collection of items in a way that allows for efficient searching and retrieval of\n",
       "items. A binary search tree is a data structure that is used to store a collection of items in a way that allows \n",
       "for efficient searching and retrieval of items.\n",
       "\n",
       "Question <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">3</span>:\n",
       "Is a binary search tree a data structure that is used to store a collection of items?\n",
       "Answer: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>\n",
       "\n",
       "Passage <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">4</span>:\n",
       "A queue is a linear data structure that follows the First-In-First-Out <span style=\"font-weight: bold\">(</span>FIFO<span style=\"font-weight: bold\">)</span> principle. A queue is a data \n",
       "structure that allows elements to be added to the end of the queue and removed from the front of the queue. A queue\n",
       "is a data structure that follows the FIFO principle, where the first element added to the queue is the first one to\n",
       "be removed.\n",
       "\n",
       "Question <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">4</span>:\n",
       "Is a queue a data structure that follows the FIFO principle?\n",
       "Answer: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>\n",
       "\n",
       "Passage <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">5</span>:\n",
       "A graph is a non-linear data structure that consists of nodes or vertices connected by edges. A graph is a data \n",
       "structure that allows for efficient searching and retrieval of nodes and edges. A graph is a non-linear data \n",
       "structure that consists of nodes or vertices connected by edges.\n",
       "\n",
       "Question <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">5</span>:\n",
       "Is a graph a non-linear data structure?\n",
       "Answer: <span style=\"color: #00ff00; text-decoration-color: #00ff00; font-style: italic\">True</span>\n",
       "\n",
       "Passage <span style=\"color: #008080; text-decoration-color: #008080; font-weight: bold\">6</span>:\n",
       "A linked list is a linear data structure that consists of a sequence of nodes, each of which contains\n",
       "</pre>\n"
      ],
      "text/plain": [
       " \n",
       "\n",
       "Passage \u001b[1;36m1\u001b[0m:\n",
       "In computing, a database is a collection of organized data that is stored in a way that facilitates efficient \n",
       "retrieval and manipulation. A database is a structured collection of data that is stored in a way that allows for \n",
       "efficient searching, sorting, and retrieval of data. A database is a collection of data that is organized in a way \n",
       "that allows for efficient querying and retrieval of data.\n",
       "\n",
       "Question \u001b[1;36m1\u001b[0m:\n",
       "Is a database a collection of organized data?\n",
       "Answer: \u001b[3;92mTrue\u001b[0m\n",
       "\n",
       "Passage \u001b[1;36m2\u001b[0m:\n",
       "A stack is a linear data structure that follows the Last-In-First-Out \u001b[1m(\u001b[0mLIFO\u001b[1m)\u001b[0m principle. A stack is a data structure\n",
       "that allows elements to be added and removed from the top of the stack. A stack is a linear data structure that \n",
       "follows the LIFO principle, where the last element added to the stack is the first one to be removed.\n",
       "\n",
       "Question \u001b[1;36m2\u001b[0m:\n",
       "Is a stack a linear data structure?\n",
       "Answer: \u001b[3;92mTrue\u001b[0m\n",
       "\n",
       "Passage \u001b[1;36m3\u001b[0m:\n",
       "A binary search tree is a data structure in which each node has a comparable value. A binary search tree is a data \n",
       "structure that is used to store a collection of items in a way that allows for efficient searching and retrieval of\n",
       "items. A binary search tree is a data structure that is used to store a collection of items in a way that allows \n",
       "for efficient searching and retrieval of items.\n",
       "\n",
       "Question \u001b[1;36m3\u001b[0m:\n",
       "Is a binary search tree a data structure that is used to store a collection of items?\n",
       "Answer: \u001b[3;92mTrue\u001b[0m\n",
       "\n",
       "Passage \u001b[1;36m4\u001b[0m:\n",
       "A queue is a linear data structure that follows the First-In-First-Out \u001b[1m(\u001b[0mFIFO\u001b[1m)\u001b[0m principle. A queue is a data \n",
       "structure that allows elements to be added to the end of the queue and removed from the front of the queue. A queue\n",
       "is a data structure that follows the FIFO principle, where the first element added to the queue is the first one to\n",
       "be removed.\n",
       "\n",
       "Question \u001b[1;36m4\u001b[0m:\n",
       "Is a queue a data structure that follows the FIFO principle?\n",
       "Answer: \u001b[3;92mTrue\u001b[0m\n",
       "\n",
       "Passage \u001b[1;36m5\u001b[0m:\n",
       "A graph is a non-linear data structure that consists of nodes or vertices connected by edges. A graph is a data \n",
       "structure that allows for efficient searching and retrieval of nodes and edges. A graph is a non-linear data \n",
       "structure that consists of nodes or vertices connected by edges.\n",
       "\n",
       "Question \u001b[1;36m5\u001b[0m:\n",
       "Is a graph a non-linear data structure?\n",
       "Answer: \u001b[3;92mTrue\u001b[0m\n",
       "\n",
       "Passage \u001b[1;36m6\u001b[0m:\n",
       "A linked list is a linear data structure that consists of a sequence of nodes, each of which contains\n"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "strength = 0\n",
    "noise = 0 \n",
    "\n",
    "responses = []\n",
    "\n",
    "aug_prompt = prompts[target_prompt] + '\\n' + prompt\n",
    "\n",
    "print ('No steering...')\n",
    "\n",
    "\n",
    "with torch.no_grad():\n",
    "    inputs = tokenizer(aug_prompt, return_tensors=\"pt\").input_ids\n",
    "    outputs = model.generate(inputs.to(device), max_new_tokens=500, do_sample=True, top_k=50, top_p=0.95, temperature=0.6, pad_token_id=tokenizer.eos_token_id, use_cache=True).detach().cpu()\n",
    "responses.append(output_text := tokenizer.batch_decode(outputs[:, inputs.shape[1]:], skip_special_tokens=True)[0])\n",
    "\n",
    "pprint (output_text)\n",
    "    \n",
    "\n",
    "print ('\\n\\n------------------------------------------------\\n\\n')\n",
    "print ('Steering...')\n",
    "\n",
    "denom = 20 \n",
    "strength = 1\n",
    "\n",
    "\n",
    "noise = torch.randn(all_prompts_attractors[target_prompt][target_layer].shape).to(torch.bfloat16)/denom\n",
    "\n",
    "\n",
    "\n",
    "with torch.no_grad():\n",
    "    inputs = tokenizer(aug_prompt, return_tensors=\"pt\").input_ids\n",
    "    outputs = model.generate(inputs.to(device), max_new_tokens=500, do_sample=True, top_k=20, top_p=0.95, temperature=0.2, pad_token_id=tokenizer.eos_token_id, use_cache=True).detach().cpu()\n",
    "responses.append(output_text := tokenizer.batch_decode(outputs[:, inputs.shape[1]:], skip_special_tokens=True)[0])\n",
    "\n",
    "pprint (output_text)\n",
    "            \n",
    "        "
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "unlearning",
   "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.11.11"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
