{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "08a0d308-c5e4-493f-a6c8-444d508d8710",
   "metadata": {},
   "outputs": [],
   "source": [
    "# --! include Koopa folder into PYTHONPATH --!\n",
    "\n",
    "import os\n",
    "import sys\n",
    "\n",
    "thisdir  = os.getcwd()\n",
    "koopadir = os.path.abspath(os.path.join(thisdir, '..', '..', 'external', 'Koopa'))\n",
    "sys.path.append(koopadir)\n",
    "\n",
    "# --! import python libraries and Koopa framework --!\n",
    "\n",
    "import torch\n",
    "import numpy as np\n",
    "from exp.exp_main import Exp_Main\n",
    "import random\n",
    "import argparse\n",
    "from   matplotlib import pyplot as plt\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "ad600560-0781-4587-92f7-1167fd2d407a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "_StoreTrueAction(option_strings=['--multistep'], dest='multistep', nargs=0, const=True, default=False, type=None, choices=None, required=False, help='whether to use approximation for multistep K', metavar=None, deprecated=False)"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# --! build a parser for arguments --!\n",
    "\n",
    "parser = argparse.ArgumentParser(description='Koopa for Time Series Forecasting')\n",
    "\n",
    "# basic config\n",
    "parser.add_argument('--is_training', type=int, required=True, default=1, help='status')\n",
    "parser.add_argument('--model_id', type=str, required=True, default='test', help='model id')\n",
    "parser.add_argument('--model', type=str, required=True, default='Koopa',\n",
    "                    help='model name, options: [Koopa]')\n",
    "\n",
    "# data loader\n",
    "parser.add_argument('--data', type=str, required=True, default='ETTh2', help='dataset type')\n",
    "parser.add_argument('--root_path', type=str, default='./data/ETT/', help='root path of the data file')\n",
    "parser.add_argument('--data_path', type=str, default='ETTh2.csv', help='data file')\n",
    "parser.add_argument('--features', type=str, default='M',\n",
    "                    help='forecasting task, options:[M, S, MS]; M:multivariate predict multivariate, S:univariate predict univariate, MS:multivariate predict univariate')\n",
    "parser.add_argument('--target', type=str, default='OT', help='target feature in S or MS task')\n",
    "parser.add_argument('--freq', type=str, default='h',\n",
    "                    help='freq for time features encoding, options:[s:secondly, t:minutely, h:hourly, d:daily, b:business days, w:weekly, m:monthly], you can also use more detailed freq like 15min or 3h')\n",
    "parser.add_argument('--checkpoints', type=str, default='./checkpoints/', help='location of model checkpoints')\n",
    "\n",
    "# forecasting task\n",
    "parser.add_argument('--seq_len', type=int, default=96, help='input sequence length')\n",
    "parser.add_argument('--label_len', type=int, default=48, help='start token length')\n",
    "parser.add_argument('--pred_len', type=int, default=48, help='prediction sequence length')\n",
    "\n",
    "# model define\n",
    "parser.add_argument('--enc_in', type=int, default=7, help='encoder input size')\n",
    "parser.add_argument('--dec_in', type=int, default=7, help='decoder input size')\n",
    "parser.add_argument('--c_out', type=int, default=7, help='output size')\n",
    "parser.add_argument('--dropout', type=float, default=0.05, help='dropout')\n",
    "parser.add_argument('--embed', type=str, default='timeF',\n",
    "                    help='time features encoding, options:[timeF, fixed, learned]')\n",
    "parser.add_argument('--do_predict', action='store_true', help='whether to predict unseen future data')\n",
    "\n",
    "# optimization\n",
    "parser.add_argument('--num_workers', type=int, default=10, help='data loader num workers')\n",
    "parser.add_argument('--itr', type=int, default=1, help='experiments times')\n",
    "parser.add_argument('--train_epochs', type=int, default=10, help='train epochs')\n",
    "parser.add_argument('--batch_size', type=int, default=32, help='batch size of train input data')\n",
    "parser.add_argument('--patience', type=int, default=3, help='early stopping patience')\n",
    "parser.add_argument('--learning_rate', type=float, default=0.001, help='optimizer learning rate')\n",
    "parser.add_argument('--des', type=str, default='test', help='exp description')\n",
    "parser.add_argument('--loss', type=str, default='mse', help='loss function')\n",
    "parser.add_argument('--lradj', type=str, default='type1', help='adjust learning rate')\n",
    "parser.add_argument('--use_amp', action='store_true', help='use automatic mixed precision training', default=False)\n",
    "\n",
    "# GPU\n",
    "parser.add_argument('--use_gpu', type=bool, default=True, help='use gpu')\n",
    "parser.add_argument('--gpu', type=int, default=0, help='gpu')\n",
    "parser.add_argument('--use_multi_gpu', action='store_true', help='use multiple gpus', default=False)\n",
    "parser.add_argument('--devices', type=str, default='0,1,2,3', help='device ids of multile gpus')\n",
    "parser.add_argument('--seed', type=int, default=2023, help='random seed')\n",
    "\n",
    "# Koopa\n",
    "parser.add_argument('--dynamic_dim', type=int, default=128, help='latent dimension of koopman embedding')\n",
    "parser.add_argument('--hidden_dim', type=int, default=64, help='hidden dimension of en/decoder')\n",
    "parser.add_argument('--hidden_layers', type=int, default=2, help='number of hidden layers of en/decoder')\n",
    "parser.add_argument('--seg_len', type=int, default=48, help='segment length of time series')\n",
    "parser.add_argument('--num_blocks', type=int, default=3, help='number of Koopa blocks')\n",
    "parser.add_argument('--alpha', type=float, default=0.2, help='spectrum filter ratio')\n",
    "parser.add_argument('--multistep', action='store_true', help='whether to use approximation for multistep K', default=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "e371369d-a617-49ca-a105-2cce4f676740",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Args in experiment:\n",
      "Namespace(is_training=1, model_id='ETTh2_96_48', model='Koopa', data='ETTh2', root_path='../../data/baselines/', data_path='ETTh2.csv', features='S', target='OT', freq='h', checkpoints='./checkpoints/', seq_len=96, label_len=48, pred_len=48, enc_in=1, dec_in=1, c_out=1, dropout=0.05, embed='timeF', do_predict=False, num_workers=10, itr=1, train_epochs=10, batch_size=32, patience=3, learning_rate=0.001, des='Exp', loss='mse', lradj='type1', use_amp=False, use_gpu=False, gpu=0, use_multi_gpu=False, devices='0,1,2,3', seed=4723, dynamic_dim=64, hidden_dim=512, hidden_layers=2, seg_len=48, num_blocks=4, alpha=0.2, multistep=False)\n"
     ]
    }
   ],
   "source": [
    "# --! specify arguments --!\n",
    "\n",
    "args = parser.parse_args(\n",
    "    args=[\n",
    "        \"--is_training\", \"1\",\n",
    "        \"--root_path\", \"../../data/baselines/\",\n",
    "        \"--data_path\", \"ETTh2.csv\",\n",
    "        \"--model_id\", \"ETTh2_96_48\",\n",
    "        \"--model\", \"Koopa\",\n",
    "        \"--data\", \"ETTh2\",\n",
    "        \"--features\", \"S\",\n",
    "        \"--seq_len\", \"96\",\n",
    "        \"--pred_len\", \"48\",\n",
    "        \"--seg_len\", \"48\",\n",
    "        \"--dynamic_dim\", \"64\",\n",
    "        \"--hidden_dim\", \"512\",\n",
    "        \"--hidden_layers\", \"2\",\n",
    "        \"--num_blocks\", \"4\",\n",
    "        \"--enc_in\", \"1\",\n",
    "        \"--dec_in\", \"1\",\n",
    "        \"--c_out\", \"1\",\n",
    "        \"--des\", \"Exp\",\n",
    "        \"--learning_rate\", \"0.001\",\n",
    "        \"--itr\", \"1\",\n",
    "        \"--gpu\", \"0\",\n",
    "        \"--seed\", \"4723\",\n",
    "    ]\n",
    ")\n",
    "args.use_gpu = True if torch.cuda.is_available() and args.use_gpu else False\n",
    "fix_seed = args.seed\n",
    "random.seed(fix_seed)\n",
    "torch.manual_seed(fix_seed)\n",
    "np.random.seed(fix_seed)\n",
    "\n",
    "if args.use_gpu:\n",
    "    if args.use_multi_gpu:\n",
    "        args.devices = args.devices.replace(' ', '')\n",
    "        device_ids = args.devices.split(',')\n",
    "        args.device_ids = [int(id_) for id_ in device_ids]\n",
    "        args.gpu = args.device_ids[0]\n",
    "    else:\n",
    "        torch.cuda.set_device(args.gpu)\n",
    "\n",
    "print('Args in experiment:')\n",
    "print(args)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "a3482668-1db0-401c-801f-86056689faf8",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Use CPU\n",
      "my hack 0\n",
      "0\n",
      "8640\n",
      "train 8497\n",
      "Total trainable parameters: 0.27M\n",
      ">>>>>>>start training : ETTh2_96_48_Koopa_ETTh2_ftS_sl96_pl48_segl48_dyna64_h512_l2_nb4_a0.2_Exp_0>>>>>>>>>>>>>>>>>>>>>>>>>>\n",
      "my hack 0\n",
      "0\n",
      "8640\n",
      "train 8497\n",
      "my hack 1\n",
      "8544\n",
      "11520\n",
      "val 2833\n",
      "my hack 2\n",
      "11424\n",
      "14400\n",
      "test 2833\n",
      "\titers: 100, epoch: 1 | loss: 0.1248656\n",
      "\tspeed: 0.1190s/iter; left time: 303.6096s\n",
      "\titers: 200, epoch: 1 | loss: 0.1741938\n",
      "\tspeed: 0.0201s/iter; left time: 49.2802s\n",
      "Epoch: 1 cost time: 64.76820111274719\n",
      "Epoch: 1, Steps: 265 | Train Loss: 0.1465977 Vali Loss: 0.1384558 Test Loss: 0.1013507\n",
      "Validation loss decreased (inf --> 0.138456).  Saving model ...\n",
      "Updating learning rate to 0.001\n",
      "\titers: 100, epoch: 2 | loss: 0.0876046\n",
      "\tspeed: 1.8566s/iter; left time: 4244.2496s\n",
      "\titers: 200, epoch: 2 | loss: 0.1095970\n",
      "\tspeed: 0.0201s/iter; left time: 43.8509s\n",
      "Epoch: 2 cost time: 64.83517384529114\n",
      "Epoch: 2, Steps: 265 | Train Loss: 0.1314647 Vali Loss: 0.1391251 Test Loss: 0.1015423\n",
      "EarlyStopping counter: 1 out of 3\n",
      "Updating learning rate to 0.0005\n",
      "\titers: 100, epoch: 3 | loss: 0.1314511\n",
      "\tspeed: 1.8511s/iter; left time: 3741.0610s\n",
      "\titers: 200, epoch: 3 | loss: 0.1630052\n",
      "\tspeed: 0.0203s/iter; left time: 38.9534s\n",
      "Epoch: 3 cost time: 64.79004383087158\n",
      "Epoch: 3, Steps: 265 | Train Loss: 0.1187189 Vali Loss: 0.1448411 Test Loss: 0.1057023\n",
      "EarlyStopping counter: 2 out of 3\n",
      "Updating learning rate to 0.00025\n",
      "\titers: 100, epoch: 4 | loss: 0.1932869\n",
      "\tspeed: 1.8507s/iter; left time: 3249.7895s\n",
      "\titers: 200, epoch: 4 | loss: 0.1856481\n",
      "\tspeed: 0.0201s/iter; left time: 33.3563s\n",
      "Epoch: 4 cost time: 64.79884696006775\n",
      "Epoch: 4, Steps: 265 | Train Loss: 0.1100094 Vali Loss: 0.1369175 Test Loss: 0.1039057\n",
      "Validation loss decreased (0.138456 --> 0.136918).  Saving model ...\n",
      "Updating learning rate to 0.000125\n",
      "\titers: 100, epoch: 5 | loss: 0.0974240\n",
      "\tspeed: 1.8536s/iter; left time: 2763.6831s\n",
      "\titers: 200, epoch: 5 | loss: 0.1038086\n",
      "\tspeed: 0.0204s/iter; left time: 28.4189s\n",
      "Epoch: 5 cost time: 64.87527298927307\n",
      "Epoch: 5, Steps: 265 | Train Loss: 0.1040654 Vali Loss: 0.1386887 Test Loss: 0.1037445\n",
      "EarlyStopping counter: 1 out of 3\n",
      "Updating learning rate to 6.25e-05\n",
      "\titers: 100, epoch: 6 | loss: 0.0854347\n",
      "\tspeed: 1.8596s/iter; left time: 2279.8871s\n",
      "\titers: 200, epoch: 6 | loss: 0.0838616\n",
      "\tspeed: 0.0210s/iter; left time: 23.6547s\n",
      "Epoch: 6 cost time: 65.27991890907288\n",
      "Epoch: 6, Steps: 265 | Train Loss: 0.1000089 Vali Loss: 0.1378460 Test Loss: 0.1032953\n",
      "EarlyStopping counter: 2 out of 3\n",
      "Updating learning rate to 3.125e-05\n",
      "\titers: 100, epoch: 7 | loss: 0.1196318\n",
      "\tspeed: 1.8537s/iter; left time: 1781.3862s\n",
      "\titers: 200, epoch: 7 | loss: 0.1113608\n",
      "\tspeed: 0.0204s/iter; left time: 17.5473s\n",
      "Epoch: 7 cost time: 64.92085075378418\n",
      "Epoch: 7, Steps: 265 | Train Loss: 0.0978645 Vali Loss: 0.1392980 Test Loss: 0.1043487\n",
      "EarlyStopping counter: 3 out of 3\n",
      "Early stopping\n",
      ">>>>>>>testing : ETTh2_96_48_Koopa_ETTh2_ftS_sl96_pl48_segl48_dyna64_h512_l2_nb4_a0.2_Exp_0<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<\n",
      "my hack 2\n",
      "11424\n",
      "14400\n",
      "test 2833\n",
      "test shape: (2833, 48, 1) (2833, 48, 1)\n",
      "test shape: (2833, 48, 1) (2833, 48, 1)\n",
      "mse:13.944741249084473, mae:2.815606117248535\n"
     ]
    }
   ],
   "source": [
    "Exp = Exp_Main\n",
    "\n",
    "if args.is_training:\n",
    "    for ii in range(args.itr):\n",
    "        # setting record of experiments\n",
    "        setting = '{}_{}_{}_ft{}_sl{}_pl{}_segl{}_dyna{}_h{}_l{}_nb{}_a{}_{}_{}'.format(\n",
    "            args.model_id,\n",
    "            args.model,\n",
    "            args.data,\n",
    "            args.features,\n",
    "            args.seq_len,\n",
    "            args.pred_len,\n",
    "            args.seg_len,\n",
    "            args.dynamic_dim,\n",
    "            args.hidden_dim,\n",
    "            args.hidden_layers,\n",
    "            args.num_blocks,\n",
    "            args.alpha,\n",
    "            args.des, ii)\n",
    "\n",
    "        exp = Exp(args)  # set experiments\n",
    "        model_size = sum(p.numel() for p in exp.model.parameters() if p.requires_grad)\n",
    "        print(f'Total trainable parameters: {model_size / 1e6:.2f}M')\n",
    "\n",
    "        print('>>>>>>>start training : {}>>>>>>>>>>>>>>>>>>>>>>>>>>'.format(setting))\n",
    "        exp.train(setting)\n",
    "\n",
    "        print('>>>>>>>testing : {}<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<'.format(setting))\n",
    "        exp.test(setting)\n",
    "\n",
    "        if args.do_predict:\n",
    "            print('>>>>>>>predicting : {}<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<'.format(setting))\n",
    "            exp.predict(setting, True)\n",
    "\n",
    "        torch.cuda.empty_cache()\n",
    "else:\n",
    "    ii = 0\n",
    "    setting = '{}_{}_{}_ft{}_sl{}_pl{}_segl{}_dyna{}_h{}_l{}_nb{}_a{}_{}_{}'.format(\n",
    "        args.model_id,\n",
    "        args.model,\n",
    "        args.data,\n",
    "        args.features,\n",
    "        args.seq_len,\n",
    "        args.pred_len,\n",
    "        args.seg_len,\n",
    "        args.dynamic_dim,\n",
    "        args.hidden_dim,\n",
    "        args.hidden_layers,\n",
    "        args.num_blocks,\n",
    "        args.alpha,\n",
    "        args.des, ii)\n",
    "\n",
    "    exp = Exp(args)  # set experiments\n",
    "    print('>>>>>>>testing : {}<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<<'.format(setting))\n",
    "    exp.test(setting, test=1)\n",
    "    torch.cuda.empty_cache()"
   ]
  }
 ],
 "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.13.0"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
