{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import warnings\n",
    "\n",
    "warnings.filterwarnings(\"ignore\", message=\"use_inf_as_na option is deprecated\")\n",
    "warnings.filterwarnings(\"ignore\", message=\"DataFrameGroupBy.apply operated on the grouping columns\")\n",
    "\n",
    "import importlib\n",
    "import json\n",
    "import os\n",
    "import sys\n",
    "import warnings\n",
    "from glob import glob\n",
    "from typing import List, Tuple\n",
    "import itertools\n",
    "import io\n",
    "import copy\n",
    "\n",
    "\n",
    "from scipy import interpolate\n",
    "import matplotlib.pyplot as plt\n",
    "import matplotlib.patches as patches\n",
    "from matplotlib.lines import Line2D\n",
    "import matplotlib.cm as cm\n",
    "import matplotlib.colors as mcolors\n",
    "import matplotlib.gridspec as gridspec\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "import seaborn as sns\n",
    "import sci_palettes\n",
    "\n",
    "sys.path.append(\"/home/lzj/work/llm-infer/other/sglang/eval\")\n",
    "import liz_plot\n",
    "from liz_plot import *\n",
    "from slo_hleper import get_capacity_and_slo, hist_cdf\n",
    "\n",
    "importlib.reload(liz_plot)\n",
    "set_style()\n",
    "paper_rc = {\n",
    "    \"lines.linewidth\": 1.5,\n",
    "    \"lines.markersize\": 5,\n",
    "    \"axes.labelweight\": \"bold\",\n",
    "    \"axes.labelsize\": 10,\n",
    "    \"axes.titlesize\": 10,\n",
    "    \"axes.titleweight\": \"bold\",\n",
    "    \"lines.markerfacecolor\": \"none\",\n",
    "    \"lines.markeredgecolor\": \"auto\",\n",
    "    \"font.size\": 10,\n",
    "    \"font.family\": \"sans-serif\",\n",
    "}\n",
    "plt.rcParams.update(paper_rc)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "figdir = \"/home/lzj/work/llm-infer/other/sglang/design/AAPlot/fig/eval\"\n",
    "\n",
    "\n",
    "def read_predict_csvs(files: List[str], slo):\n",
    "    prefill = None\n",
    "    for f in files:\n",
    "        tmp = pd.read_csv(f).iloc[1:]\n",
    "        tmp[\"request_rate\"] = float(os.path.basename(f).split(\"-\")[1][:-3])\n",
    "        prefill = pd.concat([prefill, tmp], ignore_index=True) if prefill is not None else tmp\n",
    "    prefill.fillna(0, inplace=True)\n",
    "    prefill = (\n",
    "        prefill.sort_values(by=\"start_timstamp\")\n",
    "        .groupby(\"request_rate\", group_keys=False)\n",
    "        .apply(lambda x: x.iloc[2:])\n",
    "    )\n",
    "    prefill[\"norm_start\"] = prefill.groupby(\"request_rate\")[\"start_timstamp\"].transform(lambda x: x - x.min())\n",
    "    prefill[\"norm_end\"] = prefill.groupby(\"request_rate\")[\"end_timestamp\"].transform(\n",
    "        lambda x: x - prefill.groupby(\"request_rate\")[\"start_timstamp\"].transform(\"min\")\n",
    "    )\n",
    "    prefill = prefill.query(\"predict_ms > 0\").copy()\n",
    "    prefill[\"duration\"] = (prefill[\"end_timestamp\"] - prefill[\"start_timstamp\"]) * 1000\n",
    "    prefill[\"err\"] = (prefill[\"predict_ms\"] - prefill[\"duration\"]) / prefill[\"duration\"]\n",
    "    prefill[\"rmse\"] = np.sqrt(np.mean(prefill[\"err\"] ** 2))\n",
    "    prefill[\"tf\"] = ((prefill[\"predict_ms\"] > slo) & (prefill[\"duration\"] > slo)) | ((prefill[\"predict_ms\"] < slo) & (prefill[\"duration\"] < slo))\n",
    "    return prefill\n",
    "\n",
    "\n",
    "def read_reqp_csvs(files: List[str], slo):\n",
    "    prefill = None\n",
    "    for f in files:\n",
    "        tmp = pd.read_csv(f).iloc[1:]\n",
    "        tmp[\"request_rate\"] = float(os.path.basename(f).split(\"-\")[1][:-3])\n",
    "        prefill = pd.concat([prefill, tmp], ignore_index=True) if prefill is not None else tmp\n",
    "    prefill.fillna(0, inplace=True)\n",
    "    prefill[\"end_timestamp\"] = prefill[\"arrive_tstamp\"] + prefill[\"ttft\"] / 1000\n",
    "    prefill[\"start_timstamp\"] = prefill[\"arrive_tstamp\"] + prefill[\"queue_time\"]\n",
    "    prefill = (\n",
    "        prefill.sort_values(by=\"start_timstamp\")\n",
    "        .groupby(\"request_rate\", group_keys=False)\n",
    "        .apply(lambda x: x.iloc[2:])\n",
    "    )\n",
    "    prefill[\"norm_start\"] = prefill.groupby(\"request_rate\")[\"start_timstamp\"].transform(lambda x: x - x.min())\n",
    "    prefill[\"norm_end\"] = prefill.groupby(\"request_rate\")[\"end_timestamp\"].transform(\n",
    "        lambda x: x - prefill.groupby(\"request_rate\")[\"start_timstamp\"].transform(\"min\")\n",
    "    )\n",
    "    prefill = prefill.query(\"ttft_estimate > 0\").copy()\n",
    "    # prefill[\"ttft_norm_estimate\"] = prefill[\"input_len\"] / (prefill[\"queue_time\"] + prefill[\"ttft_gpu_estimate\"])\n",
    "    # prefill[\"duration\"] = (prefill[\"end_timestamp\"] - prefill[\"start_timstamp\"]) * 1000\n",
    "    prefill[\"err\"] = (prefill[\"ttft\"] - prefill[\"ttft_estimate\"]) / prefill[\"ttft\"]\n",
    "    prefill[\"rmse\"] = np.sqrt(np.mean(prefill[\"err\"] ** 2))\n",
    "    prefill[\"tf\"] = ((prefill[\"ttft_norm_estimate\"] > slo) & (prefill[\"ttft_norm\"] > slo)) | ((prefill[\"ttft_norm_estimate\"] < slo) & (prefill[\"ttft_norm\"] < slo))\n",
    "\n",
    "    return prefill\n",
    "\n",
    "\n",
    "def read_many(dirname, backend, request_rate, ttft, tpot):\n",
    "    basedir = f\"/home/lzj/work/llm-infer/other/sglang/log/aaaa3/{dirname}\"\n",
    "    predictdir = os.path.join(basedir, \"predict\")\n",
    "    reqpdir = os.path.join(basedir, \"reqp\")\n",
    "    prefill = read_predict_csvs(glob(os.path.join(predictdir, f\"{backend}-*.prefill.predict.csv\")), ttft)\n",
    "    decode = read_predict_csvs(glob(os.path.join(predictdir, f\"{backend}-*.decode.predict.csv\")), tpot)\n",
    "    reqp = read_reqp_csvs(glob(os.path.join(reqpdir, f\"{backend}-*.prefill.req_predictor.csv\")), ttft)\n",
    "    query = f\"request_rate == {request_rate}\"\n",
    "    return prefill.query(query), decode.query(query), reqp.query(query)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "ss = read_many(\"sharegpt_0324\", \"v25\", 23, 3, 150)\n",
    "cc = read_many(\"splitcode_0222_1728-v2\", \"v27\", 4.0, 1.5, 200)\n",
    "aa = read_many(\"alpaca_0320\", \"v23\", 0.7, 1, 175)\n",
    "DATASETS = [\n",
    "    DatasetConf(name=\"sharegpt\", label=\"ShareGPT\", aitl=3, aitl_ylim=3.2, tpot=150, tpot_ylim=160, aptl=90, aptl_ylim=90, color=\"C0\"),\n",
    "    DatasetConf(name=\"splitcode\", label=\"Azure-Code\", aitl=1.5, aitl_ylim=5, tpot=200, tpot_ylim=210, aptl=4, aptl_ylim=5, color=\"C1\"),\n",
    "    DatasetConf(name=\"alpaca\", label=\"arXiv-Sum\", aitl=1.5, aitl_ylim=3, tpot=175, tpot_ylim=210, aptl=3, aptl_ylim=4, color=\"C2\"),\n",
    "]\n",
    "big = {}\n",
    "for sth, dataset in zip([ss, cc, aa], DATASETS):\n",
    "    for i in sth:\n",
    "        i[\"dataset\"] = dataset.name\n",
    "    big[dataset.name] = {\"prefill\": sth[0], \"decode\": sth[1], \"req\": sth[2]}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 600x80 with 2 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, axes = plt.subplots(1, 2, figsize=(6, 0.8), gridspec_kw={\"wspace\": 0.4})\n",
    "df = pd.DataFrame()\n",
    "for dataset in DATASETS:\n",
    "    for phase in [\"req\", \"decode\"]:\n",
    "        acc = big[dataset.name][phase][\"tf\"].sum() / big[dataset.name][phase][\"tf\"].count()\n",
    "        query = f\"dataset != ''\"\n",
    "        if phase != \"req\":\n",
    "            query = \"prefill_tpc != 54\"\n",
    "        err = big[dataset.name][phase].query(query)[\"err\"].mean()\n",
    "        std = big[dataset.name][phase].query(query)[\"err\"].std()\n",
    "        \n",
    "        df = pd.concat([df, pd.DataFrame({\"dataset\": dataset.label, \"phase\": phase,\n",
    "                                          \"accuracy\": acc, \"error\": err, \"std\": std}, index=[0])], ignore_index=True)\n",
    "ax: plt.Axes = axes[0]\n",
    "y = [\"accuracy\", \"error\", \"std\"][0]\n",
    "d2 = df.groupby([\"dataset\"]).mean(numeric_only=True).reset_index()\n",
    "sns.barplot(data=d2, x=\"accuracy\", y=\"dataset\", ax=ax, orient=\"h\",\n",
    "            palette=[f\"{i.color}\" for i in DATASETS], order=[i.label for i in DATASETS],\n",
    "            dodge=True, edgecolor=\"black\", linewidth=1)\n",
    "hatches = ['..', '//', '--', '//', '++', '\\\\\\\\', 'xx', '**']\n",
    "for hues, hatch in zip(ax.containers, hatches):\n",
    "    for i, bar in enumerate(hues):\n",
    "        bar.set_hatch(hatches[i])\n",
    "    # for hue in hues:\n",
    "    #     hue.set_hatch(hatch)\n",
    "\n",
    "ax.set_xlabel(\"Accuracy\")\n",
    "ax.set_ylabel(\"\")\n",
    "# ax.set_xticklabels([\"Prefill\", \"Decode\"], fontweight=\"bold\")\n",
    "handle, label = ax.get_legend_handles_labels()\n",
    "ax.set_xlim(0, 1)\n",
    "autogrid(ax, axises=\"x\", which=\"major\")\n",
    "\n",
    "ax = axes[1]\n",
    "for i, dataset in enumerate(DATASETS[::-1]):\n",
    "    data = big[dataset.name][\"prefill\"].query(\"prefill_tpc <= 54\")\n",
    "    x = data[\"duration\"] / 1000\n",
    "    y = data[\"predict_ms\"] / 1000\n",
    "    sns.scatterplot(data, x=x, y=y, ax=ax, label=dataset.label, s=20, \n",
    "                    alpha=0.7, marker=[\"+\", \"1\", 'x'][i], color=dataset.color)\n",
    "    sns.lineplot(x=[0, 2.5], y=[0, 2.5], ax=ax, color=\"black\", linestyle=\"--\", linewidth=1)\n",
    "    ax.set_xlim(0, 2)\n",
    "    ax.set_ylim(0, 2)\n",
    "ax.set_xlabel(\"Actual Time (s)\")\n",
    "ax.set_ylabel(\"Predicted\\nTime (s)\")\n",
    "ax.legend().remove()\n",
    "\n",
    "handle2, label = ax.get_legend_handles_labels()\n",
    "legend_scatter_handles = []\n",
    "for h in handle2:\n",
    "    h_copy = copy.copy(h)\n",
    "    h_copy.set_sizes([50])\n",
    "    h_copy.set_linewidths([2])\n",
    "    legend_scatter_handles.append(h_copy)\n",
    "super_handle = [(bar, dot) for bar, dot in zip(axes[0].containers[0], legend_scatter_handles[::-1])]\n",
    "axes[0].legend(super_handle, label, handler_map={tuple: mpl.legend_handler.HandlerTuple(ndivide=None)},\n",
    "                handlelength=4, loc=\"lower left\", bbox_to_anchor=(0, 1.0), ncol=3, frameon=False)\n",
    "fig.savefig(os.path.join(figdir, \"eval.abla.accuracy.pdf\"), bbox_inches=\"tight\")\n",
    "\n",
    "df.columns = [\"Dataset\", \"Phase\", \"Accuracy\", \"Error\", \"Std\"]\n",
    "# df = df[df.columns[:3]]\n",
    "df.replace({\"req\": \"Prefill\", \"decode\": \"Decode\"}, inplace=True)\n",
    "# df.pivot_table(index=\"Phase\", columns=\"Dataset\", values=[\"Accuracy\", \"Error\"])\n",
    "# print(df.pivot_table(index=[\"Phase\"], columns=[\"Dataset\"], values=\"Accuracy\").to_latex(float_format=\"%.2f\"))"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "gllm",
   "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.10.0"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
