{
 "cells": [
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Run length exploration\n",
    "\n",
    "Notebook for doing some EDA on run lengths as a raw measurement for a new detection algorithm/hypo test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "INPUT_DIR = \"/scratch/<username>/watermarking-root/input\"\n",
    "OUTPUT_DIR = \"/scratch/<username>/watermarking-root/output\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/scratch/<username>/miniconda3/envs/watermarking-dev/lib/python3.10/site-packages/tqdm/auto.py:22: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
      "  from .autonotebook import tqdm as notebook_tqdm\n"
     ]
    }
   ],
   "source": [
    "# Basic imports\n",
    "import os\n",
    "\n",
    "from tqdm import tqdm\n",
    "from statistics import mean\n",
    "\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import torch\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from matplotlib import rc\n",
    "rc('font', **{'family': 'serif', 'serif': ['Computer Modern']})\n",
    "rc('text', usetex=True)\n",
    "\n",
    "import cmasher as cmr"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "from datasets import load_from_disk"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Load the processed dataset/frame"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "raw_generations_df  = False\n",
    "\n",
    "# save_name = \"analysis_ds_1-21_greedy_redo\" \n",
    "# save_name = \"analysis_ds_1-21_greedy_redo_truncated_sanity_check\"\n",
    "# save_name = \"analysis_ds_1-23_greedy_gamma_0-25_truncated\" \n",
    "# save_name = \"analysis_ds_1-23_greedy_gamma_0-25_0-5_truncated\" # in figure (not 100% sure this is correct, check)\n",
    "\n",
    "save_name = \"analysis_ds_1-20_more_attack\" # in figure\n",
    "\n",
    "# save_name = \"analysis_ds_1-19_realnews_1-3_v1\" # in figure\n",
    "# save_name = \"analysis_ds_1-23_en_1-3\"\n",
    "# save_name = \"analysis_ds_1-23_pile_1-3\"\n",
    "\n",
    "# raw_generations_df  = True\n",
    "# save_name = \"analysis_ds_3-31_turbo_attack_old_realnews_1-3\" # davinci attack\n",
    "\n",
    "save_dir = f\"{INPUT_DIR}/processed_datasets/{save_name}\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "raw_data = load_from_disk(save_dir)"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### convert to pandas df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "df = raw_data.to_pandas()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Orig number of rows: 6867\n"
     ]
    },
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>idx</th>\n",
       "      <th>truncated_input</th>\n",
       "      <th>baseline_completion</th>\n",
       "      <th>orig_sample_length</th>\n",
       "      <th>prompt_length</th>\n",
       "      <th>real_completion_length</th>\n",
       "      <th>no_bl_gen_time</th>\n",
       "      <th>w_bl_gen_time</th>\n",
       "      <th>spike_entropies</th>\n",
       "      <th>no_bl_output</th>\n",
       "      <th>...</th>\n",
       "      <th>w_bl_hit_list</th>\n",
       "      <th>w_bl_attacked_num_toks_gend_eq_0</th>\n",
       "      <th>w_bl_attacked_num_tokens_generated</th>\n",
       "      <th>w_bl_attacked_hit_list</th>\n",
       "      <th>avg_spike_entropy</th>\n",
       "      <th>w_bl_exp_num_wl_tokens</th>\n",
       "      <th>w_bl_var_num_wl_tokens</th>\n",
       "      <th>exp_wl_coef</th>\n",
       "      <th>w_bl_exp_whitelist_fraction</th>\n",
       "      <th>w_bl_var_whitelist_fraction</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>6862</th>\n",
       "      <td>2445</td>\n",
       "      <td>Why is the US national security Establishment ...</td>\n",
       "      <td>get what I’ve been saying for well over a yea...</td>\n",
       "      <td>1026</td>\n",
       "      <td>826</td>\n",
       "      <td>200</td>\n",
       "      <td>9.984011</td>\n",
       "      <td>10.297770</td>\n",
       "      <td>[[0.9045331478118896, 0.8804982304573059, 0.83...</td>\n",
       "      <td>believe me when I tell them that he is the be...</td>\n",
       "      <td>...</td>\n",
       "      <td>[False, True, False, False, True, False, False...</td>\n",
       "      <td>False</td>\n",
       "      <td>230</td>\n",
       "      <td>[False, True, True, False, False, True, True, ...</td>\n",
       "      <td>0.816578</td>\n",
       "      <td>143.847859</td>\n",
       "      <td>40.386826</td>\n",
       "      <td>0.880797</td>\n",
       "      <td>0.719239</td>\n",
       "      <td>0.201934</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6863</th>\n",
       "      <td>2447</td>\n",
       "      <td>The Reagan Administration has apologized to a ...</td>\n",
       "      <td>In a letter to Mr. Cooper on June 30, Dr. Hase...</td>\n",
       "      <td>410</td>\n",
       "      <td>210</td>\n",
       "      <td>200</td>\n",
       "      <td>4.481025</td>\n",
       "      <td>4.892953</td>\n",
       "      <td>[[0.9297339916229248, 0.5696784257888794, 0.65...</td>\n",
       "      <td>In an interview, Dr. Haseltine said he was ple...</td>\n",
       "      <td>...</td>\n",
       "      <td>[False, True, False, False, False, False, Fals...</td>\n",
       "      <td>False</td>\n",
       "      <td>211</td>\n",
       "      <td>[False, True, True, False, True, True, True, F...</td>\n",
       "      <td>0.832774</td>\n",
       "      <td>146.701059</td>\n",
       "      <td>39.095055</td>\n",
       "      <td>0.880797</td>\n",
       "      <td>0.733505</td>\n",
       "      <td>0.195475</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6864</th>\n",
       "      <td>2451</td>\n",
       "      <td>Tha dòchas math gun ruig a' cho-labhairt air a...</td>\n",
       "      <td>ich seachd ministear fichead, ach dh' fheumadh...</td>\n",
       "      <td>1309</td>\n",
       "      <td>1109</td>\n",
       "      <td>200</td>\n",
       "      <td>12.412498</td>\n",
       "      <td>12.888345</td>\n",
       "      <td>[[0.9876520037651062, 0.9656899571418762, 0.98...</td>\n",
       "      <td>an t-Urramach a thabhairt a thabhairt a' thab...</td>\n",
       "      <td>...</td>\n",
       "      <td>[False, False, False, False, False, False, Fal...</td>\n",
       "      <td>False</td>\n",
       "      <td>204</td>\n",
       "      <td>[True, True, False, False, True, True, False, ...</td>\n",
       "      <td>0.767618</td>\n",
       "      <td>135.223206</td>\n",
       "      <td>43.796629</td>\n",
       "      <td>0.880797</td>\n",
       "      <td>0.676116</td>\n",
       "      <td>0.218983</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6865</th>\n",
       "      <td>2463</td>\n",
       "      <td>KYLIE Hughes has created a simple hack that ca...</td>\n",
       "      <td>a big reno for our home and I was spending li...</td>\n",
       "      <td>322</td>\n",
       "      <td>122</td>\n",
       "      <td>200</td>\n",
       "      <td>3.869065</td>\n",
       "      <td>4.088520</td>\n",
       "      <td>[[0.8722681403160095, 0.9521779417991638, 0.83...</td>\n",
       "      <td>renovating our house,” she told news.com.au.\\...</td>\n",
       "      <td>...</td>\n",
       "      <td>[False, False, False, True, False, False, True...</td>\n",
       "      <td>False</td>\n",
       "      <td>203</td>\n",
       "      <td>[True, False, True, False, True, True, True, F...</td>\n",
       "      <td>0.821787</td>\n",
       "      <td>144.765548</td>\n",
       "      <td>39.980229</td>\n",
       "      <td>0.880797</td>\n",
       "      <td>0.723828</td>\n",
       "      <td>0.199901</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6866</th>\n",
       "      <td>2466</td>\n",
       "      <td>Nortel Networks acquires CRM vendor Clarify In...</td>\n",
       "      <td>they really want a single customer franchised...</td>\n",
       "      <td>615</td>\n",
       "      <td>415</td>\n",
       "      <td>200</td>\n",
       "      <td>6.132745</td>\n",
       "      <td>6.507980</td>\n",
       "      <td>[[0.8657193183898926, 0.9246706962585449, 0.77...</td>\n",
       "      <td>they’re all treated the same,” she said. “The...</td>\n",
       "      <td>...</td>\n",
       "      <td>[False, True, False, False, False, False, Fals...</td>\n",
       "      <td>False</td>\n",
       "      <td>212</td>\n",
       "      <td>[True, True, False, True, False, False, True, ...</td>\n",
       "      <td>0.802656</td>\n",
       "      <td>141.395328</td>\n",
       "      <td>41.432134</td>\n",
       "      <td>0.880797</td>\n",
       "      <td>0.706977</td>\n",
       "      <td>0.207161</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "<p>5 rows × 96 columns</p>\n",
       "</div>"
      ],
      "text/plain": [
       "       idx                                    truncated_input  \\\n",
       "6862  2445  Why is the US national security Establishment ...   \n",
       "6863  2447  The Reagan Administration has apologized to a ...   \n",
       "6864  2451  Tha dòchas math gun ruig a' cho-labhairt air a...   \n",
       "6865  2463  KYLIE Hughes has created a simple hack that ca...   \n",
       "6866  2466  Nortel Networks acquires CRM vendor Clarify In...   \n",
       "\n",
       "                                    baseline_completion  orig_sample_length  \\\n",
       "6862   get what I’ve been saying for well over a yea...                1026   \n",
       "6863  In a letter to Mr. Cooper on June 30, Dr. Hase...                 410   \n",
       "6864  ich seachd ministear fichead, ach dh' fheumadh...                1309   \n",
       "6865   a big reno for our home and I was spending li...                 322   \n",
       "6866   they really want a single customer franchised...                 615   \n",
       "\n",
       "      prompt_length  real_completion_length  no_bl_gen_time  w_bl_gen_time  \\\n",
       "6862            826                     200        9.984011      10.297770   \n",
       "6863            210                     200        4.481025       4.892953   \n",
       "6864           1109                     200       12.412498      12.888345   \n",
       "6865            122                     200        3.869065       4.088520   \n",
       "6866            415                     200        6.132745       6.507980   \n",
       "\n",
       "                                        spike_entropies  \\\n",
       "6862  [[0.9045331478118896, 0.8804982304573059, 0.83...   \n",
       "6863  [[0.9297339916229248, 0.5696784257888794, 0.65...   \n",
       "6864  [[0.9876520037651062, 0.9656899571418762, 0.98...   \n",
       "6865  [[0.8722681403160095, 0.9521779417991638, 0.83...   \n",
       "6866  [[0.8657193183898926, 0.9246706962585449, 0.77...   \n",
       "\n",
       "                                           no_bl_output  ...  \\\n",
       "6862   believe me when I tell them that he is the be...  ...   \n",
       "6863  In an interview, Dr. Haseltine said he was ple...  ...   \n",
       "6864   an t-Urramach a thabhairt a thabhairt a' thab...  ...   \n",
       "6865   renovating our house,” she told news.com.au.\\...  ...   \n",
       "6866   they’re all treated the same,” she said. “The...  ...   \n",
       "\n",
       "                                          w_bl_hit_list  \\\n",
       "6862  [False, True, False, False, True, False, False...   \n",
       "6863  [False, True, False, False, False, False, Fals...   \n",
       "6864  [False, False, False, False, False, False, Fal...   \n",
       "6865  [False, False, False, True, False, False, True...   \n",
       "6866  [False, True, False, False, False, False, Fals...   \n",
       "\n",
       "      w_bl_attacked_num_toks_gend_eq_0  w_bl_attacked_num_tokens_generated  \\\n",
       "6862                             False                                 230   \n",
       "6863                             False                                 211   \n",
       "6864                             False                                 204   \n",
       "6865                             False                                 203   \n",
       "6866                             False                                 212   \n",
       "\n",
       "                                 w_bl_attacked_hit_list  avg_spike_entropy  \\\n",
       "6862  [False, True, True, False, False, True, True, ...           0.816578   \n",
       "6863  [False, True, True, False, True, True, True, F...           0.832774   \n",
       "6864  [True, True, False, False, True, True, False, ...           0.767618   \n",
       "6865  [True, False, True, False, True, True, True, F...           0.821787   \n",
       "6866  [True, True, False, True, False, False, True, ...           0.802656   \n",
       "\n",
       "      w_bl_exp_num_wl_tokens  w_bl_var_num_wl_tokens  exp_wl_coef  \\\n",
       "6862              143.847859               40.386826     0.880797   \n",
       "6863              146.701059               39.095055     0.880797   \n",
       "6864              135.223206               43.796629     0.880797   \n",
       "6865              144.765548               39.980229     0.880797   \n",
       "6866              141.395328               41.432134     0.880797   \n",
       "\n",
       "      w_bl_exp_whitelist_fraction  w_bl_var_whitelist_fraction  \n",
       "6862                     0.719239                     0.201934  \n",
       "6863                     0.733505                     0.195475  \n",
       "6864                     0.676116                     0.218983  \n",
       "6865                     0.723828                     0.199901  \n",
       "6866                     0.706977                     0.207161  \n",
       "\n",
       "[5 rows x 96 columns]"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "print(f\"Orig number of rows: {len(df)}\")\n",
    "df.tail()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "Index(['idx', 'truncated_input', 'baseline_completion', 'orig_sample_length',\n",
       "       'prompt_length', 'real_completion_length', 'no_bl_gen_time',\n",
       "       'w_bl_gen_time', 'spike_entropies', 'no_bl_output', 'w_bl_output',\n",
       "       'no_bl_num_tokens_generated', 'w_bl_num_tokens_generated',\n",
       "       'no_bl_sec_per_tok', 'no_bl_tok_per_sec', 'w_bl_sec_per_tok',\n",
       "       'w_bl_tok_per_sec', 'baseline_loss', 'baseline_ppl', 'no_bl_loss',\n",
       "       'no_bl_ppl', 'w_bl_loss', 'w_bl_ppl', 'w_bl_output_attacked',\n",
       "       'actual_attacked_ratio', 'w_bl_attacked_loss', 'w_bl_attacked_ppl',\n",
       "       'w_bl_whitelist_fraction', 'w_bl_blacklist_fraction',\n",
       "       'w_bl_attacked_whitelist_fraction', 'w_bl_attacked_blacklist_fraction',\n",
       "       'bert_score_attacked', 'model_name', 'attack_model_name',\n",
       "       'attack_model_max_length', 'initial_seed', 'dynamic_seed',\n",
       "       'bl_proportion', 'num_beams', 'num_return_sequences', 'max_length',\n",
       "       'oracle_model_name', 'no_wandb', 'wandb_project', 'run_name',\n",
       "       'output_dir', 'print_step', 'num_replace_per_step', 'replace_ratio',\n",
       "       'data_dir', 'targeted_attack', 'dataset_name', 'dataset_config_name',\n",
       "       'shuffle_dataset', 'shuffle_seed', 'shuffle_buffer_size',\n",
       "       'max_new_tokens', 'min_prompt_tokens', 'limit_indices',\n",
       "       'input_truncation_strategy', 'input_filtering_strategy',\n",
       "       'output_filtering_strategy', 'bl_logit_bias', 'bl_type',\n",
       "       'no_repeat_ngram_size', 'early_stopping', 'wandb_entity',\n",
       "       'load_prev_generations', 'store_bl_ids', 'store_spike_ents',\n",
       "       'use_sampling', 'sampling_temp', 'generate_only', 'SLURM_JOB_ID',\n",
       "       'SLURM_ARRAY_JOB_ID', 'SLURM_ARRAY_TASK_ID',\n",
       "       'gen_table_already_existed', 'baseline_num_toks_gend_eq_0',\n",
       "       'baseline_whitelist_fraction', 'baseline_blacklist_fraction',\n",
       "       'baseline_hit_list', 'no_bl_num_toks_gend_eq_0',\n",
       "       'no_bl_whitelist_fraction', 'no_bl_blacklist_fraction',\n",
       "       'no_bl_hit_list', 'w_bl_num_toks_gend_eq_0', 'w_bl_hit_list',\n",
       "       'w_bl_attacked_num_toks_gend_eq_0',\n",
       "       'w_bl_attacked_num_tokens_generated', 'w_bl_attacked_hit_list',\n",
       "       'avg_spike_entropy', 'w_bl_exp_num_wl_tokens', 'w_bl_var_num_wl_tokens',\n",
       "       'exp_wl_coef', 'w_bl_exp_whitelist_fraction',\n",
       "       'w_bl_var_whitelist_fraction'],\n",
       "      dtype='object')"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df.columns"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## mega filter block"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Num rows that are hard-blacklisted, and measureable, but still have a non-100% WL fraction: 0 out of 0\n",
      "Dropped 0 rows, new len 6867\n",
      "Dropped 0 rows, new len 6867\n",
      "Dropped 0 rows, new len 6867\n",
      "Dropped 0 rows, new len 6867\n",
      "Dropped 35 rows, new len 6832\n"
     ]
    }
   ],
   "source": [
    "# drop retok_problematic_rows\n",
    "retok_problematic_rows = df[(df['w_bl_whitelist_fraction'] != -1.0) & (df['w_bl_whitelist_fraction'] != 1.0) & (df['bl_type'] == 'hard')]\n",
    "print(f\"Num rows that are hard-blacklisted, and measureable, but still have a non-100% WL fraction: {len(retok_problematic_rows)} out of {len(df[df['bl_type'] == 'hard'])}\")\n",
    "\n",
    "# drop special rows marked as -1.0\n",
    "orig_len = len(df)\n",
    "\n",
    "# df['no_bl_whitelist_fraction'].mask(df['no_bl_whitelist_fraction'] == -1.0, pd.NA, inplace=True)\n",
    "# df['w_bl_whitelist_fraction'].mask(df['w_bl_whitelist_fraction'] == -1.0, pd.NA, inplace=True)\n",
    "\n",
    "df = df[df[\"no_bl_whitelist_fraction\"] != -1.0]\n",
    "df = df[df[\"w_bl_whitelist_fraction\"] != -1.0]\n",
    "\n",
    "print(f\"Dropped {orig_len-len(df)} rows, new len {len(df)}\")\n",
    "\n",
    "# drop too few tokesn rows\n",
    "\n",
    "orig_len = len(df)\n",
    "# df = df[df[\"no_bl_ppl\"].isna()]\n",
    "# df = df[df[\"w_bl_ppl\"].isna()]\n",
    "df = df[~(df[\"no_bl_ppl\"].isna() | df[\"w_bl_ppl\"].isna())]\n",
    "print(f\"Dropped {orig_len-len(df)} rows, new len {len(df)}\")\n",
    "\n",
    "# drop huge biases\n",
    "orig_len = len(df)\n",
    "\n",
    "df = df[df[\"bl_logit_bias\"] <= 100.0]\n",
    "\n",
    "print(f\"Dropped {orig_len-len(df)} rows, new len {len(df)}\")\n",
    "\n",
    "orig_len = len(df)\n",
    "\n",
    "# df = df[df[\"bl_hparams\"].apply(lambda tup: (tup[0] == False and tup[2] != 1) or (tup[0] == True and tup[2] == 1) or (tup[0] == False))]\n",
    "df = df[((df[\"use_sampling\"]==True) & (df[\"num_beams\"] == 1)) | (df[\"use_sampling\"]==False)]\n",
    "\n",
    "print(f\"Dropped {orig_len-len(df)} rows, new len {len(df)}\")\n",
    "\n",
    "# correct sampling temp\n",
    "df.loc[df[\"use_sampling\"]==False,\"sampling_temp\"] = df.loc[df[\"use_sampling\"]==False,\"sampling_temp\"].fillna(0.0)\n",
    "df.loc[df[\"use_sampling\"]==True,\"sampling_temp\"] = df.loc[df[\"use_sampling\"]==True,\"sampling_temp\"].fillna(1.0)\n",
    "\n",
    "# set to inf for hard blacklist\n",
    "df.loc[df[\"bl_type\"]==\"hard\",\"bl_logit_bias\"] = np.inf\n",
    "# df.loc[df[\"bl_type\"]==\"hard\",\"bl_logit_bias\"] = 10000 # crosscheck with whats hardcoded in the bl processor\n",
    "\n",
    "# rename some stuff\n",
    "df[\"delta\"] = df[\"bl_logit_bias\"].values\n",
    "df[\"gamma\"] = 1 - df[\"bl_proportion\"].values\n",
    "df[\"gamma\"] = df[\"gamma\"].round(3)\n",
    "\n",
    "df[\"no_bl_act_num_wl_tokens\"] = np.round(df[\"no_bl_whitelist_fraction\"].values*df[\"no_bl_num_tokens_generated\"],1) # round to 1 for sanity\n",
    "df[\"w_bl_act_num_wl_tokens\"] = np.round(df[\"w_bl_whitelist_fraction\"].values*df[\"w_bl_num_tokens_generated\"],1) # round to 1 for sanity\n",
    "\n",
    "df[\"w_bl_std_num_wl_tokens\"] = np.sqrt(df[\"w_bl_var_num_wl_tokens\"].values)\n",
    "\n",
    "if \"real_completion_length\":\n",
    "    df[\"baseline_num_tokens_generated\"] = df[\"real_completion_length\"].values\n",
    "\n",
    "if \"actual_attacked_ratio\" in df.columns:\n",
    "    df[\"actual_attacked_fraction\"] = df[\"actual_attacked_ratio\"].values*df[\"replace_ratio\"].values\n",
    "\n",
    "if \"meta\" in df.columns:\n",
    "    df[\"pile_set_name\"] = df[\"meta\"].apply(lambda dict: dict[\"pile_set_name\"])\n",
    "\n",
    "df[\"baseline_hit_list_length\"] = df[\"baseline_hit_list\"].apply(len)\n",
    "df[\"no_bl_hit_list_length\"] = df[\"no_bl_hit_list\"].apply(len)\n",
    "df[\"w_bl_hit_list_length\"] = df[\"w_bl_hit_list\"].apply(len)\n",
    "df[\"w_bl_attacked_hit_list_length\"] = df[\"w_bl_attacked_hit_list\"].apply(len)\n",
    "\n",
    "\n",
    "# for pile outlier filtering\n",
    "df[\"w_bl_space_count\"] = df[\"w_bl_output\"].apply(lambda string: string.count(\" \"))\n",
    "df[\"no_bl_space_count\"] = df[\"no_bl_output\"].apply(lambda string: string.count(\" \"))\n",
    "df[\"baseline_space_count\"] = df[\"baseline_completion\"].apply(lambda string: string.count(\" \"))\n",
    "\n",
    "df[\"w_bl_space_frac\"] = df[\"w_bl_space_count\"].values / df[\"w_bl_hit_list_length\"]\n",
    "df[\"no_bl_space_frac\"] = df[\"no_bl_space_count\"].values / df[\"no_bl_hit_list_length\"]\n",
    "df[\"baseline_space_frac\"] = df[\"baseline_space_count\"].values / df[\"baseline_hit_list_length\"]\n",
    "\n",
    "\n",
    "# Final length filtering \n",
    "\n",
    "orig_len = len(df)\n",
    "\n",
    "# # main filters\n",
    "# # df = df[(df[\"real_completion_length\"] == 200) & (df[\"w_bl_num_tokens_generated\"] == 200)]\n",
    "# df = df[(df[\"gamma\"] == 0.1) | (df[\"gamma\"] == 0.25) | (df[\"gamma\"] == 0.5)]\n",
    "# df = df[(df[\"delta\"] == 1.0) | (df[\"delta\"] == 2.0) | (df[\"delta\"] == 10.0)]\n",
    "# df = df[(df[\"use_sampling\"] == True)]\n",
    "# df = df[(df[\"bl_type\"] == \"soft\")]\n",
    "\n",
    "# df = df[(df[\"real_completion_length\"] == 200) & (df[\"no_bl_num_tokens_generated\"] == 200) & (df[\"w_bl_num_tokens_generated\"] == 200)] # now also applies to the truncated version\n",
    "# df = df[(df[\"no_bl_num_tokens_generated\"] >= 500) & (df[\"w_bl_num_tokens_generated\"] >= 500)] # all gas noop\n",
    "\n",
    "# # # attack specific\n",
    "# df = df[(df[\"real_completion_length\"] == 200) & (df[\"no_bl_num_tokens_generated\"] == 200) & (df[\"w_bl_num_tokens_generated\"] == 200)]\n",
    "# df = df[(df[\"replace_ratio\"] <= 0.7)]\n",
    "\n",
    "# # NOTE pile only\n",
    "# df = df[df[\"w_bl_space_frac\"] <= 0.9]\n",
    "# df = df[df[\"no_bl_space_frac\"] <= 0.9]\n",
    "# df = df[df[\"pile_set_name\"] != \"Github\"]\n",
    "\n",
    "upper_T = 205\n",
    "lower_T = 195\n",
    "df = df[(df[\"baseline_hit_list_length\"] >= lower_T) & (df[\"no_bl_hit_list_length\"] >= lower_T) & (df[\"w_bl_hit_list_length\"] >= lower_T)] # now also applies to the truncated version\n",
    "df = df[(df[\"baseline_hit_list_length\"] <= upper_T) & (df[\"no_bl_hit_list_length\"] <= upper_T) & (df[\"w_bl_hit_list_length\"] <= upper_T)] # now also applies to the truncated version\n",
    "\n",
    "# oops forgot to check the attacked output lengths in paper\n",
    "# df = df[(df[\"baseline_hit_list_length\"] >= lower_T) & (df[\"no_bl_hit_list_length\"] >= lower_T) & (df[\"w_bl_hit_list_length\"] >= lower_T) & (df[\"w_bl_attacked_hit_list_length\"] >= lower_T-10)] # now also applies to the truncated version\n",
    "# df = df[(df[\"baseline_hit_list_length\"] <= upper_T) & (df[\"no_bl_hit_list_length\"] <= upper_T) & (df[\"w_bl_hit_list_length\"] <= upper_T) & (df[\"w_bl_attacked_hit_list_length\"] <= upper_T+10)] # now also applies to the truncated version\n",
    "\n",
    "print(f\"Dropped {orig_len-len(df)} rows, new len {len(df)}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "from watermark_processor import WatermarkDetector\n",
    "from transformers import AutoTokenizer\n",
    "\n",
    "tokenizer = AutoTokenizer.from_pretrained(\"facebook/opt-6.7b\")\n",
    "\n",
    "# gamma = 0.25\n",
    "gamma = 0.5\n",
    "\n",
    "watermark_detector = WatermarkDetector(vocab=list(tokenizer.get_vocab().values()),\n",
    "                                        gamma=gamma,\n",
    "                                        seeding_scheme=\"simple_1\",\n",
    "                                        device=torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\"),\n",
    "                                        tokenizer=tokenizer,\n",
    "                                        z_threshold=4.0,\n",
    "                                        normalizers=\"\",\n",
    "                                        ignore_repeated_ngrams=False,\n",
    "                                        # select_green_tokens=True)\n",
    "                                        select_green_tokens=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "if raw_generations_df:\n",
    "    def get_detect_stats(text):\n",
    "        res = watermark_detector.detect(text,return_green_token_mask=True)\n",
    "        return res[\"green_fraction\"], res[\"z_score\"], res[\"p_value\"], res[\"green_token_mask\"]\n",
    "\n",
    "    df[\"delta\"] = df[\"bl_logit_bias\"].values\n",
    "    df[\"gamma\"] = 1 - df[\"bl_proportion\"].values\n",
    "    df[\"gamma\"] = df[\"gamma\"].round(3)\n",
    "    df.drop(\"replace_ratio\", axis=1, inplace=True)\n",
    "\n",
    "    # note were missing the baseline column this time\n",
    "    df[\"baseline_whitelist_fraction\"],df[\"baseline_z_score\"], df[\"baseline_p_val\"], df[\"baseline_hit_list\"] = zip(*df[[\"no_wm_output\"]].apply(lambda tup: get_detect_stats(*tup), axis=1))\n",
    "    df[\"w_bl_whitelist_fraction\"],df[\"w_bl_z_score\"], df[\"w_bl_p_val\"], df[\"w_bl_hit_list\"] = zip(*df[[\"w_wm_output\"]].apply(lambda tup: get_detect_stats(*tup), axis=1))\n",
    "    df[\"w_bl_attacked_whitelist_fraction\"],df[\"w_bl_attacked_z_score\"], df[\"w_bl_attacked_p_val\"], df[\"w_bl_attacked_hit_list\"] = zip(*df[[\"w_wm_output_attacked\"]].apply(lambda tup: get_detect_stats(*tup), axis=1))\n",
    "                    \n",
    "    # rename the output columns\n",
    "    df = df.rename(columns={\"no_wm_output\": \"baseline_completion\",\n",
    "                    \"w_wm_output\": \"w_bl_output\",\n",
    "                    \"w_wm_output_attacked\": \"w_bl_output_attacked\",})\n",
    "    \n",
    "    df[\"baseline_hit_list\"] = df[\"baseline_hit_list\"].apply(np.array)\n",
    "    df[\"w_bl_hit_list\"] = df[\"w_bl_hit_list\"].apply(np.array)\n",
    "    df[\"w_bl_attacked_hit_list\"] = df[\"w_bl_attacked_hit_list\"].apply(np.array)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "df[\"baseline_hit_list_len\"] = df[\"baseline_hit_list\"].apply(len)\n",
    "df[\"w_bl_hit_list_len\"] = df[\"w_bl_hit_list\"].apply(len)\n",
    "df[\"w_bl_attacked_hit_list_len\"] = df[\"w_bl_attacked_hit_list\"].apply(len)\n",
    "\n",
    "# df[df[\"baseline_hit_list_len\"] <= 3].head()\n",
    "# df[df[\"w_bl_hit_list_len\"] <= 3].head()\n",
    "# df[df[\"w_bl_attacked_hit_list_len\"] < 195]"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Add z-scores"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "from math import sqrt\n",
    "import scipy.stats\n",
    "def compute_z_score(observed_wl_frac, T, gamma):\n",
    "    numer = observed_wl_frac - gamma\n",
    "    denom = sqrt(gamma*(1-gamma)/T)\n",
    "    z = numer/denom\n",
    "    return z\n",
    "\n",
    "def compute_wl_for_z(z, T, gamma):\n",
    "    denom = sqrt(gamma*(1-gamma)/T)\n",
    "    numer = ((z*denom)+gamma)*T\n",
    "    return numer\n",
    "\n",
    "def compute_p_value(z):\n",
    "    p_value = scipy.stats.norm.sf(z)\n",
    "    return p_value\n",
    "\n",
    "\n",
    "if not raw_generations_df:\n",
    "    df[\"baseline_z_score\"] = df[[\"baseline_whitelist_fraction\", \"baseline_num_tokens_generated\", \"gamma\"]].apply(lambda tup: compute_z_score(*tup), axis=1)\n",
    "    df[\"baseline_p_val\"] = df[\"baseline_z_score\"].apply(lambda z: compute_p_value(z))\n",
    "\n",
    "    df[\"no_bl_z_score\"] = df[[\"no_bl_whitelist_fraction\", \"no_bl_num_tokens_generated\", \"gamma\"]].apply(lambda tup: compute_z_score(*tup), axis=1)\n",
    "    df[\"no_bl_p_val\"] = df[\"no_bl_z_score\"].apply(lambda z: compute_p_value(z))\n",
    "                                                \n",
    "    df[\"w_bl_z_score\"] = df[[\"w_bl_whitelist_fraction\", \"w_bl_num_tokens_generated\", \"gamma\"]].apply(lambda tup: compute_z_score(*tup), axis=1)\n",
    "    df[\"w_bl_p_val\"] = df[\"w_bl_z_score\"].apply(lambda z: compute_p_value(z))\n",
    "\n",
    "    if \"w_bl_attacked_whitelist_fraction\" in df.columns:\n",
    "        df[\"w_bl_attacked_z_score\"] = df[[\"w_bl_attacked_whitelist_fraction\", \"w_bl_attacked_num_tokens_generated\", \"gamma\"]].apply(lambda tup: compute_z_score(*tup), axis=1)\n",
    "        df[\"w_bl_attacked_p_val\"] = df[\"w_bl_attacked_z_score\"].apply(lambda z: compute_p_value(z))\n",
    "\n",
    "    # if attacked in df\n",
    "    if \"w_bl_attacked_whitelist_fraction\" in df.columns:\n",
    "        df[\"w_bl_attacked_act_num_wl_tokens\"] = np.round(df[\"w_bl_attacked_whitelist_fraction\"].values*df[\"w_bl_attacked_num_tokens_generated\"],1) # round to 1 for sanity"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## runs test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "%reload_ext autoreload\n",
    "%autoreload 2\n",
    "\n",
    "import numpy as np\n",
    "import numpy.ma as ma\n",
    "from scipy.stats import geom, chisquare\n",
    "\n",
    "def rle_T_and_F_runs(arr):\n",
    "    \"\"\" \n",
    "    Return run lengths and the value repeated in the run, of a boolean array.\n",
    "    This handles arrays with different values and counts up runs of each value.\n",
    "    https://stackoverflow.com/a/69693227 \n",
    "    \"\"\"\n",
    "    n = len(arr)\n",
    "    if n == 0:\n",
    "        values = np.empty(0, dtype=arr.dtype)\n",
    "        lengths = np.empty(0, dtype=np.int_)\n",
    "    else:\n",
    "        positions = np.concatenate(\n",
    "            [[-1], np.nonzero(arr[1:] != arr[:-1])[0], [n - 1]])\n",
    "        lengths = positions[1:] - positions[:-1]\n",
    "        values = arr[positions[1:]]\n",
    "    \n",
    "    return values, lengths\n",
    "\n",
    "def rle_F_succ_T_runs(arr):\n",
    "    \"\"\" \n",
    "    With the k=1,2,3 convention, where success is False, and failure is True, \n",
    "    we want to count the number of flips required before a success, i.e False.\n",
    "    This 'resets' every time we see a False. \n",
    "    Note, this truncates the tail of the array, so if the trailing elements are True,\n",
    "    then they are not counted as a run since there is no terminating False.\n",
    "\n",
    "    Note, this means if sequence is all True, then we return an empty array,\n",
    "    and if the sequence is all False, then we return an ones array of length n.\n",
    "    \"\"\"\n",
    "    n = len(arr)\n",
    "    if n == 0:\n",
    "        lengths = np.empty(0, dtype=np.int_)\n",
    "    else:\n",
    "        false_positions = np.concatenate(\n",
    "            [[-1], np.nonzero(arr == False)[0]])\n",
    "        lengths = false_positions[1:] - false_positions[:-1]\n",
    "    \n",
    "    return lengths\n",
    "\n",
    "def rle_T_succ_F_runs(arr):\n",
    "    \"\"\" \n",
    "    Opposite above\n",
    "    \"\"\"\n",
    "    n = len(arr)\n",
    "    if n == 0:\n",
    "        lengths = np.empty(0, dtype=np.int_)\n",
    "    else:\n",
    "        true_positions = np.concatenate(\n",
    "            [[-1], np.nonzero(arr == True)[0]])\n",
    "        lengths = true_positions[1:] - true_positions[:-1]\n",
    "    \n",
    "    return lengths\n",
    "\n",
    "def chi_squared_T_and_F_test(bool_arr, succ_prob, bin_spec=None, verbose=False, invert_bools=False, return_bin_counts=False, mask_zeros=False, lambda_=\"pearson\"):\n",
    "\n",
    "    if verbose: print(f\"likelihood of success=F (1-gamma), or T run length geom dist 'p' = {succ_prob}\")\n",
    "\n",
    "    if invert_bools:\n",
    "        bool_arr = ~bool_arr\n",
    "\n",
    "    values, lengths = rle_T_and_F_runs(bool_arr)\n",
    "    if verbose: print(f\"Raw run lengths and their values and types:\\n{lengths}\\n{values}\")\n",
    "\n",
    "    remove_false = False\n",
    "    remove_true = False\n",
    "    if len(lengths) == 1:\n",
    "        # lengths = np.array([len(bool_arr)+1]) # this is a HACK\n",
    "        if values[0] == True:\n",
    "            remove_false = True\n",
    "            uniq_T_lens, T_run_counts = lengths, np.array([1])\n",
    "            uniq_F_lens, F_run_counts = np.array([0]), np.array([0])\n",
    "        elif values[0] == False:\n",
    "            remove_true = True\n",
    "            uniq_T_lens, T_run_counts = np.array([0]), np.array([0])\n",
    "            uniq_F_lens, F_run_counts = lengths, np.array([1])\n",
    "        else:\n",
    "            raise ValueError(\"Unexpected value in bool array\")\n",
    "    else:\n",
    "        uniq_T_lens, T_run_counts = np.unique(lengths[values == True], return_counts=True)\n",
    "        uniq_F_lens, F_run_counts = np.unique(lengths[values == False], return_counts=True)\n",
    "\n",
    "    if verbose: print(\"Unique T run lengths: \", uniq_T_lens)\n",
    "    if verbose: print(f\"Total T runs: {sum(T_run_counts)}\")\n",
    "    if verbose: print(\"Unique F run lengths: \", uniq_F_lens)\n",
    "    if verbose: print(f\"Total F runs: {sum(F_run_counts)}\")\n",
    "\n",
    "    if bin_spec == \"max\":\n",
    "        largest_T_bin = max(uniq_T_lens)\n",
    "        largest_F_bin = max(uniq_F_lens)\n",
    "    elif bin_spec == \"max_plus_1\":\n",
    "        largest_T_bin = max(uniq_T_lens) + 1\n",
    "        largest_F_bin = max(uniq_F_lens) + 1\n",
    "    elif isinstance(bin_spec, int):\n",
    "        largest_T_bin = max(bin_spec, max(uniq_T_lens))\n",
    "        largest_F_bin = max(bin_spec, max(uniq_F_lens))\n",
    "    else:\n",
    "        raise ValueError(\"bin_spec must be 'max' or an integer\")\n",
    "\n",
    "    if not remove_true:\n",
    "        T_bins = np.arange(1, largest_T_bin+1)\n",
    "        if verbose: print(\"T Length bins: \",T_bins)\n",
    "        obs_T_counts = np.zeros_like(T_bins, dtype=float)\n",
    "        obs_T_counts[uniq_T_lens-1] = np.array(T_run_counts,dtype=float)\n",
    "        total_T_runs = sum(obs_T_counts)\n",
    "    else:\n",
    "        T_bins = uniq_T_lens\n",
    "        if verbose: print(\"Ignoring lack of T runs in combined arrays\")\n",
    "        obs_T_counts = np.array([])\n",
    "        total_T_runs = 0\n",
    "    if not remove_false:\n",
    "        F_bins = np.arange(1, largest_F_bin+1)\n",
    "        if verbose: print(\"F Length bins: \",F_bins)\n",
    "        obs_F_counts = np.zeros_like(F_bins, dtype=float)\n",
    "        obs_F_counts[uniq_F_lens-1] = np.array(F_run_counts,dtype=float)\n",
    "        total_F_runs = sum(obs_F_counts)\n",
    "    else:\n",
    "        F_bins = uniq_F_lens\n",
    "        if verbose: print(\"Ignoring lack of F runs in combined arrays\")\n",
    "        obs_F_counts = np.array([])\n",
    "        total_F_runs = 0\n",
    "\n",
    "    if bin_spec in [\"max\", \"max_plus_1\"]:\n",
    "        T_densities = geom.pmf(T_bins, succ_prob)\n",
    "        T_densities[-1] += geom.sf(T_bins[-1], succ_prob)\n",
    "        exp_T_counts = T_densities * total_T_runs\n",
    "\n",
    "        F_densities = geom.pmf(F_bins, 1-succ_prob)\n",
    "        F_densities[-1] += geom.sf(F_bins[-1], 1-succ_prob)\n",
    "        exp_F_counts = F_densities * total_F_runs\n",
    "    else:\n",
    "        T_densities = geom.pmf(T_bins, succ_prob)\n",
    "        exp_T_counts = T_densities * total_T_runs\n",
    "\n",
    "        F_densities = geom.pmf(F_bins, 1-succ_prob)\n",
    "        exp_F_counts = F_densities * total_F_runs\n",
    "\n",
    "    if remove_true:\n",
    "        exp_T_counts = np.array([])\n",
    "    if remove_false:\n",
    "        exp_F_counts = np.array([])\n",
    "\n",
    "    if verbose: print(\"Obs T counts: \", obs_T_counts)\n",
    "    if verbose: print(\"Exp T counts: \", exp_T_counts)\n",
    "    if verbose: print(f\"densities: sum={sum(T_densities)}, {T_densities}\")\n",
    "    if verbose: print(\"Obs F counts: \", obs_F_counts)\n",
    "    if verbose: print(\"Exp F counts: \", exp_F_counts)\n",
    "    if verbose: print(f\"densities: sum={sum(F_densities)}, {F_densities}\")\n",
    "\n",
    "    # concat the T and F obs and exp arrays\n",
    "    obs_counts = np.concatenate([obs_T_counts, obs_F_counts])\n",
    "    exp_counts = np.concatenate([exp_T_counts, exp_F_counts])\n",
    "\n",
    "    if mask_zeros:\n",
    "        obs_counts = ma.masked_array(obs_counts, mask=(obs_counts==0))\n",
    "\n",
    "    if verbose: print(\"Joined Obs counts: \", obs_counts)\n",
    "    if verbose: print(\"Joined Exp counts: \", exp_counts)\n",
    "\n",
    "    if lambda_ == \"g_test\":\n",
    "        statistic, p_val = scipy.stats.power_divergence(f_obs=obs_counts, f_exp=exp_counts, ddof=0, axis=0, lambda_=0)\n",
    "    elif lambda_ == \"cressie_read\":\n",
    "        statistic, p_val = scipy.stats.power_divergence(f_obs=obs_counts, f_exp=exp_counts, ddof=0, axis=0, lambda_=2/3)\n",
    "    elif lambda_ == \"pearson\":\n",
    "        statistic, p_val = chisquare(obs_counts, exp_counts)\n",
    "    else:\n",
    "        raise ValueError(f\"unrecognized lambda_={lambda_}\")\n",
    "\n",
    "\n",
    "    if return_bin_counts:\n",
    "        return statistic, p_val, total_T_runs, T_bins, obs_T_counts, exp_T_counts, total_F_runs, F_bins, obs_F_counts, exp_F_counts\n",
    "    return statistic, p_val, total_T_runs + total_F_runs\n",
    "\n",
    "\n",
    "def chi_squared_runs_test(bool_arr, succ_prob, variant=\"F_succ_T_runs\", bin_spec=200, verbose=False, invert_bools=False, return_bin_counts=False, mask_zeros=False, diy=False, lambda_=\"pearson\"):\n",
    "    \"\"\" \n",
    "    Returns the chi squared statistic and p-value for the given data.\n",
    "    The data is an array of run lengths, and a probability of success p.\n",
    "    The variant is the convention for the run lengths, i.e. if success is False or True.\n",
    "    The convention is that we are counting the number of flips required before a success.\n",
    "    bin_spec is the number of bins to use for the chi squared test, if == \"max\" then we use the max run length.\n",
    "    \"\"\"\n",
    "    if verbose: print(f\"Boolean array: {bool_arr}\")\n",
    "\n",
    "    if variant == \"F_succ_T_runs\":\n",
    "        run_func = rle_F_succ_T_runs\n",
    "    elif variant == \"T_succ_F_runs\":\n",
    "        run_func = rle_T_succ_F_runs\n",
    "    elif variant == \"T_and_F_runs\":\n",
    "        return chi_squared_T_and_F_test(bool_arr, succ_prob, bin_spec=bin_spec, verbose=verbose, invert_bools=invert_bools, return_bin_counts=return_bin_counts, mask_zeros=mask_zeros, lambda_=lambda_)\n",
    "    else:\n",
    "        raise ValueError(f\"unrecognized variant name={variant}\")\n",
    "    \n",
    "    if invert_bools:\n",
    "        bool_arr = ~bool_arr\n",
    "\n",
    "    lengths = run_func(bool_arr)\n",
    "    if len(lengths) == 0:\n",
    "        lengths = np.array([len(bool_arr)+1]) # this is a HACK\n",
    "    uniq_lens, run_counts = np.unique(lengths, return_counts=True)\n",
    "\n",
    "    if verbose: print(\"Unique run lengths: \", lengths)\n",
    "\n",
    "    if verbose: print(f\"Total runs: {sum(run_counts)}\")\n",
    "    \n",
    "    if bin_spec == \"max\":\n",
    "        largest_bin = max(uniq_lens)\n",
    "    elif bin_spec == \"max_plus_1\":\n",
    "        largest_bin = max(uniq_lens) + 1\n",
    "    elif isinstance(bin_spec, int):\n",
    "        largest_bin = max(bin_spec, max(uniq_lens))\n",
    "    else:\n",
    "        raise ValueError(\"bin_spec must be 'max' or an integer\")\n",
    "\n",
    "    bins = np.arange(1, largest_bin+1)\n",
    "    if verbose: print(\"Length bins: \",bins)\n",
    "    \n",
    "    obs_counts = np.zeros_like(bins, dtype=float)\n",
    "    obs_counts[uniq_lens-1] = np.array(run_counts,dtype=float)\n",
    "    total_runs = sum(obs_counts)\n",
    "\n",
    "    if bin_spec in [\"max\", \"max_plus_1\"]:\n",
    "        densities = geom.pmf(bins, succ_prob)\n",
    "        densities[-1] += geom.sf(bins[-1], succ_prob)\n",
    "        exp_counts = densities * total_runs\n",
    "    else:\n",
    "        densities = geom.pmf(bins, succ_prob)\n",
    "        exp_counts = densities * total_runs\n",
    "\n",
    "    if mask_zeros:\n",
    "        # exp_counts = ma.masked_array(exp_counts, mask=(obs_counts==0))\n",
    "        obs_counts = ma.masked_array(obs_counts, mask=(obs_counts==0))\n",
    "\n",
    "    if verbose: print(\"Obs counts: \", obs_counts)\n",
    "    if verbose: print(\"Exp counts: \", exp_counts)\n",
    "    if verbose: print(f\"densities: sum={sum(densities)}, {densities}\")\n",
    "\n",
    "    # from scipy.stats import power_divergence\n",
    "    # statistic, p_val = chisquare(obs_counts, exp_counts)\n",
    "    if diy:\n",
    "        # print(\"Local chi squared test\")\n",
    "        statistic, p_val = power_divergence(obs_counts, f_exp=exp_counts, ddof=0, axis=0,\n",
    "                            lambda_=\"pearson\")\n",
    "    else:\n",
    "        # print(\"Scipy chi squared test\")\n",
    "        if lambda_ == \"g_test\":\n",
    "            statistic, p_val = scipy.stats.power_divergence(f_obs=obs_counts, f_exp=exp_counts, ddof=0, axis=0, lambda_=0)\n",
    "        elif lambda_ == \"cressie_read\":\n",
    "            statistic, p_val = scipy.stats.power_divergence(f_obs=obs_counts, f_exp=exp_counts, ddof=0, axis=0, lambda_=2/3)\n",
    "        elif lambda_ == \"pearson\":\n",
    "            statistic, p_val = chisquare(obs_counts, exp_counts)\n",
    "        else:\n",
    "            raise ValueError(f\"unrecognized lambda_={lambda_}\")\n",
    "    \n",
    "    if return_bin_counts:\n",
    "        return statistic, p_val, total_runs, bins, obs_counts, exp_counts\n",
    "    return statistic, p_val, total_runs\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "# hit_list = [c for c in \"GRRGRGGR\"] # <username>'s example\n",
    "# hit_list = [c for c in \"RGGRGRRRGRGGG\"] # example with trailing True/green\n",
    "# hit_list = [c for c in \"RRGGRRRGGRGGG\"] # example with zero 2's\n",
    "# hit_list = [c for c in \"GGGGGGGGG\"] # example with all True/green\n",
    "# hit_list = [c for c in \"RRRRRRRRR\"] # example with all True/green\n",
    "\n",
    "# hit_list = [c for c in \"RRGGRRGGRGGG\"] # example with 1,and 2 for R and 2, 3 for G\n",
    "# hit_list = [c for c in \"RRRRRRR\"] # example with 1,and 2 for R and 2, 3 for G\n",
    "\n",
    "# hit_list = np.array([h==\"G\" for h in hit_list])\n",
    "# print(hit_list)\n",
    "\n",
    "# chi_squared_runs_test(hit_list, 1-gamma, variant=\"F_succ_T_runs\", bin_spec=\"max\", verbose=True)\n",
    "# chi_squared_runs_test(hit_list, 1-gamma, variant=\"F_succ_T_runs\", bin_spec=20, verbose=True)\n",
    "# chi_squared_runs_test(hit_list, 1-gamma, variant=\"F_succ_T_runs\", bin_spec=200, verbose=True)\n",
    "\n",
    "\n",
    "# chi_squared_runs_test(hit_list, 1-gamma, variant=\"F_succ_T_runs\", bin_spec=\"max\", verbose=True)\n",
    "# chi_squared_runs_test(hit_list, 1-gamma, variant=\"F_succ_T_runs\", bin_spec=\"max_plus_1\", verbose=True)\n",
    "# chi_squared_runs_test(hit_list, 1-gamma, variant=\"F_succ_T_runs\", bin_spec=\"max_plus_1\", verbose=True, mask_zeros=True, diy=False)\n",
    "# chi_squared_runs_test(hit_list, 1-gamma, variant=\"F_succ_T_runs\", bin_spec=\"max_plus_1\", verbose=True, diy=True)\n",
    "\n",
    "\n",
    "# gamma = 0.5\n",
    "# gamma = 0.25\n",
    "# chi_squared_runs_test(hit_list, 1-gamma, variant=\"T_and_F_runs\", bin_spec=\"max_plus_1\", verbose=True, mask_zeros=False)\n",
    "# chi_squared_runs_test(hit_list, 1-gamma, variant=\"T_and_F_runs\", bin_spec=\"max_plus_1\", verbose=True, mask_zeros=True)\n",
    "\n",
    "# lambda test types\n",
    "# chi_squared_runs_test(hit_list, 1-gamma, variant=\"F_succ_T_runs\", bin_spec=\"max_plus_1\", verbose=True, mask_zeros=False, lambda=\"pearson\")\n",
    "# chi_squared_runs_test(hit_list, 1-gamma, variant=\"F_succ_T_runs\", bin_spec=\"max_plus_1\", verbose=True, mask_zeros=False, lambda_=\"g_test\")\n",
    "# chi_squared_runs_test(hit_list, 1-gamma, variant=\"F_succ_T_runs\", bin_spec=\"max_plus_1\", verbose=True, mask_zeros=False, lambda_=\"cressie_read\")"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Set test type parameters"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 187,
   "metadata": {},
   "outputs": [],
   "source": [
    "# variant = \"F_succ_T_runs\"\n",
    "variant = \"T_and_F_runs\"\n",
    "\n",
    "invert_bools = False if raw_generations_df else True\n",
    "\n",
    "# mask_zeros = False\n",
    "mask_zeros = True\n",
    "\n",
    "# bin_spec = 200\n",
    "# bin_spec = \"max\"\n",
    "bin_spec = \"max_plus_1\"\n",
    "\n",
    "# lambda_ = \"pearson\"\n",
    "# lambda_ = \"g_test\"\n",
    "lambda_=\"cressie_read\""
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 188,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "df[\"baseline_chisqrd_stat\"], df[\"baseline_chisqrd_p_val\"], df[\"baseline_total_runs\"] = zip(*df[[\"baseline_hit_list\",\"gamma\"]].apply(lambda tup: chi_squared_runs_test(tup[0], 1-tup[1], variant=variant, invert_bools=invert_bools, mask_zeros=mask_zeros, bin_spec=bin_spec, lambda_=lambda_), axis=1))\n",
    "if \"no_bl_hit_list\" in df.columns:\n",
    "    df[\"no_bl_chisqrd_stat\"], df[\"no_bl_chisqrd_p_val\"], df[\"no_bl_total_runs\"] = zip(*df[[\"no_bl_hit_list\",\"gamma\"]].apply(lambda tup: chi_squared_runs_test(tup[0], 1-tup[1], variant=variant, invert_bools=invert_bools, mask_zeros=mask_zeros, bin_spec=bin_spec, lambda_=lambda_), axis=1))\n",
    "df[\"w_bl_chisqrd_stat\"], df[\"w_bl_chisqrd_p_val\"], df[\"w_bl_total_runs\"] = zip(*df[[\"w_bl_hit_list\",\"gamma\"]].apply(lambda tup: chi_squared_runs_test(tup[0], 1-tup[1], variant=variant, invert_bools=invert_bools, mask_zeros=mask_zeros, bin_spec=bin_spec, lambda_=lambda_), axis=1))\n",
    "if \"w_bl_attacked_hit_list\" in df.columns:\n",
    "    df[\"w_bl_attacked_chisqrd_stat\"], df[\"w_bl_attacked_chisqrd_p_val\"], df[\"w_bl_attacked_bl_total_runs\"] = zip(*df[[\"w_bl_attacked_hit_list\",\"gamma\"]].apply(lambda tup: chi_squared_runs_test(tup[0], 1-tup[1], variant=variant, invert_bools=invert_bools, mask_zeros=mask_zeros, bin_spec=bin_spec, lambda_=lambda_), axis=1))"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Groupby (decide which hyperparameters to groups the rows by)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 189,
   "metadata": {},
   "outputs": [],
   "source": [
    "if \"w_bl_attacked_whitelist_fraction\" in df.columns and 'replace_ratio' in df.columns: \n",
    "    groupby_fields = ['use_sampling','num_beams','delta','gamma', 'replace_ratio'] # attack grouping\n",
    "else:\n",
    "    groupby_fields = ['use_sampling','num_beams','delta','gamma'] # regular grouping"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 190,
   "metadata": {},
   "outputs": [],
   "source": [
    "grouped_df = df.groupby(groupby_fields)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 191,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Number of rows after filtering: 6832\n",
      "Number of groups: 14\n"
     ]
    }
   ],
   "source": [
    "print(f\"Number of rows after filtering: {len(df)}\")\n",
    "print(f\"Number of groups: {len(grouped_df)}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 192,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "90,000, their annual compensation for 2010 could cost about USD 9.5 million . The City ’ s proposed contract agreement for 2011-2012 Laborers Union calls for another 3.25 percent increase, again according to a July contract agreement, in the salaries of sanitation laborers . This would increase in the total costs in 2011 - 2012 from reducing the number of labor-intensive trash collections by approximately $8 million . With the reduction in laborers, the number of garbage routes could be reduced to 262, or a 17.9% reduction . This would result in 78,100 fewer collections per year based on current garbage routes (264,250/27,700). The table below shows the reduction in staffing costs associated with garbage routes on a ward-by-ward basis . In the alternative that uses regional garbage truck routing, garbage pickup would be uninterrupted on 744 garbage routes, and trash collection would be uninterrupted on 1,136 routes . Under the option that\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "{'num_tokens_scored': 192,\n",
       " 'num_green_tokens': 125,\n",
       " 'green_fraction': 0.6510416666666666,\n",
       " 'z_score': 4.185789451624787,\n",
       " 'p_value': 1.420883383516844e-05,\n",
       " 'prediction': True,\n",
       " 'confidence': 0.9999857911661648}"
      ]
     },
     "execution_count": 192,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# input_text = \"Machine learning is one of the most prominent aspects on the web. The field is still quite new (not even half a century old), but it has become one of the fastest-growing and most talked-about areas since the beginning.\"\n",
    "# input_text = \"Although it is a relatively young field, machine learning has quickly become one of the most significant features of the internet, experiencing rapid growth and generating a great deal of discussion.\"\n",
    "# input_text = \"The domain of machine learning is highly prominent on the internet. Despite being relatively new (less than 50 years old), it has rapidly developed into one of the most rapidly expanding and widely-discussed fields.\"\n",
    "\n",
    "\n",
    "# input_text = \"The diamondback terrapin or simply terrapin (Malaclemys terrapin) is a species of turtle native to the brackish coastal tidal marshes of the Northeastern and southern United States, and in Bermuda.[6] It belongs to the monotypic genus Malaclemys. It has one of the largest ranges of all turtles in North America, stretching as far south as the Florida Keys and as far north as Cape Cod.[7] The name 'terrapin' is derived from the Algonquian word torope.[8] It applies to Malaclemys terrapin in both British English and American English. The name originally was used by early European settlers in North America to describe these brackish-water turtles that inhabited neither freshwater habitats nor the sea. It retains this primary meaning in American English.[8] In British English, however, other semi-aquatic turtle species, such as the red-eared slider, might also be called terrapins. The common name refers to the diamond pattern on top of its shell (carapace), but the overall pattern and coloration vary greatly. The shell is usually wider at the back than in the front, and from above it appears wedge-shaped. The shell coloring can vary from brown to grey, and its body color can be grey, brown, yellow, or white. All have a unique pattern of wiggly, black markings or spots on their body and head. The diamondback terrapin has large webbed feet.[9] The species is\"\n",
    "# input_text = \" sexually dimorphic in that the males grow to a carapace length of approximately 13 cm (5 in), while the females grow to an average carapace length of around 19 cm (7+1⁄2 in), though they are capable of growing larger. The largest female on record was just over 23 cm (9 in) in carapace length. Specimens from regions that are consistently warmer in temperature tend to be larger than those from cooler, more northern areas.[10] Male diamondback terrapins weigh 300 g (11 oz) on average, while females weigh around 500 g (18 oz).[11] The largest females can weigh up to 1 kg (35 oz).\"\n",
    "# input_text = \"distinguished from the box turtle, which is native to North America, by its diamond-shaped carapace, which is much wider at the back than in the front. The diamondback terrapin is distinguishable from the common snapping turtle, which lives mostly in freshwater and saltwater, by its diamond pattern on the shell. The diamondback terrapin, however, also can live in freshwater in its tropical habitats, where it may occasionally be found with its well-known relatives, the box and the common snapping turtle. This is rare however.[3] The shell is also distinctive, being wedge-shaped at the back and tapering forward at the front. The shell color ranges from light tan to gray or white. The most recognizable feature is the diamond-shaped marking on the shell. The shell is usually wider at the back than in the front. The shell color ranges from light brown to grey or white. The most recognizable feature is the diamond-shaped marking on the shell. Its\"\n",
    "# input_text = \"unique diamond-shaped carapace, wider at the back than in the front, sets the diamondback terrapin apart from the North American box turtle. Its diamond pattern on the shell distinguishes it from the common snapping turtle, which typically inhabits freshwater and saltwater environments. Although the diamondback terrapin is primarily found in tropical habitats, it can also be found in freshwater, occasionally alongside its box and common snapping turtle relatives. The shell of the diamondback terrapin is wedge-shaped at the back and tapers towards the front, with a color range of light tan to gray or white. The most distinctive feature of the shell is the diamond-shaped marking.\"\n",
    "# input_text = \"unique diamond-shaped carapace, wider at the back than the front, sets the diamondback terrapin apart from the box turtle, a native of North America. Additionally, the diamond pattern on its shell helps differentiate it from the common snapping turtle, which primarily inhabits freshwater and saltwater environments. While the diamondback terrapin typically resides in tropical habitats, it may also be found in freshwater on occasion, alongside its box and common snapping turtle counterparts. Its shell is distinctively wedge-shaped at the back, gradually tapering towards the front, and comes in shades of light tan, gray, or white. But it is the diamond-shaped marking on the shell that is the most distinguishable feature.\"\n",
    "\n",
    "# input_text = input_text + input_text\n",
    "\n",
    "if \"w_bl_attacked_whitelist_fraction\" in df.columns and 'replace_ratio' in df.columns:\n",
    "    # temp_grp_df = df.groupby(groupby_fields).get_group((True, 1, 2.0, 0.5, 0.1))\n",
    "    temp_grp_df = df.groupby(groupby_fields).get_group((True, 1, 2.0, 0.5, 0.2))\n",
    "    # temp_grp_df = df.groupby(groupby_fields).get_group((True, 1, 2.0, 0.5, 0.5))\n",
    "else:\n",
    "    temp_grp_df = df.groupby(groupby_fields).get_group((True, 1, 2.0, 0.5))\n",
    "\n",
    "idx = 88\n",
    "# output_type = \"baseline_completion\"\n",
    "# output_type = \"no_bl_output\"\n",
    "# output_type = \"w_bl_output\"\n",
    "output_type = \"w_bl_output_attacked\"\n",
    "\n",
    "input_text = temp_grp_df[output_type].iloc[idx]\n",
    "\n",
    "print(input_text)\n",
    "\n",
    "\n",
    "detector_output = watermark_detector.detect(input_text,return_green_token_mask=True)\n",
    "watermarked_hit_list = [detector_output.pop(\"green_token_mask\")]\n",
    "watermarked_hit_matrix = (torch.tensor(watermarked_hit_list, dtype=bool)).numpy() # for plotting code\n",
    "detector_output"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 193,
   "metadata": {},
   "outputs": [],
   "source": [
    "# hit_list = [c for c in \"RGGRRGGRGGG\"] # example with 1,and 2 for R and 2, 3 for G\n",
    "# hit_list = np.array([h==\"G\" for h in hit_list])\n",
    "# watermarked_hit_matrix = (torch.tensor(hit_list, dtype=bool).unsqueeze(dim=0)).numpy() # for plotting code\n",
    "\n",
    "# # gamma = 0.25\n",
    "# gamma = 0.5\n",
    "\n",
    "# chi_squared_runs_test(hit_list, 1-gamma, variant=variant, bin_spec=bin_spec, verbose=True, mask_zeros=mask_zeros, lambda_=lambda_)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 194,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Boolean array: [ True False False False  True False  True  True  True  True  True False\n",
      "  True  True False False False  True  True  True False False  True  True\n",
      "  True  True False  True  True False  True False False False False False\n",
      "  True False  True  True  True  True False False  True  True  True  True\n",
      "  True False  True  True  True  True  True False False  True False False\n",
      "  True  True False False False False  True False False False  True  True\n",
      "  True  True False False False  True  True  True  True  True False  True\n",
      "  True  True False  True  True False  True  True  True  True  True False\n",
      " False  True  True  True False False  True False  True False False False\n",
      " False  True  True  True  True  True  True  True  True False  True  True\n",
      "  True False  True  True  True  True  True  True  True  True  True  True\n",
      "  True False  True  True False  True  True False  True  True  True False\n",
      "  True  True False False  True  True  True  True  True  True False False\n",
      " False  True False  True  True  True False False False  True  True False\n",
      "  True  True  True False  True  True  True  True  True False  True  True\n",
      "  True  True  True  True  True  True  True False  True False  True  True]\n",
      "likelihood of success=F (1-gamma), or T run length geom dist 'p' = 0.5\n",
      "Raw run lengths and their values and types:\n",
      "[ 1  3  1  1  5  1  2  3  3  2  4  1  2  1  1  5  1  1  4  2  5  1  5  2\n",
      "  1  2  2  4  1  3  4  3  5  1  3  1  2  1  5  2  3  2  1  1  1  4  8  1\n",
      "  3  1 11  1  2  1  2  1  3  1  2  2  6  3  1  1  3  3  2  1  3  1  5  1\n",
      "  9  1  1  1  2]\n",
      "[ True False  True False  True False  True False  True False  True False\n",
      "  True False  True False  True False  True False  True False  True False\n",
      "  True False  True False  True False  True False  True False  True False\n",
      "  True False  True False  True False  True False  True False  True False\n",
      "  True False  True False  True False  True False  True False  True False\n",
      "  True False  True False  True False  True False  True False  True False\n",
      "  True False  True False  True]\n",
      "Unique T run lengths:  [ 1  2  3  4  5  6  8  9 11]\n",
      "Total T runs: 39\n",
      "Unique F run lengths:  [1 2 3 4 5]\n",
      "Total F runs: 38\n",
      "T Length bins:  [ 1  2  3  4  5  6  7  8  9 10 11 12]\n",
      "F Length bins:  [1 2 3 4 5 6]\n",
      "Obs T counts:  [10.  9.  7.  3.  6.  1.  0.  1.  1.  0.  1.  0.]\n",
      "Exp T counts:  [1.95000000e+01 9.75000000e+00 4.87500000e+00 2.43750000e+00\n",
      " 1.21875000e+00 6.09375000e-01 3.04687500e-01 1.52343750e-01\n",
      " 7.61718750e-02 3.80859375e-02 1.90429688e-02 1.90429688e-02]\n",
      "densities: sum=1.0, [5.0000000e-01 2.5000000e-01 1.2500000e-01 6.2500000e-02 3.1250000e-02\n",
      " 1.5625000e-02 7.8125000e-03 3.9062500e-03 1.9531250e-03 9.7656250e-04\n",
      " 4.8828125e-04 4.8828125e-04]\n",
      "Obs F counts:  [22.  7.  6.  2.  1.  0.]\n",
      "Exp F counts:  [19.      9.5     4.75    2.375   1.1875  1.1875]\n",
      "densities: sum=1.0, [0.5     0.25    0.125   0.0625  0.03125 0.03125]\n",
      "Joined Obs counts:  [10.0 9.0 7.0 3.0 6.0 1.0 -- 1.0 1.0 -- 1.0 -- 22.0 7.0 6.0 2.0 1.0 --]\n",
      "Joined Exp counts:  [1.95000000e+01 9.75000000e+00 4.87500000e+00 2.43750000e+00\n",
      " 1.21875000e+00 6.09375000e-01 3.04687500e-01 1.52343750e-01\n",
      " 7.61718750e-02 3.80859375e-02 1.90429688e-02 1.90429688e-02\n",
      " 1.90000000e+01 9.50000000e+00 4.75000000e+00 2.37500000e+00\n",
      " 1.18750000e+00 1.18750000e+00]\n",
      "[ 1  2  3  4  5  6  7  8  9 10 11 12]\n",
      "run_len_w_bl_attacked_T5_T_and_F_runs-cressie_read-max_plus_1-True\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 1600x500 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# save_plot = True\n",
    "save_plot = False\n",
    "\n",
    "plt.tight_layout()\n",
    "fig, axes = plt.subplots(1, 2, figsize=(16, 5))\n",
    "\n",
    "# show_density = True\n",
    "show_density = False\n",
    "\n",
    "# print(watermarked_hit_matrix.shape)\n",
    "\n",
    "n = watermarked_hit_matrix.shape[0]\n",
    "\n",
    "t_values = torch.arange(0,watermarked_hit_matrix.shape[1])\n",
    "indices = torch.arange(0,watermarked_hit_matrix.shape[1])\n",
    "\n",
    "# print(watermarked_hit_matrix.shape)\n",
    "\n",
    "values_arrays, lengths_arrays = [], []\n",
    "\n",
    "for bool_arr in watermarked_hit_matrix:\n",
    "    # values, lengths = rle_np_neq(bool_arr)\n",
    "    lengths = rle_F_succ_T_runs(bool_arr)\n",
    "    values = np.ones_like(lengths, dtype=bool)\n",
    "    # lengths = rle_T_succ_F_runs(bool_arr)\n",
    "    # values = np.zeros_like(lengths, dtype=bool)\n",
    "    # print(lengths)\n",
    "    values_arrays.append(values)\n",
    "    lengths_arrays.append(lengths)\n",
    "\n",
    "# select the lengths for each row where values is True\n",
    "# (i.e. where the run is a hit)\n",
    "hit_lengths = [lengths[values] for values, lengths in zip(values_arrays, lengths_arrays)]\n",
    "miss_lengths = [lengths[~values] for values, lengths in zip(values_arrays, lengths_arrays)]\n",
    "\n",
    "all_lengths = hit_lengths + miss_lengths\n",
    "# remove empty lists\n",
    "all_lengths = [l for l in all_lengths if len(l) > 0]\n",
    "\n",
    "max_length = max([max(l) for l in all_lengths])\n",
    "# print(max_length)\n",
    "\n",
    "# concatenate all the lengths into one array for each type\n",
    "hit_lengths = np.concatenate(hit_lengths)\n",
    "miss_lengths = np.concatenate(miss_lengths)\n",
    "\n",
    "# then use matpltolib to plot the histograms of the lengths\n",
    "# using green for hit lengths and red for miss lengths\n",
    "# bin equally for both from 0 up to the max length\n",
    "# in a separate plot for each output_type\n",
    "\n",
    "# bin_spec = 20\n",
    "# if bin_spec == 20:\n",
    "#     bins = np.arange(0, 20+1, 1)\n",
    "# elif bin_spec == 100:\n",
    "#     bins = np.arange(0, 100+1, 5)\n",
    "# elif bin_spec == 200:\n",
    "#     bins = np.arange(0, 200+1, 10)\n",
    "\n",
    "# bins = np.arange(0, 13, 1)\n",
    "\n",
    "# hit_lengths[hit_lengths>10] = 10\n",
    "# miss_lengths[miss_lengths>10] = 10\n",
    "# print(miss_lengths)\n",
    "\n",
    "total_hit_runs = len(hit_lengths)\n",
    "total_miss_runs = len(miss_lengths)\n",
    "\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "if variant  == \"F_succ_T_runs\" :\n",
    "    statistic, p_val, total_runs, bins, obs_counts, exp_counts =  chi_squared_runs_test(bool_arr, 1-gamma, variant=\"F_succ_T_runs\", invert_bools=False, mask_zeros=mask_zeros, bin_spec=bin_spec, lambda_=lambda_, return_bin_counts=True, verbose=True)\n",
    "    total_T_runs, T_bins, obs_T_counts, exp_T_counts, total_F_runs, F_bins, obs_F_counts, exp_F_counts = None, None, None, None, None, None, None, None\n",
    "elif variant == \"T_and_F_runs\" :\n",
    "    statistic, p_val, total_T_runs, T_bins, obs_T_counts, exp_T_counts, total_F_runs, F_bins, obs_F_counts, exp_F_counts  =  chi_squared_runs_test(bool_arr, 1-gamma, variant=\"T_and_F_runs\", invert_bools=False, mask_zeros=mask_zeros, bin_spec=bin_spec, lambda_=lambda_, return_bin_counts=True, verbose=True)\n",
    "    total_runs, obs_counts, exp_counts = None, None, None\n",
    "else:\n",
    "    raise ValueError()\n",
    "\n",
    "ax = axes[0]\n",
    "# add density histogram to this subplot\n",
    "# bar_spec = ax.hist(miss_lengths, bins=bins, color='red', alpha=0.0, label='red runs', density=show_density)\n",
    "# ax.bar(bar_spec[1][:-1], bar_spec[0], label='geom pmf', color='red', alpha=0.4)\n",
    "\n",
    "# bar_spec = ax.hist(hit_lengths, bins=bins, color='green', alpha=0.0, label='green runs', density=show_density)\n",
    "# ax.bar(bar_spec[1][:-1], bar_spec[0], label='geom pmf', color='green', alpha=0.4)\n",
    "\n",
    "if T_bins is not None and F_bins is not None:\n",
    "    ax.bar(T_bins, obs_T_counts, label='Obs T runs', color='green', alpha=0.4)\n",
    "    ax.bar(F_bins, obs_F_counts, label='Obs F runs', color='red', alpha=0.4)\n",
    "else:\n",
    "    ax.bar(bins, obs_counts, label='Obs T runs', color='green', alpha=0.4)\n",
    "\n",
    "\n",
    "ax.set_title('Observed')\n",
    "if T_bins is not None:\n",
    "    print(T_bins)\n",
    "    ax.xaxis.set_ticks((T_bins if max(T_bins) >= max(F_bins) else F_bins))\n",
    "else:\n",
    "    ax.xaxis.set_ticks(bins)\n",
    "ax.set_ylabel('Density' if show_density else 'Count')\n",
    "\n",
    "\n",
    "# bins = np.arange(geom.ppf(0.01, p),\n",
    "#               geom.ppf(0.99, p))\n",
    "\n",
    "# miss_pmf = geom.pmf(bins, p)\n",
    "# hit_pmf = geom.pmf(bins, 1-p)\n",
    "\n",
    "ax = axes[1]\n",
    "if not show_density:\n",
    "    # ax.bar(bins, miss_pmf * total_miss_runs, label='geom pmf', color='red', alpha=0.4)\n",
    "    # ax.bar(bins, hit_pmf * total_hit_runs, label='geom pmf', color='green', alpha=0.4)\n",
    "\n",
    "    if T_bins is not None and F_bins is not None:\n",
    "        ax.bar(T_bins, exp_T_counts, label='Exp T Runs', color='green', alpha=0.4)\n",
    "        ax.bar(F_bins, exp_F_counts, label='Exp F Runs', color='red', alpha=0.4)\n",
    "    else:\n",
    "        ax.bar(bins, exp_counts, label='Exp T Runs', color='green', alpha=0.4)\n",
    "else:\n",
    "    # ax.bar(bins, miss_pmf, label='geom pmf', color='red', alpha=0.4)\n",
    "    # ax.bar(bins, hit_pmf, label='geom pmf', color='green', alpha=0.4)\n",
    "    raise NotImplementedError(\"No longer support density plots\")\n",
    "ax.set_title('Expected')\n",
    "if T_bins is not None:\n",
    "    ax.xaxis.set_ticks((T_bins if max(T_bins) >= max(F_bins) else F_bins))\n",
    "else:\n",
    "    ax.xaxis.set_ticks(bins)\n",
    "ax.set_ylabel('Density' if show_density else 'Count')\n",
    "\n",
    "# make figure caption\n",
    "# caption = (f\"Outcome of the chisquared test: statistic={statistic:.2f} with p-value {p_val:.2e} for {int(total_runs)} runs distibuted across {len(bins)} bins (determined via bin_spec={bin_spec}).\\nOutcome of z-test : statistic={detector_output['z_score']:.2f} with p-value {detector_output['p_value']:.2e} based on green fraction={detector_output['green_fraction']:.2f} for T={detector_output['num_tokens_scored']}.\")\n",
    "if T_bins is not None:\n",
    "    caption = (f\"Test outcome: statistic={statistic:.2f} with p-value {p_val:.2e} for {int(total_T_runs + total_F_runs)} runs in {len(T_bins) + len(F_bins)} bins | (variant,statistic,bin_spec,ignore_zeros)=({variant}, {lambda_}, {bin_spec}, {mask_zeros})\")\n",
    "else:\n",
    "    caption = (f\"Test outcome: statistic={statistic:.2f} with p-value {p_val:.2e} for {int(total_runs)} runs | (variant,statistic,bin_spec,ignore_zeros)=({variant}, {lambda_}, {bin_spec}, {mask_zeros})\")\n",
    "\n",
    "# add caption to figure\n",
    "fig.text(0.5, 0.02, caption, ha='center', fontsize=14)\n",
    "\n",
    "plot_name = f\"run_len_{output_type.replace('_output','')}_{'turbo' if 'turbo' in save_dir else 'T5'}_{variant}-{lambda_}-{bin_spec}-{mask_zeros}\"\n",
    "print(plot_name)\n",
    "\n",
    "if save_plot:\n",
    "    fname = f\"{OUTPUT_DIR}/figs/{plot_name}.png\"\n",
    "    plt.savefig(fname, format=\"png\", bbox_inches='tight', dpi=600)\n",
    "\n",
    "fig.show()"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### ROC unattacked"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 195,
   "metadata": {},
   "outputs": [],
   "source": [
    "# plt.clf()\n",
    "# plt.figure(constrained_layout=True)\n",
    "# plt.figure(figsize=(5, 4))\n",
    "\n",
    "# import sklearn.metrics as metrics\n",
    "\n",
    "# zoom = False\n",
    "# # zoom = True\n",
    "\n",
    "# beam_search = None\n",
    "# # beam_search = 1\n",
    "# # beam_search = 4\n",
    "# # beam_search = 8\n",
    "\n",
    "# deltas = [1.0,2.0,5.0,10.0]\n",
    "# # gammas = [0.25, 0.5]\n",
    "# gammas = [0.25]\n",
    "# # gammas = [0.5]\n",
    "\n",
    "# # deltas = [1.0,2.0,5.0,10.0]\n",
    "# # gammas = [0.1,0.5]\n",
    "\n",
    "# groups = []\n",
    "# names = []\n",
    "# for d in deltas:\n",
    "#     for g in gammas:\n",
    "#         if beam_search:\n",
    "#             groups.append((False, beam_search, d, g))\n",
    "#         else:\n",
    "#             groups.append((True, 1, d, g))\n",
    "#         names.append(f\"$\\delta:{d},\\gamma:{g}$\")\n",
    "# groups=groups[::-1]\n",
    "# names=names[::-1]\n",
    "\n",
    "# # Make colormap\n",
    "# import matplotlib.pyplot as plt\n",
    "# viridis = plt.colormaps['viridis'].resampled(len(groups)+1) \n",
    "# cmap = viridis.colors[:len(groups)][::-1]\n",
    "\n",
    "# # plot different parameter levels\n",
    "# for i,(group,name) in enumerate(zip(groups,names)):\n",
    "\n",
    "#     # baseline_z_scores = grouped_df.get_group(group)[\"baseline_z_score\"].values\n",
    "#     # w_bl_z_scores = grouped_df.get_group(group)[\"w_bl_z_score\"].values\n",
    "#     baseline_z_scores = grouped_df.get_group(group)[\"baseline_chisqrd_stat\"].values # HACK\n",
    "#     w_bl_z_scores = grouped_df.get_group(group)[\"w_bl_chisqrd_stat\"].values # HACK\n",
    "#     all_scores = np.concatenate([baseline_z_scores,w_bl_z_scores])\n",
    "\n",
    "#     baseline_labels = np.zeros_like(baseline_z_scores)\n",
    "#     attacked_labels = np.ones_like(w_bl_z_scores)\n",
    "#     all_labels = np.concatenate([baseline_labels,attacked_labels])\n",
    "\n",
    "#     fpr, tpr, thresholds = metrics.roc_curve(all_labels, all_scores, pos_label=1)\n",
    "#     roc_auc = metrics.auc(fpr, tpr)\n",
    "\n",
    "#     plt.plot(fpr, tpr, color=cmap[i], label = f'{name}, AUC:%0.3f, PPL:{round(grouped_df[\"w_bl_ppl\"].describe().loc[group][\"mean\"],1)}' % roc_auc, linewidth=3)\n",
    "\n",
    "# if \"w_bl_attacked_ppl\" in df.columns:\n",
    "#     pass\n",
    "# else:\n",
    "#     # # vanilla ppl value\n",
    "#     plt.scatter([-1],[-1],label=f'            $\\delta=0$, PPL: {round(grouped_df[\"no_bl_ppl\"].describe().loc[groups,\"mean\"].mean(),1)}', color=\"white\")\n",
    "\n",
    "# if zoom:\n",
    "#     if not \"w_bl_attacked_ppl\" in df.columns:\n",
    "#         plt.legend(loc = 'lower right', fontsize = 12)\n",
    "#     plt.xscale(\"log\")\n",
    "#     # plt.yscale(\"log\")\n",
    "#     plt.xlim([0, 1])\n",
    "#     plt.ylim([0.5, 1])\n",
    "#     plot_name = (\"roc_auc_zoom\" if not beam_search else f\"roc_auc_zoom_greedy_beams_{beam_search}\")\n",
    "\n",
    "# else:\n",
    "#     # if \"w_bl_attacked_ppl\" in df.columns:\n",
    "#     plt.legend(loc = 'lower right', fontsize = 12)\n",
    "    \n",
    "#     plt.plot([0, 1], [0, 1],'r--')\n",
    "#     plt.xlim([0, 1])\n",
    "#     plt.ylim([0, 1])\n",
    "#     plot_name = (\"roc_auc\" if not beam_search else f\"roc_auc_greedy_beams_{beam_search}\")\n",
    "\n",
    "# plt.ylabel('True Positive Rate', fontsize = 12)\n",
    "# plt.xlabel('False Positive Rate', fontsize = 12)\n",
    "\n",
    "# # plot_name = \"roc_auc_unattacked_z_test\"\n",
    "# plot_name = \"roc_auc_unattacked_chi_test\"\n",
    "\n",
    "# print(plot_name)\n",
    "\n",
    "# # fname = f\"{OUTPUT_DIR}/figs/{plot_name}.pdf\"\n",
    "# # plt.savefig(fname, format=\"pdf\")\n",
    "\n",
    "# plt.show()"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Attacked ROC"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 196,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "roc_auc_T5_attack_chi_test_T_and_F_runs-cressie_read-max_plus_1-True\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "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 500x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import sklearn.metrics as metrics\n",
    "\n",
    "save_plot = True\n",
    "# save_plot = False\n",
    "\n",
    "# test_type = \"z_test\"\n",
    "test_type = \"chi_test\"\n",
    "\n",
    "plt.clf()\n",
    "plt.figure(constrained_layout=True)\n",
    "plt.figure(figsize=(5, 4))\n",
    "\n",
    "if 'turbo' not in save_name:\n",
    "    attack_budgets = [0.1,0.3,0.5,0.7]\n",
    "    groups = [(True, 1, 2.0, 0.5, budget) for budget in attack_budgets]\n",
    "else:\n",
    "    groups = [(True, 1, 2.0, 0.5)] # HACK\n",
    "\n",
    "beams = False\n",
    "# groups = [(False, 8, 0.5, 2.0, budget) for budget in attack_budgets]\n",
    "# beams = True\n",
    "\n",
    "if 'turbo' not in save_name:\n",
    "    names = [f\"$\\epsilon={eps}$\" for eps in attack_budgets] # HACK\n",
    "else:\n",
    "    names = [f\"gpt-3.5-turbo\"]\n",
    "\n",
    "# Make colormap\n",
    "import matplotlib.pyplot as plt\n",
    "viridis = plt.colormaps['viridis'].resampled(len(groups)+1+1) # attack\n",
    "cmap = viridis.colors[:len(groups)+1][::-1]\n",
    "\n",
    "# plot original\n",
    "group = groups[0] # any will do\n",
    "if test_type == \"z_test\":\n",
    "    baseline_z_scores = grouped_df.get_group(group)[\"baseline_z_score\"].values\n",
    "elif test_type == \"chi_test\":\n",
    "    baseline_z_scores = grouped_df.get_group(group)[\"baseline_chisqrd_stat\"].values # HACK\n",
    "\n",
    "baseline_labels = np.zeros_like(baseline_z_scores)\n",
    "\n",
    "if test_type == \"z_test\":\n",
    "    orig_watermark_z_scores = grouped_df.get_group(group)[\"w_bl_z_score\"].values\n",
    "elif test_type == \"chi_test\":\n",
    "    orig_watermark_z_scores = grouped_df.get_group(group)[\"w_bl_chisqrd_stat\"].values # HACK\n",
    "watermark_labels = np.ones_like(orig_watermark_z_scores)\n",
    "\n",
    "all_scores = np.concatenate([baseline_z_scores,orig_watermark_z_scores])\n",
    "all_labels = np.concatenate([baseline_labels,watermark_labels])\n",
    "\n",
    "fpr, tpr, thresholds = metrics.roc_curve(all_labels, all_scores, pos_label=1)\n",
    "roc_auc = metrics.auc(fpr, tpr)\n",
    "\n",
    "if \"turbo\" not in save_name:\n",
    "    plt.plot(fpr, tpr, color=cmap[0], label = f'unattacked, AUC:%0.3f, PPL:{round(grouped_df[\"w_bl_ppl\"].describe().loc[group][\"mean\"],1)}' % roc_auc, linewidth=3)\n",
    "else:\n",
    "    plt.plot(fpr, tpr, color=cmap[0], label = f'unattacked, AUC:%0.3f, BERTscore:{1.0}' % roc_auc, linewidth=3) # HACK\n",
    "\n",
    "# plot different attack levels\n",
    "for i,(group,name) in enumerate(zip(groups,names)):\n",
    "\n",
    "    if test_type == \"z_test\":\n",
    "        baseline_z_scores = grouped_df.get_group(group)[\"baseline_z_score\"].values\n",
    "        attacked_z_scores = grouped_df.get_group(group)[\"w_bl_attacked_z_score\"].values\n",
    "    else:\n",
    "        baseline_z_scores = grouped_df.get_group(group)[\"baseline_chisqrd_stat\"].values # HACK\n",
    "        attacked_z_scores = grouped_df.get_group(group)[\"w_bl_attacked_chisqrd_stat\"].values # HACK\n",
    "    all_scores = np.concatenate([baseline_z_scores,attacked_z_scores])\n",
    "\n",
    "    baseline_labels = np.zeros_like(baseline_z_scores)\n",
    "    attacked_labels = np.ones_like(attacked_z_scores)\n",
    "    all_labels = np.concatenate([baseline_labels,attacked_labels])\n",
    "\n",
    "    fpr, tpr, thresholds = metrics.roc_curve(all_labels, all_scores, pos_label=1)\n",
    "    roc_auc = metrics.auc(fpr, tpr)\n",
    "\n",
    "    if \"turbo\" not in save_name:\n",
    "        plt.plot(fpr, tpr, color=cmap[i+1], label = f'{name}, AUC:%0.3f, PPL:{round(grouped_df[\"w_bl_attacked_ppl\"].describe().loc[group][\"mean\"],1)}' % roc_auc, linewidth=3)\n",
    "    else:\n",
    "        plt.plot(fpr, tpr, color=cmap[i+1], label = f'{name}, AUC:%0.3f, BERTscore:{round(grouped_df[\"bert_score_attacked\"].describe().loc[group][\"mean\"],1)}' % roc_auc, linewidth=3) # HACK\n",
    "\n",
    "if \"w_bl_attacked_ppl\" in df.columns:\n",
    "    pass\n",
    "elif \"no_bl_ppl\" in df.columns:\n",
    "    # # vanilla ppl value\n",
    "    plt.scatter([-1],[-1],label=f'            $\\delta=0$, PPL: {round(grouped_df[\"no_bl_ppl\"].describe().loc[groups,\"mean\"].mean(),1)}', color=\"white\")\n",
    "else:\n",
    "    pass\n",
    "\n",
    "zoom = False\n",
    "# zoom = True\n",
    "if zoom:\n",
    "    if not \"w_bl_attacked_ppl\" in df.columns:\n",
    "        plt.legend(loc = 'lower right')\n",
    "    plt.xscale(\"log\")\n",
    "    # plt.yscale(\"log\")\n",
    "    plt.xlim([0, 1])\n",
    "    plt.ylim([0.5, 1])\n",
    "    if \"w_bl_attacked_ppl\" in df.columns:\n",
    "        plot_name = \"roc_auc_untargeted_attack_no_beams_zoom\"\n",
    "        # plot_name = \"roc_auc_untargeted_attack_with_beams_zoom\"\n",
    "    else:\n",
    "        plot_name = \"roc_auc_zoom\"\n",
    "else:\n",
    "    if \"w_bl_attacked_ppl\" in df.columns:\n",
    "        plt.legend(loc = 'lower right',fontsize = 9)\n",
    "    \n",
    "    plt.plot([0, 1], [0, 1],'r--')\n",
    "    plt.xlim([0, 1])\n",
    "    plt.ylim([0, 1])\n",
    "    if \"w_bl_attacked_ppl\" in df.columns:\n",
    "        if beams: plot_name = \"roc_auc_untargeted_attack_w_beams\"\n",
    "        if not beams: plot_name = \"roc_auc_untargeted_attack_no_beams\"\n",
    "    else:\n",
    "        plot_name = \"roc_auc\"\n",
    "\n",
    "plt.legend(loc = 'lower right',fontsize = 9) # HACK\n",
    "\n",
    "plt.ylabel('True Positive Rate')\n",
    "plt.xlabel('False Positive Rate')\n",
    "\n",
    "if test_type == \"z_test\":\n",
    "    plt.title(f\"ROC-AUC | z-score\")\n",
    "elif test_type == \"chi_test\":\n",
    "    plt.title(f\"ROC-AUC | run length based chi-squared test\")\n",
    "    caption = f\"\\n(variant,statistic,bin_spec,ignore_zeros)=({variant}, {lambda_}, {bin_spec}, {mask_zeros})\"\n",
    "    plt.text(0.5, -0.2, caption, ha='center', fontsize=11)\n",
    "\n",
    "plot_name = f\"roc_auc_{'turbo' if 'turbo' in save_dir else 'T5'}_attack_{test_type}{f'_{variant}-{lambda_}-{bin_spec}-{mask_zeros}' if test_type == 'chi_test' else ''}\"\n",
    "print(plot_name)\n",
    "\n",
    "if save_plot:\n",
    "    fname = f\"{OUTPUT_DIR}/figs/{plot_name}.png\"\n",
    "    plt.savefig(fname, format=\"png\", bbox_inches='tight', dpi=600)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sklearn.metrics as metrics\n",
    "\n",
    "def reject_null_hypo(z_score=None,cuttoff=None):\n",
    "    return z_score > cuttoff\n",
    "\n",
    "records = []\n",
    "\n",
    "for group_params in tqdm(list(grouped_df.groups.keys())):\n",
    "    sub_df = grouped_df.get_group(group_params)\n",
    "    grp_size = len(sub_df)\n",
    "\n",
    "    # baseline_z_scores = sub_df[\"baseline_z_score\"].values\n",
    "    # w_bl_z_scores = sub_df[\"w_bl_z_score\"].values\n",
    "    # all_scores = np.concatenate([baseline_z_scores,w_bl_z_scores])\n",
    "\n",
    "    # baseline_labels = np.zeros_like(baseline_z_scores)\n",
    "    # attacked_labels = np.ones_like(w_bl_z_scores)\n",
    "    # all_labels = np.concatenate([baseline_labels,attacked_labels])\n",
    "\n",
    "    # fpr, tpr, thresholds = metrics.roc_curve(all_labels, all_scores, pos_label=1)\n",
    "    # roc_auc = metrics.auc(fpr, tpr)\n",
    "    record = {k:v for k,v in zip(groupby_fields,group_params)}\n",
    "\n",
    "    # for thresh in [4.0,5.0]:\n",
    "    for thresh in [4.0,50.0, 1000.0]:\n",
    "        \n",
    "        record[\"count\"] = grp_size\n",
    "        # record[f\"baseline_fpr_at_{thresh}\"] = reject_null_hypo(z_score=sub_df[\"baseline_z_score\"].values,cuttoff=thresh).sum() / grp_size\n",
    "        # record[f\"baseline_tnr_at_{thresh}\"] = (~reject_null_hypo(z_score=sub_df[\"baseline_z_score\"],cuttoff=thresh)).sum() / grp_size\n",
    "        # record[f\"no_bl_fpr_at_{thresh}\"] = reject_null_hypo(z_score=sub_df[\"no_bl_z_score\"].values,cuttoff=thresh).sum() / grp_size\n",
    "        # record[f\"no_bl_tnr_at_{thresh}\"] = (~reject_null_hypo(z_score=sub_df[\"no_bl_z_score\"].values,cuttoff=thresh)).sum() / grp_size\n",
    "        # record[f\"w_bl_tpr_at_{thresh}\"] = reject_null_hypo(z_score=sub_df[\"w_bl_z_score\"].values,cuttoff=thresh).sum() / grp_size\n",
    "        # record[f\"w_bl_fnr_at_{thresh}\"] = (~reject_null_hypo(z_score=sub_df[\"w_bl_z_score\"].values,cuttoff=thresh)).sum() / grp_size\n",
    "\n",
    "        # if \"w_bl_attacked_z_score\" in sub_df.columns:\n",
    "        #     record[f\"w_bl_attacked_tpr_at_{thresh}\"] = reject_null_hypo(z_score=sub_df[\"w_bl_attacked_z_score\"].values,cuttoff=thresh).sum() / grp_size\n",
    "        #     record[f\"w_bl_attacked_fnr_at_{thresh}\"] = (~reject_null_hypo(z_score=sub_df[\"w_bl_attacked_z_score\"].values,cuttoff=thresh)).sum() / grp_size\n",
    "\n",
    "        # use the chisquared stat instead of zscores\n",
    "        record[f\"baseline_fpr_at_{thresh}\"] = reject_null_hypo(z_score=sub_df[\"baseline_chisqrd_stat\"].values,cuttoff=thresh).sum() / grp_size\n",
    "        record[f\"baseline_tnr_at_{thresh}\"] = (~reject_null_hypo(z_score=sub_df[\"baseline_chisqrd_stat\"],cuttoff=thresh)).sum() / grp_size\n",
    "        # record[f\"no_bl_fpr_at_{thresh}\"] = reject_null_hypo(z_score=sub_df[\"no_bl_chisqrd_stat\"].values,cuttoff=thresh).sum() / grp_size\n",
    "        # record[f\"no_bl_tnr_at_{thresh}\"] = (~reject_null_hypo(z_score=sub_df[\"no_bl_chisqrd_stat\"].values,cuttoff=thresh)).sum() / grp_size\n",
    "        record[f\"w_bl_tpr_at_{thresh}\"] = reject_null_hypo(z_score=sub_df[\"w_bl_chisqrd_stat\"].values,cuttoff=thresh).sum() / grp_size\n",
    "        record[f\"w_bl_fnr_at_{thresh}\"] = (~reject_null_hypo(z_score=sub_df[\"w_bl_chisqrd_stat\"].values,cuttoff=thresh)).sum() / grp_size\n",
    "\n",
    "        if \"w_bl_attacked_chisq_stat\" in sub_df.columns:\n",
    "            record[f\"w_bl_attacked_tpr_at_{thresh}\"] = reject_null_hypo(z_score=sub_df[\"w_bl_attacked_chisqrd_stat\"].values,cuttoff=thresh).sum() / grp_size\n",
    "            record[f\"w_bl_attacked_fnr_at_{thresh}\"] = (~reject_null_hypo(z_score=sub_df[\"w_bl_attacked_chisqrd_stat\"].values,cuttoff=thresh)).sum() / grp_size\n",
    "\n",
    "    records.append(record)\n",
    "\n",
    "    #     # df[f\"baseline_fp_at_{thresh}\"] = reject_null_hypo(z_score=df[\"baseline_z_score\"].values,cuttoff=thresh)\n",
    "    #     # df[f\"baseline_tn_at_{thresh}\"] = ~reject_null_hypo(z_score=df[\"baseline_z_score\"],cuttoff=thresh)\n",
    "    #     # df[f\"no_bl_fp_at_{thresh}\"] = reject_null_hypo(z_score=df[\"no_bl_z_score\"].values,cuttoff=thresh)\n",
    "    #     # df[f\"no_bl_tn_at_{thresh}\"] = ~reject_null_hypo(z_score=df[\"no_bl_z_score\"].values,cuttoff=thresh)\n",
    "    #     # df[f\"w_bl_tp_at_{thresh}\"] = reject_null_hypo(z_score=df[\"w_bl_z_score\"].values,cuttoff=thresh)\n",
    "    #     # df[f\"w_bl_fn_at_{thresh}\"] = ~reject_null_hypo(z_score=df[\"w_bl_z_score\"].values,cuttoff=thresh)\n",
    "\n",
    "\n",
    "roc_df = pd.DataFrame.from_records(records)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "roc_df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### prototyping chisqrd code"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# uniq_hit_lens, counts = np.unique(hit_lengths, return_counts=True)\n",
    "# hit_len_count_dict = dict(zip(uniq_hit_lens, np.array(counts,dtype=float)))\n",
    "# max_hit_len = max(uniq_hit_lens)\n",
    "# # add zeros for missing lengths\n",
    "# # for i in range(1, max_hit_len):\n",
    "# for i in range(1, bin_spec+1):\n",
    "#     if i not in hit_len_count_dict:\n",
    "#         hit_len_count_dict[i] = 0\n",
    "# hit_len_count_dict = {k: v for k, v in sorted(hit_len_count_dict.items(), key=lambda item: item[0])}\n",
    "# # hit_len_freq = counts / len(hit_lengths)\n",
    "# # hit_len_freq_dict = dict(zip(uniq_hit_lens, hit_len_freq))\n",
    "\n",
    "# # uniq_miss_lens, counts = np.unique(miss_lengths, return_counts=True)\n",
    "# # miss_len_count_dict = dict(zip(uniq_miss_lens, np.array(counts,dtype=float)))\n",
    "# # max_miss_len = max(uniq_miss_lens)\n",
    "# # # add zeros for missing lengths\n",
    "# # # for i in range(1, max_miss_len):\n",
    "# # for i in range(1, bin_spec+1):\n",
    "# #     if i not in miss_len_count_dict:\n",
    "# #         miss_len_count_dict[i] = 0\n",
    "# # miss_len_count_dict = {k: v for k, v in sorted(miss_len_count_dict.items(), key=lambda item: item[0])}\n",
    "# # # miss_len_freq = counts / len(miss_lengths)\n",
    "# # # miss_len_freq_dict = dict(zip(uniq_miss_lens, miss_len_freq))\n",
    "\n",
    "\n",
    "# # hit_len_freq_dict = {k: v / sum(hit_len_count_dict.values()) for k, v in hit_len_count_dict.items()}\n",
    "# # hit_len_count_dict = hit_len_freq_dict\n",
    "# # miss_len_freq_dict\n",
    "# hit_len_count_dict\n",
    "# # miss_len_count_dict\n",
    "\n",
    "\n",
    "# # exp_hit_len_counts = geom.pmf(range(1, max(hit_len_count_dict.keys())+1), 1-p) * sum(hit_len_count_dict.values())\n",
    "# # exp_hit_len_counts = exp_hit_len_counts[uniq_hit_lens-1]\n",
    "\n",
    "# exp_hit_len_counts = geom.pmf(list(hit_len_count_dict.keys()), 1-p) * sum(hit_len_count_dict.values())\n",
    "# # exp_hit_len_counts = geom.pmf(list(hit_len_count_dict.keys()), 1-p)\n",
    "\n",
    "# expected_hit_len_count_dict = dict(zip(sorted(hit_len_count_dict.keys()), exp_hit_len_counts))\n",
    "# expected_hit_len_count_dict\n",
    "\n",
    "\n",
    "# # # exp_miss_len_counts = geom.pmf(range(1, max(miss_len_count_dict.keys())+1), p) * sum(miss_len_count_dict.values())\n",
    "# # # exp_miss_len_counts = exp_miss_len_counts[uniq_miss_lens-1]\n",
    "\n",
    "# # exp_miss_len_counts = geom.pmf(list(miss_len_count_dict.keys()), p) * sum(miss_len_count_dict.values())\n",
    "# # # exp_miss_len_counts = geom.pmf(list(miss_len_count_dict.keys()), p)\n",
    "\n",
    "# # expected_miss_len_count_dict = dict(zip(sorted(miss_len_count_dict.keys()), exp_miss_len_counts))\n",
    "# # expected_miss_len_count_dict\n",
    "\n",
    "\n",
    "\n",
    "# sum(hit_len_count_dict.values())\n",
    "# # sum(miss_len_count_dict.values())\n",
    "\n",
    "# sum(expected_hit_len_count_dict.values())\n",
    "# # sum(expected_miss_len_count_dict.values())\n",
    "\n",
    "# import numpy as np\n",
    "# from scipy.stats import chisquare\n",
    "\n",
    "# obs = np.array(list(hit_len_count_dict.values()))\n",
    "# exp = np.array(list(expected_hit_len_count_dict.values()))\n",
    "\n",
    "# # obs = np.array(list(miss_len_count_dict.values()))\n",
    "# # exp = np.array(list(expected_miss_len_count_dict.values()))\n",
    "# chisquare(obs, exp)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### could generate some synth coin data (at least the leading sequences) and show what we're looking for"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Check to see what the distros look like for hit lists"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# save_fig = True\n",
    "save_fig = False\n",
    "\n",
    "# show density\n",
    "show_density = True\n",
    "# show_density = False\n",
    "\n",
    "sweep_field = \"gamma\"\n",
    "# sweep_field = \"delta\"\n",
    "\n",
    "group_list_gamma = (\n",
    "    (True, 1, 2.0, 0.1),\n",
    "    (True, 1, 2.0, 0.25),\n",
    "    (True, 1, 2.0, 0.5),\n",
    "    (True, 1, 2.0, 0.75),\n",
    "    (True, 1, 2.0, 0.9),\n",
    ")\n",
    "group_list_delta = (\n",
    "    (True, 1, 0.1, 0.5),\n",
    "    (True, 1, 0.5, 0.5),\n",
    "    (True, 1, 1.0, 0.5),\n",
    "    (True, 1, 2.0, 0.5),\n",
    "    (True, 1, 5.0, 0.5),\n",
    ")\n",
    "group_list = group_list_gamma if sweep_field == \"gamma\" else group_list_delta\n",
    "\n",
    "single_output_idx = None # for plotting the average over all samples\n",
    "# single_output_idx = 123 # for plotting the distribution of a single sample\n",
    "\n",
    "# output_type = \"baseline_hit_list\"\n",
    "output_type = \"w_bl_hit_list\"\n",
    "\n",
    "bin_spec = 20\n",
    "# bin_spec = 100\n",
    "\n",
    "# bin_spec = 200\n",
    "# bin_spec = \"max\"\n",
    "\n",
    "\n",
    "# set sublot widths for each group\n",
    "# widths = [1, 1, 1, 1, 1]\n",
    "widths = [2, 2, 2, 2, 2]\n",
    "# set subplot heights for each group\n",
    "heights = [1]\n",
    "# create plot with this gridspec\n",
    "fig, axs = plt.subplots(ncols=len(widths), nrows=1, figsize=(18,3), gridspec_kw={'width_ratios':widths, 'height_ratios':heights}, sharey=True)\n",
    "\n",
    "for idx, group in enumerate(group_list):\n",
    "    group_df = grouped_df.get_group(group)\n",
    "\n",
    "    group = list(group)\n",
    "    group[0] = \"sampling\" if group[0] else \"greedy\"\n",
    "    group[1] = f\"{group[1]} beams\"\n",
    "    group[2] = f\"delta={group[2]}\"\n",
    "    group[3] = f\"gamma={group[3]}\"\n",
    "\n",
    "    hit_list = group_df[output_type].to_list()\n",
    "\n",
    "    hit_list = [list(l) for l in hit_list]\n",
    "\n",
    "    lengths = [len(l) for l in hit_list]\n",
    "    diff_lengths = set(lengths) \n",
    "    counter = {}\n",
    "    for l in lengths:\n",
    "        if counter.get(l):\n",
    "            counter[l] += 1\n",
    "        else:\n",
    "            counter[l] = 1\n",
    "\n",
    "    max_t = None\n",
    "    if max_t:\n",
    "        min_length = min(min(diff_lengths),max_t)\n",
    "        max_t = min_length\n",
    "    else:\n",
    "        min_length = min(diff_lengths)\n",
    "    watermarked_hit_list = [l[:min_length] for l in hit_list]\n",
    "\n",
    "    watermarked_hit_matrix = (~torch.tensor(watermarked_hit_list, dtype=bool)).numpy()\n",
    "    # watermarked_hit_matrix\n",
    "\n",
    "    n = watermarked_hit_matrix.shape[0]\n",
    "\n",
    "    if max_t:\n",
    "        t_values = torch.arange(0,max_t)\n",
    "        indices = torch.arange(0,max_t)\n",
    "    else:\n",
    "        t_values = torch.arange(0,watermarked_hit_matrix.shape[1])\n",
    "        indices = torch.arange(0,watermarked_hit_matrix.shape[1])\n",
    "\n",
    "    watermarked_hit_matrix.shape\n",
    "\n",
    "    values_arrays, lengths_arrays = [], []\n",
    "\n",
    "    if single_output_idx is not None:\n",
    "        watermarked_hit_matrix = watermarked_hit_matrix[single_output_idx:single_output_idx+1]\n",
    "\n",
    "    for bool_arr in watermarked_hit_matrix:\n",
    "        values, lengths = rle_np_neq(bool_arr)\n",
    "        values_arrays.append(values)\n",
    "        lengths_arrays.append(lengths)\n",
    "\n",
    "    # select the lengths for each row where values is True\n",
    "    # (i.e. where the run is a hit)\n",
    "    hit_lengths = [lengths[values] for values, lengths in zip(values_arrays, lengths_arrays)]\n",
    "    miss_lengths = [lengths[~values] for values, lengths in zip(values_arrays, lengths_arrays)]\n",
    "\n",
    "    all_lengths = hit_lengths + miss_lengths\n",
    "    # remove empty lists\n",
    "    all_lengths = [l for l in all_lengths if len(l) > 0]\n",
    "\n",
    "    max_length = max([max(l) for l in all_lengths])\n",
    "\n",
    "    # concatenate all the lengths into one array for each type\n",
    "    hit_lengths = np.concatenate(hit_lengths)\n",
    "    miss_lengths = np.concatenate(miss_lengths)\n",
    "\n",
    "    # then use matpltolib to plot the histograms of the lengths\n",
    "    # using green for hit lengths and red for miss lengths\n",
    "    # bin equally for both from 0 up to the max length\n",
    "    # in a separate plot for each output_type\n",
    "\n",
    "    if bin_spec == 20:\n",
    "        bins = np.arange(0, 20+1, 1)\n",
    "    elif bin_spec == 100:\n",
    "        bins = np.arange(0, 100+1, 5)\n",
    "    elif bin_spec == 200:\n",
    "        bins = np.arange(0, 200+1, 10)\n",
    "    elif bin_spec == \"max\":\n",
    "        bins = np.arange(0, max_length+1,10)\n",
    "\n",
    "    # make subplot for each histogram\n",
    "    # get subplot for each group idx\n",
    "    plt.subplot(1, len(group_list), idx+1)\n",
    "    ax = axs[idx]\n",
    "\n",
    "    # plt.hist(hit_lengths, bins=bins, color='green', alpha=0.5, label='green run')\n",
    "    # plt.hist(miss_lengths, bins=bins, color='red', alpha=0.5, label='red run')\n",
    "    # plt.hist(hit_lengths, bins=bins, color='green', alpha=0.5, label='green run', density=True)\n",
    "    # plt.hist(miss_lengths, bins=bins, color='red', alpha=0.5, label='red run', density=True)\n",
    "\n",
    "    # add density histogram to this subplot\n",
    "    ax.hist(hit_lengths, bins=bins, color='green', alpha=0.5, label='green runs', density=show_density)\n",
    "    ax.hist(miss_lengths, bins=bins, color='red', alpha=0.5, label='red runs', density=show_density)\n",
    "\n",
    "    # make xticks the bin values and center bars on the ticks\n",
    "    # for this subplot\n",
    "    ax.set_xticks(bins)\n",
    "\n",
    "    if show_density:\n",
    "    # show yticks from 0 to 1\n",
    "        plt.yticks(np.arange(0, 1.1, 0.1))\n",
    "\n",
    "    # remove extra space before first and after last bin\n",
    "    plt.xlim(bins[0], bins[-1])\n",
    "\n",
    "    # rotate xticks\n",
    "    plt.xticks(rotation=45)\n",
    "\n",
    "    if sweep_field == \"gamma\":\n",
    "        plt.title(f\"{group[-1]}\")\n",
    "    elif sweep_field == \"delta\":\n",
    "        plt.title(f\"{group[-2]}\")\n",
    "    else:\n",
    "        pass\n",
    "    plt.xlabel(\"Run length\")\n",
    "\n",
    "# add a title over all subplots\n",
    "output_type = output_type.replace(\"w_bl_hit_list\", \"Watermarked\").replace(\"no_bl_hit_list\", \"No Watermark\").replace(\"baseline_hit_list\", \"Human\")\n",
    "fig.suptitle(f\"{output_type} Run Length Distributions\")\n",
    "# add spacing below suptitle\n",
    "fig.subplots_adjust(top=0.85)\n",
    "\n",
    "plt.subplot(1, len(group_list), 1)\n",
    "plt.legend(loc='upper right')\n",
    "if show_density:\n",
    "    # show yticks from 0 to 1\n",
    "    plt.yticks(np.arange(0, 1.1, 0.1))\n",
    "    plt.ylabel(\"Density\")\n",
    "else:\n",
    "    plt.ylabel(\"Frequency\")\n",
    "plt.show()\n",
    "\n",
    "# write figure to file\n",
    "if save_fig:\n",
    "    fname = f\"{OUTPUT_DIR}/figures/histogram_{output_type}-outputs_sweep-{sweep_field}_show-{'density' if show_density else 'frequency'}_{bin_spec}-bins_{f'_single-idx-{single_output_idx}' if single_output_idx else ''}\"\n",
    "    fname += \".png\"\n",
    "    # fname += \".pdf\"\n",
    "    \n",
    "    fig.savefig(fname,\n",
    "                bbox_inches='tight', \n",
    "                # format='pdf', \n",
    "                dpi=1000)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Old Code From Prev Paper"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Z vs T (figure 3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.clf()\n",
    "plt.figure(constrained_layout=True)\n",
    "plt.figure(figsize=(5, 4))\n",
    "\n",
    "# save_fig = True\n",
    "save_fig = False\n",
    "\n",
    "z_scores = True\n",
    "# z_scores = False\n",
    "\n",
    "beam_search = None\n",
    "# beam_search = 1\n",
    "# beam_search = 4\n",
    "# beam_search = 8\n",
    "\n",
    "ablate = \"delta\"\n",
    "delta_gammas = [\n",
    "    # (0.5,0.25),\n",
    "    # (1.0,0.25),\n",
    "    # (2.0,0.25),\n",
    "    # (5.0,0.25),\n",
    "    # (10.0,0.25),\n",
    "    (0.5,0.5),\n",
    "    (1.0,0.5),\n",
    "    (2.0,0.5),\n",
    "    (5.0,0.5),\n",
    "    (10.0,0.5),\n",
    "]\n",
    "# ablate = \"gamma\"\n",
    "# delta_gammas = [\n",
    "#     # (5.0,0.9),\n",
    "#     # (5.0,0.75),\n",
    "#     # (5.0,0.5),\n",
    "#     # (5.0,0.25),\n",
    "#     # (5.0,0.1),\n",
    "#     (2.0,0.9),\n",
    "#     (2.0,0.75),\n",
    "#     (2.0,0.5),\n",
    "#     (2.0,0.25),\n",
    "#     (2.0,0.1),\n",
    "# ]\n",
    "# if not z_scores: delta_gammas = delta_gammas[::-1]\n",
    "\n",
    "groups = []\n",
    "names = []\n",
    "\n",
    "for d,g in delta_gammas:\n",
    "        if beam_search:\n",
    "            groups.append((False, beam_search, d, g))\n",
    "        else:\n",
    "            groups.append((True, 1, d, g))\n",
    "        names.append(f\"$\\delta:{d},\\gamma:{g}$\")\n",
    "\n",
    "groups=groups[::-1]\n",
    "names=names[::-1]\n",
    "\n",
    "\n",
    "axis_max_t = 200\n",
    "\n",
    "max_t = None\n",
    "# max_t = 200\n",
    "# max_t = 100\n",
    "# max_t = 50\n",
    "\n",
    "# Make colormap\n",
    "import matplotlib.pyplot as plt\n",
    "viridis = plt.colormaps['viridis'].resampled(len(groups)+1) \n",
    "cmap = viridis.colors[:len(groups)][::-1]\n",
    "\n",
    "for grp_idx,(group, name) in enumerate(zip(groups, names)):\n",
    "\n",
    "    delta, gamma = group[-2],group[-1]\n",
    "\n",
    "    # this is the series of bools corresponding to token at T being in whitelist\n",
    "    w_bl_hit_list = grouped_df.get_group(group)[\"w_bl_hit_list\"].to_list()\n",
    "\n",
    "    lengths = [len(l) for l in w_bl_hit_list]\n",
    "    diff_lengths = set(lengths) \n",
    "    counter = {}\n",
    "    for l in lengths:\n",
    "        if counter.get(l):\n",
    "            counter[l] += 1\n",
    "        else:\n",
    "            counter[l] = 1\n",
    "    if max_t:\n",
    "        min_length = min(min(diff_lengths),max_t)\n",
    "        max_t = min_length\n",
    "    else:\n",
    "        min_length = min(diff_lengths)\n",
    "    w_bl_hit_list = [l[:min_length] for l in w_bl_hit_list]\n",
    "\n",
    "    # wl_hit_matrix = ~np.matrix(w_bl_hit_list)\n",
    "    wl_hit_matrix = (~torch.tensor(w_bl_hit_list, dtype=bool)).to(torch.float)\n",
    "    # wl_hit_matrix\n",
    "\n",
    "    n = wl_hit_matrix.shape[0]\n",
    "\n",
    "    if max_t:\n",
    "        t_values = torch.arange(0,max_t)\n",
    "        indices = torch.arange(0,max_t)\n",
    "    else:\n",
    "        t_values = torch.arange(0,wl_hit_matrix.shape[1])\n",
    "        indices = torch.arange(0,wl_hit_matrix.shape[1])\n",
    "    # print(t_values[:10])\n",
    "\n",
    "    avg_cumulative = list()\n",
    "    std_cumulative = list()\n",
    "    prc_25_cumulative = list()\n",
    "    prc_50_cumulative = list()\n",
    "    prc_75_cumulative = list()\n",
    "\n",
    "    prc_25_seq_indices = list()\n",
    "\n",
    "    for idx in indices:\n",
    "\n",
    "        hits_upto_t = wl_hit_matrix[:,:idx+1]\n",
    "        cumulative_sum_to_t = hits_upto_t.sum(axis=1)\n",
    "        wl_frac_at_t = cumulative_sum_to_t/(t_values[idx]+1)\n",
    "        \n",
    "        if z_scores:\n",
    "            wl_z_score_at_t = compute_z_score(wl_frac_at_t, t_values[idx], gamma)\n",
    "            avg_at_t = torch.mean(wl_z_score_at_t,axis=0)\n",
    "            std_at_t = torch.std(wl_z_score_at_t,axis=0)\n",
    "            prc_25_at_t = torch.quantile(wl_z_score_at_t,q=0.25,axis=0)\n",
    "            prc_50_at_t = torch.quantile(wl_z_score_at_t,q=0.50,axis=0)\n",
    "            prc_75_at_t = torch.quantile(wl_z_score_at_t,q=0.75,axis=0)\n",
    "\n",
    "            if gamma == 0.9: # and idx > 20 and idx < 90:\n",
    "                pcen=np.quantile(wl_z_score_at_t,0.75,interpolation='nearest')\n",
    "                i_near=abs(wl_z_score_at_t-pcen).argmin()\n",
    "                # prc_25_seq_indices.append((i_near.item(),pcen))\n",
    "                prc_25_seq_indices.append((i_near.item()))\n",
    "        else:\n",
    "            avg_at_t = torch.mean(wl_frac_at_t,axis=0)\n",
    "            std_at_t = torch.std(wl_frac_at_t,axis=0)\n",
    "            prc_25_at_t = torch.quantile(wl_frac_at_t,q=0.25,axis=0)\n",
    "            prc_50_at_t = torch.quantile(wl_frac_at_t,q=0.50,axis=0)\n",
    "            prc_75_at_t = torch.quantile(wl_frac_at_t,q=0.75,axis=0)\n",
    "\n",
    "        avg_cumulative.append(avg_at_t.item())\n",
    "        std_cumulative.append(std_at_t.item())\n",
    "        prc_25_cumulative.append(prc_25_at_t.item())\n",
    "        prc_50_cumulative.append(prc_50_at_t.item())\n",
    "        prc_75_cumulative.append(prc_75_at_t.item())\n",
    "\n",
    "\n",
    "    print(prc_25_seq_indices)\n",
    "\n",
    "    avg_cumulative = np.array(avg_cumulative)\n",
    "    std_cumulative = np.array(std_cumulative)\n",
    "    std_err_cumulative = std_cumulative/np.sqrt(n)\n",
    "    var_cumulative = std_cumulative**2\n",
    "    \n",
    "    plt.plot(t_values, avg_cumulative, color=cmap[grp_idx],  label=name)\n",
    "\n",
    "    # bounds stuff\n",
    "\n",
    "    # plt.plot(t_values, prc_25_cumulative, color=cmap[grp_idx], linestyle=\"dashed\") #, label=name+',25th') \n",
    "    # # plt.plot(t_values, prc_50_cumulative, color=cmap[grp_idx], linestyle='--', label=name+',50th') \n",
    "    # plt.plot(t_values, prc_75_cumulative, color=cmap[grp_idx], linestyle=\"dashed\") #, label=name+',75th ') \n",
    "    # #fill between the upper and lower bands\n",
    "    # plt.fill_between(t_values, prc_25_cumulative, prc_75_cumulative, alpha = .1,color = cmap[grp_idx])\n",
    "    # or just lower\n",
    "    # plt.fill_between(t_values, prc_25_cumulative, avg_cumulative, alpha = .1,color = cmap[grp_idx])\n",
    "\n",
    "    # plt.plot(t_values, avg_cumulative-std_cumulative, color=cmap[grp_idx], linestyle=\"dashed\") #, label=name+',25th') \n",
    "    # plt.plot(t_values, avg_cumulative+std_cumulative, color=cmap[grp_idx], linestyle=\"dashed\") #, label=name+',25th') \n",
    "    # plt.plot(t_values, avg_cumulative-std_err_cumulative, color=cmap[grp_idx], linestyle=\"dashed\") #, label=name+',25th') \n",
    "    # plt.plot(t_values, avg_cumulative+std_err_cumulative, color=cmap[grp_idx], linestyle=\"dashed\") #, label=name+',25th') \n",
    "    # plt.plot(t_values, avg_cumulative-var_cumulative, color=cmap[grp_idx], linestyle=\"dashed\") #, label=name+',25th') \n",
    "    # plt.plot(t_values, avg_cumulative+var_cumulative, color=cmap[grp_idx], linestyle=\"dashed\") #, label=name+',25th') \n",
    "    # fill between the upper and lower bands\n",
    "    # plt.fill_between(t_values, avg_cumulative-std_cumulative, avg_cumulative+std_cumulative, alpha = .1,color = cmap[grp_idx])\n",
    "    # plt.fill_between(t_values, avg_cumulative-std_err_cumulative, avg_cumulative+std_err_cumulative, alpha = .1,color = cmap[grp_idx])\n",
    "    # or just lower\n",
    "    # plt.fill_between(t_values, avg_cumulative-std_cumulative, avg_cumulative, alpha = .1,color = cmap[grp_idx])\n",
    "    # plt.fill_between(t_values, avg_cumulative-std_err_cumulative, avg_cumulative, alpha = .1,color = cmap[grp_idx])\n",
    "\n",
    "# plt.plot([0.0],[0.0],label=f'25th Percentile', linestyle=\"dashed\", color=\"gray\")\n",
    "\n",
    "# if beam_search:\n",
    "#     plt.title(f\"Greedy, {beam_search}-way BS\")\n",
    "\n",
    "legend_font = 11\n",
    "\n",
    "# zoom_midrange = True\n",
    "# zoom = True\n",
    "\n",
    "zoom = False\n",
    "\n",
    "if zoom:\n",
    "    if z_scores:\n",
    "        plt.legend(loc = 'upper left', fontsize=legend_font)\n",
    "    else:\n",
    "        plt.legend(loc = 'lower right', fontsize=legend_font)\n",
    "    if zoom_midrange:\n",
    "        plt.xlim([(min_length)/4, (3*(max_t if max_t else min_length)/4)+1])\n",
    "    else:\n",
    "        plt.xlim([0, ((max_t if max_t else min_length)/4)+1])\n",
    "    plot_name = f\"z_vs_t_zoom_ablate_{ablate}\" if z_scores else f\"wl_vs_t_zoom_ablate_{ablate}\"\n",
    "else:\n",
    "    if z_scores:\n",
    "        plt.legend(loc = 'upper left', fontsize=legend_font)\n",
    "    else:\n",
    "        plt.legend(loc = 'lower right', fontsize=legend_font)\n",
    "  \n",
    "    plt.xlim([0, ((max_t if max_t else min_length))+1])\n",
    "\n",
    "    plot_name = f\"z_vs_t_ablate_{ablate}\" if z_scores else f\"wl_vs_t_ablate_{ablate}\"\n",
    "\n",
    "axes_label_fonts = 14\n",
    "if z_scores:\n",
    "    plt.ylabel('z-score',fontsize=axes_label_fonts)\n",
    "else:\n",
    "    plt.ylabel('Whitelist Fraction',fontsize=axes_label_fonts)\n",
    "plt.xlabel('T',fontsize=axes_label_fonts)\n",
    "\n",
    "# import matplotlib.ticker as ticker\n",
    "# tick_spacing = 5.0\n",
    "# plt.gca().yaxis.set_major_locator(ticker.MultipleLocator(tick_spacing))\n",
    "\n",
    "axes_tick_font = 13\n",
    "plt.xticks(fontsize=axes_tick_font)\n",
    "plt.yticks(fontsize=axes_tick_font)\n",
    "\n",
    "plt.grid()\n",
    "plt.tight_layout()\n",
    "\n",
    "if beam_search:\n",
    "    if ablate == \"gamma\":\n",
    "        plot_name = f\"greedy_{beam_search}_beams_delta_{delta}\" \n",
    "    if ablate == \"delta\":\n",
    "        plot_name = f\"greedy_{beam_search}_beams_gamma_{gamma}\" \n",
    "\n",
    "# plot_name = \"z_vs_t_ablate_gamma_boosted_delta\"\n",
    "# plot_name = \"z_vs_t_ablate_delta_boosted_gamma\"\n",
    "\n",
    "print(plot_name)\n",
    "\n",
    "\n",
    "if save_fig:\n",
    "    # fname = f\"figs/{plot_name}.pdf\"\n",
    "    fname = f\"figs_new/{plot_name}.pdf\"\n",
    "    plt.savefig(fname, format=\"pdf\")\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Set up data for charts (setup for figures 2&7)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "viz_df = pd.DataFrame()\n",
    "\n",
    "# aggregating\n",
    "\n",
    "# set the hparam keys, including an indiv column for each you want to ablate on\n",
    "viz_df[\"bl_hparams\"] = grouped_df[\"w_bl_exp_whitelist_fraction\"].describe().index.to_list()\n",
    "for i,key in enumerate(groupby_fields):\n",
    "    viz_df[key] = viz_df[\"bl_hparams\"].apply(lambda tup: tup[i])\n",
    "\n",
    "# viz_df[\"delta\"] = viz_df[\"bl_logit_bias\"].values\n",
    "viz_df[\"gamma\"] = viz_df[\"gamma\"].values\n",
    "# viz_df[\"gamma\"] = np.ones_like(viz_df[\"bl_proportion\"].values) - viz_df[\"bl_proportion\"].values\n",
    "\n",
    "# aggregate each field of interest for each hparam setting (group)\n",
    "describe_dict = grouped_df[\"w_bl_exp_whitelist_fraction\"].describe()\n",
    "viz_df[\"w_bl_exp_whitelist_fraction_mean\"] = describe_dict[\"mean\"].to_list()\n",
    "viz_df[\"w_bl_exp_whitelist_fraction_std\"] = describe_dict[\"std\"].to_list()\n",
    "\n",
    "describe_dict = grouped_df[\"w_bl_var_whitelist_fraction\"].describe()\n",
    "viz_df[\"w_bl_var_whitelist_fraction_mean\"] = describe_dict[\"mean\"].to_list()\n",
    "viz_df[\"w_bl_var_whitelist_fraction_std\"] = describe_dict[\"std\"].to_list()\n",
    "\n",
    "describe_dict = grouped_df[\"w_bl_whitelist_fraction\"].describe()\n",
    "viz_df[\"w_bl_whitelist_fraction_min\"] = describe_dict[\"min\"].to_list()\n",
    "viz_df[\"w_bl_whitelist_fraction_25\"] = describe_dict[\"25%\"].to_list()\n",
    "viz_df[\"w_bl_whitelist_fraction_50\"] = describe_dict[\"50%\"].to_list()\n",
    "viz_df[\"w_bl_whitelist_fraction_75\"] = describe_dict[\"75%\"].to_list()\n",
    "viz_df[\"w_bl_whitelist_fraction_max\"] = describe_dict[\"max\"].to_list()\n",
    "viz_df[\"w_bl_whitelist_fraction_mean\"] = describe_dict[\"mean\"].to_list()\n",
    "viz_df[\"w_bl_whitelist_fraction_std\"] = describe_dict[\"std\"].to_list()\n",
    "\n",
    "describe_dict = grouped_df[\"no_bl_whitelist_fraction\"].describe()\n",
    "viz_df[\"no_bl_whitelist_fraction_mean\"] = describe_dict[\"mean\"].to_list()\n",
    "viz_df[\"no_bl_whitelist_fraction_std\"] = describe_dict[\"std\"].to_list()\n",
    "\n",
    "\n",
    "describe_dict = grouped_df[\"w_bl_z_score\"].describe()\n",
    "viz_df[\"w_bl_z_score_mean\"] = describe_dict[\"mean\"].to_list()\n",
    "viz_df[\"w_bl_z_score_std\"] = describe_dict[\"std\"].to_list()\n",
    "\n",
    "describe_dict = grouped_df[\"no_bl_z_score\"].describe()\n",
    "viz_df[\"no_bl_z_score_mean\"] = describe_dict[\"mean\"].to_list()\n",
    "viz_df[\"no_bl_z_score_std\"] = describe_dict[\"std\"].to_list()\n",
    "\n",
    "describe_dict = grouped_df[\"baseline_z_score\"].describe()\n",
    "viz_df[\"baseline_z_score_mean\"] = describe_dict[\"mean\"].to_list()\n",
    "viz_df[\"baseline_z_score_std\"] = describe_dict[\"std\"].to_list()\n",
    "\n",
    "\n",
    "describe_dict = grouped_df[\"w_bl_ppl\"].describe()\n",
    "viz_df[\"w_bl_ppl_mean\"] = describe_dict[\"mean\"].to_list()\n",
    "viz_df[\"w_bl_ppl_std\"] = describe_dict[\"std\"].to_list()\n",
    "\n",
    "describe_dict = grouped_df[\"no_bl_ppl\"].describe()\n",
    "viz_df[\"no_bl_ppl_mean\"] = describe_dict[\"mean\"].to_list()\n",
    "viz_df[\"no_bl_ppl_std\"] = describe_dict[\"std\"].to_list()\n",
    "\n",
    "describe_dict = grouped_df[\"baseline_ppl\"].describe()\n",
    "viz_df[\"baseline_ppl_mean\"] = describe_dict[\"mean\"].to_list()\n",
    "viz_df[\"baseline_ppl_std\"] = describe_dict[\"std\"].to_list()\n",
    "\n",
    "describe_dict = grouped_df[\"avg_spike_entropy\"].describe()\n",
    "viz_df[\"avg_spike_entropy_mean\"] = describe_dict[\"mean\"].to_list()\n",
    "viz_df[\"avg_spike_entropy_std\"] = describe_dict[\"std\"].to_list()\n",
    "\n",
    "print(f\"groupby legend: {groupby_fields}\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# filtering\n",
    "\n",
    "# viz_df = viz_df[viz_df[\"bl_hparams\"].apply(lambda tup: (tup[0] == True))] # sampling\n",
    "# \n",
    "viz_df = viz_df[viz_df[\"bl_hparams\"].apply(lambda tup: (tup[0] == False))] # greedy\n",
    "\n",
    "\n",
    "# fix one of the bl params for analytic chart\n",
    "# viz_df = viz_df[(viz_df[\"gamma\"]==0.9) & (viz_df[\"delta\"]<=10.0)]\n",
    "# viz_df = viz_df[(viz_df[\"gamma\"]==0.75) & (viz_df[\"delta\"]<=10.0)]\n",
    "# viz_df = viz_df[(viz_df[\"gamma\"]==0.5) & (viz_df[\"delta\"]<=10.0)]\n",
    "# viz_df = viz_df[(viz_df[\"gamma\"]==0.25) & (viz_df[\"delta\"]<=10.0)]\n",
    "# viz_df = viz_df[(viz_df[\"gamma\"]==0.1) & (viz_df[\"delta\"]<=10.0)]\n",
    "\n",
    "# for the sample pareto chart\n",
    "# viz_df = viz_df[(viz_df[\"delta\"] > 0.5) & (viz_df[\"delta\"]<=10.0)]\n",
    "# viz_df = viz_df[(viz_df[\"delta\"]<=2.0)] # zoom in on lower deltas\n",
    "# viz_df = viz_df[(viz_df[\"delta\"] >= 2.0) & (viz_df[\"delta\"]<=10.0)] # mid deltas\n",
    "# viz_df = viz_df[(viz_df[\"gamma\"] != 0.25) & (viz_df[\"gamma\"] != 0.75) & (viz_df[\"delta\"]<=2.0)]\n",
    "# viz_df = viz_df[(viz_df[\"gamma\"] != 0.1) & (viz_df[\"gamma\"] != 0.9) & (viz_df[\"delta\"]<=2.0)]\n",
    "\n",
    "# viz_df = viz_df[(viz_df[\"delta\"]==0.5) | (viz_df[\"delta\"]==2.0) | (viz_df[\"delta\"]==10.0)]\n",
    "\n",
    "# viz_df = viz_df[(viz_df[\"delta\"]!=0.1)&(viz_df[\"delta\"]!=0.5)&(viz_df[\"delta\"]!=50.0)]\n",
    "\n",
    "# for the beams pareto\n",
    "# viz_df = viz_df[(viz_df[\"delta\"]!=50.0)]\n",
    "# viz_df = viz_df[(viz_df[\"delta\"]!=50.0) & (viz_df[\"num_beams\"]!=1)]\n",
    "\n",
    "print(len(viz_df))\n",
    "\n",
    "viz_df"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# grouped_df[\"avg_spike_entropy\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# viz_df[[\"gamma\",\"avg_spike_entropy_mean\"]]"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Basic Exp vs Empirical WL fraction chart (figure 7)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "# plt.style.use(\"classic\")\n",
    "plt.style.use(\"default\")\n",
    "# plt.style.use('ggplot') \n",
    "# plt.style.use('seaborn')\n",
    "\n",
    "rc('font', **{'family': 'serif', 'serif': ['Computer Modern']})\n",
    "rc('text', usetex=True)\n",
    "\n",
    "\n",
    "plt.clf()\n",
    "# plt.figure(figsize=(16, 4))\n",
    "# plt.figure(figsize=(8, 4))\n",
    "plt.figure(constrained_layout=True)\n",
    "plt.figure(figsize=(5, 4))\n",
    "\n",
    "\n",
    "# x_col = 'bl_hparams'\n",
    "# a = viz_df[x_col].apply(str)\n",
    "\n",
    "# x_col = 'bl_logit_bias'\n",
    "# x_col = 'bl_proportion'\n",
    "x_col = \"delta\"\n",
    "# x_col = \"gamma\"\n",
    "\n",
    "a = viz_df[x_col]\n",
    "print(f\"Num configurations: {len(a)}\")\n",
    "\n",
    "y_col = 'w_bl_whitelist_fraction_mean'\n",
    "y_col_err = 'w_bl_whitelist_fraction_std'\n",
    "\n",
    "viridis = plt.colormaps['viridis'].resampled(4)\n",
    "# cmap = viridis.colors[::-1]\n",
    "cmap = viridis.colors\n",
    "\n",
    "plt.plot(a, viz_df[\"w_bl_whitelist_fraction_mean\"].values, color=cmap[1], marker='o', label='Mean') \n",
    "plt.plot(a, viz_df[\"w_bl_whitelist_fraction_25\"].values, color=cmap[1], linestyle='-.', label='25th Percentile') \n",
    "plt.plot(a, viz_df[\"w_bl_whitelist_fraction_75\"].values, color=cmap[1], linestyle='-.', label='75th Percentile') \n",
    "# plt.plot(a, viz_df[\"w_bl_whitelist_fraction_min\"].values, color=cmap[1], linestyle='-.', label='min') \n",
    "# plt.plot(a, viz_df[\"w_bl_whitelist_fraction_max\"].values, color=cmap[1], linestyle='-.', label='max') \n",
    "\n",
    "#fill between the upper and lower bands\n",
    "plt.fill_between(a, viz_df[\"w_bl_whitelist_fraction_25\"], viz_df[\"w_bl_whitelist_fraction_75\"], alpha = .1,color = cmap[1])\n",
    "# plt.fill_between(a, viz_df[\"w_bl_whitelist_fraction_25\"], viz_df[\"w_bl_whitelist_fraction_75\"], alpha = .1,color = 'darkorchid')\n",
    "# plt.fill_between(a, y1_low, y1_high, alpha = .1,color = 'goldenrod')\n",
    "\n",
    "\n",
    "y_col = 'w_bl_exp_whitelist_fraction_mean'\n",
    "# y_col_err = 'w_bl_var_whitelist_fraction_mean'\n",
    "# d = viz_df[x_col].apply(str)\n",
    "\n",
    "# sub_df = viz_df[viz_df[\"num_beams\"]==1]\n",
    "\n",
    "a = viz_df[x_col]\n",
    "e = viz_df[y_col].values\n",
    "# plt.plot(a, e, label=\"Predicted Lower Bound\", color=cmap[-1])\n",
    "plt.plot(a, e, label=\"Analytic Bound\", color=\"r\")\n",
    "# f = viz_df[y_col_err].values\n",
    "# # f = np.sqrt(viz_df[y_col_err].values)\n",
    "# plt.errorbar(d, e, yerr=f, fmt=\"o\")\n",
    "\n",
    "plt.legend(loc=\"lower right\",frameon=True, facecolor=\"white\")\n",
    "\n",
    "# for logit bias x axis\n",
    "# log_axis = True\n",
    "log_axis = False\n",
    "if log_axis:\n",
    "    plt.xscale(\"log\")\n",
    "\n",
    "ax = plt.gca()\n",
    "plt.draw()\n",
    "\n",
    "\n",
    "\n",
    "plt.xlabel(f\"Green List Bias, $\\delta$\")\n",
    "# plt.xlabel(f\"Whitelist size := $\\gamma$\")\n",
    "\n",
    "plt.ylabel(\"Fraction in Green List\")\n",
    "\n",
    "\n",
    "plt.grid()\n",
    "\n",
    "plt.tight_layout()\n",
    "\n",
    "if log_axis:\n",
    "    plot_name = \"analytic_w_sampling_log.pdf\"\n",
    "else:\n",
    "    plot_name = \"analytic_w_sampling_linear.pdf\"\n",
    "    # plot_name = f\"analytic_w_sampling_linear_gamma_{viz_df['gamma'].values[0]}.pdf\"\n",
    "\n",
    "# plot_name = \"analytic_w_sampling_linear_greenlist.pdf\"\n",
    "print(plot_name)\n",
    "\n",
    "# fname = f\"figs/{plot_name}\"\n",
    "# plt.savefig(fname, format=\"pdf\")\n",
    "plt.show()\n"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# delta gamma sampling pareto plot (figure 2 left)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "rc('font', **{'family': 'serif', 'serif': ['Computer Modern']})\n",
    "rc('text', usetex=True)\n",
    "\n",
    "plt.clf()\n",
    "plt.figure(constrained_layout=True)\n",
    "plt.figure(figsize=(5, 4))\n",
    "\n",
    "\n",
    "x_col = 'w_bl_ppl_mean'\n",
    "y_col = 'w_bl_z_score_mean'\n",
    "\n",
    "# markers = [\"x\", \"p\", \"*\", \"P\"]\n",
    "\n",
    "deltas = sorted(np.unique(viz_df[\"delta\"].values))\n",
    "gammas = sorted(np.unique(viz_df[\"gamma\"].values), reverse=True)\n",
    "print(deltas, gammas)\n",
    "gamma_labels = [(g if g > 0.1 else 0.1) for g in gammas]\n",
    "\n",
    "markers = [\"x\", \"p\", \"*\", \"P\"][:len(deltas)]\n",
    "\n",
    "num_colors = len(gammas)\n",
    "cmap = cmr.get_sub_cmap('viridis', 0.0, 0.66, N=num_colors)\n",
    "# cmap = cmr.get_sub_cmap('plasma', 0.0, 0.66, N=num_colors)\n",
    "colors = cmap.colors#[::-1]\n",
    "\n",
    "\n",
    "for i,delta in enumerate(deltas):\n",
    "    for j,gamma in enumerate(gammas):\n",
    "        sub_df = viz_df[(viz_df[\"delta\"] == delta) & (viz_df[\"gamma\"] == gamma)]\n",
    "        a = sub_df[x_col].values\n",
    "        b = sub_df[y_col].values\n",
    "        # plt.scatter(a, b, label=f\"$\\delta={delta},\\gamma={gamma}$\", color=colors[j], marker=markers[i])\n",
    "        plt.plot(a, b, label=f\"$\\delta={delta},\\gamma={gamma}$\", color=colors[j], marker=markers[i])\n",
    "\n",
    "\n",
    "x_col = 'no_bl_ppl_mean'\n",
    "y_col = 'no_bl_z_score_mean'\n",
    "# x_col = 'baseline_ppl_mean'\n",
    "# y_col = 'baseline_z_score_mean'\n",
    "\n",
    "\n",
    "for i,delta in enumerate(deltas):\n",
    "    for j,gamma in enumerate(gammas):\n",
    "        sub_df = viz_df[(viz_df[\"delta\"] == delta) & (viz_df[\"gamma\"] == gamma)]\n",
    "        a = sub_df[x_col].values\n",
    "        b = sub_df[y_col].values\n",
    "        plt.scatter(a, b, label=f\"$\\delta={delta},\\gamma={gamma}$\", color=colors[j])\n",
    "\n",
    "# # # for manual legend\n",
    "plt.scatter([-1],[-1], label=\"Vanilla\", color=\"gray\", marker=\"o\")\n",
    "\n",
    "ax = plt.gca()\n",
    "\n",
    "from matplotlib.cm import ScalarMappable\n",
    "from matplotlib.colors import Normalize, NoNorm, ListedColormap\n",
    "cmap = ListedColormap(colors)\n",
    "cmappable = ScalarMappable(norm=NoNorm(),cmap=cmap)\n",
    "cbar = plt.colorbar(cmappable,ticks=[i for i in range(len(gammas))],shrink=0.6, pad = 0.03)\n",
    "cbar.ax.set_yticklabels(gamma_labels) \n",
    "cbar.set_label('$\\gamma$', rotation=0)\n",
    "\n",
    "\n",
    "all_x = np.concatenate([viz_df['w_bl_ppl_mean'].values,viz_df['no_bl_ppl_mean'].values])\n",
    "all_y = np.concatenate([viz_df['w_bl_z_score_mean'].values,viz_df['no_bl_z_score_mean'].values])\n",
    "# all_x = np.concatenate([viz_df['w_bl_ppl_mean'].values,viz_df['baseline_ppl_mean'].values])\n",
    "# all_y = np.concatenate([viz_df['w_bl_z_score_mean'].values,viz_df['baseline_z_score_mean'].values])\n",
    "\n",
    "min_x, max_x = np.min(all_x), np.max(all_x)\n",
    "min_y, max_y = np.min(all_y), np.max(all_y)\n",
    "\n",
    "# x_min_tick = 1.0\n",
    "x_min_tick = 3.0\n",
    "x_max_tick = np.ceil([max_x])[0]+1.0\n",
    "y_min_tick = 0.0\n",
    "y_max_tick = np.ceil([max_y])[0]+1.0\n",
    "\n",
    "x_ticks = np.arange(x_min_tick,x_max_tick,1.0)\n",
    "y_ticks = np.arange(y_min_tick,y_max_tick,5.0)\n",
    "\n",
    "\n",
    "x_lim_min = 3.0\n",
    "x_lim_max = x_max_tick\n",
    "y_lim_min = 0.45\n",
    "# y_lim_max = 1.09\n",
    "y_lim_max = 1.005\n",
    "\n",
    "\n",
    "# plt.xlim((x_min_tick-0.5,x_max_tick))\n",
    "plt.xlim((x_lim_min,x_lim_max))\n",
    "# plt.xlim((4.0,8.0))\n",
    "# plt.ylim((-1.0,20.0))\n",
    "# plt.ylim((y_lim_min,y_lim_max))\n",
    "\n",
    "ax.set_xticks(x_ticks)\n",
    "# ax.set_yticks(y_ticks)\n",
    "\n",
    "ax.invert_xaxis()\n",
    "\n",
    "# # manual legend for dual parameter visualization\n",
    "f = lambda m,c: plt.plot([],[],marker=m, color=c, ls=\"none\")[0]\n",
    "handles = [f(markers[::-1][i], \"gray\") for i in range(len(deltas))]\n",
    "handles += [f(\"o\", \"gray\")]\n",
    "labels = [f\"$\\delta={delta}$\" for delta in deltas[::-1]]+[f\"$\\delta=0.0$\"]\n",
    "plt.legend(handles, labels, loc=\"upper right\", framealpha=1)\n",
    "\n",
    "plt.grid()\n",
    "\n",
    "plt.xlabel(\"Oracle Model PPL (better →)\")\n",
    "plt.ylabel(\"z-score (better →)\")\n",
    "\n",
    "\n",
    "plt.tight_layout()\n",
    "\n",
    "# plot_name = \"pareto_sampling_no_beams\"\n",
    "# fname = f\"figs/{plot_name}.pdf\"\n",
    "# plt.savefig(fname, format=\"pdf\")\n",
    "plt.show()"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# beams pareto plot (figure 2 right)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "num_colors = 3\n",
    "cmap = cmr.get_sub_cmap('viridis', 0.0, 0.66, N=num_colors)\n",
    "colors = cmap.colors#[::-1]\n",
    "\n",
    "# plt.style.use('ggplot')\n",
    "# plt.style.use('seaborn')\n",
    "\n",
    "rc('font', **{'family': 'serif', 'serif': ['Computer Modern']})\n",
    "rc('text', usetex=True)\n",
    "\n",
    "plt.clf()\n",
    "plt.figure(constrained_layout=True)\n",
    "plt.figure(figsize=(5, 4))\n",
    "\n",
    "\n",
    "x_col = 'w_bl_ppl_mean'\n",
    "y_col = 'w_bl_z_score_mean'\n",
    "\n",
    "markers = [\"s\",\"D\", \"x\", \"p\",  \"*\", \"P\"] # <--- seems to match other pareto fig ordering\n",
    "\n",
    "deltas = sorted(np.unique(viz_df[\"delta\"].values))\n",
    "num_beams = sorted(np.unique(viz_df[\"num_beams\"].values))\n",
    "# gamma_labels = [(g if g > 0.1 else 0.1) for g in np.unique(viz_df[\"gamma\"].values)]\n",
    "\n",
    "for i,n_beams in enumerate(num_beams):\n",
    "    for j,delta in enumerate(deltas):\n",
    "        sub_df = viz_df[(viz_df[\"delta\"] == delta) & (viz_df[\"num_beams\"] == n_beams)]\n",
    "        a = sub_df[x_col].values\n",
    "        b = sub_df[y_col].values\n",
    "        # plt.scatter(a, b, label=f\"$\\delta={delta},\\gamma={gamma}$\", color=colors[j], marker=markers[i])\n",
    "        plt.plot(a, b, label=f\"$\\delta={delta}$\", color=colors[i], marker=markers[j])\n",
    "\n",
    "\n",
    "x_col = 'no_bl_ppl_mean'\n",
    "y_col = 'no_bl_z_score_mean'\n",
    "\n",
    "\n",
    "\n",
    "for i,n_beams in enumerate(num_beams):\n",
    "    for j,delta in enumerate(deltas):\n",
    "        sub_df = viz_df[(viz_df[\"delta\"] == delta) & (viz_df[\"num_beams\"] == n_beams)]\n",
    "        a = sub_df[x_col].values\n",
    "        b = sub_df[y_col].values\n",
    "        plt.scatter(a, b, label=f\"$\\delta={delta}$\", color=colors[i])\n",
    "\n",
    "# # # for manual legend\n",
    "plt.scatter([-10],[-10], label=\"$\\delta=0$\", color=\"gray\", marker=\"o\")\n",
    "\n",
    "ax = plt.gca()\n",
    "\n",
    "from matplotlib.cm import ScalarMappable\n",
    "from matplotlib.colors import Normalize, NoNorm, ListedColormap\n",
    "cmap = ListedColormap(colors)\n",
    "cmappable = ScalarMappable(norm=NoNorm(),cmap=cmap)\n",
    "cbar = plt.colorbar(cmappable,ticks=[i for i in range(len(num_beams))],shrink=0.6, pad = 0.04)\n",
    "# cbar.set_ticks(num_beams)\n",
    "cbar.set_ticklabels(num_beams)\n",
    "# cbar.ax.set_yticklabels(num_beams) \n",
    "cbar.set_label('Num Beams', rotation=90)\n",
    "\n",
    "\n",
    "all_x = np.concatenate([viz_df['w_bl_ppl_mean'].values,viz_df['no_bl_ppl_mean'].values])\n",
    "all_y = np.concatenate([viz_df['w_bl_z_score_mean'].values,viz_df['no_bl_z_score_mean'].values])\n",
    "\n",
    "min_x, max_x = np.min(all_x), np.max(all_x)\n",
    "min_y, max_y = np.min(all_y), np.max(all_y)\n",
    "\n",
    "# x_max_tick = np.ceil([max_x])[0]+1.0\n",
    "x_max_tick = np.ceil([max_x])[0]\n",
    "y_max_tick = np.ceil([max_y])[0]+1.0\n",
    "\n",
    "\n",
    "plt.xlim((1.0,x_max_tick))\n",
    "plt.ylim((-1.0,y_max_tick))\n",
    "\n",
    "# x_ticks = np.arange(x_min_tick,x_max_tick,1.0)\n",
    "# y_ticks = np.arange(y_min_tick,y_max_tick,5.0)\n",
    "\n",
    "# ax.set_xticks(x_ticks)\n",
    "# ax.set_yticks(y_ticks)\n",
    "\n",
    "ax.invert_xaxis()\n",
    "\n",
    "# # manual legend for dual parameter visualization\n",
    "f = lambda m,c: plt.plot([],[],marker=m, color=c, ls=\"none\")[0]\n",
    "handles = [f(markers[::-1][i], \"gray\") for i in range(len(deltas))]\n",
    "handles += [f(\"o\", \"gray\")]\n",
    "labels = [f\"$\\delta={delta}$\" for delta in deltas[::-1]]+[f\"$\\delta=0.0$\"]\n",
    "plt.legend(handles, labels, loc=\"lower left\", framealpha=1)\n",
    "\n",
    "plt.grid()\n",
    "\n",
    "plt.xlabel(\"Oracle Model PPL (better →)\")\n",
    "plt.ylabel(\"z-score (better →)\")\n",
    "\n",
    "\n",
    "plt.tight_layout()\n",
    "\n",
    "\n",
    "plot_name = \"pareto_greedy_w_beams\"\n",
    "print(plot_name)\n",
    "\n",
    "# fname = f\"figs/{plot_name}.pdf\"\n",
    "# plt.savefig(fname, format=\"pdf\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## z vs entropy (not in paper)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "print(f\"groupby legend: {groupby_fields}\")\n",
    "# hist_subset = grouped_df.get_group((True,1,2.0,0.1)) # needs to match the groupby keys and order\n",
    "# hist_subset = grouped_df.get_group((True,1,2.0,0.25)) \n",
    "hist_subset = grouped_df.get_group((True,1,2.0,0.5)) \n",
    "# hist_subset = grouped_df.get_group((True,1,2.0,0.75)) \n",
    "# hist_subset = grouped_df.get_group((True,1,2.0,0.9)) "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "print(len(hist_subset))\n",
    "# hist_subset = hist_subset[hist_subset[\"w_bl_space_frac\"] <= 0.9]\n",
    "# hist_subset = hist_subset[hist_subset[\"no_bl_space_frac\"] <= 0.9]\n",
    "# print(len(hist_subset))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# y = hist_subset[\"w_bl_z_score\"]\n",
    "# y = hist_subset[\"no_bl_z_score\"]\n",
    "y = hist_subset[\"baseline_z_score\"]\n",
    "\n",
    "x = hist_subset[\"avg_spike_entropy\"]\n",
    "\n",
    "plt.clf()\n",
    "\n",
    "\n",
    "plt.scatter(x, y)\n",
    "\n",
    "\n",
    "plt.grid()\n",
    "\n",
    "plt.xlabel(\"Entropy\")\n",
    "plt.ylabel(\"z-score\")\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "cols_to_tabulate = [\n",
    "    'idx', \n",
    "    'truncated_input', \n",
    "    'baseline_completion',\n",
    "    'no_bl_output', \n",
    "    'w_bl_output', \n",
    "    'avg_spike_entropy',\n",
    "    'no_bl_z_score',\n",
    "    'w_bl_z_score',\n",
    "    'w_bl_whitelist_fraction',\n",
    "    'no_bl_whitelist_fraction',\n",
    "    'baseline_ppl',\n",
    "    'no_bl_ppl',\n",
    "    'w_bl_ppl'\n",
    "]\n",
    "\n",
    "slice_size = 10\n",
    "\n",
    "num_examples = len(hist_subset)\n",
    "midpt = num_examples//5\n",
    "lower = midpt - (slice_size//2)\n",
    "upper = midpt + (slice_size//2)+1\n",
    "\n",
    "high_entropy_examples = hist_subset[cols_to_tabulate].sort_values([\"avg_spike_entropy\"],ascending=True).tail(slice_size)\n",
    "mid_entropy_examples = hist_subset[cols_to_tabulate].sort_values([\"avg_spike_entropy\"],ascending=True).iloc[lower:upper]\n",
    "low_entropy_examples = hist_subset[cols_to_tabulate].sort_values([\"avg_spike_entropy\"],ascending=True).head(slice_size)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# hist_subset[cols_to_tabulate][(hist_subset[\"avg_spike_entropy\"]<0.7)&(hist_subset[\"w_bl_z_score\"]>=14.0)]\n",
    "hist_subset[cols_to_tabulate][(hist_subset[\"avg_spike_entropy\"]<0.7)&(hist_subset[\"baseline_z_score\"]>=7.0)]\n",
    "# hist_subset[cols_to_tabulate][(hist_subset[\"avg_spike_entropy\"]<0.7)&(hist_subset[\"w_bl_z_score\"]>=12.0)]\n",
    "# print(hist_subset[cols_to_tabulate][(hist_subset[\"avg_spike_entropy\"]<0.7)&(hist_subset[\"w_bl_z_score\"]>=14.0)].iloc[6][\"w_bl_output\"])\n",
    "# .to_csv(\"input/pile_low_S_high_z_outliers.csv\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "high_entropy_examples"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "mid_entropy_examples"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "low_entropy_examples"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# plotting histograms of the metric for single runs (not in paper)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "print(f\"groupby legend: {groupby_fields}\")\n",
    "# hist_subset = grouped_df.get_group((True,1,2.0,0.1)) # needs to match the groupby keys and order\n",
    "# hist_subset = grouped_df.get_group((True,1,2.0,0.25)) \n",
    "hist_subset = grouped_df.get_group((True,1,2.0,0.5)) \n",
    "# hist_subset = grouped_df.get_group((True,1,2.0,0.75)) \n",
    "# hist_subset = grouped_df.get_group((True,1,2.0,0.9)) "
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#####  old filters to smooth the histograms"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# hist_subset = hist_subset[(hist_subset[\"no_bl_num_tokens_generated\"] == hist_subset[\"max_new_tokens\"]) & (hist_subset[\"w_bl_num_tokens_generated\"] == hist_subset[\"max_new_tokens\"])]\n",
    "# hist_subset = hist_subset[hist_subset[\"truncated_input\"] != \"\"]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "all_no_bl_wl_fractions = hist_subset[\"no_bl_chisqrd_stat\"]\n",
    "all_w_bl_wl_fractions = hist_subset[\"w_bl_chisqrd_stat\"]\n",
    "all_baseline_wl_fractions = hist_subset[\"baseline_chisqrd_stat\"]\n",
    "\n",
    "# all_no_bl_wl_fractions = hist_subset[\"no_bl_whitelist_fraction\"]\n",
    "# all_w_bl_wl_fractions = hist_subset[\"w_bl_whitelist_fraction\"]\n",
    "# all_baseline_wl_fractions = hist_subset[\"baseline_whitelist_fraction\"]\n",
    "# all_no_bl_wl_fractions = hist_subset[\"no_bl_z_score\"]\n",
    "# all_w_bl_wl_fractions = hist_subset[\"w_bl_z_score\"]\n",
    "# all_baseline_wl_fractions = hist_subset[\"baseline_z_score\"]\n",
    "\n",
    "plt.clf()\n",
    "\n",
    "all_vals = np.concatenate([all_baseline_wl_fractions, all_w_bl_wl_fractions, all_no_bl_wl_fractions])\n",
    "n_bins = 50\n",
    "bins = np.linspace(np.min(all_vals), np.max(all_vals), n_bins)\n",
    "# bins = np.linspace(0.0, 1.0, n_bins)\n",
    "\n",
    "# plt.hist(all_no_bl_wl_fractions, \n",
    "#         bins=bins,\n",
    "#         alpha=0.6,\n",
    "#         label='no blacklisting')\n",
    "\n",
    "\n",
    "plt.hist(all_w_bl_wl_fractions, \n",
    "        bins=bins,\n",
    "        alpha=0.6,\n",
    "        label='with blacklisting')\n",
    "\n",
    "plt.hist(all_baseline_wl_fractions,\n",
    "        bins=bins,\n",
    "        alpha=0.4,\n",
    "        # label='wl')\n",
    "        label='ground truth/real text')\n",
    "\n",
    "# plt.hist(all_baseline_bl_fractions, \n",
    "#         bins=bins,\n",
    "#         alpha=0.5,\n",
    "#         label='bl')\n",
    "\n",
    "plt.legend(loc='upper right')\n",
    "\n",
    "# plt.xlim((-0.1,1.1))\n",
    "# plt.xticks(np.arange(0.0,1.0,0.1))\n",
    "plt.xlabel(\"fraction of total toks gen'd in WL\")\n",
    "plt.ylabel(\"freq\")\n",
    "\n",
    "# plt.title('baseline wl/bl fractions')\n",
    "plt.title(\"Output Whitelist Token Distribution\")\n",
    "\n",
    "# plot_name = \"wl_distro\"\n",
    "# fname = f\"figs/{plot_name}.png\"\n",
    "# plt.savefig(fname, dpi=600)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.clf()\n",
    "\n",
    "all_no_bl_ppls = hist_subset[\"no_bl_ppl\"]\n",
    "all_w_bl_ppls = hist_subset[\"w_bl_ppl\"]\n",
    "all_baseline_ppls = hist_subset[\"baseline_ppl\"]\n",
    "\n",
    "all_vals = list(np.concatenate([all_no_bl_ppls, all_w_bl_ppls]))\n",
    "all_vals = sorted(all_vals)\n",
    "n_bins = 50\n",
    "# bins = np.linspace(all_vals[0], all_vals[-1], n_bins)\n",
    "bins = np.linspace(all_vals[0], 20, n_bins)\n",
    "\n",
    "plt.hist(all_no_bl_ppls, \n",
    "        bins=bins,\n",
    "        alpha=0.6,\n",
    "        label='no blacklisting')\n",
    "\n",
    "plt.hist(all_w_bl_ppls, \n",
    "        bins=bins,\n",
    "        alpha=0.6,\n",
    "        label='with blacklisting')\n",
    "\n",
    "plt.legend(loc='upper right')\n",
    "\n",
    "# plt.xlim((0,1))\n",
    "plt.xlabel(\"perplexity (lower is better)\")\n",
    "plt.ylabel(\"freq\")\n",
    "\n",
    "plt.title('Model-based Output Quality/Fluency')\n",
    "\n",
    "# plot_name = \"ppl_no_baseline\"\n",
    "# fname = f\"figs/{plot_name}.png\"\n",
    "# plt.savefig(fname, dpi=600)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.clf()\n",
    "\n",
    "all_vals = list(np.concatenate([all_no_bl_ppls, all_w_bl_ppls]))\n",
    "all_vals = sorted(all_vals)\n",
    "n_bins = 50\n",
    "# bins = np.linspace(all_vals[0], all_vals[-1], n_bins)\n",
    "bins = np.linspace(all_vals[0], 20, n_bins)\n",
    "\n",
    "plt.hist(all_no_bl_ppls, \n",
    "        bins=bins,\n",
    "        alpha=0.6,\n",
    "        label='no blacklisting')\n",
    "\n",
    "plt.hist(all_w_bl_ppls, \n",
    "        bins=bins,\n",
    "        alpha=0.6,\n",
    "        label='with blacklisting')\n",
    "\n",
    "plt.hist(all_baseline_ppls, \n",
    "        bins=bins,      \n",
    "        alpha=0.4,\n",
    "        label='ground truth/real text')\n",
    "\n",
    "plt.legend(loc='upper right')\n",
    "\n",
    "# plt.xlim((0,1))\n",
    "plt.xlabel(\"perplexity (lower is better)\")\n",
    "plt.ylabel(\"freq\")\n",
    "\n",
    "plt.title('Model-based Output Quality/Fluency')\n",
    "\n",
    "# plot_name = \"ppl_w_baseline\"\n",
    "# fname = f\"figs/{plot_name}.png\"\n",
    "# plt.savefig(fname, dpi=600)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "watermarking-dev",
   "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.10.9"
  },
  "vscode": {
   "interpreter": {
    "hash": "0a3c400d5c70e043163c46602e00ff3a948562bdc78a022eac13a63666386981"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
