{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 296,
   "metadata": {
    "slideshow": {
     "slide_type": "notes"
    }
   },
   "outputs": [],
   "source": [
    "import numpy as np\n",
    "import pandas as pd\n",
    "from scipy import stats\n",
    "from mpl_toolkits.mplot3d import Axes3D\n",
    "import matplotlib.pyplot as plt\n",
    "import itertools\n",
    "from collections import Counter\n",
    "from more_itertools import set_partitions\n",
    "import copy\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "sns.set_style(\"ticks\")\n",
    "plt.rcParams['font.family'] = 'serif'"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Dataset of $j$-th client follows $D_j\\sim (\\theta_i,\\epsilon_j^2)$.\n",
    "The following code is to simulate:\n",
    "- $\\epsilon_j^2$: the amount of noise in the sampling process \n",
    "- $\\theta_j$: the true mean of players"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 297,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[1.0, 1.0]\n",
      "200.0\n"
     ]
    }
   ],
   "source": [
    "means_dist_1 = stats.norm(loc = 0, scale = 1)\n",
    "means_dist_2 = stats.norm(loc = 0, scale =1)\n",
    "variance_dist = stats.beta(a=8, b=2, scale = 1000/4)\n",
    "print([means_dist_1.var(), means_dist_2.var()])\n",
    "print(variance_dist.mean())"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Calculate the error of clients in a coalition structure"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 298,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Reference: [https://github.com/kpdonahue/model_sharing_games]\n",
    "#Here is the function inspired by the above reference.\n",
    "def calculate_regression(var_vec = [means_dist_1.var(), means_dist_2.var()], mue = variance_dist.mean(), \n",
    "                         n_list = [10, 20, 30], w_best=False,w_list = [0.2, 0.4, 0.6], \n",
    "                         v_best=False,v_mat = [[0.1, 0.6, 0.3], [0.2, 0.8, 0.0], [0.3, 0.5, 0.2]], \n",
    "                         x_vec = [1.0, 1.0],fair=False,err_list=None, q=0):\n",
    "    '''\n",
    "    Calculate exact error for linear regression, in special case where x-distribution is 0-mean multivariate \n",
    "    normal. Assumes the parameters have 0 correlation (parameter in 1st dimension is independent of parameter\n",
    "    in 2nd dimension, for example). \n",
    "\n",
    "    Args:\n",
    "        var_vec: a list with variance of true parameter values\n",
    "        mue: mean of true error distribution. \n",
    "        n_list: a list of length M (number of players) with the number of samples each has. \n",
    "        w_list: a list of w-weights each player uses (in [0, 1]) for coarse-grained federation. \n",
    "        v_mat: a matrix (list of lists) of weights each player uses in fine-grained federation: the rows sum up \n",
    "               to 1.\n",
    "        x_vec: a list of E[X_d^2] for d in dimension D for input distribution of X. \n",
    "    Returns:\n",
    "        dataframe with average error for each player, for: local, uniform, coarse-grained, and fine-grained \n",
    "        federation.  \n",
    "    '''\n",
    "    # dataframe for storing error\n",
    "    player_error = pd.DataFrame(data = 0.0, index = ['local', 'uniform', 'coarse', 'fine',\"fair\"], \n",
    "                                columns = range(len(n_list)))\n",
    "    N = sum(n_list)\n",
    "    D = len(var_vec)\n",
    "    n_vec = pd.DataFrame(n_list)\n",
    "    \n",
    "    # for each player, calculate their true error \n",
    "    for i,n in enumerate(n_list):\n",
    "        \n",
    "        # n = n_list[i]\n",
    "        \n",
    "        \n",
    "        # local\n",
    "        player_error.loc['local'][i] = mue * D/(n - D - 1)\n",
    "        \n",
    "        # uniform\n",
    "        var_prod = pd.DataFrame(x_vec).T.dot(pd.DataFrame(var_vec))[0][0]\n",
    "        \n",
    "        player_error.loc['uniform'][i] = ((mue * D * (n_vec**2).T.dot(1/(n_vec- D -1))/(N**2))[0][0] + \n",
    "                                          var_prod*((n_vec**2).sum() - n**2 + (N- n)**2)[0]/(N**2))\n",
    "        \n",
    "        # coarse-grained\n",
    "        if w_best:\n",
    "            # v_vec = pd.DataFrame(v_mat[i])\n",
    "            w=w_list\n",
    "            player_error.loc['coarse'][i] = ((mue*D *(1-w)**2 * (n_vec**2).T.dot(1/(n_vec- D -1))/(N**2))[0][0] + \n",
    "                                            mue*D*(w**2 + 2 * (1-w) * w*n/N)/(n-D-1) + \n",
    "                                            (1-w)**2*var_prod*((n_vec**2).sum() - n**2 + (N- n)**2)[0]/(N**2))    \n",
    "        # fine-grained\n",
    "        if v_best:\n",
    "            v_vec = pd.DataFrame(v_mat[i])\n",
    "            player_error.loc['fine'][i] = (mue * D * (v_vec**2).T.dot(1/(pd.DataFrame(n_list)- D -1))[0][0] + \n",
    "                                        var_prod * ((v_vec**2).sum() - v_vec[0][i]**2 + (1 - v_vec[0][i])**2)[0])\n",
    "        if fair is True:\n",
    "            v_vec = pd.DataFrame([ni*pow(err_i,q)/sum([ni*pow(err_i,q) for ni,err_i in zip(n_list,err_list)]) for ni,err_i in zip(n_list,err_list)])\n",
    "            mu_co=(v_vec**2).T.dot(pd.DataFrame([D/(nval-D-1) for nval in n_list]))\n",
    "            var_prod_con=((v_vec**2).sum() - v_vec.iloc[i]**2 + (1 - v_vec.iloc[i])**2)\n",
    "            player_error.loc['fair'][i] = ((mue * mu_co)[0][0] + \n",
    "                                          var_prod*var_prod_con[0])\n",
    "        \n",
    "            \n",
    "            \n",
    "    return player_error"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Collect errors in all possible coalition structures"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 299,
   "metadata": {},
   "outputs": [],
   "source": [
    "#Reference: [https://github.com/kpdonahue/model_sharing_games]\n",
    "#Here is the function inspired by the above reference.\n",
    "def calc_error_groups(var_vec = [means_dist_1.var(),means_dist_2.var()], mue = variance_dist.mean(), same_size = False, \n",
    "                n = 10, M = 5, n_list = [10, 20, 30], fair=False, err_list=None, q=0):\n",
    "    '''\n",
    "    Calculates the errors that players experience, for all possible arrangements of players into groups. \n",
    "    Assumes optimal versions of coarse and fine federation are used. \n",
    "    \n",
    "    Args:\n",
    "        var: variance of mean parameter. \n",
    "        mue: mean of errors. \n",
    "        same_size: boolean, indicates whether all players are the same size (if so, runs faster). \n",
    "        n: if players are the same size, indicates the size\n",
    "        M: if players are the same size, indicates the number of players\n",
    "        n_list: if players differ in their size, provides a list of those sizes. \n",
    "        \n",
    "    Returns: \n",
    "        tables with errors for every combination of players (uniform, coarse, and fine-grained federation), \n",
    "        as well as local learning.\n",
    "        \n",
    "    \n",
    "    '''\n",
    "    \n",
    "    if same_size:\n",
    "        # all players are interchangable\n",
    "        n_list = [n for i in range(M)]\n",
    "        comb = list(itertools.combinations_with_replacement([0,1], r=M))\n",
    "    else: \n",
    "        # for combinations, the identity of player matters (not interchangable)\n",
    "        # for example: [1, 0, 1] means a group made of 1st and 3rd players only\n",
    "        comb = list(itertools.product([0,1], repeat = len(n_list)))\n",
    "    \n",
    "    string_list = [\"\".join(map(str, val)) for val in comb] # string to name groups\n",
    "    err_uniform = pd.DataFrame(data = np.nan, \n",
    "                              index = string_list,\n",
    "                              columns = ['n_' + str(i) + '_err' for i in n_list] )\n",
    "    err_best_coarse = err_uniform.copy()\n",
    "    err_best_fine = err_uniform.copy()\n",
    "    err_fair=err_uniform.copy()\n",
    "        \n",
    "    for index, group in enumerate(comb):\n",
    "        loca = \"\".join(map(str, group)) # group index\n",
    "        if sum(group) > 0: # ignore group with no members\n",
    "            \n",
    "            # drop players not in this group\n",
    "            temp_n = [n_list[i] for i in range(len(n_list)) if list(group)[i] == 1]\n",
    "            \n",
    "            # calculate error table and rename columns\n",
    "            error_table = calculate_regression(var_vec= var_vec, mue = mue, n_list = temp_n, w_best = False, v_best = False,fair=fair,err_list=err_list,q=q)\n",
    "            error_table.columns = [i for i in range(len(n_list)) if list(group)[i] == 1]  \n",
    "            \n",
    "            # copy errors into correct tables\n",
    "            for player in range(len(n_list)):\n",
    "                if list(group)[player] ==1: # if player is participating in group\n",
    "                    err_uniform.iloc[index, player] = error_table.loc['uniform', player]\n",
    "                    err_best_coarse.iloc[index, player] = error_table.loc['coarse', player]\n",
    "                    err_best_fine.iloc[index, player] = error_table.loc['fine', player]\n",
    "                    err_fair.iloc[index, player] = error_table.loc['fair', player]\n",
    "    \n",
    "    local_error = calculate_regression(var_vec=  var_vec, mue = mue, n_list = n_list, w_best = False, \n",
    "                                  v_best = False).loc['local']\n",
    "    \n",
    "    err_uniform.dropna(how = 'all', inplace = True)\n",
    "    err_best_coarse.dropna(how = 'all', inplace = True)\n",
    "    err_best_fine.dropna(how = 'all', inplace = True)\n",
    "    err_fair.dropna(how = 'all', inplace = True)\n",
    "    \n",
    "\n",
    "    return err_uniform, err_best_coarse, err_best_fine, err_fair,local_error"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Calculate utility in ACFG"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 300,
   "metadata": {},
   "outputs": [],
   "source": [
    "def calc_unity_groups(err_uniform=None,same_size = False, \n",
    "            n = 10, M = 5, n_list = [1, 2], preference=\"selfish\",friends=None,w=0.35,client_pre=None):\n",
    "    if client_pre==None:\n",
    "        client_pre=list(range(len(n_list)))\n",
    "    if same_size:\n",
    "        # all players are interchangable\n",
    "        n_list = [n for i in range(M)]\n",
    "        comb = list(itertools.combinations_with_replacement([0,1], r=M))\n",
    "    else: \n",
    "        # for combinations, the identity of player matters (not interchangable)\n",
    "        # for example: [1, 0, 1] means a group made of 1st and 3rd players only\n",
    "        comb = list(itertools.product([0,1], repeat = len(n_list)))\n",
    "    \n",
    "    string_list = [\"\".join(map(str, val)) for val in comb] # string to name groups\n",
    "    util_uniform = pd.DataFrame(data = np.nan, \n",
    "                              index = string_list,\n",
    "                              columns = ['n_' + str(i) + '_err' for i in n_list] )\n",
    "    util_uniform=copy.deepcopy(err_uniform)\n",
    "    \n",
    "    all_coalition = int('1' * len(n_list), 2)\n",
    "    for index, row_name in enumerate(util_uniform.index):\n",
    "        current_coalition= int(row_name, 2)\n",
    "        complement_coalition = all_coalition - current_coalition\n",
    "        complement_coalition = bin(complement_coalition)[2:]\n",
    "        complement_coalition = complement_coalition.zfill(len(n_list))\n",
    "        if sum([int(exist) for exist in complement_coalition])!=0:\n",
    "            complement_index = util_uniform.index.get_loc(complement_coalition)\n",
    "        else:\n",
    "            complement_index=None\n",
    "        \n",
    "        # copy errors into correct tables\n",
    "        for player in client_pre:\n",
    "        # for player in range(len(n_list)):\n",
    "            if list([exist for exist in row_name])[player]==\"1\": # if player is participating in group\n",
    "                if preference==\"mean-FR\":\n",
    "                    # if index==\"00001\":\n",
    "                    #     print()\n",
    "                    if complement_index!=None:\n",
    "                        # print(err_uniform.iloc[index,player])\n",
    "                        # print(pd.to_numeric(err_uniform.iloc[index,friends[str(player)]+[player]].replace(\"-\", np.nan), errors='coerce'))\n",
    "                        # print(pd.to_numeric(err_uniform.iloc[complement_index,friends[str(player)]].replace(\"-\", np.nan), errors='coerce'))\n",
    "                        util_uniform.iloc[index, player] =w*err_uniform.iloc[index,player]+(1-w)*(pd.to_numeric(err_uniform.iloc[index,friends[str(player)]+[player]].replace(\"-\", np.nan), errors='coerce').sum(skipna=True)+pd.to_numeric(err_uniform.iloc[complement_index,friends[str(player)]].replace(\"-\", np.nan), errors='coerce').sum(skipna=True))/(len(friends[str(player)])+1)\n",
    "                        # print( util_uniform.iloc[index, player])\n",
    "                    else:\n",
    "                        util_uniform.iloc[index, player] =w*err_uniform.iloc[index,player]+(1-w)*pd.to_numeric(err_uniform.iloc[index,friends[str(player)]+[player]]).mean(skipna=True)\n",
    "                        \n",
    "                if preference==\"min-FR\":\n",
    "                    if complement_index!=None:\n",
    "                        util_uniform.iloc[index, player] =w*err_uniform.iloc[index,player]+(1-w)*np.nanmax([pd.to_numeric(err_uniform.iloc[index,friends[str(player)]+[player]].replace(\"-\", np.nan), errors='coerce').max(skipna=True),pd.to_numeric(err_uniform.iloc[complement_index,friends[str(player)]].replace(\"-\", np.nan), errors='coerce').max(skipna=True)])\n",
    "                    else:\n",
    "                        util_uniform.iloc[index, player] =w*err_uniform.iloc[index,player]+(1-w)*pd.to_numeric(err_uniform.iloc[index,friends[str(player)]+[player]]).max(skipna=True)\n",
    "                if preference==\"mean-PA\": \n",
    "                    if complement_index!=None:\n",
    "                        util_uniform.iloc[index, player] =(pd.to_numeric(err_uniform.iloc[index,friends[str(player)]].replace(\"-\", np.nan), errors='coerce').sum(skipna=True)+pd.to_numeric(err_uniform.iloc[complement_index,friends[str(player)]].replace(\"-\", np.nan), errors='coerce').sum(skipna=True))/len(friends[str(player)])\n",
    "                    else:\n",
    "                        util_uniform.iloc[index, player] =pd.to_numeric(err_uniform.iloc[index,friends[str(player)]]).mean(skipna=True)        \n",
    "                \n",
    "                if preference==\"min-PA\":\n",
    "                    if complement_index!=None:\n",
    "                        util_uniform.iloc[index, player]=np.nanmax([pd.to_numeric(err_uniform.iloc[index,friends[str(player)]].replace(\"-\", np.nan), errors='coerce').max(skipna=True),pd.to_numeric(err_uniform.iloc[complement_index,friends[str(player)]].replace(\"-\", np.nan), errors='coerce').max(skipna=True)])\n",
    "                    else:\n",
    "                        util_uniform.iloc[index, player] =pd.to_numeric(err_uniform.iloc[index,friends[str(player)]]).max(skipna=True)\n",
    "                             \n",
    "    \n",
    "    return util_uniform"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Check whether the grand coalition structure is stable? "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 301,
   "metadata": {},
   "outputs": [],
   "source": [
    "def check_stable(utility_uniform, utility_fair,structure):\n",
    "    '''\n",
    "    Check whether the grand coalition structure is stable\n",
    "    \n",
    "    Args:\n",
    "        utility_uniform: Utility without considering fairness. \n",
    "        utility_fair: Utility  considering fairness. \n",
    "        structure:  all possible coalition structures\n",
    "        \n",
    "    Returns: \n",
    "        IF Stable? None/Subset\n",
    "        \n",
    "    \n",
    "    '''\n",
    "    row_ids = utility_uniform.index.tolist()\n",
    "   \n",
    "    grand_colition=row_ids[-1]\n",
    "    grand_value=utility_fair.loc[grand_colition]\n",
    "    for coalitions in structure:\n",
    "        for coaltion in coalitions:\n",
    "            not_nan_indices = utility_uniform.loc[coaltion].notna()\n",
    "\n",
    "            check_less_equal = utility_uniform.loc[coaltion][not_nan_indices].le(grand_value[not_nan_indices])\n",
    "            check_less = utility_uniform.loc[coaltion][not_nan_indices].lt(grand_value[not_nan_indices])\n",
    "\n",
    "            all_columns_less_equal = check_less_equal.all()\n",
    "            at_least_one_column_less = check_less.any()\n",
    "\n",
    "            result = all_columns_less_equal and at_least_one_column_less\n",
    "\n",
    "            if result:\n",
    "                return coaltion\n",
    "    return None"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 302,
   "metadata": {},
   "outputs": [],
   "source": [
    "def check_same_position(str1, str2):\n",
    "    for i in range(len(str1)):\n",
    "        if str1[i] == '1' and str2[i] == '1':\n",
    "            return 1\n",
    "    return 0\n",
    "\n",
    "def generate_combinations(numbers):\n",
    "    target_decimal = pow(2,len(numbers[0]))-1 \n",
    "\n",
    "    def backtrack(curr, idx,total):\n",
    "        if total== target_decimal:\n",
    "            combinations.append(curr)\n",
    "            return\n",
    "            \n",
    "        exist=False\n",
    "        if len(curr)!=0:\n",
    "            for curr_i in curr:\n",
    "                if check_same_position(curr_i,numbers[idx]):\n",
    "                    exist=True\n",
    "                \n",
    "            \n",
    "        if idx<len(numbers)-1:\n",
    "            if exist==False:\n",
    "                backtrack(curr+[numbers[idx]], idx + 1, total + int(numbers[idx], 2))\n",
    "            backtrack(curr, idx + 1, total)\n",
    "    combinations = []\n",
    "    backtrack([], 0, 0)\n",
    "    return combinations\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 303,
   "metadata": {},
   "outputs": [],
   "source": [
    "# clients_dataset_num = [1,5,10,20]\n",
    "clients_dataset_num = [50,100,200]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 304,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/html": [
       "<div>\n",
       "<style scoped>\n",
       "    .dataframe tbody tr th:only-of-type {\n",
       "        vertical-align: middle;\n",
       "    }\n",
       "\n",
       "    .dataframe tbody tr th {\n",
       "        vertical-align: top;\n",
       "    }\n",
       "\n",
       "    .dataframe thead th {\n",
       "        text-align: right;\n",
       "    }\n",
       "</style>\n",
       "<table border=\"1\" class=\"dataframe\">\n",
       "  <thead>\n",
       "    <tr style=\"text-align: right;\">\n",
       "      <th></th>\n",
       "      <th>n_50_err</th>\n",
       "      <th>n_100_err</th>\n",
       "      <th>n_200_err</th>\n",
       "    </tr>\n",
       "  </thead>\n",
       "  <tbody>\n",
       "    <tr>\n",
       "      <th>001</th>\n",
       "      <td>-</td>\n",
       "      <td>-</td>\n",
       "      <td>2.030457</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>010</th>\n",
       "      <td>-</td>\n",
       "      <td>4.123711</td>\n",
       "      <td>-</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>011</th>\n",
       "      <td>-</td>\n",
       "      <td>3.138393</td>\n",
       "      <td>1.80506</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>100</th>\n",
       "      <td>8.510638</td>\n",
       "      <td>-</td>\n",
       "      <td>-</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>101</th>\n",
       "      <td>4.199918</td>\n",
       "      <td>-</td>\n",
       "      <td>1.799918</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>110</th>\n",
       "      <td>4.556165</td>\n",
       "      <td>3.222832</td>\n",
       "      <td>-</td>\n",
       "    </tr>\n",
       "    <tr>\n",
       "      <th>111</th>\n",
       "      <td>3.459037</td>\n",
       "      <td>2.887608</td>\n",
       "      <td>1.744751</td>\n",
       "    </tr>\n",
       "  </tbody>\n",
       "</table>\n",
       "</div>"
      ],
      "text/plain": [
       "     n_50_err n_100_err n_200_err\n",
       "001         -         -  2.030457\n",
       "010         -  4.123711         -\n",
       "011         -  3.138393   1.80506\n",
       "100  8.510638         -         -\n",
       "101  4.199918         -  1.799918\n",
       "110  4.556165  3.222832         -\n",
       "111  3.459037  2.887608  1.744751"
      ]
     },
     "execution_count": 304,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "err_uniform, _, _, _,_ = calc_error_groups(n_list =clients_dataset_num)\n",
    "err_uniform=err_uniform.replace([np.nan],['-'])\n",
    "err_uniform"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 305,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "['001', '010', '011', '100', '101', '110', '111']"
      ]
     },
     "execution_count": 305,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "row_ids = err_uniform.index.tolist()\n",
    "row_ids"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Calculate Optimal Egalitarian Fairness Bound $\\lambda$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 306,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_coalitions(n_list, s_list):\n",
    "    coalitions = []\n",
    "    for s in s_list:\n",
    "        indices = [i for i, x in enumerate(s) if x == '1']\n",
    "        coalitions.append(n_list[indices])\n",
    "    return coalitions"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Test 1. **Calculated egalitarian fairness bound** in Selfish Clients v.s. **Obtained egalitarian fairness**"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 307,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1.7142857142857142"
      ]
     },
     "execution_count": 307,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "clients_dataset_array = np.array(clients_dataset_num)\n",
    "coalitions_except_grand=get_coalitions(clients_dataset_array,row_ids[:-1])\n",
    "\n",
    "m = np.argmin(clients_dataset_array)\n",
    "l=np.argmax(clients_dataset_array)\n",
    "N_g=np.sum(clients_dataset_array)\n",
    "N_g_2_sum=np.sum(clients_dataset_array**2)\n",
    "lambdas_cal=[]\n",
    "for coalition in coalitions_except_grand:\n",
    "    k = np.argmin(coalition)\n",
    "    N_s=np.sum(coalition)\n",
    "    N_s_2_sum=np.sum(coalition**2)\n",
    "    lambdas_cal.append(max((N_s**2/N_g**2)*((N_g*clients_dataset_array[l]+N_g_2_sum-clients_dataset_array[m]**2+(N_g-clients_dataset_array[m])**2)/(N_s*clients_dataset_array[l]+N_s_2_sum-coalition[k]**2+(N_s-coalition[k])**2)),1))\n",
    "    \n",
    "    \n",
    "lambda_cal= max(lambdas_cal)\n",
    "lambda_cal"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 308,
   "metadata": {},
   "outputs": [],
   "source": [
    "err_list = err_uniform.iloc[-1].tolist()\n",
    "\n",
    "q_list=np.arange(0,1,0.05).tolist()\n",
    "lambda_list=[]\n",
    "stable_list=[]\n",
    "\n",
    "\n",
    "coalitions = generate_combinations(row_ids)\n",
    "\n",
    "for q in q_list:\n",
    "    err_uniform, _, _, err_fair,_ = calc_error_groups(n_list =clients_dataset_num,fair=True, err_list=err_list, q=q)\n",
    "    err_fair_last_row=err_fair.iloc[-1]\n",
    "    lambda_list.append(err_list[0]/err_fair_last_row[-1])\n",
    "    stable_result=check_stable(err_uniform,err_fair,coalitions)\n",
    "    stable_list.append(stable_result!=None)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 309,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "\n",
    "\n",
    "fig, ax = plt.subplots(figsize=(4,2))\n",
    "\n",
    "ax.plot(q_list, lambda_list)\n",
    "\n",
    "fair_condition = [index for index, element in enumerate(stable_list) if element == False]\n",
    "ax.plot([q_list[i] for i in fair_condition] , [lambda_list[i] for i in fair_condition], color='red', linewidth=3, label='Core-stable')        \n",
    "ax.axhline(y=lambda_cal, linewidth=3,color='green', linestyle='--',label='Calculated $\\lambda$ bound')\n",
    "plt.xlabel('p')\n",
    "plt.ylabel('$\\lambda$')\n",
    "ax.legend() \n",
    "plt.tight_layout()\n",
    "plt.savefig(\"selfish_reg.svg\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Test 2. **Calculated egalitarian fairness bound** in Purely Altruism v.s. **Obtained egalitarian fairness**"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Full-connected Friendship"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 310,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "{'0': [1, 2], '1': [0, 2], '2': [0, 1]}"
      ]
     },
     "execution_count": 310,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "friends = {}\n",
    "for i in range(len(clients_dataset_num)):\n",
    "    friends[str(i)]=[]\n",
    "    for j in range(len(clients_dataset_num)):\n",
    "        if j!=i:\n",
    "            friends[str(i)].append(j) \n",
    "friends\n",
    "# friends={\"0\": [4,1], \"1\": [0,2],\"2\":[1,3],\"3\":[2,4],\"4\":[3,0]}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 311,
   "metadata": {},
   "outputs": [],
   "source": [
    "# friends = {}\n",
    "# for i in range(len(clients_dataset_num)):\n",
    "#     friends[str(i)]=[]\n",
    "#     friends[str(i)].append((i+1)%len(clients_dataset_num)) \n",
    "#     # friends[str(i)].append((i+2)%len(clients_dataset_num)) \n",
    "#     friends[str(i)].append((i-1)%len(clients_dataset_num) )\n",
    "# friends"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 312,
   "metadata": {},
   "outputs": [],
   "source": [
    "# friends ={'0': [3, 3], '1': [2,2], '2': [3, 1], '3': [2,0]}"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 313,
   "metadata": {},
   "outputs": [],
   "source": [
    "# clients_dataset_array = np.array(clients_dataset_num)\n",
    "# coalitions_except_grand=get_coalitions(clients_dataset_array,row_ids[:-1])\n",
    "\n",
    "# m = np.argmin(clients_dataset_array)\n",
    "# l=np.argmax(clients_dataset_array)\n",
    "# N_g=np.sum(clients_dataset_array)\n",
    "# N_g_2_sum=np.sum(clients_dataset_array**2)\n",
    "# lambdas_cal=[]\n",
    "# friends_array=np.array([np.array(friends[str(i)]) for i in range(clients_dataset_array.shape[0])])\n",
    "# friends_num_array=np.array([np.array([clients_dataset_array[f] for f in friends[str(i)]]) for i in range(clients_dataset_array.shape[0])])\n",
    "\n",
    "# for i,coalition in enumerate(coalitions_except_grand):\n",
    "#     if row_ids[i]==\"10000\":\n",
    "#         print()\n",
    "#     # k=np.argmin(np.min(friends_num_array, axis=1))\n",
    "#     # f_opt=friends_array[k,np.argmin(friends_num_array[k])]  \n",
    "\n",
    "#     # if  row_ids[i][f_opt]==\"1\":\n",
    "#     #     N_s=np.sum(coalition)\n",
    "#     #     N_s_2_sum=np.sum(coalition**2)\n",
    "#     # else:\n",
    "#     #     N_s=N_g-np.sum(coalition)\n",
    "#     #     N_s_2_sum=N_g_2_sum-np.sum(coalition**2)\n",
    "#     # k_list\n",
    "#     c_in_c=[]\n",
    "#     f_in_c=[]\n",
    "#     f_in_c_num=[]\n",
    "#     for j in range(len(row_ids[i])):\n",
    "#         if row_ids[i][j]==\"1\":\n",
    "#             c_in_c.append(j)\n",
    "#             f_in_c.append(np.array([f  if row_ids[i][f]==\"1\" else None for f in friends[str(j)] ]))\n",
    "#             # f_in_c_num.append(np.array([clients_dataset_array[f] if row_ids[i][f]=='1' else float('inf') for f in friends[str(j)]])\n",
    "#             f_in_c_num.append(np.array([clients_dataset_array[f]  for f in friends[str(j)]])\n",
    "# )\n",
    "#     f_in_c=np.array(f_in_c)\n",
    "#     f_in_c_num=np.array(f_in_c_num)\n",
    "#     k = np.argmin(np.min(f_in_c_num, axis=1))\n",
    "#     # f_opt=f_in_c[k,np.argmin(f_in_c_num[k])]\n",
    "#     f_opt=friends_array[c_in_c[k],np.argmin(friends_num_array[c_in_c[k]])]\n",
    "#     N_s=np.sum(coalition)\n",
    "#     N_s_2_sum=np.sum(coalition**2)  \n",
    "    \n",
    "#     if row_ids[i][f_opt]==\"0\":\n",
    "#         N_s=N_g-N_s\n",
    "#         N_s_2_sum=N_g_2_sum-N_s_2_sum\n",
    "#     if f_opt!=None:\n",
    "#         lambdas_cal.append((N_s**2/N_g**2)*((N_g*clients_dataset_array[l]+N_g_2_sum-clients_dataset_array[m]**2+(N_g-clients_dataset_array[m])**2)/(N_s*clients_dataset_array[l]+N_s_2_sum-clients_dataset_array[f_opt]**2+(N_s-clients_dataset_array[f_opt])**2)))\n",
    "        \n",
    "# lambda_cal= max(lambdas_cal) \n",
    "# lambda_cal"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 314,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1.1020408163265307"
      ]
     },
     "execution_count": 314,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "clients_dataset_array = np.array(clients_dataset_num)\n",
    "coalitions_except_grand=get_coalitions(clients_dataset_array,row_ids[:-1])\n",
    "\n",
    "m = np.argmin(clients_dataset_array)\n",
    "l=np.argmax(clients_dataset_array)\n",
    "N_g=np.sum(clients_dataset_array)\n",
    "N_g_2_sum=np.sum(clients_dataset_array**2)\n",
    "lambdas_cal=[]\n",
    "friends_array=np.array([np.array(friends[str(i)]) for i in range(clients_dataset_array.shape[0])])\n",
    "friends_num_array=np.array([np.array([clients_dataset_array[f] for f in friends[str(i)]]) for i in range(clients_dataset_array.shape[0])])\n",
    "\n",
    "for i,coalition in enumerate(coalitions_except_grand):\n",
    "    # k=np.argmin(np.min(friends_num_array, axis=1))\n",
    "    # f_opt=friends_array[k,np.argmin(friends_num_array[k])]  \n",
    "\n",
    "    # if  row_ids[i][f_opt]==\"1\":\n",
    "    #     N_s=np.sum(coalition)\n",
    "    #     N_s_2_sum=np.sum(coalition**2)\n",
    "    # else:\n",
    "    #     N_s=N_g-np.sum(coalition)\n",
    "    #     N_s_2_sum=N_g_2_sum-np.sum(coalition**2)\n",
    "    # k_list\n",
    "    c_in_c=[]\n",
    "    f_in_c=[]\n",
    "    f_in_c_num=[]\n",
    "    f_not_in_c=[]\n",
    "    f_not_in_c_num=[]\n",
    "    for j in range(len(row_ids[i])):\n",
    "        if row_ids[i][j]==\"1\":\n",
    "            c_in_c.append(j)\n",
    "            f_in_c.append(np.array([f  if row_ids[i][f]==\"1\" else None for f in friends[str(j)] ]))\n",
    "            f_not_in_c.append(np.array([f  if row_ids[i][f]==\"0\" else None for f in friends[str(j)] ]))\n",
    "            f_in_c_num.append(np.array([clients_dataset_array[f] if row_ids[i][f]=='1' else float('inf') for f in friends[str(j)]]))\n",
    "            f_not_in_c_num.append(np.array([clients_dataset_array[f] if row_ids[i][f]=='0' else float('inf') for f in friends[str(j)]]))\n",
    "            # f_in_c_num.append(np.array([clients_dataset_array[f]  for f in friends[str(j)]])\n",
    "\n",
    "    f_in_c=np.array(f_in_c)\n",
    "    f_in_c_num=np.array(f_in_c_num)\n",
    "    f_not_in_c=np.array(f_not_in_c)\n",
    "    f_not_in_c_num=np.array(f_not_in_c_num)\n",
    "    k_1 = np.argmin(np.min(f_in_c_num, axis=1))\n",
    "    k_2 = np.argmin(np.min(f_not_in_c_num, axis=1))\n",
    "    # f_opt=f_in_c[k,np.argmin(f_in_c_num[k])]\n",
    "    f_opt_1=f_in_c[k_1,np.argmin(f_in_c_num[k_1])]\n",
    "    f_opt_2=f_not_in_c[k_2,np.argmin(f_not_in_c_num[k_2])]\n",
    "\n",
    "    N_s=np.sum(coalition)\n",
    "    N_s_2_sum=np.sum(coalition**2)  \n",
    "    Not_N_s=N_g-N_s\n",
    "    Not_N_s_2_sum=N_g_2_sum-N_s_2_sum\n",
    "\n",
    "    lambda1=lambda2=float('inf')\n",
    "    if f_opt_1!=None:\n",
    "        lambda1=(N_s**2/N_g**2)*((N_g*clients_dataset_array[l]+N_g_2_sum-clients_dataset_array[m]**2+(N_g-clients_dataset_array[m])**2)/(N_s*clients_dataset_array[l]+N_s_2_sum-clients_dataset_array[f_opt_1]**2+(N_s-clients_dataset_array[f_opt_1])**2))\n",
    "    if f_opt_2!=None:\n",
    "        lambda2=(Not_N_s**2/N_g**2)*((N_g*clients_dataset_array[l]+N_g_2_sum-clients_dataset_array[m]**2+(N_g-clients_dataset_array[m])**2)/(Not_N_s*clients_dataset_array[l]+Not_N_s_2_sum-clients_dataset_array[f_opt_2]**2+(Not_N_s-clients_dataset_array[f_opt_2])**2))\n",
    "\n",
    "\n",
    "    lambdas_cal.append(min(lambda1,lambda2))\n",
    "        \n",
    "lambda_cal= max(lambdas_cal) \n",
    "lambda_cal"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 315,
   "metadata": {},
   "outputs": [],
   "source": [
    "q_list=np.arange(0,4,0.1).tolist()\n",
    "lambda_list=[]\n",
    "stable_list=[]\n",
    "\n",
    "coalitions = generate_combinations(row_ids)\n",
    "err_uniform, _, _, _,_ = calc_error_groups( n_list = clients_dataset_array)\n",
    "err_list = err_uniform.iloc[-1].tolist()\n",
    "util_uniform_ACFG_min_FR=calc_unity_groups(err_uniform=err_uniform, n_list = clients_dataset_array, preference=\"min-PA\",friends=friends)\n",
    "for q in q_list:\n",
    "    _, _, _, err_fair,_ = calc_error_groups( n_list = clients_dataset_array,fair=True, err_list=err_list,q=q)\n",
    "    util_uniform_ACFG_min_FR_i=calc_unity_groups(err_uniform=err_fair, n_list = clients_dataset_array, preference=\"min-PA\",friends=friends)\n",
    "    \n",
    "    err_fair_last_row=err_fair.iloc[-1]\n",
    "    lambda_list.append(err_list[0]/err_fair_last_row[-1])\n",
    "\n",
    "    stable_result=check_stable(util_uniform_ACFG_min_FR,util_uniform_ACFG_min_FR_i,coalitions)\n",
    "    stable_list.append(stable_result!=None)\n",
    "\n",
    "    # print(err_fair)\n",
    "    # print(stable_result)\n",
    "    # print(util_uniform_ACFG_min_FR_i)\n",
    "    # print(max(err_list)/min(err_fair_last_row))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 316,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(figsize=(4,2))\n",
    "\n",
    "ax.plot(q_list, lambda_list)\n",
    "\n",
    "fair_condition = [index for index, element in enumerate(stable_list) if element == False]\n",
    "ax.plot([q_list[i] for i in fair_condition] , [lambda_list[i] for i in fair_condition], color='red', linewidth=3, label='Core-stable')        \n",
    "ax.axhline(y=lambda_cal, linewidth=3,color='green', linestyle='--',label='Calculated $\\lambda$ bound')\n",
    "plt.xlabel('p')\n",
    "plt.ylabel('$\\lambda$')\n",
    "ax.legend() \n",
    "plt.tight_layout()\n",
    "# plt.savefig(\"pwa_reg.svg\")\n",
    "plt.savefig(\"pwa_reg.svg\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Test 3. **Calculated egalitarian fairness bound** in Friendly Welfare Altruism v.s. **Obtained egalitarian fairness**"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 317,
   "metadata": {},
   "outputs": [],
   "source": [
    "w=0.35"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 318,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "0.955329500221141"
      ]
     },
     "execution_count": 318,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "clients_dataset_array = np.array(clients_dataset_num)\n",
    "coalitions_except_grand=get_coalitions(clients_dataset_array,row_ids[:-1])\n",
    "\n",
    "m = np.argmin(clients_dataset_array)\n",
    "l=np.argmax(clients_dataset_array)\n",
    "N_g=np.sum(clients_dataset_array)\n",
    "N_g_2_sum=np.sum(clients_dataset_array**2)\n",
    "lambdas_cal=[]\n",
    "friends_array=np.array([np.array(friends[str(i)]) for i in range(clients_dataset_array.shape[0])])\n",
    "friends_num_array=np.array([np.array([clients_dataset_array[f] for f in friends[str(i)]]) for i in range(clients_dataset_array.shape[0])])\n",
    "\n",
    "for i,coalition in enumerate(coalitions_except_grand):\n",
    "    N_s=np.sum(coalition)\n",
    "    N_s_2_sum=np.sum(coalition**2)  \n",
    "    Not_N_s=N_g-N_s\n",
    "    Not_N_s_2_sum=N_g_2_sum-N_s_2_sum\n",
    "\n",
    "    c_in_s=[]\n",
    "    c_and_f_num=[]\n",
    "    c_and_f_not_num=[]\n",
    "    c_and_f_num_sum=[]\n",
    "    c_and_f_not_num_sum=[]\n",
    "\n",
    "    for j in range(len(row_ids[i])):\n",
    "        if row_ids[i][j]==\"1\":\n",
    "            c_in_s.append(j)\n",
    "            c_and_f_num_sum_i=[float('inf') for c_i in range(len(clients_dataset_num))]\n",
    "\n",
    "            c_and_f_num_sum_i[j]=clients_dataset_array[j]\n",
    "            for f in friends[str(j)] :\n",
    "                if row_ids[i][f]=='1':\n",
    "                    c_and_f_num_sum_i[f]=clients_dataset_array[f]\n",
    "                else:\n",
    "                    c_and_f_num_sum_i[f]=float('inf')\n",
    "            c_and_f_num.append(np.array(c_and_f_num_sum_i))\n",
    "\n",
    "        \n",
    "            c_and_f_not_num_sum_i=[float('inf') for c_i in range(len(clients_dataset_num))]\n",
    "            c_and_f_not_num_sum_i[j]=clients_dataset_array[j]\n",
    "            for f in friends[str(j)] :\n",
    "                if row_ids[i][f]=='0':\n",
    "                    c_and_f_not_num_sum_i[f]=clients_dataset_array[f]\n",
    "                else:\n",
    "                    c_and_f_not_num_sum_i[f]=float('inf')\n",
    "            c_and_f_not_num.append(np.array(c_and_f_not_num_sum_i))\n",
    "\n",
    "            c_and_f_num_sum.append(w*clients_dataset_array[j]+(1-w)*min([clients_dataset_array[j]]+[clients_dataset_array[f]   for f in friends[str(j)] if row_ids[i][f]=='1' ]))\n",
    "            c_and_f_not_num_sum.append(w*clients_dataset_array[j]/N_s+(1-w)*min([clients_dataset_array[j]/N_s]+[clients_dataset_array[f]/Not_N_s   for f in friends[str(j)] if row_ids[i][f]=='0' ]))\n",
    "            \n",
    "            # c_and_f_num_sum.append(w*clients_dataset_array[j]+(1-w)*min([clients_dataset_array[j]]+[clients_dataset_array[f]   for f in friends[str(j)] if row_ids[i][f]=='1' ]))\n",
    "            # c_and_f_not_num_sum.append(w*clients_dataset_array[j]+(1-w)*min([clients_dataset_array[j]]+[clients_dataset_array[f]   for f in friends[str(j)] if row_ids[i][f]=='0' ]))\n",
    "\n",
    "    # k=np.argmin(np.mean(f_in_c_num+f_not_in_c_num*(N_s)/(Not_N_s), axis=1))\n",
    "    \n",
    "    k1=np.argmin(np.array(c_and_f_num_sum))\n",
    "    k2=np.argmin(np.array(c_and_f_not_num_sum))\n",
    "\n",
    "    f1=np.argmin(c_and_f_num[k1])\n",
    "    f2=np.argmin(c_and_f_not_num[k2])\n",
    "    # k = np.argmin(np.mean(f_in_c_num, axis=1)+np.mean(f_not_in_c_num, axis=1))\n",
    "    \n",
    "    lambda1=(N_s**2)*(N_g*clients_dataset_array[l]+N_g_2_sum-clients_dataset_array[m]**2+(N_g-clients_dataset_array[m])**2)/((N_g**2)*((N_s*clients_dataset_array[l])+w*(N_s_2_sum-clients_dataset_array[c_in_s[k1]]**2+(N_s-clients_dataset_array[c_in_s[k1]])**2)+(1-w)*(N_s_2_sum-clients_dataset_array[f1]**2+(N_s-clients_dataset_array[f1])**2)))\n",
    "\n",
    "\n",
    "    lambda2=(N_s**2*Not_N_s**2)*(N_g*clients_dataset_array[l]+N_g_2_sum-clients_dataset_array[m]**2+(N_g-clients_dataset_array[m])**2)/((N_g**2)*(Not_N_s**2*w*(N_s*clients_dataset_array[l]+(N_s_2_sum-clients_dataset_array[c_in_s[k2]]**2+(N_s-clients_dataset_array[c_in_s[k2]])**2))+N_s**2*(1-w)*(Not_N_s*clients_dataset_array[l]+Not_N_s_2_sum-clients_dataset_array[f2]**2+(Not_N_s-clients_dataset_array[f2])**2)))\n",
    "\n",
    "    \n",
    "    lambdas_cal.append(min(lambda1,lambda2))\n",
    "        \n",
    "lambda_cal= max(lambdas_cal) \n",
    "lambda_cal"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 319,
   "metadata": {},
   "outputs": [],
   "source": [
    "q_list=np.arange(2,6,0.1).tolist()\n",
    "lambda_list=[]\n",
    "stable_list=[]\n",
    "\n",
    "coalitions = generate_combinations(row_ids)\n",
    "err_uniform, _, _, _,_ = calc_error_groups( n_list = clients_dataset_array)\n",
    "err_list = err_uniform.iloc[-1].tolist()\n",
    "util_uniform_ACFG_min_FR=calc_unity_groups(err_uniform=err_uniform, n_list = clients_dataset_array, preference=\"min-FR\",friends=friends,w=w)\n",
    "for q in q_list:\n",
    "    _, _, _, err_fair,_ = calc_error_groups( n_list = clients_dataset_array,fair=True, err_list=err_list,q=q)\n",
    "    util_uniform_ACFG_min_FR_i=calc_unity_groups(err_uniform=err_fair, n_list = clients_dataset_array, preference=\"min-FR\",friends=friends,w=w)\n",
    "    \n",
    "    err_fair_last_row=err_fair.iloc[-1]\n",
    "    lambda_list.append(err_list[0]/err_fair_last_row[-1])\n",
    "\n",
    "    stable_result=check_stable(util_uniform_ACFG_min_FR,util_uniform_ACFG_min_FR_i,coalitions)\n",
    "    stable_list.append(stable_result!=None)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 320,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(figsize=(4,2))\n",
    "\n",
    "ax.plot(q_list, lambda_list)\n",
    "\n",
    "fair_condition = [index for index, element in enumerate(stable_list) if element == False]\n",
    "ax.plot([q_list[i] for i in fair_condition] , [lambda_list[i] for i in fair_condition], color='red', linewidth=3, label='Core-stable')        \n",
    "ax.axhline(y=lambda_cal, linewidth=3,color='green', linestyle='--',label='Calculated $\\lambda$ bound')\n",
    "plt.xlabel('p')\n",
    "plt.ylabel('$\\lambda$')\n",
    "ax.legend() \n",
    "plt.tight_layout()\n",
    "plt.savefig(\"fwa_reg.svg\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Test 4. **Calculated egalitarian fairness bound** in Purely Equal Altruism v.s. **Obtained egalitarian fairness**"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 321,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1.4025974025974026"
      ]
     },
     "execution_count": 321,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "clients_dataset_array = np.array(clients_dataset_num)\n",
    "coalitions_except_grand=get_coalitions(clients_dataset_array,row_ids[:-1])\n",
    "\n",
    "m = np.argmin(clients_dataset_array)\n",
    "l=np.argmax(clients_dataset_array)\n",
    "N_g=np.sum(clients_dataset_array)\n",
    "N_g_2_sum=np.sum(clients_dataset_array**2)\n",
    "lambdas_cal=[]\n",
    "friends_array=np.array([np.array(friends[str(i)]) for i in range(clients_dataset_array.shape[0])])\n",
    "friends_num_array=np.array([np.array([clients_dataset_array[f] for f in friends[str(i)]]) for i in range(clients_dataset_array.shape[0])])\n",
    "\n",
    "for i,coalition in enumerate(coalitions_except_grand):\n",
    "    N_s=np.sum(coalition)\n",
    "    N_s_2_sum=np.sum(coalition**2)  \n",
    "    Not_N_s=N_g-N_s\n",
    "    Not_N_s_2_sum=N_g_2_sum-N_s_2_sum\n",
    "\n",
    "    f_in_c=[]\n",
    "    f_all=[]\n",
    "    f_in_c=[]\n",
    "    f_not_in_c=[]\n",
    "    f_in_c_num=[]\n",
    "    f_not_in_c_num=[]\n",
    "    f_stat=[]\n",
    "    for j in range(len(row_ids[i])):\n",
    "        if row_ids[i][j]==\"1\":\n",
    "            f_stat.append([len(friends[str(j)])])\n",
    "            f_all.append(np.array([f  for f in friends[str(j)] ]))\n",
    "            f_in_c.append(np.array([f  if row_ids[i][f]==\"1\" else None for f in friends[str(j)] ]))\n",
    "            f_not_in_c.append(np.array([f  if row_ids[i][f]==\"0\" else None for f in friends[str(j)] ]))\n",
    "            f_in_c_num.append(np.array([clients_dataset_array[f] if row_ids[i][f]=='1' else 0  for f in friends[str(j)] ]))\n",
    "            f_not_in_c_num.append(np.array([clients_dataset_array[f]  if row_ids[i][f]=='0' else 0 for f in friends[str(j)]] ))\n",
    "\n",
    "    f_stat=np.array(f_stat)\n",
    "    f_all=np.array(f_all)\n",
    "    f_in_c=np.array(f_in_c)\n",
    "    f_not_in_c=np.array(f_not_in_c)\n",
    "    f_in_c_num=np.array(f_in_c_num)\n",
    "    f_not_in_c_num=np.array(f_not_in_c_num)\n",
    "\n",
    "\n",
    "    k=np.argmin(np.sum(f_in_c_num/N_s+f_not_in_c_num/Not_N_s,axis=1).reshape(-1,1)/f_stat.reshape(-1,1))\n",
    "    \n",
    "\n",
    "    f_k_num=len(f_all[k].tolist())\n",
    "    sum_c_in_s=sum([N_s*clients_dataset_array[l]+N_s_2_sum-clients_dataset_array[f]**2+(N_s-clients_dataset_array[f])**2 for f in f_in_c[k].tolist()  if f!=None])\n",
    "    sum_c_not_in_s=sum([Not_N_s*clients_dataset_array[l]+Not_N_s_2_sum-clients_dataset_array[f]**2+(Not_N_s-clients_dataset_array[f])**2 for f in f_not_in_c[k].tolist() if f!=None ])\n",
    "    \n",
    "    lambda1=f_k_num*(N_s**2)*(Not_N_s**2)*(N_g*clients_dataset_array[l]+N_g_2_sum-clients_dataset_array[m]**2+(N_g-clients_dataset_array[m])**2)/((N_g**2)*((Not_N_s**2)*sum_c_in_s+(N_s**2)*sum_c_not_in_s))\n",
    "    \n",
    "    # if row_ids[i]==\"1000\":\n",
    "    # sum_c_in_s1=[100/N_s+(N_s_2_sum-clients_dataset_array[f]**2+(N_s-clients_dataset_array[f])**2)/N_s**2 for f in f_in_c[k].tolist()  if f!=None]\n",
    "    # sum_c_not_in_s1=[100/Not_N_s+(Not_N_s_2_sum-clients_dataset_array[f]**2+(Not_N_s-clients_dataset_array[f])**2)/Not_N_s**2 for f in f_not_in_c[k].tolist() if f!=None ]\n",
    "    \n",
    "    # sum_c_in_s1=sum([100/N_s+(N_s_2_sum-clients_dataset_array[f]**2+(N_s-clients_dataset_array[f])**2)/N_s**2 for f in f_in_c[k].tolist()  if f!=None])\n",
    "    # sum_c_not_in_s1=sum([100/Not_N_s+(Not_N_s_2_sum-clients_dataset_array[f]**2+(Not_N_s-clients_dataset_array[f])**2)/Not_N_s**2 for f in f_not_in_c[k].tolist() if f!=None ])\n",
    "    \n",
    "    # lambda_true=(100/N_g+(N_g_2_sum-clients_dataset_array[m]**2+(N_g-clients_dataset_array[m])**2)/N_g**2)/((sum_c_in_s1+sum_c_not_in_s1)/3)\n",
    "\n",
    "    # lambdas_cal.append([lambda1,lambda_true])\n",
    "    lambdas_cal.append(lambda1)\n",
    "        \n",
    "lambda_cal= max(lambdas_cal) \n",
    "lambda_cal"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 322,
   "metadata": {},
   "outputs": [],
   "source": [
    "q_list=np.arange(0,4,0.1).tolist()\n",
    "lambda_list=[]\n",
    "stable_list=[]\n",
    "\n",
    "coalitions = generate_combinations(row_ids)\n",
    "err_uniform, _, _, _,_ = calc_error_groups( n_list = clients_dataset_array)\n",
    "err_list = err_uniform.iloc[-1].tolist()\n",
    "util_uniform_ACFG_min_FR=calc_unity_groups(err_uniform=err_uniform, n_list = clients_dataset_array, preference=\"mean-PA\",friends=friends)\n",
    "for q in q_list:\n",
    "    err_uniform2, _, _, err_fair,_ = calc_error_groups( n_list = clients_dataset_array,fair=True, err_list=err_list,q=q)\n",
    "    util_uniform_ACFG_min_FR_i=calc_unity_groups(err_uniform=err_fair, n_list = clients_dataset_array, preference=\"mean-PA\",friends=friends)\n",
    "    \n",
    "    err_fair_last_row=err_fair.iloc[-1]\n",
    "    lambda_list.append(err_list[0]/err_fair_last_row[-1])\n",
    "\n",
    "    stable_result=check_stable(util_uniform_ACFG_min_FR,util_uniform_ACFG_min_FR_i,coalitions)\n",
    "    stable_list.append(stable_result!=None)\n",
    "    \n",
    "    # print(\"err_uniform2\", err_uniform2)\n",
    "    # print(\"err_fair\", err_fair)\n",
    "    # print(\"util_uniform_ACFG_min_FR\",util_uniform_ACFG_min_FR)\n",
    "    # print(stable_result)\n",
    "    # print(\"util_uniform_ACFG_min_FR_i\", util_uniform_ACFG_min_FR_i)\n",
    "    # print(err_list[0]/err_fair_last_row[-1])\n",
    "    \n",
    "    # if stable_result!=None:\n",
    "    #     print()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 323,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(figsize=(4,2))\n",
    "\n",
    "ax.plot(q_list, lambda_list)\n",
    "\n",
    "fair_condition = [index for index, element in enumerate(stable_list) if element == False]\n",
    "fair_condition.append(len(fair_condition))\n",
    "ax.plot([q_list[i] for i in fair_condition] , [lambda_list[i] for i in fair_condition], color='red', linewidth=3, label='Core-stable')        \n",
    "ax.axhline(y=lambda_cal, linewidth=3,color='green', linestyle='--',label='Calculated $\\lambda$ bound')\n",
    "plt.xlabel('p')\n",
    "plt.ylabel('$\\lambda$')\n",
    "ax.legend() \n",
    "plt.tight_layout()\n",
    "plt.savefig(\"pea_reg.svg\")\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Test 5. **Calculated egalitarian fairness bound** in Friendly Equal Altruism v.s. **Obtained egalitarian fairness**"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 324,
   "metadata": {},
   "outputs": [],
   "source": [
    "# clients_dataset_array = np.array(clients_dataset_num)\n",
    "# coalitions_except_grand=get_coalitions(clients_dataset_array,row_ids[:-1])\n",
    "\n",
    "# m = np.argmin(clients_dataset_array)\n",
    "# l=np.argmax(clients_dataset_array)\n",
    "# N_g=np.sum(clients_dataset_array)\n",
    "# N_g_2_sum=np.sum(clients_dataset_array**2)\n",
    "# lambdas_cal=[]\n",
    "# friends_array=np.array([np.array(friends[str(i)]) for i in range(clients_dataset_array.shape[0])])\n",
    "# friends_num_array=np.array([np.array([clients_dataset_array[f] for f in friends[str(i)]]) for i in range(clients_dataset_array.shape[0])])\n",
    "\n",
    "# for i,coalition in enumerate(coalitions_except_grand):\n",
    "#     N_s=np.sum(coalition)\n",
    "#     N_s_2_sum=np.sum(coalition**2)  \n",
    "#     Not_N_s=N_g-N_s\n",
    "#     Not_N_s_2_sum=N_g_2_sum-N_s_2_sum\n",
    "#     # k=np.argmin(np.min(friends_num_array, axis=1))\n",
    "#     # f_opt=friends_array[k,np.argmin(friends_num_array[k])]  \n",
    "\n",
    "#     # if  row_ids[i][f_opt]==\"1\":\n",
    "#     #     N_s=np.sum(coalition)\n",
    "#     #     N_s_2_sum=np.sum(coalition**2)\n",
    "#     # else:\n",
    "#     #     N_s=N_g-np.sum(coalition)\n",
    "#     #     N_s_2_sum=N_g_2_sum-np.sum(coalition**2)\n",
    "#     # k_list\n",
    "#     c_in_s=[]\n",
    "#     f_in_c=[]\n",
    "#     f_num=[]\n",
    "#     f_all=[]\n",
    "#     f_in_c=[]\n",
    "#     f_not_in_c=[]\n",
    "#     f_in_c_num=[]\n",
    "#     f_not_in_c_num=[]\n",
    "#     f_stat=[]\n",
    "#     for j in range(len(row_ids[i])):\n",
    "        \n",
    "#         if row_ids[i][j]==\"1\":\n",
    "#             c_in_s.append(np.array([clients_dataset_num[j]]))\n",
    "#             f_all.append(np.array([f  for f in friends[str(j)] ]))\n",
    "#             f_stat.append([len(friends[str(j)])])\n",
    "#             f_num.append(np.array([clients_dataset_array[f]  for f in friends[str(j)]]))\n",
    "#             f_in_c.append(np.array([f  if row_ids[i][f]==\"1\" else None for f in friends[str(j)] ]))\n",
    "#             f_not_in_c.append(np.array([f  if row_ids[i][f]==\"0\" else None for f in friends[str(j)] ]))\n",
    "#             f_in_c_num.append(np.array([clients_dataset_array[f] if row_ids[i][f]=='1' else 0  for f in friends[str(j)] ]))\n",
    "#             f_not_in_c_num.append(np.array([clients_dataset_array[f]  if row_ids[i][f]=='0' else 0 for f in friends[str(j)]] ))\n",
    "#             # f_in_c_num.append(np.array([clients_dataset_array[f]  for f in friends[str(j)]])\n",
    "#     c_in_s=np.array(c_in_s)\n",
    "#     f_stat=np.array(f_stat)\n",
    "#     f_all=np.array(f_all)\n",
    "#     f_num=np.array(f_num)\n",
    "#     f_in_c=np.array(f_in_c)\n",
    "#     f_not_in_c=np.array(f_not_in_c)\n",
    "#     f_in_c_num=np.array(f_in_c_num)\n",
    "#     f_not_in_c_num=np.array(f_not_in_c_num)\n",
    "#     t=f_not_in_c_num*(N_s)/(Not_N_s)\n",
    "#     # k=np.argmin(np.mean(f_in_c_num+f_not_in_c_num*(N_s)/(Not_N_s), axis=1))\n",
    "#     # k=np.argmin(w*c_in_s/N_s+(1-w)*((c_in_s+np.sum(f_in_c_num,axis=1).reshape(-1,1))/N_s+np.sum(f_not_in_c_num,axis=1).reshape(-1,1)/Not_N_s)/(f_stat+1))\n",
    "   \n",
    "#     k=np.argmin(w*c_in_s/N_s+(1-w)*((c_in_s+np.sum(f_in_c_num,axis=1).reshape(-1,1))/N_s+np.sum(f_not_in_c_num,axis=1).reshape(-1,1)/Not_N_s)/(f_stat+1))\n",
    "#     # print(k)\n",
    "#     # k=np.argmin(w*c_in_s/N_s+(1-w)*((c_in_s+np.sum(f_in_c_num,axis=1))/N_s+np.sum(f_not_in_c_num,axis=1)/Not_N_s)/(f_stat+1))\n",
    "#     # print(w*c_in_s/N_s+(1-w)*((c_in_s+np.sum(f_in_c_num,axis=1))/N_s+np.sum(f_not_in_c_num,axis=1)/Not_N_s)/(f_stat+1))\n",
    "#     # k=np.argmin(np.mean(f_in_c_num/N_s+f_not_in_c_num/Not_N_s, axis=1))\n",
    "#     # k = np.argmin(np.mean(f_in_c_num, axis=1)+np.mean(f_not_in_c_num, axis=1))\n",
    "    \n",
    "   \n",
    "\n",
    "#     f_k_num=len(f_all[k].tolist())\n",
    "#     sum_c_in_s=sum([N_s*clients_dataset_array[l]+N_s_2_sum-clients_dataset_array[f]**2+(N_s-clients_dataset_array[f])**2 for f in f_in_c[k].tolist()  if f!=None])+(N_s*clients_dataset_array[l]+N_s_2_sum-coalition[k]**2+(N_s-coalition[k])**2)\n",
    "#     sum_c_not_in_s=sum([Not_N_s*clients_dataset_array[l]+Not_N_s_2_sum-clients_dataset_array[f]**2+(Not_N_s-clients_dataset_array[f])**2 for f in f_not_in_c[k].tolist() if f!=None ])\n",
    "    \n",
    "#     lambda1=(f_k_num+1)*(N_s**2)*(Not_N_s**2)*(N_g*clients_dataset_array[l]+N_g_2_sum-clients_dataset_array[m]**2+(N_g-clients_dataset_array[m])**2)/((N_g**2)*(w*(f_k_num+1)*(Not_N_s**2)*(N_s*clients_dataset_array[l]+N_s_2_sum-coalition[k]**2+(N_s-coalition[k])**2)+(1-w)*(Not_N_s**2*sum_c_in_s+N_s**2*sum_c_not_in_s) ))\n",
    "    \n",
    "#     lambda2=(N_s**2/N_g**2)*((N_g*clients_dataset_array[l]+N_g_2_sum-clients_dataset_array[m]**2+(N_g-clients_dataset_array[m])**2)/(N_s*clients_dataset_array[l]+N_s_2_sum-coalition[k]**2+(N_s-coalition[k])**2))\n",
    "#     lambdas_cal.append(lambda1)\n",
    "# lambda_cal= max(lambdas_cal) \n",
    "# lambda_cal"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 325,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "1.2375859434682963"
      ]
     },
     "execution_count": 325,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "clients_dataset_array = np.array(clients_dataset_num)\n",
    "coalitions_except_grand=get_coalitions(clients_dataset_array,row_ids[:-1])\n",
    "\n",
    "m = np.argmin(clients_dataset_array)\n",
    "l=np.argmax(clients_dataset_array)\n",
    "N_g=np.sum(clients_dataset_array)\n",
    "N_g_2_sum=np.sum(clients_dataset_array**2)\n",
    "lambdas_cal=[]\n",
    "\n",
    "for i,coalition in enumerate(coalitions_except_grand):\n",
    "    N_s=np.sum(coalition)\n",
    "    N_s_2_sum=np.sum(coalition**2)  \n",
    "    Not_N_s=N_g-N_s\n",
    "    Not_N_s_2_sum=N_g_2_sum-N_s_2_sum\n",
    "    # k=np.argmin(np.min(friends_num_array, axis=1))\n",
    "    # f_opt=friends_array[k,np.argmin(friends_num_array[k])]  \n",
    "\n",
    "    # if  row_ids[i][f_opt]==\"1\":\n",
    "    #     N_s=np.sum(coalition)\n",
    "    #     N_s_2_sum=np.sum(coalition**2)\n",
    "    # else:\n",
    "    #     N_s=N_g-np.sum(coalition)\n",
    "    #     N_s_2_sum=N_g_2_sum-np.sum(coalition**2)\n",
    "    # k_list\n",
    "    c_in_s=[]\n",
    "    f_in_c=[]\n",
    "    f_num=[]\n",
    "    f_all=[]\n",
    "    f_in_c=[]\n",
    "    f_not_in_c=[]\n",
    "    f_in_c_num=[]\n",
    "    f_not_in_c_num=[]\n",
    "    f_stat=[]\n",
    "    for j in range(len(row_ids[i])):\n",
    "        \n",
    "        if row_ids[i][j]==\"1\":\n",
    "            c_in_s.append(np.array([clients_dataset_num[j]]))\n",
    "            f_all.append([f for f in friends[str(j)] ])\n",
    "            f_stat.append([len(friends[str(j)])])\n",
    "            f_num.append(np.array([clients_dataset_array[f]  for f in friends[str(j)]]))\n",
    "            f_in_c.append(np.array([f  if row_ids[i][f]==\"1\" else None for f in friends[str(j)] ]))\n",
    "            f_not_in_c.append(np.array([f  if row_ids[i][f]==\"0\" else None for f in friends[str(j)] ]))\n",
    "            f_in_c_num.append(np.array(sum([clients_dataset_array[f] if row_ids[i][f]=='1' else 0  for f in friends[str(j)] ])))\n",
    "            \n",
    "            f_not_in_c_num.append(np.array(sum([clients_dataset_array[f]  if row_ids[i][f]=='0' else 0 for f in friends[str(j)]]) ))\n",
    "\n",
    "            # f_in_c_num.append(np.array([clients_dataset_array[f]  for f in friends[str(j)]])\n",
    "           \n",
    "    c_in_s=np.array(c_in_s)\n",
    "    f_stat=np.array(f_stat)\n",
    "\n",
    "    f_in_c_num=np.array(f_in_c_num)\n",
    "    f_not_in_c_num=np.array(f_not_in_c_num)\n",
    "    # k=np.argmin(np.mean(f_in_c_num+f_not_in_c_num*(N_s)/(Not_N_s), axis=1))\n",
    "    # k=np.argmin(w*c_in_s/N_s+(1-w)*((c_in_s+np.sum(f_in_c_num,axis=1).reshape(-1,1))/N_s+np.sum(f_not_in_c_num,axis=1).reshape(-1,1)/Not_N_s)/(f_stat+1))\n",
    "\n",
    "    k=np.argmin(w*c_in_s/N_s+(1-w)*((c_in_s+f_in_c_num.reshape(-1,1))/N_s+f_not_in_c_num.reshape(-1,1)/Not_N_s)/(f_stat+1))\n",
    "    # print(k)\n",
    "    # k=np.argmin(w*c_in_s/N_s+(1-w)*((c_in_s+np.sum(f_in_c_num,axis=1))/N_s+np.sum(f_not_in_c_num,axis=1)/Not_N_s)/(f_stat+1))\n",
    "    # print(w*c_in_s/N_s+(1-w)*((c_in_s+np.sum(f_in_c_num,axis=1))/N_s+np.sum(f_not_in_c_num,axis=1)/Not_N_s)/(f_stat+1))\n",
    "    # k=np.argmin(np.mean(f_in_c_num/N_s+f_not_in_c_num/Not_N_s, axis=1))\n",
    "    # k = np.argmin(np.mean(f_in_c_num, axis=1)+np.mean(f_not_in_c_num, axis=1))\n",
    "    \n",
    "    f_k_num=len(f_all[k])\n",
    "    sum_c_in_s=sum([N_s*clients_dataset_array[l]+N_s_2_sum-clients_dataset_array[f]**2+(N_s-clients_dataset_array[f])**2 for f in f_in_c[k]  if f!=None])+(N_s*clients_dataset_array[l]+N_s_2_sum-coalition[k]**2+(N_s-coalition[k])**2)\n",
    "    sum_c_not_in_s=sum([Not_N_s*clients_dataset_array[l]+Not_N_s_2_sum-clients_dataset_array[f]**2+(Not_N_s-clients_dataset_array[f])**2 for f in f_not_in_c[k] if f!=None ])\n",
    "    \n",
    "    lambda1=(f_k_num+1)*(N_s**2)*(Not_N_s**2)*(N_g*clients_dataset_array[l]+N_g_2_sum-clients_dataset_array[m]**2+(N_g-clients_dataset_array[m])**2)/((N_g**2)*(w*(f_k_num+1)*(Not_N_s**2)*(N_s*clients_dataset_array[l]+N_s_2_sum-coalition[k]**2+(N_s-coalition[k])**2)+(1-w)*(Not_N_s**2*sum_c_in_s+N_s**2*sum_c_not_in_s) ))\n",
    "    \n",
    "    lambda2=(N_s**2/N_g**2)*((N_g*clients_dataset_array[l]+N_g_2_sum-clients_dataset_array[m]**2+(N_g-clients_dataset_array[m])**2)/(N_s*clients_dataset_array[l]+N_s_2_sum-coalition[k]**2+(N_s-coalition[k])**2))\n",
    "    lambdas_cal.append(lambda1)\n",
    "lambda_cal= max(lambdas_cal) \n",
    "lambda_cal"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 326,
   "metadata": {},
   "outputs": [],
   "source": [
    "# clients_dataset_array = np.array(clients_dataset_num)\n",
    "# coalitions_except_grand=get_coalitions(clients_dataset_array,row_ids[:-1])\n",
    "\n",
    "# m = np.argmin(clients_dataset_array)\n",
    "# l=np.argmax(clients_dataset_array)\n",
    "# N_g=np.sum(clients_dataset_array)\n",
    "# N_g_2_sum=np.sum(clients_dataset_array**2)\n",
    "# lambdas_cal=[]\n",
    "# friends_array=np.array([np.array(friends[str(i)]) for i in range(clients_dataset_array.shape[0])])\n",
    "# friends_num_array=np.array([np.array([clients_dataset_array[f] for f in friends[str(i)]]) for i in range(clients_dataset_array.shape[0])])\n",
    "\n",
    "# for i,coalition in enumerate(coalitions_except_grand):\n",
    "#     if row_ids[i]==\"10001\":\n",
    "#             print()\n",
    "#     N_s=np.sum(coalition)\n",
    "#     N_s_2_sum=np.sum(coalition**2)  \n",
    "#     Not_N_s=N_g-N_s\n",
    "#     Not_N_s_2_sum=N_g_2_sum-N_s_2_sum\n",
    "#     # k=np.argmin(np.min(friends_num_array, axis=1))\n",
    "#     # f_opt=friends_array[k,np.argmin(friends_num_array[k])]  \n",
    "\n",
    "#     # if  row_ids[i][f_opt]==\"1\":\n",
    "#     #     N_s=np.sum(coalition)\n",
    "#     #     N_s_2_sum=np.sum(coalition**2)\n",
    "#     # else:\n",
    "#     #     N_s=N_g-np.sum(coalition)\n",
    "#     #     N_s_2_sum=N_g_2_sum-np.sum(coalition**2)\n",
    "#     # k_list\n",
    "#     c_in_s=[]\n",
    "#     f_in_c=[]\n",
    "#     f_num=[]\n",
    "#     f_all=[]\n",
    "#     f_in_c=[]\n",
    "#     f_not_in_c=[]\n",
    "#     f_in_c_num=[]\n",
    "#     f_not_in_c_num=[]\n",
    "#     f_stat=[]\n",
    "#     for j in range(len(row_ids[i])):\n",
    "        \n",
    "#         if row_ids[i][j]==\"1\":\n",
    "#             c_in_s.append(np.array([clients_dataset_num[j]]))\n",
    "#             f_all.append(np.array([f  for f in friends[str(j)] ]))\n",
    "#             f_stat.append([len(friends[str(j)])])\n",
    "#             f_num.append(np.array([clients_dataset_array[f]  for f in friends[str(j)]]))\n",
    "#             f_in_c.append(np.array([f  if row_ids[i][f]==\"1\" else None for f in friends[str(j)] ]))\n",
    "#             f_not_in_c.append(np.array([f  if row_ids[i][f]==\"0\" else None for f in friends[str(j)] ]))\n",
    "#             f_in_c_num.append(np.array([clients_dataset_array[f] if row_ids[i][f]=='1' else 0  for f in friends[str(j)] ]))\n",
    "#             f_not_in_c_num.append(np.array([clients_dataset_array[f]  if row_ids[i][f]=='0' else 0 for f in friends[str(j)]] ))\n",
    "#             # f_in_c_num.append(np.array([clients_dataset_array[f]  for f in friends[str(j)]])\n",
    "#     c_in_s=np.array(c_in_s)\n",
    "#     f_stat=np.array(f_stat)\n",
    "#     f_all=np.array(f_all)\n",
    "#     f_num=np.array(f_num)\n",
    "#     f_in_c=np.array(f_in_c)\n",
    "#     f_not_in_c=np.array(f_not_in_c)\n",
    "#     f_in_c_num=np.array(f_in_c_num)\n",
    "#     f_not_in_c_num=np.array(f_not_in_c_num)\n",
    "#     t=f_not_in_c_num*(N_s)/(Not_N_s)\n",
    "#     # k=np.argmin(np.mean(f_in_c_num+f_not_in_c_num*(N_s)/(Not_N_s), axis=1))\n",
    "#     # k=np.argmin(w*c_in_s/N_s+(1-w)*((c_in_s+np.sum(f_in_c_num,axis=1).reshape(-1,1))/N_s+np.sum(f_not_in_c_num,axis=1).reshape(-1,1)/Not_N_s)/(f_stat+1))\n",
    "#     if row_ids[i]==\"11110\":\n",
    "#         ttt=w*c_in_s/N_s+(1-w)*((c_in_s+np.sum(f_in_c_num,axis=1).reshape(-1,1))/N_s+np.sum(f_not_in_c_num,axis=1).reshape(-1,1)/Not_N_s)/(f_stat+1)\n",
    "#         k=np.argmax(w*c_in_s/N_s+(1-w)*((c_in_s+np.sum(f_in_c_num,axis=1).reshape(-1,1))/N_s+np.sum(f_not_in_c_num,axis=1).reshape(-1,1)/Not_N_s)/(f_stat+1))\n",
    "#     # k=np.argmin(w*c_in_s/N_s+(1-w)*((c_in_s+np.sum(f_in_c_num,axis=1))/N_s+np.sum(f_not_in_c_num,axis=1)/Not_N_s)/(f_stat+1))\n",
    "#     # print(w*c_in_s/N_s+(1-w)*((c_in_s+np.sum(f_in_c_num,axis=1))/N_s+np.sum(f_not_in_c_num,axis=1)/Not_N_s)/(f_stat+1))\n",
    "#     # k=np.argmin(np.mean(f_in_c_num/N_s+f_not_in_c_num/Not_N_s, axis=1))\n",
    "#     # k = np.argmin(np.mean(f_in_c_num, axis=1)+np.mean(f_not_in_c_num, axis=1))\n",
    "    \n",
    "   \n",
    "\n",
    "#         f_k_num=len(f_all[k].tolist())\n",
    "#         sum_c_in_s=sum([N_s*clients_dataset_array[l]+N_s_2_sum-clients_dataset_array[f]**2+(N_s-clients_dataset_array[f])**2 for f in f_in_c[k].tolist()  if f!=None])+(N_s*clients_dataset_array[l]+N_s_2_sum-clients_dataset_array[k]**2+(N_s-clients_dataset_array[k])**2)\n",
    "#         sum_c_not_in_s=sum([Not_N_s*clients_dataset_array[l]+Not_N_s_2_sum-clients_dataset_array[f]**2+(Not_N_s-clients_dataset_array[f])**2 for f in f_not_in_c[k].tolist() if f!=None ])\n",
    "        \n",
    "#         lambda1=(f_k_num+1)*(N_s**2)*(Not_N_s**2)*(N_g*clients_dataset_array[l]+N_g_2_sum-clients_dataset_array[m]**2+(N_g-clients_dataset_array[m])**2)/((N_g**2)*(w*(f_k_num+1)*(Not_N_s**2)*(N_s*clients_dataset_array[l]+N_s_2_sum-clients_dataset_array[k]**2+(N_s-clients_dataset_array[k])**2)+(1-w)*(Not_N_s**2*sum_c_in_s+N_s**2*sum_c_not_in_s) ))\n",
    "        \n",
    "#         lambdas_cal.append(lambda1)\n",
    "# print(lambdas_cal)\n",
    "# lambda_cal= max(lambdas_cal) \n",
    "# lambda_cal"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 327,
   "metadata": {},
   "outputs": [],
   "source": [
    "q_list=np.arange(0,4,0.1).tolist()\n",
    "lambda_list=[]\n",
    "stable_list=[]\n",
    "\n",
    "coalitions = generate_combinations(row_ids)\n",
    "err_uniform, _, _, _,_ = calc_error_groups( n_list = clients_dataset_array)\n",
    "err_list = err_uniform.iloc[-1].tolist()\n",
    "util_uniform_ACFG_min_FR=calc_unity_groups(err_uniform=err_uniform, n_list = clients_dataset_array, preference=\"mean-FR\",friends=friends,w=w)\n",
    "for q in q_list:\n",
    "    _, _, _, err_fair,_ = calc_error_groups( n_list = clients_dataset_array,fair=True, err_list=err_list,q=q)\n",
    "    util_uniform_ACFG_min_FR_i=calc_unity_groups(err_uniform=err_fair, n_list = clients_dataset_array, preference=\"mean-FR\",friends=friends,w=w)\n",
    "    \n",
    "    err_fair_last_row=err_fair.iloc[-1]\n",
    "    lambda_list.append(err_list[0]/err_fair_last_row[-1])\n",
    "\n",
    "    stable_result=check_stable(util_uniform_ACFG_min_FR,util_uniform_ACFG_min_FR_i,coalitions)\n",
    "    stable_list.append(stable_result!=None)\n",
    "    \n",
    "    # print(err_fair)\n",
    "    # print(util_uniform_ACFG_min_FR)\n",
    "    # print(stable_result)\n",
    "    # print(util_uniform_ACFG_min_FR_i)\n",
    "    # print(err_list[0]/err_fair_last_row[-1])\n",
    "    "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 328,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 400x200 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig, ax = plt.subplots(figsize=(4,2))\n",
    "\n",
    "ax.plot(q_list, lambda_list)\n",
    "\n",
    "fair_condition = [index for index, element in enumerate(stable_list) if element == False]\n",
    "ax.plot([q_list[i] for i in fair_condition] , [lambda_list[i] for i in fair_condition], color='red', linewidth=3, label='Core-stable')        \n",
    "ax.axhline(y=lambda_cal, linewidth=3,color='green', linestyle='--',label='Calculated $\\lambda$ bound')\n",
    "plt.xlabel('p')\n",
    "plt.ylabel('$\\lambda$')\n",
    "ax.legend() \n",
    "plt.tight_layout()\n",
    "plt.savefig(\"fea_reg.svg\")\n",
    "plt.show()"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "python38",
   "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.8.16"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
