{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[INFO] running on GPU core 2\n"
     ]
    }
   ],
   "source": [
    "# restart kernel\n",
    "from IPython.core.display import HTML\n",
    "HTML(\"<script>Jupyter.notebook.kernel.restart()</script>\")\n",
    "\n",
    "import math, random, pickle, os, copy, itertools, sys, gpytorch\n",
    "import torch, logging, scipy.io\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import seaborn as sns\n",
    "from numpy import random\n",
    "from datetime import datetime\n",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "from collections.abc import Iterable\n",
    "%matplotlib inline\n",
    "\n",
    "BASE_DIR = os.path.dirname(os.path.dirname(os.getcwd()))\n",
    "sys.path.insert(1, BASE_DIR)\n",
    "\n",
    "import config\n",
    "config.init()\n",
    "from config import device\n",
    "from server.MAML_server import MAMLRegression\n",
    "from utils.analyze_results import *\n",
    "from utils.assistive_functions import load_trained_models, softplus_inverse\n",
    "from utils.pf_scheduler import PriorFactorScheduler\n",
    "from utils.train_models import train_models\n",
    "\n",
    "random_seed = 5\n",
    "random.seed(random_seed)\n",
    "np.random.seed(random_seed)\n",
    "random_state = np.random.RandomState(random_seed)\n",
    "\n",
    "if torch.cuda.is_available() and not str(device)=='cpu':\n",
    "    print('[INFO] running on GPU core ' + str(torch.cuda.current_device()))\n",
    "else:\n",
    "    print('[WARN] running on CPU')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "exp_name = 'PV_UniModal'         # PV_BiModal, PV_UniModal_augmented', PV_UniModal\n",
    "save_maml = True\n",
    "methods = [\n",
    "    'lin', \n",
    "    'nn4x2', 'nn16x2', 'nn32x2',\n",
    "    'nn4x4', 'nn16x4', 'nn32x4',\n",
    "] \n",
    "methods_to_run = ['nn4x4'] # best in Bi and uni: 4x4\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load data\n",
    "* vary the tilt of the installation and the azimuthal orientation\n",
    "* same coordinates and altitude of central Lausanne"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[INFO] loaded data for 24 clients\n",
      "\n",
      "24 households at Lausanne - tilt_std: 5.0, az_std: 15.0, weather_dev: 0.1, irrad_std: 0.2, altitude_dev: 0.1, shadow_peak_red: 0.8, different module_name, different inverter_name, \n",
      "Subset of clients for demonstration:  [12, 14, 15, 17, 22]\n",
      "\n",
      "[INFO] 15 features:  H_sun T2m WS10m station_irrad_direct_prev lag 1 lag 2 lag 4 lag 18 lag 20 lag 22 lag 43 lag 70 lag 74 lag 121 lag 145\n"
     ]
    }
   ],
   "source": [
    "# ------ LOAD DATA ------\n",
    "# NOTE: env generated in notebook 1_visualization\n",
    "filename_env = config.PVDATA_DIR + '/'+ exp_name+\"_env\"\n",
    "file = open(filename_env, 'rb')\n",
    "env_dict = pickle.load(file)\n",
    "msg = '[INFO] loaded data for {:2.0f} clients'.format(env_dict['num_clients'])\n",
    "print(msg)\n",
    "file.close()\n",
    "num_clients = env_dict['num_clients'] \n",
    "print('\\n'+env_dict['info'])\n",
    "\n",
    "# ----- SELECT A SUBSET OF CLIENTS -----\n",
    "clients_subset= [12, 14, 15, 17, 22]\n",
    "print('Subset of clients for demonstration: ', clients_subset)\n",
    "\n",
    "num_features = len(env_dict['feature_names'])\n",
    "print('\\n[INFO] {:2.0f} features: '.format(\n",
    "    len(env_dict['feature_names'])), *env_dict['feature_names'])\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "nn4x4 in sml\n",
      "[2023-05-16 03:13:02,928 -INFO]  Iter 1/5000 - Loss: 0.588021 - Time 0.79 sec Valid-RSMSE: 1.666 \n",
      "[2023-05-16 03:13:12,004 -INFO]  Iter 500/5000 - Loss: 0.372493 - Time 9.08 sec Valid-RSMSE: 0.641 \n",
      "[2023-05-16 03:13:21,096 -INFO]  Iter 1000/5000 - Loss: 0.220604 - Time 9.00 sec Valid-RSMSE: 0.596 \n",
      "[2023-05-16 03:13:29,984 -INFO]  Iter 1500/5000 - Loss: 0.194834 - Time 8.80 sec Valid-RSMSE: 0.575 \n",
      "[2023-05-16 03:13:38,903 -INFO]  Iter 2000/5000 - Loss: 0.189956 - Time 8.80 sec Valid-RSMSE: 0.568 \n",
      "[2023-05-16 03:13:47,865 -INFO]  Iter 2500/5000 - Loss: 0.204510 - Time 8.89 sec Valid-RSMSE: 0.564 \n",
      "[2023-05-16 03:13:56,876 -INFO]  Iter 3000/5000 - Loss: 0.189453 - Time 8.90 sec Valid-RSMSE: 0.559 \n",
      "[2023-05-16 03:14:05,858 -INFO]  Iter 3500/5000 - Loss: 0.195017 - Time 8.88 sec Valid-RSMSE: 0.553 \n",
      "[2023-05-16 03:14:14,822 -INFO]  Iter 4000/5000 - Loss: 0.186761 - Time 8.88 sec Valid-RSMSE: 0.550 \n",
      "[2023-05-16 03:14:23,837 -INFO]  Iter 4500/5000 - Loss: 0.186907 - Time 8.91 sec Valid-RSMSE: 0.549 \n",
      "[2023-05-16 03:14:32,834 -INFO]  Iter 5000/5000 - Loss: 0.190945 - Time 8.90 sec Valid-RSMSE: 0.546 \n",
      "\n",
      "nn4x4 in 1y\n",
      "[2023-05-16 03:14:33,058 -INFO]  Iter 1/5000 - Loss: 0.327816 - Time 0.02 sec Valid-RSMSE: 1.303 \n",
      "[2023-05-16 03:14:41,935 -INFO]  Iter 500/5000 - Loss: 0.435438 - Time 8.88 sec Valid-RSMSE: 0.659 \n",
      "[2023-05-16 03:14:50,938 -INFO]  Iter 1000/5000 - Loss: 0.221269 - Time 8.88 sec Valid-RSMSE: 0.600 \n",
      "[2023-05-16 03:15:00,001 -INFO]  Iter 1500/5000 - Loss: 0.189880 - Time 8.95 sec Valid-RSMSE: 0.585 \n",
      "[2023-05-16 03:15:09,059 -INFO]  Iter 2000/5000 - Loss: 0.181739 - Time 8.95 sec Valid-RSMSE: 0.588 \n",
      "[2023-05-16 03:15:18,065 -INFO]  Iter 2500/5000 - Loss: 0.171390 - Time 8.95 sec Valid-RSMSE: 0.589 \n",
      "[2023-05-16 03:15:27,095 -INFO]  Iter 3000/5000 - Loss: 0.179202 - Time 8.95 sec Valid-RSMSE: 0.599 \n",
      "[2023-05-16 03:15:36,173 -INFO]  Iter 3500/5000 - Loss: 0.182625 - Time 8.95 sec Valid-RSMSE: 0.600 \n",
      "[2023-05-16 03:15:45,243 -INFO]  Iter 4000/5000 - Loss: 0.171135 - Time 8.97 sec Valid-RSMSE: 0.603 \n",
      "[2023-05-16 03:15:54,270 -INFO]  Iter 4500/5000 - Loss: 0.166415 - Time 8.94 sec Valid-RSMSE: 0.597 \n",
      "[2023-05-16 03:16:03,316 -INFO]  Iter 5000/5000 - Loss: 0.173223 - Time 8.93 sec Valid-RSMSE: 0.602 \n"
     ]
    }
   ],
   "source": [
    "if len(methods_to_run)>0:\n",
    "    models_MAML = {key: {'sml':None, '1y':None} for key in methods_to_run}\n",
    "    results_MAML = {key: {'sml':None, '1y':None} for key in methods_to_run}\n",
    "    for scenario_name_fl in ['sml', '1y']:\n",
    "        clients_data = env_dict['train_scenarios'][scenario_name_fl]['clients_data']\n",
    "        meta_train_data = []\n",
    "        meta_test_data = []\n",
    "        for client_num in np.arange(num_clients):\n",
    "            meta_train_data.append((clients_data[client_num][0], clients_data[client_num][1]))\n",
    "        for method in methods_to_run:\n",
    "            if method=='lin':\n",
    "                continue\n",
    "            else:\n",
    "                print('\\n' + method + ' in ' + scenario_name_fl)\n",
    "                num_layers = int(method.split('x')[-1])\n",
    "                num_neurons = int(method.split('x')[0][2:])\n",
    "                layer_sizes = (num_neurons,)*num_layers\n",
    "                meta_learner = MAMLRegression(\n",
    "                    meta_train_data, task_batch_size=5, num_iter_fit=5000, layer_sizes=layer_sizes)\n",
    "                meta_learner.meta_fit(clients_data, log_period=500)\n",
    "                results_MAML[method][scenario_name_fl] = meta_learner.eval_datasets(clients_data, get_full_list=True)\n",
    "                models_MAML[method][scenario_name_fl] = meta_learner.best_params\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# New Clients"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[INFO] loaded data for 24 clients\n",
      "\n",
      "24 households at Lausanne - tilt_std: 5.0, az_std: 15.0, weather_dev: 0.1, irrad_std: 0.2, altitude_dev: 0.1, shadow_peak_red: 0.8, different module_name, different inverter_name, \n"
     ]
    }
   ],
   "source": [
    "exp_name_ptc = exp_name\n",
    "exp_name_new = exp_name_ptc + '_NewClients'\n",
    "\n",
    "# ----- SET UP LOGGER -----\n",
    "filename_env_new = config.PVDATA_DIR + '/'+ exp_name_ptc +\"_new_clients_env\"\n",
    "filename_res_new = os.path.join(os.getcwd(), \"saved_results\", exp_name_new)\n",
    "\n",
    "\n",
    "# ------ LOAD META_TEST DATA ------\n",
    "file = open(filename_env_new, 'rb')\n",
    "env_dict_new = pickle.load(file)\n",
    "msg = '[INFO] loaded data for {:2.0f} clients'.format(env_dict_new['num_clients'])\n",
    "print(msg)\n",
    "file.close()\n",
    "num_clients_new = env_dict_new['num_clients'] \n",
    "print('\\n'+env_dict_new['info'])\n",
    "\n",
    "clients_data_new = env_dict_new['train_scenarios']['sml']['clients_data']\n",
    "meta_train_data_new = []\n",
    "meta_test_data_new = []\n",
    "for client_num in np.arange(num_clients_new):\n",
    "    meta_train_data_new.append((clients_data_new[client_num][0], clients_data_new[client_num][1]))\n",
    "    meta_test_data_new.append((clients_data_new[client_num][2], clients_data_new[client_num][3]))\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "nn4x4 in sml\n",
      "0.545826699047857 0.5271684811844929\n",
      "\n",
      "nn4x4 in 1y\n",
      "0.6017741885075457 0.5591315079815287\n"
     ]
    }
   ],
   "source": [
    "if len(methods_to_run)>0:\n",
    "    results_MAML_new = copy.deepcopy(results_MAML)\n",
    "    for scenario_name_fl in ['sml', '1y']:\n",
    "        for method in methods_to_run:\n",
    "            if method=='lin':\n",
    "                continue\n",
    "            if models_MAML[method][scenario_name_fl] is None: \n",
    "                continue\n",
    "            print('\\n'+method+' in '+scenario_name_fl)\n",
    "            # reconstruct learner\n",
    "            num_layers = int(method.split('x')[-1])\n",
    "            num_neurons = int(method.split('x')[0][2:])\n",
    "            layer_sizes = (num_neurons,)*num_layers\n",
    "            meta_learner = MAMLRegression(\n",
    "                meta_train_data_new, task_batch_size=5, num_iter_fit=5000, layer_sizes=layer_sizes)\n",
    "            meta_learner.initial_params = [torch.nn.Parameter(torch.tensor(param).to(device)) for param in models_MAML[method][scenario_name_fl]]\n",
    "            # eval\n",
    "            results_MAML_new[method][scenario_name_fl] = meta_learner.eval_datasets(clients_data_new, get_full_list=True)\n",
    "            rsmse_test_new = results_MAML_new[method][scenario_name_fl]['rsmse']\n",
    "            print(\n",
    "                np.mean(results_MAML[method][scenario_name_fl]['rsmse']),\n",
    "                np.mean(rsmse_test_new), \n",
    "                # np.quantile(rsmse_test_new, .25), \n",
    "                # np.quantile(rsmse_test_new, .50), np.quantile(rsmse_test_new, .75)\n",
    "            )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "sml existing 0.545826699047857\n",
      "1y  existing 0.6017741885075457\n",
      "sml new      0.5271684811844929\n",
      "1y  new      0.5591315079815287\n"
     ]
    }
   ],
   "source": [
    "# To remove\n",
    "print('sml existing', np.mean(results_MAML['nn4x4']['sml']['rsmse']))\n",
    "# BiModal: [0.5198103939697368, 0.5080354630029575, 0.5133364673716602, 0.5259531514885433, 0.548085263707498]\n",
    "# Uni:     [0.568286583147538, 0.5552673549013596, 0.5771969233939762, 0.5853242865258677, 0.554966728383184]\n",
    "print('1y  existing', np.mean(results_MAML['nn4x4']['1y']['rsmse']))\n",
    "# BiModal: [0.5232343341294651, 0.5271332453300626, 0.5327929826334136, 0.5250287029008597, 0.5096271770447632]\n",
    "# Uni:     [0.5859636866974344, 0.5735252663992452, 0.5670678235617824, 0.586841881025424, 0.5873462883031865]\n",
    "print('sml new     ', np.mean(results_MAML_new['nn4x4']['sml']['rsmse']))\n",
    "# BiModal:  [0.5186647386089885, 0.513772642212023, 0.5224044290331805, 0.520295779226223, 0.5334619407686961]\n",
    "# UniModal: [0.5472391768618535, 0.532189327529406, 0.5583274480226214, 0.5644486116203647, 0.5264580733737637]\n",
    "print('1y  new     ', np.mean(results_MAML_new['nn4x4']['1y']['rsmse']))\n",
    "# BiModal:  [0.5214323239601835, 0.5373312998641454, 0.5443388072544023, 0.5342467432034309, 0.514159453762895]\n",
    "# UniModal: [0.5581600767960219, 0.5506115682181142, 0.5433789121618879, 0.5466004710359001, 0.5621243948643618]\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0.552, 0.014\n"
     ]
    }
   ],
   "source": [
    "a = [0.5581600767960219, 0.5506115682181142, 0.5433789121618879, 0.5466004710359001, 0.5621243948643618]\n",
    "print('{:1.3f}, {:1.3f}'.format(np.mean(a), 1.96 * np.std(a)))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "models saved.\n"
     ]
    }
   ],
   "source": [
    "filename_save = os.path.join(BASE_DIR, 'experiments', 'PV', 'saved_results', exp_name, 'MAML', 'MAML')\n",
    "if save_maml:\n",
    "    file = open(filename_save, 'wb')\n",
    "    pickle.dump({'results': results_MAML, 'models':models_MAML, 'results_new':results_MAML_new}, file)\n",
    "    file.close()\n",
    "    print('models saved.')\n",
    "else:\n",
    "    file = open(filename_save, 'rb')\n",
    "    res = pickle.load(file)\n",
    "    file.close()\n",
    "    results_MAML_new = res['results_new']\n",
    "    results_MAML = res['results']\n",
    "    models_MAML = res['models']"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "5:\n",
    "nn4x2 in sml\n",
    "0.5198103939697368 0.5186647386089885\n",
    "\n",
    "nn16x2 in sml\n",
    "0.578062151351349 0.5226248316108938\n",
    "\n",
    "nn32x2 in sml\n",
    "0.5730231695881834 0.5169378741713349\n",
    "\n",
    "nn4x4 in sml\n",
    "0.5260404315455733 0.5227809982504991\n",
    "\n",
    "nn16x4 in sml\n",
    "0.5703883281709586 0.5228498766694227\n",
    "\n",
    "nn32x4 in sml\n",
    "0.5623443201653552 0.526428899654507\n",
    "\n",
    "nn4x2 in 1y\n",
    "0.5232343341294651 0.5214323239601835\n",
    "\n",
    "nn16x2 in 1y\n",
    "0.5493125335919756 0.5192277579392194\n",
    "\n",
    "nn32x2 in 1y\n",
    "0.5590345594537472 0.5237550094993045\n",
    "\n",
    "nn4x4 in 1y\n",
    "0.5411756345121507 0.5395890110984914\n",
    "\n",
    "nn16x4 in 1y\n",
    "0.5523013335961507 0.5195808766908913\n",
    "\n",
    "nn32x4 in 1y\n",
    "0.5723329558598986 0.516144560780893"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3.10.4 ('python3')",
   "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.10.4"
  },
  "vscode": {
   "interpreter": {
    "hash": "98d0820a7d16c01dd14be774e41e0e13359f9b2ee065b64ad6e280dcddbc9693"
   }
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
