{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "7d895d8a",
   "metadata": {},
   "outputs": [],
   "source": [
    "#private mst and clustering\n",
    "import numpy as np\n",
    "from scipy.spatial.distance import cdist\n",
    "import networkx as nx\n",
    "\n",
    "def generate_embedding(X):\n",
    "    d, n = X.shape\n",
    "    dist_matrix = cdist(X.T, X.T)\n",
    "    G = nx.Graph()\n",
    "    for i in range(n):\n",
    "        for j in range(i + 1, n):\n",
    "            G.add_edge(i, j, weight=dist_matrix[i, j])\n",
    "    return G\n",
    "\n",
    "\n",
    "def post_processing_with_JL(X, epsilon, delta, eta):\n",
    "    # Initialize variables\n",
    "    T = set()  # This will store the MST edges\n",
    "    r = int (8 * np.log(2 / delta) / (eta ** 2))\n",
    "    # Step 2: Generate random matrix Π\n",
    "    m, n = X.shape\n",
    "    #print('reduced dimension:', m, '->', r)\n",
    "    Pi = np.random.randn(r, m)  # r x m random matrix, entries i.i.d. sampled from N(0,1)\n",
    "\n",
    "    # Step 3: Euclidean embedding X_hat = (1/sqrt(r)) * Pi * X\n",
    "    X_hat = Pi @ X\n",
    "    X_hat = (1 / np.sqrt(r)) * X_hat\n",
    "\n",
    "    # Step 4: Add Laplace noise to each column\n",
    "    X_tilde = np.zeros_like(X_hat)\n",
    "    for i in range(n):\n",
    "        zi = np.random.laplace(scale=(np.sqrt(np.log(2 / delta)) / (eta * epsilon)), size = r)\n",
    "        if i == 0:\n",
    "            #print('laplace:', np.linalg.norm(zi))\n",
    "            pass\n",
    "        X_tilde[:, i] = X_hat[:, i] + zi\n",
    "\n",
    "    # Step 5: Construct the Euclidean embedding G\n",
    "    dist_matrix = cdist(X_tilde.T, X_tilde.T)  # Pairwise distances for MST\n",
    "    # Create a graph from the distance matrix\n",
    "    G = nx.Graph()\n",
    "    for i in range(n):\n",
    "        for j in range(i + 1, n):\n",
    "            G.add_edge(i, j, weight=dist_matrix[i, j])\n",
    "\n",
    "    return G\n",
    "    # Step 7: Build the tree T using cluster-based edge selection\n",
    "    # We can implement a simplified version of the cluster-based strategy\n",
    "    \"\"\"\"\n",
    "    clusters = {i: {i} for i in range(n)}  # Each point starts as its own cluster\n",
    "    while len(clusters) > 1:\n",
    "        linkedges = set()\n",
    "        for c in clusters:\n",
    "            #need to find the minimum edge goes out of c\n",
    "                minedge = None\n",
    "                minvalue = np.inf\n",
    "                for u in c:\n",
    "                    for v in range(n):\n",
    "                        if (v not in c) and (G[u][v]['weight'] < minvalue):\n",
    "                            minedge = (u,v)\n",
    "                            minvalue = G[u][v]['weight']\n",
    "                linkedges.add(minedge)\n",
    "        for (u, v) in linkedges:\n",
    "            if clusters[u] != clusters[v]:  # If they belong to different clusters\n",
    "                T.add((u, v))\n",
    "                # Merge clusters\n",
    "                old_cluster, new_cluster = clusters[u], clusters[v]\n",
    "                for i in range(n):\n",
    "                    if clusters[i] == old_cluster:\n",
    "                        clusters[i] = new_cluster\n",
    "    \n",
    "    return T\n",
    "\n",
    "\"\"\"\n",
    "\n",
    "\n",
    "def boruvka_mst(G, k = -1):\n",
    "    \"\"\"\n",
    "    Compute the Minimum Spanning Tree (MST) of an undirected weighted graph using Borůvka's algorithm.\n",
    "    \n",
    "    Parameters:\n",
    "    - G: A connected, undirected NetworkX graph with edge weights under attribute 'weight'.\n",
    "    \n",
    "    Returns:\n",
    "    - A NetworkX graph representing the MST.\n",
    "    \"\"\"\n",
    "    # Initialize: each node is its own component\n",
    "    parent = {node: node for node in G.nodes()}\n",
    "    n = len(G.nodes())\n",
    "    \n",
    "    def find(u):\n",
    "        \"\"\"Disjoint set find with path compression.\"\"\"\n",
    "        while parent[u] != u:\n",
    "            parent[u] = parent[parent[u]]\n",
    "            u = parent[u]\n",
    "        return u\n",
    "\n",
    "    def union(u, v):\n",
    "        \"\"\"Union two components.\"\"\"\n",
    "        u_root, v_root = find(u), find(v)\n",
    "        if u_root != v_root:\n",
    "            parent[v_root] = u_root\n",
    "            return True\n",
    "        return False\n",
    "\n",
    "    mst_edges = set()\n",
    "\n",
    "    num_components = len(G.nodes())\n",
    "    i = 0\n",
    "    while num_components > 1 and (k == -1 or i < k):\n",
    "        i += 1\n",
    "        # Step 1: Find cheapest outgoing edge for each component\n",
    "        cheapest = {}\n",
    "\n",
    "        for u, v, data in G.edges(data=True):\n",
    "            w = data[\"weight\"]\n",
    "            u_root, v_root = find(u), find(v)\n",
    "            if u_root == v_root:\n",
    "                continue  # Same component, skip\n",
    "\n",
    "            # Update cheapest for u's component\n",
    "            if u_root not in cheapest or cheapest[u_root][2] > w:\n",
    "                cheapest[u_root] = (u, v, w)\n",
    "\n",
    "            # Update cheapest for v's component\n",
    "            if v_root not in cheapest or cheapest[v_root][2] > w:\n",
    "                cheapest[v_root] = (u, v, w)\n",
    "\n",
    "        # Step 2: Add selected edges and merge components\n",
    "        for u, v, w in cheapest.values():\n",
    "            if union(u, v):\n",
    "                mst_edges.add((u, v))\n",
    "                num_components -= 1\n",
    "\n",
    "    components = [set() for _ in range(n)]\n",
    "    # Build the MST graph\n",
    "    if k == -1:\n",
    "        MST = nx.Graph()\n",
    "        MST.add_edges_from(mst_edges)\n",
    "        return MST\n",
    "    else:\n",
    "        for u in G.nodes():\n",
    "            components[find(u)].add(u)\n",
    "        components_final = [x for x in components if len(x) != 0]\n",
    "        return components_final\n",
    "\n",
    "def cluster_cost(G, cluster):\n",
    "    sum_cost = 0\n",
    "    for comp in cluster:\n",
    "        G_sub = G.subgraph(comp)\n",
    "        T = nx.minimum_spanning_tree(G_sub)\n",
    "        for u,v in T.edges:\n",
    "            sum_cost += G[u][v]['weight']\n",
    "    return sum_cost\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "834251ab",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[{0, 1, 2, 3, 5, 6, 7, 8}, {9, 4}]\n",
      "[0, 1, 2, 3, 5, 6, 7, 8]\n"
     ]
    }
   ],
   "source": [
    "#user example testing\n",
    "n = 10\n",
    "d = 10\n",
    "X = np.random.normal(0,10,(d,n))\n",
    "G = generate_embedding(X)\n",
    "T = boruvka_mst(G, k =1)\n",
    "print(T)\n",
    "G1 = G.subgraph(T[0])\n",
    "print(G1.nodes())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "328cf686",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "#Generating datasets\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "from scipy.spatial import distance_matrix\n",
    "from scipy.sparse.csgraph import minimum_spanning_tree\n",
    "\n",
    "def generate_unconcentrated_dataset(n_points=100, n_clusters=4, n_outliers=5, dim=2, seed=None):\n",
    "    np.random.seed(seed)\n",
    "\n",
    "    points = []\n",
    "\n",
    "    # Generate clusters\n",
    "    for _ in range(n_clusters):\n",
    "        center = np.random.uniform(-100, 100, size=dim)\n",
    "        std_dev = np.random.uniform(1, 20)  # Different spread per cluster\n",
    "        cluster_size = np.random.randint(n_points // (2 * n_clusters), n_points // n_clusters + 1)\n",
    "        cluster_points = center + np.random.randn(cluster_size, dim) * std_dev\n",
    "        points.append(cluster_points)\n",
    "\n",
    "    # Generate outliers\n",
    "    outliers = np.random.uniform(-200, 200, size=(n_outliers, dim))\n",
    "    points.append(outliers)\n",
    "\n",
    "    all_points = np.vstack(points)\n",
    "\n",
    "    return all_points.T\n",
    "\n",
    "# Example usage:\n",
    "points = generate_unconcentrated_dataset(n_points=150, n_clusters=8, n_outliers=4, dim=2, seed=21)\n",
    "points = points.T\n",
    "# print(points)\n",
    "#Visualize\n",
    "plt.scatter(points[:, 0], points[:, 1], c='blue')\n",
    "#plt.title(\"Unconcentrated Point Cloud\")\n",
    "plt.axis('equal')\n",
    "plt.grid(linestyle = '--', linewidth = 1)\n",
    "\n",
    "plt.savefig('dataset.pdf')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "4bdd0642",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Gaussian Mechanism over edge weights\n",
    "def gaussian(X, eps, delta):\n",
    "    G = generate_embedding(X)\n",
    "    n = G.number_of_nodes()\n",
    "    sigma = np.sqrt(2 * n * np.log(1.25/delta)) / eps\n",
    "    for u, v in G.edges:\n",
    "        noise = np.random.normal(0, sigma)\n",
    "        G[u][v]['weight'] += noise\n",
    "    return G"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "fba6aecf",
   "metadata": {},
   "outputs": [],
   "source": [
    "def post_processing_without_JL(X, eps, delta):\n",
    "    m, n = X.shape\n",
    "    X_tilde = np.zeros_like(X)\n",
    "    sigma =  np.sqrt(2 * np.sqrt(n) * np.log(1.25/delta)) / eps\n",
    "    for i in range(n):\n",
    "        zi = np.random.normal(0, sigma, m)\n",
    "        if i == 0:\n",
    "            #print('gaussian:', np.linalg.norm(zi))\n",
    "            pass\n",
    "        X_tilde[:, i] = X[:, i] + zi\n",
    "\n",
    "    dist_matrix = cdist(X_tilde.T, X_tilde.T)  # Pairwise distances for MST\n",
    "    # Create a graph from the distance matrix\n",
    "    G = nx.Graph()\n",
    "    for i in range(n):\n",
    "        for j in range(i + 1, n):\n",
    "            G.add_edge(i, j, weight=dist_matrix[i, j])\n",
    "\n",
    "    return G"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "064313a4",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "5.66646550821951\n"
     ]
    }
   ],
   "source": [
    "# Example usage\n",
    "X = np.random.randn(100, 5)  # Example dataset of shape (100, 50)\n",
    "epsilon = 0.5\n",
    "delta = 0.1\n",
    "eta = 0.1\n",
    "\n",
    "G_private = post_processing_with_JL(X, epsilon, delta, eta)\n",
    "mst = boruvka_mst(G_private)\n",
    "G = generate_embedding(X)\n",
    "T = nx.minimum_spanning_tree(G)\n",
    "value_mst = 0\n",
    "for u,v in T.edges:\n",
    "    value_mst += T[u][v]['weight']\n",
    "sum_weight = 0\n",
    "for u, v in mst.edges:\n",
    "    sum_weight +=  G[u][v]['weight']\n",
    "print(sum_weight - value_mst)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "c31a07d5",
   "metadata": {},
   "outputs": [],
   "source": [
    "from scipy.interpolate import make_interp_spline\n",
    "from scipy import interpolate\n",
    "def smooth(x,y,sample_size):\n",
    "    new_x = np.linspace(x.min(),x.max(),sample_size)\n",
    "    func = interpolate.interp1d(x, y, kind='cubic')\n",
    "    new_y = func(new_x)\n",
    "    x_smooth = np.linspace(new_x.min(),new_x.max(),len(x))\n",
    "    y_smooth = make_interp_spline(new_x,new_y)(x_smooth)\n",
    "    return y_smooth"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "5b2e0aef",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "100\n",
      "300\n",
      "500\n",
      "700\n",
      "900\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "#Testing our algorithms on Gaussian data\n",
    "import numpy as np\n",
    "import matplotlib\n",
    "from matplotlib import pyplot as plt\n",
    "import scipy.stats\n",
    "\n",
    "'''\n",
    "d = 1000\n",
    "n = 10\n",
    "X = np.random.randn(n,d)\n",
    "norms = np.linalg.norm(X, axis=1, keepdims=True)\n",
    "X = X / norms\n",
    "X = X.T\n",
    "'''\n",
    "\n",
    "epsilon = 1\n",
    "delta = 0.001\n",
    "eta = 1\n",
    "d = 10\n",
    "\n",
    "real_mst = []\n",
    "private_mst_withJL = []\n",
    "private_mst_trivial = []\n",
    "\n",
    "min_n = 100\n",
    "max_n = 1000\n",
    "step_size = 200\n",
    "\n",
    "for n in range(min_n,max_n+1,step_size):\n",
    "    print(n)\n",
    "    # generate random data\n",
    "    #X = np.random.randn(n,d)\n",
    "    X = np.random.normal(0,10,(n,d))\n",
    "    #norms = np.linalg.norm(X, axis=1, keepdims=True)\n",
    "    #X = X / norms\n",
    "    X = X.T\n",
    "    # computing mst\n",
    "    G = generate_embedding(X)\n",
    "    T = nx.minimum_spanning_tree(G)\n",
    "    \n",
    "    value_mst = 0\n",
    "    for u,v in T.edges:\n",
    "        value_mst += G[u][v]['weight']\n",
    "    real_mst.append(value_mst)\n",
    "\n",
    "    G_private_JL = post_processing_with_JL(X, epsilon, delta, eta)\n",
    "\n",
    "    T = nx.minimum_spanning_tree(G_private_JL)\n",
    "    \n",
    "    value_mst = 0\n",
    "    for u,v in T.edges:\n",
    "        value_mst += G[u][v]['weight']\n",
    "    private_mst_withJL.append(value_mst)\n",
    "\n",
    "    #G_private_withoutJL = gaussian(X,epsilon,delta)\n",
    "\n",
    "    #T = nx.minimum_spanning_tree(G_private_withoutJL)\n",
    "    \n",
    "    #value_mst = 0\n",
    "    #for u,v in T.edges:\n",
    "    #    value_mst += G[u][v]['weight']\n",
    "    #private_mst_trivial.append(value_mst)\n",
    "    \n",
    "\n",
    "matplotlib.rcParams['pdf.fonttype'] = 42\n",
    "matplotlib.rcParams['ps.fonttype'] = 42\n",
    "\n",
    "x = private_mst_withJL\n",
    "#y = private_mst_trivial\n",
    "epoch = np.array(list(range(min_n,max_n+1,step_size)))\n",
    "#x_loss = smooth(epoch, x, 4)\n",
    "#y_loss = smooth(epoch, y, 4)\n",
    "plt.figure(1)\n",
    "plt.xlabel(\"Number of Vertices\",fontdict={'family' : 'Times New Roman', 'size'   : 16, 'weight' : 'bold'})\n",
    "#plt.ylabel(\"Weight of Spanning Tree\",fontdict={'family' : 'Times New Roman', 'size'   : 16, 'weight' : 'bold'})\n",
    "#plt.fill_between(epoch,x,y,facecolor=\"orchid\",alpha = 0.4)\n",
    "#plt.fill_between(epoch,y,y_loss,facecolor=\"red\",alpha = 0.5)\n",
    "\n",
    "plt.plot(epoch,real_mst,c = 'green',linestyle = ':', marker='o', label = 'real mst')\n",
    "\n",
    "plt.plot(epoch,x,c = 'fuchsia',linestyle = '-',  marker='^', label = 'JLT')\n",
    "#plt.plot(epoch,x_loss,c = 'hotpink',linestyle = '-')\n",
    "#plt.plot(epoch,y,c = 'blue',linestyle = '-.',  marker='v', label = 'trivial')\n",
    "#plt.plot(epoch,y_loss,c = 'hotpink',linestyle = '-')\n",
    "\n",
    "plt.legend()\n",
    "plt.grid(linestyle = '--', linewidth = 1)\n",
    "\n",
    "plt.savefig('gaussian_data.pdf')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "b43c6f86",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1000\n",
      "2000\n",
      "3000\n",
      "4000\n",
      "5000\n",
      "[1380412.7011931017, 1502165.8960883524, 1447938.9681769821, 1418877.4352678938, 1707496.2587785746] [1801832.0126223722, 2529383.893055293, 3294004.1420537173, 3762431.09227679, 4514821.165377545]\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "#Testing with adding noise to coordinates on unconcentrated dataset\n",
    "import numpy as np\n",
    "import matplotlib\n",
    "from matplotlib import pyplot as plt\n",
    "import scipy.stats\n",
    "\n",
    "'''\n",
    "d = 1000\n",
    "n = 10\n",
    "X = np.random.randn(n,d)\n",
    "norms = np.linalg.norm(X, axis=1, keepdims=True)\n",
    "X = X / norms\n",
    "X = X.T\n",
    "'''\n",
    "\n",
    "epsilon = 0.005 # eps = 1\n",
    "delta = 0.001\n",
    "eta = 1\n",
    "n = 800\n",
    "\n",
    "real_mst = []\n",
    "private_mst_withJL = []\n",
    "private_mst_trivial = []\n",
    "\n",
    "min_d = 1000\n",
    "max_d = 5000\n",
    "step_size = 1000\n",
    "\n",
    "for d in range(min_d,max_d+1,step_size):\n",
    "    print(d)\n",
    "    # generate random data\n",
    "    #X = np.random.randn(n,d)\n",
    "    #X = np.random.normal(0,10,(n,d))\n",
    "    X = generate_unconcentrated_dataset(n_points=n, n_clusters=10, n_outliers = int (0.1 * n), dim=d)\n",
    "    #norms = np.linalg.norm(X, axis=1, keepdims=True)\n",
    "    #X = X / norms\n",
    "    #X = X.T\n",
    "    # computing mst\n",
    "    G = generate_embedding(X)\n",
    "    T = nx.minimum_spanning_tree(G)\n",
    "    \n",
    "    value_mst = 0\n",
    "    for u,v in T.edges:\n",
    "        value_mst += G[u][v]['weight']\n",
    "    real_mst.append(value_mst)\n",
    "\n",
    "    G_private_JL = post_processing_with_JL(X, epsilon, delta, eta)\n",
    "\n",
    "    T = nx.minimum_spanning_tree(G_private_JL)\n",
    "    \n",
    "    value_mst = 0\n",
    "    for u,v in T.edges:\n",
    "        value_mst += G[u][v]['weight']\n",
    "    private_mst_withJL.append(value_mst)\n",
    "\n",
    "    G_private_withoutJL = post_processing_without_JL(X,epsilon,delta)\n",
    "\n",
    "    T = nx.minimum_spanning_tree(G_private_withoutJL)\n",
    "    \n",
    "    value_mst = 0\n",
    "    for u,v in T.edges:\n",
    "        value_mst += G[u][v]['weight']\n",
    "    private_mst_trivial.append(value_mst)\n",
    "    \n",
    "\n",
    "matplotlib.rcParams['pdf.fonttype'] = 42\n",
    "matplotlib.rcParams['ps.fonttype'] = 42\n",
    "\n",
    "x = private_mst_withJL\n",
    "y = private_mst_trivial\n",
    "print(x,y)\n",
    "epoch = np.array(list(range(min_d,max_d+1,step_size)))\n",
    "#x_loss = smooth(epoch, x, 4)\n",
    "#y_loss = smooth(epoch, y, 4)\n",
    "plt.figure(1)\n",
    "plt.xlabel(\"Number of Dimension\",fontdict={'family' : 'Times New Roman', 'size'   : 16, 'weight' : 'bold'})\n",
    "plt.ylabel(\"Weight of Spanning Tree\",fontdict={'family' : 'Times New Roman', 'size'   : 16, 'weight' : 'bold'})\n",
    "plt.fill_between(epoch,x,y,facecolor=\"orchid\",alpha = 0.4)\n",
    "#plt.fill_between(epoch,y,y_loss,facecolor=\"red\",alpha = 0.5)\n",
    "\n",
    "plt.plot(epoch,real_mst,c = 'green',linestyle = ':', marker='o', label = 'real mst')\n",
    "\n",
    "plt.plot(epoch,x,c = 'fuchsia',linestyle = '-',  marker='^', label = 'JLT')\n",
    "#plt.plot(epoch,x_loss,c = 'hotpink',linestyle = '-')\n",
    "plt.plot(epoch,y,c = 'blue',linestyle = '-.',  marker='v', label = 'trivial')\n",
    "#plt.plot(epoch,y_loss,c = 'hotpink',linestyle = '-')\n",
    "\n",
    "plt.ticklabel_format(axis='y', style='sci', scilimits=(0,0))\n",
    "plt.legend()\n",
    "plt.grid(linestyle = '--', linewidth = 1)\n",
    "\n",
    "plt.savefig('coordinate_noise.pdf')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 99,
   "id": "686abcca",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "1000\n",
      "2000\n",
      "3000\n",
      "4000\n",
      "5000\n",
      "[83353.00074716721, 106610.66037015384, 141371.2861239856, 136606.18770111533, 164564.7085096325] [127458.23029822108, 217569.41775052136, 230720.40168117703, 270633.586632006, 284587.24077041284]\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "#affinity clustering: testing on non-sequential clustering\n",
    "\n",
    "import numpy as np\n",
    "import matplotlib\n",
    "from matplotlib import pyplot as plt\n",
    "import scipy.stats\n",
    "\n",
    "'''\n",
    "d = 1000\n",
    "n = 10\n",
    "X = np.random.randn(n,d)\n",
    "norms = np.linalg.norm(X, axis=1, keepdims=True)\n",
    "X = X / norms\n",
    "X = X.T\n",
    "'''\n",
    "\n",
    "epsilon = 0.01 # eps = 1\n",
    "delta = 0.001\n",
    "eta = 1\n",
    "n = 100\n",
    "\n",
    "private_cost_withJL = []\n",
    "private_cost_trivial = []\n",
    "cost_nonprivate = []\n",
    "private_cost_edgeflip = []\n",
    "\n",
    "min_d = 1000\n",
    "max_d = 5000\n",
    "step_size = 1000\n",
    "\n",
    "num_step = 1\n",
    "\n",
    "for d in range(min_d,max_d+1,step_size):\n",
    "    print(d)\n",
    "    # generate random data\n",
    "    #X = np.random.randn(n,d)\n",
    "    #X = np.random.normal(0,10,(n,d))\n",
    "    X = generate_unconcentrated_dataset(n_points=n, n_clusters=10, n_outliers = int (0.1 * n), dim=d)\n",
    "    #norms = np.linalg.norm(X, axis=1, keepdims=True)\n",
    "    #X = X / norms\n",
    "    #X = X.T\n",
    "    # computing mst\n",
    "    G = generate_embedding(X)\n",
    "    #T = nx.minimum_spanning_tree(G)\n",
    "    \n",
    "    #value_mst = 0\n",
    "    #for u,v in T.edges:\n",
    "    #    value_mst += G[u][v]['weight']\n",
    "    #real_mst.append(value_mst)\n",
    "\n",
    "    G_private_JL = post_processing_with_JL(X, epsilon, delta, eta)\n",
    "    clusters = boruvka_mst(G_private_JL, k = num_step)\n",
    "    private_cost_withJL.append(cluster_cost(G,clusters))\n",
    "\n",
    "    #T = nx.minimum_spanning_tree(G_private_JL)\n",
    "    \n",
    "    #value_mst = 0\n",
    "    #for u,v in T.edges:\n",
    "    #    value_mst += G[u][v]['weight']\n",
    "    #private_mst_withJL.append(value_mst)\n",
    "\n",
    "    G_private_withoutJL = post_processing_without_JL(X,epsilon,delta)\n",
    "    clusters = boruvka_mst(G_private_withoutJL, k = num_step)\n",
    "    private_cost_trivial.append(cluster_cost(G,clusters))\n",
    "\n",
    "    #T = nx.minimum_spanning_tree(G_private_withoutJL)\n",
    "    clusters = boruvka_mst(G, k = num_step)\n",
    "    cost_nonprivate.append(cluster_cost(G,clusters))\n",
    "    #value_mst = 0\n",
    "    #for u,v in T.edges:\n",
    "    #    value_mst += G[u][v]['weight']\n",
    "    #private_mst_trivial.append(value_mst)\n",
    "    G_private_edgeflip = gaussian(X,epsilon,delta)\n",
    "    clusters = boruvka_mst(G_private_edgeflip, k = num_step)\n",
    "    private_cost_edgeflip.append(cluster_cost(G,clusters))\n",
    "    \n",
    "\n",
    "matplotlib.rcParams['pdf.fonttype'] = 42\n",
    "matplotlib.rcParams['ps.fonttype'] = 42\n",
    "\n",
    "x = private_cost_withJL\n",
    "y = private_cost_trivial\n",
    "z = cost_nonprivate\n",
    "w = private_cost_edgeflip\n",
    "print(x,y)\n",
    "epoch = np.array(list(range(min_d,max_d+1,step_size)))\n",
    "#x_loss = smooth(epoch, x, 4)\n",
    "#y_loss = smooth(epoch, y, 4)\n",
    "plt.figure(1)\n",
    "plt.xlabel(\"Number of Dimension\",fontdict={'family' : 'Times New Roman', 'size'   : 16, 'weight' : 'bold'})\n",
    "plt.ylabel(\"Cost of clustering (k = 1)\",fontdict={'family' : 'Times New Roman', 'size'   : 16, 'weight' : 'bold'})\n",
    "#plt.fill_between(epoch,x,y,facecolor=\"orchid\",alpha = 0.4)\n",
    "#plt.fill_between(epoch,y,y_loss,facecolor=\"red\",alpha = 0.5)\n",
    "\n",
    "plt.plot(epoch,w,c = 'slateblue',linestyle = '--', marker='s', label = 'edge-flip')\n",
    "plt.plot(epoch,z,c = 'green',linestyle = ':', marker='o', label = 'non-private')\n",
    "\n",
    "plt.plot(epoch,x,c = 'fuchsia',linestyle = '-',  marker='^', label = 'JLT')\n",
    "#plt.plot(epoch,x_loss,c = 'hotpink',linestyle = '-')\n",
    "plt.plot(epoch,y,c = 'blue',linestyle = '-.',  marker='v', label = 'trivial')\n",
    "#plt.plot(epoch,y_loss,c = 'hotpink',linestyle = '-')\n",
    "\n",
    "plt.legend()\n",
    "plt.grid(linestyle = '--', linewidth = 1)\n",
    "plt.ticklabel_format(axis='y', style='sci', scilimits=(0,0))\n",
    "\n",
    "plt.savefig('affinity_k=1.pdf')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "18b645dd",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "54.11132475878049 54.11132475878049\n"
     ]
    }
   ],
   "source": [
    "#test syntax\n",
    "import numpy as np\n",
    "import networkx as nx\n",
    "from scipy.spatial.distance import cdist\n",
    "np.inf>1000\n",
    "G = nx.Graph()\n",
    "for i in range(5):\n",
    "    for j in range(i + 1, 5):\n",
    "        G.add_edge(i, j, weight=4)\n",
    "a = set()\n",
    "a.add(1)\n",
    "\n",
    "def generate_embedding(X):\n",
    "    d, n = X.shape\n",
    "    dist_matrix = cdist(X.T, X.T)\n",
    "    G = nx.Graph()\n",
    "    for i in range(n):\n",
    "        for j in range(i + 1, n):\n",
    "            G.add_edge(i, j, weight=dist_matrix[i, j])\n",
    "    return G\n",
    "\n",
    "X = np.random.randn(3, 5)\n",
    "G = generate_embedding(X)\n",
    "sum_weight = 0\n",
    "'''\n",
    "for u, v in G.edges:\n",
    "    print(G[u][v]['weight'])\n",
    "\n",
    "G = gaussian(G,1,0.001)\n",
    "print('after')\n",
    "\n",
    "for u, v in G.edges:\n",
    "    print(G[u][v]['weight'])\n",
    "'''\n",
    "\n",
    "\n",
    "X = np.random.randn(100, 5)  # Example dataset of shape (100, 50)\n",
    "\n",
    "G = generate_embedding(X)\n",
    "T = nx.minimum_spanning_tree(G)\n",
    "mst = boruvka_mst(G)\n",
    "value_mst = 0\n",
    "value_bro = 0\n",
    "for u,v in T.edges:\n",
    "    value_mst += G[u][v]['weight']\n",
    "for u, v in mst.edges:\n",
    "    value_bro +=  G[u][v]['weight']\n",
    "print(value_mst, value_bro)\n"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "base",
   "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
}
