{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "05d0d9ac-df99-4df9-a152-ee8880d1a37a",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "import client\n",
    "import server\n",
    "import utils"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "c9dbead6-d971-4f87-92d8-45d22839f3d8",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<module 'utils' from '/Users/gael/Desktop/het-opl/src/utils.py'>"
      ]
     },
     "execution_count": 2,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "import importlib\n",
    "importlib.reload(client)\n",
    "importlib.reload(server)\n",
    "importlib.reload(utils)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "3e70eb84-1ef1-4cf1-9350-8976eaa5b27c",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "from tqdm.notebook import tqdm\n",
    "\n",
    "import random\n",
    "import numpy as np\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "sns.set_theme()\n",
    "\n",
    "from vowpalwabbit.sklearn import VW\n",
    "from sklearn.model_selection import train_test_split"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "a536414e-871c-444a-a370-3d4390d7e0cf",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "np.random.seed(42)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "afbfacfe-ea33-47f8-8968-d7a57c749722",
   "metadata": {},
   "source": [
    "### Generate data from homogeneous clients"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "4a5231e8-a8c5-4a4a-a9ab-8df7e1d29ff8",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def generate_separated_vectors(n, k, L=1.0, num_vectors=4, max_iter=1000):\n",
    "    vectors = np.zeros((num_vectors, n))\n",
    "    for i in range(num_vectors):\n",
    "        for _ in range(max_iter):\n",
    "            vector = np.random.rand(n) * 2 * L - L\n",
    "            if i == 0 or np.all(np.linalg.norm(vectors[:i] - vector, axis=1) >= k):\n",
    "                vectors[i] = vector\n",
    "                break\n",
    "        else:\n",
    "            raise RuntimeError(f\"Failed to generate a vector separated by {k} after {max_iter} iterations.\")\n",
    "    return vectors\n",
    "\n",
    "def generate_observational_data(num_clients, num_actions, num_features,\n",
    "                                train_sizes, test_sizes=None):\n",
    "    if test_sizes is None:\n",
    "        test_sizes = [10_000] * num_clients\n",
    "\n",
    "    # Generate action parameters\n",
    "    found_action_params = False\n",
    "    while not found_action_params:\n",
    "        action_params = generate_separated_vectors(num_features, 1, np.sqrt(num_features), 4)\n",
    "        found_action_params = True\n",
    "        for i in range(num_actions):\n",
    "            other_vectors = np.delete(action_params, [i], axis=0)\n",
    "            if not all(np.linalg.norm(action_params[i] - v) > np.sqrt(num_features) for v in other_vectors):\n",
    "                found_action_params = False\n",
    "                \n",
    "    print(\"Found action params.\")\n",
    "\n",
    "    data = {}\n",
    "    aux = {}\n",
    "    for client_id in range(num_clients):\n",
    "        # Get num local samples, including test samples\n",
    "        num_samples = train_sizes[client_id] + test_sizes[client_id]\n",
    "        \n",
    "        # Generate local data\n",
    "        contexts = np.random.normal(loc=0, scale=1, size=(num_samples, num_features)) \n",
    "        rewards_vectors = np.zeros((num_samples, num_actions))\n",
    "        for i in range(num_samples):\n",
    "            for a in range(num_actions):\n",
    "                rewards_vectors[i, a] += np.dot(contexts[i], action_params[a])\n",
    "                if contexts[i][0] > 0:\n",
    "                    rewards_vectors[i, a] += -np.max([np.dot(contexts[i], action_params[aprime]) for aprime in range(num_actions)])\n",
    "        # actions = np.random.choice(num_actions, p=[0.7, 0.1, 0.1, 0.1], size=num_samples)\n",
    "        actions = np.random.choice(num_actions, size=num_samples)\n",
    "        epsilons = np.random.normal(loc=0, scale=1, size=(num_samples, num_actions))\n",
    "        noisy_rewards_vectors = rewards_vectors + epsilons\n",
    "        noisy_rewards = noisy_rewards_vectors[np.arange(num_samples), actions]\n",
    "        \n",
    "        # Get local train-test split\n",
    "        (X_train, X_test,\n",
    "         A_train, A_test,\n",
    "         Y_train, Y_test,\n",
    "         true_costs_train, true_costs_test) = train_test_split(contexts, actions, noisy_rewards, -rewards_vectors,\n",
    "                                                               train_size=train_sizes[client_id],\n",
    "                                                               shuffle=False)\n",
    "\n",
    "        # Compute local AIPW scores\n",
    "        crossfit_map, mu, e = utils.cross_fit_nuisance_params(X_train, A_train, Y_train, num_actions)\n",
    "        AIPW_vectors = utils.compute_AIPW_scores(X_train, A_train, Y_train, num_actions, crossfit_map, mu, e)\n",
    "\n",
    "        # Convert local data to VW format\n",
    "        vw_data = utils.to_vw_format(X_train, A_train, -AIPW_vectors)\n",
    "\n",
    "        # Store local data\n",
    "        data[client_id] = vw_data\n",
    "        aux[client_id] = {\"X_train\": X_train, \"X_test\": X_test,\n",
    "                          \"A_train\": A_train, \"A_test\": A_test,\n",
    "                          \"Y_train\": Y_train, \"Y_test\": Y_test,\n",
    "                          \"true_costs_train\": true_costs_train, \"true_costs_test\": true_costs_test}\n",
    "    \n",
    "    return data, aux"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "4a67f2b2-df73-4cbe-a52f-6930f01644d5",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Found action params.\n"
     ]
    }
   ],
   "source": [
    "NUM_CLIENTS = 4\n",
    "NUM_ACTIONS = 4\n",
    "NUM_FEATURES = 10\n",
    "TRAIN_SIZES = np.array([1, 1, 1, 1]) * 250_000\n",
    "TEST_SIZES = np.array([1, 1, 1, 1]) * 10_000\n",
    "\n",
    "# Generate data\n",
    "data, aux = generate_observational_data(num_clients=NUM_CLIENTS,\n",
    "                                        num_actions=NUM_ACTIONS,\n",
    "                                        num_features=NUM_FEATURES,\n",
    "                                        train_sizes=TRAIN_SIZES,\n",
    "                                        test_sizes=TEST_SIZES)"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "ead70fa5-4ccb-4180-855e-02c74509adb9",
   "metadata": {},
   "source": [
    "### Train optimal model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "115935e2-8ff4-447f-b26c-875e8a850585",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Aggregate: opt_reward=3.306222474682597, reward=3.30580807557235, regret=0.0004143991102456278\n"
     ]
    }
   ],
   "source": [
    "# Aggregate all data\n",
    "X_train = []\n",
    "A_train = []\n",
    "Y_train = []\n",
    "true_costs_train = []\n",
    "true_costs_test = []\n",
    "X_test = []\n",
    "for client_id in range(NUM_CLIENTS):\n",
    "    X_train.extend(aux[client_id][\"X_train\"])\n",
    "    A_train.extend(aux[client_id][\"A_train\"])\n",
    "    Y_train.extend(aux[client_id][\"Y_train\"])\n",
    "    true_costs_train.extend(aux[client_id][\"true_costs_train\"])\n",
    "    true_costs_test.extend(aux[client_id][\"true_costs_test\"])\n",
    "    X_test.extend(aux[client_id][\"X_test\"])\n",
    "X_train = np.array(X_train)\n",
    "A_train = np.array(A_train)\n",
    "Y_train = np.array(Y_train)\n",
    "true_costs_train = np.array(true_costs_train)\n",
    "true_costs_test = np.array(true_costs_test)\n",
    "X_test = np.array(X_test)\n",
    "\n",
    "# Compute AIPW scores\n",
    "crossfit_map, mu, e = utils.cross_fit_nuisance_params(X_train, A_train, Y_train, NUM_ACTIONS)\n",
    "noisy_costs_train = -utils.compute_AIPW_scores(X_train, A_train, Y_train, NUM_ACTIONS, crossfit_map, mu, e)\n",
    "data_train = utils.to_vw_format(X_train, A_train, noisy_costs_train)\n",
    "\n",
    "# Train model\n",
    "opt_global_model = VW(csoaa=NUM_ACTIONS,\n",
    "               convert_to_vw=False,\n",
    "               convert_labels=False,\n",
    "               passes=1)\n",
    "opt_global_model.fit(data_train)\n",
    "\n",
    "# Evaluate on test data\n",
    "X_test_vw = utils.to_vw_format(X_test)\n",
    "regret, opt_reward, reward = utils.compute_regret(X_test_vw, true_costs_test, opt_global_model)\n",
    "print(f\"Aggregate: opt_reward={opt_reward}, reward={reward}, regret={regret}\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "6db1fb15-a24f-4126-9989-36a0e679d84d",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "# Define optimal local models\n",
    "opt_local_models = {k:opt_global_model for k in range(NUM_CLIENTS)}"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "fdfc730d-c2da-4338-8b23-776ff446db01",
   "metadata": {},
   "source": [
    "### Train local model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "013b9541-ced0-4dd8-9105-0baa49a72584",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def train_local_policy(train_data, sample_size):\n",
    "    assert len(train_data) >= sample_size\n",
    "    \n",
    "    # Train on local data\n",
    "    model = VW(csoaa=NUM_ACTIONS,\n",
    "               convert_to_vw=False,\n",
    "               convert_labels=False,\n",
    "               passes=1)\n",
    "    # model.fit(train_data[:sample_size])\n",
    "    model.fit(random.sample(train_data, sample_size))\n",
    "    return model\n",
    "\n",
    "def eval_local_policy(X_test, true_costs, model, opt_model=None):\n",
    "    # Evaluate on test data\n",
    "    X_test_vw = utils.to_vw_format(X_test)\n",
    "    regret, opt_reward, reward = utils.compute_regret(X_test_vw, true_costs, model, opt_model)\n",
    "    return regret, opt_reward, reward\n",
    "\n",
    "def eval_random_policy(X_test, true_costs, opt_model=None):\n",
    "    # Create random policy and evaluate on test data\n",
    "    class Model():\n",
    "        def predict(self, X):\n",
    "            return np.random.choice(NUM_ACTIONS, size=len(X))\n",
    "        \n",
    "    X_test_vw = utils.to_vw_format(X_test)\n",
    "    regret, opt_reward, reward = utils.compute_regret(X_test_vw, true_costs, Model(), opt_model)\n",
    "    return regret, opt_reward, reward"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "ca841309-29b6-4c14-8166-ecc657405d74",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "ss_range = [10, 25, 50, 75, 100, 250, 500, 750, 1000, 2500, 5000, 7500, 10000]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "3c1dc831-e7bf-4fd0-9248-c77ebc47a4ef",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def run_local_training_experiments(data, aux, client_id=0, num_iters=3):\n",
    "    local_regrets_mean = []\n",
    "    local_regrets_std = []\n",
    "    random_regrets_mean = []\n",
    "    random_regrets_std = []\n",
    "\n",
    "    for sample_size in tqdm(ss_range):\n",
    "        local_regret = 0\n",
    "        random_regret = 0\n",
    "        local_regret_iters = []\n",
    "        random_regret_iters = []\n",
    "        for i in range(num_iters):\n",
    "            local_model = train_local_policy(data[client_id], sample_size)\n",
    "            lr, _, _ = eval_local_policy(aux[client_id][\"X_test\"], aux[client_id][\"true_costs_test\"], local_model, opt_local_models[client_id])\n",
    "            rr, _, _ = eval_random_policy(aux[client_id][\"X_test\"], aux[client_id][\"true_costs_test\"], opt_local_models[client_id])\n",
    "            local_regret_iters.append(lr)\n",
    "            random_regret_iters.append(rr)\n",
    "        local_regrets_mean.append(np.mean(local_regret_iters))\n",
    "        local_regrets_std.append(np.std(local_regret_iters))\n",
    "        random_regrets_mean.append(np.mean(random_regret_iters))\n",
    "        random_regrets_std.append(np.std(random_regret_iters))\n",
    "        \n",
    "    return (np.array(local_regrets_mean),\n",
    "            np.array(local_regrets_std),\n",
    "            np.array(random_regrets_mean),\n",
    "            np.array(random_regrets_std))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "209b3430-338d-498f-8de5-53ef86238f6a",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "application/vnd.jupyter.widget-view+json": {
       "model_id": "24c7d5de0eae45f38c689417cf0c512c",
       "version_major": 2,
       "version_minor": 0
      },
      "text/plain": [
       "  0%|          | 0/13 [00:00<?, ?it/s]"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "local_regrets, local_regrets_std, random_regrets, random_regrets_std = run_local_training_experiments(data, aux, client_id=0, num_iters=3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "8ce33d90-a09a-41e5-8606-33be46f6debb",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "image/png": "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I28pcSSgzrlBnXODOucOtc6NbNU5GCzIzhiTxGWZ3bt367DDDktu3717t6ZOndpj//Hjx2vjxo1p28LhsBobGwc18DUhm4MmbdsZ0YMyc4U65wZ1zg3qnDvUOjfyWecBXYCeOnWqiouL9eqrrya3NTc3691339Xxxx/fY/9Zs2appqZGVVVVyW2J186cOXOwbQYAAAeoAfWY+Hw+nX/++brjjjtUXl6uQw45RD/60Y80fvx4LVy4ULZtq76+XiUlJQoEAqqsrNTMmTN11VVXacWKFWpvb9fy5ct11llnDanHBAAAHJgGPGR/yZIl+spXvqLrr79eixYtkmVZWrt2rXw+n6qrqzV37lxt2LBBUmyw0apVq1RRUaELL7xQV155pebNm6cVK1Zk+n0AAIADwIAWWBsOWGDt04065wZ1zg3qnDvUOjeyVeeBLLD26V/kAAAAHDAIJgAAYNggmAAAgGGDYAIAAIYNggkAABg2CCYAAGDYIJgAAIBhg2ACAACGjU/dAmuu68pxstNkyzK5a2UOUOfcoM65QZ1zh1rnRjbqbJqGDMPo176fumACAAAOXFzKAQAAwwbBBAAADBsEEwAAMGwQTAAAwLBBMAEAAMMGwQQAAAwbBBMAADBsEEwAAMCwQTABAADDBsEEAAAMGwQTAAAwbBBMAADAsEEwAQAAw8aIDyaO4+iee+7RSSedpMrKSl188cWqqqrKd7OGtcbGRn3ve9/TvHnzNHPmTC1atEibN29OPv+3v/1N559/vo477jh9/vOf19q1a9Ne35+a7+8YI82HH36oGTNmaP369clt1DmzfvOb3+j000/X9OnTdcYZZ+i5555LPketMyMSieiuu+7S5z//ec2YMUPnnnuuXn/99eTz1Hno1qxZowsuuCBtWy7qmtHPUneEu/fee90TTzzR/e///m/3b3/7m3vxxRe7CxcudEOhUL6bNmz927/9m3vmmWe6mzZtcv/+97+7N910k3vssce677//vltfX++ecMIJ7rJly9z333/ffeKJJ9zp06e7TzzxRPL1+6t5f44xkoTDYffLX/6yO2XKFPfJJ590Xbd/NaLO/feb3/zGPeqoo9wHHnjA3bFjh7tq1Sp36tSp7uuvv06tM+jHP/6x+7nPfc7905/+5O7YscNdtmyZO3PmTLempoY6Z8AvfvEL98gjj3TPP//85LZc1TWTn6UjOpiEQiF3xowZ7rp165Lbmpqa3GOPPdZ99tln89iy4WvHjh3ulClT3C1btiS3OY7jLly40L377rvdn/70p+5JJ53kRiKR5PN33nmne+qpp7qu27+a7+8YI82dd97pXnDBBWnBhDpnjuM47vz5893bbrstbfvFF1/s/vSnP6XWGXTmmWe6t956a/LnlpYWd8qUKe7vfvc76jwENTU17uLFi93jjjvOPe2009KCSS7qmunP0hF9KWfr1q1qa2vTnDlzktuCwaCOPvpobdq0KY8tG77Kysp033336ZhjjkluMwxDruuqqalJmzdv1qxZs+TxeJLPz5kzRx9++KH27t3br5rv7xgjyaZNm/TYY4/p9ttvT9tOnTPngw8+0Mcff6wvfelLadvXrl2ryy67jFpn0KhRo/Tiiy9q165dsm1bjz32mHw+n4466ijqPATvvPOOSktL9fTTT6uysjLtuVzUNdOfpSM6mNTU1EiSJkyYkLZ97Nixqq6uzkeThr1gMKiTTz5ZPp8vue25557TRx99pLlz56qmpkbjx49Pe83YsWMlSZ988km/ar6/Y4wUzc3Nuvbaa3X99df3qBd1zpwdO3ZIktrb27V48WKdeOKJOuecc/Rf//Vfkqh1Ji1btkwej0ennHKKpk+frrvuukt33323DjvsMOo8BAsWLNCdd96pQw89tMdzuahrpj9LR3Qw6ejokKS0D1lJ8vv9CoVC+WjSp86WLVv03e9+V6eccooWLFigzs7OXuspSaFQqF81398xRooVK1bouOOO6/EveWn/NaLO/dfa2ipJuu666/TFL35R999/vz73uc/pG9/4hl5++WVqnUF///vfFQwGtXr1aj322GP68pe/rOuuu05bt26lzlmSi7pm+rPUs/9dDlyBQECSFA6Hk4+lWKELCgry1axPjY0bN2rp0qWqrKzUypUrJcVqGg6H0/ZL/GIWFhb2q+b7O8ZI8Jvf/EabN2/WM8880+vz1DlzvF6vJGnx4sU6++yzJUlHHXWU3n33Xf3iF7+g1hny8ccf61vf+pYeeOABHX/88ZKk6dOn6/3339e9995LnbMkF3XN9GfpiO4xSXQ77d69O2377t27e3RbId0jjzyiK664QvPmzdPPf/7z5C/j+PHje62nJI0bN65fNd/fMUaCJ598Unv37k1Oq5wxY4Ykafny5TrjjDOocwYl6jFlypS07ZMnT9auXbuodYa89dZbikQimj59etr2yspK7dixgzpnSS7qmunP0hEdTKZOnari4mK9+uqryW3Nzc169913k4kePa1bt0433XSTzjvvPN19991p3XezZs3Sli1bZNt2ctvLL7+siRMnavTo0f2q+f6OMRLccccd2rBhg37zm98kvyRpyZIluu+++6hzBh199NEqKirSm2++mbZ9+/btOuyww6h1hiQ+vLZt25a2ffv27Tr88MOpc5bkoq4Z/ywd8DyeA8zKlSvd2bNnuxs3bkzOvf7Hf/xH1jHpwwcffOBOmzbN/eY3v+nu3r077au5udnds2ePO2vWLPe6665z33vvPffJJ590p0+f7q5fvz55jP3VvD/HGIlSpwtT58xavXq1O2PGDPeZZ55xq6qq3DVr1rhTp051X3nlFWqdIbZtu+eee6572mmnuS+//LL74YcfunfddZd71FFHuW+88QZ1zpDrrrsubbpwruqayc/SER9MotGo+8Mf/tCdM2eOe9xxx7lf//rX3Z07d+a7WcPWT37yE3fKlCm9fl133XWu67rum2++6X71q191jznmGHf+/Pnuww8/nHaM/tR8f8cYiVKDietS50y7//773QULFrjTpk1zzzzzTPeFF15IPketM6OxsdFdsWKF+/nPf96dMWOG+7Wvfc199dVXk89T56HrHkxcNzd1zeRnqeG6rjuI3iEAAICMG9FjTAAAwPBCMAEAAMMGwQQAAAwbBBMAADBsEEwAAMCwQTABAADDBsEEAAAMGwQTAAAwbBBMAADAsEEwAQAAwwbBBAAADBv/HwycSEnlL7iBAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "x = ss_range\n",
    "plt.plot(x, local_regrets, 'b-', label='Local Regret')\n",
    "plt.fill_between(x, local_regrets - local_regrets_std, local_regrets + local_regrets_std, color='b', alpha=0.2)\n",
    "plt.legend()\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "593d6f30-ed82-4325-b787-944e9f3f245c",
   "metadata": {},
   "source": [
    "### Train federated model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "id": "3ccf41b0-dc5e-42b3-aea5-8087f6b89b5e",
   "metadata": {},
   "outputs": [],
   "source": [
    "NUM_ROUNDS = 1\n",
    "\n",
    "def run_federated_training_experiments(data, aux, sample_size_fns, num_iters=3):\n",
    "\n",
    "    local_regrets_mean = {k:[] for k in range(NUM_CLIENTS)}\n",
    "    local_regrets_std = {k:[] for k in range(NUM_CLIENTS)}\n",
    "    global_regrets_mean = []\n",
    "    global_regrets_std = []\n",
    "    \n",
    "    for sample_size in tqdm(ss_range):\n",
    "        local_regrets_iters = {k:[] for k in range(NUM_CLIENTS)}\n",
    "        global_regret_iters = []\n",
    "        for i in tqdm(range(num_iters)):\n",
    "\n",
    "            # Subsample data\n",
    "            data_mod = {}\n",
    "            aux_mod = {}\n",
    "            client_weights = []\n",
    "            for client_id in range(NUM_CLIENTS):\n",
    "                client_sample_size = sample_size_fns[client_id](sample_size)\n",
    "                client_weights.append(client_sample_size)\n",
    "\n",
    "                data_mod[client_id] = random.sample(data[client_id], client_sample_size)\n",
    "                aux_mod[client_id] = {\"X_test\": aux[client_id][\"X_test\"],\n",
    "                                      \"true_costs_test\": aux[client_id][\"true_costs_test\"]}\n",
    "\n",
    "            client_weights = np.array(client_weights) / np.sum(client_weights)\n",
    "            \n",
    "            # Run federated learning\n",
    "            global_vw = server.run_federated_learning(data_mod, aux_mod,\n",
    "                                                      opt_global_model, opt_local_models,\n",
    "                                                      num_features=NUM_FEATURES,\n",
    "                                                      num_classes=NUM_ACTIONS,\n",
    "                                                      num_rounds=NUM_ROUNDS,\n",
    "                                                      num_clients=NUM_CLIENTS,\n",
    "                                                      client_weights=client_weights)\n",
    "\n",
    "            gr = global_vw.metrics_distributed[\"global_regret\"][-1][1]\n",
    "            global_regret_iters.append(gr)\n",
    "            \n",
    "            for client_id in range(NUM_CLIENTS):\n",
    "                lr = global_vw.metrics_distributed[\"local_regrets\"][-1][1][client_id]\n",
    "                local_regrets_iters[client_id].append(lr)\n",
    "\n",
    "        global_regrets_mean.append(np.mean(global_regret_iters))\n",
    "        global_regrets_std.append(np.std(global_regret_iters))\n",
    "        \n",
    "        for client_id in range(NUM_CLIENTS):\n",
    "            local_regrets_mean[client_id].append(np.mean(local_regrets_iters[client_id]))\n",
    "            local_regrets_std[client_id].append(np.std(local_regrets_iters[client_id]))\n",
    "            \n",
    "    global_regrets_mean = np.array(global_regrets_mean)\n",
    "    global_regrets_std = np.array(global_regrets_std)\n",
    "    for client_id in range(NUM_CLIENTS):\n",
    "        local_regrets_mean[client_id] = np.array(local_regrets_mean[client_id])\n",
    "        local_regrets_std[client_id] = np.array(local_regrets_std[client_id])\n",
    "        \n",
    "    return (global_regrets_mean, global_regrets_std,\n",
    "            local_regrets_mean, local_regrets_std)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "90102894-9603-4e61-af5d-e5a76725a092",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:44:32,282 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:44:39,336 | app.py:179 | Flower VCE: Ray initialized with resources: {'node:127.0.0.1': 1.0, 'memory': 6236941927.0, 'object_store_memory': 2147483648.0, 'CPU': 8.0}\n",
      "INFO flwr 2023-05-17 11:44:39,338 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:44:39,339 | server.py:270 | Requesting initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:44:40,707 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:44:40,708 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:44:40,708 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:44:40,709 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=98859)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98859)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98859)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:44:41,917 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:44:41,948 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:44:41,950 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=98863)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98863)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98862)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98862)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98858)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98858)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98859)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98863)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98862)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98858)\u001b[0m [Client 1] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:44:43,100 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:44:43,101 | server.py:144 | FL finished in 2.392037056000163\n",
      "INFO flwr 2023-05-17 11:44:43,110 | app.py:202 | app_fit: losses_distributed [(1, 4.099784909973686)]\n",
      "INFO flwr 2023-05-17 11:44:43,111 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.305808075572349)], 'reward': [(1, -0.7939768344013435)], 'global_regret': [(1, 4.099784909973687)], 'local_regrets': [(1, {0: 4.1297177411696895, 1: 4.1027911988299675, 2: 4.013618553376745, 3: 4.153012146518346})]}\n",
      "INFO flwr 2023-05-17 11:44:43,112 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:44:43,113 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:44:43,119 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:44:51,635 | app.py:179 | Flower VCE: Ray initialized with resources: {'CPU': 8.0, 'memory': 6730399744.0, 'node:127.0.0.1': 1.0, 'object_store_memory': 2147483648.0}\n",
      "INFO flwr 2023-05-17 11:44:51,637 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:44:51,638 | server.py:270 | Requesting initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:44:52,645 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:44:52,646 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:44:52,646 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:44:52,647 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=98885)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98885)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98885)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:44:53,839 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:44:53,860 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:44:53,862 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=98883)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98883)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98886)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98886)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98884)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98884)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98885)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98883)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98886)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98884)\u001b[0m [Client 1] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:44:54,991 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:44:54,992 | server.py:144 | FL finished in 2.344777833999842\n",
      "INFO flwr 2023-05-17 11:44:54,993 | app.py:202 | app_fit: losses_distributed [(1, 3.4436863587053432)]\n",
      "INFO flwr 2023-05-17 11:44:54,994 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3058080755723487)], 'reward': [(1, -0.13787828313299805)], 'global_regret': [(1, 3.4436863587053437)], 'local_regrets': [(1, {0: 3.462805438006301, 1: 3.4379627800941117, 2: 3.46687522399552, 3: 3.407101992725442})]}\n",
      "INFO flwr 2023-05-17 11:44:54,995 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:44:54,996 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:44:55,001 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:45:02,440 | app.py:179 | Flower VCE: Ray initialized with resources: {'object_store_memory': 2147483648.0, 'memory': 6741982413.0, 'node:127.0.0.1': 1.0, 'CPU': 8.0}\n",
      "INFO flwr 2023-05-17 11:45:02,442 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:45:02,443 | server.py:270 | Requesting initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:45:03,402 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:45:03,403 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:45:03,404 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:45:03,405 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=98904)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98904)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98904)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:45:04,512 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:45:04,533 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:45:04,535 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=98907)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98907)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98903)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98903)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98908)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98908)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98904)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98907)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98903)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98908)\u001b[0m [Client 0] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:45:05,574 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:45:05,575 | server.py:144 | FL finished in 2.1701704990000508\n",
      "INFO flwr 2023-05-17 11:45:05,576 | app.py:202 | app_fit: losses_distributed [(1, 3.6595610391124938)]\n",
      "INFO flwr 2023-05-17 11:45:05,577 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.305808075572349)], 'reward': [(1, -0.35375296354015456)], 'global_regret': [(1, 3.6595610391124938)], 'local_regrets': [(1, {0: 3.6310036343092755, 1: 3.714789243413611, 2: 3.6196310372693383, 3: 3.67282024145775})]}\n",
      "INFO flwr 2023-05-17 11:45:05,578 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:45:05,579 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:45:05,604 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:45:15,695 | app.py:179 | Flower VCE: Ray initialized with resources: {'object_store_memory': 2147483648.0, 'CPU': 8.0, 'node:127.0.0.1': 1.0, 'memory': 6714758349.0}\n",
      "INFO flwr 2023-05-17 11:45:15,697 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:45:15,699 | server.py:270 | Requesting initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:45:17,083 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:45:17,084 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:45:17,085 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:45:17,085 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=98930)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98930)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98930)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:45:18,333 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:45:18,353 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:45:18,355 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=98929)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98929)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98924)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98924)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98928)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98928)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98930)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98929)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98924)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98928)\u001b[0m [Client 3] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:45:19,729 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:45:19,730 | server.py:144 | FL finished in 2.6444048490000114\n",
      "INFO flwr 2023-05-17 11:45:19,732 | app.py:202 | app_fit: losses_distributed [(1, 4.020943219646165)]\n",
      "INFO flwr 2023-05-17 11:45:19,733 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3142660853604817)], 'reward': [(1, -0.7027462751560896)], 'global_regret': [(1, 4.017012360516562)], 'local_regrets': [(1, {0: 4.044528374423788, 1: 3.9797783090290912, 2: 4.031414025106607, 3: 4.028052170025175})]}\n",
      "INFO flwr 2023-05-17 11:45:19,734 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:45:19,736 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:45:19,744 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:45:29,264 | app.py:179 | Flower VCE: Ray initialized with resources: {'memory': 6709409383.0, 'object_store_memory': 2147483648.0, 'node:127.0.0.1': 1.0, 'CPU': 8.0}\n",
      "INFO flwr 2023-05-17 11:45:29,266 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:45:29,267 | server.py:270 | Requesting initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:45:30,490 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:45:30,491 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:45:30,491 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:45:30,492 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=98953)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98953)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98953)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:45:31,854 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:45:31,877 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:45:31,879 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=98949)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98949)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98952)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98952)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98954)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98954)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98953)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98949)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98952)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98954)\u001b[0m [Client 0] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:45:33,685 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:45:33,686 | server.py:144 | FL finished in 3.1934855409999727\n",
      "INFO flwr 2023-05-17 11:45:33,690 | app.py:202 | app_fit: losses_distributed [(1, 3.128356638979757)]\n",
      "INFO flwr 2023-05-17 11:45:33,692 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.2948380560395987)], 'reward': [(1, 0.1495313505288263)], 'global_regret': [(1, 3.14530670551077)], 'local_regrets': [(1, {0: 3.026656239793677, 1: 3.1427002867620586, 2: 3.1164315997179646, 3: 3.227638429645327})]}\n",
      "INFO flwr 2023-05-17 11:45:33,695 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:45:33,697 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:45:33,724 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:45:45,748 | app.py:179 | Flower VCE: Ray initialized with resources: {'CPU': 8.0, 'memory': 6597545575.0, 'object_store_memory': 2147483648.0, 'node:127.0.0.1': 1.0}\n",
      "INFO flwr 2023-05-17 11:45:45,752 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:45:45,754 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=98975)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:45:47,843 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:45:47,845 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:45:47,846 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:45:47,848 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=98975)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98975)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98972)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98972)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98974)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98974)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98977)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98977)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:45:50,345 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:45:50,377 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:45:50,380 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98974)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98975)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98972)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98977)\u001b[0m [Client 2] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:45:53,252 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:45:53,254 | server.py:144 | FL finished in 5.406595905000131\n",
      "INFO flwr 2023-05-17 11:45:53,257 | app.py:202 | app_fit: losses_distributed [(1, 2.4874046141636352)]\n",
      "INFO flwr 2023-05-17 11:45:53,260 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.30957100068308)], 'reward': [(1, 0.8264239371668147)], 'global_regret': [(1, 2.4831470635162627)], 'local_regrets': [(1, {0: 2.5129499180478723, 1: 2.48075180991252, 2: 2.473694559138322, 3: 2.4822221695558264})]}\n",
      "INFO flwr 2023-05-17 11:45:53,262 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:45:53,265 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:45:53,337 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:46:07,007 | app.py:179 | Flower VCE: Ray initialized with resources: {'node:127.0.0.1': 1.0, 'CPU': 8.0, 'object_store_memory': 2147483648.0, 'memory': 6679999284.0}\n",
      "INFO flwr 2023-05-17 11:46:07,017 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:46:07,021 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=98996)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:46:14,856 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:46:14,858 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:46:14,861 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:46:14,874 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=98996)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98996)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98999)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98999)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99001)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99001)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98997)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=98997)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:46:57,109 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:46:57,698 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:46:57,763 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98999)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98996)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99001)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=98997)\u001b[0m [Client 0] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:47:41,547 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:47:41,552 | server.py:144 | FL finished in 86.68699913700016\n",
      "INFO flwr 2023-05-17 11:47:41,558 | app.py:202 | app_fit: losses_distributed [(1, 1.071964161822634)]\n",
      "INFO flwr 2023-05-17 11:47:41,573 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.289353046273224)], 'reward': [(1, 2.213350196553291)], 'global_regret': [(1, 1.076002849719935)], 'local_regrets': [(1, {0: 1.0558094102334317, 1: 1.0616008845914464, 2: 1.1070727522715489, 3: 1.0633736001941103})]}\n",
      "INFO flwr 2023-05-17 11:47:41,577 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:47:41,582 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:47:41,632 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:48:02,564 | app.py:179 | Flower VCE: Ray initialized with resources: {'CPU': 8.0, 'object_store_memory': 2147483648.0, 'node:127.0.0.1': 1.0, 'memory': 6745831015.0}\n",
      "INFO flwr 2023-05-17 11:48:02,570 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:48:02,574 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99043)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:48:08,172 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:48:08,177 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:48:08,180 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:48:08,184 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99043)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99043)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99046)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99046)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99045)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99045)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99044)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99044)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:48:14,354 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:48:14,421 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:48:14,426 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99043)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99046)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99045)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99044)\u001b[0m [Client 0] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:48:20,012 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:48:20,014 | server.py:144 | FL finished in 11.830181181000171\n",
      "INFO flwr 2023-05-17 11:48:20,017 | app.py:202 | app_fit: losses_distributed [(1, 1.6559138297336955)]\n",
      "INFO flwr 2023-05-17 11:48:20,020 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.2893530462732246)], 'reward': [(1, 1.6375346582279602)], 'global_regret': [(1, 1.6518183880452568)], 'local_regrets': [(1, {0: 1.6722955964874497, 1: 1.6202295656055299, 2: 1.6760431333939902, 3: 1.655087023447812})]}\n",
      "INFO flwr 2023-05-17 11:48:20,023 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:48:20,027 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:48:20,051 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:48:33,256 | app.py:179 | Flower VCE: Ray initialized with resources: {'CPU': 8.0, 'node:127.0.0.1': 1.0, 'memory': 6693178164.0, 'object_store_memory': 2147483648.0}\n",
      "INFO flwr 2023-05-17 11:48:33,260 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:48:33,262 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99078)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:48:35,750 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:48:35,751 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:48:35,753 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:48:35,755 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99078)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99078)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99077)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99077)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99080)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99080)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99079)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99079)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:48:38,602 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:48:38,636 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:48:38,639 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99078)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99077)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99080)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99079)\u001b[0m [Client 0] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:48:41,530 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:48:41,532 | server.py:144 | FL finished in 5.776584402999788\n",
      "INFO flwr 2023-05-17 11:48:41,536 | app.py:202 | app_fit: losses_distributed [(1, 1.261784036269912)]\n",
      "INFO flwr 2023-05-17 11:48:41,539 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3184950902545487)], 'reward': [(1, 2.0560540540071326)], 'global_regret': [(1, 1.2624410362474163)], 'local_regrets': [(1, {0: 1.259156036359894, 1: 1.2842479359831067, 2: 1.266271459424496, 3: 1.237460713312151})]}\n",
      "INFO flwr 2023-05-17 11:48:41,541 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:48:41,544 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:48:41,663 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:48:53,834 | app.py:179 | Flower VCE: Ray initialized with resources: {'object_store_memory': 2147483648.0, 'node:127.0.0.1': 1.0, 'memory': 6753122714.0, 'CPU': 8.0}\n",
      "INFO flwr 2023-05-17 11:48:53,837 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:48:53,839 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99104)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:48:55,856 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:48:55,857 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:48:55,858 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:48:55,859 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99104)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99104)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99100)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99100)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99101)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99101)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99098)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99098)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:48:58,152 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:48:58,190 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:48:58,193 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99104)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99100)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99101)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99098)\u001b[0m [Client 2] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:49:01,457 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:49:01,459 | server.py:144 | FL finished in 5.599237139000024\n",
      "INFO flwr 2023-05-17 11:49:01,462 | app.py:202 | app_fit: losses_distributed [(1, 0.8665967172489704)]\n",
      "INFO flwr 2023-05-17 11:49:01,465 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.2886768121924375)], 'reward': [(1, 2.419426797293131)], 'global_regret': [(1, 0.869250014899301)], 'local_regrets': [(1, {0: 0.8564024683819105, 1: 0.8738695476276165, 2: 0.8592823326670576, 3: 0.8768325203192971})]}\n",
      "INFO flwr 2023-05-17 11:49:01,467 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:49:01,469 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:49:01,486 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:49:12,532 | app.py:179 | Flower VCE: Ray initialized with resources: {'memory': 6729592423.0, 'node:127.0.0.1': 1.0, 'object_store_memory': 2147483648.0, 'CPU': 8.0}\n",
      "INFO flwr 2023-05-17 11:49:12,537 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:49:12,539 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99125)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:49:14,975 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:49:14,977 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:49:14,979 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:49:14,981 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99125)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99125)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99124)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99124)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99122)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99122)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99127)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99127)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:49:17,501 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:49:17,538 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:49:17,542 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99125)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99124)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99122)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99127)\u001b[0m [Client 3] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:49:21,284 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:49:21,287 | server.py:144 | FL finished in 6.3062267159998555\n",
      "INFO flwr 2023-05-17 11:49:21,292 | app.py:202 | app_fit: losses_distributed [(1, 0.8623517688797925)]\n",
      "INFO flwr 2023-05-17 11:49:21,298 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.2886768121924375)], 'reward': [(1, 2.4312931518582808)], 'global_regret': [(1, 0.8573836603341523)], 'local_regrets': [(1, {0: 0.8814397648709359, 1: 0.8506830359603129, 2: 0.872711984752084, 3: 0.844572289935837})]}\n",
      "INFO flwr 2023-05-17 11:49:21,303 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:49:21,309 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:49:21,368 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:49:34,580 | app.py:179 | Flower VCE: Ray initialized with resources: {'object_store_memory': 2147483648.0, 'node:127.0.0.1': 1.0, 'CPU': 8.0, 'memory': 6743866164.0}\n",
      "INFO flwr 2023-05-17 11:49:34,586 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:49:34,590 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99148)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:49:38,423 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:49:38,425 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:49:38,427 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:49:38,429 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99148)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99148)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99145)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99145)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99146)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99146)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99147)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99147)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:49:42,989 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:49:43,061 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:49:43,065 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99148)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99145)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99146)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99147)\u001b[0m [Client 3] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:49:53,039 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:49:53,043 | server.py:144 | FL finished in 14.613213982999696\n",
      "INFO flwr 2023-05-17 11:49:53,048 | app.py:202 | app_fit: losses_distributed [(1, 1.2717627081713057)]\n",
      "INFO flwr 2023-05-17 11:49:53,051 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3038545913019783)], 'reward': [(1, 2.0310441184398433)], 'global_regret': [(1, 1.2728104728621321)], 'local_regrets': [(1, {0: 1.2677370859381298, 1: 1.2347860332509724, 2: 1.287167989310278, 3: 1.2973597241858423})]}\n",
      "INFO flwr 2023-05-17 11:49:53,054 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:49:53,063 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:49:53,210 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:50:13,734 | app.py:179 | Flower VCE: Ray initialized with resources: {'CPU': 8.0, 'node:127.0.0.1': 1.0, 'memory': 6622930125.0, 'object_store_memory': 2147483648.0}\n",
      "INFO flwr 2023-05-17 11:50:13,742 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:50:13,755 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99173)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:50:20,665 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:50:20,668 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:50:20,670 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:50:20,673 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99173)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99173)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99168)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99168)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99172)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99172)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99167)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99167)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:50:28,212 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:50:28,299 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:50:28,306 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99173)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99168)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99172)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99167)\u001b[0m [Client 1] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:50:37,419 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:50:37,428 | server.py:144 | FL finished in 16.755294107999816\n",
      "INFO flwr 2023-05-17 11:50:37,434 | app.py:202 | app_fit: losses_distributed [(1, 0.7203535503038875)]\n",
      "INFO flwr 2023-05-17 11:50:37,438 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.303706537757277)], 'reward': [(1, 2.5842228600388664)], 'global_regret': [(1, 0.719483677718409)], 'local_regrets': [(1, {0: 0.7234602381091674, 1: 0.7177535490332422, 2: 0.7065780639473205, 3: 0.7336223501258197})]}\n",
      "INFO flwr 2023-05-17 11:50:37,444 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:50:37,447 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:50:37,477 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:50:58,810 | app.py:179 | Flower VCE: Ray initialized with resources: {'object_store_memory': 2147483648.0, 'CPU': 8.0, 'memory': 6624430490.0, 'node:127.0.0.1': 1.0}\n",
      "INFO flwr 2023-05-17 11:50:58,823 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:50:58,827 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99207)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:51:05,655 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:51:05,657 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:51:05,659 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:51:05,674 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99207)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99207)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99201)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99201)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99203)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99203)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99200)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99200)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:51:13,059 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:51:13,151 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:51:13,157 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99207)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99201)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99203)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99200)\u001b[0m [Client 3] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:51:22,955 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:51:22,958 | server.py:144 | FL finished in 17.284396453000227\n",
      "INFO flwr 2023-05-17 11:51:22,971 | app.py:202 | app_fit: losses_distributed [(1, 1.6227332675693809)]\n",
      "INFO flwr 2023-05-17 11:51:22,974 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.312129789758377)], 'reward': [(1, 1.6848206185076315)], 'global_regret': [(1, 1.6273091712507421)], 'local_regrets': [(1, {0: 1.6063907544216611, 1: 1.6193977086208529, 2: 1.6129140842877945, 3: 1.6522305229472145})]}\n",
      "INFO flwr 2023-05-17 11:51:22,977 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:51:22,981 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:51:23,009 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:51:39,309 | app.py:179 | Flower VCE: Ray initialized with resources: {'object_store_memory': 2147483648.0, 'node:127.0.0.1': 1.0, 'memory': 6779745895.0, 'CPU': 8.0}\n",
      "INFO flwr 2023-05-17 11:51:39,313 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:51:39,316 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99227)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:51:42,286 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:51:42,288 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:51:42,290 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:51:42,292 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99227)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99227)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99232)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99232)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99233)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99233)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99231)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99231)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:51:45,945 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:51:46,001 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:51:46,006 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99227)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99232)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99233)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99231)\u001b[0m [Client 0] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:51:52,295 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:51:52,297 | server.py:144 | FL finished in 10.005451711000205\n",
      "INFO flwr 2023-05-17 11:51:52,303 | app.py:202 | app_fit: losses_distributed [(1, 1.1182890600173243)]\n",
      "INFO flwr 2023-05-17 11:51:52,309 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.2873784427573294)], 'reward': [(1, 2.17603565550347)], 'global_regret': [(1, 1.1113427872538577)], 'local_regrets': [(1, {0: 1.1430971770297054, 1: 1.1116150885489053, 2: 1.099855191786197, 3: 1.1185887827044896})]}\n",
      "INFO flwr 2023-05-17 11:51:52,312 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:51:52,315 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:51:52,432 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:52:10,739 | app.py:179 | Flower VCE: Ray initialized with resources: {'CPU': 8.0, 'memory': 6695146701.0, 'object_store_memory': 2147483648.0, 'node:127.0.0.1': 1.0}\n",
      "INFO flwr 2023-05-17 11:52:10,748 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:52:10,752 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99255)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:52:14,809 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:52:14,811 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:52:14,814 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:52:14,816 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99255)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99255)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99257)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99257)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99260)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99260)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99252)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99252)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:52:21,418 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:52:21,482 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:52:21,490 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99255)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99252)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99257)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99260)\u001b[0m [Client 1] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:52:30,503 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:52:30,505 | server.py:144 | FL finished in 15.688753103000181\n",
      "INFO flwr 2023-05-17 11:52:30,508 | app.py:202 | app_fit: losses_distributed [(1, 0.32409089053027845)]\n",
      "INFO flwr 2023-05-17 11:52:30,511 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3127270023888538)], 'reward': [(1, 2.987846133867031)], 'global_regret': [(1, 0.3248808685218197)], 'local_regrets': [(1, {0: 0.3215130676105122, 1: 0.331711987320899, 2: 0.3233130468861277, 3: 0.3198254603035748})]}\n",
      "INFO flwr 2023-05-17 11:52:30,515 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:52:30,518 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:52:30,561 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:52:48,171 | app.py:179 | Flower VCE: Ray initialized with resources: {'memory': 6768130048.0, 'node:127.0.0.1': 1.0, 'object_store_memory': 2147483648.0, 'CPU': 8.0}\n",
      "INFO flwr 2023-05-17 11:52:48,178 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:52:48,185 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99283)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:52:51,812 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:52:51,814 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:52:51,815 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:52:51,820 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99283)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99283)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99281)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99281)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99285)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99285)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99279)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99279)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:52:56,788 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:52:56,838 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:52:56,843 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99283)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99281)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99285)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99279)\u001b[0m [Client 2] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:53:05,554 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:53:05,557 | server.py:144 | FL finished in 13.737145155000235\n",
      "INFO flwr 2023-05-17 11:53:05,562 | app.py:202 | app_fit: losses_distributed [(1, 0.6463099481025997)]\n",
      "INFO flwr 2023-05-17 11:53:05,566 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3035080053830423)], 'reward': [(1, 2.6587402513173988)], 'global_regret': [(1, 0.6447677540656414)], 'local_regrets': [(1, {0: 0.6513423707495161, 1: 0.6125424784433224, 2: 0.6781786450850669, 3: 0.6431762981324933})]}\n",
      "INFO flwr 2023-05-17 11:53:05,569 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:53:05,581 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:53:05,647 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:53:19,722 | app.py:179 | Flower VCE: Ray initialized with resources: {'memory': 6668453479.0, 'node:127.0.0.1': 1.0, 'CPU': 8.0, 'object_store_memory': 2147483648.0}\n",
      "INFO flwr 2023-05-17 11:53:19,730 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:53:19,733 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99305)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:53:23,078 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:53:23,079 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:53:23,081 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:53:23,084 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99305)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99305)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99309)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99309)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99303)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99303)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99304)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99304)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:53:26,508 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:53:26,563 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:53:26,567 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99305)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99309)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99303)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99304)\u001b[0m [Client 1] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:53:32,762 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:53:32,765 | server.py:144 | FL finished in 9.680776415999844\n",
      "INFO flwr 2023-05-17 11:53:32,769 | app.py:202 | app_fit: losses_distributed [(1, 0.4182588888854754)]\n",
      "INFO flwr 2023-05-17 11:53:32,774 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3035080053830423)], 'reward': [(1, 2.886939859212006)], 'global_regret': [(1, 0.4165681461710302)], 'local_regrets': [(1, {0: 0.42377604932208585, 1: 0.4016889689581515, 2: 0.42939236598835423, 3: 0.4181781712733099})]}\n",
      "INFO flwr 2023-05-17 11:53:32,776 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:53:32,779 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:53:33,032 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:53:48,390 | app.py:179 | Flower VCE: Ray initialized with resources: {'memory': 6670495744.0, 'CPU': 8.0, 'node:127.0.0.1': 1.0, 'object_store_memory': 2147483648.0}\n",
      "INFO flwr 2023-05-17 11:53:48,396 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:53:48,399 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99336)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:53:51,314 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:53:51,315 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:53:51,318 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:53:51,321 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99336)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99336)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99335)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99335)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99338)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99338)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99334)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99334)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:53:54,766 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:53:54,813 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:53:54,817 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99336)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99335)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99338)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99334)\u001b[0m [Client 0] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:53:59,530 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:53:59,532 | server.py:144 | FL finished in 8.211493687999791\n",
      "INFO flwr 2023-05-17 11:53:59,535 | app.py:202 | app_fit: losses_distributed [(1, 0.15440944058863076)]\n",
      "INFO flwr 2023-05-17 11:53:59,538 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3129712342168722)], 'reward': [(1, 3.1576567979679364)], 'global_regret': [(1, 0.15531443624893335)], 'local_regrets': [(1, {0: 0.1515569859124872, 1: 0.15494368519969046, 2: 0.15244037545012945, 3: 0.15869671579221598})]}\n",
      "INFO flwr 2023-05-17 11:53:59,541 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:53:59,545 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:53:59,621 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:54:14,124 | app.py:179 | Flower VCE: Ray initialized with resources: {'memory': 6647872308.0, 'node:127.0.0.1': 1.0, 'CPU': 8.0, 'object_store_memory': 2147483648.0}\n",
      "INFO flwr 2023-05-17 11:54:14,128 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:54:14,131 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99361)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:54:16,927 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:54:16,929 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:54:16,931 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:54:16,933 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99361)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99361)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99358)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99358)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99356)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99356)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99357)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99357)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:54:20,040 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:54:20,089 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:54:20,094 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99361)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99358)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99356)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99357)\u001b[0m [Client 0] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:54:24,305 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:54:24,307 | server.py:144 | FL finished in 7.373996693999743\n",
      "INFO flwr 2023-05-17 11:54:24,310 | app.py:202 | app_fit: losses_distributed [(1, 0.2643989852346994)]\n",
      "INFO flwr 2023-05-17 11:54:24,313 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3034268149958903)], 'reward': [(1, 3.040111706316123)], 'global_regret': [(1, 0.26331510867976093)], 'local_regrets': [(1, {0: 0.2678152543762141, 1: 0.2755821301933197, 2: 0.2540360965555188, 3: 0.26016245981374486})]}\n",
      "INFO flwr 2023-05-17 11:54:24,316 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:54:24,319 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:54:24,352 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:54:37,152 | app.py:179 | Flower VCE: Ray initialized with resources: {'node:127.0.0.1': 1.0, 'CPU': 8.0, 'memory': 6709158708.0, 'object_store_memory': 2147483648.0}\n",
      "INFO flwr 2023-05-17 11:54:37,156 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:54:37,159 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99380)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:54:39,640 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:54:39,641 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:54:39,643 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:54:39,645 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99380)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99380)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99381)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99381)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99383)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99383)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99378)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99378)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:54:42,901 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:54:42,955 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:54:42,958 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99380)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99381)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99383)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99378)\u001b[0m [Client 2] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:54:49,697 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:54:49,699 | server.py:144 | FL finished in 10.054009329999644\n",
      "INFO flwr 2023-05-17 11:54:49,702 | app.py:202 | app_fit: losses_distributed [(1, 0.17309272916364488)]\n",
      "INFO flwr 2023-05-17 11:54:49,705 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3034268149958903)], 'reward': [(1, 3.1316975704184373)], 'global_regret': [(1, 0.17172924457744998)], 'local_regrets': [(1, {0: 0.17739029450494287, 1: 0.16328471997984578, 2: 0.17686643880239966, 3: 0.17482946336739136})]}\n",
      "INFO flwr 2023-05-17 11:54:49,707 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:54:49,709 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:54:49,834 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:55:02,774 | app.py:179 | Flower VCE: Ray initialized with resources: {'CPU': 8.0, 'node:127.0.0.1': 1.0, 'memory': 6673120461.0, 'object_store_memory': 2147483648.0}\n",
      "INFO flwr 2023-05-17 11:55:02,778 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:55:02,779 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99407)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:55:05,262 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:55:05,264 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:55:05,266 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:55:05,268 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99407)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99407)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99405)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99405)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99408)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99408)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99406)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99406)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:55:08,564 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:55:08,612 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:55:08,616 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99407)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99405)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99408)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99406)\u001b[0m [Client 2] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:55:14,487 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:55:14,490 | server.py:144 | FL finished in 9.222383071999957\n",
      "INFO flwr 2023-05-17 11:55:14,493 | app.py:202 | app_fit: losses_distributed [(1, 0.1643360313152737)]\n",
      "INFO flwr 2023-05-17 11:55:14,496 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.322182782522175)], 'reward': [(1, 3.1566491583807847)], 'global_regret': [(1, 0.16553362414138634)], 'local_regrets': [(1, {0: 0.16062448330046172, 1: 0.16564821570042937, 2: 0.16149835815544794, 3: 0.16957306810475573})]}\n",
      "INFO flwr 2023-05-17 11:55:14,498 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:55:14,501 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:55:14,542 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:55:29,089 | app.py:179 | Flower VCE: Ray initialized with resources: {'CPU': 8.0, 'object_store_memory': 2147483648.0, 'node:127.0.0.1': 1.0, 'memory': 6655982388.0}\n",
      "INFO flwr 2023-05-17 11:55:29,093 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:55:29,096 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99433)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:55:32,146 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:55:32,149 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:55:32,152 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:55:32,155 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99433)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99433)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99432)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99432)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99431)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99431)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99434)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99434)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:55:37,229 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:55:37,285 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:55:37,290 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99433)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99432)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99431)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99434)\u001b[0m [Client 2] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:55:43,984 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:55:43,987 | server.py:144 | FL finished in 11.832528581999668\n",
      "INFO flwr 2023-05-17 11:55:43,993 | app.py:202 | app_fit: losses_distributed [(1, 0.326094431758385)]\n",
      "INFO flwr 2023-05-17 11:55:43,997 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3033863034235518)], 'reward': [(1, 2.9759700455592424)], 'global_regret': [(1, 0.32741625786430784)], 'local_regrets': [(1, {0: 0.3219978632482935, 1: 0.3309613842986723, 2: 0.32182577849266275, 3: 0.3295927009939114})]}\n",
      "INFO flwr 2023-05-17 11:55:44,000 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:55:44,003 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:55:44,041 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:55:59,475 | app.py:179 | Flower VCE: Ray initialized with resources: {'node:127.0.0.1': 1.0, 'CPU': 8.0, 'object_store_memory': 2147483648.0, 'memory': 6632584807.0}\n",
      "INFO flwr 2023-05-17 11:55:59,481 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:55:59,484 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99478)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:56:02,993 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:56:02,995 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:56:02,997 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:56:02,999 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99478)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99478)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99477)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99477)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99475)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99475)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99471)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99471)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:56:07,486 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:56:07,550 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:56:07,557 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99478)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99477)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99475)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99471)\u001b[0m [Client 2] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:56:13,726 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:56:13,729 | server.py:144 | FL finished in 10.729140967000149\n",
      "INFO flwr 2023-05-17 11:56:13,731 | app.py:202 | app_fit: losses_distributed [(1, 0.1492435483972911)]\n",
      "INFO flwr 2023-05-17 11:56:13,735 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3033863034235518)], 'reward': [(1, 3.1540247730080857)], 'global_regret': [(1, 0.1493615304154646)], 'local_regrets': [(1, {0: 0.14887790164675338, 1: 0.1510441999411569, 2: 0.15232024562658886, 3: 0.1447318463746652})]}\n",
      "INFO flwr 2023-05-17 11:56:13,738 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:56:13,741 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:56:13,987 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:56:28,886 | app.py:179 | Flower VCE: Ray initialized with resources: {'CPU': 8.0, 'memory': 6653733684.0, 'object_store_memory': 2147483648.0, 'node:127.0.0.1': 1.0}\n",
      "INFO flwr 2023-05-17 11:56:28,893 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:56:28,897 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99497)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:56:31,745 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:56:31,746 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:56:31,748 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:56:31,753 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99497)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99497)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99498)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99498)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99499)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99499)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99501)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99501)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:56:34,946 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:56:34,999 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:56:35,003 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99497)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99498)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99499)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99501)\u001b[0m [Client 1] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:56:39,052 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:56:39,054 | server.py:144 | FL finished in 7.300786467999842\n",
      "INFO flwr 2023-05-17 11:56:39,057 | app.py:202 | app_fit: losses_distributed [(1, 0.10769668039882466)]\n",
      "INFO flwr 2023-05-17 11:56:39,059 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.284395124532447)], 'reward': [(1, 3.1754355755371515)], 'global_regret': [(1, 0.10895954899529095)], 'local_regrets': [(1, {0: 0.10381481662076371, 1: 0.10850994002125697, 2: 0.10857724069186436, 3: 0.10988472426141362})]}\n",
      "INFO flwr 2023-05-17 11:56:39,062 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:56:39,065 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:56:39,097 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:56:57,117 | app.py:179 | Flower VCE: Ray initialized with resources: {'CPU': 8.0, 'object_store_memory': 2147483648.0, 'node:127.0.0.1': 1.0, 'memory': 6609806541.0}\n",
      "INFO flwr 2023-05-17 11:56:57,122 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:56:57,132 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99527)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:57:02,274 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:57:02,276 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:57:02,279 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:57:02,282 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99527)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99527)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99524)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99524)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99525)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99525)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99522)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99522)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:57:07,332 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:57:07,395 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:57:07,399 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99527)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99524)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99525)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99522)\u001b[0m [Client 2] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:57:12,754 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:57:12,759 | server.py:144 | FL finished in 10.477550545999748\n",
      "INFO flwr 2023-05-17 11:57:12,762 | app.py:202 | app_fit: losses_distributed [(1, 0.13473771096256354)]\n",
      "INFO flwr 2023-05-17 11:57:12,765 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3033663488817355)], 'reward': [(1, 3.16837416624022)], 'global_regret': [(1, 0.13499218264151353)], 'local_regrets': [(1, {0: 0.1339555041709605, 1: 0.13502768284758623, 2: 0.13667675389940728, 3: 0.1332909029323002})]}\n",
      "INFO flwr 2023-05-17 11:57:12,769 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:57:12,772 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:57:12,810 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:57:26,339 | app.py:179 | Flower VCE: Ray initialized with resources: {'object_store_memory': 2147483648.0, 'CPU': 8.0, 'node:127.0.0.1': 1.0, 'memory': 6661537792.0}\n",
      "INFO flwr 2023-05-17 11:57:26,342 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:57:26,345 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99546)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:57:29,044 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:57:29,046 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:57:29,048 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:57:29,050 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99546)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99546)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99548)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99548)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99551)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99551)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99552)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99552)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:57:32,247 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:57:32,298 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:57:32,303 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99546)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99548)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99551)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99552)\u001b[0m [Client 3] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:57:36,522 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:57:36,524 | server.py:144 | FL finished in 7.4742969839999205\n",
      "INFO flwr 2023-05-17 11:57:36,529 | app.py:202 | app_fit: losses_distributed [(1, 0.14440593042433988)]\n",
      "INFO flwr 2023-05-17 11:57:36,533 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.2843951245324465)], 'reward': [(1, 3.1402499087798414)], 'global_regret': [(1, 0.1441452157526024)], 'local_regrets': [(1, {0: 0.14520732721531152, 1: 0.14761104106960982, 2: 0.13921868706795243, 3: 0.1455866663444858})]}\n",
      "INFO flwr 2023-05-17 11:57:36,536 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:57:36,539 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:57:36,775 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:57:51,141 | app.py:179 | Flower VCE: Ray initialized with resources: {'CPU': 8.0, 'object_store_memory': 2147483648.0, 'node:127.0.0.1': 1.0, 'memory': 6670879130.0}\n",
      "INFO flwr 2023-05-17 11:57:51,147 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:57:51,149 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99572)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:57:54,308 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:57:54,311 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:57:54,313 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:57:54,316 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99572)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99572)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99570)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99570)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99573)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99573)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99576)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99576)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:57:57,539 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:57:57,587 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:57:57,591 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99572)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99570)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99573)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99576)\u001b[0m [Client 1] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:58:01,041 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:58:01,043 | server.py:144 | FL finished in 6.727562224999929\n",
      "INFO flwr 2023-05-17 11:58:01,047 | app.py:202 | app_fit: losses_distributed [(1, 0.06586689125349979)]\n",
      "INFO flwr 2023-05-17 11:58:01,049 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.322534635729361)], 'reward': [(1, 3.2567130624533958)], 'global_regret': [(1, 0.06582157327596089)], 'local_regrets': [(1, {0: 0.06600438511739205, 1: 0.061756198669626804, 2: 0.06661881269358866, 3: 0.06908816853339164})]}\n",
      "INFO flwr 2023-05-17 11:58:01,053 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:58:01,057 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:58:01,095 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:58:13,378 | app.py:179 | Flower VCE: Ray initialized with resources: {'object_store_memory': 2147483648.0, 'memory': 6671859712.0, 'node:127.0.0.1': 1.0, 'CPU': 8.0}\n",
      "INFO flwr 2023-05-17 11:58:13,381 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:58:13,383 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99600)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:58:15,410 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:58:15,412 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:58:15,413 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:58:15,415 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99600)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99600)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99598)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99598)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99597)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99597)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99599)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99599)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:58:17,787 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:58:17,826 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:58:17,830 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99600)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99598)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99597)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99599)\u001b[0m [Client 2] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:58:21,420 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:58:21,422 | server.py:144 | FL finished in 6.007057064000037\n",
      "INFO flwr 2023-05-17 11:58:21,425 | app.py:202 | app_fit: losses_distributed [(1, 0.023079040909357335)]\n",
      "INFO flwr 2023-05-17 11:58:21,427 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.284113764944382)], 'reward': [(1, 3.2620391604001258)], 'global_regret': [(1, 0.022074604544254917)], 'local_regrets': [(1, {0: 0.026126481337459324, 1: 0.023069230592819345, 2: 0.023118808396091567, 3: 0.0200016433110591})]}\n",
      "INFO flwr 2023-05-17 11:58:21,429 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:58:21,431 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:58:21,461 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:58:34,716 | app.py:179 | Flower VCE: Ray initialized with resources: {'memory': 6693587354.0, 'node:127.0.0.1': 1.0, 'CPU': 8.0, 'object_store_memory': 2147483648.0}\n",
      "INFO flwr 2023-05-17 11:58:34,720 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:58:34,722 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99623)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:58:37,949 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:58:37,951 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:58:37,953 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:58:37,956 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99623)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99623)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99619)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99619)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99620)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99620)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99622)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99622)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:58:41,240 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:58:41,287 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:58:41,291 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99623)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99620)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99619)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99622)\u001b[0m [Client 1] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:58:45,904 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:58:45,907 | server.py:144 | FL finished in 7.951088132000223\n",
      "INFO flwr 2023-05-17 11:58:45,912 | app.py:202 | app_fit: losses_distributed [(1, 0.06702963071153227)]\n",
      "INFO flwr 2023-05-17 11:58:45,927 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.284113764944382)], 'reward': [(1, 3.2170711104624203)], 'global_regret': [(1, 0.06704265448196124)], 'local_regrets': [(1, {0: 0.06699011684493952, 1: 0.06557330984864453, 2: 0.061959480659103484, 3: 0.07359561549344157})]}\n",
      "INFO flwr 2023-05-17 11:58:45,933 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:58:45,941 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:58:46,205 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:58:59,432 | app.py:179 | Flower VCE: Ray initialized with resources: {'CPU': 8.0, 'node:127.0.0.1': 1.0, 'memory': 6719871386.0, 'object_store_memory': 2147483648.0}\n",
      "INFO flwr 2023-05-17 11:58:59,435 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:58:59,437 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99646)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:59:01,854 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:59:01,855 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:59:01,858 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:59:01,861 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99646)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99646)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99641)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99641)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99647)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99647)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99648)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99648)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:59:04,653 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:59:04,694 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:59:04,698 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99646)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99641)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99647)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99648)\u001b[0m [Client 1] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:59:07,932 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:59:07,934 | server.py:144 | FL finished in 6.073429708999811\n",
      "INFO flwr 2023-05-17 11:59:07,937 | app.py:202 | app_fit: losses_distributed [(1, 0.02773867427991163)]\n",
      "INFO flwr 2023-05-17 11:59:07,941 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3033222565568066)], 'reward': [(1, 3.275974720971343)], 'global_regret': [(1, 0.027347535585461706)], 'local_regrets': [(1, {0: 0.028919648598902944, 1: 0.027673010434632595, 2: 0.029173167745256228, 3: 0.025188870340854753})]}\n",
      "INFO flwr 2023-05-17 11:59:07,943 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:59:07,945 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:59:08,065 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:59:20,433 | app.py:179 | Flower VCE: Ray initialized with resources: {'memory': 6769914266.0, 'CPU': 8.0, 'object_store_memory': 2147483648.0, 'node:127.0.0.1': 1.0}\n",
      "INFO flwr 2023-05-17 11:59:20,437 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:59:20,438 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99666)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:59:22,437 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:59:22,438 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:59:22,440 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:59:22,442 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99666)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99666)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99669)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99669)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99670)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99670)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99671)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99671)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:59:24,846 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:59:24,881 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:59:24,885 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99666)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99671)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99669)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99670)\u001b[0m [Client 3] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:59:27,587 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:59:27,589 | server.py:144 | FL finished in 5.147093997999946\n",
      "INFO flwr 2023-05-17 11:59:27,592 | app.py:202 | app_fit: losses_distributed [(1, 0.0386248442212675)]\n",
      "INFO flwr 2023-05-17 11:59:27,593 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.2840084527568676)], 'reward': [(1, 3.246395537243446)], 'global_regret': [(1, 0.03761291551342276)], 'local_regrets': [(1, {0: 0.04168018452273111, 1: 0.03837491440108637, 2: 0.037950147657145224, 3: 0.03649413030410733})]}\n",
      "INFO flwr 2023-05-17 11:59:27,595 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:59:27,597 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:59:27,639 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 11:59:39,384 | app.py:179 | Flower VCE: Ray initialized with resources: {'object_store_memory': 2147483648.0, 'memory': 6763632640.0, 'node:127.0.0.1': 1.0, 'CPU': 8.0}\n",
      "INFO flwr 2023-05-17 11:59:39,389 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 11:59:39,390 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99691)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 11:59:41,636 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 11:59:41,637 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 11:59:41,638 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 11:59:41,640 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99691)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99691)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99692)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99692)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99690)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99690)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99694)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99694)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:59:44,229 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 11:59:44,270 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 11:59:44,273 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99691)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99692)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99690)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99694)\u001b[0m [Client 0] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 11:59:47,265 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 11:59:47,267 | server.py:144 | FL finished in 5.627170911000121\n",
      "INFO flwr 2023-05-17 11:59:47,271 | app.py:202 | app_fit: losses_distributed [(1, 0.029154356060291944)]\n",
      "INFO flwr 2023-05-17 11:59:47,273 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3033222565568066)], 'reward': [(1, 3.274178774499708)], 'global_regret': [(1, 0.029143482057095965)], 'local_regrets': [(1, {0: 0.029187188195545496, 1: 0.0302667372283198, 2: 0.028132617097518988, 3: 0.029030881719783498})]}\n",
      "INFO flwr 2023-05-17 11:59:47,277 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 11:59:47,281 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 11:59:47,524 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 12:00:00,508 | app.py:179 | Flower VCE: Ray initialized with resources: {'node:127.0.0.1': 1.0, 'object_store_memory': 2147483648.0, 'memory': 6734970880.0, 'CPU': 8.0}\n",
      "INFO flwr 2023-05-17 12:00:00,514 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 12:00:00,516 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99716)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 12:00:03,157 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 12:00:03,159 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 12:00:03,161 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 12:00:03,163 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99716)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99716)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99714)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99714)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99712)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99712)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99718)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99718)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 12:00:06,838 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 12:00:06,887 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 12:00:06,893 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99716)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99714)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99712)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99718)\u001b[0m [Client 0] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 12:00:12,014 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 12:00:12,016 | server.py:144 | FL finished in 8.853800843000045\n",
      "INFO flwr 2023-05-17 12:00:12,020 | app.py:202 | app_fit: losses_distributed [(1, 0.018617603246858107)]\n",
      "INFO flwr 2023-05-17 12:00:12,023 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3033169207413646)], 'reward': [(1, 3.2850433585514476)], 'global_regret': [(1, 0.018273562189914758)], 'local_regrets': [(1, {0: 0.01965414960524226, 1: 0.017572631287109543, 2: 0.019737584367154386, 3: 0.017506047727926233})]}\n",
      "INFO flwr 2023-05-17 12:00:12,025 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 12:00:12,028 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 12:00:12,192 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 12:00:26,879 | app.py:179 | Flower VCE: Ray initialized with resources: {'node:127.0.0.1': 1.0, 'object_store_memory': 2147483648.0, 'CPU': 8.0, 'memory': 6752709837.0}\n",
      "INFO flwr 2023-05-17 12:00:26,884 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 12:00:26,888 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99743)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 12:00:30,032 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 12:00:30,034 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 12:00:30,036 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 12:00:30,040 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99743)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99743)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99740)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99740)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99742)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99742)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99737)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99737)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 12:00:34,074 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 12:00:34,124 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 12:00:34,128 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99743)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99740)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99742)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99737)\u001b[0m [Client 0] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 12:00:38,695 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 12:00:38,699 | server.py:144 | FL finished in 8.658819399000095\n",
      "INFO flwr 2023-05-17 12:00:38,704 | app.py:202 | app_fit: losses_distributed [(1, 0.010287116568964166)]\n",
      "INFO flwr 2023-05-17 12:00:38,707 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3033169207413646)], 'reward': [(1, 3.292964015031018)], 'global_regret': [(1, 0.010352905710343958)], 'local_regrets': [(1, {0: 0.010088903322195556, 1: 0.009777796821593876, 2: 0.011364879645875734, 3: 0.009916886486191495})]}\n",
      "INFO flwr 2023-05-17 12:00:38,709 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 12:00:38,711 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 12:00:38,918 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 12:00:55,645 | app.py:179 | Flower VCE: Ray initialized with resources: {'node:127.0.0.1': 1.0, 'object_store_memory': 2147483648.0, 'CPU': 8.0, 'memory': 6738244404.0}\n",
      "INFO flwr 2023-05-17 12:00:55,654 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 12:00:55,659 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99767)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 12:00:59,978 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 12:00:59,981 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 12:00:59,986 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 12:00:59,989 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99767)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99767)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99765)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99765)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99764)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99764)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99761)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99761)\u001b[0m [Client 3] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 12:01:05,402 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 12:01:05,460 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 12:01:05,465 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99767)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99761)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99764)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99765)\u001b[0m [Client 1] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 12:01:11,275 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 12:01:11,278 | server.py:144 | FL finished in 11.289191667000068\n",
      "INFO flwr 2023-05-17 12:01:11,284 | app.py:202 | app_fit: losses_distributed [(1, 0.020355137872098934)]\n",
      "INFO flwr 2023-05-17 12:01:11,287 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3033169207413646)], 'reward': [(1, 3.2819243960355537)], 'global_regret': [(1, 0.021392524705808012)], 'local_regrets': [(1, {0: 0.017229640135664873, 1: 0.023027371247555464, 2: 0.02056242025450549, 3: 0.020601119850669907})]}\n",
      "INFO flwr 2023-05-17 12:01:11,294 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 12:01:11,297 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 12:01:11,821 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 12:01:27,261 | app.py:179 | Flower VCE: Ray initialized with resources: {'object_store_memory': 2147483648.0, 'memory': 6744179508.0, 'CPU': 8.0, 'node:127.0.0.1': 1.0}\n",
      "INFO flwr 2023-05-17 12:01:27,266 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 12:01:27,269 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99785)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 12:01:30,933 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 12:01:30,936 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 12:01:30,938 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 12:01:30,942 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99785)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99785)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99788)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99788)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99789)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99789)\u001b[0m [Client 2] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99791)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99791)\u001b[0m [Client 1] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 12:01:35,892 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 12:01:35,955 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 12:01:35,961 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99785)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99789)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99791)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99788)\u001b[0m [Client 2] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 12:01:41,912 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 12:01:41,915 | server.py:144 | FL finished in 10.973957113999859\n",
      "INFO flwr 2023-05-17 12:01:41,920 | app.py:202 | app_fit: losses_distributed [(1, 0.013341760509097784)]\n",
      "INFO flwr 2023-05-17 12:01:41,924 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.283947028546689)], 'reward': [(1, 3.2704871775049047)], 'global_regret': [(1, 0.013459851041784338)], 'local_regrets': [(1, {0: 0.012986208796862658, 1: 0.013748008672304494, 2: 0.013091352317912364, 3: 0.013541472249311619})]}\n",
      "INFO flwr 2023-05-17 12:01:41,927 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 12:01:41,930 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 12:01:42,034 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 12:01:55,871 | app.py:179 | Flower VCE: Ray initialized with resources: {'node:127.0.0.1': 1.0, 'memory': 6741956608.0, 'CPU': 8.0, 'object_store_memory': 2147483648.0}\n",
      "INFO flwr 2023-05-17 12:01:55,874 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 12:01:55,876 | server.py:270 | Requesting initial parameters from one random client\n",
      "INFO flwr 2023-05-17 12:01:58,442 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 12:01:58,444 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 12:01:58,445 | server.py:101 | FL starting\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99822)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 12:01:58,447 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99822)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99822)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99826)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99826)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99828)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99828)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99825)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99825)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 12:02:01,321 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 12:02:01,365 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 12:02:01,369 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99822)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99825)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99828)\u001b[0m [Client 0] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99826)\u001b[0m [Client 3] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 12:02:05,543 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 12:02:05,545 | server.py:144 | FL finished in 7.097734811000009\n",
      "INFO flwr 2023-05-17 12:02:05,549 | app.py:202 | app_fit: losses_distributed [(1, 0.013713661935776276)]\n",
      "INFO flwr 2023-05-17 12:02:05,552 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3226631913877647)], 'reward': [(1, 3.309009432320086)], 'global_regret': [(1, 0.013653759067677358)], 'local_regrets': [(1, {0: 0.013894019893656758, 1: 0.013818771007654966, 2: 0.014202207664099456, 3: 0.012939649177693923})]}\n",
      "INFO flwr 2023-05-17 12:02:05,554 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 12:02:05,556 | app.py:205 | app_fit: metrics_centralized {}\n",
      "INFO flwr 2023-05-17 12:02:05,645 | app.py:145 | Starting Flower simulation, config: ServerConfig(num_rounds=1, round_timeout=None)\n",
      "INFO flwr 2023-05-17 12:02:18,438 | app.py:179 | Flower VCE: Ray initialized with resources: {'node:127.0.0.1': 1.0, 'memory': 6752945767.0, 'object_store_memory': 2147483648.0, 'CPU': 8.0}\n",
      "INFO flwr 2023-05-17 12:02:18,443 | server.py:86 | Initializing global parameters\n",
      "INFO flwr 2023-05-17 12:02:18,445 | server.py:270 | Requesting initial parameters from one random client\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_get_parameters pid=99855)\u001b[0m [Client 0] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "INFO flwr 2023-05-17 12:02:20,753 | server.py:274 | Received initial parameters from one random client\n",
      "INFO flwr 2023-05-17 12:02:20,755 | server.py:88 | Evaluating initial parameters\n",
      "INFO flwr 2023-05-17 12:02:20,756 | server.py:101 | FL starting\n",
      "DEBUG flwr 2023-05-17 12:02:20,758 | server.py:215 | fit_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_fit pid=99855)\u001b[0m Client 0, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99855)\u001b[0m [Client 0] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99858)\u001b[0m Client 3, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99858)\u001b[0m [Client 3] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99857)\u001b[0m Client 1, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99857)\u001b[0m [Client 1] get_parameters\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99859)\u001b[0m Client 2, training finished for round 1\n",
      "\u001b[2m\u001b[36m(launch_and_fit pid=99859)\u001b[0m [Client 2] get_parameters\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 12:02:23,411 | server.py:229 | fit_round 1 received 4 results and 0 failures\n",
      "WARNING flwr 2023-05-17 12:02:23,447 | fedavg.py:242 | No fit_metrics_aggregation_fn provided\n",
      "DEBUG flwr 2023-05-17 12:02:23,450 | server.py:165 | evaluate_round 1: strategy sampled 4 clients (out of 4)\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99859)\u001b[0m [Client 1] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99855)\u001b[0m [Client 2] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99858)\u001b[0m [Client 3] evaluate, config: {}\n",
      "\u001b[2m\u001b[36m(launch_and_evaluate pid=99857)\u001b[0m [Client 0] evaluate, config: {}\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "DEBUG flwr 2023-05-17 12:02:27,748 | server.py:179 | evaluate_round 1 received 4 results and 0 failures\n",
      "INFO flwr 2023-05-17 12:02:27,750 | server.py:144 | FL finished in 6.991433505999794\n",
      "INFO flwr 2023-05-17 12:02:27,752 | app.py:202 | app_fit: losses_distributed [(1, 0.018021640674993303)]\n",
      "INFO flwr 2023-05-17 12:02:27,754 | app.py:203 | app_fit: metrics_distributed {'opt_reward': [(1, 3.3033152523315055)], 'reward': [(1, 3.2849635056121294)], 'global_regret': [(1, 0.018351746719372758)], 'local_regrets': [(1, {0: 0.017027744156550003, 1: 0.018647268615225903, 2: 0.019609712960573556, 3: 0.016801836967623735})]}\n",
      "INFO flwr 2023-05-17 12:02:27,756 | app.py:204 | app_fit: losses_centralized []\n",
      "INFO flwr 2023-05-17 12:02:27,759 | app.py:205 | app_fit: metrics_centralized {}\n"
     ]
    }
   ],
   "source": [
    "%%capture --no-stdout\n",
    "# sample_size_fns = [lambda n: n // NUM_CLIENTS for _ in range(NUM_CLIENTS)]\n",
    "\n",
    "client_0_sample_size_fn = lambda n: int(np.log(n))\n",
    "sample_size_fns = {0: client_0_sample_size_fn,\n",
    "                   1: lambda n: int((n-client_0_sample_size_fn(n))//(NUM_CLIENTS-1)),\n",
    "                   2: lambda n: int((n-client_0_sample_size_fn(n))//(NUM_CLIENTS-1)),\n",
    "                   3: lambda n: int((n-client_0_sample_size_fn(n))//(NUM_CLIENTS-1))}\n",
    "\n",
    "\n",
    "global_regrets, global_regrets_std, fed_local_regrets, fed_local_regrets_std = run_federated_training_experiments(data, aux, sample_size_fns, num_iters=3)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "id": "f4351f23-dc57-4d73-b9a2-6cae1297389e",
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "client_id = 0\n",
    "\n",
    "# Apply moving average smoothing\n",
    "window_size = 1\n",
    "local_regrets_smooth = np.convolve(local_regrets, np.ones(window_size) / window_size, mode='same')\n",
    "local_regrets_std_smooth = np.convolve(local_regrets_std, np.ones(window_size) / window_size, mode='same')\n",
    "\n",
    "global_regrets_smooth = np.convolve(global_regrets, np.ones(window_size) / window_size, mode='same')\n",
    "global_regrets_std_smooth = np.convolve(global_regrets_std, np.ones(window_size) / window_size, mode='same')\n",
    "\n",
    "fed_local_regrets_smooth = np.convolve(fed_local_regrets[client_id], np.ones(window_size) / window_size, mode='same')\n",
    "# window_size=4\n",
    "fed_local_regrets_std_smooth = np.convolve(fed_local_regrets_std[client_id], np.ones(window_size) / window_size, mode='same')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 29,
   "id": "7c0564f5-2db0-48ec-a965-ff333f5a5140",
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.legend.Legend at 0x19912fd90>"
      ]
     },
     "execution_count": 29,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAnoAAAHNCAYAAACJjdZcAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjcuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/bCgiHAAAACXBIWXMAAA9hAAAPYQGoP6dpAADLxklEQVR4nOzdeXycZb3//9e9zD6ZrE2Tpht0hy5AWQULVJAHPaiAIGABOfwAURFBDyCi54iIgh44sosIsslXDljwoFVEtoIKFCp0T/c2zdrsyyz3+vtjkmnSLJ00k2bSfJ6PB49OM/c9c4Urad65ls+luK7rIoQQQgghDjnqSDdACCGEEEIMDwl6QgghhBCHKAl6QgghhBCHKAl6QgghhBCHKAl6QgghhBCHKAl6QgghhBCHKAl6QgghhBCHKAl6QgghhBCHKH2kGzDauK6L42SuxrSqKhl9PTF00ifZRfoj+0ifZB/pk+wy3P2hqgqKoqR1rQS9QXIcl8bGjoy8lq6r5OeHaG2NYllORl5TDI30SXaR/sg+0ifZR/okuxyM/igoCKFp6QU9mboVQgghhDhESdATQgghhDhESdATQgghhDhESdATQgghhDhESdATQgghhDhESdATQgghhDhESdATQgghhDhESdATQgghhDhESdATQgghhDhESdATQgghhDhESdATQgghhDhESdATQgghhDhE6SPdgHTYts1ll13GRx99RHl5eVr3vPTSSzz11FPs2LGDcDjMmWeeyQ033EBubu4wt1YIIUR/XNfFcWwcZ3gOex+LHEchHtcwjAS27Y50c8a8/vpDVVVUVUNRlIPanlER9H75y1/y0UcfpX39o48+yr333stJJ53ETTfdxO7du3nmmWdYtWoVzz//PH6/fxhbK4QQYl+WZRGLtROLteM49kg355BTX69KeM4i/fWHqmoEAmECgTC6fnAiWNYHvdWrV/Pwww/j9XoxDGO/19fU1PDAAw+waNEiHn30UVQ1OTt95JFH8p3vfIdnnnmGq6++eribLYQQopNlmTQ01AAQCITw+QKd/zYf3JGNQ5mmKTKal0V694eL4zjE4zGi0Tai0TYKC0vQdc+wtyWrg15HRwf/8R//wac//Wk6Ojr44IMP9nvPK6+8gmmaXHHFFamQB3DOOedw7733smzZsqwPepqmYtvym5kQYvRzHJumpjo0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8smMHTuWI444guuvv57169cXjZM26TmWLl3K//2//5fx48czevRojj/+eB588EFc1y0aJ23SvSxcuJDvfOc7vPPOO+36uuveL1iwgAsuuICDDjqI7373u0ydOpUvv/xypz+LIoQQO30WSZdJp9Occ845fP7555x99tkMHz6cl156ibfffpuf/exnTJ06dXdPsVewZMkSpkyZgqZp/PjHP2bAgAF8+OGHPPPMM+y111488cQTJBIJNm7cyJlnnklzczPnnnsu/fr14y9/+QuLFy/mtttu46STTgrP+dVXX3HWWWcRjUY5++yzSSaTPPbYY6xevZr777+fgw8+eDd+4m8Wd911F7fffjvgCfIAaY+eZfr06TzzzDMcc8wxfP/732fp0qU8+uij9O/fn9mzZ1NVVSVt0oOsWrWKKVOmkMlk+PGPf8zgwYN55ZVXmDt3LmeddRY33XQTIL8n3c2yZcs455xz2LhxIw8//DDjx48P+7rr3r/77rtcdNFFDBw4kDPOOAPXdXn44YfJZDI8+eSTjBgxYsc/kJD0KPfcc49obGwUzz//fNjmuq64+OKLxb777ivWrFmzG2fXe7jooovEvvvuK/7xj38UtT/00EOisbFR/OlPfxJCCPGrX/1KjBo1Srz//vvhmGw2K04++WQxfvx4kUqlwvaLL75YjB07VqxYsSJs27Jlizj88MPFpEmThOu63fypegcLFy4U++yzjxg9erRobGws6pP26DleeeUV0djYKP7lX/6lqH327NmisbFR3HPPPUIIaZOe5OabbxaNjY3ihRdeKGo/77zzRGNjo/jyyy+FENIm3clf//pXcdBBB4nGxkbR2Ngo5s2bV9TfHffedV1xwgkniMMOO0w0NTWFY5cvXy7Gjh0rLrnkkp36TDJ028M8/fTT9OvXjxNOOCFsUxSFSy65BMuyeO6553bj7HoHpmkyf/58DjzwQBobG4v6Jk+eDMB7772H4zg8++yz7L///hxwwAHhmGg0ynnnnUdTUxOvv/46AJs2beKNN95g4sSJDB48OBxbW1vLGWecwZdffsmiRYu6/bN900mlUvz85z/ne9/7Hvvvv39Rn7RHz/L444+TTCa55ppritpPOOEELrvsMoYOHSpt0sMsW7YMgCOPPLKofeLEiQB89tln0ibdyGWXXcYVV1xBnz59OPHEE9v1d9e9/+ijj/jiiy+YMmUKNTU14dghQ4bwT//0T7z55pts2LBhhz+XFHo9SEtLC0uWLGG//fZr1xe0yS/dzqPrOs8//zy33HJLu75NmzYBoKoqX3zxBel0up3ggLw9Fi5cWPTclbGSjglyJH/961+365P26Dkcx+G9997j4IMPpqKiAoBsNotpmkQiEa655hqOO+44aZMeZtiwYQDt8rKWLl0KQL9+/aRNupElS5Zw9dVX89RTTzF06NB2/d117z/88MOi9kLGjh2L67p89NFHX/fjhEih14OsX78eIQQDBgxo1xePx6murmbVqlW7YWa9C1VVGTx4MEOGDGnXd//99wMwfvz4MOG8lD369+8PENpj3bp1HY7t169f0VhJaf7617/y//7f/+OWW26hoaGhXb+0R8+xatUqcrkcgwYN4uWXX+akk05iv/32Y//99+fiiy9myZIlgLRJT3PZZZcxbNgwrr/+et5++21WrVrFI488whNPPMGECRM48MADpU26kf/5n//hJz/5CZFIpGR/d937YOwee+yx3fPuCFLo9SBBtWcikSjZH4vFyGQyPTmlbxVPP/00s2bNYsCAAZxxxhmd2iMWiwGE9mhtbQUgmUy2GxuPx4vGStqzfv16fvnLX3L66aeHYai2SHv0HFu3bgXgrbfe4uc//zlHHXUUd911F5dffjnz58/nRz/6EStXrpQ26WH69u3LVVddxbp167jgggs45phjuOWWWxgzZgx33XUXiqJIm3QjHQm8gO6698HYUufdFXbSd/hIyddG+AXOooNCZyEEqiq1d3fw1FNPceONN5JIJLj99ttJJpMd2gHyNgrs0Znt2o6VFCOE4LrrrqOyspJf/OIXnY7bXp+0x67BNE3AC1XdcccdHHfccYCXC7bPPvswbdo0/vCHP3DEEUd0eA5pk13Pf//3f/P73/+ePffck2uvvZaGhgbmz5/Po48+yvnnn8/9998vvye7ke66991tJyn0epBA2Wez2ZL92Wy2pJtXsnMES3lUVlbyX//1X4wdOxbI26PUv5QCG1VWVn7tsZJiHnjgAebNm8ddd91FLpcjl8sBYFkWAFu2bEHTNGmPHiTwHPTr1y8UeQFHH300AwYMYO7cuUyaNAmQNukJWltbueuuu+jbty+zZs2iuroagGOPPZZ99tmH6dOn88c//pGDDjoIkDbZHXTX/6M6Gxu07YydpJTvQQYNGoSiKGE8vpB0Os22bdvCeLxk57EsixtuuIHbb7+dfv368cgjj/Dd73437B80aBBASXsEbYE9ujJWivTSvPbaawghwkVDg8cHH3wAwIQJEzj11FOlPXqQ4D6WypUM2ltaWqRNepClS5eSzWY59thjQ5EXcNJJJ5FIJHj77belTXYj3XXvg7FtFyovbNsZO0mh14Mkk0lGjBhRsnomqL4pLNmW7DiO43DNNdcwe/ZsRo0axaxZs9h7772LxgwfPpzKysqSlc5t7TFmzBhUVe107Lhx43b1x+gVXHfddTzwwAPtHsGuGA888AD//u//Lu3Rg9TV1TFkyBCWLVsWelgDXNdl1apVDBo0SNqkBwnywxzHadcnhMB1XYQQ0ia7ke6690GUqVQF9MKFC1EUpWRFbleRQq+HOfnkk1m9ejUvvPBC2CaE4L777iMSiYShEsnO8Yc//IGXX36ZsWPH8uijj4ZVToXous6kSZOYP38+CxYsCNtzuRwPP/wwDQ0NfP/73wc8D8ehhx7Kyy+/zMqVK8OxTU1NoYjcZ599uv+DfQMZPXo0hx56aLtH4LU49NBDOfDAA6U9epgpU6aQSqW49957i9qffPJJmpqaOOGEE6RNepCRI0cycOBAXnrppXaenVmzZpHNZjnssMOkTXYj3XXvx44dy7Bhw5g1axbNzc3h2BUrVvDyyy9z1FFHUVtbu8Pzllug9TDZbJYpU6awfPlyzj33XIYNG8aLL77I3LlzmT59OhdffPHunuI3njVr1nDsscfiOA5XX311SZHX0NDAYYcdxsaNGzn11FPJZDJceOGF1NfXh9vZzJgxo0h4f/7555x55pkkk0kuuOACIpEIjz76KGvWrOGBBx4oCgtLts+5557Lu+++224LNGmPnsE0Tc477zw++OADTjzxRA4++GA++eQTnnzySfbaay+efPJJ4vG4tEkP8tZbb/GTn/yEmpoazjrrLBoaGvjggw945plnGD58OH/+85/DbemkTbqXO+64gzvvvLPkFmjdce/nzp3LpZdeyqBBg/jxj3+MaZo8+OCDWJbF448/zvDhw3f4s0ihtxvYsmULM2bMYM6cOaRSKYYNG8YFF1wQ7tog2Tmefvpprrvuuk7HHHzwwcycOROAlStX8vvf/565c+diWRajRo1i2rRpJSsOP/30U2bMmMH777+PqqqMHj2aq666quSimJLOKSX0QNqjJ8lkMvzpT3/iueeeY+3atdTX13Psscdy1VVXhQspg7RJT7J48WLuvvtu5s+fTyqVCgtmLr/88qKEfGmT7qUjoQfdd+/nzZvHHXfcwccff0w8HufAAw/k6quv3rl9bpFCTyKRSCQSiaTXInP0JBKJRCKRSHopUuhJJBKJRCKR9FKk0JNIJBKJRCLppUihJ5FIJBKJRNJLkUJPIpFIJBKJpJcihZ5EIpFIJBJJL0UKPYlEIpFIJJJeihR6EolEIpFIJL0UKfQkEsluQ67XLpFIJN2LvrsnIJFIyotg65+vw6uvvsqgQYO6PH79+vX8+7//O1OmTGHChAlfd4ohwVynTp3Kz372sy4dY1kWs2bN4uWXX+bzzz+npaWFqqoqhg8fzjHHHMNZZ51FPB7f4Tntbt555x3OO+88DjjgAB5//PFuu87q1at5+OGHeeutt1izZg2O41BfX8+4ceM49dRTOfzww9sdc/TRR7N69Wr++te/sueee3bb3CQSSR4p9CQSSRGjRo3ipJNOKmrbvHkzc+fOJZFIcMwxx7Q7JpFIfK1rXHvttbzzzjucdtppOzXXr0tzczPnn38+n332GTU1NYwePZrKyko2btzIJ598wnvvvcfMmTN5+OGHv5Zw/bbx+uuvc+WVV5LL5Rg+fDiHHnooQghWrVrF888/z/PPP89pp53Gb37zGxRF2d3TlUi+1UihJ5FIijjuuOM47rjjitreeecd5s6dS21tLbfddttOX2N3hWx//etf89lnn3HKKadwyy23EI1Gw76tW7fyq1/9ipdeeokrr7yS2bNn75Y5ljtbt27l6quvxnVd7rrrLiZOnFjU/9FHHzFt2jRmz57NPvvsw7nnnhv2Pfjgg1iWxR577NHT05ZIvrXIHD2JRPKtwLIsXnzxRQzD4Oabby4SeQDV1dX87ne/o76+nsWLF7No0aLdNNPy5tVXXyWVSnH88ce3E3kAY8aM4Ve/+hUAf/7zn4v6hgwZwogRIzAMo0fmKpFIpEdPIpHsIl5//XVmzpzJRx99RCaTYY899mDixIlceuml1NTUALBq1aqi0O+FF14IwMMPP8z48eMBWLp0KQ888ADvvPMO69evx3Vd+vbty+GHH860adPo16/fDs1v27Zt2LZNJBLpMJwYi8W48MILWbZsWTsxsnXrVh5++GFee+01li9fTi6Xo6amhgMOOIBLLrmEsWPHhmODz/mDH/yA66+/nhkzZvDmm2+SzWYZOXIkV1xxBUcccQRffPEFv//975k/fz6GYXDAAQdw/fXXM3jw4PBcQR7iH/7wB1zX5Z577mHp0qXU19dz5JFHcvnll9OnT58u3YP169dzzz338Prrr7NhwwaqqqoYP34806ZNo7GxsUvn2Lx5MwCapnU45vDDD+eEE06gvr6+qL1tjl5X8kEHDhzInDlzwveO4/Dkk08ye/ZsvvzySwAaGxs566yzmDx5sgwVSyRtkEJPIpHsNLfddht/+tOf0DSNAw88kNraWj788EPuvfdeXnzxRR566CEGDx5MIpHgpJNOYu7cuWzevJkJEybQ0NBAQ0MDAPPnz+eSSy4hk8kwevRoRo0axdatW1m4cCGPP/44f/vb33juueeoqKj42nOsq6ujT58+bNy4kZ///Odcf/31DBw4sN24Sy+9tF3b5s2bOeuss1ixYgVDhgzhkEMOwbIsFi9ezMsvv8ycOXN4/PHHGTNmTNFxa9as4fTTT8d1XQ488EBWrlzJokWLmDZtGjfddBO/+c1vqKurY/z48SxevJj//d//ZeHChfz1r39tl/f49NNP89prr7Hnnnty5JFHsnjxYh577LFQYG8vp/DTTz/loosuYsuWLQwZMoQjjzyS9evX8z//8z+8+uqr3HHHHRxxxBHbvY977703AM899xz77bcfp512WjvvaCKRYMaMGds9V6l80IA333yTpqYm9tprr7DNsiwuv/xy/v73v1NRUcG4ceMwDIN3332X66+/nnfeeYff/va3272uRPKtQkgkEsl2mDdvnmhsbBRHHXVUu75XX31VNDY2ioMPPlgsWrQobM/lcuLGG28UjY2N4tRTTxWu64Z955xzjmhsbBRvvfVW0blOPPFE0djYKF5++eWi9g0bNoijjjpKNDY2imeeeSZsv/3220VjY6OYMWNGlz7HU089JRobG0VjY6MYNWqUOO2008S//du/iTlz5oiWlpYOj7vllltEY2OjuOWWW4o+RzabFVOnThWNjY3ixhtvDNtXrlwZXuecc84Rra2tQgghXNcVl19+edh3/fXXC8uyhBBCtLa2iuOOO040NjaK559/vt1nbGxsFDfffLOwbVsIIYRpmuLaa68VjY2N4rLLLgvHB7Y666yzwjbTNMXEiRNFY2OjuPfee4s+w6uvvir23XdfcdBBB4nNmzdv9x66risuuOCCcE7777+/mDp1qrj//vvFokWLhOM4HR4b2HDZsmWdXuO5554L/94K5/Sf//mf4T0tbN+4caOYPHmyaGxsFE8++eR2P4NE8m1C5uhJJJKd4sEHHwRg+vTpRR6tSCTCv/zLv7DnnnuyePFi5s2b1+l5UqkUo0ePZsqUKe2KQfr06RPmg61atWqH5zp58mTuvPNOBgwYgBCCjz/+mHvvvZepU6dy8MEHc/755xeFCQOqqqr43ve+x5VXXlkUGoxGo0yZMqXTeV133XUkk0kAFEXhhBNOALz7c8MNN6DrXmAlmUzyve99D4Dly5e3O8/w4cP5xS9+EYZMg1zD2tpaXn/9ddasWdPh537llVdYsWIFRx55JBdffHHRZzj66KP54Q9/yNatW/nLX/7S8c3zURSFu+++m/POOw/DMEin08yZM4ff/va3nH766UyYMIF//ud/ZvXq1ds9VykWLlzIL37xC+LxOHfffTd1dXUAmKbJzJkzMQyD2267LWwHaGho4Oabbwbgvvvu26HrSiS9FSn0JBLJDmPbNgsWLEBRFH7wgx+069d1PRRt77zzTqfnSiaT3HrrrfzmN78pat+wYQN/+9vf+OyzzwDvB39nOPbYY3n11Vd56KGHuOiiixg9ejSapuE4DvPmzWPatGlcf/31uK4bHnPllVdy7733UlVVFba1tLQwf/583nzzzQ7nFYlE2GeffYraAoEycODAovMB4ftcLtfuXMcff3y7vLhYLBauV9fZ/Q1EdkdrFgYh2+3ZKCAej3PjjTfy5ptv8rvf/Y6TTz6ZAQMGAN4SNrNmzWLSpEm89tprXTpfwNq1a/npT39KLpfjN7/5TRgmBli8eDEtLS0MHz68ZJ7mmDFjqK+vZ+nSpWzcuPFrXVci6c3IHD2JRLLDNDc3Y1kWtbW1HebNBbljXf3xXbBgAbNmzeKTTz5hxYoVpNNpgNALJXbB0iyapnHIIYdwyCGHANDa2sq8efN44okn+Pvf/85TTz3F2LFjOfvss8NjVq5cyWOPPcYHH3zAsmXLaGpq2u68KisrUdXif08H42tra9uN76yQoKMFhgOBtWHDhg6PXbt2LQC33nort956a4fj1q1b12FfKWpqapg8eTKTJ08GYMWKFbz22ms8+OCDrFmzhquvvppXX321yPvWEel0mmnTprFx40Z+8pOfMGnSpJKf4R//+AejRo3q9Fxr167tcoGKRNLbkUJPIpHsMIG46UygBGMikch2z3fTTTfx2GOPoWkae++9N//0T//EXnvtxX777ccbb7zBf/3Xf+3wXNeuXcuqVasYOnRoOxFQUVHBxIkTmThxIrfeeisPPvggzz77bCj0nn/+ea677jps22bPPfdk/Pjx7LXXXowePRrXdbn88stLXjMIy+4KOqpyDe5vZ1WwgXdy/Pjx9O3bt8Nx2xNkQgg+++wztm7dGorkQoYMGcL555/P5MmTmTJlCitXruR///d/+eEPf7jd806fPp1PP/2UI488kquuuqrDz7DHHntw4IEHdnq+IFQukUik0JNIJDtBTU0NhmHQ3NxMa2trSa/eypUrAdottdGWd999l8cee4wBAwZw3333MWLEiKL+l156aafm+sc//pEnnniCn//85yUrawNOP/10HnzwQZqbmwEvdzBYF+7uu+9utzPIK6+8slPz6irr168v2R7k5gWevVIEwvakk07ijDPO2Kl5nHnmmeRyOd56662wWrot1dXVHHvssdx///3hfeyM3//+97zyyisMGzaM2267rZ0XtPAz9O/ff5cs2i2RfFuQOXoSiWSHMQyDcePG4bpuScFj23bYHqyT1xEffvgh4O3M0VbkBflzsOOh23HjxgEwa9YsMplMh+OWLl0KEK4r98UXX5BKpRg5cmTJ7d+CHL1dEVLujNdff71dWzqd5q233kLTtE73DD7ooIMA+Nvf/lay/9FHH+WUU07h7rvv7nQOiqKw3377ATBz5sxOxwb3ceTIkZ2Oe/rpp/nTn/5ERUUFd911F5WVlSXHjRkzhlgsxmeffVYyTL1+/XqOP/54LrzwQlKpVKfXlEi+TUihJ5FIdorzzz8fgH/7t3/jk08+Cdsty+Kmm25ixYoVfOc73ykKtwXrrrW0tIRtQc7a22+/XSTEMpkMv/zlL/niiy+A0oUKXeHEE09k2LBhLF++nEsuuYSvvvqq3Zj58+dz8803o6pquJhzEM5cunQpS5YsCccKIXj88cd58sknd2peXeXdd98tElemafLLX/6S5uZmTjnllE7DrpMmTaJv37688sorPPDAA0WidNGiRfznf/4nn332WZcWTf7pT3+Kqqrcc8893H777WEOZeG87r77bl577TVGjhzZ6dp8CxYs4Je//CWapjFjxox2Ar+QRCLBD3/4Q9LpNNdee224cDN4XtcbbriBJUuWkEgkZOhWIilAhm4lEslOMXHiRC666CLuv/9+Tj/99HDB5IULF7Ju3ToGDhzIf/zHfxSF44YOHcobb7zBLbfcwgsvvMCFF17I8ccfz5133snnn3/OxIkT2X///TFNkw8++ICWlhZGjhzJF198waZNm3ZonoZhcN9993HJJZcwf/58Jk2aRGNjY1jk8OWXX7J06dJw2ZLAAzhkyBCOPvpo5syZw+TJkzn44IOJRqN88sknrFmzhr322osvv/xyh+fVVfr378+vf/1rZs+ezeDBg1m0aBFr167lO9/5DtOnT+/02Hg8zh/+8Acuu+wyfvvb3/LII48watQompubWbBgAUIIzjvvvJJbmrXlkEMO4V//9V+56aabuOuuu7jvvvsYO3Ys9fX1tLS0sGjRIrZt28aQIUP44x//WDIMG/DTn/4U0zQZOnQor7/+Oi+99BKWZbUbN23aNEaMGME111zDp59+yrx58zj22GMZM2YM8XicDz74gObmZoYOHcpNN920/ZspkXyLkEJPIpHsNNdddx3f/e53eeSRR/j444/J5XIMGjSIadOmceGFF1JdXV00/vLLL2f16tXMmzePN954g8MOO4xx48bx5JNPcvvtt/P222/z97//nYaGBsaMGcMZZ5zBIYccwqGHHsqbb76JZVk7tF/qwIEDefbZZ5k9ezZz5szhH//4B2+88QaqqtKvXz/OOeccfvzjHzN8+PCi4/7jP/6D++67jxdeeIH33nuPiooKBg4cyI9+9CPOP/98Tj/9dD7//HM+/vhjRo8evVP3siMmT57MoEGDeOCBB3jttdfYY489uOKKK7jooou65ME64IADwjDpG2+8wd///ndqamoYP3485557bpdEXsBpp53G+PHj+fOf/8zcuXP58ssv+eCDD0gmk+y111784Ac/4Ec/+lG7HTPasmXLFgCWLVvGsmXLOhx3xhlnMGLECGKxGPfffz9//vOfefbZZ8P9iAcNGsS5557Leeed127JGonk244iujuxRCKRSCQ7TLAf7NSpU/nZz362u6cjkUi+YcgcPYlEIpFIJJJeihR6EolEIpFIJL0UKfQkEolEIpFIeikyR08ikUgkEomklyI9ehKJRCKRSCS9FCn0JBKJRCKRSHopUuhJJBKJRCKR9FKk0JNIJBKJRCLppUihJ5FIJBKJRNJLkUJPIpFIJBKJpJcihZ5EIpFIJBJJL0UKPYlEIpFIJJJeyv8Httc/kivCPYEAAAAASUVORK5CYII=",
      "text/plain": [
       "<Figure size 707.107x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "client_id = 0\n",
    "k1 = 0\n",
    "k2 = 9\n",
    "x = ss_range\n",
    "\n",
    "c=5\n",
    "plt.rcParams['figure.figsize'] = [c*np.sqrt(2), c*1]\n",
    "plt.plot(x[k1:k2], local_regrets_smooth[k1:k2], label=r'$R_c(\\hat{\\pi}_c$)')\n",
    "plt.fill_between(x[k1:k2], local_regrets_smooth[k1:k2] - local_regrets_std_smooth[k1:k2], local_regrets_smooth[k1:k2] + local_regrets_std_smooth[k1:k2], alpha=0.4)\n",
    "\n",
    "# plt.plot([sample_size_fns[0](n) for n in x], global_regrets, label=\"Global model, global regret, local ss\")\n",
    "\n",
    "plt.plot(x[k1:k2], fed_local_regrets_smooth[k1:k2], label=r'$R_c(\\hat{\\pi}_\\lambda$)')\n",
    "plt.fill_between(x[k1:k2], fed_local_regrets_smooth[k1:k2] - fed_local_regrets_std_smooth[k1:k2], fed_local_regrets_smooth[k1:k2] + fed_local_regrets_std_smooth[k1:k2], alpha=0.4)\n",
    "\n",
    "plt.plot(x[k1:k2], global_regrets_smooth[k1:k2], label=r'$R_\\lambda(\\hat{\\pi}_\\lambda)$')\n",
    "plt.fill_between(x[k1:k2], global_regrets_smooth[k1:k2] - global_regrets_std_smooth[k1:k2], global_regrets_smooth[k1:k2] + global_regrets_std_smooth[k1:k2], alpha=0.4)\n",
    "\n",
    "plt.xlabel('Total Sample Size', fontsize=16)\n",
    "plt.ylabel('Regret', fontsize=16)\n",
    "plt.xticks(fontsize=14)\n",
    "plt.yticks(fontsize=14)\n",
    "plt.legend(fontsize=22)\n",
    "# plt.show()\n",
    "\n",
    "\n",
    "# plt.savefig('./data/homogeneous.png', dpi=300)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "0c56d533-3317-4a61-8cf9-d9e2816485b0",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.9.16"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
