{"cells":[{"cell_type":"markdown","source":[" # A simple test of using `optuna` for hyperparameter optimization"],"metadata":{}},{"cell_type":"code","execution_count":1,"source":["import time\n","from types import SimpleNamespace\n","\n","import pydove as dv\n","from tqdm.notebook import tqdm\n","\n","import torch\n","from cpcn import LinearCPCNetwork, load_mnist, train\n","\n","import optuna\n","from optuna.trial import TrialState"],"outputs":[],"metadata":{}},{"cell_type":"markdown","source":[" ## Defining the optimization"],"metadata":{}},{"cell_type":"code","execution_count":2,"source":["def optuna_reporter(trial: optuna.trial.Trial, ns: SimpleNamespace):\n","    trial.report(ns.epoch_val_loss, ns.epoch)\n","\n","    # early pruning\n","    if trial.should_prune():\n","        raise optuna.exceptions.TrialPruned()\n","\n","\n","def create_cpcn(trial):\n","    n_hidden = trial.suggest_int(\"n_hidden\", 1, 3)\n","    dims = [28 * 28]\n","    for i in range(n_hidden):\n","        n_units = trial.suggest_int(f\"n_units_l{i}\", 4, 128)\n","        dims.append(n_units)\n","    dims.append(10)\n","\n","    z_lr = trial.suggest_float(\"z_lr\", 1e-5, 1e-1, log=True)\n","\n","    # set parameters to match a simple PCN network\n","    g_a = 0.5 * torch.ones(len(dims) - 2)\n","    g_a[-1] *= 2\n","\n","    g_b = 0.5 * torch.ones(len(dims) - 2)\n","    g_b[0] *= 2\n","\n","    net = LinearCPCNetwork(dims, z_lr=z_lr, z_it=50, g_a=g_a, g_b=g_b, c_m=0, l_s=g_b)\n","\n","    return net\n","\n","\n","def objective(\n","    trial: optuna.trial.Trial, n_epochs: int, dataset: dict, device: torch.device\n",") -> float:\n","    net = create_cpcn(trial).to(device)\n","\n","    optimizer_type = trial.suggest_categorical(\"optimizer\", [\"Adam\", \"RMSprop\", \"SGD\"])\n","    lr = trial.suggest_float(\"lr\", 1e-5, 1e-1, log=True)\n","\n","    optimizer_class = getattr(torch.optim, optimizer_type)\n","    optimizer_kws = {\"lr\": lr}\n","\n","    results = train(\n","        net,\n","        n_epochs,\n","        dataset[\"train\"],\n","        dataset[\"validation\"],\n","        optimizer=optimizer_class,\n","        optimizer_kwargs=optimizer_kws,\n","        reporter=lambda ns: optuna_reporter(trial, ns),\n","    )\n","\n","    return results.validation.pc_loss[-1]"],"outputs":[],"metadata":{}},{"cell_type":"code","execution_count":3,"source":["# minimizing PC loss\n","t0 = time.time()\n","\n","device = torch.device(\"cpu\")\n","\n","n_epochs = 15\n","dataset = load_mnist(n_train=2000, n_validation=1000, batch_size=128)\n","\n","study = optuna.create_study(direction=\"minimize\")\n","study.optimize(\n","    lambda trial: objective(trial, n_epochs, dataset, device),\n","    n_trials=100,\n","    timeout=600,\n","    show_progress_bar=True,\n",")\n","\n","pruned_trials = study.get_trials(deepcopy=False, states=[TrialState.PRUNED])\n","complete_trials = study.get_trials(deepcopy=False, states=[TrialState.COMPLETE])\n","\n","t1 = time.time()"],"outputs":[{"output_type":"stream","name":"stderr","text":["\u001b[32m[I 2022-04-19 18:10:00,494]\u001b[0m A new study created in memory with name: no-name-8eb86d88-ab61-43ee-beb8-34533f70341c\u001b[0m\n","/mnt/home/ttesileanu/miniconda3/envs/cpcn/lib/python3.9/site-packages/optuna/progress_bar.py:47: ExperimentalWarning: Progress bar is experimental (supported from v1.2.0). The interface can change in the future.\n","  self._init_valid()\n"]},{"output_type":"display_data","data":{"application/vnd.jupyter.widget-view+json":{"version_major":2,"version_minor":0,"model_id":"2a28886af51542d19f7a42e3deacd18c"},"text/plain":["  0%|          | 0/100 [00:00<?, ?it/s]"]},"metadata":{}},{"output_type":"stream","name":"stdout","text":["\u001b[32m[I 2022-04-19 18:10:06,735]\u001b[0m Trial 0 finished with value: 234.62002754211426 and parameters: {'n_hidden': 1, 'n_units_l0': 68, 'z_lr': 1.396572984186087e-05, 'optimizer': 'RMSprop', 'lr': 0.0002616989298218379}. Best is trial 0 with value: 234.62002754211426.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:10:12,841]\u001b[0m Trial 1 finished with value: 58.0662145614624 and parameters: {'n_hidden': 1, 'n_units_l0': 80, 'z_lr': 4.552045845430424e-05, 'optimizer': 'Adam', 'lr': 0.000898825179763433}. Best is trial 1 with value: 58.0662145614624.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:10:26,615]\u001b[0m Trial 2 finished with value: 1344148960.0 and parameters: {'n_hidden': 3, 'n_units_l0': 38, 'n_units_l1': 120, 'n_units_l2': 42, 'z_lr': 0.005224047887963055, 'optimizer': 'Adam', 'lr': 0.05517062145129547}. Best is trial 1 with value: 58.0662145614624.\u001b[0m\n","\u001b[33m[W 2022-04-19 18:10:34,611]\u001b[0m Trial 3 failed, because the objective function returned nan.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:10:43,363]\u001b[0m Trial 4 finished with value: 1812.1015625 and parameters: {'n_hidden': 2, 'n_units_l0': 63, 'n_units_l1': 58, 'z_lr': 0.045887148620924155, 'optimizer': 'SGD', 'lr': 1.178580237546145e-05}. Best is trial 1 with value: 58.0662145614624.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:00,238]\u001b[0m Trial 5 finished with value: 17132.053588867188 and parameters: {'n_hidden': 3, 'n_units_l0': 128, 'n_units_l1': 126, 'n_units_l2': 58, 'z_lr': 0.08960494926589964, 'optimizer': 'Adam', 'lr': 0.0011358679528768363}. Best is trial 1 with value: 58.0662145614624.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:01,123]\u001b[0m Trial 6 pruned. \u001b[0m\n"]},{"output_type":"stream","name":"stderr","text":["/mnt/home/ttesileanu/miniconda3/envs/cpcn/lib/python3.9/site-packages/optuna/pruners/_percentile.py:21: RuntimeWarning: All-NaN slice encountered\n","  return np.nanmin(values)\n"]},{"output_type":"stream","name":"stdout","text":["\u001b[32m[I 2022-04-19 18:11:01,780]\u001b[0m Trial 7 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:08,876]\u001b[0m Trial 8 finished with value: 121.25624752044678 and parameters: {'n_hidden': 1, 'n_units_l0': 90, 'z_lr': 0.05695565558243151, 'optimizer': 'Adam', 'lr': 0.000524975811380883}. Best is trial 1 with value: 58.0662145614624.\u001b[0m\n"]},{"output_type":"stream","name":"stderr","text":["/mnt/home/ttesileanu/miniconda3/envs/cpcn/lib/python3.9/site-packages/optuna/pruners/_percentile.py:21: RuntimeWarning: All-NaN slice encountered\n","  return np.nanmin(values)\n"]},{"output_type":"stream","name":"stdout","text":["\u001b[32m[I 2022-04-19 18:11:09,348]\u001b[0m Trial 9 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:10,299]\u001b[0m Trial 10 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:13,687]\u001b[0m Trial 11 finished with value: 495.53832244873047 and parameters: {'n_hidden': 1, 'n_units_l0': 5, 'z_lr': 0.00012652542879170944, 'optimizer': 'Adam', 'lr': 4.4308815579826314e-05}. Best is trial 1 with value: 58.0662145614624.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:15,094]\u001b[0m Trial 12 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:15,550]\u001b[0m Trial 13 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:16,966]\u001b[0m Trial 14 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:22,171]\u001b[0m Trial 15 finished with value: 3678.4972229003906 and parameters: {'n_hidden': 1, 'n_units_l0': 46, 'z_lr': 0.0005090941651880445, 'optimizer': 'Adam', 'lr': 0.006665633203155863}. Best is trial 1 with value: 58.0662145614624.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:22,878]\u001b[0m Trial 16 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:23,463]\u001b[0m Trial 17 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:24,153]\u001b[0m Trial 18 pruned. \u001b[0m\n"]},{"output_type":"stream","name":"stderr","text":["/mnt/home/ttesileanu/miniconda3/envs/cpcn/lib/python3.9/site-packages/optuna/pruners/_percentile.py:21: RuntimeWarning: All-NaN slice encountered\n","  return np.nanmin(values)\n"]},{"output_type":"stream","name":"stdout","text":["\u001b[32m[I 2022-04-19 18:11:24,420]\u001b[0m Trial 19 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:32,001]\u001b[0m Trial 20 finished with value: 77.82842636108398 and parameters: {'n_hidden': 1, 'n_units_l0': 111, 'z_lr': 0.009478371843289975, 'optimizer': 'RMSprop', 'lr': 0.00035958561849502065}. Best is trial 1 with value: 58.0662145614624.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:32,707]\u001b[0m Trial 21 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:38,903]\u001b[0m Trial 22 finished with value: 108.68520259857178 and parameters: {'n_hidden': 1, 'n_units_l0': 82, 'z_lr': 0.002605670393026579, 'optimizer': 'RMSprop', 'lr': 0.0003477628120123441}. Best is trial 1 with value: 58.0662145614624.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:47,213]\u001b[0m Trial 23 finished with value: 77.54340934753418 and parameters: {'n_hidden': 1, 'n_units_l0': 125, 'z_lr': 0.002004449536006061, 'optimizer': 'RMSprop', 'lr': 0.00036378280721760834}. Best is trial 1 with value: 58.0662145614624.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:47,832]\u001b[0m Trial 24 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:55,870]\u001b[0m Trial 25 finished with value: 45.328901290893555 and parameters: {'n_hidden': 1, 'n_units_l0': 105, 'z_lr': 0.011500646684311623, 'optimizer': 'RMSprop', 'lr': 0.0013288501676211343}. Best is trial 25 with value: 45.328901290893555.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:56,979]\u001b[0m Trial 26 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:57,783]\u001b[0m Trial 27 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:58,312]\u001b[0m Trial 28 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:58,801]\u001b[0m Trial 29 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:59,184]\u001b[0m Trial 30 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:11:59,744]\u001b[0m Trial 31 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:07,211]\u001b[0m Trial 32 finished with value: 111.58461570739746 and parameters: {'n_hidden': 1, 'n_units_l0': 110, 'z_lr': 0.01049132732981028, 'optimizer': 'RMSprop', 'lr': 0.00031619991777505}. Best is trial 25 with value: 45.328901290893555.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:15,009]\u001b[0m Trial 33 finished with value: 235.09947776794434 and parameters: {'n_hidden': 1, 'n_units_l0': 119, 'z_lr': 0.006185149481105883, 'optimizer': 'RMSprop', 'lr': 0.001355057149030045}. Best is trial 25 with value: 45.328901290893555.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:22,392]\u001b[0m Trial 34 finished with value: 75.0975890159607 and parameters: {'n_hidden': 1, 'n_units_l0': 104, 'z_lr': 0.0016109865151488355, 'optimizer': 'RMSprop', 'lr': 0.00040556375595796756}. Best is trial 25 with value: 45.328901290893555.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:28,569]\u001b[0m Trial 35 finished with value: 54.112882137298584 and parameters: {'n_hidden': 1, 'n_units_l0': 79, 'z_lr': 0.0018107099261400331, 'optimizer': 'RMSprop', 'lr': 0.0007075889626924338}. Best is trial 25 with value: 45.328901290893555.\u001b[0m\n"]},{"output_type":"stream","name":"stderr","text":["/mnt/home/ttesileanu/miniconda3/envs/cpcn/lib/python3.9/site-packages/optuna/pruners/_percentile.py:21: RuntimeWarning: All-NaN slice encountered\n","  return np.nanmin(values)\n"]},{"output_type":"stream","name":"stdout","text":["\u001b[32m[I 2022-04-19 18:12:28,985]\u001b[0m Trial 36 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:29,912]\u001b[0m Trial 37 pruned. \u001b[0m\n"]},{"output_type":"stream","name":"stderr","text":["/mnt/home/ttesileanu/miniconda3/envs/cpcn/lib/python3.9/site-packages/optuna/pruners/_percentile.py:21: RuntimeWarning: All-NaN slice encountered\n","  return np.nanmin(values)\n"]},{"output_type":"stream","name":"stdout","text":["\u001b[32m[I 2022-04-19 18:12:30,880]\u001b[0m Trial 38 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:31,245]\u001b[0m Trial 39 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:32,029]\u001b[0m Trial 40 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:32,421]\u001b[0m Trial 41 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:40,239]\u001b[0m Trial 42 finished with value: 52.78911113739014 and parameters: {'n_hidden': 1, 'n_units_l0': 123, 'z_lr': 0.001684841256441361, 'optimizer': 'RMSprop', 'lr': 0.0004733508834503891}. Best is trial 25 with value: 45.328901290893555.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:47,435]\u001b[0m Trial 43 finished with value: 57.99083995819092 and parameters: {'n_hidden': 1, 'n_units_l0': 97, 'z_lr': 0.0015518259212979535, 'optimizer': 'RMSprop', 'lr': 0.0005068170739433618}. Best is trial 25 with value: 45.328901290893555.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:48,305]\u001b[0m Trial 44 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:48,776]\u001b[0m Trial 45 pruned. \u001b[0m\n"]},{"output_type":"stream","name":"stderr","text":["/mnt/home/ttesileanu/miniconda3/envs/cpcn/lib/python3.9/site-packages/optuna/pruners/_percentile.py:21: RuntimeWarning: All-NaN slice encountered\n","  return np.nanmin(values)\n"]},{"output_type":"stream","name":"stdout","text":["\u001b[32m[I 2022-04-19 18:12:49,181]\u001b[0m Trial 46 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:49,577]\u001b[0m Trial 47 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:50,563]\u001b[0m Trial 48 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:51,400]\u001b[0m Trial 49 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:52,303]\u001b[0m Trial 50 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:12:52,802]\u001b[0m Trial 51 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:00,893]\u001b[0m Trial 52 finished with value: 64.11477184295654 and parameters: {'n_hidden': 1, 'n_units_l0': 108, 'z_lr': 0.0016279436197696234, 'optimizer': 'RMSprop', 'lr': 0.0004799676966548949}. Best is trial 25 with value: 45.328901290893555.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:08,432]\u001b[0m Trial 53 finished with value: 52.328752517700195 and parameters: {'n_hidden': 1, 'n_units_l0': 107, 'z_lr': 0.0020466048265863077, 'optimizer': 'RMSprop', 'lr': 0.0005742634471321757}. Best is trial 25 with value: 45.328901290893555.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:08,996]\u001b[0m Trial 54 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:10,433]\u001b[0m Trial 55 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:10,975]\u001b[0m Trial 56 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:11,482]\u001b[0m Trial 57 pruned. \u001b[0m\n"]},{"output_type":"stream","name":"stderr","text":["/mnt/home/ttesileanu/miniconda3/envs/cpcn/lib/python3.9/site-packages/optuna/pruners/_percentile.py:21: RuntimeWarning: All-NaN slice encountered\n","  return np.nanmin(values)\n"]},{"output_type":"stream","name":"stdout","text":["\u001b[32m[I 2022-04-19 18:13:11,920]\u001b[0m Trial 58 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:12,491]\u001b[0m Trial 59 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:20,515]\u001b[0m Trial 60 finished with value: 47.361722469329834 and parameters: {'n_hidden': 1, 'n_units_l0': 127, 'z_lr': 0.0024707817910588565, 'optimizer': 'RMSprop', 'lr': 0.0007348143960575052}. Best is trial 25 with value: 45.328901290893555.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:28,332]\u001b[0m Trial 61 finished with value: 46.39998006820679 and parameters: {'n_hidden': 1, 'n_units_l0': 128, 'z_lr': 0.0025758638131069076, 'optimizer': 'RMSprop', 'lr': 0.0006422926472027571}. Best is trial 25 with value: 45.328901290893555.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:36,610]\u001b[0m Trial 62 finished with value: 48.26640701293945 and parameters: {'n_hidden': 1, 'n_units_l0': 126, 'z_lr': 0.002136753415751433, 'optimizer': 'RMSprop', 'lr': 0.0006653649359911815}. Best is trial 25 with value: 45.328901290893555.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:44,660]\u001b[0m Trial 63 finished with value: 49.40270709991455 and parameters: {'n_hidden': 1, 'n_units_l0': 128, 'z_lr': 0.0023365080653495723, 'optimizer': 'RMSprop', 'lr': 0.0007357470150920909}. Best is trial 25 with value: 45.328901290893555.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:45,702]\u001b[0m Trial 64 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:48,266]\u001b[0m Trial 65 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:48,813]\u001b[0m Trial 66 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:50,850]\u001b[0m Trial 67 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:51,917]\u001b[0m Trial 68 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:59,171]\u001b[0m Trial 69 finished with value: 40.018378257751465 and parameters: {'n_hidden': 1, 'n_units_l0': 112, 'z_lr': 0.004946861968902642, 'optimizer': 'RMSprop', 'lr': 0.0005594766912852229}. Best is trial 69 with value: 40.018378257751465.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:13:59,703]\u001b[0m Trial 70 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:00,249]\u001b[0m Trial 71 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:08,715]\u001b[0m Trial 72 finished with value: 53.59866952896118 and parameters: {'n_hidden': 1, 'n_units_l0': 123, 'z_lr': 0.005458687093795195, 'optimizer': 'RMSprop', 'lr': 0.0010408788735269373}. Best is trial 69 with value: 40.018378257751465.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:09,282]\u001b[0m Trial 73 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:09,859]\u001b[0m Trial 74 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:10,354]\u001b[0m Trial 75 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:10,899]\u001b[0m Trial 76 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:11,441]\u001b[0m Trial 77 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:11,957]\u001b[0m Trial 78 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:13,521]\u001b[0m Trial 79 pruned. \u001b[0m\n"]},{"output_type":"stream","name":"stderr","text":["/mnt/home/ttesileanu/miniconda3/envs/cpcn/lib/python3.9/site-packages/optuna/pruners/_percentile.py:21: RuntimeWarning: All-NaN slice encountered\n","  return np.nanmin(values)\n"]},{"output_type":"stream","name":"stdout","text":["\u001b[32m[I 2022-04-19 18:14:14,102]\u001b[0m Trial 80 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:14,602]\u001b[0m Trial 81 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:22,612]\u001b[0m Trial 82 finished with value: 54.93153476715088 and parameters: {'n_hidden': 1, 'n_units_l0': 125, 'z_lr': 0.006089703228132665, 'optimizer': 'RMSprop', 'lr': 0.0010612140002450718}. Best is trial 69 with value: 40.018378257751465.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:28,617]\u001b[0m Trial 83 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:29,647]\u001b[0m Trial 84 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:30,180]\u001b[0m Trial 85 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:31,369]\u001b[0m Trial 86 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:32,034]\u001b[0m Trial 87 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:33,203]\u001b[0m Trial 88 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:34,229]\u001b[0m Trial 89 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:34,806]\u001b[0m Trial 90 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:35,353]\u001b[0m Trial 91 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:35,694]\u001b[0m Trial 92 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:37,676]\u001b[0m Trial 93 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:45,701]\u001b[0m Trial 94 finished with value: 44.17405891418457 and parameters: {'n_hidden': 1, 'n_units_l0': 121, 'z_lr': 0.003139590540309019, 'optimizer': 'RMSprop', 'lr': 0.0007542473268225333}. Best is trial 69 with value: 40.018378257751465.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:53,806]\u001b[0m Trial 95 finished with value: 49.06181812286377 and parameters: {'n_hidden': 1, 'n_units_l0': 121, 'z_lr': 0.003052967270048183, 'optimizer': 'RMSprop', 'lr': 0.0004923606076068821}. Best is trial 69 with value: 40.018378257751465.\u001b[0m\n","\u001b[32m[I 2022-04-19 18:14:54,364]\u001b[0m Trial 96 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:15:02,486]\u001b[0m Trial 97 finished with value: 42.08882808685303 and parameters: {'n_hidden': 1, 'n_units_l0': 121, 'z_lr': 0.0035349902771580964, 'optimizer': 'RMSprop', 'lr': 0.0005346717038913246}. Best is trial 69 with value: 40.018378257751465.\u001b[0m\n"]},{"output_type":"stream","name":"stderr","text":["/mnt/home/ttesileanu/miniconda3/envs/cpcn/lib/python3.9/site-packages/optuna/pruners/_percentile.py:21: RuntimeWarning: All-NaN slice encountered\n","  return np.nanmin(values)\n"]},{"output_type":"stream","name":"stdout","text":["\u001b[32m[I 2022-04-19 18:15:02,991]\u001b[0m Trial 98 pruned. \u001b[0m\n","\u001b[32m[I 2022-04-19 18:15:03,549]\u001b[0m Trial 99 pruned. \u001b[0m\n"]}],"metadata":{}},{"cell_type":"code","execution_count":4,"source":["print(\n","    f\"{len(study.trials)} trials in {t1 - t0:.1f} seconds: \"\n","    f\"{len(complete_trials)} complete, {len(pruned_trials)} pruned.\"\n",")\n","\n","trial = study.best_trial\n","print(f\"best pc_loss: {trial.value}, for params:\")\n","for key, value in trial.params.items():\n","    print(\"    {}: {}\".format(key, value))"],"outputs":[{"output_type":"stream","name":"stdout","text":["100 trials in 303.2 seconds: 30 complete, 69 pruned.\n","best pc_loss: 40.018378257751465, for params:\n","    n_hidden: 1\n","    n_units_l0: 112\n","    z_lr: 0.004946861968902642\n","    optimizer: RMSprop\n","    lr: 0.0005594766912852229\n"]}],"metadata":{}},{"cell_type":"code","execution_count":6,"source":["optuna.visualization.matplotlib.plot_param_importances(study)"],"outputs":[{"output_type":"stream","name":"stderr","text":["<ipython-input-6-83c13c1a414d>:1: ExperimentalWarning: plot_param_importances is experimental (supported from v2.2.0). The interface can change in the future.\n","  optuna.visualization.matplotlib.plot_param_importances(study)\n"]},{"output_type":"execute_result","data":{"text/plain":["<AxesSubplot:title={'center':'Hyperparameter Importances'}, xlabel='Importance for Objective Value', ylabel='Hyperparameter'>"]},"metadata":{},"execution_count":6},{"output_type":"display_data","data":{"image/png":"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","text/plain":["<Figure size 432x288 with 1 Axes>"]},"metadata":{}}],"metadata":{}}],"nbformat":4,"nbformat_minor":2,"metadata":{"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},"orig_nbformat":4}}