{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "**Please set the PROJECT_PATH variable:**"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "PROJECT_PATH = '<ENTER/THE/PATH/TO/THIS/PROJECT/REPOSITORY/HERE>'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "import tensorboard as tb\n",
    "import matplotlib\n",
    "from matplotlib import pyplot as plt\n",
    "import pandas as pd\n",
    "import os\n",
    "\n",
    "import ast\n",
    "import argparse\n",
    "import os\n",
    "import re\n",
    "from pathlib import Path\n",
    "\n",
    "import math\n",
    "\n",
    "import pandas as pd\n",
    "import numpy as np\n",
    "import tensorflow as tf\n",
    "from tensorboard.backend.event_processing.event_accumulator import EventAccumulator\n",
    "from tensorflow.core.util.event_pb2 import Event\n",
    "\n",
    "%matplotlib inline\n",
    "\n",
    "# # Configure matplotlib to export latex-friendly plots.\n",
    "# matplotlib.use(\"pgf\")\n",
    "# matplotlib.rcParams.update({\n",
    "#     \"pgf.texsystem\": \"pdflatex\",\n",
    "#     'font.family': 'serif',\n",
    "#     'text.usetex': True,\n",
    "#     'pgf.rcfonts': False,\n",
    "# })"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "# MIT License\n",
    "# Copyright (c) 2019 Sebastian Penhouet\n",
    "# GitHub project: https://github.com/Spenhouet/tensorboard-aggregator\n",
    "# ==============================================================================\n",
    "def extract(dpath):\n",
    "    scalar_accumulators = [EventAccumulator(str(dpath)).Reload().scalars]\n",
    "    # Filter non event files\n",
    "    scalar_accumulators = [scalar_accumulator for scalar_accumulator in scalar_accumulators if scalar_accumulator.Keys()]\n",
    "    # Get and validate all scalar keys\n",
    "    all_keys = [tuple(sorted(list(scalar_accumulator.Keys()))) for scalar_accumulator in scalar_accumulators]\n",
    "    assert len(set(all_keys)) == 1, \"All runs need to have the same scalar keys. There are mismatches in {}\".format(all_keys)\n",
    "    keys = all_keys[0]\n",
    "    all_scalar_events_per_key = [[scalar_accumulator.Items(key) for scalar_accumulator in scalar_accumulators] for key in keys]\n",
    "    # Get and validate all steps per key\n",
    "    all_steps_per_key = [[tuple(scalar_event.step for scalar_event in scalar_events) for scalar_events in all_scalar_events]\n",
    "                         for all_scalar_events in all_scalar_events_per_key]\n",
    "    for i, all_steps in enumerate(all_steps_per_key):\n",
    "        assert len(set(all_steps)) == 1, \"For scalar {} the step numbering or count doesn't match. Step count for all runs: {}\".format(\n",
    "            keys[i], [len(steps) for steps in all_steps])\n",
    "    steps_per_key = [all_steps[0] for all_steps in all_steps_per_key]\n",
    "    # Get and average wall times per step per key\n",
    "    wall_times_per_key = [np.mean([tuple(scalar_event.wall_time for scalar_event in scalar_events) for scalar_events in all_scalar_events], axis=0)\n",
    "                          for all_scalar_events in all_scalar_events_per_key]\n",
    "    # Get values per step per key\n",
    "    values_per_key = [[[scalar_event.value for scalar_event in scalar_events] for scalar_events in all_scalar_events]\n",
    "                      for all_scalar_events in all_scalar_events_per_key]\n",
    "    all_per_key = dict(zip(keys, zip(steps_per_key, wall_times_per_key, values_per_key)))\n",
    "    return all_per_key\n",
    "\n",
    "def max_dir(path):\n",
    "    return max(os.listdir(path))\n",
    "\n",
    "def get_metric(experiment, metric, pad_nan=0, exps_dir = 'experiments'):\n",
    "    arrays = []\n",
    "    exp_dir = os.path.join(PROJECT_PATH, exps_dir, experiment)\n",
    "    for run_dir in os.listdir(exp_dir):\n",
    "        event_dir = os.path.join(exp_dir, run_dir, 'server')\n",
    "        event_file = os.path.join(event_dir, os.listdir(event_dir)[0])\n",
    "        rounds = extract(event_file)[f'{metric}'][0]\n",
    "        data = extract(event_file)[f'{metric}'][2][0]\n",
    "        expanded_data = []\n",
    "        for r, dat in zip(rounds, data):\n",
    "            while(len(expanded_data) <= r):\n",
    "                expanded_data.append(dat)\n",
    "        arrays.append(expanded_data)\n",
    "    return arrays\n",
    "\n",
    "def get_max_accuracy(exp, rounds_between_evaluations):\n",
    "    accuracies = get_metric(exp, 'testing/accuracy')\n",
    "    accuracies = np.array(accuracies)\n",
    "    highest_acc_idxs = np.argmax(accuracies, axis=1)\n",
    "    accs = accuracies[:, highest_acc_idxs]\n",
    "    mean_acc = accs.mean()\n",
    "    std_acc = accs.std(ddof=1)\n",
    "    return mean_acc, std_acc"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Rounds between evaluations"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "rounds_between_evaluations = {\n",
    "    \"synthetic\": 1,\n",
    "    \"femnist\": 5,\n",
    "    \"sent140\": 5,\n",
    "    \"shakespeare\": 2,\n",
    "    \"celeba\": 5\n",
    "}\n",
    "q_levels = {\n",
    "    \"synthetic\": 16,\n",
    "    \"femnist\": 4,\n",
    "    \"sent140\": 128,\n",
    "    \"shakespeare\": 512,\n",
    "    \"celeba\": 16,\n",
    "}\n",
    "\n",
    "model_params = {\n",
    "    \"synthetic\": 610,\n",
    "    \"femnist\": 6_603_710,\n",
    "    \"sent140\": 1_098_242,\n",
    "    \"shakespeare\": 133_520,\n",
    "    \"celeba\": 629_346,   \n",
    "}\n",
    "n_rounds = {\n",
    "    'synthetic': 501,\n",
    "    'femnist': 501,\n",
    "    'sent140': 1001,\n",
    "    'shakespeare': 51,\n",
    "    'celeba': 501}"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Quantization level grid search [Experiments]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "**Experiment IDs:** 5000 - 5050."
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "For each dataset, get the least quantization level that exceeds the accuracy of the dataset's uncompressed experiment.\n",
    "We use this to set the quantization level hyperparameter for the main experiments."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [],
   "source": [
    "exps = {\n",
    "    \"synthetic\": {\n",
    "        \"no_compression\": \"5000\",\n",
    "        \"1\": \"5001\",\n",
    "        \"2\": \"5002\",\n",
    "        \"4\": \"5003\",\n",
    "        \"8\": \"5004\",\n",
    "        \"16\": \"5005\",\n",
    "        \"32\": \"5006\",\n",
    "        \"64\": \"5007\",\n",
    "        \"128\": \"5008\",\n",
    "        \"256\": \"5009\",\n",
    "    }, \n",
    "    \"femnist\": {\n",
    "        \"no_compression\": \"5010\",\n",
    "        \"1\": \"5011\",\n",
    "        \"2\": \"5012\",\n",
    "        \"4\": \"5013\",\n",
    "        \"8\": \"5014\",\n",
    "        \"16\": \"5015\",\n",
    "        \"32\": \"5016\",\n",
    "        \"64\": \"5017\",\n",
    "        \"128\": \"5018\",\n",
    "        \"256\": \"5019\",\n",
    "    },\n",
    "    \"sent140\": {\n",
    "        \"no_compression\": \"5020\",\n",
    "        \"1\": \"5021\",\n",
    "        \"2\": \"5022\",\n",
    "        \"4\": \"5023\",\n",
    "        \"8\": \"5024\",\n",
    "        \"16\": \"5025\",\n",
    "        \"32\": \"5026\",\n",
    "        \"64\": \"5027\",\n",
    "        \"128\": \"5028\",\n",
    "        \"256\": \"5029\",\n",
    "    },\n",
    "    \"shakespeare\": {\n",
    "        \"no_compression\": \"5030\",\n",
    "        \"1\": \"5031\",\n",
    "        \"2\": \"5032\",\n",
    "        \"4\": \"5033\",\n",
    "        \"8\": \"5034\",\n",
    "        \"16\": \"5035\",\n",
    "        \"32\": \"5036\",\n",
    "        \"64\": \"5037\",\n",
    "        \"128\": \"5038\",\n",
    "        \"256\": \"5039\",\n",
    "        \"512\": \"5040\",\n",
    "    },\n",
    "    \"celeba\": {\n",
    "        \"no_compression\": \"5041\",\n",
    "        \"1\": \"5042\",\n",
    "        \"2\": \"5043\",\n",
    "        \"4\": \"5044\",\n",
    "        \"8\": \"5045\",\n",
    "        \"16\": \"5046\",\n",
    "        \"32\": \"5047\",\n",
    "        \"64\": \"5048\",\n",
    "        \"128\": \"5049\",\n",
    "        \"256\": \"5050\",\n",
    "    } \n",
    "}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "synthetic: q = 16\n",
      "femnist: q = 4\n",
      "sent140: q = 128\n",
      "shakespeare: q = 512\n",
      "celeba: q = 16\n"
     ]
    }
   ],
   "source": [
    "for dataset in exps.keys():\n",
    "    uncompressed_acc,_ = get_max_accuracy(exps[dataset][\"no_compression\"], rounds_between_evaluations[dataset])\n",
    "    winning_q = 1_000_000\n",
    "    for q in exps[dataset].keys() - {\"no_compression\"}:\n",
    "        acc, _ = get_max_accuracy(exps[dataset][q], rounds_between_evaluations[dataset])\n",
    "        if acc >= uncompressed_acc and int(q) < winning_q:\n",
    "            winning_q = int(q)\n",
    "    print(f\"{dataset}: q = {winning_q}\")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Main experiments"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [],
   "source": [
    "exps = {\n",
    "    \"synthetic\": {\n",
    "        \"uncompressed\": ['8000', '8100', '8200'],\n",
    "        \"qsgd\": ['8001', '8101', '8201'],\n",
    "        \"fixedpoint\": ['8001', '8101', '8201'],\n",
    "        \"monoquant\": ['8002', '8102', '8202'],\n",
    "        \"samplequant\": ['8003', '8103', '8203'],\n",
    "        \"dadaquant\": ['8004', '8104', '8204'],\n",
    "        \"uveqfed\": ['7002', '7102', '7202'],\n",
    "        \"gzip\": ['7003', '7103', '7203'],\n",
    "        \"fp8\": ['7004', '7104', '7204'],\n",
    "    },\n",
    "    \"femnist\": {\n",
    "        \"uncompressed\": ['8010', '8110', '8210'],\n",
    "        \"qsgd\": ['8011', '8111', '8211'],\n",
    "        \"fixedpoint\": ['8011', '8111', '8211'],\n",
    "        \"monoquant\": ['8012', '8112', '8212'],\n",
    "        \"samplequant\": ['8013', '8113', '8213'],\n",
    "        \"dadaquant\": ['8014', '8114', '8214'],\n",
    "        \"uveqfed\": ['7012', '7112', '7212'],\n",
    "        \"gzip\": ['7013', '7113', '7213'],\n",
    "        \"fp8\": ['7014', '7114', '7214'],\n",
    "    },\n",
    "    \"sent140\": {\n",
    "        \"uncompressed\": ['8020', '8120', '8220'],\n",
    "        \"qsgd\": ['8021', '8121', '8221'],\n",
    "        \"fixedpoint\": ['8021', '8121', '8221'],\n",
    "        \"monoquant\": ['8022', '8122', '8222'],\n",
    "        \"samplequant\": ['8023', '8123', '8223'],\n",
    "        \"dadaquant\": ['8024', '8124', '8224'],\n",
    "        \"uveqfed\": ['7022', '7122', '7222'],\n",
    "        \"gzip\": ['7023', '7123', '7223'],\n",
    "        \"fp8\": ['7024', '7124', '7224']\n",
    "    },\n",
    "    \"shakespeare\": {\n",
    "        \"uncompressed\": ['8030', '8130', '8230'],\n",
    "        \"qsgd\": ['8031', '8131', '8231'],\n",
    "        \"fixedpoint\": ['8031', '8131', '8231'],\n",
    "        \"monoquant\": ['8032', '8132', '8232'],\n",
    "        \"samplequant\": ['8033', '8133', '8233'],\n",
    "        \"dadaquant\": ['8034', '8134', '8234'],\n",
    "        \"uveqfed\": ['7032', '7132', '7232'],\n",
    "        \"gzip\": ['7033', '7133', '7233'],\n",
    "        \"fp8\": ['7034', '7134', '7234'],\n",
    "    },\n",
    "    \"celeba\": {\n",
    "        \"uncompressed\": ['8040', '8140', '8240'],\n",
    "        \"qsgd\": ['8041', '8141', '8241'],\n",
    "        \"fixedpoint\": ['8041', '8141', '8241'],\n",
    "        \"monoquant\": ['8042', '8142', '8242'],\n",
    "        \"samplequant\": ['8043', '8143', '8243'],\n",
    "        \"dadaquant\": ['8044', '8144', '8244'],\n",
    "        \"uveqfed\": ['7042', '7142', '7242'],\n",
    "        \"gzip\": ['7043', '7143', '7243'],\n",
    "        \"fp8\": ['7044', '7144', '7244'],\n",
    "    }\n",
    "}\n",
    "\n",
    "def get_max_acc_comm(exp, n_rounds):\n",
    "    accs = get_metric(exp, 'testing/accuracy')\n",
    "    accs = np.array(accs).squeeze()[0:n_rounds]\n",
    "    comm = get_metric(exp, 'cumsum data sent (acc. over all clients) in MB')\n",
    "    comm = np.array(comm).squeeze()[0:n_rounds]\n",
    "    return accs.max(), comm.max()\n",
    "\n",
    "def get_avg_var(exps, n_rounds, metric):\n",
    "#     exps = [exp for exp in exps if '71' not in exp]  # TODO: REMOVE\n",
    "    nums = np.array([np.array(get_metric(exp, metric)).squeeze()[0:n_rounds].max() for exp in exps])\n",
    "#     print(f\"exps {exps}, nums {nums}\")\n",
    "    return nums.mean(), nums.std(ddof=1)\n",
    "    \n",
    "ACC_KEY = 'testing/accuracy'\n",
    "COMM_KEY = 'cumsum data sent (acc. over all clients) in MB'\n",
    "\n",
    "# Verify exps\n",
    "for v in exps.values():\n",
    "    for v2 in v.values():\n",
    "        for idx, n in enumerate(v2):\n",
    "            assert(int(n) == int(v2[0]) + 100*idx)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Pareto curves [Experiments]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "**Experiment IDs:**\n",
    "\n",
    "Synthetic:  8000 8100 8200 8001 8101 8201 8001 8101 8201 8002 8102 8202 8003 8103 8203 8004 8104 8204 7002 7102 7202 7003 7103 7203 7004 7104 7204 \n",
    "\n",
    "FEMNIST:  8010 8110 8210 8011 8111 8211 8011 8111 8211 8012 8112 8212 8013 8113 8213 8014 8114 8214 7012 7112 7212 7013 7113 7213 7014 7114 7214 \n",
    "\n",
    "Sent140:  8020 8120 8220 8021 8121 8221 8021 8121 8221 8022 8122 8222 8023 8123 8223 8024 8124 8224 7022 7122 7222 7023 7123 7223 7024 7124 7224 \n",
    "\n",
    "Shakespeare:  8030 8130 8230 8031 8131 8231 8031 8131 8231 8032 8132 8232 8033 8133 8233 8034 8134 8234 7032 7132 7232 7033 7133 7233 7034 7134 7234 \n",
    "\n",
    "CelebA:  8040 8140 8240 8041 8141 8241 8041 8141 8241 8042 8142 8242 8043 8143 8243 8044 8144 8244 7042 7142 7242 7043 7143 7243 7044 7144 7244 "
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Appendix"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "synthetic qsgd 8001\n",
      "synthetic qsgd 8101\n",
      "synthetic qsgd 8201\n",
      "synthetic dadaquant 8004\n",
      "synthetic dadaquant 8104\n",
      "synthetic dadaquant 8204\n",
      "femnist qsgd 8011\n",
      "femnist qsgd 8111\n",
      "femnist qsgd 8211\n",
      "femnist dadaquant 8014\n",
      "femnist dadaquant 8114\n",
      "femnist dadaquant 8214\n",
      "sent140 qsgd 8021\n",
      "sent140 qsgd 8121\n",
      "sent140 qsgd 8221\n",
      "sent140 dadaquant 8024\n",
      "sent140 dadaquant 8124\n",
      "sent140 dadaquant 8224\n",
      "shakespeare qsgd 8031\n",
      "shakespeare qsgd 8131\n",
      "shakespeare qsgd 8231\n",
      "shakespeare dadaquant 8034\n",
      "shakespeare dadaquant 8134\n",
      "shakespeare dadaquant 8234\n",
      "celeba qsgd 8041\n",
      "celeba qsgd 8141\n",
      "celeba qsgd 8241\n",
      "celeba dadaquant 8044\n",
      "celeba dadaquant 8144\n",
      "celeba dadaquant 8244\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7fd741dbedd0>"
      ]
     },
     "execution_count": 19,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x288 with 5 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "titles = {\n",
    "    'synthetic': 'Synthetic',\n",
    "    'femnist': 'FEMNIST',\n",
    "    'sent140': 'Sent140',\n",
    "    'shakespeare': 'Shakespeare',\n",
    "    'celeba': 'Celeba',\n",
    "}\n",
    "labels = {\n",
    "    'qsgd': \"Federated QSGD\",\n",
    "    'dadaquant': \"DAdaQuant\",\n",
    "}\n",
    "\n",
    "fig, axs = plt.subplots(nrows=2, ncols=3, figsize=(8, 4))\n",
    "fig.delaxes(axs[1][2])\n",
    "for idx, dataset in enumerate(titles.keys()):\n",
    "    row = idx // 3\n",
    "    col = idx % 3\n",
    "    ax = axs[row,col]\n",
    "#     ax.set_xscale('log')\n",
    "    for exp_type in ['qsgd', 'dadaquant']:\n",
    "        cummax_accss = np.zeros(n_rounds[dataset])\n",
    "        for exp in exps[dataset][exp_type]:\n",
    "            print(dataset, exp_type, exp)\n",
    "            accs = get_metric(exp, 'testing/accuracy')\n",
    "            accs = np.array(accs).squeeze()[0:n_rounds[dataset]]\n",
    "            comm = get_metric(exp, 'cumsum data sent (acc. over all clients) in MB')\n",
    "            comm = np.array(comm).squeeze()[0:n_rounds[dataset]]\n",
    "            cummax_accs = np.zeros(np.shape(accs))\n",
    "            cummax_accs[0] = accs[0]\n",
    "            for idx in range(1, len(accs)):\n",
    "                cummax_accs[idx] = max(cummax_accs[idx-1], accs[idx])\n",
    "            cummax_accss += cummax_accs\n",
    "        cummax_accss /= len(exps[dataset][exp_type])\n",
    "        ax.plot(comm, cummax_accss*100, label=f\"{labels[exp_type]}\")\n",
    "    ax.set_xlabel('Communication in Megabytes')\n",
    "    ax.set_ylabel('Accuracy in %.')\n",
    "    ax.set_title(titles[dataset])\n",
    "    ax.grid()\n",
    "\n",
    "fig.subplots_adjust(hspace=0.6, wspace=0.3)  # create some space below the plots by increasing the bottom-value\n",
    "axs.flatten()[4].legend(loc='center right', bbox_to_anchor=(2,0.8), ncol=1) \n",
    "# fig.tight_layout()\n",
    "\n",
    "# fig.savefig(os.path.join(PAPER_PATH, 'pareto_appendix.pgf'))# fig, axs = plt.subplots(nrows=3, ncols=2, figsize=(8, 8))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## AdaQuantFL vs DAdaQuant [Experiments]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "**Experiment IDs:** 6000 6001 6010 6011 6020 6021 6030 6031"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x7fd73fbeb310>"
      ]
     },
     "execution_count": 21,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 216x144 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "exps = {\n",
    "    \"adaquantfl\": ['6000', '6010', '6020', '6030'],\n",
    "    \"dadaquant\": ['6001', '6011', '6021', '6031'],\n",
    "}\n",
    "\n",
    "ys = {'adaquantfl': [],\n",
    "      'dadaquant': []}\n",
    "xs = [10, 100, 200, 400]\n",
    "for name in exps.keys():\n",
    "    for exp in exps[name]:\n",
    "        comm = get_metric(exp, 'cumsum data sent (acc. over all clients) in MB', exps_dir='experiments')\n",
    "        ys[name].append(max(comm[0]))\n",
    "        \n",
    "fig,ax = plt.subplots(figsize=(3, 2))\n",
    "ax.set_xlabel('Number of clients in dataset')\n",
    "ax.set_ylabel('Communication in Megabytes')\n",
    "ax.plot(xs, ys['adaquantfl'], label=\"AdaQuantFL\")\n",
    "ax.plot(xs, ys['dadaquant'], label=\"DAdaQuant\")\n",
    "ax.grid()\n",
    "ax.set_xticks([10, 100, 200, 400])\n",
    "# ax.set_xlim([0, 150])\n",
    "# ax.set_ylim([0, 4])\n",
    "\n",
    "fig.tight_layout()\n",
    "ax.legend()#loc='center right', bbox_to_anchor=(2,0.8), ncol=1) \n",
    "\n",
    "# fig.savefig(os.path.join(PAPER_PATH, 'adaquantfl.pgf'), bbox_inches = \"tight\")# fig, axs = plt.subplots(nrows=3, ncols=2, figsize=(8, 8))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Main table [Experiments]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "**Experiment IDs:**\n",
    "\n",
    "Synthetic:  8000 8100 8200 8001 8101 8201 8001 8101 8201 8002 8102 8202 8003 8103 8203 8004 8104 8204 7002 7102 7202 7003 7103 7203 7004 7104 7204 \n",
    "\n",
    "FEMNIST:  8010 8110 8210 8011 8111 8211 8011 8111 8211 8012 8112 8212 8013 8113 8213 8014 8114 8214 7012 7112 7212 7013 7113 7213 7014 7114 7214 \n",
    "\n",
    "Sent140:  8020 8120 8220 8021 8121 8221 8021 8121 8221 8022 8122 8222 8023 8123 8223 8024 8124 8224 7022 7122 7222 7023 7123 7223 7024 7124 7224 \n",
    "\n",
    "Shakespeare:  8030 8130 8230 8031 8131 8231 8031 8131 8231 8032 8132 8232 8033 8133 8233 8034 8134 8234 7032 7132 7232 7033 7133 7233 7034 7134 7234 \n",
    "\n",
    "CelebA:  8040 8140 8240 8041 8141 8241 8041 8141 8241 8042 8142 8242 8043 8143 8243 8044 8144 8244 7042 7142 7242 7043 7143 7243 7044 7144 7244 "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 32,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset synthetic | Compressor uncompressed | max_acc_avg 0.7832384904225668 | max_acc_var 0.002880402885636174 | max_comm_avg 12.199999809265137 | max_comm_var 0.0\n",
      "Dataset synthetic | Compressor qsgd | max_acc_avg 0.7822038332621256 | max_acc_var 0.00051733851461414 | max_comm_avg 0.7135173281033834 | max_comm_var 0.003198231336070269\n",
      "Dataset synthetic | Compressor fixedpoint | max_acc_avg 0.7822038332621256 | max_acc_var 0.00051733851461414 | max_comm_avg 1.90625 | max_comm_var 0.003198231336070269\n",
      "Dataset synthetic | Compressor monoquant | max_acc_avg 0.7820313970247904 | max_acc_var 0.005207669383035579 | max_comm_avg 0.33000000317891437 | max_comm_var 0.025412952555965238\n",
      "Dataset synthetic | Compressor samplequant | max_acc_avg 0.783583382765452 | max_acc_var 0.0029416658943229603 | max_comm_avg 0.47305333614349365 | max_comm_var 0.008147684626577254\n",
      "Dataset synthetic | Compressor dadaquant | max_acc_avg 0.7813415924708048 | max_acc_var 0.0038828602774860776 | max_comm_avg 0.2535466601451238 | max_comm_var 0.024520375083890524\n",
      "Dataset synthetic | Compressor uveqfed | max_acc_avg 0.7784100770950317 | max_acc_var 0.0016629704779619146 | max_comm_avg 21.147427876790363 | max_comm_var 0.02560483949241102\n",
      "Dataset synthetic | Compressor gzip | max_acc_avg 0.7822038332621256 | max_acc_var 0.00051733851461414 | max_comm_avg 0.8655063311258951 | max_comm_var 0.0032189363563414272\n",
      "Dataset synthetic | Compressor fp8 | max_acc_avg 0.7837558190027872 | max_acc_var 0.004040481911137358 | max_comm_avg 3.049999952316284 | max_comm_var 0.0\n",
      "Dataset femnist | Compressor uncompressed | max_acc_avg 0.7765761216481527 | max_acc_var 0.004233643215710323 | max_comm_avg 132074.203125 | max_comm_var 0.0\n",
      "Dataset femnist | Compressor qsgd | max_acc_avg 0.7837047576904297 | max_acc_var 0.005313877884112052 | max_comm_avg 47.016473134358726 | max_comm_var 1.9451039045211542\n",
      "Dataset femnist | Compressor fixedpoint | max_acc_avg 0.7837047576904297 | max_acc_var 0.005313877884112052 | max_comm_avg 12381.95625 | max_comm_var 1.9451039045211542\n",
      "Dataset femnist | Compressor monoquant | max_acc_avg 0.7848212718963623 | max_acc_var 0.0019002744021435255 | max_comm_avg 29.23263422648112 | max_comm_var 1.761196855086676\n",
      "Dataset femnist | Compressor samplequant | max_acc_avg 0.7834731539090475 | max_acc_var 0.004350310071238679 | max_comm_avg 43.778350830078125 | max_comm_var 1.3807762304826323\n",
      "Dataset femnist | Compressor dadaquant | max_acc_avg 0.783688485622406 | max_acc_var 0.0013504247850201084 | max_comm_avg 27.67728042602539 | max_comm_var 0.400001735994063\n",
      "Dataset femnist | Compressor uveqfed | max_acc_avg 0.7482556502024332 | max_acc_var 0.004676571526102827 | max_comm_avg 11315.933919270834 | max_comm_var 407.07010155800964\n",
      "Dataset femnist | Compressor gzip | max_acc_avg 0.7826046148935953 | max_acc_var 0.001896113157350409 | max_comm_avg 84.82835133870442 | max_comm_var 1.0790017433607608\n",
      "Dataset femnist | Compressor fp8 | max_acc_avg 0.7756125926971436 | max_acc_var 0.0035063532604505494 | max_comm_avg 33018.55078125 | max_comm_var 0.0\n",
      "Dataset sent140 | Compressor uncompressed | max_acc_avg 0.6972458759943644 | max_acc_var 0.004796520147458373 | max_comm_avg 43929.6796875 | max_comm_var 0.0\n",
      "Dataset sent140 | Compressor qsgd | max_acc_avg 0.6968023180961609 | max_acc_var 0.005397805775912768 | max_comm_avg 486.9434814453125 | max_comm_var 9.00774311170668\n",
      "Dataset sent140 | Compressor fixedpoint | max_acc_avg 0.6968023180961609 | max_acc_var 0.005397805775912768 | max_comm_avg 10982.42 | max_comm_var 9.00774311170668\n",
      "Dataset sent140 | Compressor monoquant | max_acc_avg 0.6962602337201437 | max_acc_var 0.005773655535453024 | max_comm_avg 470.7477213541667 | max_comm_var 7.111025518652709\n",
      "Dataset sent140 | Compressor samplequant | max_acc_avg 0.6980026960372925 | max_acc_var 0.005600003071622693 | max_comm_avg 420.0984191894531 | max_comm_var 12.002110567255231\n",
      "Dataset sent140 | Compressor dadaquant | max_acc_avg 0.6962602337201437 | max_acc_var 0.004336781253206353 | max_comm_avg 407.9437967936198 | max_comm_var 10.99078180247003\n",
      "Dataset sent140 | Compressor uveqfed | max_acc_avg 0.6975345214207967 | max_acc_var 0.002032456341420588 | max_comm_avg 3007.78173828125 | max_comm_var 36.42688532897191\n",
      "Dataset sent140 | Compressor gzip | max_acc_avg 0.6969184875488281 | max_acc_var 0.00554013516877774 | max_comm_avg 617.7681070963541 | max_comm_var 9.59362626895527\n",
      "Dataset sent140 | Compressor fp8 | max_acc_avg 0.6955069104830424 | max_acc_var 0.0046281044663948724 | max_comm_avg 10982.419921875 | max_comm_var 0.0\n",
      "Dataset shakespeare | Compressor uncompressed | max_acc_avg 0.49897300203641254 | max_acc_var 0.0030588796986948112 | max_comm_avg 267.0400085449219 | max_comm_var 0.0\n",
      "Dataset shakespeare | Compressor qsgd | max_acc_avg 0.4941764573256175 | max_acc_var 0.005819811561946581 | max_comm_avg 28.05606969197591 | max_comm_var 0.13259322394366085\n",
      "Dataset shakespeare | Compressor fixedpoint | max_acc_avg 0.4941764573256175 | max_acc_var 0.005819811561946581 | max_comm_avg 83.45 | max_comm_var 0.13259322394366085\n",
      "Dataset shakespeare | Compressor monoquant | max_acc_avg 0.49410826961199444 | max_acc_var 0.0051651561098308915 | max_comm_avg 21.690466562906902 | max_comm_var 1.2534455195772978\n",
      "Dataset shakespeare | Compressor samplequant | max_acc_avg 0.4952702621618907 | max_acc_var 0.005335155805887219 | max_comm_avg 16.838189442952473 | max_comm_var 0.45900432591473644\n",
      "Dataset shakespeare | Compressor dadaquant | max_acc_avg 0.4933361013730367 | max_acc_var 0.004775142588441442 | max_comm_avg 12.689458847045898 | max_comm_var 1.040901862339181\n",
      "Dataset shakespeare | Compressor uveqfed | max_acc_avg 0.49873613317807514 | max_acc_var 0.0038990664563907603 | max_comm_avg 33.90665181477865 | max_comm_var 0.0869620229977027\n",
      "Dataset shakespeare | Compressor gzip | max_acc_avg 0.4941764573256175 | max_acc_var 0.005819811561946581 | max_comm_avg 28.854103088378906 | max_comm_var 0.07594817477103692\n",
      "Dataset shakespeare | Compressor fp8 | max_acc_avg 0.49670933683713275 | max_acc_var 0.004257997064684985 | max_comm_avg 66.76000213623047 | max_comm_var 0.0\n",
      "Dataset celeba | Compressor uncompressed | max_acc_avg 0.9037601947784424 | max_acc_var 0.0003622378464699538 | max_comm_avg 12586.919921875 | max_comm_var 0.0\n",
      "Dataset celeba | Compressor qsgd | max_acc_avg 0.9024267792701721 | max_acc_var 0.0012628569014631937 | max_comm_avg 19.422468821207683 | max_comm_var 0.37589826342505556\n",
      "Dataset celeba | Compressor fixedpoint | max_acc_avg 0.9024267792701721 | max_acc_var 0.0012628569014631937 | max_comm_avg 1966.70625 | max_comm_var 0.37589826342505556\n",
      "Dataset celeba | Compressor monoquant | max_acc_avg 0.9026071826616923 | max_acc_var 0.0016511490860296983 | max_comm_avg 17.5874449412028 | max_comm_var 0.6258765170715536\n",
      "Dataset celeba | Compressor samplequant | max_acc_avg 0.9032425284385681 | max_acc_var 0.0003668055843885077 | max_comm_avg 17.968514760335285 | max_comm_var 0.22089125061112488\n",
      "Dataset celeba | Compressor dadaquant | max_acc_avg 0.9022777676582336 | max_acc_var 0.0011968229434799478 | max_comm_avg 16.237250645955402 | max_comm_var 0.3441512103061088\n",
      "Dataset celeba | Compressor uveqfed | max_acc_avg 0.8995168407758077 | max_acc_var 0.0032239557091288964 | max_comm_avg 407.6678059895833 | max_comm_var 9.362866421742552\n",
      "Dataset celeba | Compressor gzip | max_acc_avg 0.9030150572458903 | max_acc_var 0.0018106687220540236 | max_comm_avg 25.504428227742512 | max_comm_var 0.3512442547501385\n",
      "Dataset celeba | Compressor fp8 | max_acc_avg 0.9039013584454855 | max_acc_var 0.0007133531606465389 | max_comm_avg 3146.72998046875 | max_comm_var 0.0\n",
      "\n",
      "% Please add the following required packages to your document preamble:\n",
      "% \\usepackage[table,xcdraw]{xcolor}\n",
      "% If you use beamer only pass \"xcolor=table\" option, i.e. \\documentclass[xcolor=table]{beamer}\n",
      "\\begin{table}[]\n",
      "\\begin{tabular}{lllllll}\n",
      "\\multicolumn{1}{l|}{} & \\multicolumn{2}{c|}{\\textbf{Synthetic}} & \\multicolumn{2}{c|}{\\textbf{FEMNIST}} & \\multicolumn{2}{c}{\\textbf{Sent140}} \\\\ \\hline\n",
      "\\rowcolor[HTML]{EFEFEF} \n",
      "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{Uncompressed}} & {$78.3\\pm 0.3$} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{$12.2$\\,MB}} & {$77.7\\pm 0.4$} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{$132.1$\\,GB}} & {$69.7\\pm 0.5$} & {$43.9$\\,GB} \\\\\n",
      "\\multicolumn{1}{l|}{\\textbf{QSGD}} & {$-0.1\\pm 0.1$} & \\multicolumn{1}{l|}{{$17\\times$}} & {$+0.7\\pm 0.5$} & \\multicolumn{1}{l|}{{$2809\\times$}} & {$-0.0\\pm 0.5$} & {$90\\times$} \\\\\n",
      "\\rowcolor[HTML]{EFEFEF} \n",
      "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{FP8}} & {$+0.1\\pm 0.4$} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{$4.0\\times$ ($0.23\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$-0.1\\pm 0.4$} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{$4.0\\times$ ($0.00\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$-0.2\\pm 0.5$} & {$4.0\\times$ ($0.04\\!\\times\\!\\!\\!\\!\\!\\times$)} \\\\\n",
      "\\multicolumn{1}{l|}{\\textbf{FedPAQ (FxPQ)}} & {$-0.1\\pm 0.1$} & \\multicolumn{1}{l|}{{$6.4\\times$ ($0.37\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$+0.7\\pm 0.5$} & \\multicolumn{1}{l|}{{$11\\times$ ($0.00\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$-0.0\\pm 0.5$} & {$4.0\\times$ ($0.04\\!\\times\\!\\!\\!\\!\\!\\times$)} \\\\\n",
      "\\rowcolor[HTML]{EFEFEF} \n",
      "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{\\color[HTML]{333333} \\textbf{GZip}}} & {\\color[HTML]{333333} {$-0.1\\pm 0.1$}} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{\\color[HTML]{333333} {$14\\times$ ($0.82\\!\\times\\!\\!\\!\\!\\!\\times$)}}} & {\\color[HTML]{333333} {$+0.6\\pm 0.2$}} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{\\color[HTML]{333333} {$1557\\times$ ($0.55\\!\\times\\!\\!\\!\\!\\!\\times$)}}} & {\\color[HTML]{333333} {$-0.0\\pm 0.6$}} & {\\color[HTML]{333333} {$71\\times$ ($0.79\\!\\times\\!\\!\\!\\!\\!\\times$)}} \\\\\n",
      "\\multicolumn{1}{l|}{\\textbf{UVeQFed}} & {$-0.5\\pm 0.2$} & \\multicolumn{1}{l|}{{$0.6\\times$ ($0.03\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$-2.8\\pm 0.5$} & \\multicolumn{1}{l|}{{$12\\times$ ($0.00\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$+0.0\\pm 0.2$} & {$15\\times$ ($0.16\\!\\times\\!\\!\\!\\!\\!\\times$)} \\\\ \\hline\n",
      "\\rowcolor[HTML]{EFEFEF} \n",
      "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{DAdaQuant}} & {$-0.2\\pm 0.4$} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{$48\\times$ ($2.81\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$+0.7\\pm 0.1$} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{$4772\\times$ ($1.70\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$-0.1\\pm 0.4$} & {$108\\times$ ($1.19\\!\\times\\!\\!\\!\\!\\!\\times$)} \\\\\n",
      "\\multicolumn{1}{l|}{\\textbf{DAdaQuant$_{\\text{time}}$}} & {$-0.1\\pm 0.5$} & \\multicolumn{1}{l|}{{$37\\times$ ($2.16\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$+0.8\\pm 0.2$} & \\multicolumn{1}{l|}{{$4518\\times$ ($1.61\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$-0.1\\pm 0.6$} & {$93\\times$ ($1.03\\!\\times\\!\\!\\!\\!\\!\\times$)} \\\\\n",
      "\\rowcolor[HTML]{EFEFEF} \n",
      "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{DAdaQuant$_{\\text{clients}}$}} & {$+0.0\\pm 0.3$} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{$26\\times$ ($1.51\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$+0.7\\pm 0.4$} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{$3017\\times$ ($1.07\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$+0.1\\pm 0.6$} & {$105\\times$ ($1.16\\!\\times\\!\\!\\!\\!\\!\\times$)} \\\\\n",
      " &  &  &  &  &  &  \\\\\n",
      "\\multicolumn{1}{l|}{} & \\multicolumn{2}{c|}{\\textbf{Shakespeare}} & \\multicolumn{2}{c}{\\textbf{Celeba}} &  &  \\\\ \\cline{1-5}\n",
      "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{Uncompressed}} & \\cellcolor[HTML]{EFEFEF}{$49.9\\pm 0.3$} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{$267.0$\\,MB}} & \\cellcolor[HTML]{EFEFEF}{$90.4\\pm 0.0$} & \\cellcolor[HTML]{EFEFEF}{$12.6$\\,GB} &  &  \\\\\n",
      "\\multicolumn{1}{l|}{\\textbf{QSGD}} & {$-0.5\\pm 0.6$} & \\multicolumn{1}{l|}{{$9.5\\times$}} & {$-0.1\\pm 0.1$} & {$648\\times$} &  &  \\\\\n",
      "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{FP8}} & \\cellcolor[HTML]{EFEFEF}{$-0.2\\pm 0.4$} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{$4.0\\times$ ($0.42\\!\\times\\!\\!\\!\\!\\!\\times$)}} & \\cellcolor[HTML]{EFEFEF}{$+0.0\\pm 0.1$} & \\cellcolor[HTML]{EFEFEF}{$4.0\\times$ ($0.01\\!\\times\\!\\!\\!\\!\\!\\times$)} &  &  \\\\\n",
      "\\multicolumn{1}{l|}{\\textbf{\\begin{tabular}[c]{@{}l@{}}FedPAQ (FxPQ)\\end{tabular}}} & {$-0.5\\pm 0.6$} & \\multicolumn{1}{l|}{{$3.2\\times$ ($0.34\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$-0.1\\pm 0.1$} & {$6.4\\times$ ($0.01\\!\\times\\!\\!\\!\\!\\!\\times$)} &  &  \\\\\n",
      "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{GZip}} & \\cellcolor[HTML]{EFEFEF}{$-0.5\\pm 0.6$} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{$9.3\\times$ ($0.97\\!\\times\\!\\!\\!\\!\\!\\times$)}} & \\cellcolor[HTML]{EFEFEF}{$-0.1\\pm 0.2$} & \\cellcolor[HTML]{EFEFEF}{$494\\times$ ($0.76\\!\\times\\!\\!\\!\\!\\!\\times$)} &  &  \\\\\n",
      "\\multicolumn{1}{l|}{\\textbf{UVeQFed}} & {$-0.0\\pm 0.4$} & \\multicolumn{1}{l|}{{$7.9\\times$ ($0.83\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$-0.4\\pm 0.3$} & {$31\\times$ ($0.05\\!\\times\\!\\!\\!\\!\\!\\times$)} &  &  \\\\ \\cline{1-5}\n",
      "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{DAdaQuant}} & \\cellcolor[HTML]{EFEFEF}{$-0.6\\pm 0.5$} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{$21\\times$ ($2.21\\!\\times\\!\\!\\!\\!\\!\\times$)}} & \\cellcolor[HTML]{EFEFEF}{$-0.1\\pm 0.1$} & \\cellcolor[HTML]{EFEFEF}{$775\\times$ ($1.20\\!\\times\\!\\!\\!\\!\\!\\times$)} &  &  \\\\\n",
      "\\multicolumn{1}{l|}{\\textbf{DAdaQuant$_{\\text{time}}$}} & {$-0.5\\pm 0.5$} & \\multicolumn{1}{l|}{{$12\\times$ ($1.29\\!\\times\\!\\!\\!\\!\\!\\times$)}} & {$-0.1\\pm 0.2$} & {$716\\times$ ($1.10\\!\\times\\!\\!\\!\\!\\!\\times$)} &  &  \\\\\n",
      "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{DAdaQuant$_{\\text{clients}}$}} & \\cellcolor[HTML]{EFEFEF}{$-0.4\\pm 0.5$} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{$16\\times$ ($1.67\\!\\times\\!\\!\\!\\!\\!\\times$)}} & \\cellcolor[HTML]{EFEFEF}{$-0.1\\pm 0.0$} & \\cellcolor[HTML]{EFEFEF}{$700\\times$ ($1.08\\!\\times\\!\\!\\!\\!\\!\\times$)} &  & \n",
      "\\end{tabular}\n",
      "\\end{table}\n",
      "\n"
     ]
    }
   ],
   "source": [
    "exps = {\n",
    "    \"synthetic\": {\n",
    "        \"uncompressed\": ['8000', '8100', '8200'],\n",
    "        \"qsgd\": ['8001', '8101', '8201'],\n",
    "        \"fixedpoint\": ['8001', '8101', '8201'],\n",
    "        \"monoquant\": ['8002', '8102', '8202'],\n",
    "        \"samplequant\": ['8003', '8103', '8203'],\n",
    "        \"dadaquant\": ['8004', '8104', '8204'],\n",
    "        \"uveqfed\": ['7002', '7102', '7202'],\n",
    "        \"gzip\": ['7003', '7103', '7203'],\n",
    "        \"fp8\": ['7004', '7104', '7204'],\n",
    "    },\n",
    "    \"femnist\": {\n",
    "        \"uncompressed\": ['8010', '8110', '8210'],\n",
    "        \"qsgd\": ['8011', '8111', '8211'],\n",
    "        \"fixedpoint\": ['8011', '8111', '8211'],\n",
    "        \"monoquant\": ['8012', '8112', '8212'],\n",
    "        \"samplequant\": ['8013', '8113', '8213'],\n",
    "        \"dadaquant\": ['8014', '8114', '8214'],\n",
    "        \"uveqfed\": ['7012', '7112', '7212'],\n",
    "        \"gzip\": ['7013', '7113', '7213'],\n",
    "        \"fp8\": ['7014', '7114', '7214'],\n",
    "    },\n",
    "    \"sent140\": {\n",
    "        \"uncompressed\": ['8020', '8120', '8220'],\n",
    "        \"qsgd\": ['8021', '8121', '8221'],\n",
    "        \"fixedpoint\": ['8021', '8121', '8221'],\n",
    "        \"monoquant\": ['8022', '8122', '8222'],\n",
    "        \"samplequant\": ['8023', '8123', '8223'],\n",
    "        \"dadaquant\": ['8024', '8124', '8224'],\n",
    "        \"uveqfed\": ['7022', '7122', '7222'],\n",
    "        \"gzip\": ['7023', '7123', '7223'],\n",
    "        \"fp8\": ['7024', '7124', '7224']\n",
    "    },\n",
    "    \"shakespeare\": {\n",
    "        \"uncompressed\": ['8030', '8130', '8230'],\n",
    "        \"qsgd\": ['8031', '8131', '8231'],\n",
    "        \"fixedpoint\": ['8031', '8131', '8231'],\n",
    "        \"monoquant\": ['8032', '8132', '8232'],\n",
    "        \"samplequant\": ['8033', '8133', '8233'],\n",
    "        \"dadaquant\": ['8034', '8134', '8234'],\n",
    "        \"uveqfed\": ['7032', '7132', '7232'],\n",
    "        \"gzip\": ['7033', '7133', '7233'],\n",
    "        \"fp8\": ['7034', '7134', '7234'],\n",
    "    },\n",
    "    \"celeba\": {\n",
    "        \"uncompressed\": ['8040', '8140', '8240'],\n",
    "        \"qsgd\": ['8041', '8141', '8241'],\n",
    "        \"fixedpoint\": ['8041', '8141', '8241'],\n",
    "        \"monoquant\": ['8042', '8142', '8242'],\n",
    "        \"samplequant\": ['8043', '8143', '8243'],\n",
    "        \"dadaquant\": ['8044', '8144', '8244'],\n",
    "        \"uveqfed\": ['7042', '7142', '7242'],\n",
    "        \"gzip\": ['7043', '7143', '7243'],\n",
    "        \"fp8\": ['7044', '7144', '7244'],\n",
    "    }\n",
    "}\n",
    "\n",
    "table = r\"\"\"\n",
    "% Please add the following required packages to your document preamble:\n",
    "% \\usepackage[table,xcdraw]{xcolor}\n",
    "% If you use beamer only pass \"xcolor=table\" option, i.e. \\documentclass[xcolor=table]{beamer}\n",
    "\\begin{table}[]\n",
    "\\begin{tabular}{lllllll}\n",
    "\\multicolumn{1}{l|}{} & \\multicolumn{2}{c|}{\\textbf{Synthetic}} & \\multicolumn{2}{c|}{\\textbf{FEMNIST}} & \\multicolumn{2}{c}{\\textbf{Sent140}} \\\\ \\hline\n",
    "\\rowcolor[HTML]{EFEFEF} \n",
    "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{Uncompressed}} & {synthetic_uncompressed_acc} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{synthetic_uncompressed_comm}} & {femnist_uncompressed_acc} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{femnist_uncompressed_comm}} & {sent140_uncompressed_acc} & {sent140_uncompressed_comm} \\\\\n",
    "\\multicolumn{1}{l|}{\\textbf{QSGD}} & {synthetic_qsgd_acc} & \\multicolumn{1}{l|}{{synthetic_qsgd_comm}} & {femnist_qsgd_acc} & \\multicolumn{1}{l|}{{femnist_qsgd_comm}} & {sent140_qsgd_acc} & {sent140_qsgd_comm} \\\\\n",
    "\\rowcolor[HTML]{EFEFEF} \n",
    "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{FP8}} & {synthetic_fp8_acc} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{synthetic_fp8_comm}} & {femnist_fp8_acc} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{femnist_fp8_comm}} & {sent140_fp8_acc} & {sent140_fp8_comm} \\\\\n",
    "\\multicolumn{1}{l|}{\\textbf{FedPAQ (FxPQ)}} & {synthetic_fixedpoint_acc} & \\multicolumn{1}{l|}{{synthetic_fixedpoint_comm}} & {femnist_fixedpoint_acc} & \\multicolumn{1}{l|}{{femnist_fixedpoint_comm}} & {sent140_fixedpoint_acc} & {sent140_fixedpoint_comm} \\\\\n",
    "\\rowcolor[HTML]{EFEFEF} \n",
    "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{\\color[HTML]{333333} \\textbf{GZip}}} & {\\color[HTML]{333333} {synthetic_gzip_acc}} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{\\color[HTML]{333333} {synthetic_gzip_comm}}} & {\\color[HTML]{333333} {femnist_gzip_acc}} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{\\color[HTML]{333333} {femnist_gzip_comm}}} & {\\color[HTML]{333333} {sent140_gzip_acc}} & {\\color[HTML]{333333} {sent140_gzip_comm}} \\\\\n",
    "\\multicolumn{1}{l|}{\\textbf{UVeQFed}} & {synthetic_uveqfed_acc} & \\multicolumn{1}{l|}{{synthetic_uveqfed_comm}} & {femnist_uveqfed_acc} & \\multicolumn{1}{l|}{{femnist_uveqfed_comm}} & {sent140_uveqfed_acc} & {sent140_uveqfed_comm} \\\\ \\hline\n",
    "\\rowcolor[HTML]{EFEFEF} \n",
    "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{DAdaQuant}} & {synthetic_dadaquant_acc} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{synthetic_dadaquant_comm}} & {femnist_dadaquant_acc} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{femnist_dadaquant_comm}} & {sent140_dadaquant_acc} & {sent140_dadaquant_comm} \\\\\n",
    "\\multicolumn{1}{l|}{\\textbf{DAdaQuant$_{\\text{time}}$}} & {synthetic_monoquant_acc} & \\multicolumn{1}{l|}{{synthetic_monoquant_comm}} & {femnist_monoquant_acc} & \\multicolumn{1}{l|}{{femnist_monoquant_comm}} & {sent140_monoquant_acc} & {sent140_monoquant_comm} \\\\\n",
    "\\rowcolor[HTML]{EFEFEF} \n",
    "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{DAdaQuant$_{\\text{clients}}$}} & {synthetic_samplequant_acc} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{synthetic_samplequant_comm}} & {femnist_samplequant_acc} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{femnist_samplequant_comm}} & {sent140_samplequant_acc} & {sent140_samplequant_comm} \\\\\n",
    " &  &  &  &  &  &  \\\\\n",
    "\\multicolumn{1}{l|}{} & \\multicolumn{2}{c|}{\\textbf{Shakespeare}} & \\multicolumn{2}{c}{\\textbf{Celeba}} &  &  \\\\ \\cline{1-5}\n",
    "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{Uncompressed}} & \\cellcolor[HTML]{EFEFEF}{shakespeare_uncompressed_acc} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{shakespeare_uncompressed_comm}} & \\cellcolor[HTML]{EFEFEF}{celeba_uncompressed_acc} & \\cellcolor[HTML]{EFEFEF}{celeba_uncompressed_comm} &  &  \\\\\n",
    "\\multicolumn{1}{l|}{\\textbf{QSGD}} & {shakespeare_qsgd_acc} & \\multicolumn{1}{l|}{{shakespeare_qsgd_comm}} & {celeba_qsgd_acc} & {celeba_qsgd_comm} &  &  \\\\\n",
    "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{FP8}} & \\cellcolor[HTML]{EFEFEF}{shakespeare_fp8_acc} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{shakespeare_fp8_comm}} & \\cellcolor[HTML]{EFEFEF}{celeba_fp8_acc} & \\cellcolor[HTML]{EFEFEF}{celeba_fp8_comm} &  &  \\\\\n",
    "\\multicolumn{1}{l|}{\\textbf{\\begin{tabular}[c]{@{}l@{}}FedPAQ (FxPQ)\\end{tabular}}} & {shakespeare_fixedpoint_acc} & \\multicolumn{1}{l|}{{shakespeare_fixedpoint_comm}} & {celeba_fixedpoint_acc} & {celeba_fixedpoint_comm} &  &  \\\\\n",
    "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{GZip}} & \\cellcolor[HTML]{EFEFEF}{shakespeare_gzip_acc} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{shakespeare_gzip_comm}} & \\cellcolor[HTML]{EFEFEF}{celeba_gzip_acc} & \\cellcolor[HTML]{EFEFEF}{celeba_gzip_comm} &  &  \\\\\n",
    "\\multicolumn{1}{l|}{\\textbf{UVeQFed}} & {shakespeare_uveqfed_acc} & \\multicolumn{1}{l|}{{shakespeare_uveqfed_comm}} & {celeba_uveqfed_acc} & {celeba_uveqfed_comm} &  &  \\\\ \\cline{1-5}\n",
    "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{DAdaQuant}} & \\cellcolor[HTML]{EFEFEF}{shakespeare_dadaquant_acc} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{shakespeare_dadaquant_comm}} & \\cellcolor[HTML]{EFEFEF}{celeba_dadaquant_acc} & \\cellcolor[HTML]{EFEFEF}{celeba_dadaquant_comm} &  &  \\\\\n",
    "\\multicolumn{1}{l|}{\\textbf{DAdaQuant$_{\\text{time}}$}} & {shakespeare_monoquant_acc} & \\multicolumn{1}{l|}{{shakespeare_monoquant_comm}} & {celeba_monoquant_acc} & {celeba_monoquant_comm} &  &  \\\\\n",
    "\\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}\\textbf{DAdaQuant$_{\\text{clients}}$}} & \\cellcolor[HTML]{EFEFEF}{shakespeare_samplequant_acc} & \\multicolumn{1}{l|}{\\cellcolor[HTML]{EFEFEF}{shakespeare_samplequant_comm}} & \\cellcolor[HTML]{EFEFEF}{celeba_samplequant_acc} & \\cellcolor[HTML]{EFEFEF}{celeba_samplequant_comm} &  & \n",
    "\\end{tabular}\n",
    "\\end{table}\n",
    "\"\"\"\n",
    "\n",
    "final_table = table\n",
    "for dataset in exps.keys():\n",
    "    uncompressed_acc_avg, _ = get_avg_var(exps[dataset]['uncompressed'], n_rounds[dataset], ACC_KEY)\n",
    "    uncompressed_comm_avg, _ = get_avg_var(exps[dataset]['uncompressed'], n_rounds[dataset], COMM_KEY)\n",
    "    qsgd_comm_avg,_ = get_avg_var(exps[dataset]['qsgd'], n_rounds[dataset], COMM_KEY)\n",
    "    for compressor in exps[dataset].keys():\n",
    "        exp = exps[dataset][compressor]\n",
    "        max_acc_avg, max_acc_var = get_avg_var(exp, n_rounds[dataset], ACC_KEY)\n",
    "        max_comm_avg, max_comm_var = get_avg_var(exp, n_rounds[dataset], COMM_KEY)\n",
    "        if compressor == \"fixedpoint\":\n",
    "            max_comm_avg = (n_rounds[dataset]-1) * ((1+math.ceil(math.log2(q_levels[dataset])))/8) * model_params[dataset] * 10/1_000_000  # 10 == n_clients\n",
    "        print(f\"Dataset {dataset} | Compressor {compressor} | max_acc_avg {max_acc_avg} | max_acc_var {max_acc_var} | max_comm_avg {max_comm_avg} | max_comm_var {max_comm_var}\")\n",
    "        if compressor == \"uncompressed\":\n",
    "            acc_str = f'${max_acc_avg*100:.1f}\\pm {max_acc_var*100:.1f}$'\n",
    "            if max_comm_avg > 1000:\n",
    "                comm_str = f'${max_comm_avg/1000:.1f}$\\,GB'\n",
    "            else:\n",
    "                comm_str = f'${max_comm_avg:.1f}$\\,MB'\n",
    "        elif compressor == \"qsgd\":\n",
    "            acc_diff = max_acc_avg - uncompressed_acc_avg\n",
    "            acc_str = f'${\"+\" if acc_diff > 0 else \"\"}{acc_diff*100:.1f}\\pm {max_acc_var*100:.1f}$'\n",
    "            comm_factor = uncompressed_comm_avg / max_comm_avg\n",
    "            comm_str = f'${comm_factor:.{1 if comm_factor < 10 else 0}f}\\\\times$'           \n",
    "        else:\n",
    "            acc_diff = max_acc_avg - uncompressed_acc_avg\n",
    "            acc_str = f'${\"+\" if acc_diff > 0 else \"\"}{acc_diff*100:.1f}\\pm {max_acc_var*100:.1f}$'\n",
    "            comm_factor = uncompressed_comm_avg / max_comm_avg\n",
    "#             print(\"comm_factor\", comm_factor)\n",
    "            comm_factor2 = qsgd_comm_avg / max_comm_avg\n",
    "            comm_str = f'${comm_factor:.{1 if comm_factor < 10 else 0}f}\\\\times$ (${comm_factor2:.{2 if comm_factor2 < 10 else 0}f}\\!\\\\times\\!\\!\\!\\!\\!\\\\times$)'\n",
    "        final_table = final_table.replace(f\"{dataset}_{compressor}_acc\", acc_str)\n",
    "        final_table = final_table.replace(f\"{dataset}_{compressor}_comm\", comm_str)\n",
    "    \n",
    "print(final_table)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## UVeQFed R=1 communication [Experiments]"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "**Experiment IDs:**"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Synthetic 8000 8100 8200 8001 8101 8201 8004 8104 8204 7002 7102 7202 9000 \n",
    "\n",
    "FEMNIST 8010 8110 8210 8011 8111 8211 8014 8114 8214 7012 7112 7212 9010 \n",
    "\n",
    "Sent140 8020 8120 8220 8021 8121 8221 8024 8124 8224 7022 7122 7222 9020 \n",
    "\n",
    "Shakespeare 8030 8130 8230 8031 8131 8231 8034 8134 8234 7032 7132 7232 9030 \n",
    "\n",
    "CelebA 8040 8140 8240 8041 8141 8241 8044 8144 8244 7042 7142 7242 9040 "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 31,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Dataset synthetic | Compressor uncompressed | max_comm_avg 12.199999809265137 | max_comm_var 0.0\n",
      "Dataset synthetic | Compressor qsgd | max_comm_avg 0.7135173281033834 | max_comm_var 0.003198231336070269\n",
      "Dataset synthetic | Compressor dadaquant | max_comm_avg 0.2535466601451238 | max_comm_var 0.024520375083890524\n",
      "comm_factor 48.117375327610944\n",
      "Dataset synthetic | Compressor uveqfed4 | max_comm_avg 21.147427876790363 | max_comm_var 0.02560483949241102\n",
      "comm_factor 0.5769023013269065\n",
      "Dataset synthetic | Compressor uveqfed1 | max_comm_avg 0.9278799891471863 | max_comm_var nan\n",
      "comm_factor 13.148251877355547\n",
      "Dataset femnist | Compressor uncompressed | max_comm_avg 132074.203125 | max_comm_var 0.0\n",
      "Dataset femnist | Compressor qsgd | max_comm_avg 47.016473134358726 | max_comm_var 1.9451039045211542\n",
      "Dataset femnist | Compressor dadaquant | max_comm_avg 27.67728042602539 | max_comm_var 0.400001735994063\n",
      "comm_factor 4771.935720996941\n",
      "Dataset femnist | Compressor uveqfed4 | max_comm_avg 11315.933919270834 | max_comm_var 407.07010155800964\n",
      "comm_factor 11.67152477800175\n",
      "Dataset femnist | Compressor uveqfed1 | max_comm_avg 3934.0302734375 | max_comm_var nan\n",
      "comm_factor 33.57223863190952\n",
      "Dataset sent140 | Compressor uncompressed | max_comm_avg 43929.6796875 | max_comm_var 0.0\n",
      "Dataset sent140 | Compressor qsgd | max_comm_avg 486.9434814453125 | max_comm_var 9.00774311170668\n",
      "Dataset sent140 | Compressor dadaquant | max_comm_avg 407.9437967936198 | max_comm_var 10.99078180247003\n",
      "comm_factor 107.68561756001937\n",
      "Dataset sent140 | Compressor uveqfed4 | max_comm_avg 3007.78173828125 | max_comm_var 36.42688532897191\n",
      "comm_factor 14.605341580603827\n",
      "Dataset sent140 | Compressor uveqfed1 | max_comm_avg 1071.812255859375 | max_comm_var nan\n",
      "comm_factor 40.98635693643692\n",
      "Dataset shakespeare | Compressor uncompressed | max_comm_avg 267.0400085449219 | max_comm_var 0.0\n",
      "Dataset shakespeare | Compressor qsgd | max_comm_avg 28.05606969197591 | max_comm_var 0.13259322394366085\n",
      "Dataset shakespeare | Compressor dadaquant | max_comm_avg 12.689458847045898 | max_comm_var 1.040901862339181\n",
      "comm_factor 21.044239298438537\n",
      "Dataset shakespeare | Compressor uveqfed4 | max_comm_avg 33.90665181477865 | max_comm_var 0.0869620229977027\n",
      "comm_factor 7.875741019894187\n",
      "Dataset shakespeare | Compressor uveqfed1 | max_comm_avg 12.610879898071289 | max_comm_var nan\n",
      "comm_factor 21.175366881874993\n",
      "Dataset celeba | Compressor uncompressed | max_comm_avg 12586.919921875 | max_comm_var 0.0\n",
      "Dataset celeba | Compressor qsgd | max_comm_avg 19.422468821207683 | max_comm_var 0.37589826342505556\n",
      "Dataset celeba | Compressor dadaquant | max_comm_avg 16.237250645955402 | max_comm_var 0.3441512103061088\n",
      "comm_factor 775.1878810228457\n",
      "Dataset celeba | Compressor uveqfed4 | max_comm_avg 407.6678059895833 | max_comm_var 9.362866421742552\n",
      "comm_factor 30.875432734555005\n",
      "Dataset celeba | Compressor uveqfed1 | max_comm_avg 134.90951538085938 | max_comm_var nan\n",
      "comm_factor 93.29897810648278\n",
      "\n",
      "% Please add the following required packages to your document preamble:\n",
      "% \\usepackage[table,xcdraw]{xcolor}\n",
      "% If you use beamer only pass \"xcolor=table\" option, i.e. \\documentclass[xcolor=table]{beamer}\n",
      "\\begin{table}[]\n",
      "\\begin{tabular}{l|l|l|l|l|l}\n",
      " & \\multicolumn{1}{c|}{\\textbf{Synthetic}} & \\multicolumn{1}{c|}{\\textbf{FEMNIST}} & \\multicolumn{1}{c|}{\\textbf{Sent140}} & \\multicolumn{1}{c|}{\\textbf{Shakespeare}} & \\multicolumn{1}{c}{\\textbf{Celeba}} \\\\ \\hline\n",
      "\\rowcolor[HTML]{EFEFEF} \n",
      "\\textbf{Uncompressed} & {$12.2$\\,MB} & {$132.1$\\,GB} & {$43.9$\\,GB} & \\cellcolor[HTML]{EFEFEF}{$267.0$\\,MB} & \\cellcolor[HTML]{EFEFEF}{$12.6$\\,GB} \\\\\n",
      "\\textbf{QSGD} & {$17\\times$} & {$2809\\times$} & {$90\\times$} & {$9.5\\times$} & {$648\\times$} \\\\\n",
      "\\rowcolor[HTML]{EFEFEF} \n",
      "\\textbf{UVeQFed (R=1)} & {$13\\times$ ($0.77\\!\\times\\!\\!\\!\\!\\!\\times$)} & {$34\\times$ ($0.01\\!\\times\\!\\!\\!\\!\\!\\times$)} & {$41\\times$ ($0.45\\!\\times\\!\\!\\!\\!\\!\\times$)} & {$21\\times$ ($2.22\\!\\times\\!\\!\\!\\!\\!\\times$)} & {$93\\times$ ($0.14\\!\\times\\!\\!\\!\\!\\!\\times$)} \\\\\n",
      "\\textbf{UVeQFed (R=4)} & {$0.6\\times$ ($0.03\\!\\times\\!\\!\\!\\!\\!\\times$)} & {$12\\times$ ($0.00\\!\\times\\!\\!\\!\\!\\!\\times$)} & {$15\\times$ ($0.16\\!\\times\\!\\!\\!\\!\\!\\times$)} & {$7.9\\times$ ($0.83\\!\\times\\!\\!\\!\\!\\!\\times$)} & {$31\\times$ ($0.05\\!\\times\\!\\!\\!\\!\\!\\times$)} \\\\\n",
      "\\rowcolor[HTML]{EFEFEF} \n",
      "\\textbf{DAdaQuant} & {$48\\times$ ($2.81\\!\\times\\!\\!\\!\\!\\!\\times$)} & {$4772\\times$ ($1.70\\!\\times\\!\\!\\!\\!\\!\\times$)} & {$108\\times$ ($1.19\\!\\times\\!\\!\\!\\!\\!\\times$)} & \\cellcolor[HTML]{EFEFEF}{$21\\times$ ($2.21\\!\\times\\!\\!\\!\\!\\!\\times$)} & \\cellcolor[HTML]{EFEFEF}{$775\\times$ ($1.20\\!\\times\\!\\!\\!\\!\\!\\times$)}\n",
      "\\end{tabular}\n",
      "\\end{table}\n",
      "\n"
     ]
    }
   ],
   "source": [
    "exps = {\n",
    "    \"synthetic\": {\n",
    "        \"uncompressed\": ['8000', '8100', '8200'],\n",
    "        \"qsgd\": ['8001', '8101', '8201'],\n",
    "        \"dadaquant\": ['8004', '8104', '8204'],\n",
    "        \"uveqfed4\": ['7002', '7102', '7202'],\n",
    "        \"uveqfed1\": ['9000'],\n",
    "    },\n",
    "    \"femnist\": {\n",
    "        \"uncompressed\": ['8010', '8110', '8210'],\n",
    "        \"qsgd\": ['8011', '8111', '8211'],\n",
    "        \"dadaquant\": ['8014', '8114', '8214'],\n",
    "        \"uveqfed4\": ['7012', '7112', '7212'],\n",
    "        \"uveqfed1\": ['9010'],\n",
    "    },\n",
    "    \"sent140\": {\n",
    "        \"uncompressed\": ['8020', '8120', '8220'],\n",
    "        \"qsgd\": ['8021', '8121', '8221'],\n",
    "        \"dadaquant\": ['8024', '8124', '8224'],\n",
    "        \"uveqfed4\": ['7022', '7122', '7222'],\n",
    "        \"uveqfed1\": ['9020'],\n",
    "    },\n",
    "    \"shakespeare\": {\n",
    "        \"uncompressed\": ['8030', '8130', '8230'],\n",
    "        \"qsgd\": ['8031', '8131', '8231'],\n",
    "        \"dadaquant\": ['8034', '8134', '8234'],\n",
    "        \"uveqfed4\": ['7032', '7132', '7232'],\n",
    "        \"uveqfed1\": ['9030'],\n",
    "    },\n",
    "    \"celeba\": {\n",
    "        \"uncompressed\": ['8040', '8140', '8240'],\n",
    "        \"qsgd\": ['8041', '8141', '8241'],\n",
    "        \"dadaquant\": ['8044', '8144', '8244'],\n",
    "        \"uveqfed4\": ['7042', '7142', '7242'],\n",
    "        \"uveqfed1\": ['9040'],\n",
    "    }\n",
    "}\n",
    "\n",
    "table = r\"\"\"\n",
    "% Please add the following required packages to your document preamble:\n",
    "% \\usepackage[table,xcdraw]{xcolor}\n",
    "% If you use beamer only pass \"xcolor=table\" option, i.e. \\documentclass[xcolor=table]{beamer}\n",
    "\\begin{table}[]\n",
    "\\begin{tabular}{l|l|l|l|l|l}\n",
    " & \\multicolumn{1}{c|}{\\textbf{Synthetic}} & \\multicolumn{1}{c|}{\\textbf{FEMNIST}} & \\multicolumn{1}{c|}{\\textbf{Sent140}} & \\multicolumn{1}{c|}{\\textbf{Shakespeare}} & \\multicolumn{1}{c}{\\textbf{Celeba}} \\\\ \\hline\n",
    "\\rowcolor[HTML]{EFEFEF} \n",
    "\\textbf{Uncompressed} & {synthetic_uncompressed_comm} & {femnist_uncompressed_comm} & {sent140_uncompressed_comm} & \\cellcolor[HTML]{EFEFEF}{shakespeare_uncompressed_comm} & \\cellcolor[HTML]{EFEFEF}{celeba_uncompressed_comm} \\\\\n",
    "\\textbf{QSGD} & {synthetic_qsgd_comm} & {femnist_qsgd_comm} & {sent140_qsgd_comm} & {shakespeare_qsgd_comm} & {celeba_qsgd_comm} \\\\\n",
    "\\rowcolor[HTML]{EFEFEF} \n",
    "\\textbf{UVeQFed (R=1)} & {synthetic_uveqfed1_comm} & {femnist_uveqfed1_comm} & {sent140_uveqfed1_comm} & {shakespeare_uveqfed1_comm} & {celeba_uveqfed1_comm} \\\\\n",
    "\\textbf{UVeQFed (R=4)} & {synthetic_uveqfed4_comm} & {femnist_uveqfed4_comm} & {sent140_uveqfed4_comm} & {shakespeare_uveqfed4_comm} & {celeba_uveqfed4_comm} \\\\\n",
    "\\rowcolor[HTML]{EFEFEF} \n",
    "\\textbf{DAdaQuant} & {synthetic_dadaquant_comm} & {femnist_dadaquant_comm} & {sent140_dadaquant_comm} & \\cellcolor[HTML]{EFEFEF}{shakespeare_dadaquant_comm} & \\cellcolor[HTML]{EFEFEF}{celeba_dadaquant_comm}\n",
    "\\end{tabular}\n",
    "\\end{table}\n",
    "\"\"\"\n",
    "\n",
    "final_table = table\n",
    "for dataset in exps.keys():\n",
    "    uncompressed_comm_avg, _ = get_avg_var(exps[dataset]['uncompressed'], n_rounds[dataset], COMM_KEY)\n",
    "    qsgd_comm_avg,_ = get_avg_var(exps[dataset]['qsgd'], n_rounds[dataset], COMM_KEY)\n",
    "    for compressor in exps[dataset].keys():\n",
    "        exp = exps[dataset][compressor]\n",
    "        max_comm_avg, max_comm_var = get_avg_var(exp, n_rounds[dataset], COMM_KEY)\n",
    "        if compressor == \"fixedpoint\":\n",
    "            max_comm_avg = (n_rounds[dataset]-1) * ((1+math.ceil(math.log2(q_levels[dataset])))/8) * model_params[dataset] * 10/1_000_000  # 10 == n_clients\n",
    "        print(f\"Dataset {dataset} | Compressor {compressor} | max_comm_avg {max_comm_avg} | max_comm_var {max_comm_var}\")\n",
    "        if compressor == \"uncompressed\":\n",
    "            if max_comm_avg > 1000:\n",
    "                comm_str = f'${max_comm_avg/1000:.1f}$\\,GB'\n",
    "            else:\n",
    "                comm_str = f'${max_comm_avg:.1f}$\\,MB'\n",
    "        elif compressor == \"qsgd\":\n",
    "            comm_factor = uncompressed_comm_avg / max_comm_avg\n",
    "            comm_str = f'${comm_factor:.{1 if comm_factor < 10 else 0}f}\\\\times$'           \n",
    "        else:\n",
    "            comm_factor = uncompressed_comm_avg / max_comm_avg\n",
    "            print(\"comm_factor\", comm_factor)\n",
    "            comm_factor2 = qsgd_comm_avg / max_comm_avg\n",
    "            comm_str = f'${comm_factor:.{1 if comm_factor < 10 else 0}f}\\\\times$ (${comm_factor2:.{2 if comm_factor2 < 10 else 0}f}\\!\\\\times\\!\\!\\!\\!\\!\\\\times$)'\n",
    "        final_table = final_table.replace(f\"{dataset}_{compressor}_comm\", comm_str)\n",
    "    \n",
    "print(final_table)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "flower",
   "language": "python",
   "name": "flower"
  },
  "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.7.9"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
