{
 "cells": [
  {
   "cell_type": "markdown",
   "id": "1e3635f5",
   "metadata": {},
   "source": [
    "# Sparsity evaluation on different distributions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "35205a72",
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from experiments.plot_sparsity import plot_sparsity"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "id": "be12ff08",
   "metadata": {},
   "outputs": [],
   "source": [
    "# first we define the experimental setup\n",
    "\n",
    "# random graph parameters\n",
    "T = 1000 # total number of timesteps \n",
    "d = 10 # number of nodes of the graph\n",
    "n = 100 # number of data samples = number of individual time-sequences \n",
    "sparsity =  0.05\n",
    "threshold = 0.1\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "id": "137b7367",
   "metadata": {},
   "outputs": [],
   "source": [
    "# Bernoulli & Uniform distribution\n",
    "pos = np.random.choice([0, 1], size=(n * d * T), p=[1 - sparsity, sparsity]) \n",
    "sign = np.random.choice([-1, 1], size=(n * d * T)) \n",
    "C_bernoulli = pos * np.random.uniform(0.1, 0.4, size=(n * d * T))  * sign\n",
    "\n",
    "std = threshold / 3\n",
    "Nf = np.random.normal(scale=std, size=(n * d * T))\n",
    "C_bernoulli = C_bernoulli + Nf\n",
    "\n",
    "# Gauss distribution\n",
    "sigma = 0.051\n",
    "C_gauss = np.random.normal(scale=sigma, size=(n * d * T))\n",
    "\n",
    "# Laplace distribution\n",
    "beta = threshold / 3\n",
    "C_laplace = np.random.laplace(loc=0, scale=beta, size=(n * d * T))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "32a490e1",
   "metadata": {},
   "source": [
    "## Population with regard to the threshold "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "id": "8f849a83",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Sparsity measure Bernoulli\n",
      "0.9501 of values are less than 0.10\n",
      "0.0499 of values are greater than 0.10\n",
      "0.0409 of values are greater than 0.15\n",
      "0.0329 of values are greater than 0.20\n",
      "\n",
      "Sparsity measure Gauss\n",
      "0.9503 of values are less than 0.10\n",
      "0.0497 of values are greater than 0.10\n",
      "0.0032 of values are greater than 0.15\n",
      "0.0001 of values are greater than 0.20\n",
      "\n",
      "Sparsity measure Laplace\n",
      "0.9502 of values are less than 0.10\n",
      "0.0498 of values are greater than 0.10\n",
      "0.0111 of values are greater than 0.15\n",
      "0.0025 of values are greater than 0.20\n"
     ]
    }
   ],
   "source": [
    "all = n * T * d \n",
    "\n",
    "print(\"Sparsity measure Bernoulli\")\n",
    "print(\"{:.4f} of values are less than {:.2f}\".format(np.where(np.abs(C_bernoulli) < threshold, 1, 0).sum() / all, threshold))\n",
    "print(\"{:.4f} of values are greater than {:.2f}\".format(np.where(np.abs(C_bernoulli) > threshold, 1, 0).sum() / all, threshold))\n",
    "print(\"{:.4f} of values are greater than {:.2f}\".format(np.where(np.abs(C_bernoulli) > 1.5 * threshold, 1, 0).sum() / all, 1.5 * threshold))\n",
    "print(\"{:.4f} of values are greater than {:.2f}\\n\".format(np.where(np.abs(C_bernoulli) > 2 * threshold, 1, 0).sum() / all, 2 * threshold))\n",
    "\n",
    "print(\"Sparsity measure Gauss\")\n",
    "print(\"{:.4f} of values are less than {:.2f}\".format(np.where(np.abs(C_gauss) < threshold, 1, 0).sum() / all, threshold))\n",
    "print(\"{:.4f} of values are greater than {:.2f}\".format(np.where(np.abs(C_gauss) > threshold, 1, 0).sum() / all, threshold))\n",
    "print(\"{:.4f} of values are greater than {:.2f}\".format(np.where(np.abs(C_gauss) > 1.5 * threshold, 1, 0).sum() / all, 1.5 * threshold))\n",
    "print(\"{:.4f} of values are greater than {:.2f}\\n\".format(np.where(np.abs(C_gauss) > 2 * threshold, 1, 0).sum() / all, 2 * threshold))\n",
    "\n",
    "print(\"Sparsity measure Laplace\")\n",
    "print(\"{:.4f} of values are less than {:.2f}\".format(np.where(np.abs(C_laplace) < threshold, 1, 0).sum() / all, threshold))\n",
    "print(\"{:.4f} of values are greater than {:.2f}\".format(np.where(np.abs(C_laplace) > threshold, 1, 0).sum() / all, threshold))\n",
    "print(\"{:.4f} of values are greater than {:.2f}\".format(np.where(np.abs(C_laplace) > 1.5 * threshold, 1, 0).sum() / all, 1.5 * threshold))\n",
    "print(\"{:.4f} of values are greater than {:.2f}\".format(np.where(np.abs(C_laplace) > 2 * threshold, 1, 0).sum() / all, 2 * threshold))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "id": "b55c2a38",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<Figure size 640x480 with 0 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n",
      "findfont: Font family 'gillsans' not found.\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAk0AAAGhCAYAAAB8lIA8AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy81sbWrAAAACXBIWXMAAA9hAAAPYQGoP6dpAABGEElEQVR4nO3de1hVdd7//xcHOSgCoQKSppRH0iRAkazMJNGwO7/ipI1jaB7SwEbJ4+SgHT00HlMzsxFn1HvUu1u/JRNqOFIpeYCYPHfSsNGNZgGJCirr94c/1petJAtEQHo+rmtdl6zPe631/uy9iVdrr722g2EYhgAAAHBDjjXdAAAAwO2A0AQAAGABoQkAAMACQhMAAIAFhCYAAAALCE0AAAAWEJoAAAAsIDQBAABYQGi6gfPnz+vw4cO6cOFCTbcCAABqmHNNN1Cbff/992rfvn1NtwEAAGoBzjTdwMWLFxUSEqLDhw/XdCsAAKCGEZpuwDAMffHFF7w9BwAACE0AAABWEJoAAAAsIDQBAABYQGgCAACwgFsOAAB+8/Lz85WXl1fTbaACnJ2d5efnJ0fH6jv/Q2gCAPymnT59Wg4ODmrWrJkcHBxquh1YdO7cOeXk5Khp06bVdkzengMA/KYVFhaqSZMmBKbbjIeHhy5fvlytxyQ0AQAAWEBoAgAAsIDQBAAAYAEXggMAUIYT0WHVdqzmyfuq7Vi1xfHjxxUYGKgvvvhCwcHB2rFjh3r06KGff/5Z3t7eSkpK0rhx45Sbm1vTrZo40wQAwG1o6NChcnBwMJdGjRqpd+/e+vLLL2u6tSoxcOBAffXVVzXdhh1CEwAAt6nevXvr1KlTOnXqlFJTU+Xs7Ky+fftWen9FRUVV2N3NcXd3l6+vb023YYfQBADAbcrV1VX+/v7y9/dXcHCwpkyZohMnTujMmTOSpBMnTuipp56St7e3fHx89OSTT+r48ePm9kOHDlW/fv30+uuvKyAgQG3bttXx48fl4OCg//3f/1WPHj1Uv359derUSenp6XbHfv/993XvvffK1dVVLVu21Ny5c+3GHRwctGnTJrt1JW+7WZGUlCRvb++KPiS3FNc01aDKvl/+W3zvGwBwY+fOndPq1avVqlUrNWrUSJcuXVJUVJQiIiL06aefytnZWa+99pr5Fp6Li4skKTU1VZ6entq2bZvd/l566SX95S9/UevWrfXSSy/p6aef1jfffCNnZ2dlZGToqaee0owZMzRw4EDt2rVLzz//vBo1aqShQ4fWwOyrB6EJAIDb1ObNm+Xh4SFJKigoUNOmTbV582Y5Ojpq7dq1Ki4u1ooVK8wbd65cuVLe3t7asWOHevXqJUlq0KCBVqxYYYaokjNREyZMUHR0tCTp5Zdf1r333qtvvvlG7dq107x589SzZ0/9+c9/liS1adNGhw4d0ptvvlmnQxNvzwEAcJvq0aOHsrKylJWVpT179igqKkp9+vTR999/r3//+9/65ptv1LBhQ3l4eMjDw0M+Pj66ePGivv32W3MfHTt2NANTaffdd5/575KvKjl9+rQk6fDhw+rWrZtdfbdu3fT111/rypUrt2KqtQJnmgAAuE01aNBArVq1Mn9esWKFvLy89O677+rcuXMKDQ3VmjVrrtuuSZMmdvsoS7169cx/l5ypKi4uttybg4ODDMOwW3fp0iXL29dGhCYAAOoIBwcHOTo66sKFCwoJCdG6devk6+srT0/PKj1O+/bttXPnTrt1O3fuVJs2beTk5CTpajA7deqUOf7111/r/PnzVdpHdePtOQAAblOFhYWy2Wyy2Ww6fPiwxo4dq3PnzumJJ57Q4MGD1bhxYz355JP69NNPdezYMe3YsUMvvPCCfvjhh5s67osvvqjU1FS9+uqr+uqrr7Rq1SotXrxYEyZMMGseffRRLV68WF988YX27dun0aNH2529uh1V6EzTlStXNGPGDK1evVo2m00BAQEaOnSopk2bZp66MwxD06dP17vvvqvc3Fx169ZNb7/9tlq3bm3u56efftLYsWP14YcfytHRUTExMVq4cKF5MZskffnll4qLi9PevXvVpEkTjR07VpMmTbLrZ8OGDfrzn/+s48ePq3Xr1po9e7Yef/xxc9xKLwAAlOV2+KRySkqKeb1Rw4YN1a5dO23YsEGPPPKIJOmTTz7R5MmT1b9/f/3yyy+688471bNnz5s+8xQSEqL169crMTFRr776qpo2bapXXnnF7iLwuXPnatiwYXrooYcUEBCghQsXKiMj46aOW+OMCnj99deNRo0aGZs3bzaOHTtmbNiwwfDw8DAWLlxo1syaNcvw8vIyNm3aZPz73/82/uu//ssIDAw0Lly4YNb07t3b6NSpk/H5558bn376qdGqVSvj6aefNsfz8vIMPz8/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/Hht3bpV/fv318qVK29+krUQoQkAgNvQ0qVLdfnyZYWFhWndunU6fPiwjh49qtWrV+vIkSNycnJSq1atdOnSJb311lv67rvv9Pe//13Lli2z288jjzyiM2fOaM6cOfr222+1ZMkSffTRR+b4sWPHNHXqVKWnp+v777/X1q1b9fXXX6t9+/a6cOGC4uPjtWPHDn3//ffauXOn9u7da3fNU11CaAIA4DZ0zz336IsvvlBkZKSmTp2qTp06KSwsTG+99ZYmTJigV199VZ06ddK8efM0e/ZsdejQQWvWrNHMmTPt9tO+fXstXbpUS5YsUadOnbRnzx5NmDDBHK9fv76OHDmimJgYtWnTRqNGjVJcXJyee+45OTk56ezZs3rmmWfUpk0bPfXUU+rTp49efvnl6n44qgVfo3IDde1rVLjlAABcj69RuX1V93PHmSYAAAALCE0AAAAWEJoAAAAsIDQBAABYQGgCAACwgNAEAABgAaEJAADAAkITAACABYQmAAAACwhNAADgOjNmzFBwcHBNt1GrONd0AwAA1EaV/eqpyqjM11UNHTpUubm52rRpU9U3hDJxpgkAAMACQhMAAHXMvHnz1LFjRzVo0EDNmzfX888/r3PnzpnjSUlJ8vb21qZNm9S6dWu5ubkpKipKJ06c+NV97t27V4899pgaN24sLy8vde/eXZmZmXY1ubm5eu655+Tn5yc3Nzd16NBBmzdvNsc/++wzPfTQQ3J3d1fz5s31wgsvqKCgoOofgFuE0AQAQB3j6OioRYsW6eDBg1q1apW2b9+uSZMm2dWcP39er7/+uv72t79p586dys3N1aBBg351n7/88otiY2P12Wef6fPPP1fr1q31+OOP65dffpEkFRcXq0+fPtq5c6dWr16tQ4cOadasWXJycpIkffvtt+rdu7diYmL05Zdfat26dfrss88UHx9/6x6IKsY1TQAA1DHjxo0z/92yZUu99tprGj16tJYuXWquv3TpkhYvXqzw8HBJ0qpVq9S+fXvt2bNHXbp0uW6fjz76qN3Py5cvl7e3t9LS0tS3b199/PHH2rNnjw4fPqw2bdpIku6++26zfubMmRo8eLDZW+vWrbVo0SJ1795db7/9ttzc3Kpq+rcMZ5oAAKhjPv74Y/Xs2VN33nmnGjZsqCFDhujs2bM6f/68WePs7KzOnTubP7dr107e3t46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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "plot_sparsity(C_bernoulli, C_gauss, C_laplace, threshold)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "c002988c",
   "metadata": {},
   "source": [
    "## Maximum Value"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "id": "c76ea057",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Max value for Bernoulli is 0.51\n",
      "Max value for Gauss is 0.28\n",
      "Max value for Laplace is 0.53\n"
     ]
    }
   ],
   "source": [
    "print(\"Max value for Bernoulli is {:.2f}\".format(np.abs(C_bernoulli).max()))\n",
    "print(\"Max value for Gauss is {:.2f}\".format(np.abs(C_gauss).max()))\n",
    "print(\"Max value for Laplace is {:.2f}\".format(np.abs(C_laplace).max()))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "323f8790",
   "metadata": {},
   "source": [
    "## Contrast ratio"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "id": "8af3f93d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Constrast ratio for Bernoulli is 2.50\n",
      "Constrast ratio for Gauss is 1.19\n",
      "Constrast ratio for Laplace is 1.33\n"
     ]
    }
   ],
   "source": [
    "def contrast_ratio(random_vector, eps):\n",
    "    sum_of_nonzero = np.where(np.abs(random_vector) > eps, np.abs(random_vector), 0).sum()\n",
    "    num_of_nonzero = np.where(np.abs(random_vector) > eps, 1, 0).sum()\n",
    "\n",
    "    return sum_of_nonzero / (num_of_nonzero * eps)\n",
    "\n",
    "print(\"Constrast ratio for Bernoulli is {:.2f}\".format(contrast_ratio(C_bernoulli, threshold)))\n",
    "print(\"Constrast ratio for Gauss is {:.2f}\".format(contrast_ratio(C_gauss, threshold)))\n",
    "print(\"Constrast ratio for Laplace is {:.2f}\".format(contrast_ratio(C_laplace, threshold)))"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "f82435f6",
   "metadata": {},
   "source": [
    "## SNR"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "id": "131b6217",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "SNR for Bernoulli is 3.40\n",
      "SNR for Gauss is 0.38\n",
      "SNR for Laplace is 0.73\n"
     ]
    }
   ],
   "source": [
    "def SNR(random_vector, eps, DB=False):\n",
    "    signal_power = np.where(np.abs(random_vector) > eps, np.abs(random_vector) ** 2, 0).sum()\n",
    "    noise_power = np.where(np.abs(random_vector) < eps, np.abs(random_vector) ** 2, 0).sum()\n",
    "\n",
    "    if DB:\n",
    "        return 10 * np.log10(signal_power / noise_power)\n",
    "    \n",
    "    return signal_power / noise_power\n",
    "\n",
    "print(\"SNR for Bernoulli is {:.2f}\".format(SNR(C_bernoulli, threshold)))\n",
    "print(\"SNR for Gauss is {:.2f}\".format(SNR(C_gauss, threshold)))\n",
    "print(\"SNR for Laplace is {:.2f}\".format(SNR(C_laplace, threshold)))"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "sparserc",
   "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.9.7"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
