{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import wandb\n",
    "from tqdm import tqdm\n",
    "import numpy as np\n",
    "\n",
    "api = wandb.Api()\n",
    "\n",
    "def load_synthetic_variable_cost_runs(acq, tfn, dim):\n",
    "    runs = api.runs(path=\"ziv-scully-group/PandoraBayesOPT\", filters={\n",
    "        \"sweep\": \"iwfczebd\",\n",
    "        \"config.dim\": dim, \n",
    "        \"config.problem\": tfn,\n",
    "        \"config.policy\": acq})\n",
    "    \n",
    "    configs_and_metrics = []\n",
    "    for run in tqdm(runs):\n",
    "        metric_keys = [\"cumulative cost\",\"best observed\"]\n",
    "        history = run.scan_history(keys = metric_keys, page_size=1_000_000_000)\n",
    "        metrics = {k: [d[k] for d in history] for k in metric_keys}\n",
    "        summary_metric_keys = [\"global optimum value\"]\n",
    "        summary_history = run.scan_history(keys = summary_metric_keys, page_size=1_000_000_000)\n",
    "        metrics.update({k: [d[k] for d in summary_history] for k in summary_metric_keys})\n",
    "        configs_and_metrics.append((run.config, metrics))\n",
    "\n",
    "    return configs_and_metrics"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "acquisition_functions = {\n",
    "      'RandomSearch': 'RandomSearch',\n",
    "      'ExpectedImprovementWithoutCost':'ExpectedImprovementWithoutCost',\n",
    "      'ExpectedImprovementPerUnitCost':'ExpectedImprovementPerUnitCost',\n",
    "      'ExpectedImprovementWithCostCooling':'ExpectedImprovementWithCostCooling',\n",
    "      'BudgetedMultiStepLookaheadEI': 'BudgetedMultiStepLookaheadEI',\n",
    "      'Gittins_Lambda_01':'Gittins_Lambda01',\n",
    "      'Gittins_Lambda_001':'Gittins_Lambda001',\n",
    "      'Gittins_Lambda_0001':'Gittins_Lambda0001',\n",
    "      'Gittins_Step_Divide2':'Gittins_Step_Divide2',\n",
    "      'Gittins_Step_Divide5':'Gittins_Step_Divide5',\n",
    "      'Gittins_Step_Divide10':'Gittins_Step_Divide10',\n",
    "      'Gittins_Step_EIpu':'Gittins_Step_EIPC',\n",
    "    }\n",
    "target_functions_with_optima = {\n",
    "    'Ackley':0,\n",
    "    # 'DropWave':-1,\n",
    "    # 'Shekel5':-10.1532,\n",
    "    'Rosenbrock':0,\n",
    "    'Levy':0,\n",
    "  }\n",
    "dimensions = [4, 8, 16]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.10s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:20<00:00,  1.27s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:19<00:00,  1.23s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.06s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:20<00:00,  1.25s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.09s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.05s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.13s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.14s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.02s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.03s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:20<00:00,  1.29s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.07s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.18s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.15s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.12s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.09s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.09s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.03s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.11s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:20<00:00,  1.30s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.15s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.11s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.13s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.12s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.07s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.07s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.09s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.05s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.13s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.07s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.12s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.15s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.09s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.11s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.09s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.05s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:12<00:00,  1.26it/s]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:19<00:00,  1.20s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:13<00:00,  1.15it/s]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.06s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:20<00:00,  1.28s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.02s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:13<00:00,  1.16it/s]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.14s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.07s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:20<00:00,  1.26s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.10s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:21<00:00,  1.36s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:22<00:00,  1.39s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:22<00:00,  1.42s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.09s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.09s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.09s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.13s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.08s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.11s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.19s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.03s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.16s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:21<00:00,  1.36s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:19<00:00,  1.21s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:21<00:00,  1.36s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.09s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.13s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.12s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.11s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.13s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.07s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.14s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:19<00:00,  1.22s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.15s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.06s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:20<00:00,  1.28s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.16s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.10s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.09s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.14s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.06s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:21<00:00,  1.33s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.14s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.05s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.08s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:19<00:00,  1.20s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.05s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.17s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.17s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.08s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.07s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.14s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.08s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:12<00:00,  1.30it/s]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.05s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.07s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.09s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:19<00:00,  1.24s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.06s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:12<00:00,  1.29it/s]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:19<00:00,  1.20s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.09s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.16s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:18<00:00,  1.18s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:19<00:00,  1.21s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:22<00:00,  1.39s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:20<00:00,  1.27s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:16<00:00,  1.05s/it]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:14<00:00,  1.09it/s]\n",
      "100%|███████████████████████████████████████████| 16/16 [00:17<00:00,  1.10s/it]\n"
     ]
    }
   ],
   "source": [
    "grouped_runs = {(a,t,d): load_synthetic_variable_cost_runs(a,t,d) for a in acquisition_functions.keys() for t in target_functions_with_optima.keys() for d in (dimensions)}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "RandomSearch Ackley 4 (101, 16)\n",
      "RandomSearch Ackley 8 (201, 16)\n",
      "RandomSearch Ackley 16 (401, 16)\n",
      "RandomSearch Rosenbrock 4 (101, 16)\n",
      "RandomSearch Rosenbrock 8 (201, 16)\n",
      "RandomSearch Rosenbrock 16 (401, 16)\n",
      "RandomSearch Levy 4 (101, 16)\n",
      "RandomSearch Levy 8 (201, 16)\n",
      "RandomSearch Levy 16 (401, 16)\n",
      "ExpectedImprovementWithoutCost Ackley 4 (101, 16)\n",
      "ExpectedImprovementWithoutCost Ackley 8 (201, 16)\n",
      "ExpectedImprovementWithoutCost Ackley 16 (401, 16)\n",
      "ExpectedImprovementWithoutCost Rosenbrock 4 (101, 16)\n",
      "ExpectedImprovementWithoutCost Rosenbrock 8 (201, 16)\n",
      "ExpectedImprovementWithoutCost Rosenbrock 16 (401, 16)\n",
      "ExpectedImprovementWithoutCost Levy 4 (101, 16)\n",
      "ExpectedImprovementWithoutCost Levy 8 (201, 16)\n",
      "ExpectedImprovementWithoutCost Levy 16 (401, 16)\n",
      "ExpectedImprovementPerUnitCost Ackley 4 (101, 16)\n",
      "ExpectedImprovementPerUnitCost Ackley 8 (201, 16)\n",
      "ExpectedImprovementPerUnitCost Ackley 16 (401, 16)\n",
      "ExpectedImprovementPerUnitCost Rosenbrock 4 (101, 16)\n",
      "ExpectedImprovementPerUnitCost Rosenbrock 8 (201, 16)\n",
      "ExpectedImprovementPerUnitCost Rosenbrock 16 (401, 16)\n",
      "ExpectedImprovementPerUnitCost Levy 4 (101, 16)\n",
      "ExpectedImprovementPerUnitCost Levy 8 (201, 16)\n",
      "ExpectedImprovementPerUnitCost Levy 16 (401, 16)\n",
      "ExpectedImprovementWithCostCooling Ackley 4 (101, 16)\n",
      "ExpectedImprovementWithCostCooling Ackley 8 (201, 16)\n",
      "ExpectedImprovementWithCostCooling Ackley 16 (401, 16)\n",
      "ExpectedImprovementWithCostCooling Rosenbrock 4 (101, 16)\n",
      "ExpectedImprovementWithCostCooling Rosenbrock 8 (201, 16)\n",
      "ExpectedImprovementWithCostCooling Rosenbrock 16 (401, 16)\n",
      "ExpectedImprovementWithCostCooling Levy 4 (101, 16)\n",
      "ExpectedImprovementWithCostCooling Levy 8 (201, 16)\n",
      "ExpectedImprovementWithCostCooling Levy 16 (401, 16)\n",
      "BudgetedMultiStepLookaheadEI Ackley 4 (101, 16)\n",
      "BudgetedMultiStepLookaheadEI Ackley 8 (201, 16)\n",
      "BudgetedMultiStepLookaheadEI Ackley 16 (401, 16)\n",
      "BudgetedMultiStepLookaheadEI Rosenbrock 4 (101, 16)\n",
      "BudgetedMultiStepLookaheadEI Rosenbrock 8 (201, 16)\n",
      "BudgetedMultiStepLookaheadEI Rosenbrock 16 (401, 16)\n",
      "BudgetedMultiStepLookaheadEI Levy 4 (101, 16)\n",
      "BudgetedMultiStepLookaheadEI Levy 8 (201, 16)\n",
      "BudgetedMultiStepLookaheadEI Levy 16 (401, 16)\n",
      "Gittins_Lambda_01 Ackley 4 (101, 16)\n",
      "Gittins_Lambda_01 Ackley 8 (201, 16)\n",
      "Gittins_Lambda_01 Ackley 16 (401, 16)\n",
      "Gittins_Lambda_01 Rosenbrock 4 (101, 16)\n",
      "Gittins_Lambda_01 Rosenbrock 8 (201, 16)\n",
      "Gittins_Lambda_01 Rosenbrock 16 (401, 16)\n",
      "Gittins_Lambda_01 Levy 4 (101, 16)\n",
      "Gittins_Lambda_01 Levy 8 (201, 16)\n",
      "Gittins_Lambda_01 Levy 16 (401, 16)\n",
      "Gittins_Lambda_001 Ackley 4 (101, 16)\n",
      "Gittins_Lambda_001 Ackley 8 (201, 16)\n",
      "Gittins_Lambda_001 Ackley 16 (401, 16)\n",
      "Gittins_Lambda_001 Rosenbrock 4 (101, 16)\n",
      "Gittins_Lambda_001 Rosenbrock 8 (201, 16)\n",
      "Gittins_Lambda_001 Rosenbrock 16 (401, 16)\n",
      "Gittins_Lambda_001 Levy 4 (101, 16)\n",
      "Gittins_Lambda_001 Levy 8 (201, 16)\n",
      "Gittins_Lambda_001 Levy 16 (401, 16)\n",
      "Gittins_Lambda_0001 Ackley 4 (101, 16)\n",
      "Gittins_Lambda_0001 Ackley 8 (201, 16)\n",
      "Gittins_Lambda_0001 Ackley 16 (401, 16)\n",
      "Gittins_Lambda_0001 Rosenbrock 4 (101, 16)\n",
      "Gittins_Lambda_0001 Rosenbrock 8 (201, 16)\n",
      "Gittins_Lambda_0001 Rosenbrock 16 (401, 16)\n",
      "Gittins_Lambda_0001 Levy 4 (101, 16)\n",
      "Gittins_Lambda_0001 Levy 8 (201, 16)\n",
      "Gittins_Lambda_0001 Levy 16 (401, 16)\n",
      "Gittins_Step_Divide2 Ackley 4 (101, 16)\n",
      "Gittins_Step_Divide2 Ackley 8 (201, 16)\n",
      "Gittins_Step_Divide2 Ackley 16 (401, 16)\n",
      "Gittins_Step_Divide2 Rosenbrock 4 (101, 16)\n",
      "Gittins_Step_Divide2 Rosenbrock 8 (201, 16)\n",
      "Gittins_Step_Divide2 Rosenbrock 16 (401, 16)\n",
      "Gittins_Step_Divide2 Levy 4 (101, 16)\n",
      "Gittins_Step_Divide2 Levy 8 (201, 16)\n",
      "Gittins_Step_Divide2 Levy 16 (401, 16)\n",
      "Gittins_Step_Divide5 Ackley 4 (101, 16)\n",
      "Gittins_Step_Divide5 Ackley 8 (201, 16)\n",
      "Gittins_Step_Divide5 Ackley 16 (401, 16)\n",
      "Gittins_Step_Divide5 Rosenbrock 4 (101, 16)\n",
      "Gittins_Step_Divide5 Rosenbrock 8 (201, 16)\n",
      "Gittins_Step_Divide5 Rosenbrock 16 (401, 16)\n",
      "Gittins_Step_Divide5 Levy 4 (101, 16)\n",
      "Gittins_Step_Divide5 Levy 8 (201, 16)\n",
      "Gittins_Step_Divide5 Levy 16 (401, 16)\n",
      "Gittins_Step_Divide10 Ackley 4 (101, 16)\n",
      "Gittins_Step_Divide10 Ackley 8 (201, 16)\n",
      "Gittins_Step_Divide10 Ackley 16 (401, 16)\n",
      "Gittins_Step_Divide10 Rosenbrock 4 (101, 16)\n",
      "Gittins_Step_Divide10 Rosenbrock 8 (201, 16)\n",
      "Gittins_Step_Divide10 Rosenbrock 16 (401, 16)\n",
      "Gittins_Step_Divide10 Levy 4 (101, 16)\n",
      "Gittins_Step_Divide10 Levy 8 (201, 16)\n",
      "Gittins_Step_Divide10 Levy 16 (401, 16)\n",
      "Gittins_Step_EIpu Ackley 4 (101, 16)\n",
      "Gittins_Step_EIpu Ackley 8 (201, 16)\n",
      "Gittins_Step_EIpu Ackley 16 (401, 16)\n",
      "Gittins_Step_EIpu Rosenbrock 4 (101, 16)\n",
      "Gittins_Step_EIpu Rosenbrock 8 (201, 16)\n",
      "Gittins_Step_EIpu Rosenbrock 16 (401, 16)\n",
      "Gittins_Step_EIpu Levy 4 (101, 16)\n",
      "Gittins_Step_EIpu Levy 8 (201, 16)\n",
      "Gittins_Step_EIpu Levy 16 (401, 16)\n"
     ]
    }
   ],
   "source": [
    "for a in acquisition_functions.keys():\n",
    "    for (t,opt) in target_functions_with_optima.items():\n",
    "        for d in dimensions:\n",
    "            config_and_metrics_per_seed = grouped_runs[a,t,d]\n",
    "    \n",
    "            cumulative_cost_per_seed = np.array([m['cumulative cost'] for (_,m) in config_and_metrics_per_seed]).T\n",
    "            best_observed_per_seed = np.array([m['best observed'] for (_,m) in config_and_metrics_per_seed]).T\n",
    "    \n",
    "            regret_per_seed = best_observed_per_seed - opt\n",
    "    \n",
    "            print(a, t, d, regret_per_seed.shape)\n",
    "    \n",
    "            regret_25 = np.quantile(regret_per_seed, 0.25, axis=1)\n",
    "            regret_50 = np.quantile(regret_per_seed, 0.5, axis=1)\n",
    "            regret_75 = np.quantile(regret_per_seed, 0.75, axis=1)\n",
    "    \n",
    "            output = np.stack((cumulative_cost_per_seed.mean(axis=1), regret_25, regret_50, regret_75),axis=-1)\n",
    "\n",
    "            np.savetxt(f\"results/synthetic_cost_aware/Synthetic_VariableCost_{t}_d{d}_{acquisition_functions[a]}.csv\", output, header=\"cc, q25, q50, q75\", delimiter=', ', comments='')"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.9.18"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
