{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "This notebook will study the convergence and \"error bars\" for the accuracy of $f_N^*$, the optimal function for noisy inputs, by evaluating these functions on finite datasets.\n",
    "\n",
    "The current experiment for counterexamples trains an SAN to find the lowest error function on a training (or validation!) set, call this $\\hat{f}_{train}$ or $\\hat{f}_{test}$. The definition of $\\hat{f}_{test}$ is that it minimizes error on a _specific_ test set, call it $D_N$ where $N$ is the number of data. This immediately gives two statistical fluctuations:\n",
    " 1. For different $D_N$, we get different $\\hat{f}_{test}$\n",
    " 2. Any specific $\\hat{f}_{test}$ will achieve different accuracies when evaluated on different $D_N$\n",
    "\n",
    "We want to show the (error, sensitivity) point that $\\hat{f}_{test}$ is _attracted towards_. What does this mean? Suppose that we have a function $h$ that we are computing (error, sens) for.\n",
    " - this coordinate should be(?) specific for the dataset: $\\hat{f}$ cannot \"see\" alternative datasets.\n",
    " - This coordinate could therefore be the error of the $h$ that achieves optimal performance on $D_N$\n",
    " - This coordinate _might not_ be the same as the coordinate for $f_N^*$\n",
    " - A plausible choice for $h$ should be the optimal function for the specific dataset Matheus is looking at\n",
    " - **problem**: Every time i resample a dataset, I might get a pretty different sensitivity. **however** I expect all of these sensitivities to be smaller than sens of $f$, right?"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The autoreload extension is already loaded. To reload it, use:\n",
      "  %reload_ext autoreload\n"
     ]
    }
   ],
   "source": [
    "# autoreload magic\n",
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "from mindreadingautobots.sequence_generators import make_datasets, data_io\n",
    "import numpy as np\n",
    "from mindreadingautobots.entropy_and_bayesian import boolean\n",
    "import itertools"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "true sensitivity: 3.5\n"
     ]
    },
    {
     "ename": "TypeError",
     "evalue": "compute_dataset_optimal_sens_and_err() missing 1 required positional argument: 'Z_val'",
     "output_type": "error",
     "traceback": [
      "\u001b[1;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[1;31mTypeError\u001b[0m                                 Traceback (most recent call last)",
      "Cell \u001b[1;32mIn[7], line 38\u001b[0m\n\u001b[0;32m     35\u001b[0m Z_tr \u001b[38;5;241m=\u001b[39m Z[:n_train]\n\u001b[0;32m     36\u001b[0m Z_val \u001b[38;5;241m=\u001b[39m Z[n_val:]\n\u001b[1;32m---> 38\u001b[0m err_memorize_Ztrain_evaluate_Ztrain, sens_memorize_Ztrain \u001b[38;5;241m=\u001b[39m \u001b[43mboolean\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mcompute_dataset_optimal_sens_and_err\u001b[49m\u001b[43m(\u001b[49m\u001b[43mX\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mZ_tr\u001b[49m\u001b[43m)\u001b[49m\n\u001b[0;32m     39\u001b[0m \u001b[38;5;66;03m# f_lookup_on_Z, lookup_table_on_Z = boolean.dataset_lookup_table(Z_tr)\u001b[39;00m\n\u001b[0;32m     40\u001b[0m \u001b[38;5;66;03m# f_lookup_on_X, lookup_table_on_X = boolean.dataset_lookup_table(X)\u001b[39;00m\n\u001b[0;32m     41\u001b[0m \u001b[38;5;66;03m# err_memorize_Ztrain_evaluate_Ztrain = 1 - boolean.compute_acc_on_dataset(f_lookup_on_Z, Z_tr)\u001b[39;00m\n\u001b[0;32m     42\u001b[0m \u001b[38;5;66;03m# sens_memorize_Ztrain = boolean.average_sensitivity(f_lookup_on_Z, all_bitstrings)\u001b[39;00m\n\u001b[0;32m     43\u001b[0m \n\u001b[0;32m     44\u001b[0m \u001b[38;5;66;03m# err_memorize_Ztrain_evaluate_Zval = 1 - boolean.compute_acc_on_dataset(f_lookup_on_Z, Z_val)\u001b[39;00m\n\u001b[0;32m     46\u001b[0m err_memorize_Ztrain_arr\u001b[38;5;241m.\u001b[39mappend(err_memorize_Ztrain_evaluate_Ztrain)\n",
      "\u001b[1;31mTypeError\u001b[0m: compute_dataset_optimal_sens_and_err() missing 1 required positional argument: 'Z_val'"
     ]
    }
   ],
   "source": [
    "\n",
    "p_bitflip = .2\n",
    "ntrain_vals = [5e4]\n",
    "# ntrain_vals = [1e4, 2e4, 3e4, 4e4, 5e4, 6e4, 7e4, 8e4]\n",
    "# ntrain_vals += [1e5, 3e5, 5e5, 1e6, 2e6]\n",
    "all = []\n",
    "signature = (0, 0, 0, 1, 1, 0, 0, 0, 0)\n",
    "n_bits = len(signature) - 1  # n_bits is total number of bits in X, not including the label\n",
    "subseq_idx = list(range(n_bits)) # for this exercise, we don't want/need subsets\n",
    "signature = dict(zip(range(len(signature)), signature))\n",
    "k = n_bits\n",
    "\n",
    "all_bitstrings = np.array(list(itertools.product([0, 1], repeat=n_bits)))\n",
    "true_func = lambda x: signature[sum(x)]\n",
    "true_sens = boolean.average_sensitivity(true_func, all_bitstrings)\n",
    "print(f\"true sensitivity: {true_sens}\")\n",
    "# seed = 1234\n",
    "ntrials = 10\n",
    "\n",
    "n_val = 10000\n",
    "\n",
    "#n10000, nbits14, seed1234, \n",
    "# WARNING: I'm not reproducing the literal same dataset because the nbits is different, and also\n",
    "# i don't know how n_val affects seeding..... TODO: redo this but for generating the actual dataset, \n",
    "for n_train in ntrain_vals:\n",
    "# n_train = 1000000\n",
    "    n_train = int(n_train)\n",
    "\n",
    "    err_memorize_Ztrain_arr = []\n",
    "    err_val_memorize_Ztrain_arr = []\n",
    "    sens_memorize_Ztrain_arr = []\n",
    "    for seed in list(1234 + np.arange(ntrials)):\n",
    "        # sample a variety of \"training performances\"\n",
    "        # this loop will find the f* point for the training data. To do it for the test data, we just re-run with the num data equal to val size\n",
    "        X, Z, subseq_idx = make_datasets.sparse_boolean_weightbased_k_n(n_bits + 1, k, n_train + n_val, signature, p_bitflip=p_bitflip, seed=seed, subseq_idx=subseq_idx)\n",
    "        Z_tr = Z[:n_train]\n",
    "        Z_val = Z[n_val:]\n",
    "        \n",
    "        err_memorize_Ztrain_evaluate_Ztrain, sens_memorize_Ztrain = boolean.compute_dataset_optimal_sens_and_err(X, Z_tr)\n",
    "        # f_lookup_on_Z, lookup_table_on_Z = boolean.dataset_lookup_table(Z_tr)\n",
    "        # f_lookup_on_X, lookup_table_on_X = boolean.dataset_lookup_table(X)\n",
    "        # err_memorize_Ztrain_evaluate_Ztrain = 1 - boolean.compute_acc_on_dataset(f_lookup_on_Z, Z_tr)\n",
    "        # sens_memorize_Ztrain = boolean.average_sensitivity(f_lookup_on_Z, all_bitstrings)\n",
    "\n",
    "        # err_memorize_Ztrain_evaluate_Zval = 1 - boolean.compute_acc_on_dataset(f_lookup_on_Z, Z_val)\n",
    "\n",
    "        err_memorize_Ztrain_arr.append(err_memorize_Ztrain_evaluate_Ztrain)\n",
    "        sens_memorize_Ztrain_arr.append(sens_memorize_Ztrain)\n",
    "        # err_val_memorize_Ztrain_arr.append(err_memorize_Ztrain_evaluate_Zval)\n",
    "        err_val_memorize_Ztrain_arr.append(0)\n",
    "        # training_data_memorization_X_err = 1 - boolean.compute_acc_on_dataset(f_lookup_on_Z, X)\n",
    "    all.append((sens_memorize_Ztrain_arr, err_memorize_Ztrain_arr, err_val_memorize_Ztrain_arr))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0.3753199999996568, 0.3715999999996531, 0.3743199999996558, 0.3747999999996563, 0.37593999999965744, 0.37295999999965446, 0.3738999999996554, 0.3712799999996528, 0.37403999999965554, 0.37433999999965584]\n",
      "[2.78125, 2.9375, 3.03125, 2.9375, 2.84375, 3.03125, 2.765625, 2.828125, 2.90625, 2.96875]\n"
     ]
    },
    {
     "data": {
      "image/png": "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hrp27RrKy8vDypUrqyyP8/E6n+rE5XUpH+23337pZy1ZsqQumwcAANAwYahNmzZh0KBBqTtbUUVFRZofMmRIta+Jy7cuH/32t78tWT5avnx5umaoR48eddk8AACAhhtNLg6rffPNN4fbbrstLF68OA12sGHDhjS6XDRy5MjUb7ro/PPPD7NmzQrf/e53wx//+Mdw2WWXhSeffDKMHTs2Pf/mm2+mkeYef/zx8PLLL6fgNHz48NC3b9800AIAAECTuGZoxIgRYfXq1WHChAlpEISBAwemsFMcJGHp0qVphLmio48+Otxxxx3hkksuCd/61rfCAQccEO69995w6KGHpudjt7tFixalcLV27drQs2fPcPLJJ6chuPW9BgAAmtQACrFVp9iys6046MG2PvOZz6SpOu3btw8PPPBAfTYDAABg53WTAwAAaAmEIQAAIEvCEAAAkCVhCAAAyJIwBAAAZEkYAgAAsiQMAQAAWRKGAACALAlDAABAloQhAAAgS8IQAACQJWEIAADIkjAEAABkSRgCAACyJAwBAABZEoYAAIAsCUMAAECWhCEAACBLwhAAAJAlYQgAAMiSMAQAAGRJGAIAALIkDAEAAFkShgAAgCwJQwAAQJaEIQAAIEvCEAAAkCVhCAAAyJIwBAAAZEkYAgAAsiQMAQAAWRKGAACALAlDAABAloQhAAAgS8IQAACQJWEIAADIUuvG3gCory1btoS33367sTeDauyyyy6hvLy8sTcDAKBGwhDNTqFQCCtWrAhr165t7E2hBp07dw7du3cPZWVljb0pAADVEoZodopBaK+99godOnTwZbsJhtWNGzeGVatWpfkePXo09iYBNLg//elP4bjjjgv/+7//Gw488MDG3hygloQhml3XuGIQ+od/+IfG3hxKaN++fXqMgSj+rXSZA1q6O+64I6xcuTLceeedYeLEiY29OUAtGUCBZqV4jVBsEaJpK/6NXNcF5GDGjBlVHoHmQRiiWdI1runzNwJy8fzzz4c//vGP8ZMvLF68OHWZA5oHYQgAYDv8/Oc//3t34FahVatWaR5oHoQhAIDtELvGVVQUUstQRYWuctDiw9CUKVNCnz59Qrt27cLgwYPD/Pnzayx/zz33hH79+qXyhx12WJg5c+b7Rp+aMGFCGnUqXng9dOjQ8MILL9Rn06DJuuyyy1LXsa2neFwUvfXW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//Ebkv1Tmn1Tmn2Tx77p1KlTY29Ck6Nuanj2Tc3sn9Lsm5a/bzrVsl5yCg8AAMiSMAQAAGRJGAohtG3bNkycODE98n72T2n2TWn2TWn2DbXhfVKafVMz+6c0+6a0tpnumxYxgAIAAEBdaRkCAACyJAwBAABZEoYAAIAsCUMAAECWhCEAACBLLT4MTZo0KRx55JFht912C3vttVc4/fTTw/PPP/+Br1u7dm0477zzQo8ePdIQgwceeGCYOXNmaGnqu38mT54cDjrooNC+ffvQu3fv8I1vfCO89dZboSWZOnVqGDBgQOWdmIcMGRL++7//u8bX3HPPPaFfv36hXbt24bDDDmuR75n67Jubb745HHvssaFLly5pGjp0aJg/f35oierzvim66667QllZWToOadnUTaWpl2qmbipN3VSauinjMPTwww+niuPxxx8Pv/3tb8Pbb78dTj755LBhw4aSr9m8eXM46aSTwssvvxx+9rOfpQ/heMD06tUrtDT12T933HFHuPjii9NY9IsXLw4//OEPw4wZM8K3vvWt0JLsvffe4eqrrw4LFiwITz75ZPjYxz4Whg8fHp599tlqyz/22GPhrLPOCl/84hfD73//+/ShEadnnnkmtDR13Tdz585N+2bOnDlh3rx56YtKfJ+98sorIfd9UxQ/b775zW+mipmWT91UmnqpZuqm0tRNpambalDIzKpVq+J9lQoPP/xwyTJTp04t7LfffoXNmzcXclOb/XPeeecVPvaxj1VZNm7cuMJHP/rRQkvXpUuXwi233FLtc5/97GcLp556apVlgwcPLnz5y18u5KCmfbOtd955p7DbbrsVbrvttkIOPmjfxP1x9NFHpzKjRo0qDB8+fKduH41P3VSaeumDqZtKUzeVpm56V4tvGdrWunXr0uMee+xRssyvf/3r1HwYz0x169YtHHrooeE73/lO2LJlS2jparN/jj766HRmodiU/Oc//zk1uf/zP/9zaKni3z42E8czk/G9UZ14Vik2sW9t2LBhaXlLVpt9s62NGzems701vc9y2jeXX3556g4Uz9ySJ3VTaeql0tRNpambSlM3baOQkS1btqSzIx90puiggw4qtG3btvCv//qvhSeffLJw1113FfbYY4/CZZddVmjJart/ou9///uFXXbZpdC6det0xu4rX/lKoSVatGhRYddddy2Ul5cXOnXqVPjNb35TsmzcH3fccUeVZVOmTCnstddehdz3zbbOOeecdIb7b3/7WyH3ffN///d/hV69ehVWr16d5lvy2Teqp24qTb1UPXVTaeqm0tRN1csqDMUPxn322aewbNmyGssdcMABhd69e6fmwaLvfve7he7duxdastrunzlz5hS6detWuPnmm9OB9Ytf/CLtr8svv7zQ0mzatKnwwgsvpC8eF198caFr166FZ599ttqyuVU4ddk3W5s0aVJqmv/DH/5QaKlqu2/Wr19f6NOnT2HmzJmVy1pyhUP11E2lqZeqp24qTd1Umrop8zAU+xPvvffehT//+c8fWPa4444rnHjiiVWWxTdEPNMU30i5759jjjmm8M1vfrPKsh//+MeF9u3bp7N4LVl8X5x99tnVPhcr3u9973tVlk2YMKEwYMCAQg5q2jdF1113XTob9cQTTxRyUmrf/P73v0+fK/EsXXEqKytLU/z/kiVLGmV72XnUTaWpl2pP3VSauqk0dVMm1wzFwDd27Njwy1/+Mjz00ENh3333/cDXfPSjHw1LliwJFRUVlcv+9Kc/paFM27RpE3LfP7FPbatWVd865eXlletryeJ7YtOmTdU+F/vdzp49u8qyOBJSbfsqN3c17Zvo2muvDVdccUWYNWtWOOKII0JOSu2bONTt008/HZ566qnK6bTTTgv/9E//lP4fRzaiZVI3laZeqjt1U2nqptLUTX9XaOFi/8+Y9ufOnVt47bXXKqeNGzdWlvn85z+fmguLli5dmkYTGTt2bOH5558v3H///ak5+corryy0NPXZPxMnTkz7584770xn7P7nf/6nsP/++6cRa1qS+DvH0Yteeuml1O0izsezIvH3rW6/PProo6mv+vXXX19YvHhx2k+xe8LTTz9daGnqum+uvvrqQps2bQo/+9nPqrzP3njjjULu+2ZbLbkrAu9RN5WmXqqZuqk0dVNp6qbSWnwYinmvuulHP/pRZZnjjz8+/ZG39thjj6WhJ+PFqvFiuquuuqpKP+2c98/bb7+dLtiNFU27du1SE/y5555b+Otf/1poSeJFyrGvevyg3HPPPVNzcvFDo9T75u677y4ceOCB6TWHHHJInS7cbMn7Jpat7n0WK+WWpj7vm1wqHN6jbipNvVQzdVNp6qbS1E2llcV/iq1EAAAAuWjx1wwBAABURxgCAACyJAwBAABZEoYAAIAsCUMAAECWhCEAACBLwhAAAJAlYQgAAMiSMAQAAGRJGAIAALIkDAEAACFH/x/z4Q+6ANeZEgAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 1000x600 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "print(err_memorize_Ztrain_arr)\n",
    "print(sens_memorize_Ztrain_arr)\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "fig, axes = plt.subplots(1, 2, figsize=(10, 6), sharey=True)\n",
    "\n",
    "cmap = plt.cm.inferno\n",
    "colors = [cmap(i/len(ntrain_vals)) for i in range(len(ntrain_vals))]\n",
    "\n",
    "\n",
    "for i, (x, y, yval) in enumerate(all): \n",
    "    avg_x = np.mean(x)\n",
    "    avg_y = np.mean(y)\n",
    "    avg_yval = np.mean(yval)\n",
    "    axes[0].scatter(avg_x, avg_y, marker='*', s=200, color=colors[i], edgecolor='black')\n",
    "    axes[0].scatter(x, y,  alpha=1, marker='.', color=colors[i], label=ntrain_vals[i])\n",
    "\n",
    "    axes[1].scatter(avg_x, avg_yval, marker='*', s=200, color=colors[i], edgecolor='black')\n",
    "    axes[1].scatter(x, yval,  alpha=1, marker='.', color=colors[i], label=ntrain_vals[i])\n",
    "axes[0].legend()\n",
    "\n",
    "for ax in axes:\n",
    "    ax.axvline(3.5)\n",
    "    ax.axvline(2.6)\n",
    "# ax.set_xlim(2.4, 3.6)\n",
    "# ax.set_ylim(0.34, 0.45)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "- **If we minimize on the training data**, then we will get convergence towards lower sensitivity with increasing $N$, and the valiation behavior is mostly stochastic ($N$-independent!) around the $N_{val}$ test dataset, i.e. there's noise due to small test size..?\n",
    "\n",
    "- **If we postselect for minimum validation error**, I'm not as sure what happens. "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "1. From `counterexamples_matheus/workbook_matheus` - write a downloading script, to get matheus' data locally\n",
    "2. **I'm still surprised how high the sens was for the 10000 data transformers**, figure this out\n",
    "3. rewrite plotting scripts for two plots: (i) appendix plot: overlay a bunch of n=50000 runs, each in a different color\n",
    "    - also overlay n=50000 lookup plot FOR THAT SPECIFIC DATASET, \n",
    "    - CHECKME: I'm still surprised why there was a 0.34 accuracy. This seems related to nbits=14, implying that there are 2^14 possible inputs, meaning that collision probability is smaller, meaning that memorization is easier, meaning training error is lower. BUT this shouldn't happen for test set, right? Does $n\\neq \\infty$ even make sense for validation. Think about the cases:\n",
    "\n",
    "    - case 1: model minimizes training error. let $g_{N}^*$ be the function that minimizes training set error. The model train behavior will approach $(s(g_{N}^*), err_{train}(g_{N}^*))$, while the model test behavior will approach $(s(g_{N}^*), err_{test}(g_{N}^*))$. For small $N$, these two points may be far apart\n",
    "    - case 2: model doe early stopping at minimum test error. Let $g^{test}_{N}$ denote this. The model test behavior approaches $(s(g_{N}^{test}), err_{test}(g_{N}^{test}))$, but this is _different_ than the point that minimizes test error via lookup table on the test set (which itself converges towards $f_N^*$). its still possible to get lower test error than $f_N^*$ by fluctuations, but we at least know we aren't biased downwards via memorization. <--- this seems to be the smarter option for doing a plot. **the question**: does $N$ affect this plot as much?\n",
    "4. Appendix plot: the one above: For a single _distribution_, as a function of data amount, plot convergence of memorization towards $f_N^*$ accuracy and sensitivity. Appendix discussion: memorization on $N$ training data has lower sensitivity than $f$, but that difference starts out small for small $N$ due to the ease of memorizing a small number rows of the training data. !!!!**can We can use renyi 2-entropy to estimate the size of $N$ for which sensitivity starts to decrease?**, since the inverse reenyi-2-entropy is the probability of colision? I'm not sure how to convert collision into coupon collecting though..."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 106,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.92578125"
      ]
     },
     "execution_count": 106,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "boolean.compute_acc_test(f_lookup, true_func, n_bits)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": ".venv",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.11.1"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
