{
  "cells": [
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "XydaWyMJ1vPv"
      },
      "outputs": [],
      "source": [
        "import numpy as np\n",
        "from collections import Counter\n",
        "import matplotlib.pyplot as plt\n",
        "\n",
        "import seaborn as sns\n",
        "import pandas as pd\n",
        "\n",
        "from scipy.integrate import quad, dblquad\n",
        "from scipy.optimize import fsolve\n",
        "\n",
        "import math\n",
        "from tqdm import tqdm\n",
        "import itertools\n",
        "import random"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "QC86vxph1vPx"
      },
      "outputs": [],
      "source": [
        "def plot_alloc(phi_1_in, phi_2_in, allocations_in, gamma_in, eta_1_in, eta_2_in):\n",
        "    \"\"\"Visualize allocations in the space of phi_1, phi_2\"\"\"\n",
        "    plt.figure(figsize=(8, 6))\n",
        "\n",
        "    # Plot points with color corresponding to allocation\n",
        "    palette = {0: 'slategray', 1: 'gold', 2: 'cornflowerblue'}\n",
        "    df = pd.DataFrame({'phi_1': phi_1_in, 'phi_2': phi_2_in, 'allocations': allocations_in})\n",
        "\n",
        "    sns.scatterplot(data=df, x='phi_1', y='phi_2', hue='allocations', palette=palette, alpha = 0.4, size = 0.1)\n",
        "\n",
        "    # Axis and grid formatting\n",
        "    plt.xlabel(r'$\\phi_1(v_1)$')\n",
        "    plt.ylabel(r'$\\phi_2(v_2)$')\n",
        "    plt.title('Plot of $\\phi_1(v_1)$ vs. $\\phi_2(v_2)$')\n",
        "    plt.grid(True)\n",
        "\n",
        "    # Add axes and dashed lines for relevant regions\n",
        "    plt.axhline(0, color='black', linewidth=0.5)\n",
        "    plt.axvline(0, color='black', linewidth=0.5)\n",
        "\n",
        "    plt.axhline(-gamma_in, color='gray', linestyle='--', label=f'y = -{round(gamma_in, 2)}')\n",
        "    if eta_1_in != 0: plt.axvline(-eta_1_in, color='gray', linestyle='--', label=f'y = -{round(eta_1_in, 2)}')\n",
        "    if eta_2_in != gamma_in: plt.axhline(-eta_2_in, color='gray', linestyle='--', label=f'y = -{round(eta_2_in, 2)}')\n",
        "\n",
        "    # Add line x = y\n",
        "    max_value = max(abs(np.concatenate((phi_1_in, phi_2_in))))\n",
        "    plt.plot([-max_value, max_value], [-max_value, max_value], color='gray', linestyle='--')\n",
        "\n",
        "    # Add line x = y - gamma\n",
        "    plt.plot([-max_value, max_value ], [-max_value - gamma_in, max_value - gamma_in], color='gray', linestyle='--')\n",
        "\n",
        "    # Add legent with labels\n",
        "    legend_labels = {0: 'None', 1: 'Group 1', 2: 'Group 2'}\n",
        "    handles, labels = plt.gca().get_legend_handles_labels()\n",
        "    new_labels = [legend_labels[int(label)] if label.isdigit() else label for label in labels]\n",
        "    plt.legend(handles, new_labels, title='Allocated to')\n",
        "\n",
        "    # Trim range\n",
        "    plt.xlim(min(phi_1_in)-0.1, max(phi_1_in)+0.1)\n",
        "    plt.ylim(min(phi_2_in)-0.1, max(phi_2_in)+0.1)\n",
        "\n",
        "    plt.show()\n",
        "\n",
        "def get_table(pivot_utils, pivot_errors):\n",
        "\n",
        "    pivot_errors = pivot_errors.applymap(lambda x: f'({str(x)})')\n",
        "    pivot_errors.index = [f\"{idx}_error\" for idx in pivot_errors.index]\n",
        "    combined = pd.DataFrame()\n",
        "    max_rows = max(len(pivot_utils), len(pivot_errors))\n",
        "\n",
        "    for i in range(max_rows):\n",
        "        if i < len(pivot_utils):\n",
        "            combined = pd.concat([combined, pivot_utils.iloc[[i]]])\n",
        "        if i < len(pivot_errors):\n",
        "            combined = pd.concat([combined, pivot_errors.iloc[[i]]])\n",
        "\n",
        "    indices = combined.columns.values\n",
        "    sub_indices = [' ', 'error']\n",
        "    new_index = pd.MultiIndex.from_product([indices, sub_indices])\n",
        "\n",
        "    combined.index = new_index\n",
        "\n",
        "    return combined.to_latex()\n",
        "\n",
        "def plot_heatmap(heatmap_data, file_name):\n",
        "    font_size = 16\n",
        "    heatmap_data = heatmap_data.sort_index(axis=0, ascending=False)\n",
        "\n",
        "    plt.figure(figsize=(8, 6))\n",
        "    ax = sns.heatmap(heatmap_data, annot=False, fmt=\".3f\", cmap=\"YlGnBu\")\n",
        "\n",
        "    dim = heatmap_data.shape[0]\n",
        "\n",
        "    ax.set_xticklabels([f'{float(label):.2f}' for label in list(heatmap_data.columns)], fontsize = font_size)\n",
        "    ax.set_yticklabels([f'{float(label):.2f}' for label in list(heatmap_data.columns)][::-1], fontsize = font_size)\n",
        "\n",
        "    ax.set_xlabel(r'$\\alpha_1$', fontsize = font_size)\n",
        "    ax.set_ylabel(r'$\\alpha_2$', fontsize = font_size, rotation=0, labelpad=20)\n",
        "    cbar = ax.collections[0].colorbar\n",
        "    cbar.ax.tick_params(labelsize=20)\n",
        "    plt.xticks(rotation=0)\n",
        "    plt.savefig(file_name, format='png', dpi=300, bbox_inches='tight')\n",
        "    plt.show()\n",
        "\n",
        "def unconstrained_alpha_unif(a_1, b_1, a_2, b_2):\n",
        "    \"\"\"Return alpha_1 and alpha_2, the probabilities of each group winning, in the unconstrained optimal allocation\"\"\"\n",
        "    group_1_alloc_region = (-2*a_2 + b_2)*b_1 + (0.5*b_2*b_2) + ((b_1 - b_2)*b_2)\n",
        "    group_2_alloc_region = b_2*b_2/2 + b_2*(-2*a_1+b_1)\n",
        "\n",
        "    no_alloc_region = (2*a_2 - b_2)*(2*a_1 - b_1)\n",
        "    total_region = (2*b_1 - 2*a_1)*(2*b_2 - 2*a_2)\n",
        "\n",
        "    alpha_1 = group_1_alloc_region/total_region\n",
        "    alpha_2 = group_2_alloc_region/total_region\n",
        "    alpha_0 = no_alloc_region/total_region\n",
        "\n",
        "    return alpha_1, alpha_2, alpha_0\n",
        "\n",
        "def unconstrained_alpha_dynamic(a1, b1, a2, b2, delta, T):\n",
        "    N_it = 100000\n",
        "\n",
        "    v1, v2 = np.random.uniform(a1, b1, N_it), np.random.uniform(a2, b2, N_it)\n",
        "\n",
        "    g1_alloc_prob = sum(v1>=v2)/N_it\n",
        "    g2_alloc_prob = sum(v1<v2)/N_it\n",
        "\n",
        "    last_round_alloc_g1, last_round_alloc_g2, _ = unconstrained_alpha_unif(a1, b1, a1, b2)\n",
        "\n",
        "    g1_dynamic_unconstrained = discounted_average(np.append(np.repeat(g1_alloc_prob, T-1), last_round_alloc_g1), delta)\n",
        "    g2_dynamic_unconstrained = discounted_average(np.append(np.repeat(g2_alloc_prob, T-1), last_round_alloc_g2), delta)\n",
        "    return g1_dynamic_unconstrained, g2_dynamic_unconstrained\n",
        "\n",
        "def alloc_count(alloc_list):\n",
        "  counts = {0: 0, 1: 0, 2: 0}\n",
        "  for num in alloc_list:\n",
        "    counts[num] += 1\n",
        "  return counts\n",
        "\n",
        "def discounted_sum1(util_vec, N_it, delta):\n",
        "    T = len(util_vec)\n",
        "    total_sum = 0\n",
        "    for t in range(1, T+1):\n",
        "        total_sum += (delta ** (t - 1)) * (sum([sum(p) for p in util_vec[t]]) / N_it)\n",
        "    return total_sum\n",
        "\n",
        "def discounted_sum2(util_vec, N_it, delta):\n",
        "    T = len(util_vec)\n",
        "    total_sum = 0\n",
        "    for t in range(1, T+1):\n",
        "        total_sum += (delta ** (t - 1)) * (sum([p for p in util_vec[t]]) / N_it)\n",
        "    return total_sum\n",
        "\n",
        "\n",
        "def discounted_average(vector_length_T, delta):\n",
        "    T = len(vector_length_T)\n",
        "    sum = 0\n",
        "    for index, element in enumerate(vector_length_T):\n",
        "        sum+= element* delta**index\n",
        "    return (1-delta)/(1-delta**T)*sum\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "9G7Btpvq1vPy"
      },
      "outputs": [],
      "source": [
        "def f(x, a, b):\n",
        "    \"\"\"Uniform pdf\"\"\"\n",
        "    return 1/(b - a)\n",
        "\n",
        "def F(x, a, b):\n",
        "    \"\"\"Uniform cdf\"\"\"\n",
        "    return (x - a)/(b - a)\n",
        "\n",
        "def phi_unif(v, a, b):\n",
        "    \"\"\"Virtual value function for v ~ Unif(a, b)\"\"\"\n",
        "    F = (v - a)/(b - a)\n",
        "    f = 1/(b-a)\n",
        "    return v - (1-F)/f"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "M4mkZDti1vPy"
      },
      "outputs": [],
      "source": [
        "def alpha_2_constraint_case1(eta_1, gamma, a_1, b_1, a_2, b_2):\n",
        "    a_1_p = 2*a_1 - b_1\n",
        "    num = (b_2  + gamma + eta_1)*(-eta_1 - a_1_p) + (b_2 + gamma)*eta_1 + 0.5*eta_1**2 + 0.5*(b_2+gamma)**2\n",
        "    denom = (2*b_1 - 2*a_1) * (2*b_2- 2*a_2)\n",
        "    return num / denom\n",
        "\n",
        "def alpha_1_constraint_case1(eta_1, gamma, a_1, b_1, a_2, b_2):\n",
        "    a_2_p = 2*a_2 - b_2\n",
        "    num = (b_1 + eta_1)*(-1*a_2_p - gamma - eta_1) + 0.5*eta_1**2 + b_1*eta_1 + 0.5*(b_2 + gamma)**2 + (b_1 - b_2 - gamma)*(b_2 + gamma)\n",
        "    denom = (2*b_1 - 2*a_1) * (2*b_2- 2*a_2)\n",
        "    return num / denom\n",
        "\n",
        "def alpha_equations_case1(vars, alpha_1, alpha_2, a_1, b_1, a_2, b_2):\n",
        "    eta_1, gamma = vars\n",
        "    eq1 = alpha_2_constraint_case1(eta_1, gamma, a_1, b_1, a_2, b_2) - alpha_2\n",
        "    eq2 = alpha_1_constraint_case1(eta_1, gamma, a_1, b_1, a_2, b_2) - alpha_1\n",
        "    return [eq1, eq2]\n",
        "\n",
        "def alpha_2_constraint_case2(eta_1, gamma, a_1, b_1, a_2, b_2):\n",
        "    a_1_p = 2*a_1 - b_1\n",
        "    num = (b_2  + gamma + eta_1)*(-eta_1 - a_1_p) + (b_2 + gamma)*eta_1 + 0.5*eta_1**2 + 0.5*(b_1)**2+ (b_2 + gamma - b_1)*b_1\n",
        "    denom = (2*b_1 - 2*a_1) * (2*b_2- 2*a_2)\n",
        "    return num / denom\n",
        "\n",
        "def alpha_1_constraint_case2(eta_1, gamma, a_1, b_1, a_2, b_2):\n",
        "    a_2_p = 2*a_2 - b_2\n",
        "    num = (b_1 + eta_1)*(-1*a_2_p - gamma - eta_1) + 0.5*eta_1**2 + b_1*eta_1 + 0.5*b_1**2\n",
        "    denom = (2*b_1 - 2*a_1) * (2*b_2- 2*a_2)\n",
        "    return num / denom\n",
        "\n",
        "def alpha_equations_case2(vars, alpha_1, alpha_2, a_1, b_1, a_2, b_2):\n",
        "    eta_1, gamma = vars\n",
        "    eq1 = alpha_2_constraint_case2(eta_1, gamma, a_1, b_1, a_2, b_2) - alpha_2\n",
        "    eq2 = alpha_1_constraint_case2(eta_1, gamma, a_1, b_1, a_2, b_2) - alpha_1\n",
        "    return [eq1, eq2]\n",
        "\n",
        "\n",
        "def alpha_2_constraint_case1_eta1_zero(gamma, a_1, b_1, a_2, b_2):\n",
        "    eta_1 = 0.0001\n",
        "    a_1_p = 2*a_1 - b_1\n",
        "    num = (b_2  + gamma + eta_1)*(-eta_1 - a_1_p) + (b_2 + gamma)*eta_1 + 0.5*eta_1**2 + 0.5*(b_2+gamma)**2\n",
        "    denom = (2*b_1 - 2*a_1) * (2*b_2- 2*a_2)\n",
        "    return num / denom\n",
        "\n",
        "def alpha_2_constraint_case2_eta1_zero(gamma, a_1, b_1, a_2, b_2):\n",
        "    eta_1 = 0.0001\n",
        "    a_1_p = 2*a_1 - b_1\n",
        "    num = (b_2  + gamma + eta_1)*(-eta_1 - a_1_p) + (b_2 + gamma)*eta_1 + 0.5*eta_1**2 + 0.5*(b_1)**2+ (b_2 + gamma - b_1)*b_1\n",
        "    denom = (2*b_1 - 2*a_1) * (2*b_2- 2*a_2)\n",
        "    return num / denom\n",
        "\n",
        "\n",
        "def alpha_equations_case1_eta1_zero(gamma, alpha_1, alpha_2, a_1, b_1, a_2, b_2):\n",
        "    eq1 = alpha_2_constraint_case1_eta1_zero(gamma, a_1, b_1, a_2, b_2) - alpha_2\n",
        "    return eq1\n",
        "\n",
        "def alpha_equations_case2_eta1_zero(gamma, alpha_1, alpha_2, a_1, b_1, a_2, b_2):\n",
        "    eq1 = alpha_2_constraint_case2_eta1_zero(gamma, a_1, b_1, a_2, b_2) - alpha_2\n",
        "    return eq1\n",
        "\n",
        "\n",
        "def get_allocation_params(alpha_1, alpha_2, a_1, b_1, a_2, b_2):\n",
        "    # initial guess for eta_1 and gamma\n",
        "    initial_guess = [0.5, 0.5]\n",
        "\n",
        "    # solve the system of equations\n",
        "    eta_1, gamma = fsolve(alpha_equations_case1, initial_guess, args=(alpha_1, alpha_2, a_1, b_1, a_2, b_2))\n",
        "\n",
        "    if eta_1 < 0:\n",
        "        #solve for gamma\n",
        "        eta_1 = 0\n",
        "        gamma = fsolve(alpha_equations_case1_eta1_zero, 0.0, args=(alpha_1, alpha_2, a_1, b_1, a_2, b_2))[0]\n",
        "\n",
        "    gamma = max(0, gamma)\n",
        "    eta_1 = max(0, eta_1)\n",
        "    eta_2 = eta_1 + gamma\n",
        "\n",
        "    #large gamma changes geometry\n",
        "    if b_1 < b_2 + gamma:\n",
        "        eta_1, gamma = fsolve(alpha_equations_case2, initial_guess, args=(alpha_1, alpha_2, a_1, b_1, a_2, b_2))\n",
        "\n",
        "        if eta_1 < 0:\n",
        "            #solve for gamma\n",
        "            eta_1 = 0\n",
        "            gamma = fsolve(alpha_equations_case2_eta1_zero, 0.0, args=(alpha_1, alpha_2, a_1, b_1, a_2, b_2))[0]\n",
        "\n",
        "        gamma = max(0, gamma)\n",
        "        eta_1 = max(0, eta_1)\n",
        "        eta_2 = eta_1 + gamma\n",
        "\n",
        "    return eta_1, eta_2, gamma"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "L11aNp6m1vPz"
      },
      "outputs": [],
      "source": [
        "#Functions to calculate nu, mu, G1, G2 in round T and general t\n",
        "\n",
        "def G1_indicator_static(v1, v2, a1, b1, a2, b2, gamma, eta1, eta2):\n",
        "    \"\"\"Indicator function of whether (v1, v2) is in region G1 defined in Theorem 1\"\"\"\n",
        "    phi1, phi2 = phi_unif(v1, a1, b1), phi_unif(v2, a2, b2)\n",
        "    return int(phi1 >= phi2 + gamma)*int(phi1 >= -1*eta1)\n",
        "\n",
        "def G2_indicator_static(v2, v1, a1, b1, a2, b2, gamma, eta1, eta2):\n",
        "    \"\"\"Indicator function of whether (v1, v2) is in region G2 defined in Theorem 1\"\"\"\n",
        "    phi1, phi2 = phi_unif(v1, a1, b1), phi_unif(v2, a2, b2)\n",
        "    return int(phi2 >= phi1 - gamma)*int(phi2 >= -1*eta2)\n",
        "\n",
        "def nu_1_T_integrand(v1, v2, a1, b1, a2, b2, gamma, eta1, eta2, n = 1):\n",
        "    \"\"\"Integrand of nu_1 for equation 16a\"\"\"\n",
        "    return G1_indicator_static(v1, v2, a1, b1, a2, b2, gamma, eta1, eta2) * (1-F(v1, a1, b1))/f(v1, a1, b1) * f(v1, a1, b1)**n * f(v2, a2, b2)**n\n",
        "\n",
        "def nu_2_T_integrand(v1, v2, a1, b1, a2, b2, gamma, eta1, eta2, n = 1):\n",
        "    \"\"\"Integrand of nu_2 equation 16a\"\"\"\n",
        "    return G2_indicator_static(v2, v1, a1, b1, a2, b2, gamma, eta1, eta2) * (1-F(v2, a2, b2))/f(v2, a2, b2) * f(v2, a2, b2)**n * f(v1, a1, b1)**n\n",
        "\n",
        "def mu_T_term_1_integrand(v1, v2, a1, b1, a2, b2, gamma, eta1, eta2, n = 1):\n",
        "    \"\"\"Integrand of first term of mu\"\"\"\n",
        "    return G1_indicator_static(v1, v2, a1, b1, a2, b2, gamma, eta1, eta2) * phi_unif(v1, a1, b1) * f(v1, a1, b1)**n * f(v2, a2, b2)**n\n",
        "\n",
        "def mu_T_term_2_integrand(v1, v2, a1, b1, a2, b2, gamma, eta1, eta2, n = 1):\n",
        "    \"\"\"Integrand of second term of mu\"\"\"\n",
        "    return G2_indicator_static(v2, v1, a1, b1, a2, b2, gamma, eta1, eta2) * phi_unif(v2, a2, b2) * f(v1, a1, b1)**n * f(v2, a2, b2)**n\n",
        "\n",
        "\n",
        "def nu_1_T(R_1_T, R_2_T, delta, T, a1, b1, a2, b2, n=1):\n",
        "    \"\"\"Eqn 16a in long version of paper\"\"\"\n",
        "\n",
        "    alpha1_effective = max(0, R_1_T)\n",
        "    alpha2_effective = max(0, R_2_T)\n",
        "\n",
        "    eta_1, eta_2, gamma = get_allocation_params(alpha1_effective, alpha2_effective, a1, b1, a2, b2)\n",
        "    return (1/n)*dblquad( nu_1_T_integrand, a2, b2, a1, b1, args = (a1, b1, a2, b2, gamma, eta_1, eta_2, 1) )[0]\n",
        "\n",
        "def nu_2_T(R_1_T, R_2_T, delta, T, a1, b1, a2, b2, n=1):\n",
        "    \"\"\"Eqn 16a in long version of paper\"\"\"\n",
        "    alpha1_effective = max(0, R_1_T)\n",
        "    alpha2_effective = max(0, R_2_T)\n",
        "\n",
        "    eta_1, eta_2, gamma = get_allocation_params(alpha1_effective, alpha2_effective, a1, b1, a2, b2)\n",
        "    return (1/n)* dblquad( nu_2_T_integrand, a2, b2, a1, b1, args = (a1, b1, a2, b2, gamma, eta_1, eta_2, 1) )[0]\n",
        "\n",
        "\n",
        "def mu_T(R_1_T, R_2_T, delta, T, a1, b1, a2, b2, n=1):\n",
        "    \"\"\"Eqn 16b in long version of paper\"\"\"\n",
        "    alpha1_effective = max(0, R_1_T)\n",
        "    alpha2_effective = max(0, R_2_T)\n",
        "\n",
        "    eta_1, eta_2, gamma = get_allocation_params(alpha1_effective, alpha2_effective, a1, b1, a2, b2)\n",
        "\n",
        "    term1 = dblquad( mu_T_term_1_integrand, a2, b2, a1, b1, args = (a1, b1, a2, b2, gamma, eta_1, eta_2, 1) )[0]\n",
        "    term2 = dblquad( mu_T_term_2_integrand, a2, b2, a1, b1, args = (a1, b1, a2, b2, gamma, eta_1, eta_2, 1) )[0]\n",
        "    return term1 + term2\n",
        "\n",
        "################################################################################\n",
        "def G1_indicator_dynamic(v1, v2, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1):\n",
        "    \"\"\"Indicator function of whether (v1, v2) is in region G1 defined in Theorem 2\"\"\"\n",
        "    phi1, phi2 = phi_unif(v1, a1, b1), phi_unif(v2, a2, b2)\n",
        "    return int(phi1 - phi2 >= n*delta*(Delta_1_t - Delta_2_t - delta*Delta_0_t))\n",
        "\n",
        "def nu_1_t_integrand(v1, v2, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1):\n",
        "    \"\"\"Integrand of nu_1_t\"\"\"\n",
        "    return G1_indicator_dynamic(v1, v2, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1) * (1-F(v1, a1, b1))/f(v1, a1, b1) * f(v1, a1, b1)**n * f(v2, a2, b2)**n\n",
        "\n",
        "def nu_1_t_integral(a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1):\n",
        "    \"\"\"First term in eqn 23a in long version of paper\"\"\"\n",
        "    return (1/n)*dblquad( nu_1_t_integrand, a2, b2, a1, b1, args = (a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, 1) )[0]\n",
        "\n",
        "def G2_indicator_dynamic(v2, v1, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1):\n",
        "    \"\"\"Indicator function of whether (v1, v2) is in region G2 defined in Theorem 2\"\"\"\n",
        "    phi1, phi2 = phi_unif(v1, a1, b1), phi_unif(v2, a2, b2)\n",
        "    return int(phi1 - phi2 >= n*delta*(Delta_2_t - Delta_1_t + delta*Delta_0_t))\n",
        "\n",
        "def nu_2_t_integrand(v1, v2, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1):\n",
        "    \"\"\"Integrand of nu_2_t\"\"\"\n",
        "    return G2_indicator_dynamic(v2, v1, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1) * (1-F(v2, a2, b2))/f(v2, a2, b2) * f(v1, a1, b1)**n * f(v2, a2, b2)**n\n",
        "\n",
        "def nu_2_t_integral(a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1):\n",
        "    \"\"\"First term in eqn 23a in long version of paper\"\"\"\n",
        "    return (1/n)*dblquad( nu_2_t_integrand, a2, b2, a1, b1, args = (a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, 1) )[0]"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "3SUUTWjv1vPz"
      },
      "outputs": [],
      "source": [
        "def alpha_to_R(alpha, delta, T):\n",
        "    \"\"\"determine residual minimum allocation in round 1 given alpha1, alpha2\"\"\"\n",
        "    #return (1/delta**(T-2)) * (1-delta**T)/(1-delta)*alpha\n",
        "    return  ((1-delta**T)/(1-delta))*alpha\n",
        "\n",
        "\n",
        "def effective_alpha_i(R_i_t, delta, t, T):\n",
        "    \"\"\"Given a residual minimum allocation at round t, return the equivalent fairness constraint if you were to start from round t\"\"\"\n",
        "    return R_i_t*(1-delta)/(1-delta**(T-t))"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "6X6v1f1U1vPz"
      },
      "outputs": [],
      "source": [
        "\n",
        "def generate_possible_allocations(T):\n",
        "    \"\"\"Return x_1=(x_1(t=1)...x_1(t=T)) (allocation for group 1 over T round) and x_2 (same for group 2)\"\"\"\n",
        "    all = list(itertools.product([0, 1], repeat=T-1))\n",
        "    pairs = []\n",
        "    for vector in all:\n",
        "        flip = tuple(1 - bit for bit in vector)\n",
        "        pairs.append((vector, flip))\n",
        "    return pairs\n",
        "\n",
        "def generate_possible_rounds(T):\n",
        "    rounds = []\n",
        "    for t in range(1, T):\n",
        "        combinations = list(itertools.product([0, 1], repeat=t))\n",
        "        str_combinations = [''.join(map(str, comb)) for comb in combinations]\n",
        "        rounds.append(str_combinations)\n",
        "    return rounds\n",
        "\n",
        "def residual_minimum_allocation_update(R_init, alloc, delta):\n",
        "    return 1/delta*(R_init - alloc)\n",
        "\n",
        "def R(R_init, alloc_string, delta):\n",
        "    R_current = R_init\n",
        "    for alloc in list(map(int, alloc_string)):\n",
        "        R_current = residual_minimum_allocation_update(R_current, alloc, delta)\n",
        "    return max(R_current, 0)\n",
        "\n",
        "def get_possible_R_T_pairs(R1_1, R2_1, delta, T):\n",
        "    \"\"\"Return list of all possible pairs of R_1_T, R_2_T (residual minimum allocation in last round)\"\"\"\n",
        "    possible_allocations = generate_possible_allocations(T)\n",
        "\n",
        "    possible_R_pairs = [(R(R1_1, ''.join(str(num) for num in a[0]), delta), R(R2_1, ''.join(str(num) for num in a[1]), delta)) for a in possible_allocations]\n",
        "\n",
        "    return (possible_allocations, possible_R_pairs)\n",
        "\n",
        "def eliminate_infeasible(value_vec, infeasible_vec):\n",
        "    copy_vec = value_vec\n",
        "    for inf in infeasible_vec.keys():\n",
        "        if inf in value_vec.keys() and infeasible_vec[inf] == 1:\n",
        "            copy_vec[inf] = -99999\n",
        "    return copy_vec\n",
        "\n",
        "def mu_update_integral_1_group_infeasible(v, a1, b1):\n",
        "    return phi_unif(v, a1, b1)*F(v, a1, b1)\n",
        "\n",
        "def nu_update_integral_1_group_infeasible(v, a1, b1, n):\n",
        "    return (1-F(v, a1, b1))*F(v, a1, b1)**n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "rJXuVW-U1vP0"
      },
      "outputs": [],
      "source": [
        "def group1_static_payment_integrand(z, b_1, v_2, b_2, eta_1, eta_2, gamma_in):\n",
        "    #phi_1 > phi_2 + gamma and phi_1 > -eta_1\n",
        "    return int(2*z - b_1 > (2*v_2 - b_2) + gamma_in) * int(2*z - b_1 > -eta_1)\n",
        "\n",
        "def group2_static_payment_integrand(z, b_1, v_1, b_2, eta_1, eta_2, gamma_in, static = True):\n",
        "    #phi_2 > phi_1 - gamma and phi_1 > -eta_2\n",
        "    return int(2*z - b_2 > (2*v_1 - b_1) - gamma_in) * int( (2*v_1 - b_1) > -eta_2)"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "O-0o-5zn1vP0"
      },
      "outputs": [],
      "source": [
        "def mu_t_integrand_G1(v1, v2, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, n = 1):\n",
        "    phi1, phi2 = phi_unif(v1, a1, b1), phi_unif(v2, a2, b2)\n",
        "    return G1_indicator_dynamic(v1, v2, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta) * phi1 * f(v1, a1, b1)**n * f(v2, a2, b2)**n\n",
        "\n",
        "def mu_t_integral_G1(a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n = 1):\n",
        "    \"\"\"G1 integrand of mu_t\"\"\"\n",
        "    return dblquad( mu_t_integrand_G1, a2, b2, a1, b1, args = (a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta) )[0]\n",
        "\n",
        "def mu_t_integrand_G2(v1, v2, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, n = 1):\n",
        "    phi1, phi2 = phi_unif(v1, a1, b1), phi_unif(v2, a2, b2)\n",
        "    return G2_indicator_dynamic(v2, v1, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1) * phi2 * f(v1, a1, b1)**n * f(v2, a2, b2)**n\n",
        "\n",
        "def mu_t_integral_G2(a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n = 1):\n",
        "    \"\"\"G2 integrand of mu_t\"\"\"\n",
        "    return dblquad( mu_t_integrand_G2, a2, b2, a1, b1, args = (a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta) )[0]\n",
        "\n",
        "\n",
        "def P_G1_integrand(v1, v2, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1):\n",
        "    return G1_indicator_dynamic(v1, v2, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1)*f(v1, a1, b1)**n * f(v2, a2, b2)**n\n",
        "\n",
        "def P_G1( a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1):\n",
        "    \"\"\"Calculate P(G_1)\"\"\"\n",
        "    return dblquad( P_G1_integrand, a2, b2, a1, b1, args = (a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta) )[0]\n",
        "\n",
        "\n",
        "def P_G2_integrand(v1, v2, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1):\n",
        "    return G2_indicator_dynamic(v2, v1, a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1)*f(v1, a1, b1)**n * f(v2, a2, b2)**n\n",
        "\n",
        "def P_G2( a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta, n=1):\n",
        "    \"\"\"Calculate P(G_2)\"\"\"\n",
        "    return dblquad( P_G2_integrand, a2, b2, a1, b1, args = (a1, b1, a2, b2, Delta_1_t, Delta_2_t, Delta_0_t, delta) )[0]"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "psTjeUd01vP2"
      },
      "outputs": [],
      "source": [
        "def get_utilities(alpha1, alpha2, N_it, T):\n",
        "\n",
        "    n = 1\n",
        "    delta = 0.99\n",
        "\n",
        "    a1 = 0.5\n",
        "    b1 = 1.5\n",
        "\n",
        "    a2 = 0\n",
        "    b2 = 1\n",
        "\n",
        "    neg_infinity = -9999\n",
        "\n",
        "    g1_dynamic_unconstrained, g2_dynamic_unconstrained = unconstrained_alpha_dynamic(a1, b1, a2, b2, delta, T)\n",
        "\n",
        "    print('Unconstrained: Group 1 wins with probability ',  g1_dynamic_unconstrained)\n",
        "    print('Unconstrained: Group 2 wins with probability ',  g2_dynamic_unconstrained)\n",
        "\n",
        "    ##############################################################\n",
        "    #determine residual minimum allocation in round t given alpha1, alpha2\n",
        "    R_1_1 = max(0, (1 - delta**T)/(1-delta)*alpha1)\n",
        "    R_2_1 = max(0,(1 - delta**T)/(1-delta)*alpha2)\n",
        "\n",
        "    R_dict = {}\n",
        "    for t in range(1,T+2):\n",
        "        possible_allocations, possible_R_T_pairs = get_possible_R_T_pairs(R_1_1, R_2_1, delta, t)\n",
        "        possible_rounds = generate_possible_rounds(T)\n",
        "        R_dict[t] = dict(zip([''.join(map(str, item[0])) for item in possible_allocations], possible_R_T_pairs))\n",
        "\n",
        "    R_sum = {}\n",
        "    for key, values in R_dict.items():\n",
        "        for v in values.items():\n",
        "            R_sum[v[0]] = sum(v[1])\n",
        "\n",
        "    #determine infeasibility based on possible R values\n",
        "    infeasible = {}\n",
        "\n",
        "    for round in [r for rs in possible_rounds for r in rs]:\n",
        "        gr_1 = [key for key, value in R_sum.items() if value > 1]\n",
        "        subsequent_rounds = [key for key in R_sum.keys() if key.startswith(round)]\n",
        "        infeasible[round]=int(all(item in gr_1 for item in subsequent_rounds))\n",
        "\n",
        "    if sum([infeasible[p]==1 for p in possible_rounds[-1]]) == len(possible_rounds[-1]):\n",
        "        print('Auction is not feasible with these parameters')\n",
        "        return (np.nan, np.nan), (np.nan, np.nan), (np.nan, np.nan)\n",
        "\n",
        "    ##############################################################\n",
        "\n",
        "    nu_1_dict = {i: {} for i in range(1, T + 1)}\n",
        "    nu_2_dict = {i: {} for i in range(1, T + 1)}\n",
        "    mu_dict = {i: {} for i in range(1, T + 1)}\n",
        "\n",
        "    Delta_1_dict = {i: {} for i in range(1, T + 1)}\n",
        "    Delta_2_dict = {i: {} for i in range(1, T + 1)}\n",
        "    Delta_0_dict = {i: {} for i in range(1, T + 1)}\n",
        "\n",
        "    nu_integral_G1_dict = {i: {} for i in range(1, T + 1)}\n",
        "    nu_integral_G2_dict = {i: {} for i in range(1, T + 1)}\n",
        "    mu_integral_G1_dict = {i: {} for i in range(1, T + 1)}\n",
        "    mu_integral_G2_dict = {i: {} for i in range(1, T + 1)}\n",
        "    P_G1_dict = {i: {} for i in range(1, T + 1)}\n",
        "    P_G2_dict = {i: {} for i in range(1, T + 1)}\n",
        "    ##############################################################\n",
        "    #calculate all possible nu_i_T and mu_T values\n",
        "    nu_1_dict[T] = {key: nu_1_T(value[0], value[1], delta, T, a1, b1, a2, b2, n=1) for key, value in R_dict[T].items()}\n",
        "    nu_2_dict[T] = {key: nu_2_T(value[0], value[1], delta, T, a1, b1, a2, b2, n=1) for key, value in R_dict[T].items()}\n",
        "    mu_dict[T] = {key: mu_T(value[0], value[1], delta, T, a1, b1, a2, b2, n=1) for key, value in R_dict[T].items()}\n",
        "\n",
        "    for round in mu_dict[T]:\n",
        "        if infeasible[round] == 1:\n",
        "            mu_dict[T][round] = neg_infinity\n",
        "\n",
        "    zeta_1 = 0 #note that we set zeta to 0 for n = 1.\n",
        "    zeta_2 = 0 #note that we set zeta to 0 for n = 1.\n",
        "\n",
        "    ##############################################################\n",
        "    possible_rounds = generate_possible_rounds(T)\n",
        "\n",
        "    for i in range(3, T+1):\n",
        "        j = i-3\n",
        "        for r in possible_rounds[T-i]:\n",
        "            Delta_1_dict[T-1-j][r] = nu_1_dict[T-j][r+'0'] - nu_1_dict[T-j][r+'1']\n",
        "            Delta_2_dict[T-1-j][r] = nu_2_dict[T-j][r+'1'] - nu_2_dict[T-j][r+'0']\n",
        "            Delta_0_dict[T-1-j][r] = mu_dict[T-j][r+'1'] - mu_dict[T-j][r+'0']\n",
        "\n",
        "            g1_infeasible = mu_dict[T-j][r+'1'] == neg_infinity\n",
        "            g2_infeasible = mu_dict[T-j][r+'0'] == neg_infinity\n",
        "\n",
        "            #calculate nu by equation 23a\n",
        "            #add some if statement here that checks if the later mu is negative infinty and do 19c rule instead\n",
        "\n",
        "            nu_integral_G1_dict[T-1-j][r] = nu_1_t_integral(a1, b1, a2, b2, Delta_1_dict[T-1-j][r], Delta_2_dict[T-1-j][r], Delta_0_dict[T-1-j][r], delta, n=1)\n",
        "            nu_integral_G2_dict[T-1-j][r] = nu_2_t_integral(a1, b1, a2, b2, Delta_1_dict[T-1-j][r], Delta_2_dict[T-1-j][r], Delta_0_dict[T-1-j][r], delta, n=1)\n",
        "\n",
        "            nu_1_dict[T-1-j][r] = nu_integral_G1_dict[T-1-j][r] + delta*nu_1_dict[T-j][r+'0'] - delta*Delta_1_dict[T-1-j][r]*zeta_1\n",
        "            nu_2_dict[T-1-j][r] = nu_integral_G2_dict[T-1-j][r] + delta*nu_2_dict[T-j][r+'1'] - delta*Delta_2_dict[T-1-j][r]*zeta_2\n",
        "\n",
        "            #calculate mu by equation 23b or 19c\n",
        "\n",
        "            if g1_infeasible and not g2_infeasible:\n",
        "                #print('Have to give to group 2')\n",
        "                mu_dict[T-1-j][r] = quad(mu_update_integral_1_group_infeasible, a2, b2, args=(a2, b2))[0] + delta*mu_dict[T-j][r + '0']\n",
        "                nu_2_dict[T-1-j][r] = quad(nu_update_integral_1_group_infeasible, a2, b2, args=(a2, b2, n))[0] + delta*nu_2_dict[T-j][r + '0']\n",
        "                nu_1_dict[T-1-j][r] = delta*nu_1_dict[T-j][r + '0']\n",
        "\n",
        "\n",
        "            if g2_infeasible and not g1_infeasible:\n",
        "                #print('Have to give to group 1')\n",
        "                mu_dict[T-1-j][r] = quad(mu_update_integral_1_group_infeasible, a1, b1, args=(a1, b1))[0] + delta*mu_dict[T-j][r + '1']\n",
        "                nu_1_dict[T-1-j][r] = quad(nu_update_integral_1_group_infeasible, a1, b1, args=(a1, b1, n))[0] + delta*nu_1_dict[T-j][r + '1']\n",
        "                nu_2_dict[T-1-j][r] = delta*nu_2_dict[T][r + '1']\n",
        "\n",
        "            if g1_infeasible and g2_infeasible:\n",
        "                print('Neither feasible. Issue.', r, T-1-j, alpha1, alpha2, T)\n",
        "                mu_dict[T-1-j][r] = 0\n",
        "                nu_1_dict[T-1-j][r] = 0\n",
        "                nu_2_dict[T-1-j][r] = 0\n",
        "\n",
        "            if not g1_infeasible and not g2_infeasible:\n",
        "                #print('Either are feasible. ')\n",
        "                mu_integral_G1_dict[T-1-j][r] = mu_t_integral_G1(a1, b1, a2, b2, Delta_1_dict[T-1-j][r], Delta_2_dict[T-1-j][r], Delta_0_dict[T-1-j][r], delta, n=1)\n",
        "                mu_integral_G2_dict[T-1-j][r] = mu_t_integral_G2(a1, b1, a2, b2, Delta_1_dict[T-1-j][r], Delta_2_dict[T-1-j][r], Delta_0_dict[T-1-j][r], delta, n=1)\n",
        "\n",
        "                P_G1_dict[T-1-j][r] = P_G1( a1, b1, a2, b2, Delta_1_dict[T-1-j][r], Delta_2_dict[T-1-j][r], Delta_0_dict[T-1-j][r], delta, 1)\n",
        "                P_G2_dict[T-1-j][r] = P_G2( a1, b1, a2, b2, Delta_1_dict[T-1-j][r], Delta_2_dict[T-1-j][r], Delta_0_dict[T-1-j][r], delta, 1)\n",
        "\n",
        "                term1 = mu_integral_G1_dict[T-1-j][r] + delta*P_G1_dict[T-1-j][r]*mu_dict[T-j][r+'1'] + delta*n*Delta_1_dict[T-1-j][r]*(zeta_1-P_G1_dict[T-1-j][r])\n",
        "                term2 = mu_integral_G2_dict[T-1-j][r] + delta*P_G2_dict[T-1-j][r]*mu_dict[T-j][r+'0'] + delta*n*Delta_2_dict[T-1-j][r]*(zeta_2-P_G2_dict[T-1-j][r])\n",
        "\n",
        "                mu_dict[T-1-j][r] = term1 + term2\n",
        "\n",
        "                #update nu1 and nu2\n",
        "                nu_integral_G1_dict[T-1-j][r] = nu_1_t_integral(a1, b1, a2, b2, Delta_1_dict[T-1-j][r], Delta_2_dict[T-1-j][r], Delta_0_dict[T-1-j][r], delta, n=1)\n",
        "                nu_integral_G2_dict[T-1-j][r] = nu_2_t_integral(a1, b1, a2, b2, Delta_1_dict[T-1-j][r], Delta_2_dict[T-1-j][r], Delta_0_dict[T-1-j][r], delta, n=1)\n",
        "\n",
        "                nu_1_dict[T-1-j][r] = nu_integral_G1_dict[T-1-j][r] + delta*nu_1_dict[T-j][r+'0'] - delta*Delta_1_dict[T-1-j][r]*zeta_1\n",
        "                nu_2_dict[T-1-j][r] = nu_integral_G2_dict[T-1-j][r] + delta*nu_2_dict[T-j][r+'1'] - delta*Delta_2_dict[T-1-j][r]*zeta_2\n",
        "\n",
        "    ##############################################################\n",
        "    #calculate first round parameters after calculating all inner round parameters\n",
        "    RHS_G1 = {}\n",
        "    Delta_1_dict[1][''] = nu_1_dict[2]['0'] - nu_1_dict[2]['1']\n",
        "    Delta_2_dict[1][''] = nu_2_dict[2]['1'] - nu_2_dict[2]['0']\n",
        "    Delta_0_dict[1][''] = mu_dict[2]['1'] - mu_dict[2]['0']\n",
        "\n",
        "    alloc_dict = {i: [] for i in range(1, T + 1)}\n",
        "    payment_dict = {i: [] for i in range(1, T + 1)}\n",
        "    phi_1_dict = {i: [] for i in range(1, T + 1)}\n",
        "    phi_2_dict = {i: [] for i in range(1, T + 1)}\n",
        "    buyer_1_val_dict = {i: [] for i in range(1, T + 1)}\n",
        "    buyer_2_val_dict = {i: [] for i in range(1, T + 1)}\n",
        "    history_vec = []\n",
        "    ##############################################################\n",
        "\n",
        "\n",
        "    for i in range(N_it):\n",
        "        infeasible_record = {i: False for i in range(1, T + 1)}\n",
        "\n",
        "        #draw values and virtual values\n",
        "        v1, v2 = np.random.uniform(a1, b1, T), np.random.uniform(a2, b2, T)\n",
        "        phi_1, phi_2 = phi_unif(v1, a1, b1), phi_unif(v2, a2, b2)\n",
        "\n",
        "        for i in range(1,T+1):\n",
        "            phi_1_dict[i].append(phi_1[i-1])\n",
        "            phi_2_dict[i].append(phi_2[i-1])\n",
        "\n",
        "        #rounds 1...T-1\n",
        "        history = ''\n",
        "\n",
        "        for t in range(1, T):\n",
        "            RHS_G1[t] = n*delta*(Delta_1_dict[t][history] - Delta_2_dict[t][history] - delta*Delta_0_dict[t][history])\n",
        "            x_1_t = (phi_1[t-1] - phi_2[t-1] > RHS_G1[t]).astype(int)\n",
        "            #if this allocation would be infeasible, allocate to the other group\n",
        "            if infeasible[history + str(x_1_t)] == 1:\n",
        "                x_1_t = 1-x_1_t\n",
        "                infeasible_record[t] = True\n",
        "            history += str(x_1_t)\n",
        "            alloc_dict[t].append((0, x_1_t, (1-x_1_t)))\n",
        "\n",
        "            #get payment in round 1\n",
        "            payment_integral_1 = quad(G1_indicator_dynamic, a1, v1[t-1], args = (v2[t-1], a1, b1, a2, b2, Delta_1_dict[t][history[:-1]], Delta_2_dict[t][history[:-1]], Delta_0_dict[t][history[:-1]], delta, 1))[0]\n",
        "            payment_1 = x_1_t*v1[t-1] - payment_integral_1 + delta*Delta_1_dict[t][history[:-1]]*(zeta_1 - x_1_t) #second argument of this term is 1 for n=1\n",
        "\n",
        "            payment_integral_2 = quad(G2_indicator_dynamic, a2, v2[t-1], args = (v1[t-1], a1, b1, a2, b2, Delta_1_dict[t][history[:-1]], Delta_2_dict[t][history[:-1]], Delta_0_dict[t][history[:-1]], delta, 1))[0]\n",
        "            payment_2 = (1-x_1_t)*v2[t-1] - payment_integral_2 + delta*Delta_2_dict[t][history[:-1]]*(zeta_2 - (1-x_1_t)) #second argument of this term is 1 for n=1\n",
        "\n",
        "            if infeasible_record[t] == True:\n",
        "                    payment_1 = 0\n",
        "                    payment_2 = 0\n",
        "\n",
        "            payment_dict[t].append((payment_1, payment_2))\n",
        "            buyer_1_val_dict[t].append(x_1_t*v1[t-1]-payment_1)\n",
        "            buyer_2_val_dict[t].append((1-x_1_t)*v2[t-1]-payment_2)\n",
        "    ##############################################################\n",
        "\n",
        "        #T-th round is static\n",
        "        R_1_T, R_2_T = R_dict[T][history]\n",
        "\n",
        "        alpha1_T = R_1_T\n",
        "        alpha2_T = R_2_T\n",
        "        eta_1, eta_2, gamma = get_allocation_params(alpha1_T, alpha2_T, a1, b1, a2, b2)\n",
        "\n",
        "        x_1_T = (phi_1[T-1] > phi_2[T-1] + gamma).astype(int) * (phi_1[T-1] > - eta_1).astype(int)\n",
        "        x_2_T = (phi_2[T-1] > phi_1[T-1] - gamma).astype(int) * (phi_2[T-1] > - eta_2).astype(int)\n",
        "        x_0_T = 1 - (x_1_T+x_2_T)\n",
        "        history += str(x_1_T)\n",
        "\n",
        "        alloc_dict[T].append((x_0_T, x_1_T, x_2_T))\n",
        "\n",
        "        #3rd round payment\n",
        "        payment_integral_1 = quad(group1_static_payment_integrand, a1, v1[T-1], args = (b1, v2[T-1], b2, eta_1, eta_2, gamma))[0]\n",
        "        payment_1 = x_1_T*v1[T-1] - payment_integral_1\n",
        "\n",
        "        payment_integral_2 = quad(group2_static_payment_integrand, a2, v2[T-1], args = (b1, v1[T-1], b2, eta_1, eta_2, gamma))[0]\n",
        "        payment_2 = (1-x_1_T)*v2[T-1] - payment_integral_2\n",
        "\n",
        "        payment_dict[T].append((x_1_T*payment_1, x_2_T*payment_2))\n",
        "        buyer_1_val_dict[T].append(x_1_T*v1[T-1] - payment_1)\n",
        "        buyer_2_val_dict[T].append((1-x_1_T)*v2[T-1]-payment_2)\n",
        "\n",
        "        history_vec.append(history)\n",
        "\n",
        "\n",
        "    ##############################################################\n",
        "    alloc_ind_dict = {i: [] for i in range(1, T + 1)}\n",
        "    alloc_counts = {i: [] for i in range(1, T + 1)}\n",
        "\n",
        "    for i in range(1, T+1):\n",
        "        alloc_ind_dict[i] = [row.index(1)  for row in alloc_dict[i]]\n",
        "        alloc_counts[i] = alloc_count(alloc_ind_dict[i])\n",
        "\n",
        "    print('Constrained: Group 1 wins with probability ', (1-delta)/(1-delta**T)*(alloc_counts[1][1]/N_it + delta * alloc_counts[2][1]/N_it + delta**2 * alloc_counts[3][1]/N_it))\n",
        "    print('Constrained: Group 2 wins with probability ', (1-delta)/(1-delta**T)*(alloc_counts[1][2]/N_it + delta * alloc_counts[2][2]/N_it + delta**2 * alloc_counts[3][2]/N_it))\n",
        "\n",
        "\n",
        "    seller_discounted_avg_util = discounted_sum1(payment_dict, N_it, delta)\n",
        "    buyer1_discounted_avg_util = discounted_sum2(buyer_1_val_dict, N_it, delta)\n",
        "    buyer2_discounted_avg_util = discounted_sum2(buyer_2_val_dict, N_it, delta)\n",
        "\n",
        "    summed_payment_dict = {key: [sum(tup) for tup in value] for key, value in payment_dict.items()}\n",
        "\n",
        "    seller_avg_util = np.mean(np.sum([delta**(t-1) * np.array(summed_payment_dict[t]) for t in range(1, T + 1)], axis=0))\n",
        "    buyer1_avg_util = np.mean(np.sum([delta**(t-1) * np.array(buyer_1_val_dict[t]) for t in range(1, T + 1)], axis=0))\n",
        "    buyer2_avg_util = np.mean(np.sum([delta**(t-1) * np.array(buyer_2_val_dict[t]) for t in range(1, T + 1)], axis=0))\n",
        "\n",
        "    seller_err = np.std(np.sum([delta**(t-1) * np.array(summed_payment_dict[t]) for t in range(1, T + 1)], axis=0))/math.sqrt(N_it)\n",
        "    buyer1_err = np.std(np.sum([delta**(t-1) * np.array(buyer_1_val_dict[t]) for t in range(1, T + 1)], axis=0))/math.sqrt(N_it)\n",
        "    buyer2_err = np.std(np.sum([delta**(t-1) * np.array(buyer_2_val_dict[t]) for t in range(1, T + 1)], axis=0))/math.sqrt(N_it)\n",
        "\n",
        "    return (seller_avg_util, seller_err), (buyer1_avg_util, buyer1_err), (buyer2_avg_util, buyer2_err)\n",
        "\n",
        "\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "VJ80MumW1vP3"
      },
      "outputs": [],
      "source": [
        "delta = 0.99"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "b2cb-cRd1vP3",
        "outputId": "a8a87415-fcdc-4787-8cd7-f20282c714e5"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r  0%|          | 0/36 [00:00<?, ?it/s]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Unconstrained: Group 1 wins with probability  0.7839324060837488\n",
            "Unconstrained: Group 2 wins with probability  0.2160675939162505\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r  3%|▎         | 1/36 [03:50<2:14:20, 230.30s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.5416599613389402\n",
            "Constrained: Group 2 wins with probability  0.21209619381184477\n",
            "Unconstrained: Group 1 wins with probability  0.7822515298577627\n",
            "Unconstrained: Group 2 wins with probability  0.21774847014223675\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r  6%|▌         | 2/36 [07:36<2:09:10, 227.95s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.5408954905328113\n",
            "Constrained: Group 2 wins with probability  0.21286066461797384\n",
            "Unconstrained: Group 1 wins with probability  0.7840153192608156\n",
            "Unconstrained: Group 2 wins with probability  0.21598468073918392\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r  8%|▊         | 3/36 [11:39<2:09:01, 234.60s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.5396919525154682\n",
            "Constrained: Group 2 wins with probability  0.21406420263531678\n",
            "Unconstrained: Group 1 wins with probability  0.7835630655677249\n",
            "Unconstrained: Group 2 wins with probability  0.21643693443227438\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 11%|█         | 4/36 [15:31<2:04:34, 233.57s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.5445253259885607\n",
            "Constrained: Group 2 wins with probability  0.2092308291622243\n",
            "Unconstrained: Group 1 wins with probability  0.7809098439015942\n",
            "Unconstrained: Group 2 wins with probability  0.21909015609840515\n",
            "Constrained: Group 1 wins with probability  0.5414580173226109\n",
            "Constrained: Group 2 wins with probability  0.2122981378281741\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 14%|█▍        | 5/36 [19:26<2:01:00, 234.20s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Unconstrained: Group 1 wins with probability  0.783042973820671\n",
            "Unconstrained: Group 2 wins with probability  0.2169570261793284\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 17%|█▋        | 6/36 [23:55<2:02:58, 245.95s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.5784817324336947\n",
            "Constrained: Group 2 wins with probability  0.17527442271709023\n",
            "Unconstrained: Group 1 wins with probability  0.7826510206199927\n",
            "Unconstrained: Group 2 wins with probability  0.21734897938000683\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 19%|█▉        | 7/36 [27:35<1:54:53, 237.72s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.5068850184968575\n",
            "Constrained: Group 2 wins with probability  0.24687113665392754\n",
            "Unconstrained: Group 1 wins with probability  0.7828168469741257\n",
            "Unconstrained: Group 2 wins with probability  0.21718315302587365\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 22%|██▏       | 8/36 [31:17<1:48:31, 232.57s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.5090757636472851\n",
            "Constrained: Group 2 wins with probability  0.24468039150349988\n",
            "Unconstrained: Group 1 wins with probability  0.7832841757903193\n",
            "Unconstrained: Group 2 wins with probability  0.21671582420968016\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 25%|██▌       | 9/36 [35:14<1:45:18, 234.02s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.5076236289776743\n",
            "Constrained: Group 2 wins with probability  0.2461325261731107\n",
            "Unconstrained: Group 1 wins with probability  0.7813470224715818\n",
            "Unconstrained: Group 2 wins with probability  0.2186529775284177\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 28%|██▊       | 10/36 [39:00<1:40:20, 231.56s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.5068376908023778\n",
            "Constrained: Group 2 wins with probability  0.2469184643484072\n",
            "Unconstrained: Group 1 wins with probability  0.7827640840432653\n",
            "Unconstrained: Group 2 wins with probability  0.21723591595673422\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 31%|███       | 11/36 [42:46<1:35:40, 229.64s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.507560698802329\n",
            "Constrained: Group 2 wins with probability  0.24619545634845594\n",
            "Unconstrained: Group 1 wins with probability  0.7830580489437741\n",
            "Unconstrained: Group 2 wins with probability  0.2169419510562254\n",
            "Constrained: Group 1 wins with probability  0.5551832517468406\n",
            "Constrained: Group 2 wins with probability  0.1985729034039445\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 33%|███▎      | 12/36 [47:13<1:36:30, 241.25s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Unconstrained: Group 1 wins with probability  0.7814751610179573\n",
            "Unconstrained: Group 2 wins with probability  0.21852483898204206\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 36%|███▌      | 13/36 [51:08<1:31:41, 239.19s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.48183094148587463\n",
            "Constrained: Group 2 wins with probability  0.27192521366491046\n",
            "Unconstrained: Group 1 wins with probability  0.7826887084277502\n",
            "Unconstrained: Group 2 wins with probability  0.2173112915722493\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 39%|███▉      | 14/36 [55:04<1:27:19, 238.17s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.48388898180108125\n",
            "Constrained: Group 2 wins with probability  0.2698671733497038\n",
            "Unconstrained: Group 1 wins with probability  0.7828243845356773\n",
            "Unconstrained: Group 2 wins with probability  0.21717561546432215\n",
            "Constrained: Group 1 wins with probability  0.479391043901899\n",
            "Constrained: Group 2 wins with probability  0.274365111248886\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 42%|████▏     | 15/36 [59:17<1:24:55, 242.64s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Unconstrained: Group 1 wins with probability  0.7823872059656898\n",
            "Unconstrained: Group 2 wins with probability  0.21761279403430958\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 44%|████▍     | 16/36 [1:03:20<1:20:58, 242.90s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.4793863946265337\n",
            "Constrained: Group 2 wins with probability  0.2743697605242513\n",
            "Unconstrained: Group 1 wins with probability  0.7823344430348292\n",
            "Unconstrained: Group 2 wins with probability  0.21766555696517015\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 47%|████▋     | 17/36 [1:07:21<1:16:44, 242.35s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.48156622210085803\n",
            "Constrained: Group 2 wins with probability  0.272189933049927\n",
            "Unconstrained: Group 1 wins with probability  0.783163574805495\n",
            "Unconstrained: Group 2 wins with probability  0.21683642519450427\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 50%|█████     | 18/36 [1:12:07<1:16:38, 255.49s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.5258538868779529\n",
            "Constrained: Group 2 wins with probability  0.22790226827283214\n",
            "Unconstrained: Group 1 wins with probability  0.7818671142186358\n",
            "Unconstrained: Group 2 wins with probability  0.21813288578136364\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "<ipython-input-4-ee1a5c35cb3f>:67: RuntimeWarning: The iteration is not making good progress, as measured by the \n",
            "  improvement from the last ten iterations.\n",
            "  eta_1, gamma = fsolve(alpha_equations_case1, initial_guess, args=(alpha_1, alpha_2, a_1, b_1, a_2, b_2))\n",
            " 53%|█████▎    | 19/36 [1:16:14<1:11:36, 252.73s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.4542181489742534\n",
            "Constrained: Group 2 wins with probability  0.2995380061765317\n",
            "Unconstrained: Group 1 wins with probability  0.7818671142186358\n",
            "Unconstrained: Group 2 wins with probability  0.21813288578136364\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 56%|█████▌    | 20/36 [1:20:17<1:06:39, 249.96s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.45288434496100494\n",
            "Constrained: Group 2 wins with probability  0.3008718101897801\n",
            "Unconstrained: Group 1 wins with probability  0.7836158284985857\n",
            "Unconstrained: Group 2 wins with probability  0.21638417150141384\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 58%|█████▊    | 21/36 [1:24:47<1:03:57, 255.86s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.45287869324908436\n",
            "Constrained: Group 2 wins with probability  0.3008774619017006\n",
            "Unconstrained: Group 1 wins with probability  0.7820706283805264\n",
            "Unconstrained: Group 2 wins with probability  0.21792937161947296\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 61%|██████    | 22/36 [1:28:51<58:51, 252.28s/it]  "
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.4493606155112716\n",
            "Constrained: Group 2 wins with probability  0.3043955396395135\n",
            "Unconstrained: Group 1 wins with probability  0.7830957367515317\n",
            "Unconstrained: Group 2 wins with probability  0.21690426324846787\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 64%|██████▍   | 23/36 [1:32:52<53:57, 249.08s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.4541275540877965\n",
            "Constrained: Group 2 wins with probability  0.2996286010629886\n",
            "Unconstrained: Group 1 wins with probability  0.783103274313083\n",
            "Unconstrained: Group 2 wins with probability  0.21689672568691637\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 67%|██████▋   | 24/36 [1:37:54<52:58, 264.91s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.5011387197083338\n",
            "Constrained: Group 2 wins with probability  0.25261743544245124\n",
            "Unconstrained: Group 1 wins with probability  0.7817540507953632\n",
            "Unconstrained: Group 2 wins with probability  0.21824594920463627\n",
            "Constrained: Group 1 wins with probability  0.4441076576255346\n",
            "Constrained: Group 2 wins with probability  0.3096484975252505\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 69%|██████▉   | 25/36 [1:41:34<46:04, 251.29s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Unconstrained: Group 1 wins with probability  0.7839625563299549\n",
            "Unconstrained: Group 2 wins with probability  0.21603744367004446\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 72%|███████▏  | 26/36 [1:45:13<40:16, 241.69s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.44850978796817254\n",
            "Constrained: Group 2 wins with probability  0.3052463671826125\n",
            "Unconstrained: Group 1 wins with probability  0.7822515298577627\n",
            "Unconstrained: Group 2 wins with probability  0.21774847014223675\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 75%|███████▌  | 27/36 [1:49:02<35:39, 237.77s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.44866669593612196\n",
            "Constrained: Group 2 wins with probability  0.30508945921466313\n",
            "Unconstrained: Group 1 wins with probability  0.781068132694176\n",
            "Unconstrained: Group 2 wins with probability  0.21893186730582348\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 78%|███████▊  | 28/36 [1:53:01<31:45, 238.21s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.4457700933331875\n",
            "Constrained: Group 2 wins with probability  0.30798606181759747\n",
            "Unconstrained: Group 1 wins with probability  0.7815354615103695\n",
            "Unconstrained: Group 2 wins with probability  0.21846453848962996\n",
            "Constrained: Group 1 wins with probability  0.44871155433751736\n",
            "Constrained: Group 2 wins with probability  0.30504460081326773\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 81%|████████  | 29/36 [1:57:00<27:49, 238.47s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Unconstrained: Group 1 wins with probability  0.7825002693889624\n",
            "Unconstrained: Group 2 wins with probability  0.21749973061103697\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 83%|████████▎ | 30/36 [2:01:52<25:27, 254.62s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.49946931008763296\n",
            "Constrained: Group 2 wins with probability  0.2542868450631521\n",
            "Unconstrained: Group 1 wins with probability  0.7814073229639937\n",
            "Unconstrained: Group 2 wins with probability  0.2185926770360056\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 86%|████████▌ | 31/36 [2:05:26<20:12, 242.42s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.356920994549029\n",
            "Constrained: Group 2 wins with probability  0.3968351606017561\n",
            "Unconstrained: Group 1 wins with probability  0.7813168722253756\n",
            "Unconstrained: Group 2 wins with probability  0.21868312777462373\n",
            "Constrained: Group 1 wins with probability  0.35427547311833124\n",
            "Constrained: Group 2 wins with probability  0.39948068203245385\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 89%|████████▉ | 32/36 [2:08:59<15:34, 233.67s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Unconstrained: Group 1 wins with probability  0.7823344430348292\n",
            "Unconstrained: Group 2 wins with probability  0.21766555696517015\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 92%|█████████▏| 33/36 [2:12:37<11:26, 228.76s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.353990468985501\n",
            "Constrained: Group 2 wins with probability  0.39976568616528413\n",
            "Unconstrained: Group 1 wins with probability  0.7831484996823921\n",
            "Unconstrained: Group 2 wins with probability  0.21685150031760733\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 94%|█████████▍| 34/36 [2:16:09<07:27, 223.95s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.3489062224409252\n",
            "Constrained: Group 2 wins with probability  0.40484993270985986\n",
            "Unconstrained: Group 1 wins with probability  0.7821761542422476\n",
            "Unconstrained: Group 2 wins with probability  0.21782384575775182\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "\r 97%|█████████▋| 35/36 [2:19:45<03:41, 221.47s/it]"
          ]
        },
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Constrained: Group 1 wins with probability  0.3548369847825053\n",
            "Constrained: Group 2 wins with probability  0.3989191703682798\n",
            "Unconstrained: Group 1 wins with probability  0.7818897269032903\n",
            "Unconstrained: Group 2 wins with probability  0.21811027309670913\n",
            "Constrained: Group 1 wins with probability  0.4031155550491204\n",
            "Constrained: Group 2 wins with probability  0.35064060010166465\n"
          ]
        },
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "100%|██████████| 36/36 [2:24:27<00:00, 240.76s/it]\n"
          ]
        }
      ],
      "source": [
        "random.seed(2)\n",
        "\n",
        "#calculate utilities across grid\n",
        "T_in = 4\n",
        "start, end = 0, 0.5\n",
        "step = 0.1\n",
        "\n",
        "values = np.arange(start, end + step, step)\n",
        "Alpha1, Alpha2 = np.meshgrid(values, values)\n",
        "grid_pairs = np.vstack([Alpha1.ravel(), Alpha2.ravel()]).T\n",
        "\n",
        "\n",
        "utils = []\n",
        "errors = []\n",
        "\n",
        "for i in tqdm(range(len(grid_pairs))):\n",
        "    alpha_1_i = grid_pairs[i][0]\n",
        "    alpha_2_i = grid_pairs[i][1]\n",
        "\n",
        "    (seller, seller_error), (group1, group1_error), (group2, group2_error) = get_utilities(alpha_1_i, alpha_2_i, N_it = 10000, T=T_in)\n",
        "    utils.append((seller, group1, group2))\n",
        "    errors.append((seller_error, group1_error, group2_error))\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "oUliH5al1vP3"
      },
      "outputs": [],
      "source": [
        "#make tables and plots\n",
        "\n",
        "seller_utils = [u[0] for u in utils]\n",
        "group1_utils = [u[1] for u in utils]\n",
        "group2_utils = [u[2] for u in utils]\n",
        "\n",
        "seller_utils_err = [u[0] for u in errors]\n",
        "group1_utils_err = [u[1] for u in errors]\n",
        "group2_utils_err = [u[2] for u in errors]\n",
        "\n",
        "res = pd.DataFrame({'alpha1': [g[0] for g in grid_pairs], 'alpha2': [g[1] for g in grid_pairs],\n",
        "                    'seller_utils': seller_utils, 'group1_utils': group1_utils, 'group2_utils': group2_utils})\n",
        "\n",
        "table = pd.DataFrame({'alpha1': [g[0] for g in grid_pairs], 'alpha2': [g[1] for g in grid_pairs],\n",
        "                    'seller_utils': seller_utils, 'group1_utils': group1_utils, 'group2_utils': group2_utils,\n",
        "                    'seller_utils_err': seller_utils_err, 'group1_utils_err': group1_utils_err, 'group2_utils_err': group2_utils_err})\n",
        "\n",
        "unconstrained_seller_utility = res['seller_utils'][0]\n",
        "unconstrained_group1_utility= res['group1_utils'][0]\n",
        "unconstrained_group2_utility= res['group2_utils'][0]\n",
        "\n",
        "res['seller_utils'] = res['seller_utils'] - unconstrained_seller_utility\n",
        "res['group1_utils'] = res['group1_utils'] - unconstrained_group1_utility\n",
        "res['group2_utils'] = res['group2_utils'] - unconstrained_group2_utility\n",
        "table['seller_utils'] = table['seller_utils'] - unconstrained_seller_utility\n",
        "table['group1_utils'] = table['group1_utils'] - unconstrained_group1_utility\n",
        "table['group2_utils'] = table['group2_utils'] - unconstrained_group2_utility"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "4hWl3SLw1vP4",
        "outputId": "60876946-7bee-4636-ed76-77588a0d0743"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Mounted at drive\n",
            "cp: cannot stat 'experiment_results.csv': No such file or directory\n"
          ]
        }
      ],
      "source": [
        "from google.colab import drive\n",
        "drive.mount('drive')\n",
        "table.to_csv('experiment_results_T4.csv')\n",
        "!cp experiment_results.csv \"drive/My Drive/\""
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "Wjy6ar-Z1vP4"
      },
      "outputs": [],
      "source": [
        "table.to_csv('T4.csv')"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 565
        },
        "id": "vScJ_wZp1vP4",
        "outputId": "c09b6f51-d29d-4edc-9d91-495adb8f8753"
      },
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 800x600 with 2 Axes>"
            ],
            "image/png": "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E8tgAAAAAA9FZR0AAMAPUFn3TlTWAQAAAA9FZR0AAMAPmLjC1CtRWQcAAAA8FJV1AAAAP8Cade9EZR0AAADwUI1K1vPy8jRu3DjFxsYqJCREHTt21DPPPKPy8nJJ0ogRIxQSEqIffvjBKcECAACgYUwm1z3gOg1eBrNw4UJNnDhRFRUVuvbaa9W7d2+tW7dOTz/9tCwWiwYNGqSMjAxNnjxZsbGxzowZAAAADiKp9k4NStZXrFih++67T1FRUVq/fr2uvPJKSdK6det03XXX6R//+Ic2b96siIgIPf74404NGAAAAPAXDi+DKSsr06RJk2QYhv7617/aEnVJGjp0qMLCwpSdna01a9bokUceUcuWLSVJH3zwgYYPH642bdooLCxM3bt31//93//ZlswAAADAdQJMrnt4qtLSUqWlpSkhIUHNmzeXxWJRly5dNHXqVJcv0x49erRMJpPtceDAgQb143BlPSMjQz/++KO6deume+65p8rxZs2a6fDhw2rZsqWmTp1qe/3FF19U27ZtlZaWplatWmnbtm164okntGvXLi1evLhBwQMAAADV2bt3r4YNG6a8vDy71/fs2aM9e/bozTff1LvvvqubbrrJ6WN/9NFHWrZsmVP6cjhZX7t2rSTp9ttvr7XdjBkzFBERYfv/1atX26rsknTttdfKMAzNnDnTlsADAADANfxpzXpxcbFuvPFGW6J+3333KTk5WaGhodq0aZNSU1NVVFSk0aNHa+vWrbriiiucNvbp06c1YcIESdIll1yio0ePNqo/h5fBZGVlSZIGDBhQ5Vh5eblKS0vVsmVLPfjgg3bHLkzUraxLaI4cOeJoGAAAAEC1XnjhBeXm5kqS0tLS9MYbb2jQoEHq37+/ZsyYoU8++URBQUEqLS3VlClTnDr2E088oYMHD2rw4MG64YYbGt2fw8m6dX3P5ZdfXuXYSy+9pJMnT+qSSy6R2Wyus69PP/1UwcHB6tChg6NhAAAAwAGmANc9PEl5ebnmzp0rSeratavdsmyrq6++Wvfee68kacuWLbZidGNlZ2frlVdekdls1quvvuqUPh2e3oCAX04pLCy0e33//v167rnnJEmBgYF19vPNN98oPT1d48aNU2RkpKNhAAAAAFVs2rRJp06dkiTdfffdttz118aOHWt7npGR0ehxKyoqdN9996myslLTp09XXFxc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78dk1PR1Ax4ttM0d7g4BsNMkyPEbPaB6ltBW7g7BJwSYuCzLWSxh/Jl0ltyd4905ugv7jnNh344rLi5WfHy8cnN/ec/jxo1TcnKyQkNDtWnTJs2ePVunT59WaGiotm3bpiuuuKLafqyV89jYWB04cKDK8c2bN2vIkCE6f/68TCaT7rrrLo0aNUrR0dEqLi7Wli1b9PLLL6ugoECSdO+99+rNN9906L2QrHspknV4GpJ15yFZdw6SdechWXcekvWLZ+/evRo2bJjy8vKqPR4ZGal3331XN910U4191JWsS9J7772ncePG2W0DWZ3k5GQtXry4XjdrsouBZB2uUFZWptTUVKWkpLjtDma+grl0DubReZhL52EunYN5RE1KSko0f/58LV++XHv37tW5c+cUExOjYcOGafLkyYqNja31/Pok65J06NAhvfHGG1q/fr327NmjoqIimc1mRUdHq1+/frr77rt17bXXNug9kKzDJYqKihQVFaVTp04pMjLS3eF4NebSOZhH52EunYe5dA7mEb6MC0wBAAAAD0WyDgAAAHgoknUAAADAQ5GswyXMZrOeeuopLvRxAubSOZhH52EunYe5dA7mEb6MC0wBAAAAD0VlHQAAAPBQJOsAAACAhyJZBwAAADwUyTpsli9froEDB6pZs2ayWCzq1auX0tLSVF5e3qD+vvjiC40cOVKtWrVSSEiI2rVrp0mTJuno0aO1nvfzzz9r4sSJateuncxms1q1aqWRI0fqyy+/bFAcF5uz5rGgoECLFi3SpEmTdPXVVyssLEwmk0lDhgyp1/l79+7V2LFjFR0dbbuL2tixY7Vv376GvC23cNZc5uTkKDU1VYMHD1arVq3UpEkTNWvWTNdcc43mz59fZ3/M5X9t27ZN48ePV//+/XX55ZcrJCREFotF3bp106RJk2q9w5/k/XPp7H8nL7R27VqZTKZ6/T339nmUnDeXixYtss1bTY+PP/64xvO9/WcO/IABGIYxefJkQ5IRFBRkDB061BgxYoTRtGlTQ5KRlJRklJaWOtTf8uXLjaCgIEOSkZCQYIwaNcpo3769Iclo1aqVkZeXV+15e/bsMS655BJDktG+fXtj1KhRRkJCgi22lStXOuPtuowz5zEjI8OQVOUxePDgOs/NzMw0wsLCDElG9+7djdGjRxvdu3c3JBkWi8XYvn17Y97mReGs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\n"
          },
          "metadata": {}
        }
      ],
      "source": [
        "#plot heatmaps\n",
        "heatmap_data = res.pivot(index='alpha2', columns='alpha1', values='seller_utils').round(4)\n",
        "plot_heatmap(heatmap_data, 'seller_utility_T4.png')"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 554
        },
        "id": "WAf07Tbi1vP4",
        "outputId": "34629894-47bf-4c2f-9eea-b3ad8f48b13f"
      },
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 800x600 with 2 Axes>"
            ],
            "image/png": "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3X333ZKknTt3KiUlpVHX+sUvfqHHH39cv/3tb9WuXbvGB10LknUAAAD4je3btysvL0+SdOeddyogoPp0Nz4+3vk6KSmpOUJrFKbBAAAAWEALbwfQTJKTk52vhw0bVmO7yMhIBQUFqaioSLt27WqO0BqFyjoAAAD8xldffeV83adPnxrbBQYGKiIiQpL09ddfezyuxiJZBwAAsIAAm+GxzZccOXJEkuRwONS2bdta23bp0kWSdOLECZWUlHg6tEZhGgwAAACapKSkpEqya7fbZbfbmz2WgoICSVJwcHCdbR0Oh/P1mTNnvBJvXaisAwAAWIAnV4OZP3++wsLCXLb58+d75X2ePXtWktSqVas6216YnBcXF3sspqYgWQcAAECTJCQkKC8vz2VLSEio9Rybzdbkbfny5VX6bd26tSTp3LlzdcZ94V8D2rRp07A33UyYBgMAAGABnlwP3VtTXqoTEhIi6fy0lroUFhY6X9dn2ow3kKwDAABYQAsfe3iRO1Zg+dWvflVlX3h4uPbu3avCwkLl5ubWepNpdna2JKljx44+88vGz5GsAwAAoNnVtqxiU/Tt21fr1q2TJH3zzTcaMmRIte3Kysr07bffSjr/pFNfxZx1AAAAC/DkDaa+JCYmxvl6586dNbZLTU11ToOJjo72eFyNRbIOAAAAvzF8+HCFhYVJklasWCHDqH4d+AtvTh07dmxzhNYoJOsAAAAWYJWHIrVq1UoPPPCApPPz4p977rkqbfbs2aPXXntNkjRs2DBFRUVV21flqjPdu3f3WLx1Yc46AAAA/Mojjzyi1atXKyMjQ48++qiysrIUFxenNm3aaPv27Zo3b57KysrUpk0b/e1vf2vStd5//339+OOPzn9/8803zteff/65SwU/ODhYt912W4P6txk1/W0APu1Pe7d5OwS/UOHtAPzIuQofm7RoYp8fb+ntEPyCwRe42xxf+k3djVAvmR/+0WvX/vtXWzzW9/S+oz3Wd2NlZWVpzJgxyszMrPZ4aGio3n77bd1www019mGznf/Z1q1bNx06dKjaNsOHD691bvyFauunJkyDAQAAgN+JiIhQWlqann32WUVGRqpt27YKCgpS79699dBDD+mLL76oNVH3FUyDAQAAsIAW3g7ACxwOhx599FE9+uijjTq/PhNQduzY0ai+64vKOgAAAOCjqKwDAABYgK+t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3LlzdY7JhX8NaNOmTZ3t62vHjh3OpSHbt2+vdevWNbp/psEAAACgSbw15aU6ISEhks5Pa6lLYWGh83V9ps3UR2pqqm666SaVlJQoODhYmzdvbtSNpZVI1gEAACyghuf2eI07VmD51a9+VWVfeHi49u7dq8LCQuXm5tZ6k2l2drYkqWPHjm75ZePLL7/Utddeq4KCAtntdm3YsEFXXnllk/okWQcAAECzq21Zxabo27ev1q1bJ0n65ptvNGTIkGrblZWVOR9S1JTKd6Vvv/1Wv/3tb5WTk6PAwECtXr1aI0eObHK/zFkHAACwAKussx4TE+N8feEa5z+XmprqnAYTHR3dpGseOXJEo0aN0g8//KCAgACtWLGiUcs0VodkHQAAwAICPLj5kuHDhyssLEyStGLFChlG9UtWXnhz6tixYxt9vePHj2vUqFE6dOiQJOnll1/WxIkTG93fz/na+AIAAACN1qpVKz3wwAOSzs+Lf+6556q02bNnj1577TVJ0rBhwxQVFVVtX5WrznTv3r3a47m5ufrd736nAwcOSJJeeOEF3XPPPW54F//BnHUAAAAL8LUbTD3pkUce0erVq5WRkaFHH31UWVlZiouLU5s2bbR9+3bNmzdPZWVlatOmjfMJow1VUlKi66+/Xp9//rkk6Q9/+INGjRql9PT0Gs9xOBzq0aNHg65jM2r62wB82uVvfOTtEPxCWbm3I/Af5WXejsB/VOw/6e0Q/AN/O3abHzau9XYIfqP48EqvXftUyXse67u9/SaP9d1YWVlZGjNmjDIzM6s9Hhoaqrfffls33HBDjX3Y/v9vON26dXNOc6l06NChBifew4YN044dOxp0Dt/KAAAALMEqt5ieFxERobS0ND377LOKjIxU27ZtFRQUpN69e+uhhx7SF198UWui7iuorJsUlXX3oLLuPlTW3YfKuptQjnIbKuvu493K+kaP9d3efqPH+rY65qwDAABYgM1HK+CoHXUHAAAAwEdRWQcAALAAm40arRmRrAMAAFgC02DMiF+xAAAAAB9FZR0AAMACuMHUnKisAwAAAD7Kr5P1tWvXavjw4WrXrp0cDocGDBighQsXqrS0tFH9ffbZZxo/frw6deqk1q1bq0ePHpo+fbqOHz9e63k//fSTpk2bph49eshut6tTp04aP3689u3b16g4AAAAGs5aD0XyF36brD/4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\n"
          },
          "metadata": {}
        }
      ],
      "source": [
        "heatmap_data = res.pivot(index='alpha2', columns='alpha1', values='group1_utils').round(4)\n",
        "plot_heatmap(heatmap_data, 'group1_utility_T4.png')\n"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 554
        },
        "id": "Cy08u05h1vP4",
        "outputId": "b93cd2b6-a97a-4f1f-b8a9-6c2c56b197c3"
      },
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 800x600 with 2 Axes>"
            ],
            "image/png": "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sFJSUjR16lS1bdvWnTEDAADARd6e9FrFVDK9bt06PfTQQwoJCdEnn3yiXr16SZI2b96s22+/XX/729+0detWNWnSRM8995xbAwYAAADqCpeneRQVFWnKlCkyDEP/8z//40ikJWnYsGEKCAhQRkaGUlNT9dRTT6lFixaSpLVr12rEiBFq06aNAgIC1LVrV82bN88xJQQAAADW8bFZd3gzl5PplJQUff/99+rSpYsmTJhQ5nzTpk116dIltWjRwmnrx9dff11+fn6aO3euUlNTNXbsWD3//PP63e9+V7NPAAAAANQSl6d5bNq0SZJ03333VdouMTHRaevHjRs3OqrUkjRo0CAZhqEZM2Zo7ty5atmypauhAAAAoJqYM20NlyvT6enpkqQBAwaUOXflyhUVFBSoRYsWevTRR53OXZtI29mniJw8edLVMAAAAIBa53Jl2r7MXevWrcuc+8Mf/qBz586pa9eu8vPzq7Kvzz//XI0aNVKHDh1cDQMAAAAusLG7iCVcHlYfn6uX5ObmOr1/5MgRvfrqq5KkBg0aVNnPP//5TyUnJ2vixIkKDg52NQwAAACg1rmcTHfv3l2StHDhQhmGIUnKycnRr371K+Xn56thw4bKzs5WYWFhhX2cOXNGw4cPV0REhObMmWMydAAAAFQXm7ZYw+Vk2r5Cx1/+8hf17NlTcXFx6tSpkzIzMzV37lz17t1bZ8+e1cCBAzV//vwy11+4cEG/+MUvdPnyZX344YcKDAys+acAAAAAaoHLyfR9992nd999V9HR0crKytLGjRt144036t1339V///d/680331SPHj2Unp5e5sHCoqIi3XvvvTp69Kg++ugjtWrVym0fBAAAABWz2WyWHd7M1A6IY8aM0ZgxY8o916NHD/39738v835JSYnGjBmj9PR0ffrpp+rUqZOZWwMAAMAEL895LWMqmTbjscce04YNG/Tqq6+qpKREX3zxheNcly5deAgRAAAA9c51S6Y//PBDSdKMGTM0Y8YMp3OfffaZBg4ceL1CAQAA8DpUpq1x3ZLpo0ePXq9bAQAAANfFdUumAQAAUHuoTFuDvXAAAAAAk6hM11PBo++q7RA8w//feAhuUMpYuouNsXSPEsbRXUpaN6ntEOAGPlSmLUFlGgAAADCJyjQAAIAXoDJtDZJpAAAAL+BjY+qTFZjmAQAAAJhEZRoA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lSuSpJEjR8rf31/fffddLUcKAKiKb20HAADeZMmSJZo8ebKKi4s1aNAgRUVFafPmzXrppZcUGBiowYMHKyUlRVOnTlXbtm1rO1wAQBVshmEYtR0EAHiDdevWadSoUQoJCdEnn3yiXr16SZI2b96s22+/XbGxsQoJCdHnn3+uf/3rX2rRokUtRwwAqArJNABcB0VFRWrfvr2+//57vfPOO5owYYLT+cDAQJWWlurSpUt68cUX9dJLL0mSDh8+rNdff127d+/WP/7xD7Vu3VpHjx69/h8AAFAu5kwDwHWQkpKi77//Xl26dCmTSEtS06ZNdenSJbVo0ULTpk1zvP/VV1/p/fffV7t27dStW7frGTIAoBpIpgHgOti0aZMk6b777qu0XWJiopo0aeL48913363s7GytX79et912m6UxAgBcRzINANdBenq6JGnAgAFlzl25ckUFBQVq0aKFHn30UadzPj58mQaAuoyv0gBwHdiXuWvdunWZc3/4wx907tw53XjjjfLz87veoQEAaoBkGgCuA3uFOTc31+n9I0eO6NVXX5UkNWjQ4HqHBQCoIZJpALgOunfvLklauHCh7Iso5eTk6Fe/+pXy8/PVsGFDZWdnq7CwsDbDBAC4iGQaAK4D+wodf/nLX9SzZ0/FxcWpU6dOyszM1Ny5c9W7d2+dPXtWAwcO1Pz582s5WgBAdZFMA8B1cN999+ndd99VdHS0srKytHHjRt14441699139d///d9688031aNHD6Wnp+vkyZO1HS4AoJrYtAUA6olHHnlEH374IZu2AEAd4lvbAQAAKlZQUOBYo/rbb79VQUGB1q5dK0mKjo5W27ZtazM8APB6VKYBoA47evSo2rdvX+65pUuXKj4+/voGBABwQjINAAAAmMQDiAAAAIBJJNMAAACASSTTAAAAgEkk0wAAAIBJJNMAAACASSTTAAAAgEkk0wAAAIBJJNMAAACASSTTAAAAgEkk0wAAAIBJJNMAAACASSTTAAAAgEn/D249iGG7krgPAAAAAElFTkSuQmCC\n"
          },
          "metadata": {}
        }
      ],
      "source": [
        "heatmap_data = res.pivot(index='alpha2', columns='alpha1', values='group2_utils').round(4)\n",
        "plot_heatmap(heatmap_data, 'group2_utility_T4.png')"
      ]
    },
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "id": "QWtJbnJ61vP4"
      },
      "outputs": [],
      "source": []
    }
  ],
  "metadata": {
    "interpreter": {
      "hash": "284adbd07680ab978261f82881c8f376b9317887ca1a5755e8ed17ea6b98df15"
    },
    "kernelspec": {
      "display_name": "Python 3.9.2 ('.venv': venv)",
      "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.8"
    },
    "orig_nbformat": 4,
    "colab": {
      "provenance": []
    }
  },
  "nbformat": 4,
  "nbformat_minor": 0
}