{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "2ccafcbf",
   "metadata": {},
   "outputs": [],
   "source": [
    "import os, sys\n",
    "path_to_this_notebook = os.path.abspath('.')\n",
    "path_to_project = path_to_this_notebook[:path_to_this_notebook.find('note')]\n",
    "sys.path.append(path_to_project)\n",
    "%load_ext autoreload\n",
    "%autoreload 2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "e00047ee",
   "metadata": {},
   "outputs": [],
   "source": [
    "import yaml\n",
    "import wandb\n",
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "35c8bc38",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_method_name(runner_config):\n",
    "    r_weight = runner_config['reward_weight']\n",
    "    p_weight = runner_config['proximity_weight']\n",
    "    sd_weight = runner_config['solution_distance_weight']\n",
    "    pd_weight = runner_config['projection_distance_weight']\n",
    "    mse_weight = runner_config['mse_weight']\n",
    "    loc_weight = runner_config['local_loss_weight']\n",
    "    \n",
    "    if r_weight == 1 and p_weight == 0 and sd_weight == 0 and pd_weight == 0 and mse_weight == 0 and loc_weight == 0:\n",
    "        name = 'vanilla'\n",
    "        \n",
    "    elif r_weight == 1 and p_weight > 0 and sd_weight == 0 and pd_weight == 0 and mse_weight == 0 and loc_weight == 0:\n",
    "        name = 'prox=%s' % p_weight\n",
    "        \n",
    "    elif p_weight == 0 and sd_weight > 0 and pd_weight == 0 and mse_weight == 0 and loc_weight == 0:\n",
    "        name = 'SD=%s-R=%s' % (sd_weight, r_weight)\n",
    "        \n",
    "    elif r_weight == 1 and p_weight == 0 and sd_weight == 0 and pd_weight > 0 and mse_weight == 0 and loc_weight == 0:\n",
    "        name = 'PD=%s' % pd_weight\n",
    "    \n",
    "    elif r_weight == 0 and p_weight == 0 and sd_weight == 0 and pd_weight == 0 and mse_weight == 1 and loc_weight == 0:\n",
    "        name = 'MSE'\n",
    "    elif r_weight == 0 and p_weight == 0 and sd_weight == 0 and pd_weight == 0 and mse_weight == 0 and loc_weight > 0:\n",
    "        if runner_config['add_inwards_gradient']:\n",
    "            name = 'LOC=%s+IN=%s' % (loc_weight, runner_config['inwards_gradient_th'])\n",
    "        else:\n",
    "            name = 'LOC=%s' % loc_weight\n",
    "        \n",
    "    else:\n",
    "        name = 'R=%s' % r_weight\n",
    "        if p_weight != 0:\n",
    "            name = name + '_P=%s' % p_weight\n",
    "        if sd_weight != 0:\n",
    "            name = name + '_SD=%s' % sd_weight\n",
    "        if pd_weight != 0:\n",
    "            name = name + '_PD=%s' % pd_weight\n",
    "        if mse_weight != 0:\n",
    "            name = name + '_MSE=%s' % mse_weight\n",
    "        if loc_weight != 0:\n",
    "            if runner_config['add_inwards_gradient']:\n",
    "                name = name + '_LOC=%s+IN=%s' % (loc_weight, runner_config['inwards_gradient_th'])\n",
    "            else:\n",
    "                name = name + '_LOC=%s' % loc_weight\n",
    "    if runner_config['surrogate_problem_weight'] != 0:\n",
    "        name = name +  '_SurPr%s' % runner_config['surrogate_problem_weight']\n",
    "    return name\n",
    "\n",
    "def generate_problem_name(problem_config):\n",
    "    \n",
    "    data_str = '%d-samples' % problem_config['n_samples'] if problem_config['n_samples'] < 2500 else 'full'\n",
    "    if problem_config['use_train_only']:\n",
    "        data_str = data_str + '_train-only'    \n",
    "    name = 'portfolio-problem_n=%d_alpha=%s_%s' % (problem_config['n_variables'], problem_config['alpha'], data_str)\n",
    "    return name\n",
    "\n",
    "def generate_run_name(problem_config, predictor_config, runner_config, training_config):\n",
    "    name = get_method_name(runner_config)\n",
    "    if runner_config['pretrain_for']:\n",
    "        name = name + '_pretrain%s' % runner_config['pretrain_for']\n",
    "    \n",
    "    predictor_str = 'pred-xu' if predictor_config['predict_x_u'] else 'pred-wlin'\n",
    "    if predictor_config['use_covariance']:\n",
    "        predictor_str += '_cov-mat'    \n",
    "        \n",
    "    predictor_str = predictor_str + '_scale%s-add%s' % (predictor_config['scale_x_u'],\n",
    "                                                        predictor_config['x_u_shift'])\n",
    "    if not predictor_config['train_W_sq']:\n",
    "        predictor_str = predictor_str +  '_frozen-Wsq'\n",
    "        \n",
    "    bs_str = 'bs=%s' % training_config['batch_size']\n",
    "    lr_str = 'lr=%s' % runner_config['lr']\n",
    "    ep_str = 'epochs=%s' % training_config['n_epochs']\n",
    "    seed_str = '(%s, %s)' % (predictor_config['init_seed'], problem_config['data_sample_seed'])\n",
    "    \n",
    "    return '%s_%s_%s_%s_%s_%s' % (name, lr_str, bs_str, ep_str, predictor_str, seed_str)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "566c5d05",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'/home/gr1/Projects/study-diff-opt-submission//experiments/portfolio_optimization_n=50/portfolio-problem_n=50_alpha=2_full_test/vanilla_lr=5e-05_bs=1_epochs=16_pred-xu_scale0.1-add1_frozen-Wsq_(0, 0)/'"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# Define initial values\n",
    "\n",
    "N = 50\n",
    "folder_name = 'portfolio_optimization_n=%d' % N\n",
    "problem_config = {'n_variables': N, \n",
    "                  'path_to_data': path_to_project + '/portfolio_data/',\n",
    "                  'n_samples': 10000,\n",
    "                  'alpha': 2,\n",
    "                  'train_ratio': 0.7,\n",
    "                  'validate_ratio': 0.15,\n",
    "                  'use_train_only': False,\n",
    "                  'data_sample_seed': 0,\n",
    "                  'proximity_method': 'mean',\n",
    "                  }\n",
    "\n",
    "predictor_config = {'init_seed': 0,\n",
    "                    'latent_dim': 32,\n",
    "                    'x_u_shift': 1, \n",
    "                    'scale_x_u': .1,\n",
    "                    'predict_x_u': True,\n",
    "                    'use_covariance': False,\n",
    "                    'train_W_sq': False,\n",
    "                    }\n",
    "\n",
    "runner_config = {'proximity_weight': 0,\n",
    "                 'reward_weight': 1,\n",
    "                 'solution_distance_weight': 0,\n",
    "                 'projection_distance_weight': 0,\n",
    "                 'mse_weight': 0,\n",
    "                 'surrogate_problem_weight': 0,\n",
    "                 'local_loss_weight': 0,\n",
    "                 'add_inwards_gradient': False,\n",
    "                 'inwards_gradient_th': 0,\n",
    "                 'lr': 5e-5, \n",
    "                 'optimizer_class': 'adam',\n",
    "                 'surrogate_problem_min_radius': 0.1,\n",
    "                 'pretrain_for': False, \n",
    "                 'pretraining_proximity_weight': 0,\n",
    "                 'pretraining_solution_distance_weight': 0,\n",
    "                 'pretraining_reward_weight': 1,\n",
    "                }\n",
    "\n",
    "training_config = {'n_epochs': 16,\n",
    "                   'batch_size': 1,\n",
    "                   'print_each': 5,\n",
    "                   'validate_each': 1, \n",
    "                   'track_gradients': False,\n",
    "                   'normalize_losses': False,\n",
    "                   'early_stopping': False,\n",
    "                   'early_stopping_th': 5,\n",
    "                  }\n",
    "\n",
    "suffix = '_test'\n",
    "problem_name = generate_problem_name(problem_config) + suffix\n",
    "run_name = generate_run_name(problem_config, predictor_config, runner_config, training_config)\n",
    "base_path = path_to_project + '/experiments/%s/%s/%s/' % (folder_name, problem_name, run_name)\n",
    "base_path"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "10a1825e",
   "metadata": {},
   "outputs": [],
   "source": [
    "N_SEEDS = 1\n",
    "\n",
    "data_seeds = list(np.random.randint(0, 9999, size=N_SEEDS))\n",
    "init_seeds = list(np.random.randint(0, 9999, size=N_SEEDS))\n",
    "\n",
    "#init_seeds = [16, 8945, 95, 14][:4] # alpha=2\n",
    "#data_seeds  = [7254, 3266, 5890, 64][:4]\n",
    "\n",
    "#init_seeds = [4334, 1375, 1142, 3986][:2] # alpha=1\n",
    "#data_seeds  = [6702, 9759, 1374, 7909][:2]\n",
    "\n",
    "#init_seeds = [6174, 3516, 7600, 165][:2] # alpha=0.1\n",
    "#data_seeds  = [9578, 9559, 1840, 8851][:2]\n",
    "\n",
    "#init_seeds = [6691, 4957, 7841, 2869][:3] # alpha=0\n",
    "#data_seeds  = [9914, 4380, 8908, 7906][:3]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "13fbb24b",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "data_seeds = list(np.random.randint(0, 9999, size=N_SEEDS))\n",
    "init_seeds = list(np.random.randint(0, 9999, size=N_SEEDS))\n",
    "\n",
    "weights = [\n",
    "            {'mse_weight': 0, 'local_loss_weight': 0, 'add_inwards_gradient': False, 'lr': 3e-5, # Vanilla\n",
    "             'surrogate_problem_weight': 0, 'reward_weight': 1, 'projection_distance_weight': 0},\n",
    "        \n",
    "            #{'mse_weight': 0, 'local_loss_weight': 1, 'add_inwards_gradient': False, 'lr': 5e-5, # Local\n",
    "            # 'surrogate_problem_weight': 0, 'reward_weight': 0, 'projection_distance_weight': 0},\n",
    "    \n",
    "            #{'mse_weight': 0, 'local_loss_weight': 0, 'add_inwards_gradient': False, 'lr': 5e-5, # Vanilla + PD#1\n",
    "            # 'surrogate_problem_weight': 0, 'reward_weight': 1, 'projection_distance_weight': 0.1},\n",
    "    \n",
    "            #{'mse_weight': 0, 'local_loss_weight': 1, 'add_inwards_gradient': False, 'lr': 5e-5, # Local + PD#1\n",
    "            # 'surrogate_problem_weight': 0, 'reward_weight': 0, 'projection_distance_weight': 0.1},\n",
    "    \n",
    "            #{'mse_weight': 1, 'local_loss_weight': 0, 'add_inwards_gradient': False, # MSE\n",
    "            #'lr': 1e-4, 'surrogate_problem_weight': 0, 'reward_weight': 0, 'projection_distance_weight': 0},\n",
    "    \n",
    "            #{'local_loss_weight': 1, 'add_inwards_gradient': False, 'lr': 5e-6, # Loccal + PD#1 diff lr\n",
    "             #'surrogate_problem_weight': 0, 'reward_weight': 0, 'projection_distance_weight': 0.1},\n",
    "        \n",
    "            #{'local_loss_weight': 0, 'add_inwards_gradient': False, 'lr': 1e-5, # Vanilla + PD#2\n",
    "            #'surrogate_problem_weight': 0, 'reward_weight': 1, 'projection_distance_weight': 0.15},\n",
    "    \n",
    "            #{'local_loss_weight': 1, 'add_inwards_gradient': False, 'lr': 1e-5, # Local + PD#2\n",
    "            #'surrogate_problem_weight': 0, 'reward_weight': 0, 'projection_distance_weight': 0.15},\n",
    "    \n",
    "            #'local_loss_weight': 1, 'add_inwards_gradient': False, 'lr': 7e-6, # Loccal + PD#2 diff lr\n",
    "            #'surrogate_problem_weight': 0, 'reward_weight': 0, 'projection_distance_weight': 0.15},\n",
    "    \n",
    "           \n",
    "          ]\n",
    "\n",
    "predictor_params = [{'train_W_sq': False,  'x_u_shift': 1, 'scale_x_u': .1, 'use_covariance': False},\n",
    "                    #{'train_W_sq': True,  'x_u_shift': 1, 'scale_x_u': .1, 'use_covariance': True},\n",
    "                    #{'train_W_sq': True,  'x_u_shift': 1, 'scale_x_u': .1, 'use_covariance': True},\n",
    "                   ]\n",
    "\n",
    "training_params = [\n",
    "                    {'batch_size': 1},\n",
    "                   #{'batch_size': 4}, \n",
    "                   #{'batch_size': 16}, \n",
    "                   #{'batch_size': 32}, \n",
    "                  ]\n",
    "\n",
    "problem_params = [\n",
    "                  {'alpha': 2},\n",
    "                  #{'alpha': 0.1},\n",
    "                  #{'alpha': 0.25},\n",
    "                  #{'alpha': 0.5}, \n",
    "                  #{'alpha': 1},\n",
    "                  #{'alpha': 2},\n",
    "                  ]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "c0402f3b",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Problem: portfolio-problem_n=50_alpha=2_full_test\n",
      "Created vanilla_lr=3e-05_bs=1_epochs=16_pred-xu_scale0.1-add1_frozen-Wsq_(3698, 2430)\n"
     ]
    }
   ],
   "source": [
    "for data_sample_seed, init_seed in zip(data_seeds, init_seeds):\n",
    "    problem_config['data_sample_seed'] = int(data_sample_seed)\n",
    "    predictor_config['init_seed'] = int(init_seed)\n",
    "    \n",
    "    \n",
    "    for problem_param_dict in problem_params:\n",
    "        for key, val in problem_param_dict.items():\n",
    "                problem_config[key] = val\n",
    "        problem_name = generate_problem_name(problem_config) + suffix\n",
    "        print('Problem:', problem_name)\n",
    "        for predictor_params_dict in predictor_params:\n",
    "            for key, val in predictor_params_dict.items():\n",
    "                predictor_config[key] = val\n",
    "        \n",
    "            for weights_dict in weights:\n",
    "                for key, val in weights_dict.items():\n",
    "                    runner_config[key] = val\n",
    "                    \n",
    "                for training_dict in training_params:\n",
    "                    for key, val in training_dict.items():\n",
    "                        training_config[key] = val\n",
    "                \n",
    "                    run_name = generate_run_name(problem_config, predictor_config, runner_config, training_config)\n",
    "                    base_path = path_to_project + '/experiments/%s/%s/%s/' % (folder_name, problem_name, run_name)\n",
    "\n",
    "                    if not os.path.isdir(base_path):\n",
    "                        os.makedirs(base_path)\n",
    "                        print('Created', run_name)\n",
    "                        full_config = {'problem_config': problem_config,\n",
    "                                       'predictor_config': predictor_config,\n",
    "                                       'runner_config': runner_config,\n",
    "                                       'training_config': training_config, \n",
    "                                       'wandb_id': wandb.util.generate_id(), \n",
    "                                       'wandb_project_name': problem_name,\n",
    "                                       'problem_name': problem_name,\n",
    "                                       'run_name': run_name}\n",
    "                        with open(base_path + 'config.yml', 'w') as f:\n",
    "                            yaml.dump(full_config, f, default_flow_style=False)\n",
    "                    else:\n",
    "                        print('The experiment already exists!')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "324b15e3",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "b2ec65b0",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dd4f55af",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python [conda env:diffopt] *",
   "language": "python",
   "name": "conda-env-diffopt-py"
  },
  "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.8.16"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
