{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# RCT for ML Tools Synthetic Experiments\n",
    "\n",
    "Code for synthetic simulations of randomized controlled trials for ML tools.\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 92,
   "metadata": {},
   "outputs": [],
   "source": [
    "import pandas as pd\n",
    "import numpy as np\n",
    "\n",
    "import matplotlib.pyplot as plt"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "Generate synthetic dataset."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 93,
   "metadata": {},
   "outputs": [],
   "source": [
    "def generate_data(n=5000):\n",
    "    \"\"\"\n",
    "    Generate synthetic data for the experiments.\n",
    "    Input:\n",
    "    n: the size of the synthetic dataset\n",
    "\n",
    "    Output:\n",
    "    a pandas dataframe with the synthetic data\n",
    "    \"\"\"\n",
    "\n",
    "    # generate a variable representing baseline patient covariates where\n",
    "    # each category is assigned unformly at random\n",
    "    X = np.random.choice([0,1,2,3], size=n, p=[1/4., 1/4., 1/4., 1/4.])\n",
    "\n",
    "    # generate a variable to represent randomization to different cluster\n",
    "    # 0 means control cluster, 1 means policy1, and 2 means policy2\n",
    "    # assignment to clusters is uniformly at random\n",
    "    D = np.random.choice([0,1,2], size=n, p=[1/3., 1/3., 1/3.])\n",
    "\n",
    "    # define the behavior of policy 1\n",
    "    def policy1(x):\n",
    "        if x == 1:\n",
    "            return 1\n",
    "        else:\n",
    "            return 0\n",
    "    \n",
    "    # define the behavior of policy 2\n",
    "    def policy2(x):\n",
    "        if x == 2 or x == 3:\n",
    "            return 1\n",
    "        else:\n",
    "            return 0\n",
    "        \n",
    "    # define the behavior of policy e\n",
    "    def policye(x):\n",
    "        if x == 1 or x == 2 or x == 3:\n",
    "            return 1\n",
    "        else:\n",
    "            return 0\n",
    "        \n",
    "    # define the behavior of policy e2\n",
    "    def policye2(x):\n",
    "        if x == 0 or x == 1:\n",
    "            return 1\n",
    "        else:\n",
    "            return 0\n",
    "    \n",
    "    # generate the policy outputs for each patient covariate\n",
    "    P0 = np.array([0 for x in X])\n",
    "    P1 = np.array([policy1(x) for x in X])\n",
    "    P2 = np.array([policy2(x) for x in X])\n",
    "    Pe = np.array([policye(x) for x in X])\n",
    "    Pe2 = np.array([policye2(x) for x in X])\n",
    "\n",
    "    # generate whether the deployed model alerts, which is a deterministic function of D and the policy\n",
    "    # that was deployed\n",
    "    A = []\n",
    "    for i in range(n):\n",
    "        a = int(D[i] == 0)*0 + int(D[i] == 1)*P1[i] + int(D[i] == 2)*P2[i]\n",
    "        A.append(a)\n",
    "\n",
    "    # generate whether each patient experiences sepsis, which is defined as a Bernoulli trial\n",
    "    # depending on the patient's covariates\n",
    "    O = []\n",
    "    for i in range(n):\n",
    "        if X[i] == 0:\n",
    "            p = 0.9\n",
    "        elif X[i] == 1:\n",
    "            p = 0.7\n",
    "        elif X[i] == 2:\n",
    "            p = 0.6\n",
    "        elif X[i] == 3:\n",
    "            p = 0.5\n",
    "        O.append(np.random.binomial(1, p, 1)[0])\n",
    "    O = np.array(O)\n",
    "\n",
    "    # define model performance for each of the policies\n",
    "    # model performance is deterministic based on the accuracy of the deployed model defined by D\n",
    "    M = []\n",
    "    for i in range(n):\n",
    "        m = int(D[i] == 0)*round(np.mean(P0 == O), 2) + int(D[i] == 1)*round(np.mean(P1 == O), 2) + int(D[i] == 2)*round(np.mean(P2 == O), 2)\n",
    "\n",
    "        M.append(m)\n",
    "\n",
    "    # define the data generating process of Y, which is also a Bernoulli trial where the probability of success\n",
    "    # is defined as a function of x, a, and m below\n",
    "    def pY(x, a, m):\n",
    "        if x == 0:\n",
    "            p = 0.3\n",
    "            if a == 1:\n",
    "                p += 0.1\n",
    "        elif x == 1:\n",
    "            p = 0.4\n",
    "            if a == 1:\n",
    "                p += 0.5\n",
    "        elif x == 2:\n",
    "            p = 0.3\n",
    "            if a == 1:\n",
    "                p += 0.5\n",
    "        elif x == 3:\n",
    "            p = 0.2\n",
    "            if a == 1:\n",
    "                p += 0.5\n",
    "\n",
    "        p += 0.1*m*a\n",
    "        return p\n",
    "\n",
    "    Y = []\n",
    "    for i in range(n):\n",
    "        Y.append(np.random.binomial(1, pY(X[i], A[i], M[i]), size=1)[0])\n",
    "\n",
    "    # compute the true effect of deploying P0 and print it\n",
    "    effect_p0 = 0\n",
    "    for x in [0,1,2,3]:\n",
    "        a = 0\n",
    "        effect_p0 += pY(x, a, round(np.mean(0 == O), 2))\n",
    "    print('true effect of p0', effect_p0/4)\n",
    "\n",
    "    # compute the true effect of deploying P1 and print it\n",
    "    effect_p1 = 0\n",
    "    for x in [0,1,2,3]:\n",
    "        a = policy1(x)\n",
    "        effect_p1 += pY(x, a, round(np.mean(P1 == O), 2))\n",
    "    print('true effect of p1', effect_p1/4)\n",
    "\n",
    "    # compute the true effect of deploying P1 and print it\n",
    "    effect_p2 = 0\n",
    "    for x in [0,1,2,3]:\n",
    "        a = policy2(x)\n",
    "        effect_p2 += pY(x, a, round(np.mean(P2 == O), 2))\n",
    "    print('true effect of p1', effect_p2/4)\n",
    "\n",
    "    # compute the true effect of deploying Pe and print it\n",
    "    effect_pe = 0\n",
    "    for x in [0,1,2,3]:\n",
    "        a = policye(x)\n",
    "        effect_pe += pY(x, a, round(np.mean(Pe == O), 2))\n",
    "    print('true effect of pe', effect_pe/4)\n",
    "\n",
    "    # compute the true effect of deploying Pe2 and print it\n",
    "    effect_pe2 = 0\n",
    "    for x in [0,1,2,3]:\n",
    "        a = policye2(x)\n",
    "        effect_pe2 += pY(x, a, round(np.mean(Pe2 == O), 2))\n",
    "    print('true effect of pe2', effect_pe2/4)\n",
    "\n",
    "    # combine the data to make the final dataframe\n",
    "    data = pd.DataFrame({'X': X, 'D': D, 'P0': P0, 'P1': P1, 'P2': P2, 'Pe': Pe, 'Pe2': Pe2, 'A': A, 'Y': Y, 'M': M, 'O': O})\n",
    "\n",
    "    return (data, effect_p0/4, effect_p1/4, effect_p2/4, effect_pe/4, effect_pe2/4)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 94,
   "metadata": {},
   "outputs": [],
   "source": [
    "def estimate_counterfactual(data, Pe, M_Pe, A, X, D, M, Y, Ymin, Ymax, a_0):\n",
    "    \"\"\"\n",
    "    Input:\n",
    "    data - a pandas dataframe containing the data\n",
    "    Pe - string containing the name of the policy we want to evaluate\n",
    "    M_Pe - the performance of the model Pe\n",
    "    A - string containing the name of the variable for A\n",
    "    X - string containing the name of the variable for X\n",
    "    D - string containing the name of the variable for D\n",
    "    M - string containing the name of the variable for M\n",
    "    Y - string containing the name of the variable for Y\n",
    "    Ymin - the minimum value for Y\n",
    "    Ymax - the maximum value for Y\n",
    "    a_0 - the value of the neutral action\n",
    "\n",
    "    Output:\n",
    "    a tuple containing the lower and upper bounds for the performance of the model Pe,\n",
    "    if the performance is point identified, then the lower and upper bounds will have\n",
    "    the same value\n",
    "    \"\"\"\n",
    "    # define sets with values of X where at least one trialed model matches in behavior and \n",
    "    # has different same, worse, or better performance\n",
    "    x_equal = set()\n",
    "    x_less = set()\n",
    "    x_great = set()\n",
    "    x_star = set()\n",
    "\n",
    "    # get the unique values of X in the data\n",
    "    X_values = np.unique(data[X])\n",
    "\n",
    "    # get the unique values of D in the data\n",
    "    D_values = np.unique(data[D])\n",
    "\n",
    "    for x in X_values:\n",
    "        # first subset to values in data where X=x\n",
    "        data_subset_x = data[data[X] == x]\n",
    "\n",
    "        for d in D_values:\n",
    "            # next, subset to values where X=x and D=d\n",
    "            data_subset_xd = data_subset_x[data_subset_x[D] == d]\n",
    "\n",
    "            # see if pi_d has the same model behavior as pi_e and has the same performance under X=x\n",
    "            if np.mean(data_subset_xd[A] == data_subset_xd[Pe]) == 1 and round(np.mean(data_subset_xd[M]), 2) == M_Pe:\n",
    "                x_equal.add(x)\n",
    "            \n",
    "            # see if pi_d has the same model behavior as pi_e and has worse performance under X=x\n",
    "            if np.mean(data_subset_xd[A] == data_subset_xd[Pe]) == 1 and round(np.mean(data_subset_xd[M]), 2) < M_Pe:\n",
    "                x_less.add(x)\n",
    "\n",
    "            # see if pi_d has the same model behavior as pi_e and has better performance under X=x\n",
    "            if np.mean(data_subset_xd[A] == data_subset_xd[Pe]) == 1 and round(np.mean(data_subset_xd[M]), 2) > M_Pe:\n",
    "                x_great.add(x)\n",
    "\n",
    "            # see if pi_d has the same model behavior as pi_e\n",
    "            if np.mean(data_subset_xd[A] == data_subset_xd[Pe]) == 1:\n",
    "                x_star.add(x)\n",
    "\n",
    "    # subtract out x_equal from x_less and x_great so that we are not double counting\n",
    "    x_less = x_less - x_equal\n",
    "    x_great = x_great - x_equal\n",
    "\n",
    "    # get the values of x that do not fit into either of the above two sets\n",
    "    x_lessequal = set(X_values) - x_equal - x_less\n",
    "    x_greatequal = set(X_values) - x_equal - x_great\n",
    "\n",
    "    # define functions of X that return a set in domain of D for each value of x\n",
    "    # define them as dictionaries since the domain of these functions is just the domain of X (0,1,2,3);\n",
    "    # the key of the dictionary is the input to the function, and the value is the output of the function (a set in the domain of D)\n",
    "    S_equal = dict()\n",
    "    S_less = dict()\n",
    "    S_great = dict()\n",
    "    S_star = dict()\n",
    "\n",
    "    for x in X_values:\n",
    "        # compute the output to the functions above for values of X=x\n",
    "        data_subset_x = data[data[X] == x]\n",
    "\n",
    "        cur_s_equal = set()\n",
    "        cur_s_less = set()\n",
    "        cur_s_great = set()\n",
    "        cur_s_star = set()\n",
    "\n",
    "        for d in D_values:\n",
    "            data_subset_xd = data_subset_x[data_subset_x[D] == d]\n",
    "\n",
    "            # see if pi_d has the same model behavior as pi_e and has the same performance under X=x\n",
    "            if np.mean(data_subset_xd[A] == data_subset_xd[Pe]) == 1 and round(np.mean(data_subset_xd[M]), 2) == M_Pe:\n",
    "                cur_s_equal.add(d)\n",
    "            \n",
    "            # see if pi_d has the same model behavior as pi_e and has worse performance under X=x\n",
    "            if np.mean(data_subset_xd[A] == data_subset_xd[Pe]) == 1 and round(np.mean(data_subset_xd[M]), 2) < M_Pe:\n",
    "                cur_s_less.add(d)\n",
    "\n",
    "            # see if pi_d has the same model behavior as pi_e and has better performance under X=x\n",
    "            if np.mean(data_subset_xd[A] == data_subset_xd[Pe]) == 1 and round(np.mean(data_subset_xd[M]), 2) > M_Pe:\n",
    "                cur_s_great.add(d)\n",
    "\n",
    "            # see if pi_d has the same model behavior as pi_e\n",
    "            if np.mean(data_subset_xd[A] == data_subset_xd[Pe]) == 1:\n",
    "                cur_s_star.add(d)\n",
    "\n",
    "        S_equal[x] = cur_s_equal\n",
    "        S_less[x] = cur_s_less\n",
    "        S_great[x] = cur_s_great\n",
    "        S_star[x] = cur_s_star\n",
    "\n",
    "    # define a function that computes the probability that D is in a set d_set\n",
    "    def p_D(d_set):\n",
    "        return len(data[data[D].isin(d_set)]) / len(data)\n",
    "    \n",
    "    # define a special divide operation that prevents us from dividing by 0\n",
    "    # we divide by 0 sometimes in the estimating functional but that doesn't hurt us because the indicator for that divide by 0\n",
    "    # will be \"turned off\" anyways\n",
    "    def special_divide(numerator, denominator):\n",
    "        if denominator == 0:\n",
    "            return 0\n",
    "        else:\n",
    "            return numerator/denominator\n",
    "    \n",
    "    # choose a critical value of 1.96 corresponding to 95% confidence intervals\n",
    "    crit = 1.96\n",
    "\n",
    "    # calculate the value of the target estimand for each row of data\n",
    "    L_values = []\n",
    "    for index, row in data.iterrows():\n",
    "        L_values.append( row[Y] * ( int(row[X] in x_equal) * int(row[Pe] != a_0) * (special_divide(int(row[D] in S_equal[row[X]]), p_D(S_equal[row[X]]))) \\\n",
    "                            + int(row[X] in x_less) * int(row[Pe] != a_0) * (special_divide(int(row[D] in S_less[row[X]]), p_D(S_less[row[X]])))  ) \\\n",
    "                            + Ymin * int(row[X] in x_lessequal) * int(row[Pe] != 0) \\\n",
    "                            + row[Y] * int(row[X] in x_star) * int(row[Pe] == a_0) * (special_divide(int(row[D] in S_star[row[X]]), p_D(S_star[row[X]]))) \\\n",
    "                            + Ymin * int(row[X] not in x_star) * int(row[Pe] == a_0) )\n",
    "    lower_mean = np.mean(L_values)\n",
    "    # compute confidence intervals for the lower bound value\n",
    "    lower_se = np.std(L_values)/np.sqrt(len(data))\n",
    "    lower_lowbound = lower_mean - (crit * lower_se)\n",
    "    lower_upbound = lower_mean + (crit * lower_se)\n",
    "\n",
    "    U_values = []\n",
    "    for index, row in data.iterrows():\n",
    "        U_values.append( row[Y] * ( int(row[X] in x_equal) * int(row[Pe] != a_0) * (special_divide(int(row[D] in S_equal[row[X]]), p_D(S_equal[row[X]]))) \\\n",
    "                            + int(row[X] in x_great) * int(row[Pe] != a_0) * (special_divide(int(row[D] in S_great[row[X]]), p_D(S_great[row[X]])))  ) \\\n",
    "                            + Ymax * int(row[X] in x_greatequal) * int(row[Pe] != a_0) \\\n",
    "                            + row[Y] * int(row[X] in x_star) * int(row[Pe] == a_0) * (special_divide(int(row[D] in S_star[row[X]]), p_D(S_star[row[X]]))) \\\n",
    "                            + Ymax * int(row[X] not in x_star) * int(row[Pe] == a_0) \n",
    "                            )\n",
    "    upper_mean = np.mean(U_values)\n",
    "    # compute confidence intervals for the upper bound value\n",
    "    upper_se = np.std(U_values)/np.sqrt(len(data))\n",
    "    upper_lowbound = upper_mean - (crit * upper_se)\n",
    "    upper_upbound = upper_mean + (crit * upper_se)\n",
    "\n",
    "    return (lower_mean, upper_mean, lower_lowbound, upper_upbound)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 95,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "true effect of p0 0.3\n",
      "true effect of p1 0.43575\n",
      "true effect of p1 0.5680000000000001\n",
      "true effect of pe 0.71025\n",
      "true effect of pe2 0.482\n",
      "model accuracy\n",
      "P0 0.3244\n",
      "P1 0.4346\n",
      "P2 0.3646\n",
      "Pe 0.4748\n",
      "Pe2 0.6354\n",
      "model effect on outcome\n",
      "P0 0.30382435492142823 0.30382435492142823\n",
      "P1 0.4258170801275342 0.4258170801275342\n",
      "P2 0.5557459242295476 0.5557459242295476\n",
      "Pe 0.6777386494356535 0.8274\n",
      "Pe2 0.3492170801275342 0.6297651135005974\n"
     ]
    }
   ],
   "source": [
    "np.random.seed(0)\n",
    "\n",
    "(data, effect_p0, effect_p1, effect_p2, effect_pe, effect_pe2) = generate_data(5000)\n",
    "\n",
    "print('model accuracy')\n",
    "print('P0', np.mean(data['P0'] == data['O']))\n",
    "print('P1', np.mean(data['P1'] == data['O']))\n",
    "print('P2', np.mean(data['P2'] == data['O']))\n",
    "print('Pe', np.mean(data['Pe'] == data['O']))\n",
    "print('Pe2', np.mean(data['Pe2'] == data['O']))\n",
    "\n",
    "print('model effect on outcome')\n",
    "P0_effect = estimate_counterfactual(data, 'P0', round(np.mean(data['P0'] == data['O']), 2), 'A', 'X', 'D', 'M', 'Y', 0, 1, 0)\n",
    "P1_effect = estimate_counterfactual(data, 'P1', round(np.mean(data['P1'] == data['O']), 2), 'A', 'X', 'D', 'M', 'Y', 0, 1, 0)\n",
    "P2_effect = estimate_counterfactual(data, 'P2', round(np.mean(data['P2'] == data['O']), 2), 'A', 'X', 'D', 'M', 'Y', 0, 1, 0)\n",
    "Pe_effect = estimate_counterfactual(data, 'Pe', round(np.mean(data['Pe'] == data['O']), 2), 'A', 'X', 'D', 'M', 'Y', 0, 1, 0)\n",
    "Pe2_effect = estimate_counterfactual(data, 'Pe2', round(np.mean(data['Pe2'] == data['O']), 2), 'A', 'X', 'D', 'M', 'Y', 0, 1, 0)\n",
    "\n",
    "print('P0', P0_effect[0], P0_effect[1])\n",
    "print('P1', P1_effect[0], P1_effect[1])\n",
    "print('P2', P2_effect[0], P2_effect[1])\n",
    "print('Pe', Pe_effect[0], Pe_effect[1])\n",
    "print('Pe2', Pe2_effect[0], Pe2_effect[1])\n",
    "\n",
    "objects = [r'$\\pi_0$', r'$\\pi_1$', r'$\\pi_2$', r'$\\pi_{e0}$', r'$\\pi_{e1}$']\n",
    "performances = []\n",
    "performances.append(np.mean(data['P0'] == data['O']))\n",
    "performances.append(np.mean(data['P1'] == data['O']))\n",
    "performances.append(np.mean(data['P2'] == data['O']))\n",
    "performances.append(np.mean(data['Pe'] == data['O']))\n",
    "performances.append(np.mean(data['Pe2'] == data['O']))\n",
    "\n",
    "expected_outcomes = []\n",
    "expected_outcomes.append(effect_p0)\n",
    "expected_outcomes.append(effect_p1)\n",
    "expected_outcomes.append(effect_p2)\n",
    "expected_outcomes.append(effect_pe)\n",
    "expected_outcomes.append(effect_pe2)\n",
    "\n",
    "midpoints = [effect_p0, effect_p1, effect_p2, effect_pe, effect_pe2]\n",
    "intervals = []\n",
    "intervals.append((P0_effect[2], P0_effect[3]))\n",
    "intervals.append((P1_effect[2], P1_effect[3]))\n",
    "intervals.append((P2_effect[2], P2_effect[3]))\n",
    "intervals.append((Pe_effect[2], Pe_effect[3]))\n",
    "intervals.append((Pe2_effect[2], Pe2_effect[3]))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 101,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[np.float64(0.3), np.float64(0.43575), np.float64(0.5680000000000001), np.float64(0.71025), np.float64(0.482)]\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "[(np.float64(0.28802551041206936), np.float64(0.3196231994307871)),\n",
       " (np.float64(0.40172945623131845), np.float64(0.44990470402375)),\n",
       " (np.float64(0.5272458302957539), np.float64(0.5842460181633413)),\n",
       " (np.float64(0.6446381879466753), np.float64(0.837874884022117)),\n",
       " (np.float64(0.32540587183795566), np.float64(0.6447727442189711))]"
      ]
     },
     "execution_count": 101,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "print(midpoints)\n",
    "intervals"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 123,
   "metadata": {},
   "outputs": [],
   "source": [
    "# bar plots\n",
    "def create_bar_plots(objects, performances, expected_outcomes, intervals, midpoints):\n",
    "    SMALL_SIZE = 10\n",
    "    MEDIUM_SIZE = 14\n",
    "    BIGGER_SIZE = 16\n",
    "\n",
    "    plt.rc('font', size=SMALL_SIZE)  # controls default text sizes\n",
    "    plt.rc('axes', titlesize=MEDIUM_SIZE)  # fontsize of the axes title\n",
    "    plt.rc('axes', labelsize=MEDIUM_SIZE)  # fontsize of the x and y labels\n",
    "    plt.rc('xtick', labelsize=SMALL_SIZE)  # fontsize of the tick labels\n",
    "    plt.rc('ytick', labelsize=SMALL_SIZE)  # fontsize of the tick labels\n",
    "    plt.rc('legend', fontsize=SMALL_SIZE)  # legend fontsize\n",
    "    plt.rc('figure', titlesize=BIGGER_SIZE)  # fontsize of the figure \n",
    "\n",
    "    bar_width = 0.4\n",
    "    # positions for first cluster\n",
    "    x = [1, 1.4, 1.8, 2.3, 2.7]\n",
    "    # Positions for second cluster\n",
    "    x2 = [3.5, 3.9, 4.3, 4.8, 5.2]\n",
    "\n",
    "    # Create the plot\n",
    "    fig, ax = plt.subplots(figsize=(8, 5))\n",
    "\n",
    "    colors = ['#B8BBB7', '#6D6960', '#7F7F95', '#E09D00', '#CC77CE']\n",
    "\n",
    "    # Plot first cluster\n",
    "    for i in range(len(objects)):\n",
    "        ax.bar(x[i], performances[i], color=colors[i], width=bar_width)\n",
    "    # Plot second cluster\n",
    "    for i in range(len(objects)):\n",
    "        ax.bar(x2[i], expected_outcomes[i], color=colors[i], width=bar_width)\n",
    "\n",
    "    # Set x-axis labels\n",
    "    ax.set_xticks(np.concatenate([x, x2]))\n",
    "    ax.set_xticklabels(objects * 2, fontsize=14)\n",
    "\n",
    "    # ax.set_ylim(0, 1)\n",
    "    # set y ticks\n",
    "    ax.set_yticks([i*0.1 for i in range(0, 11)])\n",
    "    \n",
    "    # Add labels for the two clusters\n",
    "    mid_x1 = np.mean(x)  # Midpoint for first cluster\n",
    "    mid_x2 = np.mean(x2)  # Midpoint for second cluster\n",
    "    ax.text(mid_x1, -0.12, \"Accuracy\", ha='center', fontsize=14, fontweight='bold')\n",
    "    ax.text(mid_x2, -0.12, \"Expected Survival Rate\", ha='center', fontsize=14, fontweight='bold')\n",
    "\n",
    "    # Labels and title\n",
    "    # ax.set_xlabel(\"Accuracy                                               Expected Survival Rate\")\n",
    "    ax.set_ylabel(\"Accuracy/Expected Survival Rate\", fontsize=14)\n",
    "    # ax.set_title(\"Accuracies and Expected Survival Rates Under Deployed\\nand New Models\")\n",
    "    # ax.legend()\n",
    "\n",
    "    error_bar_pos = [3.5, 3.9, 4.3, 4.8, 5.2]\n",
    "    errors = []\n",
    "    i = 0\n",
    "    for interval in intervals:\n",
    "        mid = midpoints[i]\n",
    "        i += 1\n",
    "\n",
    "        errors.append([[abs(mid - interval[0])],\n",
    "                       [abs(mid - interval[1])]])\n",
    "    print(errors)\n",
    "    \n",
    "    for i in range(len(error_bar_pos)):\n",
    "        ax.errorbar(error_bar_pos[i], midpoints[i], yerr=errors[i], fmt=' ', color='b', ecolor='b', capsize=5, capthick=2, elinewidth=2)\n",
    "\n",
    "    plt.savefig('sims.pdf')\n",
    "\n",
    "    # Show the plot\n",
    "    plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 124,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[[[np.float64(0.011974489587930626)], [np.float64(0.01962319943078711)]], [[np.float64(0.03402054376868158)], [np.float64(0.014154704023749964)]], [[np.float64(0.04075416970424617)], [np.float64(0.016246018163341258)]], [[np.float64(0.06561181205332478)], [np.float64(0.1276248840221169)]], [[np.float64(0.15659412816204432)], [np.float64(0.16277274421897114)]]]\n"
     ]
    },
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 800x500 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "create_bar_plots(objects, performances, expected_outcomes, intervals, midpoints)"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "rct_ml_tools",
   "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.13.0"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
