{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "84198f79",
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "    The code for all our experiments is provided in this document.\n",
    "    We run it with Python 3.10.9 environment.\n",
    "    We provide detailed instructions on how to run the code with example usage.\n",
    "    We write the codes in a iPython notebook, all functions defined earlier are needed to run the final test.\n",
    "    We provide the URL of the testing datasets acquired from existing repositories.\n",
    "'''\n",
    "\n",
    "import networkx as nx\n",
    "import numpy as np\n",
    "from scipy.sparse import csgraph, diags\n",
    "from scipy.sparse.linalg import cg\n",
    "from scipy.stats import gmean\n",
    "from scipy.sparse.linalg import eigsh\n",
    "\n",
    "# Check the multiplicative error of a sparsifier G' with respect to G\n",
    "def check(G, G_prime, epsilon=1e-6, kernel_threshold=1e-5):\n",
    "    # Compute Laplacian matrices\n",
    "    A = nx.to_scipy_sparse_array(G, weight='weight', format='csr')\n",
    "    degrees = np.array(A.sum(axis=1)).flatten()\n",
    "    L = diags(degrees) - A  # Original graph Laplacian\n",
    "    \n",
    "    B = nx.to_scipy_sparse_array(G_prime, weight='weight', format='csr')\n",
    "    degrees_prime = np.array(B.sum(axis=1)).flatten()\n",
    "    L_prime = diags(degrees_prime) - B  # Sparsified graph Laplacian\n",
    "    \n",
    "    # Compute difference matrix\n",
    "    diff = L - L_prime\n",
    "    \n",
    "    # Regularize L to avoid singularity\n",
    "    n = L.shape[0]\n",
    "    L_reg = L + epsilon * diags(np.ones(n))\n",
    "    \n",
    "    # Solve generalized eigenvalue problem: (L - L')x = lambda L_reg x\n",
    "    try:\n",
    "        # Compute the 2 largest eigenvalues and eigenvectors\n",
    "        eigenvalues, eigenvectors = eigsh(diff, k=2, M=L_reg, which='LA', tol=1e-8)\n",
    "        eigenvalues = np.real(eigenvalues)  # Ensure real eigenvalues\n",
    "        eigenvectors = eigenvectors[:, ~np.isnan(eigenvalues)]  # Remove NaN-related columns\n",
    "        eigenvalues = eigenvalues[~np.isnan(eigenvalues)]\n",
    "    except Exception as e:\n",
    "        print(f\"Error in eigenvalue computation: {e}\")\n",
    "        return 1.0  # Neutral default value\n",
    "    \n",
    "    # Find maximum eigenvalue whose eigenvector is not in kernel of L or L - L'\n",
    "    max_valid_eigenvalue = None\n",
    "    for i, eigenvalue in enumerate(eigenvalues):\n",
    "        eigenvector = eigenvectors[:, i]\n",
    "        \n",
    "        # Check if eigenvector is in the kernel of L\n",
    "        Lx = L @ eigenvector\n",
    "        Lx_norm = np.linalg.norm(Lx)\n",
    "        \n",
    "        # Check if eigenvector is in the kernel of L - L'\n",
    "        diff_x = diff @ eigenvector\n",
    "        diff_x_norm = np.linalg.norm(diff_x)\n",
    "        \n",
    "        # Consider eigenvalue only if eigenvector is not in either kernel\n",
    "        if Lx_norm > kernel_threshold and diff_x_norm > kernel_threshold:\n",
    "            if max_valid_eigenvalue is None or abs(eigenvalue) > abs(max_valid_eigenvalue):\n",
    "                max_valid_eigenvalue = eigenvalue\n",
    "    #print(eigenvector)\n",
    "    #print(eigenvector.T @ L @ eigenvector)\n",
    "    #print(eigenvector.T @ L_prime @ eigenvector)\n",
    "    # If no valid eigenvalues found, return neutral value\n",
    "    if max_valid_eigenvalue is None:\n",
    "        print(\"No valid eigenvalues found (all eigenvectors in kernel)\")\n",
    "        return 1.0\n",
    "    \n",
    "    # Return largest eigenvalue magnitude as quality metric\n",
    "    return np.abs(max_valid_eigenvalue)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "04846539",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Offline algorithm\n",
    "def offline(G, C_off, seed = None):\n",
    "    if seed is not None:\n",
    "        np.random.seed(seed)\n",
    "    \n",
    "    # Convert to adjacency matrix and Laplacian\n",
    "    n = G.number_of_nodes()\n",
    "    A = nx.to_scipy_sparse_array(G, weight = 'weight', format = 'csr')\n",
    "    degrees = np.array(A.sum(axis=1)).flatten()\n",
    "    L = diags(degrees) - A  # Laplacian matrix\n",
    "    L_reg = L + 1e-1 * diags(np.ones(L.shape[0])) #Add a regularity\n",
    "\n",
    "    # Construct sparsified graph\n",
    "    G_sparse = nx.MultiGraph()\n",
    "    G_sparse.add_nodes_from(G.nodes())\n",
    "    \n",
    "    # Sample the edges\n",
    "    for u, v, key, data in G.edges(keys = True, data = True):\n",
    "\n",
    "        orig_weight = data.get('weight', 1.0)\n",
    "        e = np.zeros(n)\n",
    "        e[u] = 1\n",
    "        e[v] = -1\n",
    "        x, _ = cg(L_reg, e)#, tol=1e-6)\n",
    "        r_eff = e.T @ x * orig_weight\n",
    "        prob = min(1, r_eff * C_off * np.log(n))\n",
    "        if np.random.uniform(0, 1) < prob:\n",
    "            new_weight = orig_weight / prob\n",
    "            G_sparse.add_edge(u, v, key = key, weight=new_weight)\n",
    "    \n",
    "    return G_sparse"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Online algorithm - batch implementation. C_ol is the constant factor\n",
    "def online_batch(G, C_ol, batch_size = 100, seed = None):\n",
    "    if seed is not None:\n",
    "        np.random.seed(seed)\n",
    "    n = G.number_of_nodes()\n",
    "    \n",
    "    # Initialize sparsified multi-graph with all nodes\n",
    "    G_sparse = nx.MultiGraph()\n",
    "    G_sparse.add_nodes_from(G.nodes())\n",
    "\n",
    "    # Get edge list\n",
    "    edges = list(G.edges(keys = True, data = True))\n",
    "    m = len(edges)\n",
    "    \n",
    "    # Process edges in batches\n",
    "    for batch_start in range(0, m, batch_size):\n",
    "        batch_edges = edges[batch_start:batch_start + batch_size]\n",
    "        any_edge_added = False\n",
    "        \n",
    "        # Compute Laplacian for the current G_sparse\n",
    "        A = nx.to_scipy_sparse_array(G_sparse, weight = 'weight', format = 'csr')\n",
    "        degrees = np.array(A.sum(axis=1)).flatten()\n",
    "        L = diags(degrees) - A + 1e-5 * np.eye(n)\n",
    "        L_reg = L #+ 1e-3 * diags(np.ones(L.shape[0])) #Add a regularity\n",
    "        \n",
    "        # Process each edge in the batch\n",
    "        for u, v, key, data in batch_edges:\n",
    "            orig_weight = data.get('weight', 1.0)\n",
    "            \n",
    "            # Compute effective resistance\n",
    "            e = np.zeros(n)\n",
    "            e[u] = 1\n",
    "            e[v] = -1\n",
    "            x, _ = cg(L_reg, e)#, tol = 1e-6)\n",
    "            r_eff = e.T @ x * orig_weight\n",
    "            prob = min(r_eff * C_ol * np.log(n), 1.0)\n",
    "            \n",
    "            # Sample edge\n",
    "            if np.random.uniform(0, 1) < prob:\n",
    "                new_weight = orig_weight / prob\n",
    "                #print(new_weight)\n",
    "                G_sparse.add_edge(u, v, key = key, weight = new_weight)\n",
    "                any_edge_added = True\n",
    "        \n",
    "        # Update Laplacian only if an edge was added in this batch\n",
    "        if not any_edge_added:\n",
    "            continue\n",
    "        \n",
    "    print(f\"Online Batch MultiGraph - Edges in G_sparse: {G_sparse.number_of_edges()}\")\n",
    "    \n",
    "    return G_sparse"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "70e3eddd",
   "metadata": {},
   "outputs": [],
   "source": [
    "import time\n",
    "\n",
    "# Merge-and-reduce method. C_off is the constant factor\n",
    "def merge_and_reduce(G, C_off, seed = None):\n",
    "    if seed is not None:\n",
    "        np.random.seed(seed)\n",
    "    \n",
    "    # Get edge list as stream\n",
    "    edges = list(G.edges(keys = True, data = True))\n",
    "    m = len(edges)\n",
    "    \n",
    "    # Level 0: Divide edges into coresets\n",
    "    level_coresets = []\n",
    "    n = G.number_of_nodes()\n",
    "    coreset_size = max(1,int(np.round(C_off * n * np.log(n))))\n",
    "    for i in range(0, m, coreset_size):\n",
    "        coreset_edges = edges[i:i + coreset_size]\n",
    "        G_coreset = nx.MultiGraph()\n",
    "        G_coreset.add_nodes_from(G.nodes())  # Include all nodes\n",
    "        G_coreset.add_edges_from((u, v, key, data) for u, v, key, data in coreset_edges)\n",
    "        level_coresets.append(G_coreset)\n",
    "    #print(f\"Level 0: {len(level_coresets)} coresets\")\n",
    "    \n",
    "    # Process levels until one coreset remains\n",
    "    level = 0\n",
    "    while len(level_coresets) > 1:\n",
    "        next_level = []\n",
    "        #print(f\"\\nProcessing Level {level}\")\n",
    "        \n",
    "        # Pair neighboring coresets\n",
    "        for i in range(0, len(level_coresets), 2):\n",
    "            if i + 1 < len(level_coresets):\n",
    "                # Merge pair\n",
    "                G1 = level_coresets[i]\n",
    "                G2 = level_coresets[i + 1]\n",
    "                G_merged = nx.MultiGraph()\n",
    "                G_merged.add_nodes_from(G.nodes())\n",
    "                G_merged.add_edges_from(G1.edges(keys = True, data = True))\n",
    "                G_merged.add_edges_from(G2.edges(keys = True, data = True))\n",
    "                #print(f\"  Merging coresets {i} and {i+1}: {G_merged.number_of_edges()} edges\")\n",
    "            else:\n",
    "                # Single coreset\n",
    "                G_merged = level_coresets[i]\n",
    "                #print(f\"  Single coreset {i}: {G_merged.number_of_edges()} edges\")\n",
    "            \n",
    "            # Apply offline sparsification\n",
    "            G_reduced = offline(G_merged, C_off, seed=seed)\n",
    "            #print(f\"  Reduced to: {G_reduced.number_of_edges()} edges\")\n",
    "            next_level.append(G_reduced)\n",
    "        \n",
    "        level_coresets = next_level\n",
    "        level += 1\n",
    "        #print(f\"Level {level}: {len(level_coresets)} coresets\")\n",
    "    \n",
    "    # Final coreset\n",
    "    if level_coresets:\n",
    "        final_graph = level_coresets[0]\n",
    "        #print(f\"\\nFinal graph: {final_graph.number_of_nodes()} nodes, {final_graph.number_of_edges()} edges\")\n",
    "        return final_graph\n",
    "\n",
    "# Our streaming algorithm. C_ol_str is the constant factor for the online substream. C_off is the constant factor for the offline algorithm\n",
    "def streaming(G, C_ol_str, C_off, seed = None):\n",
    "\n",
    "    #Run online algorithm\n",
    "    G_prime = online_batch(G, C_ol_str, 1000, seed)\n",
    "\n",
    "    #Passing the online result to the merge-and-reduce\n",
    "    G_sparse = merge_and_reduce(G_prime, C_off, seed)\n",
    "    return G_sparse"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "26965f8e",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "'\\nExample usage\\nprint(comparison(n = 100, m = 50000, \\n                 C_ol = 0.01, C_off = 2.2, C_ol_str = 2.5, \\n                 iter = 10, seed = None))\\n'"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "\n",
    "'''\n",
    "    This block implement our comparison under different budgets using the synthetic graph. \n",
    "    All hyperparameters can be found in the appendix of our paper, they are also listed in the experiment results.\n",
    "    C_ol is the constant factor for the online algorithm. \n",
    "    C_off is the constant factor for the offline algorithm.\n",
    "    C_ol_str is the constant factor for the online substream in our streaming algorithm. \n",
    "'''\n",
    "import json\n",
    "\n",
    "def comparison(n, m, C_ol, C_off, C_ol_str, iter = 10, seed = None):\n",
    "    list_ol = []\n",
    "    list_mr = []\n",
    "    list_str = []\n",
    "    output_file = \"-\" # Choose a output directory\n",
    "\n",
    "    for i in range(iter):\n",
    "        #Generate the graph\n",
    "        G = nx.MultiGraph()\n",
    "        G.add_nodes_from(range(n))\n",
    "        for _ in range(m):\n",
    "            u, v = np.random.choice(n, 2, replace=False)\n",
    "            weight = np.random.uniform(1, 10)\n",
    "            G.add_edge(u, v, weight = weight)\n",
    "\n",
    "        G_ol = online_batch(G, C_ol, 100, seed)\n",
    "        approx_ol = check(G, G_ol)\n",
    "        list_ol.append(approx_ol)\n",
    "\n",
    "        G_mr = merge_and_reduce(G, C_off, seed)\n",
    "        approx_mr = check(G, G_mr)\n",
    "        list_mr.append(approx_mr)\n",
    "\n",
    "        G_str = streaming(G, C_ol_str, C_off, seed)\n",
    "        approx_str = check(G, G_str)\n",
    "        list_str.append(approx_str)\n",
    "\n",
    "    results = { \"budget\": 2500,\n",
    "        \"results\": {\n",
    "            \"batch_online\": {\n",
    "                \"eigenvalues\": list_ol,\n",
    "                \"arithmetic_mean\": np.mean(np.array(list_ol))\n",
    "            },\n",
    "            \"merge_and_reduce\": {\n",
    "                \"eigenvalues\": list_mr,\n",
    "                \"arithmetic_mean\": np.mean(np.array(list_mr))\n",
    "            },\n",
    "            \"our_algorithm\": {\n",
    "                \"eigenvalues\": list_str,\n",
    "                \"arithmetic_mean\": np.mean(np.array(list_str))\n",
    "            },\n",
    "            \"parameters\": {\n",
    "                \"n\": n,\n",
    "                \"m\": m,\n",
    "                \"C_ol\": C_ol,\n",
    "                \"C_off\": C_off,\n",
    "                \"C_ol_str\": C_ol_str,\n",
    "                \"iterations\": iter,\n",
    "                \"seed\": seed\n",
    "            }\n",
    "        }\n",
    "    }\n",
    "\n",
    "    # Save to JSON file\n",
    "    with open(output_file, 'w') as f:\n",
    "        json.dump(results, f, indent=4)\n",
    "    \n",
    "    return np.mean(np.array(list_ol)), np.mean(np.array(list_mr)), np.mean(np.array(list_str))\n",
    "\n",
    "'''\n",
    "Example usage\n",
    "print(comparison(n = 100, m = 50000, \n",
    "                 C_ol = 0.01, C_off = 2.2, C_ol_str = 2.5, \n",
    "                 iter = 10, seed = None))\n",
    "'''\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "11f67928",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Original graph: 1035 nodes, 53498 edges\n",
      "Online Batch MultiGraph - Edges in G_sparse: 13528\n",
      "Online Batch MultiGraph - Edges in G_sparse: 33062\n",
      "Online Batch MultiGraph - Edges in G_sparse: 13523\n",
      "Online Batch MultiGraph - Edges in G_sparse: 33182\n",
      "Online Batch MultiGraph - Edges in G_sparse: 13606\n",
      "Online Batch MultiGraph - Edges in G_sparse: 33155\n",
      "{'budget': 1500, 'results': {'batch_online': {'eigenvalues': [0.693785087839378, 0.5873733221698637, 0.595830893461069], 'arithmetic_mean': 0.6256631011567703, 'num': 13606}, 'merge_and_reduce': {'eigenvalues': [0.6077975034945373, 0.4638690355633387, 0.46186049695384285], 'arithmetic_mean': 0.511175678670573, 'num': 14644}, 'our_algorithm': {'eigenvalues': [0.49252281885542193, 0.5346961433447353, 0.42258009544066827], 'arithmetic_mean': 0.48326635254694184, 'num': 14737}, 'parameters': {'n': 1035, 'm': 53498, 'C_ol': 0.35, 'C_off': 2.3333333333333335, 'C_ol_str': 1.8, 'iterations': 3, 'seed': None}}}\n",
      "(0.6238603701799513, 0.506860410776717, 0.4810026639366142)\n",
      "Original graph: 1035 nodes, 53498 edges\n",
      "Online Batch MultiGraph - Edges in G_sparse: 21265\n",
      "Online Batch MultiGraph - Edges in G_sparse: 42853\n",
      "Online Batch MultiGraph - Edges in G_sparse: 21121\n",
      "Online Batch MultiGraph - Edges in G_sparse: 42872\n",
      "Online Batch MultiGraph - Edges in G_sparse: 21250\n",
      "Online Batch MultiGraph - Edges in G_sparse: 42805\n",
      "{'budget': 1500, 'results': {'batch_online': {'eigenvalues': [0.44094100674249753, 0.48818935512543327, 0.45960335439002975], 'arithmetic_mean': 0.4629112387526535, 'num': 21250}, 'merge_and_reduce': {'eigenvalues': [0.43673552562943396, 0.35190719356933664, 0.35268664256935184], 'arithmetic_mean': 0.38044312058937413, 'num': 19635}, 'our_algorithm': {'eigenvalues': [0.36027153419660923, 0.3740206619622199, 0.3496385583367137], 'arithmetic_mean': 0.36131025149851426, 'num': 19826}, 'parameters': {'n': 1035, 'm': 53498, 'C_ol': 0.8, 'C_off': 3.3333333333333335, 'C_ol_str': 4, 'iterations': 3, 'seed': None}}}\n",
      "(0.4625059542508006, 0.3784529477465849, 0.36117273240001024)\n",
      "Original graph: 1035 nodes, 53498 edges\n",
      "Online Batch MultiGraph - Edges in G_sparse: 27015\n",
      "Online Batch MultiGraph - Edges in G_sparse: 45934\n",
      "Online Batch MultiGraph - Edges in G_sparse: 26975\n",
      "Online Batch MultiGraph - Edges in G_sparse: 45916\n",
      "Online Batch MultiGraph - Edges in G_sparse: 26942\n",
      "Online Batch MultiGraph - Edges in G_sparse: 45947\n",
      "{'budget': 1500, 'results': {'batch_online': {'eigenvalues': [0.37285794564086827, 0.376133184624098, 0.3768628784065277], 'arithmetic_mean': 0.3752846695571647, 'num': 26942}, 'merge_and_reduce': {'eigenvalues': [0.293521354929374, 0.3451724656750011, 0.3092167951550648], 'arithmetic_mean': 0.3159702052531466, 'num': 24692}, 'our_algorithm': {'eigenvalues': [0.31577741527667397, 0.32447138801795145, 0.31223241024779264], 'arithmetic_mean': 0.3174937378474727, 'num': 24723}, 'parameters': {'n': 1035, 'm': 53498, 'C_ol': 1.3, 'C_off': 4.5, 'C_ol_str': 5.5, 'iterations': 3, 'seed': None}}}\n",
      "(0.37528062057439976, 0.31524362125447564, 0.31745230484277454)\n"
     ]
    }
   ],
   "source": [
    "'''\n",
    "    This block implements our comparison under different budgets using the Facebook graph. \n",
    "    We download the facebook.tar.gz file on the snap webpage https://snap.stanford.edu/data/ego-Facebook.html.\n",
    "    Open the graph of the user 107 from directory \"/facebook/107.edges/\".\n",
    "    All hyperparameters can be found in the appendix of our paper, they are also listed in the experiment results.\n",
    "    C_ol is the constant factor for the online algorithm. \n",
    "    C_off is the constant factor for the offline algorithm.\n",
    "    C_ol_str is the constant factor for the online substream in our streaming algorithm. \n",
    "    The provided code only supports one choice of parameters, one might need to copy-paste the results to a new json file.\n",
    "'''\n",
    "\n",
    "import random\n",
    "import json\n",
    "\n",
    "def load_ego_network(file_path, seed=None):\n",
    "    # Set random seed if provided\n",
    "    if seed is not None:\n",
    "        random.seed(seed)\n",
    "        np.random.seed(seed)\n",
    "    \n",
    "    # Load the ego network from a SNAP edge list file\n",
    "    G = nx.read_edgelist(file_path, nodetype=int, create_using=nx.MultiGraph())\n",
    "    \n",
    "    # Get sorted nodes and create mapping to consecutive integers starting from 1\n",
    "    nodes = sorted(G.nodes())\n",
    "    mapping = {old_node: new_node for new_node, old_node in enumerate(nodes, start=1)}\n",
    "    \n",
    "    # Create a new MultiGraph\n",
    "    GG = nx.MultiGraph()\n",
    "    \n",
    "    # Add nodes in sorted order (1 to n)\n",
    "    for i in range(len(nodes)):\n",
    "        GG.add_node(i)\n",
    "    \n",
    "    # Copy all edges (including parallel edges) with random weights\n",
    "    for u, v, key in G.edges(keys=True):\n",
    "        new_u = mapping[u]\n",
    "        new_v = mapping[v]\n",
    "        weight = np.random.uniform(1, 10)\n",
    "        GG.add_edge(new_u, new_v, key=key, weight=weight)\n",
    "    \n",
    "    return GG\n",
    "\n",
    "#Comparison - budget\n",
    "def comparison(C_ol, C_off, C_ol_str, iter = 10, seed = None):\n",
    "    list_ol = []\n",
    "    list_mr = []\n",
    "    list_str = []\n",
    "    output_file = \"-\" # Choose a output directory\n",
    "\n",
    "    file_path = \"-\" # Open the downloaded file\n",
    "    G = load_ego_network(file_path, seed)\n",
    "    n = G.number_of_nodes()\n",
    "    m = G.number_of_edges()\n",
    "    print(f\"Original graph: {n} nodes, {m} edges\")\n",
    "\n",
    "    for i in range(iter):\n",
    "        #Generate the graph\n",
    "        #G = nx.MultiGraph()\n",
    "        #G.add_nodes_from(range(n))\n",
    "        #for _ in range(m):\n",
    "        #    u, v = np.random.choice(n, 2, replace=False)\n",
    "        #    weight = np.random.uniform(1, 10)\n",
    "        #    G.add_edge(u, v, weight = weight)\n",
    "\n",
    "        G_ol = online_batch(G, C_ol, 100, seed)\n",
    "        approx_ol = check(G, G_ol)\n",
    "        list_ol.append(approx_ol)\n",
    "\n",
    "        G_mr = merge_and_reduce(G, C_off, seed)\n",
    "        approx_mr = check(G, G_mr)\n",
    "        list_mr.append(approx_mr)\n",
    "\n",
    "        G_str = streaming(G, C_ol_str, C_off, seed)\n",
    "        approx_str = check(G, G_str)\n",
    "        list_str.append(approx_str)\n",
    "\n",
    "    results = { \"budget\": 1500, # Need to change it here\n",
    "        \"results\": {\n",
    "            \"batch_online\": {\n",
    "                \"eigenvalues\": list_ol,\n",
    "                \"arithmetic_mean\": np.mean(np.array(list_ol)),\n",
    "                \"num\": G_ol.number_of_edges()\n",
    "            },\n",
    "            \"merge_and_reduce\": {\n",
    "                \"eigenvalues\": list_mr,\n",
    "                \"arithmetic_mean\": np.mean(np.array(list_mr)),\n",
    "                \"num\": G_mr.number_of_edges()\n",
    "            },\n",
    "            \"our_algorithm\": {\n",
    "                \"eigenvalues\": list_str,\n",
    "                \"arithmetic_mean\": np.mean(np.array(list_str)),\n",
    "                \"num\": G_str.number_of_edges()\n",
    "            },\n",
    "            \"parameters\": {\n",
    "                \"n\": n,\n",
    "                \"m\": m,\n",
    "                \"C_ol\": C_ol,\n",
    "                \"C_off\": C_off,\n",
    "                \"C_ol_str\": C_ol_str,\n",
    "                \"iterations\": iter,\n",
    "                \"seed\": seed\n",
    "            }\n",
    "        }\n",
    "    }\n",
    "\n",
    "    # Save to JSON file\n",
    "    # with open(output_file, 'w') as f:\n",
    "        # json.dump(results, f, indent=4)\n",
    "    print(results)\n",
    "    return gmean(list_ol), gmean(list_mr), gmean(list_str)\n",
    "\n",
    "\n",
    "# print(comparison(0.145, 4/3, 1, iter = 5, seed = None))\n",
    "# print(comparison(0.35, 7/3, 1.8, iter = 3, seed = None))\n",
    "# print(comparison(0.8, 10/3, 4, iter = 3, seed = None))\n",
    "# print(comparison(1.3, 27/6, 5.5, iter = 3, seed = None))\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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ivT0dKxduxb79u3DpEmTsHDhQlRWVrqsn5WVhWXLlmHHjh3Iy8tDVFQUFixYgLKyskFuORERERF5mseT2ZdffhkrV65EWloaxo8fj7feegtarRbvvfeey/off/wxHnroIUyePBljx47Fu+++C5vNhszMzEFuORERERF5msKTN29ra0NBQQFWr17tOCeTyZCSkoK8vLx+vUZzczOsViuCg4NdllssFlgsFsfz+vp6AEBNTQ2sVus1tL5/rFYrmpubUV1dDaVS6fb7EXkS452GE8Y7DSeDHe8NDQ0AACHEZet6NJm9cOECOjo6YDAYnM4bDAYcO3asX6/xxBNPICIiAikpKS7L169fj3Xr1vU6HxMTc+UNJiIiIqJB09DQgICAgEvW8Wgye602bNiAzZs3IysrCxqNxmWd1atXIz093fHcZrOhpqYGISEhkCTJ7W00m82IiorC2bNnodPp3H4/Ik9ivNNwwnin4WSw410IgYaGBkRERFy2rkeT2dDQUMjlclRUVDidr6iogNFovOS1L730EjZs2IBvvvkGN954Y5/11Go11Gq107nAwMCrbvPV0ul0/LKjYYPxTsMJ452Gk8GM98v1yHbx6AQwlUqFadOmOU3e6prMlZyc3Od1L7zwAv70pz8hIyMD06dPH4ymEhEREdEQ5PFhBunp6Vi+fDmmT5+OxMREvPLKK2hqakJaWhoA4N5770VkZCTWr18PAHj++eexZs0afPLJJxg1ahRMJhMAwM/PD35+fh57H0REREQ0+DyezN51112oqqrCmjVrYDKZMHnyZGRkZDgmhZ05cwYy2cUO5DfffBNtbW347//+b6fXWbt2LZ555pnBbHq/qNVqrF27ttdQB6LrEeOdhhPGOw0nQzneJdGfNQ+IiIiIiIYgj2+aQERERER0tZjMEhEREZHXYjJLRERERF6LySwREREReS0msy58//33WLJkCSIiIiBJEj7//HOnciEE1qxZg/DwcPj4+CAlJQUnTpxwqlNTU4O7774bOp0OgYGBWLFiBRobG53qHDhwAHPmzIFGo0FUVBReeOGFXm3ZsmULxo4dC41Gg4kTJ2Lbtm0D/n5peLtcvN93332QJMnpSE1NdarDeCdvsH79esyYMQP+/v7Q6/X48Y9/jKKiIqc6ra2tePjhhxESEgI/Pz/813/9V6+Nfc6cOYPFixdDq9VC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hqdLxXtzc7NYsGCBCAsLE0qlUowcOVKsXLmy1xcQ4528gas4ByDef/99R52Wlhbx0EMPiaCgIKHVasWdd94pysvLnV6ntLRULFq0SPj4+IjQ0FDxP//zP8JqtTrV2bFjh5g8ebJQqVQiNjbW6R5dXnvtNREdHS1UKpVITEwUu3btcsfbpmHqcvF+5swZMXfuXBEcHCzUarWIj48Xjz32mNM6s0J4R7xLnW+YiIiIiMjrcMwsEREREXktJrNERERE5LWYzBIRERGR12IyS0RERERei8ksEREREXktJrNERERE5LWYzBIRERGR12IyS0RERERei8ksEZGX27hxIwIDAz3dDCIij2AyS0Q0SO677z5IkuQ4QkJCkJqaigMHDni6ab2UlpZCkiQUFhZ6uilERJfEZJaIaBClpqaivLwc5eXlyMzMhEKhwO233+7pZhEReS0ms0REg0itVsNoNMJoNGLy5Ml48skncfbsWVRVVSErKwuSJKGurs5Rv7CwEJIkobS01HFu48aNiI6OhlarxZ133onq6upe93n22Weh1+vh7++PX/3qV3jyyScxefJkpzrvvvsuxo0bB41Gg7Fjx+KNN95wlMXExAAApkyZAkmSMG/evIH8GIiIBgyTWSIiD2lsbMRHH32E+Ph4hISE9Oua/Px8rFixAqtWrUJhYSFuueUWPPvss051Pv74Yzz33HN4/vnnUVBQgOjoaLz55pu96qxZswbPPfccjh49ij//+c94+umn8Y9//AMAsHv3bgDAN998g/LycmzdunUA3jER0cBTeLoBRETDyRdffAE/Pz8AQFNTE8LDw/HFF19AJutf38Krr76K1NRUPP744wCAMWPGYOfOncjIyHDUee2117BixQqkpaUBANasWYOvv/4ajY2Njjpr167FX/7yF/zkJz8BYO+JPXLkCN5++20sX74cYWFhAICQkBAYjcZrf+NERG7CnlkiokF0yy23oLCwEIWFhdi9ezcWLlyIRYsW4fTp0/26/ujRo0hKSnI6l5yc7PS8qKgIiYmJTue6P29qakJxcTFWrFgBPz8/x/Hss8+iuLj4Kt8ZEZFnsGeWiGgQ+fr6Ij4+3vH83XffRUBAAN555x0sWLAAACCEcJRbrdYBb0NXD+0777zTKzGWy+UDfj8iIndizywRkQdJkgSZTIaWlhbHn/bLy8sd5T2Xxho3bhzy8/Odzu3atcvpeUJCAvbs2eN0rvtzg8GAiIgInDp1CvHx8U5H18QvlUoFAOjo6Li2N0hE5GbsmSUiGkQWiwUmkwkAUFtbi7/97W9obGzEkiVLEB8fj6ioKDzzzDN47rnncPz4cfzlL39xuv6RRx7BrFmz8NJLL2Hp0qX46quvnMbLAsBvfvMbrFy5EtOnT8dNN92ETz/9FAcOHEBsbKyjzrp16/DII48gICAAqampsFgs2Lt3L2pra5Geng69Xg8fHx9kZGRgxIgR0Gg0CAgIcP8HRER0hdgzS0Q0iDIyMhAeHo7w8HAkJSVhz5492LJlC+bNmwelUolNmzbh2LFjuPHGG/H888/3Wqlg5syZeOedd/Dqq69i0qRJ+Prrr/HUU0851bn77ruxevVq/O53v8PUqVNRUlKC++67DxqNxlHnV7/6Fd599128//77mDhxIm6++WZs3LjR0TOrUCjw17/+FW+//TYiIiKwdOlS9384RERXQRLdB2cREdF1af78+TAajfjwww893RQiogHFYQZERNeZ5uZmvPXWW1i4cCHkcjk2bdqEb775Btu3b/d004iIBhx7ZomIrjMtLS1YsmQJ9u/fj9bWViQkJOCpp55yrClLRHQ9YTJLRERERF6LE8CIiIiIyGsxmSUiIiIir8VkloiIiIi8FpNZIiIiIvJaTGaJiIiIyGsxmSUiIiIir8VkloiIiIi8FpNZIiIiIvJa/x9VUVVAt6yKmQAAAABJRU5ErkJggg==",
      "text/plain": [
       "<Figure size 800x400 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "'''\n",
    "    This block plots the figure for the comparison under different budgets.\n",
    "    It can be directly ran on the json file created by previous blocks.\n",
    "'''\n",
    "\n",
    "import json\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "file_path = \"-\" # Choose the file path\n",
    "\n",
    "# Read JSON file\n",
    "with open(file_path, \"r\") as file:\n",
    "    data = json.load(file)\n",
    "\n",
    "# Extract data\n",
    "labels = [f\"{exp['budget']}\" for exp in data]\n",
    "methods = [\"our_algorithm\", \"merge_and_reduce\", \"batch_online\"]\n",
    "results = {method: [] for method in methods}\n",
    "\n",
    "# Collect results, capping errors >= 0.8 at 0.8\n",
    "for exp in data:\n",
    "    for method in methods:\n",
    "        error = exp[\"results\"][method][\"arithmetic_mean\"]\n",
    "        # Cap error at 0.8\n",
    "        # capped_error = min(error, 0.8)\n",
    "        results[method].append(error)\n",
    "\n",
    "# Plotting\n",
    "plt.figure(figsize=(8, 4))\n",
    "x = range(len(labels))  # Positions for all budget settings\n",
    "for method in methods:\n",
    "    plt.plot(x, results[method], marker='o', label=method)\n",
    "\n",
    "plt.xticks(x, labels)\n",
    "plt.xlabel(\"Budget\")\n",
    "plt.ylabel(\"Multiplicative Error\")\n",
    "plt.title(\"Comparison of Methods (Facebook, n = 1034, m = 53498)\")\n",
    "plt.ylim(0.2, 0.9) \n",
    "# Customize y-axis ticks to label 0.8 as \">0.8\"\n",
    "# plt.yticks([0.2, 0.4, 0.6, 0.8], [0.2, 0.4, 0.6, \">0.8\"])\n",
    "plt.legend()\n",
    "plt.grid(True)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "04f21a4b",
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "    This block implements our comparison under different budgets using the synthetic graph. \n",
    "    All hyperparameters can be found in the appendix of our paper, they are also listed in the experiment results.\n",
    "    C_ol is the constant factor for the online algorithm. \n",
    "    C_off is the constant factor for the offline algorithm.\n",
    "    C_ol_str is the constant factor for the online substream in our streaming algorithm. \n",
    "'''\n",
    "\n",
    "def comparison(n, edge_counts, C_ol_list, C_off, C_ol_str_list, iter=10, seed=None):\n",
    "    # Initialize result storage\n",
    "    results_by_edges = {m: {\n",
    "        \"batch_online\": {\"eigenvalues\": []},\n",
    "        \"merge_and_reduce\": {\"eigenvalues\": []},\n",
    "        \"our_algorithm\": {\"eigenvalues\": []},\n",
    "        \"parameters\": {\"n\": n, \"m\": m, \"C_ol\": C_ol, \"C_off\": C_off, \"C_ol_str\": C_ol_str, \"iterations\": iter, \"seed\": seed}\n",
    "    } for m, C_ol, C_ol_str in zip(edge_counts, C_ol_list, C_ol_str_list)}\n",
    "    output_file = \"-\" # Choose the output directory\n",
    "\n",
    "    # Set random seed for reproducibility\n",
    "    if seed is not None:\n",
    "        np.random.seed(seed)\n",
    "\n",
    "    for i in range(iter):\n",
    "        # Generate the base graph with the minimum number of edges\n",
    "        min_edges = min(edge_counts)\n",
    "        G_base = nx.MultiGraph()\n",
    "        G_base.add_nodes_from(range(n))\n",
    "        base_edges = []\n",
    "        for _ in range(min_edges):\n",
    "            u, v = np.random.choice(n, 2, replace=False)\n",
    "            weight = np.random.uniform(1, 10)\n",
    "            base_edges.append((u, v, weight))\n",
    "            G_base.add_edge(u, v, weight=weight)\n",
    "\n",
    "        for idx, m in enumerate(edge_counts):\n",
    "            # Create graph with m edges by adding to the base graph\n",
    "            G = nx.MultiGraph()\n",
    "            G.add_nodes_from(range(n))\n",
    "            # Add base edges\n",
    "            for u, v, weight in base_edges:\n",
    "                G.add_edge(u, v, weight=weight)\n",
    "            # Add additional edges if m > min_edges\n",
    "            if m > min_edges:\n",
    "                for _ in range(m - min_edges):\n",
    "                    u, v = np.random.choice(n, 2, replace=False)\n",
    "                    weight = np.random.uniform(1, 10)\n",
    "                    G.add_edge(u, v, weight=weight)\n",
    "\n",
    "            C_ol = C_ol_list[idx]\n",
    "            C_ol_str = C_ol_str_list[idx]\n",
    "\n",
    "            # Run online_batch\n",
    "            G_ol = online_batch(G, C_ol, 100, seed)\n",
    "            approx_ol = check(G, G_ol)\n",
    "            results_by_edges[m][\"batch_online\"][\"eigenvalues\"].append(approx_ol)\n",
    "\n",
    "            # Run merge_and_reduce\n",
    "            G_mr = merge_and_reduce(G, C_off, seed)\n",
    "            approx_mr = check(G, G_mr)\n",
    "            results_by_edges[m][\"merge_and_reduce\"][\"eigenvalues\"].append(approx_mr)\n",
    "\n",
    "            # Run streaming\n",
    "            G_str = streaming(G, C_ol_str, C_off, seed)\n",
    "            approx_str = check(G, G_str)\n",
    "            results_by_edges[m][\"our_algorithm\"][\"eigenvalues\"].append(approx_str)\n",
    "\n",
    "    # Compute means for each edge count\n",
    "    for m in edge_counts:\n",
    "        results_by_edges[m][\"batch_online\"][\"arithmetic_mean\"] = np.mean(np.array(results_by_edges[m][\"batch_online\"][\"eigenvalues\"]))\n",
    "        results_by_edges[m][\"merge_and_reduce\"][\"arithmetic_mean\"] = np.mean(np.array(results_by_edges[m][\"merge_and_reduce\"][\"eigenvalues\"]))\n",
    "        results_by_edges[m][\"our_algorithm\"][\"arithmetic_mean\"] = np.mean(np.array(results_by_edges[m][\"our_algorithm\"][\"eigenvalues\"]))\n",
    "        results_by_edges[m][\"m\"] = m\n",
    "\n",
    "    # Save results to JSON file\n",
    "    results = {\n",
    "        \"budget\": 1500,\n",
    "        \"results_by_edges\": results_by_edges\n",
    "    }\n",
    "    with open(output_file, 'w') as f:\n",
    "        json.dump(results, f, indent=4)\n",
    "\n",
    "    # Return geometric means for each edge count\n",
    "    return [(m, results_by_edges[m][\"batch_online\"][\"arithmetic_mean\"],\n",
    "             results_by_edges[m][\"merge_and_reduce\"][\"arithmetic_mean\"],\n",
    "             results_by_edges[m][\"our_algorithm\"][\"arithmetic_mean\"]) for m in edge_counts]\n",
    "\n",
    "'''\n",
    "# Example usage\n",
    "edge_counts = [(i + 1) * 10000 for i in range(10)]  # List of edge counts to compare\n",
    "print(comparison(n = 100, edge_counts = edge_counts, \n",
    "                 C_ol_list=[0.3, 0.2, 0.15, 0.1, 0.08, 0.05, 0.04, 0.02, 0.005, 0.002], \n",
    "                 C_off = 3.3, \n",
    "                 C_ol_str_list = [15, 7, 5.5, 4.8, 4.3, 4.05, 3.75, 3.75, 3.65, 3.2], iter=5, seed=42))\n",
    "                '''"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "e8d05271",
   "metadata": {},
   "outputs": [],
   "source": [
    "'''\n",
    "    This block plots the figure for the comparison under different number of edges.\n",
    "    It can be directly ran on the json file created by previous blocks.\n",
    "'''\n",
    "\n",
    "import json\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "file_path = '-' # Choose the file path\n",
    "\n",
    "# Read JSON file\n",
    "with open(file_path, \"r\") as file:\n",
    "    data = json.load(file)\n",
    "\n",
    "# Extract data\n",
    "labels = list(data[\"results_by_edges\"].keys())  # Get edge count keys\n",
    "methods = [\"our_algorithm\", \"merge_and_reduce\"]\n",
    "results = {method: [] for method in methods}\n",
    "\n",
    "# Collect results\n",
    "for edge_count in labels:\n",
    "    for method in methods:\n",
    "        error = data[\"results_by_edges\"][edge_count][method][\"arithmetic_mean\"]\n",
    "        results[method].append(error)\n",
    "\n",
    "# Plotting\n",
    "plt.figure(figsize=(10, 4))\n",
    "x = range(len(labels))  # Positions for edge counts\n",
    "for method in methods:\n",
    "    plt.plot(x, results[method], marker='o', label=method)\n",
    "\n",
    "plt.xticks(x, labels)\n",
    "plt.xlabel(\"Number of Edges (m)\")\n",
    "plt.ylabel(\"Multiplicative Error\")\n",
    "plt.title(\"Comparison of Methods (n = 100, Budget = 1500)\")\n",
    "plt.ylim(0.35, 0.55)  \n",
    "plt.yticks([0.35, 0.4, 0.45, 0.5, 0.55])  \n",
    "plt.legend()\n",
    "plt.grid(True)\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
