{
  "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": "5KrxAlvzxPKt"
      },
      "outputs": [],
      "source": [
        "import numpy as np, torch, matplotlib.pyplot as plt\n",
        "from sklearn.discriminant_analysis import LinearDiscriminantAnalysis as LDA\n",
        "from sklearn.metrics import roc_auc_score\n",
        "\n",
        "# ────────────────────────────────────────────────────────────\n",
        "# helpers\n",
        "# ────────────────────────────────────────────────────────────\n",
        "def vne_entropy(cov, beta=2.0):\n",
        "    M = torch.as_tensor(cov, dtype=torch.float64)\n",
        "    rho = torch.matrix_exp(-beta*M); rho /= rho.trace()\n",
        "    lam = torch.linalg.eigvalsh(rho).clamp_min(1e-12)\n",
        "    return float(-(lam*lam.log2()).sum())\n",
        "\n",
        "def naive_entropy(cov):\n",
        "    lam = np.linalg.eigvalsh(cov); lam /= lam.sum(); eps=1e-12\n",
        "    return -np.sum(lam*np.log2(lam+eps))\n",
        "\n",
        "# ────────────────────────────────────────────────────────────\n",
        "# 1)  two 3-D rank-2 Gaussians  (Σ₂ = 4 Σ₁)\n",
        "# ────────────────────────────────────────────────────────────\n",
        "lam1 = np.array([1.5, .5, 0.0])\n",
        "lam2 = lam1*[2,2,1.3]\n",
        "Σ1, Σ2 = np.diag(lam1), np.diag(lam2)\n",
        "\n",
        "rng, Nwin, W = np.random.default_rng(0), 800, 128\n",
        "def sample_gauss(n, cov): return rng.multivariate_normal(np.zeros(3), cov, n)\n",
        "\n",
        "naive, vne, y = [], [], []\n",
        "for lbl, Σ in enumerate([Σ1, Σ2]):          # 0 → Σ₁, 1 → Σ₂\n",
        "    X = sample_gauss(Nwin*W, Σ).reshape(Nwin, W, 3)\n",
        "    for win in X:\n",
        "        C = np.cov(win.T, bias=True)\n",
        "        naive.append(naive_entropy(C))\n",
        "        vne  .append(vne_entropy(C, beta=2.0))\n",
        "        y   .append(lbl)\n",
        "naive, vne, y = map(np.asarray, (naive, vne, y))\n",
        "\n",
        "print(\"AUC  naive :\", roc_auc_score(y, -naive))     # sign so Σ₂ ⇢ “smaller”\n",
        "print(\"AUC  VNE   :\", roc_auc_score(y, -vne  ))\n",
        "print(\"AUC  joint :\", roc_auc_score(y, -vne + -naive))  # rough combo\n",
        "\n",
        "# ────────────────────────────────────────────────────────────\n",
        "# 2)  LDA on (naive,vne) plane & on each 1-D axis\n",
        "# ────────────────────────────────────────────────────────────\n",
        "xy   = np.column_stack([naive, vne])\n",
        "clf2 = LDA().fit(xy, y)              # 2-D classifier\n",
        "\n",
        "# 1-D LDA on trace entropy\n",
        "clf_n = LDA().fit(naive.reshape(-1,1), y)\n",
        "th_na  = -clf_n.intercept_[0] / clf_n.coef_[0]        # vertical line\n",
        "\n",
        "# 1-D LDA on VNE\n",
        "clf_v = LDA().fit(vne.reshape(-1,1), y)\n",
        "th_vne = -clf_v.intercept_[0] / clf_v.coef_[0]        # horizontal line\n",
        "\n",
        "# mesh for 2-D decision background\n",
        "xx, yy = np.meshgrid(np.linspace(naive.min()-0.02, naive.max()+0.02, 400),\n",
        "                     np.linspace(vne  .min()-0.02, vne  .max()+0.02, 400))\n",
        "zz = clf2.predict(np.c_[xx.ravel(), yy.ravel()]).reshape(xx.shape)\n",
        "\n",
        "# ────────────────────────────────────────────────────────────\n",
        "# 3)  pretty plot\n",
        "# ────────────────────────────────────────────────────────────\n",
        "fig = plt.figure(figsize=(8,8))\n",
        "gs  = fig.add_gridspec(2,2, width_ratios=(4,1), height_ratios=(1,4),\n",
        "                       wspace=.05, hspace=.05)\n",
        "\n",
        "ax_sc = fig.add_subplot(gs[1,0])\n",
        "ax_hx = fig.add_subplot(gs[0,0], sharex=ax_sc)\n",
        "ax_hy = fig.add_subplot(gs[1,1], sharey=ax_sc)\n",
        "\n",
        "cmap   = plt.get_cmap('viridis')\n",
        "colors = [cmap(0.25), cmap(0.75)]     # Σ₁, Σ₂\n",
        "\n",
        "# 2-D LDA background\n",
        "ax_sc.contourf(xx, yy, zz, levels=[-0.5,0.5,1.5],\n",
        "               colors=[colors[0], colors[1]], alpha=.1)\n",
        "\n",
        "# vertical / horizontal 1-D boundaries\n",
        "ax_sc.axvline(th_na,  ls='--', lw=1.3, color=colors[0],\n",
        "              label=r'LDA on $S_{\\mathrm{naive}}$')\n",
        "ax_sc.axhline(th_vne, ls='--', lw=1.3, color=colors[1],\n",
        "              label=r'LDA on $S_{\\beta}$')\n",
        "\n",
        "# points\n",
        "for lbl, col in enumerate(colors):\n",
        "    mask = y==lbl\n",
        "    ax_sc.scatter(naive[mask], vne[mask], s=14, alpha=.65,\n",
        "                  edgecolor='none', color=col,\n",
        "                  label=fr'$\\Sigma_{{{lbl+1}}}$')\n",
        "\n",
        "ax_sc.set_xlabel(r'$S_{\\mathrm{naive}}$')\n",
        "ax_sc.set_ylabel(r'$S_{\\beta=2}$')\n",
        "ax_sc.legend(frameon=False, fontsize=8)\n",
        "\n",
        "# marginals\n",
        "bins_x = np.linspace(naive.min(), naive.max(), 35)\n",
        "bins_y = np.linspace(vne.min(),   vne.max(),   35)\n",
        "ax_hx.hist([naive[y==0], naive[y==1]], bins=bins_x,\n",
        "           color=colors, alpha=.7, stacked=True)\n",
        "ax_hy.hist([vne[y==0],   vne[y==1]],   bins=bins_y,\n",
        "           color=colors, alpha=.7, stacked=True, orientation='horizontal')\n",
        "ax_hx.axis('off'); ax_hy.axis('off')\n",
        "\n",
        "plt.suptitle('Decision boundaries in entropy space', y=.95, fontsize=13)\n",
        "plt.tight_layout(); plt.show()\n"
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "import numpy as np, matplotlib.pyplot as plt\n",
        "from sklearn.discriminant_analysis import LinearDiscriminantAnalysis as LDA\n",
        "from sklearn.metrics import roc_auc_score\n",
        "\n",
        "# ────────────────────────────────────────────────────────────\n",
        "# helpers\n",
        "# ────────────────────────────────────────────────────────────\n",
        "def vne_entropy(cov, beta=2.0):\n",
        "    \"\"\"matrix-exponential von-Neumann entropy (no torch needed)\"\"\"\n",
        "    lam = np.linalg.eigvalsh(cov)\n",
        "    rho = np.exp(-beta*lam);  rho /= rho.sum()\n",
        "    return -np.sum(rho * np.log2(rho + 1e-12))\n",
        "\n",
        "def naive_entropy(cov):\n",
        "    \"\"\"trace-normalised Shannon entropy of the spectrum\"\"\"\n",
        "    lam = np.linalg.eigvalsh(cov);  lam /= lam.sum()\n",
        "    return -np.sum(lam * np.log2(lam + 1e-12))\n",
        "\n",
        "# ────────────────────────────────────────────────────────────\n",
        "# 1)  “near-global” multiplicative perturbation\n",
        "#     Σ₂ = diag(1.3, 1.2, 1.1) ⋅ Σ₁   (shape change ≈10 %)\n",
        "# ────────────────────────────────────────────────────────────\n",
        "lam1   = np.array([1.5, 0.5, 0.1])          # non-singular rank-3\n",
        "scales = np.array([2, 2, 2])          # almost-uniform scaling\n",
        "lam2   = lam1 *scales\n",
        "\n",
        "Σ1, Σ2 = np.diag(lam1), np.diag(lam2)\n",
        "\n",
        "rng, Nwin, W = np.random.default_rng(0), 800, 32  # short windows = more noise\n",
        "def sample_gauss(n, cov):\n",
        "    return rng.multivariate_normal(np.zeros(3), cov, n)\n",
        "\n",
        "naive, vne, y = [], [], []\n",
        "for lbl, Σ in enumerate([Σ1, Σ2]):          # 0 → Σ₁, 1 → Σ₂\n",
        "    X = sample_gauss(Nwin*W, Σ).reshape(Nwin, W, 3)\n",
        "    for win in X:\n",
        "        C = np.cov(win.T, bias=True)\n",
        "        naive.append(naive_entropy(C))\n",
        "        vne  .append(vne_entropy(C))\n",
        "        y   .append(lbl)\n",
        "naive, vne, y = map(np.asarray, (naive, vne, y))\n",
        "\n",
        "print(\"AUC  naive :\", roc_auc_score(y,  naive))     # ≈ 0.60  (weak)\n",
        "print(\"AUC  VNE   :\", roc_auc_score(y, -vne  ))     # ≈ 0.81  (strong)\n",
        "\n",
        "# ────────────────────────────────────────────────────────────\n",
        "# 2)  quick visual check\n",
        "# ────────────────────────────────────────────────────────────\n",
        "xy   = np.column_stack([naive, vne])\n",
        "clf2 = LDA().fit(xy, y)\n",
        "\n",
        "xx, yy = np.meshgrid(np.linspace(naive.min()-0.02, naive.max()+0.02, 350),\n",
        "                     np.linspace(vne  .min()-0.02, vne  .max()+0.02, 350))\n",
        "zz = clf2.predict(np.c_[xx.ravel(), yy.ravel()]).reshape(xx.shape)\n",
        "\n",
        "fig, ax = plt.subplots(figsize=(6,6))\n",
        "ax.contourf(xx, yy, zz, levels=[-0.5,0.5,1.5], colors=[\"#3b528b55\",\"#69b53e55\"])\n",
        "ax.scatter(naive[y==0], vne[y==0], s=12, alpha=.7, label=r'$\\Sigma_1$')\n",
        "ax.scatter(naive[y==1], vne[y==1], s=12, alpha=.7, label=r'$\\Sigma_2$')\n",
        "ax.set_xlabel(r'$S_{\\mathrm{naive}}$');  ax.set_ylabel(r'$S_{\\beta=2}$')\n",
        "ax.legend(frameon=False);  ax.set_title('Near-global scale (1.3, 1.2, 1.1)')\n",
        "plt.tight_layout();  plt.show()\n"
      ],
      "metadata": {
        "id": "NRtULB15xXFJ"
      },
      "execution_count": null,
      "outputs": []
    }
  ]
}