{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import networkx as nx\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "from electrostatic import compute_electrostatic_interactions\n",
    "from graphormer import compute_graphormer_distances\n",
    "from commute import compute_commute_distances\n",
    "\n",
    "from positional_encoding import nodepair_to_edge, nodepair_to_node"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Test graphs\n",
    "adj_dict = {}\n",
    "\n",
    "# 6-ring\n",
    "adj = np.asarray([[0, 1, 0, 0, 0, 1],\n",
    "                [1, 0, 1, 0, 0, 0],\n",
    "                [0, 1, 0, 1, 0, 0],\n",
    "                [0, 0, 1, 0, 1, 0],\n",
    "                [0, 0, 0, 1, 0, 1],\n",
    "                [1, 0, 0, 0, 1, 0]])\n",
    "adj_dict['6-ring'] = adj\n",
    "\n",
    "# 5-path\n",
    "G = nx.path_graph(5)\n",
    "adj = nx.to_numpy_array(G)\n",
    "adj_dict['5-path'] = adj\n",
    "\n",
    "# 4-clique\n",
    "adj = 1 - np.eye(4)\n",
    "adj_dict['4-clique'] = adj\n",
    "\n",
    "# 4-barbell\n",
    "H = nx.barbell_graph(4, 0)\n",
    "adj = nx.to_numpy_array(H)\n",
    "adj_dict['4-barbell'] = adj"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "4-barbell graph\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "print(\"4-barbell graph\")\n",
    "nx.draw(H)\n"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Test new positional encodings"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Graph type: 6-ring\n",
      "[[ 0.   -0.42 -0.67 -0.75 -0.67 -0.42]\n",
      " [-0.42  0.   -0.42 -0.67 -0.75 -0.67]\n",
      " [-0.67 -0.42  0.   -0.42 -0.67 -0.75]\n",
      " [-0.75 -0.67 -0.42  0.   -0.42 -0.67]\n",
      " [-0.67 -0.75 -0.67 -0.42  0.   -0.42]\n",
      " [-0.42 -0.67 -0.75 -0.67 -0.42  0.  ]]\n",
      "Graph type: 5-path\n",
      "[[ 0.  -0.2 -0.6 -1.2 -2. ]\n",
      " [-0.8  0.  -0.4 -1.  -1.8]\n",
      " [-1.4 -0.6  0.  -0.6 -1.4]\n",
      " [-1.8 -1.  -0.4  0.  -0.8]\n",
      " [-2.  -1.2 -0.6 -0.2  0. ]]\n",
      "Graph type: 4-clique\n",
      "[[ 0.   -0.25 -0.25 -0.25]\n",
      " [-0.25  0.   -0.25 -0.25]\n",
      " [-0.25 -0.25  0.   -0.25]\n",
      " [-0.25 -0.25 -0.25  0.  ]]\n",
      "Graph type: 4-barbell\n",
      "[[ 0.   -0.25 -0.25 -0.12 -0.62 -1.   -1.   -1.  ]\n",
      " [-0.25  0.   -0.25 -0.12 -0.62 -1.   -1.   -1.  ]\n",
      " [-0.25 -0.25  0.   -0.12 -0.62 -1.   -1.   -1.  ]\n",
      " [-0.38 -0.38 -0.38  0.   -0.5  -0.87 -0.87 -0.87]\n",
      " [-0.88 -0.88 -0.87 -0.5   0.   -0.38 -0.37 -0.37]\n",
      " [-1.   -1.   -1.   -0.63 -0.12  0.   -0.25 -0.25]\n",
      " [-1.   -1.   -1.   -0.63 -0.13 -0.25  0.   -0.25]\n",
      " [-1.   -1.   -1.   -0.63 -0.12 -0.25 -0.25  0.  ]]\n"
     ]
    }
   ],
   "source": [
    "# Test electrostatic interaction\n",
    "for graph_type, adj in adj_dict.items():\n",
    "    print(\"Graph type:\", graph_type)\n",
    "    print(compute_electrostatic_interactions(adj, cache={})[0].round(2))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Graph type: 6-ring\n",
      "[[0 1 2 3 2 1]\n",
      " [1 0 1 2 3 2]\n",
      " [2 1 0 1 2 3]\n",
      " [3 2 1 0 1 2]\n",
      " [2 3 2 1 0 1]\n",
      " [1 2 3 2 1 0]]\n",
      "Graph type: 5-path\n",
      "[[0 1 2 3 4]\n",
      " [1 0 1 2 3]\n",
      " [2 1 0 1 2]\n",
      " [3 2 1 0 1]\n",
      " [4 3 2 1 0]]\n",
      "Graph type: 4-clique\n",
      "[[0 1 1 1]\n",
      " [1 0 1 1]\n",
      " [1 1 0 1]\n",
      " [1 1 1 0]]\n",
      "Graph type: 4-barbell\n",
      "[[0 1 1 1 2 3 3 3]\n",
      " [1 0 1 1 2 3 3 3]\n",
      " [1 1 0 1 2 3 3 3]\n",
      " [1 1 1 0 1 2 2 2]\n",
      " [2 2 2 1 0 1 1 1]\n",
      " [3 3 3 2 1 0 1 1]\n",
      " [3 3 3 2 1 1 0 1]\n",
      " [3 3 3 2 1 1 1 0]]\n"
     ]
    }
   ],
   "source": [
    "# Test graphormer distance\n",
    "for graph_type, adj in adj_dict.items():\n",
    "    print(\"Graph type:\", graph_type)\n",
    "    print(compute_graphormer_distances(adj, adj.shape[0], cache={})[0])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Graph type: 6-ring\n",
      "[[ 0. 10. 16. 18. 16. 10.]\n",
      " [10.  0. 10. 16. 18. 16.]\n",
      " [16. 10.  0. 10. 16. 18.]\n",
      " [18. 16. 10.  0. 10. 16.]\n",
      " [16. 18. 16. 10.  0. 10.]\n",
      " [10. 16. 18. 16. 10.  0.]]\n",
      "Graph type: 5-path\n",
      "[[ 0.  8. 16. 24. 32.]\n",
      " [ 8.  0.  8. 16. 24.]\n",
      " [16.  8.  0.  8. 16.]\n",
      " [24. 16.  8.  0.  8.]\n",
      " [32. 24. 16.  8.  0.]]\n",
      "Graph type: 4-clique\n",
      "[[0. 6. 6. 6.]\n",
      " [6. 0. 6. 6.]\n",
      " [6. 6. 0. 6.]\n",
      " [6. 6. 6. 0.]]\n",
      "Graph type: 4-barbell\n",
      "[[ 0. 13. 13. 13. 39. 52. 52. 52.]\n",
      " [13.  0. 13. 13. 39. 52. 52. 52.]\n",
      " [13. 13.  0. 13. 39. 52. 52. 52.]\n",
      " [13. 13. 13.  0. 26. 39. 39. 39.]\n",
      " [39. 39. 39. 26.  0. 13. 13. 13.]\n",
      " [52. 52. 52. 39. 13.  0. 13. 13.]\n",
      " [52. 52. 52. 39. 13. 13.  0. 13.]\n",
      " [52. 52. 52. 39. 13. 13. 13.  0.]]\n"
     ]
    }
   ],
   "source": [
    "# Test commute distance\n",
    "for graph_type, adj in adj_dict.items():\n",
    "    print(\"Graph type:\", graph_type)\n",
    "    print(compute_commute_distances(adj, adj.shape[0], cache={})[0])"
   ]
  },
  {
   "attachments": {},
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Test conversion functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Graph type: 6-ring (pair -> edge)\n",
      "tensor([-0.4167, -0.4167, -0.4167, -0.4167, -0.4167, -0.4167, -0.4167, -0.4167,\n",
      "        -0.4167, -0.4167, -0.4167, -0.4167], dtype=torch.float64)\n",
      "Graph type: 5-path (pair -> edge)\n",
      "tensor([-0.2000, -0.8000, -0.4000, -0.6000, -0.6000, -0.4000, -0.8000, -0.2000],\n",
      "       dtype=torch.float64)\n",
      "Graph type: 4-clique (pair -> edge)\n",
      "tensor([-0.2500, -0.2500, -0.2500, -0.2500, -0.2500, -0.2500, -0.2500, -0.2500,\n",
      "        -0.2500, -0.2500, -0.2500, -0.2500], dtype=torch.float64)\n",
      "Graph type: 4-barbell (pair -> edge)\n",
      "tensor([-0.2500, -0.2500, -0.1250, -0.2500, -0.2500, -0.1250, -0.2500, -0.2500,\n",
      "        -0.1250, -0.3750, -0.3750, -0.3750, -0.5000, -0.5000, -0.3750, -0.3750,\n",
      "        -0.3750, -0.1250, -0.2500, -0.2500, -0.1250, -0.2500, -0.2500, -0.1250,\n",
      "        -0.2500, -0.2500], dtype=torch.float64)\n"
     ]
    }
   ],
   "source": [
    "# nodepair -> edge\n",
    "for graph_type, adj in adj_dict.items():\n",
    "    pe, base_level, _ = compute_electrostatic_interactions(adj, cache={})\n",
    "\n",
    "    # Convert to edge level\n",
    "    pos_level = 'edge'\n",
    "    print(\"Graph type:\", graph_type, f\"({base_level} -> {pos_level})\")\n",
    "    print(nodepair_to_edge(pe, adj))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Graph type: 6-ring (pair -> node)\n",
      "[[-0.75       -0.75       -0.48611111 -0.48611111  0.25192163  0.25192163]\n",
      " [-0.75       -0.75       -0.48611111 -0.48611111  0.25192163  0.25192163]\n",
      " [-0.75       -0.75       -0.48611111 -0.48611111  0.25192163  0.25192163]\n",
      " [-0.75       -0.75       -0.48611111 -0.48611111  0.25192163  0.25192163]\n",
      " [-0.75       -0.75       -0.48611111 -0.48611111  0.25192163  0.25192163]\n",
      " [-0.75       -0.75       -0.48611111 -0.48611111  0.25192163  0.25192163]]\n",
      "Graph type: 5-path (pair -> node)\n",
      "[[-2.         -2.         -1.2        -0.8         0.72663608  0.72663608]\n",
      " [-1.2        -1.8        -0.6        -0.8         0.45607017  0.60663004]\n",
      " [-0.6        -1.4        -0.4        -0.8         0.21908902  0.53665631]\n",
      " [-1.2        -1.8        -0.6        -0.8         0.45607017  0.60663004]\n",
      " [-2.         -2.         -1.2        -0.8         0.72663608  0.72663608]]\n",
      "Graph type: 4-clique (pair -> node)\n",
      "[[-0.25       -0.25       -0.1875     -0.1875      0.10825318  0.10825318]\n",
      " [-0.25       -0.25       -0.1875     -0.1875      0.10825318  0.10825318]\n",
      " [-0.25       -0.25       -0.1875     -0.1875      0.10825318  0.10825318]\n",
      " [-0.25       -0.25       -0.1875     -0.1875      0.10825318  0.10825318]]\n",
      "Graph type: 4-barbell (pair -> node)\n",
      "[[-1.         -1.         -0.59375    -0.53125     0.38905936  0.39897329]\n",
      " [-1.         -1.         -0.59375    -0.53125     0.38905936  0.39897329]\n",
      " [-1.         -1.         -0.59375    -0.53125     0.38905936  0.39897329]\n",
      " [-0.625      -0.875      -0.34375    -0.53125     0.25579227  0.298106  ]\n",
      " [-0.625      -0.875      -0.34375    -0.53125     0.25579227  0.298106  ]\n",
      " [-1.         -1.         -0.59375    -0.53125     0.38905936  0.39897329]\n",
      " [-1.         -1.         -0.59375    -0.53125     0.38905936  0.39897329]\n",
      " [-1.         -1.         -0.59375    -0.53125     0.38905936  0.39897329]]\n"
     ]
    }
   ],
   "source": [
    "# nodepair -> node\n",
    "for graph_type, adj in adj_dict.items():\n",
    "    pe, base_level, _ = compute_electrostatic_interactions(adj, cache={})\n",
    "\n",
    "    # Convert to node level\n",
    "    pos_level = 'node'\n",
    "    print(\"Graph type:\", graph_type, f\"({base_level} -> {pos_level})\")\n",
    "    print(nodepair_to_node(pe, stats_list=[np.min, np.mean, np.std]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "goli",
   "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.10.10"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
