{
  "cells": [
    {
      "cell_type": "code",
      "execution_count": 1,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "BN_y7UhVzT-5",
        "outputId": "613122f0-3019-4917-8bb4-f3811d0f8078"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Configuration loaded. Device used: cuda. Node Dim: 3.\n"
          ]
        }
      ],
      "source": [
        "import torch\n",
        "import torch.nn as nn\n",
        "import torch.nn.functional as F\n",
        "from torch.utils.data import Dataset, DataLoader\n",
        "import numpy as np\n",
        "import math\n",
        "import networkx as nx\n",
        "import matplotlib.pyplot as plt\n",
        "import time\n",
        "\n",
        "# --- Global Configuration ---\n",
        "class Config:\n",
        "    num_nodes = 32\n",
        "    node_in_dim = 3    # Positions (x,y) + Parity (1)\n",
        "    edge_in_dim = 1\n",
        "    hidden_dim = 32\n",
        "    num_layers = 2\n",
        "    num_heads = 8\n",
        "    num_cams = 8  # or other value\n",
        "\n",
        "    batch_size = 64\n",
        "    lr = 1e-3\n",
        "    epochs = 30\n",
        "    timesteps = 500\n",
        "\n",
        "    # ADDED: Cosine Schedule Parameters\n",
        "    s_cosine = 0.008 # 's' parameter for the cosine formula, essential for noise addition\n",
        "\n",
        "    parite_penalisation_weight = 0.5\n",
        "    device = torch.device(\"cuda\" if torch.cuda.is_available() else \"cpu\")\n",
        "\n",
        "cfg = Config()\n",
        "print(f\"Configuration loaded. Device used: {cfg.device}. Node Dim: {cfg.node_in_dim}.\")"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 2,
      "metadata": {
        "id": "rxvMN29z2T5G",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "b59644f3-0315-47a2-ba2a-05eff3769892"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Generating 1000 graphs of 32 nodes...\n"
          ]
        }
      ],
      "source": [
        "import torch\n",
        "from torch.utils.data import Dataset\n",
        "import numpy as np\n",
        "import math\n",
        "\n",
        "class SpatialNetworkDataset(Dataset):\n",
        "    def __init__(self, num_samples=None, num_nodes=None, data=None):\n",
        "        \"\"\"\n",
        "        If data=None -> a dataset is generated.\n",
        "        If data is provided -> a saved dataset is loaded.\n",
        "        \"\"\"\n",
        "        if data is not None:\n",
        "            # Loading from an existing file or list\n",
        "            self.data = data\n",
        "        else:\n",
        "            # Generating a full dataset\n",
        "            assert num_samples is not None and num_nodes is not None\n",
        "            print(f\"Generating {num_samples} graphs of {num_nodes} nodes...\")\n",
        "            self.data = [self.generate_graph(num_nodes) for _ in range(num_samples)]\n",
        "\n",
        "    # ----------------------------------------------------------\n",
        "    # GENERATION METHODS (IMPROVED)\n",
        "    # ----------------------------------------------------------\n",
        "\n",
        "    def get_quadrant(self, dx, dy):\n",
        "        \"\"\"\n",
        "        Returns the quadrant (0, 1, 2, 3) based on the angle.\n",
        "        0: East, 1: North, 2: West, 3: South\n",
        "        \"\"\"\n",
        "        angle = math.degrees(math.atan2(dy, dx))\n",
        "        if -45 < angle <= 45: return 0\n",
        "        if 45 < angle <= 135: return 1\n",
        "        if 135 < angle <= 180 or -180 <= angle <= -135: return 2\n",
        "        return 3\n",
        "\n",
        "    def generate_graph(self, n):\n",
        "        # 1. Feature Generation (Positions + Parity)\n",
        "        pos = np.random.rand(n, 2)\n",
        "        parity = np.random.randint(0, 2, size=(n, 1))\n",
        "\n",
        "        # Concatenation for model input (Pos, Parity)\n",
        "        x_feat = np.concatenate([pos, parity], axis=-1)\n",
        "\n",
        "        adj = np.zeros((n, n), dtype=np.float32)\n",
        "\n",
        "        # Structure to track quadrant occupancy for each node\n",
        "        # quadrants_status[i][q] = True if quadrant q of node i is already occupied\n",
        "        quadrants_status = np.zeros((n, 4), dtype=bool)\n",
        "        current_degrees = np.zeros(n, dtype=int)\n",
        "\n",
        "        # 2. Pre-calculation of all distances\n",
        "        # Distance matrix (N, N)\n",
        "        dist_matrix = np.sum((pos[:, None, :] - pos[None, :, :]) ** 2, axis=-1)\n",
        "\n",
        "        # Create a list of potential candidates: (distance, u, v)\n",
        "        # Only consider pairs with different parity (fundamental constraint)\n",
        "        candidates = []\n",
        "        for i in range(n):\n",
        "            for j in range(i + 1, n): # i+1 to avoid duplicates (u,v) = (v,u) and diagonal\n",
        "                if parity[i] != parity[j]:\n",
        "                    candidates.append((dist_matrix[i, j], i, j))\n",
        "\n",
        "        # Global sorting of candidates by increasing distance (Global Greedy approach)\n",
        "        # This allows forming short links first across the entire graph\n",
        "        candidates.sort(key=lambda x: x[0])\n",
        "\n",
        "        # 3. Strict Filling (Main Loop)\n",
        "        for dist, i, j in candidates:\n",
        "            # Check max degrees (Max 4 links per node)\n",
        "            if current_degrees[i] >= 4 or current_degrees[j] >= 4:\n",
        "                continue\n",
        "\n",
        "            # Calculate directional vectors for i->j and j->i\n",
        "            dx_ij, dy_ij = pos[j, 0] - pos[i, 0], pos[j, 1] - pos[i, 1]\n",
        "            dx_ji, dy_ji = pos[i, 0] - pos[j, 0], pos[i, 1] - pos[j, 1]\n",
        "\n",
        "            # Calculate respective quadrants\n",
        "            q_i = self.get_quadrant(dx_ij, dy_ij) # In which quadrant of i is j?\n",
        "            q_j = self.get_quadrant(dx_ji, dy_ji) # In which quadrant of j is i?\n",
        "\n",
        "            # STRICT BIDIRECTIONAL VERIFICATION\n",
        "            # Link is added ONLY if the quadrant is free for both i AND j\n",
        "            if not quadrants_status[i, q_i] and not quadrants_status[j, q_j]:\n",
        "                # Validate the link\n",
        "                adj[i, j] = 1\n",
        "                adj[j, i] = 1\n",
        "\n",
        "                # Update statuses\n",
        "                quadrants_status[i, q_i] = True\n",
        "                quadrants_status[j, q_j] = True\n",
        "                current_degrees[i] += 1\n",
        "                current_degrees[j] += 1\n",
        "\n",
        "        # 4. Rescue Isolated Nodes\n",
        "        # Isolated nodes are connected to the nearest neighbor, regardless of quadrant constraints.\n",
        "        isolated_nodes = np.where(current_degrees == 0)[0]\n",
        "\n",
        "        for i in isolated_nodes:\n",
        "            # Create a copy of distances to find the min by excluding itself\n",
        "            dists = dist_matrix[i].copy()\n",
        "            dists[i] = np.inf\n",
        "\n",
        "            # Sort neighbors by distance\n",
        "            nearest_neighbors = np.argsort(dists)\n",
        "\n",
        "            target = -1\n",
        "\n",
        "            # Strategy 1: Try to find the nearest with different parity\n",
        "            # (to respect the parity rule if possible)\n",
        "            for neighbor in nearest_neighbors:\n",
        "                if parity[i] != parity[neighbor]:\n",
        "                    target = neighbor\n",
        "                    break\n",
        "\n",
        "            # Strategy 2: If failed (all neighbors have same parity), take the absolute nearest\n",
        "            if target == -1:\n",
        "                target = nearest_neighbors[0]\n",
        "\n",
        "            # Force connection (without checking quadrants_status)\n",
        "            adj[i, target] = 1\n",
        "            adj[target, i] = 1\n",
        "            # Note: quadrants_status is not updated here because it's a forced connection\n",
        "\n",
        "        return (\n",
        "            torch.tensor(x_feat, dtype=torch.float32),\n",
        "            torch.tensor(adj, dtype=torch.float32)\n",
        "        )\n",
        "\n",
        "    # ----------------------------------------------------------\n",
        "    # Standard PyTorch Dataset Methods\n",
        "    # ----------------------------------------------------------\n",
        "\n",
        "    def __len__(self):\n",
        "        return len(self.data)\n",
        "\n",
        "    def __getitem__(self, idx):\n",
        "        return self.data[idx]\n",
        "\n",
        "    # ----------------------------------------------------------\n",
        "    # ☢️ SAVE / LOAD METHODS\n",
        "    # ----------------------------------------------------------\n",
        "\n",
        "    def save(self, path):\n",
        "        \"\"\"\n",
        "        Saves the complete dataset to a .pt file.\n",
        "        \"\"\"\n",
        "        torch.save(self.data, path)\n",
        "        print(f\"[OK] Dataset saved to {path} ({len(self.data)} graphs)\")\n",
        "\n",
        "    @staticmethod\n",
        "    def load(path):\n",
        "        \"\"\"\n",
        "        Loads a previously saved dataset.\n",
        "        Returns a SpatialNetworkDataset ready for use.\n",
        "        \"\"\"\n",
        "        data = torch.load(path)\n",
        "        print(f\"[OK] Dataset loaded from {path} ({len(data)} graphs)\")\n",
        "        return SpatialNetworkDataset(data=data)\n",
        "\n",
        "# --- Example usage (Cell 3) ---\n",
        "dataset = SpatialNetworkDataset(num_samples=1000, num_nodes=cfg.num_nodes)\n",
        "dataloader = DataLoader(dataset, batch_size=cfg.batch_size, shuffle=True)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 3,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 559
        },
        "id": "0-cWL71h2as6",
        "outputId": "c721f31e-c93e-436b-98b1-051b42950e83"
      },
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 500x500 with 1 Axes>"
            ],
            "image/png": "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jzyC3/3udwCAmJgY52fn/Pnz2LBhAzIzM132t379enAch6+++srlGBzHYfPmzc7b2tvbcffddyMyMhJSqRTx8fF46qmnXHpFAf4C+PzzzyMlJQUymQzBwcH4+c9/3uf8NZrPh6M9N998M3x8fKBWq3HTTTf1OwTZO4fJcU3atWsXTpw44XxtHHlk586dw7fffuvymgFAT08PHn74YcTHxzu/3/fddx96enpcjsVxHH7xi1/gvffeQ0pKCqRSKb777jsAQF1dHf7nf/4HwcHBkEqlSElJwZtvvunyeEc7PvroI/z5z39GREQEZDIZVqxYgcrKSud2S5cuxbfffguNRuNs61Byiv7zn/9g/vz58PLygq+vL5YsWYKtW7e6bPOPf/zD2fawsDDcddddQxraNRqNuPfee52fj8TERPz1r38FY2zIr9EHH3yArKwsKJVKqFQqzJkzBy+88MKgxx4UG4a33nqLAWDbt29nLS0tLn+tra3O7cxmM8vIyGDR0dGso6ODMcbYd999xwCwxx9/3LldS0sLCw0NZffccw975ZVX2NNPP80SExOZWCxmxcXFzu3OnTvHALD09HSWnJzMnnvuOfZ///d/TCKRsAULFrA//OEPLCcnh7344ovsV7/6FeM4jt1yyy0ubY+OjmYJCQlMrVaz+++/nz333HNszpw5TCAQsK1btzq327VrFwPAdu3a5bxtx44dTCKRsIULF7Jnn32W/e1vf2OpqalMIpGwgwcPXvQ1u/766xkAVltbO+zXOTk5meXl5bGXXnqJ/eUvf2GMMfbxxx+ztLQ09tBDD7HXX3+d/eEPf2C+vr4sOjqaGY3GPvuYM2cOy83NZS+++CK76667mEAgYEuWLGF2u93ltUlMTGTBwcHsD3/4A3v55ZdZZmYm4ziOlZWVXbStZrOZyeVyds8997jcXl5ezgCw//mf/xnSczaZTGzWrFlMLBaz3/zmN+zFF19kubm5DAB7/vnnXbYFwB5++GHnv8vKypi3tzcLDQ1ljz/+OPvLX/7CYmJimFQqZYWFhUN6XQdy4ed8oL/u7u5BnyMAdtddd7ncptPpmFAoZAsWLGCMMfbLX/6SXXrppeyJJ55gr732Grv11luZUChkV199tcvjbrrpJiaVSllcXBy76aab2Kuvvsr+/e9/O++Ljo5mjDFmMBjYK6+8wgCwq666ir377rvs3XffZceOHWMdHR1MLpeze++9t09bk5OT2fLly11us1gsA24/FDfddBPz9va+6DaO9+jcuXPO24b6+XQ89vDhw4wx/r1LT09nUVFRrLKy0rldTk4O8/HxYf/3f//H3njjDfbEE0+wZcuWsd27d1+0bdu2bWMikYglJCSwp59+mj366KMsICCA+fr6urT34YcfZgBYRkYG27BhA/vHP/7BbrvtNgaA3XfffS77jI6OZjfddJPz34sWLWJZWVl9jn3nnXcypVLp8h2/kOPcNWfOHJaamsqee+45dv/99zOZTMYSEhKYyWRybjuUc9qxY8ec56+//e1vzs+OwWBgzz33HBMIBEyv1zPGGLPb7czX15cJBAL229/+1nmcZ555xmU7o9HIUlNTmb+/P/vDH/7AXn31VbZx40bGcRz79a9/7fJ8brvtNiYSidjtt9/OXn31Vfb73/+eeXt7s3nz5jGz2ezyGo70/GW329mSJUuYQCBgd955J3vppZfY8uXLWWpqKgPA3nrrLee2jveVMf579e6777KkpCQWERHhfG0aGxvZu+++ywICAlh6errLa2az2djq1auZl5cXu/vuu9lrr73GfvGLXzCRSMSuuOIKl3YBYLNmzWKBgYHs0UcfZX//+99ZcXExa2xsZBERESwyMpI99thj7JVXXmGXX3658z268LOQkZHBsrKy2N/+9jf2yCOPMC8vLzZ//nzndlu3bmXp6eksICDA2dbPP//8oq/ZI488wgCwnJwc9swzz7AXXniB3XDDDez3v/99n9dq5cqV7KWXXmK/+MUvmFAo7PPe9T5XOd6P5cuXM47j2G233cZefvlltn79egaA3X333UN6jbZu3coAsBUrVrC///3v7O9//zv7xS9+wa655pqLPq+hGFHA1N+fVCp12ba0tJRJJBJ22223sba2NhYeHs7mzp3LLBaLcxur1cp6enpcHtfW1saCg4NdLrSOgCkwMJC1t7c7b3/ggQcYAJaWluay3+uvv55JJBKXi1h0dDQDwD799FPnbXq9noWGhrKMjAznbRcGTHa7nc2cOZOtWbPGJcgwmUwsJiaGrVq16qKvWUZGBlOr1X1uNxgMLhdcxwmFsR9e58WLFzOr1eryuN4nPYcDBw4wAM4LZu99ZGVluXxAn376aQaAffnll31emz179jhva25uZlKpdNCLY2VlJQPAXnrpJZfbv/zyyz5f4ot5/vnnGQD2n//8x3mb2WxmCxcuZAqFwhl4M9Y3YLryyiuZRCJhZ8+edd5WX1/PlEolW7JkifO2i72uAxno837hX+8T68X2deutt7KWlhbW3NzMDh48yFasWMEAsGeffZYx1v/7++STTzKO45hGo3HedtNNNzEA7P777++z/YUnoZaWlj6vmcP111/PwsLCmM1mc95WVFQ04HNKSEhga9euHfS59mc0AdNQPp+9A6aGhgaWkpLCYmNj2fnz553btLW1MQDsmWeeGXb709PTWVBQENNqtc7bjh07xgQCAdu4caPzNsfF4sIfC1dddRXz9/d3ue3CgOm1115jANipU6ect5nNZhYQEOCyXX8c567w8HCX78tHH33EALAXXniBMTa8c9ozzzzT5/1gjLHDhw8zAGzTpk2MMcaOHz/OALBrrrmGZWdnO7e7/PLLXc6vjz/+OPP29mbl5eUu+7v//vuZUChk1dXVjDHG9u7dywCw9957z2U7xw/v3reP5vz1xRdfMADs6aefdt5mtVqdP9YGCpgc8vLyWEpKSp/9RkdHs8suu8zltnfffZcJBAK2d+9el9tfffVVBoDt37/feRsAJhAI2IkTJ1y2vfXWW1loaKhLBwVjjF133XXMx8fHef5wfBZmzZrlco194YUXGABWWlrqvO2yyy5zOV9cTEVFBRMIBOyqq65yOWcwxpyfpebmZiaRSNjq1atdtnn55ZcZAPbmm286b7vwXOV4P/70pz+57Pvqq69mHMe5/PAZ6DX69a9/zVQq1ZDP8cMxoiG5v//979i2bZvLX+8uVwCYPXs2Hn30UbzxxhtYs2YNWltb8c4777iMAQuFQkgkEgB896tOp4PVasXcuXNRVFTU57jXXHMNfHx8nP/Ozs4GAPz0pz912W92djbMZrPLcAIAhIWF4aqrrnL+2zFEVVxcjMbGxn6fa0lJCSoqKnDDDTdAq9WitbUVra2tMBqNWLFiBfbs2dOnK7m3jo6OfvM9/vjHPyIwMND511/i8+23394nn0Mulzv/32KxQKvVIj4+Hmq1ut/X7Gc/+5nLcNgdd9wBkUiETZs2uWyXnJyM3Nxc578DAwORmJiIqqqqAZ8bAGi1WgCAr6+vy+0dHR0AAKVSedHHO2zatAkhISG4/vrrnbeJxWL86le/gsFgwO7du/t9nM1mw9atW3HllVciNjbWeXtoaChuuOEG7Nu3z9kWh/5e14Fc+Dkf6G/NmjVD2t+//vUvBAYGIigoCNnZ2di/fz/uueceZ8Jl7/fXaDSitbUVOTk5YIz1O3R2xx13DOm4A9m4cSPq6+tdhsHfe+89yOVy/OhHP+qzva+vL1pbW0d1zJEYzueztrYWeXl5sFgs2LNnD6Kjo533yeVySCQS5OfnDzg03Z+GhgaUlJTg5ptvhp+fn/P21NRUrFq1qs/3CQD+93//1+Xfubm50Gq1fT6PvV177bWQyWR47733nLdt2bIFra2tLrlvF7Nx40aX793VV1+N0NBQZxtHe04D+EkACoUCe/bsAcAPJUdERGDjxo0oKiqCyWQCYwz79u1zed8+/vhj5ObmOj9Hjr+VK1fCZrM59/fxxx/Dx8cHq1atctkuKysLCoWiT9rGSM9fmzZtgkgkcvkeCYVC/PKXv7zo40bi448/xqxZs5CUlOTynJYvXw4AfZ5TXl6eSx4UYwyffvop1q9fD8aYyz7WrFkDvV7f5xpwyy23OK+xAJyv0WCvy0C++OIL2O12PPTQQxAIXMMHx3Dl9u3bYTabcffdd7tsc/vtt0OlUl10VvimTZsgFArxq1/9yuX2e++9F4yxPnHGha8RAKjV6jFLHRhR0vf8+fOHlPT9u9/9Dh988AEOHTqEJ554ot8kuHfeeQfPPvssTp8+7ZIA3N8svKioKJd/O4KnyMjIfm+/8IQYHx/fp45GQkICAH5MOiQkpM8xKyoqAAA33XRT/08SgF6v7xMwOCiVSmdQ0dudd97pzN8Z6ETY32vQ1dWFJ598Em+99Rbq6upcxnX1en2f7WfOnOnyb4VCgdDQ0D75Ghe+tgB/cRzqRaV3OwA+GAWAzs7OIT1eo9Fg5syZfb6Es2bNct7fn5aWFphMJiQmJva5b9asWbDb7aipqUFKSorz9v5e14GsXLlyyNsOxRVXXIFf/OIX4DgOSqUSKSkpLknn1dXVeOihh/DVV1/1ee0vfH9FItGoc7BWrVqF0NBQvPfee1ixYgXsdjv++9//4oorrug32GWM9fkOjYfhfD5vvPFGiEQinDp1qs93WiqV4qmnnsK9996L4OBgLFiwAOvWrcPGjRv7/f47OD5/A33OtmzZ0mcCwYVtdpwj2tranN+PC6nVaqxfvx7vv/8+Hn/8cQB8ABseHu68sA7mwu88x3GIj493fudHe04D+KBi4cKF2Lt3LwA+YMrNzcXixYths9lQWFiI4OBg6HQ6l0CmoqICx48fd+bVXai5udm5nV6vR1BQ0EW3cxjp+Uuj0SA0NLTPj9r+3ufRqqiowKlTpwZ97g4XnqdaWlrQ3t6O119/Ha+//vqQ9nGxz+BInD17FgKB4KIJ7QN9VyQSCWJjYwc8lzseGxYW1ufcM9B1oL9z+Z133omPPvoIa9euRXh4OFavXo1rr70Wl1xyycWf3BCMKGAaqqqqKueXs7S0tM/9//nPf3DzzTfjyiuvxO9+9zsEBQVBKBTiySefxNmzZ/tsP1CvwEC3X3gRHwnHL61nnnkG6enp/W5zsRlDSUlJKCkpQV1dHcLDw523JyQkOIO1gbL6e/c2OPzyl7/EW2+9hbvvvhsLFy6Ej48POI7DddddN+ivwosZ6Wvo7+8PoO8XMCkpCUD/77un9fe6DmSgnscL+fj4DGm/ERERAwZhNpsNq1atgk6nw+9//3skJSXB29sbdXV1uPnmm/u8v1KptE+AOVxCoRA33HAD/vnPf+If//gH9u/fj/r6+gGD+La2tj4X5PEwnM/nhg0b8O9//xsvvPACnnzyyT7333333Vi/fj2++OILbNmyBQ8++CCefPJJ7Ny5ExkZGR5pc28bN27Exx9/jIKCAsyZMwdfffUV7rzzzlG/1w6jPac5LF68GH/+85/R3d2NvXv34o9//CPUajVmz56NvXv3Ijg4GABcAia73Y5Vq1bhvvvu63efjnOi3W5HUFCQS09bbxcGHWN5DXAXu92OOXPm4Lnnnuv3/gt/+F94PnG8bz/96U8HDHZTU1Nd/j0ZXpfR6O+cGxQUhJKSEmzZsgWbN2/G5s2b8dZbb2Hjxo1DnkQ0kDELmOx2O26++WaoVCrcfffdeOKJJ3D11Vdjw4YNzm0++eQTxMbG4rPPPnP51TpWtSsqKyv7/EIuLy8HgAFnBsTFxQHge0xG0tuwbt06fPDBB3jvvfcGPEkMxyeffIKbbroJzz77rPO27u7uAWcfVFRUYNmyZc5/GwwGNDQ04NJLLx11WwD+F4xcLnfO8nJISEhAYmIivvzyS7zwwguDnoCjo6Nx/Phx2O12lwvD6dOnnff3JzAwEF5eXs4ZUL2dPn0aAoGgz4loOEJDQ4e03VtvvTXqis2lpaUoLy/HO++8g40bNzpvH23X8mA9Qhs3bsSzzz6Lr7/+Gps3b0ZgYGC/Q4xWqxU1NTW4/PLLR9WesfbLX/4S8fHxeOihh+Dj49Ondg7Af6/vvfde3HvvvaioqEB6ejqeffZZ/Oc//+l3n47P30Cfs4CAALeVp7jkkksQGBiI9957D9nZ2TCZTMOq0eb4kerAGENlZaXzYjqcc9rFPju5ubkwm83473//i7q6OmdgtGTJEmfAlJCQ4AycHMc2GAyDHjcuLg7bt2/HokWLhvUDZ7iio6OxY8cOGAwGl3NUf+/zaMXFxeHYsWNYsWLFiHppAwMDoVQqYbPZ3NrzPZy2xMXFwW634+TJkwMG272/K73TJMxmM86dO3fRtkdHR2P79u3o7Ox06WUa7DpwIYlEgvXr12P9+vWw2+2488478dprr+HBBx8cVb3IMVsa5bnnnkNBQQFef/11PP7448jJycEdd9zhkv/giH57R7sHDx7EgQMHxqRN9fX1+Pzzz53/7ujowL///W+kp6cP2B2flZWFuLg4/PWvf4XBYOhzf0tLy0WPee211yI5ORmPP/74gJXQhxPtC4XCPtu/9NJLA9Zxev31112GOl955RVYrVa3FUkUi8WYO3duv8tRPProo9BqtbjttttgtVr73L9161Z88803AIBLL70UjY2N+PDDD533W61WvPTSS1AoFMjLy+v3+EKhEKtXr8aXX37pMszY1NSE999/H4sXLx5w+GMo3J3DdDH9fR8YY6OeDuvl5QUAAwbVqampSE1NxRtvvIFPP/0U1113XZ96MwBw8uRJdHd3Iycnx+X206dPo7q6elRtdLcHH3wQv/3tb/HAAw/glVdecd5uMpnQ3d3tsm1cXByUSmWfqd29hYaGIj09He+8847L61hWVoatW7e67QcIwA+1Xn/99fjoo4/w9ttvY86cOX16Di7m3//+t8tQ+CeffIKGhgbnd3445zRHENjfZyc7OxtisRhPPfUU/Pz8nMPeubm5KCwsxO7du116lwD+fHjgwAFs2bKlz/7a29ud54lrr70WNpvNOSzZm9VqdVvl8UsvvRRWq9XlM2Kz2fDSSy+5Zf+9XXvttairq8M///nPPvd1dXXBaDRe9PFCoRA/+tGP8Omnn6KsrKzP/YNdiwbi7e3dbzpHf6688koIBAI89thjfXq8HeetlStXQiKR4MUXX3Q5l/3rX/+CXq/HZZddNuD+L730UthsNrz88ssut//tb38Dx3FDum5dmAIjEAic3x/Hd9xiseD06dNoaGgYdH+9jaiHafPmzc6Ir7ecnBzExsbi1KlTePDBB3HzzTc7ixm+/fbbSE9Pd44vAnzvy2effYarrroKl112Gc6dO4dXX30VycnJ/X6RRyshIQG33norDh8+jODgYLz55ptoamrCW2+9NeBjBAIB3njjDaxduxYpKSm45ZZbEB4ejrq6OuzatQsqlQpff/31gI8Xi8X4/PPPsWbNGixevBgbNmxw1nGpq6vDV199herq6ot+iHpbt24d3n33Xfj4+CA5ORkHDhzA9u3bnUNjFzKbzVixYgWuvfZanDlzBv/4xz+wePFit/YSXHHFFfjjH/+Ijo4Ol+Dkxz/+MUpLS/HnP/8ZxcXFuP76652Vvr/77jvs2LED77//PgA+Of21117DzTffjKNHj2LGjBn45JNPsH//fjz//PMXTR7/05/+hG3btmHx4sW48847IRKJ8Nprr6GnpwdPP/30qJ6bu3OYLiYpKQlxcXH47W9/i7q6OqhUKnz66acjzjdwkMvlSE5OxocffoiEhAT4+flh9uzZmD17tnObjRs34re//S2AgXPqtm3bBi8vL6xatcrl9lmzZiEvL29IS69YLBb86U9/6nO7n58f7rzzzmE8q8E988wz0Ov1uOuuu6BUKvHTn/4U5eXlzu9DcnIyRCIRPv/8czQ1NeG6664bdH9r167FwoULceutt6KrqwsvvfQSfHx83L624caNG/Hiiy9i165deOqpp4b1WD8/PyxevBi33HILmpqa8PzzzyM+Ph633347gOGd07KysgDwk1Suu+46iMVirF+/Ht7e3vDy8kJWVhYKCwudNZgAvofJaDTCaDT2CZh+97vf4auvvsK6detw8803IysrC0ajEaWlpfjkk09w/vx5BAQEIC8vDz//+c/x5JNPoqSkBKtXr4ZYLEZFRQU+/vhjvPDCC7j66qtH+zJj/fr1WLRoEe6//36cP3/eWZtvqAHEcNx444346KOP8L//+7/YtWsXFi1aBJvNhtOnT+Ojjz7Cli1bBs0N/stf/oJdu3YhOzsbt99+O5KTk6HT6VBUVITt27f3qT84FFlZWfjwww9xzz33YN68eVAoFH2KEDvEx8fjj3/8Ix5//HHk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\n"
          },
          "metadata": {}
        }
      ],
      "source": [
        "# Example Visualization\n",
        "# Retrieve features from the first dataset element\n",
        "x_feat_viz, adj_viz = dataset[0]\n",
        "\n",
        "# Extract positions (x, y) and parity (last column)\n",
        "pos_viz = x_feat_viz[:, :2].numpy()\n",
        "parity_viz = x_feat_viz[:, 2].numpy()\n",
        "\n",
        "# Create NetworkX graph\n",
        "G_viz = nx.from_numpy_array(adj_viz.numpy())\n",
        "\n",
        "# Color based on parity: visually check that there are no links between\n",
        "# two nodes of the same color (Red <-> Red or Blue <-> Blue)\n",
        "colors = ['red' if p == 0 else 'blue' for p in parity_viz]\n",
        "\n",
        "plt.figure(figsize=(5, 5))\n",
        "nx.draw(G_viz,\n",
        "        pos=pos_viz,\n",
        "        node_size=100,\n",
        "        node_color=colors,\n",
        "        edge_color=\"gray\",\n",
        "        alpha=0.7)\n",
        "plt.title(\"Example Graph (Color = Parity). Links only between different colors.\")\n",
        "plt.show()"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 4,
      "metadata": {
        "id": "SmHRY_2xTJIE"
      },
      "outputs": [],
      "source": [
        "# --- Base Modules (FiLM, Graph Transformer Layer) ---\n",
        "\n",
        "class FiLM(nn.Module):\n",
        "    \"\"\"\n",
        "    Feature-wise Linear Modulation (FiLM)\n",
        "    FiLM(x | cond) = x * (1 + s(cond)) + b(cond)\n",
        "    with condition normalization for stability.\n",
        "    \"\"\"\n",
        "    def __init__(self, dim):\n",
        "        super().__init__()\n",
        "        self.norm_cond = nn.LayerNorm(dim)\n",
        "        self.scale = nn.Linear(dim, dim)\n",
        "        self.shift = nn.Linear(dim, dim)\n",
        "\n",
        "    def forward(self, x, condition):\n",
        "        \"\"\"\n",
        "        x: (B, N, C) or (B, N, N, C)\n",
        "        condition: (B, C)\n",
        "        \"\"\"\n",
        "        cond = self.norm_cond(condition)\n",
        "        s = self.scale(cond)\n",
        "        b = self.shift(cond)\n",
        "\n",
        "        if x.dim() == 3:        # (B, N, C)\n",
        "            s = s.unsqueeze(1)  # (B, 1, C)\n",
        "            b = b.unsqueeze(1)\n",
        "        elif x.dim() == 4:      # (B, N, N, C)\n",
        "            s = s.unsqueeze(1).unsqueeze(1)  # (B, 1, 1, C)\n",
        "            b = b.unsqueeze(1).unsqueeze(1)\n",
        "\n",
        "        return x * (1.0 + s) + b\n",
        "\n",
        "\n",
        "class GraphTransformerLayer(nn.Module):\n",
        "    \"\"\"\n",
        "    Implementation of Dwivedi's Graph Transformer (Multiplication by E).\n",
        "\n",
        "    x : (B, N, C)  (node features)\n",
        "    e : (B, N, N, C) (edge features)\n",
        "    \"\"\"\n",
        "    def __init__(self, dim, num_heads):\n",
        "        super().__init__()\n",
        "        self.dim = dim\n",
        "        self.num_heads = num_heads\n",
        "        self.d_k = dim // num_heads\n",
        "\n",
        "        # Attention projections\n",
        "        self.q_proj = nn.Linear(dim, dim)\n",
        "        self.k_proj = nn.Linear(dim, dim)\n",
        "        self.v_proj = nn.Linear(dim, dim)\n",
        "        self.o_proj = nn.Linear(dim, dim)\n",
        "\n",
        "        # Edge feature injection into attention\n",
        "        self.edge_proj_for_attn = nn.Linear(dim, num_heads)\n",
        "\n",
        "        # Normalizations\n",
        "        self.norm_node1 = nn.LayerNorm(dim)\n",
        "        self.norm_node2 = nn.LayerNorm(dim)\n",
        "        self.norm_edge = nn.LayerNorm(dim)\n",
        "\n",
        "        # Node FFN\n",
        "        self.ffn_node = nn.Sequential(\n",
        "            nn.Linear(dim, dim * 4),\n",
        "            nn.GELU(),\n",
        "            nn.Linear(dim * 4, dim)\n",
        "        )\n",
        "\n",
        "        # Edge FFN\n",
        "        self.ffn_edge = nn.Sequential(\n",
        "            nn.Linear(dim * 3, dim * 4),\n",
        "            nn.GELU(),\n",
        "            nn.Linear(dim * 4, dim)\n",
        "        )\n",
        "\n",
        "    def forward(self, x, e):\n",
        "        \"\"\"\n",
        "        x: (B, N, C)\n",
        "        e: (B, N, N, C)\n",
        "        \"\"\"\n",
        "        B, N, C = x.shape\n",
        "\n",
        "        # --- 1. Node Attention ---\n",
        "        x_res, e_res = x, e\n",
        "\n",
        "        x_norm = self.norm_node1(x)\n",
        "        Q = self.q_proj(x_norm).view(B, N, self.num_heads, self.d_k).transpose(1, 2)\n",
        "        K = self.k_proj(x_norm).view(B, N, self.num_heads, self.d_k).transpose(1, 2)\n",
        "        V = self.v_proj(x_norm).view(B, N, self.num_heads, self.d_k).transpose(1, 2)\n",
        "\n",
        "        attn_score_qk = (Q @ K.transpose(-2, -1)) / math.sqrt(self.d_k)  # (B, H, N, N)\n",
        "\n",
        "        edge_factor = self.edge_proj_for_attn(e).permute(0, 3, 1, 2)     # (B, H, N, N)\n",
        "        attn_unnorm = attn_score_qk * edge_factor\n",
        "\n",
        "        attn_prob = F.softmax(attn_unnorm, dim=-1)\n",
        "        out = (attn_prob @ V).transpose(1, 2).reshape(B, N, C)\n",
        "        out = self.o_proj(out)\n",
        "\n",
        "        x = x_res + out\n",
        "\n",
        "        # --- 1.b Node FFN ---\n",
        "        x = x + self.ffn_node(self.norm_node2(x))\n",
        "\n",
        "        # --- 2. Edge Update ---\n",
        "        e_norm = self.norm_edge(e)\n",
        "        x_i = x.unsqueeze(2).expand(-1, -1, N, -1)  # (B, N, N, C)\n",
        "        x_j = x.unsqueeze(1).expand(-1, N, -1, -1)  # (B, N, N, C)\n",
        "        e_in = torch.cat([e_norm, x_i, x_j], dim=-1)\n",
        "        e_out = self.ffn_edge(e_in)\n",
        "        e = e_res + e_out\n",
        "\n",
        "        return x, e"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 5,
      "metadata": {
        "id": "kqTrf9oz2gSX"
      },
      "outputs": [],
      "source": [
        "\n",
        "# --- Utility Functions ---\n",
        "\n",
        "def node_connex(adj):\n",
        "    \"\"\" Calculates the degree of each node. \"\"\"\n",
        "    return torch.sum(adj, dim=-1)\n",
        "\n",
        "def edge_angles(pos):\n",
        "    \"\"\" Calculates the angle of the vector j -> i (B, N, N, 1). \"\"\"\n",
        "    B, N, _ = pos.shape\n",
        "\n",
        "    # Directional vectors (j - i)\n",
        "    vecs = pos.unsqueeze(1) - pos.unsqueeze(2) # (B, N, N, 2)\n",
        "\n",
        "    # Calculate angle atan2(dy, dx)\n",
        "    dx = vecs[:, :, :, 0]\n",
        "    dy = vecs[:, :, :, 1]\n",
        "\n",
        "    angles = torch.atan2(dy, dx) # (-pi to pi)\n",
        "\n",
        "    return angles.unsqueeze(-1) # (B, N, N, 1)\n",
        "\n",
        "def compute_quadrants_batched(adj, pos):\n",
        "    \"\"\"\n",
        "    Estimates the violation of the 'max 1 link per quadrant' constraint on the graph.\n",
        "    The penalty is applied to the sum of links > 1 per quadrant/node.\n",
        "    \"\"\"\n",
        "    B, N, _ = pos.shape\n",
        "    angles = edge_angles(pos).squeeze(-1) # (B, N, N)\n",
        "\n",
        "    # Define the 4 quadrants by angle (0: E, 1: N, 2: W, 3: S)\n",
        "    degrees = torch.rad2deg(angles)\n",
        "    quadrant_index = torch.zeros_like(degrees, dtype=torch.long, device=adj.device)\n",
        "\n",
        "    quadrant_index[(-45 < degrees) & (degrees <= 45)] = 0\n",
        "    quadrant_index[(45 < degrees) & (degrees <= 135)] = 1\n",
        "    quadrant_index[(135 < degrees) & (degrees <= 180) | (-180 <= degrees) & (degrees <= -135)] = 2\n",
        "    quadrant_index[(-135 < degrees) & (degrees <= -45)] = 3\n",
        "\n",
        "    # Initialize quadrant link count matrix [B, N, 4]\n",
        "    quadrant_counts = torch.zeros((B, N, 4), device=adj.device)\n",
        "\n",
        "    # Count active links (adj) per quadrant for each node\n",
        "    # Note: We count the number of neighbors for each node 'i' in each quadrant.\n",
        "\n",
        "    adj_clamp = torch.clamp(adj, min=0) # Ensure adj is positive/probabilistic\n",
        "\n",
        "    for q in range(4):\n",
        "        # Boolean mask for links in quadrant q\n",
        "        q_mask = (quadrant_index == q).float()\n",
        "\n",
        "        # Counting: Sum of links (adj) that fall into this quadrant\n",
        "        # (B, N, N) * (B, N, N) -> Sum over neighbor dimension (dim=2)\n",
        "        quadrant_counts[:, :, q] = torch.sum(adj_clamp * q_mask, dim=2)\n",
        "\n",
        "    # The penalty is F.relu(quads - 1).sum(). We return the count matrix.\n",
        "    # Here, we return the sum of violations [B, N, 4] -> [B, N]\n",
        "    # The original loss uses `F.relu(quads-1).sum()`, we return the full matrix.\n",
        "    return quadrant_counts # (B, N, 4)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 6,
      "metadata": {
        "id": "YVHoQwWO57cN"
      },
      "outputs": [],
      "source": [
        "\n",
        "# --- Diffusion Models (Corrected I/O) ---\n",
        "\n",
        "class BaseGraphNet(nn.Module):\n",
        "    # node_in_dim = (Pos+Parity) + (Noisy Parity) + (Sector)\n",
        "    # edge_in_dim = (Noisy Edge) + (Angle)\n",
        "    def __init__(self, cfg):\n",
        "        super().__init__()\n",
        "        # Total inputs after concatenation in p_loss:\n",
        "        # Node In: Pos (2) + Parity (1) + Sector(4) + Noisy Parity (1) = 8\n",
        "        # Edge In: Edge (1) + Angle (1) = 2\n",
        "\n",
        "        self.hidden_dim = cfg.hidden_dim\n",
        "\n",
        "        self.node_emb = nn.Linear(8, cfg.hidden_dim)\n",
        "        self.edge_emb = nn.Linear(1, cfg.hidden_dim)\n",
        "\n",
        "        self.node_emb_norm = nn.LayerNorm(cfg.hidden_dim)\n",
        "        self.edge_emb_norm = nn.LayerNorm(cfg.hidden_dim)\n",
        "\n",
        "        self.time_emb = nn.Sequential(\n",
        "            nn.Linear(1, cfg.hidden_dim),\n",
        "            nn.SiLU(),\n",
        "            nn.Linear(cfg.hidden_dim, cfg.hidden_dim),\n",
        "        )\n",
        "        self.time_emb_norm = nn.LayerNorm(cfg.hidden_dim)\n",
        "\n",
        "        # Output: Adjacency (N, N) and Parity (N)\n",
        "        self.final_proj_adj = nn.Linear(cfg.hidden_dim, 1)   # For edges\n",
        "        self.final_proj_parity = nn.Linear(cfg.hidden_dim, 1)  # For nodes\n",
        "\n",
        "    def encode_inputs(self, x_node_feat, x_edge_feat, t):\n",
        "        \"\"\"\n",
        "        Common utility for models:\n",
        "        - Encode node/edge features\n",
        "        - Encode time\n",
        "        \"\"\"\n",
        "        h = self.node_emb_norm(self.node_emb(x_node_feat))   # (B, N, D)\n",
        "        e = self.edge_emb_norm(self.edge_emb(x_edge_feat))   # (B, N, N, D)\n",
        "        t_emb = self.time_emb_norm(self.time_emb(t.view(-1, 1).float()))  # (B, D)\n",
        "        return h, e, t_emb\n",
        "\n",
        "    def decode_outputs(self, h, e):\n",
        "        \"\"\"\n",
        "        Projects to adj (B,N,N) and parity (B,N).\n",
        "        \"\"\"\n",
        "        out_adj = self.final_proj_adj(e).squeeze(-1)     # (B, N, N)\n",
        "        out_parity = self.final_proj_parity(h).squeeze(-1)  # (B, N)\n",
        "\n",
        "        # Symmetrization of links + stable sigmoid\n",
        "        adj_sym = (out_adj + out_adj.transpose(1, 2)) / 2.0\n",
        "        return torch.sigmoid(adj_sym), torch.sigmoid(out_parity)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 7,
      "metadata": {
        "id": "rv7NfBPSHabY"
      },
      "outputs": [],
      "source": [
        "class ACAMCountBlockCosine(nn.Module):\n",
        "    def __init__(self, dim, num_heads, eps=1e-8, relu_scores=False):\n",
        "        super().__init__()\n",
        "        assert dim % num_heads == 0\n",
        "        self.dim = dim\n",
        "        self.num_heads = num_heads\n",
        "        self.d_k = dim // num_heads\n",
        "        self.eps = eps\n",
        "        self.relu_scores = relu_scores\n",
        "\n",
        "\n",
        "        self.q_proj = nn.Linear(dim, dim)\n",
        "        self.k_proj = nn.Linear(dim, dim)\n",
        "        self.v_proj = nn.Linear(dim, dim)\n",
        "\n",
        "        self.out_proj = nn.Linear(dim, dim)\n",
        "        self.norm = nn.LayerNorm(dim)\n",
        "\n",
        "    def forward(self, cam, h):\n",
        "        # cam: (B,M,D), h: (B,N,D)\n",
        "        B, M, D = cam.shape\n",
        "        _, N, _ = h.shape\n",
        "\n",
        "        Q = self.q_proj(cam).view(B, M, self.num_heads, self.d_k).transpose(1, 2)  # (B,H,M,d)\n",
        "        K = self.k_proj(h  ).view(B, N, self.num_heads, self.d_k).transpose(1, 2)  # (B,H,N,d)\n",
        "        V = self.v_proj(h  ).view(B, N, self.num_heads, self.d_k).transpose(1, 2)  # (B,H,N,d)\n",
        "\n",
        "        Q = F.normalize(Q, dim=-1, eps=self.eps)\n",
        "        K = F.normalize(K, dim=-1, eps=self.eps)\n",
        "\n",
        "        scores = torch.einsum(\"bhmd,bhnd->bhmn\", Q, K)  # (B,H,M,N)\n",
        "        if self.relu_scores:\n",
        "            scores = F.relu(scores)\n",
        "\n",
        "        ctx = scores @ V  # (B,H,M,d)\n",
        "\n",
        "        # merge heads -> (B,M,D)\n",
        "        ctx = ctx.transpose(1, 2).contiguous().view(B, M, D)\n",
        "\n",
        "        upd = self.out_proj(ctx)  # (B,M,D)\n",
        "\n",
        "        cam = self.norm(upd)\n",
        "        return cam"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 8,
      "metadata": {
        "id": "9DjbvkAgTZYs"
      },
      "outputs": [],
      "source": [
        "class CAMGraphTransformer(BaseGraphNet):\n",
        "    def __init__(self, cfg):\n",
        "        super().__init__(cfg)\n",
        "\n",
        "        self.layers = nn.ModuleList([\n",
        "            GraphTransformerLayer(cfg.hidden_dim, cfg.num_heads)\n",
        "            for _ in range(cfg.num_layers)\n",
        "        ])\n",
        "\n",
        "        self.num_cams = getattr(cfg, \"num_cams\", 4)\n",
        "\n",
        "        # FiLM for time\n",
        "        self.film_nodes_time = nn.ModuleList([FiLM(cfg.hidden_dim) for _ in range(cfg.num_layers)])\n",
        "        self.film_edges_time = nn.ModuleList([FiLM(cfg.hidden_dim) for _ in range(cfg.num_layers)])\n",
        "\n",
        "        # FiLM for CAM\n",
        "        self.film_nodes_cam = nn.ModuleList([FiLM(cfg.hidden_dim) for _ in range(cfg.num_layers)])\n",
        "        self.film_edges_cam = nn.ModuleList([FiLM(cfg.hidden_dim) for _ in range(cfg.num_layers)])\n",
        "\n",
        "        # Multiple CAM tokens (1, M, D)\n",
        "        self.cam_tokens = nn.Parameter(\n",
        "            torch.randn(1, self.num_cams, cfg.hidden_dim)\n",
        "        )\n",
        "\n",
        "        # Pre-MLP on CAMs (one per layer)\n",
        "        self.cam_pre = nn.ModuleList([\n",
        "            nn.Sequential(\n",
        "                nn.Linear(cfg.hidden_dim, cfg.hidden_dim),\n",
        "                nn.SiLU(),\n",
        "                nn.Linear(cfg.hidden_dim, cfg.hidden_dim)\n",
        "            )\n",
        "            for _ in range(cfg.num_layers)\n",
        "        ])\n",
        "\n",
        "        # 🌟 CAM Block without softmax (count-friendly)\n",
        "        self.cam_blocks = nn.ModuleList([\n",
        "            ACAMCountBlockCosine(cfg.hidden_dim, num_heads=4)\n",
        "            for _ in range(cfg.num_layers)\n",
        "        ])\n",
        "\n",
        "        # Combine flattened CAMs: (B, M*D) -> (B, D)\n",
        "        self.cam_combine = nn.Linear(self.num_cams * cfg.hidden_dim, cfg.hidden_dim)\n",
        "\n",
        "    def forward(self, x_node_feat, x_edge_feat, t):\n",
        "        # Common encoding (nodes, edges, time)\n",
        "        h, e, t_emb = self.encode_inputs(x_node_feat, x_edge_feat, t)\n",
        "\n",
        "        B, N, D = h.shape\n",
        "        cam = self.cam_tokens.expand(B, self.num_cams, D)  # (B, M, D)\n",
        "\n",
        "        for i in range(len(self.layers)):\n",
        "\n",
        "             # --- FiLM for time ---\n",
        "            h = self.film_nodes_time[i](h, t_emb)\n",
        "            e = self.film_edges_time[i](e, t_emb)\n",
        "\n",
        "\n",
        "            # --- Graph Transformer layer ---\n",
        "            h, e = self.layers[i](h, e)\n",
        "\n",
        "            # --- Pre-MLP on CAMs ---\n",
        "            #cam_proj = self.cam_pre[i](cam)                 # (B, M, D)\n",
        "\n",
        "            # --- CAM Block without softmax ---\n",
        "            cam = self.cam_blocks[i](cam, h)           # (B, M, D)\n",
        "\n",
        "            # Flattening + projection to get a global condition\n",
        "            cam_flat = cam.reshape(B, -1)                   # (B, M*D)\n",
        "            cam_cond = self.cam_combine(cam_flat)           # (B, D)\n",
        "\n",
        "\n",
        "\n",
        "            # --- FiLM for CAM ---\n",
        "            h = self.film_nodes_cam[i](h, cam_cond)\n",
        "            e = self.film_edges_cam[i](e, cam_cond)\n",
        "\n",
        "        return self.decode_outputs(h, e)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 9,
      "metadata": {
        "id": "UJdsOmgA59kf"
      },
      "outputs": [],
      "source": [
        "# --- Cell 6: Diffusion Framework & Loss (COMPLETE) ---\n",
        "\n",
        "# 1. Schedule Parameters (Cosine / DiGress style)\n",
        "n_steps = cfg.timesteps\n",
        "s = cfg.s_cosine\n",
        "\n",
        "# Calculate alpha_cumprod Schedule\n",
        "# Formula: |cos(0.5 * pi * (t/T + s) / (1 + s))|^2\n",
        "alphas_cumprod = torch.zeros(n_steps).to(cfg.device)\n",
        "for t in range(n_steps) :\n",
        "    arg = (0.5 * torch.tensor(math.pi)) * ((t / n_steps + s) / (1 + s))\n",
        "    alphas_cumprod[t] = torch.abs(torch.cos(arg))**2\n",
        "\n",
        "# Derived variables for q_sample\n",
        "sqrt_alphas_cumprod = alphas_cumprod\n",
        "one_minus_alphas_cumprod = (1 - alphas_cumprod)\n",
        "sqrt_one_minus_alphas_cumprod = one_minus_alphas_cumprod\n",
        "\n",
        "\n",
        "# 2. Noising Functions (Forward Diffusion - Bernoulli Relaxation)\n",
        "\n",
        "def q_sample(x_0, t):\n",
        "    \"\"\"\n",
        "    Forward diffusion for adjacency (B, N, N).\n",
        "    Mixes x_0 and symmetric noise, then samples via Bernoulli.\n",
        "    \"\"\"\n",
        "    B, N, _ = x_0.shape\n",
        "\n",
        "    # Base noise (0.3 * U[0,1])\n",
        "    noise_base = (0.3 * torch.rand(B, N, N)).to(cfg.device)\n",
        "\n",
        "    # Symmetrize noise (Upper triangle copied)\n",
        "    mask = torch.triu(torch.ones(B, N, N), diagonal=1).to(cfg.device)\n",
        "    upper_tri_noise = noise_base * mask\n",
        "    symmetric_noise = upper_tri_noise + upper_tri_noise.permute(0,2,1)\n",
        "\n",
        "    # Retrieve alpha coefficients for the batch\n",
        "    alpha_t = alphas_cumprod.gather(-1, t).reshape(-1, 1, 1).repeat(1, N, N)\n",
        "    one_minus_alpha_t = one_minus_alphas_cumprod.gather(-1, t).reshape(-1, 1, 1).repeat(1, N, N)\n",
        "\n",
        "    # Mixing probability\n",
        "    p = alpha_t * x_0 + one_minus_alpha_t * symmetric_noise\n",
        "\n",
        "    # Discrete sampling (Bernoulli)\n",
        "    return torch.bernoulli(p), symmetric_noise\n",
        "\n",
        "def q_sample_parity(x_0, t):\n",
        "    \"\"\"\n",
        "    Forward diffusion for parity (B, N).\n",
        "    Mixes x_0 and uniform noise, then samples via Bernoulli.\n",
        "    \"\"\"\n",
        "    B, N = x_0.shape\n",
        "    parity_noise = torch.rand(B, N).to(cfg.device) # U[0,1] noise\n",
        "\n",
        "    alpha_t = alphas_cumprod.gather(-1, t).reshape(-1, 1).repeat(1, N)\n",
        "    one_minus_alpha_t = one_minus_alphas_cumprod.gather(-1, t).reshape(-1, 1).repeat(1, N)\n",
        "\n",
        "    p = alpha_t * x_0 + one_minus_alpha_t * parity_noise\n",
        "\n",
        "    return torch.bernoulli(p), parity_noise\n",
        "\n",
        "\n",
        "# 3. Loss Function (p_loss)\n",
        "\n",
        "def p_loss(model, x_feat, adj, t):\n",
        "    \"\"\"\n",
        "    Calculates reconstruction loss (BCE) and structural penalties.\n",
        "    Handles input normalization to prevent NaNs.\n",
        "    \"\"\"\n",
        "    B, N, _ = x_feat.shape\n",
        "\n",
        "    # Extract Ground Truth Features\n",
        "    x_pos = x_feat[:, :, :2]\n",
        "    x_parity_0 = x_feat[:, :, 2]\n",
        "\n",
        "    # --- A. Forward Process (Noising) ---\n",
        "    noisy_e, symmetric_noise = q_sample(adj, t)\n",
        "    noisy_parity, parity_noise = q_sample_parity(x_parity_0, t)\n",
        "\n",
        "    # --- B. Model Input Preparation ---\n",
        "\n",
        "    # 1. Edge Features: Only noisy adjacency (B, N, N, 1)\n",
        "    x_edge_feat_concatenated = noisy_e.unsqueeze(-1)\n",
        "\n",
        "    # 2. Node Features: [Pos, Parity, Sectors, Noisy Parity]\n",
        "    # Calculate sector counts on the noisy graph\n",
        "    sector_counts = compute_quadrants_batched(noisy_e, x_pos).to(cfg.device)\n",
        "\n",
        "    # !! IMPORTANT NORMALIZATION !! (Divide by N to bring between 0 and 1)\n",
        "    sector_counts_normalized = sector_counts / float(N)\n",
        "\n",
        "    x_node_feat_concatenated = torch.cat([\n",
        "        x_pos,                      # (B, N, 2)\n",
        "        x_parity_0.unsqueeze(-1),   # (B, N, 1)\n",
        "        sector_counts_normalized,   # (B, N, 4) -> Normalized\n",
        "        noisy_parity.unsqueeze(-1)  # (B, N, 1)\n",
        "    ], dim=2) # Total dim = 8\n",
        "\n",
        "    # --- C. Model Prediction ---\n",
        "    # The model returns LOGITS\n",
        "    noise_computed_prob, parity_prob = model(x_node_feat_concatenated, x_edge_feat_concatenated, t)\n",
        "\n",
        "    # Sigmoid + CLAMPING (Vital for BCE stability)\n",
        "    #noise_computed_prob = torch.sigmoid(noise_computed_logits)\n",
        "    #noise_computed_prob = torch.clamp(noise_computed_prob, 1e-6, 1.0 - 1e-6)\n",
        "\n",
        "    # --- D. Penalty Calculation ---\n",
        "\n",
        "    # Quadrant Penalty (on predicted probabilities)\n",
        "    # We calculate how many 'probable' links fall into each quadrant\n",
        "    quads = compute_quadrants_batched(noise_computed_prob, x_pos).to(cfg.device)\n",
        "    # Penalty if sum > 1 in a quadrant\n",
        "    pen = F.relu(quads - 1).sum()\n",
        "\n",
        "    # --- E. Total Loss Calculation ---\n",
        "\n",
        "    # 1. BCE Loss (Adjacency Reconstruction)\n",
        "    # Normalized by total number of elements (N*N*B)\n",
        "    bce_loss = F.binary_cross_entropy(noise_computed_prob, adj, reduction='sum')\n",
        "    bce_loss = bce_loss / (N * N * B)\n",
        "\n",
        "    # 2. Normalized Penalty Term\n",
        "    # Divide by B so the penalty doesn't grow with batch size\n",
        "    #normalized_pen_term = 0.5 * pen / (1 + torch.sum(noise_computed_prob)) / B\n",
        "\n",
        "    total_loss = bce_loss # + normalized_pen_term\n",
        "\n",
        "    return total_loss"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": 10,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 755
        },
        "id": "Nt6UpWuM6J_I",
        "outputId": "81cc0dc4-42a4-45ed-cd71-712cd424eea5"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "\n",
            "--- Training model: CAM Model ---\n",
            "Epoch 1/30 | Loss: 0.2253\n",
            "  -> New best model saved to: ./checkpoints/cam_model_abscosine_best.pt\n",
            "Epoch 2/30 | Loss: 0.1765\n",
            "  -> New best model saved to: ./checkpoints/cam_model_abscosine_best.pt\n",
            "Epoch 3/30 | Loss: 0.1492\n",
            "  -> New best model saved to: ./checkpoints/cam_model_abscosine_best.pt\n",
            "Epoch 4/30 | Loss: 0.1254\n",
            "  -> New best model saved to: ./checkpoints/cam_model_abscosine_best.pt\n",
            "Epoch 5/30 | Loss: 0.1125\n",
            "  -> New best model saved to: ./checkpoints/cam_model_abscosine_best.pt\n",
            "Epoch 6/30 | Loss: 0.1042\n",
            "  -> New best model saved to: ./checkpoints/cam_model_abscosine_best.pt\n",
            "Epoch 7/30 | Loss: 0.0933\n",
            "  -> New best model saved to: ./checkpoints/cam_model_abscosine_best.pt\n",
            "Epoch 8/30 | Loss: 0.0866\n",
            "  -> New best model saved to: ./checkpoints/cam_model_abscosine_best.pt\n",
            "Epoch 9/30 | Loss: 0.0892\n",
            "Epoch 10/30 | Loss: 0.0883\n",
            "Epoch 11/30 | Loss: 0.0849\n",
            "  -> New best model saved to: ./checkpoints/cam_model_abscosine_best.pt\n",
            "Epoch 12/30 | Loss: 0.0852\n"
          ]
        },
        {
          "output_type": "error",
          "ename": "KeyboardInterrupt",
          "evalue": "",
          "traceback": [
            "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
            "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
            "\u001b[0;32m/tmp/ipython-input-4288408390.py\u001b[0m in \u001b[0;36m<cell line: 0>\u001b[0;34m()\u001b[0m\n\u001b[1;32m     57\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     58\u001b[0m \u001b[0;31m# Launch training\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 59\u001b[0;31m \u001b[0mcam_model\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcam_losses\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtrain_model\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mCAMGraphTransformer\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m\"CAM Model\"\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     60\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     61\u001b[0m \u001b[0;31m# Display loss curves\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/tmp/ipython-input-4288408390.py\u001b[0m in \u001b[0;36mtrain_model\u001b[0;34m(model_cls, name)\u001b[0m\n\u001b[1;32m     27\u001b[0m             \u001b[0mt\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mtorch\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mrandint\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcfg\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtimesteps\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m(\u001b[0m\u001b[0mx_feat\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0msize\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mdevice\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mcfg\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdevice\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     28\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 29\u001b[0;31m             \u001b[0mloss\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mp_loss\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mmodel\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mx_feat\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0madj\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mt\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     30\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     31\u001b[0m             \u001b[0mopt\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mzero_grad\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/tmp/ipython-input-1893179650.py\u001b[0m in \u001b[0;36mp_loss\u001b[0;34m(model, x_feat, adj, t)\u001b[0m\n\u001b[1;32m     85\u001b[0m     \u001b[0;31m# 2. Node Features: [Pos, Parity, Sectors, Noisy Parity]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     86\u001b[0m     \u001b[0;31m# Calculate sector counts on the noisy graph\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 87\u001b[0;31m     \u001b[0msector_counts\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mcompute_quadrants_batched\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mnoisy_e\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mx_pos\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mto\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mcfg\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdevice\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     88\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     89\u001b[0m     \u001b[0;31m# !! IMPORTANT NORMALIZATION !! (Divide by N to bring between 0 and 1)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;32m/tmp/ipython-input-3820301890.py\u001b[0m in \u001b[0;36mcompute_quadrants_batched\u001b[0;34m(adj, pos)\u001b[0m\n\u001b[1;32m     47\u001b[0m     \u001b[0;32mfor\u001b[0m \u001b[0mq\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mrange\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;36m4\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     48\u001b[0m         \u001b[0;31m# Boolean mask for links in quadrant q\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 49\u001b[0;31m         \u001b[0mq_mask\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m(\u001b[0m\u001b[0mquadrant_index\u001b[0m \u001b[0;34m==\u001b[0m \u001b[0mq\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mfloat\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     50\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     51\u001b[0m         \u001b[0;31m# Counting: Sum of links (adj) that fall into this quadrant\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
            "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
          ]
        }
      ],
      "source": [
        "# --- Cell 7: Training ---\n",
        "import os\n",
        "\n",
        "# Optional: folder to save models\n",
        "save_dir = getattr(cfg, \"save_dir\", \"./checkpoints\")\n",
        "os.makedirs(save_dir, exist_ok=True)\n",
        "\n",
        "def train_model(model_cls, name):\n",
        "    print(f\"\\n--- Training model: {name} ---\")\n",
        "    model = model_cls(cfg).to(cfg.device)\n",
        "    opt = torch.optim.Adam(model.parameters(), lr=cfg.lr)\n",
        "\n",
        "    losses = []\n",
        "    model.train()\n",
        "\n",
        "    best_loss = float(\"inf\")\n",
        "    best_path = os.path.join(\n",
        "        save_dir,\n",
        "        f\"{name.lower().replace(' ', '_')}_abscosine_best.pt\"\n",
        "    )\n",
        "\n",
        "    start_time = time.time()\n",
        "    for epoch in range(5000):\n",
        "        ep_loss = 0.0\n",
        "        for x_feat, adj in dataloader:\n",
        "            x_feat, adj = x_feat.to(cfg.device), adj.to(cfg.device)\n",
        "            t = torch.randint(0, cfg.timesteps, (x_feat.size(0),), device=cfg.device)\n",
        "\n",
        "            loss = p_loss(model, x_feat, adj, t)\n",
        "\n",
        "            opt.zero_grad()\n",
        "            loss.backward()\n",
        "            opt.step()\n",
        "            ep_loss += loss.item()\n",
        "\n",
        "        avg_loss = ep_loss / len(dataloader)\n",
        "        losses.append(avg_loss)\n",
        "\n",
        "        # Display\n",
        "        print(f\"Epoch {epoch+1}/{cfg.epochs} | Loss: {avg_loss:.4f}\")\n",
        "\n",
        "        # 🔥 Save best model\n",
        "        if avg_loss < best_loss:\n",
        "            best_loss = avg_loss\n",
        "            torch.save({\n",
        "                \"model_state_dict\": model.state_dict(),\n",
        "                \"optimizer_state_dict\": opt.state_dict(),\n",
        "                \"epoch\": epoch + 1,\n",
        "                \"loss\": avg_loss,\n",
        "                \"cfg\": cfg.__dict__,  # useful if you want to reload the config\n",
        "            }, best_path)\n",
        "            print(f\"  -> New best model saved to: {best_path}\")\n",
        "\n",
        "    print(f\"Total duration: {(time.time() - start_time):.2f}s\")\n",
        "    print(f\"Best loss: {best_loss:.4f} (checkpoint: {best_path})\")\n",
        "    return model, losses\n",
        "\n",
        "# Launch training\n",
        "cam_model, cam_losses = train_model(CAMGraphTransformer, \"CAM Model\")\n",
        "\n",
        "# Display loss curves\n",
        "plt.figure(figsize=(8, 4))\n",
        "plt.plot(cam_losses, label='CAM Model')\n",
        "plt.title(\"Training Loss (Faithful to original formula)\")\n",
        "plt.xlabel(\"Epoch\")\n",
        "plt.ylabel(\"Total Loss\")\n",
        "plt.legend()\n",
        "plt.show()"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "yx0nf_nxUaDw"
      },
      "outputs": [],
      "source": [
        "import os, time\n",
        "import torch\n",
        "\n",
        "save_dir = getattr(cfg, \"save_dir\", \"./checkpoints\")\n",
        "os.makedirs(save_dir, exist_ok=True)\n",
        "\n",
        "def resume_train(model_cls, name, ckpt_path, extra_epochs=1000):\n",
        "    print(f\"\\n--- Resuming training: {name} ---\")\n",
        "    model = model_cls(cfg).to(cfg.device)\n",
        "    opt = torch.optim.Adam(model.parameters(), lr=cfg.lr)\n",
        "\n",
        "    ckpt = torch.load(ckpt_path, map_location=cfg.device)\n",
        "    model.load_state_dict(ckpt[\"model_state_dict\"])\n",
        "    opt.load_state_dict(ckpt[\"optimizer_state_dict\"])\n",
        "\n",
        "    for g in opt.param_groups:\n",
        "      g[\"lr\"] = cfg.lr\n",
        "\n",
        "    start_epoch = ckpt.get(\"epoch\", 0)\n",
        "    best_loss = ckpt.get(\"loss\", float(\"inf\"))\n",
        "\n",
        "    # (optional) if you want to continue saving the \"best\"\n",
        "    best_path = os.path.join(save_dir, f\"{name.lower().replace(' ', '_')}_abscosine_best.pt\")\n",
        "\n",
        "    model.train()\n",
        "    losses = []\n",
        "\n",
        "    start_time = time.time()\n",
        "    for epoch in range(start_epoch, start_epoch + extra_epochs):\n",
        "        ep_loss = 0.0\n",
        "        for x_feat, adj in dataloader:\n",
        "            x_feat, adj = x_feat.to(cfg.device), adj.to(cfg.device)\n",
        "            t = torch.randint(0, cfg.timesteps, (x_feat.size(0),), device=cfg.device)\n",
        "\n",
        "            loss = p_loss(model, x_feat, adj, t)\n",
        "\n",
        "            opt.zero_grad()\n",
        "            loss.backward()\n",
        "            opt.step()\n",
        "            ep_loss += loss.item()\n",
        "\n",
        "        avg_loss = ep_loss / len(dataloader)\n",
        "        losses.append(avg_loss)\n",
        "        print(f\"Epoch {epoch+1} | Loss: {avg_loss:.4f}\")\n",
        "\n",
        "        if avg_loss < best_loss:\n",
        "            best_loss = avg_loss\n",
        "            torch.save({\n",
        "                \"model_state_dict\": model.state_dict(),\n",
        "                \"optimizer_state_dict\": opt.state_dict(),\n",
        "                \"epoch\": epoch + 1,\n",
        "                \"loss\": avg_loss,\n",
        "                \"cfg\": cfg.__dict__,\n",
        "            }, best_path)\n",
        "            print(f\"  -> New best model saved to: {best_path}\")\n",
        "\n",
        "    print(f\"Resume duration: {(time.time() - start_time):.2f}s\")\n",
        "    print(f\"Current best loss: {best_loss:.4f}\")\n",
        "    return model, losses, best_loss"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "0sZA9zdJeFm1"
      },
      "outputs": [],
      "source": [
        "ckpt_path = os.path.join(save_dir, \"cam_model_abscosine_best.pt\")\n",
        "cam_model, more_losses, best_loss = resume_train(\n",
        "    CAMGraphTransformer, \"CAM Model\", ckpt_path, extra_epochs=3900\n",
        ")"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "xnSEoTWnul0s"
      },
      "outputs": [],
      "source": []
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "zOwm6wTK6ki4"
      },
      "outputs": [],
      "source": [
        "# --- Cell 8: Generation Function (Reverse Diffusion) ---\n",
        "\n",
        "@torch.no_grad()\n",
        "def sample(model, num_samples=1):\n",
        "    model.eval()\n",
        "\n",
        "    # 1. Initialization of fixed conditions (Positions and target Parity)\n",
        "    # We generate random positions and random parities to guide generation\n",
        "    pos = torch.rand(num_samples, cfg.num_nodes, 2).to(cfg.device)\n",
        "    parity_target = torch.randint(0, 2, size=(num_samples, cfg.num_nodes)).float().to(cfg.device)\n",
        "\n",
        "    # 2. Initialization of noisy state (x_T)\n",
        "    # Adjacency: Symmetric random noise [0, 1]\n",
        "    adj = torch.rand(num_samples, cfg.num_nodes, cfg.num_nodes).to(cfg.device)\n",
        "    adj = (adj + adj.transpose(1, 2)) / 2\n",
        "\n",
        "    # Initial noisy parity\n",
        "    parity_noisy = torch.rand(num_samples, cfg.num_nodes).to(cfg.device)\n",
        "\n",
        "    print(f\"Starting generation for {num_samples} graph(s)...\")\n",
        "\n",
        "    # 3. Denoising Loop (Reverse Process)\n",
        "    for i in reversed(range(cfg.timesteps)):\n",
        "        t_idx = i\n",
        "        t = torch.full((num_samples,), t_idx, dtype=torch.long, device=cfg.device)\n",
        "\n",
        "        # --- A. Input Construction (Identical to p_loss) ---\n",
        "\n",
        "        # Edge Features: (B, N, N, 1)\n",
        "        # Use current adjacency as input \"noise\"\n",
        "        x_edge_feat = adj.unsqueeze(-1)\n",
        "\n",
        "        # Node Features:\n",
        "        # Recalculate sectors based on current adjacency (adj)\n",
        "        sector_counts = compute_quadrants_batched(adj, pos).to(cfg.device)\n",
        "        sector_counts_normalized = sector_counts / float(cfg.num_nodes) # Normalization!\n",
        "\n",
        "        x_node_feat = torch.cat([\n",
        "            pos,                            # (B, N, 2)\n",
        "            parity_target.unsqueeze(-1),    # (B, N, 1)\n",
        "            sector_counts_normalized,       # (B, N, 4)\n",
        "            parity_noisy.unsqueeze(-1)      # (B, N, 1)\n",
        "        ], dim=2)\n",
        "\n",
        "        # --- B. Model Prediction (x0_pred) ---\n",
        "        # The model predicts the \"clean\" state (x0) from the noisy state (xt)\n",
        "        adj_logits, parity_logits = model(x_node_feat, x_edge_feat, t)\n",
        "\n",
        "        adj_pred = adj_logits\n",
        "        # Enforce prediction symmetry\n",
        "        adj_pred = (adj_pred + adj_pred.transpose(1, 2)) / 2\n",
        "\n",
        "        # --- C. Sampling Step (Reverse Transition) ---\n",
        "        # Use interpolation based on alphas to go from t to t-1\n",
        "\n",
        "        alpha_t = alphas_cumprod[t_idx]\n",
        "        one_minus_alpha_t = one_minus_alphas_cumprod[t_idx]\n",
        "\n",
        "        # Heuristic for relaxed discrete diffusion:\n",
        "        # Mix current state (adj) and prediction (adj_pred)\n",
        "        # Closer to the end (t=0), more trust in adj_pred.\n",
        "\n",
        "        # If t > 0, keep some noise. If t=0, take the pure prediction.\n",
        "        if t_idx > 0:\n",
        "            # Empirical weights to stabilize the reverse process\n",
        "            # (Inspired by simplified D3PM / VDM reverse steps)\n",
        "            noise_factor = one_minus_alpha_t\n",
        "            signal_factor = alpha_t\n",
        "\n",
        "            # Smooth update: xt-1 = (1-w)*xt + w*x0_pred\n",
        "            # w is often small at the beginning and large at the end.\n",
        "            # Here, we simplify by linear interpolation towards the predicted target\n",
        "            update_rate = 1.0 / (t_idx + 1) # Larger towards the end\n",
        "\n",
        "            # New edge probability\n",
        "            adj_next_prob = (1 - update_rate) * adj + update_rate * adj_pred\n",
        "\n",
        "            # Bernoulli sampling to keep discrete nature\n",
        "            adj = torch.bernoulli(torch.clamp(adj_next_prob, 0, 1))\n",
        "\n",
        "            # Update noisy parity (optional if not used as final output)\n",
        "            parity_noisy = (1 - update_rate) * parity_noisy + update_rate * torch.sigmoid(parity_logits)\n",
        "\n",
        "        else:\n",
        "            # Last step: Take the final thresholded prediction\n",
        "            adj = (adj_pred > 0.5).float()\n",
        "\n",
        "        # Resymmetrize after sampling\n",
        "        adj = (adj + adj.transpose(1, 2)) / 2\n",
        "\n",
        "    return adj.cpu().numpy(), pos.cpu().numpy(), parity_target.cpu().numpy()\n",
        "\n",
        "# --- Visualization ---\n",
        "\n",
        "# Generate 2 graphs with the CAM model\n",
        "adjs, pos, parities = sample(cam_model, num_samples=2)\n",
        "\n",
        "plt.figure(figsize=(12, 6))\n",
        "\n",
        "for i in range(2):\n",
        "    ax = plt.subplot(1, 2, i+1)\n",
        "\n",
        "    # Create NetworkX graph\n",
        "    # Convert to int and ensure it's binary\n",
        "    adj_mat = (adjs[i] > 0.5).astype(int)\n",
        "\n",
        "    # Remove diagonal for clean display\n",
        "    np.fill_diagonal(adj_mat, 0)\n",
        "\n",
        "    G = nx.from_numpy_array(adj_mat)\n",
        "\n",
        "    # Colors based on target parity\n",
        "    node_colors = ['blue' if p == 1 else 'red' for p in parities[i]]\n",
        "\n",
        "    # Draw\n",
        "    # Use generated positions (pos[i]) which are in [0,1]\n",
        "    nx.draw(G, pos=pos[i], ax=ax, node_size=100, node_color=node_colors, edge_color='gray', with_labels=False)\n",
        "\n",
        "    # Calculate basic metrics\n",
        "    num_edges = G.number_of_edges()\n",
        "    avg_deg = num_edges * 2 / cfg.num_nodes\n",
        "    ax.set_title(f\"Generated Graph {i+1}\\nEdges: {num_edges} | Avg Deg: {avg_deg:.2f}\")\n",
        "\n",
        "plt.tight_layout()\n",
        "plt.show()"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "1bN1sXa2Y06e"
      },
      "outputs": [],
      "source": [
        "# --- Cell 8: Generation + Evaluation on 100 graphs ---\n",
        "\n",
        "@torch.no_grad()\n",
        "def sample(model, num_samples=1):\n",
        "    model.eval()\n",
        "\n",
        "    # 1) Fixed conditions\n",
        "    pos = torch.rand(num_samples, cfg.num_nodes, 2).to(cfg.device)\n",
        "    parity_target = torch.randint(0, 2, size=(num_samples, cfg.num_nodes)).float().to(cfg.device)\n",
        "\n",
        "    # 2) Initial noisy state x_T\n",
        "    adj = torch.rand(num_samples, cfg.num_nodes, cfg.num_nodes).to(cfg.device)\n",
        "    adj = (adj + adj.transpose(1, 2)) / 2\n",
        "\n",
        "    parity_noisy = torch.rand(num_samples, cfg.num_nodes).to(cfg.device)\n",
        "\n",
        "    # 3) Reverse process\n",
        "    for i in reversed(range(cfg.timesteps)):\n",
        "        t_idx = i\n",
        "        t = torch.full((num_samples,), t_idx, dtype=torch.long, device=cfg.device)\n",
        "\n",
        "        # Inputs (same as p_loss)\n",
        "        x_edge_feat = adj.unsqueeze(-1)\n",
        "\n",
        "        sector_counts = compute_quadrants_batched(adj, pos).to(cfg.device)\n",
        "        sector_counts_normalized = sector_counts / float(cfg.num_nodes)\n",
        "\n",
        "        x_node_feat = torch.cat([\n",
        "            pos,                          # (B,N,2)\n",
        "            parity_target.unsqueeze(-1),  # (B,N,1)\n",
        "            sector_counts_normalized,     # (B,N,4)\n",
        "            parity_noisy.unsqueeze(-1)    # (B,N,1)\n",
        "        ], dim=2)\n",
        "\n",
        "        # Model prediction\n",
        "        adj_out, parity_logits = model(x_node_feat, x_edge_feat, t)\n",
        "\n",
        "        # IMPORTANT: here we assume the model already returns probabilities (sigmoid done in the model)\n",
        "        adj_pred = adj_out\n",
        "        adj_pred = (adj_pred + adj_pred.transpose(1, 2)) / 2\n",
        "\n",
        "        # Reverse step (heuristic)\n",
        "        if t_idx > 0:\n",
        "            update_rate = 1.0 / (t_idx + 1)\n",
        "\n",
        "            adj_next_prob = (1 - update_rate) * adj + update_rate * adj_pred\n",
        "            adj = torch.bernoulli(torch.clamp(adj_next_prob, 0, 1))\n",
        "\n",
        "            # Noisy parity (if parity_logits are logits)\n",
        "            parity_noisy = (1 - update_rate) * parity_noisy + update_rate * torch.sigmoid(parity_logits)\n",
        "        else:\n",
        "            adj = (adj_pred > 0.5).float()\n",
        "\n",
        "        # Resymmetrize + diag=0 (otherwise biased degrees)\n",
        "        adj = (adj + adj.transpose(1, 2)) / 2\n",
        "        adj = adj * (1 - torch.eye(cfg.num_nodes, device=cfg.device).unsqueeze(0))\n",
        "\n",
        "    return adj.cpu().numpy(), pos.cpu().numpy(), parity_target.cpu().numpy()\n",
        "\n",
        "\n",
        "# --- Evaluation on 100 graphs ---\n",
        "num_gen = 100\n",
        "adjs, pos, parities = sample(cam_model, num_samples=num_gen)\n",
        "\n",
        "viol_nodes_counts = []  # nb of nodes with deg > 4\n",
        "num_edges_list = []     # total nb of links\n",
        "\n",
        "for i in range(num_gen):\n",
        "    adj_mat = (adjs[i] > 0.5).astype(int)\n",
        "    np.fill_diagonal(adj_mat, 0)\n",
        "\n",
        "    # Degrees (symmetric, undirected)\n",
        "    deg = adj_mat.sum(axis=1)\n",
        "    viol_nodes = int((deg > 4).sum())\n",
        "    viol_nodes_counts.append(viol_nodes)\n",
        "\n",
        "    # Total number of edges (divide by 2 because symmetric matrix)\n",
        "    num_edges = int(adj_mat.sum() // 2)\n",
        "    num_edges_list.append(num_edges)\n",
        "\n",
        "print(f\"Evaluation on {num_gen} graphs\")\n",
        "print(f\"- Average number of nodes with deg>4: {np.mean(viol_nodes_counts):.2f}  (min={np.min(viol_nodes_counts)}, max={np.max(viol_nodes_counts)})\")\n",
        "print(f\"- Average number of links           : {np.mean(num_edges_list):.2f}  (min={np.min(num_edges_list)}, max={np.max(num_edges_list)})\")\n",
        "\n",
        "# (optional) quick histograms\n",
        "plt.figure(figsize=(10,4))\n",
        "plt.subplot(1,2,1)\n",
        "plt.hist(viol_nodes_counts, bins=range(0, cfg.num_nodes+2))\n",
        "plt.title(\"Nb of nodes with deg>4\")\n",
        "plt.xlabel(\"count\"); plt.ylabel(\"freq\")\n",
        "\n",
        "plt.subplot(1,2,2)\n",
        "plt.hist(num_edges_list, bins=20)\n",
        "plt.title(\"Total number of links\")\n",
        "plt.xlabel(\"edges\"); plt.ylabel(\"freq\")\n",
        "plt.tight_layout()\n",
        "plt.show()"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "K9diKqYvxm60"
      },
      "outputs": [],
      "source": [
        "import matplotlib.pyplot as plt\n",
        "import numpy as np\n",
        "\n",
        "def compute_metrics(adjs, parities=None):\n",
        "    \"\"\"\n",
        "    Calculates vectorized metrics on a batch of adjacency matrices.\n",
        "    Returns dictionaries of statistics.\n",
        "    \"\"\"\n",
        "    # Clean binarization\n",
        "    adjs_bin = (adjs > 0.5).astype(int)\n",
        "\n",
        "    # Ensure diagonal is 0\n",
        "    B, N, _ = adjs_bin.shape\n",
        "    for i in range(B):\n",
        "        np.fill_diagonal(adjs_bin[i], 0)\n",
        "\n",
        "    # 1. Number of edges\n",
        "    # Sum of matrix / 2 (because symmetric)\n",
        "    num_edges = np.sum(adjs_bin, axis=(1, 2)) / 2\n",
        "\n",
        "    # 2. Node degrees\n",
        "    degrees = np.sum(adjs_bin, axis=2) # (B, N)\n",
        "\n",
        "    # 3. Degree > 4 constraint violation (Proxy for quadrants)\n",
        "    # Count how many nodes per graph have a degree > 4\n",
        "    nodes_violating_deg4 = np.sum(degrees > 4, axis=1)\n",
        "\n",
        "    # 4. Parity adherence (If parities is provided)\n",
        "    parity_respect_ratio = []\n",
        "    if parities is not None:\n",
        "        for i in range(B):\n",
        "            # Indices where there is a link\n",
        "            u, v = np.where(np.triu(adjs_bin[i], k=1) == 1)\n",
        "            if len(u) > 0:\n",
        "                p_u = parities[i, u].flatten()\n",
        "                p_v = parities[i, v].flatten()\n",
        "                # A link is \"good\" if p_u != p_v\n",
        "                good_links = np.sum(p_u != p_v)\n",
        "                ratio = good_links / len(u)\n",
        "                parity_respect_ratio.append(ratio)\n",
        "            else:\n",
        "                parity_respect_ratio.append(1.0) # Empty graph = trivial adherence\n",
        "        parity_respect_ratio = np.array(parity_respect_ratio)\n",
        "\n",
        "    return {\n",
        "        \"num_edges\": num_edges,\n",
        "        \"degrees\": degrees.flatten(), # Flatten for global histogram\n",
        "        \"viol_deg4\": nodes_violating_deg4,\n",
        "        \"parity_ratio\": parity_respect_ratio\n",
        "    }\n",
        "\n",
        "# --- 1. Prepare Real Data (Ground Truth) ---\n",
        "print(\">>> Generating reference dataset (1000 graphs)...\")\n",
        "# Use your SpatialNetworkDataset class\n",
        "real_dataset = SpatialNetworkDataset(num_samples=1000, num_nodes=cfg.num_nodes)\n",
        "\n",
        "# Extract numpy tensors\n",
        "real_adjs = np.array([item[1].numpy() for item in real_dataset.data])\n",
        "real_parities = np.array([item[0][:, 2].numpy() for item in real_dataset.data]) # Feature 2 is parity\n",
        "\n",
        "# --- 2. Generate Synthetic Data (Model) ---\n",
        "print(\">>> Generating model graphs (1000 graphs)...\")\n",
        "gen_adjs_list = []\n",
        "gen_parities_list = []\n",
        "\n",
        "# Batching to avoid OOM on GPU\n",
        "batch_size = 100\n",
        "num_batches = 1000 // batch_size\n",
        "\n",
        "for b in range(num_batches):\n",
        "    print(f\"   Batch {b+1}/{num_batches}...\")\n",
        "    # sample returns (adj, pos, parity_target)\n",
        "    # parity_target is what guided generation, use it to check consistency\n",
        "    adj, _, parity = sample(cam_model, num_samples=batch_size)\n",
        "    gen_adjs_list.append(adj)\n",
        "    gen_parities_list.append(parity)\n",
        "\n",
        "gen_adjs = np.concatenate(gen_adjs_list, axis=0)\n",
        "gen_parities = np.concatenate(gen_parities_list, axis=0)\n",
        "\n",
        "# --- 3. Calculate Metrics ---\n",
        "print(\">>> Calculating metrics...\")\n",
        "metrics_real = compute_metrics(real_adjs, real_parities)\n",
        "metrics_gen = compute_metrics(gen_adjs, gen_parities)\n",
        "\n",
        "# --- 4. Display and Visualize ---\n",
        "\n",
        "def plot_comparison(metric_real, metric_gen, title, xlabel, bins=20):\n",
        "    plt.hist(metric_real, bins=bins, alpha=0.5, label='Real Dataset', density=True, color='blue', edgecolor='black')\n",
        "    plt.hist(metric_gen, bins=bins, alpha=0.5, label='Generated (Model)', density=True, color='red', edgecolor='black')\n",
        "    plt.title(title)\n",
        "    plt.xlabel(xlabel)\n",
        "    plt.legend()\n",
        "\n",
        "plt.figure(figsize=(15, 10))\n",
        "\n",
        "# Plot 1: Edge count distribution\n",
        "plt.subplot(2, 2, 1)\n",
        "plot_comparison(metrics_real['num_edges'], metrics_gen['num_edges'],\n",
        "                \"Distribution of number of edges\", \"Number of edges\")\n",
        "\n",
        "# Plot 2: Degree distribution (all nodes combined)\n",
        "plt.subplot(2, 2, 2)\n",
        "plot_comparison(metrics_real['degrees'], metrics_gen['degrees'],\n",
        "                \"Distribution of Node Degrees\", \"Degree\", bins=range(0, 10))\n",
        "\n",
        "# Plot 3: Deg > 4 constraint violation (Quadrant Proxy)\n",
        "plt.subplot(2, 2, 3)\n",
        "plot_comparison(metrics_real['viol_deg4'], metrics_gen['viol_deg4'],\n",
        "                \"Nodes violating constraint (Deg > 4)\", \"Nb nodes / graph\", bins=range(0, 15))\n",
        "\n",
        "# Plot 4: Parity Adherence\n",
        "if metrics_real['parity_ratio'] is not None:\n",
        "    plt.subplot(2, 2, 4)\n",
        "    plot_comparison(metrics_real['parity_ratio'], metrics_gen['parity_ratio'],\n",
        "                    \"Ratio of links respecting parity\", \"Ratio (1.0 = Perfect)\", bins=20)\n",
        "\n",
        "plt.tight_layout()\n",
        "plt.show()\n",
        "\n",
        "# --- Textual Summary ---\n",
        "print(\"\\n=== STATISTICAL SUMMARY (Averages) ===\")\n",
        "print(f\"{'Metric':<25} | {'Real':<10} | {'Generated':<10} | {'Diff':<10}\")\n",
        "print(\"-\" * 60)\n",
        "print(f\"{'Nb Edges / Graph':<25} | {np.mean(metrics_real['num_edges']):.2f}       | {np.mean(metrics_gen['num_edges']):.2f}       | {abs(np.mean(metrics_real['num_edges']) - np.mean(metrics_gen['num_edges'])):.2f}\")\n",
        "print(f\"{'Average Degree':<25} | {np.mean(metrics_real['degrees']):.2f}       | {np.mean(metrics_gen['degrees']):.2f}       | {abs(np.mean(metrics_real['degrees']) - np.mean(metrics_gen['degrees'])):.2f}\")\n",
        "print(f\"{'Nodes Deg > 4 (Avg)':<25} | {np.mean(metrics_real['viol_deg4']):.2f}       | {np.mean(metrics_gen['viol_deg4']):.2f}       | {abs(np.mean(metrics_real['viol_deg4']) - np.mean(metrics_gen['viol_deg4'])):.2f}\")\n",
        "print(f\"{'Parity Adherence (%)':<25} | {np.mean(metrics_real['parity_ratio'])*100:.1f}%      | {np.mean(metrics_gen['parity_ratio'])*100:.1f}%      | {abs(np.mean(metrics_real['parity_ratio']) - np.mean(metrics_gen['parity_ratio']))*100:.1f}%\")"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "goc4wqVhmKqq"
      },
      "outputs": [],
      "source": [
        "import matplotlib.pyplot as plt\n",
        "import numpy as np\n",
        "import math\n",
        "\n",
        "# --- Helper Function for Quadrants (Numpy Version) ---\n",
        "def get_quadrant_numpy(dx, dy):\n",
        "    \"\"\"\n",
        "    Vectorized or unitary version of the quadrant logic.\n",
        "    Returns 0, 1, 2, 3 based on the angle.\n",
        "    \"\"\"\n",
        "    angle = np.degrees(np.arctan2(dy, dx))\n",
        "    # Logic identical to generation\n",
        "    q = np.full_like(angle, 3, dtype=int)\n",
        "    q[(-45 < angle) & (angle <= 45)] = 0\n",
        "    q[(45 < angle) & (angle <= 135)] = 1\n",
        "    q[(135 < angle) & (angle <= 180) | (-180 <= angle) & (angle <= -135)] = 2\n",
        "    return q\n",
        "\n",
        "def compute_metrics(adjs, pos, parities=None):\n",
        "    \"\"\"\n",
        "    Calculates metrics including strict sector violation.\n",
        "    \"\"\"\n",
        "    # Binarization and diagonal cleaning\n",
        "    adjs_bin = (adjs > 0.5).astype(int)\n",
        "    B, N, _ = adjs_bin.shape\n",
        "    for i in range(B):\n",
        "        np.fill_diagonal(adjs_bin[i], 0)\n",
        "\n",
        "    # 1. Number of edges & Degrees\n",
        "    num_edges = np.sum(adjs_bin, axis=(1, 2)) / 2\n",
        "    degrees = np.sum(adjs_bin, axis=2)\n",
        "\n",
        "    # 2. Strict Sector Violation\n",
        "    # For each graph, count the number of nodes that have >1 link in the same quadrant\n",
        "    sector_violation_counts = []\n",
        "\n",
        "    for i in range(B):\n",
        "        # For this graph i\n",
        "        p = pos[i] # (N, 2)\n",
        "        adj = adjs_bin[i] # (N, N)\n",
        "\n",
        "        nodes_in_violation = 0\n",
        "\n",
        "        # Iterate over each node u\n",
        "        for u in range(N):\n",
        "            neighbors = np.where(adj[u] == 1)[0]\n",
        "            if len(neighbors) <= 1:\n",
        "                continue # No conflict possible with 0 or 1 neighbor\n",
        "\n",
        "            # Calculate vectors to neighbors\n",
        "            # p[neighbors] : (K, 2)\n",
        "            # p[u] : (2,)\n",
        "            vecs = p[neighbors] - p[u]\n",
        "            dx = vecs[:, 0]\n",
        "            dy = vecs[:, 1]\n",
        "\n",
        "            # Neighbors' quadrants\n",
        "            qs = get_quadrant_numpy(dx, dy)\n",
        "\n",
        "            # Check for duplicates in quadrants\n",
        "            # np.unique returns sorted unique values\n",
        "            if len(np.unique(qs)) < len(qs):\n",
        "                nodes_in_violation += 1\n",
        "\n",
        "        sector_violation_counts.append(nodes_in_violation)\n",
        "\n",
        "    sector_violation_counts = np.array(sector_violation_counts)\n",
        "\n",
        "    # 3. Parity adherence\n",
        "    parity_respect_ratio = []\n",
        "    if parities is not None:\n",
        "        for i in range(B):\n",
        "            u, v = np.where(np.triu(adjs_bin[i], k=1) == 1)\n",
        "            if len(u) > 0:\n",
        "                p_u = parities[i, u].flatten()\n",
        "                p_v = parities[i, v].flatten()\n",
        "                good_links = np.sum(p_u != p_v)\n",
        "                parity_respect_ratio.append(good_links / len(u))\n",
        "            else:\n",
        "                parity_respect_ratio.append(1.0)\n",
        "        parity_respect_ratio = np.array(parity_respect_ratio)\n",
        "\n",
        "    return {\n",
        "        \"num_edges\": num_edges,\n",
        "        \"degrees\": degrees.flatten(),\n",
        "        \"sector_violation\": sector_violation_counts, # This is the new key metric\n",
        "        \"parity_ratio\": parity_respect_ratio\n",
        "    }\n",
        "\n",
        "# --- 1. Prepare Real Data ---\n",
        "print(\">>> Generating reference dataset (1000 graphs)...\")\n",
        "real_dataset = SpatialNetworkDataset(num_samples=1000, num_nodes=cfg.num_nodes)\n",
        "real_adjs = np.array([item[1].numpy() for item in real_dataset.data])\n",
        "real_pos = np.array([item[0][:, :2].numpy() for item in real_dataset.data]) # Also retrieve POS\n",
        "real_parities = np.array([item[0][:, 2].numpy() for item in real_dataset.data])\n",
        "\n",
        "# --- 2. Generate Synthetic Data ---\n",
        "print(\">>> Generating model graphs (1000 graphs)....\")\n",
        "gen_adjs_list = []\n",
        "gen_pos_list = []      # Need generated positions to verify angles\n",
        "gen_parities_list = []\n",
        "\n",
        "batch_size = 100\n",
        "num_batches = 1000 // batch_size\n",
        "\n",
        "for b in range(num_batches):\n",
        "    print(f\"   Batch {b+1}/{num_batches}...\")\n",
        "    # sample returns (adj, pos, parity_target)\n",
        "    adj, pos, parity = sample(cam_model, num_samples=batch_size)\n",
        "    gen_adjs_list.append(adj)\n",
        "    gen_pos_list.append(pos)\n",
        "    gen_parities_list.append(parity)\n",
        "\n",
        "gen_adjs = np.concatenate(gen_adjs_list, axis=0)\n",
        "gen_pos = np.concatenate(gen_pos_list, axis=0)\n",
        "gen_parities = np.concatenate(gen_parities_list, axis=0)\n",
        "\n",
        "# --- 3. Calculate Metrics (With POS) ---\n",
        "print(\">>> Calculating strict metrics...\")\n",
        "# Pass real and generated positions\n",
        "metrics_real = compute_metrics(real_adjs, real_pos, real_parities)\n",
        "metrics_gen = compute_metrics(gen_adjs, gen_pos, gen_parities)\n",
        "\n",
        "# --- 4. Display ---\n",
        "def plot_comparison(metric_real, metric_gen, title, xlabel, bins=20, range_bins=None):\n",
        "    if range_bins is not None:\n",
        "        bins = range_bins\n",
        "    plt.hist(metric_real, bins=bins, alpha=0.5, label='Real Dataset', density=True, color='blue', edgecolor='black')\n",
        "    plt.hist(metric_gen, bins=bins, alpha=0.5, label='Generated (Model)', density=True, color='red', edgecolor='black')\n",
        "    plt.title(title, fontsize=10)\n",
        "    plt.xlabel(xlabel)\n",
        "    plt.legend()\n",
        "\n",
        "plt.figure(figsize=(15, 10))\n",
        "\n",
        "# Plot 1: Nb links\n",
        "plt.subplot(2, 2, 1)\n",
        "plot_comparison(metrics_real['num_edges'], metrics_gen['num_edges'],\n",
        "                \"Number of edges\", \"Number of edges\")\n",
        "\n",
        "# Plot 2: Degrees\n",
        "plt.subplot(2, 2, 2)\n",
        "plot_comparison(metrics_real['degrees'], metrics_gen['degrees'],\n",
        "                \"Degree Distribution\", \"Degree\", range_bins=range(0, 10))\n",
        "\n",
        "# Plot 3: STRICT SECTOR VIOLATION (The requested metric)\n",
        "plt.subplot(2, 2, 3)\n",
        "plot_comparison(metrics_real['sector_violation'], metrics_gen['sector_violation'],\n",
        "                \"Nodes violating Sector constraint\", \"Nb nodes / graph\", range_bins=range(0, 15))\n",
        "\n",
        "# Plot 4: Parity\n",
        "plt.subplot(2, 2, 4)\n",
        "plot_comparison(metrics_real['parity_ratio'], metrics_gen['parity_ratio'],\n",
        "                \"Parity Adherence\", \"Ratio (1.0 = Perfect)\", bins=20)\n",
        "\n",
        "plt.tight_layout()\n",
        "plt.show()\n",
        "\n",
        "# --- Summary ---\n",
        "print(\"\\n=== STATISTICAL SUMMARY (Averages over 1000 graphs) ===\")\n",
        "print(f\"{'Metric':<35} | {'Real':<10} | {'Generated':<10} | {'Diff':<10}\")\n",
        "print(\"-\" * 70)\n",
        "print(f\"{'Sector Violation (Nb nodes/graph)':<35} | {np.mean(metrics_real['sector_violation']):.2f}       | {np.mean(metrics_gen['sector_violation']):.2f}       | {abs(np.mean(metrics_real['sector_violation']) - np.mean(metrics_gen['sector_violation'])):.2f}\")\n",
        "print(f\"{'Average Degree':<35} | {np.mean(metrics_real['degrees']):.2f}       | {np.mean(metrics_gen['degrees']):.2f}       | {abs(np.mean(metrics_real['degrees']) - np.mean(metrics_gen['degrees'])):.2f}\")\n",
        "print(f\"{'Parity Adherence (%)':<35} | {np.mean(metrics_real['parity_ratio'])*100:.1f}%      | {np.mean(metrics_gen['parity_ratio'])*100:.1f}%      | {abs(np.mean(metrics_real['parity_ratio']) - np.mean(metrics_gen['parity_ratio']))*100:.1f}%\")"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "ze2ongdlWz6C"
      },
      "outputs": [],
      "source": [
        "import matplotlib.pyplot as plt\n",
        "import numpy as np\n",
        "import math\n",
        "import networkx as nx # Needed for connectivity test\n",
        "\n",
        "# --- Helper Function for Quadrants (Numpy Version) ---\n",
        "def get_quadrant_numpy(dx, dy):\n",
        "    angle = np.degrees(np.arctan2(dy, dx))\n",
        "    q = np.full_like(angle, 3, dtype=int)\n",
        "    q[(-45 < angle) & (angle <= 45)] = 0\n",
        "    q[(45 < angle) & (angle <= 135)] = 1\n",
        "    q[(135 < angle) & (angle <= 180) | (-180 <= angle) & (angle <= -135)] = 2\n",
        "    return q\n",
        "\n",
        "def compute_metrics(adjs, pos, parities=None):\n",
        "    \"\"\"\n",
        "    Calculates metrics including strict sector violation and connectivity.\n",
        "    \"\"\"\n",
        "    # Binarization and diagonal cleaning\n",
        "    adjs_bin = (adjs > 0.5).astype(int)\n",
        "    B, N, _ = adjs_bin.shape\n",
        "    for i in range(B):\n",
        "        np.fill_diagonal(adjs_bin[i], 0)\n",
        "\n",
        "    # 1. Number of edges & Degrees\n",
        "    num_edges = np.sum(adjs_bin, axis=(1, 2)) / 2\n",
        "    degrees = np.sum(adjs_bin, axis=2)\n",
        "\n",
        "    # 2. Strict Sector Violation\n",
        "    sector_violation_counts = []\n",
        "\n",
        "    # 3. Connectivity (New)\n",
        "    is_connected_list = []\n",
        "\n",
        "    for i in range(B):\n",
        "        # --- A. Sector Violation ---\n",
        "        p = pos[i]\n",
        "        adj = adjs_bin[i]\n",
        "        nodes_in_violation = 0\n",
        "\n",
        "        for u in range(N):\n",
        "            neighbors = np.where(adj[u] == 1)[0]\n",
        "            if len(neighbors) <= 1: continue\n",
        "\n",
        "            vecs = p[neighbors] - p[u]\n",
        "            qs = get_quadrant_numpy(vecs[:, 0], vecs[:, 1])\n",
        "\n",
        "            if len(np.unique(qs)) < len(qs):\n",
        "                nodes_in_violation += 1\n",
        "        sector_violation_counts.append(nodes_in_violation)\n",
        "\n",
        "        # --- B. Connectivity ---\n",
        "        # Use NetworkX to check if the graph is connected\n",
        "        G = nx.from_numpy_array(adj)\n",
        "        is_connected_list.append(1 if nx.is_connected(G) else 0)\n",
        "\n",
        "    sector_violation_counts = np.array(sector_violation_counts)\n",
        "    connectivity = np.array(is_connected_list)\n",
        "\n",
        "    # 4. Parity adherence\n",
        "    parity_respect_ratio = []\n",
        "    if parities is not None:\n",
        "        for i in range(B):\n",
        "            u, v = np.where(np.triu(adjs_bin[i], k=1) == 1)\n",
        "            if len(u) > 0:\n",
        "                p_u = parities[i, u].flatten()\n",
        "                p_v = parities[i, v].flatten()\n",
        "                good_links = np.sum(p_u != p_v)\n",
        "                parity_respect_ratio.append(good_links / len(u))\n",
        "            else:\n",
        "                parity_respect_ratio.append(1.0)\n",
        "        parity_respect_ratio = np.array(parity_respect_ratio)\n",
        "\n",
        "    return {\n",
        "        \"num_edges\": num_edges,\n",
        "        \"degrees\": degrees.flatten(),\n",
        "        \"sector_violation\": sector_violation_counts,\n",
        "        \"parity_ratio\": parity_respect_ratio,\n",
        "        \"connectivity\": connectivity # New\n",
        "    }\n",
        "\n",
        "# --- 1. Prepare Real Data ---\n",
        "print(\">>> Generating reference dataset (1000 graphs)...\")\n",
        "real_dataset = SpatialNetworkDataset(num_samples=1000, num_nodes=cfg.num_nodes)\n",
        "real_adjs = np.array([item[1].numpy() for item in real_dataset.data])\n",
        "real_pos = np.array([item[0][:, :2].numpy() for item in real_dataset.data])\n",
        "real_parities = np.array([item[0][:, 2].numpy() for item in real_dataset.data])\n",
        "\n",
        "# --- 2. Generate Synthetic Data ---\n",
        "print(\">>> Generating model graphs (1000 graphs)...\")\n",
        "gen_adjs_list = []\n",
        "gen_pos_list = []\n",
        "gen_parities_list = []\n",
        "\n",
        "batch_size = 100\n",
        "num_batches = 1000 // batch_size\n",
        "\n",
        "for b in range(num_batches):\n",
        "    print(f\"   Batch {b+1}/{num_batches}...\")\n",
        "    adj, pos, parity = sample(cam_model, num_samples=batch_size)\n",
        "    gen_adjs_list.append(adj)\n",
        "    gen_pos_list.append(pos)\n",
        "    gen_parities_list.append(parity)\n",
        "\n",
        "gen_adjs = np.concatenate(gen_adjs_list, axis=0)\n",
        "gen_pos = np.concatenate(gen_pos_list, axis=0)\n",
        "gen_parities = np.concatenate(gen_parities_list, axis=0)\n",
        "\n",
        "# --- 3. Calculate Metrics ---\n",
        "print(\">>> Calculating strict metrics...\")\n",
        "metrics_real = compute_metrics(real_adjs, real_pos, real_parities)\n",
        "metrics_gen = compute_metrics(gen_adjs, gen_pos, gen_parities)\n",
        "\n",
        "# --- 4. Display ---\n",
        "def plot_comparison(metric_real, metric_gen, title, xlabel, bins=20, range_bins=None):\n",
        "    if range_bins is not None: bins = range_bins\n",
        "    plt.hist(metric_real, bins=bins, alpha=0.5, label='Real Dataset', density=True, color='blue', edgecolor='black')\n",
        "    plt.hist(metric_gen, bins=bins, alpha=0.5, label='Generated (Model)', density=True, color='red', edgecolor='black')\n",
        "    plt.title(title, fontsize=10)\n",
        "    plt.xlabel(xlabel)\n",
        "    plt.legend()\n",
        "\n",
        "plt.figure(figsize=(15, 12)) # Slightly larger to accommodate the 5th metric\n",
        "\n",
        "# Plot 1: Nb links\n",
        "plt.subplot(3, 2, 1)\n",
        "plot_comparison(metrics_real['num_edges'], metrics_gen['num_edges'],\n",
        "                \"Number of edges\", \"Number of edges\")\n",
        "\n",
        "# Plot 2: Degrees\n",
        "plt.subplot(3, 2, 2)\n",
        "plot_comparison(metrics_real['degrees'], metrics_gen['degrees'],\n",
        "                \"Degree Distribution\", \"Degree\", range_bins=range(0, 10))\n",
        "\n",
        "# Plot 3: STRICT SECTOR VIOLATION\n",
        "plt.subplot(3, 2, 3)\n",
        "plot_comparison(metrics_real['sector_violation'], metrics_gen['sector_violation'],\n",
        "                \"Nodes violating Sector constraint\", \"Nb nodes / graph\", range_bins=range(0, 15))\n",
        "\n",
        "# Plot 4: Parity\n",
        "plt.subplot(3, 2, 4)\n",
        "plot_comparison(metrics_real['parity_ratio'], metrics_gen['parity_ratio'],\n",
        "                \"Parity Adherence\", \"Ratio (1.0 = Perfect)\", bins=20)\n",
        "\n",
        "# Plot 5: Connectivity (Bar Chart)\n",
        "plt.subplot(3, 2, 5)\n",
        "conn_real = np.mean(metrics_real['connectivity']) * 100\n",
        "conn_gen = np.mean(metrics_gen['connectivity']) * 100\n",
        "plt.bar([\"Real\", \"Generated\"], [conn_real, conn_gen], color=['blue', 'red'], alpha=0.6, edgecolor='black')\n",
        "plt.ylabel(\"% of connected graphs\")\n",
        "plt.title(f\"Connectivity (Real: {conn_real:.1f}% vs Gen: {conn_gen:.1f}%)\")\n",
        "plt.ylim(0, 100)\n",
        "\n",
        "plt.tight_layout()\n",
        "plt.show()\n",
        "\n",
        "# --- Summary ---\n",
        "print(\"\\n=== STATISTICAL SUMMARY (Averages over 1000 graphs) ===\")\n",
        "print(f\"{'Metric':<35} | {'Real':<10} | {'Generated':<10} | {'Diff':<10}\")\n",
        "print(\"-\" * 70)\n",
        "print(f\"{'Connected Graphs (%)':<35} | {conn_real:.1f}%      | {conn_gen:.1f}%      | {abs(conn_real - conn_gen):.1f}%\")\n",
        "print(f\"{'Sector Violation (Nb nodes)':<35} | {np.mean(metrics_real['sector_violation']):.2f}       | {np.mean(metrics_gen['sector_violation']):.2f}       | {abs(np.mean(metrics_real['sector_violation']) - np.mean(metrics_gen['sector_violation'])):.2f}\")\n",
        "print(f\"{'Average Degree':<35} | {np.mean(metrics_real['degrees']):.2f}       | {np.mean(metrics_gen['degrees']):.2f}       | {abs(np.mean(metrics_real['degrees']) - np.mean(metrics_gen['degrees'])):.2f}\")\n",
        "print(f\"{'Parity Adherence (%)':<35} | {np.mean(metrics_real['parity_ratio'])*100:.1f}%      | {np.mean(metrics_gen['parity_ratio'])*100:.1f}%      | {abs(np.mean(metrics_real['parity_ratio']) - np.mean(metrics_gen['parity_ratio']))*100:.1f}%\")"
      ]
    }
  ],
  "metadata": {
    "accelerator": "GPU",
    "colab": {
      "gpuType": "A100",
      "machine_shape": "hm",
      "provenance": []
    },
    "kernelspec": {
      "display_name": "Python 3",
      "name": "python3"
    },
    "language_info": {
      "name": "python"
    }
  },
  "nbformat": 4,
  "nbformat_minor": 0
}