{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_2052/2221352350.py:18: FutureWarning: You are using `torch.load` with `weights_only=False` (the current default value), which uses the default pickle module implicitly. It is possible to construct malicious pickle data which will execute arbitrary code during unpickling (See https://github.com/pytorch/pytorch/blob/main/SECURITY.md#untrusted-models for more details). In a future release, the default value for `weights_only` will be flipped to `True`. This limits the functions that could be executed during unpickling. Arbitrary objects will no longer be allowed to be loaded via this mode unless they are explicitly allowlisted by the user via `torch.serialization.add_safe_globals`. We recommend you start setting `weights_only=True` for any use case where you don't have full control of the loaded file. Please open an issue on GitHub for any issues related to this experimental feature.\n",
      "  population = torch.load(f'{folder_name}/popl_lastlayers.pt')\n",
      "/tmp/ipykernel_2052/2221352350.py:21: FutureWarning: You are using `torch.load` with `weights_only=False` (the current default value), which uses the default pickle module implicitly. It is possible to construct malicious pickle data which will execute arbitrary code during unpickling (See https://github.com/pytorch/pytorch/blob/main/SECURITY.md#untrusted-models for more details). In a future release, the default value for `weights_only` will be flipped to `True`. This limits the functions that could be executed during unpickling. Arbitrary objects will no longer be allowed to be loaded via this mode unless they are explicitly allowlisted by the user via `torch.serialization.add_safe_globals`. We recommend you start setting `weights_only=True` for any use case where you don't have full control of the loaded file. Please open an issue on GitHub for any issues related to this experimental feature.\n",
      "  model = torch.load(f'{folder_name}/popl_model.pt')\n"
     ]
    }
   ],
   "source": [
    "import torch\n",
    "import os\n",
    "\n",
    "#import from outer folder\n",
    "import sys\n",
    "\n",
    "sys.path.append('..')\n",
    "\n",
    "from experiments.synthetic_stateless_utils import BaseRewardModel\n",
    "from experiments.synthetic_stateless import get_monotone\n",
    "from src.popl import select_one_best\n",
    "\n",
    "device = torch.device('cuda' if torch.cuda.is_available() else 'cpu')\n",
    "\n",
    "folder_name = '../multirun/2024-08-23/14-52-16/+learning_rate=0.001,+method=popl,+seed=1/'\n",
    "\n",
    "# Load the population\n",
    "population = torch.load(f'{folder_name}/popl_lastlayers.pt')\n",
    "\n",
    "# Load the best individual\n",
    "model = torch.load(f'{folder_name}/popl_model.pt')\n",
    "\n",
    "model = model.to(device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([200, 128])\n"
     ]
    }
   ],
   "source": [
    "print(population.shape)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([100, 128])\n",
      "torch.Size([100, 128])\n"
     ]
    }
   ],
   "source": [
    "sample_num = 100\n",
    "feature_model = model\n",
    "\n",
    "\n",
    "s0 = torch.linspace(0, 1, sample_num).to(device)\n",
    "s1 = torch.linspace(0, 1, sample_num).to(device)\n",
    "f0 = feature_model.get_penultimate_layer(s0[:, None])\n",
    "f1 = feature_model.get_penultimate_layer(s1[:, None])\n",
    "\n",
    "print(f0.shape)\n",
    "print(f1.shape)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n",
      "torch.Size([100])\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAigAAAGzCAYAAAAFROyYAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjkuMiwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy8hTgPZAAAACXBIWXMAAA9hAAAPYQGoP6dpAAAzZUlEQVR4nO3deXhU5aHH8d8kJJMIySQs2SQsggoqiIKGCIjaWOrClUovYlFBQWxBK5sW6gVE0LCpCBURrwWuS7H6qHVBFEHgihApikVBQEGIS4IISSCsSd77x9wMmWQSMmGWMzPfz/PMY/LOmZM3B3R+vr9zztiMMUYAAAAWEhXsCQAAAFRHQAEAAJZDQAEAAJZDQAEAAJZDQAEAAJZDQAEAAJZDQAEAAJZDQAEAAJZDQAEAAJZDQAEQkmbNmqVzzjlH0dHR6tKlS7Cn49Hq1atls9m0evXqYE8FCDkEFACSpE8++UQPP/ywioqKgj2V0/rggw/04IMPqkePHlq0aJEee+yxoM5n/vz5Wrx4cVDnAIQbG5/FA0CSZs+erQceeEC7d+9WmzZtgj2dOo0fP16zZs3S0aNHFRsbG+zp6KKLLlLz5s1rrJRUVFToxIkTio2NVVQU/z8IeKNRsCcAAN7at2+f4uPjLRFO6hIVFaW4uLhgTwMISUR6wIcefvhh2Ww27dixQ7fddpscDodatGihiRMnyhij/Px83XTTTUpMTFRaWpoef/zxGvvYt2+fhg4dqtTUVMXFxeniiy/WkiVL3Lb57rvvZLPZNHv2bC1cuFDt2rWT3W7XZZddpo0bN9bY56pVq9SrVy81btxYSUlJuummm7Rt2za3eT/wwAOSpLZt28pms8lms+m7776TJJWVlWnq1Kmun9OmTRv95S9/0fHjx91+Tps2bXTjjTfq448/1uWXX664uDidc845+p//+Z8acyoqKtKoUaOUmZkpu92u9u3ba8aMGaqoqKjzGNtsNi1atEilpaWueS5evNh1TDxVLTabTQ8//LDb72uz2fTNN99oyJAhSkpKksPh0J133qkjR47UeP2LL76oyy+/XGeddZaSk5N15ZVX6oMPPnD9zl999ZXWrFnjms9VV10lqfZzUF599VV17dpV8fHxat68uW677Tb98MMPbtsMGTJETZo00Q8//KB+/fqpSZMmatGihcaNG6fy8nK3bZcuXaquXbsqISFBiYmJ6tSpk5566qk6jyNgdQQUwA9uueUWVVRUaPr06crKytK0adM0Z84cXXvttTr77LM1Y8YMtW/fXuPGjdPatWtdrzt69KiuuuoqvfDCCxo0aJBmzZolh8OhIUOGeHzDefnllzVr1izdc889mjZtmr777jvdfPPNOnnypGubDz/8UH369NG+ffv08MMPa8yYMfrkk0/Uo0cPVwC5+eabdeutt0qSnnzySb3wwgt64YUX1KJFC0nSsGHDNGnSJF166aV68skn1bt3b+Xm5mrgwIE15vTNN9/od7/7na699lo9/vjjSk5O1pAhQ/TVV1+5tjly5Ih69+6tF198UXfccYfmzp2rHj16aMKECRozZkydx/aFF15Qr169ZLfbXfO88sor6/+HU8WAAQN06NAh5ebmasCAAVq8eLGmTJnits2UKVN0++23KyYmRo888oimTJmizMxMrVq1SpI0Z84ctWzZUh06dHDN56GHHqr1Zy5evFgDBgxQdHS0cnNzdffdd+v1119Xz549a5z/U15erj59+qhZs2aaPXu2evfurccff1wLFy50bbNixQrdeuutSk5O1owZMzR9+nRdddVVWrduXYOOCWAZBoDPTJ482Ugyw4cPd42VlZWZli1bGpvNZqZPn+4aP3jwoImPjzeDBw92jc2ZM8dIMi+++KJr7MSJEyY7O9s0adLElJSUGGOM2b17t5FkmjVrZg4cOODa9p///KeRZN5++23XWJcuXUxKSor55ZdfXGNffPGFiYqKMnfccYdrbNasWUaS2b17t9vvtHnzZiPJDBs2zG183LhxRpJZtWqVa6x169ZGklm7dq1rbN++fcZut5uxY8e6xqZOnWoaN25sduzY4bbP8ePHm+joaLN3715Tl8GDB5vGjRu7jVUek0WLFtXYXpKZPHmy6/vKP6e77rrLbbvf/va3plmzZq7vd+7caaKiosxvf/tbU15e7rZtRUWF6+sLL7zQ9O7du8bP/eijj4wk89FHHxljnH+WKSkp5qKLLjJHjx51bffOO+8YSWbSpEluv6Mk88gjj7jt85JLLjFdu3Z1fX///febxMREU1ZWVuPnA6GMFRTAD4YNG+b6Ojo6Wt26dZMxRkOHDnWNJyUl6fzzz9euXbtcY8uWLVNaWpprNUOSYmJi9Kc//UmHDx/WmjVr3H7OLbfcouTkZNf3vXr1kiTXPn/66Sdt3rxZQ4YMUdOmTV3bde7cWddee62WLVt22t+lcpvqKxtjx46VJL377rtu4xdccIFrHpLUokWLGr/nq6++ql69eik5OVn79+93PXJyclReXu62quRPf/jDH9y+79Wrl3755ReVlJRIkt58801VVFRo0qRJNU5ytdlsXv+8f/3rX9q3b59GjBjhdm7KDTfcoA4dOtQ4lrXNseqxTEpKUmlpqVasWOH1fAArI6AAftCqVSu37x0Oh+Li4tS8efMa4wcPHnR9v2fPHp177rk13gw7duzoer6un1MZVir3Wbn9+eefX2OOHTt21P79+1VaWlrn77Jnzx5FRUWpffv2buNpaWlKSko67Zwq51X199y5c6eWL1+uFi1auD1ycnIkOc/DCYTTHb9vv/1WUVFRuuCCC3zy8+r68+jQoUONYxkXF+eq2arOseqxHDFihM477zxdd911atmype666y4tX77cJ/MFgomreAA/iI6OrteYJJkzuNLfH/usTX1XDOozp4qKCl177bV68MEHPW573nnn+Wx+1U8orSqQx68haptfVSkpKdq8ebPef/99vffee3rvvfe0aNEi3XHHHTVOrgZCCQEFsJDWrVvr3//+tyoqKtxWUb7++mvX897uT5K2b99e47mvv/5azZs3V+PGjSXV/gbfunVrVVRUaOfOna6VHEkqLCxUUVGR13OSpHbt2unw4cOuFRNfqFz9qH6iafVVCW+0a9dOFRUV2rp1a513q61veKv653HNNde4Pbd9+/YGHUtJio2NVd++fdW3b19VVFRoxIgRevbZZzVx4sQaK19AqKDiASzk+uuvV0FBgV555RXXWFlZmebNm6cmTZqod+/eXu0vPT1dXbp00ZIlS9zeuL/88kt98MEHuv76611jlUGl+ht85TZz5sxxG3/iiSckOc+f8NaAAQO0fv16vf/++zWeKyoqUllZmdf7TExMVPPmzWucvzJ//nyv91WpX79+ioqK0iOPPFLj8ueqqyyNGzeu1x14u3XrppSUFC1YsMDtEu333ntP27Zta9Cx/OWXX9y+j4qKUufOnSWpxmXgQChhBQWwkOHDh+vZZ5/VkCFDtGnTJrVp00avvfaa1q1bpzlz5ighIcHrfc6aNUvXXXedsrOzNXToUB09elTz5s2Tw+FwuzdI165dJUkPPfSQBg4cqJiY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5ORo586d/pgKEJYIKQDCnc8DysGDB9WjRw/FxMTovffe09atW/X4448rOTnZtc3MmTM1d+5cLViwQHl5eWrcuLH69OmjY8eO+Xo6QNii8gEQzmzGePp/sYYbP3681q1bp//93//1+LwxRhkZGRo7dqzGjRsnSSouLlZqaqoWL16sgQMHnvZnlJSUyOFwqLi4WImJib6cPhCSPIUS3/6bDQBnzpv3b5+voLz11lvq1q2b/vM//1MpKSm65JJL9Nxzz7me3717twoKCpSTk+MaczgcysrK0vr16z3u8/jx4yopKXF7ADiltsqne/fAzwUAfMHnAWXXrl165plndO655+r999/XH//4R/3pT3/SkiVLJEkFBQWSpNTUVLfXpaamup6rLjc3Vw6Hw/XIzMz09bSBkOep8snLo/IBEJp8HlAqKip06aWX6rHHHtMll1yi4cOH6+6779aCBQsavM8JEyaouLjY9cjPz/fhjIHwwVU+AMKFzwNKenq6LrjgArexjh07au/evZKktLQ0SVJhYaHbNoWFha7nqrPb7UpMTHR7AKgdlQ+AUOfzgNKjRw9t377dbWzHjh1q3bq1JKlt27ZKS0vTypUrXc+XlJQoLy9P2dnZvp4OELGofACEska+3uHo0aN1xRVX6LHHHtOAAQP06aefauHChVq4cKEkyWazadSoUZo2bZrOPfdctW3bVhMnTlRGRob69evn6+kAEW3DBuc/q4cSm42rfABYm88DymWXXaY33nhDEyZM0COPPKK2bdtqzpw5GjRokGubBx98UKWlpRo+fLiKiorUs2dPLV++XHFxcb6eDgA5w4inkJKVdSrEAICV+Pw+KIHAfVCAhune3VnzVBd6/xUAEIqCeh8UANbFVT4AQgUBBYhAXOUDwOoIKECEMqZmUOEqHwBWQUABIhyVDwArIqAAoPIBYDkEFACSqHwAWAsBBYAbKh8AVkBAAVADlQ+AYCOgAPCIygdAMBFQANSJygdAMBBQAJwWlQ+AQCOgAKgXKh8AgURAAeAVKh8AgUBAAeC12kIKQQWArxBQADSIp8pHIqQA8A0CCoAzQkgB4A8EFABnjMoHgK8RUAD4BJUPAF8ioADwKUIKAF8goADwOWOkrCz3MSofAN4goADwiw0bWE0B0HAEFAB+RUgB0BAEFAB+R+UDwFsEFAABQeUDwBsEFAABRUgB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81//9V8mJibGbNmyJcAzDy3eHuff//735umnnzaff/652bZtmxkyZIhxOBzm+++/D/DMQ4u3x7nS7t27zdlnn2169eplbrrppsBMNoR5e5yPHz9uunXrZq6//nrz8ccfm927d5vVq1ebzZs3B3jmocXb4/zSSy8Zu91uXnrpJbN7927z/vvvm/T0dDN69OgAzzy0LFu2zDz00EPm9ddfN5LMG2+8Uef2u3btMmeddZYZM2aM2bp1q5k3b56Jjo42y5cv9+s8Iy6gXH755WbkyJGu78vLy01GRobJzc31uP2AAQPMDTfc4DaWlZVl7rnnHr/OM9R5e5yrKysrMwkJCWbJkiX+mmJYaMhxLisrM1dccYX57//+bzN48GACSj14e5yfeeYZc84555gTJ04EaophwdvjPHLkSHPNNde4jY0ZM8b06NHDr/MMJ/UJKA8++KC58MIL3cZuueUW06dPHz/OzJiIqnhOnDihTZs2KScnxzUWFRWlnJwcrV+/3uNr1q9f77a9JPXp06fW7dGw41zdkSNHdPLkSTVt2tRf0wx5DT3OjzzyiFJSUjR06NBATDPkNeQ4v/XWW8rOztbIkSOVmpqqiy66SI899pjKy8sDNe2Q05DjfMUVV2jTpk2uGmjXrl1atmyZrr/++oDMOVIE630wJD/NuKH279+v8vJypaamuo2npqbq66+/9viagoICj9sXFBT4bZ6hriHHubo///nPysjIqPEvBU5pyHH++OOP9fzzz2vz5s0BmGF4aMhx3rVrl1atWqVBgwZp2bJl+uabbzRixAidPHlSkydPDsS0Q05DjvPvf/977d+/Xz179pQxRmVlZfrDH/6gv/zlL4GYcsSo7X2wpKRER48eVXx8vF9+bkStoCA0TJ8+XUuXLtUbb7yhuLi4YE8nbBw6dEi33367nnvuOTVv3jzY0wlrFRUVSklJ0cKFC9W1a1fdcssteuihh7RgwYJgTy2srF69Wo899pjmz5+vzz77TK+//rreffddTZ06NdhTgw9E1ApK8+bNFR0drcLCQrfxwsJCpaWleXxNWlqaV9ujYce50uzZszV9+nR9+OGH6ty5sz+nGfK8Pc7ffvutvvvuO/Xt29c1VlFRIUlq1KiRtm/frnbt2vl30iGoIX+f09PTFRMTo+joaNdYx44dVVBQoBMnTig2Ntavcw5FDTnOEydO1O23365hw4ZJkjp16qTS0lINHz5cDz30kKKi+H9wX6jtfTAxMdFvqydShK2gxMbGqmvXrlq5cqVrrKKiQitXrlR2drbH12RnZ7ttL0krVqyodXs07DhL0syZMzV16lQtX75c3bp1C8RUQ5q3x7lDhw7asmWLNm/e7Hr8x3/8h66++mpt3rxZmZmZgZx+yGjI3+cePXrom2++cQVASdqxY4fS09MJJ7VoyHE+cuRIjRBSGQoNn4PrM0F7H/TrKbgWtHTpUmO3283ixYvN1q1bzfDhw01SUpIpKCgwxhhz++23m/Hjx7u2X7dunWnUqJGZPXu22bZtm5k8eTKXGdeDt8d5+vTpJjY21rz22mvmp59+cj0OHToUrF8hJHh7nKvjKp768fY479271yQkJJh7773XbN++3bzzzjsmJSXFTJs2LVi/Qkjw9jhPnjzZJCQkmL///e9m165d5oMPPjDt2rUzAwYMCNavEBIOHTpkPv/8c/P5558bSeaJJ54wn3/+udmzZ48xxpjx48eb22+/3bV95WXGDzzwgNm2bZt5+umnuczYX+bNm2datWplYmNjzeWXX242bNjgeq53795m8ODBbtv/4x//MOedd56JjY01F154oXn33XcDPOPQ5M1xbt26tZFU4zF58uTATzzEePv3uSoCSv15e5w/+eQTk5WVZex2uznnnHPMo48+asrKygI869DjzXE+efKkefjhh027du1MXFycyczMNCNGjDAHDx4M/MRDyEcffeTxv7eVx3bw4MGmd+/eNV7TpUsXExsba8455xyzaNEiv8/TZgzrYAAAwFoi6hwUAAAQGggoAADAcggoAADAcggoAADAcggoAADAcggoAADAcggoAADAcggoAADAcggoAADAcggoAADAcggoAADAcv4PVMpbs90wJvAAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "plt.figure()\n",
    "\n",
    "for last_layer in population:\n",
    "    outputs0 = f0 @ last_layer\n",
    "\n",
    "    monotone = get_monotone(outputs0)\n",
    "    print(monotone.shape)\n",
    "    plt.plot(s0.detach().cpu().numpy(), get_monotone(outputs0).detach().cpu().numpy(), 'b', alpha=0.1)\n",
    "\n",
    "plt.title(\"monotone functions\")\n",
    "plt.show()\n",
    "\n",
    "for last_layer in population:\n",
    "    outputs0 = f0 @ last_layer\n",
    "\n",
    "    plt.plot(s0.detach().cpu().numpy(), outputs0.detach().cpu().numpy(), 'b', alpha=0.1)\n",
    "\n",
    "plt.title(\"outputs\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "metadata": {},
   "outputs": [],
   "source": [
    "def reward_fn_1d(state: torch.Tensor, identity: torch.Tensor) -> torch.Tensor:\n",
    "    # identity ignored\n",
    "    state = state.squeeze(-1)\n",
    "    rewards = state.clone()\n",
    "    double_rewards = torch.rand(rewards.shape, device=rewards.device) < 0.5\n",
    "    rewards[(state >= 0.8) & double_rewards] *= 2\n",
    "    rewards[(state >= 0.8) & ~double_rewards] *= 0\n",
    "    return rewards\n",
    "\n",
    "\n",
    "sample_num = 100\n",
    "# states are interpolated from 0 to 1\n",
    "states0 = []\n",
    "states1 = []\n",
    "#get a full nxn arrangement\n",
    "for i in range(1, sample_num):\n",
    "    for j in range(1, sample_num):\n",
    "        states0.append(i / sample_num)\n",
    "        states1.append(j / sample_num)\n",
    "\n",
    "s0 = torch.tensor(states0).to(device)\n",
    "s1 = torch.tensor(states1).to(device)\n",
    "\n",
    "f0 = feature_model.get_penultimate_layer(s0[:, None])\n",
    "f1 = feature_model.get_penultimate_layer(s1[:, None])\n",
    "\n",
    "# get ground truth preferences for each hidden context group\n",
    "r0_z1 = reward_fn_1d(s0, torch.tensor(1))\n",
    "r1_z1 = reward_fn_1d(s1, torch.tensor(1))\n",
    "r0_z0 = reward_fn_1d(s0, torch.tensor(0))\n",
    "r1_z0 = reward_fn_1d(s1, torch.tensor(0))\n",
    "pp0 = (r1_z0 > r0_z0).long()\n",
    "pp1 = (r1_z1 > r0_z1).long()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "metadata": {},
   "outputs": [],
   "source": [
    "def select_one_best(population, features0, features1, pairwise_prefs):\n",
    "    max_score = 0\n",
    "    best_ind = None\n",
    "    for i in range(len(population)):\n",
    "        last_layer = population[i]\n",
    "\n",
    "        output0 = features0 @ last_layer\n",
    "        output1 = features1 @ last_layer\n",
    "\n",
    "        pred_output = (output1 > output0).to(device=last_layer.device)\n",
    "        scores = torch.logical_not(\n",
    "            (torch.logical_xor(pairwise_prefs, pred_output)))\n",
    "        score = torch.sum(scores).item()\n",
    "        if score > max_score:\n",
    "            max_score = score\n",
    "            best_ind = i\n",
    "\n",
    "    return population[best_ind], max_score"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 73,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "best individual got score 8112/9801\n",
      "best individual got score 8032/9801\n"
     ]
    }
   ],
   "source": [
    "ind, score = select_one_best(population, f0, f1, pp0)\n",
    "\n",
    "print(f\"best individual got score {score}/{len(pp0)}\")\n",
    "\n",
    "ind2, score2 = select_one_best(population, f0, f1, pp1)\n",
    "\n",
    "print(f\"best individual got score {score2}/{len(pp1)}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "called get_monotonie with r_values of shape: torch.Size([100])\n",
      "called get_monotonie with r_values of shape: torch.Size([100])\n"
     ]
    },
    {
     "data": {
      "image/png": "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Ro4cWbr/jh07DG9vb2PixInGoUOHjLFjx+pjxpegrOM8YcIEw9fX13jnnXeM06dPl96ys7PN+hVcQlnH+ff0KZ5LU9ZxPnHihBEcHGw89dRTxuHDh421a9catWrVMhISEsz6FVxCWcd57NixRnBwsLFixQrj2LFjxkcffWQ0bNjQiIuLM+tXcAnZ2dnG/v37jf379xuAMXnyZGP//v3GTz/9ZBiGYYwcOdLo06dP6f7/+5jxc889Zxw6dMiYMWOGPmZcUaZNm2bUrVvX8PX1NW688Ubjs88+K72vY8eORt++fS/YPzU11bj66qsNX19fo2nTpsa6desqObFrKss416tXzwD+cBs7dmzlB3cxZX0//18qKJeurOO8c+dOo02bNoafn5/RoEEDY9y4cUZxcXElp3Y9ZRnnoqIi4+WXXzYaNmxo+Pv7G5GRkcbAgQONzMzMyg/uQj755JOL/r3939j27dvX6Nix4x8eExUVZfj6+hoNGjQwFi5cWOE5PQxD82AiIiLiXKrUOSgiIiLiGlRQRERExOmooIiIiIjTUUERERERp6OCIiIiIk5HBUVEREScjgqKiIiIOB0VFBEREXE6KigiIiLidFRQRERExOmooIiIiIjTUUERERERp/P/AZR6FgsqvKVHAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "#graph best individuals\n",
    "\n",
    "all_states = torch.linspace(0, 1, 100).to(device)\n",
    "features = feature_model.get_penultimate_layer(all_states[:, None])\n",
    "\n",
    "outputs0 = features @ ind\n",
    "\n",
    "plt.plot(all_states.detach().cpu().numpy(), get_monotone(outputs0).detach().cpu().numpy(), 'r', alpha=1)\n",
    "\n",
    "outputs1 = features @ ind2\n",
    "\n",
    "plt.plot(all_states.detach().cpu().numpy(), get_monotone(outputs1).detach().cpu().numpy(), 'g', alpha=1)\n",
    "\n",
    "plt.show()\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 56,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "r1: 0.2752780318260193, r0: 0.2799714207649231\n",
      "r1: 0.8146107196807861, r0: 0.8215488195419312\n",
      "r1: 0.8109376430511475, r0: 0.8133863806724548\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.2881338596343994, r0: 0.27935925126075745\n",
      "r1: 0.8139985799789429, r0: 0.8203244805335999\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.7942046523094177, r0: 0.8027752041816711\n",
      "r1: 0.211202934384346, r0: 0.20304051041603088\n",
      "r1: 0.8137944936752319, r0: 0.8186919689178467\n",
      "r1: 0.8107336163520813, r0: 0.8158351182937622\n",
      "r1: 0.19018466770648956, r0: 0.1814100593328476\n",
      "r1: 0.3485358655452728, r0: 0.35322925448417664\n",
      "r1: 0.8119579553604126, r0: 0.815018892288208\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.2014080137014389, r0: 0.19304151833057404\n",
      "r1: 0.8021630644798279, r0: 0.8107336163520813\n",
      "r1: 0.8131822943687439, r0: 0.8184878826141357\n",
      "r1: 0.6780940890312195, r0: 0.6734006404876709\n",
      "r1: 0.2226303368806839, r0: 0.21385572850704193\n",
      "r1: 0.815018892288208, r0: 0.8219569325447083\n",
      "r1: 0.8037955164909363, r0: 0.8119579553604126\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.19997958838939667, r0: 0.19304151833057404\n",
      "r1: 0.8154270052909851, r0: 0.8225691318511963\n",
      "r1: 0.8154270052909851, r0: 0.8223650455474854\n",
      "r1: 0.8172635436058044, r0: 0.8246097564697266\n",
      "r1: 0.8176716566085815, r0: 0.8258340954780579\n",
      "r1: 0.7909396886825562, r0: 0.7880828380584717\n",
      "r1: 0.7993062138557434, r0: 0.8070604801177979\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8115498423576355, r0: 0.81440669298172\n",
      "r1: 0.8195081949234009, r0: 0.827466607093811\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.34567901492118835, r0: 0.35322925448417664\n",
      "r1: 0.291398823261261, r0: 0.282624214887619\n",
      "r1: 0.8154270052909851, r0: 0.8223650455474854\n",
      "r1: 0.19365370273590088, r0: 0.1856953352689743\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.778287947177887, r0: 0.7860422134399414\n",
      "r1: 0.8180797696113586, r0: 0.8254259824752808\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8193041682243347, r0: 0.8270584344863892\n",
      "r1: 0.8172635436058044, r0: 0.8246097564697266\n",
      "r1: 0.790123462677002, r0: 0.7980818152427673\n",
      "r1: 0.8093051910400391, r0: 0.8156310319900513\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8137944936752319, r0: 0.8180797696113586\n",
      "r1: 0.20079582929611206, r0: 0.1958983838558197\n",
      "r1: 0.8172635436058044, r0: 0.8248137831687927\n",
      "r1: 0.8115498423576355, r0: 0.8156310319900513\n",
      "r1: 0.6758493781089783, r0: 0.6831955909729004\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.81440669298172, r0: 0.821140706539154\n",
      "r1: 0.19018466770648956, r0: 0.1814100593328476\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8180797696113586, r0: 0.8256300091743469\n",
      "r1: 0.2999693751335144, r0: 0.29119476675987244\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.1963064968585968, r0: 0.18855218589305878\n",
      "r1: 0.2367105334997177, r0: 0.22793592512607574\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8184878826141357, r0: 0.826242208480835\n",
      "r1: 0.8113457560539246, r0: 0.8156310319900513\n",
      "r1: 0.8182838559150696, r0: 0.8264462947845459\n",
      "r1: 0.8174675703048706, r0: 0.8256300091743469\n",
      "r1: 0.8168554306030273, r0: 0.8242015838623047\n",
      "r1: 0.796245276927948, r0: 0.802367091178894\n",
      "r1: 0.7209468483924866, r0: 0.7258443236351013\n",
      "r1: 0.8172635436058044, r0: 0.8246097564697266\n",
      "r1: 0.7919600009918213, r0: 0.7960411906242371\n",
      "r1: 0.8180797696113586, r0: 0.8258340954780579\n",
      "r1: 0.21059074997901917, r0: 0.2018161416053772\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8050199151039124, r0: 0.8113457560539246\n",
      "r1: 0.19018466770648956, r0: 0.1810019314289093\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8076726794242859, r0: 0.8121620416641235\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8184878826141357, r0: 0.8260381817817688\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8188960552215576, r0: 0.8264462947845459\n",
      "r1: 0.8191000819206238, r0: 0.8272625207901001\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.765023946762085, r0: 0.7735944986343384\n",
      "r1: 0.7950209379196167, r0: 0.8031833171844482\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "r1: 0.8125701546669006, r0: 0.81440669298172\n",
      "r1: 0.19018466770648956, r0: 0.1814100593328476\n",
      "r1: 0.8195081949234009, r0: 0.8276706337928772\n",
      "tensor(1., device='cuda:0')\n",
      "tensor(1., device='cuda:0')\n"
     ]
    }
   ],
   "source": [
    "max_r1 = 0\n",
    "max_r0 = 0\n",
    "\n",
    "for last_layer in population:\n",
    "    outputs0 = f0 @ last_layer\n",
    "    outputs1 = f1 @ last_layer\n",
    "\n",
    "    # get preferences for each hidden context group\n",
    "    p0 = (outputs1 > outputs0).long()\n",
    "    p1 = (outputs1 > outputs0).long()\n",
    "\n",
    "    r1 = (p1 == pp1).float()\n",
    "    r0 = (p0 == pp0).float()\n",
    "\n",
    "    print(f\"r1: {r1.mean()}, r0: {r0.mean()}\")\n",
    "\n",
    "    max_r1 = max(max_r1, r1.max())\n",
    "    max_r0 = max(max_r0, r0.max())\n",
    "\n",
    "\n",
    "print(max_r0)\n",
    "print(max_r1)\n"
   ]
  },
  {
   "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.10.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
