{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from __future__ import annotations\n",
    "\n",
    "from typing import Optional, Tuple\n",
    "\n",
    "import matplotlib as mpl\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import sympy as sp\n",
    "from IPython.display import display\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "In this notebook, I will present what I believe is a counterexample that Resistance-WL $\\ngtr$ WL *for weighted graphs* (here, when I say WL for weighted graphs, I mean RPE-WL using the weighted adjacency matrix. To me, this is the natural equivalent of WL for weighted graphs, as WL and RPE-WL with the adjacency matrix are equivalent for unweighted graphs.) This also implies that the Resistance-WL $\\lneq$ Eigenspace-Projection-WL."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Here, we use the characterization of the effective resistance of weighted graphs from the paper [Effective Resistance is more than Distance](https://arxiv.org/abs/2010.04521). This paper gives the characterization of effective resistance as a certain kind of squared Euclidean distance (Theorem 1). Our strategy is therefore to find two squared Euclidean distances that correspond to the resistance distances of two weighted graphs, show they cannot be distinguished by resistance-WL, and show that their Laplacian *can* be distinguised by WL.\n",
    "\n",
    "The observation we need from this paper is that we can search for resistance matrices by searching for point sets. The squared Euclidean distance matrix of a point is a resistance matrix iff the pseudoinverse of its centered Gram matrix is a valid graph Laplacian (Proposition 2).\n",
    "\n",
    "We use the following two observations.\n",
    "\n",
    "__Observation__ Let $R$ be the resistance matrix of a graph. Then $L^\\dagger = -\\frac{1}{2} (I-\\frac{1}{n}11^T) R (I-\\frac{1}{n}11^T)$. \n",
    "\n",
    "Call $-\\frac{1}{2}(I-\\frac{1}{n}11^T) R (I-\\frac{1}{n}11^T)$ the __centering__ of $R$ (it ensures that its rows and columns have mean 0, i.e., it centers it).\n",
    "\n",
    "__Observation__(Devriendt, Page 4) A matrix $M$ is a weighted graph Laplacian iff (1) $M$ is symmetric, (2) $M1=0$, and (3) the off-diagonals of $M$ are negative.  \n",
    "\n",
    "These imply the following corrolary:\n",
    "\n",
    "__Corollary__ A squared Euclidean distance matrix is the resistance matrix of a weighted graph iff the pseudoinverse of it centering satisfies properties (1)-(3)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def squared_distance_matrix(P: sp.Matrix) -> sp.Matrix:\n",
    "    \"\"\"Compute the exact squared Euclidean distance matrix for a point set matrix. \"\"\"\n",
    "    n, d = P.rows, P.cols\n",
    "    R = sp.zeros(n, n)\n",
    "    for i in range(n):\n",
    "        for j in range(i + 1, n):\n",
    "            s = sp.Integer(0)\n",
    "            for k in range(d):\n",
    "                diff = P[i, k] - P[j, k]\n",
    "                s += diff * diff\n",
    "            R[i, j] = s\n",
    "            R[j, i] = s\n",
    "    return R\n",
    "\n",
    "\n",
    "def compute_laplacian_pseudoinverse_from_resistance(R: sp.Matrix) -> sp.Matrix:\n",
    "    \"\"\"Compute the Laplacian pseudoinverse L^+ from an exact resistance matrix.\n",
    "\n",
    "    Uses the identity (see Spielman lecture notes)\n",
    "        R = diag(L^+) 1^T + 1 diag(L^+)^T - 2 L^+,\n",
    "    equivalently\n",
    "        L^+ = -(1/2) * Pi R Pi,\n",
    "    where Pi = I - (1/n) 11^T is the projection onto 1^⊥.\n",
    "\n",
    "    \"\"\"\n",
    "    n = R.rows\n",
    "    one = sp.ones(n, 1)\n",
    "    Pi = sp.eye(n) - (sp.Rational(1, n) * (one * one.T))\n",
    "    return sp.Rational(-1, 2) * (Pi * R * Pi) \n",
    "\n",
    "\n",
    "def compute_laplacian_from_point_set(P: sp.Matrix) -> sp.Matrix:\n",
    "    \"\"\" Compute the Laplacian from a resistance embedding.\n",
    "\n",
    "    Steps:\n",
    "      1) Form the squared distance matrix R from P.\n",
    "      2) Compute L^+ = -(1/2) Π R Π.\n",
    "      3) Take the Moore–Penrose pseudoinverse again to recover L.\n",
    "\n",
    "    \"\"\"\n",
    "    R = squared_distance_matrix(P)\n",
    "    L_pinv = compute_laplacian_pseudoinverse_from_resistance(R)\n",
    "    L = L_pinv.pinv()\n",
    "    return sp.Matrix(L)\n",
    "\n",
    "\n",
    "def check_laplacian(L: sp.Matrix) -> Tuple[bool, str]:\n",
    "    \"\"\"Check whether an exact matrix satisfies basic Laplacian conditions.\n",
    "\n",
    "    Conditions checked:\n",
    "      1) Symmetric.\n",
    "      2) Off-diagonals are <= 0.\n",
    "      3) Row sums are 0.\n",
    "\n",
    "    \"\"\"\n",
    "    n = L.rows\n",
    "    is_symmetric =  (L == L.T)\n",
    "    non_positive_off_diagonals = True\n",
    "    for i in range(n):\n",
    "        for j in range(n):\n",
    "            if i == j:\n",
    "                continue\n",
    "            if L[i, j] > 0:\n",
    "                non_positive_off_diagonals = False\n",
    "    zero_row_and_column_sums = True\n",
    "    for i in range(n):\n",
    "        if sp.simplify(sum(L[i, j] for j in range(n))) != 0:\n",
    "            zero_row_and_column_sums = False\n",
    "    return is_symmetric and non_positive_off_diagonals and zero_row_and_column_sums\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Our goal is to now construct point sets that are resistance embeddings of graphs and that are indistinguishable by resistance WL. Our construction are based on regular graphs. Each point will have the same multiset of distances to every other point, so the two graphs will always have the same resistance-WL colors at each iteration. However, the distances will be in different patterns (think 2 length $n/2$ cycles vs a single length $n$ cycle), so their weighted adjacency matrices will have different entries. (The construction of these point sets is inspired by the construction of unit disk graphs; in fact, these point sets realize  $2$ length $n/2$ cycles and a length $n$ cycle as their unit disk graphs. See this [Stack Exchange post](https://cs.stackexchange.com/questions/149386/spatial-embedding-of-graph/149388#149388) for the construction.)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "def generate_cyclic_matrix(n: int) -> sp.Matrix:\n",
    "    \"\"\"Return the adjacency matrix for a cycle C_n \"\"\"\n",
    "    C = sp.zeros(n, n)\n",
    "    for i in range(n):\n",
    "        C[i, (i + 1) % n] = 1\n",
    "        C[i, (i - 1) % n] = 1\n",
    "    return C\n",
    "\n",
    "\n",
    "def generate_point_set_1(n: int, M: Optional[int] = None) -> sp.Matrix:\n",
    "    if M is None:\n",
    "        M = n // 2\n",
    "    P = (sp.Integer(M) * sp.eye(n) + generate_cyclic_matrix(n))\n",
    "    return sp.Rational(1, n) * P\n",
    "\n",
    "\n",
    "def generate_point_set_2(n: int, M: Optional[int] = None) -> sp.Matrix:\n",
    "    if n % 2 != 0:\n",
    "        raise ValueError(\"require n is even\")\n",
    "    if M is None:\n",
    "        M = n // 2\n",
    "\n",
    "    P = sp.Integer(M) * sp.eye(n)\n",
    "    half = n // 2\n",
    "    C = generate_cyclic_matrix(half)\n",
    "    # add two independent cyclic blocks on the diagonal\n",
    "    for i in range(half):\n",
    "        for j in range(half):\n",
    "            P[i, j] += C[i, j]\n",
    "            P[i + half, j + half] += C[i, j]\n",
    "\n",
    "    return sp.Rational(1, n) * P\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We can check that the squared Euclidean distance matrix of these two point sets are resistance matrices"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Matrix([[17801021/10979298, -5822669/10979298, -202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298, -202579/10979298, -5822669/10979298], [-5822669/10979298, 17801021/10979298, -5822669/10979298, -202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298, -202579/10979298], [-202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298, -202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298], [-1249229/10979298, -202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298, -202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298], [-1073419/10979298, -1249229/10979298, -202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298, -202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298], [-1105229/10979298, -1073419/10979298, -1249229/10979298, -202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298, -202579/10979298, -1249229/10979298, -1073419/10979298], [-1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298, -202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298, -202579/10979298, -1249229/10979298], [-1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298, -202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298, -202579/10979298], [-202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298, -202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298], [-5822669/10979298, -202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298, -202579/10979298, -5822669/10979298, 17801021/10979298]])\n",
      "Matrix([[1961/1210, -639/1210, -39/1210, -39/1210, -639/1210, -1/10, -1/10, -1/10, -1/10, -1/10], [-639/1210, 1961/1210, -639/1210, -39/1210, -39/1210, -1/10, -1/10, -1/10, -1/10, -1/10], [-39/1210, -639/1210, 1961/1210, -639/1210, -39/1210, -1/10, -1/10, -1/10, -1/10, -1/10], [-39/1210, -39/1210, -639/1210, 1961/1210, -639/1210, -1/10, -1/10, -1/10, -1/10, -1/10], [-639/1210, -39/1210, -39/1210, -639/1210, 1961/1210, -1/10, -1/10, -1/10, -1/10, -1/10], [-1/10, -1/10, -1/10, -1/10, -1/10, 1961/1210, -639/1210, -39/1210, -39/1210, -639/1210], [-1/10, -1/10, -1/10, -1/10, -1/10, -639/1210, 1961/1210, -639/1210, -39/1210, -39/1210], [-1/10, -1/10, -1/10, -1/10, -1/10, -39/1210, -639/1210, 1961/1210, -639/1210, -39/1210], [-1/10, -1/10, -1/10, -1/10, -1/10, -39/1210, -39/1210, -639/1210, 1961/1210, -639/1210], [-1/10, -1/10, -1/10, -1/10, -1/10, -639/1210, -39/1210, -39/1210, -639/1210, 1961/1210]])\n",
      "L1 is Laplacian: True\n",
      "L1 is Laplacian: True\n"
     ]
    }
   ],
   "source": [
    "n = 10\n",
    "# Setting M to a multiple of n tends to keep denominators small.\n",
    "P1 = generate_point_set_1(n=n, M=8)\n",
    "P2 = generate_point_set_2(n=n, M=8)\n",
    "\n",
    "L1 = compute_laplacian_from_point_set(P1)\n",
    "L2 = compute_laplacian_from_point_set(P2)\n",
    "\n",
    "is_L1_laplacian = check_laplacian(L1)\n",
    "is_L2_laplacian = check_laplacian(L2)\n",
    "\n",
    "print(L1)\n",
    "print(L2)\n",
    "\n",
    "print(\"L1 is Laplacian: \" + str(is_L1_laplacian))\n",
    "print(\"L2 is Laplacian: \" + str(is_L2_laplacian))\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We can also plot the resistance matrices see what they look like. Notice that each row has the same sets of values, so the resistance-WL test will always assign each node the same color."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 800x300 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "R1 = squared_distance_matrix(P1)\n",
    "R2 = squared_distance_matrix(P2)\n",
    "\n",
    "R1f = np.array(R1.tolist(), dtype=float)\n",
    "R2f = np.array(R2.tolist(), dtype=float)\n",
    "\n",
    "_, axes = plt.subplots(1, 2, figsize=(8, 3))\n",
    "axes[0].set_title(\"R1\")\n",
    "axes[0].matshow(R1f, cmap=mpl.cm.inferno)\n",
    "axes[1].set_title(\"R2\")\n",
    "axes[1].matshow(R2f, cmap=mpl.cm.inferno)\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "R1 (exact):\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\left[\\begin{matrix}0 & 1 & \\frac{13}{10} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{13}{10} & 1\\\\1 & 0 & 1 & \\frac{13}{10} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{13}{10}\\\\\\frac{13}{10} & 1 & 0 & 1 & \\frac{13}{10} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25}\\\\\\frac{33}{25} & \\frac{13}{10} & 1 & 0 & 1 & \\frac{13}{10} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25}\\\\\\frac{33}{25} & \\frac{33}{25} & \\frac{13}{10} & 1 & 0 & 1 & \\frac{13}{10} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25}\\\\\\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{13}{10} & 1 & 0 & 1 & \\frac{13}{10} & \\frac{33}{25} & \\frac{33}{25}\\\\\\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{13}{10} & 1 & 0 & 1 & \\frac{13}{10} & \\frac{33}{25}\\\\\\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{13}{10} & 1 & 0 & 1 & \\frac{13}{10}\\\\\\frac{13}{10} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{13}{10} & 1 & 0 & 1\\\\1 & \\frac{13}{10} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{13}{10} & 1 & 0\\end{matrix}\\right]$"
      ],
      "text/plain": [
       "Matrix([\n",
       "[    0,     1, 13/10, 33/25, 33/25, 33/25, 33/25, 33/25, 13/10,     1],\n",
       "[    1,     0,     1, 13/10, 33/25, 33/25, 33/25, 33/25, 33/25, 13/10],\n",
       "[13/10,     1,     0,     1, 13/10, 33/25, 33/25, 33/25, 33/25, 33/25],\n",
       "[33/25, 13/10,     1,     0,     1, 13/10, 33/25, 33/25, 33/25, 33/25],\n",
       "[33/25, 33/25, 13/10,     1,     0,     1, 13/10, 33/25, 33/25, 33/25],\n",
       "[33/25, 33/25, 33/25, 13/10,     1,     0,     1, 13/10, 33/25, 33/25],\n",
       "[33/25, 33/25, 33/25, 33/25, 13/10,     1,     0,     1, 13/10, 33/25],\n",
       "[33/25, 33/25, 33/25, 33/25, 33/25, 13/10,     1,     0,     1, 13/10],\n",
       "[13/10, 33/25, 33/25, 33/25, 33/25, 33/25, 13/10,     1,     0,     1],\n",
       "[    1, 13/10, 33/25, 33/25, 33/25, 33/25, 33/25, 13/10,     1,     0]])"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "R2 (exact):\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\left[\\begin{matrix}0 & 1 & \\frac{13}{10} & \\frac{13}{10} & 1 & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25}\\\\1 & 0 & 1 & \\frac{13}{10} & \\frac{13}{10} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25}\\\\\\frac{13}{10} & 1 & 0 & 1 & \\frac{13}{10} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25}\\\\\\frac{13}{10} & \\frac{13}{10} & 1 & 0 & 1 & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25}\\\\1 & \\frac{13}{10} & \\frac{13}{10} & 1 & 0 & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25}\\\\\\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & 0 & 1 & \\frac{13}{10} & \\frac{13}{10} & 1\\\\\\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & 1 & 0 & 1 & \\frac{13}{10} & \\frac{13}{10}\\\\\\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{13}{10} & 1 & 0 & 1 & \\frac{13}{10}\\\\\\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{13}{10} & \\frac{13}{10} & 1 & 0 & 1\\\\\\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & \\frac{33}{25} & 1 & \\frac{13}{10} & \\frac{13}{10} & 1 & 0\\end{matrix}\\right]$"
      ],
      "text/plain": [
       "Matrix([\n",
       "[    0,     1, 13/10, 13/10,     1, 33/25, 33/25, 33/25, 33/25, 33/25],\n",
       "[    1,     0,     1, 13/10, 13/10, 33/25, 33/25, 33/25, 33/25, 33/25],\n",
       "[13/10,     1,     0,     1, 13/10, 33/25, 33/25, 33/25, 33/25, 33/25],\n",
       "[13/10, 13/10,     1,     0,     1, 33/25, 33/25, 33/25, 33/25, 33/25],\n",
       "[    1, 13/10, 13/10,     1,     0, 33/25, 33/25, 33/25, 33/25, 33/25],\n",
       "[33/25, 33/25, 33/25, 33/25, 33/25,     0,     1, 13/10, 13/10,     1],\n",
       "[33/25, 33/25, 33/25, 33/25, 33/25,     1,     0,     1, 13/10, 13/10],\n",
       "[33/25, 33/25, 33/25, 33/25, 33/25, 13/10,     1,     0,     1, 13/10],\n",
       "[33/25, 33/25, 33/25, 33/25, 33/25, 13/10, 13/10,     1,     0,     1],\n",
       "[33/25, 33/25, 33/25, 33/25, 33/25,     1, 13/10, 13/10,     1,     0]])"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "print(\"R1 (exact):\")\n",
    "display(R1)\n",
    "print()\n",
    "print(\"R2 (exact):\")\n",
    "display(R2)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Conversely, if we look at the Laplacian of these point sets, we see that they have different values."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "L1 (exact):\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\left[\\begin{matrix}\\frac{17801021}{10979298} & - \\frac{5822669}{10979298} & - \\frac{202579}{10979298} & - \\frac{1249229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1105229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1249229}{10979298} & - \\frac{202579}{10979298} & - \\frac{5822669}{10979298}\\\\- \\frac{5822669}{10979298} & \\frac{17801021}{10979298} & - \\frac{5822669}{10979298} & - \\frac{202579}{10979298} & - \\frac{1249229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1105229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1249229}{10979298} & - \\frac{202579}{10979298}\\\\- \\frac{202579}{10979298} & - \\frac{5822669}{10979298} & \\frac{17801021}{10979298} & - \\frac{5822669}{10979298} & - \\frac{202579}{10979298} & - \\frac{1249229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1105229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1249229}{10979298}\\\\- \\frac{1249229}{10979298} & - \\frac{202579}{10979298} & - \\frac{5822669}{10979298} & \\frac{17801021}{10979298} & - \\frac{5822669}{10979298} & - \\frac{202579}{10979298} & - \\frac{1249229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1105229}{10979298} & - \\frac{1073419}{10979298}\\\\- \\frac{1073419}{10979298} & - \\frac{1249229}{10979298} & - \\frac{202579}{10979298} & - \\frac{5822669}{10979298} & \\frac{17801021}{10979298} & - \\frac{5822669}{10979298} & - \\frac{202579}{10979298} & - \\frac{1249229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1105229}{10979298}\\\\- \\frac{1105229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1249229}{10979298} & - \\frac{202579}{10979298} & - \\frac{5822669}{10979298} & \\frac{17801021}{10979298} & - \\frac{5822669}{10979298} & - \\frac{202579}{10979298} & - \\frac{1249229}{10979298} & - \\frac{1073419}{10979298}\\\\- \\frac{1073419}{10979298} & - \\frac{1105229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1249229}{10979298} & - \\frac{202579}{10979298} & - \\frac{5822669}{10979298} & \\frac{17801021}{10979298} & - \\frac{5822669}{10979298} & - \\frac{202579}{10979298} & - \\frac{1249229}{10979298}\\\\- \\frac{1249229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1105229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1249229}{10979298} & - \\frac{202579}{10979298} & - \\frac{5822669}{10979298} & \\frac{17801021}{10979298} & - \\frac{5822669}{10979298} & - \\frac{202579}{10979298}\\\\- \\frac{202579}{10979298} & - \\frac{1249229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1105229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1249229}{10979298} & - \\frac{202579}{10979298} & - \\frac{5822669}{10979298} & \\frac{17801021}{10979298} & - \\frac{5822669}{10979298}\\\\- \\frac{5822669}{10979298} & - \\frac{202579}{10979298} & - \\frac{1249229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1105229}{10979298} & - \\frac{1073419}{10979298} & - \\frac{1249229}{10979298} & - \\frac{202579}{10979298} & - \\frac{5822669}{10979298} & \\frac{17801021}{10979298}\\end{matrix}\\right]$"
      ],
      "text/plain": [
       "Matrix([\n",
       "[17801021/10979298, -5822669/10979298,  -202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298,  -202579/10979298, -5822669/10979298],\n",
       "[-5822669/10979298, 17801021/10979298, -5822669/10979298,  -202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298,  -202579/10979298],\n",
       "[ -202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298,  -202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298],\n",
       "[-1249229/10979298,  -202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298,  -202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298],\n",
       "[-1073419/10979298, -1249229/10979298,  -202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298,  -202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298],\n",
       "[-1105229/10979298, -1073419/10979298, -1249229/10979298,  -202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298,  -202579/10979298, -1249229/10979298, -1073419/10979298],\n",
       "[-1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298,  -202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298,  -202579/10979298, -1249229/10979298],\n",
       "[-1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298,  -202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298,  -202579/10979298],\n",
       "[ -202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298,  -202579/10979298, -5822669/10979298, 17801021/10979298, -5822669/10979298],\n",
       "[-5822669/10979298,  -202579/10979298, -1249229/10979298, -1073419/10979298, -1105229/10979298, -1073419/10979298, -1249229/10979298,  -202579/10979298, -5822669/10979298, 17801021/10979298]])"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "L2 (exact):\n"
     ]
    },
    {
     "data": {
      "text/latex": [
       "$\\displaystyle \\left[\\begin{matrix}\\frac{1961}{1210} & - \\frac{639}{1210} & - \\frac{39}{1210} & - \\frac{39}{1210} & - \\frac{639}{1210} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10}\\\\- \\frac{639}{1210} & \\frac{1961}{1210} & - \\frac{639}{1210} & - \\frac{39}{1210} & - \\frac{39}{1210} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10}\\\\- \\frac{39}{1210} & - \\frac{639}{1210} & \\frac{1961}{1210} & - \\frac{639}{1210} & - \\frac{39}{1210} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10}\\\\- \\frac{39}{1210} & - \\frac{39}{1210} & - \\frac{639}{1210} & \\frac{1961}{1210} & - \\frac{639}{1210} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10}\\\\- \\frac{639}{1210} & - \\frac{39}{1210} & - \\frac{39}{1210} & - \\frac{639}{1210} & \\frac{1961}{1210} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10}\\\\- \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & \\frac{1961}{1210} & - \\frac{639}{1210} & - \\frac{39}{1210} & - \\frac{39}{1210} & - \\frac{639}{1210}\\\\- \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{639}{1210} & \\frac{1961}{1210} & - \\frac{639}{1210} & - \\frac{39}{1210} & - \\frac{39}{1210}\\\\- \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{39}{1210} & - \\frac{639}{1210} & \\frac{1961}{1210} & - \\frac{639}{1210} & - \\frac{39}{1210}\\\\- \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{39}{1210} & - \\frac{39}{1210} & - \\frac{639}{1210} & \\frac{1961}{1210} & - \\frac{639}{1210}\\\\- \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{1}{10} & - \\frac{639}{1210} & - \\frac{39}{1210} & - \\frac{39}{1210} & - \\frac{639}{1210} & \\frac{1961}{1210}\\end{matrix}\\right]$"
      ],
      "text/plain": [
       "Matrix([\n",
       "[1961/1210, -639/1210,  -39/1210,  -39/1210, -639/1210,     -1/10,     -1/10,     -1/10,     -1/10,     -1/10],\n",
       "[-639/1210, 1961/1210, -639/1210,  -39/1210,  -39/1210,     -1/10,     -1/10,     -1/10,     -1/10,     -1/10],\n",
       "[ -39/1210, -639/1210, 1961/1210, -639/1210,  -39/1210,     -1/10,     -1/10,     -1/10,     -1/10,     -1/10],\n",
       "[ -39/1210,  -39/1210, -639/1210, 1961/1210, -639/1210,     -1/10,     -1/10,     -1/10,     -1/10,     -1/10],\n",
       "[-639/1210,  -39/1210,  -39/1210, -639/1210, 1961/1210,     -1/10,     -1/10,     -1/10,     -1/10,     -1/10],\n",
       "[    -1/10,     -1/10,     -1/10,     -1/10,     -1/10, 1961/1210, -639/1210,  -39/1210,  -39/1210, -639/1210],\n",
       "[    -1/10,     -1/10,     -1/10,     -1/10,     -1/10, -639/1210, 1961/1210, -639/1210,  -39/1210,  -39/1210],\n",
       "[    -1/10,     -1/10,     -1/10,     -1/10,     -1/10,  -39/1210, -639/1210, 1961/1210, -639/1210,  -39/1210],\n",
       "[    -1/10,     -1/10,     -1/10,     -1/10,     -1/10,  -39/1210,  -39/1210, -639/1210, 1961/1210, -639/1210],\n",
       "[    -1/10,     -1/10,     -1/10,     -1/10,     -1/10, -639/1210,  -39/1210,  -39/1210, -639/1210, 1961/1210]])"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "print(\"L1 (exact):\")\n",
    "display(L1)\n",
    "print()\n",
    "print(\"L2 (exact):\")\n",
    "display(L2)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "We can also print the two Laplacians, although admittedly it is hard to tell them apart just from a picture"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\\left[\\begin{matrix}0 & \\frac{639}{1210} & \\frac{39}{1210} & \\frac{39}{1210} & \\frac{639}{1210} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10}\\\\\\frac{639}{1210} & 0 & \\frac{639}{1210} & \\frac{39}{1210} & \\frac{39}{1210} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10}\\\\\\frac{39}{1210} & \\frac{639}{1210} & 0 & \\frac{639}{1210} & \\frac{39}{1210} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10}\\\\\\frac{39}{1210} & \\frac{39}{1210} & \\frac{639}{1210} & 0 & \\frac{639}{1210} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10}\\\\\\frac{639}{1210} & \\frac{39}{1210} & \\frac{39}{1210} & \\frac{639}{1210} & 0 & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10}\\\\\\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & 0 & \\frac{639}{1210} & \\frac{39}{1210} & \\frac{39}{1210} & \\frac{639}{1210}\\\\\\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{639}{1210} & 0 & \\frac{639}{1210} & \\frac{39}{1210} & \\frac{39}{1210}\\\\\\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{39}{1210} & \\frac{639}{1210} & 0 & \\frac{639}{1210} & \\frac{39}{1210}\\\\\\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{39}{1210} & \\frac{39}{1210} & \\frac{639}{1210} & 0 & \\frac{639}{1210}\\\\\\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{1}{10} & \\frac{639}{1210} & \\frac{39}{1210} & \\frac{39}{1210} & \\frac{639}{1210} & 0\\end{matrix}\\right]\n"
     ]
    }
   ],
   "source": [
    "from copy import deepcopy\n",
    "\n",
    "def convert_laplacian_to_adjacency(laplacian):\n",
    "    adjacency = deepcopy(laplacian)\n",
    "    for i in range(adjacency.rows):\n",
    "        adjacency[i,i] = 0\n",
    "    return -1*adjacency\n",
    "\n",
    "A2 = convert_laplacian_to_adjacency(L2)\n",
    "sp.print_latex(A2)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 800x300 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "L1f = np.array(L1.tolist(), dtype=float)\n",
    "L2f = np.array(L2.tolist(), dtype=float)\n",
    "\n",
    "vmin = min(np.min(L1f), np.min(L2f))\n",
    "vmax = max(np.max(L1f), np.max(L2f))\n",
    "\n",
    "_, axes = plt.subplots(1, 2, figsize=(8, 3))\n",
    "axes[0].set_title(\"L1 (float view)\")\n",
    "axes[0].matshow(L1f, vmin=vmin, vmax=vmax, cmap=mpl.cm.plasma)\n",
    "axes[1].set_title(\"L2 (float view)\")\n",
    "axes[1].matshow(L2f, vmin=vmin, vmax=vmax, cmap=mpl.cm.plasma)\n",
    "plt.show()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "torch_geometric",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.13.0"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
