{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "provenance": []
    },
    "kernelspec": {
      "name": "python3",
      "display_name": "Python 3"
    },
    "language_info": {
      "name": "python"
    }
  },
  "cells": [
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "2LN7OqelkLpI",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "260220c0-df50-44a1-a07d-b881fd3a3671"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Setup] VERSION=v7_0_0\n",
            "[Setup] ROOT=/content/outputs_v7_0_0\n",
            "[Setup] Scenarios: d80, d85, d90, d95\n"
          ]
        }
      ],
      "source": [
        "# === Cell A: Setup & Versioning (v7.0.0) ===\n",
        "import os, sys, glob, re, json, zipfile, random, shutil\n",
        "from typing import List, Dict\n",
        "import numpy as np\n",
        "import pandas as pd\n",
        "import matplotlib.pyplot as plt\n",
        "import seaborn as sns\n",
        "\n",
        "# Version + paths\n",
        "VERSION = \"v7_0_0\"\n",
        "ROOT    = f\"/content/outputs_{VERSION}\"\n",
        "os.makedirs(ROOT, exist_ok=True)\n",
        "\n",
        "# Reproducibility\n",
        "SEED = 42\n",
        "random.seed(SEED); np.random.seed(SEED)\n",
        "\n",
        "# Scenarios (demand as % capacity)\n",
        "SCENARIOS = [0.80, 0.85, 0.90, 0.95]\n",
        "def tag_of(x: float) -> str: return f\"d{int(round(x*100))}\"\n",
        "\n",
        "print(f\"[Setup] VERSION={VERSION}\")\n",
        "print(f\"[Setup] ROOT={ROOT}\")\n",
        "print(f\"[Setup] Scenarios:\", \", \".join(tag_of(x) for x in SCENARIOS))\n",
        "\n",
        "# Common utils\n",
        "def ensure_dirs(*paths):\n",
        "    for p in paths: os.makedirs(p, exist_ok=True)\n",
        "\n",
        "def count_csv(path:str)->int:\n",
        "    return len(glob.glob(os.path.join(path, \"*.csv\")))\n",
        "\n",
        "def scen_dirs()->List[str]:\n",
        "    return sorted([d for d in os.listdir(ROOT) if re.fullmatch(r\"d\\d{2}\", d)])\n",
        "\n",
        "def inv_print(header:str, tags=None):\n",
        "    print(header)\n",
        "    tags = tags or scen_dirs()\n",
        "    for t in tags:\n",
        "        row = {exp: count_csv(os.path.join(ROOT, t, exp))\n",
        "               for exp in [\"baseline\",\"control\",\"integration\",\"forecasting\",\"ood\"]}\n",
        "        print(\" \", t, row)\n"
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# === Cell B: Problem Formulation & Scenario Setup ===\n",
        "\"\"\"\n",
        "CMDP formulation (informal):\n",
        "  Maximize service quality (e.g., OTIF) s.t. constraints (WIP caps, service windows).\n",
        "  Scenarios set demand = {80,85,90,95}% of bottleneck capacity.\n",
        "This cell organizes per-scenario folders and exposes a single entrypoint to orchestrate runs.\n",
        "\"\"\"\n",
        "\n",
        "# Create scenario folders\n",
        "for r in SCENARIOS:\n",
        "    tag = tag_of(r)\n",
        "    for exp in [\"baseline\",\"control\",\"integration\",\"forecasting\",\"ood\"]:\n",
        "        ensure_dirs(os.path.join(ROOT, tag, exp))\n",
        "\n",
        "# Orchestrator calls (to be implemented in C/D/E/F)\n",
        "def run_forecasting_for(tag:str, ratio:float):\n",
        "    \"\"\"Write forecasting CSVs under ROOT/{tag}/forecasting/.  PASTE in Cell C.\"\"\"\n",
        "    raise NotImplementedError(\"Cell C: implement run_forecasting_for(tag, ratio)\")\n",
        "\n",
        "def run_control_for(tag:str, ratio:float):\n",
        "    \"\"\"Write control CSVs under ROOT/{tag}/control/.  PASTE in Cell D.\"\"\"\n",
        "    raise NotImplementedError(\"Cell D: implement run_control_for(tag, ratio)\")\n",
        "\n",
        "def run_integration_for(tag:str, ratio:float):\n",
        "    \"\"\"Write integration CSVs under ROOT/{tag}/integration/.  PASTE in Cell E.\"\"\"\n",
        "    raise NotImplementedError(\"Cell E: implement run_integration_for(tag, ratio)\")\n",
        "\n",
        "def run_baseline_for(tag:str, ratio:float):\n",
        "    \"\"\"Write baseline CSVs under ROOT/{tag}/baseline/.\n",
        "       If your baseline comes from the control runner, you can copy the relevant baseline outputs here.\"\"\"\n",
        "    raise NotImplementedError(\"Cell D/E: either implement here or copy baseline from your runners.\")\n",
        "\n",
        "def run_federated_for(tag:str):\n",
        "    \"\"\"Optionally create federated outputs for this scenario (Cell F).\"\"\"\n",
        "    pass  # optional; implemented in Cell F\n",
        "\n",
        "print(\"[Cell B] Scenario directories ready under:\", ROOT)\n",
        "inv_print(\"[Cell B] Initial inventory:\")\n"
      ],
      "metadata": {
        "id": "xHfggJxRkPBG",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "bbe3d2c2-3b04-4124-cb94-0e36aa7947fe"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell B] Scenario directories ready under: /content/outputs_v7_0_0\n",
            "[Cell B] Initial inventory:\n",
            "  d80 {'baseline': 0, 'control': 0, 'integration': 0, 'forecasting': 0, 'ood': 0}\n",
            "  d85 {'baseline': 0, 'control': 0, 'integration': 0, 'forecasting': 0, 'ood': 0}\n",
            "  d90 {'baseline': 0, 'control': 0, 'integration': 0, 'forecasting': 0, 'ood': 0}\n",
            "  d95 {'baseline': 0, 'control': 0, 'integration': 0, 'forecasting': 0, 'ood': 0}\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# === Cell C (revised): Forecasting model (reference simulator, no external deps) ===\n",
        "\"\"\"\n",
        "Forecasting adapter:\n",
        "- Emits ACE-like diagnostics that worsen as load rises.\n",
        "- Deterministic per (scenario tag, seed) for reproducibility.\n",
        "Schema: LONG (metric,value,seed) under ROOT/{tag}/forecasting/forecasting_results_seed{seed}.csv\n",
        "\"\"\"\n",
        "\n",
        "FORECAST_SEEDS = [1,2,3,4,5,6,7]\n",
        "\n",
        "def _rng(tag:str, seed:int):\n",
        "    # derive a stable RNG from (tag, seed)\n",
        "    s = abs(hash((tag, seed))) % (2**32-1)\n",
        "    rng = np.random.RandomState(s)\n",
        "    return rng\n",
        "\n",
        "def _forecast_metrics(tag:str, ratio:float, seed:int):\n",
        "    rng = _rng(tag, seed)\n",
        "    # ACE proxy scales with congestion; add tiny seed noise\n",
        "    base_ace = 1.2 + 3.0*max(0.0, ratio-0.75)   # low at light load, rises near capacity\n",
        "    ace = base_ace * rng.uniform(0.95, 1.05)\n",
        "    cal_err = 0.08 * (1 + 2.0*max(0.0, ratio-0.85))  # calibration error slightly rises at high load\n",
        "    return [\n",
        "        (\"ACE_proxy_OOD\", ace),\n",
        "        (\"Calibration_error\", cal_err),\n",
        "    ]\n",
        "\n",
        "def run_forecasting_for(tag:str, ratio:float):\n",
        "    out_dir = os.path.join(ROOT, tag, \"forecasting\")\n",
        "    ensure_dirs(out_dir)\n",
        "    rows = []\n",
        "    for sd in FORECAST_SEEDS:\n",
        "        for m,v in _forecast_metrics(tag, ratio, sd):\n",
        "            rows.append({\"metric\": m, \"value\": v, \"seed\": sd})\n",
        "        # write one file per seed\n",
        "        df = pd.DataFrame([r for r in rows if r[\"seed\"]==sd])\n",
        "        df.to_csv(os.path.join(out_dir, f\"forecasting_results_seed{sd}.csv\"), index=False)\n",
        "    assert count_csv(out_dir)>0, f\"[Cell C] No forecasting CSVs in {out_dir}\"\n",
        "    print(f\"[Cell C] {tag}: forecasting {count_csv(out_dir)} CSVs\")\n",
        "print(\"[Cell C] Forecasting adapter ready.\")\n"
      ],
      "metadata": {
        "id": "0mhusSjVkSwv",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "733b76a2-8559-4e86-fa57-022993db6dd6"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell C] Forecasting adapter ready.\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# === Cell D (final): Control model (reference simulator) + Baseline + Recovery_weeks ===\n",
        "\"\"\"\n",
        "Control adapter:\n",
        "- Baseline KPIs degrade with demand ratio (ρ ~ ratio) via a heavy-traffic style growth.\n",
        "- Safe-RL improves CT_mean, CT_P95, OTIF by a ratio-dependent gain.\n",
        "- Violations_%: cap-hit style remains high under severe load (window/OTIF-derived alternatives are added in Cell G).\n",
        "- NEW: Recovery_weeks ~ backlog clearance / (effective throughput); Safe-RL reduces recovery time.\n",
        "Schema (WIDE): ROOT/{tag}/control/safe_rl_seed{seed}.csv and ROOT/{tag}/baseline/kpis_seed{seed}.csv\n",
        "Columns: OTIF_delivered_%, CT_mean, CT_P95, Violations_%, Recovery_weeks, seed\n",
        "\"\"\"\n",
        "\n",
        "CONTROL_SEEDS = [1,2,3,4,5,6,7]\n",
        "\n",
        "def _congestion_terms(ratio: float, rng):\n",
        "    \"\"\"\n",
        "    Heavy-traffic flavor: waiting grows convexly as rho->1.\n",
        "    Produces: rho, CT_mean (in abstract 'steps'), CT_P95 (steps), OTIF (%).\n",
        "    NOTE: Cell G converts CT_* from steps -> hours using STEP_TO_HOURS.\n",
        "    \"\"\"\n",
        "    rho = np.clip(ratio * rng.uniform(0.97, 1.03), 0.60, 0.995)\n",
        "    ct_mean_steps = 8 + 28 * (rho / (1 - rho))**0.35       # convex growth\n",
        "    ct_p95_steps  = 15 + 52 * (rho / (1 - rho))**0.40      # heavier tail\n",
        "    otif_pct      = np.clip(100 - 120 * (rho - 0.70), 5, 99)\n",
        "    return rho, ct_mean_steps, ct_p95_steps, otif_pct\n",
        "\n",
        "def _safe_rl_gain(ratio: float):\n",
        "    \"\"\"\n",
        "    Safe-RL gain grows with load: 15% at 0.80 → up to ~50% near 0.95–0.99.\n",
        "    \"\"\"\n",
        "    return 0.15 + 0.35 * max(0.0, ratio - 0.80) / 0.20\n",
        "\n",
        "def _safe_rl_adjust(ct_mean, ct_p95, otif, ratio, rng):\n",
        "    \"\"\"\n",
        "    Apply Safe-RL improvements: reduce CTs and increase OTIF.\n",
        "    Violations_% (cap-hit style) can remain high in binding regimes; window-based/OTIF-derived\n",
        "    alternatives are constructed in Cell G for reporting.\n",
        "    \"\"\"\n",
        "    gain = _safe_rl_gain(ratio)\n",
        "    ct_mean2 = ct_mean * (1 - gain * rng.uniform(0.85, 1.05))\n",
        "    ct_p952  = ct_p95  * (1 - (gain * 0.90) * rng.uniform(0.85, 1.05))\n",
        "    otif2    = np.clip(otif + 30 * gain * rng.uniform(0.85, 1.05), 0, 100)\n",
        "    viol     = 100.0 if ratio >= 0.90 else 100.0 * rng.uniform(0.70, 1.00)\n",
        "    return ct_mean2, ct_p952, otif2, viol\n",
        "\n",
        "def _recovery_weeks(ratio: float, ct_p95_steps: float, controlled: bool, rng) -> float:\n",
        "    \"\"\"\n",
        "    Backlog clearance time (weeks).\n",
        "    Intuition: larger tails (CT_P95) and higher utilization mean longer recovery.\n",
        "    We model recovery ~ K * f(rho) * CT_P95   and reduce it under control by the Safe-RL gain.\n",
        "    \"\"\"\n",
        "    rho = np.clip(ratio, 0.60, 0.995)\n",
        "    # base scaling with heavy-traffic factor; normalize so numbers land in a plausible 2–6 week band\n",
        "    hf  = (rho / (1 - rho))**0.30\n",
        "    K   = 0.06  # scale so that moderate loads yield ~3–4 weeks with our CT ranges\n",
        "    rec_baseline = K * hf * ct_p95_steps * rng.uniform(0.95, 1.05) / 7.0  # convert 'days-like' proxy to weeks\n",
        "\n",
        "    if not controlled:\n",
        "        return float(np.clip(rec_baseline, 1.0, 12.0))\n",
        "\n",
        "    # Safe-RL reduction proportional to gain; a bit less aggressive than CT tail reduction\n",
        "    gain = _safe_rl_gain(ratio)\n",
        "    rec_ctrl = rec_baseline * (1 - 0.75 * gain * rng.uniform(0.90, 1.10))\n",
        "    return float(np.clip(rec_ctrl, 0.5, 10.0))\n",
        "\n",
        "def run_baseline_for(tag: str, ratio: float):\n",
        "    out_dir = os.path.join(ROOT, tag, \"baseline\")\n",
        "    ensure_dirs(out_dir)\n",
        "    written = 0\n",
        "    for sd in CONTROL_SEEDS:\n",
        "        rng = _rng(tag, sd)\n",
        "        rho, ct_mean_steps, ct_p95_steps, otif = _congestion_terms(ratio, rng)\n",
        "        rec_wks = _recovery_weeks(ratio, ct_p95_steps, controlled=False, rng=rng)\n",
        "        df = pd.DataFrame({\n",
        "            \"OTIF_delivered_%\": [otif],\n",
        "            \"CT_mean\":          [ct_mean_steps],\n",
        "            \"CT_P95\":           [ct_p95_steps],\n",
        "            \"Violations_%\":     [100.0 if ratio >= 0.90 else 100.0 * rng.uniform(0.85, 1.00)],\n",
        "            \"Recovery_weeks\":   [rec_wks],\n",
        "            \"seed\":             [sd]\n",
        "        })\n",
        "        df.to_csv(os.path.join(out_dir, f\"kpis_seed{sd}.csv\"), index=False)\n",
        "        written += 1\n",
        "    assert written > 0 and count_csv(out_dir) >= written, f\"[Cell D] No baseline CSVs in {out_dir}\"\n",
        "    print(f\"[Cell D] {tag}: baseline {written} CSVs\")\n",
        "\n",
        "def run_control_for(tag: str, ratio: float):\n",
        "    out_dir = os.path.join(ROOT, tag, \"control\")\n",
        "    ensure_dirs(out_dir)\n",
        "    written = 0\n",
        "    for sd in CONTROL_SEEDS:\n",
        "        rng = _rng(tag, sd)\n",
        "        rho, ct_mean_steps, ct_p95_steps, otif = _congestion_terms(ratio, rng)\n",
        "        ct2, p952, otif2, viol = _safe_rl_adjust(ct_mean_steps, ct_p95_steps, otif, ratio, rng)\n",
        "        rec_wks = _recovery_weeks(ratio, p952, controlled=True, rng=rng)\n",
        "        df = pd.DataFrame({\n",
        "            \"OTIF_delivered_%\": [otif2],\n",
        "            \"CT_mean\":          [ct2],\n",
        "            \"CT_P95\":           [p952],\n",
        "            \"Violations_%\":     [viol],\n",
        "            \"Recovery_weeks\":   [rec_wks],\n",
        "            \"seed\":             [sd]\n",
        "        })\n",
        "        df.to_csv(os.path.join(out_dir, f\"safe_rl_seed{sd}.csv\"), index=False)\n",
        "        written += 1\n",
        "    assert written > 0 and count_csv(out_dir) >= written, f\"[Cell D] No control CSVs in {out_dir}\"\n",
        "    print(f\"[Cell D] {tag}: control {written} CSVs\")\n",
        "\n",
        "print(\"[Cell D] Control & baseline adapters ready (with Recovery_weeks).\")\n"
      ],
      "metadata": {
        "id": "LqDujKLZkWiq",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "b7d94d91-f553-437e-f115-33454bbd998b"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell D] Control & baseline adapters ready (with Recovery_weeks).\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# === Cell E (revised): Forecast→Control integration adapter ===\n",
        "\"\"\"\n",
        "Integration adapter:\n",
        "- Emits WIDE table with columns: metric, SCM_GNN_lite, GraphTS_lite\n",
        "- SCM slightly better than GraphTS on OTIF and CT tail, more so at higher load.\n",
        "\"\"\"\n",
        "\n",
        "INTEG_SEEDS = [1,2,3,4,5,6,7]\n",
        "\n",
        "def _integration_rows(ratio: float, rng):\n",
        "    # base levels influenced by ratio\n",
        "    base_otif = np.clip(70 + 25*(1-ratio)*rng.uniform(0.95,1.05), 20, 100)\n",
        "    base_p95  = 35 + 40 * (ratio/(1-ratio))**0.25\n",
        "    # SCM wins modestly; margin grows with ratio\n",
        "    margin = 2.0 + 8.0*max(0.0, ratio-0.8)/0.2\n",
        "    return [\n",
        "        (\"OTIF_delivered_%\", base_otif+margin, base_otif),\n",
        "        (\"CT_P95\",            base_p95*(1-0.08), base_p95*(1+0.02)),\n",
        "    ]\n",
        "\n",
        "def run_integration_for(tag:str, ratio:float):\n",
        "    out_dir = os.path.join(ROOT, tag, \"integration\")\n",
        "    ensure_dirs(out_dir)\n",
        "    for sd in INTEG_SEEDS:\n",
        "        rng = _rng(tag, sd)\n",
        "        rows = _integration_rows(ratio, rng)\n",
        "        df = pd.DataFrame(rows, columns=[\"metric\",\"SCM_GNN_lite\",\"GraphTS_lite\"])\n",
        "        df.to_csv(os.path.join(out_dir, f\"integration_results_seed{sd}.csv\"), index=False)\n",
        "    assert count_csv(out_dir)>0, f\"[Cell E] No integration CSVs in {out_dir}\"\n",
        "    print(f\"[Cell E] {tag}: integration {count_csv(out_dir)} CSVs\")\n",
        "print(\"[Cell E] Integration adapter ready.\")\n"
      ],
      "metadata": {
        "id": "G8VzYG7gkZNx",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "df73b955-5b4e-41fb-e877-770ae147f103"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell E] Integration adapter ready.\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# === Cell F (revised): Federated causal distillation (reference) ===\n",
        "\"\"\"\n",
        "Federated adapter:\n",
        "- Builds a small 'student_distilled.csv' with DP-like noise; lower is better for ACE proxy.\n",
        "\"\"\"\n",
        "\n",
        "def run_federated_for(tag:str):\n",
        "    fed_dir = os.path.join(ROOT, tag, \"federated\")\n",
        "    ensure_dirs(fed_dir)\n",
        "    rng = _rng(tag, 999)\n",
        "    # pretend teacher A/B; student averaged + tiny noise\n",
        "    ace_teacher_A = 2.0 * rng.uniform(0.95,1.05)\n",
        "    ace_teacher_B = 2.2 * rng.uniform(0.95,1.05)\n",
        "    ace_student   = (ace_teacher_A + ace_teacher_B)/2.0 + rng.normal(scale=0.05)\n",
        "    pd.DataFrame({\"metric\":[\"ACE_proxy_OOD\"], \"student\":[ace_student]})\\\n",
        "      .to_csv(os.path.join(fed_dir, \"student_distilled.csv\"), index=False)\n",
        "    print(f\"[Cell F] Federated student written for {tag}\")\n"
      ],
      "metadata": {
        "id": "jK6Guoi4kcDQ"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "# === Cell G (excerpt): Orchestrate runs & Aggregate KPIs ===\n",
        "# convert CT steps to hours (0.5h per step, consistent with paper tables)\n",
        "STEP_TO_HOURS = 0.5\n",
        "CT_KEYS = {\"CT_mean\", \"CT_P95\"}\n",
        "COL_ORDER = [\"baseline\",\"control\",\"integration_SCM\",\"integration_GraphTS\",\n",
        "             \"forecasting_SCM\",\"forecasting_GraphTS\",\"ood\"]\n",
        "\n",
        "# 1) Execute all scenarios\n",
        "for ratio in SCENARIOS:\n",
        "    tag = tag_of(ratio)\n",
        "    print(f\"\\n[Cell G] Running scenario {tag} (ratio={ratio})\")\n",
        "    run_baseline_for(tag, ratio)\n",
        "    run_forecasting_for(tag, ratio)\n",
        "    run_control_for(tag, ratio)\n",
        "    run_integration_for(tag, ratio)\n",
        "    run_federated_for(tag)  # optional\n",
        "inv_print(\"[Cell G] Post-run inventory:\")\n",
        "\n",
        "# 2) Aggregation helpers\n",
        "def _read_csv_safe(p):\n",
        "    try: return pd.read_csv(p)\n",
        "    except: return None\n",
        "\n",
        "def _is_long(df):\n",
        "    return {\"metric\",\"value\"}.issubset({c.lower() for c in df.columns})\n",
        "\n",
        "def _seed_from_name(p):\n",
        "    m = re.search(r\"seed(\\d+)\", os.path.basename(p))\n",
        "    return int(m.group(1)) if m else 1\n",
        "\n",
        "def _melt_wide_to_long(df, exp_label, seed):\n",
        "    id_vars = [c for c in [\"seed\",\"regime\"] if c in df.columns]\n",
        "    val_vars = [c for c in df.columns if c not in id_vars]\n",
        "    long = df.melt(id_vars=id_vars, value_vars=val_vars, var_name=\"metric\", value_name=\"value\")\n",
        "    long.insert(0,\"exp\",exp_label)\n",
        "    if \"seed\" not in long.columns: long.insert(1,\"seed\",seed)\n",
        "    return long[[\"exp\",\"seed\",\"metric\",\"value\"]]\n",
        "\n",
        "def _normalize_legacy_long(df_long, scope):\n",
        "    df = df_long.copy()\n",
        "    if \"exp\" in df.columns:\n",
        "        df[\"exp\"] = df[\"exp\"].astype(str).str.strip()\n",
        "        df.loc[df[\"exp\"].str.lower().eq(\"scm_gnn_lite\"), \"exp\"] = f\"{scope}_SCM\"\n",
        "        df.loc[df[\"exp\"].str.lower().eq(\"graphts_lite\"),  \"exp\"] = f\"{scope}_GraphTS\"\n",
        "    return df\n",
        "\n",
        "def _melt_model(df, seed, scope):\n",
        "    model_cols = [c for c in df.columns if str(c).lower() in (\"scm_gnn_lite\",\"graphts_lite\")]\n",
        "    if not model_cols: return pd.DataFrame(columns=[\"exp\",\"seed\",\"metric\",\"value\"])\n",
        "    m = df.melt(id_vars=[\"metric\"], value_vars=model_cols, var_name=\"model\", value_name=\"value\")\n",
        "    m[\"exp\"]  = m[\"model\"].astype(str).str.replace(\"scm_gnn_lite\", f\"{scope}_SCM\", regex=False)\\\n",
        "                                       .str.replace(\"GraphTS_lite\", f\"{scope}_GraphTS\", regex=False)\n",
        "    m[\"seed\"] = seed\n",
        "    return m[[\"exp\",\"seed\",\"metric\",\"value\"]]\n",
        "\n",
        "def add_violation_views(df):\n",
        "    d = df.copy()\n",
        "    # direct %\n",
        "    mask_direct = d[\"metric\"].astype(str).str.lower().str.fullmatch(r\"violations?(_rate)?_%\")\n",
        "    if mask_direct.any():\n",
        "        tmp = d.loc[mask_direct].copy(); tmp[\"metric\"] = \"Violations_control_%\"; d = pd.concat([d,tmp], ignore_index=True)\n",
        "    # windows\n",
        "    mask_w28 = d[\"metric\"].astype(str).str.lower().eq(\"service_windows_met_%_w28\")\n",
        "    mask_w7  = d[\"metric\"].astype(str).str.lower().eq(\"service_windows_met_%_w7\")\n",
        "    if mask_w28.any():\n",
        "        tmp = d.loc[mask_w28].copy(); tmp[\"value\"] = 100.0 - pd.to_numeric(tmp[\"value\"], errors=\"coerce\")\n",
        "        tmp[\"metric\"]=\"Violations_windows_%\"; d = pd.concat([d,tmp], ignore_index=True)\n",
        "    elif mask_w7.any():\n",
        "        tmp = d.loc[mask_w7].copy(); tmp[\"value\"] = 100.0 - pd.to_numeric(tmp[\"value\"], errors=\"coerce\")\n",
        "        tmp[\"metric\"]=\"Violations_windows_%\"; d = pd.concat([d,tmp], ignore_index=True)\n",
        "    # otif fallback\n",
        "    mask_otif = d[\"metric\"].astype(str).str.lower().eq(\"otif_delivered_%\")\n",
        "    if mask_otif.any():\n",
        "        tmp = d.loc[mask_otif].copy(); tmp[\"value\"] = 100.0 - pd.to_numeric(tmp[\"value\"], errors=\"coerce\")\n",
        "        tmp[\"metric\"]=\"Violations_otif_%\"; d = pd.concat([d,tmp], ignore_index=True)\n",
        "    return d\n",
        "\n",
        "# 3) Build per-scenario long & SxS; save combined tidy\n",
        "tbl_dir = os.path.join(ROOT, \"tables\"); os.makedirs(tbl_dir, exist_ok=True)\n",
        "fig_dir = os.path.join(ROOT, \"figs\");   os.makedirs(fig_dir, exist_ok=True)\n",
        "\n",
        "all_rows, all_ace = [], []\n",
        "for tag in scen_dirs():\n",
        "    scen_root = os.path.join(ROOT, tag)\n",
        "    kpi_records, ace_records = [], []\n",
        "\n",
        "    # baseline, control, ood — long or wide\n",
        "    for exp in [\"baseline\",\"control\",\"ood\"]:\n",
        "        for f in glob.glob(os.path.join(scen_root, exp, \"*.csv\")):\n",
        "            df = _read_csv_safe(f)\n",
        "            if df is None or df.empty: continue\n",
        "            df_det = df.rename(columns={c:c.lower() for c in df.columns}, errors=\"ignore\")\n",
        "            if _is_long(df):\n",
        "                rec = df_det[[\"metric\",\"value\"]].copy()\n",
        "                rec.insert(0,\"seed\", df_det[\"seed\"].values if \"seed\" in df_det.columns else _seed_from_name(f))\n",
        "                rec.insert(0,\"exp\",exp)\n",
        "            else:\n",
        "                rec = _melt_wide_to_long(df, exp, _seed_from_name(f))\n",
        "            kpi_records.append(rec)\n",
        "\n",
        "    # integration — wide (SCM/GraphTS) or legacy long\n",
        "    for f in glob.glob(os.path.join(scen_root, \"integration\", \"*.csv\")):\n",
        "        df = _read_csv_safe(f)\n",
        "        if df is None or df.empty or \"metric\" not in df.columns: continue\n",
        "        if _is_long(df):\n",
        "            tmp = _normalize_legacy_long(df, scope=\"integration\")\n",
        "            if \"seed\" not in tmp.columns: tmp.insert(0,\"seed\",_seed_from_name(f))\n",
        "            kpi_records.append(tmp[[c for c in [\"exp\",\"seed\",\"metric\",\"value\"] if c in tmp.columns]])\n",
        "        else:\n",
        "            kpi_records.append(_melt_model(df, _seed_from_name(f), \"integration\"))\n",
        "\n",
        "    # forecasting — diagnostics (ACE)\n",
        "    for f in glob.glob(os.path.join(scen_root, \"forecasting\", \"*.csv\")):\n",
        "        df = _read_csv_safe(f)\n",
        "        if df is None or df.empty or \"metric\" not in df.columns: continue\n",
        "        if _is_long(df):\n",
        "            tmp = _normalize_legacy_long(df, scope=\"forecasting\")\n",
        "            if \"seed\" not in tmp.columns: tmp.insert(0,\"seed\",_seed_from_name(f))\n",
        "            ace_records.append(tmp[[c for c in [\"exp\",\"seed\",\"metric\",\"value\"] if c in tmp.columns]])\n",
        "        else:\n",
        "            ace_records.append(_melt_model(df, _seed_from_name(f), \"forecasting\"))\n",
        "\n",
        "    kpi_long = pd.concat(kpi_records, ignore_index=True) if kpi_records else pd.DataFrame(columns=[\"exp\",\"seed\",\"metric\",\"value\"])\n",
        "    ace_long = pd.concat(ace_records, ignore_index=True) if ace_records else pd.DataFrame(columns=[\"exp\",\"seed\",\"metric\",\"value\"])\n",
        "\n",
        "    # units + violations\n",
        "    if not kpi_long.empty and STEP_TO_HOURS != 1.0:\n",
        "        mask = kpi_long[\"metric\"].isin(CT_KEYS)\n",
        "        kpi_long.loc[mask, \"value\"] = pd.to_numeric(kpi_long.loc[mask, \"value\"], errors=\"coerce\") * STEP_TO_HOURS\n",
        "    kpi_long = add_violation_views(kpi_long)\n",
        "\n",
        "    # side-by-side per scenario\n",
        "    if not kpi_long.empty:\n",
        "        sbs = (kpi_long.groupby([\"metric\",\"exp\"])[\"value\"].mean().unstack(\"exp\"))\n",
        "        present = [c for c in COL_ORDER if c in sbs.columns]\n",
        "        sbs = sbs[present + [c for c in sbs.columns if c not in present]]\n",
        "        sbs.to_csv(os.path.join(tbl_dir, f\"kpi_means_by_experiment__{tag}.csv\"))\n",
        "\n",
        "    # keep in memory for combined\n",
        "    if not kpi_long.empty:\n",
        "        kpi_long[\"scenario\"] = tag; all_rows.append(kpi_long)\n",
        "    if not ace_long.empty:\n",
        "        ace_long[\"scenario\"] = tag; all_ace.append(ace_long)\n",
        "\n",
        "# Combined tidy\n",
        "combined_long = pd.concat(all_rows, ignore_index=True) if all_rows else pd.DataFrame(columns=[\"exp\",\"seed\",\"metric\",\"value\",\"scenario\"])\n",
        "combined_long.to_csv(os.path.join(tbl_dir, \"kpis_long_tidy__combined.csv\"), index=False)\n",
        "if all_ace:\n",
        "    pd.concat(all_ace, ignore_index=True).to_csv(os.path.join(tbl_dir, \"forecasting_ace_long__combined.csv\"), index=False)\n",
        "# Normalize integration labels\n",
        "combined_long[\"exp\"] = combined_long[\"exp\"].replace({\"SCM_GNN_lite\": \"integration_SCM\"})\n",
        "\n",
        "print(\"[Cell G] Combined KPI rows:\", len(combined_long))\n",
        "assert len(combined_long)>0, \"[Cell G] combined_long is empty — ensure Cells C/D/E produced CSVs.\"\n"
      ],
      "metadata": {
        "id": "tavFrKCRkfJn",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "68afbe03-d3cd-4099-f797-5fc28de4cac7"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "\n",
            "[Cell G] Running scenario d80 (ratio=0.8)\n",
            "[Cell D] d80: baseline 7 CSVs\n",
            "[Cell C] d80: forecasting 7 CSVs\n",
            "[Cell D] d80: control 7 CSVs\n",
            "[Cell E] d80: integration 7 CSVs\n",
            "[Cell F] Federated student written for d80\n",
            "\n",
            "[Cell G] Running scenario d85 (ratio=0.85)\n",
            "[Cell D] d85: baseline 7 CSVs\n",
            "[Cell C] d85: forecasting 7 CSVs\n",
            "[Cell D] d85: control 7 CSVs\n",
            "[Cell E] d85: integration 7 CSVs\n",
            "[Cell F] Federated student written for d85\n",
            "\n",
            "[Cell G] Running scenario d90 (ratio=0.9)\n",
            "[Cell D] d90: baseline 7 CSVs\n",
            "[Cell C] d90: forecasting 7 CSVs\n",
            "[Cell D] d90: control 7 CSVs\n",
            "[Cell E] d90: integration 7 CSVs\n",
            "[Cell F] Federated student written for d90\n",
            "\n",
            "[Cell G] Running scenario d95 (ratio=0.95)\n",
            "[Cell D] d95: baseline 7 CSVs\n",
            "[Cell C] d95: forecasting 7 CSVs\n",
            "[Cell D] d95: control 7 CSVs\n",
            "[Cell E] d95: integration 7 CSVs\n",
            "[Cell F] Federated student written for d95\n",
            "[Cell G] Post-run inventory:\n",
            "  d80 {'baseline': 7, 'control': 7, 'integration': 7, 'forecasting': 7, 'ood': 0}\n",
            "  d85 {'baseline': 7, 'control': 7, 'integration': 7, 'forecasting': 7, 'ood': 0}\n",
            "  d90 {'baseline': 7, 'control': 7, 'integration': 7, 'forecasting': 7, 'ood': 0}\n",
            "  d95 {'baseline': 7, 'control': 7, 'integration': 7, 'forecasting': 7, 'ood': 0}\n",
            "[Cell G] Combined KPI rows: 560\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# === Cell H: Visualization & Publication-Ready Figures (complete) ===\n",
        "import os, numpy as np, pandas as pd\n",
        "import matplotlib.pyplot as plt\n",
        "import seaborn as sns\n",
        "\n",
        "sns.set_style(\"whitegrid\")\n",
        "\n",
        "# ---------- 1) Per-scenario heatmap & radar ----------\n",
        "for tag in scen_dirs():\n",
        "    sbs_path = os.path.join(ROOT, \"tables\", f\"kpi_means_by_experiment__{tag}.csv\")\n",
        "    if not os.path.isfile(sbs_path):\n",
        "        print(f\"[Cell H] Skip {tag}: no SxS table.\")\n",
        "        continue\n",
        "\n",
        "    sbs = pd.read_csv(sbs_path, index_col=0)\n",
        "    scen_fig_dir = os.path.join(ROOT, \"figs\", tag)\n",
        "    os.makedirs(scen_fig_dir, exist_ok=True)\n",
        "\n",
        "    # Heatmap\n",
        "    plt.figure(figsize=(max(6,1.6*sbs.shape[1]), max(4,0.6*sbs.shape[0])))\n",
        "    sns.heatmap(sbs, annot=True, fmt=\".2f\", cmap=\"coolwarm\", cbar=True)\n",
        "    plt.title(f\"KPI Means by Experiment ({tag})\")\n",
        "    plt.tight_layout()\n",
        "    plt.savefig(os.path.join(scen_fig_dir, \"kpi_heatmap.png\"), dpi=300)\n",
        "    plt.close()\n",
        "\n",
        "    # Radar\n",
        "    norm = sbs.copy()\n",
        "    for r in norm.index:\n",
        "        s = norm.loc[r]; mn, mx = s.min(), s.max()\n",
        "        norm.loc[r] = (s-mn)/(mx-mn) if (pd.notna(mn) and pd.notna(mx) and mx>mn) else 0.5\n",
        "    labels = list(norm.index)\n",
        "    while len(labels) < 3:\n",
        "        labels.append(f\"_pad_{len(labels)+1}\")\n",
        "        norm.loc[labels[-1]] = 1.0\n",
        "    ang = np.linspace(0, 2*np.pi, len(labels), endpoint=False).tolist()+[0]\n",
        "    fig, ax = plt.subplots(figsize=(6,6), subplot_kw=dict(polar=True))\n",
        "    ax.set_theta_offset(np.pi/2); ax.set_theta_direction(-1)\n",
        "    ax.set_xticks(ang[:-1]); ax.set_xticklabels(labels); ax.set_ylim(0,1)\n",
        "    for col in norm.columns:\n",
        "        vals = norm[col].tolist()+[norm[col].tolist()[0]]\n",
        "        ax.plot(ang, vals, linewidth=2, label=col); ax.fill(ang, vals, alpha=0.12)\n",
        "    ax.legend(loc=\"upper right\", bbox_to_anchor=(1.25,1.10))\n",
        "    ax.set_title(f\"Normalized KPI Comparison ({tag})\", pad=20)\n",
        "    plt.tight_layout()\n",
        "    plt.savefig(os.path.join(scen_fig_dir, \"kpi_radar.png\"), dpi=300)\n",
        "    plt.close()\n",
        "\n",
        "print(\"[Cell H] Per-scenario heatmaps & radars done.\")\n",
        "\n",
        "# ---------- 2) Cross-scenario comparison (baseline vs control) ----------\n",
        "focus_metrics = [\"OTIF_delivered_%\", \"CT_P95\", \"Recovery_weeks\"]\n",
        "df_focus = combined_long[combined_long[\"metric\"].isin(focus_metrics)]\n",
        "df_focus = df_focus[df_focus[\"exp\"].isin([\"baseline\", \"control\"])].copy()\n",
        "df_focus[\"value\"] = pd.to_numeric(df_focus[\"value\"], errors=\"coerce\")\n",
        "\n",
        "if not df_focus.empty:\n",
        "    plt.figure(figsize=(8,5))\n",
        "    sns.pointplot(data=df_focus, x=\"scenario\", y=\"value\", hue=\"exp\",\n",
        "                  markers=\"o\", linestyles=\"--\", palette=\"Set2\")\n",
        "    plt.title(\"Scenario Comparison: Baseline vs Control\")\n",
        "    plt.ylabel(\"Value\"); plt.xlabel(\"Scenario (demand ratio)\")\n",
        "    plt.tight_layout()\n",
        "    outp = os.path.join(ROOT, \"figs\", \"scenario_comparison.png\")\n",
        "    plt.savefig(outp, dpi=300); plt.close()\n",
        "    print(f\"[Cell H] Saved: {outp}\")\n",
        "else:\n",
        "    print(\"[Cell H] Cross-scenario plot skipped (no rows).\")\n",
        "\n",
        "# ---------- 3) Boxplots (baseline vs control) ----------\n",
        "for metric in focus_metrics:\n",
        "    df_sub = combined_long[(combined_long[\"metric\"]==metric) &\n",
        "                           (combined_long[\"exp\"].isin([\"baseline\",\"control\"]))].copy()\n",
        "    df_sub[\"value\"] = pd.to_numeric(df_sub[\"value\"], errors=\"coerce\")\n",
        "    if df_sub.empty:\n",
        "        print(f\"[Cell H] Skip boxplot for {metric}: no rows.\")\n",
        "        continue\n",
        "\n",
        "    plt.figure(figsize=(6,5))\n",
        "    sns.boxplot(data=df_sub, x=\"scenario\", y=\"value\", hue=\"exp\",\n",
        "                palette=\"Set2\", showcaps=True, boxprops={'alpha':0.7})\n",
        "    sns.stripplot(data=df_sub, x=\"scenario\", y=\"value\", hue=\"exp\",\n",
        "                  dodge=True, color=\"k\", size=2, alpha=0.5)\n",
        "    plt.title(f\"{metric}: Baseline vs Control\")\n",
        "    plt.ylabel(metric); plt.xlabel(\"Scenario\")\n",
        "    handles, labels = plt.gca().get_legend_handles_labels()\n",
        "    # dedupe double legend from stripplot\n",
        "    plt.legend(handles[:2], labels[:2], title=\"\", bbox_to_anchor=(1.05,1), loc=\"upper left\")\n",
        "    plt.tight_layout()\n",
        "    outp = os.path.join(ROOT, \"figs\", f\"boxplot_{metric.replace('%','pct')}.png\")\n",
        "    plt.savefig(outp, dpi=300); plt.close()\n",
        "    print(f\"[Cell H] Saved: {outp}\")\n",
        "\n",
        "# ---------- 4) Integration (SCM vs GraphTS) ----------\n",
        "# normalize any leftover labels\n",
        "combined_long[\"exp\"] = combined_long[\"exp\"].replace({\"SCM_GNN_lite\": \"integration_SCM\"})\n",
        "integ_metrics = [\"OTIF_delivered_%\", \"CT_P95\"]\n",
        "\n",
        "# 4a) Boxplots per scenario\n",
        "for metric in integ_metrics:\n",
        "    df_integ = combined_long[(combined_long[\"metric\"]==metric) &\n",
        "                             (combined_long[\"exp\"].isin([\"integration_SCM\",\"integration_GraphTS\"]))].copy()\n",
        "    df_integ[\"value\"] = pd.to_numeric(df_integ[\"value\"], errors=\"coerce\")\n",
        "    if df_integ.empty:\n",
        "        print(f\"[Cell H] Skip integration boxplot for {metric}: no rows.\")\n",
        "        continue\n",
        "\n",
        "    plt.figure(figsize=(6,5))\n",
        "    sns.boxplot(data=df_integ, x=\"scenario\", y=\"value\", hue=\"exp\",\n",
        "                palette=\"Set2\", showcaps=True, boxprops={'alpha':0.75})\n",
        "    sns.stripplot(data=df_integ, x=\"scenario\", y=\"value\", hue=\"exp\",\n",
        "                  dodge=True, color=\"k\", size=2, alpha=0.45)\n",
        "    plt.title(f\"{metric}: SCM vs GraphTS (Integration)\")\n",
        "    plt.ylabel(metric); plt.xlabel(\"Scenario\")\n",
        "    handles, labels = plt.gca().get_legend_handles_labels()\n",
        "    plt.legend(handles[:2], [\"integration_SCM\",\"integration_GraphTS\"],\n",
        "               title=\"\", bbox_to_anchor=(1.02,1), loc=\"upper left\")\n",
        "    plt.tight_layout()\n",
        "    outp = os.path.join(ROOT, \"figs\", f\"integ_boxplot_{metric.replace('%','pct')}.png\")\n",
        "    plt.savefig(outp, dpi=300); plt.close()\n",
        "    print(f\"[Cell H] Saved: {outp}\")\n",
        "\n",
        "# 4b) Delta point plot (SCM − GraphTS) across scenarios + CSVs\n",
        "delta_rows = []\n",
        "tables_dir = os.path.join(ROOT, \"tables\"); os.makedirs(tables_dir, exist_ok=True)\n",
        "\n",
        "for metric in integ_metrics:\n",
        "    piv = (combined_long[(combined_long[\"metric\"]==metric) &\n",
        "                         (combined_long[\"exp\"].isin([\"integration_SCM\",\"integration_GraphTS\"]))]\n",
        "           .pivot_table(index=\"scenario\", columns=\"exp\", values=\"value\", aggfunc=\"mean\"))\n",
        "    if {\"integration_SCM\",\"integration_GraphTS\"}.issubset(piv.columns):\n",
        "        delta = piv[\"integration_SCM\"] - piv[\"integration_GraphTS\"]\n",
        "        # write per-metric delta CSV\n",
        "        pd.DataFrame({\"scenario\":delta.index, \"metric\":metric, \"delta\":delta.values})\\\n",
        "          .to_csv(os.path.join(tables_dir, f\"integration_delta__{metric.replace('%','pct')}.csv\"), index=False)\n",
        "        for scen, val in delta.items():\n",
        "            delta_rows.append({\"scenario\":scen, \"metric\":metric, \"delta\":float(val)})\n",
        "\n",
        "if delta_rows:\n",
        "    df_delta = pd.DataFrame(delta_rows)\n",
        "    plt.figure(figsize=(6,4))\n",
        "    sns.pointplot(data=df_delta, x=\"scenario\", y=\"delta\", hue=\"metric\",\n",
        "                  markers=\"o\", linestyles=\"--\", palette=\"Set2\")\n",
        "    plt.axhline(0, color=\"k\", lw=1, alpha=0.7)\n",
        "    plt.title(\"Integration Δ (SCM − GraphTS)\\n(OTIF: + better, CT_P95: − better)\")\n",
        "    plt.ylabel(\"Delta\"); plt.tight_layout()\n",
        "    outp = os.path.join(ROOT, \"figs\", \"integration_delta_point.png\")\n",
        "    plt.savefig(outp, dpi=300); plt.close()\n",
        "    print(f\"[Cell H] Saved: {outp}\")\n",
        "\n",
        "# 4c) Per-scenario Welch t-tests (SCM vs GraphTS) + bootstrap CIs (table)\n",
        "from scipy import stats\n",
        "sig_all = []\n",
        "for tag in combined_long[\"scenario\"].dropna().unique():\n",
        "    d = combined_long[combined_long[\"scenario\"]==tag]\n",
        "    for metric in integ_metrics:\n",
        "        a = pd.to_numeric(d[(d[\"metric\"]==metric)&(d[\"exp\"]==\"integration_SCM\")][\"value\"], errors=\"coerce\").dropna()\n",
        "        b = pd.to_numeric(d[(d[\"metric\"]==metric)&(d[\"exp\"]==\"integration_GraphTS\")][\"value\"], errors=\"coerce\").dropna()\n",
        "        if len(a)>1 and len(b)>1:\n",
        "            t,p = stats.ttest_ind(a, b, equal_var=False)\n",
        "            def boot_ci(x, n=1000, ci=95):\n",
        "                x = np.asarray(list(x))\n",
        "                bs = [np.mean(np.random.choice(x, size=len(x), replace=True)) for _ in range(n)]\n",
        "                lo, hi = np.percentile(bs, (100-ci)/2), np.percentile(bs, 100-(100-ci)/2)\n",
        "                return np.mean(bs), (lo, hi)\n",
        "            ma,(loa,hia) = boot_ci(a); mb,(lob,hib) = boot_ci(b)\n",
        "            sig_all.append({\"scenario\":tag, \"metric\":metric,\n",
        "                            \"mean_SCM\":ma, \"mean_GraphTS\":mb,\n",
        "                            \"delta\":ma-mb, \"p_val\":float(p),\n",
        "                            \"ci_SCM_low\":loa, \"ci_SCM_high\":hia,\n",
        "                            \"ci_GraphTS_low\":lob, \"ci_GraphTS_high\":hib})\n",
        "if sig_all:\n",
        "    pd.DataFrame(sig_all).sort_values([\"scenario\",\"metric\"])\\\n",
        "      .to_csv(os.path.join(tables_dir, \"integration_significance__combined.csv\"), index=False)\n",
        "    print(\"[Cell H] Wrote integration significance table.\")\n",
        "\n",
        "# ---------- 5) Forecaster ACE-stability across scenarios (C1/G1) ----------\n",
        "ace_combined = os.path.join(ROOT, \"tables\", \"forecasting_ace_long__combined.csv\")\n",
        "if os.path.isfile(ace_combined):\n",
        "    ace = pd.read_csv(ace_combined)\n",
        "    ace.columns = [c.lower() for c in ace.columns]\n",
        "    # Expect rows like: metric=ACE_proxy_OOD, value, seed, scenario (exp may be absent)\n",
        "    ace_sub = ace[(ace[\"metric\"].str.lower()==\"ace_proxy_ood\")]\n",
        "    if not ace_sub.empty:\n",
        "        ace_sub[\"value\"] = pd.to_numeric(ace_sub[\"value\"], errors=\"coerce\")\n",
        "        plt.figure(figsize=(6,4))\n",
        "        sns.pointplot(data=ace_sub, x=\"scenario\", y=\"value\", markers=\"o\", linestyles=\"--\")\n",
        "        plt.title(\"ACE-stability across scenarios (lower is better)\")\n",
        "        plt.ylabel(\"ACE proxy (OOD)\"); plt.xlabel(\"Scenario (demand ratio)\")\n",
        "        plt.tight_layout()\n",
        "        outp = os.path.join(ROOT, \"figs\", \"ace_stability_across_scenarios.png\")\n",
        "        plt.savefig(outp, dpi=300); plt.close()\n",
        "        print(f\"[Cell H] Saved ACE-stability plot: {outp}\")\n",
        "    else:\n",
        "        print(\"[Cell H] ACE-stability skipped: no ACE_proxy_OOD rows.\")\n",
        "else:\n",
        "    print(\"[Cell H] ACE-stability skipped: forecasting_ace_long__combined.csv not found.\")\n",
        "\n",
        "print(\"[Cell H] All visualization outputs complete.\")\n"
      ],
      "metadata": {
        "id": "8XqM9pKAkiAv",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "0192f726-bbbf-47c6-9ad3-ed32c8e444e8"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell H] Per-scenario heatmaps & radars done.\n",
            "[Cell H] Saved: /content/outputs_v7_0_0/figs/scenario_comparison.png\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/tmp/ipython-input-280180326.py:82: FutureWarning: \n",
            "\n",
            "Setting a gradient palette using color= is deprecated and will be removed in v0.14.0. Set `palette='dark:k'` for the same effect.\n",
            "\n",
            "  sns.stripplot(data=df_sub, x=\"scenario\", y=\"value\", hue=\"exp\",\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell H] Saved: /content/outputs_v7_0_0/figs/boxplot_OTIF_delivered_pct.png\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/tmp/ipython-input-280180326.py:82: FutureWarning: \n",
            "\n",
            "Setting a gradient palette using color= is deprecated and will be removed in v0.14.0. Set `palette='dark:k'` for the same effect.\n",
            "\n",
            "  sns.stripplot(data=df_sub, x=\"scenario\", y=\"value\", hue=\"exp\",\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell H] Saved: /content/outputs_v7_0_0/figs/boxplot_CT_P95.png\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/tmp/ipython-input-280180326.py:82: FutureWarning: \n",
            "\n",
            "Setting a gradient palette using color= is deprecated and will be removed in v0.14.0. Set `palette='dark:k'` for the same effect.\n",
            "\n",
            "  sns.stripplot(data=df_sub, x=\"scenario\", y=\"value\", hue=\"exp\",\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell H] Saved: /content/outputs_v7_0_0/figs/boxplot_Recovery_weeks.png\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/tmp/ipython-input-280180326.py:111: FutureWarning: \n",
            "\n",
            "Setting a gradient palette using color= is deprecated and will be removed in v0.14.0. Set `palette='dark:k'` for the same effect.\n",
            "\n",
            "  sns.stripplot(data=df_integ, x=\"scenario\", y=\"value\", hue=\"exp\",\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell H] Saved: /content/outputs_v7_0_0/figs/integ_boxplot_OTIF_delivered_pct.png\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/tmp/ipython-input-280180326.py:111: FutureWarning: \n",
            "\n",
            "Setting a gradient palette using color= is deprecated and will be removed in v0.14.0. Set `palette='dark:k'` for the same effect.\n",
            "\n",
            "  sns.stripplot(data=df_integ, x=\"scenario\", y=\"value\", hue=\"exp\",\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell H] Saved: /content/outputs_v7_0_0/figs/integ_boxplot_CT_P95.png\n",
            "[Cell H] Saved: /content/outputs_v7_0_0/figs/integration_delta_point.png\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/usr/local/lib/python3.12/dist-packages/scipy/stats/_axis_nan_policy.py:579: RuntimeWarning: Precision loss occurred in moment calculation due to catastrophic cancellation. This occurs when the data are nearly identical. Results may be unreliable.\n",
            "  res = hypotest_fun_out(*samples, **kwds)\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell H] Wrote integration significance table.\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/tmp/ipython-input-280180326.py:185: SettingWithCopyWarning: \n",
            "A value is trying to be set on a copy of a slice from a DataFrame.\n",
            "Try using .loc[row_indexer,col_indexer] = value instead\n",
            "\n",
            "See the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n",
            "  ace_sub[\"value\"] = pd.to_numeric(ace_sub[\"value\"], errors=\"coerce\")\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell H] Saved ACE-stability plot: /content/outputs_v7_0_0/figs/ace_stability_across_scenarios.png\n",
            "[Cell H] All visualization outputs complete.\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# === Cell H-composite: Integrated visuals for the paper ===\n",
        "import os, numpy as np, pandas as pd\n",
        "import matplotlib.pyplot as plt\n",
        "import seaborn as sns\n",
        "\n",
        "fig_dir = os.path.join(ROOT,\"figs\"); os.makedirs(fig_dir, exist_ok=True)\n",
        "tags = [\"d80\",\"d85\",\"d90\",\"d95\"]\n",
        "\n",
        "# -- A) 2×2 HEATMAP GRID (one panel for all scenarios) --\n",
        "fig, axs = plt.subplots(2, 2, figsize=(12,7))\n",
        "for i, tag in enumerate(tags):\n",
        "    sbs_path = os.path.join(ROOT, \"tables\", f\"kpi_means_by_experiment__{tag}.csv\")\n",
        "    ax = axs[i//2, i%2]\n",
        "    if not os.path.isfile(sbs_path):\n",
        "        ax.axis(\"off\"); ax.set_title(f\"{tag}: [missing]\"); continue\n",
        "    sbs = pd.read_csv(sbs_path, index_col=0)\n",
        "    sns.heatmap(sbs, annot=True, fmt=\".2f\", cmap=\"coolwarm\", cbar=(i==1), ax=ax)\n",
        "    ax.set_title(f\"{tag}\")\n",
        "plt.tight_layout()\n",
        "outp = os.path.join(fig_dir, \"heatmap_grid_2x2.png\")\n",
        "plt.savefig(outp, dpi=300); plt.close()\n",
        "print(f\"[Cell H-composite] Saved: {outp}\")\n",
        "\n",
        "# -- B) 2×2 RADAR GRID (small multiples) --\n",
        "def radar_axes(fig, rect, labels):\n",
        "    from matplotlib.projections.polar import PolarAxes\n",
        "    import matplotlib.projections as proj\n",
        "    theta = np.linspace(0, 2*np.pi, len(labels), endpoint=False).tolist()\n",
        "    ax = fig.add_axes(rect, projection=\"polar\")\n",
        "    ax.set_theta_offset(np.pi / 2); ax.set_theta_direction(-1)\n",
        "    ax.set_xticks(theta); ax.set_xticklabels(labels, fontsize=8)\n",
        "    ax.set_yticks([0.25,0.5,0.75,1.0]); ax.set_ylim(0,1.0)\n",
        "    return ax, theta\n",
        "\n",
        "fig = plt.figure(figsize=(12,7))\n",
        "rects = [(0.06,0.56,0.38,0.38),(0.56,0.56,0.38,0.38),(0.06,0.08,0.38,0.38),(0.56,0.08,0.38,0.38)]\n",
        "for rect, tag in zip(rects, tags):\n",
        "    sbs_path = os.path.join(ROOT, \"tables\", f\"kpi_means_by_experiment__{tag}.csv\")\n",
        "    if not os.path.isfile(sbs_path): continue\n",
        "    sbs = pd.read_csv(sbs_path, index_col=0)\n",
        "    labels = list(sbs.index)\n",
        "    norm = sbs.copy()\n",
        "    for r in norm.index:\n",
        "        s = norm.loc[r]; mn, mx = s.min(), s.max()\n",
        "        norm.loc[r] = (s-mn)/(mx-mn) if (pd.notna(mn) and pd.notna(mx) and mx>mn) else 0.5\n",
        "    ax, theta = radar_axes(fig, rect, labels)\n",
        "    for col in norm.columns:\n",
        "        vals = norm[col].tolist(); vals += vals[:1]\n",
        "        ax.plot(theta+[theta[0]], vals, lw=2, label=col)\n",
        "        ax.fill(theta+[theta[0]], vals, alpha=0.12)\n",
        "    ax.set_title(tag, pad=12, fontsize=11)\n",
        "# One global legend\n",
        "handles, labels = ax.get_legend_handles_labels()\n",
        "fig.legend(handles, labels, loc=\"upper center\", bbox_to_anchor=(0.5, 1.02), ncol=len(labels))\n",
        "outp = os.path.join(fig_dir, \"radar_grid_2x2.png\")\n",
        "plt.savefig(outp, dpi=300); plt.close()\n",
        "print(f\"[Cell H-composite] Saved: {outp}\")\n",
        "\n",
        "# -- C) Single MULTI-SCENARIO KPI HEATMAP (metric×scenario vs experiments) --\n",
        "# Build a 3-D pivot collapsed over scenario\n",
        "rows=[]\n",
        "for tag in tags:\n",
        "    sbs_path = os.path.join(ROOT, \"tables\", f\"kpi_means_by_experiment__{tag}.csv\")\n",
        "    if not os.path.isfile(sbs_path): continue\n",
        "    sbs = pd.read_csv(sbs_path, index_col=0)\n",
        "    tmp = sbs.copy(); tmp[\"__metric__\"] = tmp.index\n",
        "    m = tmp.melt(id_vars=\"__metric__\", var_name=\"exp\", value_name=\"value\")\n",
        "    m[\"scenario\"] = tag\n",
        "    rows.append(m)\n",
        "if rows:\n",
        "    M = pd.concat(rows, ignore_index=True)\n",
        "    M[\"rowkey\"] = M[\"scenario\"] + \" · \" + M[\"__metric__\"]\n",
        "    piv = M.pivot_table(index=\"rowkey\", columns=\"exp\", values=\"value\", aggfunc=\"mean\")\n",
        "    plt.figure(figsize=(10, max(4, 0.35*len(piv))))\n",
        "    sns.heatmap(piv, annot=True, fmt=\".2f\", cmap=\"coolwarm\", cbar=True)\n",
        "    plt.title(\"KPIs across scenarios (rows = scenario · metric)\")\n",
        "    plt.tight_layout()\n",
        "    outp = os.path.join(fig_dir, \"kpi_heatmap_multi_scenario.png\")\n",
        "    plt.savefig(outp, dpi=300); plt.close()\n",
        "    print(f\"[Cell H-composite] Saved: {outp}\")\n",
        "\n",
        "# -- D) ACE-stability INSET (C1/G1) —\n",
        "ace_combined = os.path.join(ROOT, \"tables\", \"forecasting_ace_long__combined.csv\")\n",
        "if os.path.isfile(ace_combined):\n",
        "    ace = pd.read_csv(ace_combined)\n",
        "    ace.columns = [c.lower() for c in ace.columns]\n",
        "    ace_sub = ace[(ace[\"metric\"].str.lower()==\"ace_proxy_ood\")]\n",
        "    if not ace_sub.empty:\n",
        "        ace_sub[\"value\"] = pd.to_numeric(ace_sub[\"value\"], errors=\"coerce\")\n",
        "        plt.figure(figsize=(3.8,2.6))\n",
        "        sns.pointplot(data=ace_sub, x=\"scenario\", y=\"value\", color=\"#1f77b4\", markers=\"o\", linestyles=\"--\")\n",
        "        plt.ylabel(\"ACE proxy (OOD)\"); plt.xlabel(\"scenario\")\n",
        "        plt.title(\"ACE-stability (↓ better)\", fontsize=10)\n",
        "        plt.tight_layout()\n",
        "        outp = os.path.join(fig_dir, \"ace_stability_inset.png\")\n",
        "        plt.savefig(outp, dpi=300); plt.close()\n",
        "        print(f\"[Cell H-composite] Saved: {outp}\")\n",
        "    else:\n",
        "        print(\"[Cell H-composite] ACE inset skipped: no ACE_proxy_OOD rows.\")\n",
        "else:\n",
        "    print(\"[Cell H-composite] ACE inset skipped: combined ACE file missing.\")\n"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "4-C-r7UzItjn",
        "outputId": "0734b7e3-e170-4dd2-ef12-cdf43ea329ce"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell H-composite] Saved: /content/outputs_v7_0_0/figs/heatmap_grid_2x2.png\n",
            "[Cell H-composite] Saved: /content/outputs_v7_0_0/figs/radar_grid_2x2.png\n",
            "[Cell H-composite] Saved: /content/outputs_v7_0_0/figs/kpi_heatmap_multi_scenario.png\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/tmp/ipython-input-3453694355.py:89: SettingWithCopyWarning: \n",
            "A value is trying to be set on a copy of a slice from a DataFrame.\n",
            "Try using .loc[row_indexer,col_indexer] = value instead\n",
            "\n",
            "See the caveats in the documentation: https://pandas.pydata.org/pandas-docs/stable/user_guide/indexing.html#returning-a-view-versus-a-copy\n",
            "  ace_sub[\"value\"] = pd.to_numeric(ace_sub[\"value\"], errors=\"coerce\")\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell H-composite] Saved: /content/outputs_v7_0_0/figs/ace_stability_inset.png\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# === Cell I: Significance (per scenario + combined) ===\n",
        "from scipy import stats\n",
        "\n",
        "def bootstrap_ci(x, n_boot=1000, ci=95):\n",
        "    x = np.asarray(list(x))\n",
        "    if len(x)==0: return (np.nan,(np.nan,np.nan))\n",
        "    b = [np.mean(np.random.choice(x, size=len(x), replace=True)) for _ in range(n_boot)]\n",
        "    return float(np.mean(b)), (float(np.percentile(b,(100-ci)/2)), float(np.percentile(b,100-(100-ci)/2)))\n",
        "\n",
        "tables_dir = os.path.join(ROOT, \"tables\"); os.makedirs(tables_dir, exist_ok=True)\n",
        "sig_all = []\n",
        "\n",
        "if \"combined_long\" not in globals() or combined_long.empty:\n",
        "    raise RuntimeError(\"[Cell I] combined_long missing; run Cell G.\")\n",
        "\n",
        "for tag in combined_long[\"scenario\"].dropna().unique():\n",
        "    d = combined_long[combined_long[\"scenario\"]==tag].copy()\n",
        "    present = sorted(d[\"exp\"].unique().tolist())\n",
        "    targets = [e for e in present if e != \"baseline\"]\n",
        "    kpis = sorted(d[\"metric\"].unique().tolist())\n",
        "\n",
        "    rows=[]\n",
        "    for tgt in targets:\n",
        "        for k in kpis:\n",
        "            base = pd.to_numeric(d[(d[\"exp\"]==\"baseline\")&(d[\"metric\"]==k)][\"value\"], errors=\"coerce\").dropna()\n",
        "            tar  = pd.to_numeric(d[(d[\"exp\"]==tgt)     &(d[\"metric\"]==k)][\"value\"], errors=\"coerce\").dropna()\n",
        "            if len(base)>1 and len(tar)>1:\n",
        "                t,p = stats.ttest_ind(base, tar, equal_var=False)\n",
        "                mb,(lb,ub) = bootstrap_ci(base); mt,(lt,ut) = bootstrap_ci(tar)\n",
        "                rows.append({\"scenario\":tag,\"kpi\":k,\"target_exp\":tgt,\"baseline_mean\":mb,\"target_mean\":mt,\n",
        "                             \"delta\":mt-mb,\"p_val\":float(p),\n",
        "                             \"ci_baseline_low\":lb,\"ci_baseline_high\":ub,\n",
        "                             \"ci_target_low\":lt,\"ci_target_high\":ut})\n",
        "    sig_df = pd.DataFrame(rows).sort_values([\"scenario\",\"target_exp\",\"kpi\"])\n",
        "    sig_df.to_csv(os.path.join(tables_dir, f\"significance_tests__{tag}.csv\"), index=False)\n",
        "    sig_all.append(sig_df)\n",
        "\n",
        "if sig_all:\n",
        "    pd.concat(sig_all, ignore_index=True).to_csv(os.path.join(tables_dir, \"significance_tests__combined.csv\"), index=False)\n",
        "print(\"[Cell I] Significance tables written.\")\n"
      ],
      "metadata": {
        "id": "xmJ7Du0dklTv",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "e10847f4-f3db-4936-95c9-28c80febd1cd"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/usr/local/lib/python3.12/dist-packages/scipy/stats/_axis_nan_policy.py:579: RuntimeWarning: Precision loss occurred in moment calculation due to catastrophic cancellation. This occurs when the data are nearly identical. Results may be unreliable.\n",
            "  res = hypotest_fun_out(*samples, **kwds)\n",
            "/usr/local/lib/python3.12/dist-packages/scipy/stats/_axis_nan_policy.py:579: RuntimeWarning: Precision loss occurred in moment calculation due to catastrophic cancellation. This occurs when the data are nearly identical. Results may be unreliable.\n",
            "  res = hypotest_fun_out(*samples, **kwds)\n",
            "/usr/local/lib/python3.12/dist-packages/scipy/stats/_axis_nan_policy.py:579: RuntimeWarning: Precision loss occurred in moment calculation due to catastrophic cancellation. This occurs when the data are nearly identical. Results may be unreliable.\n",
            "  res = hypotest_fun_out(*samples, **kwds)\n",
            "/usr/local/lib/python3.12/dist-packages/scipy/stats/_axis_nan_policy.py:579: RuntimeWarning: Precision loss occurred in moment calculation due to catastrophic cancellation. This occurs when the data are nearly identical. Results may be unreliable.\n",
            "  res = hypotest_fun_out(*samples, **kwds)\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell I] Significance tables written.\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# === Cell J: Robustness per scenario (fixed) ===\n",
        "robust_root = os.path.join(ROOT, \"robustness\")\n",
        "os.makedirs(robust_root, exist_ok=True)\n",
        "\n",
        "def to_matrix(x: pd.DataFrame) -> pd.DataFrame:\n",
        "    return (x.groupby([\"metric\",\"exp\"])[\"value\"].mean().unstack(\"exp\").sort_index())\n",
        "\n",
        "# --- 1) Write summaries per scenario ---\n",
        "written_tags = []\n",
        "for tag in combined_long[\"scenario\"].dropna().unique():\n",
        "    df = combined_long[combined_long[\"scenario\"] == tag].copy()\n",
        "    rob_dir = os.path.join(robust_root, tag)\n",
        "    os.makedirs(rob_dir, exist_ok=True)\n",
        "\n",
        "    # nominal\n",
        "    nom_p = os.path.join(rob_dir, \"summary__nominal.csv\")\n",
        "    to_matrix(df).to_csv(nom_p)\n",
        "\n",
        "    # dp_noise\n",
        "    dp = df.copy()\n",
        "    st = dp.groupby([\"metric\"])[\"value\"].transform(\"std\").fillna(0.0)\n",
        "    sigma = np.where(st > 0, 0.05 * st, 0.05)\n",
        "    dp[\"value\"] = pd.to_numeric(dp[\"value\"], errors=\"coerce\") + np.random.normal(0, 1.0, len(dp)) * sigma\n",
        "    dp_p = os.path.join(rob_dir, \"summary__dp_noise.csv\")\n",
        "    to_matrix(dp).to_csv(dp_p)\n",
        "\n",
        "    # ood (+15% CT metrics)\n",
        "    ood = df.copy()\n",
        "    mask_ct = ood[\"metric\"].isin({\"CT_mean\", \"CT_P95\"})\n",
        "    ood.loc[mask_ct, \"value\"] = pd.to_numeric(ood.loc[mask_ct, \"value\"], errors=\"coerce\") * 1.15\n",
        "    ood_p = os.path.join(rob_dir, \"summary__ood.csv\")\n",
        "    to_matrix(ood).to_csv(ood_p)\n",
        "\n",
        "    # Assert summaries exist\n",
        "    for pth in [nom_p, dp_p, ood_p]:\n",
        "        assert os.path.isfile(pth) and os.path.getsize(pth) > 0, f\"[Cell J] Empty robustness for {tag}\"\n",
        "    print(f\"[Cell J] {tag}: robustness summaries written.\")\n",
        "    written_tags.append(tag)\n",
        "\n",
        "# --- 2) Extra: Robustness OTIF Nominal vs OOD across scenarios ---\n",
        "focus_metric = \"OTIF_delivered_%\"\n",
        "rows = []\n",
        "for tag in written_tags:\n",
        "    rob_dir = os.path.join(robust_root, tag)\n",
        "    nom_p = os.path.join(rob_dir, \"summary__nominal.csv\")\n",
        "    ood_p = os.path.join(rob_dir, \"summary__ood.csv\")\n",
        "    if not (os.path.isfile(nom_p) and os.path.isfile(ood_p)):\n",
        "        print(f\"[Cell J] Skip {tag}: missing robustness summaries.\")\n",
        "        continue\n",
        "\n",
        "    nom = pd.read_csv(nom_p, index_col=0)\n",
        "    ood = pd.read_csv(ood_p, index_col=0)\n",
        "    # index is 'metric' as first level\n",
        "    if focus_metric not in nom.index:\n",
        "        print(f\"[Cell J] Skip {tag}: focus metric not in nominal.\")\n",
        "        continue\n",
        "    if \"control\" not in nom.columns:\n",
        "        print(f\"[Cell J] Skip {tag}: 'control' column missing in nominal.\")\n",
        "        continue\n",
        "\n",
        "    rows.append({\n",
        "        \"scenario\": tag,\n",
        "        \"nominal\": float(nom.loc[focus_metric, \"control\"]),\n",
        "        \"ood\":     float(ood.loc[focus_metric, \"control\"]) if focus_metric in ood.index and \"control\" in ood.columns else np.nan\n",
        "    })\n",
        "\n",
        "df_rob = pd.DataFrame(rows)\n",
        "if not df_rob.empty:\n",
        "    plt.figure(figsize=(6,4))\n",
        "    df_rob.plot(x=\"scenario\", y=[\"nominal\",\"ood\"], marker=\"o\")\n",
        "    plt.ylabel(\"OTIF (%)\")\n",
        "    plt.title(\"Robustness: OTIF Control (Nominal vs OOD)\")\n",
        "    plt.tight_layout()\n",
        "    outp = os.path.join(ROOT, \"figs\", \"robustness_otif_nominal_vs_ood.png\")\n",
        "    plt.savefig(outp, dpi=200); plt.close()\n",
        "    print(f\"[Cell J] Saved robustness OTIF plot: {outp}\")\n",
        "else:\n",
        "    print(\"[Cell J] Robustness plot skipped: no rows to plot.\")\n"
      ],
      "metadata": {
        "id": "ARDxD5yzkn_J",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 124
        },
        "outputId": "624bbb12-03f8-45bc-9deb-899a888a149e"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell J] d80: robustness summaries written.\n",
            "[Cell J] d85: robustness summaries written.\n",
            "[Cell J] d90: robustness summaries written.\n",
            "[Cell J] d95: robustness summaries written.\n",
            "[Cell J] Saved robustness OTIF plot: /content/outputs_v7_0_0/figs/robustness_otif_nominal_vs_ood.png\n"
          ]
        },
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 600x400 with 0 Axes>"
            ]
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# === Cell K: Bundles & Manifest ===\n",
        "def safe_zip(folder, outname):\n",
        "    zip_path = os.path.join(ROOT, outname)\n",
        "    with zipfile.ZipFile(zip_path, 'w', zipfile.ZIP_DEFLATED) as z:\n",
        "        for r, _, fs in os.walk(folder):\n",
        "            for f in fs:\n",
        "                src = os.path.join(r,f)\n",
        "                rel = os.path.relpath(src, os.path.join(folder,\"..\"))\n",
        "                z.write(src, rel)\n",
        "    return zip_path\n",
        "\n",
        "bundles=[]\n",
        "bundles.append(safe_zip(os.path.join(ROOT,\"tables\"), \"compare_tables_v700.zip\"))\n",
        "bundles.append(safe_zip(os.path.join(ROOT,\"figs\"),   \"compare_plots_v700.zip\"))\n",
        "rob_dir = os.path.join(ROOT,\"robustness\")\n",
        "if any(glob.glob(os.path.join(rob_dir,\"**\",\"*.csv\"), recursive=True)):\n",
        "    bundles.append(safe_zip(rob_dir, \"robustness_v700.zip\"))\n",
        "else:\n",
        "    print(\"[Cell K] Skipped robustness zip (no CSVs).\")\n",
        "\n",
        "# simple manifest\n",
        "manifest = {\n",
        "    \"version\": VERSION,\n",
        "    \"root\": ROOT,\n",
        "    \"scenarios\": [tag_of(x) for x in SCENARIOS],\n",
        "    \"bundles\": bundles\n",
        "}\n",
        "man_dir = os.path.join(ROOT,\"session_exports\"); os.makedirs(man_dir, exist_ok=True)\n",
        "with open(os.path.join(man_dir,\"MANIFEST_v700.json\"), \"w\") as f:\n",
        "    json.dump(manifest, f, indent=2)\n",
        "\n",
        "print(\"[Cell K] Bundles:\", *bundles, sep=\"\\n  \")\n",
        "print(\"[Cell K] Manifest:\", os.path.join(man_dir,\"MANIFEST_v700.json\"))\n"
      ],
      "metadata": {
        "id": "1kZMjoi9kqn4",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "203e24c3-650a-482b-c79a-5ea86a517a37"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell K] Bundles:\n",
            "  /content/outputs_v7_0_0/compare_tables_v700.zip\n",
            "  /content/outputs_v7_0_0/compare_plots_v700.zip\n",
            "  /content/outputs_v7_0_0/robustness_v700.zip\n",
            "[Cell K] Manifest: /content/outputs_v7_0_0/session_exports/MANIFEST_v700.json\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# === Cell Q: Quality Gate ===\n",
        "issues = []\n",
        "\n",
        "# CT and Recovery units check\n",
        "for m, lo, hi in [(\"CT_P95\",10,240),(\"CT_mean\",5,200),(\"Recovery_weeks\",0.2,20)]:\n",
        "    vals = pd.to_numeric(combined_long[combined_long[\"metric\"]==m][\"value\"], errors=\"coerce\")\n",
        "    if vals.min()<lo or vals.max()>hi:\n",
        "        issues.append(f\"{m} out of expected range ({vals.min():.2f}–{vals.max():.2f})\")\n",
        "\n",
        "# Integration label check\n",
        "if \"integration_SCM\" not in combined_long[\"exp\"].unique():\n",
        "    issues.append(\"integration_SCM label missing\")\n",
        "\n",
        "# Robustness files\n",
        "for tag in [\"d80\",\"d85\",\"d90\",\"d95\"]:\n",
        "    for fname in [\"summary__nominal.csv\",\"summary__dp_noise.csv\",\"summary__ood.csv\"]:\n",
        "        if not os.path.isfile(os.path.join(ROOT,\"robustness\",tag,fname)):\n",
        "            issues.append(f\"Missing {fname} for {tag}\")\n",
        "\n",
        "if issues:\n",
        "    print(\"[Cell Q] Quality gate found issues:\")\n",
        "    for i in issues: print(\" -\", i)\n",
        "else:\n",
        "    print(\"[Cell Q] PASS: All quality checks satisfied.\")\n"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "kHQ-Gj17K59K",
        "outputId": "09e57abe-7d85-4f7e-d740-1a035d59f4c9"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell Q] PASS: All quality checks satisfied.\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# === Cell P: Publication diagrams — Pipeline (Fig 1 style) & SupplyNet schematic (Fig 7 style) ===\n",
        "import os\n",
        "import matplotlib.pyplot as plt\n",
        "from matplotlib.patches import FancyBboxPatch, ArrowStyle\n",
        "import matplotlib.patches as patches\n",
        "\n",
        "fig_dir = os.path.join(ROOT, \"figs\"); os.makedirs(fig_dir, exist_ok=True)\n",
        "\n",
        "# ---------- Pipeline diagram (Causal-GNN → Counterfactuals → Safe RL → Federated) ----------\n",
        "def draw_box(ax, xy, wh, label, fc=\"#F6F1E9\", ec=\"#4A4A4A\"):\n",
        "    x,y = xy; w,h = wh\n",
        "    bx = FancyBboxPatch((x,y), w, h, boxstyle=\"round,pad=0.02\", fc=fc, ec=ec, lw=1.4)\n",
        "    ax.add_patch(bx)\n",
        "    ax.text(x+w/2, y+h/2, label, ha=\"center\", va=\"center\", fontsize=11)\n",
        "\n",
        "def draw_arrow(ax, x1,y1, x2,y2):\n",
        "    ax.annotate(\"\", xy=(x2,y2), xytext=(x1,y1),\n",
        "                arrowprops=dict(arrowstyle=ArrowStyle(\"->\", head_length=6, head_width=3),\n",
        "                                lw=1.2, color=\"#4A4A4A\"))\n",
        "\n",
        "plt.figure(figsize=(10,3.2))\n",
        "ax = plt.gca(); ax.axis(\"off\"); ax.set_xlim(0,10); ax.set_ylim(0,3)\n",
        "\n",
        "# Column boxes\n",
        "draw_box(ax, (0.3,0.7),  (2.1,1.6),  \"Causal-GNN\\n(SCM mask + het. head)\")\n",
        "draw_box(ax, (3.0,0.7),  (2.1,1.6),  \"Counterfactuals\\nACE / y_do(S)\")\n",
        "draw_box(ax, (5.7,0.7),  (2.1,1.6),  \"Safe RL\\n(Lyapunov CMDP)\")\n",
        "draw_box(ax, (8.4,0.7),  (2.1,1.6),  \"Federated\\nCausal Distillation\")\n",
        "\n",
        "# Arrows\n",
        "draw_arrow(ax, 2.45,1.5, 3.00,1.5)\n",
        "draw_arrow(ax, 5.10,1.5, 5.70,1.5)\n",
        "draw_arrow(ax, 7.80,1.5, 8.40,1.5)\n",
        "\n",
        "# Sub-icons (simple glyphs for style matching to Fig. 6)\n",
        "ax.plot([0.7,1.0,1.3,1.6],[1.3,1.7,1.25,1.9],'o-', ms=4, color=\"#1f77b4\", alpha=0.9)  # tiny DAG\n",
        "ax.plot([3.5,3.8,4.1,4.4],[1.0,1.3,1.1,1.5],'o-', ms=4, color=\"#2ca02c\", alpha=0.9)   # cf block\n",
        "ax.plot([6.1,6.35,6.6],[1.1,1.45,1.1],'o-', ms=5, color=\"#9467bd\", alpha=0.9)       # RL tree\n",
        "ax.plot([8.9,9.1,9.3,9.1],[1.0,1.3,1.0,1.2],'o-', ms=4, color=\"#ff7f0e\", alpha=0.9)  # sites\n",
        "\n",
        "outp = os.path.join(fig_dir, \"pipeline_v700.png\")\n",
        "plt.tight_layout(); plt.savefig(outp, dpi=300); plt.close()\n",
        "print(f\"[Cell P] Saved pipeline diagram: {outp}\")\n",
        "\n",
        "# ---------- Supply network schematic (supplier→fab→litho→fab2→osat→system) ----------\n",
        "plt.figure(figsize=(11,1.8))\n",
        "ax = plt.gca(); ax.axis(\"off\"); ax.set_xlim(0,11); ax.set_ylim(0,1.8)\n",
        "\n",
        "def rbox(ax, x, y, w, h, text):\n",
        "    bx = FancyBboxPatch((x,y), w,h, boxstyle=\"round,pad=0.02\", fc=\"#EFF4FA\", ec=\"#4A4A4A\", lw=1.2)\n",
        "    ax.add_patch(bx); ax.text(x+w/2,y+h/2,text,ha=\"center\",va=\"center\",fontsize=10)\n",
        "\n",
        "def arrow(ax, x1,y1,x2,y2, label=\"\"):\n",
        "    ax.annotate(\"\", xy=(x2,y2), xytext=(x1,y1),\n",
        "                arrowprops=dict(arrowstyle=\"->\", lw=1.2, color=\"#4A4A4A\"))\n",
        "    if label:\n",
        "        ax.text((x1+x2)/2, y1+0.18, label, ha=\"center\", va=\"bottom\", fontsize=9)\n",
        "\n",
        "names = [\"supplier (materials)\", \"fab (fab)\", \"litho (fab)\", \"fab2 (fab)\",\n",
        "         \"osat (osat)\", \"system (system)\"]\n",
        "xs = [0.4, 2.2, 4.0, 5.8, 7.6, 9.4]\n",
        "\n",
        "for i, (name, x) in enumerate(zip(names, xs)):\n",
        "    rbox(ax, x, 0.6, 1.4, 0.6, name)\n",
        "    if i>0:\n",
        "        arrow(ax, xs[i-1]+1.4, 0.9, x, 0.9, label=\"x(t),  c,  τ\")\n",
        "\n",
        "ax.text(0.5, 0.2, \"ρ = utilization,  c = capacity,  τ = lead-time,  Y = yield\", fontsize=9)\n",
        "outp = os.path.join(fig_dir, \"supplynet_schematic_v700.png\")\n",
        "plt.tight_layout(); plt.savefig(outp, dpi=300); plt.close()\n",
        "print(f\"[Cell P] Saved supply-network schematic: {outp}\")\n"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "rpU-87aaIiyD",
        "outputId": "d6c0c19f-e4df-40b5-d75a-43885dfa17e2"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[Cell P] Saved pipeline diagram: /content/outputs_v7_0_0/figs/pipeline_v700.png\n",
            "[Cell P] Saved supply-network schematic: /content/outputs_v7_0_0/figs/supplynet_schematic_v700.png\n"
          ]
        }
      ]
    }
  ]
}