{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## CountSketch (CS)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from utils import hash_func\n",
    "def count_sketch(stream, r, n, b, eps):\n",
    "    '''\n",
    "    Description:\n",
    "        implementation of count-sketch algorithm to determine heavy-hitters\n",
    "    Parameters:\n",
    "        stream: stream vector of random values\n",
    "        r: number of algorithm iterations\n",
    "        n: size of frequency vector\n",
    "        b: number of buckets (hyperparam)\n",
    "        eps: desired accuracy (hyperparam)\n",
    "    Outputs:\n",
    "        returns list of heavy hitter stream values\n",
    "    '''\n",
    "    heavy_hitters = []\n",
    "    buckets_list = np.zeros((r, b))\n",
    "    approx_norms = np.zeros(r)\n",
    "    # process the stream\n",
    "    for s_i in stream:\n",
    "        for i in range(r):\n",
    "            curr_buckets = buckets_list[i]\n",
    "            v_i = 2*hash_func(i+100, s_i, 2) - 1\n",
    "            bucket = hash_func(i, s_i, b)\n",
    "            curr_buckets[bucket] += v_i\n",
    "            approx_norms[i] += v_i\n",
    "\n",
    "    # determine the threshold\n",
    "    thresh = eps * np.sqrt(np.median(approx_norms**2))\n",
    "    # print(buckets_list)\n",
    "    for j in range(1, n+1):\n",
    "        f_j = []\n",
    "        for i in range(r):\n",
    "            sign = 2*hash_func(i+100, j, 2) - 1\n",
    "            bucket = hash_func(i, j, b)\n",
    "            f_j.append(buckets_list[i][bucket] * sign)\n",
    "        est_norm = np.median(f_j)\n",
    "        if (est_norm > thresh):\n",
    "                heavy_hitters.append(j)\n",
    "    return heavy_hitters"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Uniform Stream Test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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uCDbtm3zlNGAp61Y+hrcMmvWrDEz61w6065kyZI+M/UAAEDylaDQpAOm586dKz/99JOZebZgwQKzErg7+8xfevToIRs3bjRBTC/RoksEfPTRR9KlSxdzf4oUKaR79+4yZMgQM2hclwrQy7fojLgmTZp4WqYaNGhguvX0osLr1q2Trl27mhYxLad0zSmtt67ftHfvXvPa9Np6ekFiAACABC854Lr77rvNlljuu+8+WbhwoRk/NGjQINOypEsM6LpLrt69e8vly5fNeCdtUdILB+sSA+4aTUqDnQYlbRnTWXPNmzc3azu5dCD3ihUrTBirVKmSmRmoC2ay3AAAALit0KRjmPQyKdotpuOFtKvLm3Z/+YuuNq5bXLS1SQOVbnHRmXLuQpZxKV++vKxdu/a26goAAJKuBIUmHfCtoalhw4YSHh5uggsAAEBSlqDQNGfOHJk3b55ZVRsAACA5SPBA8OLFi/u/NgAAAEkpNL366qtmdple+w0AACA5SFD33A8//GAWtly6dKm5uG3q1Kl97tclCAAAACS5hya9AG7Tpk39XxsAAICkFJqmTp3q/5oAAAAktTFN7vXbvvnmG/nwww/lr7/+MsdOnDghly5d8mf9AAAAQrel6ciRI+bSJHrh3MjISHnkkUckc+bMMmLECLM/adIk/9cUAAAg1FqadHHLypUry/nz583Fel06zsn74rkAAADJuqVJLzeyfv36GBfnLVKkiBw/ftxfdQMAAAjtlia91pxefy6633//3XTTAQAAJDUJCk316tWTsWPHevb12nM6AHzAgAFcWgUAACRJCeqeGzVqlNSvX1/KlCkjV69elZYtW8qhQ4ckV65c8tlnn/m/lgAAAKEYmgoUKCC7du0yF+7dvXu3aWXq0KGDtGrVymdgOAAAQLIOTeYbw8KkdevW/q0NAABAUgpNM2bMuOX9bdq0SWh9AAAAkk5o0nWavF27dk2uXLliliDIkCEDoQkAACQ5CZo9p4taem86punAgQNSo0YNBoIDAIAkKcHXnouuRIkSMnz48BitUAAAAEmB30KTOzhcL9oLAACQ1CRoTNPixYt99h3HkZMnT8qECROkevXq/qobAABAaIemJk2a+OzriuB33XWX1K5d2yx8CQAAkNSEJfTacwAAAMmJX8c0AQAAJFUJamnq2bOnddnRo0cn5CkAAABCPzTt2LHD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BXj88cfNopa6DIF2tS1YsMBzf6pUqUzXnt5qmGrdurW0adNGBg0a5CmjLVgakLR16Z577jFLD3zyyScsNwAAAEKjpUm75/5JunTp5P333zdbXAoXLixff/31LR9Hg9mOHTsSVE8AAJD0BXVLEwAAQLAgNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNAEAAFggNEXz/vvvS5EiRSRdunRSpUoV2bx5c6CrBAAAggChycvcuXOlZ8+eMmDAANm+fbvcc889Ur9+fTlz5kygqwYAAAKM0ORl9OjR0rFjR2nXrp2UKVNGJk2aJBkyZJApU6YEumoAACDACE3/JyoqSrZt2yZ169b1HEuZMqXZ37BhQ0DrBgAAAi8s0BUIFn/88YfcuHFD8uTJ43Nc93/66acY5SMjI83munjxormNiIhIlPrdjLzi+Vqfw3s/tmOhWCY2CS0T7K81OZeJDe+FpFcmNvyck2cZf/+c/c19TMdx/rmwA+P48eN6tpz169f7HO/Vq5dz//33xyg/YMAAU56NjY2NjY1NQn47duzYP2YFWpr+T65cuSRVqlRy+vRpn+O6nzdv3hjl+/TpYwaNu27evCnnzp2TnDlzSooUKfySfAsWLCjHjh2TLFmy3PbjIXac5zuHc31ncJ7vDM5z0jnX2sL0119/Sf78+f+xLKHp/6RJk0YqVaokq1atkiZNmniCkO537do1Rvm0adOazVu2bNn8Xi99g/AfMvFxnu8czvWdwXm+MzjPSeNcZ82a1aococmLthy1bdtWKleuLPfff7+M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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Heavy Hitters from CS:\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "[1,\n",
       " 2,\n",
       " 3,\n",
       " 4,\n",
       " 5,\n",
       " 6,\n",
       " 7,\n",
       " 8,\n",
       " 9,\n",
       " 10,\n",
       " 11,\n",
       " 12,\n",
       " 13,\n",
       " 14,\n",
       " 15,\n",
       " 16,\n",
       " 17,\n",
       " 18,\n",
       " 19,\n",
       " 20,\n",
       " 21,\n",
       " 22,\n",
       " 23,\n",
       " 24,\n",
       " 25,\n",
       " 26,\n",
       " 27,\n",
       " 28,\n",
       " 29,\n",
       " 30,\n",
       " 31,\n",
       " 32,\n",
       " 33,\n",
       " 34,\n",
       " 35,\n",
       " 36,\n",
       " 37,\n",
       " 38,\n",
       " 39,\n",
       " 40,\n",
       " 41,\n",
       " 42,\n",
       " 43,\n",
       " 44,\n",
       " 45,\n",
       " 46,\n",
       " 47,\n",
       " 49,\n",
       " 50,\n",
       " 51,\n",
       " 52,\n",
       " 53,\n",
       " 54,\n",
       " 55,\n",
       " 56,\n",
       " 57,\n",
       " 58,\n",
       " 59,\n",
       " 60,\n",
       " 61,\n",
       " 62,\n",
       " 63,\n",
       " 64,\n",
       " 65,\n",
       " 66,\n",
       " 67,\n",
       " 68,\n",
       " 69,\n",
       " 70,\n",
       " 71,\n",
       " 72,\n",
       " 73,\n",
       " 74,\n",
       " 75,\n",
       " 76,\n",
       " 77,\n",
       " 78,\n",
       " 79,\n",
       " 80,\n",
       " 81,\n",
       " 82,\n",
       " 83,\n",
       " 84,\n",
       " 85,\n",
       " 86,\n",
       " 87,\n",
       " 88,\n",
       " 89,\n",
       " 90,\n",
       " 91,\n",
       " 92,\n",
       " 93,\n",
       " 94,\n",
       " 95,\n",
       " 96,\n",
       " 97,\n",
       " 98,\n",
       " 99,\n",
       " 100]"
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from utils import generate_stream_uniform, plot_stream\n",
    "\n",
    "n=100\n",
    "unif_strm, unif_freq = generate_stream_uniform(n=n, m=int(1e6))\n",
    "plot_stream(unif_strm)\n",
    "hh=count_sketch(stream=unif_strm, r=5, n=n, b=300, eps=0.1)\n",
    "print(\"Heavy Hitters from CS:\")\n",
    "hh"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Skewed Stream Test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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3Wr+KNPB1ZznnIfuPz9TrFgx89dQ0quhdL1EiRKZVq/sQu/rqIMZV65cKaVLl3a367nVrtdTp075lOe8p4127+kFFXXq1DF/Meqig9F1sKl+rX9lcp7t0Cuiqlat6rPtlltukUOHDpmvPeeT3yV/T79+/UxrVbt27czVlZ06dTIXW+gVxYrznDFSc171MekFXJcuXTJXBGbUuSdU+Rltuq9bt67pw/f+i1TX69evn6l182c6DlID1dy5c2XFihXm8mhves71Kirv865TLugHFOc99Ro3biw7duwwf817Fm1N0a4Sz9ecZzu0+zrptCA67ic8PNx8rT/j+sHifa61G0vHmnCuU+/s2bNmjI43/cNXfy8rznPGSM151Uf9A03/mPPQ3+/6vdGxVxkiQ4a/I0PNmDHDXOEwdepUc3VDjx49nEKFCjnHjh3L7Kr5rWeffdYpWLCgs2rVKufo0aPucvbsWbfMM88845QtW9ZZsWKF8+OPPzr169c3C/4e76v/FOfZjg0bNji5c+d23nzzTWfv3r3O559/7uTLl8/57LPP3DIjR440vzvmz5/vbN++3WnZsqVTrlw559y5c5lad3/SuXNn56abbnIWLFjg7N+/35kzZ45TrFgxp3///m4ZznP6rxLesmWLWTSujB071nx98ODBVJ/X5s2bO7feequzfv165/vvvzdXHbdv397JKIQqP/Xuu++aD56goCAzxcK6desyu0p+Tf/DprRMmTLFLaP/UZ977jmncOHC5sPp4YcfNsELdkMV59meb775xqlevbr5I6xKlSrOhx9+6LNfL0sfPHiwExYWZso0btzYiYmJybT6+qP4+Hjz86u/j0NCQpybb77ZeeWVV5yEhAS3DOc5fVauXJni72UNsqk9rydOnDAhqkCBAk5oaKjTtWtXE9YySoD+kzFtYAAAADkHY6oAAAAsIFQBAABYQKgCAACwgFAFAABgAaEKAADAAkIVAACABYQqAAAACwhVAAAAFhCqAGR5f/zxhzz77LNStmxZCQ4ONvf8atasmfzwww9umYCAAJk3b574m4iICBk3bpzfvw8AIrkzuwIAcC1t2rSRCxcuyLRp0+Tmm282d6LXG6meOHEiTcfRY+hNyQEgI9BSBSBL07vMf/fdd/L2229Lo0aNJDw8XOrVqyeDBg2Shx56yG3tUQ8//LBp6fGsDxs2TGrXri0ff/yxuat9SEiIe8ynnnpKbrzxRgkNDZV//OMfsm3bNvc19+3bJy1btpSwsDApUKCA3H777bJs2TKfeulrvPHGG/LEE0+YMlqvr7/+2rSq6XN1W82aNeXHH39M9Xu90vtQ8+fPlzp16pj3oMFy+PDhcunSJXe/lv/ggw/kwQcflHz58sktt9wi0dHR8ssvv8i9994r+fPnlzvvvNO8NwAZg1AFIEvTcKKLdoklJCSkWGbjxo3mccqUKXL06FF3XWmo+Oqrr2TOnDmydetWs+3RRx+V2NhYWbx4sWzatMmElcaNG8uff/5p9p8+fVoeeOAB0xq2ZcsWad68ubRo0UIOHTrk87rvvPOO3HXXXaZMVFSUdOrUyYSsjh07yubNm6V8+fJmPbW3WL3S+9BQqcd58cUX5aeffjLhaerUqfLmm2/6PP/111835fR9VqlSRTp06CBPP/20CaAa7rQezz//fBrOPoA0ybBbNQOAJf/5z3+cwoULOyEhIc6dd97pDBo0yNm2bZtPGf11NnfuXJ9tQ4cOdfLkyePExsa627777jtzt/rz58/7lC1fvrzzwQcfXLEO1apVc9599113PTw83OnYsaO7fvToUVOHwYMHu9uio6PNNt13JXqcd95556rvo3Hjxs5bb73ls+3f//63U7JkSZ/nvfrqq8le+5NPPnG3ffnll+YcAsgYtFQB8IsxVUeOHDHda9pqtGrVKtO6pK0116LdctrN56HdfNoSVbRoUbcVTJf9+/e7XWO6/+WXXzZdaIUKFTL7d+/enaylSrv3PLSrUNWoUSPZNm0V+zu0zq+99ppPfbt3725as86ePZum+pw/f17i4+P/Vn0ApIyB6gD8go4luu+++8wyePBgMyZq6NCh0qVLl6s+T8cSedPAVLJkSRPMktIApTRQLV26VP71r39JhQoVJG/evPLII4+Yge7e8uTJ4zOm6UrbEhMT0/WeveusY6hat26dbJ9nnNj1rA+AlBGqAPilqlWr+kw9oOHh8uXL13yetnAdO3ZMcufO7TMQ3JtO1aBhTQeMe0LNgQMH5HpI6X1onWNiYkzAA5B10f0HIEvTaRP06rzPPvtMtm/fbrrpZs+eLaNGjTJX2XloQNKB5RqYTp48ecXjNWnSROrXry+tWrWS//73vyYsrV27Vl555RX3Sr2KFSu6A9u1600HfF+v1p2U3seQIUNk+vTpprVq165dpityxowZ8uqrr16XOgFIHUIVgCxNxw9FRkaaK+0aNmwo1atXN91/OqZo4sSJbrkxY8aYLrsyZcrIrbfeesXjaRfYokWLzLG6du0qlSpVknbt2snBgwfdcUhjx46VwoULmykI9Ko/nWhUW4uuh5Teh77+ggULTAjU6R3uuOMOcz50vBiArCNAR6tndiUAAAD8HS1VAAAAFhCqAAAALCBUAQAAWECoAgAAsIBQBQAAYAGhCgAAwAJCFQAAgAWEKgAAAAsIVQAAABYQqgAAACwgVAEAAFhAqAIAAJC/7/8BJyh/H7jThM8AAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Heavy Hitters from CS:\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "[1]"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "from utils import generate_stream_skewed, plot_stream\n",
    "\n",
    "n=100\n",
    "skwd_strm, skwd_freq = generate_stream_skewed(n=n, m=int(1e6), q=1, k=2)\n",
    "plot_stream(skwd_strm)\n",
    "hh=count_sketch(stream=skwd_strm, r=5, n=n, b=300, eps=0.1)\n",
    "print(\"Heavy Hitters from CS:\")\n",
    "hh"
   ]
  }
 ],
 "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.12.0"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
