{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "d4608677",
   "metadata": {},
   "source": [
    "To put as the very first image\n",
    "\n",
    "An image of linear programming, which should be later editted by the \n",
    "\n",
    "# Runner"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "04e99424",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "\n",
    "lam_list = np.linspace(0.01, 1, 100)\n",
    "alpha = 0.05"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "80475b88",
   "metadata": {},
   "outputs": [],
   "source": [
    "from model.v1.v1.sampler import Sampler\n",
    "import numpy as np\n",
    "from model.v1.crc import ConformalRegretControl\n",
    "from utils.setting import config\n",
    "import matplotlib.pyplot as plt\n",
    "from tqdm import tqdm\n",
    "\n",
    "def vis_1_runner(crc : ConformalRegretControl, mean, sigma, nsample, ntrials = 10, verbose = True, B = 1,\n",
    "                   width = None, use_gaussian = False   # 9/30 update\n",
    "                   ):\n",
    "    # CREME true\n",
    "    true_list = []\n",
    "    x = Sampler().uniform(mean, sigma, 100) if not use_gaussian else Sampler().gaussian(mean, width, sigma, 100)\n",
    "    y = Sampler().uniform(mean, sigma, 100) if not use_gaussian else Sampler().gaussian(mean, width, sigma, 100) # [ nsample, 2 ]\n",
    "    pbar = tqdm(total = len(lam_list), desc = 'True') if verbose else None\n",
    "    for lam in lam_list:\n",
    "        pbar.update(1) if verbose else None\n",
    "        _, _, alpha_R_true, alpha_I_true = crc.estimate(x, y, lam=lam, B=B, also_mc=True)   # all scalars\n",
    "        true_list.append(np.array([alpha_R_true, alpha_I_true]))\n",
    "    true_list = np.stack(true_list, axis=0)             # [ nlams, 2 ]\n",
    "    true_list = true_list.reshape(len(lam_list), -1)    # [ nlams, 2 ]\n",
    "\n",
    "    # CREME estimate and the associated baselines\n",
    "    estimate_list = []\n",
    "    baseline_list = []\n",
    "    pbar = tqdm(total = ntrials * len(lam_list), desc = 'Estimate') if verbose else None\n",
    "    for i in range(ntrials):\n",
    "\n",
    "        x = Sampler().uniform(mean, sigma, nsample) if not use_gaussian else Sampler().gaussian(mean, width, sigma, nsample)\n",
    "        y = Sampler().uniform(mean, sigma, nsample) if not use_gaussian else Sampler().gaussian(mean, width, sigma, nsample) # [ nsample, 2 ]\n",
    "\n",
    "        model       = crc.optimzation.model\n",
    "        # baselines   = MiscoverageToLam(x, y, model)\n",
    "\n",
    "        # lam_cp  = baselines.get_conformal_lam(alpha)\n",
    "        # lam_q   = baselines.get_quantile_lam(alpha)\n",
    "        # lam_g   = baselines.get_gaussian_lam(alpha)\n",
    "        # baseline_list.append(np.array([lam_g, lam_q, lam_cp]))\n",
    "\n",
    "        for lam in lam_list:\n",
    "            pbar.update(1) if verbose else None\n",
    "            alpha_R, alpha_I, alpha_R_mc, alpha_I_mc = crc.estimate(x, y, lam=lam, B=B, also_mc=True)   # all scalars\n",
    "            estimate_list.append(np.array([alpha_R, alpha_I, alpha_R_mc, alpha_I_mc]))\n",
    "    estimate_list = np.stack(estimate_list, axis=0)                     # [ ntrials * nlams, 4 ]\n",
    "    estimate_list = estimate_list.reshape(ntrials, len(lam_list), -1)   # [ ntrials, nlams, 4 ]\n",
    "    # pbar.close() if verbose else None\n",
    "    estimate_list = estimate_list.mean(0)   # [ nlams, 4 ] NOTE: multi-shot estimator\n",
    "    estimate_list = estimate_list[:, :2]    \n",
    "\n",
    "    # baseline_list = np.stack(baseline_list, axis=0) # [ ntrial, 3 ] np\n",
    "    # baseline_list = baseline_list.mean(0)   # [ 3 ] np\n",
    "\n",
    "    # both [ nlams, 2 ] NOTE: multi-shot estimator unless ntrails=1\n",
    "    # return true_list, estimate_list, baseline_list                  \n",
    "    return np.concatenate([true_list,\n",
    "                           estimate_list\n",
    "                        #    baseline_list[None, :] * np.ones([len(true_list), 3])\n",
    "                           ],\n",
    "                           axis=1)\n",
    "\n",
    "def vis_1_plotter(ax, arr, add_label = True, s = 100, add_dots = True, w = np.array([-1., -1.])):\n",
    "\n",
    "    true_list = arr[:, [0, 1]]\n",
    "    estimate_list = arr[:, [2, 3]]\n",
    "\n",
    "    coord_list = [true_list, estimate_list][::-1]\n",
    "    color_list = ['black', 'red'][::-1]\n",
    "    label_list = [r'True Pareto frontier $(\\alpha_I(\\lambda), \\alpha_R(\\lambda))$', r'Estimated Pareto frontier $(\\hat \\alpha_I(\\lambda), \\hat \\alpha_R(\\lambda))$ via \\texttt{CREME}'][::-1]\n",
    "\n",
    "    # Plotting the Pareto frontiers\n",
    "    for i in range(2):\n",
    "        ax.plot(coord_list[i][:, 1], coord_list[i][:, 0],\n",
    "                color = color_list[i],\n",
    "                label = label_list[i] if add_label else None)\n",
    "    ylow, ylim    = ax.get_ylim()\n",
    "    for i in range(2):\n",
    "        ax.fill_between(coord_list[i][:, 1],\n",
    "                        coord_list[i][:, 0],\n",
    "                        ylim,\n",
    "                        color = color_list[i],\n",
    "                        alpha = 0.1,\n",
    "                        lw = 0)\n",
    "    \n",
    "    # Plotting the estimate and true\n",
    "    if add_dots:\n",
    "        # ------- estimate --------\n",
    "        y = np.inner(w.reshape(1, 2), estimate_list)    # [ 1, num_lam ] np\n",
    "        index = y.argmax(1).item()\n",
    "        ax.scatter(true_list[index, 1], true_list[index, 0], s = s / 3, color = color_list[0], zorder = 99)\n",
    "        # ax.scatter(estimate_list[index, 1], estimate_list[index, 0], s = s / 2, edgecolor = color_list[0], facecolor = 'none')\n",
    "        ax.plot(\n",
    "            [true_list[index, 1], estimate_list[index, 1]],\n",
    "            [true_list[index, 0], estimate_list[index, 0]],\n",
    "            color = color_list[0], ls = 'dotted', lw = 2\n",
    "        )\n",
    "        # ------- true --------\n",
    "        y = np.inner(w.reshape(1, 2), true_list)    # [ 1, num_lam ] np\n",
    "        index = y.argmax(1).item()\n",
    "        ax.scatter(true_list[index, 1], true_list[index, 0], s = s + 80, color = color_list[1], marker = '*')\n",
    "\n",
    "    return ax"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "e0cf57e9",
   "metadata": {},
   "source": [
    "# Exp"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f1b7ecd5",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from model.v1.predictor import OffsetScalarPredictor\n",
    "from model.v1.crc import ConformalRegretControl\n",
    "from model.v1.newsvendor import Newsvendor\n",
    "from model.v1.lp import LinearProgramming, circle_as_polytope\n",
    "from model.v1.shortest_path import ShortestPath, build_incidence_many_chains\n",
    "from model.v1.portfolio import PortfolioOptimization\n",
    "from utils.setting import config"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "95ad2bd1",
   "metadata": {},
   "outputs": [],
   "source": [
    "mean    = [-1.1, -1.]\n",
    "sigma   = [1., 1.]\n",
    "\n",
    "model_kwds = {'offset': mean}\n",
    "model = OffsetScalarPredictor(**model_kwds)\n",
    "A, b = circle_as_polytope(R=1.0, m=128)\n",
    "A_pos = np.array([\n",
    "    [-1., 0.],\n",
    "    [0., -1.]\n",
    "])\n",
    "b_pos = np.array([0., 0.])\n",
    "A, b = np.concatenate([A, A_pos]), np.concatenate([b, b_pos])\n",
    "\n",
    "opt_kwds = {\n",
    "    'model':    model,\n",
    "    'feasible_region':  None,    # [ nres**2, 2 ]\n",
    "    'outcome_space':    None,    # [ nres**2, 2 ]\n",
    "    'A':    A,\n",
    "    'b':    b,\n",
    "}\n",
    "optimization = LinearProgramming(**opt_kwds)\n",
    "\n",
    "crc_kwds = {'optimization': optimization}\n",
    "crc = ConformalRegretControl(**crc_kwds)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "23cc9432",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "True: 100%|██████████| 100/100 [00:24<00:00,  4.09it/s]\n",
      "True: 100%|██████████| 100/100 [00:24<00:00,  4.08it/s]\n",
      "\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "\u001b[A\n",
      "Estimate: 100%|██████████| 100/100 [00:06<00:00, 15.73it/s]\n"
     ]
    }
   ],
   "source": [
    "import os\n",
    "\n",
    "config()\n",
    "\n",
    "exp_kwds = {\n",
    "    'crc':      crc,\n",
    "    'mean':     mean,\n",
    "    'sigma':    sigma,\n",
    "    'ntrials':  1,\n",
    "    'nsample':  10,\n",
    "    'verbose':  True,\n",
    "    'B':        0.8,\n",
    "    'width':    None,\n",
    "    'use_gaussian': False\n",
    "}\n",
    "path = 'cache/10_1/result_list.npy'\n",
    "\n",
    "if os.path.exists(path) and True:\n",
    "    pass\n",
    "else:\n",
    "    result_list = vis_1_runner(**exp_kwds)\n",
    "    np.save(path, result_list)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 39,
   "id": "3a33ba61",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plt.figure(figsize=[4, 3])\n",
    "\n",
    "arr = np.load(path)\n",
    "ax = vis_1_plotter(plt.gca(), arr, w = np.array([-1., -0.8]), add_dots=False)\n",
    "ax.axis('off')\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "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.12.3"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
