{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "from sklearn import metrics\n",
    "import numpy as np\n",
    "import os\n",
    "import matplotlib.pylab as plt"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_auc(id_score, ood_score):\n",
    "    id_label = np.ones(len(id_score))\n",
    "    ood_label = np.zeros(len(ood_score))\n",
    "    conf = np.concatenate((id_score, ood_score))\n",
    "    label = np.concatenate((id_label, ood_label))\n",
    "    return metrics.roc_auc_score(label, conf)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "base_path = 'path-to-where-scores-are-stored'\n",
    "def get_scores_dataset(model, dataset, statistic):\n",
    "    path = os.path.join(base_path, model, dataset, statistic)\n",
    "    if statistic == 'doctor/alpha.npy':\n",
    "        return -np.load(path)\n",
    "    else:\n",
    "        return np.load(path)\n",
    "\n",
    "# for error detection, we need to retrieve which samples have been correctly classified\n",
    "def get_correct_dataset(model, dataset):\n",
    "    path_correct = os.path.join(base_path, model, dataset, 'base/correct_preds.npy')\n",
    "    return np.load(path_correct)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "# to combine mds scores with a sum\n",
    "\n",
    "def get_scores_mds_combined(model, id_dataset, ood_dataset, layers, correct=None):\n",
    "    id_path = os.path.join(base_path, model, id_dataset, 'mds')\n",
    "    ood_path = os.path.join(base_path, model, ood_dataset, 'mds')\n",
    "    id_score = []\n",
    "    ood_score = []\n",
    "    for layer in layers:\n",
    "        id_score.append(np.load(os.path.join(id_path, 'layer_' + str(layer) + '.npy')))\n",
    "        ood_score.append(np.load(os.path.join(ood_path, 'layer_' + str(layer) + '.npy')))\n",
    "    if correct is not None:\n",
    "        id_score = [id_score_layer[correct] for id_score_layer in id_score]\n",
    "    id_vars = [np.var(scores) for scores in id_score]\n",
    "    for layer_idx in range(len(layers)):\n",
    "        id_score[layer_idx] = id_score[layer_idx] / np.sqrt(id_vars[layer_idx])\n",
    "        ood_score[layer_idx] = ood_score[layer_idx] / np.sqrt(id_vars[layer_idx])\n",
    "    id_score = sum(id_score)\n",
    "    ood_score = sum(ood_score)\n",
    "    return id_score, ood_score"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "statistic ash/ash.npy\n",
      "\n",
      "dataset iNaturalist AUC : 0.9868996200000001\n",
      "\n",
      "dataset openimage-o AUC : 0.9725740131578946\n",
      "\n",
      "dataset imagenet-o AUC : 0.9061425999999999\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_resnet AUC base : 0.63682004\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_resnet AUC correct : 0.6877357334458953\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_vit AUC base : 0.25200528\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_vit AUC correct : 0.2806230291023459\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_vit AUC base : 0.73482342\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_vit AUC correct : 0.7953645152203945\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_resnet AUC base : 0.7650621\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_resnet AUC correct : 0.9013324964890376\n",
      "\n",
      "dataset clean_val_test AUC : 0.5\n",
      "\n",
      "dataset clean_val_test AUC correct : 0.7835249998719662\n",
      "\n",
      "dataset biggan_imagenet AUC : 0.5480314692587703\n",
      "\n",
      "dataset biggan_imagenet AUC correct : 0.8240000620069999\n",
      "\n",
      "dataset synthetic_imagenet AUC : 0.7702259359999999\n",
      "\n",
      "dataset synthetic_imagenet AUC correct : 0.9068447841927394\n",
      "\n",
      "dataset defocus_blur AUC : 0.9453810600000001\n",
      "\n",
      "dataset defocus_blur AUC correct : 0.9890080468265632\n",
      "\n",
      "dataset gaussian_noise AUC : 0.9119782139999999\n",
      "\n",
      "dataset gaussian_noise AUC correct : 0.9782432385853457\n",
      "\n",
      "dataset snow AUC : 0.8348496200000001\n",
      "\n",
      "dataset snow AUC correct : 0.9547930245273198\n",
      "\n",
      "dataset brightness AUC : 0.661110088\n",
      "\n",
      "dataset brightness AUC correct : 0.8781849227147501\n",
      "\n",
      "dataset coco-multiclass AUC : 0.7126941176470588\n",
      "\n",
      "dataset coco-monoclass AUC : 0.5665117647058824\n",
      "\n",
      "statistic she/she.npy\n",
      "\n",
      "dataset iNaturalist AUC : 0.91543733\n",
      "\n",
      "dataset openimage-o AUC : 0.9254773309891108\n",
      "\n",
      "dataset imagenet-o AUC : 0.888352\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_resnet AUC base : 0.64314348\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_resnet AUC correct : 0.7643584646178566\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_vit AUC base : 0.71492168\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_vit AUC correct : 0.7634809052706641\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_vit AUC base : 0.75118302\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_vit AUC correct : 0.8821806241427916\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_resnet AUC base : 0.77679272\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_resnet AUC correct : 0.9313387060517642\n",
      "\n",
      "dataset clean_val_test AUC : 0.5\n",
      "\n",
      "dataset clean_val_test AUC correct : 0.7896354110651896\n",
      "\n",
      "dataset biggan_imagenet AUC : 0.5731984679387175\n",
      "\n",
      "dataset biggan_imagenet AUC correct : 0.8780539471826012\n",
      "\n",
      "dataset synthetic_imagenet AUC : 0.7705295399999998\n",
      "\n",
      "dataset synthetic_imagenet AUC correct : 0.9202886355520457\n",
      "\n",
      "dataset defocus_blur AUC : 0.927390498\n",
      "\n",
      "dataset defocus_blur AUC correct : 0.9888112807511176\n",
      "\n",
      "dataset gaussian_noise AUC : 0.8907737499999999\n",
      "\n",
      "dataset gaussian_noise AUC correct : 0.9764058631575279\n",
      "\n",
      "dataset snow AUC : 0.820672754\n",
      "\n",
      "dataset snow AUC correct : 0.9618764694329299\n",
      "\n",
      "dataset brightness AUC : 0.656320966\n",
      "\n",
      "dataset brightness AUC correct : 0.9009024042529629\n",
      "\n",
      "dataset coco-multiclass AUC : 0.6091529411764707\n",
      "\n",
      "dataset coco-monoclass AUC : 0.5355235294117647\n",
      "\n",
      "statistic relation/relation.npy\n",
      "\n",
      "dataset iNaturalist AUC : 0.96947121\n",
      "\n",
      "dataset openimage-o AUC : 0.9526655058983666\n",
      "\n",
      "dataset imagenet-o AUC : 0.88623695\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_resnet AUC base : 0.6243497\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_resnet AUC correct : 0.7107916558870385\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_vit AUC base : 0.6576028\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_vit AUC correct : 0.7055095650533225\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_vit AUC base : 0.73062148\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_vit AUC correct : 0.8438958570563186\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_resnet AUC base : 0.73827686\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_resnet AUC correct : 0.9037925813159673\n",
      "\n",
      "dataset clean_val_test AUC : 0.5\n",
      "\n",
      "dataset clean_val_test AUC correct : 0.7690795909065302\n",
      "\n",
      "dataset biggan_imagenet AUC : 0.5535108484339374\n",
      "\n",
      "dataset biggan_imagenet AUC correct : 0.8302496232460144\n",
      "\n",
      "dataset synthetic_imagenet AUC : 0.759866676\n",
      "\n",
      "dataset synthetic_imagenet AUC correct : 0.9104911784966357\n",
      "\n",
      "dataset defocus_blur AUC : 0.8926754520000001\n",
      "\n",
      "dataset defocus_blur AUC correct : 0.9796551379465692\n",
      "\n",
      "dataset gaussian_noise AUC : 0.878297372\n",
      "\n",
      "dataset gaussian_noise AUC correct : 0.9720705546064254\n",
      "\n",
      "dataset snow AUC : 0.793217592\n",
      "\n",
      "dataset snow AUC correct : 0.9458439023102668\n",
      "\n",
      "dataset brightness AUC : 0.633793736\n",
      "\n",
      "dataset brightness AUC correct : 0.8695500358016105\n",
      "\n",
      "dataset coco-multiclass AUC : 0.6479117647058823\n",
      "\n",
      "dataset coco-monoclass AUC : 0.5225882352941176\n",
      "\n",
      "statistic combined MDS \n",
      "\n",
      "dataset iNaturalist cls_tokens AUC : 0.9036732399999999\n",
      "\n",
      "dataset iNaturalist all tokens AUC : 0.89394753\n",
      "\n",
      "dataset iNaturalist patch tokens AUC : 0.8696268699999999\n",
      "\n",
      "dataset openimage-o cls_tokens AUC : 0.8988232815335754\n",
      "\n",
      "dataset openimage-o all tokens AUC : 0.8943183700090745\n",
      "\n",
      "dataset openimage-o patch tokens AUC : 0.87652458030853\n",
      "\n",
      "dataset imagenet-o cls_tokens AUC : 0.8725012000000001\n",
      "\n",
      "dataset imagenet-o all tokens AUC : 0.8777727499999999\n",
      "\n",
      "dataset imagenet-o patch tokens AUC : 0.8690665\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_resnet cls tokens AUC : 0.80062874\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_resnet cls tokens AUC correct: 0.8307874538745387\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_resnet all AUC : 0.7311207599999999\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_resnet all AUC correct: 0.7598773062730628\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_resnet patch tokens AUC : 0.62891938\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_resnet patch tokens AUC correct: 0.6512195571955719\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_vit cls tokens AUC : 0.78493572\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_vit cls tokens AUC correct: 0.8100996466431095\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_vit all AUC : 0.68258238\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_vit all AUC correct: 0.7080135689045937\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_vit patch tokens AUC : 0.5388948\n",
      "\n",
      "dataset val_attack_LinfPGD_whitebox_0.050000_40_vit patch tokens AUC correct: 0.5577\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_vit cls tokens AUC : 0.92257718\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_vit cls tokens AUC correct: 0.98423645213629\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_vit all AUC : 0.8995583999999999\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_vit all AUC correct: 0.9728574364521363\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_vit patch tokens AUC : 0.8582618\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_vit patch tokens AUC correct: 0.9449507842076799\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_resnet cls tokens AUC : 0.9174191399999998\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_resnet cls tokens AUC correct: 0.9524475452196383\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_resnet all AUC : 0.8960273199999998\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_resnet all AUC correct: 0.9377900086132644\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_resnet patch tokens AUC : 0.86069744\n",
      "\n",
      "dataset val_attack_AA_whitebox_0.050000_Linf_resnet patch tokens AUC correct: 0.9126487510766581\n",
      "\n",
      "dataset clean_val_test cls_tokens AUC : 0.49999999999999994\n",
      "\n",
      "dataset clean_val_test cls_tokens AUC correct: 0.5493394098155802\n",
      "\n",
      "dataset clean_val_test all tokens AUC : 0.5\n",
      "\n",
      "dataset clean_val_test all tokens AUC correct: 0.5371981284024971\n",
      "\n",
      "dataset clean_val_test patch tokens AUC : 0.5\n",
      "\n",
      "dataset clean_val_test patch tokens AUC correct: 0.5219044957774466\n",
      "\n",
      "dataset biggan_imagenet cls_tokens AUC : 0.5409186647465898\n",
      "\n",
      "dataset biggan_imagenet cls_tokens AUC correct: 0.5677462326650144\n",
      "\n",
      "dataset biggan_imagenet all tokens AUC : 0.4498507060282412\n",
      "\n",
      "dataset biggan_imagenet all tokens AUC correct: 0.47192312115375634\n",
      "\n",
      "dataset biggan_imagenet patch tokens AUC : 0.36127879915196615\n",
      "\n",
      "dataset biggan_imagenet patch tokens AUC correct: 0.37560525114610516\n",
      "\n",
      "dataset synthetic_imagenet cls_tokens AUC : 0.9080869400000001\n",
      "\n",
      "dataset synthetic_imagenet cls_tokens AUC correct: 0.9301887810789435\n",
      "\n",
      "dataset synthetic_imagenet all tokens AUC : 0.869785352\n",
      "\n",
      "dataset synthetic_imagenet all tokens AUC correct: 0.8976811942953185\n",
      "\n",
      "dataset synthetic_imagenet patch tokens AUC : 0.8044441119999999\n",
      "\n",
      "dataset synthetic_imagenet patch tokens AUC correct: 0.8400929972734791\n",
      "\n",
      "dataset defocus_blur cls_tokens AUC : 0.9973982280000001\n",
      "\n",
      "dataset defocus_blur cls_tokens AUC correct: 0.9979676137225456\n",
      "\n",
      "dataset defocus_blur all tokens AUC : 0.995104612\n",
      "\n",
      "dataset defocus_blur all tokens AUC correct: 0.9958913434029314\n",
      "\n",
      "dataset defocus_blur patch tokens AUC : 0.990670516\n",
      "\n",
      "dataset defocus_blur patch tokens AUC correct: 0.9925207123292035\n",
      "\n",
      "dataset gaussian_noise cls_tokens AUC : 0.999980524\n",
      "\n",
      "dataset gaussian_noise cls_tokens AUC correct: 0.9999788056792842\n",
      "\n",
      "dataset gaussian_noise all tokens AUC : 0.9997574080000001\n",
      "\n",
      "dataset gaussian_noise all tokens AUC correct: 0.9997400436624324\n",
      "\n",
      "dataset gaussian_noise patch tokens AUC : 0.9978367260000001\n",
      "\n",
      "dataset gaussian_noise patch tokens AUC correct: 0.9980542137769546\n",
      "\n",
      "dataset snow cls_tokens AUC : 0.9613872360000001\n",
      "\n",
      "dataset snow cls_tokens AUC correct: 0.9691568806492692\n",
      "\n",
      "dataset snow all tokens AUC : 0.961797506\n",
      "\n",
      "dataset snow all tokens AUC correct: 0.9691854629051266\n",
      "\n",
      "dataset snow patch tokens AUC : 0.953874994\n",
      "\n",
      "dataset snow patch tokens AUC correct: 0.9616155389965512\n",
      "\n",
      "dataset brightness cls_tokens AUC : 0.863145774\n",
      "\n",
      "dataset brightness cls_tokens AUC correct: 0.8818884743145521\n",
      "\n",
      "dataset brightness all tokens AUC : 0.838473438\n",
      "\n",
      "dataset brightness all tokens AUC correct: 0.8566881218631405\n",
      "\n",
      "dataset brightness patch tokens AUC : 0.792867736\n",
      "\n",
      "dataset brightness patch tokens AUC correct: 0.8106219214166622\n",
      "\n",
      "dataset coco-multiclass cls_tokens AUC : 0.2605882352941176\n",
      "\n",
      "dataset coco-multiclass all tokens AUC : 0.23104117647058822\n",
      "\n",
      "dataset coco-multiclass patch tokens AUC : 0.2215176470588235\n",
      "\n",
      "dataset coco-monoclass cls_tokens AUC : 0.46655882352941175\n",
      "\n",
      "dataset coco-monoclass all tokens AUC : 0.4613882352941176\n",
      "\n",
      "dataset coco-monoclass patch tokens AUC : 0.4630235294117646\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# this will compute the AUC in DSD setting (and Error detection when relevant) for all the ood splits listed. Results will be printed separately per method and per split.\n",
    "\n",
    "id_dataset = 'clean_val_test'\n",
    "id_dataset_coco = 'imagenet-coco-classes'\n",
    "model = 'vit'\n",
    "\n",
    "ood_datasets_coco = ['coco-multiclass', 'coco-monoclass']\n",
    "ood_datasets_osr = ['iNaturalist', 'openimage-o', 'imagenet-o']\n",
    "ood_datasets_adv = ['val_attack_LinfPGD_whitebox_0.050000_40_resnet', \n",
    "                    'val_attack_LinfPGD_whitebox_0.050000_40_vit',\n",
    "                    'val_attack_AA_whitebox_0.050000_Linf_vit',\n",
    "                    'val_attack_AA_whitebox_0.050000_Linf_resnet']\n",
    "ood_datasets_synth_corrupt = ['clean_val_test', 'biggan_imagenet', 'synthetic_imagenet', \n",
    "                              'defocus_blur', 'gaussian_noise', 'snow', 'brightness']\n",
    "statistics = ['ash/ash.npy', 'she/she.npy', 'relation/relation.npy', 'cadet/m_in.npy', 'cadet/m_in_logits.npy', 'maxlogit/logitnorm.npy', 'maxlogit/maxlogit.npy', 'maxlogit/softmax.npy',\n",
    "              'mds/layer_0.npy', 'odin/odin.npy', 'react/react.npy', 'gradnorm/gradnorm.npy', 'ebo/energy_scores.npy',\n",
    "              'doctor/alpha.npy', 'dice/dice.npy', 'vim/vim.npy']\n",
    "\n",
    "id_correct = get_correct_dataset(model, id_dataset)\n",
    "id_correct_coco = get_correct_dataset(model, id_dataset_coco)\n",
    "\n",
    "for stat in statistics:\n",
    "    print('statistic ' + stat + '\\n')\n",
    "    id_score = get_scores_dataset(model, id_dataset, stat)\n",
    "    id_score_coco = get_scores_dataset(model, id_dataset_coco, stat)\n",
    "    for ood_dataset in ood_datasets_osr:\n",
    "        ood_score = get_scores_dataset(model, ood_dataset, stat)\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' AUC : ' + str(auc) + '\\n')\n",
    "    for ood_dataset in ood_datasets_adv:\n",
    "        ood_score = get_scores_dataset(model, ood_dataset, stat)\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' AUC base : ' + str(auc) + '\\n')\n",
    "        ood_correct = get_correct_dataset(model, ood_dataset)\n",
    "        ood_score = ood_score[id_correct]\n",
    "        id_score_adv = id_score[id_correct]\n",
    "        ood_correct = ood_correct[id_correct]\n",
    "        ood_score = ood_score[np.invert(ood_correct)]\n",
    "        id_score_adv = id_score_adv[np.invert(ood_correct)]\n",
    "        auc = get_auc(id_score_adv, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' AUC correct : ' + str(auc) + '\\n')\n",
    "    for ood_dataset in ood_datasets_synth_corrupt:\n",
    "        ood_score = get_scores_dataset(model, ood_dataset, stat)\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' AUC : ' + str(auc) + '\\n')\n",
    "        ood_correct = get_correct_dataset(model, ood_dataset)\n",
    "        id_score_correct = id_score[id_correct]\n",
    "        ood_score = ood_score[np.invert(ood_correct)]\n",
    "        auc = get_auc(id_score_correct, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' AUC correct : ' + str(auc) + '\\n')\n",
    "    for ood_dataset in ood_datasets_coco:\n",
    "        ood_score = get_scores_dataset(model, ood_dataset, stat)\n",
    "        auc = get_auc(id_score_coco, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' AUC : ' + str(auc) + '\\n')\n",
    "\n",
    "        \n",
    "\n",
    "\n",
    "print('statistic combined MDS \\n')\n",
    "for ood_dataset in ood_datasets_osr:\n",
    "    if model == 'resnet50':\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(5)))\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' AUC : ' + str(auc) + '\\n')\n",
    "    elif model == 'vit':\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(12)))\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' cls_tokens AUC : ' + str(auc) + '\\n')\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(24)))\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' all tokens AUC : ' + str(auc) + '\\n')\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(24))[12:])\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' patch tokens AUC : ' + str(auc) + '\\n')\n",
    "\n",
    "for ood_dataset in ood_datasets_adv:\n",
    "    if model == 'resnet50':\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(5)))\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' AUC : ' + str(auc) + '\\n')\n",
    "        id_score_adv, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(5)), correct=id_correct)\n",
    "        ood_correct = get_correct_dataset(model, ood_dataset)\n",
    "        ood_correct = ood_correct[id_correct]\n",
    "        ood_score = ood_score[id_correct]\n",
    "        ood_score = ood_score[np.invert(ood_correct)]\n",
    "        id_score_adv = id_score_adv[np.invert(ood_correct)]\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' AUC correct: ' + str(auc) + '\\n')\n",
    "    elif model == 'vit':\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(12)))\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' cls tokens AUC : ' + str(auc) + '\\n')\n",
    "        id_score_adv, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(12)), correct=id_correct)\n",
    "        ood_correct = get_correct_dataset(model, ood_dataset)\n",
    "        ood_correct = ood_correct[id_correct]\n",
    "        ood_score = ood_score[id_correct]\n",
    "        ood_score = ood_score[np.invert(ood_correct)]\n",
    "        id_score_adv = id_score_adv[np.invert(ood_correct)]\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' cls tokens AUC correct: ' + str(auc) + '\\n')\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(24)))\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' all AUC : ' + str(auc) + '\\n')\n",
    "        id_score_adv, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(24)), correct=id_correct)\n",
    "        ood_correct = get_correct_dataset(model, ood_dataset)\n",
    "        ood_correct = ood_correct[id_correct]\n",
    "        ood_score = ood_score[id_correct]\n",
    "        ood_score = ood_score[np.invert(ood_correct)]\n",
    "        id_score_adv = id_score_adv[np.invert(ood_correct)]\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' all AUC correct: ' + str(auc) + '\\n')\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(24))[12:])\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' patch tokens AUC : ' + str(auc) + '\\n')\n",
    "        id_score_adv, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(24))[12:], correct=id_correct)\n",
    "        ood_correct = get_correct_dataset(model, ood_dataset)\n",
    "        ood_correct = ood_correct[id_correct]\n",
    "        ood_score = ood_score[id_correct]\n",
    "        ood_score = ood_score[np.invert(ood_correct)]\n",
    "        id_score_adv = id_score_adv[np.invert(ood_correct)]\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' patch tokens AUC correct: ' + str(auc) + '\\n')\n",
    "\n",
    "for ood_dataset in ood_datasets_synth_corrupt:\n",
    "    if model == 'resnet50':\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(5)))\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' AUC : ' + str(auc) + '\\n')\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(5)), correct=id_correct)\n",
    "        ood_correct = get_correct_dataset(model, ood_dataset)\n",
    "        ood_score = ood_score[np.invert(ood_correct)]\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' AUC correct: ' + str(auc) + '\\n')\n",
    "    elif model == 'vit':\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(12)))\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' cls_tokens AUC : ' + str(auc) + '\\n')\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(12)), correct=id_correct)\n",
    "        ood_correct = get_correct_dataset(model, ood_dataset)\n",
    "        ood_score = ood_score[np.invert(ood_correct)]\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' cls_tokens AUC correct: ' + str(auc) + '\\n')\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(24)))\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' all tokens AUC : ' + str(auc) + '\\n')\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(24)), correct=id_correct)\n",
    "        ood_correct = get_correct_dataset(model, ood_dataset)\n",
    "        ood_score = ood_score[np.invert(ood_correct)]\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' all tokens AUC correct: ' + str(auc) + '\\n')\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(24))[12:])\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' patch tokens AUC : ' + str(auc) + '\\n')\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset, ood_dataset, list(range(24))[12:], correct=id_correct)\n",
    "        ood_correct = get_correct_dataset(model, ood_dataset)\n",
    "        ood_score = ood_score[np.invert(ood_correct)]\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' patch tokens AUC correct: ' + str(auc) + '\\n')\n",
    "\n",
    "for ood_dataset in ood_datasets_coco:\n",
    "    if model == 'resnet50':\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset_coco, ood_dataset, list(range(5)))\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' AUC : ' + str(auc) + '\\n')\n",
    "    elif model == 'vit':\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset_coco, ood_dataset, list(range(12)))\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' cls_tokens AUC : ' + str(auc) + '\\n')\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset_coco, ood_dataset, list(range(24)))\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' all tokens AUC : ' + str(auc) + '\\n')\n",
    "        id_score, ood_score = get_scores_mds_combined(model, id_dataset_coco, ood_dataset, list(range(24))[12:])\n",
    "        auc = get_auc(id_score, ood_score)\n",
    "        print('dataset ' + ood_dataset + ' patch tokens AUC : ' + str(auc) + '\\n')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "# to compute the covariance matrix\n",
    "model = 'vit'\n",
    "id_train_dataset = 'clean_val_train'\n",
    "\n",
    "id_datasets = ['clean_val_train']\n",
    "ood_datasets_osr = ['iNaturalist', 'openimage-o', 'imagenet-o']\n",
    "ood_datasets_adv = ['val_attack_LinfPGD_whitebox_0.050000_40_resnet', \n",
    "                    'val_attack_LinfPGD_whitebox_0.050000_40_vit',\n",
    "                    'val_attack_AA_whitebox_0.050000_Linf_vit',\n",
    "                    'val_attack_AA_whitebox_0.050000_Linf_resnet',]\n",
    "ood_datasets_synth = ['biggan_imagenet', 'synthetic_imagenet'] \n",
    "ood_datasets_corrupt = ['defocus_blur', 'gaussian_noise', 'snow', 'brightness']\n",
    "\n",
    "target_datasets = id_datasets\n",
    "\n",
    "statistics = ['gradnorm/gradnorm.npy', 'cadet/m_in.npy', 'maxlogit/logitnorm.npy', 'odin/odin.npy', 'dice/dice.npy', 'ebo/energy_scores.npy',\n",
    "              'maxlogit/maxlogit.npy',\n",
    "              'react/react.npy',\n",
    "              'vim/vim.npy', 'maxlogit/softmax.npy', 'doctor/alpha.npy', 'mds/layer_0.npy', 'mds-all', 'ash/ash.npy', 'she/she.npy', 'relation/relation.npy']\n",
    "\n",
    "if model == 'vit':\n",
    "    statistics.append('mds/layer_11.npy')\n",
    "elif model == 'resnet50':\n",
    "    statistics.append('mds/layer_4.npy')\n",
    "\n",
    "total_score = []\n",
    "\n",
    "for stat in statistics:\n",
    "    stat_score = []\n",
    "    for dataset in target_datasets:\n",
    "        if stat == 'mds-all':\n",
    "            if model == 'vit':\n",
    "                _, score = get_scores_mds_combined(model, id_train_dataset, dataset, list(range(12)))\n",
    "            else:\n",
    "                _, score = get_scores_mds_combined(model, id_train_dataset, dataset, list(range(5)))\n",
    "        else:\n",
    "            score = get_scores_dataset(model, dataset, stat)\n",
    "        stat_score.extend(score)\n",
    "    stat_score = np.array(stat_score)\n",
    "    stat_score = (stat_score - stat_score.mean()) / stat_score.std()\n",
    "\n",
    "    total_score.append(stat_score)\n",
    "\n",
    "total_score = np.array(total_score)\n",
    "\n",
    "cov = np.cov(total_score)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAS4AAAEyCAYAAABEVD2jAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjUuMSwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy/YYfK9AAAACXBIWXMAAAsTAAALEwEAmpwYAAA7IklEQVR4nO2dd7xcVbXHv7+bTgIhkACCQCA0KSGBoKiUALZnA0UIiAhKU6kWnh0DPttTH1JEDEpREVAEVESKQDAUISGVjrQAASFAQhop9673x96Te+7cc2bumTtzp7C+n8987px99jp7nzkz6+6yiswMx3GcZqKt3h1wHMfJiysux3GaDldcjuM0Ha64HMdpOlxxOY7TdLjichyn6XDF5ThO0+GKy3GcpqN/vTvgOK2EpI2AwYVjM5tfx+60LD7icpwqIOmjkh4HngLuAJ4G/l7XTrUwrrgcpzp8F9gTeMzMtgIOAP5V3y61Lq64HKc6rDazV4A2SW1mdjswod6dalV8javJkLQeiedmZq/WsTtOJ4skDQP+CVwu6SVgWZ371LLIo0M0B5JOAM4E3gAKD83MbOv69copIGko4dkIOAIYDlweR2FOlXHF1STEhd93mtnCevfFceqNr3E1D08Ay+vdCScdSR+X9LikxZJel7RE0uv17ler4iOuJkHSeOAS4F5gZaHczE6pW6ectUj6N/ARM3u43n15M+CL883DL4HbgHlAR5374nTnP660+g4fcTUJkmaZ2fh698PpiqSPx7f7ApsA19F1RHxNHbrV8rjiahIkfZ9gjf1Xuv4w3Byijki6pMRpM7PP9lln3kS44moSJD2VUtyU5hCSxgDPmdlKSROBscBvzGxRPfvlNA+uuKqApK2Ak4HRdDUO/WiVrt8GHGJmV1XjevVG0myCVflo4Abgz8BOZvbBOnarV0jaGjiH4PZjwD3AaWaW9g/H6SWuuKqApDnArylaODezO6rYxgwzawkXEkkzzWw3SacDb5jZec2+hifpX8DPgSti0WHAyWb2jvr1qnXxXcXq8IaZnVvjNv4h6SvAVSRcSZp0jWu1pMOBo4CPxLIBdexPNVjHzH6bOP5dVMxODfARVxWQ9ElgW+Bmui6cz6xiG620xrUj8DngHjO7Ik61DzWzH9W5axUj6UfAa8CVhKniJGAE8GNo2n8wDYuPuKrDLsCRwP50ThUtHleFGCqlVXhv0nDWzJ6S9EY9O1QFDo1/TygqP4zwXWi6fzCNjI+4qkC0mt7RzFbVsI0BwOeBfWLRVOCXZra6Vm3WisIaV1FZU69xOX2Lj7iqwwPA+sBLNWzjF4R1oAvi8ZGx7NgatllV4rrWJ4GtJP0lcWpdoCmnUgkD1FTcALU2uOKqDusDj0iaTtc1rqqYQ0T2MLNdE8e3xd3MZuJu4AVgJPDTRPkSYG5detR7PlLinAGuuGqAK67q8J0+aKNd0hgzewLW2g2190G7VcPMngGeAd5Z775UCzP7TL378GbE17h6iaR+wINmtkON2zmAEB3iSUKwui2Bz8QQwU2BpDvNbC9JS+gMhgjhfszM1qtT16qCpA8BO9E1y89Z9etR6+Ijrl5iZu2SHpW0RS1TUZnZrZK2BbaPRY+a2cpSMo2Gme0V/65b775UG0kXAusA+wG/Aj4B3FfXTrUwPuKqApL+CYwnfFGTxqHVXONC0rvo7lb0m2q20RdI2iCleEkz7pAWkDTXzMYm/g4D/m5me9e7b42IpO2A0wkzh+T3uUcmRD7iqg7frnUDkn4LjAFm07m2ZUDTKS5gJrA5wWBThM2NFyX9BzjOzO6vY98qZUX8u1zSpsArwFuqdXFJ59F1et2FJgwo+UfgQuAiKlirdcVVBczsDkkbA3vEovvMrNqmERMItmKtMES+BbjazG4CkPQ+4GDCGt4FQDP6910vaX2CpfxMgpK5qIrXn1HFazUCa8zsF5UK+1SxCkg6lPCFnUoYQewNnG5mV1exjT8Cp5jZC9W6Zr2QNM/MdikqK0yxZpvZuDp1rSpIGgQMNrPF9e5LoyJpMsHu8VoqiC/niqsKRHuq9xZGWZJGAf8osrvqbRu3A+MI62i1shXrEyTdDNxK8OuD4Nf3XuADwPRiq/pmQ9IUMzu+ytf8K6Wnik31Peit761PFatDW9HU8BWqn0FpcpWvV08+SbB9uy4e3xXL+tHp89fM1CL80E9qcM260VvfWx9xVQFJPyZE8SzEYpoEzDWzr9avV42PpHUJ/2WX1rsvvSEGetzTzO6Oxzea2Qfq3K2Gpre+t664qoSkg4F3x8NpZnZtPfvTyEjahbAbWjCLWAgcZWYP1K9XvaOvnMSjLd8PgB3paujaVNEnJP2K4Ht7WSw6Emg3sx753rriqhOS3m1md5Ura0Uk3Q18s2D1H+POf9/M3lXPfvUGST8hhGu+ptzOr6RDzOyPkrbKG9pZ0p2EafbZBD/JzxCWKs6osOt1QdKc4jXgtLJMeVdclRMXGLM+QDOzMSVk00K7dCvLkB0BbG5mTemY3NsvbSMS3ZiGEmySCrHFUt2YEqGre/S8i2TvN7PdkzuzhbLe3kNfImkmIY9C0vf26p5+Hr443zuKF2HbCIvLXwFmpQlIeifwLmCUpC8lTq1HWJxORdJU4KOEZ3Y/8JKku8zsS1kyDcyTkr4NFEIdf4rgg9m05HRjeiXurBaH9ylcq9QO4cq4pva4pJOA54FhxZWawGD1dOB2SV18b3sq7IqrF5jZK7B2cfZIwsOYDXzIzB7KEBtI+KL1J8ShKvA6wb8ti+Fm9rqkYwmpvL4jqSlHXMBngTPpDPkyLZaVpDDFKldWLyR9lMRis5ldn1H1Q8BuBMX904w6WZxK8Ik8BfguwTfyqJR6DW2w2lvfW58q9oK4M/JZ4IvAncAPzezfPZTd0syekbSOmS3vQf15wPsIi5nfNLPpBaPNXtxCU9Gb6XWtkfRDgufE5bHocGCGmX29hMwoM3u5xPnzzOzknP3IJVNJG71B0v5mdltWAMaeBl70EVfveApYA/wMmA+MlbRWkZR5CJtK+jth9LWFpF2BE8zsCxn1zwRuAu6MSmtr4PEq3EOfUakRpaT/Aj4IbCYpmU1pPcLn3wh8EBhnZh0Aki4jLBdkKq5SSivy7jLnqyFTSRu9YV/gNtIDMPY48KIrrt7xD8KHvWt8JSn3EH4GvB/4C4CZzZG0T4n6LyRHV2b2pKT/q6TTdaRSI8oFhKnPRwnrewWWEEa7jcL6dIagHl7HfjQsZlYIunlW8Y6qQranHuGKqxeY2dG9lH9WUrKolJf8eYR1kXJlabudShyX3O2sJVZhglwzmwPMkXQtsMzM2mFtEMdBVexib/gBMCu6Zomw1pU52nL4E92/u1cDPdoddcXVC4p2BbthZqVGRM/G+FoW18pOBR5OaaOSXcjcu51Nws3Ae4CCpf2QWFZ3+y8L+SGn0hkh5Ktm9mIvL6vyVXotU0kbFSNpB0KU2OFF61zrkTCoLYcrrt5R2BXcnvCFLWxtf4Ty0S8/B5wDbEbY0r4ZODGlXu5dyAp3O5uBwUn3IDNbKmmdenaogKRbzewAOr8DybKeyI8AFhUZr55TTZlK2qgB2wMfJkyrk+tcS4DjenwVM/NXL1/AP4F1E8frAv+schtb5qg7gJCY9BFCGOFtcra1DiE44kXxeFvgw1W8l8EpZSN7IHcXsFvieHdCNux6PvvBBNelOYTM1RvE12jgkQyZM4Ad4vtBwO2EtbGXgPdUQ6aSNvr4c3tnb+R9xFUdNgaSyWBXxbJUJO0HnEynDcvDwPlmNjWl7s/M7DTgfEndduQsfSeuN7udEAL63U9nNp7nCRErs+ySkDQY2CYe/tvMSmWmni7pODP7V5Q9mLBGtF2Zfp0G/FHSAsIUZxOCQ3vVkHQQ4T7mWQx0WIYTYr82JXxmhanX68D5GTKTCDZY0GmDNYpw/5cRNn16K1NJG33JLEkn0j25SFl7PvCpYrX4DXBfXDwGOIhO59EuKGSCOR84i2DiIMIi5cWSTjKzG4pECtbleXbkerPbCTDGzCYpJHDFzJaraBehgKT+wPcJ9mzPEO5nc0mXEOzN0rz9P0m436mEH/yGQNlY4xbMQHYgYbRIp6N2r5F0AeGHdDfwXUlvN7PvlpIxs3OAcySdbGbn9bCpVRaHHYSd5SstbDg8HD/Pasjkql+JqUrBJiu+7+J3KenjZf5B/pYwI3g/4bdwBClrvJnUe8jYKi/CtOXU+Bpfot5UYNeU8rHAHfW+j9iXuwkL3zPj8RhCOOq0umcTpqPJqfJ6wBTgnBJtHERY11hA/qns+sAxhGCEC6p43w8A/eL7dYD7c8ieCKyfOB4BfCGj7r+AnQkjoFeBrRLnsqaXuWQqqL9vfE0k2Afum3xl9Glm2vu04xTZWfHv3Ph3APCvnn7ePuKqEmZ2v6RnicNeZacr28TC9n6x/FyFuPWpRMv54v+Iiwn2Tf9jcUE+1u3NbieE6AM3EkZOlxOMFI/OqPthYDuL3754/dclfZ7wH/XUlHv5NUEZjiVMXa6PFtw/z+qQpCHAgYTR2njCOuJBhPXF4rolLenNbGbGqVUWTS2sxCgzg+OS/Tez1yQdR4ihX8yphK3/UcDZFkcqkj5I9q5vXplc9S1hqiJpqfXMdEUZ79OOiymMxBdJ2hl4EdioB20CPlWsCtFH7aeEac9LwBaEH+1OKdWXpZT15NzfCXZev4/HhxFGBS8Cl9J1h6ZXeQvN7Jbovb8n4Qt4qpktzK5uxQoVC/kms6Ye84Bjo9xTkt4BZCpTSb8nxPG/mWC7dhthHW1qhsgMwuip0Ofkj8jInpbukPD/FDAmHhcS1pZyr+onSYXPItqYDUyraGb3At0SCFtYJiheKqhIppI2ktXKnE+rVyxT7hpT4i7ntwk7scMIGwo9wn0Vq4BCzPn9CXHmx8fF90+Z2TEpdReRMkog/Dj2MrMRGW1k+ukpJflEb5D0MeA2i8keFLLXTDSz61LqXkeIQfWbovJPAYdaFWKhS5pNsEX7DWGt5jlJT1pG8DxJpxFMRRYT4tpfaz2Isippy1LnzeyZErI/JkQ4+GUsOgF41sy+nFF/e+B4OpXLw8AUM3usRBu5ZPLUV9dcl7cTpoxrFb6lJLFIfJcLCWIK3+uS3+Vq4IqrCkiaYWYTogIbb2YdyogvJWnfUtfKGqLHax9nZvfF4z2AX5nZrkqJvhmV50l0/dKm7lymtDXbijLtpLURyzcjLPavoNMdZwJhjexjZvZ8ikzuKJ5xUf5wwm7ZQsIC/c5m9p8SMlsTRqYHEjYOvm9ms7Pqp8hvSLCAn29lcj1Gm7kTgILd1i2E59PNGyIaFV9DWAecSfihjyfYMX3c4m5rb2QqqF/wtkib4lnas6nku1yFZYy1Ff3VyxdhF28YYRpzBcGo7+6c19ickNIs6/wehCnWU8DTwNxYNpQwsknW/VCs9xnCruI4wq7fk8AHe9CXuSll88rI7E8w8TgZOKBM3TsJP/C5hFHKZILvWk8/qwmEXdb55T5nwnT9u/EzO7RM3esJyhBCMtcXgL8CDwGn9aBfA4FdCIviA0rU+zthBFtcvi8h+3WvZSppoy9ehPXTzFePr1OvG2ilV1QebYQ1w6MIsZI27IHcKOALhHhUTwA/6YHMcEJsrlJ1ptKLnUvgYsKa05j4+j/g0hL1+9M5et+cME0bV6L+/fHvvOKynJ+7gH1SyrcGvgHcS1ig/gQwpAfXezDx/huEuGcQ1gy7KfMi2YmEUd0dhCnTU2l9i3UfK3GdR6shU0H9LZPfK0Kcr3MITuwDM64zj/DPJ/WV93nmeVU7hdabjrgIe72ZdZjZGjO7zMzOtcQuX1H9dSUdJekmglvQGMJW9Rgz+0qJdoYrRIO4FbhV0k8lDc+onrlzSQnD2AQnE4xor4qvlaS7IxF3zl4CnonvbyUoiqskZWU56hLFM66pdYvimdLWdpJulVRIqrELnYH7kvyb4Jt5IyEO/BbA5yV9qcxUJWlzdgBxEdvMlgAdZbr3U+B9Zravme1DsE86O6PukhLXydqgySuTt/4fCP+AkTSOYHA8nzBaT9sZhbCj/BFC1I5+8X3ylUnxs5Q0VtK3Sskk8V3FXmJh96xD0nDrWebilwgK61uE2FoWf7jluJiwU1bIO3gkwcI9LSBbpTuXAJjZMuBrPegTBKvxMYRRycME16SFCj6E04EfpcicStconvuTHsWzmIsIfpe/jP2cG3cc/6eo3ll07mqVVYgJnpV0MvAcwSj4RlhrijGgjOwAM3u0cGBmjyk4z6exubrGFSsggu9qNWTy1h9iZgvi+08BF5vZT+M/mNlpHbLEZoWklVZi8yKFnj7LVFxxVYelwDxJt5BQDJYe1/vrhAXjC4ArJF3VwzbGmNnBieMz425bal2lxDInfGlLLYD/zMxOy7KitvQdwlVm9hrwmqR/WzSbsGAHtSqlPmY2Pb5dSo4448A6ZnZfkXlVt0CCZjY5xzWTHENQeu8BJpnZoli+J+GfRClmKKTc+l08PoLs8Mmnl7pOlWTy1k9+qPsTQ/JY2GiqRQSJHj3LLFxxVYdr6HSjKfzgUx+2mf0M+Flix+s6QjTU/waus+zt8BWS9jKzOwEkvZuwk5fGgfHvEIKDtBGmTyso7TpUiXvREEnjCWt8A+N7xVeXMCUZynQtGYoxyUJJY4ifsaRPEBbQuyDpv83sf5WRMCLjHwoWspF/LqX8doKJQCk+T5hOF649jYwplpmluoOVIq9MBW3cJukPhM9zBMFWDklvoTNrURfU1dB3SNExlm3oCz18llm4OUQvkHQg8FaLFtOS7iMsuBshHlO3JA6StgE2tkT+RIUEqecQXCtSY2wphHb+DZ2RNV8jJFHtljAjTlG+R9hJLFjvb04wVP2G9SBbsKRRUD68sIK/YeaXyMz2S9R9GXiWsPN6L0XK3cpYa0dlP4UQf+s1wgL4EcVTFIXMMZ8mOEunKa4sP9JeKdYcn1nudvLKVFBfBFOTtwB/sGjGImlvwsZMt+CTCkETkyYUXT5rM8v0P+3ps8yUd8VVOZLuAg4zs2fj8WzCMHsYcImlxGKSdD3wdTObV1Q+lmBn9OEyba4Ha91qTosjuOI6Z8c+fCkuLBfkfgKsMLNTS1x/MsH+q43whVwDnGdmZ5XqV0+IGxnvJdhjjQX+BlxhZg/mvE5hF3c54fO/vOj8aSR+hLGNLFeapFxuxRp/8N+h8zOD4OGQ+ZlV2E4umd78k4ij5k8ChxAUyjWW4kAu6e0EI9sX4vFRwMEE05PJlmK0mnKNks8yk1puWbb6C5hedHx+4n2qw2ixTNG5krZSKfXnZ5Q/TvynVFTeD3i8xPW+RDCcTDrkbk1I0vHFDJn/Trw/pOjc90u0NYjg//gycFKZ+1yPsOZyPkHxiaAongL+XEJuS+CrBN+8RwguJduWqN8P+AAhsscswkLxTmX6VslnVkk7uWQqqL8dQQE/QrCzOxl4pkyfZgIbxPf7EBzmDyZsuFxdzWfZ7Tp5fij+6vYQ/l3i3BMZ5aUUR+b1Muo/m1Feyoan1LlZpAT0I0x/Z2XI5IoQEBXWxwnb7dMJvmqblbnPPxOmuScQRlBTCfZS43J8VuPj/bX3sH6PFGsln1kl7fRGpif1CeYed5CI1AE8Wea6cxLvf04YZRWOZ9fqWZp5dIjecq9CQLyLkoWSTiA7dPOMDJlj6ZrBpidkzfMfkvRpS/cffKTE9QZYijO1mb1cYmu/xxECJP2GYFV+A3CmmT1Az9jaOtPN/4qwiLuFlQ5WWIgV9l+ETZADCD+SyWVkBhE8Dw4nRDE9F7i2hEgln1kl7eSWyVn/44TP6XZJNxJ8PMvtJvaT1N/M1hA+3+MT57J0S0XPshhXXL3ji8B1kj5JGDZDiMs1iBByJY3TgGslHUFX376BQDd7LklLSFdQIuwapnEicI2kz5LiP5h9O6SaL5Q5Zxnv044/RTAXORU4JbEVXoi+sF5GG2s3EyzYzT1X6osuqbCO9kHCP5ArgeMt2KdlUqFizf2ZVdJOXpm89S040F8X15wOJHxPN5L0C4KT+s0pYlcAd0haSNixnhbb3obg4J5GrmeZhS/OVwFJ+9MZwuZBi1Ehy8jsR/hi9Viml/16yMxuLVO/nXQDVRHixHcbQSRkCop0eTmZvBT1K9lOqsKTdBsh/M+fLNiY9bSdjkQ7yR9GpmKt8DOrpJ1cMpW0kdLmCMIC/STLSPohaU/CJsjNhX8MkrYDhlmKOUTeZ5nZN1dcjuM0G+6r6DhO0+GKy3GcpsMVVw2QdHz5WpXXbzWZRu1XX8k0ar8qkemrfrniqg15H0TuB9diMo3ar76SadR+VSLTJ/1yxeU4TtPhu4oVMnKDfjZ68/Sd/pdfaWfUht19pee9Niq1fvvSZfQbNjT13KBF6c9n1aplDBzYXWblyOzn2b5kGf3WTWlnTfb/r6y+tQ3sFkq9U+b15fRbb53uMm3pfVuzeDn9h3evD9Dent639teX0W+99M8s6yuddf/9+2fHCMzq24C27PtftWgFA9fvbmK35qVBqfVXr1zKgEHpYcMswwR0zRvL6D84/f47RqT3Leu5dFTw/LNMUzO/YwCr04U6li2jbWi6zKrnn1toZt1+OG6AWiGjNx/AfTdtnktmzB+6RUwpy1Z/LmXf2J2nj83/j8gyflClGLZVT2ImdmXIwLJBKbrx2pJ0hVaKNatTA2xkMmqDUsFC09lkaH6Zl88fnVumo3/+UFjLJ+V7Nq8vzFA0pahgrtb/pfwmfU9+9cup0SJ8qug4TtNRE8UlaWNJv5f0pKT7Jd2jnoUnzrreZElfie8vlfR89MNC0khJT1ep647jNAFVV1wxPtF1wD/NbGsz253gvPnWonq9maa2E4LkVdI/nx47TpNTixHX/oQ45BcWCszsGTM7T9LRkv4SfclulTRMIdPHTEnzFCKKAiDpm5Iek3QnIflnkp8BXyxWQgr8WNID8XqTYvlESdMUokI+FI/vkPTnOCr8oaQjJN0X5bpFe3Qcp3GoxehjJzojJaSxGzDWzF6NiudjFqJ5jgT+FZXLboRR2rjYx5l0DfkynxDs7EhCws4CH48yuwIjgemSCmnBdyMk+3xK0sRY523Aq4REqb8ys7dLOpUQRO204o5HQ7njAbbYzAdujlMvar44L+nnkuZIKmR2ucU6Q7oK+L6kuYRs0JsR8v7tTQilsdzMXgfS4mf/gJDJJHkPexHC9LZbSM1+ByHbM8B9ZvZUou50M3vBzFYSkrEWwnbMI8Qu6oaZTTGzCWY2Ic3cwXGcvqEWw4YHCeFbATCzE+NoqpASKRkC5AhCpMjdzWx1XGTvkhkmCzN7XCHG+6Hl6qa0CyHJaYGOxHEHbibiOA1NLUZctwGDJX0+UZZljDMceCkqrf0IMcIhpDA/SNIQSeuSnRX3e0Ay+/M0YJKkfgoZV/YhOxKp4zhNStUVlwVT/IOAfSU9pZCy6zJC0oJiLgcmSJpHSCf1SLzGTELq9znA3wmxydPaepCu62nXAnOj3G2ERA4vVuG2HMdpIGoyJbKQruiwjNOXJuotBN6ZcY3vEUZUxeVHFx1/PPHeCOtepxfVmUqIN551PDHrnOM4jYev5VTIvNdG5XbheeLQC8tXKmLrEfnM1Ta6Ib/7zsBP/ye3zMrfbZJbJtOJsASbLMr2I8yiY2A+N5mO/iNzt/HKmg1zy6iC+c2I2a/klsl7P+2b5ncrGrog/7PccEa3nCJleTKj3F1+HMdpOlxxOY7TdDSF4pK0iaQrJT0RfR9viJlEkHSapDckDU/UnyhpsaRZkh6V9E9JJVPbJ+TeVct7cRyn9zS84oq+j9cCU81sTPR9/DrBUBVC/rzpBKv5JNPMbLyZbQ+cApwvKTXFUoKJgCsux2lwGl5xAfsBq4t8H+eY2bToUzgM+BZBgaViZrOBs4CTACSNkvQnSdPj692SRgOfI/hAzpa0d+1uyXGc3tAMu4o7k52a/jBCluJpwPaSNo6uPmnMpNNM4hzgbDO7U9IWwE1m9jZJFwJLzewnVey/4zhVphkUVykOJzhpd0j6EyHr7vkZdZN7vu8BdlRnCvj1JKXHzk1eIOFk3W/EiIo77ThO72gGxfUg8IniQkm7ANsCt0QFNBB4imzFNR54OL5vA/Y0szeKrlmyI2Y2BZgCMGiLzT1Yv+PUiWZY47oNGJTMvSZpLHAuMNnMRsfXpsCmkrYsvkCs/23g57HoZkLomsL5cfHtEmDdmtyF4zhVo+EVV3Tj+RjwnmgO8SAhpM1Ewm5jkmvpdDXau2AOQVBYp5jZrfHcKQQfybmSHiIsykOI7fUxX5x3nMamGaaKmNkCehC+xsy+lDgcXqLeQmBSSvljwNhK+ug4Tt/R8CMux3GcYppixNWIDFpkuXMe5nWYBnjyvRfnqj/xD8flbuPFe/M7TG8157XcMlqT32FaS5bnllm1ZT4n4wGPL8jdBiPz7yov3im/zIot188ts0HOZzPyrjfKVyqiY0H+aFHtb+RvJwsfcTmO03S44nIcp+moSHFJWtrbhiVNkHRufO/OzY7j9Ji6rXGZ2Qw6E2hMBJYCd/dF25L6mVl7X7TlOE71qdpUUdI4Sf+KtlHXShoRy/eIZbMLyVpj+URJ16c5N0s6JCZ1nZPIi5hsa6KkqZKulvSIpMtjFAkkHRDtt+ZJuljSoFj+tKQfSZoJHBKPfxDbnCFpN0k3RVuxfKFNHcfpU6q5xvUb4KtmNpaQm/A7sfwS4AQzGwd0G+WY2dPAhQSn53FmNg04A3i/me0KfDSjvfGEpK07AlsD75Y0mBDTfpKZ7UIYUSazDb1iZruZ2ZXxeH7s17Qo9wlgT+DMnPfuOE4fUhXFFYP4rW9md8Siy4B9JK0PrGtm98Ty3/fwkncBl0o6DsjKvHqfmT1nZh3AbEIS1+2Bp6Ih6dp+JGSuKrpGIdHsPOBeM1tiZi8DK2PfuyDp+Dg6m7FqVXGaRsdx+oqG3FU0s88RYmxtDtwvKS0zQTKhazs9W6/LSgqbTAhbOO52vWQm64EDh/agOcdxakFVFJeZLQZeS/j3HQncYWaLgCWS3hHLs1KWdXFuljTGzO41szOAlwkKrCc8CoyWtE2yHz2/E8dxmoFKdxXXkfRc4vj/gKOACyWtQ8gq9Jl47hjgIkkdBCWyOOV6fwWulnQgIWrDFyVtS4ihdSshwWtZzOwNSZ8B/iipPyGkc/6cYI7jNDQVKS4zyxqp7ZlS9mBcsEfS14gmEMnEqynOzdPKtL9WNh6flHh/K2HhvlhmdNaxmV1K10S1Xeo6jtNY9IUd14ckfT229QxwdB+0WXNWjjSePjZfLMFKkrXm9T2cetFFuduY8J3Pl69UxLFX35BbphIWrM7v3ze8X76Nk4vn75W7jfkLygbM7cZbKvjIFo4dkFtm8rn5Gjr97kNyt6H++Z9LW1sFsTcPvzq1uOaKy8yuovtunuM4TsU05K6i4zhOKVxxOY7TdDS14pL0Vkl/lvR4dNU5R9LAcpmsJU2W9JX4/lJJzydcg0ZKerpOt+Q4Tg9oWsUVfROvAa4zs22B7QjJYb8Xq+TJZN0O5I/y5zhOXWhaxQXsD7xhZpcAxGgPXyQooHWSFYszWafwM4LtmEeEdZwmoJkV104UZbg2s9eB+cA2KfVnAjtkXGs+cCfB0t5xnAanmRVXXkpnew0pz06nxGeSdLJuX+JO1o5TL5pZcT0E7J4skLQesAXw75T6yUzW3TCzxwlRJjLToCWdrPut607WjlMvmllx3Urwmfw0hKimwE8JrjtdUsOkZLLO4nvAV6reU8dxqkrTKq5EhutDJD0OPAa8AXwjVimVyTrrmg8S1sIcx2lgmnoXzcyeBT6ScmoqpTNZT068P7ro3Mer0zvHcWpFUyuuurKmDXspn9P0wE//J3czeZO1VuIwPePMX+SW2e2s/O2oAh9b5c8hS1u+PL20tefv2Carc4vQUcGvbZN7VuSW+d7rR+SqP3RYuX2r7gx6Lf9nNuyFNbllnswob9qpouM4b15ccTmO03S44nIcp+loGcUlqT3mSHww5mP8sqS2eG5t1mzHcZqfVlqcXxFzJCJpI0IqtPWA7xRlzXYcp8lpmRFXEjN7CTgeOEmBiZKuB5A0TNIlMdP1XEkHx/L3SbpH0kxJf5SUPzav4zh9QksqLgAze5KQTHajolPfBhab2S4xicdtkkYS8ji+x8x2I4zOvlR8zS6+ikvdV9Fx6kUrTRV7yntI5Hc0s9dikMEdgbtCmC8GAvcUC5rZFGAKwKAtNq/AKslxnGrQsopL0taEAIEvAW8rVx24xcwOr3nHHMfpNS05VZQ0ipAI9vzo05jkFuDERN0RwL+AdxcyYEsaKmm7vuqv4zj5aCXFNaRgDgH8A7gZODOl3v8AIyQ9IGkOsJ+ZvUzI93iFpLmEaWJW0EHHcepMy0wVzaxfiXNT6cyavRQ4KqXObcAeNeqe4zhVpGUUV1/TNrCdYVstziWz8nf5HKYBtprzWq76lWSYrsRheuYZ+R2zK+G19uXlKxUxKGfqgBOfy8qhks0980fnltnw6vzBJxeOHZJbZtY3L8hV/5gKMnkP6Zffy7yjbBDi7kz7W3p5K00VHcd5k+CKy3GcpsMVl+M4TUdLKa6Eo3Xh9bVYPjVmtJ4t6WFJxydkUrNh1+8uHMcpR6stzq91tE7hCDObIWkD4AlJlwKrCdmwf2FmB8aEG1MISTNO74sOO46Tn5YacfWQYcAyglV9ZjZsSetkX8JxnHrSaoprSNFUcVLi3OXRuPRR4LtRSeXKht3Fyfr1/Nv0juNUhzfjVHEUcLekG/NePOlkPWSbTd3J2nHqRKuNuMoS3XtmAu8gfzZsx3EagDed4oprV+OBJyiRDdvMfC7oOA1Kqymu4jWuHybOXS5pNmFN61Izu78H2bAdx2lAWmqNK8vR2swmlpDJyobtOE6D0lKKqy9pazOGDMzpaNotNFh5tKaCVM5522jgbYZ28nduNe256o8e8kruNh4cmt9hHvI7WfcF/dvyfV4Ag9ryO1mv7BiQWyaLVpsqOo7zJsAVl+M4TUdNFJckk/S7xHF/SS8XUoT18toTe3MdSXfHv6MlfbK3/XEcp++p1YhrGbCzpEIUtPcCz9eorVyY2bvi29GAKy7HaUJqOVW8AfhQfH84cEXhhKS3x+SrsyTdLWn7WP5FSRfH97vEuPCZPoOSNpB0XUzs+i9JY2P5KEm3SHpQ0q8kPRNzJyJpaRT/IbB3NJv4oqSdJN0Xj+dK2rbqn4jjOFWhlorrSuAwSYOBscC9iXOPAHub2XjgDOD7sfwcYBtJHwMuAU4oYwh6JjArJnb9BvCbWP4d4DYz2wm4mmAJX8zXgGlmNs7MzgY+B5wTXYYmAM/lvWHHcfqGmplDmNlcSaMJo63iQOjDgcviqMaAAVGmQ9LRwFzgl2Z2V5lm9gIOjrK3SdowuuzsRTAsxcxulNSTwO33AN+U9FbgGjN7vLhCjON1PMDAjdbrwSUdx6kFtd5V/AvwExLTxMh3gdvNbGeC8efgxLltgaXApjXuWxfM7PfAR4EVwA2S9k+pM8XMJpjZhP7DPeqN49SLWiuui4EzzWxeUflwOhfrjy4UShoOnAvsA2wo6RNlrj8NOCLKTgQWxrA0dwGHxvL3ASNSZJcA6yba3hp40szOBf5MmN46jtOA1FRxmdlzUREU87/ADyTNout09Wzg52b2GHAM8ENJG5VoYjKwe4yz9UM68yWeCbxP0gPAIcCLBEWVZC7QLmmOpC8SFN0D0Z9xZzrXyxzHaTBqssZlZsNSyqbSmZT1HiCZ4v5bsfyzifrPkhLMr+g6rwIHpXRhMfB+M1sj6Z3AHma2Mtk3M1tNiICa5Ic4jtPwtKqv4hbAHyS1AauA4+rcH8dxqkhLKq64Izi+lm20t7fx2pJ8C/SbLMrvMK0l+cKCLVidtpxXpo0K/LgryTBdicP0yH75HZOfW7O0fKUEDy55S+42Xl3UbVJRlo0rcLKvhLzPZlVHfjXQXkEm6yH9VuWWycJ9FR3HaTpccTmO03Q0neJKJH19QNJfJa3fQ7nZkq7sQb1xkj7Y6446jlMzmk5xETP5ROPVV4ETywlIehvQj+CbWG7RZBzgistxGphmVFxJ7gE2A5A0RtKNku6XNE3SDol6hwO/BW4GDiwUStojOnnPiQ7Ww4GzgEkpeRkdx2kQmnZXMWbkOQD4dSyaAnzOzB6X9A7gAjrttCYRQuvsAJwM/F7SQOAqYJKZTY8+jssJTt8TzOykvrsbx3Hy0IyKa0i0bt8MeBi4RdIw4F3AHyUV6g0CkDSB4Ao0X9LzwMWSNojyL5jZdFibwZqEfDeSTtb9Rw6v/p05jtMjmnGqWMhWvSUgwhpXG7Aorn0VXm+L9Q8HdpD0NCGX4nrEiBJ5STpZ91uvMRMfOM6bgWZUXADEOF2nAF8mTPGeknQIgAK7Rsv5Q4FdzGy0mY0mrHEdDjwKvEXSHlFmXUn9KXK+dhyn8WhaxQVgZrMIztKHE6JEHCNpDvAgQUHtDTxvZgsSYv8EdgQ2JKx9nRdlbiGE17kd2NEX5x2ncWm6Na5iB24zSyZz/UCKyJ5F9duBQlK8F4rPR/boTR8dx6ktTT3ichznzUnTjbgaBTNYs7pfLpmOgdk7llms2nJkrvrD+y3L3UZbBb6vg5T/q5M3wzTkd5gGeGv/fA7Qq9rz34sqSf9dYsc6i0qSP1fybPKyoj1/xwb0z//8s/ARl+M4TYcrLsdxmo6WVFySbpf0/qKy0yQ9Jelr8XhyzLi9TVEdi0arjuM0KC2puAhZhQ4rKjsMOMrMkuGZ5xXVO4RgSuE4TgPTqorrauBD0R+RmN9xU2CMpPMT9a4jOl1LGkOIVb+wT3vqOE5uWlJxxSQa9wH/FYsOA/4A3WIHvw48K2nnWOeqPuuk4zgV05KKK5KcLh5G96S0Ba6M5w8Cri11QUnHS5ohaUb7kvxmB47jVIdWVlx/Bg6QtBuwjpndn1HveuBIYH4hQkQWXZys13Una8epFy1rgGpmSyXdTsimnTXawsyWS/oq8Fifdc5xnF7RsoorcgVh+le8w9gFMysbi95xnMahpRWXmV1HiNlVOL4UuDS+n5whM7HmHXMcp1e0tOKqJf37dzBqgyW5ZDr65/M7BBjw+ILylRJcPH+v3G20tef3uzvxuQNyy4we8kpumUqSteb1Pbxu25tyt3HK0PwBRO633XLLDFha+2fzxOL838sNBudPCDxr6Vtzy2TRyovzjuO0KK64HMdpOlxxOY7TdLSU4ooO0r9LHPeX9LKk6+PxxpKuj3kUH5J0QywfLWlFDNf8kKQLY7x6x3EakFb7cS4DdpY0JB6/F3g+cf4s4BYz29XMdgS+ljj3RMweNJYQk/6g2nfXcZxKaDXFBXAD8KH4/nC6Gp++BXiucGBmc4uFzWwNcDewTfE5x3Eag1ZUXFcCh0kaTBg93Zs493Pg1zFe1zclbVosLGkdQobseSnn1voqrlmcfzvYcZzq0HKKK46iRhNGWzcUnbsJ2Bq4CNgBmCVpVDw9JmbIvgv4m5n9PeXaa30V+w9fp3Y34ThOSVrVAPUvwE+AiYT8iWuJIW9+D/w+LtrvA9xP5xqX4zgNTsuNuCIXA2eaWZfpnqT941QQSesCY4D5deif4zi9oCVHXGb2HHBuyqndgfMlrSEo7V+Z2fQYIdVxnCahpRRXcZbrWDYVmBrf/xj4cUqdp4Gda9s7x3GqRUsprr5kQFs7mwzN52T9ypoNy1cqZuSIXNXnL8iXDBVgk9W5Rbhn/ujcMg8O3SS3zKuL8t9P3mStlThMn7vp9Nwye1XgZK0KcqjmfTYrFw/O3cYLQ/N/adYsr566adU1LsdxWhhXXI7jNB2uuBzHaTrKKi5J7dH5+MHonPzlSh2QY6bomlhuSrpU0id6W8dxnManJwpohZmNM7OdCE7L/wV8p8L2TgNyKS5J/Spsy3GcFiXXyMnMXgKOB05SYLCkSyTNkzRL0n4QlI2kn0h6QNJcSSdLOoWQTfr2mH0HSYdH2Qck/ajQjqSlkn4qaQ7wzmQfJJ0haXqUmSJJFCHpaUn/G699n6Skw/Q+ku6W9GRh9CVpmKRbJc2MMgfm+Vwcx+lbck/5zOxJoB+wEXBiKLJdCL6Bl0Xn5uMJ/oLjzGwscLmZnQssAPYzs/2ig/OPgP2BccAekg6KzQwF7o3hZ+4s6sL5ZraHme0MDAE+nNHVxbFf5wM/S5S/Bdgryv0wlr0BfMzMdgP2A36aoRDXOlmvWrSi3EflOE6N6O3i/F7A7wDM7BHgGWA74D3AL2OImIJ/YDF7AFPN7OVY73KC3yBAO/CnjDb3k3SvpHkEpbdTRr0rEn+To7brzKzDzB4CNo5lAr4vaS7wD2CzxLm1JJ2sB64/pPi04zh9RG6LMElbExTLS9XvzlreMLP22N5NBCUyAzgJuACYYGbPSpoMZFnPWcb7lYn3hVHVEcAoYHczWy3p6RLXdRynzuQaccUQMBcSpmsGTCP86JG0HbAF8ChwC3CCpP7x3AbxEkuAdeP7+4B9JY2MC/CHA3cUt2lm74+bA8fSqUwWShoGlNohnJT4e0+ZWxsOvBSV1n7AlmXqO45TR3oy4hoS41QNANYAvwX+L567APhFnLatAY42s5WSfkWYMs6VtJoQ/+p8YApwo6QFcZ3ra8DthJHP38zsz6U6YmaLJF0EPAC8CJTyuxgRp34rCUqxFJcDf433MQN4pEx9x3HqSFnFZWaZ5ghm9gbwmZTyNcCX4itZfh5wXuL4CrqGVi6UZzqomdm3gG+llB9dVPRjM/tqqTqFdsxsIUW7l47jNC4KM77WIq5RTYgKqSYM22Bz2+W9p+aSyen7C4C1ddvcrDodFfi+tq2pfj9SqeT72X1DuOptVPIs7zz3l7ll9vnC8bll2gfm23MbsCy/J7fl/Yyhc0U5B3f+5b/vN7MJxeUtGR3CzEbXuw+O49QO91V0HKfpcMXlOE7T0RSKS+UzVB8dj2dJelzSTZLelai/ZzRanS3p4Wj/ldXWIEn/iHUnZdVzHKd+NMsa19oM1Wa2gu4ZqgGuMrOTAKIt1jWS9jOzh4HLgEPNbE60Gdu+RFvjATzjj+M0Lk0x4oqUylDdBTO7nWAzVtiS2Qh4IZ5rj+4+3ZC0EcGFaY844hpTpb47jlNFmklxlcpQncZMQtJXgLOBRyVdK+mEeI1uxOgXxwLTorX+E8nzSSfr1SuX9upmHMepnKZRXKUyVGew1mrEzM4CJgA3A58EbqywD2udrAcMyp/EwXGc6tA0iitSyFCdOU1MMB54uHBgZk+Y2S+AA4BdJVWQcsdxnEagWRbnC1wMLDKzeZImZlWStC9hfasQ2PBDwA3RMXxbQnSLRbXurOM4taGpFFeJDNUAkyTtRQgN/RRwcNxRBDgSOFvScoIz+BGFsDmO4zQfTaG4epCh+lLg0hLyh+Voa+11HcdpTJpCcTUiJujon89rdMTsV3K3s2LL9XPVXzh2QO42NrknfxjqhWMbNwJsR86PYMDSCpysKxivV+Iw/c8LpuSWGf+9L+Sqv2bL/EvdVsHq+IqNO/IL/SW9+E2ruCR9BigO73CXmZ1Yj/44jtNz3rSKy8wuAS6pdz8cx8lPs5lDOI7jNIbi6ksn6hJ9mFjU3vlVuDXHcWpAo0wV+9KJ2nGcJqchRlyRmjtRA0h6u6R74ujtbkmu5BynyWgkxVVzJ+rII8DeZjYeOAP4fk87mHSyXvPGsp6KOY5TZRpGcfWhE/Vw4I+SHiAovKxM2Gl9XOtk3X/w0J6KOY5TZRpGcUX6won6u8DtZrYz8BE8Y7XjNB2NsjhfoC+cqIfTufB/dDU67ThO39JQiquPnKj/F7hM0reAv1Wv947j9BUNobj62In6HmC7RNG38rbnOE59aQjF1Yx0jGhn+aTF+WT6j8zdzgZzXstVf/K5PdnX6Mr3Xj8it8ysb16QW6YSXmtfnltmkPJ9rU987oDcbdwzf3RumYFX59/QyeswDfmfzTHz98rdxpB+q3PLdFSQyjor93fLKi53onac1qVlFZc7UTtO69Jo5hCO4zhlaRrF1cfZrN3J2nEamGaaKrojtuM4QBONuCJ94ojtOE5j02yKq68csVNJOlm3v55/m95xnOrQVIqr3tmsk07W/dZbJ6+44zhVoqkUV8SzWTvOm5xmWpwv4NmsHedNTtMpLs9m7ThO0yiuPnbELnktx3HqS9MorkajY00bry/M5zTbvml+J9ORd72Rq/7pdx+Su42hw/L3qxLH3P5t+Qe4qzpq/xV9YnF+5/eVi/PHnxywLP/9V5JlOu+z+fUWd+Zu44nVS3PLPLI6/+f8pnOy7gnuiO04zcmbWnG5I7bjNCfNaA5RFkkHRd/GHeJxm6RzJT0gaZ6k6ZK2iueeljQyIbs2MazjOI1JSyougoHqnfEvwCRgU2Csme0CfAw3hXCcpqXlFJekYcBewDFAYSfxLcALZtYBwaTCzPKFFnUcp2FoxTWuA4EbzewxSa9I2h34A3CnpL2BW4HfmdmshMztkgpbPsMISWO7Iel4otN2vw3Xr1X/HccpQ8uNuAjTwyvj+yuBw6PR6vbA14EO4FZJyUDj+5nZODMbBxybdeEuvorDPCGs49SLlhpxSdoA2B/YRZIB/QCTdLqZrQT+Dvxd0n+AgwijL8dxmoxWG3F9AvitmW1pZqPNbHOC68/ekjaFsMNICInzTB376ThOL2ipERdhmvijorI/EaKfvippUCy7D/DQzI7TpLSU4jKz/VLKziXbKRszG110PJXo/+g4TmPSUoqrTxG5J9pDF1juZjoWvJirvvqPyN3GoNfy96uShKCD2vLLtFfQzor2AbnqbzA4fzTbF4bm75dpYH6ZChZz8j6bSvwOxwzoFvOgLE+vyf+ZZdFqa1yO47wJcMXlOE7T0ZKKS9I3JT0oaW7Mo/gOSVMlTUjUGS3pgfh+oqTFsW7h9Z763YHjOKVouTUuSe8EPgzsZmYrowN1TxYXppnZh2vbO8dxqkHLKS6CX+LCaHCKmS0EkPIHy3McpzFpRcV1M3CGpMeAfxCyW98Rz10uaUV8P5Dg/lNgb0mzE8cHm9kTNe+t4zi5aTnFZWZLo2P13oQMP1dJ+lo8fYSZzYCwxgUk426VnSq6k7XjNAYtp7gAYvaeqcBUSfOAo6p03SnAFIBBW701v/GT4zhVoeV2FSVtL2nbRNE43C/RcVqKllNchHhal0l6SNJcYEdgcg/k9i4yh/hETXvpOE7FtNxU0czuB96VcmpiUb2ngZ3j+6nA8Bp3zXGcKtGKIy7HcVqclhtx9RmrRf+X8jnzbjhjYe5m2t/IlxC2rS3/nsGwF9bklukgv13cyo58nxfAkH6rcssM6J8v8eqspW/N3caa5RX8dCowJVyxcUf5SkXkfTaVJGqtxGH6gCH5E+Jm4SMux3GaDldcjuM0HXVXXJLa4y7eA5L+Kmn9MvUnS/pKmToHSdoxcXyWO007TutQd8UFrIgZdnYGXgVOrMI1DyKYQQBgZmeY2T+qcF3HcRqARlBcSe4BNgOQNEbSjZLulzRN0g7FlSUdJ2m6pDmS/iRpHUnvAj4K/DiO5MZIurRglyXpAEmzJM2TdHEhDr2kpyWdKWlmPNetPcdxGoOGUVyS+gEHAH+JRVOAk81sd+ArwAUpYteY2R5mtivwMHCMmd0dr3F6HMmtdZSWNBi4FJhkZrsQdlU/n7jeQjPbDfhFbNNxnAakERTXkBiV4UVgY+AWScMIRqR/jOd+SQhXU8zOcTQ2DzgC2KlMW9sDT5nZY/H4MmCfxPlr4t/7gdHFwpKOlzRD0oyOZct6cm+O49SARlBcK2IG6S0Jli4nEvq1qJBdOr7eliJ7KXBSHD2dCQzuZV9Wxr/tpNi4JTNZtw31TNaOUy8aQXEBYGbLgVOALwPLgackHQKgwK4pYusCL0gaQBhxFVgSzxXzKDBa0jbx+EjgjpR6juM0MA2juADMbBYwl5DY9QjgGElzgAeBA1NEvg3cC9wFPJIovxI4PS7Cj0lc/w3gM4Qp6DxCIMELa3EvjuPUjrq7/JjZsKLjjyQOP5BSf3Li/S8IC+nFde4iYQ4BHJ04dyswPkVmdOL9DIqcsh3HaRwaasTlOI7TE2TmgTwrQdLLZAcoHAnk8ajOW7/VZBq1X30l06j9qkSm2m1saWajupWamb+q/AJm1LJ+q8k0ar/8/hv3Xnyq6DhO0+GKy3GcpsMVV22YUuP6rSbTqP3qK5lG7VclMn3SL1+cdxyn6fARl+M4TYcrLsdxmg5XXI7jNB2uuBzHaTpccTmO03T8Py/pzm5MyR+xAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# to plot the covariance matrix\n",
    "\n",
    "labels = ['GradNorm', 'CADet', 'Logits norm', 'ODIN', 'Dice', 'EBO',\n",
    "              'Max logits',\n",
    "              'ReAct',\n",
    "              'ViM', 'MSP', 'Doctor-alpha', 'MDS_f', 'MDS_all', 'MDS_l', 'ASH', 'SHE', 'Relation']\n",
    "\n",
    "fig = plt.figure()\n",
    "ax = fig.add_subplot(111)\n",
    "\n",
    "ax.matshow(cov)\n",
    "ax.set_xticks(range(len(labels)))\n",
    "ax.set_yticks(range(len(labels)))\n",
    "ax.set_xticklabels(labels,rotation=90)\n",
    "ax.set_yticklabels(labels)\n",
    "plt.savefig('vit-cov.pdf')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# To evaluate the GMM ensemble method. Specify below the methods, models and dataset to use.\n",
    "\n",
    "from sklearn.mixture import GaussianMixture\n",
    "import seaborn as sns\n",
    "from matplotlib import pyplot as plt\n",
    "import os\n",
    "import numpy as np\n",
    "\n",
    "vit_stats = ['gradnorm/gradnorm.npy', 'odin/odin.npy', 'dice/dice.npy', 'maxlogit/softmax.npy', 'mds-all', 'mds-last', 'maxlogit/maxlogit.npy', 'cadet/m_in.npy',\n",
    "             ]\n",
    "resnet_stats = ['gradnorm/gradnorm.npy', 'odin/odin.npy', 'react/react.npy', 'vim/vim.npy', 'doctor/alpha.npy', 'mds-all', 'mds-last', 'cadet/m_in.npy',]\n",
    "cheap_stats = ['dice/dice.npy', 'maxlogit/softmax.npy', 'mds-all', 'mds-last', 'maxlogit/maxlogit.npy', 'ebo/energy_scores.npy']\n",
    "all_stats = ['gradnorm/gradnorm.npy', 'cadet/m_in.npy', 'maxlogit/logitnorm.npy', 'odin/odin.npy', 'dice/dice.npy', 'ebo/energy_scores.npy',\n",
    "             'maxlogit/softmax.npy', 'doctor/alpha.npy', 'mds/layer_0.npy', 'mds-all', 'mds-last']\n",
    "\n",
    "id_train = 'clean_val_train'\n",
    "ood_datasets = ['clean_val_test', 'iNaturalist', 'openimage-o', 'imagenet-o', 'val_attack_LinfPGD_whitebox_0.050000_40_resnet', \n",
    "                'val_attack_LinfPGD_whitebox_0.050000_40_vit',\n",
    "                'val_attack_AA_whitebox_0.050000_Linf_vit', 'val_attack_AA_whitebox_0.050000_Linf_resnet',\n",
    "                'biggan_imagenet', 'synthetic_imagenet', \n",
    "                'defocus_blur', 'gaussian_noise', 'snow', 'brightness','coco-multiclass', 'coco-monoclass', 'imagenet-coco-classes']\n",
    "\n",
    "model = 'vit'\n",
    "stats = vit_stats\n",
    "n_components = 10\n",
    "train_only_correct = True\n",
    "\n",
    "id_correct = get_correct_dataset(model, id_train)\n",
    "if train_only_correct:\n",
    "    correct = id_correct\n",
    "else:\n",
    "    correct = None\n",
    "\n",
    "stat_scores = []\n",
    "stat_means = []\n",
    "stat_stds = []\n",
    "for stat in stats:\n",
    "    if stat == 'mds-all':\n",
    "        if model == 'vit':\n",
    "            _, score = get_scores_mds_combined(model, id_train, id_train, list(range(12)), correct=correct)\n",
    "        else:\n",
    "            _, score = get_scores_mds_combined(model, id_train, id_train, list(range(5)), correct=correct)\n",
    "    elif stat == 'mds-last':\n",
    "        if model == 'vit':\n",
    "            score = get_scores_dataset(model, id_train, 'mds/layer_11.npy')\n",
    "        else:\n",
    "            score = get_scores_dataset(model, id_train, 'mds/layer_4.npy')\n",
    "    else:\n",
    "        score = get_scores_dataset(model, id_train, stat)\n",
    "\n",
    "    score = np.array(score)\n",
    "    if train_only_correct:\n",
    "        score = score[id_correct]\n",
    "    stat_means.append(score.mean())\n",
    "    stat_stds.append(score.std())\n",
    "    score = (score - score.mean()) / score.std()\n",
    "    stat_scores.append(score)\n",
    "stat_scores = np.array(stat_scores)\n",
    "\n",
    "gm = GaussianMixture(n_components=n_components).fit(X=np.transpose(stat_scores))\n",
    "\n",
    "ood_gm_scores = []\n",
    "for dataset in ood_datasets:\n",
    "    ood_scores = []\n",
    "    for i, stat in enumerate(stats):\n",
    "        if stat == 'mds-all':\n",
    "            if model == 'vit':\n",
    "                _, score = get_scores_mds_combined(model, id_train, dataset, list(range(12)), correct=correct)\n",
    "            else:\n",
    "                _, score = get_scores_mds_combined(model, id_train, dataset, list(range(5)), correct=correct)\n",
    "        elif stat == 'mds-last':\n",
    "            if model == 'vit':\n",
    "                score = get_scores_dataset(model, dataset, 'mds/layer_11.npy')\n",
    "            else:\n",
    "                score = get_scores_dataset(model, dataset, 'mds/layer_4.npy')\n",
    "        else:\n",
    "            score = get_scores_dataset(model, dataset, stat)\n",
    "        score = np.array(score)\n",
    "        score = (score - stat_means[i]) / stat_stds[i]\n",
    "        ood_scores.append(score)\n",
    "    ood_scores = np.array(ood_scores)\n",
    "    \n",
    "    gm_log_llh = gm.score_samples(X=np.transpose(ood_scores))\n",
    "    ood_gm_scores.append(gm_log_llh)\n",
    "\n",
    "total_auc = []\n",
    "for i, dataset in enumerate(ood_datasets[:-1]):\n",
    "    if i == 0:\n",
    "        continue\n",
    "    auc = get_auc(ood_gm_scores[0], ood_gm_scores[i])\n",
    "    print('AUC for dataset ' + dataset + ': ' + str(auc) + '\\n')\n",
    "    if dataset in ['iNaturalist', 'openimage-o', 'imagenet-o']:\n",
    "        total_auc.append(auc/3)\n",
    "    if dataset in ['val_attack_LinfPGD_whitebox_0.050000_40_resnet', \n",
    "                'val_attack_LinfPGD_whitebox_0.050000_40_vit',\n",
    "                'val_attack_AA_whitebox_0.050000_Linf_vit', 'val_attack_AA_whitebox_0.050000_Linf_resnet']:\n",
    "        total_auc.append(auc/4)\n",
    "    if dataset in ['biggan_imagenet', 'synthetic_imagenet']:\n",
    "        total_auc.append(auc/2)\n",
    "    if dataset in ['defocus_blur', 'gaussian_noise', 'snow', 'brightness']:\n",
    "        total_auc.append(auc/4)\n",
    "    if dataset in ['coco-multiclass']:\n",
    "        total_auc.append(auc)\n",
    "print('total auc: ' + str(np.sum(total_auc) / 5))\n",
    "\n",
    "\n",
    "test_id_correct = get_correct_dataset(model, 'clean_val_test')\n",
    "id_gm_score = ood_gm_scores[0][test_id_correct]\n",
    "ood_gm_scores_correct = []\n",
    "for dataset in ood_datasets:\n",
    "    if dataset in ['iNaturalist', 'openimage-o', 'imagenet-o', 'biggan_imagenet', 'synthetic_imagenet']:\n",
    "        ood_gm_scores_correct.append(None)\n",
    "        continue\n",
    "    ood_scores = []\n",
    "    ood_correct = get_correct_dataset(model, dataset)\n",
    "    for i, stat in enumerate(stats):\n",
    "        if stat == 'mds-all':\n",
    "            if model == 'vit':\n",
    "                _, score = get_scores_mds_combined(model, id_train, dataset, list(range(12)), correct=correct)\n",
    "            else:\n",
    "                _, score = get_scores_mds_combined(model, id_train, dataset, list(range(5)), correct=correct)\n",
    "        elif stat == 'mds-last':\n",
    "            if model == 'vit':\n",
    "                score = get_scores_dataset(model, dataset, 'mds/layer_11.npy')\n",
    "            else:\n",
    "                score = get_scores_dataset(model, dataset, 'mds/layer_4.npy')\n",
    "        else:\n",
    "            score = get_scores_dataset(model, dataset, stat)\n",
    "        score = np.array(score)\n",
    "        score = (score - stat_means[i]) / stat_stds[i]\n",
    "        score = score[np.invert(ood_correct)]\n",
    "        ood_scores.append(score)\n",
    "    ood_scores = np.array(ood_scores)\n",
    "    \n",
    "    gm_log_llh = gm.score_samples(X=np.transpose(ood_scores))\n",
    "    ood_gm_scores_correct.append(gm_log_llh)\n",
    "\n",
    "for i, dataset in enumerate(ood_datasets[:-1]):\n",
    "    if dataset in ['iNaturalist', 'openimage-o', 'imagenet-o', 'biggan_imagenet', 'synthetic_imagenet']:\n",
    "        continue\n",
    "    auc = get_auc(id_gm_score, ood_gm_scores_correct[i])\n",
    "    print('AUC for dataset CORRECT' + dataset + ': ' + str(auc) + '\\n')"
   ]
  }
 ],
 "metadata": {
  "interpreter": {
   "hash": "31c703b0062de80b194fe46bf23e8915293999ff8f84ef8abedaea965e59484c"
  },
  "kernelspec": {
   "display_name": "Python 3.9.5 ('miniconda': virtualenv)",
   "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.9.5"
  },
  "orig_nbformat": 4
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
