{
  "nbformat": 4,
  "nbformat_minor": 0,
  "metadata": {
    "colab": {
      "provenance": [],
      "machine_shape": "hm"
    },
    "kernelspec": {
      "name": "python3",
      "display_name": "Python 3"
    },
    "language_info": {
      "name": "python"
    }
  },
  "cells": [
    {
      "cell_type": "code",
      "execution_count": null,
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "NMUnAwwqUBFO",
        "outputId": "d95339e1-7d81-4e6c-81dc-6320e3000dbc"
      },
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Mounted at /content/drive\n"
          ]
        }
      ],
      "source": [
        "from google.colab import drive\n",
        "drive.mount('/content/drive', force_remount=True)"
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "cd \"/content/drive/MyDrive/My projects/Input_adaptive_BMA/Code\""
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "BVA4eJvDUEuJ",
        "outputId": "60266b43-50fc-45a0-f13d-8ddb8465f8d6"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "/content/drive/MyDrive/My projects/Input_adaptive_BMA/Code\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "SAVE = False"
      ],
      "metadata": {
        "id": "bn8eKAXuq40o"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "import numpy as np\n",
        "import tensorflow as tf\n",
        "from scipy import sparse\n",
        "import pandas as pd\n",
        "from sklearn.metrics import accuracy_score, brier_score_loss, r2_score, mean_squared_error, mean_absolute_error\n",
        "from sklearn.preprocessing import StandardScaler\n",
        "import matplotlib.pyplot as plt\n",
        "from matplotlib.patches import Patch\n",
        "import matplotlib.ticker as ticker\n",
        "from packaging import version\n",
        "import time"
      ],
      "metadata": {
        "id": "PxS_YeiXfLGh"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "from previous_methods import baseline_methods, moe_gating, moe_prediction, dynamic_local_accuracy, smc_ensemble, bhs_train, bhs_predict\n",
        "from IABMA import train_adaptive_BMA\n",
        "from utils import log_run, plot_confidence_error_bars, compute_ece, zscore_numeric_df"
      ],
      "metadata": {
        "id": "lfE0geD7eNDQ"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "EXPERIMENT = \"synthetic_binary\"\n",
        "# EXPERIMENT = \"cancer\"\n",
        "# EXPERIMENT = \"fraud\"\n",
        "\n",
        "# EXPERIMENT = \"UCI\"\n",
        "# # Supported datasets:\n",
        "# # Classification: \"spambase\", \"bank-marketing\", \"credit-g\"\n",
        "# # Regression: \"california-housing\", \"bike-sharing\", \"communities\"\n",
        "# UCI_DATASET = \"bank-marketing\"\n",
        "\n",
        "\n",
        "if EXPERIMENT != \"UCI\":\n",
        "  UCI_DATASET = None\n",
        "  TASK = None\n",
        "else:\n",
        "  try:\n",
        "    UCI_DATASET\n",
        "    if UCI_DATASET in [\"spambase\", \"bank-marketing\", \"credit-g\"]:\n",
        "      TASK = \"classification\"\n",
        "    elif UCI_DATASET in [\"bike-sharing\", \"california-housing\", \"communities\"]:\n",
        "      TASK = \"regression\"\n",
        "  except NameError:\n",
        "      print(\"UCI dataset is not defined\")"
      ],
      "metadata": {
        "id": "2d4C35G9WSf3"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "if EXPERIMENT == \"synthetic_binary\":\n",
        "  from synthetic import generate_circle_line, plot_circle_line, baseline_models, train_models, accuracy_by_region\n",
        "  from previous_methods import compute_cv_pred_probas\n",
        "elif EXPERIMENT == \"cancer\":\n",
        "  from cancer import load_cancer_data, train_models, predict_table, plot_winner_enrichment, metrics_by_region, mlp_predict, attach_annotations, extract_mlp_params_from_pipeline, build_gate_features, make_case_table, plot_cases_from_table\n",
        "  from previous_methods import compute_oos_pred_values, moe_gating_predict\n",
        "elif EXPERIMENT == \"fraud\":\n",
        "  from fraud import load_fraud_data, train_models, evaluate_models, format_results, plot_confidence_error_bars\n",
        "  from previous_methods import compute_cv_pred_probas, moe_gating_predict\n",
        "  from utils import downsample_majority\n",
        "elif EXPERIMENT == \"UCI\":\n",
        "  from UCI import load_uci_data, train_models, evaluate_models, build_gate_features\n",
        "  from previous_methods import compute_cv_pred_probas, compute_oos_pred_values, moe_gating_predict"
      ],
      "metadata": {
        "id": "f5ksmOsAY_d1"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "source": [
        "### Load Data"
      ],
      "metadata": {
        "id": "XaLhS8ThUUVg"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "if EXPERIMENT == \"synthetic_binary\":\n",
        "  x_train, y_train, y_oh_train, types_train, x_test, y_test, y_oh_test, types_test = generate_circle_line(t=1.0)\n",
        "  plot_circle_line(x_test, y_test, types_test)\n",
        "\n",
        "elif EXPERIMENT == \"cancer\":\n",
        "    # DepMap data release of CRISPR\n",
        "    import os\n",
        "    COL = \"primary_site\"\n",
        "    DATA_PATH = \"/content/drive/MyDrive/My projects/Input_adaptive_BMA/Code/Cancer_data\"\n",
        "\n",
        "    x_train, y_train, x_test, y_test, meta_df, df = load_cancer_data(\n",
        "      PRISM_PATH=os.path.join(DATA_PATH, \"Repurposing_Public_23Q2_Extended_Primary_Data_Matrix.csv\"),\n",
        "      EXPR_PATH=os.path.join(DATA_PATH, \"OmicsExpressionProteinCodingGenesTPMLogp1.csv\"),\n",
        "      TOP_K_DRUGS=40,\n",
        "      CRITERION=\"site_heterogeneity\",\n",
        "      MIN_PER_DRUG=40,\n",
        "      PER_DRUG_CAP=None,\n",
        "      MAX_GENES=100,\n",
        "      TEST_SIZE=0.2,\n",
        "      RSEED=42,\n",
        "      CELL_META_PATH=os.path.join(DATA_PATH, \"Cell_lines_annotations_20181226.txt\"),\n",
        "      MODEL_PATH=os.path.join(DATA_PATH, \"Model.csv\"),\n",
        "      SITE_COL=\"primary_site\",\n",
        "      COVERAGE_SITE_MIN=20,\n",
        "      PER_SITE_FRACTION_CAP=1.1,\n",
        "      USE_PAIR_STRAT=False,\n",
        "      USE_SITE_UNION_GENES=False\n",
        "  )\n",
        "\n",
        "elif EXPERIMENT == \"fraud\":\n",
        "  # IEEE-CIS Fraud Detection\n",
        "  import os\n",
        "  DATA_PATH = \"/content/drive/MyDrive/My projects/Input_adaptive_BMA/Code/Fraud_data\"\n",
        "  x_train, y_train, x_test, y_test = load_fraud_data(\n",
        "    os.path.join(DATA_PATH, \"train_transaction.csv\"),\n",
        "    os.path.join(DATA_PATH, \"train_identity.csv\"),\n",
        "    test_size=0.5,\n",
        "    sample_size=50**3\n",
        ")\n",
        "\n",
        "  TRAIN_COLS = list(x_train.columns)\n",
        "  NUM_COLS   = x_train.select_dtypes(include=[np.number]).columns.tolist()\n",
        "  CAT_COLS   = [c for c in TRAIN_COLS if c not in NUM_COLS]\n",
        "\n",
        "  def make_schema_locker(df_train: pd.DataFrame):\n",
        "    cols = list(df_train.columns)\n",
        "    num  = df_train.select_dtypes(include=[np.number, \"bool\"]).columns.tolist()\n",
        "    cat  = [c for c in cols if c not in num]\n",
        "\n",
        "    def lock(X: pd.DataFrame) -> pd.DataFrame:\n",
        "        X2 = X.reindex(columns=cols, fill_value=np.nan).copy()\n",
        "        # keep numerics numeric (bad parses -> NaN; your imputers handle it)\n",
        "        X2[num] = X2[num].apply(pd.to_numeric, errors=\"coerce\")\n",
        "        # keep categoricals as object (DON'T cast to str; preserve NaN)\n",
        "        for c in cat:\n",
        "            X2[c] = X2[c].astype(\"object\")\n",
        "        return X2\n",
        "\n",
        "    return lock\n",
        "\n",
        "  lock     = make_schema_locker(x_train)\n",
        "  x_train  = lock(x_train)\n",
        "  x_test   = lock(x_test)\n",
        "\n",
        "  x_test_b, y_test_b = downsample_majority(x_test, y_test)\n",
        "  x_test_b = lock(x_test_b)\n",
        "\n",
        "elif EXPERIMENT == \"UCI\":\n",
        "  x_train, y_train, x_test, y_test = load_uci_data(UCI_DATASET, test_size=0.2)\n",
        "  if x_train.shape[0] > 4000:\n",
        "    rnd_idx = np.random.choice(len(x_train), 4000, replace=False)\n",
        "    x_train, y_train = x_train.iloc[rnd_idx], y_train.iloc[rnd_idx]\n",
        "    rnd_idx = np.random.choice(len(x_test), 1000, replace=False)\n",
        "    x_test, y_test = x_test.iloc[rnd_idx], y_test.iloc[rnd_idx]\n",
        "  if UCI_DATASET in [\"bike-sharing\", \"california-housing\"]:\n",
        "    y_train = np.log1p(y_train)\n",
        "    y_test  = np.log1p(y_test)\n",
        "\n"
      ],
      "metadata": {
        "id": "k46BWE4BURT9"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "N_train, D = x_train.shape\n",
        "N_test, _ = x_test.shape"
      ],
      "metadata": {
        "id": "EDQyVArClV9-"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "N_train, N_test"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "YNZ_q8-IRk-h",
        "outputId": "b158afcf-7b43-4a73-847e-33e5e0d077ea"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "(62500, 62500)"
            ]
          },
          "metadata": {},
          "execution_count": 10
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "y_train = pd.Series(y_train.ravel())"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "-rEBWLR66XWf",
        "outputId": "b92a0340-8582-4df9-b65a-445cb59d3167"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/tmp/ipython-input-1683706722.py:1: FutureWarning: Series.ravel is deprecated. The underlying array is already 1D, so ravel is not necessary.  Use `to_numpy()` for conversion to a numpy array instead.\n",
            "  y_train = pd.Series(y_train.ravel())\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "### Candidate Predictors"
      ],
      "metadata": {
        "id": "sKGo7y6wY3-O"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "if EXPERIMENT == \"synthetic_binary\":\n",
        "  candidate_models = baseline_models()\n",
        "  fitted_models = train_models(x_train, y_train, candidate_models)\n",
        "\n",
        "elif EXPERIMENT == \"cancer\":\n",
        "  fitted_models_, results = train_models(x_train, y_train, x_test, y_test)\n",
        "  mlp_model = fitted_models_[\"mlp\"]\n",
        "  mlp_preds = mlp_predict(fitted_models_, x_test)\n",
        "\n",
        "  class PrecomputedModel:\n",
        "        def __init__(self, yhat):\n",
        "          self.yhat = np.asarray(yhat)\n",
        "        def predict(self, X):\n",
        "          return self.yhat\n",
        "\n",
        "  fitted_models = {\"logistic\": fitted_models_[\"ridge\"], \"xgb\": fitted_models_[\"xgb\"], \"hgb\": fitted_models_[\"hgb\"], \"MLP\": PrecomputedModel(mlp_preds)}\n",
        "  preds = predict_table(fitted_models, x_test, y_test, meta_df)\n",
        "\n",
        "elif EXPERIMENT == \"fraud\":\n",
        "  fitted_models = train_models(x_train, y_train)\n",
        "\n",
        "elif EXPERIMENT == \"UCI\":\n",
        "  fitted_models = train_models(x_train, y_train, task=TASK)"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "dppuetxvUW08",
        "outputId": "a88f364b-6f53-44ed-ee0c-fa1c66800c33"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "--- Fitting Logistic ---\n",
            "--- Fitting MLP ---\n",
            "--- Fitting XGB ---\n",
            "--- Fitting HGB ---\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "M = len(fitted_models)\n",
        "print(\"Number of candidate models:\", M)"
      ],
      "metadata": {
        "id": "0Caz43rIlQKk",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "226bc38f-12ed-4691-d45c-e398b4e5ad49"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Number of candidate models: 4\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "if EXPERIMENT == \"syntethic_binary\":\n",
        "  eval_res = accuracy_by_region(x_test, y_test, types_test, fitted_models)\n",
        "  eval_res"
      ],
      "metadata": {
        "id": "rc-9cbKZV_Nl"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "if EXPERIMENT == \"cancer\":\n",
        "    if \"primary_site\" in preds.columns:\n",
        "        types_test = preds[\"primary_site\"]\n",
        "    elif \"primary_disease\" in preds.columns:\n",
        "        types_test = preds[\"primary_disease\"]\n",
        "    else:\n",
        "        types_test = preds[\"compound\"]\n",
        "\n",
        "    eval_res = metrics_by_region(x_test, y_test, types_test, fitted_models)\n",
        "    display(eval_res.sort_values([\"type\",\"model\"]))"
      ],
      "metadata": {
        "id": "JpLuv-tsjxIo"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "if EXPERIMENT == \"fraud\":\n",
        "  results_b = evaluate_models(fitted_models, x_test_b, y_test_b)\n",
        "  disp_results_b = format_results(results_b)\n",
        "  display(disp_results_b)"
      ],
      "metadata": {
        "id": "hNdrQSRlt4w-",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 206
        },
        "outputId": "406334de-cb76-4738-dfad-ce30e7beb62e"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "          Accuracy       ECE\n",
              "model                       \n",
              "XGB       0.821624  0.091939\n",
              "MLP       0.690237  0.299396\n",
              "HGB       0.685903  0.296229\n",
              "Logistic  0.560447  0.392548"
            ],
            "text/html": [
              "\n",
              "  <div id=\"df-c605d53b-1adc-48c4-a44a-abbb6096ae99\" class=\"colab-df-container\">\n",
              "    <div>\n",
              "<style scoped>\n",
              "    .dataframe tbody tr th:only-of-type {\n",
              "        vertical-align: middle;\n",
              "    }\n",
              "\n",
              "    .dataframe tbody tr th {\n",
              "        vertical-align: top;\n",
              "    }\n",
              "\n",
              "    .dataframe thead th {\n",
              "        text-align: right;\n",
              "    }\n",
              "</style>\n",
              "<table border=\"1\" class=\"dataframe\">\n",
              "  <thead>\n",
              "    <tr style=\"text-align: right;\">\n",
              "      <th></th>\n",
              "      <th>Accuracy</th>\n",
              "      <th>ECE</th>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>model</th>\n",
              "      <th></th>\n",
              "      <th></th>\n",
              "    </tr>\n",
              "  </thead>\n",
              "  <tbody>\n",
              "    <tr>\n",
              "      <th>XGB</th>\n",
              "      <td>0.821624</td>\n",
              "      <td>0.091939</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>MLP</th>\n",
              "      <td>0.690237</td>\n",
              "      <td>0.299396</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>HGB</th>\n",
              "      <td>0.685903</td>\n",
              "      <td>0.296229</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>Logistic</th>\n",
              "      <td>0.560447</td>\n",
              "      <td>0.392548</td>\n",
              "    </tr>\n",
              "  </tbody>\n",
              "</table>\n",
              "</div>\n",
              "    <div class=\"colab-df-buttons\">\n",
              "\n",
              "  <div class=\"colab-df-container\">\n",
              "    <button class=\"colab-df-convert\" onclick=\"convertToInteractive('df-c605d53b-1adc-48c4-a44a-abbb6096ae99')\"\n",
              "            title=\"Convert this dataframe to an interactive table.\"\n",
              "            style=\"display:none;\">\n",
              "\n",
              "  <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\">\n",
              "    <path d=\"M120-120v-720h720v720H120Zm60-500h600v-160H180v160Zm220 220h160v-160H400v160Zm0 220h160v-160H400v160ZM180-400h160v-160H180v160Zm440 0h160v-160H620v160ZM180-180h160v-160H180v160Zm440 0h160v-160H620v160Z\"/>\n",
              "  </svg>\n",
              "    </button>\n",
              "\n",
              "  <style>\n",
              "    .colab-df-container {\n",
              "      display:flex;\n",
              "      gap: 12px;\n",
              "    }\n",
              "\n",
              "    .colab-df-convert {\n",
              "      background-color: #E8F0FE;\n",
              "      border: none;\n",
              "      border-radius: 50%;\n",
              "      cursor: pointer;\n",
              "      display: none;\n",
              "      fill: #1967D2;\n",
              "      height: 32px;\n",
              "      padding: 0 0 0 0;\n",
              "      width: 32px;\n",
              "    }\n",
              "\n",
              "    .colab-df-convert:hover {\n",
              "      background-color: #E2EBFA;\n",
              "      box-shadow: 0px 1px 2px rgba(60, 64, 67, 0.3), 0px 1px 3px 1px rgba(60, 64, 67, 0.15);\n",
              "      fill: #174EA6;\n",
              "    }\n",
              "\n",
              "    .colab-df-buttons div {\n",
              "      margin-bottom: 4px;\n",
              "    }\n",
              "\n",
              "    [theme=dark] .colab-df-convert {\n",
              "      background-color: #3B4455;\n",
              "      fill: #D2E3FC;\n",
              "    }\n",
              "\n",
              "    [theme=dark] .colab-df-convert:hover {\n",
              "      background-color: #434B5C;\n",
              "      box-shadow: 0px 1px 3px 1px rgba(0, 0, 0, 0.15);\n",
              "      filter: drop-shadow(0px 1px 2px rgba(0, 0, 0, 0.3));\n",
              "      fill: #FFFFFF;\n",
              "    }\n",
              "  </style>\n",
              "\n",
              "    <script>\n",
              "      const buttonEl =\n",
              "        document.querySelector('#df-c605d53b-1adc-48c4-a44a-abbb6096ae99 button.colab-df-convert');\n",
              "      buttonEl.style.display =\n",
              "        google.colab.kernel.accessAllowed ? 'block' : 'none';\n",
              "\n",
              "      async function convertToInteractive(key) {\n",
              "        const element = document.querySelector('#df-c605d53b-1adc-48c4-a44a-abbb6096ae99');\n",
              "        const dataTable =\n",
              "          await google.colab.kernel.invokeFunction('convertToInteractive',\n",
              "                                                    [key], {});\n",
              "        if (!dataTable) return;\n",
              "\n",
              "        const docLinkHtml = 'Like what you see? Visit the ' +\n",
              "          '<a target=\"_blank\" href=https://colab.research.google.com/notebooks/data_table.ipynb>data table notebook</a>'\n",
              "          + ' to learn more about interactive tables.';\n",
              "        element.innerHTML = '';\n",
              "        dataTable['output_type'] = 'display_data';\n",
              "        await google.colab.output.renderOutput(dataTable, element);\n",
              "        const docLink = document.createElement('div');\n",
              "        docLink.innerHTML = docLinkHtml;\n",
              "        element.appendChild(docLink);\n",
              "      }\n",
              "    </script>\n",
              "  </div>\n",
              "\n",
              "\n",
              "    <div id=\"df-783f8da5-0f71-4e32-986d-e2f7aa6990c0\">\n",
              "      <button class=\"colab-df-quickchart\" onclick=\"quickchart('df-783f8da5-0f71-4e32-986d-e2f7aa6990c0')\"\n",
              "                title=\"Suggest charts\"\n",
              "                style=\"display:none;\">\n",
              "\n",
              "<svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\"viewBox=\"0 0 24 24\"\n",
              "     width=\"24px\">\n",
              "    <g>\n",
              "        <path d=\"M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zM9 17H7v-7h2v7zm4 0h-2V7h2v10zm4 0h-2v-4h2v4z\"/>\n",
              "    </g>\n",
              "</svg>\n",
              "      </button>\n",
              "\n",
              "<style>\n",
              "  .colab-df-quickchart {\n",
              "      --bg-color: #E8F0FE;\n",
              "      --fill-color: #1967D2;\n",
              "      --hover-bg-color: #E2EBFA;\n",
              "      --hover-fill-color: #174EA6;\n",
              "      --disabled-fill-color: #AAA;\n",
              "      --disabled-bg-color: #DDD;\n",
              "  }\n",
              "\n",
              "  [theme=dark] .colab-df-quickchart {\n",
              "      --bg-color: #3B4455;\n",
              "      --fill-color: #D2E3FC;\n",
              "      --hover-bg-color: #434B5C;\n",
              "      --hover-fill-color: #FFFFFF;\n",
              "      --disabled-bg-color: #3B4455;\n",
              "      --disabled-fill-color: #666;\n",
              "  }\n",
              "\n",
              "  .colab-df-quickchart {\n",
              "    background-color: var(--bg-color);\n",
              "    border: none;\n",
              "    border-radius: 50%;\n",
              "    cursor: pointer;\n",
              "    display: none;\n",
              "    fill: var(--fill-color);\n",
              "    height: 32px;\n",
              "    padding: 0;\n",
              "    width: 32px;\n",
              "  }\n",
              "\n",
              "  .colab-df-quickchart:hover {\n",
              "    background-color: var(--hover-bg-color);\n",
              "    box-shadow: 0 1px 2px rgba(60, 64, 67, 0.3), 0 1px 3px 1px rgba(60, 64, 67, 0.15);\n",
              "    fill: var(--button-hover-fill-color);\n",
              "  }\n",
              "\n",
              "  .colab-df-quickchart-complete:disabled,\n",
              "  .colab-df-quickchart-complete:disabled:hover {\n",
              "    background-color: var(--disabled-bg-color);\n",
              "    fill: var(--disabled-fill-color);\n",
              "    box-shadow: none;\n",
              "  }\n",
              "\n",
              "  .colab-df-spinner {\n",
              "    border: 2px solid var(--fill-color);\n",
              "    border-color: transparent;\n",
              "    border-bottom-color: var(--fill-color);\n",
              "    animation:\n",
              "      spin 1s steps(1) infinite;\n",
              "  }\n",
              "\n",
              "  @keyframes spin {\n",
              "    0% {\n",
              "      border-color: transparent;\n",
              "      border-bottom-color: var(--fill-color);\n",
              "      border-left-color: var(--fill-color);\n",
              "    }\n",
              "    20% {\n",
              "      border-color: transparent;\n",
              "      border-left-color: var(--fill-color);\n",
              "      border-top-color: var(--fill-color);\n",
              "    }\n",
              "    30% {\n",
              "      border-color: transparent;\n",
              "      border-left-color: var(--fill-color);\n",
              "      border-top-color: var(--fill-color);\n",
              "      border-right-color: var(--fill-color);\n",
              "    }\n",
              "    40% {\n",
              "      border-color: transparent;\n",
              "      border-right-color: var(--fill-color);\n",
              "      border-top-color: var(--fill-color);\n",
              "    }\n",
              "    60% {\n",
              "      border-color: transparent;\n",
              "      border-right-color: var(--fill-color);\n",
              "    }\n",
              "    80% {\n",
              "      border-color: transparent;\n",
              "      border-right-color: var(--fill-color);\n",
              "      border-bottom-color: var(--fill-color);\n",
              "    }\n",
              "    90% {\n",
              "      border-color: transparent;\n",
              "      border-bottom-color: var(--fill-color);\n",
              "    }\n",
              "  }\n",
              "</style>\n",
              "\n",
              "      <script>\n",
              "        async function quickchart(key) {\n",
              "          const quickchartButtonEl =\n",
              "            document.querySelector('#' + key + ' button');\n",
              "          quickchartButtonEl.disabled = true;  // To prevent multiple clicks.\n",
              "          quickchartButtonEl.classList.add('colab-df-spinner');\n",
              "          try {\n",
              "            const charts = await google.colab.kernel.invokeFunction(\n",
              "                'suggestCharts', [key], {});\n",
              "          } catch (error) {\n",
              "            console.error('Error during call to suggestCharts:', error);\n",
              "          }\n",
              "          quickchartButtonEl.classList.remove('colab-df-spinner');\n",
              "          quickchartButtonEl.classList.add('colab-df-quickchart-complete');\n",
              "        }\n",
              "        (() => {\n",
              "          let quickchartButtonEl =\n",
              "            document.querySelector('#df-783f8da5-0f71-4e32-986d-e2f7aa6990c0 button');\n",
              "          quickchartButtonEl.style.display =\n",
              "            google.colab.kernel.accessAllowed ? 'block' : 'none';\n",
              "        })();\n",
              "      </script>\n",
              "    </div>\n",
              "\n",
              "  <div id=\"id_f5695390-af81-4afe-9624-34838d58496b\">\n",
              "    <style>\n",
              "      .colab-df-generate {\n",
              "        background-color: #E8F0FE;\n",
              "        border: none;\n",
              "        border-radius: 50%;\n",
              "        cursor: pointer;\n",
              "        display: none;\n",
              "        fill: #1967D2;\n",
              "        height: 32px;\n",
              "        padding: 0 0 0 0;\n",
              "        width: 32px;\n",
              "      }\n",
              "\n",
              "      .colab-df-generate:hover {\n",
              "        background-color: #E2EBFA;\n",
              "        box-shadow: 0px 1px 2px rgba(60, 64, 67, 0.3), 0px 1px 3px 1px rgba(60, 64, 67, 0.15);\n",
              "        fill: #174EA6;\n",
              "      }\n",
              "\n",
              "      [theme=dark] .colab-df-generate {\n",
              "        background-color: #3B4455;\n",
              "        fill: #D2E3FC;\n",
              "      }\n",
              "\n",
              "      [theme=dark] .colab-df-generate:hover {\n",
              "        background-color: #434B5C;\n",
              "        box-shadow: 0px 1px 3px 1px rgba(0, 0, 0, 0.15);\n",
              "        filter: drop-shadow(0px 1px 2px rgba(0, 0, 0, 0.3));\n",
              "        fill: #FFFFFF;\n",
              "      }\n",
              "    </style>\n",
              "    <button class=\"colab-df-generate\" onclick=\"generateWithVariable('disp_results_b')\"\n",
              "            title=\"Generate code using this dataframe.\"\n",
              "            style=\"display:none;\">\n",
              "\n",
              "  <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\"viewBox=\"0 0 24 24\"\n",
              "       width=\"24px\">\n",
              "    <path d=\"M7,19H8.4L18.45,9,17,7.55,7,17.6ZM5,21V16.75L18.45,3.32a2,2,0,0,1,2.83,0l1.4,1.43a1.91,1.91,0,0,1,.58,1.4,1.91,1.91,0,0,1-.58,1.4L9.25,21ZM18.45,9,17,7.55Zm-12,3A5.31,5.31,0,0,0,4.9,8.1,5.31,5.31,0,0,0,1,6.5,5.31,5.31,0,0,0,4.9,4.9,5.31,5.31,0,0,0,6.5,1,5.31,5.31,0,0,0,8.1,4.9,5.31,5.31,0,0,0,12,6.5,5.46,5.46,0,0,0,6.5,12Z\"/>\n",
              "  </svg>\n",
              "    </button>\n",
              "    <script>\n",
              "      (() => {\n",
              "      const buttonEl =\n",
              "        document.querySelector('#id_f5695390-af81-4afe-9624-34838d58496b button.colab-df-generate');\n",
              "      buttonEl.style.display =\n",
              "        google.colab.kernel.accessAllowed ? 'block' : 'none';\n",
              "\n",
              "      buttonEl.onclick = () => {\n",
              "        google.colab.notebook.generateWithVariable('disp_results_b');\n",
              "      }\n",
              "      })();\n",
              "    </script>\n",
              "  </div>\n",
              "\n",
              "    </div>\n",
              "  </div>\n"
            ],
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "dataframe",
              "variable_name": "disp_results_b",
              "summary": "{\n  \"name\": \"disp_results_b\",\n  \"rows\": 4,\n  \"fields\": [\n    {\n      \"column\": \"model\",\n      \"properties\": {\n        \"dtype\": \"string\",\n        \"num_unique_values\": 4,\n        \"samples\": [\n          \"MLP\",\n          \"Logistic\",\n          \"XGB\"\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"Accuracy\",\n      \"properties\": {\n        \"dtype\": \"number\",\n        \"std\": 0.10665348762658289,\n        \"min\": 0.5604470802919708,\n        \"max\": 0.8216240875912408,\n        \"num_unique_values\": 4,\n        \"samples\": [\n          0.6902372262773723,\n          0.5604470802919708,\n          0.8216240875912408\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"ECE\",\n      \"properties\": {\n        \"dtype\": \"number\",\n        \"std\": 0.12685412601517432,\n        \"min\": 0.09193863740350616,\n        \"max\": 0.3925479947326746,\n        \"num_unique_values\": 4,\n        \"samples\": [\n          0.299395609039799,\n          0.3925479947326746,\n          0.09193863740350616\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    }\n  ]\n}"
            }
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "if EXPERIMENT == \"UCI\":\n",
        "  results = evaluate_models(fitted_models, x_test, y_test)\n",
        "  display(results)"
      ],
      "metadata": {
        "id": "278E0KHufQ0C"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "source": [
        "## Previous methods"
      ],
      "metadata": {
        "id": "COnmGU4od7os"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "if EXPERIMENT == \"synthetic_binary\":\n",
        "  EPOCHS = 10\n",
        "  BATCH_SIZE = 64\n",
        "if EXPERIMENT == \"cancer\":\n",
        "  EPOCHS = 20\n",
        "  BATCH_SIZE = 128\n",
        "if EXPERIMENT == \"fraud\":\n",
        "  EPOCHS = 10\n",
        "  BATCH_SIZE = 64\n",
        "if EXPERIMENT == \"UCI\":\n",
        "  BATCH_SIZE = 64\n",
        "  EPOCHS = 10"
      ],
      "metadata": {
        "id": "-u1CYcVPC9-s"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "class BasicNet(tf.keras.Model):\n",
        "    def __init__(self, feature_dim, num_models):\n",
        "        super().__init__()\n",
        "        self.d1 = tf.keras.layers.Dense(64, activation=\"relu\")\n",
        "        self.d2 = tf.keras.layers.Dense(32, activation=\"relu\")\n",
        "        self.d3 = tf.keras.layers.Dense(16, activation=\"relu\")\n",
        "        self.m_head = tf.keras.layers.Dense(num_models)\n",
        "\n",
        "    def call(self, x):\n",
        "        h = self.d1(x)\n",
        "        h = self.d2(h)\n",
        "        h = self.d3(h)\n",
        "        return self.m_head(h)\n",
        "\n",
        "    def build(self, input_shape):\n",
        "        super().build(input_shape)"
      ],
      "metadata": {
        "id": "1sf_begwd9M5"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "source": [
        "### Baseline methods"
      ],
      "metadata": {
        "id": "CSibZRkLd-sG"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "if EXPERIMENT == \"synthetic_binary\":\n",
        "  best_model, uniform_avg, freq_avg, bma = baseline_methods(\"binary\", x_train, y_train, x_test, fitted_models)\n",
        "\n",
        "elif EXPERIMENT == \"cancer\":\n",
        "  mlp_preds_tr = mlp_predict(fitted_models_, x_train)\n",
        "  mlp_preds_te = mlp_predict(fitted_models_, x_test)\n",
        "\n",
        "  class PrecomputedModel:\n",
        "      def __init__(self, yhat_test, yhat_train=None, name=\"\"):\n",
        "          self.yhat_test  = np.asarray(yhat_test).ravel()\n",
        "          self.yhat_train = None if yhat_train is None else np.asarray(yhat_train).ravel()\n",
        "          self.name = name\n",
        "      def predict(self, X):\n",
        "          return self.yhat_test\n",
        "\n",
        "  tr_te_fitted_models = {\n",
        "      \"ridge\": fitted_models_[\"ridge\"],\n",
        "      \"xgb\":   fitted_models_[\"xgb\"],\n",
        "      \"hgb\":   fitted_models_[\"hgb\"],\n",
        "      \"MLP\":   PrecomputedModel(mlp_preds_te, yhat_train=mlp_preds_tr, name=\"MLP\"),\n",
        "  }\n",
        "\n",
        "  best_model, uniform_avg, freq_avg, bma = baseline_methods(\"regression\", x_train, y_train, x_test, tr_te_fitted_models)\n",
        "\n",
        "elif EXPERIMENT == \"fraud\":\n",
        "  best_model, uniform_avg, freq_avg, bma = baseline_methods(\"binary\", x_train, y_train, x_test_b, fitted_models)\n",
        "\n",
        "elif EXPERIMENT == \"UCI\":\n",
        "  if TASK == \"classification\":\n",
        "    best_model, uniform_avg, freq_avg, bma = baseline_methods(\"binary\", x_train, y_train, x_test, fitted_models)\n",
        "  elif TASK == \"regression\":\n",
        "    best_model, uniform_avg, freq_avg, bma = baseline_methods(\"regression\", x_train, y_train, x_test, fitted_models)"
      ],
      "metadata": {
        "id": "XsGZLNHJdPsD"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "markdown",
      "source": [
        "### Previous adaptive methods"
      ],
      "metadata": {
        "id": "caKwJEOSiOeX"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "if EXPERIMENT == \"synthetic_binary\":\n",
        "  cv_pred_train = compute_cv_pred_probas(fitted_models, x_train, y_train, cv=5)\n",
        "\n",
        "elif EXPERIMENT == \"cancer\":\n",
        "  mlp_params = extract_mlp_params_from_pipeline(fitted_models_[\"mlp\"])\n",
        "  mlp_context = {\"mlp_params\": mlp_params, \"RSEED\": 42}\n",
        "\n",
        "  models_for_oos = {\n",
        "      \"ridge\": fitted_models_[\"ridge\"],\n",
        "      \"xgb\":   fitted_models_[\"xgb\"],\n",
        "      \"hgb\":   fitted_models_[\"hgb\"],\n",
        "      \"MLP\": object()\n",
        "  }\n",
        "\n",
        "  cv_pred_train = compute_oos_pred_values(\n",
        "      models_for_oos, x_train, y_train,\n",
        "      mode=\"holdout\", holdout_size=0.2,\n",
        "      mlp_context=mlp_context,\n",
        "  )\n",
        "\n",
        "if EXPERIMENT == \"fraud\":\n",
        "\n",
        "  from sklearn.base import clone\n",
        "\n",
        "  import numpy as np\n",
        "\n",
        "  def cv_to_Z(cv_pred_train, keys, n_train):\n",
        "    if isinstance(cv_pred_train, dict):\n",
        "        cols = [k for k in keys if k in cv_pred_train] or list(cv_pred_train.keys())\n",
        "        Z = np.column_stack([np.asarray(cv_pred_train[k]).ravel() for k in cols])\n",
        "        if Z.shape[0] != n_train:\n",
        "            raise ValueError(f\"Dict: got {Z.shape[0]} rows, expected n_train={n_train}\")\n",
        "        return Z.astype(np.float32), cols\n",
        "\n",
        "    if isinstance(cv_pred_train, pd.DataFrame):\n",
        "        cols = [k for k in keys if k in cv_pred_train.columns] or list(cv_pred_train.columns)\n",
        "        try:\n",
        "          Z = cv_pred_train[cols].to_numpy(dtype=np.float32)\n",
        "        except:\n",
        "          Z = cv_pred_train[cols]\n",
        "        if Z.shape[0] != n_train:\n",
        "            raise ValueError(f\"DataFrame: got {Z.shape[0]} rows, expected n_train={n_train}\")\n",
        "        return Z, cols\n",
        "\n",
        "    if isinstance(cv_pred_train, (list, tuple)):\n",
        "        arrs = [np.asarray(a).ravel() for a in cv_pred_train]\n",
        "        Z = np.column_stack(arrs).astype(np.float32)\n",
        "        if Z.shape[0] != n_train and Z.shape[1] == n_train:\n",
        "            Z = Z.T\n",
        "        if Z.shape[0] != n_train:\n",
        "            raise ValueError(f\"List/tuple: got {Z.shape}, expected first dim n_train={n_train}\")\n",
        "        return Z, keys[:Z.shape[1]]\n",
        "\n",
        "    Z = np.asarray(cv_pred_train)\n",
        "    if Z.ndim == 1:\n",
        "        Z = Z[:, None]\n",
        "    if Z.shape[0] != n_train and Z.shape[1] == n_train:\n",
        "        Z = Z.T\n",
        "    if Z.shape[0] != n_train:\n",
        "        raise ValueError(f\"Array: got {Z.shape}, expected first dim n_train={n_train}\")\n",
        "    return Z.astype(np.float32), keys[:Z.shape[1]]\n",
        "\n",
        "  def predict_proba_aligned(pipe, X):\n",
        "    pre = getattr(pipe, \"named_steps\", {}).get(\"preproc\", None)\n",
        "    cols = None\n",
        "    if pre is not None and hasattr(pre, \"feature_names_in_\"):\n",
        "        cols = list(pre.feature_names_in_)\n",
        "    elif hasattr(pipe, \"feature_names_in_\"):\n",
        "        cols = list(pipe.feature_names_in_)\n",
        "    X2 = X.reindex(columns=cols, fill_value=np.nan) if cols is not None else X\n",
        "    return pipe.predict_proba(X2)[:, 1]\n",
        "\n",
        "  cv_pred_train = compute_cv_pred_probas(\n",
        "    {k: clone(m) for k, m in fitted_models.items()},\n",
        "    x_train, y_train, cv=2)\n",
        "\n",
        "  KEYS = list(fitted_models.keys())\n",
        "  if isinstance(cv_pred_train, dict):\n",
        "      Ztr = np.column_stack([np.asarray(cv_pred_train[k]).ravel() for k in KEYS]).astype(np.float32)\n",
        "  else:\n",
        "      Ztr = np.asarray(cv_pred_train)\n",
        "      if Ztr.ndim == 1: Ztr = Ztr[:, None]\n",
        "      if Ztr.shape[0] != len(x_train) and Ztr.shape[1] == len(x_train): Ztr = Ztr.T\n",
        "      Ztr = Ztr.astype(np.float32)\n",
        "      KEYS = KEYS[:Ztr.shape[1]]\n",
        "  if Ztr.ndim == 3 and Ztr.shape[-1] == 2:\n",
        "      Ztr = Ztr[..., 1]\n",
        "  assert Ztr.ndim == 2\n",
        "\n",
        "  Zte = np.column_stack([fitted_models[k].predict_proba(x_test_b)[:, 1] for k in KEYS]).astype(np.float32)\n",
        "  assert Zte.ndim == 2 and Zte.shape[1] == Ztr.shape[1]\n",
        "\n",
        "\n",
        "if EXPERIMENT == \"UCI\":\n",
        "  if TASK == \"classification\":\n",
        "    cv_pred_train = compute_cv_pred_probas(fitted_models, x_train, y_train, cv=5)\n",
        "\n",
        "  elif TASK == \"regression\":\n",
        "    cv_pred_train = compute_oos_pred_values(fitted_models, x_train, y_train, cv=5)"
      ],
      "metadata": {
        "id": "qDI59TBpo1gE",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "0500b242-0a70-4060-f2e2-0273b04785cd"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stderr",
          "text": [
            "/usr/local/lib/python3.12/dist-packages/sklearn/neural_network/_multilayer_perceptron.py:609: UserWarning: Got `batch_size` less than 1 or larger than sample size. It is going to be clipped\n",
            "  warnings.warn(\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "##### Mixture of Experts Gating (Jacobs et al., 1991)"
      ],
      "metadata": {
        "id": "2TU_4-sRiiMM"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "MoEGatingNet = BasicNet"
      ],
      "metadata": {
        "id": "fwhNJtg0nXu_"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "moe_start = time.time()\n",
        "\n",
        "if EXPERIMENT == \"synthetic_binary\":\n",
        "\n",
        "  moe_gate = moe_gating(x_train, y_train, fitted_models, D, M, MoEGatingNet, task=\"binary\", epochs=EPOCHS, lr=1e-3, l2=0.0, batch_size=BATCH_SIZE, verbose=True)\n",
        "  moe_pred = moe_prediction(x_test, moe_gate, fitted_models, task=\"binary\")\n",
        "\n",
        "elif EXPERIMENT == \"cancer\":\n",
        "\n",
        "  Xtr_gate = build_gate_features(fitted_models_, x_train)\n",
        "  Xte_gate = build_gate_features(fitted_models_, x_test)\n",
        "\n",
        "  base_pred_train = np.column_stack([\n",
        "      fitted_models_[\"ridge\"].predict(x_train),\n",
        "      fitted_models_[\"xgb\"].predict(x_train),\n",
        "      fitted_models_[\"hgb\"].predict(x_train),\n",
        "      mlp_predict(fitted_models_, x_train),\n",
        "  ]).astype(np.float32)\n",
        "\n",
        "  base_pred_test = np.column_stack([\n",
        "      fitted_models_[\"ridge\"].predict(x_test),\n",
        "      fitted_models_[\"xgb\"].predict(x_test),\n",
        "      fitted_models_[\"hgb\"].predict(x_test),\n",
        "      mlp_predict(fitted_models_, x_test),\n",
        "  ]).astype(np.float32)\n",
        "\n",
        "  gate_net = moe_gating(\n",
        "      Xtr_gate, y_train, fitted_models,\n",
        "      feature_dim=Xtr_gate.shape[1],\n",
        "      num_models=base_pred_train.shape[1],\n",
        "      MoEGatingNet=MoEGatingNet,\n",
        "      task=\"regression\",\n",
        "      base_pred_train=base_pred_train,\n",
        "      epochs=20, lr=1e-3, l2=0.0, batch_size=BATCH_SIZE, verbose=True\n",
        "  )\n",
        "\n",
        "  moe_pred, moe_w = moe_gating_predict(gate_net, Xte_gate, base_pred_test, task=\"regression\", return_weights=True)\n",
        "\n",
        "elif EXPERIMENT == \"fraud\":\n",
        "\n",
        "  gate_net = moe_gating(\n",
        "    x_train, y_train, fitted_models,\n",
        "    feature_dim=Ztr.shape[1], num_models=len(KEYS),\n",
        "    MoEGatingNet=MoEGatingNet, task=\"binary\",\n",
        "    epochs=EPOCHS, lr=1e-3, l2=0.0, batch_size=BATCH_SIZE, verbose=True,\n",
        "    gate_features=Ztr)\n",
        "\n",
        "  p1 = np.asarray(Zte, dtype=np.float32)\n",
        "  p1 = np.clip(p1, 1e-7, 1-1e-7)\n",
        "  BASE3 = np.stack([1.0 - p1, p1], axis=-1).astype(np.float32)\n",
        "\n",
        "  moe_pred, moe_w = moe_gating_predict(\n",
        "      gate_net,\n",
        "      Zte,\n",
        "      BASE3,\n",
        "      task=\"binary\",\n",
        "      return_weights=True\n",
        "  )\n",
        "\n",
        "elif EXPERIMENT == \"UCI\":\n",
        "  if TASK == \"classification\":\n",
        "    PREFER = (\"LinearSVM (B5)\", \"NB (B2)\", \"KNN (B1)\", \"RF (B3)\", \"ExtraTrees (B4)\")\n",
        "    moe_gate = moe_gating(x_train, y_train, fitted_models, D, M, MoEGatingNet, task=\"binary\", epochs=EPOCHS, lr=1e-3, l2=0.0, batch_size=BATCH_SIZE, verbose=True)\n",
        "    moe_pred = moe_prediction(x_test, moe_gate, fitted_models, task=\"binary\")\n",
        "  elif TASK == \"regression\":\n",
        "    PREFER = (\"Ridge(B6)\", \"Lasso(B6)\", \"KNN(B7)\", \"RF(B7)\", \"ExtraTrees(B7)\")\n",
        "    moe_gate = moe_gating(x_train, y_train, fitted_models, D, M, MoEGatingNet, task=\"regression\", epochs=EPOCHS, lr=1e-3, l2=0.0, batch_size=BATCH_SIZE, verbose=True)\n",
        "    moe_pred = moe_prediction(x_test, moe_gate, fitted_models, task=\"regression\")\n",
        "  else:\n",
        "    raise ValueError(\"Task must be 'binary' or 'regression'\")\n",
        "\n",
        "moe_end = time.time()\n",
        "moe_time = moe_end - moe_start\n",
        "print(f\"Execution time: {moe_time:.4f} seconds\")"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "Ts0qRfk3irb4",
        "outputId": "2440cb3e-3167-401f-b7c9-24980bed47fe"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "[MoE-gate/binary] epoch 1/10 loss=0.0435\n",
            "[MoE-gate/binary] epoch 2/10 loss=0.0401\n",
            "[MoE-gate/binary] epoch 3/10 loss=0.0398\n",
            "[MoE-gate/binary] epoch 4/10 loss=0.0396\n",
            "[MoE-gate/binary] epoch 5/10 loss=0.0395\n",
            "[MoE-gate/binary] epoch 6/10 loss=0.0395\n",
            "[MoE-gate/binary] epoch 7/10 loss=0.0395\n",
            "[MoE-gate/binary] epoch 8/10 loss=0.0394\n",
            "[MoE-gate/binary] epoch 9/10 loss=0.0394\n",
            "[MoE-gate/binary] epoch 10/10 loss=0.0394\n",
            "Execution time: 365.3818 seconds\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "Dynamic Local-Accuracy Combination (Woods et al. ,1997)"
      ],
      "metadata": {
        "id": "N4fPaj6wlhIt"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "dla_start = time.time()\n",
        "\n",
        "if EXPERIMENT == \"synthetic_binary\":\n",
        "  dla_pred = dynamic_local_accuracy(\n",
        "    x_train, y_train, x_test,\n",
        "    base_pred_train_cv=cv_pred_train,\n",
        "    fitted_models=fitted_models,\n",
        "    k=50, temp=1.0, alpha=1.0,\n",
        "    task=\"binary\")\n",
        "\n",
        "elif EXPERIMENT == \"cancer\":\n",
        "  dla_pred, dla_w = dynamic_local_accuracy(\n",
        "    x_train=None, y_train=y_train, x_test=None,\n",
        "    base_pred_train_cv=None, fitted_models=None,\n",
        "    k=50, temp=1.0, alpha=1.0,\n",
        "    task=\"regression\",\n",
        "    X_train_feat=Xtr_gate,\n",
        "    X_test_feat=Xte_gate,\n",
        "    base_pred_train_oos=cv_pred_train,\n",
        "    base_pred_test=base_pred_test,\n",
        "    return_weights=True\n",
        ")\n",
        "\n",
        "elif EXPERIMENT == \"fraud\":\n",
        "  dla_pred = dynamic_local_accuracy(\n",
        "    x_train, y_train, x_test_b,\n",
        "    base_pred_train_cv=cv_pred_train,\n",
        "    fitted_models=fitted_models,\n",
        "    k=50, temp=1.0, alpha=1.0,\n",
        "    task=\"binary\")\n",
        "\n",
        "elif EXPERIMENT == \"UCI\":\n",
        "  if TASK == \"classification\":\n",
        "    dla_pred = dynamic_local_accuracy(\n",
        "      x_train, y_train, x_test,\n",
        "      base_pred_train_cv=cv_pred_train,\n",
        "      fitted_models=fitted_models,\n",
        "      k=50, temp=1.0, alpha=1.0,\n",
        "      task=\"binary\")\n",
        "  if TASK == \"regression\":\n",
        "    dla_pred, dla_w = dynamic_local_accuracy(\n",
        "        x_train=None, y_train=y_train, x_test=None,\n",
        "        base_pred_train_cv=None, fitted_models=None,\n",
        "        k=50, temp=1.0, alpha=1.0,\n",
        "        task=\"regression\",\n",
        "        X_train_feat=Xtr_gate,\n",
        "        X_test_feat=Xte_gate,\n",
        "        base_pred_train_oos=cv_pred_train,\n",
        "        base_pred_test=base_pred_test,\n",
        "        return_weights=True\n",
        "        )\n",
        "\n",
        "dla_end = time.time()\n",
        "dla_time = dla_end - dla_start\n",
        "print(f\"Execution time: {dla_time:.4f} seconds\")"
      ],
      "metadata": {
        "id": "nKVeQeq0lhqe",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "bf575465-2160-4ab9-98f4-eea4d7ce7e51"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Execution time: 7.9718 seconds\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "Synthetic Model Combination (Chan & van der Schaar, 2022)"
      ],
      "metadata": {
        "id": "BVNT_q-InkWy"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "smc_start = time.time()\n",
        "\n",
        "if EXPERIMENT == \"synthetic_binary\":\n",
        "  smc_pred = smc_ensemble(\n",
        "    x_train, y_train, x_test,\n",
        "    base_pred_train_cv=cv_pred_train,\n",
        "    fitted_models=fitted_models,\n",
        "    conf_threshold=0.6, bandwidth=None, temp=1.0,\n",
        "    task=\"binary\"\n",
        ")\n",
        "\n",
        "elif EXPERIMENT == \"cancer\":\n",
        "  smc_pred, smc_w = smc_ensemble(\n",
        "    x_train=None, y_train=y_train, x_test=None,\n",
        "    base_pred_train_cv=None, fitted_models=fitted_models,\n",
        "    conf_threshold=0.6, bandwidth=None, temp=1.0,\n",
        "    task=\"regression\",\n",
        "    X_train_feat=Xtr_gate, X_test_feat=Xte_gate,\n",
        "    base_pred_train_oos=cv_pred_train,\n",
        "    base_pred_test=base_pred_test,\n",
        "    cover_q=0.30,\n",
        "    return_weights=True\n",
        ")\n",
        "\n",
        "elif EXPERIMENT == \"fraud\":\n",
        "  smc_pred = smc_ensemble(\n",
        "    x_train, y_train, x_test_b,\n",
        "    base_pred_train_cv=cv_pred_train,\n",
        "    fitted_models=fitted_models,\n",
        "    conf_threshold=0.6, bandwidth=None, temp=1.0,\n",
        "    task=\"binary\"\n",
        ")\n",
        "\n",
        "elif EXPERIMENT == \"UCI\":\n",
        "  if TASK == \"classification\":\n",
        "      smc_pred = smc_ensemble(\n",
        "        x_train, y_train, x_test,\n",
        "        base_pred_train_cv=cv_pred_train,\n",
        "        fitted_models=fitted_models,\n",
        "        conf_threshold=0.6, bandwidth=None, temp=1.0,\n",
        "        task=\"binary\"\n",
        "    )\n",
        "  if TASK == \"regression\":\n",
        "      smc_pred, smc_w = smc_ensemble(\n",
        "        x_train=None, y_train=y_train, x_test=None,\n",
        "        base_pred_train_cv=None, fitted_models=fitted_models,\n",
        "        conf_threshold=0.6, bandwidth=None, temp=1.0,\n",
        "        task=\"regression\",\n",
        "        X_train_feat=Xtr_gate, X_test_feat=Xte_gate,\n",
        "        base_pred_train_oos=cv_pred_train,\n",
        "        base_pred_test=base_pred_test,\n",
        "        cover_q=0.30,\n",
        "        return_weights=True\n",
        "    )\n",
        "\n",
        "smc_end = time.time()\n",
        "smc_time = smc_end - smc_start\n",
        "print(f\"Execution time: {smc_time:.4f} seconds\")"
      ],
      "metadata": {
        "id": "JVc6hV_XnwMa",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "50425c8a-3819-4f8d-e830-8927e8330e6c"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Execution time: 548.1578 seconds\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "Bayesian Hierarchical Stacking (Yao et al., 2022)"
      ],
      "metadata": {
        "id": "BDrG21C_RVtn"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "bhs_start = time.time()\n",
        "\n",
        "if EXPERIMENT == \"synthetic_binary\":\n",
        "  base_pred_test = np.stack([m.predict_proba(x_test) for m in fitted_models.values()], axis=1)\n",
        "  bhs_params = bhs_train(\n",
        "    x_train, y_train,\n",
        "    cv_pred_train,\n",
        "    epochs=EPOCHS, lr=5e-3, batch_size=BATCH_SIZE,\n",
        "    temp=1.0, prior_weight=1.0, s0=5.0,\n",
        "    verbose=False,\n",
        "    task=\"binary\"\n",
        ")\n",
        "  bhs_pred = bhs_predict(\n",
        "      bhs_params,\n",
        "      x_test,\n",
        "      base_pred_test\n",
        "  )\n",
        "\n",
        "elif EXPERIMENT == \"cancer\":\n",
        "\n",
        "  bhs_params = bhs_train(\n",
        "      Xtr_gate, y_train, cv_pred_train,\n",
        "      epochs=100, lr=5e-3, batch_size=BATCH_SIZE,\n",
        "      temp=1.0, prior_weight=1.0, s0=5.0,\n",
        "      verbose=True,\n",
        "      task=\"regression\")\n",
        "\n",
        "  bhs_pred, bhs_w = bhs_predict(\n",
        "      bhs_params,\n",
        "      Xte_gate,\n",
        "      base_pred_test,\n",
        "      return_weights=True\n",
        "  )\n",
        "\n",
        "elif EXPERIMENT == \"fraud\":\n",
        "  base_pred_test = np.stack([m.predict_proba(x_test_b) for m in fitted_models.values()], axis=1)\n",
        "  bhs_params = bhs_train(\n",
        "    x_train, y_train,\n",
        "    cv_pred_train,\n",
        "    epochs=EPOCHS, lr=5e-3, batch_size=BATCH_SIZE,\n",
        "    temp=1.0, prior_weight=1.0, s0=5.0,\n",
        "    verbose=False,\n",
        "    task=\"binary\"\n",
        ")\n",
        "  bhs_pred = bhs_predict(\n",
        "      bhs_params,\n",
        "      x_test_b,\n",
        "      base_pred_test\n",
        "  )\n",
        "\n",
        "elif EXPERIMENT == \"UCI\":\n",
        "  if TASK == \"classification\":\n",
        "\n",
        "    base_pred_test = np.stack([m.predict_proba(x_test) for m in fitted_models.values()], axis=1)\n",
        "\n",
        "    bhs_params = bhs_train(\n",
        "        x_train, y_train,\n",
        "        cv_pred_train,\n",
        "        epochs=EPOCHS, lr=5e-3, batch_size=BATCH_SIZE,\n",
        "        temp=1.0, prior_weight=1.0, s0=5.0,\n",
        "        verbose=False,\n",
        "        task=\"binary\"\n",
        "    )\n",
        "\n",
        "    D_tr = bhs_params[\"beta\"].shape[0]\n",
        "    X_feat = base_pred_test.reshape(base_pred_test.shape[0], -1).astype(np.float32)\n",
        "\n",
        "    if X_feat.shape[1] > D_tr:\n",
        "        X_feat = X_feat[:, :D_tr]\n",
        "    elif X_feat.shape[1] < D_tr:\n",
        "        pad = np.zeros((X_feat.shape[0], D_tr - X_feat.shape[1]), dtype=np.float32)\n",
        "        X_feat = np.concatenate([X_feat, pad], axis=1)\n",
        "\n",
        "    bhs_pred = bhs_predict(\n",
        "        bhs_params,\n",
        "        X_feat,\n",
        "        base_pred_test\n",
        "    )\n",
        "  if TASK == \"regression\":\n",
        "    bhs_params = bhs_train(\n",
        "      Xtr_gate, y_train, cv_pred_train,\n",
        "      epochs=200, lr=5e-3, batch_size=BATCH_SIZE,\n",
        "      temp=1.0, prior_weight=1.0, s0=5.0,\n",
        "      verbose=True,\n",
        "      task=\"regression\")\n",
        "\n",
        "    bhs_pred, bhs_w = bhs_predict(\n",
        "        bhs_params,\n",
        "        Xte_gate,\n",
        "        base_pred_test,\n",
        "        return_weights=True\n",
        "    )\n",
        "\n",
        "bhs_end = time.time()\n",
        "bhs_time = bhs_end - bhs_start\n",
        "print(f\"Execution time: {bhs_time:.4f} seconds\")"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "54kPPXlxRQvO",
        "outputId": "e0e248fe-eb07-4079-abb7-cc72f06b0d76"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Execution time: 18.0993 seconds\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "## Input Adaptive BMA"
      ],
      "metadata": {
        "id": "ndLD04hlsChq"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "if EXPERIMENT == \"synthetic_binary\":\n",
        "  # pred_test = np.stack([m.predict_proba(x_test) for m in fitted_models.values()], axis=1)\n",
        "  pred_test = np.column_stack([mdl.predict_proba(x_test)[:, 1] for mdl in fitted_models.values()]).astype(np.float32)\n",
        "elif EXPERIMENT in [\"cancer\", \"UCI\"]:\n",
        "  pred_test = base_pred_test\n",
        "elif EXPERIMENT == \"fraud\":\n",
        "  pred_test = np.stack([m.predict_proba(x_test) for m in fitted_models.values()], axis=1)"
      ],
      "metadata": {
        "id": "3xQYwZ1AyXkh"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "if EXPERIMENT in [\"synthetic_binary\", \"cancer\", \"fraud\"]:\n",
        "  posterior_net = BasicNet(D, M)\n",
        "  lr = 1e-3\n",
        "elif EXPERIMENT == \"UCI\":\n",
        "  if TASK == \"classification\":\n",
        "    M = cv_pred_train.shape[1]\n",
        "    C = cv_pred_train.shape[2]\n",
        "    posterior_net = BasicNet(M * C, M)\n",
        "    lr = 5e-3\n",
        "  elif TASK == \"regression\":\n",
        "    posterior_net = BasicNet(D, M)\n",
        "    lr = 1e-3\n",
        "optimizer = tf.keras.optimizers.Adam(lr)"
      ],
      "metadata": {
        "id": "MjHGpUHmsezq"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "if UCI_DATASET==\"bike-sharing\":\n",
        "  ymin, ymax = 0, 7\n",
        "elif UCI_DATASET==\"california-housing\":\n",
        "  ymin, ymax = 10, 14\n",
        "elif UCI_DATASET==\"communities\":\n",
        "  ymin, ymax = -0.1, 1.1"
      ],
      "metadata": {
        "id": "QNMbWiNI4iKa"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "iabma_start = time.time()\n",
        "\n",
        "if EXPERIMENT == \"synthetic_binary\":\n",
        "  posterior_net = train_adaptive_BMA(\"binary\", posterior_net, \"categorical\", optimizer, x_train, y_train, fitted_models, batch_size=BATCH_SIZE, epochs=EPOCHS,  kl_f_weight=0.05)\n",
        "\n",
        "elif EXPERIMENT == \"cancer\":\n",
        "  posterior_net = train_adaptive_BMA(\"regression\", posterior_net, \"categorical\", optimizer, x_train, y_train, fitted_models_, batch_size=BATCH_SIZE, epochs=30, kl_f_weight=0.2,\n",
        "                                      X_train_feat=Xtr_gate, y_min=-6, y_max=6)\n",
        "\n",
        "elif EXPERIMENT == \"fraud\":\n",
        "  posterior_net = train_adaptive_BMA(\"binary\", posterior_net, \"categorical\", optimizer, x_train, y_train, fitted_models, batch_size=BATCH_SIZE, epochs=EPOCHS,  kl_f_weight=0.2)\n",
        "\n",
        "elif EXPERIMENT == \"UCI\":\n",
        "  if TASK == \"classification\":\n",
        "    posterior_net = train_adaptive_BMA(\"binary\", posterior_net, \"categorical\", optimizer, x_train, y_train, fitted_models, batch_size=BATCH_SIZE, epochs=EPOCHS,  kl_f_weight=0.1)\n",
        "  elif TASK == \"regression\":\n",
        "    if UCI_DATASET==\"bike-sharing\":\n",
        "      posterior_net = train_adaptive_BMA(\"regression\", posterior_net, \"categorical\", optimizer, x_train, y_train, fitted_models, batch_size=BATCH_SIZE, epochs=10, kl_f_weight=0.8,\n",
        "                                      X_train_feat=Xtr_gate, y_min=ymin, y_max=ymax)\n",
        "    elif UCI_DATASET==\"california-housing\":\n",
        "      posterior_net = train_adaptive_BMA(\"regression\", posterior_net, \"categorical\", optimizer, x_train, y_train, fitted_models, batch_size=BATCH_SIZE, epochs=10, kl_f_weight=3.0,\n",
        "                                      X_train_feat=Xtr_gate, y_min=ymin, y_max=ymax)\n",
        "    elif UCI_DATASET==\"communities\":\n",
        "      posterior_net = train_adaptive_BMA(\"regression\", posterior_net, \"categorical\", optimizer, x_train, y_train, fitted_models, batch_size=BATCH_SIZE, epochs=10, kl_f_weight=5.0,\n",
        "                                      X_train_feat=Xtr_gate, y_min=ymin, y_max=ymax)\n",
        "\n",
        "iabma_end = time.time()\n",
        "iabma_time = iabma_end - iabma_start\n",
        "print(f\"Execution time: {iabma_time:.4f} seconds\")"
      ],
      "metadata": {
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "id": "Z5UQ5j4XvwOx",
        "outputId": "89358fe9-9c19-4b7a-be36-91f861775adf"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "Epoch 1/10 — ELBO loss = -1.3165, -E[ll] = 1.8153, KL = 2.4937\n",
            "Epoch 2/10 — ELBO loss = -1.3395, -E[ll] = 1.8163, KL = 2.3838\n",
            "Epoch 3/10 — ELBO loss = -1.3410, -E[ll] = 1.8169, KL = 2.3796\n",
            "Epoch 4/10 — ELBO loss = -1.3426, -E[ll] = 1.8175, KL = 2.3748\n",
            "Epoch 5/10 — ELBO loss = -1.3438, -E[ll] = 1.8179, KL = 2.3704\n",
            "Epoch 6/10 — ELBO loss = -1.3448, -E[ll] = 1.8179, KL = 2.3656\n",
            "Epoch 7/10 — ELBO loss = -1.3457, -E[ll] = 1.8180, KL = 2.3613\n",
            "Epoch 8/10 — ELBO loss = -1.3463, -E[ll] = 1.8179, KL = 2.3576\n",
            "Epoch 9/10 — ELBO loss = -1.3469, -E[ll] = 1.8176, KL = 2.3532\n",
            "Epoch 10/10 — ELBO loss = -1.3477, -E[ll] = 1.8178, KL = 2.3507\n",
            "Execution time: 411.2164 seconds\n"
          ]
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "def adapt_features(X, fitted_models, task=\"binary\"):\n",
        "    try:\n",
        "        return np.asarray(X, dtype=np.float32)\n",
        "    except Exception:\n",
        "        if task == \"binary\":\n",
        "            bp = np.stack([m.predict_proba(X) for m in fitted_models.values()], axis=1).astype(np.float32)\n",
        "        else:\n",
        "            bp = np.stack([np.asarray(m.predict(X)).ravel() for m in fitted_models.values()], axis=1).astype(np.float32)\n",
        "        return bp.reshape(bp.shape[0], -1)\n",
        "\n",
        "if EXPERIMENT in [\"synthetic_binary\"]:\n",
        "  logits_f_test = posterior_net.predict(x_test)\n",
        "  q_f_test = tf.nn.softmax(logits_f_test, axis=1).numpy()\n",
        "  iabma_pred = (q_f_test * pred_test).sum(axis=1)\n",
        "\n",
        "elif EXPERIMENT == \"cancer\":\n",
        "  base_pred_test = np.column_stack([\n",
        "    fitted_models_[\"ridge\"].predict(x_test),\n",
        "    fitted_models_[\"xgb\"].predict(x_test),\n",
        "    fitted_models_[\"hgb\"].predict(x_test),\n",
        "    mlp_predict(fitted_models_, x_test)]).astype(np.float32)\n",
        "\n",
        "  logits_f_test = posterior_net.predict(Xte_gate)\n",
        "  iabma_w = tf.nn.softmax(logits_f_test, axis=1).numpy()\n",
        "\n",
        "  iabma_pred = np.sum(iabma_w * base_pred_test, axis=1)\n",
        "\n",
        "elif EXPERIMENT == \"fraud\":\n",
        "\n",
        "  Xb_feat = adapt_features(x_test_b, fitted_models, task=\"binary\")\n",
        "  logits_f_test = posterior_net.predict(Xb_feat)\n",
        "  q_f_test = tf.nn.softmax(logits_f_test, axis=1).numpy()\n",
        "  pred_test_b = np.stack([m.predict_proba(x_test_b) for m in fitted_models.values()], axis=1).astype(np.float32)\n",
        "  iabma_pred = (q_f_test[:, :, None] * pred_test_b).sum(axis=1)\n",
        "\n",
        "elif EXPERIMENT == \"UCI\":\n",
        "  if TASK == \"classification\":\n",
        "    x_feat = adapt_features(x_test, fitted_models, task=\"binary\")\n",
        "    logits_f_test = posterior_net.predict(x_feat)\n",
        "    q_f_test = tf.nn.softmax(logits_f_test, axis=1).numpy()\n",
        "    iabma_pred = np.sum(q_f_test[:, :, None] * pred_test, axis=1)\n",
        "  elif TASK == \"regression\":\n",
        "    x_feat = adapt_features(Xte_gate, fitted_models, task=\"regression\")\n",
        "    logits_f_test = posterior_net.predict(x_feat)\n",
        "    q_f_test = tf.nn.softmax(logits_f_test, axis=1).numpy()\n",
        "    iabma_pred = (q_f_test * pred_test).sum(axis=1)"
      ],
      "metadata": {
        "id": "kbrO1hUqxCKV",
        "colab": {
          "base_uri": "https://localhost:8080/"
        },
        "outputId": "63d098a1-3eb3-4b1c-a575-be33e20695b2"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "stream",
          "name": "stdout",
          "text": [
            "\u001b[1m137/137\u001b[0m \u001b[32m━━━━━━━━━━━━━━━━━━━━\u001b[0m\u001b[37m\u001b[0m \u001b[1m0s\u001b[0m 2ms/step\n"
          ]
        }
      ]
    },
    {
      "cell_type": "markdown",
      "source": [
        "### Evaluation"
      ],
      "metadata": {
        "id": "Ue21zqGq-DDx"
      }
    },
    {
      "cell_type": "code",
      "source": [
        "methods = {\n",
        "    \"Best-single\": best_model,\n",
        "    \"Uniform Avg.\": uniform_avg,\n",
        "    \"Freq Avg.\": freq_avg,\n",
        "    \"BMA\": bma,\n",
        "\n",
        "    \"MoE\": moe_pred,\n",
        "    \"DLA\": dla_pred,\n",
        "    \"SMC\": smc_pred,\n",
        "    \"BHS\": bhs_pred,\n",
        "\n",
        "    \"IABMA\": iabma_pred,\n",
        "}\n"
      ],
      "metadata": {
        "id": "bihyL1vb-GPu"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "def get_p1(probs):\n",
        "    p = np.asarray(probs)\n",
        "    if p.ndim == 1:\n",
        "        return p\n",
        "    if p.ndim == 2 and p.shape[1] == 1:\n",
        "        return p[:, 0]\n",
        "    if p.ndim == 2 and p.shape[1] == 2:\n",
        "        return p[:, 1]\n",
        "    raise ValueError(f\"Unexpected probs shape {p.shape}\")\n",
        "\n",
        "def to_two_cols(p1):\n",
        "    p1 = np.asarray(p1).ravel()\n",
        "    return np.column_stack([1.0 - p1, p1])"
      ],
      "metadata": {
        "id": "gKS0lPtt7pRf"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "results = {}\n",
        "\n",
        "if EXPERIMENT == \"synthetic_binary\":\n",
        "  for name, probs in methods.items():\n",
        "      res = {}\n",
        "      p1 = get_p1(probs)\n",
        "      acc = accuracy_score(y_test.ravel(), p1>0.5)\n",
        "      ece = compute_ece(y_test.ravel(), p1)\n",
        "      bri = brier_score_loss(y_test.ravel(), p1)\n",
        "      res['Accuracy'] = acc\n",
        "      res['ECE'] = ece\n",
        "      res['Brier'] = bri\n",
        "      results[name] = res\n",
        "\n",
        "  res_df = pd.DataFrame(results).T\n",
        "  res_df = res_df.round(3)\n",
        "\n",
        "\n",
        "elif EXPERIMENT == \"cancer\":\n",
        "\n",
        "  for name, yhat in methods.items():\n",
        "      res = {}\n",
        "      y_true = np.asarray(y_test).ravel()\n",
        "      y_pred = np.asarray(yhat).ravel()\n",
        "      r2   = r2_score(y_true, y_pred)\n",
        "      rmse = mean_squared_error(y_true, y_pred)\n",
        "      mae  = mean_absolute_error(y_true, y_pred)\n",
        "      res['R2']   = r2\n",
        "      res['RMSE'] = rmse\n",
        "      res['MAE']  = mae\n",
        "      results[name] = res\n",
        "\n",
        "  res_df = pd.DataFrame(results).T\n",
        "  res_df = res_df.round(3)\n",
        "\n",
        "\n",
        "elif EXPERIMENT == \"fraud\":\n",
        "  for name, probs in methods.items():\n",
        "      res = {}\n",
        "      p1 = probs[:,1]\n",
        "      if len(p1) == len(y_test):\n",
        "        y = np.asarray(y_test)\n",
        "      elif 'y_test_b' in globals() and y_test_b is not None and len(p1) == len(y_test_b):\n",
        "        y = np.asarray(y_test_b)\n",
        "      acc = accuracy_score(y, p1>0.5)\n",
        "      ece = compute_ece(y, p1)\n",
        "      bri = brier_score_loss(y, p1)\n",
        "      res['Accuracy'] = acc\n",
        "      res['ECE'] = ece\n",
        "      res['Brier'] = bri\n",
        "      results[name] = res\n",
        "\n",
        "  res_df = pd.DataFrame(results).T\n",
        "  res_df = res_df.round(3)\n",
        "\n",
        "\n",
        "elif EXPERIMENT == \"UCI\":\n",
        "\n",
        "  if TASK == \"classification\":\n",
        "    for name, probs in methods.items():\n",
        "        res = {}\n",
        "        p1 = probs[:,1]\n",
        "        p1 = np.nan_to_num(np.clip(p1, 0, 1), nan=0.0)\n",
        "        acc = accuracy_score(y_test, p1>0.5)\n",
        "        ece = compute_ece(y_test, p1)\n",
        "        bri = brier_score_loss(y_test, p1)\n",
        "        res['Accuracy'] = acc\n",
        "        res['ECE'] = ece\n",
        "        res['Brier'] = bri\n",
        "        results[name] = res\n",
        "\n",
        "    res_df = pd.DataFrame(results).T\n",
        "    res_df = res_df.round(3)\n",
        "\n",
        "  if TASK == \"regression\":\n",
        "    for name, yhat in methods.items():\n",
        "      res = {}\n",
        "      y_true = np.asarray(y_test).ravel()\n",
        "      y_pred = np.asarray(yhat).ravel()\n",
        "      r2   = r2_score(y_true, y_pred)\n",
        "      rmse = mean_squared_error(y_true, y_pred)\n",
        "      mae  = mean_absolute_error(y_true, y_pred)\n",
        "      res['R2']   = r2\n",
        "      res['RMSE'] = rmse\n",
        "      res['MAE']  = mae\n",
        "      results[name] = res\n",
        "\n",
        "    res_df = pd.DataFrame(results).T\n",
        "    res_df = res_df.round(3)"
      ],
      "metadata": {
        "id": "GiQusMAJBRUj"
      },
      "execution_count": null,
      "outputs": []
    },
    {
      "cell_type": "code",
      "source": [
        "res_df"
      ],
      "metadata": {
        "id": "cvaBFmNoFR47",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 332
        },
        "outputId": "59a97136-5002-49f0-ed4e-71ba1bf523bb"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "execute_result",
          "data": {
            "text/plain": [
              "              Accuracy    ECE  Brier\n",
              "Best-single      0.690  0.299  0.278\n",
              "Uniform Avg.     0.691  0.270  0.222\n",
              "Freq Avg.        0.690  0.271  0.223\n",
              "BMA              0.688  0.273  0.225\n",
              "MoE              0.730  0.262  0.235\n",
              "DLA              0.706  0.265  0.222\n",
              "SMC              0.656  0.305  0.281\n",
              "BHS              0.691  0.270  0.222\n",
              "IABMA            0.745  0.203  0.178"
            ],
            "text/html": [
              "\n",
              "  <div id=\"df-c8db6202-d31e-486c-8a86-9954abcce253\" class=\"colab-df-container\">\n",
              "    <div>\n",
              "<style scoped>\n",
              "    .dataframe tbody tr th:only-of-type {\n",
              "        vertical-align: middle;\n",
              "    }\n",
              "\n",
              "    .dataframe tbody tr th {\n",
              "        vertical-align: top;\n",
              "    }\n",
              "\n",
              "    .dataframe thead th {\n",
              "        text-align: right;\n",
              "    }\n",
              "</style>\n",
              "<table border=\"1\" class=\"dataframe\">\n",
              "  <thead>\n",
              "    <tr style=\"text-align: right;\">\n",
              "      <th></th>\n",
              "      <th>Accuracy</th>\n",
              "      <th>ECE</th>\n",
              "      <th>Brier</th>\n",
              "    </tr>\n",
              "  </thead>\n",
              "  <tbody>\n",
              "    <tr>\n",
              "      <th>Best-single</th>\n",
              "      <td>0.690</td>\n",
              "      <td>0.299</td>\n",
              "      <td>0.278</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>Uniform Avg.</th>\n",
              "      <td>0.691</td>\n",
              "      <td>0.270</td>\n",
              "      <td>0.222</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>Freq Avg.</th>\n",
              "      <td>0.690</td>\n",
              "      <td>0.271</td>\n",
              "      <td>0.223</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>BMA</th>\n",
              "      <td>0.688</td>\n",
              "      <td>0.273</td>\n",
              "      <td>0.225</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>MoE</th>\n",
              "      <td>0.730</td>\n",
              "      <td>0.262</td>\n",
              "      <td>0.235</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>DLA</th>\n",
              "      <td>0.706</td>\n",
              "      <td>0.265</td>\n",
              "      <td>0.222</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>SMC</th>\n",
              "      <td>0.656</td>\n",
              "      <td>0.305</td>\n",
              "      <td>0.281</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>BHS</th>\n",
              "      <td>0.691</td>\n",
              "      <td>0.270</td>\n",
              "      <td>0.222</td>\n",
              "    </tr>\n",
              "    <tr>\n",
              "      <th>IABMA</th>\n",
              "      <td>0.745</td>\n",
              "      <td>0.203</td>\n",
              "      <td>0.178</td>\n",
              "    </tr>\n",
              "  </tbody>\n",
              "</table>\n",
              "</div>\n",
              "    <div class=\"colab-df-buttons\">\n",
              "\n",
              "  <div class=\"colab-df-container\">\n",
              "    <button class=\"colab-df-convert\" onclick=\"convertToInteractive('df-c8db6202-d31e-486c-8a86-9954abcce253')\"\n",
              "            title=\"Convert this dataframe to an interactive table.\"\n",
              "            style=\"display:none;\">\n",
              "\n",
              "  <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\" viewBox=\"0 -960 960 960\">\n",
              "    <path d=\"M120-120v-720h720v720H120Zm60-500h600v-160H180v160Zm220 220h160v-160H400v160Zm0 220h160v-160H400v160ZM180-400h160v-160H180v160Zm440 0h160v-160H620v160ZM180-180h160v-160H180v160Zm440 0h160v-160H620v160Z\"/>\n",
              "  </svg>\n",
              "    </button>\n",
              "\n",
              "  <style>\n",
              "    .colab-df-container {\n",
              "      display:flex;\n",
              "      gap: 12px;\n",
              "    }\n",
              "\n",
              "    .colab-df-convert {\n",
              "      background-color: #E8F0FE;\n",
              "      border: none;\n",
              "      border-radius: 50%;\n",
              "      cursor: pointer;\n",
              "      display: none;\n",
              "      fill: #1967D2;\n",
              "      height: 32px;\n",
              "      padding: 0 0 0 0;\n",
              "      width: 32px;\n",
              "    }\n",
              "\n",
              "    .colab-df-convert:hover {\n",
              "      background-color: #E2EBFA;\n",
              "      box-shadow: 0px 1px 2px rgba(60, 64, 67, 0.3), 0px 1px 3px 1px rgba(60, 64, 67, 0.15);\n",
              "      fill: #174EA6;\n",
              "    }\n",
              "\n",
              "    .colab-df-buttons div {\n",
              "      margin-bottom: 4px;\n",
              "    }\n",
              "\n",
              "    [theme=dark] .colab-df-convert {\n",
              "      background-color: #3B4455;\n",
              "      fill: #D2E3FC;\n",
              "    }\n",
              "\n",
              "    [theme=dark] .colab-df-convert:hover {\n",
              "      background-color: #434B5C;\n",
              "      box-shadow: 0px 1px 3px 1px rgba(0, 0, 0, 0.15);\n",
              "      filter: drop-shadow(0px 1px 2px rgba(0, 0, 0, 0.3));\n",
              "      fill: #FFFFFF;\n",
              "    }\n",
              "  </style>\n",
              "\n",
              "    <script>\n",
              "      const buttonEl =\n",
              "        document.querySelector('#df-c8db6202-d31e-486c-8a86-9954abcce253 button.colab-df-convert');\n",
              "      buttonEl.style.display =\n",
              "        google.colab.kernel.accessAllowed ? 'block' : 'none';\n",
              "\n",
              "      async function convertToInteractive(key) {\n",
              "        const element = document.querySelector('#df-c8db6202-d31e-486c-8a86-9954abcce253');\n",
              "        const dataTable =\n",
              "          await google.colab.kernel.invokeFunction('convertToInteractive',\n",
              "                                                    [key], {});\n",
              "        if (!dataTable) return;\n",
              "\n",
              "        const docLinkHtml = 'Like what you see? Visit the ' +\n",
              "          '<a target=\"_blank\" href=https://colab.research.google.com/notebooks/data_table.ipynb>data table notebook</a>'\n",
              "          + ' to learn more about interactive tables.';\n",
              "        element.innerHTML = '';\n",
              "        dataTable['output_type'] = 'display_data';\n",
              "        await google.colab.output.renderOutput(dataTable, element);\n",
              "        const docLink = document.createElement('div');\n",
              "        docLink.innerHTML = docLinkHtml;\n",
              "        element.appendChild(docLink);\n",
              "      }\n",
              "    </script>\n",
              "  </div>\n",
              "\n",
              "\n",
              "    <div id=\"df-c52c3184-17ee-476c-8358-ed05c89a6eaa\">\n",
              "      <button class=\"colab-df-quickchart\" onclick=\"quickchart('df-c52c3184-17ee-476c-8358-ed05c89a6eaa')\"\n",
              "                title=\"Suggest charts\"\n",
              "                style=\"display:none;\">\n",
              "\n",
              "<svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\"viewBox=\"0 0 24 24\"\n",
              "     width=\"24px\">\n",
              "    <g>\n",
              "        <path d=\"M19 3H5c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h14c1.1 0 2-.9 2-2V5c0-1.1-.9-2-2-2zM9 17H7v-7h2v7zm4 0h-2V7h2v10zm4 0h-2v-4h2v4z\"/>\n",
              "    </g>\n",
              "</svg>\n",
              "      </button>\n",
              "\n",
              "<style>\n",
              "  .colab-df-quickchart {\n",
              "      --bg-color: #E8F0FE;\n",
              "      --fill-color: #1967D2;\n",
              "      --hover-bg-color: #E2EBFA;\n",
              "      --hover-fill-color: #174EA6;\n",
              "      --disabled-fill-color: #AAA;\n",
              "      --disabled-bg-color: #DDD;\n",
              "  }\n",
              "\n",
              "  [theme=dark] .colab-df-quickchart {\n",
              "      --bg-color: #3B4455;\n",
              "      --fill-color: #D2E3FC;\n",
              "      --hover-bg-color: #434B5C;\n",
              "      --hover-fill-color: #FFFFFF;\n",
              "      --disabled-bg-color: #3B4455;\n",
              "      --disabled-fill-color: #666;\n",
              "  }\n",
              "\n",
              "  .colab-df-quickchart {\n",
              "    background-color: var(--bg-color);\n",
              "    border: none;\n",
              "    border-radius: 50%;\n",
              "    cursor: pointer;\n",
              "    display: none;\n",
              "    fill: var(--fill-color);\n",
              "    height: 32px;\n",
              "    padding: 0;\n",
              "    width: 32px;\n",
              "  }\n",
              "\n",
              "  .colab-df-quickchart:hover {\n",
              "    background-color: var(--hover-bg-color);\n",
              "    box-shadow: 0 1px 2px rgba(60, 64, 67, 0.3), 0 1px 3px 1px rgba(60, 64, 67, 0.15);\n",
              "    fill: var(--button-hover-fill-color);\n",
              "  }\n",
              "\n",
              "  .colab-df-quickchart-complete:disabled,\n",
              "  .colab-df-quickchart-complete:disabled:hover {\n",
              "    background-color: var(--disabled-bg-color);\n",
              "    fill: var(--disabled-fill-color);\n",
              "    box-shadow: none;\n",
              "  }\n",
              "\n",
              "  .colab-df-spinner {\n",
              "    border: 2px solid var(--fill-color);\n",
              "    border-color: transparent;\n",
              "    border-bottom-color: var(--fill-color);\n",
              "    animation:\n",
              "      spin 1s steps(1) infinite;\n",
              "  }\n",
              "\n",
              "  @keyframes spin {\n",
              "    0% {\n",
              "      border-color: transparent;\n",
              "      border-bottom-color: var(--fill-color);\n",
              "      border-left-color: var(--fill-color);\n",
              "    }\n",
              "    20% {\n",
              "      border-color: transparent;\n",
              "      border-left-color: var(--fill-color);\n",
              "      border-top-color: var(--fill-color);\n",
              "    }\n",
              "    30% {\n",
              "      border-color: transparent;\n",
              "      border-left-color: var(--fill-color);\n",
              "      border-top-color: var(--fill-color);\n",
              "      border-right-color: var(--fill-color);\n",
              "    }\n",
              "    40% {\n",
              "      border-color: transparent;\n",
              "      border-right-color: var(--fill-color);\n",
              "      border-top-color: var(--fill-color);\n",
              "    }\n",
              "    60% {\n",
              "      border-color: transparent;\n",
              "      border-right-color: var(--fill-color);\n",
              "    }\n",
              "    80% {\n",
              "      border-color: transparent;\n",
              "      border-right-color: var(--fill-color);\n",
              "      border-bottom-color: var(--fill-color);\n",
              "    }\n",
              "    90% {\n",
              "      border-color: transparent;\n",
              "      border-bottom-color: var(--fill-color);\n",
              "    }\n",
              "  }\n",
              "</style>\n",
              "\n",
              "      <script>\n",
              "        async function quickchart(key) {\n",
              "          const quickchartButtonEl =\n",
              "            document.querySelector('#' + key + ' button');\n",
              "          quickchartButtonEl.disabled = true;  // To prevent multiple clicks.\n",
              "          quickchartButtonEl.classList.add('colab-df-spinner');\n",
              "          try {\n",
              "            const charts = await google.colab.kernel.invokeFunction(\n",
              "                'suggestCharts', [key], {});\n",
              "          } catch (error) {\n",
              "            console.error('Error during call to suggestCharts:', error);\n",
              "          }\n",
              "          quickchartButtonEl.classList.remove('colab-df-spinner');\n",
              "          quickchartButtonEl.classList.add('colab-df-quickchart-complete');\n",
              "        }\n",
              "        (() => {\n",
              "          let quickchartButtonEl =\n",
              "            document.querySelector('#df-c52c3184-17ee-476c-8358-ed05c89a6eaa button');\n",
              "          quickchartButtonEl.style.display =\n",
              "            google.colab.kernel.accessAllowed ? 'block' : 'none';\n",
              "        })();\n",
              "      </script>\n",
              "    </div>\n",
              "\n",
              "  <div id=\"id_f95862a5-588a-46f3-8783-49fda78bc9a7\">\n",
              "    <style>\n",
              "      .colab-df-generate {\n",
              "        background-color: #E8F0FE;\n",
              "        border: none;\n",
              "        border-radius: 50%;\n",
              "        cursor: pointer;\n",
              "        display: none;\n",
              "        fill: #1967D2;\n",
              "        height: 32px;\n",
              "        padding: 0 0 0 0;\n",
              "        width: 32px;\n",
              "      }\n",
              "\n",
              "      .colab-df-generate:hover {\n",
              "        background-color: #E2EBFA;\n",
              "        box-shadow: 0px 1px 2px rgba(60, 64, 67, 0.3), 0px 1px 3px 1px rgba(60, 64, 67, 0.15);\n",
              "        fill: #174EA6;\n",
              "      }\n",
              "\n",
              "      [theme=dark] .colab-df-generate {\n",
              "        background-color: #3B4455;\n",
              "        fill: #D2E3FC;\n",
              "      }\n",
              "\n",
              "      [theme=dark] .colab-df-generate:hover {\n",
              "        background-color: #434B5C;\n",
              "        box-shadow: 0px 1px 3px 1px rgba(0, 0, 0, 0.15);\n",
              "        filter: drop-shadow(0px 1px 2px rgba(0, 0, 0, 0.3));\n",
              "        fill: #FFFFFF;\n",
              "      }\n",
              "    </style>\n",
              "    <button class=\"colab-df-generate\" onclick=\"generateWithVariable('res_df')\"\n",
              "            title=\"Generate code using this dataframe.\"\n",
              "            style=\"display:none;\">\n",
              "\n",
              "  <svg xmlns=\"http://www.w3.org/2000/svg\" height=\"24px\"viewBox=\"0 0 24 24\"\n",
              "       width=\"24px\">\n",
              "    <path d=\"M7,19H8.4L18.45,9,17,7.55,7,17.6ZM5,21V16.75L18.45,3.32a2,2,0,0,1,2.83,0l1.4,1.43a1.91,1.91,0,0,1,.58,1.4,1.91,1.91,0,0,1-.58,1.4L9.25,21ZM18.45,9,17,7.55Zm-12,3A5.31,5.31,0,0,0,4.9,8.1,5.31,5.31,0,0,0,1,6.5,5.31,5.31,0,0,0,4.9,4.9,5.31,5.31,0,0,0,6.5,1,5.31,5.31,0,0,0,8.1,4.9,5.31,5.31,0,0,0,12,6.5,5.46,5.46,0,0,0,6.5,12Z\"/>\n",
              "  </svg>\n",
              "    </button>\n",
              "    <script>\n",
              "      (() => {\n",
              "      const buttonEl =\n",
              "        document.querySelector('#id_f95862a5-588a-46f3-8783-49fda78bc9a7 button.colab-df-generate');\n",
              "      buttonEl.style.display =\n",
              "        google.colab.kernel.accessAllowed ? 'block' : 'none';\n",
              "\n",
              "      buttonEl.onclick = () => {\n",
              "        google.colab.notebook.generateWithVariable('res_df');\n",
              "      }\n",
              "      })();\n",
              "    </script>\n",
              "  </div>\n",
              "\n",
              "    </div>\n",
              "  </div>\n"
            ],
            "application/vnd.google.colaboratory.intrinsic+json": {
              "type": "dataframe",
              "variable_name": "res_df",
              "summary": "{\n  \"name\": \"res_df\",\n  \"rows\": 9,\n  \"fields\": [\n    {\n      \"column\": \"Accuracy\",\n      \"properties\": {\n        \"dtype\": \"number\",\n        \"std\": 0.025942778913944007,\n        \"min\": 0.656,\n        \"max\": 0.745,\n        \"num_unique_values\": 7,\n        \"samples\": [\n          0.69,\n          0.691,\n          0.656\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"ECE\",\n      \"properties\": {\n        \"dtype\": \"number\",\n        \"std\": 0.028805381441668148,\n        \"min\": 0.203,\n        \"max\": 0.305,\n        \"num_unique_values\": 8,\n        \"samples\": [\n          0.27,\n          0.265,\n          0.299\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    },\n    {\n      \"column\": \"Brier\",\n      \"properties\": {\n        \"dtype\": \"number\",\n        \"std\": 0.0313678887470044,\n        \"min\": 0.178,\n        \"max\": 0.281,\n        \"num_unique_values\": 7,\n        \"samples\": [\n          0.278,\n          0.222,\n          0.281\n        ],\n        \"semantic_type\": \"\",\n        \"description\": \"\"\n      }\n    }\n  ]\n}"
            }
          },
          "metadata": {},
          "execution_count": 36
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "# Groups\n",
        "baselines   = [\"Best-single\",\"Uniform Avg.\",\"Freq Avg.\",\"BMA\"]\n",
        "competitors = [\"MoE\",\"DLA\",\"SMC\", \"BHS\"]\n",
        "ours        = [\"IABMA\"]\n",
        "\n",
        "# Colors\n",
        "baseline_colors   = ['#a63603', '#e6550d', '#ff7f00', '#fdb863']\n",
        "competitor_colors = ['#08306b', '#08519c', '#3182bd', '#6baed6']\n",
        "ours_colors       = ['#00441b']\n",
        "\n",
        "methods_l   = baselines + competitors + ours\n",
        "color_map = {m: c for m, c in zip(methods_l, baseline_colors + competitor_colors + ours_colors)}\n",
        "\n",
        "if EXPERIMENT in [\"synthetic_binary\", \"fraud\"]:\n",
        "  metrics = ['Accuracy', 'ECE']\n",
        "  metric_arrows = {\"Accuracy\": \"↑\",  \"ECE\": \"↓\"}\n",
        "\n",
        "elif EXPERIMENT == \"cancer\":\n",
        "  metrics = ['R2', 'RMSE']\n",
        "  metric_arrows = {\"R2\": \"↑\",  \"RMSE\": \"↓\"}\n",
        "\n",
        "elif EXPERIMENT == \"UCI\":\n",
        "  if TASK == \"classification\":\n",
        "    metrics = ['Accuracy', 'ECE']\n",
        "    metric_arrows = {\"Accuracy\": \"↑\",  \"ECE\": \"↓\"}\n",
        "  elif TASK == \"regression\":\n",
        "    metrics = ['R2', 'RMSE']\n",
        "    metric_arrows = {\"R2\": \"↑\",  \"RMSE\": \"↓\"}\n",
        "\n",
        "values = {metric: [results[m][metric] for m in methods] for metric in metrics}\n",
        "\n",
        "fig, axes = plt.subplots(1, 2, figsize=(9, 4), sharex=True)\n",
        "\n",
        "for i, metric in enumerate(metrics):\n",
        "    for j, m in enumerate(methods_l):\n",
        "        axes[i].set_axisbelow(True)\n",
        "        axes[i].grid(axis=\"y\", linewidth=0.5, color='gainsboro')\n",
        "        if m == \"IABMA\":\n",
        "            axes[i].bar(\n",
        "                j, values[metric][j],\n",
        "                color=color_map[m],\n",
        "                hatch='//',\n",
        "                edgecolor='white',\n",
        "                linewidth=0.0,\n",
        "                alpha=0.9\n",
        "            )\n",
        "        else:\n",
        "            axes[i].bar(j, values[metric][j], color=color_map[m], alpha=0.9)\n",
        "    axes[i].set_title(f\"{metric} ({metric_arrows[metric]})\")\n",
        "    axes[i].set_xticks(np.arange(len(methods_l)))\n",
        "    axes[i].set_xticklabels(methods_l, rotation=45, ha='right')\n",
        "    axes[i].set_ylabel(metric)\n",
        "\n",
        "legend_elements = [\n",
        "    Patch(facecolor=baseline_colors[1], label='Baselines'),\n",
        "    Patch(facecolor=competitor_colors[1], label='Competitors'),\n",
        "    Patch(facecolor=ours_colors[0], label='Ours', hatch='//', edgecolor='white')\n",
        "]\n",
        "\n",
        "fig.legend(\n",
        "    handles=legend_elements,\n",
        "    loc='upper center',\n",
        "    ncol=3,\n",
        "    bbox_to_anchor=(0.5, 1.1)\n",
        ")\n",
        "\n",
        "\n",
        "plt.tight_layout()"
      ],
      "metadata": {
        "id": "ddkjR1aWBVei",
        "colab": {
          "base_uri": "https://localhost:8080/",
          "height": 455
        },
        "outputId": "135c94e3-dd40-4d94-ceee-353f8487b967"
      },
      "execution_count": null,
      "outputs": [
        {
          "output_type": "display_data",
          "data": {
            "text/plain": [
              "<Figure size 900x400 with 2 Axes>"
            ],
            "image/png": "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\n"
          },
          "metadata": {}
        }
      ]
    },
    {
      "cell_type": "code",
      "source": [
        "if SAVE:\n",
        "  run_times = {\"MoE\": moe_time, \"DLA\": dla_time, \"SMC\": smc_time, \"BHS\": bhs_time, \"IABMA\": iabma_time}\n",
        "  log_run(results, csv_path=\"results_log.csv\", method_times=run_times)"
      ],
      "metadata": {
        "id": "vulmaAutFuNT"
      },
      "execution_count": null,
      "outputs": []
    }
  ]
}