{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "import matplotlib.pyplot as plt\n",
    "import networkx as nx\n",
    "import random\n",
    "\n",
    "\n",
    "import time\n",
    "from marlAcrl import marlAcrl\n",
    "from tqdm.notebook import tqdm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def experiment_scaling_fixed_p(num_nodes_list, p, repeats=5, max_trials=100):\n",
    "    \"\"\"\n",
    "    Ejecuta experimentos con un valor fijo de p para grafos Erdos-Rényi,\n",
    "    asegurándose de que el grafo sea conexo.\n",
    "\n",
    "    Parameters\n",
    "    ----------\n",
    "    num_nodes_list : list of int\n",
    "        Lista de tamaños de grafo (número de nodos) a probar.\n",
    "    p : float\n",
    "        Probabilidad fija de conexión entre dos nodos (Erdos-Rényi).\n",
    "    repeats : int\n",
    "        Número de repeticiones por cada tamaño de grafo.\n",
    "    max_trials : int\n",
    "        Número máximo de intentos para generar un grafo conexo.\n",
    "\n",
    "    Returns\n",
    "    -------\n",
    "    results : list of (n, mean_time, std_time)\n",
    "        Tuplas con (n, tiempo_medio, desviación_estándar).\n",
    "    \"\"\"\n",
    "    results = []\n",
    "    plt.figure(figsize=(10, 6))\n",
    "\n",
    "    for n in tqdm(num_nodes_list):\n",
    "        times = []\n",
    "\n",
    "        for seed in range(repeats):\n",
    "            # Intentamos generar un grafo conexo hasta max_trials veces.\n",
    "            for _ in range(max_trials):\n",
    "                G = nx.erdos_renyi_graph(n, p, seed=np.random.randint(1e9))\n",
    "                if nx.is_connected(G):\n",
    "                    # Éxito: grafo conexo\n",
    "                    break\n",
    "            else:\n",
    "                # Si no se logra tras max_trials intentos\n",
    "                raise ValueError(f\"No se pudo generar un grafo conexo para n={n} con p={p}.\")\n",
    "\n",
    "            # Construimos la lista de conexiones para el 'manager'\n",
    "            connections = list(G.edges)\n",
    "            \n",
    "            # Ejemplo simple: buildings al azar 80% tipo 1, 20% tipo 5\n",
    "            buildings = [1 if np.random.random() > 0.2 else 5 for _ in range(n)]\n",
    "            \n",
    "            # crear manager\n",
    "            manager = marlAcrl(\n",
    "                connections=connections,\n",
    "                buildings=buildings,\n",
    "                models=['all_reg' if b == 1 else 'all_double_reg' for b in buildings],\n",
    "                nu_val=1.5,\n",
    "                lambda_val=1.5,\n",
    "                derenv=True\n",
    "            )\n",
    "            \n",
    "            # Configuramos constraints\n",
    "            manager.set_costraint(thresh=0.27)\n",
    "            \n",
    "            start_time = time.time()\n",
    "            # Ejecutar algoritmo\n",
    "            manager.run_episode(\n",
    "                epochs=2000,\n",
    "                consensus=True,\n",
    "                lamdaStep=0.01,\n",
    "                consensus_steps=1\n",
    "            )\n",
    "            elapsed = time.time() - start_time\n",
    "            times.append(elapsed)\n",
    "        \n",
    "        mean_time = np.mean(times)\n",
    "        std_time = np.std(times)\n",
    "\n",
    "        def runnin(arr):\n",
    "            return np.cumsum(arr) / np.arange(1, len(arr) + 1)\n",
    "\n",
    "        samples, num_agents = manager.lambdas_list.shape\n",
    "        t = np.arange(samples)\n",
    "        # Compute running averages for each agent\n",
    "        running_lambdas = np.array([runnin(manager.lambdas_list[:, i]) for i in range(num_agents)]).T\n",
    "\n",
    "        # Compute mean, min, and max of the running averages across agents\n",
    "        mean_lambda = np.mean(running_lambdas, axis=1)\n",
    "        min_lambda = np.min(running_lambdas, axis=1)\n",
    "        max_lambda = np.max(running_lambdas, axis=1)\n",
    "\n",
    "        # Plot the mean with fill between min and max\n",
    "        \n",
    "        plt.plot(t, mean_lambda, label=f'Mean Lambda (Running Average)- {n} agents')\n",
    "        plt.fill_between(t, min_lambda, max_lambda, alpha=0.2)\n",
    "        plt.title('Mean Lambda Across Agents (Running Average)')\n",
    "        plt.xlabel('Time')\n",
    "        plt.ylabel('Lambda Value')\n",
    "        \n",
    "        \n",
    "        results.append((n, mean_time, std_time))\n",
    "        print(f\"n = {n}, p = {p:.4f}, tiempo medio = {mean_time:.4f}s ± {std_time:.4f}s\")\n",
    "\n",
    "    plt.legend()\n",
    "    plt.tight_layout()\n",
    "    plt.savefig(\"lambda_conv_increa_ag.pdf\")\n",
    "    plt.show()\n",
    "    \n",
    "    return results\n",
    "\n",
    "\n",
    "# Ejemplo de uso y graficado\n",
    "if __name__ == \"__main__\":\n",
    "    # Diferentes tamaños de grafo\n",
    "    num_nodes_list = [10, 20, 50, 300, 1000]\n",
    "    \n",
    "    # Probabilidad fija de conexión\n",
    "    p = 0.2  # Ajusta según tus necesidades (densidad)\n",
    "    \n",
    "    # Ejecutar experimento\n",
    "    res = experiment_scaling_fixed_p(num_nodes_list, p, repeats=5, max_trials=100)\n",
    "    \n",
    "    # Organizar resultados para graficar\n",
    "    ns = [r[0] for r in res]\n",
    "    mean_times = [r[1] for r in res]\n",
    "    std_times = [r[2] for r in res]\n",
    "    \n",
    "    plt.figure(figsize=(8,5))\n",
    "    plt.errorbar(ns, mean_times, yerr=std_times, fmt='o-', capsize=5, label=\"Execution time\")\n",
    "    plt.xlabel(\"Number of agents (n)\")\n",
    "    plt.ylabel(\"Execution time (segundos)\")\n",
    "    plt.title(\"Algorithm scalability\")\n",
    "    plt.grid(True)\n",
    "    plt.legend()\n",
    "    plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pickle\n",
    "\n",
    "# Save the variables to a file\n",
    "with open('time_complexity_results.pkl', 'wb') as f:\n",
    "    pickle.dump({'ns': ns, 'mean_times': mean_times, 'std_times': std_times}, f)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "marl",
   "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.12.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
