{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "fdc0055b",
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "import math\n",
    "from tqdm import tqdm\n",
    "from itertools import product\n",
    "\n",
    "import numpy as np"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "78286a79",
   "metadata": {},
   "outputs": [],
   "source": [
    "class DynamicGraph():\n",
    "    def __init__(self, w_list):\n",
    "        \"\"\"\n",
    "        Parameter\n",
    "        --------\n",
    "        w_list (list of torch.tensor):\n",
    "            list of mixing matrix\n",
    "        \"\"\"\n",
    "        self.w_list = w_list\n",
    "        self.n_nodes = w_list[0].size()[0]\n",
    "        self.length = len(w_list)\n",
    "        self.itr = 0\n",
    "        \n",
    "    def get_in_neighbors(self, i):\n",
    "        \"\"\"\n",
    "        Parameter\n",
    "        ----------\n",
    "        i (int):\n",
    "            a node index\n",
    "        Return\n",
    "        ----------\n",
    "            dictionary of (neighbors's index: weight of the edge (i,j))\n",
    "        \"\"\"\n",
    "        w = self.w_list[self.itr%self.length]        \n",
    "\n",
    "        return {idx.item(): w[idx, i].item() for idx in torch.nonzero(w[:,i])}\n",
    "\n",
    "    def get_out_neighbors(self, i):\n",
    "        \"\"\"\n",
    "        Parameter\n",
    "        ----------\n",
    "        i (int):\n",
    "            a node index\n",
    "        Return\n",
    "        ----------\n",
    "            dictionary of (neighbors's index: weight of the edge (i,j))\n",
    "        \"\"\"\n",
    "        w = self.w_list[self.itr%self.length]        \n",
    "        \n",
    "        return {idx.item(): w[i,idx].item() for idx in torch.nonzero(w[i])}\n",
    "\n",
    "    \n",
    "    def get_neighbors(self, i):\n",
    "        in_neighbors = self.get_in_neighbors(i)\n",
    "        out_neighbors = self.get_out_neighbors(i)\n",
    "        self.itr += 1\n",
    "        return in_neighbors, out_neighbors\n",
    "        \n",
    "    \n",
    "    def get_w(self):\n",
    "        w = self.w_list[self.itr%self.length]        \n",
    "        self.itr += 1\n",
    "        return w\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "14bc1d04",
   "metadata": {},
   "outputs": [],
   "source": [
    "import torch\n",
    "\n",
    "class Ring():\n",
    "    def __init__(self, n_nodes):\n",
    "        self.w = torch.zeros((n_nodes, n_nodes))\n",
    "\n",
    "        for i in range(n_nodes):\n",
    "            self.w[i,i] = 1/3\n",
    "            self.w[i, (i+1)%n_nodes] = 1/3\n",
    "            self.w[i, (i-1)%n_nodes] = 1/3\n",
    "\n",
    "class kRing():\n",
    "    def __init__(self, n_nodes, k):\n",
    "        self.w = torch.zeros((n_nodes, n_nodes))\n",
    "\n",
    "        for i in range(n_nodes):\n",
    "            self.w[i,i] = 1/ (2*k + 1)\n",
    "            for j in range(k+1):\n",
    "                self.w[i, (i+j)%n_nodes] = 1 / (2*k + 1)\n",
    "                self.w[i, (i-j)%n_nodes] = 1 / (2*k + 1)\n",
    "          \n",
    "            \n",
    "class Torus(DynamicGraph):\n",
    "    def __init__(self, p, q):\n",
    "        self.n_nodes = p * q\n",
    "        self.p = p\n",
    "        self.q = q\n",
    "\n",
    "        if p <=2 or q<=2:\n",
    "            print(\"ERROR\")\n",
    "        \n",
    "        w = torch.zeros((self.n_nodes, self.n_nodes))\n",
    "\n",
    "        node_list_list = self.split_nodes()\n",
    "        \n",
    "        for sub_node_list in node_list_list:\n",
    "            for i in range(len(sub_node_list)):\n",
    "                i_idx = sub_node_list[i]\n",
    "                j_idx = sub_node_list[(i+1) % len(sub_node_list)]\n",
    "                w[i_idx, j_idx] = 1/5\n",
    "                w[j_idx, i_idx] = 1/5\n",
    "                w[i_idx, i_idx] = 1/5\n",
    "                w[j_idx, j_idx] = 1/5\n",
    "                \n",
    "\n",
    "        node_list_list2 = self.split_nodes2()\n",
    "        \n",
    "        for sub_node_list in node_list_list2:\n",
    "            for i in range(len(sub_node_list)):\n",
    "                i_idx = sub_node_list[i]\n",
    "                j_idx = sub_node_list[(i+1) % len(sub_node_list)]\n",
    "                w[i_idx, j_idx] = 1/5\n",
    "                w[j_idx, i_idx] = 1/5\n",
    "                w[i_idx, i_idx] = 1/5\n",
    "                w[j_idx, j_idx] = 1/5\n",
    "                \n",
    "\n",
    "        super().__init__([w])\n",
    "\n",
    "    def split_nodes(self):\n",
    "        node_list = list(range(self.n_nodes))\n",
    "        node_list_list = [node_list[i*self.q:(i+1)*self.q] for i in range(self.p)]\n",
    "        return node_list_list\n",
    "\n",
    "    \n",
    "    def split_nodes2(self):\n",
    "        node_list_list = self.split_nodes()\n",
    "        node_list_list2 = [[] for _ in range(self.q)]\n",
    "\n",
    "        for i in range(self.q):\n",
    "            for j in range(self.p):\n",
    "                node_list_list2[i].append(node_list_list[j][i])\n",
    "\n",
    "\n",
    "        return node_list_list2\n",
    "        \n",
    "class ExponentialGraph(DynamicGraph):\n",
    "    def __init__(self, n_nodes):\n",
    "        w = torch.zeros((n_nodes, n_nodes))\n",
    "\n",
    "        n_neighbors = int(math.log2(n_nodes-1))\n",
    "        for i in range(n_nodes):\n",
    "            w[i,i] = 1 / (math.ceil(math.log2(n_nodes)) + 1)\n",
    "            \n",
    "            for j in range(n_neighbors+1):\n",
    "                w[i, (i+2**j)%n_nodes] = 1 / (math.ceil(math.log2(n_nodes)) + 1)\n",
    "\n",
    "        super().__init__([w])\n",
    "        \n",
    "class Line():\n",
    "    def __init__(self, n_nodes):\n",
    "        self.w = torch.zeros((n_nodes, n_nodes))\n",
    "\n",
    "        for i in range(n_nodes):\n",
    "            if i == 0:\n",
    "                self.w[0,0] = 2/3\n",
    "                self.w[0,1] = 1/3\n",
    "            elif i == n_nodes - 1:\n",
    "                self.w[n_nodes-1, n_nodes-1] = 2/3\n",
    "            else:\n",
    "                self.w[i,i] = 1/3\n",
    "                self.w[i, (i+1)] = 1/3\n",
    "                self.w[i, (i-1)] = 1/3\n",
    "\n",
    "class FC():\n",
    "    def __init__(self, n_nodes):\n",
    "        self.w = torch.ones((n_nodes, n_nodes)) / n_nodes\n",
    "        \n",
    "class Hypercube():\n",
    "    def __init__(self, n_nodes):\n",
    "        self.w = torch.zeros((n_nodes, n_nodes))\n",
    "        \n",
    "        dim = int(math.log2(n_nodes))\n",
    "        \n",
    "        node_index = self.generate_pm_vectors(dim)\n",
    "        \n",
    "        for i in range(n_nodes):\n",
    "            for j in range(n_nodes):\n",
    "                if sum((node_index[i] - node_index[j])**2) == 4:\n",
    "                    self.w[i, j] = 1/(1 + dim)\n",
    "                if i == j:\n",
    "                    self.w[i, i] = 1/ (1 + dim)\n",
    "                \n",
    "    def generate_pm_vectors(self, d):\n",
    "        return [np.array(list(vec)) for vec in product([1, -1], repeat=d)]\n",
    "\n",
    "class Star():\n",
    "    def __init__(self, n_nodes):\n",
    "        self.w = torch.zeros((n_nodes, n_nodes))\n",
    "        \n",
    "        self.w[0,0] = 1/n_nodes\n",
    "        for i in range(n_nodes):\n",
    "            if i == 0:\n",
    "                for j in range(n_nodes):\n",
    "                    self.w[i,j] = 1 / n_nodes\n",
    "            else:\n",
    "                self.w[i, 0] = 1 / n_nodes\n",
    "                self.w[0, i] = 1 / n_nodes\n",
    "                self.w[i, i] = 1 - 1 / n_nodes\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "63694ce8",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "\n",
    "def calc_average_spectral_gap(matrix):\n",
    "    singular_values = np.linalg.eig(matrix)[0] \n",
    "    #singular_values = np.linalg.svd(matrix, compute_uv=False)\n",
    "    \n",
    "    \"\"\"\n",
    "    if len(singular_values) <= 4:\n",
    "        print(sorted(singular_values))\n",
    "    else:\n",
    "        print(sorted(singular_values)[:5], \" ... \", sorted(singular_values)[-3:])\n",
    "    \"\"\" \n",
    "    return sum(v**2 / (1 - v**2) for v in sorted(singular_values)[:-1]) / len(singular_values)\n",
    "\n",
    "def calc_spectral_gap(matrix):\n",
    "    singular_values = np.linalg.eig(matrix)[0] \n",
    "    #singular_values = np.linalg.svd(matrix, compute_uv=False)\n",
    "    \n",
    "    \"\"\"\n",
    "    if len(singular_values) <= 4:\n",
    "        print(sorted(singular_values))\n",
    "    else:\n",
    "        print(sorted(singular_values)[:5], \" ... \", sorted(singular_values)[-3:])\n",
    "    \"\"\" \n",
    "    return max([v**2 / (1 - v**2) for v in sorted(singular_values)[:-1]])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "a089b34e",
   "metadata": {},
   "outputs": [],
   "source": [
    "plt.rcParams['mathtext.fontset'] = 'stix'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "bff3bca3",
   "metadata": {},
   "outputs": [
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-6-7141c73cc62b>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m     10\u001b[0m \u001b[0;32mfor\u001b[0m \u001b[0mn\u001b[0m \u001b[0;32min\u001b[0m \u001b[0mnode_list\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     11\u001b[0m     \u001b[0mring\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mRing\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mn\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 12\u001b[0;31m     \u001b[0mlist_of_sing_1\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mappend\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mcalc_spectral_gap\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mring\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mw\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     13\u001b[0m \u001b[0mplt\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mplot\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mnode_list\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mlist_of_sing_1\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmarker\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;34m\"o\"\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mcolor\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;34m\"red\"\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     14\u001b[0m \u001b[0mplt\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mtitle\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34mr\"$\\frac{1 - p}{p}$\"\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0my\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;36m1.03\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mfontsize\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;36m20\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m<ipython-input-4-1794a0ac5c2c>\u001b[0m in \u001b[0;36mcalc_spectral_gap\u001b[0;34m(matrix)\u001b[0m\n\u001b[1;32m     14\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     15\u001b[0m \u001b[0;32mdef\u001b[0m \u001b[0mcalc_spectral_gap\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mmatrix\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m---> 16\u001b[0;31m     \u001b[0msingular_values\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mnp\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mlinalg\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0meig\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mmatrix\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m[\u001b[0m\u001b[0;36m0\u001b[0m\u001b[0;34m]\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m     17\u001b[0m     \u001b[0;31m#singular_values = np.linalg.svd(matrix, compute_uv=False)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m     18\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.pyenv/versions/3.8.5/envs/base/lib/python3.8/site-packages/numpy/core/overrides.py\u001b[0m in \u001b[0;36meig\u001b[0;34m(*args, **kwargs)\u001b[0m\n",
      "\u001b[0;32m~/.pyenv/versions/3.8.5/envs/base/lib/python3.8/site-packages/numpy/linalg/linalg.py\u001b[0m in \u001b[0;36meig\u001b[0;34m(a)\u001b[0m\n\u001b[1;32m   1309\u001b[0m         _raise_linalgerror_eigenvalues_nonconvergence)\n\u001b[1;32m   1310\u001b[0m     \u001b[0msignature\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;34m'D->DD'\u001b[0m \u001b[0;32mif\u001b[0m \u001b[0misComplexType\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mt\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32melse\u001b[0m \u001b[0;34m'd->DD'\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 1311\u001b[0;31m     \u001b[0mw\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mvt\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0m_umath_linalg\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0meig\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0ma\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0msignature\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0msignature\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mextobj\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mextobj\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   1312\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   1313\u001b[0m     \u001b[0;32mif\u001b[0m \u001b[0;32mnot\u001b[0m \u001b[0misComplexType\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mt\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mand\u001b[0m \u001b[0mall\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mw\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mimag\u001b[0m \u001b[0;34m==\u001b[0m \u001b[0;36m0.0\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 648x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig = plt.figure(figsize=(9, 4))\n",
    "\n",
    "ax1 = fig.add_subplot(1, 2, 1)\n",
    "\n",
    "list_of_sing_1 = []\n",
    "list_of_sing_2 = []\n",
    "\n",
    "node_list = [10] + list(range(100, 2100, 100))\n",
    "\n",
    "for n in node_list:\n",
    "    ring = Ring(n)\n",
    "    list_of_sing_1.append(calc_spectral_gap(ring.w))\n",
    "plt.plot(node_list, list_of_sing_1, marker=\"o\", color=\"red\")\n",
    "plt.title(r\"$\\frac{1 - p}{p}$\", y=1.03, fontsize=20)\n",
    "plt.xlabel(\"# Nodes\", fontsize=20)\n",
    "plt.xticks(fontsize=13)\n",
    "plt.yticks(fontsize=13)\n",
    "\n",
    "ax1 = fig.add_subplot(1, 2, 2)\n",
    "\n",
    "list_of_sing_1 = []\n",
    "list_of_sing_2 = []\n",
    "\n",
    "\n",
    "for n in node_list:\n",
    "    ring = Ring(n)\n",
    "    list_of_sing_2.append(calc_average_spectral_gap(ring.w))\n",
    "plt.plot(node_list, list_of_sing_2, marker=\"s\", color=\"blue\")\n",
    "plt.title(r\"$\\frac{1}{n} \\sum_{i=2}^n \\frac{\\lambda_i^2}{1 - \\lambda_i^2}$\", fontsize=20)\n",
    "plt.xlabel(\"#Nodes\", fontsize=20)\n",
    "\n",
    "plt.xticks(fontsize=13)\n",
    "plt.yticks(fontsize=13)\n",
    "\n",
    "plt.savefig(\"pic/ring.pdf\", bbox_inches='tight', )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "4b13938b",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "77694e4b",
   "metadata": {
    "scrolled": false
   },
   "outputs": [],
   "source": [
    "fig = plt.figure(figsize=(9, 4))\n",
    "\n",
    "ax1 = fig.add_subplot(1, 2, 1)\n",
    "\n",
    "list_of_sing_1 = []\n",
    "list_of_sing_2 = []\n",
    "\n",
    "node_list = [n**2 for n in range(10, 50)]\n",
    "\n",
    "for n in node_list:\n",
    "    ring = Torus(int(math.sqrt(n)), int(math.sqrt(n)))\n",
    "    list_of_sing_1.append(calc_spectral_gap(ring.w_list[0]))\n",
    "plt.plot(node_list, list_of_sing_1, marker=\"o\", color=\"red\")\n",
    "plt.title(r\"$\\frac{1 - p}{p}$\", y=1.03, fontsize=20)\n",
    "plt.xlabel(\"# Nodes\", fontsize=20)\n",
    "plt.xticks(fontsize=13)\n",
    "plt.yticks(fontsize=13)\n",
    "\n",
    "ax1 = fig.add_subplot(1, 2, 2)\n",
    "\n",
    "list_of_sing_1 = []\n",
    "list_of_sing_2 = []\n",
    "\n",
    "\n",
    "for n in node_list:\n",
    "    ring = Torus(int(math.sqrt(n)), int(math.sqrt(n)))\n",
    "    list_of_sing_2.append(calc_average_spectral_gap(ring.w_list[0]))\n",
    "plt.plot(node_list, list_of_sing_2, marker=\"s\", color=\"blue\")\n",
    "plt.title(r\"$\\frac{1}{n} \\sum_{i=2}^n \\frac{\\lambda_i^2}{1 - \\lambda_i^2}$\", fontsize=20)\n",
    "plt.xlabel(\"#Nodes\", fontsize=20)\n",
    "\n",
    "plt.xticks(fontsize=13)\n",
    "plt.yticks(fontsize=13)\n",
    "\n",
    "plt.savefig(\"pic/torus.pdf\", bbox_inches='tight', )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "6a24a502",
   "metadata": {},
   "outputs": [],
   "source": [
    "fig = plt.figure(figsize=(9, 4))\n",
    "\n",
    "ax1 = fig.add_subplot(1, 2, 1)\n",
    "\n",
    "list_of_sing_1 = []\n",
    "list_of_sing_2 = []\n",
    "\n",
    "node_list = [n**2 for n in range(5, 50)]\n",
    "\n",
    "for n in node_list:\n",
    "    ring = Torus(int(math.sqrt(n)), int(math.sqrt(n)))\n",
    "    list_of_sing_1.append(calc_spectral_gap(ring.w_list[0]))\n",
    "plt.plot(node_list, list_of_sing_1, marker=\"o\", color=\"red\")\n",
    "plt.title(r\"$\\frac{1 - p}{p}$\", y=1.03, fontsize=20)\n",
    "plt.xlabel(\"# Nodes\", fontsize=20)\n",
    "plt.xticks(fontsize=13)\n",
    "plt.yticks([0, 20, 40, 60, 80, 100, 120, 140, 160], fontsize=13)\n",
    "\n",
    "ax1 = fig.add_subplot(1, 2, 2)\n",
    "\n",
    "list_of_sing_1 = []\n",
    "list_of_sing_2 = []\n",
    "\n",
    "\n",
    "for n in node_list:\n",
    "    ring = Torus(int(math.sqrt(n)), int(math.sqrt(n)))\n",
    "    list_of_sing_2.append(calc_average_spectral_gap(ring.w_list[0]))\n",
    "plt.plot(node_list, list_of_sing_2, marker=\"s\", color=\"blue\")\n",
    "plt.title(r\"$\\frac{1}{n} \\sum_{i=2}^n \\frac{\\lambda_i^2}{1 - \\lambda_i^2}$\", fontsize=20)\n",
    "plt.xlabel(\"#Nodes\", fontsize=20)\n",
    "\n",
    "plt.xticks(fontsize=13)\n",
    "plt.yticks(fontsize=13)\n",
    "plt.xscale(\"log\")\n",
    "#plt.subplots_adjust(wspace=0.25)\n",
    "#plt.tight_layout()\n",
    "\n",
    "plt.savefig(\"pic/torus_log.pdf\", bbox_inches='tight', )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "ed9774be",
   "metadata": {},
   "outputs": [],
   "source": [
    "fig = plt.figure(figsize=(9, 4))\n",
    "\n",
    "ax1 = fig.add_subplot(1, 2, 1)\n",
    "\n",
    "list_of_sing_1 = []\n",
    "list_of_sing_2 = []\n",
    "\n",
    "node_list = [10] + list(range(100, 2100, 100))\n",
    "\n",
    "for n in node_list:\n",
    "    ring = Line(n)\n",
    "    list_of_sing_1.append(calc_spectral_gap(ring.w))\n",
    "plt.plot(node_list, list_of_sing_1, marker=\"o\", color=\"red\")\n",
    "plt.title(r\"$\\frac{1 - p}{p}$\", y=1.03, fontsize=20)\n",
    "plt.xlabel(\"# Nodes\", fontsize=20)\n",
    "plt.xticks(fontsize=13)\n",
    "plt.yticks(fontsize=13)\n",
    "\n",
    "ax1 = fig.add_subplot(1, 2, 2)\n",
    "\n",
    "list_of_sing_1 = []\n",
    "list_of_sing_2 = []\n",
    "\n",
    "\n",
    "for n in node_list:\n",
    "    ring = Line(n)\n",
    "    list_of_sing_2.append(calc_average_spectral_gap(ring.w))\n",
    "plt.plot(node_list, list_of_sing_2, marker=\"s\", color=\"blue\")\n",
    "plt.title(r\"$\\frac{1}{n} \\sum_{i=2}^n \\frac{\\lambda_i^2}{1 - \\lambda_i^2}$\", fontsize=20)\n",
    "plt.xlabel(\"#Nodes\", fontsize=20)\n",
    "\n",
    "plt.xticks(fontsize=13)\n",
    "plt.yticks(fontsize=13)\n",
    "\n",
    "plt.savefig(\"pic/line.pdf\", bbox_inches='tight', )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "681c116d",
   "metadata": {},
   "outputs": [],
   "source": [
    "fig = plt.figure(figsize=(9, 4))\n",
    "\n",
    "ax1 = fig.add_subplot(1, 2, 1)\n",
    "\n",
    "list_of_sing_1 = []\n",
    "list_of_sing_2 = []\n",
    "\n",
    "node_list = [2**n for n in range(1, 12)]\n",
    "\n",
    "for n in node_list:\n",
    "    ring = Hypercube(n)\n",
    "    list_of_sing_1.append(calc_spectral_gap(ring.w))\n",
    "plt.plot(node_list, list_of_sing_1, marker=\"o\", color=\"red\")\n",
    "plt.title(r\"$\\frac{1 - p}{p}$\", y=1.03, fontsize=20)\n",
    "plt.xlabel(\"# Nodes\", fontsize=20)\n",
    "plt.xticks(fontsize=13)\n",
    "plt.yticks(fontsize=13)\n",
    "\n",
    "ax1 = fig.add_subplot(1, 2, 2)\n",
    "\n",
    "list_of_sing_1 = []\n",
    "list_of_sing_2 = []\n",
    "\n",
    "\n",
    "for n in node_list:\n",
    "    ring = Hypercube(n)\n",
    "    list_of_sing_2.append(calc_average_spectral_gap(ring.w))\n",
    "plt.plot(node_list, list_of_sing_2, marker=\"s\", color=\"blue\")\n",
    "plt.title(r\"$\\frac{1}{n} \\sum_{i=2}^n \\frac{\\lambda_i^2}{1 - \\lambda_i^2}$\", fontsize=20)\n",
    "plt.xlabel(\"#Nodes\", fontsize=20)\n",
    "\n",
    "plt.xticks(fontsize=13)\n",
    "plt.yticks(fontsize=13)\n",
    "\n",
    "plt.savefig(\"pic/hypercube.pdf\", bbox_inches='tight', )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "82d33fe7",
   "metadata": {},
   "outputs": [],
   "source": [
    "fig = plt.figure(figsize=(9, 4))\n",
    "\n",
    "ax1 = fig.add_subplot(1, 2, 1)\n",
    "\n",
    "list_of_sing_1 = []\n",
    "list_of_sing_2 = []\n",
    "\n",
    "node_list = [2**n for n in range(1, 12)]\n",
    "\n",
    "for n in node_list:\n",
    "    ring = Hypercube(n)\n",
    "    list_of_sing_1.append(calc_spectral_gap(ring.w))\n",
    "plt.plot(node_list, list_of_sing_1, marker=\"o\", color=\"red\")\n",
    "plt.title(r\"$\\frac{1 - p}{p}$\", y=1.03, fontsize=20)\n",
    "plt.xlabel(\"# Nodes\", fontsize=20)\n",
    "plt.xticks(fontsize=13)\n",
    "plt.yticks(fontsize=13)\n",
    "plt.xscale(\"log\")\n",
    "\n",
    "ax1 = fig.add_subplot(1, 2, 2)\n",
    "\n",
    "list_of_sing_1 = []\n",
    "list_of_sing_2 = []\n",
    "\n",
    "\n",
    "for n in node_list:\n",
    "    ring = Hypercube(n)\n",
    "    list_of_sing_2.append(calc_average_spectral_gap(ring.w))\n",
    "plt.plot(node_list, list_of_sing_2, marker=\"s\", color=\"blue\")\n",
    "plt.title(r\"$\\frac{1}{n} \\sum_{i=2}^n \\frac{\\lambda_i^2}{1 - \\lambda_i^2}$\", fontsize=20)\n",
    "plt.xlabel(\"#Nodes\", fontsize=20)\n",
    "\n",
    "plt.xticks(fontsize=13)\n",
    "plt.yticks(fontsize=13)\n",
    "\n",
    "plt.savefig(\"pic/hypercube_log.pdf\", bbox_inches='tight', )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "c63b1886",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.8.5"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
