{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import sys\n",
    "sys.path.append(os.path.abspath(\"<path>\"))\n",
    "\n",
    "import gymnasium as gym\n",
    "import numpy as np \n",
    "import pandas as pd \n",
    "import pickle as pkl \n",
    "import tqdm \n",
    "import torch\n",
    "import torch.nn as nn \n",
    "import torch.optim as optim \n",
    "import random\n",
    "\n",
    "from torch.utils.data import DataLoader \n",
    "#from utils.utils import Specs\n",
    "\n",
    "from ctrl_data import CollectRandomTransitions, TransitionDataset, BiCoGAN\n",
    "from ctrl_models import Generator, Discriminator, Encoder\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "\n",
    "\n",
    "import math\n",
    "import mlflow"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "envName = 'LunarLander-v3'\n",
    "class Specs:\n",
    "  def __init__(self,\n",
    "               continuous: bool= False,\n",
    "               gravity: float = -10.,\n",
    "               enable_wind: bool = False,\n",
    "               wind_power: float = 15.,\n",
    "               turbulence_power: float = 1.5):\n",
    "    self.continuous = continuous\n",
    "    self.gravity = gravity\n",
    "    self.enable_wind = enable_wind\n",
    "    self.wind_power = wind_power\n",
    "    self.turbulence_power = turbulence_power\n",
    "\n",
    "class Conf:\n",
    "    bsize = 128\n",
    "    udim = 128\n",
    "    alpha = 5\n",
    "    rho = 0.1 # 0.25\n",
    "    phi = 10\n",
    "    beta1 = 0.5 \n",
    "    beta2 = 0.999 \n",
    "    lr = 1e-4\n",
    "    disc_lr = 1e-5\n",
    "    num_epochs=50\n",
    "    pre_train_epochs=50\n",
    "    pre_train_lr=1e-3\n",
    "    action_dim =4\n",
    "\n",
    "    def __init__(self): \n",
    "        super().__init__()\n",
    "\n",
    "args = Conf()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Generating random transitions\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████████████████████████████████████████████████████████| 500/500 [00:01<00:00, 457.95it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "collected 47062 transitions.\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "# collect random transitions from a domain. \n",
    "tgtDom = Specs(enable_wind=False,turbulence_power=0., )\n",
    "\n",
    "traindata = CollectRandomTransitions(envName, tgtDom, Tmax=500)\n",
    "print(f\"collected {len(traindata)} transitions.\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "dataset = TransitionDataset(traindata,  num_actions=args.action_dim)\n",
    "loader = DataLoader(dataset, batch_size=args.bsize)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "368\n"
     ]
    }
   ],
   "source": [
    "s, a, r, s_, done = iter(loader).__next__()\n",
    "print(len(loader))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 58,
   "metadata": {},
   "outputs": [],
   "source": [
    "import math\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "import torch.optim as optim\n",
    "import mlflow\n",
    "import tqdm\n",
    "\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "\n",
    "\n",
    "class BiCoGAN: \n",
    "\n",
    "    def __init__(self, \n",
    "                 forward,\n",
    "                 backward,\n",
    "                 discriminator, \n",
    "                 args):\n",
    "        \n",
    "        self.forward = forward\n",
    "        self.backward = backward\n",
    "        self.args = args\n",
    "        self.discriminator = discriminator\n",
    "\n",
    "        self.global_steps = 0\n",
    "        self.global_pretrain_teps = 0\n",
    "\n",
    "        if args is not None:\n",
    "            # gamma schedule\n",
    "            self.getGamma = lambda t: min(self.args.alpha * math.exp(self.args.rho * t),\n",
    "                                          self.args.phi)\n",
    "            \n",
    "            self.fwd_optim = optim.Adam(self.forward.parameters(), \n",
    "                                        lr=self.args.pre_train_lr)\n",
    "            \n",
    "            self.disc_optim = optim.Adam(self.discriminator.parameters(),\n",
    "                                         lr=self.args.disc_lr)\n",
    "            \n",
    "            self.fwd_bwd_optim = optim.Adam(\n",
    "                list(self.forward.parameters()) + list(self.backward.parameters()), \n",
    "                lr=self.args.lr, betas=(self.args.beta1, self.args.beta2)\n",
    "            )\n",
    "            \n",
    "            self.adversarialLoss = nn.BCEWithLogitsLoss()\n",
    "            self.EFL = nn.MSELoss()\n",
    "\n",
    "    def TrainForwardBackward(self, batch):\n",
    "        s, a, r, nexts, done = batch\n",
    "        bsize = s.size(0)\n",
    "        \n",
    "        # forward model: noise → next state, done, reward\n",
    "        noise = torch.randn(bsize, self.args.udim).to(device)\n",
    "        nexts_hat, done_hat, r_hat = self.forward(s, a, noise)\n",
    "\n",
    "        # backward model: infer (s_hat, a_hat, u_hat) from (nexts, r)\n",
    "        s_hat, a_hat, u_hat = self.backward(nexts, r)\n",
    "\n",
    "        # forward loss (adversarial): (s,a,r_hat,nexts_hat,noise) as \"real\"\n",
    "        fwd_logits = self.discriminator(s, a, r_hat, nexts_hat, noise)\n",
    "        real_labels = torch.ones(bsize, 1).to(device)\n",
    "        fwd_loss = self.adversarialLoss(fwd_logits, real_labels)\n",
    "\n",
    "        # backward loss (adversarial): (s,a,r_hat,nexts_hat,u_hat) as \"fake\"\n",
    "        bwd_logits = self.discriminator(s, a, r_hat, nexts_hat, u_hat)\n",
    "        fake_labels = torch.zeros(bsize, 1).to(device)\n",
    "        bwd_loss = self.adversarialLoss(bwd_logits, fake_labels)\n",
    "\n",
    "        # encoder–forward consistency loss (EFL)\n",
    "        efl_loss = self.EFL(\n",
    "            torch.cat([s, a, r, nexts], dim=1),\n",
    "            torch.cat([s_hat, a_hat, r_hat, nexts_hat], dim=1)\n",
    "        )\n",
    "\n",
    "        total_loss = fwd_loss + bwd_loss + self.gamma * efl_loss\n",
    "        self.fwd_bwd_optim.zero_grad()\n",
    "        total_loss.backward()\n",
    "        self.fwd_bwd_optim.step()\n",
    "\n",
    "        return fwd_loss.item(), bwd_loss.item(), efl_loss.item()\n",
    "\n",
    "    def TrainDiscriminator(self, batch):\n",
    "        s, a, r, nexts, done = batch\n",
    "        bsize = s.size(0)\n",
    "\n",
    "        # \"real\" logits from backward\n",
    "        with torch.no_grad():\n",
    "            s_hat, a_hat, u_hat = self.backward(nexts, r)\n",
    "        real_logits = self.discriminator(s_hat, a_hat, r, nexts, u_hat)\n",
    "        real_labels = torch.ones(bsize, 1).to(device)\n",
    "\n",
    "        # \"fake\" logits from forward\n",
    "        noise = torch.randn(bsize, self.args.udim).to(device)\n",
    "        with torch.no_grad(): \n",
    "            nexts_hat, done_hat, r_hat = self.forward(s, a, noise)\n",
    "        fake_logits = self.discriminator(s, a, r_hat, nexts_hat, noise)\n",
    "        fake_labels = torch.zeros(bsize, 1).to(device)\n",
    "\n",
    "        logits = torch.cat([real_logits, fake_logits], dim=0)\n",
    "        labels = torch.cat([real_labels, fake_labels], dim=0)\n",
    "\n",
    "        disc_loss = self.adversarialLoss(logits, labels)\n",
    "\n",
    "        self.disc_optim.zero_grad()\n",
    "        disc_loss.backward()\n",
    "        self.disc_optim.step()\n",
    "\n",
    "        return disc_loss.item()\n",
    "\n",
    "    def PreTrainForward(self, batch): \n",
    "        \"\"\"Pre-train forward model to predict next state + done + reward.\"\"\"\n",
    "        s, a, r, nexts, done = batch\n",
    "        bsize = s.size(0)\n",
    "        \n",
    "        # noise is zero in pretrain\n",
    "        noise = torch.zeros(bsize, self.args.udim).to(device)\n",
    "        nexts_hat, done_hat, r_hat = self.forward(s, a, noise)\n",
    "\n",
    "        stateloss = F.mse_loss(nexts_hat, nexts)\n",
    "        doneloss = F.binary_cross_entropy_with_logits(done_hat, done)\n",
    "        rewardloss = F.mse_loss(r_hat, r)\n",
    "\n",
    "        total_loss = stateloss + 0.1 * rewardloss + 10.0 * doneloss\n",
    "        self.fwd_optim.zero_grad()\n",
    "        total_loss.backward()\n",
    "        self.fwd_optim.step()\n",
    "\n",
    "        self.global_pretrain_teps += 1\n",
    "\n",
    "        return stateloss.item(), doneloss.item(), rewardloss.item()\n",
    "\n",
    "    def train(self, trainloader, pretrainonly=False, testloader=None):\n",
    "        with mlflow.start_run(description=\"BiCoGAN training\"):\n",
    "            \n",
    "            mlflow.log_params(vars(self.args))\n",
    "\n",
    "            # ---------- Pre-train ----------\n",
    "            preTrainLossStateEpoch = 0.0\n",
    "            preTrainLossDoneEpoch = 0.0\n",
    "            preTrainLossRewardEpoch = 0.0\n",
    "\n",
    "            for epoch in (pbar := tqdm.tqdm(range(self.args.pre_train_epochs),\n",
    "                                            desc=\"Pretrain forward\")):\n",
    "                for batch_idx, batch in enumerate(trainloader):\n",
    "                    ls_state, ls_done, ls_reward = self.PreTrainForward(batch)\n",
    "                    preTrainLossDoneEpoch   += ls_done\n",
    "                    preTrainLossStateEpoch  += ls_state\n",
    "                    preTrainLossRewardEpoch += ls_reward\n",
    "\n",
    "                preTrainLossDoneEpoch   /= (batch_idx + 1)\n",
    "                preTrainLossStateEpoch  /= (batch_idx + 1)\n",
    "                preTrainLossRewardEpoch /= (batch_idx + 1)\n",
    "\n",
    "                mlflow.log_metric(\"preTrainLossDoneEpoch\",\n",
    "                                  preTrainLossDoneEpoch,\n",
    "                                  step=self.global_pretrain_teps)\n",
    "                mlflow.log_metric(\"preTrainLossStateEpoch\",\n",
    "                                  preTrainLossStateEpoch,\n",
    "                                  step=self.global_pretrain_teps)\n",
    "                mlflow.log_metric(\"preTrainLossRewardEpoch\",\n",
    "                                  preTrainLossRewardEpoch,\n",
    "                                  step=self.global_pretrain_teps)\n",
    "\n",
    "                preTrainLossStateEpoch  = 0.0\n",
    "                preTrainLossDoneEpoch   = 0.0\n",
    "                preTrainLossRewardEpoch = 0.0\n",
    "\n",
    "            if pretrainonly:\n",
    "                return\n",
    "\n",
    "            # ---------- Adversarial + EFL training ----------\n",
    "            for epoch in (pbar := tqdm.tqdm(range(self.args.num_epochs),\n",
    "                                            desc=\"BiCoGAN main training\")):\n",
    "                self.gamma = self.getGamma(epoch)\n",
    "                batch_efl_loss = 0.0\n",
    "\n",
    "                for batch_idx, batch in enumerate(trainloader):\n",
    "                    disc_loss = self.TrainDiscriminator(batch)\n",
    "                    fwdloss, bwdloss, efl_loss = self.TrainForwardBackward(batch)\n",
    "                    batch_efl_loss += efl_loss\n",
    "\n",
    "                    mlflow.log_metric(\"fwdloss\",   fwdloss, step=self.global_steps)\n",
    "                    mlflow.log_metric(\"bwdloss\",   bwdloss, step=self.global_steps)\n",
    "                    mlflow.log_metric(\"efl_loss\",  efl_loss, step=self.global_steps)\n",
    "                    mlflow.log_metric(\"discloss\",  disc_loss, step=self.global_steps)\n",
    "                    mlflow.log_metric(\"gamma\",     self.gamma, step=self.global_steps)\n",
    "\n",
    "                    self.global_steps += 1\n",
    "\n",
    "                    msgDict = {\n",
    "                        \"fwdloss\": fwdloss,\n",
    "                        \"bwdloss\": bwdloss,\n",
    "                        \"EFL\": efl_loss,\n",
    "                        \"Gamma\": self.gamma,\n",
    "                        \"steps\": self.global_steps,\n",
    "                        \"ep\": epoch    \n",
    "                    }\n",
    "                    pbar.set_postfix(msgDict)\n",
    "\n",
    "                batch_efl_loss /= (batch_idx + 1)\n",
    "                mlflow.log_metric(\"batch_efl_loss\",\n",
    "                                  batch_efl_loss,\n",
    "                                  step=self.global_steps)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 59,
   "metadata": {},
   "outputs": [],
   "source": [
    "env = gym.make(envName, **vars(tgtDom))\n",
    "mlflow.set_experiment(\"BiCOGAN-v2\")\n",
    "udim = args.udim\n",
    "hid_dim = 128\n",
    "obsdim = env.observation_space.shape[0]\n",
    "actiondim = env.action_space.n\n",
    "\n",
    "gen     = Generator(obsdim, udim, actiondim, hid_dim)\n",
    "disc    = Discriminator(obsdim, udim, actiondim, hid_dim )\n",
    "enc     = Encoder(obsdim, udim, actiondim, hid_dim)\n",
    "\n",
    "gen     = gen.to(device)\n",
    "disc    = disc.to(device)\n",
    "enc     = enc.to(device)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 60,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Pretrain forward: 100%|██████████████████████████████████████████████████████████████████████████████| 50/50 [01:42<00:00,  2.05s/it]\n",
      "BiCoGAN main training: 100%|███| 50/50 [10:46<00:00, 12.93s/it, fwdloss=0.687, bwdloss=0.698, EFL=5.55, Gamma=10, steps=18400, ep=49]\n"
     ]
    }
   ],
   "source": [
    "bicogan = BiCoGAN( forward=gen, \n",
    "                  backward=enc, \n",
    "                  discriminator=disc,\n",
    "                  args=args)\n",
    "bicogan.train(loader, pretrainonly=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 61,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Using run: 22bc99f6499642319f56a9e7f3783265\n",
      "Epochs: 50\n",
      "Batches per epoch: 368\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAABW0AAAMWCAYAAACKoqSLAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjAsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvlHJYcgAAAAlwSFlzAAAPYQAAD2EBqD+naQABAABJREFUeJzs3Xd4FOX6//H3pjdISIAkBAhFOigYOqKoEKTYEQQVC6gcjn4PctADoj8BEY7gQdQj2EBEFFCxg0CkCYcqvRcpoSTEkEAgIckm2d8fwy7EBEifzebzuq65dnbmmZl78gTdvfPM/VhsNpsNEREREREREREREXEKbmYHICIiIiIiIiIiIiKXKWkrIiIiIiIiIiIi4kSUtBURERERERERERFxIkraioiIiIiIiIiIiDgRJW1FREREREREREREnIiStiIiIiIiIiIiIiJORElbERERERERERERESeipK2IiIiIiIiIiIiIE1HSVkRERERERERERMSJKGkrIk5h1qxZWCwWx+Lj40NYWBi33347EydOJCEhIc8xY8aMwWKxlHmsFouFMWPGOP05/2rRokWlfo2/euKJJwgICCjTa4qIiIi4ih07djBo0CDq16+Pr68vvr6+NGjQgGeffZbff//d7PBK1Nq1axkzZgxnz54ts2vav08kJiaW2TVFRApKSVsRcSqffvop69atIyYmhvfff5+WLVvy5ptv0qRJE3799ddcbQcPHsy6devKPMZ169YxePBgpz/nXy1atIixY8eW6jVEREREpGR8+OGHREVFsWHDBv7xj3/w888/s3DhQoYNG8bu3btp06YNf/zxh9lhlpi1a9cyduzYMk3aiog4Mw+zAxARuVLz5s1p3bq14/2DDz7ICy+8wC233MIDDzzAwYMHCQ0NBaBmzZrUrFmzTOKy2Wykp6fj6+tL+/btS/z8pXHOsnLx4kV8fX3NDkNERETEZfzvf/9j6NCh9OrVi2+++QYvLy/HvjvuuIO///3vfP311079GSwtLQ0/Pz+zw3CaOERECksjbUXE6dWuXZv//Oc/nD9/ng8//NCxPb/yCMuXL6dLly6EhITg6+tL7dq1efDBB0lLS3O0ycjIYNy4cTRp0gQfHx9CQkK4/fbbWbt2raONxWLhueee44MPPqBJkyZ4e3vz2WefOfZdWWbAXtph+fLlPP3004SEhFC5cmUGDhxIamoq8fHx9O3bl6CgIMLDwxkxYgRWqzVX3Fc754oVK/jb3/5G1apVCQkJ4YEHHuDUqVO5jp0/fz7R0dGEh4fj6+tLkyZNGDlyJKmpqY42TzzxBO+//77jWvbl6NGjAKSnpzNq1Cjq1q2Ll5cXERER/P3vf88z0qFOnTr07t2bb7/9llatWuHj41Mio3dnzpzJTTfdhI+PD8HBwdx///3s3bs3V5vDhw/z8MMPU6NGDby9vQkNDeXOO+9k27ZtjjYF6X8RERERZzdhwgTc3d358MMPcyVsr/TQQw9Ro0aNXNt+//137rnnHoKDg/Hx8aFVq1Z89dVXudoU5nMmGJ81O3TogL+/PwEBAXTv3p2tW7fmamMvibVz506io6OpVKkSd955JwAxMTHce++91KxZEx8fH2644QaeffbZXCUJxowZw4svvghA3bp1HZ9VV65cCUBOTg6TJk2icePGeHt7U716dQYOHMiJEydyxdGlSxeaN2/Ob7/9RseOHfHz8+Opp54qwE/82n788Uc6dOiAn58flSpVolu3bnme+Pvzzz955plnqFWrFt7e3lSrVo1OnTrlelpw69at9O7dm+rVq+Pt7U2NGjXo1atXnvsQEQGNtBWRcqJnz564u7vz22+/XbXN0aNH6dWrF507d2bmzJkEBQVx8uRJFi9eTGZmJn5+fmRlZdGjRw9Wr17NsGHDuOOOO8jKymL9+vXExsbSsWNHx/m+//57Vq9ezf/7f/+PsLAwqlevfs0YBw8ezAMPPMC8efPYunUrL7/8MllZWezfv58HHniAZ555hl9//ZU333yTGjVqMHz48Ove9+DBg+nVqxdffvklx48f58UXX+TRRx9l+fLljjYHDx6kZ8+eDBs2DH9/f/bt28ebb77Jxo0bHe1effVVUlNT+eabb3J9wAwPD8dms3HfffexbNkyRo0aRefOndmxYwevvfYa69atY926dXh7ezuO2bJlC3v37uWVV16hbt26+Pv7X/c+rmXixIm8/PLL9O/fn4kTJ3LmzBnGjBlDhw4d2LRpEw0aNACM34Hs7GwmTZpE7dq1SUxMZO3atY7EckH6X0RERMTZZWdns2LFClq3bk14eHiBj1uxYgV33XUX7dq144MPPiAwMJB58+bRr18/0tLSeOKJJ3K1L8jnzAkTJvDKK6/w5JNP8sorr5CZmcnkyZPp3LkzGzdupGnTpo62mZmZ3HPPPTz77LOMHDmSrKwsAP744w86dOjA4MGDCQwM5OjRo0yZMoVbbrmFnTt34unpyeDBg0lKSuK9997j22+/ddy3/fx/+9vf+Oijj3juuefo3bs3R48e5dVXX2XlypVs2bKFqlWrOuKIi4vj0Ucf5aWXXmLChAm4uRVvrNqXX37JI488QnR0NHPnziUjI4NJkybRpUsXli1bxi233ALAY489xpYtW3jjjTdo2LAhZ8+eZcuWLZw5cwaA1NRUunXrRt26dXn//fcJDQ0lPj6eFStWcP78+WLFKCIuyiYi4gQ+/fRTG2DbtGnTVduEhobamjRp4nj/2muv2a78z9g333xjA2zbtm276jlmz55tA2wff/zxNeMBbIGBgbakpKR897322mt5Yn/++edztbvvvvtsgG3KlCm5trds2dJ28803F+icQ4cOzdVu0qRJNsAWFxeXb9w5OTk2q9VqW7VqlQ2wbd++3bHv73//uy2//+wvXrzYBtgmTZqUa/v8+fNtgO2jjz5ybIuMjLS5u7vb9u/fn+/1/+rxxx+3+fv7X3V/cnKyzdfX19azZ89c22NjY23e3t62AQMG2Gw2my0xMdEG2KZOnXrVcxWk/0VEREScXXx8vA2wPfzww3n2ZWVl2axWq2PJyclx7GvcuLGtVatWNqvVmuuY3r1728LDw23Z2dk2m63gnzNjY2NtHh4eeT7jnj9/3hYWFmbr27evY9vjjz9uA2wzZ8685r3ZP6seO3bMBth++OEHx77JkyfbANuRI0dyHbN37958492wYYMNsL388suObbfddpsNsC1btuyacdjZv0/8+eef+e7Pzs621ahRw9aiRQvHz89mM34G1atXt3Xs2NGxLSAgwDZs2LCrXuv333+3Abbvv/++QLGJiKg8goiUGzab7Zr7W7ZsiZeXF8888wyfffYZhw8fztPml19+wcfHp0CPSd1xxx1UqVKlwPH17t071/smTZoA0KtXrzzbjx07VqBz3nPPPbne33jjjQC5jj98+DADBgwgLCwMd3d3PD09ue222wDylBjIj300xV9HXzz00EP4+/uzbNmyPDE0bNiwQPFfz7p167h48WKea9eqVYs77rjDce3g4GDq16/P5MmTmTJlClu3biUnJyfXMQXpfxEREZHyLCoqCk9PT8fyn//8B4BDhw6xb98+HnnkEQCysrIcS8+ePYmLi2P//v25znW9z5lLliwhKyuLgQMH5jqfj48Pt912m6N0wZUefPDBPNsSEhIYMmQItWrVwsPDA09PTyIjI4GCfVZdsWIFkPezatu2bWnSpEmez6pVqlThjjvuuO55C2L//v2cOnWKxx57LNeI3YCAAB588EHWr1/vKMPVtm1bZs2axfjx41m/fn2ecmg33HADVapU4V//+hcffPABe/bsKZEYRcR1KWkrIuVCamoqZ86cyVO360r169fn119/pXr16vz973+nfv361K9fn3feecfR5s8//6RGjRoFekyqMI+jgZFYvJK9/lh+29PT0wt0zpCQkFzv7WUKLl68CMCFCxfo3LkzGzZsYPz48axcuZJNmzbx7bff5mp3LWfOnMHDw4Nq1arl2m6xWAgLC3M80mVX2J/L9a59tXPWqFHDsd9isbBs2TK6d+/OpEmTuPnmm6lWrRr/93//53icrCD9LyIiIuLsqlatiq+vb75/5P/yyy/ZtGkTP/74Y67tp0+fBmDEiBG5krqenp4MHToUIFcNWbj+50z7Odu0aZPnnPPnz89zPj8/PypXrpxrW05ODtHR0Xz77be89NJLLFu2jI0bN7J+/fpc17qWgn5etCvLz6o5OTkkJycDRu3fxx9/nE8++YQOHToQHBzMwIEDiY+PByAwMJBVq1bRsmVLXn75ZZo1a0aNGjV47bXX8iR4RURANW1FpJxYuHAh2dnZdOnS5ZrtOnfuTOfOncnOzub333/nvffeY9iwYYSGhvLwww9TrVo11qxZQ05OznUTt3+d5MwZLV++nFOnTrFy5UrH6FogzwRi1xISEkJWVhZ//vlnrsStzWYjPj6eNm3a5Gpfkj8X+5eFuLi4PPtOnTqVqz5ZZGQkM2bMAODAgQN89dVXjBkzhszMTD744APg+v0vIiIi4uzc3d254447WLp0KXFxcbkShvYar/bJZO3sn5lGjRrFAw88kO95GzVqVKg47Of85ptvHCNjryW/z4i7du1i+/btzJo1i8cff9yx/dChQwWO48rPizVr1sy176+fF68WR1Fd77Oqm5ub48m8qlWrMnXqVKZOnUpsbCw//vgjI0eOJCEhgcWLFwPQokUL5s2bh81mY8eOHcyaNYtx48bh6+vLyJEjSyxuEXENGmkrIk4vNjaWESNGEBgYyLPPPlugY9zd3WnXrh3vv/8+YEyeBdCjRw/S09OZNWtWaYVbpuwfSq+cKAzgww8/zNP2r6Mn7Owz+86ZMyfX9gULFpCamurYXxo6dOiAr69vnmufOHGC5cuXX/XaDRs25JVXXqFFixaOvr3S1fpfREREpDwYNWoU2dnZDBkypECjMBs1akSDBg3Yvn07rVu3znepVKlSoWLo3r07Hh4e/PHHH1c95/WUxGdVe6mDv35e3LRpE3v37i3Vz6qNGjUiIiKCL7/8MlepttTUVBYsWECHDh3yney2du3aPPfcc3Tr1i3fz6EWi4WbbrqJt99+m6CgIH1WFZF8aaStiDiVXbt2OeplJSQksHr1aj799FPc3d357rvv8jzCf6UPPviA5cuX06tXL2rXrk16ejozZ84EoGvXrgD079+fTz/9lCFDhrB//35uv/12cnJy2LBhA02aNCl3ozE7duxIlSpVGDJkCK+99hqenp588cUXbN++PU/bFi1aAPDmm2/So0cP3N3dufHGG+nWrRvdu3fnX//6FykpKXTq1IkdO3bw2muv0apVKx577LFixZidnc0333yTZ7u/vz89evTg1Vdf5eWXX2bgwIH079+fM2fOMHbsWHx8fHjttdcA2LFjB8899xwPPfQQDRo0wMvLi+XLl7Njxw7HqISC9L+IiIhIedCpUyfef/99nn/+eW6++WaeeeYZmjVrhpubG3FxcSxYsAAgVzmCDz/8kB49etC9e3eeeOIJIiIiSEpKYu/evWzZsoWvv/66UDHUqVOHcePGMXr0aA4fPsxdd91FlSpVOH36NBs3bsTf35+xY8de8xyNGzemfv36jBw5EpvNRnBwMD/99BMxMTF52to/q77zzjs8/vjjeHp60qhRIxo1asQzzzzDe++9h5ubGz169ODo0aO8+uqr1KpVixdeeKFQ95Wfn376Kd+kdp8+fZg0aRKPPPIIvXv35tlnnyUjI4PJkydz9uxZ/v3vfwNw7tw5br/9dgYMGEDjxo2pVKkSmzZtYvHixY6Rzz///DPTpk3jvvvuo169ethsNr799lvOnj1Lt27din0PIuJ6lLQVEafy5JNPAkbd16CgIJo0acK//vUvBg8efM2ELRgTUS1dupTXXnuN+Ph4AgICaN68OT/++CPR0dEAeHh4sGjRIiZOnMjcuXOZOnUqlSpV4qabbuKuu+4q9fsraSEhISxcuJB//vOfPProo/j7+3Pvvfcyf/58br755lxtBwwYwP/+9z+mTZvGuHHjsNlsHDlyhDp16vD9998zZswYPv30U9544w2qVq3KY489xoQJE/KMjCis9PR0HnrooTzbIyMjOXr0KKNGjaJ69eq8++67zJ8/H19fX7p06cKECRNo0KABAGFhYdSvX59p06Zx/PhxLBYL9erV4z//+Q/PP/88ULD+FxERESkvhgwZQocOHXjnnXd4++23OXXqFBaLhZo1a9KxY0eWLVuWa8Kt22+/nY0bN/LGG28wbNgwkpOTCQkJoWnTpvTt27dIMYwaNYqmTZvyzjvvMHfuXDIyMggLC6NNmzYMGTLkusd7enry008/8Y9//INnn30WDw8Punbtyq+//krt2rVzte3SpQujRo3is88+4+OPPyYnJ4cVK1bQpUsXpk+fTv369ZkxYwbvv/8+gYGB3HXXXUycODFPbd6iuNokxTabjQEDBuDv78/EiRPp168f7u7utG/fnhUrVtCxY0cAfHx8aNeuHZ9//jlHjx7FarVSu3Zt/vWvf/HSSy8B0KBBA4KCgpg0aRKnTp3Cy8uLRo0a5SkdISJiZ7Fdbzp2ERERERERERERESkzqmkrIiIiIiIiIiIi4kSUtBURERERERERERFxIkraioiIiIiIiIiIiDgRJW1FREREREREREREnIiStiIiIiIiIiIiIiJORElbERERERERERERESfiYXYAJSUnJ4dTp05RqVIlLBaL2eGIiIiISDHZbDbOnz9PjRo1cHOrmGMN9BlXRERExLUU9DOuyyRtT506Ra1atcwOQ0RERERK2PHjx6lZs6bZYZhCn3FFREREXNP1PuO6TNK2UqVKgHHDlStXLtI5rFYrS5cuJTo6Gk9Pz5IMT0yiPnU96lPXpH51PepT12NGn6akpFCrVi3H57yKSJ9xJT/qU9ejPnVN6lfXoz51Pc78Gddlkrb2x8UqV65crA+0fn5+VK5cWf/4XIT61PWoT12T+tX1qE9dj5l9WpHLAugzruRHfep61KeuSf3qetSnrseZP+NWzOJgIiIiIiIiIiIiIk5KSVsRERERERERERERJ6KkrYiIiIiIiIiIiIgTUdJWRERERERERERExIkoaSsiIiIiIiIiIiLiRJS0FREREREREREREXEiStqKiIiIiIiIiIiI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8kNJ22LSSNtra1MnmO0nzrHxaBL3tYowOxwRERGpoIYOHcrQoUPz3Tdr1qw82xo3bnzdCSn69OlDnz59inxdqRjefResVujcGa7I6YuIiIhck8ojFJNq2l5bG9W1FREREZEK6vx5+OADY33ECHNjERERkfJFSdtiStZI22tqHVkFgIMJFxwJbhERERGRiuCTT+DcOWjcGHr3NjsaERERKU+UtC2m5DR7TVslbfMTEuBN/Wr+AGw6qtG2IiIiIlIxWK3w9tvG+j//CW765iUiIiKFoI8OxZCTY+OsY6StJiK7mraXSiT8fizZ5EhERERERMrGV1/B8eMQGgqPPmp2NCIiIlLeKGlbDCnpVnJsxrpG2l5dmzpG0naj6tqKiIiISAVgs8Fbbxnrzz8PPj7mxiMiIiLlj5K2xZB0qUZrJW8PPN31o7wae9J218lzpGVmmRyNiIiIiEjpWr4ctm0DPz/429/MjkZERETKI2Uai0GTkBVMzSq+hAf6kJVjY1vsWbPDEREREREpVXPmGK9PPAHBwaaGIiIiIuWUkrbFkJx6aRIyJW2vyWKx0PrSaNtNR1XXVkRERERcl80GS5ca6/fdZ2ooIiIiUo4paVsMSfaRtn6ahOx62tapAsCmo6prKyIiIiKua88eOHXKqGN7yy1mRyMiIiLllZK2xZB8qaZtsCYhu642dY2Rtltik8nKzjE5GhERERGR0rFkifF6223g62tuLCIiIlJ+KWlbDMlpKo9QUA2rVyLQ15O0zGx2n0oxOxwRERERkVJhL40QHW1uHCIiIlK+KWlbDPaRtiqPcH1ubhZaR6pEgoiIiIi4rosXYdUqY717d3NjERERkfJNSdticNS01UjbArk8GZmStiIiIiLietasgfR0qFEDmjY1OxoREREpz5S0LYazaappWxht6xojbX8/mozNZjM5GhERERGRknVlaQSLxdxYREREpHxT0rYYki6VRwhS0rZAWkQE4e3hxpnUTP74M9XscERERERESpR9EjKVRhAREZHiUtK2GOwTkQWrPEKBeHm40bJWEKASCSIiIiLiWuLiYOdOY4Rt165mRyMiIiLlnZK2RZSTY3OUR6jir4nICqqNva7tESVtRURERMR12EsjREVB1armxiIiIiLln5K2RZSSbiXnUlnWIF+NtC2oNnUvJW2PKWkrIiIiIq7jynq2IiIi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CEVbAAAAAAAAAPAiFG0BAAAAAAAAwItQtAUAAAAAAAAAL0LRFoDb9OnTZbPZznhbsGCB2SEWasGCBeUe386dOzVq1Cg1btxYgYGBCgoKUosWLfTYY49p//795RZHYUr6erz++uuaPn16mcRU3nbv3i2bzaYXXnjB7FAAAADy+euvv3TLLbeoQYMGCgwMVGBgoBo1aqTbb79dK1asMDs802zatEnXX3+96tevr4CAANWsWVPt27fXqFGjlJqaWqxz3XjjjapSpUoZRZpfbGysbrzxxhId27t3b/Xu3duj8QCwDl+zAwDgfd5//301bdq0wPbmzZubEI33+f777zVkyBDVrFlTo0aNUrt27WSz2bRu3Tq99957+uGHH7R69WrT4mvfvr2WLFlS7Hy9/vrrqlmzZokHnQAAADi7t956S6NGjVKTJk107733qkWLFrLZbNq0aZM++eQTderUSdu3b1eDBg3MDrVcrV69Wj169FCzZs30xBNPKDY2VkeOHNHatWv16aef6oEHHlBoaKjZYQJAuaJoC6CAli1bqmPHjmaHYZr09HQFBQUV+tiuXbs0ZMgQNW7cWPPnz1dYWJj7sQsvvFD33HOPZs+eXV6hFio0NFRdu3Y1NQYAAADk98cff2jkyJHq37+/vvjiC/n7+7sfu/DCC3XXXXfp888/V2BgoIlRmmPatGny8fHRggULFBIS4t5+1VVX6cknn5RhGCZGBwDmoD0CgBKx2WwaNWqUPvzwQzVr1kxBQUFq06aNvv/++wL7bt68WUOHDlV4eLgcDofq1q2rG264QZmZme591q9fr4EDB6patWoKCAhQ27ZtNWPGjELP9Z///EdBQUGqWbOm7rjjDqWlpRUa4y+//KI+ffooNDRUQUFB6tGjh3799dd8+0yYMEE2m02rVq3SVVddpWrVqp11ZsPUqVN16tQpvf766/kKtv98Xa644op829577z21adNGAQEBql69ugYNGqRNmzbl28f1Fa7t27erX79+qlKlimJiYnT//ffne50k6Y033lCbNm1UpUoVhYSEqGnTpnrkkUfcjxfWHmHnzp0aMmSIoqKi5HA4FB4erj59+mjNmjWS8r7WtWHDBi1cuNDdDiM2NtZ9fGpqqh544AHFxcXJ399f0dHRGj16tE6dOlXg+XvifbF79275+vpq8uTJBY777bffZLPZ9Pnn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jz/+WD/88IOmTJmisLAwSdLo0aP13nvvqX///nrqqacUHh6ujz/+uMBMxipVquiVV17R8OHDdezYMV111VWqXbu2Dh8+rLVr1+rw4cN64403SvTaxMXF6dNPP9U111yjtm3batSoUWrXrp0kaePGjXrvvfdkGIYGDRqkqlWr6vHHH9cjjzyiG264QUOHDtXRo0c1ceJEBQQEaPz48cW+/q233qrAwED16NFDkZGRSk5O1uTJkxUWFqZOnToVesxff/2lUaNG6eqrr1ajRo3k7++vefPm6a+//so3a6NVq1b69NNPNWvWLNWvX18BAQFq1aqVRo8erS+//FLnn3++7rvvPrVu3Vq5ublKTEzU3Llzdf/996tLly7Feh5FeV+4jBw5UlOmTNHKlSv1zjvvFOs669at0xdffFFge6dOnXTffffpgw8+UP/+/TVp0iTVq1dPP/zwg15//XXdeeedaty4saSiveZdunTRpZdeqtatW6tatWratGmTPvzwQ3Xr1k1BQUHFihkAAFhTjx499Nprr+nuu+9W+/btddttt6lFixby8fFRUlKSvvzyS0lSaGio+7/nn3++nn/+edWsWVOxsbFauHCh3n33XVWtWtXEZ1JQ9+7dVa1aNd1xxx0aP368/Pz89PHHHxeYbHEmt912m06cOKErr7xSLVu2lN1u1+bNm/Xiiy/Kx8dHDz/8sKS8cf5///tfjRgxQhdddJFuvfVWhYeHa/v27Vq7dq1effXVYsU9ZMgQffzxx+rXr5/uvfdede7cWX5+ftq3b5/mz5+vgQMHatCgQWrWrJmuu+46TZs2TX5+frrooou0fv16vfDCC+58ufTr10/Vq1fXLbfcokmTJsnX11fTp0/X3r17PRYPgErCzFXQAHgX14qmZ7r973//c+8rybjrrrsKnKNevXr5Vk81DMPYuHGjcfXVVxs1atQw/P39jbp16xo33nijkZGR4d5n3bp1xoABA4ywsDDD39/faNOmjfH+++8XOP/GjRuNvn37GgEBAUb16tWNW265xfjmm28MScb8+fPz7btw4UKjf//+RvXq1Q0/Pz8jOjra6N+/v/H555+793Gt7nr48OFivVY7duwwRo4caTRs2NBwOBxGYGCg0bx5c2PMmDH5VoQ1DMN45513jNatWxv+/v5GWFiYMXDgQGPDhg359hk+fLgRHBxc4Dr/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",
      "text/plain": [
       "<Figure size 1400x800 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import mlflow\n",
    "from mlflow.tracking import MlflowClient\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# -------------------------------------------------------\n",
    "# 1. Load experiment and get last run\n",
    "# -------------------------------------------------------\n",
    "exp = mlflow.get_experiment_by_name(\"BiCOGAN-v2\")\n",
    "client = MlflowClient()\n",
    "\n",
    "runs = mlflow.search_runs(experiment_ids=[exp.experiment_id])\n",
    "# get latest run\n",
    "run_id = runs.sort_values(\"start_time\", ascending=False).iloc[0][\"run_id\"]\n",
    "\n",
    "print(\"Using run:\", run_id)\n",
    "\n",
    "# -------------------------------------------------------\n",
    "# 2. Pull metric histories from MLflow\n",
    "# -------------------------------------------------------\n",
    "metric_keys = [\"fwdloss\", \"bwdloss\", \"efl_loss\", \"discloss\", \"gamma\"]\n",
    "\n",
    "metric_hist = {}\n",
    "for key in metric_keys:\n",
    "    metric_hist[key] = client.get_metric_history(run_id, key)\n",
    "\n",
    "# -------------------------------------------------------\n",
    "# 3. Convert to simple lists\n",
    "# -------------------------------------------------------\n",
    "history_raw = {\n",
    "    \"fwd\":   [m.value for m in metric_hist[\"fwdloss\"]],\n",
    "    \"bwd\":   [m.value for m in metric_hist[\"bwdloss\"]],\n",
    "    \"EFL\":   [m.value for m in metric_hist[\"efl_loss\"]],\n",
    "    \"D\":     [m.value for m in metric_hist[\"discloss\"]],\n",
    "    \"Gamma\": [m.value for m in metric_hist[\"gamma\"]]\n",
    "}\n",
    "\n",
    "# -------------------------------------------------------\n",
    "# 4. Determine batches-per-epoch from training metadata\n",
    "# -------------------------------------------------------\n",
    "# If you logged batch_efl_loss once per epoch, we can infer epochs:\n",
    "epoch_count = len([m.value for m in client.get_metric_history(run_id, \"batch_efl_loss\")])\n",
    "\n",
    "steps_total = len(history_raw[\"fwd\"])\n",
    "batches_per_epoch = steps_total // epoch_count\n",
    "\n",
    "print(\"Epochs:\", epoch_count)\n",
    "print(\"Batches per epoch:\", batches_per_epoch)\n",
    "\n",
    "def group_by_epoch(metric_list, bpe):\n",
    "    return [\n",
    "        np.mean(metric_list[i*bpe:(i+1)*bpe])\n",
    "        for i in range(len(metric_list) // bpe)\n",
    "    ]\n",
    "\n",
    "# -------------------------------------------------------\n",
    "# 5. Build history dict compatible with your plotter\n",
    "# -------------------------------------------------------\n",
    "history = {\n",
    "    \"D\":     group_by_epoch(history_raw[\"D\"], batches_per_epoch),\n",
    "    \"Adv\":   group_by_epoch(history_raw[\"fwd\"], batches_per_epoch),  # generator loss\n",
    "    \"EFL\":   group_by_epoch(history_raw[\"EFL\"], batches_per_epoch),\n",
    "    \"Gamma\": group_by_epoch(history_raw[\"Gamma\"], batches_per_epoch)\n",
    "}\n",
    "\n",
    "# -------------------------------------------------------\n",
    "# 6. Plot nicely\n",
    "# -------------------------------------------------------\n",
    "epochs = np.arange(1, len(history[\"D\"]) + 1)\n",
    "\n",
    "plt.figure(figsize=(14,8))\n",
    "\n",
    "# 1. Discriminator Loss\n",
    "plt.subplot(2,2,1)\n",
    "plt.plot(epochs, history[\"D\"], label=\"D-loss\")\n",
    "plt.grid(True); plt.title(\"Discriminator Loss\")\n",
    "\n",
    "# 2. Generator Adv Loss\n",
    "plt.subplot(2,2,2)\n",
    "plt.plot(epochs, history[\"Adv\"], label=\"Adv-loss (G)\", color=\"blue\")\n",
    "plt.grid(True); plt.title(\"Generator Loss\")\n",
    "\n",
    "# 3. Encoder Forward Loss\n",
    "plt.subplot(2,2,3)\n",
    "plt.plot(epochs, history[\"EFL\"], label=\"EFL Loss\", color=\"green\")\n",
    "plt.grid(True); plt.title(\"Encoder Consistency Loss\")\n",
    "\n",
    "# 4. Gamma schedule\n",
    "plt.subplot(2,2,4)\n",
    "plt.plot(epochs, history[\"Gamma\"], label=\"Gamma\", color=\"purple\")\n",
    "plt.grid(True); plt.title(\"Gamma Schedule\")\n",
    "\n",
    "plt.tight_layout()\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 62,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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oUOHqmnTprLZbFq/fr0++OADffnll1q7dq1p8V111VVasWLFZedrypQpqlSpUoE71wAAAAX19ttva+jQoapXr54effRRJSQkyGazafPmzZo9e7ZatGih7du3q1atWmaHWqLWrl2rNm3aqEGDBnruuecUFxenI0eO6Ndff9WcOXP0xBNPKDw83OwwvcKIESM0ceJEdezYUVOmTFFkZKS2bt2qSZMm6aqrrtKsWbPUo0cPs8OUJM2bN08ZGRnux++9957ef/99ff311ypXrpy7vbD/Hrp06aIVK1YoOjr6so+Njo7WihUrSt2/SaAoULQFUGCNGjVS8+bNzQ7DNOnp6QoJCcnzuT/++EO9evVS3bp1tWTJEo9O0w033KBHHnlE8+bNK6lQ8xQeHq5WrVqZGgMAAEB+/fjjjxo8eLC6dOmif//73ypTpoz7uRtuuEFDhgzRZ599puDgYBOjNMfkyZPl5+enpUuXKiwszN3es2dPPf/88zIMw8TovMfs2bM1ceJEPfTQQ5oyZYq7/brrrlPv3r3Vtm1b3XPPPbryyitVs2bNEovrQt87mjZt6vH466+/liQ1a9ZMlSpVuuzzXUjlypVVuXLlfO9/PrvdzncOoICYHgFAsbLZbBo6dKg++ugjNWjQQCEhIWrSpIm++OKLXPv+/vvv6t27tyIjI2W321W9enX17dvX46/HGzZsUPfu3RUREaGgoCBdeeWVmj59ep7n6tixo0JCQlSpUiUNGjRIp06dyjPGxYsXq3379goPD1dISIjatGmjb775xmOfMWPGyGaz6ZdfflHPnj0VERFx0b8WT5o0SWfOnNGUKVM8Crbnvy9//wv9Bx98oCZNmigoKEgVKlTQrbfeqs2bN3vs47pVbfv27ercubPKli2r2NhYPf744x7vkyRNnTpVTZo0UdmyZRUWFqb69evrqaeecj+f1/QIO3fuVK9evRQTEyO73a7IyEi1b99e69atk+S8BWvjxo1atmyZ+3aruLg49/FpaWl64oknFB8frzJlyqhq1aoaNmyYzpw5k+v1F8XnYteuXQoICFBycnKu47777jvZbDZ99tlnuZ67XHv27NHdd9+tKlWqyG63q0GDBnr11VeVk5Pjsd+l3vP09HT3++PKc/PmzTV79uxCxwgAgK978cUX5e/vr7ffftujYHu+22+/XTExMe7Ha9asUa9evRQXF6fg4GDFxcWpd+/e2r17t8dxrtu/v/32W91///2qWLGiwsPD1bdvX505c0YHDx7UHXfcofLlyys6OlpPPPGEHA6H+3jXLeATJ07Uyy+/7L5eu3bttHXrVjkcDo0cOVIxMTEqV66cbr31Vh0+fNgjhk8++URJSUmKjo5WcHCwGjRooJEjR+bqR+Xl6NGjCg8Pv+CUBjabzePx119/rfbt26tcuXIKCQlRgwYN8uxP5afPmZmZqfHjx6t+/fqy2+2qXLmyBgwYoD///NNjP4fDoREjRigqKkohISG65pprtGrVqlzXdPW7/y6/t+jnN568vPDCC4qIiMhz6qzQ0FC9/vrrSk9P12uvvSbJWSy32Wzavn17rv2ffPJJlSlTRkeOHHG3Fcf3jktxfX9Yv369kpKSFBYWpvbt20uSUlJS1L17d1WrVk1BQUGqXbu2HnzwQY+Ypbzf+3bt2qlRo0ZavXq1rr32WoWEhKhmzZp66aWXPPrIeU2P4HqNGzduVO/evVWuXDlFRkbq3nvv1cmTJz2ufeLECd13332qUKGCypYtqy5dumjnzp2y2WwaM2ZMgd8XwBsw0hZAgWVnZysrK8ujzWazyd/f36Ptyy+/1OrVqzVu3DiVLVtWEyZM0K233qotW7a4/0L966+/6pprrlGlSpU0btw41alTR6mpqZo/f74yMzNlt9u1ZcsWtW7dWlWqVNG//vUvVaxYUTNnzlT//v116NAhjRgxQpJ06NAhtW3bVoGBge5bmj7++GMNHTo012uYOXOm+vbtq+7du2v69OkKDAzU22+/rZtuukkLFy50d2hcevTooV69emnQoEEX7UAvWrRIkZGR+f6rcnJysp566in17t1bycnJOnr0qMaMGaPExEStXr1aderUce/rcDh0880367777tPjjz+u7777Ts8//7zKlSun5557TpI0Z84cDR48WA8//LBeeeUV+fn5afv27Zech6tz587Kzs7WhAkTVL16dR05ckTLly/XiRMnJDlvwerZs6fKlSvnHn1gt9slOQuSbdu21b59+/TUU0/piiuu0MaNG/Xcc89p/fr1Wrx4sUcHvCg+F3Fxcbr55pv11ltvacSIER6fvTfeeEMxMTG69dZb85WDC/nzzz/VunVrZWZm6vnnn1dcXJy++OILPfHEE9qxY4f7fcjPez58+HB99NFHGj9+vJo2baozZ85ow4YNOnr0aKFiBADA12VnZ2vJkiVq3rz5Zd2ivWvXLtWrV0+9evVShQoVlJqaqqlTp6pFixbatGlTrtGIAwcOVI8ePTRnzhytXbtWTz31lLKysrRlyxb16NFDDzzwgBYvXqyXX35ZMTExGj58uMfxb775pq644gq9+eabOnHihB5//HF169ZNLVu2VGBgoD744APt3r1bTzzxhAYOHKj58+e7j922bZs6d+6sYcOGKTQ0VL///rtefvllrVq1St9+++1FX2diYqK+/PJL3XXXXXrwwQd19dVXX3DE8fvvv6/7779fbdu21VtvvaUqVapo69at2rBhg8d++elz5uTkqHv37vr+++81YsQItW7dWrt379bo0aPVrl07rVmzxh3H/fffrxkzZuiJJ55Qhw4dtGHDBvXo0eOCAysK4nLi+bvU1FRt3LhRd9555wVHoSYmJqpKlSpKSUmRJN1999168skn9eGHH2r8+PHu/bKzszVz5kx169bN/Rkrru8d+ZGZmambb75ZDz74oEaOHOn+Drdjxw4lJiZq4MCBKleunHbt2qVJkybpmmuu0fr16xUYGHjR8x48eFB33XWXHn/8cY0ePVrz5s3TqFGjFBMTo759+14yrttuu0133nmn7rvvPq1fv16jRo2S5BzMIjnz2a1bN61Zs0ZjxoxxT+/WsWPHQr0fgNcwAOAyTZs2zZCU54+/v7/HvpKMyMhIIy0tzd128OBBw8/Pz0hOTna33XDDDUb58uWNw4cPX/C6vXr1Mux2u7Fnzx6P9k6dOhkhISHGiRMnDMMwjCeffNKw2WzGunXrPPbr0KGDIclYsmSJYRiGcebMGaNChQpGt27dPPbLzs42mjRpYlx99dXuttGjRxuSjOeeey4f75BhBAUFGa1atcrXvsePHzeCg4ONzp07e7Tv2bPHsNvtRp8+fdxt/fr1MyQZn376qce+nTt3NurVq+d+PHToUKN8+fIXve6SJUs83o8jR44YkozJkydf9LiEhASjbdu2udqTk5MNPz8/Y/Xq1R7t//73vw1JxoIFC9xtRfm5cL2OefPmudv2799vBAQEGGPHjr3oa/njjz8MScbEiRMvuM/IkSMNScZPP/3k0f7QQw8ZNpvN2LJli2EY+XvPGzVqZNxyyy0X3QcAAOR28OBBQ5LRq1evXM9lZWUZDofD/ZOT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4YYbtGjRIqWmpnoUtBo2bChJuRaoOnnypL744guNHj1aI0eOdLdnZGTo2LFjJRJ3fn377bc6cOCAli5d6h5dK8m9rkBB2Gw2PfbYYxo3bpx7vlrX3Uv56V/lR6VKlVSxYkV9/fXXeT7v+qO/q7978OBBVa1a1f18VlZWrrn9XYXijIwMdxFPunjB9XLjyUt0dLQSEhK0aNEipaen5zmv7YoVK3To0CHdfvvt7jZ/f3/dc889+te//qUTJ05o1qxZysjI0IABAzzikorne0d+5HWuDRs26Ndff9WHH36ofv36udvzWlTNLBUrVlRWVpaOHTvmUbjN67se4IuYHgGA6YKDg9W2bVt99tlnF+2MtW/f3t2hPd+MGTMUEhLi7gBdf/312rhxo3799VeP/WbNmuXxuE2bNipfvrw2bdqk5s2b5/lzob/SX8pjjz2m0NBQDR48ONcKqJJz5MO8efMkORc0CA4O1syZMz322bdvn3tKiMIIDQ1Vp06d9PTTTyszM1MbN27M13F169bVM888o8aNG+uXX35xt1/or+ddu3bVjh07VLFixTzfy7xGq15Mfj8XkrNz/8ADD2j69OmaNGmSrrzySrVp0+ayrnch7du316ZNmzzeA8n5ubPZbLr++utzHZOf9zwyMlL9+/dX7969tWXLFqWnpxdJvAAA+KpRo0YpOztbgwYNksPhuOT+NptNhmF4FP4k6b333lN2dnZxhVkgrqLa32N9++2383V8Xn9glpx/ZE5LS3PfIdS6dWuVK1dOb731lgzDKETETl27dtXRo0eVnZ2dZ/+vXr16kpzTWUnSxx9/7HH8p59+mmthY1ef8bfffvNo/9///ldk8VzI008/rePHj+uJJ57I9dyZM2f0yCOPKCQkRI899pjHcwMGDNC5c+c0e/Zsffjhh0pMTFT9+vXdzxfn946CKuxnriS4/oDxySefeLTPmTPHjHCAEsdIWwAFtmHDhlydLEmqVavWRecgzYtrldKWLVtq5MiRql27tg4dOqT58+fr7bffVlhYmEaPHq0vvvhC119/vZ577jlVqFBBH3/8sb788ktNmDBB5cqVkyQNGzZMH3zwgbp06aLx48crMjJSH3/8ca6RjGXLltXrr7+ufv366dixY+rZs6eqVKmiP//8U7/++qv+/PNPTZ06tUDvTXx8vObMmaM777xTV155pYYOHaqmTZtKkjZt2qQPPvhAhmHo1ltvVfny5fXss8/qqaeeUt++fdW7d28dPXpUY8eOVVBQkEaPHn3Z17///vsVHBysNm3aKDo6WgcPHlRycrLKlSunFi1a5HnMb7/9pqFDh+r2229XnTp1VKZMGX377bf67bffPEanNG7cWHPmzNEnn3yimjVrKigoSI0bN9awYcM0d+5cXXfddXrsscd0xRVXKCcnR3v27NGiRYv0+OOPq2XLlpf1OvLzuXAZPHiwJkyYoJ9//lnvvffeZV1n/fr1+ve//52rvUWLFnrsscc0Y8YMdenSRePGjVONGjX05ZdfasqUKXrooYdUt25dSfl7z1u2bKmuXbvqiiuuUEREhDZv3qyPPvpIiYmJF1ylGAAAOLVp00ZvvvmmHn74YV111VV64IEHlJCQID8/P6Wmpmru3LmS5L4NPDw8XNddd50mTpyoSpUqKS4uTsuWLdP777+v8uXLm/hKcmvdurUiIiI0aNAgjR49WoGBgfr4449zDUK4kAceeEAnTpzQbbfdpkaNGsnf31+///67XnvtNfn5+enJJ5+U5Oz/vvrqqxo4cKBuvPFG3X///YqMjNT27dv166+/6o033risuHv16qWPP/5YnTt31qOPPqqrr75agYGB2rdvn5YsWaLu3bvr1ltvVYMGDXT33Xdr8uTJCgwM1I033qgNGzbolVdeyXXbfufOnVWhQgXdd999GjdunAICAvThhx9q7969RRbPhfTu3Vu//PKLXnnlFe3atUv33nuvIiMjtWXLFr322mvasWOHZs2apZo1a3ocV79+fSUmJio5OVl79+7VO++84/F8cX7vKKj69eurVq1aGjlypAzDUIUKFfS///0v19RsZurYsaPatGmjxx9/XGlpaWrWrJlWrFihGTNmSJLH/NGATzJvDTQA3sq1euiFft599133vpKMIUOG5DpHjRo1PFaJNQzD2LRpk3H77bcbFStWNMqUKWNUr17d6N+/v3Hu3Dn3PuvXrze6detmlCtXzihTpozRpEkTj5VIzz9Xhw4djKCgIKNChQrGfffdZ/z3v/81JBlLlizx2HfZsmVGly5djAoVKhiBgYFG1apVjS5duhifffaZe5+8VmLNjx07dhiDBw82ateubdjtdiM4ONho2LChMXz4cI/VVw3DMN577z3jiiuuMMqUKWOUK1fO6N69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zGcLAACswHJDNmvXrq3PP/9ce/fu1cKFC5WUlCRJOnz4sMLDS2exMiggSHZ/uySmSAAAADBbTk6OcnJyVL16dR0+fNj9OCcnRxkZGdqyZYu6du1qdpgoIOazBQAAVmC5ou1zzz2nJ554QnFxcbr66quVmJgoyTnqtmnTpiZHZx7mtQUAALCWP/74Q5UqVTI7DBShE7tO6MSuE/IL8FP1a5jPFgAAmMdyRduePXtqz549WrNmjRYuXOhub9++vXuu29KIoi0AAID55syZk+999+7dqx9//LEYo0FRc81nG9MiRmXKljE5GgAAUJpZrmgrSVFRUWratKkOHDjgXnn36quvVv369U2OzDwUbQEAAMw3depU1a9fXy+//LI2b96c6/mTJ09qwYIF6tOnj5o1a6Zjx46ZECUKamfKTklS/A1MjQAAAMxluaJtTk6Oxo0bp3LlyqlGjRqqXr26ypcvr+eff145OTlmh2cairYAAADmW7ZsmV555RV9++23atSokcLDw1WnTh01btxY1apVU8WKFXXfffcpLi5OGzZsULdu3cwOGfmUk52jHYt2SJJqd6xtcjQAAKC0CzA7gL97+umn9f777+ull15SmzZtZBiGfvzxR40ZM0bnzp3TCy+8YHaIpqBoCwAAYA1du3ZV165ddfToUf3www/atWuXzp49q0qVKqlp06Zq2rSp/PwsNzYCl5D6c6rOHj0rezm7qrWqZnY4AACglLNc0Xb69Ol67733dPPNN7vbmjRpoqpVq2rw4MEUbSnaAgAAWMLAgQM1cOBAPfzwwxRpfcD2r7dLkmreWFN+AeQTAACYy3K9kWPHjuU5d239+vVL9ZxgFG0BAACspWzZsrrjjjsUGxurp59+Wjt27DA7JBTCjoXO/NW6qZbJkQAAAFiwaNukSRO98cYbudrfeOMNXXHFFSZEZA0UbQEAAKzlo48+Umpqqp566il9/fXXqlOnjtq1a6eZM2fq3LlzZoeHy3D2+FntW7lPklT7JuazBQAA5rPc9AgTJkxQly5dtHjxYiUmJspms2n58uXau3evFixYYHZ4pnEVbY+fO25uIAAAAHALDw/XkCFDNGTIEP3yyy96++23NWDAAA0dOlQ1a9Z07/fLL7+YGCUuZefinTJyDFVuWFnlqpczOxwAAADrFW3btm2rrVu36s0339Tvv/8uwzDUo0cP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",
      "text/plain": [
       "<Figure size 1400x800 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# After building 'history' dict\n",
    "from ctrl_utilities import evaluate_policy,plot_losses\n",
    "class FakeTrainer:\n",
    "    pass\n",
    "\n",
    "trainer = FakeTrainer()\n",
    "trainer.history = history\n",
    "\n",
    "plot_losses(trainer)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "metadata": {},
   "outputs": [],
   "source": [
    "from math import log\n",
    "import torch.nn.functional as F\n",
    "\n",
    "LOG2PI = float(np.log(2.0 * np.pi))\n",
    "\n",
    "\n",
    "# ===============================================================\n",
    "# 1. Base-S Model (State + Reward Gaussian World Model)\n",
    "# ===============================================================\n",
    "class BaseS(nn.Module):\n",
    "    def __init__(self, state_dim: int, num_actions: int, hidden: int = 128):\n",
    "        super().__init__()\n",
    "        self.state_dim = state_dim\n",
    "        self.num_actions = num_actions\n",
    "        \n",
    "        # state (8) + action one-hot (4)\n",
    "        in_dim = state_dim + num_actions\n",
    "        out_dim = 2 * (state_dim + 1)   # mu/logvar for next-state + reward\n",
    "\n",
    "        self.fc1 = nn.utils.weight_norm(nn.Linear(in_dim, hidden))\n",
    "        self.fc2 = nn.utils.weight_norm(nn.Linear(hidden, hidden))\n",
    "        self.fc3 = nn.utils.weight_norm(nn.Linear(hidden, hidden))\n",
    "        self.fc4 = nn.Linear(hidden, out_dim)\n",
    "\n",
    "        for m in [self.fc1, self.fc2, self.fc3, self.fc4]:\n",
    "            nn.init.xavier_uniform_(m.weight)\n",
    "            if m.bias is not None:\n",
    "                nn.init.zeros_(m.bias)\n",
    "\n",
    "    def forward(self, s: torch.Tensor, a: torch.Tensor):\n",
    "        \"\"\"\n",
    "        Handles both:\n",
    "          a = (B,)       → integer actions   → convert to one-hot\n",
    "          a = (B,4)      → already one-hot  → use directly\n",
    "        \"\"\"\n",
    "        # Case 1: Already one-hot\n",
    "        if a.dim() == 2 and a.size(1) == self.num_actions:\n",
    "            a_oh = a.float()\n",
    "\n",
    "        # Case 2: scalar actions → convert to one-hot\n",
    "        else:\n",
    "            if a.dim() == 2:\n",
    "                a = a.squeeze(-1)\n",
    "            a_oh = F.one_hot(a.long(), num_classes=self.num_actions).float()\n",
    "\n",
    "        sa = torch.cat([s, a_oh], dim=-1)  # (B, 8+4 = 12)\n",
    "\n",
    "        h = F.relu(self.fc1(sa))\n",
    "        h = F.relu(self.fc2(h))\n",
    "        h = F.relu(self.fc3(h))\n",
    "        out = self.fc4(h)\n",
    "\n",
    "        mu, logvar = out.chunk(2, dim=-1)\n",
    "\n",
    "        mu_s     = mu[..., :self.state_dim]\n",
    "        mu_r     = mu[..., self.state_dim:].unsqueeze(-1)\n",
    "\n",
    "        logvar_s = logvar[..., :self.state_dim]\n",
    "        logvar_r = logvar[..., self.state_dim:].unsqueeze(-1)\n",
    "\n",
    "        return mu_s, mu_r, logvar_s, logvar_r\n",
    "\n",
    "    def sample_next(self, s, a):\n",
    "        mu_s, mu_r, logvar_s, logvar_r = self.forward(s, a)\n",
    "\n",
    "        logvar_s = torch.clamp(logvar_s, -5.0, 5.0)\n",
    "        logvar_r = torch.clamp(logvar_r, -5.0, 5.0)\n",
    "\n",
    "        eps_s = torch.randn_like(mu_s)\n",
    "        eps_r = torch.randn_like(mu_r)\n",
    "\n",
    "        s_next = mu_s + torch.exp(0.5 * logvar_s) * eps_s\n",
    "        r_next = mu_r + torch.exp(0.5 * logvar_r) * eps_r\n",
    "\n",
    "        return s_next, r_next\n",
    "\n",
    "\n",
    "\n",
    "# ===============================================================\n",
    "# 2. Gaussian Negative Log-Likelihood Loss\n",
    "# ===============================================================\n",
    "def gaussian_nll(x, mu, logvar):\n",
    "    return 0.5 * (LOG2PI + logvar + (x - mu)**2 / torch.exp(logvar))\n",
    "\n",
    "\n",
    "# ===============================================================\n",
    "# 3. Train Base-S Model\n",
    "# ===============================================================\n",
    "from tqdm import tqdm\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "def train_base_s(loader, state_dim, num_actions, epochs=20, lr=1e-3):\n",
    "    mlflow.set_experiment(\"BaseS-LunarLander-v3\")\n",
    "\n",
    "    model = BaseS(state_dim, num_actions).to(device)\n",
    "    optim = torch.optim.Adam(model.parameters(), lr=lr)\n",
    "\n",
    "    loss_history = []\n",
    "\n",
    "    with mlflow.start_run(description=\"Base-S World Model Training\"):\n",
    "        mlflow.log_param(\"state_dim\", state_dim)\n",
    "        mlflow.log_param(\"num_actions\", num_actions)\n",
    "        mlflow.log_param(\"epochs\", epochs)\n",
    "        mlflow.log_param(\"lr\", lr)\n",
    "\n",
    "        pbar = tqdm(range(epochs), desc=\"Training Base-S\")\n",
    "\n",
    "        for epoch in pbar:\n",
    "            epoch_loss = 0.0\n",
    "            \n",
    "            for batch in loader:\n",
    "                s, a, r, s_next, done = batch\n",
    "\n",
    "                s       = s.to(device).float()\n",
    "                a       = a.to(device)         # shape (B,4) one-hot or (B,) int\n",
    "                r       = r.to(device).float().unsqueeze(-1)\n",
    "                s_next  = s_next.to(device).float()\n",
    "\n",
    "                mu_s, mu_r, logvar_s, logvar_r = model(s, a)\n",
    "\n",
    "                nll_s = gaussian_nll(s_next, mu_s, logvar_s).sum(dim=-1)\n",
    "                nll_r = gaussian_nll(r, mu_r, logvar_r).squeeze(-1)\n",
    "\n",
    "                loss = (nll_s + nll_r).mean()\n",
    "\n",
    "                optim.zero_grad()\n",
    "                loss.backward()\n",
    "                optim.step()\n",
    "\n",
    "                epoch_loss += loss.item()\n",
    "\n",
    "            epoch_loss /= len(loader)\n",
    "            loss_history.append(epoch_loss)\n",
    "\n",
    "            mlflow.log_metric(\"nll_epoch\", epoch_loss, step=epoch)\n",
    "\n",
    "            pbar.set_postfix({\"NLL\": f\"{epoch_loss:.4f}\"})\n",
    "\n",
    "    return model, loss_history\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 67,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "s shape      = torch.Size([128, 8])\n",
      "a shape      = torch.Size([128, 4])\n",
      "r shape      = torch.Size([128, 1])\n",
      "s_next shape = torch.Size([128, 8])\n",
      "done shape   = torch.Size([128, 1])\n"
     ]
    }
   ],
   "source": [
    "s, a, r, s_next, done = next(iter(loader))\n",
    "\n",
    "print(\"s shape      =\", s.shape)\n",
    "print(\"a shape      =\", a.shape)\n",
    "print(\"r shape      =\", r.shape)\n",
    "print(\"s_next shape =\", s_next.shape)\n",
    "print(\"done shape   =\", done.shape)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 68,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "Training Base-S: 100%|█████████████████████████████████████████████████████████████████| 50/50 [01:03<00:00,  1.27s/it, NLL=-21.0877]\n"
     ]
    }
   ],
   "source": [
    "base_s,loss_history, = train_base_s(\n",
    "    loader,\n",
    "    state_dim=obsdim,        # 8\n",
    "    num_actions=actiondim,   # 4\n",
    "    epochs=50,               # or 50\n",
    "    lr=1e-4\n",
    ")\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 69,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Using run: 6267dc50d1404d7f992552a021b5cd46\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAroAAAHUCAYAAADV64uuAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjAsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvlHJYcgAAAAlwSFlzAAAPYQAAD2EBqD+naQAAV35JREFUeJzt3Xd4VFX+x/HPpE0SUiAJKZRAQBAQ6YLBQhOkyGJbCxZYd63gyrL+LLhKWQUXkdW1YEdYVLALFgQFUZcOhg4Khp5QAiQhIf3+/ogzJmQmmUlmMoX363nyLHPvmTtn5gzy2ZPvPcdkGIYhAAAAwM8EeLoDAAAAgDsQdAEAAOCXCLoAAADwSwRdAAAA+CWCLgAAAPwSQRcAAAB+iaALAAAAv0TQBQAAgF8i6AIAAMAvEXQBP/D222/LZDJV+mncuLH69u2rzz//3NPdq5ZhGJo/f74uu+wyxcfHKzQ0VM2aNdOVV16pN954w+7zbL1nWz8tW7ascx9btmyp0aNH1+q5o0ePdkkfavvaERERHnltZxUXF6tdu3Z6+umnrcdqGuPvvvvOcx2WtHfvXplMJs2YMaNWz//uu++s72XVqlVVztsav759+6pjx47VXnfSpEkymUw6fvy43TbffvutIiIidOjQoVr1HfAVQZ7uAADXmT17ttq1ayfDMJSZmakXX3xRw4cP18KFCzV8+HBPd8+mRx99VP/6179055136v/+7/8UGRmpffv2admyZfrss8/0l7/8xebzhg0bViUcpKam6vrrr9ff//536zGz2VznPn7yySeKioqq1XMff/xxPfDAA3Xug797+eWXdfLkSd1///1Vzlm+12fr0KFDfXStXjz00EP64Ycf6u31BgwYoJ49e2rChAmaM2dOvb0uUN8IuoAf6dixo3r06GF9PHjwYDVq1EjvvfeeVwbdM2fO6LnnntPtt9+u1157rdK50aNHq6yszO5zGzdurMaNG1c5npCQoIsvvtju80pLS1VSUuJUAO7atavDbc/WunXrWj/3XFFSUqJnnnlGd9xxhxo0aFDl/Nnfa38zePBgLV68WIsWLarXv6djxozRjTfeqCeffFLNmzevt9cF6hOlC4AfCw0NVUhIiIKDgysdnzx5snr16qWYmBhFRUWpW7duevPNN2UYRqV2y5YtU9++fRUbG6uwsDAlJyfruuuuU35+vrVNUVGRnnzySbVr105ms1mNGzfWn/70Jx07dqzG/uXl5amwsFBJSUk2zwcE1O0/UZZfLU+fPl1PPvmkUlJSZDabtXz5chUUFOjvf/+7unTpoujoaMXExCg1NVWfffZZleucXbpg+ZXze++9p8cee0xNmjRRVFSUrrjiCu3atavSc22VLphMJo0dO1b//e9/1b59e4WHh6tz5842y0w+++wzderUSWazWa1atdLzzz9v/dW0q7z11lvq3LmzQkNDFRMTo2uuuUY7duyo1ObXX3/VTTfdpCZNmshsNishIUEDBgxQWlqatY0j3xdbFi5cqEOHDum2226r9XuwfKavvvqq2rZtK7PZrA4dOmj+/PlV2m7dulUjRoxQo0aNFBoaqi5dutic1Tx16pT+/ve/q1WrVjKbzYqPj9fQoUO1c+fOKm1nzpyplJQURUREKDU1VatXr3a476NHj1aHDh306KOPqrS01Lk3XgfDhw9XRESEXn/99Xp7TaC+MaML+BHLbKVhGDpy5IieeeYZ5eXlaeTIkZXa7d27V3fffbeSk5MlSatXr9b999+vQ4cO6YknnrC2GTZsmC677DK99dZbatiwoQ4dOqTFixerqKhI4eHhKisr04gRI/TDDz/ooYceUu/evbVv3z5NnDhRffv21fr16xUWFma3v3FxcTrvvPP08ssvW0PE+eef79IQJ0n/+c9/1LZtW82YMUNRUVFq06aNCgsLdeLECT344INq2rSpioqK9M033+jaa6/V7Nmzdfvtt9d43QkTJuiSSy7RG2+8oZycHD388MMaPny4duzYocDAwGqf+8UXX2jdunWaMmWKIiIiNH36dF1zzTXatWuXWrVqJUlavHixrr32Wl1++eVasGCBSkpKNGPGDB05csQln4skTZs2TRMmTNDNN9+sadOmKSsrS5MmTVJqaqrWrVunNm3aSJKGDh2q0tJSTZ8+XcnJyTp+/LhWrlypU6dOSXLs+1LdZxEfH2+3FMHyva7IZDJV+YwXLlyo5cuXa8qUKWrQoIFefvll3XzzzQoKCtL1118vSdq1a5d69+6t+Ph4/ec//1FsbKzmzZun0aNH68iRI3rooYckSbm5ubr00ku1d+9ePfzww+rVq5dOnz6t77//XhkZGZVKKV566SW1a9dOzz33nKTycpWhQ4cqPT1d0dHRNY5BYGCgpk2bphEjRmjOnDm64447anyOK4SEhKh379764osvNGXKlHp5TaDeGQB83uzZsw1JVX7MZrPx8ssvV/vc0tJSo7i42JgyZYoRGxtrlJWVGYZhGB9++KEhyUhLS7P73Pfee8+QZHz00UeVjq9bt86QVONrG4ZhrF271khOTrb2OTIy0rjqqquMuXPnWvviKEnGmDFjrI/T09MNSUbr1q2NoqKiap9bUlJiFBcXG3/+85+Nrl27VjrXokULY9SoUdbHy5cvNyQZQ4cOrdTu/fffNyQZq1atsh4bNWqU0aJFiyr9TEhIMHJycqzHMjMzjYCAAGPatGnWYxdddJHRvHlzo7Cw0HosNzfXiI2NNRz5z/eoUaOMBg0a2D1/8uRJIywsrMr72L9/v2E2m42RI0cahmEYx48fNyQZzz33nN1rOfJ9sad9+/bG4MGDqxy3972WZAQGBlZqK8kICwszMjMzrcdKSkqMdu3aGeedd5712E033WSYzWZj//79lZ4/ZMgQIzw83Dh16pRhGIYxZcoUQ5KxdOlSu/22fL8uvPBCo6SkxHp87dq1hiTjvffeq/Z9W75HH3zwgWEYhnHppZcazZo1M86cOWMYhu3x69Onj3HBBRdUe92JEycakoxjx45V284wDOOxxx4zAgICjNOnT9fYFvBFlC4AfmTu3Llat26d1q1bp6+++kqjRo3SmDFj9OKLL1Zqt2zZMl1xxRWKjo5WYGCggoOD9cQTTygrK0tHjx6VJHXp0kUhISG66667NGfOHP36669VXu/zzz9Xw4YNNXz4cJWUlFh/unTposTEROtd8WVlZZXOV/z17EUXXaTdu3dr8eLFmjBhglJTU/Xtt9/q9ttv1x/+8Icq5RS18Yc//KFK+YYkffDBB7rkkksUERGhoKAgBQcH680336zya/vqrltRp06dJEn79u2r8bn9+vVTZGSk9XFCQoLi4+Otz83Ly9P69et19dVXKyQkxNouIiLCZXWcq1at0pkzZ6qsKNG8eXP1799f3377rSQpJiZGrVu31jPPPKOZM2fqp59+qlI/7cj3xZ7Dhw8rPj7e7vmK32vLz5o1a6q0GzBggBISEqyPAwMDdeONN2r37t06ePCgpPLv/oABA6rUpI4ePVr5+fnWGxy/+uortW3bVldccUWN/R82bFil2WVnvgcV/etf/9LBgwf1/PPPO/W8uoiPj1dZWZkyMzPr7TWB+kTQBfxI+/bt1aNHD/Xo0UODBw/Wq6++qkGDBumhhx6y/op57dq1GjRokCTp9ddf1//+9z+tW7dOjz32mKTyG8Sk8puovvnmG8XHx2vMmDFq3bq1WrduXekf4SNHjujUqVPWOuCKP5mZmdbljaZMmVLp3Nk3aAUHB+vKK6/UU089pa+//loHDhywLo321Vdf1flzsVUD/PHHH+uGG25Q06ZNNW/ePK1atUrr1q3THXfcoYKCAoeuGxsbW+mx5QY3y2fozHMtz7c89+TJkzIMo1Jws7B1rDaysrIk2f58mjRpYj1vMpn07bff6sorr9T06dPVrVs3NW7cWH/961+Vm5srybHviz1nzpxRaGio3fMVv9eWn+7du1dpl5iYaPeY5b1kZWXZfb8V2x07dkzNmjWrse9S3b4HFfXu3VtXX321nn76aZ08edKp59aW5XN3tq+Ar6BGF/BznTp10tdff62ff/5ZPXv21Pz58xUcHKzPP/+8Urj49NNPqzz3sssu02WXXabS0lKtX79eL7zwgsaNG6eEhATddNNNiouLU2xsrBYvXmzztS0zlnfddZeuuuoq6/GaVjyIjY3VuHHj9N1332nr1q0aOnRoLd7572zV/M6bN08pKSlasGBBpfOFhYV1ei1XadSokUwmk816XFfNvlkCWkZGRpVzhw8fVlxcnPVxixYt9Oabb0qSfv75Z73//vuaNGmSioqK9Morr0iq+ftiT1xcnE6cOFHn92Prc7Ecs7zX2NhYu+/X0hepfFUPyyxwfZo2bZo6duyoqVOn1svrWT73imMN+BNmdAE/Z7kr3rIUl8lkUlBQUKVftZ45c0b//e9/7V4jMDBQvXr10ksvvSRJ2rhxoyTpqquuUlZWlkpLS6vMuPXo0UPnn3++pPLZsorHL7zwQknlmwRYZtDOZikfsMy0uZrJZFJISEilkJuZmWlz1QVPaNCggXr06KFPP/1URUVF1uOnT5922SYgqampCgsL07x58yodP3jwoPVX/La0bdtW//jHP3ThhRdavwsV2fu+2NOuXTvt2bOnlu/id99++22l/2NQWlqqBQsWqHXr1tbZ2QEDBmjZsmXWYGsxd+5chYeHW5emGzJkiH7++WctW7aszv1yRrt27XTHHXfohRde0P79+93+er/++qtiY2Nd9lsCwNswowv4ka1bt1rvTs/KytLHH3+spUuX6pprrlFKSoqk8nrCmTNnauTIkbrrrruUlZWlGTNmVJllfeWVV7Rs2TINGzZMycnJKigo0FtvvSVJ1rrFm266Se+8846GDh2qBx54QD179lRwcLAOHjyo5cuXa8SIEbrmmmvs9jc7O1stW7bUH//4R11xxRVq3ry5Tp8+re+++07PP/+82rdvr2uvvdYdH5Wuuuoqffzxx7rvvvt0/fXX68CBA/rnP/+ppKQk/fLLL255TWdNmTJFw4YN05VXXqkHHnhApaWleuaZZxQREeHwDGhpaak+/PDDKscbNGigIUOG6PHHH9eECRN0++236+abb1ZWVpYmT56s0NBQTZw4UZK0efNmjR07Vn/84x/Vpk0bhYSEaNmyZdq8ebMeeeQRSY59X+zp27evpkyZovz8fJurM1T8XlfUunXrSmspx8XFqX///nr88cetqy7s3Lmz0hJjEydO1Oeff65+/frpiSeeUExMjN555x198cUXmj59unWVhHHjxmnBggUaMWKEHnnkEfXs2VNnzpzRihUrdNVVV6lfv341ffS1NmnSJL3zzjtavny5zXWFc3JybI5p48aN1adPH+vjRYsWVaoDt7CsQCGVr7jSp08fl690AngNT98NB6DubN2dHh0dbXTp0sWYOXOmUVBQUKn9W2+9ZZx//vmG2Ww2WrVqZUybNs148803DUlGenq6YRiGsWrVKuOaa64xWrRoYZjNZiM2Ntbo06ePsXDhwkrXKi4uNmbMmGF07tzZCA0NNSIiIox27doZd999t/HLL79U2+/CwkJjxowZxpAhQ4zk5GTDbDYboaGhRvv27Y2HHnrIyMrKcupzkJ1VF5555hmb7Z9++mmjZcuWhtlsNtq3b2+8/vrr1jvWK7K36oLlbvmzX2/27NnWY/ZWXajYT3uvYxiG8cknnxgXXnihERISYiQnJxtPP/208de//tVo1KhRNZ/E76999vfC8lOxT2+88YbRqVMnIyQkxIiOjjZGjBhhbNu2zXr+yJEjxujRo4127doZDRo0MCIiIoxOnToZ//73v62rDTj6fbFl9+7dhslkMt5///1Kx6tbdUGS8frrr1f5TF9++WWjdevWRnBwsNGuXTvjnXfeqfJ6W7ZsMYYPH25ER0cbISEhRufOnSuNmcXJkyeNBx54wEhOTjaCg4ON+Ph4Y9iwYcbO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",
      "text/plain": [
       "<Figure size 800x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "def plot_base_s_loss():\n",
    "    # Load experiment\n",
    "    exp = mlflow.get_experiment_by_name(\"BaseS-LunarLander-v3\")\n",
    "    client = MlflowClient()\n",
    "\n",
    "    # Get latest run\n",
    "    runs = mlflow.search_runs(experiment_ids=[exp.experiment_id])\n",
    "    run_id = runs.sort_values(\"start_time\", ascending=False).iloc[0][\"run_id\"]\n",
    "\n",
    "    print(\"Using run:\", run_id)\n",
    "\n",
    "    # Pull metric history\n",
    "    metric_history = client.get_metric_history(run_id, \"nll_epoch\")\n",
    "\n",
    "    # Extract (step, value) pairs\n",
    "    steps  = [m.step for m in metric_history]\n",
    "    values = [m.value for m in metric_history]\n",
    "\n",
    "    # Plot\n",
    "    plt.figure(figsize=(8, 5))\n",
    "    plt.plot(steps, values, marker=\"o\")\n",
    "    plt.xlabel(\"Epoch\")\n",
    "    plt.ylabel(\"NLL\")\n",
    "    plt.title(\"Base-S Training Loss (Epoch NLL)\")\n",
    "    plt.grid(True)\n",
    "    plt.show()\n",
    "plot_base_s_loss()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 70,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "BaseS(\n",
       "  (fc1): Linear(in_features=12, out_features=128, bias=True)\n",
       "  (fc2): Linear(in_features=128, out_features=128, bias=True)\n",
       "  (fc3): Linear(in_features=128, out_features=128, bias=True)\n",
       "  (fc4): Linear(in_features=128, out_features=18, bias=True)\n",
       ")"
      ]
     },
     "execution_count": 70,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "base_s"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 71,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Generating Base-S CF...\n",
      "Base-S CF samples: 141186\n"
     ]
    }
   ],
   "source": [
    "# %% GENERATE BASE-S COUNTERFACTUAL TRANSITIONS\n",
    "# base_s is already trained\n",
    "\n",
    "def generate_cf_base_s(model, transitions, num_actions=4):\n",
    "    cf_data = []\n",
    "\n",
    "    for (s, a_real, r, s_next, done) in transitions:\n",
    "        s_t = torch.tensor(s, dtype=torch.float32).unsqueeze(0).to(device)\n",
    "\n",
    "        for a_cf in range(num_actions):\n",
    "            if a_cf == a_real:\n",
    "                continue\n",
    "\n",
    "            a_t = torch.tensor([a_cf]).long().to(device)\n",
    "\n",
    "            with torch.no_grad():\n",
    "                mu_s, mu_r, logvar_s, logvar_r = model(s_t, a_t)\n",
    "                s_hat = mu_s.squeeze(0).cpu().numpy()\n",
    "                r_hat = mu_r.item()\n",
    "                done_hat = False  # Base-S cannot predict done\n",
    "\n",
    "            cf_data.append([s, a_cf, r_hat, s_hat, done_hat])\n",
    "\n",
    "    return cf_data\n",
    "\n",
    "print(\"Generating Base-S CF...\")\n",
    "cf_base_s = generate_cf_base_s(base_s, traindata, num_actions=actiondim)\n",
    "print(\"Base-S CF samples:\", len(cf_base_s))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 72,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Generating BiCoGAN CF...\n",
      "BiCoGAN CF samples: 141186\n"
     ]
    }
   ],
   "source": [
    "# %% GENERATE BICOGAN COUNTERFACTUAL TRANSITIONS\n",
    "\n",
    "def generate_cf_bicogan(bicogan, transitions, num_actions=4):\n",
    "    cf_data = []\n",
    "\n",
    "    for (s, a_real, r, s_next, done) in transitions:\n",
    "\n",
    "        # prepare tensors\n",
    "        s_t       = torch.tensor(s, dtype=torch.float32).unsqueeze(0).to(device)\n",
    "        s_next_t  = torch.tensor(s_next, dtype=torch.float32).unsqueeze(0).to(device)\n",
    "        r_t       = torch.tensor([r], dtype=torch.float32).unsqueeze(0).to(device)\n",
    "\n",
    "        # Step 1: invert the forward pass → get latent u_hat\n",
    "        with torch.no_grad():\n",
    "            _, _, u_hat = bicogan.backward(s_next_t, r_t)\n",
    "\n",
    "        # Step 2: generate CF for all alternative actions\n",
    "        for a_cf in range(num_actions):\n",
    "            if a_cf == a_real:\n",
    "                continue   # skip the real action\n",
    "\n",
    "            a_onehot = F.one_hot(torch.tensor(a_cf), num_actions)\\\n",
    "                         .float().unsqueeze(0).to(device)\n",
    "\n",
    "            # forward pass with replaced action + same u_hat\n",
    "            with torch.no_grad():\n",
    "                s_hat, done_hat, r_hat = bicogan.forward(s_t, a_onehot, u_hat)\n",
    "\n",
    "            cf_data.append([\n",
    "                s,                                           # state\n",
    "                a_cf,                                        # CF action\n",
    "                float(r_hat.item()),                         # CF reward\n",
    "                s_hat.squeeze(0).cpu().numpy(),              # CF next_state\n",
    "                bool(done_hat.item() > 0)                    # CF termination\n",
    "            ])\n",
    "\n",
    "    return cf_data\n",
    "\n",
    "\n",
    "print(\"Generating BiCoGAN CF...\")\n",
    "cf_bicogan = generate_cf_bicogan(bicogan, traindata, num_actions=actiondim)\n",
    "print(\"BiCoGAN CF samples:\", len(cf_bicogan))\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### “In this setup, we use counterfactual samples generated by both Base-S and BiCoGAN from previously stored transitions, and train the D3QN agent using this combined CF buffer.”"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from ctrl_data import ReplayBuffer\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from tqdm import tqdm\n",
    "import mlflow\n",
    "\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "\n",
    "\n",
    "# ===============================================================\n",
    "# Dueling DQN network (unchanged, correct)\n",
    "# ===============================================================\n",
    "class DuelingDQN(nn.Module):\n",
    "    def __init__(self, state_size, action_size):\n",
    "        super().__init__()\n",
    "        self.feature_layer = nn.Sequential(\n",
    "            nn.Linear(state_size, 64),\n",
    "            nn.ReLU()\n",
    "        )\n",
    "        self.value_stream = nn.Sequential(\n",
    "            nn.Linear(64, 64),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(64, 1)\n",
    "        )\n",
    "        self.advantage_stream = nn.Sequential(\n",
    "            nn.Linear(64, 64),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(64, action_size)\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        features = self.feature_layer(x)\n",
    "        value = self.value_stream(features)\n",
    "        advantages = self.advantage_stream(features)\n",
    "        q_values = value + (advantages - advantages.mean(dim=-1, keepdim=True))\n",
    "        return q_values\n",
    "\n",
    "\n",
    "\n",
    "# ===============================================================\n",
    "# Default D3QN hyperparameters\n",
    "# ===============================================================\n",
    "class D3QNParams:\n",
    "    eps_start = 1.0\n",
    "    eps_end = 0.01\n",
    "    eps_decay = 300\n",
    "    lr = 5e-4\n",
    "    min_buffer = 256\n",
    "    batch_size = 128\n",
    "    gamma = 0.99\n",
    "    max_episodes = 200\n",
    "    max_iter = 1000\n",
    "    tau = 0.005\n",
    "    optimizer = torch.optim.Adam\n",
    "    criteria = torch.nn.MSELoss\n",
    "    mix_ratio = 0.5   # 50% REAL + 50% CF during offline mixing\n",
    "\n",
    "\n",
    "\n",
    "# ===============================================================\n",
    "# Patched D3QNAgent\n",
    "# ===============================================================\n",
    "class D3QNAgent:\n",
    "\n",
    "    def __init__(self, env, policy_net, target_net, scm=None):\n",
    "        self.env = env\n",
    "        self.policy_net = policy_net.to(device)\n",
    "        self.target_net = target_net.to(device)\n",
    "        self.target_net.load_state_dict(self.policy_net.state_dict())\n",
    "        self.target_net.eval()\n",
    "\n",
    "        self.hparams = D3QNParams()\n",
    "        self.optimizer = self.hparams.optimizer(\n",
    "            self.policy_net.parameters(), lr=self.hparams.lr\n",
    "        )\n",
    "\n",
    "        # --------------------------------------------\n",
    "        # ❗ Most important fix: correct number of actions\n",
    "        # --------------------------------------------\n",
    "        self.actiondim = env.action_space.n\n",
    "\n",
    "        # --------------------------------------------\n",
    "        # Replay buffers\n",
    "        # --------------------------------------------\n",
    "        self.ReplayMem = ReplayBuffer(\n",
    "            max_size=100000,\n",
    "            state_dim=env.observation_space.shape[0],\n",
    "            action_dim=1,\n",
    "            batch_size=self.hparams.batch_size\n",
    "        )\n",
    "\n",
    "        self.Counterfactual = ReplayBuffer(\n",
    "            max_size=500000,\n",
    "            state_dim=env.observation_space.shape[0],\n",
    "            action_dim=1,\n",
    "            batch_size=self.hparams.batch_size\n",
    "        )\n",
    "\n",
    "        self.global_steps = 0\n",
    "        self.learn_steps = 0\n",
    "        self.eps_threshold = self.hparams.eps_start\n",
    "\n",
    "        self.scm = scm   # Base-S or BiCoGAN (optional)\n",
    "        self.scm_specified = scm is not None\n",
    "\n",
    "\n",
    "\n",
    "    # ===============================================================\n",
    "    # Update target network (soft update)\n",
    "    # ===============================================================\n",
    "    def update_network_parameters(self, tau=None):\n",
    "        if tau is None:\n",
    "            tau = self.hparams.tau\n",
    "\n",
    "        for tgt, src in zip(self.target_net.parameters(), self.policy_net.parameters()):\n",
    "            tgt.data.copy_((1 - tau) * tgt + tau * src)\n",
    "\n",
    "\n",
    "\n",
    "    # ===============================================================\n",
    "    # Epsilon schedule\n",
    "    # ===============================================================\n",
    "    def decrement_epsilon(self):\n",
    "        eps_start, eps_end, decay = (\n",
    "            self.hparams.eps_start, self.hparams.eps_end, self.hparams.eps_decay\n",
    "        )\n",
    "        self.eps_threshold = eps_end + (eps_start - eps_end) * \\\n",
    "                             np.exp(-1.0 * self.global_steps / decay)\n",
    "\n",
    "\n",
    "\n",
    "    # ===============================================================\n",
    "    # The fully corrected hybrid training function\n",
    "    # ===============================================================\n",
    "    def TrainFromReplayMem(\n",
    "            self,\n",
    "            replay='replay',      # 'replay', 'counterfactual', 'offline'\n",
    "            mix=False,\n",
    "            batch_size=None):\n",
    "\n",
    "        DEFAULT_LOSS = 0.0\n",
    "        if batch_size is None:\n",
    "            batch_size = self.hparams.batch_size\n",
    "\n",
    "        # Mixing ratio for hybrid CTRL training\n",
    "        mix_ratio = self.hparams.mix_ratio\n",
    "        k_cf   = int(batch_size * mix_ratio)\n",
    "        k_real = batch_size - k_cf\n",
    "\n",
    "        # ------------------------------------------\n",
    "        # Real-only\n",
    "        # ------------------------------------------\n",
    "        if replay == 'replay':\n",
    "            if not self.ReplayMem.ready():\n",
    "                return DEFAULT_LOSS\n",
    "            batch = self.ReplayMem.sample_buffer()\n",
    "            states, actions, rewards, next_states, terminals = batch\n",
    "\n",
    "        # ------------------------------------------\n",
    "        # CF-only\n",
    "        # ------------------------------------------\n",
    "        elif replay == 'counterfactual':\n",
    "            if not self.Counterfactual.ready():\n",
    "                return DEFAULT_LOSS\n",
    "            batch = self.Counterfactual.sample_buffer()\n",
    "            states, actions, rewards, next_states, terminals = batch\n",
    "\n",
    "        # ------------------------------------------\n",
    "        # Hybrid CTRL (REAL + CF mix)\n",
    "        # ------------------------------------------\n",
    "        elif replay == 'offline':\n",
    "            if not self.ReplayMem.ready():\n",
    "                return DEFAULT_LOSS\n",
    "            if self.Counterfactual.mem_cnt < k_cf:\n",
    "                return DEFAULT_LOSS\n",
    "\n",
    "            states_real, actions_real, rewards_real, next_real, terms_real = \\\n",
    "                self.ReplayMem.sample_buffer(batch_size=k_real)\n",
    "\n",
    "            states_cf, actions_cf, rewards_cf, next_cf, terms_cf = \\\n",
    "                self.Counterfactual.sample_buffer(batch_size=k_cf)\n",
    "\n",
    "            states      = np.concatenate([states_real,  states_cf], axis=0)\n",
    "            actions     = np.concatenate([actions_real, actions_cf], axis=0)\n",
    "            rewards     = np.concatenate([rewards_real, rewards_cf], axis=0)\n",
    "            next_states = np.concatenate([next_real,   next_cf],   axis=0)\n",
    "            terminals   = np.concatenate([terms_real,  terms_cf],  axis=0)\n",
    "        else:\n",
    "            raise ValueError(f\"Unknown replay={replay}\")\n",
    "\n",
    "        # Convert to tensors\n",
    "        states      = torch.tensor(states, dtype=torch.float32, device=device)\n",
    "        actions     = torch.tensor(actions, dtype=torch.long,   device=device).view(-1)\n",
    "        rewards     = torch.tensor(rewards, dtype=torch.float32, device=device).view(-1)\n",
    "        next_states = torch.tensor(next_states, dtype=torch.float32, device=device)\n",
    "        terminals   = torch.tensor(terminals, dtype=torch.float32, device=device).view(-1)\n",
    "\n",
    "        # ---------------------------------\n",
    "        # Double-DQN target\n",
    "        # ---------------------------------\n",
    "        with torch.no_grad():\n",
    "            next_q_eval = self.policy_net(next_states)\n",
    "            best_actions = next_q_eval.argmax(dim=1)\n",
    "\n",
    "            next_q_target = self.target_net(next_states)\n",
    "            next_vals = next_q_target.gather(1, best_actions.unsqueeze(1)).squeeze(1)\n",
    "\n",
    "            target = rewards + self.hparams.gamma * (1 - terminals) * next_vals\n",
    "\n",
    "        # ---------------------------------\n",
    "        # Q(s,a)\n",
    "        # ---------------------------------\n",
    "        q_pred = self.policy_net(states)\n",
    "        q_taken = q_pred.gather(1, actions.unsqueeze(1)).squeeze(1)\n",
    "\n",
    "        # ---------------------------------\n",
    "        # Loss\n",
    "        # ---------------------------------\n",
    "        loss = F.smooth_l1_loss(q_taken, target)\n",
    "\n",
    "        self.optimizer.zero_grad()\n",
    "        loss.backward()\n",
    "        self.optimizer.step()\n",
    "\n",
    "        self.learn_steps += 1\n",
    "        self.update_network_parameters()\n",
    "\n",
    "        return float(loss.item())\n",
    "\n",
    "\n",
    "\n",
    "    # ===============================================================\n",
    "    # Rollout on real environment\n",
    "    # ===============================================================\n",
    "    def RolloutTrajectory(self, num_episodes=10):\n",
    "        self.policy_net.eval()\n",
    "        rewards = []\n",
    "\n",
    "        for _ in range(num_episodes):\n",
    "            obs, _ = self.env.reset()\n",
    "            terminated = truncated = False\n",
    "            total_r = 0\n",
    "\n",
    "            while not (terminated or truncated):\n",
    "                obs_t = torch.tensor(obs, dtype=torch.float32, device=device)\n",
    "                with torch.no_grad():\n",
    "                    a = self.policy_net(obs_t).argmax().item()\n",
    "\n",
    "                obs, r, terminated, truncated, _ = self.env.step(a)\n",
    "                total_r += r\n",
    "\n",
    "            rewards.append(total_r)\n",
    "\n",
    "        return rewards\n",
    "\n",
    "\n",
    "\n",
    "    # ===============================================================\n",
    "    # Episode loop (collect real experience)\n",
    "    # ===============================================================\n",
    "    def ElapseOneEpisode(self, useCfData=False):\n",
    "\n",
    "        obs, _ = self.env.reset()\n",
    "        epreward = 0\n",
    "        terminated = truncated = False\n",
    "\n",
    "        losses = []\n",
    "\n",
    "        while not (terminated or truncated):\n",
    "\n",
    "            self.decrement_epsilon()\n",
    "\n",
    "            # ε-greedy\n",
    "            if np.random.rand() < self.eps_threshold:\n",
    "                action = self.env.action_space.sample()\n",
    "            else:\n",
    "                obs_t = torch.tensor(obs, dtype=torch.float32, device=device)\n",
    "                with torch.no_grad():\n",
    "                    action = self.policy_net(obs_t).argmax().item()\n",
    "\n",
    "            next_obs, reward, terminated, truncated, _ = self.env.step(action)\n",
    "\n",
    "            self.ReplayMem.store_transition(obs, action, reward, next_obs, terminated)\n",
    "\n",
    "            self.global_steps += 1\n",
    "            epreward += reward\n",
    "            obs = next_obs\n",
    "\n",
    "            # Train\n",
    "            if useCfData:\n",
    "                loss = self.TrainFromReplayMem(replay='offline')\n",
    "            else:\n",
    "                loss = self.TrainFromReplayMem(replay='replay')\n",
    "\n",
    "            if loss != 0.0:\n",
    "                losses.append(loss)\n",
    "\n",
    "        return epreward, np.mean(losses) if losses else 0.0\n",
    "\n",
    "\n",
    "\n",
    "    # ===============================================================\n",
    "    # FULL TRAINING LOOP\n",
    "    # ===============================================================\n",
    "    def train(self, useCfData=False):\n",
    "\n",
    "        # -------------------------------\n",
    "        # Warm-up phase\n",
    "        # -------------------------------\n",
    "        warmup_steps = 300\n",
    "        for _ in range(warmup_steps):\n",
    "            if useCfData:\n",
    "                self.TrainFromReplayMem(replay='offline')\n",
    "            else:\n",
    "                self.TrainFromReplayMem(replay='replay')\n",
    "\n",
    "        # -------------------------------\n",
    "        # Main phase\n",
    "        # -------------------------------\n",
    "        rewards = []\n",
    "        for ep in tqdm(range(self.hparams.max_episodes)):\n",
    "\n",
    "            r, loss = self.ElapseOneEpisode(useCfData=useCfData)\n",
    "            rewards.append(r)\n",
    "\n",
    "            mlflow.log_metric(\"Reward\", r, step=self.global_steps)\n",
    "            mlflow.log_metric(\"Loss\", loss,   step=self.global_steps)\n",
    "\n",
    "        return rewards\n",
    "\n",
    "\n",
    "def apply_dqn_overrides(agent, overrides: dict):\n",
    "    \"\"\"Apply your custom dqnParams to the agent's hparams.\"\"\"\n",
    "    for k, v in overrides.items():\n",
    "        if hasattr(agent.hparams, k):\n",
    "            setattr(agent.hparams, k, v)\n",
    "\n",
    "\n",
    "def prepare_buffer(agent, real_data, cf_data=None):\n",
    "    \"\"\"\n",
    "    Populate an agent's ReplayMem (real data) and Counterfactual (optional).\n",
    "    real_data, cf_data: lists of [s, a, r, s_next, done]\n",
    "    \"\"\"\n",
    "    state_dim = agent.env.observation_space.shape[0]\n",
    "\n",
    "    # ----------------------------\n",
    "    # Real Replay Buffer\n",
    "    # ----------------------------\n",
    "    agent.ReplayMem = ReplayBuffer(\n",
    "        max_size=100000,\n",
    "        state_dim=state_dim,\n",
    "        action_dim=1,                         # scalar action index\n",
    "        batch_size=agent.hparams.batch_size\n",
    "    )\n",
    "\n",
    "    for (s, a, r, s_next, done) in real_data:\n",
    "        agent.ReplayMem.store_transition(s, a, r, s_next, done)\n",
    "\n",
    "    # ----------------------------\n",
    "    # Counterfactual Buffer\n",
    "    # ----------------------------\n",
    "    if cf_data is not None:\n",
    "        agent.Counterfactual = ReplayBuffer(\n",
    "            max_size=500000,\n",
    "            state_dim=state_dim,\n",
    "            action_dim=1,\n",
    "            batch_size=agent.hparams.batch_size\n",
    "        )\n",
    "        for (s, a, r, s_next, done) in cf_data:\n",
    "            agent.Counterfactual.store_transition(s, a, r, s_next, done)\n",
    "\n",
    "\n",
    "def run_experiment(name, agent, use_cf: bool):\n",
    "    \"\"\"\n",
    "    Train the agent and evaluate on the real env.\n",
    "    use_cf=False → Real-only training\n",
    "    use_cf=True  → Hybrid CTRL (real + CF mixing)\n",
    "    \"\"\"\n",
    "    print(f\"\\n========== {name} ==========\")\n",
    "\n",
    "    # Train the agent\n",
    "    agent.train(useCfData=use_cf)\n",
    "\n",
    "    # Evaluate on real env\n",
    "    eval_returns = agent.RolloutTrajectory(num_episodes=10)\n",
    "    mean_ret = np.mean(eval_returns)\n",
    "    std_ret = np.std(eval_returns)\n",
    "\n",
    "    print(f\"[{name}] Eval returns: {eval_returns}\")\n",
    "    print(f\"[{name}] Mean = {mean_ret:.2f}, Std = {std_ret:.2f}\")\n",
    "\n",
    "    return mean_ret, std_ret\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 114,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Real transitions: 47062\n"
     ]
    }
   ],
   "source": [
    "# =====================================================\n",
    "# 0. Convert traindata → numpy transitions\n",
    "# =====================================================\n",
    "\n",
    "real_data = []\n",
    "for s, a, r, s_next, done in traindata:\n",
    "\n",
    "    # a is ALREADY scalar int (0..3) in your dataset\n",
    "    a_idx = int(a)\n",
    "\n",
    "    real_data.append([\n",
    "        s.astype(np.float32),\n",
    "        a_idx,\n",
    "        float(r),\n",
    "        s_next.astype(np.float32),\n",
    "        bool(done)\n",
    "    ])\n",
    "\n",
    "print(\"Real transitions:\", len(real_data))\n",
    "\n",
    "\n",
    "\n",
    "# =====================================================\n",
    "# 1. Agent Hyperparameters\n",
    "# =====================================================\n",
    "\n",
    "tgtEnv   = env\n",
    "obs_dim  = obsdim      # 8\n",
    "act_dim  = actiondim   # 4\n",
    "\n",
    "dqnParams = {\n",
    "    \"min_buffer\":   256,\n",
    "    \"batch_size\":   256,\n",
    "    \"lr\":           5e-4,\n",
    "    \"tau\":          0.001,\n",
    "    \"max_iter\":     1000,\n",
    "    \"eps_start\":    1.0,\n",
    "    \"eps_end\":      0.01,\n",
    "    \"max_episodes\": 50\n",
    "}\n",
    "\n",
    "\n",
    "# =====================================================\n",
    "# 2. Create three agents\n",
    "# =====================================================\n",
    "\n",
    "agent_real = D3QNAgent(\n",
    "    tgtEnv,\n",
    "    DuelingDQN(obs_dim, act_dim),      # actiondim=4\n",
    "    DuelingDQN(obs_dim, act_dim)\n",
    ")\n",
    "apply_dqn_overrides(agent_real, dqnParams)\n",
    "\n",
    "agent_base_s = D3QNAgent(\n",
    "    tgtEnv,\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    DuelingDQN(obs_dim, act_dim)\n",
    ")\n",
    "apply_dqn_overrides(agent_base_s, dqnParams)\n",
    "\n",
    "agent_bicogan = D3QNAgent(\n",
    "    tgtEnv,\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    DuelingDQN(obs_dim, act_dim)\n",
    ")\n",
    "apply_dqn_overrides(agent_bicogan, dqnParams)\n",
    "\n",
    "\n",
    "\n",
    "# =====================================================\n",
    "# 3. Fill Buffers\n",
    "# =====================================================\n",
    "\n",
    "prepare_buffer(agent_real, real_data, cf_data=None)\n",
    "prepare_buffer(agent_base_s, real_data, cf_data=cf_base_s)\n",
    "prepare_buffer(agent_bicogan, real_data, cf_data=cf_bicogan)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 115,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "========== Real Only ==========\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████████████████████████████████████████████████████████████| 50/50 [02:02<00:00,  2.45s/it]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[Real Only] Eval returns: [-41.85250359657477, 8.137275036829307, -28.78263244478804, -46.448018621752965, 96.3049618326196, -37.4313531596586, -71.35196958532521, -66.94594592235777, -73.24738707595348, -37.87506734425104]\n",
      "[Real Only] Mean = -29.95, Std = 47.84\n"
     ]
    }
   ],
   "source": [
    "mean_real, std_real = run_experiment(\"Real Only\", agent_real, use_cf=False)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 118,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "========== Real + Base-S CF ==========\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████████████████████████████████████████████████████████████| 50/50 [02:24<00:00,  2.89s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[Real + Base-S CF] Eval returns: [20.396421589891112, -213.6993834655533, -57.05859902129478, -381.61120379919123, -212.4600222330521, 172.2558848420001, -30.48652753992053, -57.293087721764735, -209.39727698486465, 12.596158425041423]\n",
      "[Real + Base-S CF] Mean = -95.68, Std = 150.48\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "mean_base_s, std_base_s = run_experiment(\"Real + Base-S CF\", agent_base_s, use_cf=True)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 122,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "========== Real + BiCoGAN CF ==========\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████████████████████████████████████████████████████████████| 50/50 [00:41<00:00,  1.19it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[Real + BiCoGAN CF] Eval returns: [27.906098580242926, -53.25468911230733, -6.67551692074619, 204.18040473731935, -162.13550759977247, -163.35593594319553, 14.147390718567877, 11.185126788824448, -11.752284740292751, -95.28589513352135]\n",
      "[Real + BiCoGAN CF] Mean = -23.50, Std = 101.16\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "mean_bicogan, std_bicogan = run_experiment(\"Real + BiCoGAN CF\", agent_bicogan, use_cf=True)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 📌 Counterfactual Selection Logic (Base-S and BiCoGAN SCM)-BellMan Score\n",
    "\n",
    "In this experiment, we interact with the environment normally and, for every\n",
    "real transition $ (s_t, a_t, r_t, s_{t+1}) $, we generate *counterfactual*\n",
    "next-states using either:\n",
    "\n",
    "- the **Base-S Gaussian world model**, or  \n",
    "- the **BiCoGAN structural causal model (SCM)**.\n",
    "\n",
    "Both models can propose alternative next-states for actions *different* from\n",
    "the one actually taken.\n",
    "\n",
    "### 🔍 How the \"best\" counterfactual is chosen\n",
    "\n",
    "For each candidate counterfactual action $ a_{\\text{cf}} \\neq a_t $:\n",
    "\n",
    "1. We pass the current state into the SCM with action $ a_{\\text{cf}} $.\n",
    "\n",
    "2. The SCM predicts:\n",
    "   - a counterfactual next state $ \\hat{s}' $\n",
    "   - a counterfactual reward $ \\hat{r} $\n",
    "   - (BiCoGAN only) a predicted termination flag $ \\hat{d} $.\n",
    "\n",
    "3. For each counterfactual next-state, we compute a **Bellman score**:\n",
    "\n",
    "$$\n",
    "B_i = \\hat{r}_i \\;+\\; \\gamma \\; (1 - \\hat{d}_i)\\; \n",
    "\\max_{a'} Q_{\\text{target}}(\\hat{s}'_i, a')\n",
    "$$\n",
    "\n",
    "4. The counterfactual with the **highest Bellman score** is selected:\n",
    "\n",
    "$$\n",
    "(a_{\\text{best}},\\; \\hat{s}'_{\\text{best}},\\; B_{\\text{best}})\n",
    "$$\n",
    "\n",
    "### ✔ What gets stored\n",
    "\n",
    "Only **one** CF transition is added to the counterfactual buffer, containing:\n",
    "\n",
    "- the original state $s_t$\n",
    "- the selected counterfactual action $a_{\\text{best}}$\n",
    "- the CF reward $ \\hat{r}_{\\text{best}} $\n",
    "- the CF next state $ \\hat{s}'_{\\text{best}} $\n",
    "- the CF termination flag $ \\hat{d}_{\\text{best}} $\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 135,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from tqdm import tqdm\n",
    "import gymnasium as gym\n",
    "from ctrl_data import ReplayBuffer\n",
    "\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "\n",
    "# =====================================================\n",
    "# 1. Dueling DQN network\n",
    "# =====================================================\n",
    "\n",
    "class DuelingDQN(nn.Module):\n",
    "    def __init__(self, state_size, action_size):\n",
    "        super(DuelingDQN, self).__init__()\n",
    "        self.feature_layer = nn.Sequential(\n",
    "            nn.Linear(state_size, 64),\n",
    "            nn.ReLU()\n",
    "        )\n",
    "        self.value_stream = nn.Sequential(\n",
    "            nn.Linear(64, 64),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(64, 1)\n",
    "        )\n",
    "        self.advantage_stream = nn.Sequential(\n",
    "            nn.Linear(64, 64),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(64, action_size)\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        features = self.feature_layer(x)\n",
    "        value = self.value_stream(features)\n",
    "        advantages = self.advantage_stream(features)\n",
    "        q_values = value + (advantages - advantages.mean(dim=-1, keepdim=True))\n",
    "        return q_values\n",
    "\n",
    "\n",
    "# =====================================================\n",
    "# 2. Hyperparameters\n",
    "# =====================================================\n",
    "\n",
    "class D3QNParams:\n",
    "    eps_start = 0.9\n",
    "    eps_end   = 0.05\n",
    "    eps_decay = 200\n",
    "    lr        = 5e-4\n",
    "    min_buffer = 256\n",
    "    batch_size = 64\n",
    "    gamma      = 0.99\n",
    "    reward_min = -500\n",
    "    reward_max = 500\n",
    "    max_episodes = 100\n",
    "    max_iter     = 1000\n",
    "    update_tgt_nw_ep = 50\n",
    "    criteria = torch.nn.MSELoss\n",
    "    optimizer = torch.optim.Adam\n",
    "    decay_steps = 1e8\n",
    "    tau = 0.005\n",
    "    mix_ratio = 0.5\n",
    "\n",
    "\n",
    "# =====================================================\n",
    "# 3. D3QN Agent (Fully Updated with Bellman CF)\n",
    "# =====================================================\n",
    "\n",
    "class D3QNAgent:\n",
    "    def __init__(self, env: gym.Env,\n",
    "                 policy_net: nn.Module,\n",
    "                 target_net: nn.Module,\n",
    "                 scm=None,\n",
    "                 scm_type=None,   # None, \"base_s\", \"bicogan\"\n",
    "                 **kwargs):\n",
    "\n",
    "        self.SetHP()\n",
    "\n",
    "        self.env        = env\n",
    "        self.policy_net = policy_net.to(device)\n",
    "        self.target_net = target_net.to(device)\n",
    "\n",
    "        self.optimizer = self.hparams.optimizer(\n",
    "            self.policy_net.parameters(),\n",
    "            lr=self.hparams.lr\n",
    "        )\n",
    "\n",
    "        self.target_net.eval()\n",
    "        self.policy_net.train()\n",
    "\n",
    "        self.update_network_parameters(tau=1.0)\n",
    "        self.criteria = self.hparams.criteria()\n",
    "\n",
    "        # action dim\n",
    "        if isinstance(env.action_space, gym.spaces.Discrete):\n",
    "            actiondim = 1\n",
    "            self.num_actions = env.action_space.n\n",
    "        else:\n",
    "            actiondim = env.action_space.shape[0]\n",
    "            self.num_actions = actiondim\n",
    "\n",
    "        self.ReplayMem = ReplayBuffer(\n",
    "            max_size=100000,\n",
    "            state_dim=self.env.observation_space.shape[0],\n",
    "            action_dim=actiondim,\n",
    "            batch_size=self.hparams.batch_size\n",
    "        )\n",
    "        self.Counterfactual = ReplayBuffer(\n",
    "            max_size=400000,\n",
    "            state_dim=self.env.observation_space.shape[0],\n",
    "            action_dim=actiondim,\n",
    "            batch_size=self.hparams.batch_size\n",
    "        )\n",
    "\n",
    "        self.global_steps = 0\n",
    "        self.learn_steps  = 0\n",
    "        self.history      = []\n",
    "\n",
    "        self.scm      = scm\n",
    "        self.scm_type = scm_type\n",
    "\n",
    "        self.eps_threshold = self.hparams.eps_start\n",
    "\n",
    "\n",
    "    def SetHP(self):\n",
    "        self.hparams = D3QNParams()\n",
    "\n",
    "\n",
    "    def update_network_parameters(self, tau=None):\n",
    "        if tau is None:\n",
    "            tau = self.hparams.tau\n",
    "        for q_target, q_eval in zip(self.target_net.parameters(),\n",
    "                                    self.policy_net.parameters()):\n",
    "            q_target.data.copy_(\n",
    "                (1 - tau) * q_target.data + tau * q_eval.data\n",
    "            )\n",
    "\n",
    "\n",
    "    def decrement_epsilon(self):\n",
    "        eps_start = self.hparams.eps_start\n",
    "        eps_end   = self.hparams.eps_end\n",
    "        t         = self.global_steps\n",
    "        eps_decay = self.hparams.eps_decay\n",
    "        self.eps_threshold = eps_end + (eps_start - eps_end) * np.exp(-t / eps_decay)\n",
    "\n",
    "\n",
    "    # =================================================\n",
    "    # 3.1 Bellman CF scoring (NEW)\n",
    "    # =================================================\n",
    "    def _select_best_cf_bellman(self, cf_candidates):\n",
    "        \"\"\"\n",
    "        cf_candidates: list of dicts:\n",
    "            {\n",
    "                'a_cf': int,\n",
    "                's_hat': np.array,\n",
    "                'r_hat': float,\n",
    "                'done_hat': bool\n",
    "            }\n",
    "        Returns best candidate using:\n",
    "            Bellman = r_hat + gamma * max_a' Q_target(s_hat,a')\n",
    "        \"\"\"\n",
    "\n",
    "        best_score = -1e15\n",
    "        best_cand  = None\n",
    "\n",
    "        for cand in cf_candidates:\n",
    "            s_hat_tensor = torch.tensor(\n",
    "                cand['s_hat'], dtype=torch.float32, device=device\n",
    "            ).unsqueeze(0)\n",
    "\n",
    "            with torch.no_grad():\n",
    "                q_next = self.target_net(s_hat_tensor).max(dim=1)[0].item()\n",
    "\n",
    "            bellman = cand['r_hat'] + \\\n",
    "                       self.hparams.gamma * (1.0 - float(cand['done_hat'])) * q_next\n",
    "\n",
    "            if bellman > best_score:\n",
    "                best_score = bellman\n",
    "                best_cand  = cand\n",
    "\n",
    "        return best_cand\n",
    "\n",
    "\n",
    "    # =================================================\n",
    "    # 3.2 Base-S Bellman CF selection\n",
    "    # =================================================\n",
    "    def _add_best_cf_base_s(self, state_np, action_real):\n",
    "        if self.scm is None:\n",
    "            return\n",
    "\n",
    "        s_t = torch.tensor(state_np, dtype=torch.float32, device=device).unsqueeze(0)\n",
    "\n",
    "        cf_candidates = []\n",
    "\n",
    "        for a_cf in range(self.num_actions):\n",
    "            if a_cf == action_real:\n",
    "                continue\n",
    "\n",
    "            a_t = torch.tensor([a_cf], dtype=torch.long, device=device)\n",
    "\n",
    "            with torch.no_grad():\n",
    "                mu_s, mu_r, _, _ = self.scm(s_t, a_t)\n",
    "\n",
    "            cf_candidates.append({\n",
    "                'a_cf': a_cf,\n",
    "                's_hat': mu_s.squeeze(0).cpu().numpy(),\n",
    "                'r_hat': float(mu_r.item()),\n",
    "                'done_hat': False\n",
    "            })\n",
    "\n",
    "        best_cf = self._select_best_cf_bellman(cf_candidates)\n",
    "\n",
    "        self.Counterfactual.store_transition(\n",
    "            state_np,\n",
    "            best_cf['a_cf'],\n",
    "            best_cf['r_hat'],\n",
    "            best_cf['s_hat'],\n",
    "            best_cf['done_hat']\n",
    "        )\n",
    "\n",
    "\n",
    "    # =================================================\n",
    "    # 3.3 BiCoGAN Bellman CF selection\n",
    "    # =================================================\n",
    "    def _add_best_cf_bicogan(self, state_np, action_real, reward_real, next_state_np):\n",
    "        if self.scm is None:\n",
    "            return\n",
    "\n",
    "        s_t      = torch.tensor(state_np,      dtype=torch.float32, device=device).unsqueeze(0)\n",
    "        s_next_t = torch.tensor(next_state_np, dtype=torch.float32, device=device).unsqueeze(0)\n",
    "        r_t      = torch.tensor([[reward_real]], dtype=torch.float32, device=device)\n",
    "\n",
    "        # infer u_hat from real transition\n",
    "        with torch.no_grad():\n",
    "            _, _, u_hat = self.scm.backward(s_next_t, r_t)\n",
    "\n",
    "        cf_candidates = []\n",
    "\n",
    "        for a_cf in range(self.num_actions):\n",
    "            if a_cf == action_real:\n",
    "                continue\n",
    "\n",
    "            a_onehot = F.one_hot(\n",
    "                torch.tensor(a_cf, device=device),\n",
    "                self.num_actions\n",
    "            ).float().unsqueeze(0)\n",
    "\n",
    "            with torch.no_grad():\n",
    "                s_hat, done_hat, r_hat = self.scm.forward(s_t, a_onehot, u_hat)\n",
    "\n",
    "            cf_candidates.append({\n",
    "                'a_cf': a_cf,\n",
    "                's_hat': s_hat.squeeze(0).cpu().numpy(),\n",
    "                'r_hat': float(r_hat.item()),\n",
    "                'done_hat': bool(done_hat.item() > 0)\n",
    "            })\n",
    "\n",
    "        best_cf = self._select_best_cf_bellman(cf_candidates)\n",
    "\n",
    "        self.Counterfactual.store_transition(\n",
    "            state_np,\n",
    "            best_cf['a_cf'],\n",
    "            best_cf['r_hat'],\n",
    "            best_cf['s_hat'],\n",
    "            best_cf['done_hat']\n",
    "        )\n",
    "\n",
    "\n",
    "    # =================================================\n",
    "    # 3.4 Replay Training (unchanged)\n",
    "    # =================================================\n",
    "    def TrainFromReplayMem(self,\n",
    "                           replay='replay',\n",
    "                           batch_size=None,\n",
    "                           **kwargs):\n",
    "\n",
    "        DEFAULT_LOSS = 0.0\n",
    "\n",
    "        if batch_size is None:\n",
    "            batch_size = self.hparams.batch_size\n",
    "\n",
    "        mix_ratio = self.hparams.mix_ratio\n",
    "        k_cf   = int(batch_size * mix_ratio)\n",
    "        k_real = batch_size - k_cf\n",
    "\n",
    "        if replay == 'replay':\n",
    "            if not self.ReplayMem.ready(): return DEFAULT_LOSS\n",
    "            states, actions, rewards, next_states, terminals = \\\n",
    "                self.ReplayMem.sample_buffer()\n",
    "\n",
    "        elif replay == 'counterfactual':\n",
    "            if not self.Counterfactual.ready(): return DEFAULT_LOSS\n",
    "            states, actions, rewards, next_states, terminals = \\\n",
    "                self.Counterfactual.sample_buffer()\n",
    "\n",
    "        elif replay == 'offline':\n",
    "            if not self.ReplayMem.ready(): return DEFAULT_LOSS\n",
    "            if self.Counterfactual.mem_cnt < k_cf: return DEFAULT_LOSS\n",
    "\n",
    "            states_real, actions_real, rewards_real, next_real, terms_real = \\\n",
    "                self.ReplayMem.sample_buffer(batch_size=k_real)\n",
    "\n",
    "            states_cf, actions_cf, rewards_cf, next_cf, terms_cf = \\\n",
    "                self.Counterfactual.sample_buffer(batch_size=k_cf)\n",
    "\n",
    "            states      = np.concatenate([states_real, states_cf], axis=0)\n",
    "            actions     = np.concatenate([actions_real, actions_cf], axis=0)\n",
    "            rewards     = np.concatenate([rewards_real, rewards_cf], axis=0)\n",
    "            next_states = np.concatenate([next_real,  next_cf],   axis=0)\n",
    "            terminals   = np.concatenate([terms_real, terms_cf],  axis=0)\n",
    "\n",
    "        else:\n",
    "            raise ValueError(f\"Unknown replay mode: {replay}\")\n",
    "\n",
    "        states_tensor      = torch.tensor(states, dtype=torch.float32, device=device)\n",
    "        actions_tensor     = torch.tensor(actions, dtype=torch.long, device=device).view(-1)\n",
    "        rewards_tensor     = torch.tensor(rewards, dtype=torch.float32, device=device).view(-1)\n",
    "        next_states_tensor = torch.tensor(next_states, dtype=torch.float32, device=device)\n",
    "        terminals_tensor   = torch.tensor(terminals, dtype=torch.float32, device=device).view(-1)\n",
    "\n",
    "        # Double DQN target\n",
    "        with torch.no_grad():\n",
    "            q_eval_next  = self.policy_net(next_states_tensor)\n",
    "            best_actions = q_eval_next.argmax(dim=1)\n",
    "            q_tgt_next   = self.target_net(next_states_tensor)\n",
    "            next_vals    = q_tgt_next.gather(1, best_actions.unsqueeze(1)).squeeze(1)\n",
    "\n",
    "            target = rewards_tensor + self.hparams.gamma * (1 - terminals_tensor) * next_vals\n",
    "\n",
    "        # Q(s,a)\n",
    "        q_pred_all = self.policy_net(states_tensor)\n",
    "        q_taken    = q_pred_all.gather(1, actions_tensor.unsqueeze(1)).squeeze(1)\n",
    "\n",
    "        loss = F.smooth_l1_loss(q_taken, target)\n",
    "\n",
    "        self.optimizer.zero_grad()\n",
    "        loss.backward()\n",
    "        self.optimizer.step()\n",
    "\n",
    "        self.learn_steps += 1\n",
    "        self.update_network_parameters()\n",
    "\n",
    "        return float(loss.item())\n",
    "\n",
    "\n",
    "    # =================================================\n",
    "    # 3.5 Rollout + Training Loops\n",
    "    # =================================================\n",
    "    def RolloutTrajectory(self, num_episodes=20):\n",
    "        self.policy_net.eval()\n",
    "        ep_hist = []\n",
    "\n",
    "        for _ in range(num_episodes):\n",
    "            obs, _ = self.env.reset()\n",
    "            done = False\n",
    "            total = 0\n",
    "\n",
    "            while not done:\n",
    "                obs_t = torch.tensor(obs, dtype=torch.float32, device=device)\n",
    "                with torch.no_grad():\n",
    "                    qvals = self.policy_net(obs_t)\n",
    "                a = int(qvals.argmax().item())\n",
    "                obs, r, terminated, truncated, _ = self.env.step(a)\n",
    "                done = terminated or truncated\n",
    "                total += r\n",
    "\n",
    "            ep_hist.append(total)\n",
    "\n",
    "        return ep_hist\n",
    "\n",
    "\n",
    "    def ElapseOneEpisode(self, episode, useCfData=False):\n",
    "        epreward = 0\n",
    "        losshist = []\n",
    "        trun = term = False\n",
    "        obs, _ = self.env.reset()\n",
    "\n",
    "        for _ in range(self.hparams.max_iter):\n",
    "\n",
    "            self.decrement_epsilon()\n",
    "\n",
    "            # epsilon-greedy\n",
    "            if np.random.rand() < self.eps_threshold:\n",
    "                action = int(self.env.action_space.sample())\n",
    "            else:\n",
    "                with torch.no_grad():\n",
    "                    qvals = self.policy_net(torch.tensor(obs, dtype=torch.float32, device=device))\n",
    "                action = int(qvals.argmax().item())\n",
    "\n",
    "            next_obs, r, term, trun, _ = self.env.step(action)\n",
    "            done = term or trun\n",
    "\n",
    "            self.global_steps += 1\n",
    "            epreward += r\n",
    "\n",
    "            self.ReplayMem.store_transition(\n",
    "                obs.astype(np.float32),\n",
    "                action,\n",
    "                float(r),\n",
    "                next_obs.astype(np.float32),\n",
    "                done\n",
    "            )\n",
    "\n",
    "            # add Bellman CF\n",
    "            if useCfData and self.scm is not None:\n",
    "                if self.scm_type == \"base_s\":\n",
    "                    self._add_best_cf_base_s(obs.astype(np.float32), action)\n",
    "                elif self.scm_type == \"bicogan\":\n",
    "                    self._add_best_cf_bicogan(\n",
    "                        obs.astype(np.float32),\n",
    "                        action,\n",
    "                        float(r),\n",
    "                        next_obs.astype(np.float32)\n",
    "                    )\n",
    "\n",
    "            obs = next_obs\n",
    "\n",
    "            # training\n",
    "            if useCfData and self.scm is not None:\n",
    "                loss = self.TrainFromReplayMem(replay='offline')\n",
    "            else:\n",
    "                loss = self.TrainFromReplayMem(replay='replay')\n",
    "\n",
    "            if loss != 0:\n",
    "                losshist.append(loss)\n",
    "\n",
    "            if done:\n",
    "                break\n",
    "\n",
    "        mean_loss = float(np.mean(losshist)) if len(losshist) > 0 else 0.0\n",
    "        return epreward, len(losshist), mean_loss\n",
    "\n",
    "\n",
    "    def train(self, useCfData=False):\n",
    "        rewardhist = []\n",
    "        self.eps_threshold = self.hparams.eps_start\n",
    "\n",
    "        tq = tqdm(range(self.hparams.max_episodes))\n",
    "        for ep in tq:\n",
    "            r, _, L = self.ElapseOneEpisode(ep, useCfData=useCfData)\n",
    "            rewardhist.append(r)\n",
    "            avgR = np.mean(rewardhist[-100:])\n",
    "\n",
    "            tq.set_postfix({\n",
    "                \"EP\": ep,\n",
    "                \"Reward\": r,\n",
    "                \"Avg100\": avgR,\n",
    "                \"Eps\": self.eps_threshold,\n",
    "                \"Loss\": L\n",
    "            })\n",
    "\n",
    "\n",
    "# =====================================================\n",
    "# 4. Helpers\n",
    "# =====================================================\n",
    "\n",
    "def apply_dqn_overrides(agent, overrides: dict):\n",
    "    for k, v in overrides.items():\n",
    "        if hasattr(agent.hparams, k):\n",
    "            setattr(agent.hparams, k, v)\n",
    "\n",
    "\n",
    "def run_experiment(name, agent, use_cf: bool):\n",
    "    print(f\"\\n========== {name} ==========\")\n",
    "    agent.train(useCfData=use_cf)\n",
    "    eval_returns = agent.RolloutTrajectory(num_episodes=20)\n",
    "    mean_ret = float(np.mean(eval_returns))\n",
    "    std_ret  = float(np.std(eval_returns))\n",
    "    print(f\"[{name}] Eval returns: {eval_returns}\")\n",
    "    print(f\"[{name}] Mean = {mean_ret:.2f}, Std = {std_ret:.2f}\")\n",
    "    return mean_ret, std_ret\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 136,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "========== Real Only ==========\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████| 50/50 [00:22<00:00,  2.18it/s, EP=49, Reward=-250, Avg100=-123, Eps=0.01, Loss=0.721]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[Real Only] Eval returns: [-122.76602445862746, -80.50771865679171, -229.62944144408877, -149.6822336699587, -110.5936117845836, -178.96986954984368, -253.18140881491777, -213.9318208742468, -121.90996664886782, -64.6220148462819, -100.08285514119923, -167.98847789609337, -116.97080101380229, -253.87859777103606, -203.48623263204894, -121.53711729062293, -104.71332963435042, -109.0306058365573, -110.9166666935925, -104.58593831556706]\n",
      "[Real Only] Mean = -145.95, Std = 55.89\n",
      "\n",
      "========== Real + Base-S (best CF) ==========\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█████████████████████████████████████████| 50/50 [00:47<00:00,  1.06it/s, EP=49, Reward=-257, Avg100=-296, Eps=0.01, Loss=0.685]\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[Real + Base-S (best CF)] Eval returns: [-462.2654869449439, -279.25538024788045, -452.7195489616071, -226.80521074992413, -317.314389314108, -209.89133879911958, -459.7898588717712, -524.6399957928978, -186.5066451080918, -422.8147340611467, -435.89205857780894, -169.72656804933743, -280.72154131848686, -171.87461541472095, -306.0212517125792, -295.4958062839528, -221.21206718483847, -487.71031561683793, -209.7882750106632, -419.3641604449976]\n",
      "[Real + Base-S (best CF)] Mean = -326.99, Std = 116.05\n",
      "\n",
      "========== Real + BiCoGAN (best CF) ==========\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████████████████████████████████████| 50/50 [00:32<00:00,  1.52it/s, EP=49, Reward=-190, Avg100=-219, Eps=0.01, Loss=2.17]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[Real + BiCoGAN (best CF)] Eval returns: [-235.833290961278, -316.04090196184427, -297.6428691736561, -338.2177584956844, -249.07834590642665, -216.58151480638082, -355.80971774301366, -152.39870377371415, -97.55562755203884, -58.57358886729654, -285.6171630597247, -345.9310650281652, -130.00229033464095, -323.1548521448624, -323.87229722568657, -283.28019939270786, -241.6226293791774, -333.4175336564732, -322.0988468991379, -342.876508397681]\n",
      "[Real + BiCoGAN (best CF)] Mean = -262.48, Std = 86.85\n",
      "\n",
      "================= FINAL RESULTS (20 eval episodes) =================\n",
      "Real Only:              Mean=-145.95, Std=55.89\n",
      "Real + Base-S (bestCF): Mean=-326.99, Std=116.05\n",
      "Real + BiCoGAN (bestCF):Mean=-262.48, Std=86.85\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "# =====================================================\n",
    "# Create env + sizes (if not already done)\n",
    "# =====================================================\n",
    "# envName = \"LunarLander-v3\"\n",
    "# tgtDom  = Specs(enable_wind=False, turbulence_power=0.0)\n",
    "# env     = gym.make(envName, **vars(tgtDom))\n",
    "# obsdim  = env.observation_space.shape[0]\n",
    "# actiondim = env.action_space.n\n",
    "\n",
    "tgtEnv   = env\n",
    "obs_dim  = obsdim\n",
    "act_dim  = actiondim\n",
    "\n",
    "# DQN overrides\n",
    "dqnParams = {\n",
    "    \"min_buffer\":   256,\n",
    "    \"batch_size\":   256,\n",
    "    \"lr\":           5e-4,\n",
    "    \"tau\":          0.001,\n",
    "    \"max_iter\":     1000,\n",
    "    \"eps_start\":    1.0,\n",
    "    \"eps_end\":      0.01,\n",
    "    \"max_episodes\": 50,   # you can crank to 200 later\n",
    "    \"mix_ratio\":    0.5   # 50% CF in hybrid batches\n",
    "}\n",
    "\n",
    "# =====================================================\n",
    "# 1) Real-only agent\n",
    "# =====================================================\n",
    "agent_real = D3QNAgent(\n",
    "    tgtEnv,\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    scm=None,\n",
    "    scm_type=None\n",
    ")\n",
    "apply_dqn_overrides(agent_real, dqnParams)\n",
    "\n",
    "# =====================================================\n",
    "# 2) Base-S hybrid agent\n",
    "# =====================================================\n",
    "agent_base_s = D3QNAgent(\n",
    "    tgtEnv,\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    scm=base_s,\n",
    "    scm_type=\"base_s\"\n",
    ")\n",
    "apply_dqn_overrides(agent_base_s, dqnParams)\n",
    "\n",
    "# =====================================================\n",
    "# 3) BiCoGAN hybrid agent\n",
    "# =====================================================\n",
    "agent_bicogan = D3QNAgent(\n",
    "    tgtEnv,\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    scm=bicogan,\n",
    "    scm_type=\"bicogan\"\n",
    ")\n",
    "apply_dqn_overrides(agent_bicogan, dqnParams)\n",
    "\n",
    "# =====================================================\n",
    "# Train + evaluate (this will take some time)\n",
    "# =====================================================\n",
    "mean_real,     std_real     = run_experiment(\"Real Only\",           agent_real,     use_cf=False)\n",
    "mean_base_s,   std_base_s   = run_experiment(\"Real + Base-S (best CF)\", agent_base_s,   use_cf=True)\n",
    "mean_bicogan,  std_bicogan  = run_experiment(\"Real + BiCoGAN (best CF)\", agent_bicogan, use_cf=True)\n",
    "\n",
    "print(\"\\n================= FINAL RESULTS (20 eval episodes) =================\")\n",
    "print(f\"Real Only:              Mean={mean_real:.2f}, Std={std_real:.2f}\")\n",
    "print(f\"Real + Base-S (bestCF): Mean={mean_base_s:.2f}, Std={std_base_s:.2f}\")\n",
    "print(f\"Real + BiCoGAN (bestCF):Mean={mean_bicogan:.2f}, Std={std_bicogan:.2f}\")\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## 📌 Counterfactual Selection Logic (Base-S and BiCoGAN SCM) _max Reward\n",
    "\n",
    "In this experiment, we interact with the environment normally and, for every\n",
    "real transition $ (s_t, a_t, r_t, s_{t+1}) $, we generate *counterfactual*\n",
    "next-states using either:\n",
    "\n",
    "- the **Base-S Gaussian world model**, or  \n",
    "- the **BiCoGAN structural causal model (SCM)**.\n",
    "\n",
    "Both models can propose alternative next-states for actions *different* from\n",
    "the one actually taken.\n",
    "\n",
    "### 🔍 How the \"best\" counterfactual is chosen\n",
    "\n",
    "For each candidate counterfactual action $ a_{\\text{cf}} \\neq a_t $:\n",
    "\n",
    "1. We pass the current state into the SCM with action $ a_{\\text{cf}} $.\n",
    "\n",
    "2. The SCM predicts:\n",
    "   - a counterfactual next state $ \\hat{s}' $\n",
    "   - a counterfactual reward $ \\hat{r} $\n",
    "   - (BiCoGAN only) a predicted termination flag.\n",
    "\n",
    "3. We **compare the SCM-predicted rewards**:\n",
    "\n",
    "$$\n",
    "\\hat{r}_{\\text{best}} = \\max_i \\hat{r}_i\n",
    "$$\n",
    "\n",
    "4. The counterfactual with the **highest predicted reward** is selected:\n",
    "\n",
    "$$\n",
    "(a_{\\text{best}},\\; \\hat{s}'_{\\text{best}},\\; \\hat{r}_{\\text{best}})\n",
    "$$\n",
    "\n",
    "### ✔ What gets stored\n",
    "\n",
    "Only **one** CF transition is added to the counterfactual buffer.\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 131,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from tqdm import tqdm\n",
    "import gymnasium as gym\n",
    "from ctrl_data import ReplayBuffer\n",
    "\n",
    "try:\n",
    "    import mlflow\n",
    "    USE_MLFLOW = True\n",
    "except ImportError:\n",
    "    USE_MLFLOW = False\n",
    "\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "\n",
    "# =====================================================\n",
    "# 1. Dueling DQN network\n",
    "# =====================================================\n",
    "\n",
    "class DuelingDQN(nn.Module):\n",
    "    def __init__(self, state_size, action_size):\n",
    "        super(DuelingDQN, self).__init__()\n",
    "        self.feature_layer = nn.Sequential(\n",
    "            nn.Linear(state_size, 64),\n",
    "            nn.ReLU()\n",
    "        )\n",
    "        self.value_stream = nn.Sequential(\n",
    "            nn.Linear(64, 64),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(64, 1)\n",
    "        )\n",
    "        self.advantage_stream = nn.Sequential(\n",
    "            nn.Linear(64, 64),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(64, action_size)\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        features = self.feature_layer(x)\n",
    "        value = self.value_stream(features)\n",
    "        advantages = self.advantage_stream(features)\n",
    "        q_values = value + (advantages - advantages.mean(dim=-1, keepdim=True))\n",
    "        return q_values\n",
    "\n",
    "# =====================================================\n",
    "# 2. Hyperparameter container\n",
    "# =====================================================\n",
    "\n",
    "class D3QNParams:\n",
    "    eps_start = 0.9\n",
    "    eps_end   = 0.05\n",
    "    eps_decay = 200\n",
    "    lr        = 5e-4\n",
    "    min_buffer = 256\n",
    "    batch_size = 64\n",
    "    gamma      = 0.99\n",
    "    reward_min = -500\n",
    "    reward_max = 500\n",
    "    max_episodes = 100\n",
    "    max_iter     = 1000\n",
    "    update_tgt_nw_ep = 50\n",
    "    criteria = torch.nn.MSELoss\n",
    "    optimizer = torch.optim.Adam\n",
    "    decay_steps = 1e8\n",
    "    tau = 0.005\n",
    "    mix_ratio = 0.5      # fraction of CF in hybrid batches\n",
    "\n",
    "# =====================================================\n",
    "# 3. D3QN Agent with online CF (Base-S / BiCoGAN)\n",
    "# =====================================================\n",
    "\n",
    "class D3QNAgent:\n",
    "    def __init__(self, env: gym.Env,\n",
    "                 policy_net: nn.Module,\n",
    "                 target_net: nn.Module,\n",
    "                 scm=None,\n",
    "                 scm_type=None,   # None, \"base_s\", \"bicogan\"\n",
    "                 **kwargs):\n",
    "\n",
    "        self.SetHP()\n",
    "\n",
    "        if USE_MLFLOW:\n",
    "            mlflow.log_param(\"algo\", \"d3qn\")\n",
    "            mlflow.log_params(vars(self.hparams))\n",
    "\n",
    "        self.env        = env\n",
    "        self.policy_net = policy_net.to(device)\n",
    "        self.target_net = target_net.to(device)\n",
    "\n",
    "        self.optimizer = self.hparams.optimizer(\n",
    "            self.policy_net.parameters(),\n",
    "            lr=self.hparams.lr\n",
    "        )\n",
    "\n",
    "        self.target_net.eval()\n",
    "        self.policy_net.train()\n",
    "\n",
    "        self.update_network_parameters(tau=1.0)\n",
    "        self.criteria = self.hparams.criteria()\n",
    "\n",
    "        # discrete vs continuous action spaces\n",
    "        if isinstance(env.action_space, gym.spaces.Discrete):\n",
    "            actiondim      = 1\n",
    "            self.num_actions = env.action_space.n\n",
    "        else:\n",
    "            actiondim        = env.action_space.shape[0]\n",
    "            self.num_actions = actiondim\n",
    "\n",
    "        self.ReplayMem = ReplayBuffer(\n",
    "            max_size=100000,\n",
    "            state_dim=self.env.observation_space.shape[0],\n",
    "            action_dim=actiondim,\n",
    "            batch_size=self.hparams.batch_size\n",
    "        )\n",
    "        self.Counterfactual = ReplayBuffer(\n",
    "            max_size=400000,\n",
    "            state_dim=self.env.observation_space.shape[0],\n",
    "            action_dim=actiondim,\n",
    "            batch_size=self.hparams.batch_size\n",
    "        )\n",
    "\n",
    "        self.global_steps = 0\n",
    "        self.learn_steps  = 0\n",
    "        self.history      = []\n",
    "\n",
    "        self.scm      = scm\n",
    "        self.scm_type = scm_type  # \"base_s\", \"bicogan\", or None\n",
    "\n",
    "        self.eps_threshold = self.hparams.eps_start\n",
    "\n",
    "    def SetHP(self):\n",
    "        self.hparams = D3QNParams()\n",
    "\n",
    "    def update_network_parameters(self, tau=None):\n",
    "        if tau is None:\n",
    "            tau = self.hparams.tau\n",
    "        for q_target_params, q_eval_params in zip(self.target_net.parameters(),\n",
    "                                                  self.policy_net.parameters()):\n",
    "            q_target_params.data.copy_(\n",
    "                (1. - tau) * q_target_params.data + tau * q_eval_params.data\n",
    "            )\n",
    "\n",
    "    def decrement_epsilon(self):\n",
    "        eps_start = self.hparams.eps_start\n",
    "        eps_end   = self.hparams.eps_end\n",
    "        t         = self.global_steps\n",
    "        eps_decay = self.hparams.eps_decay\n",
    "        self.eps_threshold = eps_end + (eps_start - eps_end) * np.exp(-1. * t / eps_decay)\n",
    "\n",
    "    # ---------------- Best-CF helpers ----------------\n",
    "\n",
    "    def _add_best_cf_base_s(self, state_np, action_real):\n",
    "        if self.scm is None:\n",
    "            return\n",
    "\n",
    "        s_t = torch.tensor(state_np, dtype=torch.float32, device=device).unsqueeze(0)\n",
    "\n",
    "        best_r = -1e9\n",
    "        best_a = None\n",
    "        best_s_hat = None\n",
    "\n",
    "        for a_cf in range(self.num_actions):\n",
    "            if a_cf == action_real:\n",
    "                continue\n",
    "            a_t = torch.tensor([a_cf], dtype=torch.long, device=device)\n",
    "            with torch.no_grad():\n",
    "                mu_s, mu_r, logvar_s, logvar_r = self.scm(s_t, a_t)\n",
    "            r_hat = mu_r.item()\n",
    "            if r_hat > best_r:\n",
    "                best_r = r_hat\n",
    "                best_a = a_cf\n",
    "                best_s_hat = mu_s.squeeze(0).detach().cpu().numpy()\n",
    "\n",
    "        if best_a is not None and best_s_hat is not None:\n",
    "            self.Counterfactual.store_transition(\n",
    "                state_np,\n",
    "                best_a,\n",
    "                best_r,\n",
    "                best_s_hat,\n",
    "                False\n",
    "            )\n",
    "\n",
    "    def _add_best_cf_bicogan(self, state_np, action_real, reward_real, next_state_np):\n",
    "        if self.scm is None:\n",
    "            return\n",
    "\n",
    "        s_t      = torch.tensor(state_np,      dtype=torch.float32, device=device).unsqueeze(0)\n",
    "        s_next_t = torch.tensor(next_state_np, dtype=torch.float32, device=device).unsqueeze(0)\n",
    "        r_t      = torch.tensor([[reward_real]], dtype=torch.float32, device=device)\n",
    "\n",
    "        with torch.no_grad():\n",
    "            _, _, u_hat = self.scm.backward(s_next_t, r_t)\n",
    "\n",
    "        best_r = -1e9\n",
    "        best_a = None\n",
    "        best_s_hat = None\n",
    "        best_done = None\n",
    "\n",
    "        for a_cf in range(self.num_actions):\n",
    "            if a_cf == action_real:\n",
    "                continue\n",
    "\n",
    "            a_onehot = F.one_hot(torch.tensor(a_cf, device=device), self.num_actions)\\\n",
    "                        .float().unsqueeze(0)\n",
    "\n",
    "            with torch.no_grad():\n",
    "                s_hat, done_hat, r_hat = self.scm.forward(s_t, a_onehot, u_hat)\n",
    "\n",
    "            r_val = float(r_hat.item())\n",
    "            if r_val > best_r:\n",
    "                best_r = r_val\n",
    "                best_a = a_cf\n",
    "                best_s_hat = s_hat.squeeze(0).detach().cpu().numpy()\n",
    "                best_done = bool(done_hat.item() > 0)\n",
    "\n",
    "        if best_a is not None and best_s_hat is not None:\n",
    "            self.Counterfactual.store_transition(\n",
    "                state_np,\n",
    "                best_a,\n",
    "                best_r,\n",
    "                best_s_hat,\n",
    "                best_done if best_done is not None else False\n",
    "            )\n",
    "\n",
    "    # ---------------- Train from replay ----------------\n",
    "\n",
    "    def TrainFromReplayMem(self,\n",
    "                           replay='replay',   # 'replay', 'counterfactual', 'offline'\n",
    "                           batch_size=None,\n",
    "                           **kwargs):\n",
    "\n",
    "        DEFAULT_LOSS = 0.0\n",
    "\n",
    "        if batch_size is None:\n",
    "            batch_size = self.hparams.batch_size\n",
    "\n",
    "        mix_ratio = self.hparams.mix_ratio\n",
    "        k_cf   = int(batch_size * mix_ratio)\n",
    "        k_real = batch_size - k_cf\n",
    "\n",
    "        if replay == 'replay':\n",
    "            if not self.ReplayMem.ready():\n",
    "                return DEFAULT_LOSS\n",
    "            states, actions, rewards, next_states, terminals = \\\n",
    "                self.ReplayMem.sample_buffer()\n",
    "\n",
    "        elif replay == 'counterfactual':\n",
    "            if not self.Counterfactual.ready():\n",
    "                return DEFAULT_LOSS\n",
    "            states, actions, rewards, next_states, terminals = \\\n",
    "                self.Counterfactual.sample_buffer()\n",
    "\n",
    "        elif replay == 'offline':\n",
    "            if not self.ReplayMem.ready():\n",
    "                return DEFAULT_LOSS\n",
    "            if self.Counterfactual.mem_cnt < k_cf:\n",
    "                return DEFAULT_LOSS\n",
    "\n",
    "            states_real, actions_real, rewards_real, next_real, terms_real = \\\n",
    "                self.ReplayMem.sample_buffer(batch_size=k_real)\n",
    "\n",
    "            states_cf, actions_cf, rewards_cf, next_cf, terms_cf = \\\n",
    "                self.Counterfactual.sample_buffer(batch_size=k_cf)\n",
    "\n",
    "            states      = np.concatenate([states_real,  states_cf], axis=0)\n",
    "            actions     = np.concatenate([actions_real, actions_cf], axis=0)\n",
    "            rewards     = np.concatenate([rewards_real, rewards_cf], axis=0)\n",
    "            next_states = np.concatenate([next_real,   next_cf],   axis=0)\n",
    "            terminals   = np.concatenate([terms_real,  terms_cf],  axis=0)\n",
    "        else:\n",
    "            raise ValueError(f\"Unknown replay mode: {replay}\")\n",
    "\n",
    "        states_tensor      = torch.tensor(states, dtype=torch.float32, device=device)\n",
    "        actions_tensor     = torch.tensor(actions, dtype=torch.long,   device=device).view(-1)\n",
    "        rewards_tensor     = torch.tensor(rewards, dtype=torch.float32, device=device).view(-1)\n",
    "        next_states_tensor = torch.tensor(next_states, dtype=torch.float32, device=device)\n",
    "        terminals_tensor   = torch.tensor(terminals, dtype=torch.float32, device=device).view(-1)\n",
    "\n",
    "        with torch.no_grad():\n",
    "            q_eval_next  = self.policy_net(next_states_tensor)\n",
    "            best_actions = q_eval_next.argmax(dim=1)\n",
    "            q_tgt_next   = self.target_net(next_states_tensor)\n",
    "            next_vals    = q_tgt_next.gather(1, best_actions.unsqueeze(1)).squeeze(1)\n",
    "            target = rewards_tensor + self.hparams.gamma * (1 - terminals_tensor) * next_vals\n",
    "\n",
    "        q_pred_all = self.policy_net(states_tensor)\n",
    "        q_taken    = q_pred_all.gather(1, actions_tensor.unsqueeze(1)).squeeze(1)\n",
    "\n",
    "        loss = F.smooth_l1_loss(q_taken, target)\n",
    "\n",
    "        self.optimizer.zero_grad()\n",
    "        loss.backward()\n",
    "        self.optimizer.step()\n",
    "\n",
    "        self.learn_steps += 1\n",
    "        self.update_network_parameters()\n",
    "\n",
    "        return float(loss.item())\n",
    "\n",
    "    # ---------------- Rollout & Training ----------------\n",
    "\n",
    "    def RolloutTrajectory(self, num_episodes=20):\n",
    "        self.policy_net.eval()\n",
    "        self.policy_net.to(device)\n",
    "        eprewardHist = []\n",
    "\n",
    "        for _ in range(num_episodes):\n",
    "            obs, _ = self.env.reset()\n",
    "            eprewards = 0\n",
    "            terminated = truncated = False\n",
    "\n",
    "            while not (terminated or truncated):\n",
    "                obs_t = torch.tensor(obs, dtype=torch.float32, device=device)\n",
    "                with torch.no_grad():\n",
    "                    action_logits = self.policy_net(obs_t)\n",
    "                action = int(action_logits.argmax(dim=-1).item())\n",
    "\n",
    "                obs, reward, terminated, truncated, _ = self.env.step(action)\n",
    "                eprewards += reward\n",
    "\n",
    "            eprewardHist.append(eprewards)\n",
    "\n",
    "        self.history.append(eprewardHist)\n",
    "        return eprewardHist\n",
    "\n",
    "    def ElapseOneEpisode(self, episode, useCfData: bool = False, **kwargs):\n",
    "        epreward = 0.0\n",
    "        ep_len   = 0\n",
    "        losshist = []\n",
    "\n",
    "        observation, _ = self.env.reset()\n",
    "        terminated = truncated = False\n",
    "\n",
    "        while not (terminated or truncated) and ep_len < self.hparams.max_iter:\n",
    "            self.decrement_epsilon()\n",
    "\n",
    "            if np.random.rand() < self.eps_threshold:\n",
    "                action = int(self.env.action_space.sample())\n",
    "            else:\n",
    "                with torch.no_grad():\n",
    "                    qvals = self.policy_net(\n",
    "                        torch.tensor(observation, dtype=torch.float32, device=device)\n",
    "                    )\n",
    "                    action = int(qvals.argmax().item())\n",
    "\n",
    "            next_observation, reward, terminated, truncated, _ = self.env.step(action)\n",
    "            done_flag = terminated or truncated\n",
    "\n",
    "            self.global_steps += 1\n",
    "            ep_len   += 1\n",
    "            epreward += reward\n",
    "\n",
    "            self.ReplayMem.store_transition(\n",
    "                observation.astype(np.float32),\n",
    "                action,\n",
    "                float(reward),\n",
    "                next_observation.astype(np.float32),\n",
    "                bool(done_flag)\n",
    "            )\n",
    "\n",
    "            if useCfData and (self.scm is not None):\n",
    "                if self.scm_type == \"base_s\":\n",
    "                    self._add_best_cf_base_s(observation.astype(np.float32), action)\n",
    "                elif self.scm_type == \"bicogan\":\n",
    "                    self._add_best_cf_bicogan(\n",
    "                        observation.astype(np.float32),\n",
    "                        action,\n",
    "                        float(reward),\n",
    "                        next_observation.astype(np.float32)\n",
    "                    )\n",
    "\n",
    "            observation = next_observation\n",
    "\n",
    "            if useCfData and (self.scm is not None):\n",
    "                currloss = self.TrainFromReplayMem(replay='offline')\n",
    "            else:\n",
    "                currloss = self.TrainFromReplayMem(replay='replay')\n",
    "\n",
    "            if currloss != 0.0:\n",
    "                losshist.append(currloss)\n",
    "\n",
    "        mean_loss = float(np.mean(losshist)) if len(losshist) > 0 else 0.0\n",
    "        return epreward, ep_len, mean_loss\n",
    "\n",
    "    def train(self, useCfData: bool = False, **kwargs):\n",
    "        rewardhist = []\n",
    "        self.eps_threshold = self.hparams.eps_start\n",
    "\n",
    "        tq = tqdm(range(self.hparams.max_episodes))\n",
    "        for episode in tq:\n",
    "            ep_reward, ep_len, loss = self.ElapseOneEpisode(\n",
    "                episode=episode,\n",
    "                useCfData=useCfData\n",
    "            )\n",
    "            rewardhist.append(ep_reward)\n",
    "            avgReward = float(np.mean(rewardhist[-100:]))\n",
    "\n",
    "            msgDict = {\n",
    "                'EP': episode,\n",
    "                'reward': ep_reward,\n",
    "                'avg reward': avgReward,\n",
    "                'len': ep_len,\n",
    "                'eps': self.eps_threshold,\n",
    "                'mix': self.hparams.mix_ratio,\n",
    "                'loss': loss,\n",
    "                'glob_steps': self.global_steps,\n",
    "                'learnsteps': self.learn_steps\n",
    "            }\n",
    "\n",
    "            if USE_MLFLOW:\n",
    "                mlflow.log_metric(\"Reward\", ep_reward, step=self.global_steps)\n",
    "                mlflow.log_metric(\"Average Reward\", avgReward, step=self.global_steps)\n",
    "                mlflow.log_metric(\"Loss\", loss, step=self.global_steps)\n",
    "\n",
    "            tq.set_postfix(msgDict)\n",
    "\n",
    "# =====================================================\n",
    "# 4. Helpers\n",
    "# =====================================================\n",
    "\n",
    "def apply_dqn_overrides(agent, overrides: dict):\n",
    "    for k, v in overrides.items():\n",
    "        if hasattr(agent.hparams, k):\n",
    "            setattr(agent.hparams, k, v)\n",
    "\n",
    "def run_experiment(name, agent, use_cf: bool):\n",
    "    print(f\"\\n========== {name} ==========\")\n",
    "    agent.train(useCfData=use_cf)\n",
    "    eval_returns = agent.RolloutTrajectory(num_episodes=20)\n",
    "    mean_ret = float(np.mean(eval_returns))\n",
    "    std_ret  = float(np.std(eval_returns))\n",
    "    print(f\"[{name}] Eval returns: {eval_returns}\")\n",
    "    print(f\"[{name}] Mean = {mean_ret:.2f}, Std = {std_ret:.2f}\")\n",
    "    return mean_ret, std_ret\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 134,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "========== Real Only ==========\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█| 50/50 [00:28<00:00,  1.78it/s, EP=49, reward=-177, avg reward=-74.6, len=311, eps=0.05, mix=0.5, loss=0.676, glob_steps=30963\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[Real Only] Eval returns: [-158.8190608485723, -97.5534321830307, -199.2325403983093, -96.68018128666371, -273.11911168438576, 9.35198413877876, -148.7240848380039, -72.29184034956793, -226.8634125457581, -207.05551458777336, -96.90509902575054, -50.15870127878584, -115.9797437852657, -96.34780976056103, -231.93997118102328, 25.246800162947636, -230.37032677222098, -118.69893530682202, -110.13199561431554, -147.6337680068501]\n",
      "[Real Only] Mean = -132.20, Std = 77.72\n",
      "\n",
      "========== Real + Base-S (best CF) ==========\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█| 50/50 [00:19<00:00,  2.52it/s, EP=49, reward=-131, avg reward=-236, len=60, eps=0.05, mix=0.5, loss=0.809, glob_steps=12948, \n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[Real + Base-S (best CF)] Eval returns: [-258.5613276882102, -314.48087794935464, -129.31627063257713, -127.57552931137377, -296.2513593245949, -178.94263633783476, -270.42843320179793, -141.4833046525607, -204.50543242211893, -209.24922287500902, -262.23266137405096, -145.3664484478238, -134.50827925211064, -143.0598768652602, -308.9964739652541, -136.0515694400691, -180.53585945709057, -182.65068568502656, -127.43749175879461, -312.23536830316334]\n",
      "[Real + Base-S (best CF)] Mean = -203.19, Std = 68.44\n",
      "\n",
      "========== Real + BiCoGAN (best CF) ==========\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|█| 50/50 [00:16<00:00,  3.06it/s, EP=49, reward=-119, avg reward=-149, len=102, eps=0.05, mix=0.5, loss=2.72, glob_steps=13561, \n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[Real + BiCoGAN (best CF)] Eval returns: [-369.09168430977047, -104.65104374403424, -100.45537719558257, -214.58242400465696, -101.96316525011551, -236.05796694420047, -194.80067025247922, -60.588110116219426, -99.64540494231431, -113.94534871909457, -73.48412473819866, -210.29446158339644, -81.40923771191495, -212.268111815066, -124.02015387101267, -484.1524886318632, -273.1000017177307, -58.52145672264591, -86.0215379303566, -280.55876513700366]\n",
      "[Real + BiCoGAN (best CF)] Mean = -173.98, Std = 110.41\n"
     ]
    }
   ],
   "source": [
    "# env, base_s, bicogan already trained\n",
    "env      = gym.make(envName, **vars(tgtDom))\n",
    "obs_dim  = env.observation_space.shape[0]\n",
    "act_dim  = env.action_space.n\n",
    "\n",
    "agent_real = D3QNAgent(\n",
    "    env,\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    scm=None,\n",
    "    scm_type=None\n",
    ")\n",
    "\n",
    "agent_base_s = D3QNAgent(\n",
    "    env,\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    scm=base_s,\n",
    "    scm_type=\"base_s\"\n",
    ")\n",
    "\n",
    "agent_bicogan = D3QNAgent(\n",
    "    env,\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    DuelingDQN(obs_dim, act_dim),\n",
    "    scm=bicogan,\n",
    "    scm_type=\"bicogan\"\n",
    ")\n",
    "\n",
    "# optional hyperparam overrides\n",
    "overrides = {\"max_episodes\": 50, \"batch_size\": 64}\n",
    "apply_dqn_overrides(agent_real, overrides)\n",
    "apply_dqn_overrides(agent_base_s, overrides)\n",
    "apply_dqn_overrides(agent_bicogan, overrides)\n",
    "\n",
    "mean_real,    std_real    = run_experiment(\"Real Only\", agent_real,    use_cf=False)\n",
    "mean_base_s,  std_base_s  = run_experiment(\"Real + Base-S (best CF)\", agent_base_s, use_cf=True)\n",
    "mean_bicogan, std_bicogan = run_experiment(\"Real + BiCoGAN (best CF)\", agent_bicogan, use_cf=True)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "device = \"cuda\" if torch.cuda.is_available() else \"cpu\"\n",
    "class DuelingDQN(nn.Module):\n",
    "    def __init__(self, state_size, action_size):\n",
    "        super(DuelingDQN, self).__init__()\n",
    "        self.feature_layer = nn.Sequential(\n",
    "            nn.Linear(state_size, 64),\n",
    "            nn.ReLU()\n",
    "        )\n",
    "        self.value_stream = nn.Sequential(\n",
    "            nn.Linear(64, 64),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(64, 1)\n",
    "        )\n",
    "        self.advantage_stream = nn.Sequential(\n",
    "            nn.Linear(64, 64),\n",
    "            nn.ReLU(),\n",
    "            nn.Linear(64, action_size)\n",
    "        )\n",
    "\n",
    "    def forward(self, x):\n",
    "        features = self.feature_layer(x)\n",
    "        value = self.value_stream(features)\n",
    "        advantages = self.advantage_stream(features)\n",
    "        q_values = value + (advantages - advantages.mean(dim=-1, keepdim=True))\n",
    "        return q_values\n",
    "\n",
    "\n",
    "class D3QNParams:\n",
    "    \"\"\"default set of DQN params. \n",
    "    Mostly copied from the deepmind paper. \n",
    "    \"\"\"\n",
    "    eps_start = 0.9 # deepmind\n",
    "    eps_end = 0.05\n",
    "    eps_decay = 200\n",
    "    lr =  5e-4\n",
    "    min_buffer = 256\n",
    "    batch_size = 64  # deepmind\n",
    "    gamma = 0.99  # deepmind / discount.\n",
    "    reward_min = -500\n",
    "    reward_max = 500\n",
    "    max_episodes = 100 # 500 for lunar lander, 1000 for mountain car.\n",
    "    max_iter = 500\n",
    "    update_tgt_nw_ep = 50 # update target nw every 'n' steps. \n",
    "    criteria = torch.nn.MSELoss # not used. a smooth L1 loss is used later in the code. \n",
    "    optimizer = torch.optim.Adam\n",
    "    decay_steps = 1e8\n",
    "    tau = 0.005\n",
    "\n",
    "class D3QNAgent:\n",
    "\n",
    "    def __init__(self, env: gym.Env, \n",
    "                 policy_net: torch.nn.Module,\n",
    "                target_net: torch.nn.Module,\n",
    "                scm=None,\n",
    "                 **kwargs):\n",
    "        self.SetHP()\n",
    "        mlflow.log_param(\"algo\", \"d3qn\")\n",
    "        mlflow.log_params(vars(self.hparams))\n",
    "\n",
    "        self.learn_steps = 0\n",
    "        self.scm_specified=False\n",
    "\n",
    "        if scm is not None:\n",
    "            self.scm=scm\n",
    "            self.scm_specified=True\n",
    "            # self.scm.gamma=1\n",
    "        \n",
    "        self.env = env\n",
    "        self.policy_net = policy_net\n",
    "        self.target_net = target_net\n",
    "\n",
    "        self.target_net.eval()\n",
    "        self.policy_net.train()\n",
    "        \n",
    "        self.update_network_parameters(tau=1.0)\n",
    "        self.criteria = self.hparams.criteria()\n",
    "            \n",
    "        actiondim = env.action_space.shape[0] if isinstance(env.action_space.sample(), np.ndarray) else 1\n",
    "\n",
    "        self.ReplayMem = ReplayBuffer(max_size = 10000,\n",
    "                                      state_dim= self.env.observation_space.shape[0],\n",
    "                                      action_dim= actiondim, \n",
    "                                      batch_size=self.hparams.batch_size)\n",
    "        \n",
    "        self.Counterfactual = ReplayBuffer(max_size = 400000,\n",
    "                                      state_dim= self.env.observation_space.shape[0],\n",
    "                                      action_dim= actiondim, \n",
    "                                      batch_size=self.hparams.batch_size)\n",
    "        \n",
    "        self.global_steps = 0\n",
    "        self.policy_net = self.policy_net.to(device)\n",
    "        self.target_net = self.target_net.to(device)\n",
    "        self.optimizer = self.hparams.optimizer(self.policy_net.parameters(), \n",
    "                                                lr=self.hparams.lr)\n",
    "        self.history = []\n",
    "        self.hparams.mix_ratio = 0.9\n",
    "\n",
    "    # def TrainGAN(self, batch):\n",
    "    #     fwdloss, bwdloss, efl_loss=self.scm.TrainForwardBackward(batch)\n",
    "    #     discloss=self.scm.TrainDiscriminator(batch)\n",
    "    #     mlflow.log_metric(\"fwdloss\", fwdloss, step = self.global_steps)\n",
    "    #     mlflow.log_metric(\"bwdloss\", bwdloss, step = self.global_steps)\n",
    "    #     mlflow.log_metric(\"efl_loss\", efl_loss, step = self.global_steps)\n",
    "    #     mlflow.log_metric(\"discloss\", discloss, step = self.global_steps)\n",
    "\n",
    "            \n",
    "    def GenerateCounterfactuals(self, state, action, reward, next_state):\n",
    "        \"\"\"given a specific 1-step transition generate countefactual outcomes\n",
    "        and insert to replay memory.\n",
    "        \"\"\"\n",
    "        for a in range(4):\n",
    "            if action != a:\n",
    "                counterfactual_action = torch.tensor(a)\n",
    "                for i in range(10):\n",
    "                    with torch.no_grad():\n",
    "                        shat, ahat, uhat = self.scm.backward(next_state.unsqueeze(0), reward.unsqueeze(0)) # backward. \n",
    "                        action_tensor = nn.functional.one_hot(counterfactual_action, 4).to(device)\n",
    "                        obsaction_tensor = nn.functional.one_hot(torch.tensor(action), 4).to(device)\n",
    "                        s_hat, done_hat, reward_hat = self.scm.forward(state.unsqueeze(0), \n",
    "                                                                       obsaction_tensor.unsqueeze(0), \n",
    "                                                                       uhat)\n",
    "                    self.Counterfactual.store_transition(state.cpu().numpy(), \n",
    "                                                    action, \n",
    "                                                    reward_hat[0].item(), \n",
    "                                                    s_hat.cpu().numpy(), done_hat > 0)\n",
    "                    \n",
    "    def LoadPriorData(self, transdt, scm): \n",
    "        \"\"\" load the prior replay memory so that \n",
    "        inference time is quicker.\n",
    "        \"\"\"\n",
    "        dt = {}\n",
    "        dt[\"h\"] = torch.cat([i[0].unsqueeze(0) for i in transdt])\n",
    "        dt[\"s\"] = torch.vstack([i[1] for i in transdt])\n",
    "        dt[\"r\"] = torch.FloatTensor([i[3] for i in transdt]).unsqueeze(1)\n",
    "        dt[\"s_\"] = torch.vstack([i[4] for i in transdt])\n",
    "        self.Counterfactual = dt\n",
    "        self.scm = scm\n",
    "\n",
    "    def Act(self, explore: bool = True, **kwargs):\n",
    "        pass\n",
    "\n",
    "    def SetHP(self,):\n",
    "        self.hparams = D3QNParams()\n",
    "        # print(vars(D3QNParams()))\n",
    "        \n",
    "    def update_network_parameters(self, tau =None ):\n",
    "        if tau is None:\n",
    "            tau = tau = self.hparams.tau\n",
    "\n",
    "        for q_target_params, q_eval_params in zip(self.target_net.parameters(), self.policy_net.parameters()):\n",
    "            q_target_params.data.copy_((1. - tau) * q_target_params + tau * q_eval_params)\n",
    "    \n",
    "    def decrement_epsilon(self,):\n",
    "        eps_start = self.hparams.eps_start\n",
    "        eps_end = self.hparams.eps_end\n",
    "        t = self.global_steps\n",
    "        eps_decay = self.hparams.eps_decay\n",
    "        self.eps_threshold = eps_end + (eps_start - eps_end) * np.exp(-1. * t / eps_decay)\n",
    "\n",
    "    def SetReplayMem(self, \n",
    "                     path:str = None,\n",
    "                     d_aug = None,\n",
    "                      **kwargs):\n",
    "        \"\"\"set replay memory to a specific dataset and start training from\n",
    "        there. \n",
    "        \"\"\"\n",
    "        with open(path, 'rb') as f: \n",
    "            d_aug = pkl.load(f)\n",
    "            f.close()\n",
    "        \n",
    "        print(\"Loading counterfactual data\")\n",
    "        for old_observation, action, reward, observation, terminated in tqdm.tqdm(d_aug):\n",
    "            # print(str(reward) + \"---\\n\")\n",
    "            self.Counterfactual.store_transition(old_observation, \n",
    "                                             action, \n",
    "                                             reward, \n",
    "                                             observation, \n",
    "                                             terminated)\n",
    "\n",
    "    def TrainFromReplayMem(self, \n",
    "                           replay='replay',\n",
    "                           mix = False,\n",
    "                           mix_ratio = 0.9,# mix_ratio% in offline samples. \n",
    "                           batch_size = None,\n",
    "                             **kwargs):\n",
    "\n",
    "        \"\"\"train policy using a buffer.\n",
    "        For offline problems, use this to\n",
    "        train using a previous buffer.\n",
    "        \"\"\"\n",
    "        if batch_size is not None: batch_size = self.hparams.batch_size\n",
    "       \n",
    "        mix_ratio = self.hparams.mix_ratio\n",
    "        # print(\"mix ration: \", mix_ratio)\n",
    "\n",
    "        if replay == 'replay':\n",
    "            # print(\"sampling from replay memory.\")\n",
    "            if not self.ReplayMem.ready() : return 0\n",
    "            states, actions, rewards, next_states, terminals = self.ReplayMem.sample_buffer()\n",
    "        \n",
    "        else:\n",
    "            if not mix: \n",
    "                states, actions, rewards, next_states, terminals = self.Counterfactual.sample_buffer()\n",
    "            else:\n",
    "                # mix offline & online replay buffer \n",
    "                if self.ReplayMem.mem_cnt >= (1 - mix_ratio) * self.hparams.batch_size:\n",
    "                    batch_size_off = int(mix_ratio * self.hparams.batch_size)\n",
    "                    batch_size_on = self.hparams.batch_size - batch_size_off\n",
    "                    states_on, actions_on, rewards_on, next_states_on, terminals_on = self.ReplayMem.sample_buffer(batch_size=batch_size_on)\n",
    "                    states_off, actions_off, rewards_off, next_states_off, terminals_off = self.Counterfactual.sample_buffer(batch_size=batch_size_off)\n",
    "                    \n",
    "                    states      = np.append(states_on, states_off, axis= 0)\n",
    "                    actions     = np.append(actions_on, actions_off, axis = 0)\n",
    "                    rewards     = np.append(rewards_on, rewards_off, axis = 0)\n",
    "                    next_states = np.append(next_states_on, next_states_off, axis  = 0)\n",
    "                    terminals   = np.append(terminals_off, terminals_on, axis = 0)\n",
    "                \n",
    "                else: return 0\n",
    "        \n",
    "        # batch_idx          = torch.arange(self.hparams.batch_size, dtype=torch.long).to(device)\n",
    "        states_tensor      = torch.tensor(states, dtype=torch.float32).to(device)\n",
    "        actions_tensor     = torch.tensor(actions, dtype=torch.long).to(device)\n",
    "        rewards_tensor     = torch.tensor(rewards, dtype=torch.float32).to(device)\n",
    "        next_states_tensor = torch.tensor(next_states, dtype=torch.float32).to(device)\n",
    "        terminals_tensor   = torch.tensor(terminals, dtype=torch.float).to(device)\n",
    "\n",
    "        actions_onehot= F.one_hot(actions_tensor, 4).resize(actions_tensor.size()[0], 4)\n",
    "\n",
    "        with torch.no_grad():\n",
    "            q_ = self.target_net(next_states_tensor)\n",
    "            max_actions = torch.argmax(input=self.policy_net(next_states_tensor), dim=-1)\n",
    "            # q_[terminals_tensor] = 0.0\n",
    "            nextstatevals  = torch.gather(q_, 1, max_actions.unsqueeze(1) )\n",
    "            target = rewards_tensor + self.hparams.gamma * (1 - terminals_tensor) * nextstatevals.squeeze(1)\n",
    "        \n",
    "        preds = self.policy_net.forward(states_tensor) # bsize X n-\n",
    "        q = torch.gather(preds, 1, actions_tensor)# self.policy_net.forward(states_tensor)[batch_idx, actions_tensor]\n",
    "        # loss =  self.criteria(q.squeeze(1), target.detach())      \n",
    "        loss = F.smooth_l1_loss(q.squeeze(1), target)\n",
    "        self.optimizer.zero_grad()\n",
    "        loss.backward()\n",
    "        self.optimizer.step() \n",
    "        self.learn_steps += 1\n",
    "        self.update_network_parameters()\n",
    "        self.hparams.mix_ratio = 0.0005 if self.hparams.mix_ratio < 0.0005 else self.hparams.mix_ratio - 5e-3\n",
    "        return loss.item() if loss.item() is not None else 0 \n",
    " \n",
    "    def RolloutTrajectory(self, num_episodes=50):\n",
    "        \"\"\"perform infernece with trained policy\n",
    "        \"\"\"\n",
    "        self.policy_net.eval()\n",
    "        self.policy_net.to(device)\n",
    "        eprewardHist = []\n",
    "\n",
    "        for ep in range(num_episodes):\n",
    "            obs = self.env.reset()\n",
    "            eprewards = 0\n",
    "            terminated = truncated = False\n",
    "            while not (terminated or truncated):\n",
    "                obs = torch.tensor(obs, dtype=torch.float32).to(device)\n",
    "                with torch.no_grad():\n",
    "                    action_logits = self.policy_net(obs.to(device))\n",
    "                action = action_logits.argmax(dim=-1).item()\n",
    "                obs, reward, terminated,  _ = self.env.step(action)\n",
    "                eprewards += reward\n",
    "                if terminated or truncated: eprewardHist.append(eprewards)\n",
    "        # print(f\"Average Reward: {np.mean(eprewardHist): 0.2f} +/- {np.std(eprewardHist): 0.2f}\")\n",
    "        self.history.append(eprewardHist)\n",
    "        return eprewardHist    \n",
    "    \n",
    "    def ElapseOneEpisode(self,\n",
    "                         episode, \n",
    "                         useCfData: bool= False,\n",
    "                          **kwargs):\n",
    "        \"\"\"agent training.\"\"\"\n",
    "\n",
    "        epreward = 0\n",
    "        ep_len = 0\n",
    "        observation,_ = self.env.reset()\n",
    "        terminated, truncated = False, False\n",
    "        losshist = []\n",
    "        while (not terminated):  # and (ep_len < self.hparams.max_iter): \n",
    "            self.decrement_epsilon()\n",
    "\n",
    "            # also perform random exploration. \n",
    "            if np.random.random() < self.eps_threshold: \n",
    "                action=self.env.action_space.sample() \n",
    "            else:\n",
    "                with torch.no_grad():\n",
    "                    qvals = self.policy_net(torch.tensor(observation, dtype=torch.float32).to(device))\n",
    "                    action = torch.argmax(qvals).item()\n",
    "    \n",
    "            next_observation, reward, terminated, truncated,  info = self.env.step(action)\n",
    "            self.global_steps+=1\n",
    "            ep_len += 1\n",
    "            epreward += reward \n",
    "            self.ReplayMem.store_transition(observation,\n",
    "                                             action,\n",
    "                                             reward,\n",
    "                                            next_observation,\n",
    "                                            terminated)\n",
    "            # self.GenerateCounterfactuals(torch.tensor(observation, dtype=torch.float32).to(device),\n",
    "            #                               action, \n",
    "            #                               torch.tensor([reward], dtype=torch.float32).to(device), \n",
    "            #                               torch.tensor(next_observation, dtype=torch.float32).to(device))\n",
    "            observation = next_observation\n",
    "            terminated = terminated or truncated\n",
    "                        \n",
    "            if not useCfData: currloss = self.TrainFromReplayMem()\n",
    "            else: currloss = self.TrainFromReplayMem(replay='offline', mix=True)\n",
    "            # self.decrement_epsilon()\n",
    "            \n",
    "            losshist.append(currloss)\n",
    "\n",
    "        return epreward, ep_len, np.mean(losshist)\n",
    "      \n",
    "    def train(self, \n",
    "              useCfData: bool= False,\n",
    "              **kwargs): \n",
    "        \"\"\" full training loop. \n",
    "        \"\"\"\n",
    "\n",
    "        # self.eps_threshold = self.hparams.eps_start\n",
    "        \n",
    "        rewardhist = []\n",
    "\n",
    "        if useCfData:\n",
    "            [self.TrainFromReplayMem(replay = 'counterfactual', batch_size=64) for i in range(500)]\n",
    "        \n",
    "        self.eps_threshold = self.hparams.eps_start\n",
    "        # self.hparams.mix_ratio = 0.9\n",
    "        for episode in (pbar:= tqdm.tqdm(range(self.hparams.max_episodes))): \n",
    "            ep_reward, ep_len, loss = self.ElapseOneEpisode(episode=episode,\n",
    "                                                                   useCfData=useCfData,\n",
    "                                                                   ) \n",
    "            rewardhist.append(ep_reward)\n",
    "            avgReward = np.mean(rewardhist[-100:])\n",
    "            \n",
    "            msgDict  = {'EP': episode, \n",
    "                        'reward': ep_reward, \n",
    "                        'avg reward': avgReward,\n",
    "                        # 'eval reward': np.mean(evalReward),\n",
    "                        'len' : ep_len,\n",
    "                        'eps' : self.eps_threshold, \n",
    "                        'mix': self.hparams.mix_ratio,\n",
    "                        'loss' : loss, \n",
    "                        'glob_steps': self.global_steps,\n",
    "                        'learnsteps:': self.learn_steps}\n",
    "            \n",
    "            mlflow.log_metric(\"Reward\", ep_reward,\n",
    "                              step = self.global_steps)\n",
    "            \n",
    "            mlflow.log_metric(\"Average Reward\", avgReward, \n",
    "                            step = self.global_steps)  \n",
    "            \n",
    "            # mlflow.log_metric(\"Eval Reward\", np.mean(evalReward), \n",
    "            #                 step = episode)      \n",
    "            \n",
    "            mlflow.log_metric(\"Loss\", loss, \n",
    "                            step = self.global_steps)            \n",
    "            \n",
    "            \n",
    "            # mlflow.log_metric(\"Reward steps \", ep_reward, \n",
    "            #                 step = steps)\n",
    "\n",
    "            # torch.save(self.policy_net.state_dict(), f'trained/dqn_{self.env}')\n",
    "            \n",
    "            pbar.set_postfix(msgDict)\n",
    "            # print(msgDict)\n",
    "if True:\n",
    "    \n",
    "    dqnParams = {\"min_buffer\": 256,\n",
    "                 \"batch_size\": 256,\n",
    "                 \"lr\": 0.0005,\n",
    "                 \"tau\": 0.001,\n",
    "                 \"max_iter\": 1000,\n",
    "                 \"eps_start\": 1,\n",
    "                 \"eps_end\" : 0.01,\n",
    "                 \"max_episodes\" : 500}\n",
    "\n",
    "    q_policy   = DuelingDQN(obs_dim, action_dim) \n",
    "    q_target   = DuelingDQN(obs_dim, action_dim)    \n",
    "\n",
    "\n",
    "    \n",
    "    # with open(\"data/lunarLander\", \"rb\") as f: \n",
    "    #     d_aug = pkl.load(f)\n",
    "    #     f.close()\n",
    "    \n",
    "    srcAgent = D3QNAgent(tgtEnv,\n",
    "                            q_policy,\n",
    "                            q_target)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Actual action:  2  Actual reward:  2.3974270820617676\n",
      "Counterfactual Action:  0  Reward:  -1.4742648601531982 -1.4540550708770752\n",
      "Counterfactual Action:  1  Reward:  -0.9091733694076538 -0.920695424079895\n",
      "Counterfactual Action:  2  Reward:  1.9800838232040405 1.983275294303894\n",
      "Counterfactual Action:  3  Reward:  -2.1101512908935547 -2.0957343578338623\n"
     ]
    }
   ],
   "source": [
    "## even if the reward prediction is not good \n",
    "# can we say something about the ordering of rewards ?\n",
    "\n",
    "s, a, r, s_, done = dataset[5]\n",
    "\n",
    "# T(s, a' | (s, a, r, s_)) -> cf_nextstate, cfrward .\n",
    "\n",
    "\n",
    "print(\"Actual action: \", torch.argmax(a).cpu().item(),  \\\n",
    "      \" Actual reward: \", r.cpu().item())\n",
    "# s, a, r, s_\n",
    "for a in range(4):\n",
    "    counterfactual_action = torch.tensor(a)\n",
    "    with torch.no_grad():\n",
    "        shat, ahat, uhat = bicogan.backward(s_.unsqueeze(0), r.unsqueeze(0)) # backward. \n",
    "        action = nn.functional.one_hot(counterfactual_action, 4).to(device)\n",
    "        s_hat, donehat,  reward = bicogan.forward(s.unsqueeze(0), action.unsqueeze(0), uhat)\n",
    "        s_hat1, donehat,  reward1 = bicogan.forward(shat, action.unsqueeze(0), uhat)\n",
    "        print(\"Counterfactual Action: \", counterfactual_action.item(), \n",
    "              \" Reward: \", reward[0].item(), reward1[0].item())\n",
    "        \n",
    "      #   print(\"l1 loss own state: \", nn.functional.l1_loss(shat, s).item())\n",
    "      #   print(\"l1 loss next state: \", nn.functional.l1_loss(s_hat, s_).item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [],
   "source": [
    "# get a heatmap analysing the best action in the observation space. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Actual action:  1  Actual reward:  -1.1468390226364136\n",
      "Counterfactual Action:  0  Reward:  -1.603095372915268\n",
      "Counterfactual Action:  1  Reward:  -1.2067727942466735\n",
      "Counterfactual Action:  2  Reward:  0.6174236614704132\n",
      "Counterfactual Action:  3  Reward:  -2.035371540904045\n"
     ]
    }
   ],
   "source": [
    "# maybe we just need the do-operator. \n",
    "\n",
    "s, a, r, s_,done = dataset[6]\n",
    "print(\"Actual action: \", torch.argmax(a).cpu().item(),  \\\n",
    "      \" Actual reward: \", r.cpu().item())\n",
    "# s, a, r, s_\n",
    "NUM_SAMP=500\n",
    "for a in range(4):\n",
    "    counterfactual_action = torch.tensor(a)\n",
    "    mean_reward = 0.\n",
    "    for samp in range(NUM_SAMP):\n",
    "        with torch.no_grad():\n",
    "            noise = torch.randn(1, 128).to(device)\n",
    "            action = nn.functional.one_hot(counterfactual_action, 4).to(device)\n",
    "            s_hat, donehat, reward = bicogan.forward(s.unsqueeze(0), action.unsqueeze(0), noise)\n",
    "            mean_reward += reward[0].item()\n",
    "    print(\"Counterfactual Action: \", counterfactual_action.item(), \" Reward: \", mean_reward/NUM_SAMP)\n",
    "      \n",
    "            \n",
    "\n",
    "      # with torch.no_grad():\n",
    "      #       noise = torch.randn(1, 128).to(device)\n",
    "      #       action = nn.functional.one_hot(counterfactual_action, 4).to(device)\n",
    "      #       s_hat, donehat, reward = bicogan.forward(s.unsqueeze(0), action.unsqueeze(0), noise)\n",
    "      #       mean_reward += reward[0].item()\n",
    "      \n",
    "#      print(\"Counterfactual Action: \", counterfactual_action.item(), \" Reward: \", mean_reward/5)\n",
    "      \n",
    "      # #   print(\"l1 loss own state: \", nn.functional.l1_loss(shat, s).item())\n",
    "      # print(\"l1 loss next state: \", nn.functional.l1_loss(s_hat, s_).item())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(bicogan.forward, \"trained_forward\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [],
   "source": [
    "torch.save(bicogan.backward, \"trained_backward\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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L3whenp64vjx47Uere6NjY3FjTfeGJs2bar1KJPO0NBQrFq1Kvbu3RuDg4Px9ttvR3d3d4yNjdV6tEnDr9pOAn/3d38XAwMD8bvf/a7Wo0wqTz31VKxZsyZOnz5d61Hq1vz58+Pzn/98DAwMlLZ95jOfiWXLlkV/f38NJ5tccrlcbN++PZYtW1brUSal3//+99Hc3BxDQ0Nx66231nqcScGVj0lgdHQ0pk+fXusxYJzz58/HgQMHoru7e9z27u7u2LNnT42mgvKNjo5GRPh3toLER537z//8z/j+97/vo+u54rzxxhtx4cKFCV8o2dLSMuGLJ+FKlWVZ9Pb2xi233BJz5syp9TiThvi4QvT19UUul3vfZf/+/eMec+LEibjjjjvinnvuia9//es1mrw+XMrrS2Xkcrlx61mWTdgGV6qHHnoofvvb38Yvf/nLWo8yqVTlu10o30MPPRRf/epX3/eY6667rvTnEydOxOLFi0tf3Mf7K/f15fLNmDEjrrrqqglXOU6dOjXhaghciVavXh3PPfdc7N69O2bOnFnrcSYV8XGFmDFjRsyYMeNDHfs///M/sXjx4pg7d25s3rw5PvYxF7A+SDmvL5UxderUmDt3bgwODsZdd91V2j44OBh33nlnDSeD95dlWaxevTq2b98eu3btiq6urlqPNOmIjzpz4sSJWLRoUVx77bXx93//9/H73/++tK+1tbWGk00ex48fjz/84Q9x/PjxuHDhQhw8eDAiIq6//vq45pprajtcnent7Y377rsv5s2bV7pKd/z4cfcoVcDZs2fjtddeK60fPXo0Dh48GNOnT49rr722hpPVv1WrVsWWLVviV7/6VTQ0NJSu3jU1NcW0adNqPN0kkVFXNm/enEXERRcqY8WKFRd9fX/961/XerS69IMf/CDr7OzMpk6dmn3+85/PhoaGaj3SpPDrX//6ov+drlixotaj1b33+jd28+bNtR5t0vA5HwBAUm4WAACSEh8AQFLiAwBISnwAAEmJDwAgKfEBACQlPgCApMQHAJCU+AAAkhIfAEBS4gMASEp8AABJ/T/3YKsmQktLTwAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt \n",
    "plt.hist(uhat.cpu().numpy().flatten())\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## INFERENCE: "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/var/folders/1_/jmlm_4tj4k18gv9r77l0vwkw0000gn/T/ipykernel_87575/516470868.py:1: FutureWarning: You are using `torch.load` with `weights_only=False` (the current default value), which uses the default pickle module implicitly. It is possible to construct malicious pickle data which will execute arbitrary code during unpickling (See https://github.com/pytorch/pytorch/blob/main/SECURITY.md#untrusted-models for more details). In a future release, the default value for `weights_only` will be flipped to `True`. This limits the functions that could be executed during unpickling. Arbitrary objects will no longer be allowed to be loaded via this mode unless they are explicitly allowlisted by the user via `torch.serialization.add_safe_globals`. We recommend you start setting `weights_only=True` for any use case where you don't have full control of the loaded file. Please open an issue on GitHub for any issues related to this experimental feature.\n",
      "  gen = torch.load(\"trained_forward\")\n",
      "/var/folders/1_/jmlm_4tj4k18gv9r77l0vwkw0000gn/T/ipykernel_87575/516470868.py:2: FutureWarning: You are using `torch.load` with `weights_only=False` (the current default value), which uses the default pickle module implicitly. It is possible to construct malicious pickle data which will execute arbitrary code during unpickling (See https://github.com/pytorch/pytorch/blob/main/SECURITY.md#untrusted-models for more details). In a future release, the default value for `weights_only` will be flipped to `True`. This limits the functions that could be executed during unpickling. Arbitrary objects will no longer be allowed to be loaded via this mode unless they are explicitly allowlisted by the user via `torch.serialization.add_safe_globals`. We recommend you start setting `weights_only=True` for any use case where you don't have full control of the loaded file. Please open an issue on GitHub for any issues related to this experimental feature.\n",
      "  backward = torch.load(\"trained_backward\")\n"
     ]
    }
   ],
   "source": [
    "gen = torch.load(\"trained_forward\")\n",
    "backward = torch.load(\"trained_backward\")\n",
    "\n",
    "bicogan = BiCoGAN(forward=gen, \n",
    "            backward=backward, \n",
    "            discriminator=None,\n",
    "            args=None)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [],
   "source": [
    "def GenerateCounterfactuals(self, state, action, reward, next_state):\n",
    "        \"\"\"given a specific 1-step transition generate countefactual outcomes\n",
    "        and insert to replay memory.\n",
    "        \"\"\"\n",
    "        for a in range(4):\n",
    "            if action != a:\n",
    "                counterfactual_action = torch.tensor(a)\n",
    "                for i in range(10):\n",
    "                    with torch.no_grad():\n",
    "                        shat, ahat, uhat = self.scm.backward(next_state.unsqueeze(0), reward.unsqueeze(0)) # backward. \n",
    "                        action_tensor = nn.functional.one_hot(counterfactual_action, 4).to(device)\n",
    "                        obsaction_tensor = nn.functional.one_hot(torch.tensor(action), 4).to(device)\n",
    "                        s_hat, done_hat, reward_hat = self.scm.forward(state.unsqueeze(0), \n",
    "                                                                       obsaction_tensor.unsqueeze(0), \n",
    "                                                                       uhat)\n",
    "                    # self.Counterfactual.store_transition(state.cpu().numpy(), \n",
    "                    #                                 action, \n",
    "                    #                                 reward_hat[0].item(), \n",
    "                    #                                 s_hat.cpu().numpy(), done_hat > 0)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Generate counterfactual data. "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "metadata": {},
   "outputs": [],
   "source": [
    "d_aug = []\n",
    "# s, a, r, s_\n",
    "for s, a, r, s_, done in dataset: \n",
    "    for a_ in range(4):\n",
    "        if torch.argmax(a).cpu().item() != a_:\n",
    "            counterfactual_action = torch.tensor(a_)\n",
    "            with torch.no_grad():\n",
    "                shat, ahat, uhat = bicogan.backward(s_.unsqueeze(0), \n",
    "                                                    r.unsqueeze(0)) # backward. \n",
    "                action = nn.functional.one_hot(counterfactual_action, 4).to(device)\n",
    "                s_hat, donehat,  reward = bicogan.forward(s.unsqueeze(0), \n",
    "                                                          action.unsqueeze(0), \n",
    "                                                          uhat)\n",
    "                toappend = [s.cpu().numpy(), \n",
    "                            a_, \n",
    "                            reward[0].item() , \n",
    "                            s_hat.cpu().numpy(), \n",
    "                            donehat[0].item() > 0 ]\n",
    "                d_aug.append(toappend)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "2"
      ]
     },
     "execution_count": 28,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "# random.choice(range(4))\n",
    "torch.argmax(a).cpu().item()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|████████████████████████████████████████████████████████████████████████████████████████| 47062/47062 [00:07<00:00, 6051.72it/s]\n"
     ]
    }
   ],
   "source": [
    "# generate just 1 cf- data point. \n",
    "d_aug = []\n",
    "# s, a, r, s_\n",
    "for s, a, r, s_, done in tqdm.tqdm(dataset): \n",
    "    a_ = torch.argmax(a).cpu().item() #random.choice(range(4))\n",
    "    while a_ == torch.argmax(a).cpu().item():\n",
    "        a_ =  random.choice(range(4))\n",
    "    \n",
    "    counterfactual_action = torch.tensor(a_)\n",
    "    with torch.no_grad():\n",
    "        shat, ahat, uhat = bicogan.backward(s_.unsqueeze(0), \n",
    "                                            r.unsqueeze(0)) # backward. \n",
    "        action = nn.functional.one_hot(counterfactual_action, 4).to(device)\n",
    "        s_hat, donehat,  reward = bicogan.forward(s.unsqueeze(0), \n",
    "                                                    action.unsqueeze(0), \n",
    "                                                    uhat)\n",
    "        toappend = [s.cpu().numpy(), \n",
    "                    a_, \n",
    "                    reward[0].item() , \n",
    "                    s_hat.cpu().numpy(), \n",
    "                    donehat[0].item() > 0 ]\n",
    "        d_aug.append(toappend)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "47062"
      ]
     },
     "execution_count": 30,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "len(d_aug)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "[array([ 0.00326729,  1.4007356 ,  0.33091703, -0.45263883, -0.00377909,\n",
       "        -0.07495759,  0.        ,  0.        ], dtype=float32),\n",
       " 1,\n",
       " -0.525330662727356,\n",
       " array([[ 0.2512148 ,  1.0998038 ,  0.34717417, -0.6013429 , -0.12733069,\n",
       "          0.01942215,  0.10060097,  0.12164406]], dtype=float32),\n",
       " False]"
      ]
     },
     "execution_count": 31,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "d_aug[0]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [],
   "source": [
    "with open(\"data/lunarLander-1\", 'wb') as f: \n",
    "    pkl.dump(d_aug, f)\n",
    "    f.close() "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "ece60146",
   "language": "python",
   "name": "ece60146"
  },
  "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": 4
}
