{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Parallel SS/SSA"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "def estimate_norm(timestep, order, r1, r2, q):\n",
    "    '''\n",
    "    Description:\n",
    "        Helper function to estimate norm at timestep based on its histogram\n",
    "    Input:\n",
    "        hist: histogram to use for norms\n",
    "        timestep: timestep for which to estimte order norm\n",
    "        order: order of norm\n",
    "        r1: number of values to take median of\n",
    "        r2: number of values to take mean of\n",
    "        q: threshold of ssa algorithm\n",
    "    Returns:\n",
    "        Order norm at timestep\n",
    "    '''\n",
    "    t = hist_ssa[timestep] # keeps track of samples\n",
    "    r1_norms = []\n",
    "    for i in range(r1):\n",
    "        r2_norms = []\n",
    "        t_i = t[i] # sample for current iteration\n",
    "        for j in range(r2):\n",
    "            curr_t = t_i[j]\n",
    "            moment = np.sum(np.power(list(curr_t.values()), order))\n",
    "            r2_norms.append(\n",
    "                    np.power((1/q * moment), 1/order)\n",
    "                )\n",
    "        r1_norms.append(np.mean(r2_norms))\n",
    "    rest_est = np.median(r1_norms)\n",
    "    temp_arr = np.concatenate((hist_hh_sums[timestep], [rest_est]))\n",
    "    return np.linalg.norm(temp_arr, order)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "def prune_threads_ssa(order, r1, r2, q):\n",
    "    '''\n",
    "    Description:\n",
    "        Removes adjacent threads that are within a threshold of 2\n",
    "    Inputs:\n",
    "        order: order of norm\n",
    "        r1: number of values to take median of\n",
    "        r2: number of values to take mean of\n",
    "        q: threshold for ssa algorithm\n",
    "    '''\n",
    "    i = 0 # index variable\n",
    "    while (i < len(threads_vec_ssa)):\n",
    "        # boundary check\n",
    "        if (i >= len(threads_vec_ssa) - 2):\n",
    "            return\n",
    "        t_i = threads_vec_ssa[i] # thread identity\n",
    "        ssa_ti = estimate_norm(t_i, order, r1, r2, q) # ssa estimate for thread\n",
    "\n",
    "        t_i2 = threads_vec_ssa[i+2] # adjcaent thread\n",
    "        ssa_ti2 = estimate_norm(t_i2, order, r1, r2, q) # ssa estimate for adjancent thread\n",
    "        \n",
    "        # check distance, prune hist, threads\n",
    "        denom = min(ssa_ti, ssa_ti2)\n",
    "        if (denom != 0 and (max(ssa_ti, ssa_ti2)/denom <= 2)):\n",
    "            # remove the thread in between\n",
    "            t_i1 = threads_vec_ssa[i+1]\n",
    "            threads_vec_ssa.pop(i+1)\n",
    "            hist_ssa.pop(t_i1)\n",
    "            hist_ss.pop(t_i1)\n",
    "            hist_hh_sums.pop(t_i1)\n",
    "            i -= 1\n",
    "        i += 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "def spawn_threads_ssa(timestep, order, l, r1, r2, q):\n",
    "    '''\n",
    "    Description:\n",
    "        Helper function to spawn threads for ssa\n",
    "    Inputs:\n",
    "        timestep: timestep identifier to spawn threads for ssa\n",
    "        order: order of norm\n",
    "        l: number of heavy hitters\n",
    "        r1: number of values to take median of\n",
    "        r2: number of values to take mean of\n",
    "        q: threshold for ssa algorithm\n",
    "    '''\n",
    "    if (len(threads_vec_ssa) == num_threads_max):\n",
    "        prune_threads_ssa(order, r1, r2, q)\n",
    "    if (len(threads_vec_ssa) < num_threads_max):\n",
    "        threads_vec_ssa.append(timestep)\n",
    "        hist_ssa[timestep] = [[{} for _ in range(r2)] for _ in range(r1)]\n",
    "        hist_ss[timestep] = [[{} for _ in range(r2)] for _ in range(r1)]\n",
    "        hist_hh_sums[timestep] = np.zeros(l)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def selective_subsampling(s_i, r1, r2, q):\n",
    "    '''\n",
    "    Description:\n",
    "        Helper method to run ss in parallel with ssa\n",
    "    Inputs:\n",
    "        s_i: the current stream value\n",
    "        r1: number of values to take median of\n",
    "        r2: number of values to take mean of\n",
    "        \n",
    "    '''\n",
    "    for thread in threads_vec_ssa:\n",
    "        t = hist_ss[thread]\n",
    "        for i in range(r1):\n",
    "            t_i = t[i]\n",
    "            for j in range(r2):\n",
    "                curr_t = t_i[j]\n",
    "                prob = seeded_input_random((i+1)*(j+1) + s_i, s_i)\n",
    "                if (prob < q):\n",
    "                    curr_t.setdefault(s_i, 0)\n",
    "                    curr_t[s_i] += 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "from utils import seeded_input_random\n",
    "import csv\n",
    "import ipaddress\n",
    "\n",
    "def selective_subsampling_augmented(heavy, filepath, order, limit, r1, r2, q, q_ss):\n",
    "    '''\n",
    "    Description:\n",
    "        Uses augmented selective subsampling algorithm to estimate order norm of frequencies\n",
    "    Inputs:\n",
    "        heavy: list of heavy hitters IP addresses\n",
    "        filepath: file to run algorithm on\n",
    "        order: order of norm\n",
    "        limit: number of addresses to process\n",
    "        r1: number of values to take median of\n",
    "        r2: number of values to take mean of\n",
    "        q: threshold for ssa algorithm\n",
    "        q_ss: threshold for ss algorithm\n",
    "    Outputs:\n",
    "        returns estimated order norm of frequency vector\n",
    "    '''\n",
    "    \n",
    "    with open(filepath) as csvfile:\n",
    "        stream = csv.reader(csvfile, delimiter=\",\")\n",
    "        stream.__next__() # get rid of the header\n",
    "        time = 1\n",
    "        for row in stream:\n",
    "            if (stream.line_num == limit):\n",
    "                break\n",
    "            try:\n",
    "                s_i = int(ipaddress.ip_address(row[0]))\n",
    "            except:\n",
    "                # ip address is not valid IPv4 or IPv6 address\n",
    "                continue\n",
    "            vals.append(s_i)\n",
    "            # run augmented algorithm\n",
    "            spawn_threads_ssa(time, order, len(heavy), r1, r2, q)\n",
    "            try:\n",
    "                index = heavy.index(s_i)\n",
    "                # s_i is a heavy hitter\n",
    "                for thread in threads_vec_ssa:\n",
    "                    t = hist_hh_sums[thread]\n",
    "                    t[index] += 1\n",
    "            except ValueError:\n",
    "                # s_i is not a heavy hitter\n",
    "                for thread in threads_vec_ssa:\n",
    "                    t = hist_ssa[thread]\n",
    "                    for i in range(r1):\n",
    "                        t_i = t[i]\n",
    "                        for j in range(r2):\n",
    "                            curr_t = t_i[j]\n",
    "                            prob = seeded_input_random((i+1)*(j+1) + s_i, s_i)\n",
    "                            if (prob < q):\n",
    "                                curr_t.setdefault(s_i, 0)\n",
    "                                curr_t[s_i] += 1\n",
    "            finally:\n",
    "                # run ss in parallel\n",
    "                selective_subsampling(s_i, r1, r2, q_ss)\n",
    "                time += 1"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Test Parallel SS/SSA on Data - Vary Window Size"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "# from CS\n",
    "hh = [\n",
    " 2401796176,\n",
    " 3573875943,\n",
    " 879396542,\n",
    " 1572803056,\n",
    " 3451786411,\n",
    " 1298962670,\n",
    " 607033987,\n",
    " 2497912303,\n",
    " 3339135037,\n",
    " 887074222,\n",
    " 1115991985,\n",
    " 2613392396,\n",
    " 1604957565,\n",
    " 1072021340,\n",
    " 2987213905,\n",
    " 3124349474,\n",
    " 3124349526,\n",
    " 517524860,\n",
    " 595152676,\n",
    " 3568914218,\n",
    " 2839797836,\n",
    " 1039166085,\n",
    " 1115974613,\n",
    " 3174560376,\n",
    " 3124327772,\n",
    " 3016127925\n",
    " ]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "# #setup shared variables\n",
    "# hist_ssa = {} # maps timestep to ssa value\n",
    "# hist_ss = {} # maps timestep to ss value\n",
    "# hist_hh_sums = {} # maps timestep to hh sum value\n",
    "# threads_vec_ssa = [] # stores thread identities\n",
    "# num_threads_max = 20 # hyperparam for max counters\n",
    "# vals = []\n",
    "# r1 = 3\n",
    "# r2 = 5\n",
    "# order = 3\n",
    "# q = 1/50\n",
    "# q_ss = 1/50\n",
    "# filepath=\"CAIDA_12mins_sender_IPs/125910_ip_timestamps.csv\"\n",
    "# m = int(2e6)\n",
    "\n",
    "# selective_subsampling_augmented(heavy=hh, filepath=filepath, order=order, limit=m, r1=r1, r2=r2, q=q, q_ss=q_ss)\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Visualize Results"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "def norm_helper(time, order, r1, r2, q):\n",
    "    '''\n",
    "        Description:\n",
    "            Helper function to calculate order norm for ss\n",
    "        Inputs:\n",
    "            time: timestep for calculation\n",
    "            order: order of norm\n",
    "            r1: number of values to take median of\n",
    "            r2: number of values to take mean of\n",
    "            q: threshold for ss algorithm\n",
    "        Outputs:\n",
    "            order norm\n",
    "    '''\n",
    "    t = hist_ss[time] # keeps track of samples\n",
    "    r1_norms = []\n",
    "    for i in range(r1):\n",
    "        r2_norms = []\n",
    "        t_i = t[i] # sample for current iteration\n",
    "        for j in range(r2):\n",
    "            curr_t = t_i[j]\n",
    "            moment = np.sum(np.power(list(curr_t.values()), order))\n",
    "            r2_norms.append(\n",
    "                    np.power((1/q * moment), 1/order)\n",
    "                )\n",
    "        r1_norms.append(np.mean(r2_norms))\n",
    "    \n",
    "    return np.median(r1_norms)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "def calculate_estimates():\n",
    "    '''\n",
    "    Description:\n",
    "        Calculates estimates for ssa/ss and computes ground truth\n",
    "    Inputs:\n",
    "        None\n",
    "    Outputs:\n",
    "        timesteps: unpruned timesteps\n",
    "        est_ssa: ssa estimates corresponding to timesteps\n",
    "        est_ss: ss estimates corresponding to timesteps\n",
    "        gt: ground truth vaues corresponding to timesteps\n",
    "    '''\n",
    "    timesteps = []\n",
    "    est_ssa = []\n",
    "    est_ss = []\n",
    "    gt = []\n",
    "    for time in hist_ssa.keys():\n",
    "        timesteps.append(time)\n",
    "        est_ssa.append(estimate_norm(time, order, r1, r2, q))\n",
    "        est_ss.append(norm_helper(time, order, r1, r2, q_ss))    \n",
    "        _, temp_freq = np.unique(vals[time:], return_counts=True)\n",
    "        gt.append(np.linalg.norm(temp_freq, order))\n",
    "    return timesteps, est_ssa, est_ss, gt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "def plot_estimates(timesteps, est_ssa, est_ss, gt):\n",
    "    '''\n",
    "    Description:\n",
    "        Plots ground truth and estimates for ss/ssa\n",
    "    Inputs:\n",
    "        timesteps: unpruned timesteps\n",
    "        est_ssa: ssa estimates corresponding to timesteps\n",
    "        est_ss: ss estimates corresponding to timesteps\n",
    "        gt: ground truth values corresponding to timesteps\n",
    "    '''\n",
    "    _, ax = plt.subplots()\n",
    "\n",
    "    ax.plot(timesteps, est_ssa, 'rx', label=\"SSA Estimation\", markersize=10)\n",
    "    ax.plot(timesteps, est_ss, 'bP', label=\"SS Estimation\")\n",
    "    ax.plot(timesteps, gt, 'go', label=\"Ground Truth\", fillstyle='none')\n",
    "    ax.set_title(f\"SSA/SS Estimation on CAIDA (n={m:.1e})\")\n",
    "    ax.set_xlabel(\"Starting Timestep\")\n",
    "    ax.set_ylabel(\"Estimated 3 Norm\")\n",
    "    ax.legend()\n",
    "    plt.show()\n",
    "\n",
    "def plot_error_ratio(timesteps, est_ssa, est_ss, gt):\n",
    "    '''\n",
    "    Description:\n",
    "        Computes and plots ss/ssa to ground truth ratio\n",
    "    Inputs:\n",
    "        timesteps: unpruned timesteps\n",
    "        est_ssa: ssa estimates corresponding to timesteps\n",
    "        est_ss: ss estimates corresponding to timesteps\n",
    "        gt: ground truth values corresponding to timesteps\n",
    "    '''\n",
    "    # plot the ratio\n",
    "    ratio = lambda x, y: max(x, y)/min(x, y)\n",
    "    err_ss = np.array(list(map(ratio, est_ss, gt)))\n",
    "    err_ssa = np.array(list(map(ratio, est_ssa, gt)))\n",
    "\n",
    "    _, ax2 = plt.subplots()\n",
    "    ax2.plot(timesteps[:-1], err_ssa[:-1], 'r--', label=\"SSA Estimation\")\n",
    "    ax2.plot(timesteps[:-1], err_ss[:-1], 'b-.',label=\"SS Estimation\")\n",
    "    ax2.set_xlabel(\"Starting Timestep\")\n",
    "    ax2.set_ylabel(\"L3 Norm Estimate to Ground Truth Ratio\")\n",
    "    ax2.set_title(f\"SSA/SS Estimation Error on CAIDA (n={m:.1e})\")\n",
    "    ax2.legend()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [],
   "source": [
    "# timesteps, est_ssa, est_ss, gt = calculate_estimates()\n",
    "# plot_estimates(timesteps, est_ssa, est_ss, gt)\n",
    "# plot_error_ratio(timesteps, est_ssa, est_ss, gt)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Test Parallel SS/SSA on Data - Vary Selection Probability"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "\n",
    "q_range = [\n",
    "    1/10, 1/20, 1/40, 1/80, 1/160, 1/320, 1/640\n",
    "]\n",
    "\n",
    "est_ssa_2 = []\n",
    "est_ss_2 = []\n",
    "gt_2 = []\n",
    "\n",
    "for q in q_range:\n",
    "    hist_ssa = {} # maps timestep to ssa value\n",
    "    hist_ss = {} # maps timestep to ss value\n",
    "    hist_hh_sums = {} # maps timestep to hh sum value\n",
    "    threads_vec_ssa = [] # stores thread identities\n",
    "    num_threads_max = 20 # hyperparam for max counters\n",
    "    vals = []\n",
    "    r1 = 3\n",
    "    r2 = 5\n",
    "    order = 3\n",
    "    filepath=\"CAIDA_12mins_sender_IPs/125910_ip_timestamps.csv\"\n",
    "    m = int(2e5)\n",
    "    W = int(m/2) # window size\n",
    "    sel_idx = m - W\n",
    "    q_ss = q\n",
    "    selective_subsampling_augmented(heavy=hh, filepath=filepath, order=order, limit=m, r1=r1, r2=r2, q=q, q_ss=q_ss)\n",
    "    timesteps, est_ssa, est_ss, _ = calculate_estimates()\n",
    "    diffs = np.absolute(np.array(timesteps)-sel_idx)\n",
    "    est_idx = diffs.argmin()\n",
    "    est_ssa_2.append(est_ssa[est_idx])\n",
    "    est_ss_2.append(est_ss[est_idx])\n",
    "    _, temp_freq = np.unique(vals[:-W], return_counts=True)\n",
    "    gt_2.append(np.linalg.norm(temp_freq, order))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Visualize Results "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAjsAAAHLCAYAAAAurFnfAAAAOnRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjEwLjEsIGh0dHBzOi8vbWF0cGxvdGxpYi5vcmcvc2/+5QAAAAlwSFlzAAAPYQAAD2EBqD+naQAAX2VJREFUeJzt3Qd8E/X7B/CHUmbZo8y27ELZU4YyZA8BQRBlgyh7g6BsZMgSnDgZAipTBBREprK3CBUBWf7Ze6/2/q/P19/FS5q0SZs0yfXz5hXSXC65by6Xu+ee77hkmqZpQkRERGRSAd4uABEREZEnMdghIiIiU2OwQ0RERKbGYIeIiIhMjcEOERERmRqDHSIiIjI1BjtERERkagx2iIiIyNQY7BAREZGpMdghv7R582ZJliyZuvclNWvWVDfyf1OmTJGiRYtKdHS0t4tCZApt2rSR1q1be2XZDHY84PDhw/LSSy9JWFiYpE6dWvLkySN169aVDz74wGq+x48fy6xZs6Rs2bKSIUMGyZQpkxQvXlxef/11+fPPP+2+d2RkpDrI431v3rwZazlWrVolAQEBcvHiRfX4ypUr0q9fP7UDT5MmjQQHB0ulSpXkzTfflLt378Z4bY0aNdQ8adOmlQIFCqiNdO3atXF+/nz58qky2rs1aNBAXPHxxx/L3LlzxZccPXpUxowZI6dPn5ak6uTJk/LGG2+o7QLbIrbfatWqqe35wYMHMeaPioqS3Llzq23gp59+svueWKd4/urVq5ZpnTp1stp+0qVLp5aJ39eyZctiDUTw+0DZ8Dr8blxx+/Zteffdd9VvA7+hxHLs2DEZMGCAVK1a1VJ2V7czfFb8zrCusmTJIu3bt1e/fW/48ssvpVixYuqzFC5cOMY+0B7sK/G5e/fuHe8gFa8/cOCA1XRcGSlz5szquVOnTlk99/DhQ0mVKpW8+uqrklCO9n2TJ0926vWPHj1S2x1+L9hPP/PMM7J+/XrxhkgntqXz589Lu3btJDw8XNKnT6+OYziuzJs3T61zI3wu/G4PHTqUyJ9EJDDRl2hy27dvl1q1akloaKh069ZNcubMKefOnZOdO3eqA0GfPn0s87Zs2VLt+F955RU175MnT1SQs3r1arWzQ1Bia8GCBeo9b9y4IUuXLpXXXnvNYVnWrFkj5cuXV/Nfv35dKlSooHbiXbp0Ue997do1+f333+WTTz6RHj16qA0apk2bJkOGDFHBzvDhw1Wwc+LECfnll1/k22+/dSpgKVOmjAwaNCjGdPyAXQ12smXLpg56RtWrV1cH1ZQpU4o3gp2xY8eqDA4CO6Off/5ZzA7bVatWrdTBoUOHDlKiRAkVuP/2229quzly5Ih89tlnVq/ZuHGjXLhwQa2vhQsXSsOGDZ1eHpbzxRdfqL/xnZ85c0YF4wh48B2sXLlSBVu2lixZog4y2P6xzHfeecfpZX711Vfy9OlT9dtMTDt27JD3339fIiIiVJBw8OBBl17/zz//qN9GxowZZeLEieokBr9nnIDt3r07UX8vn376qXTv3l3t5wYOHCi//vqr9O3bV+7fv68OevYsX75crYOEePbZZ9U9tkecSOqwXSIADgwMlG3btkn+/Pktz+3Zs0dtw/prEwoBG34bRsayxAb7Ouzb+/fvrwJEnOw1atRINm3a5LbyuXNbwskJ5sXvEcc9HMcQnOFzIHjHa43rAMeh6dOny/z58yVR4UKg5D6NGjXSsmfPrt24cSPGc5cuXbL8vXv3boS82oQJE2LM9/TpU+3q1asxpkdHR2v58uXTBg4cqL344otazZo1Yy1LSEiINnr0aPX3lClT1PK2bdsWY75bt25pDx48UH8/efJEy5Ahg1a3bl2772n8DI6EhYVpjRs31tyhePHiWo0aNTRfsmTJErUuN23apCU1f//9t5YuXTqtaNGi2vnz52M8f/z4cW3mzJkxpnfo0EErV66cNmvWLC0oKEi7e/dujHmwrWK9XrlyxTKtY8eOan57Jk2apOZv3bq13eerV6+utWjRQhswYICWP39+lz5nqVKltHbt2mmJ7dq1a9rt27fV31OnTlWf79SpU06/vkePHlqaNGm0M2fOWKatX79evc+nn37qljLOmTNHvV9s7t+/r2XNmjXGfqBt27bq+7x+/XqM12AfhP3buHHj1Pv36tUrXuV79OiRljp16hjbxezZs1WZ6tevr73xxhtWz02cOFEt89ChQ1pCJaTsu3btUq/Hd29cLwULFtSqVKmS4LLpsAx8j57clpo0aaK+axzPjKZNm6am37lzR0tMDHbcLDw8PM4gBL755hu10WzevNnp9/7111/VaxAofffdd1pAQIB27tw5u/P+/vvvlnkBP+7kyZNrUVFRsS7jwoUL6nVjxozR4svZYAfL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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "_, ax = plt.subplots()\n",
    "\n",
    "ax.plot(q_range, gt_2, 'g--', label='Ground Truth')\n",
    "ax.plot(q_range, est_ss_2, 'bP', label='SS Estimation')\n",
    "ax.plot(q_range, est_ssa_2, 'rs', label='SSA Estimation')\n",
    "ax.set_xscale('log', base=2)\n",
    "ax.set_title(f\"SSA/SS Estimation on CAIDA (n={m:.1e}, W={W:.1e})\")\n",
    "ax.set_ylabel(\"Estimated 3 Norm\")\n",
    "ax.set_xlabel(\"Sample Selection Probability\")\n",
    "ax.legend()\n",
    "plt.show()"
   ]
  }
 ],
 "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
}
