{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "c62e1b42",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "import torch.nn as nn\n",
    "import numpy as np\n",
    "# Sample‑or‑deterministic wrapper for the Gaussian actor\n",
    "from torch.distributions import Normal\n",
    "\n",
    "def env_action(raw_actor_out, sample=False):\n",
    "    mu, log_sigma = raw_actor_out.split(2, dim=-1)\n",
    "    a = Normal(mu, log_sigma.exp()).sample() if sample else mu\n",
    "    # map (-inf, inf) -> (0, 1)\n",
    "    return (torch.tanh(a) + 1) * 0.5\n",
    "\n",
    "\n",
    "class MLP(nn.Module):\n",
    "    def __init__(self, input_dim, output_dim, config):\n",
    "        \"\"\"\n",
    "        Build an MLP network using config.\n",
    "        \n",
    "        config should provide:\n",
    "          - num_cells: list of hidden layer sizes, e.g. [256, 256]\n",
    "          - layer_class: e.g. torch.nn.Linear\n",
    "          - activation_class: e.g. torch.nn.Tanh (or None)\n",
    "          - activation_kwargs: dict or None\n",
    "        \"\"\"\n",
    "        super(MLP, self).__init__()\n",
    "        layers = []\n",
    "        current_dim = input_dim\n",
    "        for cell_size in config.num_cells:\n",
    "            layers.append(config.layer_class(current_dim, cell_size))\n",
    "            if config.activation_class is not None:\n",
    "                if config.activation_kwargs is None:\n",
    "                    layers.append(config.activation_class())\n",
    "                else:\n",
    "                    layers.append(config.activation_class(**config.activation_kwargs))\n",
    "            current_dim = cell_size\n",
    "        # Output layer (no activation)\n",
    "        layers.append(config.layer_class(current_dim, output_dim))\n",
    "        self.net = nn.Sequential(*layers)\n",
    "        \n",
    "    def forward(self, x):\n",
    "        return self.net(x)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "3e0166d4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Model configuration:\n",
      "  num_cells: [256, 256]\n",
      "  layer_class: <class 'torch.nn.modules.linear.Linear'>\n",
      "  activation_class: <class 'torch.nn.modules.activation.Tanh'>\n"
     ]
    }
   ],
   "source": [
    "# Adjust the import path as necessary; here we assume the configuration is stored in benchmarl/conf/model/layers/mlp.yaml\n",
    "from benchmarl.models.mlp import MlpConfig\n",
    "\n",
    "# Load model configuration from YAML. This returns an object with the attributes shown.\n",
    "model_config = MlpConfig.get_from_yaml()\n",
    "critic_model_config = MlpConfig.get_from_yaml()\n",
    "\n",
    "print(\"Model configuration:\")\n",
    "print(\"  num_cells:\", model_config.num_cells)\n",
    "print(\"  layer_class:\", model_config.layer_class)\n",
    "print(\"  activation_class:\", model_config.activation_class)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "3999f37c",
   "metadata": {},
   "outputs": [],
   "source": [
    "def convert_policy_state_dict(state_dict_policy):\n",
    "    \"\"\"\n",
    "    Convert keys from the checkpoint's policy state dict into keys that match the keys of our MLP\n",
    "    instance (i.e., keys should be prefixed with \"net.\" and should not include extra keys like \n",
    "    __batch_size or __device).\n",
    "    \"\"\"\n",
    "    new_state_dict = {}\n",
    "    prefix = \"module.0.module.0.module.0.mlp.params.\"\n",
    "    for k, v in state_dict_policy.items():\n",
    "        # Skip keys that include \"__batch_size\" or \"__device\"\n",
    "        if \"__batch_size\" in k or \"__device\" in k:\n",
    "            continue\n",
    "        if k.startswith(prefix):\n",
    "            new_key = \"net.\" + k[len(prefix):]  # remove the original prefix and replace with \"net.\"\n",
    "            new_state_dict[new_key] = v\n",
    "        else:\n",
    "            # Optionally log or skip other keys.\n",
    "            print(\"Skipping key:\", k)\n",
    "    return new_state_dict\n",
    "\n",
    "\n",
    "def convert_critic_state_dict(state_dict_loss_agents):\n",
    "    \"\"\"\n",
    "    Convert keys for the critic network. We expect keys starting with\n",
    "    'critic_network_params.module.0.mlp.params.' and we convert them to 'net.*'.\n",
    "    \"\"\"\n",
    "    new_state_dict = {}\n",
    "    prefix = \"critic_network_params.module.0.mlp.params.\"\n",
    "    for k, v in state_dict_loss_agents.items():\n",
    "        # Skip keys that include \"__batch_size\" or \"__device\"\n",
    "        if \"__batch_size\" in k or \"__device\" in k:\n",
    "            continue\n",
    "        if k.startswith(prefix):\n",
    "            new_key = \"net.\" + k[len(prefix):]\n",
    "            new_state_dict[new_key] = v\n",
    "        else:\n",
    "            # Uncomment the next line to see which keys are skipped\n",
    "            # print(\"Skipping key in critic state dict:\", k)\n",
    "            pass\n",
    "    return new_state_dict\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "f4fe2db3",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Inferred Policy Network: input_dim=4, output_dim=4\n",
      "Inferred Critic Network: input_dim=24, output_dim=1\n"
     ]
    }
   ],
   "source": [
    "checkpoint_path = \"mappoSmart1/checkpoints/checkpoint_2000000.pt\"\n",
    "\n",
    "# Load the checkpoint state dictionary (it is an OrderedDict)\n",
    "checkpoint = torch.load(checkpoint_path, map_location=torch.device(\"cpu\"))\n",
    "\n",
    "# Prefer using the 'collector.policy_state_dict' if available.\n",
    "# (Based on your state_dict structure, policy parameters are stored in collector.)\n",
    "collector = checkpoint[\"collector\"]\n",
    "policy_state_dict_ckpt = collector[\"policy_state_dict\"]\n",
    "\n",
    "\n",
    "# Infer policy input and output dimensions.\n",
    "actor_first_key = \"module.0.module.0.module.0.mlp.params.0.weight\"\n",
    "actor_last_key = \"module.0.module.0.module.0.mlp.params.4.weight\"\n",
    "\n",
    "if actor_first_key not in policy_state_dict_ckpt:\n",
    "    raise KeyError(f\"Key {actor_first_key} not found in policy state dict.\")\n",
    "\n",
    "w0 = policy_state_dict_ckpt[actor_first_key]  # shape: (hidden_dim, input_dim)\n",
    "input_dim_policy = w0.shape[1]\n",
    "\n",
    "if actor_last_key not in policy_state_dict_ckpt:\n",
    "    raise KeyError(f\"Key {actor_last_key} not found in policy state dict.\")\n",
    "\n",
    "w_last = policy_state_dict_ckpt[actor_last_key] # shape: (output_dim, hidden_dim_last)\n",
    "output_dim_policy = w_last.shape[0]\n",
    "\n",
    "print(f\"Inferred Policy Network: input_dim={input_dim_policy}, output_dim={output_dim_policy}\")\n",
    "\n",
    "# Similarly, infer critic dimensions from loss_agents dictionary.\n",
    "loss_agents_dict = checkpoint[\"loss_agents\"]\n",
    "critic_first_key = \"critic_network_params.module.0.mlp.params.0.weight\"\n",
    "critic_last_key = \"critic_network_params.module.0.mlp.params.4.weight\"\n",
    "\n",
    "if critic_first_key not in loss_agents_dict:\n",
    "    raise KeyError(f\"Key {critic_first_key} not found in critic state dict.\")\n",
    "    \n",
    "w0_critic = loss_agents_dict[critic_first_key]\n",
    "input_dim_critic = w0_critic.shape[1]\n",
    "\n",
    "if critic_last_key not in loss_agents_dict:\n",
    "    raise KeyError(f\"Key {critic_last_key} not found in critic state dict.\")\n",
    "    \n",
    "w_last_critic = loss_agents_dict[critic_last_key]\n",
    "output_dim_critic = w_last_critic.shape[0]\n",
    "\n",
    "print(f\"Inferred Critic Network: input_dim={input_dim_critic}, output_dim={output_dim_critic}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "5c7bf1e9",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Policy and critic networks instantiated with inferred dimensions.\n"
     ]
    }
   ],
   "source": [
    "# Instantiate the networks using the MLP builder and the loaded configuration.\n",
    "policy_net = MLP(input_dim_policy, output_dim_policy, model_config)\n",
    "critic_net = MLP(input_dim_critic, output_dim_critic, critic_model_config)\n",
    "\n",
    "print(\"Policy and critic networks instantiated with inferred dimensions.\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "1ae905f7",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Networks loaded with checkpoint parameters.\n"
     ]
    }
   ],
   "source": [
    "# Convert the policy state dict.\n",
    "converted_policy_dict = convert_policy_state_dict(policy_state_dict_ckpt)\n",
    "policy_net.load_state_dict(converted_policy_dict)\n",
    "policy_net.eval()\n",
    "\n",
    "# Convert the critic state dict.\n",
    "converted_critic_dict = convert_critic_state_dict(loss_agents_dict)\n",
    "critic_net.load_state_dict(converted_critic_dict)\n",
    "critic_net.eval()\n",
    "\n",
    "print(\"Networks loaded with checkpoint parameters.\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "2d648b09",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Evaluation environment built.\n"
     ]
    }
   ],
   "source": [
    "from vmas.simulator.environment import Environment\n",
    "from smart_grid import SmartGridScenario\n",
    "\n",
    "device = torch.device(\"cpu\")  # or \"cuda\" if available\n",
    "\n",
    "scenario = SmartGridScenario()\n",
    "env_kwargs = dict(\n",
    "    building_types=[5, 1, 1, 1, 5],\n",
    "    episode_length=80,\n",
    "    verbose=False,\n",
    ")\n",
    "env = Environment(\n",
    "    scenario=scenario,\n",
    "    max_steps=env_kwargs[\"episode_length\"],\n",
    "    device=device,\n",
    "    **env_kwargs,\n",
    ")\n",
    "\n",
    "\n",
    "print(\"Evaluation environment built.\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "f634e326",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Rollouts complete.\n"
     ]
    }
   ],
   "source": [
    "num_episodes = 5\n",
    "episode_length = env_kwargs[\"episode_length\"]\n",
    "\n",
    "all_episode_data = []  # To store metrics for each episode\n",
    "\n",
    "for ep in range(num_episodes):\n",
    "    # Reset environment. \"obs\" is a list where each element corresponds to an agent's observations.\n",
    "    obs = env.reset()  # Each element should be a torch tensor of shape (n_env, n_obs), e.g., (32, 4)\n",
    "    ep_data = {\"grid\": [], \"battery\": [], \"postponed\": []}\n",
    "\n",
    "    for t in range(episode_length):\n",
    "        actions_list = []\n",
    "        # Process each agent's observation individually.\n",
    "        for obs_i in obs:\n",
    "            # obs_i already is a tensor of shape (batch_dim, n_obs)\n",
    "            with torch.no_grad():\n",
    "                a_i = policy_net(obs_i)  # Expected shape: (batch_dim, action_dim)\n",
    "            actions_list.append(env_action(a_i).cpu())  # Convert each action tensor to numpy\n",
    "        \n",
    "        # Step the environment with the list of actions.\n",
    "        next_obs, rewards, dones, infos = env.step(actions_list)\n",
    "        \n",
    "        # Collect metrics from each agent:\n",
    "        grid_vals = []\n",
    "        battery_vals = []\n",
    "        postponed_vals = []\n",
    "        for agent in env.world.agents:\n",
    "            act = agent.action.u      # (batch_dim, 2)\n",
    "            grid_vals.append(act[:, 0].mean().item())\n",
    "            battery_vals.append(act[:, 1].mean().item())\n",
    "            postponed_vals.append(agent.state.postponed.mean().item())\n",
    "        \n",
    "        ep_data[\"grid\"].append(np.mean(grid_vals))\n",
    "        ep_data[\"battery\"].append(np.mean(battery_vals))\n",
    "        ep_data[\"postponed\"].append(np.mean(postponed_vals))\n",
    "        \n",
    "        obs = next_obs  # Update obs for the next timestep\n",
    "    all_episode_data.append(ep_data)\n",
    "\n",
    "print(\"Rollouts complete.\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "f5897697",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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9YmIiIjow6mU2Lm4uGDGjBmIiYkpsI4QAiEhIWjdujUWLFhQaAESERERkWpUOhV75MgRjB8/HlOmTEGNGjXg7u4OOzs7GBgY4NmzZ7h+/TpOnz4NPT09+Pv744cffijquImIiIjoDWrdoPjRo0fYsmULjh07hvv37+PFixcoUaIEatasiZYtW6JNmzbQ0VH7DiofFWfFEhER0edEndxF7SdPfO6Y2BEREdHnpEifPEFEREREnyYmdkRERERagokdERERkZZgYkdERESkJZjYEREREWmJ90rsjh8/ju+//x4eHh54/PgxAGDt2rU4ceJEoQZHRERERKpTO7Hbtm0bWrZsCUNDQ4SHhyM9PR0AkJycjBkzZhR6gERERESkGrUTu2nTpmHp0qVYsWIF9PT0pPL69evj4sWLhRocEREREalO7cTu5s2baNy4cZ5yMzMzPH/+vDBiIiIiIqL3oHZiZ2trizt37uQpP3HiBMqUKVMoQRERERGR+tRO7Pr3748hQ4bg7NmzkMlkiImJwfr16zFixAj89NNPRREjEREREalAV90NRo0ahcTERHh6euLly5do3LgxFAoFRowYgUGDBhVFjERERESkApkQQrzPhmlpabh+/TpycnJQqVIlmJiYFHZsRUKdB+kSERERaZo6uYvaI3a5jIyM4O7u/r6bExEREVEhUzux8/T0hEwmK3D9oUOHPiggIiIiIno/aid2NWrUUFrOzMxEREQErl69it69exdWXERERESkJrUTu99//z3f8smTJyMlJeWDAyIiIiKi9/Nez4rNz/fff49Vq1YV1u6IiIiISE2FltidPn0aBgYGhbU7IiIiIlKT2qdiO3XqpLQshEBsbCzCwsIwYcKEQguMiIiIiNSjdmJnbm6utKyjo4MKFSpg6tSp8PLyKrTAiIiIiEg9aid2AQEBRREHEREREX2gQrvGjoiIiIg0S6URu2LFir31psSve/r06QcFRERERETvR6XEbv78+UUcBhERERF9KJUSOz5RgoiIiOjTp/bkide9ePECmZmZSmVmZmYfFBARERERvR+1J0+kpqZi0KBBsLKygomJCYoVK6b0IiIiIiLNUDuxGzVqFA4dOoTFixdDoVDgr7/+wpQpU2BnZ4c1a9YURYxEREREpAK1T8Xu3r0ba9asQdOmTeHr64tGjRrBxcUFjo6OWL9+PXr0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WoKJHREREZGWYGJHREREpCWY2BERERFpCSZ2RERERFqCiR0RERGRlmBiR0RERKQlmNgRERERaQkmdkRERERagokdERERkZZgYkdERESkJZjYEREREWkJJnZEREREWoKJHREREZGWYGJHREREpCWY2BERERFpCSZ2RERERFqCiR0RERGRlmBiR0RERKQlmNgRERERaQkmdkRERERagokdERERkZZgYkdERESkJZjYEREREWkJJnZEREREWoKJHREREZGWYGJHREREpCWY2BERERFpCSZ2RERERFqCiR0RERGRlmBiR0RERKQlmNgRERERaQkmdkRERERagokdERERkZZgYkdERESkJZjYEREREWkJJnZEREREWoKJHREREZGWYGJHREREpCWY2BERERFpCY0ndosXL4azszMMDAzg5uaG48ePv7X+0aNH4ebmBgMDA5QpUwZLly79SJESERERfdo0mtgFBQVh6NChGDduHMLDw9GoUSO0bt0a0dHR+daPiopCmzZt0KhRI4SHh2Ps2LEYPHgwtm3b9pEjJyIiIvr0yIQQQlMHr1u3LmrVqoUlS5ZIZa6urvD29sbMmTPz1B89ejR27dqFyMhIqWzAgAG4dOkSTp8+rdIxk5KSYG5ujsTERJiZmX14I/IhhMCLrBdFsm8iIiL69BjqGkImkxXJvtXJXXSLJAIVZGRk4MKFCxgzZoxSuZeXF06dOpXvNqdPn4aXl5dSWcuWLbFy5UpkZmZCT08vzzbp6elIT0+XlpOSkgoh+rd7kfUCdTfULfLjEBER0afhbPezMNIz0nQYmjsVm5CQgOzsbFhbWyuVW1tbIy4uLt9t4uLi8q2flZWFhISEfLeZOXMmzM3NpZe9vX3hNICIiIjoE6OxEbtcbw5bCiHeOpSZX/38ynP5+/vDz89PWk5KSiry5M5Q1xBnu58t0mMQERHRp8NQ11DTIQDQYGJXokQJyOXyPKNz8fHxeUblctnY2ORbX1dXF5aWlvluo1AooFAoCidoFclksk9iOJaIiIi+LBo7Fauvrw83NzeEhIQolYeEhKB+/fr5buPh4ZGn/oEDB+Du7p7v9XVEREREXxKN3u7Ez88Pf/31F1atWoXIyEgMGzYM0dHRGDBgAIBXp1F79eol1R8wYAAePHgAPz8/REZGYtWqVVi5ciVGjBihqSYQERERfTI0eo1d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8ePH6NKlC4oVKwZLS0t06NAB9+/fl9YfOXIEderUgbGxMSwsLNCgQQM8ePAAgYGBmDJlCi5duiSNJAYGBgIAEhMT8cMPP8DKygpmZmb46quvcOnSJWmfubEvW7YM9vb2MDIywnfffYfnz58XYa8S0eeCiR0REYBDhw4hJiYGx44dw7x58zB58mS0a9cOxYoVw9mzZzFgwAAMGDAADx8+BACkpaXB09MTJiYmOHbsGE6cOAETExO0atUKGRkZyMrKgre3N5o0aYLLly/j9OnT+OGHHyCTydClSxcMHz4clStXRmxsLGJjY9GlSxcIIdC2bVvExcUhODgYFy5cQK1atdCsWTM8ffpUivXOnTvYvHkzdu/ejX379iEiIgIDBw7UVNcR0adEEBFpqYCAAGFubp6nfNKkSaJ69erScu/evYWjo6PIzs6WyipUqCAaNWokLWdlZQljY2OxceNGIYQQK1euFBUqVBA5OTlSnfT0dGFoaCj2798vnjx5IgCII0eO5BvbmzEIIURoaKgwMzMTL1++VCovW7asWLZsmbSdXC4XDx8+lNb//fffQkdHR8TGxr69Q4hI63HEjogIQOXKlaGj879fidbW1qhataq0LJfLYWlpifj4eADAhQsXcOfOHZiamkrX7BUvXhwvX77E3bt3Ubx4cfj4+KBly5Zo3749/vjjD6Xr//Jz4cIFpKSkwNLSUtqniYkJoqKicPfuXameg4MDSpcuLS17eHggJydH6VpCIvoycfIEEREAPT09pWWZTJZvWU5ODgAgJycHbm5uWL9+fZ59lSxZEgAQEBCAwYMHY9++fQgKCsL48eMREhKCevXq5RtDTk4ObG1tceTIkTzrLCwsCoxdJpMp/UtEXy4mdkRE76FWrVoICgqSJjkUpGbNmqhZsyb8/f3h4eGBDRs2oF69etDX10d2dnaefcbFxUFXVxdOTk4F7jM6OhoxMTGws7MDAJw+fRo6OjooX758obSNiD5fPBVLRPQeevTogRIlSqBDhw44fvw4oqKicPToUQwZMgSPHj1CVFQU/P39cfr0aTx48AAHDhzArVu34OrqCgBwcnJCVFQUIiIikJCQgPT0dDRv3hweHh7w9vbG/v37cf/+fZw6dQrjx49HWFiYdGwDAwP07t0bly5dwvHjxzF48GB07twZNjY2muoOIvpEMLEjInoPRkZGOHbsGBwcHNCpUye4urrC19cXL168gJmZGYyMjHDjxg188803KF++PH744QcMGjQI/fv3BwB88803aNWqFTw9PVGyZEls3LgRMpkMwcHBaNy4MXx9fVG+fHl07doV9+/fh7W1tXRsFxcXdOrUCW3atIGXlxeqVKmCxYsXa6oriOgTIhNCCE0HQUREqpk8eTJ27tyJiIgITYdCRJ8gjtgRERERaQkmdkRERERagqdiiYiIiLQER+yIiIiItAQTOyIiIiItwcSOiIiISEswsSMiIiLSEkzsiIiIiLQEEzsiIiIiLcHEjoiIiEhLMLEjIiIi0hJM7IiIiIi0xP8BCW6WQ94FoKEAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# ⬛▸ Average metrics across the evaluation episodes and plot them\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# Build (num_episodes, episode_length) arrays\n",
    "grid_arr      = np.array([ep[\"grid\"]      for ep in all_episode_data])\n",
    "battery_arr   = np.array([ep[\"battery\"]   for ep in all_episode_data])\n",
    "postponed_arr = np.array([ep[\"postponed\"] for ep in all_episode_data])\n",
    "\n",
    "# Mean over the episode axis\n",
    "mean_grid      = grid_arr.mean(axis=0)\n",
    "mean_battery   = battery_arr.mean(axis=0)\n",
    "mean_postponed = postponed_arr.mean(axis=0)\n",
    "\n",
    "timesteps = np.arange(mean_grid.shape[0])\n",
    "\n",
    "plt.figure()\n",
    "plt.plot(timesteps, mean_grid,      label=\"Grid consumption\")\n",
    "plt.plot(timesteps, mean_battery,   label=\"Battery consumption\")\n",
    "plt.plot(timesteps, mean_postponed, label=\"Postponed demand\")\n",
    "plt.xlabel(\"Timestep\")\n",
    "plt.ylabel(\"Average value (batch mean)\")\n",
    "plt.title(f\"Average behaviour over {grid_arr.shape[0]} evaluation episodes\")\n",
    "plt.legend()\n",
    "plt.tight_layout()\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "bee71a17",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "t=0 [0.        1.1352838 3.025     0.       ]\n"
     ]
    },
    {
     "ename": "AssertionError",
     "evalue": "Action for agent agent_0 has shape 4, but should have shape 2",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mAssertionError\u001b[0m                            Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[11], line 7\u001b[0m\n\u001b[1;32m      4\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m t \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(\u001b[38;5;241m5\u001b[39m):\n\u001b[1;32m      5\u001b[0m     \u001b[38;5;28mprint\u001b[39m(\u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mt=\u001b[39m\u001b[38;5;132;01m{\u001b[39;00mt\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m\"\u001b[39m,\n\u001b[1;32m      6\u001b[0m           obs[\u001b[38;5;241m0\u001b[39m][\u001b[38;5;241m0\u001b[39m, :]\u001b[38;5;241m.\u001b[39mnumpy())          \u001b[38;5;66;03m# battery, demand, price, postponed\u001b[39;00m\n\u001b[0;32m----> 7\u001b[0m     obs, \u001b[38;5;241m*\u001b[39m_ \u001b[38;5;241m=\u001b[39m \u001b[43mtmp_env\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mstep\u001b[49m\u001b[43m(\u001b[49m\u001b[43m[\u001b[49m\u001b[43mtorch\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mzeros_like\u001b[49m\u001b[43m(\u001b[49m\u001b[43mo\u001b[49m\u001b[43m)\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43;01mfor\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mo\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;129;43;01min\u001b[39;49;00m\u001b[43m \u001b[49m\u001b[43mobs\u001b[49m\u001b[43m]\u001b[49m\u001b[43m)\u001b[49m  \u001b[38;5;66;03m# dummy zero action\u001b[39;00m\n",
      "File \u001b[0;32m/opt/anaconda3/envs/benchmarl/lib/python3.10/site-packages/vmas/simulator/environment/environment.py:253\u001b[0m, in \u001b[0;36mEnvironment.step\u001b[0;34m(self, actions)\u001b[0m\n\u001b[1;32m    249\u001b[0m         actions[i]\u001b[38;5;241m.\u001b[39munsqueeze_(\u001b[38;5;241m-\u001b[39m\u001b[38;5;241m1\u001b[39m)\n\u001b[1;32m    250\u001b[0m     \u001b[38;5;28;01massert\u001b[39;00m (\n\u001b[1;32m    251\u001b[0m         actions[i]\u001b[38;5;241m.\u001b[39mshape[\u001b[38;5;241m0\u001b[39m] \u001b[38;5;241m==\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mnum_envs\n\u001b[1;32m    252\u001b[0m     ), \u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mActions used in input of env must be of len \u001b[39m\u001b[38;5;132;01m{\u001b[39;00m\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mnum_envs\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m, got \u001b[39m\u001b[38;5;132;01m{\u001b[39;00mactions[i]\u001b[38;5;241m.\u001b[39mshape[\u001b[38;5;241m0\u001b[39m]\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m\"\u001b[39m\n\u001b[0;32m--> 253\u001b[0m     \u001b[38;5;28;01massert\u001b[39;00m actions[i]\u001b[38;5;241m.\u001b[39mshape[\u001b[38;5;241m1\u001b[39m] \u001b[38;5;241m==\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mget_agent_action_size(\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39magents[i]), (\n\u001b[1;32m    254\u001b[0m         \u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mAction for agent \u001b[39m\u001b[38;5;132;01m{\u001b[39;00m\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39magents[i]\u001b[38;5;241m.\u001b[39mname\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m has shape \u001b[39m\u001b[38;5;132;01m{\u001b[39;00mactions[i]\u001b[38;5;241m.\u001b[39mshape[\u001b[38;5;241m1\u001b[39m]\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m,\u001b[39m\u001b[38;5;124m\"\u001b[39m\n\u001b[1;32m    255\u001b[0m         \u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124m but should have shape \u001b[39m\u001b[38;5;132;01m{\u001b[39;00m\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mget_agent_action_size(\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39magents[i])\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m\"\u001b[39m\n\u001b[1;32m    256\u001b[0m     )\n\u001b[1;32m    258\u001b[0m \u001b[38;5;66;03m# set action for each agent\u001b[39;00m\n\u001b[1;32m    259\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m i, agent \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28menumerate\u001b[39m(\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39magents):\n",
      "\u001b[0;31mAssertionError\u001b[0m: Action for agent agent_0 has shape 4, but should have shape 2"
     ]
    }
   ],
   "source": [
    "# --- peek at the first agent in env 0 for a few steps --------------\n",
    "tmp_env = env  # reuse the env you've already built\n",
    "obs = tmp_env.reset()\n",
    "for t in range(5):\n",
    "    print(f\"t={t}\",\n",
    "          obs[0][0, :].numpy())          # battery, demand, price, postponed\n",
    "    obs, *_ = tmp_env.step([torch.zeros_like(o) for o in obs])  # dummy zero action\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "benchmarl",
   "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.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
