{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": "# FedQAP Multi-Model Experiments\n\nThis notebook runs federated learning experiments with different time series forecasting models using the FedQAP framework.\n\n## Available Models\n- **SpikeRNN**: Spiking neural network for time series\n- **iTransformer**: Inverted transformer architecture\n- **Spikformer**: Spike-based transformer\n- **TimeMixer**: Time mixing neural network\n- **DLinear**: Decomposition linear model"
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import sys\n",
    "import runpy\n",
    "import os\n",
    "import time\n",
    "from datetime import datetime\n",
    "\n",
    "# Change to parent directory\n",
    "os.chdir(\"..\")\n",
    "\n",
    "print(f\"Current working directory: {os.getcwd()}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Configuration\n",
    "\n",
    "Common experimental settings for all models:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": "# Common configuration\nCOMMON_CONFIG = {\n    \"dataset\": \"electricity\",\n    \"num_clients\": \"20\",\n    \"max_features\": \"20\",\n    \"window\": \"96\",\n    \"horizon\": \"48\",\n    \"batch_size\": \"128\",\n    \"global_epochs\": \"50\",\n    \"local_epochs\": \"1\",\n    \"lr\": \"1e-3\",\n    \"optimizer\": \"adam\",\n    \"seed\": \"42\",\n    \"device\": \"cuda\",\n    \"frac\": \"1.0\",\n    \"dim\": \"128\",\n    \"early_stop\": True,\n    \"patience\": \"10\"\n}\n\n# Available models\nMODELS = [\"spikernn\", \"itransformer\", \"spikformer\", \"timemixer\", \"dlinear\"]\n\nprint(\"Configuration loaded:\")\nfor key, value in COMMON_CONFIG.items():\n    print(f\"  {key}: {value}\")\nprint(f\"\\nAvailable models: {', '.join(MODELS)}\")"
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Helper Functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def run_model_experiment(model_name: str, config: dict = None):\n",
    "    \"\"\"\n",
    "    Run experiment for a specific model\n",
    "    \n",
    "    Args:\n",
    "        model_name: Name of the model to run\n",
    "        config: Optional config overrides\n",
    "    \"\"\"\n",
    "    if config is None:\n",
    "        config = COMMON_CONFIG.copy()\n",
    "    \n",
    "    print(f\"\\n{'='*60}\")\n",
    "    print(f\"🚀 Starting experiment: {model_name.upper()}\")\n",
    "    print(f\"⏰ Time: {datetime.now().strftime('%Y-%m-%d %H:%M:%S')}\")\n",
    "    print(f\"{'='*60}\")\n",
    "    \n",
    "    # Build sys.argv\n",
    "    args = [\"main.py\"]\n",
    "    for key, value in config.items():\n",
    "        if key == \"early_stop\" and value:\n",
    "            args.append(\"--early_stop\")\n",
    "        else:\n",
    "            args.extend([f\"--{key}\", str(value)])\n",
    "    \n",
    "    # Add model-specific settings\n",
    "    args.extend([\"--model_name\", model_name])\n",
    "    args.extend([\"--experiment\", f\"electricity_{model_name}_50global_96-48_dim128\"])\n",
    "    \n",
    "    start_time = time.time()\n",
    "    \n",
    "    try:\n",
    "        # Set sys.argv and run\n",
    "        sys.argv = args\n",
    "        runpy.run_path(\"main.py\", run_name=\"__main__\")\n",
    "        \n",
    "        elapsed = time.time() - start_time\n",
    "        print(f\"\\n✅ {model_name.upper()} completed successfully!\")\n",
    "        print(f\"⏱️  Time: {elapsed:.1f}s ({elapsed/60:.1f}min)\")\n",
    "        return True, elapsed\n",
    "        \n",
    "    except Exception as e:\n",
    "        elapsed = time.time() - start_time\n",
    "        print(f\"\\n❌ {model_name.upper()} failed after {elapsed:.1f}s\")\n",
    "        print(f\"Error: {e}\")\n",
    "        import traceback\n",
    "        traceback.print_exc()\n",
    "        return False, elapsed"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Individual Model Experiments\n",
    "\n",
    "Run each model individually. Uncomment the cells for the models you want to run:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# SpikeRNN Experiment\n",
    "success, time_taken = run_model_experiment(\"spikernn\")\n",
    "print(f\"SpikeRNN result: {'Success' if success else 'Failed'} ({time_taken/60:.1f} min)\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": "# iTransformer Experiment  \n# success, time_taken = run_model_experiment(\"itransformer\")\n# print(f\"iTransformer result: {'Success' if success else 'Failed'} ({time_taken/60:.1f} min)\")"
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": "# Spikformer Experiment\n# success, time_taken = run_model_experiment(\"spikformer\")\n# print(f\"Spikformer result: {'Success' if success else 'Failed'} ({time_taken/60:.1f} min)\")"
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Spikformer Experiment\n",
    "# success, time_taken = run_model_experiment(\"spikformer\")\n",
    "# print(f\"Spikformer result: {'Success' if success else 'Failed'} ({time_taken/60:.1f} min)\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# TimeMixer Experiment\n",
    "# success, time_taken = run_model_experiment(\"timemixer\")\n",
    "# print(f\"TimeMixer result: {'Success' if success else 'Failed'} ({time_taken/60:.1f} min)\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# DLinear Experiment\n",
    "# success, time_taken = run_model_experiment(\"dlinear\")\n",
    "# print(f\"DLinear result: {'Success' if success else 'Failed'} ({time_taken/60:.1f} min)\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Run All Models Sequentially\n",
    "\n",
    "**⚠️ Warning**: This will take several hours to complete!"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Uncomment to run all models (takes several hours!)\n",
    "# results = {}\n",
    "# total_start = time.time()\n",
    "\n",
    "# print(\"🔬 Starting multi-model comparison...\")\n",
    "# print(f\"📊 Models: {', '.join(MODELS)}\")\n",
    "\n",
    "# for i, model in enumerate(MODELS, 1):\n",
    "#     print(f\"\\n📈 Progress: {i}/{len(MODELS)}\")\n",
    "#     success, elapsed = run_model_experiment(model)\n",
    "#     results[model] = {'success': success, 'time': elapsed}\n",
    "\n",
    "# total_time = time.time() - total_start\n",
    "\n",
    "# print(f\"\\n{'='*80}\")\n",
    "# print(\"📊 EXPERIMENT SUMMARY\")\n",
    "# print(f\"{'='*80}\")\n",
    "# print(f\"🕐 Total time: {total_time/60:.1f} min ({total_time/3600:.2f} hours)\")\n",
    "# print(\"\\n📋 Results:\")\n",
    "# for model, result in results.items():\n",
    "#     status = \"✅ Success\" if result['success'] else \"❌ Failed\"\n",
    "#     print(f\"  {model:15s}: {status} ({result['time']/60:.1f} min)\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Local-Only Training Example\n",
    "\n",
    "Run local-only training (no federation) for comparison:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Local-only training example\n",
    "local_config = COMMON_CONFIG.copy()\n",
    "local_config[\"local_epochs\"] = \"100\"  # More epochs for local training\n",
    "\n",
    "sys.argv = [\n",
    "    \"main.py\",\n",
    "    \"--dataset\", \"electricity\",\n",
    "    \"--num_clients\", \"20\", \n",
    "    \"--max_features\", \"20\",\n",
    "    \"--model_name\", \"spikernn\",\n",
    "    \"--window\", \"96\",\n",
    "    \"--horizon\", \"48\",\n",
    "    \"--batch_size\", \"128\",\n",
    "    \"--local_epochs\", \"100\",\n",
    "    \"--lr\", \"1e-3\",\n",
    "    \"--optimizer\", \"adam\",\n",
    "    \"--seed\", \"42\",\n",
    "    \"--device\", \"cuda\",\n",
    "    \"--experiment\", \"electricity_spikernn_local_only_96-48\",\n",
    "    \"--dim\", \"128\",\n",
    "    \"--early_stop\",\n",
    "    \"--patience\", \"10\",\n",
    "    \"--local_only\"  # Enable local-only mode\n",
    "]\n",
    "\n",
    "print(\"🏠 Running local-only training...\")\n",
    "# runpy.run_path(\"main.py\", run_name=\"__main__\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Results Analysis\n",
    "\n",
    "After running experiments, you can analyze results:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import json\n",
    "import pandas as pd\n",
    "from pathlib import Path\n",
    "\n",
    "def load_experiment_results():\n",
    "    \"\"\"Load results from all experiments\"\"\"\n",
    "    output_dir = Path(\"output\")\n",
    "    results = []\n",
    "    \n",
    "    for exp_dir in output_dir.glob(\"electricity_*\"):\n",
    "        meta_file = exp_dir / \"results\" / \"meta.json\"\n",
    "        if meta_file.exists():\n",
    "            with open(meta_file) as f:\n",
    "                meta = json.load(f)\n",
    "            results.append({\n",
    "                'experiment': exp_dir.name,\n",
    "                'model': meta.get('model_name', 'unknown'),\n",
    "                **meta\n",
    "            })\n",
    "    \n",
    "    return pd.DataFrame(results)\n",
    "\n",
    "# Load and display results\n",
    "# results_df = load_experiment_results()\n",
    "# print(\"📊 Experiment Results:\")\n",
    "# print(results_df[['experiment', 'model', 'hidden_size', 'lr']].head())"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## TensorBoard Visualization\n",
    "\n",
    "Launch TensorBoard to compare results:"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# TensorBoard setup\n",
    "print(\"📊 TensorBoard Commands:\")\n",
    "print(\"\\n1. Compare all experiments:\")\n",
    "print(\"   tensorboard --logdir=output/\")\n",
    "print(\"\\n2. View specific experiment:\")\n",
    "print(\"   tensorboard --logdir=output/electricity_spikernn_50global_96-48_dim128/\")\n",
    "print(\"\\n3. Then open: http://localhost:6006\")\n",
    "\n",
    "# Uncomment to launch TensorBoard directly\n",
    "# import subprocess\n",
    "# subprocess.Popen([\"tensorboard\", \"--logdir=output/\"])\n",
    "# print(\"🚀 TensorBoard launched! Open http://localhost:6006\")"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}