{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Setup"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "%load_ext autoreload\n",
    "%load_ext tensorboard\n",
    "%matplotlib inline"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import matplotlib\n",
    "import numpy as np\n",
    "import os\n",
    "import random\n",
    "import tensorflow as tf\n",
    "import yaml\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "from matplotlib import rc\n",
    "from matplotlib import cm\n",
    "import seaborn as sns\n",
    "from importlib import reload\n",
    "from pathlib import Path\n",
    "import sklearn\n",
    "from tensorflow.keras.models import load_model\n",
    "from joblib import dump, load\n",
    "import pandas as pd\n",
    "import cub_experiments as cub\n",
    "import models\n",
    "from CUB200.cub_loader import load_data, find_class_imbalance\n",
    "import torch\n",
    "import pytorch_lightning as pl\n",
    "from pytorch_lightning import seed_everything"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "################################################################################\n",
    "## Global Variables Defining Experiment Flow\n",
    "################################################################################\n",
    "\n",
    "GPU = 1\n",
    "NUM_WORKERS = 10\n",
    "LATEX_SYMBOL = \"$\"\n",
    "RESULTS_DIR = \"./results/\"\n",
    "CUB_RESULTS_DIR = os.path.join(\n",
    "    RESULTS_DIR,\n",
    "    \"cub\",\n",
    ")\n",
    "\n",
    "CUB_ABLATION_RESULTS_DIR = os.path.join(\n",
    "    \"results/cub_randint_ablation\",\n",
    ")\n",
    "\n",
    "CUB_EMB_ABLATION_RESULTS_DIR = \"results/cub_emb_size_ablation/\"\n",
    "\n",
    "rc('text', usetex=(LATEX_SYMBOL == \"$\"))\n",
    "plt.style.use('seaborn-whitegrid')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load Model Configs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "from collections import defaultdict\n",
    "cub_configs = defaultdict(dict)\n",
    "for file in os.listdir(CUB_RESULTS_DIR):\n",
    "    if '_experiment_config.joblib' in file:\n",
    "        config = load(os.path.join(CUB_RESULTS_DIR, file))\n",
    "        fold = int(file[file.find(\"_fold_\") + len(\"_fold_\"):file.find(\"_experiment_config\")]) - 1\n",
    "        if \"Sequential\" in file:\n",
    "            extra = \"Sequential\"\n",
    "        elif \"Independent\" in file:\n",
    "            extra = \"Independent\"\n",
    "        else:\n",
    "            extra = \"\"\n",
    "            \n",
    "        model_name = f\"{extra}{config['architecture']}{config.get('extra_name', '')}\"\n",
    "        cub_configs[str(fold)][model_name] = config\n",
    "\n",
    "print(\"CUB Model names:\")\n",
    "for model_name, _ in cub_configs['0'].items(): \n",
    "    print(\"\\t\", model_name)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# See Selected Attributes for CUB"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "selected_attributes = [\n",
    "    1,\n",
    "    4,\n",
    "    6,\n",
    "    7,\n",
    "    10,\n",
    "    14,\n",
    "    15,\n",
    "    20,\n",
    "    21,\n",
    "    23,\n",
    "    25,\n",
    "    29,\n",
    "    30,\n",
    "    35,\n",
    "    36,\n",
    "    38,\n",
    "    40,\n",
    "    44,\n",
    "    45,\n",
    "    50,\n",
    "    51,\n",
    "    53,\n",
    "    54,\n",
    "    56,\n",
    "    57,\n",
    "    59,\n",
    "    63,\n",
    "    64,\n",
    "    69,\n",
    "    70,\n",
    "    72,\n",
    "    75,\n",
    "    80,\n",
    "    84,\n",
    "    90,\n",
    "    91,\n",
    "    93,\n",
    "    99,\n",
    "    101,\n",
    "    106,\n",
    "    110,\n",
    "    111,\n",
    "    116,\n",
    "    117,\n",
    "    119,\n",
    "    125,\n",
    "    126,\n",
    "    131,\n",
    "    132,\n",
    "    134,\n",
    "    145,\n",
    "    149,\n",
    "    151,\n",
    "    152,\n",
    "    153,\n",
    "    157,\n",
    "    158,\n",
    "    163,\n",
    "    164,\n",
    "    168,\n",
    "    172,\n",
    "    178,\n",
    "    179,\n",
    "    181,\n",
    "    183,\n",
    "    187,\n",
    "    188,\n",
    "    193,\n",
    "    194,\n",
    "    196,\n",
    "    198,\n",
    "    202,\n",
    "    203,\n",
    "    208,\n",
    "    209,\n",
    "    211,\n",
    "    212,\n",
    "    213,\n",
    "    218,\n",
    "    220,\n",
    "    221,\n",
    "    225,\n",
    "    235,\n",
    "    236,\n",
    "    238,\n",
    "    239,\n",
    "    240,\n",
    "    242,\n",
    "    243,\n",
    "    244,\n",
    "    249,\n",
    "    253,\n",
    "    254,\n",
    "    259,\n",
    "    260,\n",
    "    262,\n",
    "    268,\n",
    "    274,\n",
    "    277,\n",
    "    283,\n",
    "    289,\n",
    "    292,\n",
    "    293,\n",
    "    294,\n",
    "    298,\n",
    "    299,\n",
    "    304,\n",
    "    305,\n",
    "    308,\n",
    "    309,\n",
    "    310,\n",
    "    311,\n",
    "]"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "attributes = [\n",
    "    \"has_bill_shape::curved_(up_or_down)\",\n",
    "    \"has_bill_shape::dagger\",\n",
    "    \"has_bill_shape::hooked\",\n",
    "    \"has_bill_shape::needle\",\n",
    "    \"has_bill_shape::hooked_seabird\",\n",
    "    \"has_bill_shape::spatulate\",\n",
    "    \"has_bill_shape::all-purpose\",\n",
    "    \"has_bill_shape::cone\",\n",
    "    \"has_bill_shape::specialized\",\n",
    "    \"has_wing_color::blue\",\n",
    "    \"has_wing_color::brown\",\n",
    "    \"has_wing_color::iridescent\",\n",
    "    \"has_wing_color::purple\",\n",
    "    \"has_wing_color::rufous\",\n",
    "    \"has_wing_color::grey\",\n",
    "    \"has_wing_color::yellow\",\n",
    "    \"has_wing_color::olive\",\n",
    "    \"has_wing_color::green\",\n",
    "    \"has_wing_color::pink\",\n",
    "    \"has_wing_color::orange\",\n",
    "    \"has_wing_color::black\",\n",
    "    \"has_wing_color::white\",\n",
    "    \"has_wing_color::red\",\n",
    "    \"has_wing_color::buff\",\n",
    "    \"has_upperparts_color::blue\",\n",
    "    \"has_upperparts_color::brown\",\n",
    "    \"has_upperparts_color::iridescent\",\n",
    "    \"has_upperparts_color::purple\",\n",
    "    \"has_upperparts_color::rufous\",\n",
    "    \"has_upperparts_color::grey\",\n",
    "    \"has_upperparts_color::yellow\",\n",
    "    \"has_upperparts_color::olive\",\n",
    "    \"has_upperparts_color::green\",\n",
    "    \"has_upperparts_color::pink\",\n",
    "    \"has_upperparts_color::orange\",\n",
    "    \"has_upperparts_color::black\",\n",
    "    \"has_upperparts_color::white\",\n",
    "    \"has_upperparts_color::red\",\n",
    "    \"has_upperparts_color::buff\",\n",
    "    \"has_underparts_color::blue\",\n",
    "    \"has_underparts_color::brown\",\n",
    "    \"has_underparts_color::iridescent\",\n",
    "    \"has_underparts_color::purple\",\n",
    "    \"has_underparts_color::rufous\",\n",
    "    \"has_underparts_color::grey\",\n",
    "    \"has_underparts_color::yellow\",\n",
    "    \"has_underparts_color::olive\",\n",
    "    \"has_underparts_color::green\",\n",
    "    \"has_underparts_color::pink\",\n",
    "    \"has_underparts_color::orange\",\n",
    "    \"has_underparts_color::black\",\n",
    "    \"has_underparts_color::white\",\n",
    "    \"has_underparts_color::red\",\n",
    "    \"has_underparts_color::buff\",\n",
    "    \"has_breast_pattern::solid\",\n",
    "    \"has_breast_pattern::spotted\",\n",
    "    \"has_breast_pattern::striped\",\n",
    "    \"has_breast_pattern::multi-colored\",\n",
    "    \"has_back_color::blue\",\n",
    "    \"has_back_color::brown\",\n",
    "    \"has_back_color::iridescent\",\n",
    "    \"has_back_color::purple\",\n",
    "    \"has_back_color::rufous\",\n",
    "    \"has_back_color::grey\",\n",
    "    \"has_back_color::yellow\",\n",
    "    \"has_back_color::olive\",\n",
    "    \"has_back_color::green\",\n",
    "    \"has_back_color::pink\",\n",
    "    \"has_back_color::orange\",\n",
    "    \"has_back_color::black\",\n",
    "    \"has_back_color::white\",\n",
    "    \"has_back_color::red\",\n",
    "    \"has_back_color::buff\",\n",
    "    \"has_tail_shape::forked_tail\",\n",
    "    \"has_tail_shape::rounded_tail\",\n",
    "    \"has_tail_shape::notched_tail\",\n",
    "    \"has_tail_shape::fan-shaped_tail\",\n",
    "    \"has_tail_shape::pointed_tail\",\n",
    "    \"has_tail_shape::squared_tail\",\n",
    "    \"has_upper_tail_color::blue\",\n",
    "    \"has_upper_tail_color::brown\",\n",
    "    \"has_upper_tail_color::iridescent\",\n",
    "    \"has_upper_tail_color::purple\",\n",
    "    \"has_upper_tail_color::rufous\",\n",
    "    \"has_upper_tail_color::grey\",\n",
    "    \"has_upper_tail_color::yellow\",\n",
    "    \"has_upper_tail_color::olive\",\n",
    "    \"has_upper_tail_color::green\",\n",
    "    \"has_upper_tail_color::pink\",\n",
    "    \"has_upper_tail_color::orange\",\n",
    "    \"has_upper_tail_color::black\",\n",
    "    \"has_upper_tail_color::white\",\n",
    "    \"has_upper_tail_color::red\",\n",
    "    \"has_upper_tail_color::buff\",\n",
    "    \"has_head_pattern::spotted\",\n",
    "    \"has_head_pattern::malar\",\n",
    "    \"has_head_pattern::crested\",\n",
    "    \"has_head_pattern::masked\",\n",
    "    \"has_head_pattern::unique_pattern\",\n",
    "    \"has_head_pattern::eyebrow\",\n",
    "    \"has_head_pattern::eyering\",\n",
    "    \"has_head_pattern::plain\",\n",
    "    \"has_head_pattern::eyeline\",\n",
    "    \"has_head_pattern::striped\",\n",
    "    \"has_head_pattern::capped\",\n",
    "    \"has_breast_color::blue\",\n",
    "    \"has_breast_color::brown\",\n",
    "    \"has_breast_color::iridescent\",\n",
    "    \"has_breast_color::purple\",\n",
    "    \"has_breast_color::rufous\",\n",
    "    \"has_breast_color::grey\",\n",
    "    \"has_breast_color::yellow\",\n",
    "    \"has_breast_color::olive\",\n",
    "    \"has_breast_color::green\",\n",
    "    \"has_breast_color::pink\",\n",
    "    \"has_breast_color::orange\",\n",
    "    \"has_breast_color::black\",\n",
    "    \"has_breast_color::white\",\n",
    "    \"has_breast_color::red\",\n",
    "    \"has_breast_color::buff\",\n",
    "    \"has_throat_color::blue\",\n",
    "    \"has_throat_color::brown\",\n",
    "    \"has_throat_color::iridescent\",\n",
    "    \"has_throat_color::purple\",\n",
    "    \"has_throat_color::rufous\",\n",
    "    \"has_throat_color::grey\",\n",
    "    \"has_throat_color::yellow\",\n",
    "    \"has_throat_color::olive\",\n",
    "    \"has_throat_color::green\",\n",
    "    \"has_throat_color::pink\",\n",
    "    \"has_throat_color::orange\",\n",
    "    \"has_throat_color::black\",\n",
    "    \"has_throat_color::white\",\n",
    "    \"has_throat_color::red\",\n",
    "    \"has_throat_color::buff\",\n",
    "    \"has_eye_color::blue\",\n",
    "    \"has_eye_color::brown\",\n",
    "    \"has_eye_color::purple\",\n",
    "    \"has_eye_color::rufous\",\n",
    "    \"has_eye_color::grey\",\n",
    "    \"has_eye_color::yellow\",\n",
    "    \"has_eye_color::olive\",\n",
    "    \"has_eye_color::green\",\n",
    "    \"has_eye_color::pink\",\n",
    "    \"has_eye_color::orange\",\n",
    "    \"has_eye_color::black\",\n",
    "    \"has_eye_color::white\",\n",
    "    \"has_eye_color::red\",\n",
    "    \"has_eye_color::buff\",\n",
    "    \"has_bill_length::about_the_same_as_head\",\n",
    "    \"has_bill_length::longer_than_head\",\n",
    "    \"has_bill_length::shorter_than_head\",\n",
    "    \"has_forehead_color::blue\",\n",
    "    \"has_forehead_color::brown\",\n",
    "    \"has_forehead_color::iridescent\",\n",
    "    \"has_forehead_color::purple\",\n",
    "    \"has_forehead_color::rufous\",\n",
    "    \"has_forehead_color::grey\",\n",
    "    \"has_forehead_color::yellow\",\n",
    "    \"has_forehead_color::olive\",\n",
    "    \"has_forehead_color::green\",\n",
    "    \"has_forehead_color::pink\",\n",
    "    \"has_forehead_color::orange\",\n",
    "    \"has_forehead_color::black\",\n",
    "    \"has_forehead_color::white\",\n",
    "    \"has_forehead_color::red\",\n",
    "    \"has_forehead_color::buff\",\n",
    "    \"has_under_tail_color::blue\",\n",
    "    \"has_under_tail_color::brown\",\n",
    "    \"has_under_tail_color::iridescent\",\n",
    "    \"has_under_tail_color::purple\",\n",
    "    \"has_under_tail_color::rufous\",\n",
    "    \"has_under_tail_color::grey\",\n",
    "    \"has_under_tail_color::yellow\",\n",
    "    \"has_under_tail_color::olive\",\n",
    "    \"has_under_tail_color::green\",\n",
    "    \"has_under_tail_color::pink\",\n",
    "    \"has_under_tail_color::orange\",\n",
    "    \"has_under_tail_color::black\",\n",
    "    \"has_under_tail_color::white\",\n",
    "    \"has_under_tail_color::red\",\n",
    "    \"has_under_tail_color::buff\",\n",
    "    \"has_nape_color::blue\",\n",
    "    \"has_nape_color::brown\",\n",
    "    \"has_nape_color::iridescent\",\n",
    "    \"has_nape_color::purple\",\n",
    "    \"has_nape_color::rufous\",\n",
    "    \"has_nape_color::grey\",\n",
    "    \"has_nape_color::yellow\",\n",
    "    \"has_nape_color::olive\",\n",
    "    \"has_nape_color::green\",\n",
    "    \"has_nape_color::pink\",\n",
    "    \"has_nape_color::orange\",\n",
    "    \"has_nape_color::black\",\n",
    "    \"has_nape_color::white\",\n",
    "    \"has_nape_color::red\",\n",
    "    \"has_nape_color::buff\",\n",
    "    \"has_belly_color::blue\",\n",
    "    \"has_belly_color::brown\",\n",
    "    \"has_belly_color::iridescent\",\n",
    "    \"has_belly_color::purple\",\n",
    "    \"has_belly_color::rufous\",\n",
    "    \"has_belly_color::grey\",\n",
    "    \"has_belly_color::yellow\",\n",
    "    \"has_belly_color::olive\",\n",
    "    \"has_belly_color::green\",\n",
    "    \"has_belly_color::pink\",\n",
    "    \"has_belly_color::orange\",\n",
    "    \"has_belly_color::black\",\n",
    "    \"has_belly_color::white\",\n",
    "    \"has_belly_color::red\",\n",
    "    \"has_belly_color::buff\",\n",
    "    \"has_wing_shape::rounded-wings\",\n",
    "    \"has_wing_shape::pointed-wings\",\n",
    "    \"has_wing_shape::broad-wings\",\n",
    "    \"has_wing_shape::tapered-wings\",\n",
    "    \"has_wing_shape::long-wings\",\n",
    "    \"has_size::large_(16_-_32_in)\",\n",
    "    \"has_size::small_(5_-_9_in)\",\n",
    "    \"has_size::very_large_(32_-_72_in)\",\n",
    "    \"has_size::medium_(9_-_16_in)\",\n",
    "    \"has_size::very_small_(3_-_5_in)\",\n",
    "    \"has_shape::upright-perching_water-like\",\n",
    "    \"has_shape::chicken-like-marsh\",\n",
    "    \"has_shape::long-legged-like\",\n",
    "    \"has_shape::duck-like\",\n",
    "    \"has_shape::owl-like\",\n",
    "    \"has_shape::gull-like\",\n",
    "    \"has_shape::hummingbird-like\",\n",
    "    \"has_shape::pigeon-like\",\n",
    "    \"has_shape::tree-clinging-like\",\n",
    "    \"has_shape::hawk-like\",\n",
    "    \"has_shape::sandpiper-like\",\n",
    "    \"has_shape::upland-ground-like\",\n",
    "    \"has_shape::swallow-like\",\n",
    "    \"has_shape::perching-like\",\n",
    "    \"has_back_pattern::solid\",\n",
    "    \"has_back_pattern::spotted\",\n",
    "    \"has_back_pattern::striped\",\n",
    "    \"has_back_pattern::multi-colored\",\n",
    "    \"has_tail_pattern::solid\",\n",
    "    \"has_tail_pattern::spotted\",\n",
    "    \"has_tail_pattern::striped\",\n",
    "    \"has_tail_pattern::multi-colored\",\n",
    "    \"has_belly_pattern::solid\",\n",
    "    \"has_belly_pattern::spotted\",\n",
    "    \"has_belly_pattern::striped\",\n",
    "    \"has_belly_pattern::multi-colored\",\n",
    "    \"has_primary_color::blue\",\n",
    "    \"has_primary_color::brown\",\n",
    "    \"has_primary_color::iridescent\",\n",
    "    \"has_primary_color::purple\",\n",
    "    \"has_primary_color::rufous\",\n",
    "    \"has_primary_color::grey\",\n",
    "    \"has_primary_color::yellow\",\n",
    "    \"has_primary_color::olive\",\n",
    "    \"has_primary_color::green\",\n",
    "    \"has_primary_color::pink\",\n",
    "    \"has_primary_color::orange\",\n",
    "    \"has_primary_color::black\",\n",
    "    \"has_primary_color::white\",\n",
    "    \"has_primary_color::red\",\n",
    "    \"has_primary_color::buff\",\n",
    "    \"has_leg_color::blue\",\n",
    "    \"has_leg_color::brown\",\n",
    "    \"has_leg_color::iridescent\",\n",
    "    \"has_leg_color::purple\",\n",
    "    \"has_leg_color::rufous\",\n",
    "    \"has_leg_color::grey\",\n",
    "    \"has_leg_color::yellow\",\n",
    "    \"has_leg_color::olive\",\n",
    "    \"has_leg_color::green\",\n",
    "    \"has_leg_color::pink\",\n",
    "    \"has_leg_color::orange\",\n",
    "    \"has_leg_color::black\",\n",
    "    \"has_leg_color::white\",\n",
    "    \"has_leg_color::red\",\n",
    "    \"has_leg_color::buff\",\n",
    "    \"has_bill_color::blue\",\n",
    "    \"has_bill_color::brown\",\n",
    "    \"has_bill_color::iridescent\",\n",
    "    \"has_bill_color::purple\",\n",
    "    \"has_bill_color::rufous\",\n",
    "    \"has_bill_color::grey\",\n",
    "    \"has_bill_color::yellow\",\n",
    "    \"has_bill_color::olive\",\n",
    "    \"has_bill_color::green\",\n",
    "    \"has_bill_color::pink\",\n",
    "    \"has_bill_color::orange\",\n",
    "    \"has_bill_color::black\",\n",
    "    \"has_bill_color::white\",\n",
    "    \"has_bill_color::red\",\n",
    "    \"has_bill_color::buff\",\n",
    "    \"has_crown_color::blue\",\n",
    "    \"has_crown_color::brown\",\n",
    "    \"has_crown_color::iridescent\",\n",
    "    \"has_crown_color::purple\",\n",
    "    \"has_crown_color::rufous\",\n",
    "    \"has_crown_color::grey\",\n",
    "    \"has_crown_color::yellow\",\n",
    "    \"has_crown_color::olive\",\n",
    "    \"has_crown_color::green\",\n",
    "    \"has_crown_color::pink\",\n",
    "    \"has_crown_color::orange\",\n",
    "    \"has_crown_color::black\",\n",
    "    \"has_crown_color::white\",\n",
    "    \"has_crown_color::red\",\n",
    "    \"has_crown_color::buff\",\n",
    "    \"has_wing_pattern::solid\",\n",
    "    \"has_wing_pattern::spotted\",\n",
    "    \"has_wing_pattern::striped\",\n",
    "    \"has_wing_pattern::multi-colored\",\n",
    "]\n",
    "selected_attribute_names = list(np.array(attributes)[selected_attributes])"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load the data"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "seed_everything(42)\n",
    "N_CONCEPTS, N_TASKS = 112, 200\n",
    "og_config = cub_configs['0']['MixtureEmbModelSharedProb_AdaptiveDropout_NoProbConcat']\n",
    "if og_config['weight_loss']:\n",
    "    imbalance = find_class_imbalance(os.path.join(cub.BASE_DIR, 'train.pkl'), True)\n",
    "else:\n",
    "    imbalance = None\n",
    "sampling_percent = og_config.get(\"sampling_percent\", 1)\n",
    "if sampling_percent != 1:\n",
    "    # Do the subsampling\n",
    "    new_n_concepts = int(np.ceil(N_CONCEPTS * sampling_percent))\n",
    "    selected_concepts_file = os.path.join(\n",
    "        CUB_RESULTS_DIR,\n",
    "        f\"selected_concepts_sampling_{sampling_percent}.npy\",\n",
    "    )\n",
    "    selected_concepts = np.load(selected_concepts_file)\n",
    "    print(\"\\t\\tSelected concepts:\", selected_concepts)\n",
    "    def subsample_transform(sample):\n",
    "        if isinstance(sample, list):\n",
    "            sample = np.array(sample)\n",
    "        return sample[selected_concepts]\n",
    "\n",
    "    if og_config['weight_loss']:\n",
    "        imbalance = np.array(imbalance)[selected_concepts]\n",
    "\n",
    "    train_dl = load_data(\n",
    "        pkl_paths=[os.path.join(cub.BASE_DIR, 'train.pkl')],\n",
    "        use_attr=True,\n",
    "        no_img=False,\n",
    "        batch_size=og_config['batch_size'],\n",
    "        uncertain_label=False,\n",
    "        n_class_attr=2,\n",
    "        image_dir='images',\n",
    "        resampling=False,\n",
    "        root_dir=cub.CUB_DIR,\n",
    "        num_workers=NUM_WORKERS,\n",
    "        concept_transform=subsample_transform,\n",
    "    )\n",
    "    test_dl = load_data(\n",
    "        pkl_paths=[os.path.join(cub.BASE_DIR, 'test.pkl')],\n",
    "        use_attr=True,\n",
    "        no_img=False,\n",
    "        batch_size=og_config['batch_size'],\n",
    "        uncertain_label=False,\n",
    "        n_class_attr=2,\n",
    "        image_dir='images',\n",
    "        resampling=False,\n",
    "        root_dir=cub.CUB_DIR,\n",
    "        num_workers=NUM_WORKERS,\n",
    "        concept_transform=subsample_transform,\n",
    "    )\n",
    "\n",
    "    # And set the right number of concepts to be used\n",
    "    N_CONCEPTS = new_n_concepts\n",
    "else:\n",
    "    selected_concepts = list(range(N_CONCEPTS))\n",
    "    train_dl = load_data(\n",
    "        pkl_paths=[os.path.join(cub.BASE_DIR, 'train.pkl')],\n",
    "        use_attr=True,\n",
    "        no_img=False,\n",
    "        batch_size=og_config['batch_size'],\n",
    "        uncertain_label=False,\n",
    "        n_class_attr=2,\n",
    "        image_dir='images',\n",
    "        resampling=False,\n",
    "        root_dir=cub.CUB_DIR,\n",
    "        num_workers=NUM_WORKERS,\n",
    "    )\n",
    "    test_dl = load_data(\n",
    "        pkl_paths=[os.path.join(cub.BASE_DIR, 'test.pkl')],\n",
    "        use_attr=True,\n",
    "        no_img=False,\n",
    "        batch_size=og_config['batch_size'],\n",
    "        uncertain_label=False,\n",
    "        n_class_attr=2,\n",
    "        image_dir='images',\n",
    "        resampling=False,\n",
    "        root_dir=cub.CUB_DIR,\n",
    "        num_workers=NUM_WORKERS,\n",
    "    )\n",
    "\n",
    "total_c = []\n",
    "for (_, _, c) in train_dl:\n",
    "    total_c.append(c.cpu().detach())\n",
    "total_c = np.concatenate(total_c, axis=0)\n",
    "concept_corr_matrix = np.corrcoef(total_c.T)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "concept_group_map = defaultdict(list)\n",
    "for i, concept_name in enumerate(np.array(selected_attribute_names)[selected_concepts]):\n",
    "    group = concept_name[:concept_name.find(\"::\")]\n",
    "    concept_group_map[group].append(i)\n",
    "concept_group_map"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Load CUB Models"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def load_cub_model(\n",
    "    config,\n",
    "    n_tasks=N_TASKS,\n",
    "    result_dir=CUB_RESULTS_DIR,\n",
    "    n_concepts=N_CONCEPTS,\n",
    "    split=0,\n",
    "    imbalance=None,\n",
    "    intervention_idxs=None,\n",
    "    adversarial_intervention=False,\n",
    "    train_dl=None,\n",
    "    sequential=False,\n",
    "    independent=False,\n",
    "):\n",
    "    if split is not None:\n",
    "        full_run_name = f\"{config['architecture']}{config.get('extra_name', '')}_{config['c_extractor_arch']}_fold_{split + 1}\"\n",
    "    else:\n",
    "        full_run_name = f\"{config['architecture']}{config.get('extra_name', '')}_{config['c_extractor_arch']}\"\n",
    "    selected_concepts = np.arange(n_concepts)\n",
    "    if config.get(\"message_passing_layers\"):\n",
    "        edges = []\n",
    "        edge_weights = []\n",
    "        corr_thresh = config.get('corr_thresh')\n",
    "        sorted_selected = sorted(selected_concepts)\n",
    "        for i in range(n_concepts):\n",
    "            i_idx = sorted_selected[i]\n",
    "            for j in range(i + 1, n_concepts):\n",
    "                j_idx = sorted_selected[j]\n",
    "                if np.abs(concept_corr_matrix[i_idx, j_idx]) >= corr_thresh:\n",
    "                    edges.append(np.array([[i, j], [j, i]]))\n",
    "                    if config.get(\"weighted_edges\"):\n",
    "                        weight = np.abs(concept_corr_matrix[i_idx, j_idx])\n",
    "                    else:\n",
    "                        weight = 1\n",
    "                    edge_weights.extend([weight, weight])\n",
    "        concept_edge_list = torch.cuda.LongTensor(np.concatenate(edges, axis=-1))\n",
    "        concept_edge_weights = torch.cuda.FloatTensor(np.array(edge_weights))\n",
    "    else:\n",
    "        concept_edge_list = None\n",
    "        concept_edge_weights = None\n",
    "    if (\n",
    "        (intervention_idxs is not None) and\n",
    "        (train_dl is not None) and\n",
    "        (config['architecture'] == \"ConceptBottleneckModel\") and\n",
    "        (not config.get('sigmoidal_prob', True))\n",
    "    ):\n",
    "        # Then let's look at the empirical distribution of the logits in order to\n",
    "        # be able to intervene\n",
    "        model = models.construct_model(\n",
    "            n_concepts=n_concepts,\n",
    "            n_tasks=n_tasks,\n",
    "            config=config,\n",
    "            imbalance=imbalance,\n",
    "            concept_edge_list=concept_edge_list,\n",
    "            concept_edge_weights=concept_edge_weights,\n",
    "        )\n",
    "        trainer = pl.Trainer(\n",
    "            gpus=GPU,\n",
    "        )\n",
    "        batch_results = trainer.predict(model, train_dl)\n",
    "        out_embs = np.concatenate(\n",
    "            list(map(lambda x: x[1], batch_results)),\n",
    "            axis=0,\n",
    "        )\n",
    "        active_intervention_values = []\n",
    "        inactive_intervention_values = []\n",
    "        for idx in range(n_concepts):\n",
    "            active_intervention_values.append(np.percentile(out_embs[:, idx], 95))\n",
    "            inactive_intervention_values.append(np.percentile(out_embs[:, idx], 5))\n",
    "        active_intervention_values = torch.cuda.FloatTensor(active_intervention_values)\n",
    "        inactive_intervention_values = torch.cuda.FloatTensor(inactive_intervention_values)\n",
    "    else:\n",
    "        active_intervention_values = inactive_intervention_values = None\n",
    "    \n",
    "    if sequential:\n",
    "        _, c2y_model = models.construct_sequential_models(\n",
    "            n_concepts=n_concepts,\n",
    "            n_tasks=n_tasks,\n",
    "            config=config,\n",
    "            imbalance=imbalance,\n",
    "        )\n",
    "    elif independent:\n",
    "        _, c2y_model = models.construct_sequential_models(\n",
    "            n_concepts=n_concepts,\n",
    "            n_tasks=n_tasks,\n",
    "            config=config,\n",
    "            imbalance=imbalance,\n",
    "        )\n",
    "    else:\n",
    "        c2y_model = None\n",
    "    model = models.construct_model(\n",
    "        n_concepts=n_concepts,\n",
    "        n_tasks=n_tasks,\n",
    "        config=config,\n",
    "        imbalance=imbalance,\n",
    "        concept_edge_list=concept_edge_list,\n",
    "        concept_edge_weights=concept_edge_weights,\n",
    "        intervention_idxs=intervention_idxs,\n",
    "        adversarial_intervention=adversarial_intervention,\n",
    "        active_intervention_values=active_intervention_values,\n",
    "        inactive_intervention_values=inactive_intervention_values,\n",
    "        c2y_model=c2y_model,\n",
    "    )\n",
    "    print(\"sequential =\", sequential)\n",
    "    print(\"independent =\", independent)\n",
    "    if sequential:\n",
    "        extra = \"Sequential\"\n",
    "    elif independent:\n",
    "        extra = \"Independent\"\n",
    "    else:\n",
    "        extra = \"\"\n",
    "    print(\"extra =\", extra)\n",
    "    model_saved_path = os.path.join(\n",
    "        result_dir or \".\",\n",
    "        f'{extra}{full_run_name}.pt'\n",
    "    )\n",
    "    model.load_state_dict(torch.load(model_saved_path))\n",
    "    return model"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Test interventions for all models in CUB"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import joblib\n",
    "if os.path.exists(\n",
    "    os.path.join(\"results/cub_intervention\", f'total_y_accs.joblib')\n",
    "):\n",
    "    total_y_accs = joblib.load(os.path.join(\"results/cub_intervention\", f'total_y_accs.joblib'))\n",
    "    total_c_accs = joblib.load(os.path.join(\"results/cub_intervention\", f'total_c_accs.joblib'))\n",
    "else:\n",
    "    total_y_accs = []\n",
    "    total_c_accs = []"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "import models_cub\n",
    "reload(cub)\n",
    "reload(models_cub)\n",
    "\n",
    "WHITELIST = [\n",
    "    \"ConceptBottleneckModelFuzzy\",\n",
    "    \"ConceptBottleneckModelBool\",\n",
    "    \"SequentialConceptBottleneckModel\",\n",
    "    \"IndependentConceptBottleneckModel\",\n",
    "    \"ConceptBottleneckModelFuzzyExtraCapacity_Logit\",\n",
    "    'MixtureEmbModelSharedProb_AdaptiveDropout_NoProbConcat',\n",
    "    'MixtureEmbModelSharedProb_Adaptive_NoProbConcat',\n",
    "]\n",
    "\n",
    "# for j, num_groups_intervened in enumerate(range(0, len(concept_group_map) + 1, 4)):\n",
    "groups = list(range(0, len(concept_group_map) + 1, 4))\n",
    "for j, num_groups_intervened in enumerate(groups):\n",
    "    while len(total_y_accs) <= j:\n",
    "        total_y_accs.append(defaultdict(list))\n",
    "        total_c_accs.append(defaultdict(list))\n",
    "    intervention_y_accs = total_y_accs[j]\n",
    "    c_accs = total_c_accs[j]\n",
    "    print(\"Intervening with\", num_groups_intervened, \"our of\", len(concept_group_map), \"concept groups\")\n",
    "    for split, runs in cub_configs.items():\n",
    "        selected_groups_for_split = np.random.choice(\n",
    "            list(concept_group_map.keys()),\n",
    "            size=num_groups_intervened,\n",
    "            replace=False,\n",
    "        )\n",
    "        intervention_idxs = []\n",
    "        for selected_group in selected_groups_for_split:\n",
    "            intervention_idxs.extend(concept_group_map[selected_group])\n",
    "        intervention_idxs = sorted(intervention_idxs)\n",
    "        print(\"\\tFor fold\", split, \"we will intervene on groups\", selected_groups_for_split)\n",
    "        print(\"\\t\\tThis results in concepts\", intervention_idxs, \"being modified\")\n",
    "        for model_name, config in runs.items(): \n",
    "            config[\"shared_prob_gen\"] = config.get(\"shared_prob_gen\", False)\n",
    "            config[\"per_concept_weight\"] = config.get(\"per_concept_weight\", False)\n",
    "            if (model_name not in WHITELIST) or (\"NoConceptSupervision\" in model_name):\n",
    "                continue\n",
    "            if (model_name in intervention_y_accs) and len(intervention_y_accs[model_name]) >= (int(split) + 1):\n",
    "                print(\"\\tSkipping\", model_name, \"as it was found in cached results\")\n",
    "                continue\n",
    "            print(\"Computing for\", model_name)\n",
    "            model = load_cub_model(\n",
    "                config=config,\n",
    "                n_tasks=N_TASKS,\n",
    "                n_concepts=N_CONCEPTS,\n",
    "                result_dir=CUB_RESULTS_DIR,\n",
    "                split=int(split),\n",
    "                imbalance=imbalance,\n",
    "                intervention_idxs=intervention_idxs,\n",
    "                train_dl=train_dl,\n",
    "                sequential=(\"Sequential\" in model_name),\n",
    "                independent=(\"Independent\" in model_name),\n",
    "            )\n",
    "            trainer = pl.Trainer(\n",
    "                gpus=GPU,\n",
    "            )\n",
    "            [test_results] = trainer.test(model, test_dl, verbose=False,)\n",
    "            print(\"\\tFor model:\", model_name, f\"(fold {split}) intervening with\", num_groups_intervened, \"gives:\")\n",
    "            print(\"\\t\\tTest task accuracy:\", test_results['test_y_accuracy'])\n",
    "            intervention_y_accs[model_name].append(test_results['test_y_accuracy'])\n",
    "            print(\"\\t\\tOriginal concept accuracy:\", test_results['test_c_accuracy'])\n",
    "            c_accs[model_name].append(test_results['test_c_accuracy'])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "import joblib\n",
    "joblib.dump(total_y_accs, os.path.join(\"results/cub_intervention\", f'total_y_accs.joblib'))\n",
    "joblib.dump(groups, os.path.join(\"results/cub_intervention\", f'groups.joblib'))\n",
    "joblib.dump(total_c_accs, os.path.join(\"results/cub_intervention\", f'total_c_accs.joblib'))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "{'ConceptBottleneckModelFuzzy': 0.9693821191787719, 'ConceptBottleneckModelBool': 0.980669641494751, 'ConceptBottleneckModelFuzzyExtraCapacity_Logit': 0.8002071142196655, 'MixtureEmbModelSharedProb_AdaptiveDropout_NoProbConcat': 0.9680704116821289, 'MixtureEmbModelSharedProb_Adaptive_NoProbConcat': 0.8123576283454895, 'IndependentConceptBottleneckModel': 1.0, 'SequentialConceptBottleneckModel': 0.9405246853828431}\n",
      "ConceptBottleneckModelFuzzy\n",
      "ConceptBottleneckModelBool\n",
      "ConceptBottleneckModelFuzzyExtraCapacity_Logit\n",
      "MixtureEmbModelSharedProb_AdaptiveDropout_NoProbConcat\n",
      "MixtureEmbModelSharedProb_Adaptive_NoProbConcat\n",
      "IndependentConceptBottleneckModel\n",
      "SequentialConceptBottleneckModel\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAbAAAAGHCAYAAADRMQc/AAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAADscUlEQVR4nOy9d7hcVb3//9pt+pze0vs5SUijJJDQQpMqJQpckQ7KTxRR0IuCXkRU9MpXBUTloqiIErqAooQmNYSSQgopJ8npvU6f3dbvj5kzOZNTckpCCvv1PPPMzF67rL1nZr/nU9ZnSUIIgYODg4ODw0GGvL874ODg4ODgMBIcAXNwcHBwOChxBMzBwcHB4aDEETAHBwcHh4MSR8AcHBwcHA5KHAFzcHBwcDgocQTMwcHBYZjcd999VFRUcNlll+3vrnyqUfd3Bw4FmpqaePLJJ3nrrbeora2lu7sbTdMYO3Ys8+fP59xzz+WYY47ps11dXR2nnHIKAH//+9+ZNWvWoMdZtWoVl19+OQDvv/8+OTk5mbaKiop+t1EUhWAwyOTJk1myZAmXXnophYWFIz1VAFauXMm//vUv3n//fVpbW4nH4wSDQaZMmcKxxx7LhRdeSGlp6aiO4TAww/3eOOx9pk6dyimnnMKMGTP2d1c+1TgCNkp+97vf8Zvf/IZkMommaVRUVHDYYYcRjUbZtGkTlZWVPPXUU5xwwgn8/Oc/Jy8vb5/2Z968eRQXF2fe67pOc3Mz69atY+3atfz1r3/lwQcfZP78+cPed3t7O9/61rd45513ACgoKKCiogKfz0dTUxNr165l9erVPPDAA9x8881ceeWVe+u0DlmuvPJKampqePXVV/drPwzD4KijjuLMM8/kpz/96X7ty4HEb37zG+655x5eeeUVxo8fn1l+9tlnc/bZZ+/HnjmAI2Cj4o477uBvf/sbmqZxww03cMUVVxAMBjPtsViM5cuXc8899/DGG29w1VVXsXz5ctxu9z7r03XXXcepp57aZ/n27dv5+te/TmVlJd/61rdYsWIFkiQNeb9tbW1cfPHF1NXVMWnSJG677TaOP/54ZHmXF3rHjh388pe/ZMWKFdx1113ous6Xv/zlvXJehyK2bbN+/Xpyc3P3d1fYvHkziURif3fjgGPdunX7uwsOg+DEwEbIP//5T/72t78hyzL33XcfX/va17LEC8Dn83H11Vfz29/+FlVV2bRpE7/97W/3S3+nTZvGHXfcAUBNTQ2bNm0a1va33HILdXV1TJ8+neXLl3PiiSdmiRek3Cr33nsvy5YtA+Cee+5hx44de+cEDkEqKyuJRCL7uxuAc6MeCOe6HNg4AjYCbNvmnnvuAeC//uu/OOmkkwZdf8mSJVx++eWcc845LFq06JPoYr/0jpW0t7cPebv333+ft956C4C77rqLgoKCAdeVJInbbruNBQsWcO211+L1evus89JLL/GlL32JJUuWMGfOHI4++mguu+wyHn/8cUzT7LP+ZZddRkVFBY8++ijRaJRf/vKXnH766cybN4+FCxfypS99iQ0bNvTbHyEETz/9NJdddhlHH300c+fO5YwzzuCXv/wl4XC4322am5v52c9+xllnncW8efM4/PDDOeecc/jFL35BV1dXn/V7Avpf//rXMQyDX//615n+LVq0iGuuuYYPPvgga5uTTz6Zz372swDU19dTUVFBRUUFq1atGvDaDoXhXqu6ujoqKiq48847AXjmmWcyfemNZVk88cQTXHbZZSxcuJA5c+Zwwgkn8M1vfpM1a9b025ee/Wzfvp3ly5dzyimnMHfuXBobG5k5cyYVFRWsXbt2wHN58MEHqaio6ONR6O7u5te//jXnn38+hx9+OAsWLOD000/nzjvvpKmpqc9+Vq1aRUVFBcceeywA7777Ltdccw2LFy9mzpw5nH766dx3331Z373vfOc7VFRU0NnZCcApp5xCRUUF9913HzB4EkcymeRPf/oTF198ceZaHXfccXzta1/LuN8Hu1Y1NTV85zvfYenSpZltv/vd79LW1jbgtfq04rgQR8B7771HdXU1kiRx1VVXDWmbW265ZR/3as/0vvmWlZUNebsnn3wSgIULFzJv3rw9rh8IBHjsscf6bbvtttsy+5s2bRpz5syhra2NDz74gPfee48XX3yR3/72t7hcrj7b6rrOlVdeybZt21iwYAEFBQVs2bKFN954gw8++IBnn32WiRMnZtY3DIOvf/3rvPrqq7hcLubPn48kSWzatInf/e53vPDCCzzyyCNZCSfr1q3jy1/+Ml1dXeTl5bFgwQLi8TiVlZU88MADPP300/zlL39hypQpffqXTCa56aabePXVV5k+fToLFy5k8+bNvPXWW7zzzjv85je/yfzZOfbYY9mwYQOb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h7Jnzx7GjBnDuHHj2Lx582M/wz8/rQWdTseIESO4cOECs2fP5umnn86WZ/To0bzwwgt8+umnBAQE0L59+4I097HmYVpuAkFhIPrAHiH++OMP9u7dS4cOHRg5cqSdeFlp0aIFEyZMoHPnznaxysxmM4sXL6ZHjx7Ur1+f+vXr0717d7755huMRqMtn9VH/+abb3Lu3DlefPFFQkNDqVevHs899xwnTpzIVuaff/7J888/T2hoKPXr12fgwIE5xm46fPgwL730Eo0bN6Zu3bq0b9+eWbNmkZCQYJcvPDycZs2aERsby2uvvUbTpk2pV68evXr1YteuXXb5PvroIwDatm1LUFBQnvdv+fLlnD17lqFDh+YoXgAVKlTgs88+Y9q0adStWzfH+7J27Vpat25Nnz59bPt1Oh3z5s2jS5cu1KtXj5CQEPr27cuKFSvszr969WqCgoJYvXp1trJfeOEFgoKCuHbtml2Zr7zyCidOnGDQoEGEhITQsGFDRowYwaVLl+yOj4qK4s0336Rt27bUq1ePJ554gsGDB7Nz584870teDBo0iFq1aqHX65k9ezatWrWiTp06hIeH8/3339vlGz9+PACDBw+2uw6wrMY+cOBAQkJCqFevHl26dGH+/PnodDq78oKCgnj++efZs2cPHTt25Mknn2TFihUEBQXx5Zdf5mjj0KFDCQoKsq0vCJbQJX369KF+/fqEhITQs2dPlixZgtlstuV5kHfd0XcYYPHixXTs2JE6derQqlUrPv74YwwGg2M3XlAgiBbYI8TatWsByxc2L4YMGZIt7a233mLVqlW0aNGCvn37olQq2bVrF7Nnz+bff/+1CYGV27dvM3ToULp27UrXrl05e/YsixcvZuTIkezcudMWamXTpk1MmDCBhg0bMmnSJEwmEz/99BNDhw5l/vz5toCEO3bsYNy4cQQGBjJ27Fjc3Nw4duwYS5YsYe/evaxYscIu1pVOp+Pll1+mWrVqTJw4kaSkJL7++mtGjRrFjz/+SEhICO+88w6LFi3i4MGDvPPOO3h7e+d5X9atW4dCobjv/QsODs51jctbt24RERHByy+/bItEYDabGTlyJPv27aNr1648//zz6PV6tmzZwltvvcW1a9dslfuDcP36dUaNGkWPHj3o06cP586d4/vvv2fw4MFs2rQJd3d3kpKS6N+/P3q9niFDhlC5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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "y_acc_means = [\n",
    "    {\n",
    "        method: np.mean(vals)\n",
    "        for method, vals in results.items()\n",
    "    } for results in total_y_accs\n",
    "]\n",
    "y_acc_stds = [\n",
    "    {\n",
    "        method: np.std(vals)\n",
    "        for method, vals in results.items()\n",
    "    } for results in total_y_accs\n",
    "]\n",
    "method_names = list(y_acc_means[-1].keys())\n",
    "print(y_acc_means[-1])\n",
    "\n",
    "x_vals = groups\n",
    "fig, ax = plt.subplots(figsize=(6, 4))\n",
    "clrs = sns.color_palette(\"colorblind\", 30)\n",
    "color_map = {\n",
    "    'No Concepts': clrs[1],\n",
    "    'Bool': clrs[2],\n",
    "    'Fuzzy': clrs[5],\n",
    "    'Hybrid': clrs[6],\n",
    "    'Isolated Concept Embeddings (ours)': clrs[9],\n",
    "    \"Isolated Concept Embeddings + RandInt (ours)\": clrs[10],\n",
    "    \"MixCEM + RandInt (ours)\": clrs[3],\n",
    "    \"MixCEM (ours)\": clrs[4],\n",
    "}\n",
    "method_to_actual_name = {\n",
    "    'ConceptBottleneckModelBool': 'Bool',\n",
    "    'ConceptBottleneckModelFuzzy': 'Fuzzy',\n",
    "    \"SequentialConceptBottleneckModel\": \"Sequential-CBM\",\n",
    "    \"IndependentConceptBottleneckModel\": \"Independent-CBM\",\n",
    "    'ConceptBottleneckModelFuzzyExtraCapacity_Logit': 'Hybrid',\n",
    "    'MixtureEmbModelSharedProb_AdaptiveDropout_NoProbConcat': \"MixCEM + RandInt (ours)\",\n",
    "    'MixtureEmbModelSharedProb_Adaptive_NoProbConcat': \"MixCEM (ours)\",\n",
    "}\n",
    "\n",
    "for method_name in method_names:\n",
    "    print(method_name)\n",
    "    if method_name not in method_to_actual_name:\n",
    "        continue\n",
    "    means = np.array([item[method_name] * 100 for item in y_acc_means])\n",
    "    stds = np.array([item[method_name] * 100 for item in y_acc_stds])\n",
    "    method_name = method_to_actual_name.get(method_name, method_name)\n",
    "    if method_name not in color_map:\n",
    "        color_map[method_name] = clrs[len(color_map)]\n",
    "    color = color_map[method_name]\n",
    "    ax.plot(\n",
    "        x_vals,\n",
    "        means,\n",
    "        c=color,\n",
    "        zorder=1,\n",
    "    )\n",
    "    ax.scatter(\n",
    "        x_vals,\n",
    "        means,\n",
    "        s=50,\n",
    "        label=method_name,\n",
    "        color=color,\n",
    "        zorder=2,\n",
    "        marker='x' if \"ours\" in method_name else \".\",\n",
    "    )\n",
    "    ax.fill_between(\n",
    "        x_vals,\n",
    "        means - stds,\n",
    "        means + stds,\n",
    "        alpha=0.3,\n",
    "        facecolor=color,\n",
    "    )\n",
    "\n",
    "ax.set_xlabel(\"Concept Groups Intervened\", fontsize=20)\n",
    "ax.set_ylabel(\"Task Accuracy (%)\", fontsize=20)\n",
    "\n",
    "ax.grid(False)\n",
    "handles, labels = ax.get_legend_handles_labels()\n",
    "plt.title('CUB Concept Intervention', fontsize=25)\n",
    "lgd = fig.legend(handles, labels, fontsize=15, loc='upper center', bbox_to_anchor=(0.5,-0.001), ncol=2) #(num_models - 1)//3)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Test incorrect interventions in CUB"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [],
   "source": [
    "import joblib\n",
    "if os.path.exists(\n",
    "    os.path.join(\"results/cub_intervention\", f'adversarial_total_y_accs.joblib')\n",
    "):\n",
    "    adversarial_total_y_accs = joblib.load(os.path.join(\"results/cub_intervention\", f'adversarial_total_y_accs.joblib'))\n",
    "else:\n",
    "    adversarial_total_y_accs = []\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import models_cub\n",
    "reload(cub)\n",
    "reload(models_cub)\n",
    "\n",
    "groups = list(range(0, len(concept_group_map) + 1, 4))\n",
    "for j, num_groups_intervened in enumerate(groups):\n",
    "    while len(adversarial_total_y_accs) <= j:\n",
    "        adversarial_total_y_accs.append(defaultdict(list))\n",
    "    intervention_y_accs = adversarial_total_y_accs[j]\n",
    "    print(\"Intervening with\", num_groups_intervened, \"our of\", len(concept_group_map), \"concept groups\")\n",
    "    for split, runs in cub_configs.items():\n",
    "        selected_groups_for_split = np.random.choice(\n",
    "            list(concept_group_map.keys()),\n",
    "            size=num_groups_intervened,\n",
    "            replace=False,\n",
    "        )\n",
    "        intervention_idxs = []\n",
    "        for selected_group in selected_groups_for_split:\n",
    "            intervention_idxs.extend(concept_group_map[selected_group])\n",
    "        intervention_idxs = sorted(intervention_idxs)\n",
    "        print(\"\\tFor fold\", split, \"we will intervene on groups\", selected_groups_for_split)\n",
    "        print(\"\\t\\tThis results in concepts\", intervention_idxs, \"being modified\")\n",
    "        for model_name, config in runs.items(): \n",
    "            config[\"shared_prob_gen\"] = config.get(\"shared_prob_gen\", False)\n",
    "            config[\"per_concept_weight\"] = config.get(\"per_concept_weight\", False)\n",
    "            if (model_name not in WHITELIST) or (\"NoConceptSupervision\" in model_name):\n",
    "                continue\n",
    "            if (model_name in intervention_y_accs) and len(intervention_y_accs[model_name]) >= (int(split) + 1):\n",
    "                print(\"\\tSkipping\", model_name, \"as it was found in cached results\")\n",
    "                continue\n",
    "            model = load_cub_model(\n",
    "                config=config,\n",
    "                n_tasks=N_TASKS,\n",
    "                n_concepts=N_CONCEPTS,\n",
    "                result_dir=CUB_RESULTS_DIR,\n",
    "                split=int(split),\n",
    "                imbalance=imbalance,\n",
    "                intervention_idxs=intervention_idxs,\n",
    "                train_dl=train_dl,\n",
    "                adversarial_intervention=True,\n",
    "                sequential=(\"Sequential\" in model_name),\n",
    "                independent=(\"Independent\" in model_name),\n",
    "            )\n",
    "            trainer = pl.Trainer(\n",
    "                gpus=GPU,\n",
    "            )\n",
    "            [test_results] = trainer.test(model, test_dl, verbose=False,)\n",
    "            print(\"\\tFor model:\", model_name, f\"(fold {split}) intervening with\", num_groups_intervened, \"gives:\")\n",
    "            print(\"\\t\\tTest task accuracy:\", test_results['test_y_accuracy'])\n",
    "            intervention_y_accs[model_name].append(test_results['test_y_accuracy'])\n",
    "            print(\"\\t\\tOriginal concept accuracy:\", test_results['test_c_accuracy'])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import joblib\n",
    "joblib.dump(\n",
    "    adversarial_total_y_accs,\n",
    "    os.path.join(\"results/cub_intervention\", f'adversarial_total_y_accs.joblib'),\n",
    ")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "y_acc_means = [\n",
    "    {\n",
    "        method: np.mean(vals)\n",
    "        for method, vals in results.items()\n",
    "    } for results in adversarial_total_y_accs\n",
    "]\n",
    "y_acc_stds = [\n",
    "    {\n",
    "        method: np.std(vals)\n",
    "        for method, vals in results.items()\n",
    "    } for results in adversarial_total_y_accs\n",
    "]\n",
    "method_names = list(y_acc_means[-1].keys())\n",
    "print(y_acc_means[-1])\n",
    "\n",
    "x_vals = groups\n",
    "fig, ax = plt.subplots(figsize=(6, 4))\n",
    "clrs = sns.color_palette(\"colorblind\", 30)\n",
    "color_map = {\n",
    "    'No Concepts': clrs[1],\n",
    "    'Bool': clrs[2],\n",
    "    'Fuzzy': clrs[5],\n",
    "    'Hybrid': clrs[6],\n",
    "    'Isolated Concept Embeddings (ours)': clrs[9],\n",
    "    \"Isolated Concept Embeddings + RandInt (ours)\": clrs[10],\n",
    "    \"MixCEM + RandInt (ours)\": clrs[3],\n",
    "    \"MixCEM (ours)\": clrs[4],\n",
    "}\n",
    "method_to_actual_name = {\n",
    "    'ConceptBottleneckModelBool': 'Bool',\n",
    "    'ConceptBottleneckModelFuzzy': 'Fuzzy',\n",
    "    \"SequentialConceptBottleneckModel\": \"Sequential-CBM\",\n",
    "    \"IndependentConceptBottleneckModel\": \"Independent-CBM\",\n",
    "    'ConceptBottleneckModelFuzzyExtraCapacity_Logit': 'Hybrid',\n",
    "    'MixtureEmbModelSharedProb_AdaptiveDropout_NoProbConcat': \"MixCEM + RandInt (ours)\",\n",
    "    'MixtureEmbModelSharedProb_Adaptive_NoProbConcat': \"MixCEM (ours)\",\n",
    "}\n",
    "\n",
    "for method_name in method_names:\n",
    "    print(method_name)\n",
    "    if method_name not in method_to_actual_name:\n",
    "        continue\n",
    "    means = np.array([item[method_name] * 100 for item in y_acc_means])\n",
    "    stds = np.array([item[method_name] * 100 for item in y_acc_stds])\n",
    "    method_name = method_to_actual_name.get(method_name, method_name)\n",
    "    if method_name not in color_map:\n",
    "        color_map[method_name] = clrs[len(color_map)]\n",
    "    color = color_map[method_name]\n",
    "    ax.plot(\n",
    "        x_vals,\n",
    "        means,\n",
    "        c=color,\n",
    "        zorder=1,\n",
    "    )\n",
    "    ax.scatter(\n",
    "        x_vals,\n",
    "        means,\n",
    "        s=50,\n",
    "        label=method_name,\n",
    "        color=color,\n",
    "        zorder=2,\n",
    "        marker='x' if \"ours\" in method_name else \".\",\n",
    "    )\n",
    "    ax.fill_between(\n",
    "        x_vals,\n",
    "        means - stds,\n",
    "        means + stds,\n",
    "        alpha=0.3,\n",
    "        facecolor=color,\n",
    "    )\n",
    "\n",
    "ax.set_xlabel(\"Concept Groups Incorrectly Intervened\", fontsize=20)\n",
    "ax.set_ylabel(\"Task Accuracy (%)\", fontsize=20)\n",
    "\n",
    "ax.grid(False)\n",
    "handles, labels = ax.get_legend_handles_labels()\n",
    "plt.title('CUB Incorrect Concept Intervention', fontsize=25)\n",
    "lgd = fig.legend(handles, labels, fontsize=15, loc='upper center', bbox_to_anchor=(0.5,-0.001), ncol=2) #(num_models - 1)//3)\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# And let's look at how interventions change as the probability of train-time intervention varies"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import joblib\n",
    "if os.path.exists(\n",
    "    os.path.join(\"results/cub_intervention\", f'prob_ablation_total_y_accs.joblib')\n",
    "):\n",
    "    prob_ablation_total_y_accs = joblib.load(os.path.join(\"results/cub_intervention\", f'prob_ablation_total_y_accs.joblib'))\n",
    "else:\n",
    "    prob_ablation_total_y_accs = defaultdict(lambda: defaultdict(list))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import models_cub\n",
    "reload(cub)\n",
    "reload(models_cub)\n",
    "\n",
    "prob_ablation_configs = defaultdict(dict)\n",
    "for file in os.listdir(CUB_ABLATION_RESULTS_DIR):\n",
    "    if '_experiment_config.joblib' in file:\n",
    "        config = load(os.path.join(CUB_ABLATION_RESULTS_DIR, file))\n",
    "        fold = int(file[file.find(\"_fold_\") + len(\"_fold_\"):file.find(\"_experiment_config\")]) - 1\n",
    "        model_name = f\"{config['architecture']}{config.get('extra_name', '')}\"\n",
    "        prob_ablation_configs[str(fold)][model_name] = config\n",
    "print(\"CUB Prob Ablation Model names:\")\n",
    "for model_name, _ in prob_ablation_configs['0'].items(): \n",
    "    print(\"\\t\", model_name)\n",
    "    \n",
    "\n",
    "WHITELIST = [\n",
    "    'MixtureEmbModelSharedProb_AdaptiveDropout_NoProbConcat',\n",
    "]\n",
    "\n",
    "\n",
    "splits = sorted(prob_ablation_configs.keys())\n",
    "\n",
    "for split in splits:\n",
    "    runs = prob_ablation_configs[split]\n",
    "    for num_groups_intervened in [\n",
    "        int(len(concept_group_map) * 0.25),\n",
    "        int(len(concept_group_map) * 0.5),\n",
    "        int(len(concept_group_map) * 0.75),\n",
    "        len(concept_group_map),\n",
    "    ]:\n",
    "        selected_groups_for_split = np.random.choice(\n",
    "            list(concept_group_map.keys()),\n",
    "            size=num_groups_intervened,\n",
    "            replace=False,\n",
    "        )\n",
    "        intervention_idxs = []\n",
    "        for selected_group in selected_groups_for_split:\n",
    "            intervention_idxs.extend(concept_group_map[selected_group])\n",
    "        intervention_idxs = sorted(intervention_idxs)\n",
    "        print(\"\\tFor fold\", int(split) + 1, \"we will intervene on groups\", selected_groups_for_split)\n",
    "        print(\"\\t\\tThis results in concepts\", intervention_idxs, \"being modified\")\n",
    "        for model_name, config in runs.items(): \n",
    "            config[\"shared_prob_gen\"] = config.get(\"shared_prob_gen\", False)\n",
    "            config[\"per_concept_weight\"] = config.get(\"per_concept_weight\", False)\n",
    "            if (\"MixtureEmbModelSharedProb_AdaptiveDropout_NoProbConcat\" not in model_name):\n",
    "                print(\"Skipping\", model_name)\n",
    "                continue\n",
    "            model = load_cub_model(\n",
    "                config=config,\n",
    "                n_tasks=N_TASKS,\n",
    "                n_concepts=N_CONCEPTS,\n",
    "                result_dir=CUB_ABLATION_RESULTS_DIR,\n",
    "                split=int(split),\n",
    "                imbalance=imbalance,\n",
    "                intervention_idxs=intervention_idxs,\n",
    "                train_dl=train_dl,\n",
    "            )\n",
    "            trainer = pl.Trainer(\n",
    "                gpus=GPU,\n",
    "            )\n",
    "            [test_results] = trainer.test(model, test_dl, verbose=False,)\n",
    "            print(\n",
    "                \"\\tFor model:\", model_name, f\"(fold {split} trained with prob {config['training_intervention_prob']})\",\n",
    "                \"intervening with\", num_groups_intervened, \"gives:\"\n",
    "            )\n",
    "            print(\"\\t\\tTest task accuracy:\", test_results['test_y_accuracy'])\n",
    "            prob_ablation_total_y_accs[model_name][config['training_intervention_prob']].append(\n",
    "                test_results['test_y_accuracy']\n",
    "            )\n",
    "            print(\"\\t\\tOriginal concept accuracy:\", test_results['test_c_accuracy'])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import joblib\n",
    "joblib.dump(\n",
    "    prob_ablation_total_y_accs,\n",
    "    os.path.join(\"results/cub_intervention\", f'prob_ablation_total_y_accs.joblib'),\n",
    ")"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Finally, let's see how the embedding size affects interventions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [],
   "source": [
    "import joblib\n",
    "if os.path.exists(\n",
    "    os.path.join(\"results/cub_intervention\", f'emb_ablation_total_y_accs.joblib')\n",
    "):\n",
    "    emb_seen = joblib.load(os.path.join(\"results/cub_intervention\", f'emb_ablation_seen.joblib'))\n",
    "    emb_ablation_total_y_accs = joblib.load(os.path.join(\"results/cub_intervention\", f'emb_ablation_total_y_accs.joblib'))\n",
    "else:\n",
    "    emb_ablation_total_y_accs = defaultdict(lambda: defaultdict(list))\n",
    "    emb_seen = set()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import models_cub\n",
    "reload(cub)\n",
    "reload(models_cub)\n",
    "\n",
    "emb_ablation_configs = defaultdict(dict)\n",
    "for file in os.listdir(CUB_EMB_ABLATION_RESULTS_DIR):\n",
    "    if '_experiment_config.joblib' in file:\n",
    "        config = load(os.path.join(CUB_EMB_ABLATION_RESULTS_DIR, file))\n",
    "        fold = int(file[file.find(\"_fold_\") + len(\"_fold_\"):file.find(\"_experiment_config\")]) - 1\n",
    "        model_name = f\"{config['architecture']}{config.get('extra_name', '')}\"\n",
    "        emb_ablation_configs[str(fold)][model_name] = config\n",
    "print(\"CUB Emb Ablation Model names:\")\n",
    "for model_name, _ in emb_ablation_configs['0'].items(): \n",
    "    print(\"\\t\", model_name)\n",
    "\n",
    "sub_selected_concepts = np.load(os.path.join(\n",
    "    CUB_EMB_ABLATION_RESULTS_DIR,\n",
    "    \"selected_concepts_sampling_0.25.npy\",\n",
    "))\n",
    "\n",
    "# Do the subsampling\n",
    "print(\"\\t\\tSelected concepts:\", selected_concepts)\n",
    "def subsample_transform(sample):\n",
    "    if isinstance(sample, list):\n",
    "        sample = np.array(sample)\n",
    "    return sample[sub_selected_concepts]\n",
    "\n",
    "sub_test_dl = load_data(\n",
    "    pkl_paths=[os.path.join(cub.BASE_DIR, 'test.pkl')],\n",
    "    use_attr=True,\n",
    "    no_img=False,\n",
    "    batch_size=og_config['batch_size'],\n",
    "    uncertain_label=False,\n",
    "    n_class_attr=2,\n",
    "    image_dir='images',\n",
    "    resampling=False,\n",
    "    root_dir=cub.CUB_DIR,\n",
    "    num_workers=NUM_WORKERS,\n",
    "    concept_transform=subsample_transform,\n",
    ")\n",
    "\n",
    "sub_selected_concept_group_map = defaultdict(list)\n",
    "for i, concept_name in enumerate(np.array(selected_attribute_names)[selected_concepts][sub_selected_concepts]):\n",
    "    group = concept_name[:concept_name.find(\"::\")]\n",
    "    sub_selected_concept_group_map[group].append(i)\n",
    "\n",
    "\n",
    "WHITELIST = [\n",
    "    \"ConceptBottleneckModelFuzzyExtraCapacity_Logit\",\n",
    "    'MixtureEmbModelSharedProb_AdaptiveDropout_NoProbConcat',\n",
    "    'MixtureEmbModelSharedProb_Adaptive_NoProbConcat',\n",
    "]\n",
    "\n",
    "\n",
    "splits = sorted(emb_ablation_configs.keys())\n",
    "\n",
    "for split in splits:\n",
    "    runs = emb_ablation_configs[split]\n",
    "    model_names = sorted(runs.keys())\n",
    "    for num_groups_intervened in [\n",
    "        0,\n",
    "        int(len(sub_selected_concept_group_map) * 0.25),\n",
    "        int(len(sub_selected_concept_group_map) * 0.5),\n",
    "        int(len(sub_selected_concept_group_map) * 0.75),\n",
    "        len(sub_selected_concept_group_map),\n",
    "    ]:\n",
    "        selected_groups_for_split = np.random.choice(\n",
    "            list(concept_group_map.keys()),\n",
    "            size=num_groups_intervened,\n",
    "            replace=False,\n",
    "        )\n",
    "        intervention_idxs = []\n",
    "        for selected_group in selected_groups_for_split:\n",
    "            intervention_idxs.extend(sub_selected_concept_group_map[selected_group])\n",
    "        intervention_idxs = sorted(intervention_idxs)\n",
    "        print(\"\\tFor fold\", int(split) + 1, \"we will intervene on groups\", selected_groups_for_split)\n",
    "        print(\"\\t\\tThis results in concepts\", intervention_idxs, \"being modified\")\n",
    "        for model_name in model_names: \n",
    "            config = runs[model_name]\n",
    "            config[\"shared_prob_gen\"] = config.get(\"shared_prob_gen\", False)\n",
    "            config[\"per_concept_weight\"] = config.get(\"per_concept_weight\", False)\n",
    "            print(\"model_name =\", model_name)\n",
    "            \n",
    "            if \"NoConceptSupervision\" in model_name or (\n",
    "                \"ConceptBottleneckModelFuzzy\" in model_name and config.get('extra_dims', 0) == 0\n",
    "            ):\n",
    "                continue\n",
    "            if config[\"emb_size\"] > 32:\n",
    "                print(\"Skipping model\", model_name, \"as it has embedding size\", config[\"emb_size\"])\n",
    "                continue\n",
    "            key = (model_name, split, num_groups_intervened, config[\"emb_size\"])\n",
    "            if key in emb_seen:\n",
    "                print(\"Skipping this as we have already seen it for\", model_name)\n",
    "                continue\n",
    "            print(\"Loading model\", model_name)\n",
    "            model = load_cub_model(\n",
    "                config=config,\n",
    "                n_tasks=N_TASKS,\n",
    "                n_concepts=len(sub_selected_concepts),\n",
    "                result_dir=CUB_EMB_ABLATION_RESULTS_DIR,\n",
    "                split=int(split),\n",
    "                imbalance=np.array(imbalance)[sub_selected_concepts],\n",
    "                intervention_idxs=intervention_idxs,\n",
    "                train_dl=train_dl,\n",
    "            )\n",
    "            trainer = pl.Trainer(\n",
    "                gpus=GPU,\n",
    "            )\n",
    "            [test_results] = trainer.test(model, sub_test_dl, verbose=False,)\n",
    "            print(\n",
    "                \"\\tFor model:\", model_name, f\"(fold {int(split) + 1} trained with emb {config['emb_size']})\",\n",
    "                \"intervening with\", num_groups_intervened, \"gives:\"\n",
    "            )\n",
    "            print(\"\\t\\tTest task accuracy:\", test_results['test_y_accuracy'])\n",
    "            emb_ablation_total_y_accs[model_name][config['emb_size']].append(\n",
    "                test_results['test_y_accuracy']\n",
    "            )\n",
    "            emb_seen.add(key)\n",
    "            print(\"\\t\\tOriginal concept accuracy:\", test_results['test_c_accuracy'])\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "import joblib\n",
    "joblib.dump(\n",
    "    emb_ablation_total_y_accs,\n",
    "    os.path.join(\"results/cub_intervention\", f'emb_ablation_total_y_accs.joblib'),\n",
    ")\n",
    "joblib.dump(emb_seen, os.path.join(\"results/cub_intervention\", f'emb_ablation_seen.joblib'))"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.6.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
