{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "import os\n",
    "import json\n",
    "import numpy as np\n",
    "import pandas as pd\n",
    "\n",
    "def extract_metrics(data, share_ttft_con, share_tpot_cons):\n",
    "    ttfts = data['ttfts']\n",
    "    tpots = data['tpots']\n",
    "    e2els = data['e2els']\n",
    "\n",
    "    ttft_class = []  # sharegpt, splitwise\n",
    "    tpot_class = []\n",
    "    e2el_class = []\n",
    "    goodput = 0\n",
    "\n",
    "\n",
    "    for i in range(len(ttfts)):\n",
    "        ttft_class.append(ttfts[i])\n",
    "        tpot_class.append(tpots[i])\n",
    "        e2el_class.append(e2els[i])\n",
    "\n",
    "\n",
    "        if ttfts[i] <= share_ttft_con and tpots[i] <= share_tpot_cons:\n",
    "            goodput += 1\n",
    "\n",
    "    scale = 1000  # to ms\n",
    "\n",
    "    result = {\n",
    "        'splitwise_avg_ttft': round(np.mean(ttft_class) * scale, 4),\n",
    "        'splitwise_avg_tpot': round(np.mean(tpot_class) * scale, 4),\n",
    "        'splitwise_avg_e2el': round(np.mean(e2el_class) * scale, 4),\n",
    "        'splitwise_goodput': round(goodput / len(ttft_class) * 100, 4)\n",
    "    }\n",
    "    return result\n",
    "\n",
    "def process_all_results(base_dir, share_ttft_con, share_tpot_cons):\n",
    "    all_data = []\n",
    "    for filename in os.listdir(base_dir):\n",
    "        if filename.startswith('rate_') and filename.endswith('.json'):\n",
    "            rate_value = int(filename.split('_')[1].split('.')[0])\n",
    "            filepath = os.path.join(base_dir, filename)\n",
    "            with open(filepath, 'r') as f:\n",
    "                data = json.load(f)\n",
    "                metrics = extract_metrics(data, share_ttft_con, share_tpot_cons)\n",
    "                token_throughput = round(data['total_token_throughput'], 4)\n",
    "                metrics['rate'] = rate_value\n",
    "                metrics['token_throughput'] = token_throughput\n",
    "                all_data.append(metrics)\n",
    "    if len(all_data) != 0:\n",
    "        df = pd.DataFrame(all_data)\n",
    "        df = df.sort_values('rate')  # 按rate排序\n",
    "        df = df.set_index('rate')\n",
    "        return df\n",
    "    else:\n",
    "        return None\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "base_dir = '/mnt/sda/2022-0526/home/xuhx/projects/gLLM/experiments/results/32B/splitwise_conv_filiter/'\n",
    "# base_dir = '/mnt/sda/2022-0526/home/xuhx/projects/gLLM/backup/results_429_1800/splitwise_conv/'\n",
    "share_ttft_con = 8\n",
    "share_tpot_cons = 0.15"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>splitwise_avg_ttft</th>\n",
       "      <th>splitwise_avg_tpot</th>\n",
       "      <th>splitwise_avg_e2el</th>\n",
       "      <th>splitwise_goodput</th>\n",
       "      <th>token_throughput</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>rate</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>711.9677</td>\n",
       "      <td>63.7622</td>\n",
       "      <td>12986.0382</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>535.3208</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>743.6176</td>\n",
       "      <td>64.9414</td>\n",
       "      <td>14006.7684</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>1148.9412</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>756.0523</td>\n",
       "      <td>68.4926</td>\n",
       "      <td>14352.8667</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>1729.8845</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>766.2083</td>\n",
       "      <td>73.3335</td>\n",
       "      <td>14515.4166</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>2193.5728</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>771.1655</td>\n",
       "      <td>78.4683</td>\n",
       "      <td>15114.6904</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>2623.9551</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>802.6225</td>\n",
       "      <td>88.3668</td>\n",
       "      <td>17402.6022</td>\n",
       "      <td>99.6094</td>\n",
       "      <td>3331.1319</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7</th>\n",
       "      <td>825.0664</td>\n",
       "      <td>102.2331</td>\n",
       "      <td>19977.7846</td>\n",
       "      <td>99.6652</td>\n",
       "      <td>3926.8526</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>8</th>\n",
       "      <td>909.0211</td>\n",
       "      <td>121.8171</td>\n",
       "      <td>23370.5778</td>\n",
       "      <td>86.9141</td>\n",
       "      <td>4520.0727</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>9</th>\n",
       "      <td>3377.4723</td>\n",
       "      <td>134.5341</td>\n",
       "      <td>28320.6313</td>\n",
       "      <td>64.8438</td>\n",
       "      <td>4788.9724</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>10</th>\n",
       "      <td>7929.1330</td>\n",
       "      <td>138.2735</td>\n",
       "      <td>33004.1628</td>\n",
       "      <td>46.6406</td>\n",
       "      <td>4960.6040</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>11</th>\n",
       "      <td>13690.3025</td>\n",
       "      <td>143.0766</td>\n",
       "      <td>38990.1707</td>\n",
       "      <td>27.5568</td>\n",
       "      <td>5120.4447</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>12</th>\n",
       "      <td>19324.2522</td>\n",
       "      <td>149.0577</td>\n",
       "      <td>44998.7866</td>\n",
       "      <td>21.2240</td>\n",
       "      <td>5256.2783</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>13</th>\n",
       "      <td>37617.0658</td>\n",
       "      <td>173.4772</td>\n",
       "      <td>66939.6689</td>\n",
       "      <td>0.7212</td>\n",
       "      <td>4880.0143</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>14</th>\n",
       "      <td>31089.6192</td>\n",
       "      <td>154.0907</td>\n",
       "      <td>57066.2777</td>\n",
       "      <td>12.2210</td>\n",
       "      <td>5436.0350</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "      splitwise_avg_ttft  splitwise_avg_tpot  splitwise_avg_e2el  \\\n",
       "rate                                                               \n",
       "1               711.9677             63.7622          12986.0382   \n",
       "2               743.6176             64.9414          14006.7684   \n",
       "3               756.0523             68.4926          14352.8667   \n",
       "4               766.2083             73.3335          14515.4166   \n",
       "5               771.1655             78.4683          15114.6904   \n",
       "6               802.6225             88.3668          17402.6022   \n",
       "7               825.0664            102.2331          19977.7846   \n",
       "8               909.0211            121.8171          23370.5778   \n",
       "9              3377.4723            134.5341          28320.6313   \n",
       "10             7929.1330            138.2735          33004.1628   \n",
       "11            13690.3025            143.0766          38990.1707   \n",
       "12            19324.2522            149.0577          44998.7866   \n",
       "13            37617.0658            173.4772          66939.6689   \n",
       "14            31089.6192            154.0907          57066.2777   \n",
       "\n",
       "      splitwise_goodput  token_throughput  \n",
       "rate                                       \n",
       "1              100.0000          535.3208  \n",
       "2              100.0000         1148.9412  \n",
       "3              100.0000         1729.8845  \n",
       "4              100.0000         2193.5728  \n",
       "5              100.0000         2623.9551  \n",
       "6               99.6094         3331.1319  \n",
       "7               99.6652         3926.8526  \n",
       "8               86.9141         4520.0727  \n",
       "9               64.8438         4788.9724  \n",
       "10              46.6406         4960.6040  \n",
       "11              27.5568         5120.4447  \n",
       "12              21.2240         5256.2783  \n",
       "13               0.7212         4880.0143  \n",
       "14              12.2210         5436.0350  "
      ]
     },
     "execution_count": 3,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df_dynamic= process_all_results(base_dir + 'dynamic', share_ttft_con, share_tpot_cons)\n",
    "df_dynamic"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>splitwise_avg_ttft</th>\n",
       "      <th>splitwise_avg_tpot</th>\n",
       "      <th>splitwise_avg_e2el</th>\n",
       "      <th>splitwise_goodput</th>\n",
       "      <th>token_throughput</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>rate</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>744.3359</td>\n",
       "      <td>66.5637</td>\n",
       "      <td>13639.7264</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>528.9632</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>795.3028</td>\n",
       "      <td>70.5100</td>\n",
       "      <td>15270.3222</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>1128.0506</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>825.6241</td>\n",
       "      <td>75.5900</td>\n",
       "      <td>15892.6353</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>1698.3582</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>821.8712</td>\n",
       "      <td>80.6436</td>\n",
       "      <td>15990.3340</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>2151.7364</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>845.6045</td>\n",
       "      <td>88.6456</td>\n",
       "      <td>17087.0131</td>\n",
       "      <td>99.8438</td>\n",
       "      <td>2579.4485</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>873.5825</td>\n",
       "      <td>99.4698</td>\n",
       "      <td>19603.8771</td>\n",
       "      <td>99.4792</td>\n",
       "      <td>3268.4489</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7</th>\n",
       "      <td>907.4180</td>\n",
       "      <td>116.3296</td>\n",
       "      <td>22608.5409</td>\n",
       "      <td>94.7545</td>\n",
       "      <td>3815.2457</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>8</th>\n",
       "      <td>952.0268</td>\n",
       "      <td>138.8960</td>\n",
       "      <td>26446.0141</td>\n",
       "      <td>70.5078</td>\n",
       "      <td>4380.6579</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>9</th>\n",
       "      <td>3816.6060</td>\n",
       "      <td>153.2010</td>\n",
       "      <td>32222.7644</td>\n",
       "      <td>24.8264</td>\n",
       "      <td>4595.1372</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>10</th>\n",
       "      <td>8773.6669</td>\n",
       "      <td>158.3983</td>\n",
       "      <td>37546.5098</td>\n",
       "      <td>9.6094</td>\n",
       "      <td>4721.3616</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>11</th>\n",
       "      <td>15134.9678</td>\n",
       "      <td>164.5170</td>\n",
       "      <td>44359.2811</td>\n",
       "      <td>5.2557</td>\n",
       "      <td>4862.7532</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>12</th>\n",
       "      <td>20866.1601</td>\n",
       "      <td>169.1288</td>\n",
       "      <td>50234.1814</td>\n",
       "      <td>2.4740</td>\n",
       "      <td>5011.0841</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>13</th>\n",
       "      <td>26495.3156</td>\n",
       "      <td>173.9077</td>\n",
       "      <td>56155.9274</td>\n",
       "      <td>1.6226</td>\n",
       "      <td>5144.4410</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>14</th>\n",
       "      <td>34273.1219</td>\n",
       "      <td>134.6765</td>\n",
       "      <td>64567.7140</td>\n",
       "      <td>1.0045</td>\n",
       "      <td>5160.9551</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "      splitwise_avg_ttft  splitwise_avg_tpot  splitwise_avg_e2el  \\\n",
       "rate                                                               \n",
       "1               744.3359             66.5637          13639.7264   \n",
       "2               795.3028             70.5100          15270.3222   \n",
       "3               825.6241             75.5900          15892.6353   \n",
       "4               821.8712             80.6436          15990.3340   \n",
       "5               845.6045             88.6456          17087.0131   \n",
       "6               873.5825             99.4698          19603.8771   \n",
       "7               907.4180            116.3296          22608.5409   \n",
       "8               952.0268            138.8960          26446.0141   \n",
       "9              3816.6060            153.2010          32222.7644   \n",
       "10             8773.6669            158.3983          37546.5098   \n",
       "11            15134.9678            164.5170          44359.2811   \n",
       "12            20866.1601            169.1288          50234.1814   \n",
       "13            26495.3156            173.9077          56155.9274   \n",
       "14            34273.1219            134.6765          64567.7140   \n",
       "\n",
       "      splitwise_goodput  token_throughput  \n",
       "rate                                       \n",
       "1              100.0000          528.9632  \n",
       "2              100.0000         1128.0506  \n",
       "3              100.0000         1698.3582  \n",
       "4              100.0000         2151.7364  \n",
       "5               99.8438         2579.4485  \n",
       "6               99.4792         3268.4489  \n",
       "7               94.7545         3815.2457  \n",
       "8               70.5078         4380.6579  \n",
       "9               24.8264         4595.1372  \n",
       "10               9.6094         4721.3616  \n",
       "11               5.2557         4862.7532  \n",
       "12               2.4740         5011.0841  \n",
       "13               1.6226         5144.4410  \n",
       "14               1.0045         5160.9551  "
      ]
     },
     "execution_count": 4,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "\n",
    "# 用法\n",
    "df_0 = process_all_results(base_dir + 'baseline0',  share_ttft_con, share_tpot_cons)\n",
    "df_0"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>splitwise_avg_ttft</th>\n",
       "      <th>splitwise_avg_tpot</th>\n",
       "      <th>splitwise_avg_e2el</th>\n",
       "      <th>splitwise_goodput</th>\n",
       "      <th>token_throughput</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>rate</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>711.8897</td>\n",
       "      <td>62.9558</td>\n",
       "      <td>12863.2957</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>535.2314</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>763.7168</td>\n",
       "      <td>66.1132</td>\n",
       "      <td>14331.3249</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>1138.4844</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>775.1148</td>\n",
       "      <td>70.1272</td>\n",
       "      <td>14759.0227</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>1715.2022</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>793.3429</td>\n",
       "      <td>75.9195</td>\n",
       "      <td>15096.3045</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>2177.5195</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>814.9423</td>\n",
       "      <td>82.9944</td>\n",
       "      <td>16020.4942</td>\n",
       "      <td>99.8438</td>\n",
       "      <td>2608.1001</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>811.1806</td>\n",
       "      <td>90.8720</td>\n",
       "      <td>17949.5465</td>\n",
       "      <td>99.7396</td>\n",
       "      <td>3305.2570</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7</th>\n",
       "      <td>849.0798</td>\n",
       "      <td>107.3787</td>\n",
       "      <td>21009.9067</td>\n",
       "      <td>99.4420</td>\n",
       "      <td>3874.1596</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>8</th>\n",
       "      <td>924.8551</td>\n",
       "      <td>130.6919</td>\n",
       "      <td>24917.0161</td>\n",
       "      <td>77.3438</td>\n",
       "      <td>4478.7733</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>9</th>\n",
       "      <td>3058.6101</td>\n",
       "      <td>142.1365</td>\n",
       "      <td>29413.3185</td>\n",
       "      <td>49.8264</td>\n",
       "      <td>4732.4321</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>10</th>\n",
       "      <td>6926.8355</td>\n",
       "      <td>146.1642</td>\n",
       "      <td>33489.3911</td>\n",
       "      <td>20.4688</td>\n",
       "      <td>4905.9164</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>11</th>\n",
       "      <td>12536.0491</td>\n",
       "      <td>150.7311</td>\n",
       "      <td>39335.2887</td>\n",
       "      <td>11.8608</td>\n",
       "      <td>5091.6771</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>12</th>\n",
       "      <td>18423.7877</td>\n",
       "      <td>157.0368</td>\n",
       "      <td>45689.0439</td>\n",
       "      <td>9.7656</td>\n",
       "      <td>5228.4078</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "      splitwise_avg_ttft  splitwise_avg_tpot  splitwise_avg_e2el  \\\n",
       "rate                                                               \n",
       "1               711.8897             62.9558          12863.2957   \n",
       "2               763.7168             66.1132          14331.3249   \n",
       "3               775.1148             70.1272          14759.0227   \n",
       "4               793.3429             75.9195          15096.3045   \n",
       "5               814.9423             82.9944          16020.4942   \n",
       "6               811.1806             90.8720          17949.5465   \n",
       "7               849.0798            107.3787          21009.9067   \n",
       "8               924.8551            130.6919          24917.0161   \n",
       "9              3058.6101            142.1365          29413.3185   \n",
       "10             6926.8355            146.1642          33489.3911   \n",
       "11            12536.0491            150.7311          39335.2887   \n",
       "12            18423.7877            157.0368          45689.0439   \n",
       "\n",
       "      splitwise_goodput  token_throughput  \n",
       "rate                                       \n",
       "1              100.0000          535.2314  \n",
       "2              100.0000         1138.4844  \n",
       "3              100.0000         1715.2022  \n",
       "4              100.0000         2177.5195  \n",
       "5               99.8438         2608.1001  \n",
       "6               99.7396         3305.2570  \n",
       "7               99.4420         3874.1596  \n",
       "8               77.3438         4478.7733  \n",
       "9               49.8264         4732.4321  \n",
       "10              20.4688         4905.9164  \n",
       "11              11.8608         5091.6771  \n",
       "12               9.7656         5228.4078  "
      ]
     },
     "execution_count": 5,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df_1 = process_all_results(base_dir + 'baseline1',  share_ttft_con, share_tpot_cons)\n",
    "df_1\n",
    "# rate 8 有问题"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>splitwise_avg_ttft</th>\n",
       "      <th>splitwise_avg_tpot</th>\n",
       "      <th>splitwise_avg_e2el</th>\n",
       "      <th>splitwise_goodput</th>\n",
       "      <th>token_throughput</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>rate</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>692.6691</td>\n",
       "      <td>60.8598</td>\n",
       "      <td>12494.8216</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>538.0820</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>740.0014</td>\n",
       "      <td>64.1198</td>\n",
       "      <td>13900.0302</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>1144.9716</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>761.1571</td>\n",
       "      <td>68.0622</td>\n",
       "      <td>14320.5303</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>1725.4153</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>768.3487</td>\n",
       "      <td>72.7022</td>\n",
       "      <td>14445.7275</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>2193.3647</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>772.7495</td>\n",
       "      <td>77.9635</td>\n",
       "      <td>15049.2425</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>2620.9482</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>801.6183</td>\n",
       "      <td>87.6572</td>\n",
       "      <td>17317.0986</td>\n",
       "      <td>99.7396</td>\n",
       "      <td>3336.4749</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7</th>\n",
       "      <td>832.6950</td>\n",
       "      <td>102.4006</td>\n",
       "      <td>20048.1877</td>\n",
       "      <td>99.6652</td>\n",
       "      <td>3914.5807</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>8</th>\n",
       "      <td>887.0373</td>\n",
       "      <td>122.3780</td>\n",
       "      <td>23445.9578</td>\n",
       "      <td>85.9375</td>\n",
       "      <td>4533.1471</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>9</th>\n",
       "      <td>2464.4372</td>\n",
       "      <td>137.6267</td>\n",
       "      <td>27954.8237</td>\n",
       "      <td>60.5903</td>\n",
       "      <td>4822.4821</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>10</th>\n",
       "      <td>6723.0415</td>\n",
       "      <td>144.2638</td>\n",
       "      <td>32912.7896</td>\n",
       "      <td>20.2344</td>\n",
       "      <td>4997.3206</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>11</th>\n",
       "      <td>11820.5009</td>\n",
       "      <td>148.6018</td>\n",
       "      <td>38193.1830</td>\n",
       "      <td>15.4119</td>\n",
       "      <td>5162.9009</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>12</th>\n",
       "      <td>17654.0032</td>\n",
       "      <td>154.9337</td>\n",
       "      <td>44481.7849</td>\n",
       "      <td>12.7604</td>\n",
       "      <td>5303.4693</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "      splitwise_avg_ttft  splitwise_avg_tpot  splitwise_avg_e2el  \\\n",
       "rate                                                               \n",
       "1               692.6691             60.8598          12494.8216   \n",
       "2               740.0014             64.1198          13900.0302   \n",
       "3               761.1571             68.0622          14320.5303   \n",
       "4               768.3487             72.7022          14445.7275   \n",
       "5               772.7495             77.9635          15049.2425   \n",
       "6               801.6183             87.6572          17317.0986   \n",
       "7               832.6950            102.4006          20048.1877   \n",
       "8               887.0373            122.3780          23445.9578   \n",
       "9              2464.4372            137.6267          27954.8237   \n",
       "10             6723.0415            144.2638          32912.7896   \n",
       "11            11820.5009            148.6018          38193.1830   \n",
       "12            17654.0032            154.9337          44481.7849   \n",
       "\n",
       "      splitwise_goodput  token_throughput  \n",
       "rate                                       \n",
       "1              100.0000          538.0820  \n",
       "2              100.0000         1144.9716  \n",
       "3              100.0000         1725.4153  \n",
       "4              100.0000         2193.3647  \n",
       "5              100.0000         2620.9482  \n",
       "6               99.7396         3336.4749  \n",
       "7               99.6652         3914.5807  \n",
       "8               85.9375         4533.1471  \n",
       "9               60.5903         4822.4821  \n",
       "10              20.2344         4997.3206  \n",
       "11              15.4119         5162.9009  \n",
       "12              12.7604         5303.4693  "
      ]
     },
     "execution_count": 6,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df_2 = process_all_results(base_dir + 'baseline2',  share_ttft_con, share_tpot_cons)\n",
    "df_2"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>splitwise_avg_ttft</th>\n",
       "      <th>splitwise_avg_tpot</th>\n",
       "      <th>splitwise_avg_e2el</th>\n",
       "      <th>splitwise_goodput</th>\n",
       "      <th>token_throughput</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>rate</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>708.0732</td>\n",
       "      <td>61.6378</td>\n",
       "      <td>12630.2876</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>534.6448</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>750.7808</td>\n",
       "      <td>64.5250</td>\n",
       "      <td>14006.7838</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>1147.9584</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>779.7235</td>\n",
       "      <td>68.7179</td>\n",
       "      <td>14487.6197</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>1726.8190</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>775.4437</td>\n",
       "      <td>73.4732</td>\n",
       "      <td>14603.0873</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>2190.9209</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>783.6952</td>\n",
       "      <td>79.4002</td>\n",
       "      <td>15337.3218</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>2610.5015</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>837.4477</td>\n",
       "      <td>90.7321</td>\n",
       "      <td>17783.0949</td>\n",
       "      <td>99.3490</td>\n",
       "      <td>3334.5074</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7</th>\n",
       "      <td>836.0708</td>\n",
       "      <td>102.9909</td>\n",
       "      <td>20117.8924</td>\n",
       "      <td>99.5536</td>\n",
       "      <td>3923.1691</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>8</th>\n",
       "      <td>906.6666</td>\n",
       "      <td>124.1219</td>\n",
       "      <td>23719.1578</td>\n",
       "      <td>83.1055</td>\n",
       "      <td>4538.8636</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>9</th>\n",
       "      <td>2731.5947</td>\n",
       "      <td>139.3892</td>\n",
       "      <td>28535.5942</td>\n",
       "      <td>55.4688</td>\n",
       "      <td>4821.6487</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>10</th>\n",
       "      <td>6761.5403</td>\n",
       "      <td>144.4382</td>\n",
       "      <td>32956.4223</td>\n",
       "      <td>19.6094</td>\n",
       "      <td>5005.8133</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>11</th>\n",
       "      <td>11503.3666</td>\n",
       "      <td>147.9420</td>\n",
       "      <td>37787.5132</td>\n",
       "      <td>13.7784</td>\n",
       "      <td>5168.7044</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>12</th>\n",
       "      <td>17973.6982</td>\n",
       "      <td>155.6798</td>\n",
       "      <td>44955.5895</td>\n",
       "      <td>11.9792</td>\n",
       "      <td>5288.3043</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>13</th>\n",
       "      <td>24122.5716</td>\n",
       "      <td>153.4537</td>\n",
       "      <td>51523.4356</td>\n",
       "      <td>10.7572</td>\n",
       "      <td>5415.9904</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>14</th>\n",
       "      <td>28840.6245</td>\n",
       "      <td>122.0274</td>\n",
       "      <td>56156.2705</td>\n",
       "      <td>9.0960</td>\n",
       "      <td>5539.6481</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "      splitwise_avg_ttft  splitwise_avg_tpot  splitwise_avg_e2el  \\\n",
       "rate                                                               \n",
       "1               708.0732             61.6378          12630.2876   \n",
       "2               750.7808             64.5250          14006.7838   \n",
       "3               779.7235             68.7179          14487.6197   \n",
       "4               775.4437             73.4732          14603.0873   \n",
       "5               783.6952             79.4002          15337.3218   \n",
       "6               837.4477             90.7321          17783.0949   \n",
       "7               836.0708            102.9909          20117.8924   \n",
       "8               906.6666            124.1219          23719.1578   \n",
       "9              2731.5947            139.3892          28535.5942   \n",
       "10             6761.5403            144.4382          32956.4223   \n",
       "11            11503.3666            147.9420          37787.5132   \n",
       "12            17973.6982            155.6798          44955.5895   \n",
       "13            24122.5716            153.4537          51523.4356   \n",
       "14            28840.6245            122.0274          56156.2705   \n",
       "\n",
       "      splitwise_goodput  token_throughput  \n",
       "rate                                       \n",
       "1              100.0000          534.6448  \n",
       "2              100.0000         1147.9584  \n",
       "3              100.0000         1726.8190  \n",
       "4              100.0000         2190.9209  \n",
       "5              100.0000         2610.5015  \n",
       "6               99.3490         3334.5074  \n",
       "7               99.5536         3923.1691  \n",
       "8               83.1055         4538.8636  \n",
       "9               55.4688         4821.6487  \n",
       "10              19.6094         5005.8133  \n",
       "11              13.7784         5168.7044  \n",
       "12              11.9792         5288.3043  \n",
       "13              10.7572         5415.9904  \n",
       "14               9.0960         5539.6481  "
      ]
     },
     "execution_count": 7,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df_3 = process_all_results(base_dir + 'baseline3',  share_ttft_con, share_tpot_cons)\n",
    "df_3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>splitwise_avg_ttft</th>\n",
       "      <th>splitwise_avg_tpot</th>\n",
       "      <th>splitwise_avg_e2el</th>\n",
       "      <th>splitwise_goodput</th>\n",
       "      <th>token_throughput</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>rate</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>1</th>\n",
       "      <td>236.4249</td>\n",
       "      <td>70.3987</td>\n",
       "      <td>14164.6340</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>537.8618</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>294.7376</td>\n",
       "      <td>88.8235</td>\n",
       "      <td>18660.9006</td>\n",
       "      <td>99.6094</td>\n",
       "      <td>1137.9421</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>3</th>\n",
       "      <td>348.0235</td>\n",
       "      <td>113.5910</td>\n",
       "      <td>22673.8907</td>\n",
       "      <td>96.0938</td>\n",
       "      <td>1709.2028</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>475.4114</td>\n",
       "      <td>160.8393</td>\n",
       "      <td>30094.8419</td>\n",
       "      <td>33.7891</td>\n",
       "      <td>2149.7253</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>5</th>\n",
       "      <td>490.5190</td>\n",
       "      <td>177.3461</td>\n",
       "      <td>32064.9901</td>\n",
       "      <td>22.5000</td>\n",
       "      <td>2561.7741</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>1607.1245</td>\n",
       "      <td>225.9839</td>\n",
       "      <td>41755.7748</td>\n",
       "      <td>12.3698</td>\n",
       "      <td>3070.5476</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>7</th>\n",
       "      <td>9359.3506</td>\n",
       "      <td>243.1472</td>\n",
       "      <td>52752.6839</td>\n",
       "      <td>1.2277</td>\n",
       "      <td>3168.5647</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>8</th>\n",
       "      <td>19774.9450</td>\n",
       "      <td>257.7719</td>\n",
       "      <td>65429.0650</td>\n",
       "      <td>0.8789</td>\n",
       "      <td>3268.5698</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>9</th>\n",
       "      <td>35890.5529</td>\n",
       "      <td>281.0924</td>\n",
       "      <td>86237.3719</td>\n",
       "      <td>0.5208</td>\n",
       "      <td>3119.2142</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>10</th>\n",
       "      <td>41919.0492</td>\n",
       "      <td>267.2541</td>\n",
       "      <td>89355.2445</td>\n",
       "      <td>0.4688</td>\n",
       "      <td>3295.6968</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>11</th>\n",
       "      <td>48568.3656</td>\n",
       "      <td>258.9026</td>\n",
       "      <td>94019.0036</td>\n",
       "      <td>0.4261</td>\n",
       "      <td>3462.7640</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>12</th>\n",
       "      <td>60371.2332</td>\n",
       "      <td>263.0544</td>\n",
       "      <td>105897.6033</td>\n",
       "      <td>0.2604</td>\n",
       "      <td>3492.6698</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "      splitwise_avg_ttft  splitwise_avg_tpot  splitwise_avg_e2el  \\\n",
       "rate                                                               \n",
       "1               236.4249             70.3987          14164.6340   \n",
       "2               294.7376             88.8235          18660.9006   \n",
       "3               348.0235            113.5910          22673.8907   \n",
       "4               475.4114            160.8393          30094.8419   \n",
       "5               490.5190            177.3461          32064.9901   \n",
       "6              1607.1245            225.9839          41755.7748   \n",
       "7              9359.3506            243.1472          52752.6839   \n",
       "8             19774.9450            257.7719          65429.0650   \n",
       "9             35890.5529            281.0924          86237.3719   \n",
       "10            41919.0492            267.2541          89355.2445   \n",
       "11            48568.3656            258.9026          94019.0036   \n",
       "12            60371.2332            263.0544         105897.6033   \n",
       "\n",
       "      splitwise_goodput  token_throughput  \n",
       "rate                                       \n",
       "1              100.0000          537.8618  \n",
       "2               99.6094         1137.9421  \n",
       "3               96.0938         1709.2028  \n",
       "4               33.7891         2149.7253  \n",
       "5               22.5000         2561.7741  \n",
       "6               12.3698         3070.5476  \n",
       "7                1.2277         3168.5647  \n",
       "8                0.8789         3268.5698  \n",
       "9                0.5208         3119.2142  \n",
       "10               0.4688         3295.6968  \n",
       "11               0.4261         3462.7640  \n",
       "12               0.2604         3492.6698  "
      ]
     },
     "execution_count": 8,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df_vllm = process_all_results(base_dir + 'vllm',  share_ttft_con, share_tpot_cons)\n",
    "df_vllm"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>splitwise_avg_ttft</th>\n",
       "      <th>splitwise_avg_tpot</th>\n",
       "      <th>splitwise_avg_e2el</th>\n",
       "      <th>splitwise_goodput</th>\n",
       "      <th>token_throughput</th>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>rate</th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "      <th></th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>2</th>\n",
       "      <td>284.0428</td>\n",
       "      <td>35.3849</td>\n",
       "      <td>11258.8111</td>\n",
       "      <td>100.0000</td>\n",
       "      <td>1408.7614</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>4</th>\n",
       "      <td>484.2710</td>\n",
       "      <td>57.3376</td>\n",
       "      <td>16586.8557</td>\n",
       "      <td>99.6094</td>\n",
       "      <td>2632.8177</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>6</th>\n",
       "      <td>1654.0817</td>\n",
       "      <td>170.2722</td>\n",
       "      <td>45639.4524</td>\n",
       "      <td>28.3854</td>\n",
       "      <td>3677.6580</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>8</th>\n",
       "      <td>19061.8729</td>\n",
       "      <td>233.4118</td>\n",
       "      <td>77776.8841</td>\n",
       "      <td>1.5625</td>\n",
       "      <td>3815.6269</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>10</th>\n",
       "      <td>43680.9648</td>\n",
       "      <td>239.4381</td>\n",
       "      <td>103445.0282</td>\n",
       "      <td>0.9375</td>\n",
       "      <td>3871.2655</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>12</th>\n",
       "      <td>67893.3369</td>\n",
       "      <td>247.9801</td>\n",
       "      <td>128411.3661</td>\n",
       "      <td>0.4557</td>\n",
       "      <td>3923.4726</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "      splitwise_avg_ttft  splitwise_avg_tpot  splitwise_avg_e2el  \\\n",
       "rate                                                               \n",
       "2               284.0428             35.3849          11258.8111   \n",
       "4               484.2710             57.3376          16586.8557   \n",
       "6              1654.0817            170.2722          45639.4524   \n",
       "8             19061.8729            233.4118          77776.8841   \n",
       "10            43680.9648            239.4381         103445.0282   \n",
       "12            67893.3369            247.9801         128411.3661   \n",
       "\n",
       "      splitwise_goodput  token_throughput  \n",
       "rate                                       \n",
       "2              100.0000         1408.7614  \n",
       "4               99.6094         2632.8177  \n",
       "6               28.3854         3677.6580  \n",
       "8                1.5625         3815.6269  \n",
       "10               0.9375         3871.2655  \n",
       "12               0.4557         3923.4726  "
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "df_sgl = process_all_results(base_dir + 'sglang',  share_ttft_con, share_tpot_cons)\n",
    "df_sgl"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 600x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# 设置图表样式\n",
    "plt.style.use('seaborn-v0_8')\n",
    "plt.rcParams['axes.unicode_minus'] = False\n",
    "\n",
    "# 选择第三个指标（索引为2）\n",
    "metric = 'splitwise_avg_e2el'\n",
    "title = 'E2EL (ms)'\n",
    "\n",
    "# 设置图表\n",
    "fig, ax = plt.subplots(figsize=(6, 5))\n",
    "# ax.set_title(title, fontsize=14)\n",
    "ax.set_xlabel('Request Rate (req/s)', fontsize=12)\n",
    "ax.set_ylabel('E2EL (ms)', fontsize=12)\n",
    "\n",
    "selected_rates = [6,8,10,12] # 32b\n",
    "# selected_rates = [4,8,12,16] # 14b\n",
    "\n",
    "# 颜色、样式设置\n",
    "colors = ['#1f77b4', '#ff7f0e', '#2ca02c', '#9467bd']\n",
    "markers = ['o', 's', '^', '*']\n",
    "linestyles = ['-', '-', '-', '-']\n",
    "linewidths = [2, 2, 2, 2]\n",
    "\n",
    "# 绘图\n",
    "ax.plot(selected_rates, df_0.loc[selected_rates, metric], marker=markers[0], color=colors[0], linestyle=linestyles[0], linewidth=linewidths[0])\n",
    "ax.plot(selected_rates, df_vllm.loc[selected_rates, metric], marker=markers[1], color=colors[1], linestyle=linestyles[1], linewidth=linewidths[1])\n",
    "ax.plot(selected_rates, df_sgl.loc[selected_rates, metric], marker=markers[2], color=colors[2], linestyle=linestyles[2], linewidth=linewidths[2])\n",
    "ax.plot(selected_rates, df_dynamic.loc[selected_rates, metric], marker=markers[3], color=colors[3], linestyle=linestyles[3], linewidth=linewidths[3])\n",
    "\n",
    "# 标注加速比\n",
    "for rate in selected_rates:\n",
    "    base_val = df_0.loc[rate, metric]\n",
    "    dyn_val = df_dynamic.loc[rate, metric]\n",
    "    speedup = base_val / dyn_val if dyn_val != 0 else float('nan')\n",
    "    ax.annotate(f'{speedup:.2f}x', \n",
    "                (rate, dyn_val), \n",
    "                textcoords=\"offset points\", \n",
    "                xytext=(0, 8), \n",
    "                ha='center', \n",
    "                fontsize=10, \n",
    "                color='black')\n",
    "\n",
    "# 美化\n",
    "ax.set_ylim(0, 80000)\n",
    "ax.set_xticks(selected_rates)\n",
    "ax.grid(True)\n",
    "ax.spines['top'].set_visible(False)\n",
    "ax.spines['right'].set_visible(False)\n",
    "\n",
    "# 保存图片\n",
    "plt.tight_layout()\n",
    "plt.savefig(\"e2el_32B_splitwise.pdf\", format=\"pdf\", bbox_inches=\"tight\")\n",
    "# plt.close()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "ename": "KeyError",
     "evalue": "'[16] not in index'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyError\u001b[0m                                  Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[11], line 47\u001b[0m\n\u001b[1;32m     44\u001b[0m ax \u001b[38;5;241m=\u001b[39m axes[idx]  \u001b[38;5;66;03m# 直接使用一维索引\u001b[39;00m\n\u001b[1;32m     46\u001b[0m \u001b[38;5;66;03m# 绘制每个baseline的数据，只显示选定的rate值\u001b[39;00m\n\u001b[0;32m---> 47\u001b[0m ax\u001b[38;5;241m.\u001b[39mplot(selected_rates, \u001b[43mdf_0\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mloc\u001b[49m\u001b[43m[\u001b[49m\u001b[43mselected_rates\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mmetric\u001b[49m\u001b[43m]\u001b[49m, marker\u001b[38;5;241m=\u001b[39mmarkers[\u001b[38;5;241m0\u001b[39m], color\u001b[38;5;241m=\u001b[39mcolors[\u001b[38;5;241m0\u001b[39m], linestyle\u001b[38;5;241m=\u001b[39mlinestyles[\u001b[38;5;241m0\u001b[39m], linewidth\u001b[38;5;241m=\u001b[39mlinewidths[\u001b[38;5;241m0\u001b[39m], label\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mBaseline\u001b[39m\u001b[38;5;124m'\u001b[39m)\n\u001b[1;32m     48\u001b[0m \u001b[38;5;66;03m# ax.plot(selected_rates, df_1.loc[selected_rates, metric], marker=markers[1], color=colors[1], linestyle=linestyles[1], linewidth=linewidths[1], label='Baseline 1')\u001b[39;00m\n\u001b[1;32m     49\u001b[0m \u001b[38;5;66;03m# ax.plot(selected_rates, df_2.loc[selected_rates, metric], marker=markers[2], color=colors[2], linestyle=linestyles[2], linewidth=linewidths[2], label='Baseline 2')\u001b[39;00m\n\u001b[1;32m     50\u001b[0m \u001b[38;5;66;03m# ax.plot(selected_rates, df_3.loc[selected_rates, metric], marker=markers[3], color=colors[3], linestyle=linestyles[3], linewidth=linewidths[3], label='Baseline 3')\u001b[39;00m\n\u001b[1;32m     51\u001b[0m ax\u001b[38;5;241m.\u001b[39mplot(selected_rates, df_dynamic\u001b[38;5;241m.\u001b[39mloc[selected_rates, metric], marker\u001b[38;5;241m=\u001b[39mmarkers[\u001b[38;5;241m4\u001b[39m], color\u001b[38;5;241m=\u001b[39mcolors[\u001b[38;5;241m4\u001b[39m], linestyle\u001b[38;5;241m=\u001b[39mlinestyles[\u001b[38;5;241m4\u001b[39m], linewidth\u001b[38;5;241m=\u001b[39mlinewidths[\u001b[38;5;241m4\u001b[39m], label\u001b[38;5;241m=\u001b[39m\u001b[38;5;124m'\u001b[39m\u001b[38;5;124mDynamic\u001b[39m\u001b[38;5;124m'\u001b[39m)\n",
      "File \u001b[0;32m~/software/miniconda3/envs/gllm/lib/python3.10/site-packages/pandas/core/indexing.py:1184\u001b[0m, in \u001b[0;36m_LocationIndexer.__getitem__\u001b[0;34m(self, key)\u001b[0m\n\u001b[1;32m   1182\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_is_scalar_access(key):\n\u001b[1;32m   1183\u001b[0m         \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mobj\u001b[38;5;241m.\u001b[39m_get_value(\u001b[38;5;241m*\u001b[39mkey, takeable\u001b[38;5;241m=\u001b[39m\u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_takeable)\n\u001b[0;32m-> 1184\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_getitem_tuple\u001b[49m\u001b[43m(\u001b[49m\u001b[43mkey\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1185\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m   1186\u001b[0m     \u001b[38;5;66;03m# we by definition only have the 0th axis\u001b[39;00m\n\u001b[1;32m   1187\u001b[0m     axis \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39maxis \u001b[38;5;129;01mor\u001b[39;00m \u001b[38;5;241m0\u001b[39m\n",
      "File \u001b[0;32m~/software/miniconda3/envs/gllm/lib/python3.10/site-packages/pandas/core/indexing.py:1368\u001b[0m, in \u001b[0;36m_LocIndexer._getitem_tuple\u001b[0;34m(self, tup)\u001b[0m\n\u001b[1;32m   1366\u001b[0m \u001b[38;5;28;01mwith\u001b[39;00m suppress(IndexingError):\n\u001b[1;32m   1367\u001b[0m     tup \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_expand_ellipsis(tup)\n\u001b[0;32m-> 1368\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_getitem_lowerdim\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtup\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1370\u001b[0m \u001b[38;5;66;03m# no multi-index, so validate all of the indexers\u001b[39;00m\n\u001b[1;32m   1371\u001b[0m tup \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_validate_tuple_indexer(tup)\n",
      "File \u001b[0;32m~/software/miniconda3/envs/gllm/lib/python3.10/site-packages/pandas/core/indexing.py:1089\u001b[0m, in \u001b[0;36m_LocationIndexer._getitem_lowerdim\u001b[0;34m(self, tup)\u001b[0m\n\u001b[1;32m   1087\u001b[0m             \u001b[38;5;28;01mreturn\u001b[39;00m section\n\u001b[1;32m   1088\u001b[0m         \u001b[38;5;66;03m# This is an elided recursive call to iloc/loc\u001b[39;00m\n\u001b[0;32m-> 1089\u001b[0m         \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mgetattr\u001b[39;49m\u001b[43m(\u001b[49m\u001b[43msection\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mname\u001b[49m\u001b[43m)\u001b[49m\u001b[43m[\u001b[49m\u001b[43mnew_key\u001b[49m\u001b[43m]\u001b[49m\n\u001b[1;32m   1091\u001b[0m \u001b[38;5;28;01mraise\u001b[39;00m IndexingError(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mnot applicable\u001b[39m\u001b[38;5;124m\"\u001b[39m)\n",
      "File \u001b[0;32m~/software/miniconda3/envs/gllm/lib/python3.10/site-packages/pandas/core/indexing.py:1191\u001b[0m, in \u001b[0;36m_LocationIndexer.__getitem__\u001b[0;34m(self, key)\u001b[0m\n\u001b[1;32m   1189\u001b[0m maybe_callable \u001b[38;5;241m=\u001b[39m com\u001b[38;5;241m.\u001b[39mapply_if_callable(key, \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mobj)\n\u001b[1;32m   1190\u001b[0m maybe_callable \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_check_deprecated_callable_usage(key, maybe_callable)\n\u001b[0;32m-> 1191\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_getitem_axis\u001b[49m\u001b[43m(\u001b[49m\u001b[43mmaybe_callable\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43maxis\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43maxis\u001b[49m\u001b[43m)\u001b[49m\n",
      "File \u001b[0;32m~/software/miniconda3/envs/gllm/lib/python3.10/site-packages/pandas/core/indexing.py:1420\u001b[0m, in \u001b[0;36m_LocIndexer._getitem_axis\u001b[0;34m(self, key, axis)\u001b[0m\n\u001b[1;32m   1417\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mhasattr\u001b[39m(key, \u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mndim\u001b[39m\u001b[38;5;124m\"\u001b[39m) \u001b[38;5;129;01mand\u001b[39;00m key\u001b[38;5;241m.\u001b[39mndim \u001b[38;5;241m>\u001b[39m \u001b[38;5;241m1\u001b[39m:\n\u001b[1;32m   1418\u001b[0m         \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mValueError\u001b[39;00m(\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mCannot index with multidimensional key\u001b[39m\u001b[38;5;124m\"\u001b[39m)\n\u001b[0;32m-> 1420\u001b[0m     \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_getitem_iterable\u001b[49m\u001b[43m(\u001b[49m\u001b[43mkey\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43maxis\u001b[49m\u001b[38;5;241;43m=\u001b[39;49m\u001b[43maxis\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1422\u001b[0m \u001b[38;5;66;03m# nested tuple slicing\u001b[39;00m\n\u001b[1;32m   1423\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m is_nested_tuple(key, labels):\n",
      "File \u001b[0;32m~/software/miniconda3/envs/gllm/lib/python3.10/site-packages/pandas/core/indexing.py:1360\u001b[0m, in \u001b[0;36m_LocIndexer._getitem_iterable\u001b[0;34m(self, key, axis)\u001b[0m\n\u001b[1;32m   1357\u001b[0m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_validate_key(key, axis)\n\u001b[1;32m   1359\u001b[0m \u001b[38;5;66;03m# A collection of keys\u001b[39;00m\n\u001b[0;32m-> 1360\u001b[0m keyarr, indexer \u001b[38;5;241m=\u001b[39m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_get_listlike_indexer\u001b[49m\u001b[43m(\u001b[49m\u001b[43mkey\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43maxis\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1361\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mobj\u001b[38;5;241m.\u001b[39m_reindex_with_indexers(\n\u001b[1;32m   1362\u001b[0m     {axis: [keyarr, indexer]}, copy\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mTrue\u001b[39;00m, allow_dups\u001b[38;5;241m=\u001b[39m\u001b[38;5;28;01mTrue\u001b[39;00m\n\u001b[1;32m   1363\u001b[0m )\n",
      "File \u001b[0;32m~/software/miniconda3/envs/gllm/lib/python3.10/site-packages/pandas/core/indexing.py:1558\u001b[0m, in \u001b[0;36m_LocIndexer._get_listlike_indexer\u001b[0;34m(self, key, axis)\u001b[0m\n\u001b[1;32m   1555\u001b[0m ax \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mobj\u001b[38;5;241m.\u001b[39m_get_axis(axis)\n\u001b[1;32m   1556\u001b[0m axis_name \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mobj\u001b[38;5;241m.\u001b[39m_get_axis_name(axis)\n\u001b[0;32m-> 1558\u001b[0m keyarr, indexer \u001b[38;5;241m=\u001b[39m \u001b[43max\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_get_indexer_strict\u001b[49m\u001b[43m(\u001b[49m\u001b[43mkey\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43maxis_name\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   1560\u001b[0m \u001b[38;5;28;01mreturn\u001b[39;00m keyarr, indexer\n",
      "File \u001b[0;32m~/software/miniconda3/envs/gllm/lib/python3.10/site-packages/pandas/core/indexes/base.py:6200\u001b[0m, in \u001b[0;36mIndex._get_indexer_strict\u001b[0;34m(self, key, axis_name)\u001b[0m\n\u001b[1;32m   6197\u001b[0m \u001b[38;5;28;01melse\u001b[39;00m:\n\u001b[1;32m   6198\u001b[0m     keyarr, indexer, new_indexer \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39m_reindex_non_unique(keyarr)\n\u001b[0;32m-> 6200\u001b[0m \u001b[38;5;28;43mself\u001b[39;49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43m_raise_if_missing\u001b[49m\u001b[43m(\u001b[49m\u001b[43mkeyarr\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mindexer\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43maxis_name\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m   6202\u001b[0m keyarr \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mself\u001b[39m\u001b[38;5;241m.\u001b[39mtake(indexer)\n\u001b[1;32m   6203\u001b[0m \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28misinstance\u001b[39m(key, Index):\n\u001b[1;32m   6204\u001b[0m     \u001b[38;5;66;03m# GH 42790 - Preserve name from an Index\u001b[39;00m\n",
      "File \u001b[0;32m~/software/miniconda3/envs/gllm/lib/python3.10/site-packages/pandas/core/indexes/base.py:6252\u001b[0m, in \u001b[0;36mIndex._raise_if_missing\u001b[0;34m(self, key, indexer, axis_name)\u001b[0m\n\u001b[1;32m   6249\u001b[0m     \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mKeyError\u001b[39;00m(\u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;124mNone of [\u001b[39m\u001b[38;5;132;01m{\u001b[39;00mkey\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m] are in the [\u001b[39m\u001b[38;5;132;01m{\u001b[39;00maxis_name\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m]\u001b[39m\u001b[38;5;124m\"\u001b[39m)\n\u001b[1;32m   6251\u001b[0m not_found \u001b[38;5;241m=\u001b[39m \u001b[38;5;28mlist\u001b[39m(ensure_index(key)[missing_mask\u001b[38;5;241m.\u001b[39mnonzero()[\u001b[38;5;241m0\u001b[39m]]\u001b[38;5;241m.\u001b[39munique())\n\u001b[0;32m-> 6252\u001b[0m \u001b[38;5;28;01mraise\u001b[39;00m \u001b[38;5;167;01mKeyError\u001b[39;00m(\u001b[38;5;124mf\u001b[39m\u001b[38;5;124m\"\u001b[39m\u001b[38;5;132;01m{\u001b[39;00mnot_found\u001b[38;5;132;01m}\u001b[39;00m\u001b[38;5;124m not in index\u001b[39m\u001b[38;5;124m\"\u001b[39m)\n",
      "\u001b[0;31mKeyError\u001b[0m: '[16] not in index'"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 2000x500 with 4 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "import matplotlib.pyplot as plt\n",
    "\n",
    "# 设置图表样式\n",
    "# plt.style.available\n",
    "plt.style.use('seaborn-v0_8')\n",
    "# plt.rcParams['font.sans-serif'] = ['SimHei']  # 用来正常显示中文标签\n",
    "plt.rcParams['axes.unicode_minus'] = False  # 用来正常显示负号\n",
    "\n",
    "# 创建图表\n",
    "fig, axes = plt.subplots(1, 4, figsize=(20, 5))\n",
    "# fig.suptitle('Performance Metrics Comparison Across Different Baselines', \n",
    "#              fontsize=16, y=1.05)  # 增加y值使标题位置更高\n",
    "\n",
    "\n",
    "\n",
    "# 定义要比较的指标\n",
    "metrics = [\n",
    "    ('splitwise_avg_ttft', 'Splitwise TTFT (ms)'),\n",
    "    ('splitwise_avg_tpot', 'Splitwise TPOT (ms)'),\n",
    "    ('splitwise_avg_e2el', 'Splitwise E2EL (ms)'),\n",
    "    ('splitwise_goodput', 'Splitwise Goodput (%)'),\n",
    "    # ('token_throughput', 'Token Throughput (tokens/s)'),\n",
    "]\n",
    "\n",
    "# 设置每个指标对应的y轴范围\n",
    "y_limits = {\n",
    "    'splitwise_avg_ttft': (0, 30000),\n",
    "    'splitwise_avg_tpot': (0, 400),\n",
    "    'splitwise_avg_e2el': (0, 65000),\n",
    "    'splitwise_goodput': (0, 100),\n",
    "}\n",
    "\n",
    "# 只选择特定的rate值\n",
    "# selected_rates = [6,8,10,12] # 32b\n",
    "selected_rates = [4,8,12,16] # 14b\n",
    "\n",
    "colors = ['#1f77b4', '#ff7f0e', '#2ca02c', '#d62728', '#9467bd']  # 蓝、橙、绿、红、紫\n",
    "markers = ['o', 's', '^', 'v', '*']\n",
    "linestyles = ['-', '-', '-', '-', '-']\n",
    "linewidths = [2, 2, 2, 2, 2] \n",
    "\n",
    "# 绘制每个指标\n",
    "for idx, (metric, title) in enumerate(metrics):\n",
    "    ax = axes[idx]  # 直接使用一维索引\n",
    "\n",
    "    # 绘制每个baseline的数据，只显示选定的rate值\n",
    "    ax.plot(selected_rates, df_0.loc[selected_rates, metric], marker=markers[0], color=colors[0], linestyle=linestyles[0], linewidth=linewidths[0], label='Baseline')\n",
    "    # ax.plot(selected_rates, df_1.loc[selected_rates, metric], marker=markers[1], color=colors[1], linestyle=linestyles[1], linewidth=linewidths[1], label='Baseline 1')\n",
    "    # ax.plot(selected_rates, df_2.loc[selected_rates, metric], marker=markers[2], color=colors[2], linestyle=linestyles[2], linewidth=linewidths[2], label='Baseline 2')\n",
    "    # ax.plot(selected_rates, df_3.loc[selected_rates, metric], marker=markers[3], color=colors[3], linestyle=linestyles[3], linewidth=linewidths[3], label='Baseline 3')\n",
    "    ax.plot(selected_rates, df_dynamic.loc[selected_rates, metric], marker=markers[4], color=colors[4], linestyle=linestyles[4], linewidth=linewidths[4], label='Dynamic')\n",
    "    ax.plot(selected_rates, df_vllm.loc[selected_rates, metric], marker=markers[1], color=colors[1], linestyle=linestyles[1], linewidth=linewidths[1], label='vllm')\n",
    "    ax.plot(selected_rates, df_sgl.loc[selected_rates, metric], marker=markers[2], color=colors[2], linestyle=linestyles[2], linewidth=linewidths[2], label='sglang')\n",
    "\n",
    "\n",
    "    # 计算并标注加速比\n",
    "    for rate in selected_rates:\n",
    "        base_val = df_0.loc[rate, metric]\n",
    "        dyn_val = df_dynamic.loc[rate, metric]\n",
    "        # 时间类指标用 baseline0/dynamic，goodput 用 dynamic/baseline0\n",
    "        if 'Goodput' in title or 'Throughput' in title:\n",
    "            speedup = dyn_val / base_val if base_val != 0 else float('nan')\n",
    "        else:\n",
    "            speedup = base_val / dyn_val if dyn_val != 0 else float('nan')\n",
    "        # 在 dynamic 曲线的点上方标注加速比\n",
    "        ax.annotate(f'{speedup:.2f}x', \n",
    "                    (rate, dyn_val), \n",
    "                    textcoords=\"offset points\", \n",
    "                    xytext=(0,8), \n",
    "                    ha='center', \n",
    "                    fontsize=10, \n",
    "                    color='black')\n",
    "\n",
    "    ax.set_title(title)\n",
    "    ax.set_xlabel('Rate')\n",
    "    ax.set_xticks(selected_rates)  # 设置 x 轴刻度只显示 selected_rates 中的值\n",
    "    if metric in y_limits:\n",
    "        ax.set_ylim(y_limits[metric])\n",
    "    ax.grid(True)\n",
    "    ax.legend()\n",
    "\n",
    "# 调整布局\n",
    "plt.tight_layout()\n",
    "plt.subplots_adjust(top=0.95)  # 减小top值，给标题留出更多空间\n",
    "plt.show()\n"
   ]
  }
 ],
 "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
}
