{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Alon, Matias, Szegedy (AMS)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from utils import hash_func\n",
    "def AMS(stream, r):\n",
    "    '''\n",
    "    Description:\n",
    "        Uses Alon, Matias, Szegedy (AMS) algorithm to approximate 2 norm\n",
    "    Inputs:\n",
    "        stream: stream vector of random values\n",
    "        r: number of algorithm iterations\n",
    "    Outputs:\n",
    "        est_norm: estimated 2 norm\n",
    "        random_signs: random_signs for the last iteration \n",
    "    '''\n",
    "    \n",
    "    norms = np.zeros(r)\n",
    "    for s_i in stream:\n",
    "        for i in range(r):\n",
    "            v_i = 2*hash_func(i, s_i, 2) - 1 # randomize v_i for each iteration of the algorithm\n",
    "            norms[i] += v_i\n",
    "    return np.sqrt(np.median(norms**2))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Uniform Stream Test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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LT5FixaVb7/99raP++STT8rOnTt9Vk8AAJC8BPSYJgAAgEBBaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAALBAaAIAAAj20DRixAh55JFHJGvWrJI3b15p1qyZHDx40KvM1atXpUePHpI7d27JkiWLtGzZUs6cOeNV5tixY9K4cWPJlCmTeZx+/frJjRs3vMr88MMPUqlSJUmfPr2ULFlSZs6cmSSvEQAABIeADk1r1641gWjTpk2ycuVKuX79utSvX18uX77sLtOnTx/5+uuvZeHChab8yZMnpUWLFu7bb968aQLTtWvXZOPGjTJr1iwTiAYPHuwuc+TIEVOmdu3asmvXLundu7e8+OKL8t133yX5awYAAIEpRALY8uXLva5r2NGWou3bt0utWrXk4sWLMm3aNJk7d67UqVPHlJkxY4aUKVPGBK3HHntMVqxYIfv375fvv/9e8uXLJxUrVpRhw4bJ66+/Lm+//bakS5dOwsPDpXjx4jJmzBjzGHr/DRs2yLhx4yQsLMwvrx0AAASWgG5piklDksqVK5e51PCkrU/16tVzlyldurQUKVJEIiIizHW9LF++vAlMLhqEoqKiZN++fe4yno/hKuN6jLhER0ebx/DcAABA8hU0oenWrVum26xGjRoSGhpq9p0+fdq0FOXIkcOrrAYkvc1VxjMwuW533Xa7MhqE/vnnn3jHW2XPnt29FS5c2IevFgAABJqgCU06tumnn36SefPmSSAYMGCAaflybcePH/d3lQAAQEod0+TSs2dPWbp0qaxbt04KFSrk3p8/f34zwPvChQterU06e05vc5XZsmWL1+O5Ztd5lok5406vZ8uWTTJmzBhnnXSWnW4AACBlCOiWJsdxTGBavHixrF692gzW9lS5cmVJmzatrFq1yr1PlyTQJQaqVatmruvl3r17JTIy0l1GZ+JpICpbtqy7jOdjuMq4HgMAACAk0LvkdGbcl19+adZqco1B0jFE2gKkl507d5a+ffuaweEahF555RUTdnTmnNIlCjQcPf/88zJ69GjzGAMHDjSP7Wop6tatm0yePFn69+8vnTp1MgFtwYIF8s033/j19QMAgMAR0C1NH3/8sRkv9OSTT0qBAgXc2/z5891ldFmAp59+2ixqqcsQaFfbokWL3LenSZPGdO3ppYapdu3aSfv27WXo0KHuMtqCpQFJW5ceeughs/TAp59+ynIDAAAgOFqatHvuTjJkyCAffvih2eJTtGhR+fbbb2/7OBrMdu7cmaB6AgCA5C+gW5oAAAACBaEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEJAADAAqEphg8//FCKFSsmGTJkkKpVq8qWLVv8XSUAABAACE0e5s+fL3379pUhQ4bIjh075KGHHpKwsDCJjIz0d9UAAICfEZo8jB07Vrp06SIdO3aUsmXLSnh4uGTKlEmmT5/u76oBAAA/IzT9n2vXrsn27dulXr167n2pU6c21yMiIvxaNwAA4H8h/q5AoPjzzz/l5s2bki9fPq/9ev3nn3+OVT46OtpsLhcvXjSXUVFRiVK/W9FX3D/rc3hej2tfMJaJS0LL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110iIiLMPr0tPvo448aNu+3r0LO3v/fee177/vOf/zgFChTwut/AgQNjPfe0adPc+z7//HNzDAEkHlqaAPh9TNPJkydN95e2+vzwww+mdUhbW+5Eu820G85Fu+G0JSl37tzuVizdjhw54u660ttfe+0108WVI0cOc/uBAwditTRp95uLduWp8uXLx9qnrVr3Qus8dOhQr/p26dLFtEZduXLlrupz9epViYqKuqf6AIgfA8EB+J2O5XnqqafMNmjQIDMmaciQIfLCCy/c9n46lseTBqICBQqY4BWTBiSlgWnlypXywQcfSMmSJSVjxozyzDPPmIHkntKmTes1pii+fbdu3UrQa/ass45hatGiRazbXOO0krI+AOJHaAIQcMqWLes1NV/Dwc2bN+94P22hOn36tISEhHgNtPakSxloGNMB2a7Q8vvvv0tSiOt1aJ0PHjxoAhyAwEb3HAC/0WUFdHbbZ599Jnv27DHdaAsXLpTRo0ebWWouGoB04LYGovPnz8f7ePXq1ZNq1apJs2bNZMWKFSYMbdy4Ud566y33TLdSpUq5B45r15gOqE6q1pm4XsfgwYNl9uzZprVp3759pqtw3rx5MnDgwCSpEwB7hCYAfqPjd6pWrWpmqtWqVUtCQ0NN95yO6Zk8ebK73JgxY0yXWuHCheXhhx+O9/G0i+rbb781j9WxY0d54IEHpHXr1nL06FH3OKCxY8dKzpw5zRR9nTWnC2lqa09SiOt16PMvXbrUhDxd/uCxxx4zx0PHawEILKl0NLi/KwEAABDoaGkCAACwQGgCAACwQGgCAACwQGgCAACwQGgCAACwQGgCAACwQGgCAACwQGgCAACwQGgCAACwQGgCAACwQGgCAACwQGgCAACQO/t/TxnNWo+on7UAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "AMS Value: 79990.0\n",
      "Ground Truth: 100004.46851016209\n"
     ]
    }
   ],
   "source": [
    "from utils import generate_stream_uniform, plot_stream\n",
    "\n",
    "unif_strm, unif_freq = generate_stream_uniform(n=100, m=int(1e6))\n",
    "plot_stream(unif_strm)\n",
    "\n",
    "print(\"AMS Estimate:\", AMS(unif_strm, 5))\n",
    "print(\"Ground Truth:\", np.linalg.norm(unif_freq, 2))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Skewed Stream Test"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "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": [
      "AMS Value: 191556.0\n",
      "Ground Truth: 215688.21204229034\n"
     ]
    }
   ],
   "source": [
    "from utils import generate_stream_skewed, plot_stream\n",
    "\n",
    "skwd_strm, skwd_freq = generate_stream_skewed(n=100, m=int(1e6), q=1, k=2)\n",
    "plot_stream(skwd_strm)\n",
    "\n",
    "print(\"AMS Estimate:\", AMS(skwd_strm, 5))\n",
    "print(\"Ground Truth:\", np.linalg.norm(skwd_freq, 2))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Alon, Matias, Szegedy Augmented (AMSA)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from utils import hash_func\n",
    "\n",
    "def AMSA(stream, heavy, r):\n",
    "    '''\n",
    "    Description:\n",
    "        Uses AMS algorithm + heavy hitter knowledge to approximate 2 norm\n",
    "    Inputs:\n",
    "        stream: stream vector of random values\n",
    "        r: number of algorithm iterations\n",
    "        heavy: list of heavy hitters\n",
    "    Outputs:\n",
    "        Estimated 2 norm of frequency vector\n",
    "    '''\n",
    "    l = len(heavy)\n",
    "    buckets = np.zeros(l+r)\n",
    "    # process the heavy hitters and non-heavy hitters\n",
    "    for s_i in stream:\n",
    "        try:\n",
    "            index = heavy.index(s_i)\n",
    "            buckets[index] += 1\n",
    "        except ValueError:\n",
    "            # s_i is not a heavy hitter\n",
    "            for i in range(r):\n",
    "                v_i = 2*hash_func(i, s_i, 2) - 1\n",
    "                buckets[l+i] += v_i\n",
    "    # approximate all non-heavy hitters\n",
    "    buckets[l] = np.sqrt(np.median(buckets[l:]**2))\n",
    "    return np.linalg.norm(buckets[:l+1], 2)"
   ]
  },
  {
   "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": [
      "AMSA Estimate: 79990.0\n",
      "Ground Truth: 100004.46851016209\n"
     ]
    }
   ],
   "source": [
    "from utils import generate_stream_uniform, plot_stream\n",
    "\n",
    "unif_strm, unif_freq = generate_stream_uniform(n=100, m=int(1e6))\n",
    "plot_stream(unif_strm)\n",
    "\n",
    "print(\"AMSA Estimate:\", AMSA(unif_strm,[], 5))\n",
    "print(\"Ground Truth:\", np.linalg.norm(unif_freq, 2))"
   ]
  },
  {
   "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": [
      "AMSA Estimate: 207835.37094537108\n",
      "Ground Truth: 215688.21204229034\n"
     ]
    }
   ],
   "source": [
    "from utils import generate_stream_skewed, plot_stream\n",
    "\n",
    "skwd_strm, skwd_freq = generate_stream_skewed(n=100, m=int(1e6), q=1, k=2)\n",
    "plot_stream(skwd_strm)\n",
    "\n",
    "print(\"AMSA Estimate:\", AMSA(skwd_strm,[1], 5))\n",
    "print(\"Ground Truth:\", np.linalg.norm(skwd_freq, 2))"
   ]
  }
 ],
 "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
}
