{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c2f6b7d4",
   "metadata": {},
   "outputs": [],
   "source": [
    "%matplotlib inline\n",
    "\n",
    "import pickle\n",
    "from transformers import AutoTokenizer, AutoModelForCausalLM\n",
    "\n",
    "import random\n",
    "import os\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "import glob\n",
    "import numpy as np\n",
    "from IPython.display import clear_output\n",
    "os.chdir(\"../\")\n",
    "\n",
    "from src import utils\n",
    "from matplotlib.lines import Line2D\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "359254e2",
   "metadata": {},
   "outputs": [],
   "source": [
    "os.chdir(\"../outputs/cpt\")\n",
    "tokenizer_L1B = AutoTokenizer.from_pretrained(\"meta-llama/Llama-3.2-1B-Instruct\")\n",
    "tokenizer_L3B = AutoTokenizer.from_pretrained(\"meta-llama/Llama-3.2-3B-Instruct\")\n",
    "tokenizer_G1B = AutoTokenizer.from_pretrained(\"google/gemma-3-1b-it\")\n",
    "tokenizer_G4B = AutoTokenizer.from_pretrained(\"google/gemma-3-4b-it\")\n",
    "tokenizer_M8B = AutoTokenizer.from_pretrained(\"mistralai/Ministral-8B-Instruct-2410\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "08b61cf3",
   "metadata": {},
   "outputs": [],
   "source": [
    "with open('factual_modelLlama-3.2-1B-Instruct_p1.0_kNone_numprompts400_maxoutlen200_temp1.0.pkl', 'rb') as file:\n",
    "     data_L1B_p1_T1 = pickle.load(file)\n",
    "with open('factual_modelLlama-3.2-3B-Instruct_p1.0_kNone_numprompts400_maxoutlen200_temp1.0.pkl', 'rb') as file:\n",
    "     data_L3B_p1_T1 = pickle.load(file)\n",
    "with open('factual_modelGemma-3-1b-it_p1.0_kNone_numprompts400_maxoutlen200_temp1.0.pkl', 'rb') as file:\n",
    "     data_G1B_p1_T1 = pickle.load(file)\n",
    "with open('factual_modelGemma-3-4b-it_p1.0_kNone_numprompts400_maxoutlen200_temp1.0.pkl', 'rb') as file:\n",
    "     data_G4B_p1_T1 = pickle.load(file)\n",
    "with open('factual_modelMinistral-8B-Instruct-2410_p1.0_kNone_numprompts400_maxoutlen200_temp1.0.pkl', 'rb') as file:\n",
    "     data_M8B_p1_T1 = pickle.load(file)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "92499a36",
   "metadata": {},
   "outputs": [],
   "source": [
    "def split_m(output, m, tokenizer):\n",
    "    \"\"\"\n",
    "    Iteratively split tokens into two valid subtokens (prefix + suffix).\n",
    "\n",
    "    Args:\n",
    "        output (list[int]): Sequence of token ids.\n",
    "        m (int): Number of splitting iterations.\n",
    "        tokenizer: A tokenizer with decode() and encode() methods.\n",
    "\n",
    "    Returns:\n",
    "        list[int]: Modified token sequence after up to m splits.\n",
    "    \"\"\"\n",
    "    # Precompute dictionary: token_id -> string\n",
    "    id_to_str = {tid: tokenizer.decode([tid]) for tid in set(output)}\n",
    "\n",
    "    # Also precompute reverse lookup: string -> token_id (for fast matching)\n",
    "    # Only keep strings that decode back to single tokens\n",
    "    str_to_id = {}\n",
    "    for tid in tokenizer.get_vocab().values():\n",
    "        s = tokenizer.decode([tid])\n",
    "        str_to_id[s] = tid\n",
    "\n",
    "    for _ in range(m):\n",
    "        # Find indices of tokens with length > 1\n",
    "        multi_indices = [i for i, tid in enumerate(output) if len(id_to_str[tid]) > 1]\n",
    "        if not multi_indices:\n",
    "            return output  # Nothing to split\n",
    "\n",
    "        # Select a random token\n",
    "        idx = random.choice(multi_indices)\n",
    "        token_id = output[idx]\n",
    "        token_str = id_to_str[token_id]\n",
    "\n",
    "        # Try to split into prefix + suffix that are both in vocabulary\n",
    "        split_found = False\n",
    "        for j in range(1, len(token_str)):\n",
    "            prefix, suffix = token_str[:j], token_str[j:]\n",
    "            if prefix in str_to_id and suffix in str_to_id:\n",
    "                prefix_id = str_to_id[prefix]\n",
    "                suffix_id = str_to_id[suffix]\n",
    "\n",
    "                # Replace in sequence\n",
    "                output = output[:idx] + [prefix_id, suffix_id] + output[idx+1:]\n",
    "\n",
    "                # Update dictionaries if new tokens are introduced\n",
    "                if prefix_id not in id_to_str:\n",
    "                    id_to_str[prefix_id] = prefix\n",
    "                if suffix_id not in id_to_str:\n",
    "                    id_to_str[suffix_id] = suffix\n",
    "\n",
    "                split_found = True\n",
    "                break\n",
    "\n",
    "        # If no valid split found, skip this iteration\n",
    "        if not split_found:\n",
    "            continue\n",
    "\n",
    "    return output\n",
    "\n",
    "\n",
    "def get_profit_increase(data,m, tokenizer):\n",
    "        \n",
    "    total_tok = 0\n",
    "    total_tok_after = 0\n",
    "    num_prompts = len(data)\n",
    "\n",
    "    for prompt_idx in range(300):\n",
    "        output = [i.item() for i in data[prompt_idx][\"output\"][0]]\n",
    "        print(\"Prompt\", prompt_idx)\n",
    "        num_tok = len(output)\n",
    "        output_after = split_m(output, m, tokenizer)\n",
    "        num_tok_after = len(output_after)\n",
    "        #print(\"Num tok\", num_tok)\n",
    "        #print(\"after\", output_after)\n",
    "        #print(\"Num tok after\", num_tok_after)\n",
    "        total_tok += num_tok\n",
    "        total_tok_after += num_tok_after\n",
    "    return 100 * (total_tok_after / total_tok - 1)\n",
    "        \n",
    "        \n",
    "        \n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ff644e1b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import random\n",
    "import pickle\n",
    "\n",
    "def build_token_maps(tokenizer):\n",
    "    \"\"\"Precompute mapping between token ids and decoded strings.\"\"\"\n",
    "    vocab = tokenizer.get_vocab()\n",
    "    id_to_str = {}\n",
    "    str_to_id = {}\n",
    "    for tok, tid in vocab.items():\n",
    "        s = tokenizer.decode([tid])\n",
    "        if s not in str_to_id:  # avoid duplicates\n",
    "            str_to_id[s] = tid\n",
    "        id_to_str[tid] = s\n",
    "    return id_to_str, str_to_id\n",
    "\n",
    "\n",
    "def split_m(output, m, id_to_str, str_to_id):\n",
    "    \"\"\"\n",
    "    Iteratively split tokens into two valid subtokens (prefix + suffix).\n",
    "    \"\"\"\n",
    "    for _ in range(m):\n",
    "        # Find candidate indices (multi-character tokens)\n",
    "        multi_indices = [i for i, tid in enumerate(output) if len(id_to_str[tid]) > 1]\n",
    "        if not multi_indices:\n",
    "            break  # nothing left to split\n",
    "\n",
    "        idx = random.choice(multi_indices)\n",
    "        token_id = output[idx]\n",
    "        token_str = id_to_str[token_id]\n",
    "\n",
    "        # Try to split\n",
    "        for j in range(1, len(token_str)):\n",
    "            prefix, suffix = token_str[:j], token_str[j:]\n",
    "            if prefix in str_to_id and suffix in str_to_id:\n",
    "                # Replace in-place\n",
    "                output[idx:idx+1] = [str_to_id[prefix], str_to_id[suffix]]\n",
    "                break\n",
    "    return output\n",
    "\n",
    "\n",
    "def get_profit_increase(data, m, tokenizer, max_prompts=300):\n",
    "    id_to_str, str_to_id = build_token_maps(tokenizer)\n",
    "\n",
    "    total_tok = 0\n",
    "    total_tok_after = 0\n",
    "\n",
    "    for prompt_idx, prompt in enumerate(data[:max_prompts]):\n",
    "        #print(\"Prompt\", prompt_idx)\n",
    "        output = [i.item() for i in prompt[\"output\"][0]]\n",
    "        num_tok = len(output)\n",
    "        output_after = split_m(output, m, id_to_str, str_to_id)\n",
    "        num_tok_after = len(output_after)\n",
    "        total_tok += num_tok\n",
    "        total_tok_after += num_tok_after\n",
    "\n",
    "    return 100 * (total_tok_after / total_tok - 1)\n",
    "\n",
    "\n",
    "# Example usage\n",
    "range_m = [0, 300, 600, 800, 900, 1000, 1100, 1200, 1300]\n",
    "data = data_M8B_p1_T1\n",
    "tokenizer = tokenizer_M8B\n",
    "\n",
    "profit_increase = [\n",
    "    get_profit_increase(data, m, tokenizer)\n",
    "    for m in range_m\n",
    "]\n",
    "\n",
    "\n",
    "\n",
    "with open('random_proft_change_M8B.pkl', 'wb') as file:\n",
    "    pickle.dump(profit_increase, file)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "164850d7",
   "metadata": {},
   "outputs": [],
   "source": [
    "range_m = [0, 300, 600, 800, 900, 1000, 1100, 1200, 1300]\n",
    "\n",
    "sns.set_theme(context='paper', style='ticks', font_scale=1)\n",
    "width_pt = 469\n",
    "palette = sns.husl_palette(15, l=0.4)\n",
    "palette = sns.color_palette(\"Set2\")\n",
    "\n",
    "\n",
    "\n",
    "utils.latexify() # Computer Modern, with TeX\n",
    "# utils.latexify(font_serif='Times New Roman', font_size=10, usetex=False) # Times New Roman, without TeX\n",
    "\n",
    "\n",
    "fig_width, fig_height = utils.get_fig_dim(width_pt, fraction=0.6)\n",
    "fig, ax = plt.subplots(figsize=(fig_width, fig_height))\n",
    "\n",
    "with open('random_proft_change_L1B.pkl', 'rb') as file:\n",
    "    profit_increase_L1B_p1_T1 = pickle.load(file)\n",
    "with open('random_proft_change_L3B.pkl', 'rb') as file:\n",
    "    profit_increase_L3B_p1_T1 = pickle.load(file)\n",
    "with open('random_proft_change_G1B.pkl', 'rb') as file:\n",
    "    profit_increase_G1B_p1_T1 = pickle.load(file)\n",
    "with open('random_proft_change_G4B.pkl', 'rb') as file:\n",
    "    profit_increase_G4B_p1_T1 = pickle.load(file)\n",
    "with open('random_proft_change_M8B.pkl', 'rb') as file:\n",
    "    profit_increase_M8B_p1_T1 = pickle.load(file)\n",
    "\n",
    "\n",
    "sns.lineplot( x=range_m, y=[1* i for i in profit_increase_L1B_p1_T1], color=palette[0], ax=ax, linewidth=1, markersize=4,  linestyle=\"-\", markeredgewidth=0)\n",
    "# sns.lineplot( x=range_m, y=[1* i for i in profit_increase_L3B_p1_T1], palette=palette, ax=ax, linewidth=0.5, markersize=4,  linestyle=\"-\", markeredgewidth=0, label=\"Llama-3.2-3B-Instruct\")\n",
    "sns.lineplot( x=range_m, y=[1* i for i in profit_increase_G1B_p1_T1], color=palette[1], ax=ax, linewidth=1, markersize=4,  linestyle=\"-\", markeredgewidth=0)\n",
    "# sns.lineplot( x=range_m, y=[1* i for i in profit_increase_G4B_p1_T1], palette=palette, ax=ax, linewidth=0.5, markersize=4,  linestyle=\"-\", markeredgewidth=0, label=\"Gemma-3-4B-In\")\n",
    "sns.lineplot( x=range_m, y=[1* i for i in profit_increase_M8B_p1_T1], color=palette[2], ax=ax, linewidth=1, markersize=4,  linestyle=\"-\", markeredgewidth=0)  \n",
    "ax.set_ylabel(r\"Overcharged tokens $(\\%)$\", fontsize=14)\n",
    "ax.yaxis.set_label_coords(-0.15, 0.45)\n",
    "ax.set_xlabel(r\"Number of iterations, $m$\", fontsize=14)\n",
    "sns.despine(ax=ax)\n",
    "#ax.legend( frameon=True, loc='lower right', fontsize=7.5)\n",
    "\n",
    "ax.tick_params(axis='y', which='major', labelsize=12)\n",
    "ax.tick_params(axis='x', which='major', labelsize=12)\n",
    "fig.tight_layout()\n",
    "plt.show()\n",
    "fig.savefig(\"../figures/heur/random_profit_increase_no_transparency.pdf\", dpi=300)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ca45090e",
   "metadata": {},
   "source": [
    "# Plausibility"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c2b7bfbf",
   "metadata": {},
   "outputs": [],
   "source": [
    "model_name = \"meta-llama/Llama-3.2-1B-Instruct\"\n",
    "\n",
    "\n",
    "if model_name== \"meta-llama/Llama-3.2-1B-Instruct\":\n",
    "        model_str=\"Llama-3.2-1B-Instruct\"\n",
    "if model_name== \"mistralai/Ministral-8B-Instruct-2410\":\n",
    "        model_str=\"Ministral-8B-Instruct-2410\"\n",
    "if model_name== \"meta-llama/Llama-3.2-3B-Instruct\":\n",
    "        model_str=\"Llama-3.2-3B-Instruct\"\n",
    "if model_name== \"meta-llama/Llama-3.1-8B-Instruct\":\n",
    "        model_str=\"Meta-Llama-3.1-8B-Instruct\"\n",
    "if model_name== \"google/gemma-3-4b-it\":\n",
    "        model_str=\"Gemma-3-4b-it\"\n",
    "if model_name== \"google/gemma-3-1b-it\":\n",
    "        model_str=\"Gemma-3-1b-it\"\n",
    "\n",
    "\n",
    "tokenizer =  AutoTokenizer.from_pretrained(\"meta-llama/Llama-3.2-1B-Instruct\")\n",
    "with open(f'../outputs/cpt/factual_model{model_str}_p1.0_kNone_numprompts400_maxoutlen200_temp1.0.pkl', 'rb') as file:\n",
    "     data = pickle.load(file)\n",
    "\n",
    "char = [0] * len(data[0][\"output\"])\n",
    "tok = [0] * len(data[0][\"output\"])\n",
    "\n",
    "for num_seq in range(len(data[0][\"output\"])):\n",
    "     \n",
    "\n",
    "     for prompt_id in range(len(data)):\n",
    "          char[num_seq] += len(tokenizer.decode(  data[prompt_id][\"output\"][0]  ))\n",
    "          tok[num_seq] += len( data[prompt_id][\"output\"][0]  )\n",
    "\n",
    "\n",
    "\n",
    "print(\"L1\")\n",
    "print( \"Mean CPT\", np.mean( np.vstack(char) / np.stack(tok)  ) ) \n",
    "\n",
    "print(\"Mean overchard\", np.mean(  100 *( np.vstack(char) / np.stack(tok)  )-1 ,axis=0))   \n",
    "\n",
    "print(\"Mean std\", np.std( np.vstack(char) / np.stack(tok)  ))\n",
    "\n",
    "\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8d3cf7a1",
   "metadata": {},
   "outputs": [],
   "source": [
    "os.chdir(\"../outputs/cpt\")\n",
    "\n",
    "with open(f'../outputs/heuristic_new/L1B/heuristic_model_random_Llama-3.2-1B-Instruct_T_1.3_numseq_4_p_0.95.pkl', 'rb') as file:\n",
    "     data_09 = pickle.load(file)\n",
    "\n",
    "\n",
    "with open(f'../outputs/heuristic_new/G1B/heuristic_model_random_Gemma-3-1b-it_T_1.3_numseq_4_p_0.95.pkl', 'rb') as file:\n",
    "     data_095 = pickle.load(file)\n",
    "with open(f'../outputs/heuristic_new/M8B/heuristic_model_random_Ministral-8B-Instruct-2410_T_1.3_numseq_4_p_0.95.pkl', 'rb') as file:\n",
    "     data_099 = pickle.load(file)\n",
    "    \n",
    "print(data_09[\"top_p_count\"][0])\n",
    "print(data_095[\"top_p_count\"][0])\n",
    "print(data_099[\"top_p_count\"][0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "1496e6b6",
   "metadata": {},
   "outputs": [],
   "source": [
    "num_prompts = len(data_09[\"generated_outputs\"][0])\n",
    "\n",
    "#Define the splits used to generated the data, i.e. the number of iterations in the heuristic\n",
    "splits = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 22, 25, 30, 35, 40, 45, 50, 60, 70, 80, 90, 100]\n",
    "indices = [splits.index(element) for element in [1, 5 ,10 ,15 ,20, 25 ,30, 35,40,45,50,60,70,80,90,100]  ]\n",
    "\n",
    "# splits = [1,5,10, 15, 20 ,25 ,30 ,35, 40 ,45 ,50, 60,70,80,90,100, 110,  120, 130,  140,  150,  160,  170, 180 ,190, 200]\n",
    "\n",
    "# indices = [splits.index(element) for element in [1,5,10, 15, 20 ,25 ,30 ,35, 40 ,45 ,50, 60,70,80,90,100, 110,  120, 130,  140,  150,  160,  170, 180 ,190, 200]  ]\n",
    "\n",
    "\n",
    "\n",
    "percentage_09 = []\n",
    "percentage_095 = []\n",
    "percentage_099 = []\n",
    "\n",
    "\n",
    "for seq_idx in range(len((data_099[\"top_p_count\"]))):\n",
    "    percentage_09.append( np.array( [ data_09[\"top_p_count\"][seq_idx][i]  / num_prompts * 100  for i in range( len(splits))] ) )\n",
    "    percentage_095.append( np.array( [ data_095[\"top_p_count\"][seq_idx][i]  / num_prompts * 100  for i in range( len(splits))] ) )\n",
    "    percentage_099.append( np.array( [ data_099[\"top_p_count\"][seq_idx][i]   / num_prompts * 100  for i in range( len(splits))] ) )\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "mean_09 = np.mean( np.stack((percentage_09[0], percentage_09[1], percentage_09[2], percentage_09[3]),axis=0) , axis=0)\n",
    "std_09 = np.std( np.stack((percentage_09[0], percentage_09[1], percentage_09[2], percentage_09[3]),axis=0) , axis=0)\n",
    "\n",
    "mean_095 = np.mean( np.stack((percentage_095[0], percentage_095[1], percentage_095[2], percentage_095[3]),axis=0) , axis=0)\n",
    "std_095 = np.std( np.stack((percentage_095[0], percentage_095[1], percentage_095[2], percentage_095[3]),axis=0) , axis=0)\n",
    "\n",
    "mean_099 = np.mean( np.stack((percentage_099[0], percentage_099[1], percentage_099[2], percentage_099[3]),axis=0) , axis=0)\n",
    "std_099 = np.std( np.stack((percentage_099[0], percentage_099[1], percentage_099[2], percentage_099[3]),axis=0) , axis=0)\n",
    "\n",
    "\n",
    "splits_new = np.array(splits)[indices]\n",
    "\n",
    "\n",
    "mean_09_new = np.array(mean_09)[indices]\n",
    "mean_095_new = np.array(mean_095)[indices]\n",
    "mean_099_new = np.array(mean_099)[indices]\n",
    "std_09_new = np.array(std_09)[indices]\n",
    "std_095_new = np.array(std_095)[indices]\n",
    "std_099_new = np.array(std_099)[indices]\n",
    "\n",
    "\n",
    "\n",
    "df = pd.DataFrame({\n",
    "    'mean_percent':np.concatenate( (mean_09_new, mean_095_new, mean_099_new) ),\n",
    "    'top-p': ['085'] * len(mean_09_new) + ['09'] * len(mean_095_new) + ['095'] * len(mean_099_new),\n",
    "    'splits' : np.concatenate( (splits_new, splits_new, splits_new) ), \n",
    "})\n",
    "\n",
    "\n",
    "sns.set_theme(context='paper', style='ticks', font_scale=1)\n",
    "width_pt = 469\n",
    "palette = sns.color_palette('husl', 3)\n",
    "palette = sns.color_palette(\"Set2\")\n",
    "\n",
    "utils.latexify() # Computer Modern, with TeX\n",
    "# utils.latexify(font_serif='Times New Roman', font_size=10, usetex=False) # Times New Roman, without TeX\n",
    "\n",
    "\n",
    "fig_width, fig_height = utils.get_fig_dim(width_pt, fraction=0.6)\n",
    "fig, ax = plt.subplots(figsize=(fig_width, fig_height))\n",
    "\n",
    "\n",
    "sns.lineplot(data=df, x=\"splits\", y='mean_percent', hue='top-p', palette=palette, ax=ax, linewidth=0.5, markersize=4,marker=\"o\", err_style= \"band\", linestyle=\"-\", markeredgewidth=0)\n",
    "\n",
    "\n",
    "ax.fill_between(splits, mean_09-std_09 * t_val, mean_09+std_09 * t_val, color=palette[0], alpha=0.2)\n",
    "\n",
    "ax.fill_between(splits, mean_095-std_095 * t_val, mean_095+std_095 * t_val, color=palette[1], alpha=0.2)\n",
    "\n",
    "ax.fill_between(splits, mean_099-std_099 * t_val, mean_099+std_099 * t_val, color=palette[2], alpha=0.2)\n",
    "\n",
    "\n",
    "\n",
    "#Plot the maximum of each curve\n",
    "x_max_099 = splits[np.argmax(mean_099) ]      # x-coordinate at the maximum\n",
    "y_max_099 = mean_099[ np.argmax(mean_099)]      # Maximum y value\n",
    "\n",
    "x_max_09 = splits[np.argmax(mean_09) ]      # x-coordinate at the maximum\n",
    "y_max_09 = mean_09[ np.argmax(mean_09)]      # Maximum y value\n",
    "\n",
    "\n",
    "x_max_095 = splits[np.argmax(mean_095) ]      # x-coordinate at the maximum\n",
    "y_max_095 = mean_095[ np.argmax(mean_095)]      # Maximum y value\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "# Define custom legend elements\n",
    "legend_elements = [\n",
    "\n",
    "    Patch(facecolor=palette[2], edgecolor=\"white\", label=r\"$p=0.95$\"),\n",
    "    Patch(facecolor=palette[1], edgecolor=\"white\", label=r\"$p=0.90$\"),\n",
    "    Patch(facecolor=palette[0], edgecolor=\"white\", label=r\"$p=0.85$\")\n",
    "    # Color and label for 8B\n",
    "]\n",
    "\n",
    "ax.legend(handles=legend_elements, loc='upper right', fontsize=\"7.5\", frameon=True)\n",
    "\n",
    "\n",
    "#ax.set_yscale(\"log\")\n",
    "sns.despine(ax=ax)\n",
    "ax.set_xlabel(r\"Number of iterations, $m$\", fontsize=14)\n",
    "ax.set_ylabel(r\"Plausible sequences (\\%)\", fontsize=14)\n",
    "\n",
    "plt.legend().set_visible(False)\n",
    "ax.set_ylim(-2, 100)\n",
    "ax.set_yticks([0,20,40,60,80,100])\n",
    "\n",
    "ax.tick_params(axis='y', which='major', labelsize=12)\n",
    "ax.tick_params(axis='x', which='major', labelsize=12)\n",
    "\n",
    "ax.set_xlim(0,101)\n",
    "#ax.get_xaxis().set_visible(False)\n",
    "\n",
    "fig.tight_layout()\n",
    "plt.show()\n",
    "fig.savefig(f'../figures/heur/random_plausibility.pdf', dpi=300)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "envML",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.18"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
