{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "f7e99f4e",
   "metadata": {},
   "outputs": [],
   "source": [
    "%matplotlib inline\n",
    "\n",
    "import pickle\n",
    "from transformers import AutoTokenizer, AutoModelForCausalLM, pipeline\n",
    "\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",
    "import pandas as pd\n",
    "from src import utils\n",
    "from matplotlib.lines import Line2D\n",
    "import pynvml\n",
    "\n",
    "import time\n",
    "from matplotlib.patches import Patch\n",
    "from matplotlib.ticker import LogLocator\n",
    "\n",
    "t_val = 2.3\n",
    "\n",
    "class CustomLogLocator(LogLocator):\n",
    "    def __init__(self, base=10.0, subs=None, numticks=10, threshold=0.2):\n",
    "        super().__init__(base=base, subs=subs, numticks=numticks)\n",
    "        self.threshold = threshold\n",
    " \n",
    "    def tick_values(self, vmin, vmax):\n",
    "        ticks = super().tick_values(vmin, vmax)\n",
    "        # Filter ticks greater than the threshold\n",
    "        return [tick for tick in ticks if tick > self.threshold]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "442e097e",
   "metadata": {},
   "outputs": [],
   "source": [
    "with open(\"/../outputs/energy_outputs/energy_results_Gemma-3-1b-it.pkl\", \"rb\") as f:\n",
    "    data_G1 = pickle.load(f)\n",
    "\n",
    "with open(\"/../outputs/energy_outputs/energy_results_Gemma-3-4b-it.pkl\", \"rb\") as f:\n",
    "    data_G4 = pickle.load(f)\n",
    "    \n",
    "with open(\"/../outputs/energy_outputs/energy_results_Llama-3.2-1B-Instruct.pkl\", \"rb\") as f:\n",
    "    data_L1 = pickle.load(f)\n",
    "    \n",
    "with open(\"/../outputs/energy_outputs/energy_results_Llama-3.2-3B-Instruct.pkl\", \"rb\") as f:\n",
    "    data_L3 = pickle.load(f)\n",
    "    \n",
    "with open(\"/../outputs/energy_outputs/energy_results_Ministral-8B-Instruct-2410.pkl\", \"rb\") as f:\n",
    "    data_M8 = pickle.load(f)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5991550b",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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zS7vT6VSDA4Oks1mefvtt0mlUrz66qu89dZbPP/88/WTeEeOHAHgiSeeAGp/4EeOHGH//v0kk8n6cw4dOsTBgwc5evQor732Grt27eLgwYPE4/El+33hhReIxWK33O/evXtv+FquDXSpVIpDhw7VR5lDQ0P09/fXpxJms1mefPLJG45OP/jgg5v22a2O80nlEgAOHz58yxHujcol9PX1LfkdXNufyWSSb33rW/UPiMUPq1QqRV9fX70PFvu1u7u7PkXy6NGjjIyM8K1vfeu6sgyLgXzxvjfffJPTp09z6NCh+u/hyJEjJBIJkskkmUxmyQfZtcfTVpcO9tfQwV7TtPuRTuNomqatATrYa5qmrQE62Guapq0BOthrmqatATrYa5qmrQE62Guapq0BOthrmqatATrYa5qmrQE62Guapq0BOthrmqatATrYa5qmrQE62Guapq0BOthrmqatATrYa5qmrQE62Guapq0BOthrmqatATrYa5qmrQH3xILjR44cqS8xB7Xl0YaGhurrUl69VNpKt2mapq1p6i4bGRlRW7duVZlMpn7f17/+9frPY2Nj6s///M9ve9unbc/IyMinfq6madq96q6ncVKp1JJFi69eoR5qixefPHnytrZpmqatdXc12A8NDdVXnV80PDxcX+1+UTweZ3R0dMXbNE3T1rq7lrPPZrM3zKdns9kbPj6Tyax4241MT08zMzNz3f3nzp27WZM1TdM+t+5asB8cHKSvr2/Zj79ZMF/ptoGBAX70ox8t+/iapmmfZ3cl2A8PD7Nv374bbovFYteNxjOZDLFYbMXbbqSvr4+vfvWr191/7tw5Dh8+/GlejqZp2j3vro7sF6VSKV599VX2799Pb28vAwMD1z1+165dJJPJFW27kdbWVlpbW2/jFWiapn1+3JVg39vbu+T2Sy+9xLPPPrtkVs6iVCrFrl276qP3lWzTNE1b6z51sB8fH2doaIjh4eElaZNEIkFvby9PP/00XV1dy9pXNputj8aPHj1KX18fPT09vPLKKxw5coTdu3dz6tQpXnnllfpzVrpN0zRtLRNKKbXcB/f39yOEYN++fezcufO67adPn+bNN99ECMFf/dVfrWpDPyujo6McOHCA48eP09PTc7ebo2matiqWPbJ//fXXeeGFF4hGozd9zM6dO9m5cye5XI7vf//7n9uAr2madr9Z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",
      "text/plain": [
       "<Figure size 389.373x240.646 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.clf()\n",
    "sns.set_theme(context='paper', style='ticks', font_scale=1)\n",
    "width_pt = 469\n",
    "palette = sns.color_palette('husl', 3)\n",
    "\n",
    "utils.latexify() # Computer Modern, with TeX\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",
    "data_frame = pd.DataFrame(data_L3)\n",
    "\n",
    "sns.scatterplot(data=data_frame, x='out_length', y='gen_energy', label='Generation Energy', color=palette[2], s=15, alpha=0.5, legend = False)\n",
    "sns.scatterplot(data=data_frame, x='out_length', y='score_energy', label='Scoring Energy', color=palette[1], s=15, alpha=0.5, legend = False)\n",
    "\n",
    "ax.set_xlabel(\"Output Length (tokens)\")\n",
    "ax.set_ylabel(\"Energy (Joules)\")\n",
    "#Remove top and right spines\n",
    "sns.despine(ax=ax)\n",
    "ax.set_xlim(100, 510)\n",
    "ax.set_ylim(-100, 5000)\n",
    "\n",
    "#Add a text with the model name\n",
    "ax.text(0.5, 0.95, \"Llama-3.2-3B-Instruct\", transform=ax.transAxes, ha='center', fontsize=12)\n",
    "\n",
    "\n",
    "\n",
    "fig.tight_layout()\n",
    "fig.savefig('/../figures/energy/energy_L3.pdf', dpi=300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "3c94b227",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Average scoring cost for L1:  15.188899605847121\n",
      "Standard deviation:  0.17998998395693833\n",
      "Average generation cost per token for L1:  2.642089241176297\n",
      "Standard deviation:  0.13943733022530838\n",
      "----------------------------------------\n",
      "Average scoring cost  for L3:  15.903015182751416\n",
      "Standard deviation:  0.2251979090361956\n",
      "Average generation cost per token for L3:  4.805773798810014\n",
      "Standard deviation:  0.151391018785545\n",
      "----------------------------------------\n",
      "Average scoring cost for G1:  12.737264256523133\n",
      "Standard deviation:  0.08006648731049276\n",
      "Average generation cost per token for G1:  6.482024071432994\n",
      "Standard deviation:  0.20635111273991025\n",
      "----------------------------------------\n",
      "Average scoring cost for G4:  33.51756702235285\n",
      "Standard deviation:  4.285303360603443\n",
      "Average generation cost per token for G4:  9.992732220092629\n",
      "Standard deviation:  0.1786623631454739\n",
      "----------------------------------------\n",
      "Average scoring cost  for M8:  17.740518088517906\n",
      "Standard deviation:  0.2610677358736029\n",
      "Average generation cost per token for M8:  7.342451551694223\n",
      "Standard deviation:  0.2913893686473248\n",
      "----------------------------------------\n"
     ]
    }
   ],
   "source": [
    "#For each model, print the average scoring cost, and the average generation cost per token\n",
    "#Then print their standar deviations\n",
    "\n",
    "print(\"Average scoring cost for L1: \", np.mean( [data_L1[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))\n",
    "print(\"Standard deviation: \", np.std( [data_L1[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))\n",
    "\n",
    "print(\"Average generation cost per token for L1: \", np.mean( [data_L1[i][\"gen_energy\"]/data_L1[i][\"out_length\"] for i in range(len(data_G1))  ] )  )\n",
    "print(\"Standard deviation: \", np.std( [data_L1[i][\"gen_energy\"]/data_L1[i][\"out_length\"] for i in range(len(data_G1))  ] )  )\n",
    "print(\"----------------------------------------\")\n",
    "print(\"Average scoring cost  for L3: \", np.mean( [data_L3[i][\"score_energy\"] for i in range(len(data_L3))  ]    ))\n",
    "print(\"Standard deviation: \", np.std( [data_L3[i][\"score_energy\"] for i in range(len(data_L3))  ]    ))\n",
    "\n",
    "print(\"Average generation cost per token for L3: \", np.mean( [data_L3[i][\"gen_energy\"]/data_L3[i][\"out_length\"] for i in range(len(data_L3))  ] )  )\n",
    "print(\"Standard deviation: \", np.std( [data_L3[i][\"gen_energy\"]/data_L3[i][\"out_length\"] for i in range(len(data_L3))  ] )  )\n",
    "print(\"----------------------------------------\")\n",
    "print(\"Average scoring cost for G1: \", np.mean( [data_G1[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))\n",
    "print(\"Standard deviation: \", np.std( [data_G1[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))\n",
    "\n",
    "print(\"Average generation cost per token for G1: \", np.mean( [data_G1[i][\"gen_energy\"]/data_G1[i][\"out_length\"] for i in range(len(data_G1))  ] )  )\n",
    "print(\"Standard deviation: \", np.std( [data_G1[i][\"gen_energy\"]/data_G1[i][\"out_length\"] for i in range(len(data_G1))  ] )  )\n",
    "print(\"----------------------------------------\")\n",
    "print(\"Average scoring cost for G4: \", np.mean( [data_G4[i][\"score_energy\"] for i in range(len(data_G4))  ]    ))\n",
    "print(\"Standard deviation: \", np.std( [data_G4[i][\"score_energy\"] for i in range(len(data_G4))  ]    ))\n",
    "\n",
    "print(\"Average generation cost per token for G4: \", np.mean( [data_G4[i][\"gen_energy\"]/data_G4[i][\"out_length\"] for i in range(len(data_G4))  ] )  )\n",
    "print(\"Standard deviation: \", np.std( [data_G4[i][\"gen_energy\"]/data_G4[i][\"out_length\"] for i in range(len(data_G4))  ] )  )\n",
    "print(\"----------------------------------------\")\n",
    "print(\"Average scoring cost  for M8: \", np.mean( [data_M8[i][\"score_energy\"] for i in range(len(data_G4))  ]    ))\n",
    "print(\"Standard deviation: \", np.std( [data_M8[i][\"score_energy\"] for i in range(len(data_G4))  ]    ))\n",
    "\n",
    "print(\"Average generation cost per token for M8: \", np.mean( [data_M8[i][\"gen_energy\"]/data_M8[i][\"out_length\"] for i in range(len(data_G4))  ] )  )\n",
    "print(\"Standard deviation: \", np.std( [data_M8[i][\"gen_energy\"]/data_M8[i][\"out_length\"] for i in range(len(data_G4))  ] )  )\n",
    "print(\"----------------------------------------\")\n",
    "\n",
    "\n",
    "#save the (mean per token generation) / (mean scoring cost) for each model\n",
    "\n",
    "gen_to_score_ratio_L1 =  (np.mean( [data_L1[i][\"gen_energy\"]/data_L1[i][\"out_length\"] for i in range(len(data_G1))  ] )) / (np.mean( [data_L1[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))\n",
    "\n",
    "gen_to_score_ratio_L3 =  (np.mean( [data_L3[i][\"gen_energy\"]/data_L3[i][\"out_length\"] for i in range(len(data_G1))  ] )) / (np.mean( [data_L3[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))\n",
    "\n",
    "gen_to_score_ratio_G1 =  (np.mean( [data_G1[i][\"gen_energy\"]/data_G1[i][\"out_length\"] for i in range(len(data_G1))  ] )) / (np.mean( [data_G1[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))\n",
    "\n",
    "\n",
    "gen_to_score_ratio_G4 =  (np.mean( [data_G4[i][\"gen_energy\"]/data_G4[i][\"out_length\"] for i in range(len(data_G1))  ] )) / (np.mean( [data_G4[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))\n",
    "\n",
    "\n",
    "gen_to_score_ratio_M8 =  (np.mean( [data_M8[i][\"gen_energy\"]/data_M8[i][\"out_length\"] for i in range(len(data_G1))  ] )) / (np.mean( [data_M8[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "46bb634b",
   "metadata": {},
   "outputs": [],
   "source": [
    "#For each model, compute the erropr on the per-token geneartion cost, and the error in the scoring cost\n",
    "gen_cost_error_L1 = np.std( [data_L1[i][\"gen_energy\"]/data_L1[i][\"out_length\"] for i in range(len(data_G1))  ] ) / np.sqrt(len(data_L1))\n",
    "score_cost_error_L1 = np.std( [data_L1[i][\"score_energy\"] for i in range(len(data_G1))  ] ) / np.sqrt(len(data_L1))\n",
    "\n",
    "gen_cost_error_L3 = np.std( [data_L3[i][\"gen_energy\"]/data_L3[i][\"out_length\"] for i in range(len(data_G1))  ] ) / np.sqrt(len(data_L3))\n",
    "score_cost_error_L3 = np.std( [data_L3[i][\"score_energy\"] for i in range(len(data_G1))  ] ) / np.sqrt(len(data_L3))\n",
    "\n",
    "gen_cost_error_G1 = np.std( [data_G1[i][\"gen_energy\"]/data_G1[i][\"out_length\"] for i in range(len(data_G1))  ] ) / np.sqrt(len(data_G1))\n",
    "score_cost_error_G1 = np.std( [data_G1[i][\"score_energy\"] for i in range(len(data_G1))  ] ) / np.sqrt(len(data_G1))\n",
    "\n",
    "gen_cost_error_G4 = np.std( [data_G4[i][\"gen_energy\"]/data_G4[i][\"out_length\"] for i in range(len(data_G1))  ] ) / np.sqrt(len(data_G4))\n",
    "score_cost_error_G4 = np.std( [data_G4[i][\"score_energy\"] for i in range(len(data_G1))  ] ) / np.sqrt(len(data_G4))\n",
    "\n",
    "gen_cost_error_M8 = np.std( [data_M8[i][\"gen_energy\"]/data_M8[i][\"out_length\"] for i in range(len(data_G1))  ] ) / np.sqrt(len(data_M8))\n",
    "score_cost_error_M8 = np.std( [data_M8[i][\"score_energy\"] for i in range(len(data_G1))  ] ) / np.sqrt(len(data_M8))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "77567f40",
   "metadata": {},
   "outputs": [],
   "source": [
    "#Now, for each model, compute the error in the ratio\n",
    "gen_to_score_ratio_L1_error = np.std( [data_L1[i][\"gen_energy\"]/data_L1[i][\"out_length\"] for i in range(len(data_G1))  ] ) / (np.mean( [data_L1[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))\n",
    "\n",
    "gen_to_score_ratio_L3_error = np.std( [data_L3[i][\"gen_energy\"]/data_L3[i][\"out_length\"] for i in range(len(data_G1))  ] ) / (np.mean( [data_L3[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))\n",
    "\n",
    "gen_to_score_ratio_G1_error = np.std( [data_G1[i][\"gen_energy\"]/data_G1[i][\"out_length\"] for i in range(len(data_G1))  ] ) / (np.mean( [data_G1[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))\n",
    "\n",
    "gen_to_score_ratio_G4_error = np.std( [data_G4[i][\"gen_energy\"]/data_G4[i][\"out_length\"] for i in range(len(data_G1))  ] ) / (np.mean( [data_G4[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))\n",
    "\n",
    "gen_to_score_ratio_M8_error = np.std( [data_M8[i][\"gen_energy\"]/data_M8[i][\"out_length\"] for i in range(len(data_G1))  ] ) / (np.mean( [data_M8[i][\"score_energy\"] for i in range(len(data_G1))  ]    ))\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "71afd039",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Rho min=  -14.489471499708221\n",
      "Optimal m  90\n",
      "Rho min=  -6.154956273913318\n",
      "Optimal m  45\n",
      "Rho min=  -2.138592532784343\n",
      "Optimal m  35\n",
      "Model:  M8B\n",
      "Temp:  1.45\n",
      "-------------------------\n",
      "Lim 0.99:  19.210300782367163\n",
      "Lim 0.95:  9.075525353804315\n",
      "Lim 0.9:  3.903731524285874\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 389.373x240.646 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "model = \"M8B\"\n",
    "temp = \"1.45\"\n",
    "\n",
    "#Define the gen_to_score based on the model name\n",
    "if model == \"L1B\":\n",
    "    gen_to_score = gen_to_score_ratio_L1\n",
    "elif model == \"L3B\":\n",
    "    gen_to_score = gen_to_score_ratio_L3\n",
    "elif model == \"G1B\":\n",
    "    gen_to_score = gen_to_score_ratio_G1\n",
    "elif model == \"G4B\":\n",
    "    gen_to_score = gen_to_score_ratio_G4\n",
    "elif model == \"M8B\":\n",
    "    gen_to_score = gen_to_score_ratio_M8\n",
    "    \n",
    "\n",
    "    \n",
    "    \n",
    "if temp == \"1.15\" or temp == \"1.3\":\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",
    "elif temp == \"1.45\":\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",
    "\n",
    "if temp == \"1.15\":\n",
    "    if model == \"M8B\":\n",
    "        with open(\"/../outputs/heuristic_new/M8B/heuristic_model_Ministral-8B-Instruct-2410_T_1.15_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]_numseq_3_p_0.9_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/M8B/heuristic_model_Ministral-8B-Instruct-2410_T_1.15_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]_numseq_3_p_0.95_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/M8B/heuristic_model_Ministral-8B-Instruct-2410_T_1.15_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]_numseq_3_p_0.99_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "    elif model == \"L1B\":\n",
    "        with open(\"/../outputs/heuristic_new/L1B/heuristic_model_Llama-3.2-1B-Instruct_T_1.15_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]_numseq_3_p_0.9_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/L1B/heuristic_model_Llama-3.2-1B-Instruct_T_1.15_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]_numseq_3_p_0.95_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/L1B/heuristic_model_Llama-3.2-1B-Instruct_T_1.15_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]_numseq_3_p_0.99_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "    elif model == \"L3B\":\n",
    "        with open(\"/../outputs/heuristic_new/L3B/heuristic_model_Llama-3.2-3B-Instruct_T_1.15_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]_numseq_3_p_0.9_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/L3B/heuristic_model_Llama-3.2-3B-Instruct_T_1.15_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]_numseq_3_p_0.95_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/L3B/heuristic_model_Llama-3.2-3B-Instruct_T_1.15_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]_numseq_3_p_0.99_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "    elif model == \"G1B\":\n",
    "        with open(\"/../outputs/heuristic_new/G1B/heuristic_model_Gemma-3-1b-it_T_1.15_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]_numseq_3_p_0.9_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/G1B/heuristic_model_Gemma-3-1b-it_T_1.15_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]_numseq_3_p_0.95_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/G1B/heuristic_model_Gemma-3-1b-it_T_1.15_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]_numseq_3_p_0.99_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "    elif model == \"G4B\":\n",
    "        with open(\"/../outputs/heuristic_new/G4B/heuristic_model_Gemma-3-4b-it_T_1.15_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]_numseq_3_p_0.9_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/G4B/heuristic_model_Gemma-3-4b-it_T_1.15_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]_numseq_3_p_0.95_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/G4B/heuristic_model_Gemma-3-4b-it_T_1.15_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]_numseq_3_p_0.99_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "    \n",
    "    \n",
    "        \n",
    "    \n",
    "if temp == \"1.3\":\n",
    "    if model == \"M8B\":\n",
    "        with open(\"/../outputs/heuristic_new/M8B/heuristic_model_Ministral-8B-Instruct-2410_T_1.3_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]_numseq_3_p_0.9_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/M8B/heuristic_model_Ministral-8B-Instruct-2410_T_1.3_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]_numseq_3_p_0.95_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/M8B/heuristic_model_Ministral-8B-Instruct-2410_T_1.3_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]_numseq_3_p_0.99_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "    elif model == \"L1B\":\n",
    "        with open(\"/../outputs/heuristic_new/L1B/heuristic_model_Llama-3.2-1B-Instruct_T_1.3_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]_numseq_3_p_0.9_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/L1B/heuristic_model_Llama-3.2-1B-Instruct_T_1.3_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]_numseq_3_p_0.95_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/L1B/heuristic_model_Llama-3.2-1B-Instruct_T_1.3_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]_numseq_3_p_0.99_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "    elif model == \"L3B\":\n",
    "        with open(\"/../outputs/heuristic_new/L3B/heuristic_model_Llama-3.2-3B-Instruct_T_1.3_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]_numseq_3_p_0.9_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/L3B/heuristic_model_Llama-3.2-3B-Instruct_T_1.3_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]_numseq_3_p_0.95_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/L3B/heuristic_model_Llama-3.2-3B-Instruct_T_1.3_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]_numseq_3_p_0.99_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "    elif model == \"G1B\":\n",
    "        with open(\"/../outputs/heuristic_new/G1B/heuristic_model_Gemma-3-1b-it_T_1.3_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]_numseq_3_p_0.9_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/G1B/heuristic_model_Gemma-3-1b-it_T_1.3_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]_numseq_3_p_0.95_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/G1B/heuristic_model_Gemma-3-1b-it_T_1.3_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]_numseq_3_p_0.99_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "    elif model == \"G4B\":\n",
    "        with open(\"/../outputs/heuristic_new/G4B/heuristic_model_Gemma-3-4b-it_T_1.3_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]_numseq_3_p_0.9_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/G4B/heuristic_model_Gemma-3-4b-it_T_1.3_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]_numseq_3_p_0.95_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/G4B/heuristic_model_Gemma-3-4b-it_T_1.3_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]_numseq_3_p_0.99_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "\n",
    "    \n",
    "\n",
    "if temp == \"1.45\":\n",
    "    if model == \"M8B\":\n",
    "        with open(\"/../outputs/heuristic_new/M8B/heuristic_model_Ministral-8B-Instruct-2410_T_1.45_splits_numseq_3_p_0.9.pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/M8B/heuristic_model_Ministral-8B-Instruct-2410_T_1.45_splits_numseq_3_p_0.95.pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/M8B/heuristic_model_Ministral-8B-Instruct-2410_T_1.45_splits_numseq_3_p_0.99.pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "    elif model == \"L1B\":\n",
    "        with open(\"/../outputs/heuristic_new/L1B/heuristic_model_Llama-3.2-1B-Instruct_T_1.45_splits_numseq_3_p_0.9.pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/L1B/heuristic_model_Llama-3.2-1B-Instruct_T_1.45_splits_numseq_3_p_0.95.pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/L1B/heuristic_model_Llama-3.2-1B-Instruct_T_1.45_splits_numseq_3_p_0.99.pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "    elif model == \"L3B\":\n",
    "        with open(\"/../outputs/heuristic_new/L3B/heuristic_model_Llama-3.2-3B-Instruct_T_1.45_splits_numseq_3_p_0.9.pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/L3B/heuristic_model_Llama-3.2-3B-Instruct_T_1.45_splits_numseq_3_p_0.95.pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/L3B/heuristic_model_Llama-3.2-3B-Instruct_T_1.45_splits_numseq_3_p_0.99.pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "    elif model == \"G1B\":\n",
    "        with open(\"/../outputs/heuristic_new/G1B/heuristic_model_Gemma-3-1b-it_T_1.45_splits_numseq_3_p_0.9_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/G1B/heuristic_model_Gemma-3-1b-it_T_1.45_splits_numseq_3_p_0.95_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/G1B/heuristic_model_Gemma-3-1b-it_T_1.45_splits_numseq_3_p_0.99_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "    elif model == \"G4B\":\n",
    "        with open(\"/../outputs/heuristic_new/G4B/heuristic_model_Gemma-3-4b-it_T_1.45_splits_numseq_3_p_0.9_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_09 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/G4B/heuristic_model_Gemma-3-4b-it_T_1.45_splits_numseq_3_p_0.95_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_095 = pickle.load(f)\n",
    "        with open(\"/../outputs/heuristic_new/G4B/heuristic_model_Gemma-3-4b-it_T_1.45_splits_numseq_3_p_0.99_k_None_prompt_idare you .pkl\", \"rb\") as f:\n",
    "            data_099 = pickle.load(f)\n",
    "\n",
    "    \n",
    " #-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- \n",
    "def get_profit(splits, c0_to_cv, rho_array, data):\n",
    "    misreport = False\n",
    "    #Obtain if misreport is viable for the rho, ie, determine the m mazimizin splits * probs - (1-rho) * energy_ratio\n",
    "    \n",
    "\n",
    "    probs_0 = np.array([data[\"top_p_count\"][0][i] / len(data[\"generated_outputs\"][0]) for i in range(len(splits))])\n",
    "    probs_1 = np.array([data[\"top_p_count\"][1][i] / len(data[\"generated_outputs\"][0]) for i in range(len(splits))])\n",
    "    probs_2 = np.array([data[\"top_p_count\"][2][i] / len(data[\"generated_outputs\"][0]) for i in range(len(splits))])\n",
    "\n",
    "    mean_probs = np.mean(np.stack((probs_0, probs_1, probs_2), axis=0), axis=0)\n",
    "    rho_min = 1 - c0_to_cv * max([mean_probs[i] * splits[i]  for i in range(len(splits))])\n",
    "    print(\"Rho min= \", rho_min)\n",
    "    \n",
    "    coef_0 = max([probs_0[i] * splits[i]  - 1 / c0_to_cv   for i in range(len(splits))])\n",
    "    coef_1 = max([probs_1[i] * splits[i]  - 1 / c0_to_cv  for i in range(len(splits))])\n",
    "    coef_2 = max([probs_2[i] * splits[i]  - 1 / c0_to_cv  for i in range(len(splits))])\n",
    "    \n",
    "\n",
    "    \n",
    "    num_prompts = len(data_095[\"generated_outputs\"][0])\n",
    "    total_num_tok = len(data[\"generated_outputs\"][0])\n",
    "    \n",
    "    total_num_tokens = sum(data_095[\"generated_outputs\"][0])\n",
    "\n",
    "    increase_0 = [100 * (num_prompts) / total_num_tokens * max(0, 1/rho * (coef_0) + 1/ c0_to_cv ) for rho in rho_array] \n",
    "    \n",
    "    increase_1 = [100 * (num_prompts) / total_num_tokens * max(0, 1/rho * (coef_1) + 1/ c0_to_cv ) for rho in rho_array]\n",
    "    \n",
    "    increase_2 = [100 * (num_prompts) / total_num_tokens * max(0, 1/rho * (coef_2) + 1/ c0_to_cv ) for rho in rho_array] \n",
    "    \n",
    "    mean = np.mean( np.stack((increase_0, increase_1, increase_2),axis=0) , axis=0)\n",
    "    std = np.std( np.stack((increase_0, increase_1, increase_2),axis=0) , axis=0)\n",
    "        \n",
    "    return  mean,std,rho_min\n",
    "\n",
    "rho_range = np.arange(0.01, 1.05, 0.05)\n",
    "\n",
    "plt.clf()\n",
    "sns.set_theme(context='paper', style='ticks', font_scale=1)\n",
    "width_pt = 469\n",
    "palette = sns.color_palette('husl', 3)\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",
    "\n",
    "mean_099, std_099 ,rho_min_099 = get_profit(splits, gen_to_score, rho_range, data_099)\n",
    "\n",
    "mean_095, std_095, rho_min_095 = get_profit(splits, gen_to_score, rho_range, data_095)\n",
    "\n",
    "mean_09, std_09 , rho_min_09= get_profit(splits, gen_to_score, rho_range, data_09)\n",
    "print(\"Model: \", model)\n",
    "print(\"Temp: \", temp)\n",
    "print(\"-------------------------\")\n",
    "print(\"Lim 0.99: \", mean_099[-1])\n",
    "print(\"Lim 0.95: \", mean_095[-1])\n",
    "print(\"Lim 0.9: \", mean_09[-1])\n",
    "\n",
    "sns.lineplot(x=rho_range[mean_099>0], y=mean_099[mean_099>0] , color=palette[2], linewidth=0.5, markersize=4, marker=\"o\", linestyle=\"-\", markeredgewidth=0)\n",
    "sns.lineplot(x=rho_range[mean_095>0], y=mean_095[mean_095>0] , color=palette[1], linewidth=0.5, markersize=4, marker=\"o\", linestyle=\"-\", markeredgewidth=0)\n",
    "sns.lineplot(x=rho_range[mean_09>0], y=mean_09[mean_09>0] , color=palette[0], linewidth=0.5, markersize=4, marker=\"o\", linestyle=\"-\", markeredgewidth=0)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "#Plot the error band\n",
    "ax.fill_between(rho_range[mean_099>0], mean_099[mean_099>0] - std_099[mean_099>0]* t_val, mean_099[mean_099>0] + std_099[mean_099>0] * t_val, alpha=0.2, color=palette[2])\n",
    "ax.fill_between(rho_range[mean_095>0], mean_095[mean_095>0] - std_095[mean_095>0] * t_val, mean_095[mean_095>0] + std_095[mean_095>0] * t_val, alpha=0.2, color=palette[1])\n",
    "ax.fill_between(rho_range[mean_09>0], mean_09[mean_09>0] - std_09[mean_09>0] * t_val, mean_09[mean_09>0] + std_09[mean_09>0] * t_val, alpha=0.2, color=palette[0])\n",
    "\n",
    "\n",
    "ax.axvline(x=rho_range[mean_099>0][0], color=palette[2], linestyle='--', linewidth=0.5, label=\"No margin\") \n",
    "ax.axvline(x=rho_range[mean_095>0][0], color=palette[1], linestyle='--', linewidth=0.5, label=\"No margin\") \n",
    "ax.axvline(x=rho_range[mean_09>0][0], color=palette[0], linestyle='--', linewidth=0.5, label=\"No margin\") \n",
    "\n",
    "ax.set_ylabel(r\"Increase in utility $(\\%)$\")\n",
    "ax.set_xlim(0.1, 1.05)\n",
    "ax.set_ylim(-0.1, 100)\n",
    "\n",
    "ax.set_yscale(\"symlog\", linthresh=1)\n",
    "ax.get_yaxis().set_major_formatter(plt.ScalarFormatter())\n",
    "\n",
    "ax.yaxis.set_minor_locator(plt.FixedLocator(\n",
    "    [0, 0.1, 0.2, 0.3, 0.4, 0.5, 0.6, 0.7, 0.8, 0.9, 1,\n",
    "     2, 3, 4, 5, 6, 7, 8, 9, 10, 20, 30, 40, 50, 60, 70, 80, 90, 100]\n",
    "))\n",
    "\n",
    "\n",
    "if model != \"M8B\":\n",
    "    ax.get_xaxis().set_visible(False)\n",
    "\n",
    "if temp != \"1.45\":\n",
    "    ax.set_ylabel(\"\")\n",
    "    #remove all tick labels in y axis, but keep the ticks\n",
    "    ax.tick_params(axis='y', which='both', labelleft=False, labelright=False)\n",
    "\n",
    "sns.despine(ax=ax)\n",
    "ax.set_xlabel(r\"Provider's profit margin, $\\rho_o$\")\n",
    "plt.tight_layout()\n",
    "\n",
    "fig.savefig(f'/../figures/heur/profit/{model}_profit_{temp}.pdf', dpi=300)\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "env",
   "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.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
