{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "8e6fc04d",
   "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": 2,
   "id": "f8a99bb3",
   "metadata": {},
   "outputs": [],
   "source": [
    "import yaml\n",
    "import wandb\n",
    "\n",
    "from src.diffopt_new.portfolio_optimization.predictor import NetworkPredictor\n",
    "from src.diffopt_new.portfolio_optimization.portfolio_problem import PortfolioProblem\n",
    "from src.diffopt_new.runner import Runner\n",
    "from collections import defaultdict\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "%matplotlib inline\n",
    "import time\n",
    "\n",
    "\n",
    "def run(problem_config, predictor_config, runner_config, training_config, n_epochs=10):\n",
    "    training_history = defaultdict(list) \n",
    "    validate_history = defaultdict(list) \n",
    "    test_history = defaultdict(list) \n",
    "    problem = PortfolioProblem(**problem_config)\n",
    "    predictor = NetworkPredictor(problem.n_variables, problem.observation_size, **predictor_config)\n",
    "    runner = Runner(problem, predictor, **runner_config)\n",
    "    problem_params_batch, _ = problem.sample_parameters()\n",
    "    \n",
    "    for ep in range(n_epochs):\n",
    "        time_start = time.time()\n",
    "        if ep % 1 == 0:\n",
    "            print('Epoch:', ep)\n",
    "        \n",
    "        validate_epoch_results =runner.run_epoch(batch_size=training_config['batch_size'], \n",
    "                                                 track_gradients=training_config['track_gradients'], \n",
    "                                                 normalize_losses=training_config['normalize_losses'],\n",
    "                                                 mode='validate', )\n",
    "        for key, val in validate_epoch_results.items():\n",
    "            if key in ['indices', 'prediction_diff']:\n",
    "                validate_history[key].append(np.concatenate(val))\n",
    "            else:\n",
    "                validate_history[key].append(np.mean(val, axis=0))\n",
    "                \n",
    "        test_epoch_results =runner.run_epoch(batch_size=training_config['batch_size'], \n",
    "                                             track_gradients=training_config['track_gradients'], \n",
    "                                             normalize_losses=training_config['normalize_losses'],\n",
    "                                             mode='test', )\n",
    "        for key, val in test_epoch_results.items():\n",
    "            if key in ['indices', 'prediction_diff']:\n",
    "                test_history[key].append(np.concatenate(val))\n",
    "            else:\n",
    "                test_history[key].append(np.mean(val, axis=0))\n",
    "        \n",
    "        \n",
    "        \n",
    "        train_epoch_results = runner.run_epoch(batch_size=training_config['batch_size'], \n",
    "                                               track_gradients=training_config['track_gradients'], \n",
    "                                               normalize_losses=training_config['normalize_losses'],\n",
    "                                               mode='train')\n",
    "        for key, val in train_epoch_results.items():\n",
    "            if key in ['indices', 'prediction_diff']:\n",
    "                training_history[key].append(np.concatenate(val))\n",
    "            else:\n",
    "                training_history[key].append(np.mean(val, axis=0))\n",
    "        if problem.validate_size == 0:\n",
    "            for key, val in train_epoch_results.items():\n",
    "                if key in ['indices', 'prediction_diff']:\n",
    "                    training_history[key].append(np.concatenate(val))\n",
    "                else:\n",
    "                    training_history[key].append(np.mean(val, axis=0))\n",
    "        print('Epoch results: train=%.4f || validate=%.4f' % (training_history['regret'][-1], \n",
    "                                                              validate_history['regret'][-1]))\n",
    "        print('It took %.2f seconds' % (time.time() - time_start))\n",
    "        \n",
    "\n",
    "    plt.figure(figsize=(8, 8))\n",
    "    plt.subplot(121)\n",
    "    plt.plot(training_history['regret'], label='train')\n",
    "    plt.plot(validate_history['regret'], label='validate')\n",
    "    _ = plt.legend()\n",
    "    plt.subplot(122)\n",
    "    plt.plot(training_history['projection_distance'], label='train')\n",
    "    plt.plot(validate_history['projection_distance'], label='validate')\n",
    "    _ = plt.legend()\n",
    "    print('Train:', np.mean(training_history['regret'][-10:]))\n",
    "    print('Validate:', np.mean(validate_history['regret'][-10:]))\n",
    "    _ = plt.legend()\n",
    "    return runner, training_history, validate_history, test_history"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "eeaa1a13",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/gr1/miniconda3/envs/evcp/lib/python3.8/site-packages/cvxpy/reductions/solvers/solving_chain.py:187: UserWarning: Your problem has too many parameters for efficient DPP compilation. We suggest setting 'ignore_dpp = True'.\n",
      "  warnings.warn(\n"
     ]
    }
   ],
   "source": [
    "# Define initial values\n",
    "\n",
    "N = 200\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",
    "problem = PortfolioProblem(**problem_config)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "9cd2a0f4",
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "predictor_config = {'init_seed': 0,\n",
    "                    'latent_dim': 32,\n",
    "                    'x_u_shift': -0.5, \n",
    "                    'scale_x_u': .1,\n",
    "                    'predict_x_u': True,\n",
    "                    'use_covariance': True,\n",
    "                    'train_W_sq': True,\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': 32,\n",
    "                   'batch_size': 16,\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",
    "                  }"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "d8c78d86",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/gr1/Projects/study-diff-opt/src/diffopt_new/runner.py:290: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  reward_loss_true, reward_true, r_max_true = torch.tensor(reward_loss), torch.tensor(reward), torch.tensor(r_max)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch results: train=0.0444 || validate=0.0427\n",
      "It took 321.57 seconds\n",
      "Epoch: 1\n",
      "Epoch results: train=0.0413 || validate=0.0405\n",
      "It took 242.55 seconds\n",
      "Epoch: 2\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n",
      "KeyboardInterrupt\n",
      "\n"
     ]
    }
   ],
   "source": [
    "runner, training_history, validate_history, test_history = run(problem_config, predictor_config,\n",
    "                                                               runner_config, training_config, n_epochs=32)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f81cda1a",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 0\n"
     ]
    }
   ],
   "source": [
    "runner, training_history, validate_history, test_history = run(problem_config, predictor_config,\n",
    "                                                               runner_config, training_config, n_epochs=32)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "f092edd3",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "df4e57f4",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "dbeb8314",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "33290950",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "ea4958f6",
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/gr1/Projects/study-diff-opt/src/diffopt_new/runner.py:290: UserWarning: To copy construct from a tensor, it is recommended to use sourceTensor.clone().detach() or sourceTensor.clone().detach().requires_grad_(True), rather than torch.tensor(sourceTensor).\n",
      "  reward_loss_true, reward_true, r_max_true = torch.tensor(reward_loss), torch.tensor(reward), torch.tensor(r_max)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch results: train=0.0325 || validate=0.0216\n",
      "Epoch: 1\n",
      "Epoch results: train=0.0322 || validate=0.0216\n",
      "Epoch: 2\n",
      "Epoch results: train=0.0319 || validate=0.0215\n",
      "Epoch: 3\n",
      "Epoch results: train=0.0317 || validate=0.0215\n",
      "Epoch: 4\n",
      "Epoch results: train=0.0314 || validate=0.0215\n",
      "Epoch: 5\n",
      "Epoch results: train=0.0312 || validate=0.0214\n",
      "Epoch: 6\n",
      "Epoch results: train=0.0310 || validate=0.0214\n",
      "Epoch: 7\n",
      "Epoch results: train=0.0308 || validate=0.0214\n",
      "Epoch: 8\n",
      "Epoch results: train=0.0307 || validate=0.0213\n",
      "Epoch: 9\n",
      "Epoch results: train=0.0305 || validate=0.0213\n",
      "Epoch: 10\n",
      "Epoch results: train=0.0303 || validate=0.0213\n",
      "Epoch: 11\n",
      "Epoch results: train=0.0302 || validate=0.0213\n",
      "Epoch: 12\n",
      "Epoch results: train=0.0300 || validate=0.0213\n",
      "Epoch: 13\n",
      "Epoch results: train=0.0298 || validate=0.0212\n",
      "Epoch: 14\n",
      "Epoch results: train=0.0297 || validate=0.0212\n",
      "Epoch: 15\n",
      "Epoch results: train=0.0295 || validate=0.0212\n",
      "Epoch: 16\n",
      "Epoch results: train=0.0294 || validate=0.0212\n",
      "Epoch: 17\n",
      "Epoch results: train=0.0293 || validate=0.0212\n",
      "Epoch: 18\n",
      "Epoch results: train=0.0291 || validate=0.0212\n",
      "Epoch: 19\n",
      "Epoch results: train=0.0290 || validate=0.0211\n",
      "Train: 0.029631633\n",
      "Validate: 0.021223184\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x576 with 2 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "predictor_config = {'init_seed': 123,\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",
    "\n",
    "runner_config = {'proximity_weight': 0,\n",
    "                 'reward_weight': 0,\n",
    "                 'solution_distance_weight': 0,\n",
    "                 'projection_distance_weight': 0,\n",
    "                 'mse_weight': 0,\n",
    "                 'surrogate_problem_weight': 1,\n",
    "                 'local_loss_weight': 0,\n",
    "                 'add_inwards_gradient': False,\n",
    "                 'inwards_gradient_th': -1,\n",
    "                 'lr': 1e-5, \n",
    "                 'optimizer_class': 'adam',\n",
    "                 'surrogate_problem_min_radius': 0.1,\n",
    "                 'pretrain_for': False, \n",
    "                 'pretraining_proximity_weight': 1,\n",
    "                 'pretraining_solution_distance_weight': 0,\n",
    "                 'pretraining_reward_weight': 0,\n",
    "                }\n",
    "\n",
    "training_config = {'n_epochs': 12,\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",
    "runner, training_history, validate_history, test_history = run(problem_config, predictor_config,\n",
    "                                                               runner_config, training_config, n_epochs=20)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "81732078",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "508e5045",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e2f44a6f",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e030b627",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "5c895e90",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "8f71d6d4",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "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.8.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
