{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Imports"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1779/2162111138.py:25: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
      "  from tqdm.autonotebook import tqdm\n"
     ]
    }
   ],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "%matplotlib inline\n",
    "import os\n",
    "os.environ['CUDA_DEVICE_ORDER']='PCI_BUS_ID'\n",
    "os.environ['CUDA_VISIBLE_DEVICES']='0'\n",
    "import variational\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "from matplotlib.ticker import FuncFormatter\n",
    "from itertools import cycle\n",
    "import os\n",
    "import time\n",
    "import math\n",
    "import pandas as pd\n",
    "from collections import OrderedDict\n",
    "from sklearn.linear_model import LogisticRegression\n",
    "from sklearn.metrics import confusion_matrix\n",
    "    \n",
    "import copy\n",
    "import torch.nn as nn\n",
    "from torch.autograd import Variable\n",
    "from typing import List\n",
    "import itertools\n",
    "from tqdm.autonotebook import tqdm\n",
    "from models import *\n",
    "import models\n",
    "from logger import *\n",
    "\n",
    "from thirdparty.repdistiller.helper.util import adjust_learning_rate as sgda_adjust_learning_rate\n",
    "from thirdparty.repdistiller.distiller_zoo import DistillKL, HintLoss, Attention, Similarity, Correlation, VIDLoss, RKDLoss\n",
    "from thirdparty.repdistiller.distiller_zoo import PKT, ABLoss, FactorTransfer, KDSVD, FSP, NSTLoss\n",
    "\n",
    "from thirdparty.repdistiller.helper.loops import train_distill, train_distill_hide, train_distill_linear, train_vanilla, train_negrad, train_bcu, train_bcu_distill, validate\n",
    "from thirdparty.repdistiller.helper.pretrain import init"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Metrics1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "from utils import *\n",
    "def get_metrics(model,dataloader,criterion,samples_correctness=False,use_bn=False,delta_w=None,scrub_act=False):\n",
    "    activations=[]\n",
    "    predictions=[]\n",
    "    if use_bn:\n",
    "        model.train()\n",
    "        dataloader = torch.utils.data.DataLoader(retain_loader.dataset, batch_size=128, shuffle=True)\n",
    "        for i in range(10):\n",
    "            for batch_idx, (data, target) in enumerate(dataloader):\n",
    "                data, target = data.to(args.device), target.to(args.device)            \n",
    "                output = model(data)\n",
    "    dataloader = torch.utils.data.DataLoader(dataloader.dataset, batch_size=1, shuffle=False)\n",
    "    model.eval()\n",
    "    metrics = AverageMeter()\n",
    "    mult = 0.5 if args.lossfn=='mse' else 1\n",
    "    for batch_idx, (data, target) in enumerate(dataloader):\n",
    "        data, target = data.to(args.device), target.to(args.device)            \n",
    "        if args.lossfn=='mse':\n",
    "            target=(2*target-1)\n",
    "            target = target.type(torch.cuda.FloatTensor).unsqueeze(1)\n",
    "        if 'mnist' in args.dataset:\n",
    "            data=data.view(data.shape[0],-1)\n",
    "        output = model(data)\n",
    "        loss = mult*criterion(output, target)\n",
    "        if samples_correctness:\n",
    "            activations.append(torch.nn.functional.softmax(output,dim=1).cpu().detach().numpy().squeeze())\n",
    "            predictions.append(get_error(output,target))\n",
    "        metrics.update(n=data.size(0), loss=loss.item(), error=get_error(output, target))\n",
    "    if samples_correctness:\n",
    "        return metrics.avg,np.stack(activations),np.array(predictions)\n",
    "    else:\n",
    "        return metrics.avg\n",
    "\n",
    "def activations_predictions(model,dataloader,name):\n",
    "    criterion = torch.nn.CrossEntropyLoss()\n",
    "    metrics,activations,predictions=get_metrics(model,dataloader,criterion,True)\n",
    "    print(f\"{name} -> Loss:{np.round(metrics['loss'],3)}, Error:{metrics['error']}\")\n",
    "    log_dict[f\"{name}_loss\"]=metrics['loss']\n",
    "    log_dict[f\"{name}_error\"]=metrics['error']\n",
    "\n",
    "    return activations,predictions\n",
    "\n",
    "def predictions_distance(l1,l2,name):\n",
    "    dist = np.sum(np.abs(l1-l2))\n",
    "    print(f\"Predictions Distance {name} -> {dist}\")\n",
    "    log_dict[f\"{name}_predictions\"]=dist\n",
    "\n",
    "def activations_distance(a1,a2,name):\n",
    "    dist = np.linalg.norm(a1-a2,ord=1,axis=1).mean()\n",
    "    print(f\"Activations Distance {name} -> {dist}\")\n",
    "    log_dict[f\"{name}_activations\"]=dist\n",
    "\n",
    "def interclass_confusion(model, dataloader, class_to_forget, name):\n",
    "    criterion = torch.nn.CrossEntropyLoss()\n",
    "    dataloader = torch.utils.data.DataLoader(dataloader.dataset, batch_size=128, shuffle=False)\n",
    "    model.eval()\n",
    "    reals=[]\n",
    "    predicts=[]\n",
    "    for batch_idx, (data, target) in enumerate(dataloader):\n",
    "        data, target = data.to(args.device), target.to(args.device) \n",
    "        if 'mnist' in args.dataset:\n",
    "            data=data.view(data.shape[0],-1)\n",
    "        output = model(data)\n",
    "        probs = torch.nn.functional.softmax(output, dim=1)\n",
    "        predict = np.argmax(probs.cpu().detach().numpy(),axis=1)\n",
    "        reals = reals + list(target.cpu().detach().numpy())\n",
    "        predicts = predicts + list(predict)\n",
    "    \n",
    "    classes = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]\n",
    "    cm = confusion_matrix(reals, predicts, labels=classes)\n",
    "    counts = 0\n",
    "    for i in range(len(cm)):\n",
    "        if i != class_to_forget[0]:\n",
    "            counts += cm[class_to_forget[0]][i]\n",
    "        if i != class_to_forget[1]:\n",
    "            counts += cm[class_to_forget[1]][i]\n",
    "    \n",
    "    ic_err = counts / (np.sum(cm[class_to_forget[0]]) + np.sum(cm[class_to_forget[1]]))\n",
    "    fgt = cm[class_to_forget[0]][class_to_forget[1]] + cm[class_to_forget[1]][class_to_forget[0]]\n",
    "    #print (cm)\n",
    "    return ic_err, fgt"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Helper and Utils"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "from utils import *\n",
    "def get_metrics(model,dataloader,criterion,samples_correctness=False,use_bn=False,delta_w=None,scrub_act=False):\n",
    "    activations=[]\n",
    "    predictions=[]\n",
    "    if use_bn:\n",
    "        model.train()\n",
    "        dataloader = torch.utils.data.DataLoader(retain_loader.dataset, batch_size=128, shuffle=True)\n",
    "        for i in range(10):\n",
    "            for batch_idx, (data, target) in enumerate(dataloader):\n",
    "                data, target = data.to(args.device), target.to(args.device)            \n",
    "                output = model(data)\n",
    "    dataloader = torch.utils.data.DataLoader(dataloader.dataset, batch_size=1, shuffle=False)\n",
    "    model.eval()\n",
    "    metrics = AverageMeter()\n",
    "    mult = 0.5 if args.lossfn=='mse' else 1\n",
    "    for batch_idx, (data, target) in enumerate(dataloader):\n",
    "        data, target = data.to(args.device), target.to(args.device)            \n",
    "        if args.lossfn=='mse':\n",
    "            target=(2*target-1)\n",
    "            target = target.type(torch.cuda.FloatTensor).unsqueeze(1)\n",
    "        if 'mnist' in args.dataset:\n",
    "            data=data.view(data.shape[0],-1)\n",
    "        output = model(data)\n",
    "        if scrub_act:\n",
    "            G = []\n",
    "            for cls in range(num_classes):\n",
    "                grads = torch.autograd.grad(output[0,cls],model.parameters(),retain_graph=True)\n",
    "                grads = torch.cat([g.view(-1) for g in grads])\n",
    "                G.append(grads)\n",
    "            grads = torch.autograd.grad(output_sf[0,cls],model_scrubf.parameters(),retain_graph=False)\n",
    "            G = torch.stack(G).pow(2)\n",
    "            delta_f = torch.matmul(G,delta_w)\n",
    "            output += delta_f.sqrt()*torch.empty_like(delta_f).normal_()\n",
    "\n",
    "        loss = mult*criterion(output, target)\n",
    "        if samples_correctness:\n",
    "            activations.append(torch.nn.functional.softmax(output,dim=1).cpu().detach().numpy().squeeze())\n",
    "            predictions.append(get_error(output,target))\n",
    "        metrics.update(n=data.size(0), loss=loss.item(), error=get_error(output, target))\n",
    "    if samples_correctness:\n",
    "        return metrics.avg,np.stack(activations),np.array(predictions)\n",
    "    else:\n",
    "        return metrics.avg\n",
    "\n",
    "def l2_penalty(model,model_init,weight_decay):\n",
    "    l2_loss = 0\n",
    "    for (k,p),(k_init,p_init) in zip(model.named_parameters(),model_init.named_parameters()):\n",
    "        if p.requires_grad:\n",
    "            l2_loss += (p-p_init).pow(2).sum()\n",
    "    l2_loss *= (weight_decay/2.)\n",
    "    return l2_loss\n",
    "\n",
    "def run_train_epoch(model: nn.Module, model_init, data_loader: torch.utils.data.DataLoader, \n",
    "                    loss_fn: nn.Module,\n",
    "                    optimizer: torch.optim.SGD, split: str, epoch: int, ignore_index=None,\n",
    "                    negative_gradient=False, negative_multiplier=-1, random_labels=False,\n",
    "                    quiet=False,delta_w=None,scrub_act=False):\n",
    "    model.eval()\n",
    "    metrics = AverageMeter()    \n",
    "    num_labels = data_loader.dataset.targets.max().item() + 1\n",
    "    \n",
    "    with torch.set_grad_enabled(split != 'test'):\n",
    "        for idx, batch in enumerate(tqdm(data_loader, leave=False)):\n",
    "            batch = [tensor.to(next(model.parameters()).device) for tensor in batch]\n",
    "            input, target = batch\n",
    "            output = model(input)\n",
    "            if split=='test' and scrub_act:\n",
    "                G = []\n",
    "                for cls in range(num_classes):\n",
    "                    grads = torch.autograd.grad(output[0,cls],model.parameters(),retain_graph=True)\n",
    "                    grads = torch.cat([g.view(-1) for g in grads])\n",
    "                    G.append(grads)\n",
    "                grads = torch.autograd.grad(output_sf[0,cls],model_scrubf.parameters(),retain_graph=False)\n",
    "                G = torch.stack(G).pow(2)\n",
    "                delta_f = torch.matmul(G,delta_w)\n",
    "                output += delta_f.sqrt()*torch.empty_like(delta_f).normal_()\n",
    "            loss = loss_fn(output, target) + l2_penalty(model,model_init,args.weight_decay)\n",
    "            metrics.update(n=input.size(0), loss=loss_fn(output,target).item(), error=get_error(output, target))\n",
    "            \n",
    "            if split != 'test':\n",
    "                model.zero_grad()\n",
    "                loss.backward()\n",
    "                optimizer.step()\n",
    "    if not quiet:\n",
    "        log_metrics(split, metrics, epoch)\n",
    "    return metrics.avg\n",
    "\n",
    "def run_neggrad_epoch(model: nn.Module, model_init, data_loader: torch.utils.data.DataLoader, \n",
    "                    forget_loader: torch.utils.data.DataLoader,\n",
    "                    alpha: float,\n",
    "                    loss_fn: nn.Module,\n",
    "                    optimizer: torch.optim.SGD, split: str, epoch: int, ignore_index=None,\n",
    "                    quiet=False):\n",
    "    model.eval()\n",
    "    metrics = AverageMeter()    \n",
    "    num_labels = data_loader.dataset.targets.max().item() + 1\n",
    "    \n",
    "    with torch.set_grad_enabled(split != 'test'):\n",
    "        for idx, (batch_retain,batch_forget) in enumerate(tqdm(zip(data_loader,cycle(forget_loader)), leave=False)):\n",
    "            batch_retain = [tensor.to(next(model.parameters()).device) for tensor in batch_retain]\n",
    "            batch_forget = [tensor.to(next(model.parameters()).device) for tensor in batch_forget]\n",
    "            input_r, target_r = batch_retain\n",
    "            input_f, target_f = batch_forget\n",
    "            output_r = model(input_r)\n",
    "            output_f = model(input_f)\n",
    "            loss = alpha*(loss_fn(output_r, target_r) + l2_penalty(model,model_init,args.weight_decay)) - (1-alpha)*loss_fn(output_f, target_f)\n",
    "            metrics.update(n=input_r.size(0), loss=loss_fn(output_r,target_r).item(), error=get_error(output_r, target_r))\n",
    "            if split != 'test':\n",
    "                model.zero_grad()\n",
    "                loss.backward()\n",
    "                optimizer.step()\n",
    "    if not quiet:\n",
    "        log_metrics(split, metrics, epoch)\n",
    "    return metrics.avg"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "def test(model, data_loader):\n",
    "    loss_fn = nn.CrossEntropyLoss()\n",
    "    model_init=copy.deepcopy(model)\n",
    "    return run_train_epoch(model, model_init, data_loader, loss_fn, optimizer=None, split='test', epoch=epoch, ignore_index=None, quiet=True)\n",
    "\n",
    "def readout_retrain(model, data_loader, test_loader, lr=0.1, epochs=500, threshold=0.01, quiet=True):\n",
    "    torch.manual_seed(seed)\n",
    "    model = copy.deepcopy(model)\n",
    "    loss_fn = nn.CrossEntropyLoss()\n",
    "    optimizer = torch.optim.SGD(model.parameters(), lr=lr, weight_decay=0.0)\n",
    "    sampler = torch.utils.data.RandomSampler(data_loader.dataset, replacement=True, num_samples=500)\n",
    "    data_loader_small = torch.utils.data.DataLoader(data_loader.dataset, batch_size=data_loader.batch_size, sampler=sampler, num_workers=data_loader.num_workers)\n",
    "    metrics = []\n",
    "    model_init=copy.deepcopy(model)\n",
    "    for epoch in range(epochs):\n",
    "        metrics.append(run_train_epoch(model, model_init, test_loader, loss_fn, optimizer, split='test', epoch=epoch, ignore_index=None, quiet=quiet))\n",
    "        if metrics[-1]['loss'] <= threshold:\n",
    "            break\n",
    "        run_train_epoch(model, model_init, data_loader_small, loss_fn, optimizer, split='train', epoch=epoch, ignore_index=None, quiet=quiet)\n",
    "    return epoch, metrics\n",
    "\n",
    "def extract_retrain_time(metrics, threshold=0.1):\n",
    "    losses = np.array([m['loss'] for m in metrics])\n",
    "    return np.argmax(losses < threshold)\n",
    "\n",
    "def all_readouts(test_loader, retain_loader, forget_loader, model, wandb=None,thresh=0.1,name='method'):\n",
    "    #train_loader = torch.utils.data.DataLoader(train_loader_full.dataset, batch_size=args.batch_size, shuffle=True)\n",
    "    #retrain_time, _ = readout_retrain(model, train_loader, forget_loader, epochs=100, lr=0.1, threshold=thresh)\n",
    "    test_error = test(model, test_loader)['error']\n",
    "    forget_error = test(model, forget_loader)['error']\n",
    "    retain_error = test(model, retain_loader)['error']\n",
    "    ic_err_test, fgt_test = interclass_confusion(model, test_loader, class_to_forget, name)\n",
    "    ic_err_retain, fgt_retain = interclass_confusion(model, retain_loader, class_to_forget, name)\n",
    "    print(f\"{name} ->\"\n",
    "          f\"\\ttest: {test_error:.2%}\"\n",
    "          f\"\\tForget: {forget_error:.2%}\\tRetain: {retain_error:.2%}\"\n",
    "          f\"\\tIC-test: {ic_err_test}\\tfgt-test: {fgt_test}\"\n",
    "          f\"\\tIC-retain: {ic_err_retain}\\tfgt-retain: {fgt_retain}\")\n",
    "    #log_dict[f\"{name}_retrain_time\"]=retrain_time+1\n",
    "    if wandb is not None:\n",
    "        #wandb.log({f\"{name}_RLT\": retrain_time + 1})\n",
    "        wandb.log({f\"{name}_test_error\": test_error})\n",
    "        wandb.log({f\"{name}_retain_error\": retain_error})\n",
    "        wandb.log({f\"{name}_forget_error\": forget_error})\n",
    "        wandb.log({f\"{name}_IC_err_test\": ic_err_test})\n",
    "        wandb.log({f\"{name}_IC_err_retain\": ic_err_retain})\n",
    "        wandb.log({f\"{name}_fgt_test\": fgt_test})\n",
    "        wandb.log({f\"{name}_fgt_retain\": fgt_retain})\n",
    "    return(dict(test_error=test_error, forget_error=forget_error, retain_error=retain_error))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Forgetting helper functions"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def finetune(model: nn.Module, data_loader: torch.utils.data.DataLoader, lr=0.01, epochs=10, quiet=False):\n",
    "    loss_fn = nn.CrossEntropyLoss()\n",
    "    optimizer = torch.optim.SGD(model.parameters(), lr=lr, weight_decay=0.0)\n",
    "    model_init=copy.deepcopy(model)\n",
    "    for epoch in range(epochs):\n",
    "        run_train_epoch(model, model_init, data_loader, loss_fn, optimizer, split='train', epoch=epoch, ignore_index=None, quiet=quiet)\n",
    "        #train_vanilla(epoch, data_loader, model, loss_fn, optimizer, args)\n",
    "\n",
    "def negative_grad(model: nn.Module, data_loader: torch.utils.data.DataLoader, forget_loader: torch.utils.data.DataLoader, alpha: float, lr=0.01, epochs=10, quiet=False, args=None):\n",
    "    loss_fn = nn.CrossEntropyLoss()\n",
    "    optimizer = torch.optim.SGD(model.parameters(), lr=lr, weight_decay=0.0)\n",
    "    model_init=copy.deepcopy(model)\n",
    "    for epoch in range(epochs):\n",
    "        #run_neggrad_epoch(model, model_init, data_loader, forget_loader, alpha, loss_fn, optimizer, split='train', epoch=epoch, ignore_index=None, quiet=quiet)\n",
    "        train_negrad(epoch, data_loader, forget_loader, model, loss_fn, optimizer,  alpha, args)\n",
    "\n",
    "def fk_fientune(model: nn.Module, data_loader: torch.utils.data.DataLoader, args, lr=0.01, epochs=10, quiet=False):\n",
    "    loss_fn = nn.CrossEntropyLoss()\n",
    "    optimizer = torch.optim.SGD(model.parameters(), lr=lr, weight_decay=0.0005)\n",
    "    model_init=copy.deepcopy(model)\n",
    "    for epoch in range(epochs):\n",
    "        sgda_adjust_learning_rate(epoch, args, optimizer)\n",
    "        #train_vanilla(epoch, data_loader, model, loss_fn, optimizer, args)\n",
    "        run_train_epoch(model, model_init, data_loader, loss_fn, optimizer, split='train', epoch=epoch, ignore_index=None, quiet=quiet)\n",
    "\n",
    "def pdb():\n",
    "    import pdb\n",
    "    pdb.set_trace\n",
    "    \n",
    "def parameter_count(model):\n",
    "    count=0\n",
    "    for p in model.parameters():\n",
    "        count+=np.prod(np.array(list(p.shape)))\n",
    "    print(f'Total Number of Parameters: {count}')\n",
    "    \n",
    "def vectorize_params(model):\n",
    "    param = []\n",
    "    for p in model.parameters():\n",
    "        param.append(p.data.view(-1).cpu().numpy())\n",
    "    return np.concatenate(param)\n",
    "\n",
    "def print_param_shape(model):\n",
    "    for k,p in model.named_parameters():\n",
    "        print(k,p.shape)\n",
    "\n",
    "def distance(model,model0):\n",
    "    distance=0\n",
    "    normalization=0\n",
    "    for (k, p), (k0, p0) in zip(model.named_parameters(), model0.named_parameters()):\n",
    "        space='  ' if 'bias' in k else ''\n",
    "        current_dist=(p.data0-p0.data0).pow(2).sum().item()\n",
    "        current_norm=p.data0.pow(2).sum().item()\n",
    "        distance+=current_dist\n",
    "        normalization+=current_norm\n",
    "    print(f'Distance: {np.sqrt(distance)}')\n",
    "    print(f'Normalized Distance: {1.0*np.sqrt(distance/normalization)}')\n",
    "    return 1.0*np.sqrt(distance/normalization)\n",
    "\n",
    "def ntk_init(resume,seed=1):\n",
    "    manual_seed(seed)\n",
    "    model_init = models.get_model(arch, num_classes=num_classes, filters_percentage=filters).to(args.device)\n",
    "    model_init.load_state_dict(torch.load(resume))\n",
    "    return model_init"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Pre-training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checkpoint name: cifar100_resnet_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1\n",
      "[Logging in cifar100_resnet_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1_training]\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "confuse mode: False\n",
      "split mode: None\n",
      "Number of Classes: 100\n",
      "[0] train metrics:{\"loss\": 4.110220985412598, \"error\": 0.9239}\n",
      "Learning Rate : 0.1\n",
      "[0] dry_run metrics:{\"loss\": 3.7692216533660887, \"error\": 0.87745}\n",
      "Learning Rate : 0.1\n",
      "[0] test metrics:{\"loss\": 3.779144535446167, \"error\": 0.8744}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 42.82 sec\n",
      "[1] train metrics:{\"loss\": 3.6166512355804445, \"error\": 0.84515}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.37 sec\n",
      "[2] train metrics:{\"loss\": 3.262185221862793, \"error\": 0.767475}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.25 sec\n",
      "[3] train metrics:{\"loss\": 2.9241571712493895, \"error\": 0.6889}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.38 sec\n",
      "[4] train metrics:{\"loss\": 2.6573989303588865, \"error\": 0.61595}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.36 sec\n",
      "[5] train metrics:{\"loss\": 2.456750422286987, \"error\": 0.55295}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.53 sec\n",
      "[6] train metrics:{\"loss\": 2.316490512084961, \"error\": 0.494525}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.36 sec\n",
      "[7] train metrics:{\"loss\": 2.2176277866363527, \"error\": 0.44985}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.4 sec\n",
      "[8] train metrics:{\"loss\": 2.1540924285888674, \"error\": 0.404375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.33 sec\n",
      "[9] train metrics:{\"loss\": 2.1174500984191895, \"error\": 0.37015}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.39 sec\n",
      "[10] train metrics:{\"loss\": 2.0962327156066896, \"error\": 0.33205}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.38 sec\n",
      "[11] train metrics:{\"loss\": 2.089509313583374, \"error\": 0.2955}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.4 sec\n",
      "[12] train metrics:{\"loss\": 2.1187580627441407, \"error\": 0.27275}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.43 sec\n",
      "[13] train metrics:{\"loss\": 2.1520710746765137, \"error\": 0.2503}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.34 sec\n",
      "[14] train metrics:{\"loss\": 2.1659040351867676, \"error\": 0.222975}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 25.96 sec\n",
      "[15] train metrics:{\"loss\": 2.180283085632324, \"error\": 0.202875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.4 sec\n",
      "[16] train metrics:{\"loss\": 2.237203318405151, \"error\": 0.195875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 25.99 sec\n",
      "[17] train metrics:{\"loss\": 2.256527083206177, \"error\": 0.18455}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 26.32 sec\n",
      "[18] train metrics:{\"loss\": 2.3006882884979247, \"error\": 0.180475}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 24.7 sec\n",
      "[19] train metrics:{\"loss\": 2.2771842372894286, \"error\": 0.164375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 22.87 sec\n",
      "[20] train metrics:{\"loss\": 2.332470198059082, \"error\": 0.1652}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 22.08 sec\n",
      "[21] train metrics:{\"loss\": 2.330265349960327, \"error\": 0.1593}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 22.87 sec\n",
      "[22] train metrics:{\"loss\": 2.4015751735687254, \"error\": 0.164125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 18.48 sec\n",
      "[23] train metrics:{\"loss\": 2.3445411685943602, \"error\": 0.1472}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 17.01 sec\n",
      "[24] train metrics:{\"loss\": 2.3646815967559816, \"error\": 0.151725}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 17.2 sec\n",
      "[25] train metrics:{\"loss\": 2.3665282775878906, \"error\": 0.14435}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 17.15 sec\n",
      "[26] train metrics:{\"loss\": 2.345229354476929, \"error\": 0.14115}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 16.89 sec\n",
      "[27] train metrics:{\"loss\": 2.3579494285583498, \"error\": 0.142225}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 16.91 sec\n",
      "[28] train metrics:{\"loss\": 2.3899746318817137, \"error\": 0.1456}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 16.84 sec\n",
      "[29] train metrics:{\"loss\": 2.4067670585632324, \"error\": 0.145975}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 16.97 sec\n",
      "[30] train metrics:{\"loss\": 2.4091349899291994, \"error\": 0.14255}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 17.28 sec\n",
      "Pure training time: 721.0800000000002 sec\n"
     ]
    }
   ],
   "source": [
    "%run main.py --dataset cifar100 --dataroot=data/cifar-100-python/ --model resnet --filters 1.0 --lr 0.1 --lossfn ce --num-classes 100"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Original"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "#%run main.py --dataset small_lacuna6 --model allcnn --dataroot=data/lacuna10/ --filters 1.0 --lr 0.001 \\\n",
    "#--resume checkpoints/lacuna100_allcnn_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1_30.pt --disable-bn \\\n",
    "#--weight-decay 0.1 --batch-size 128 --epochs 31 --seed 3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checkpoint name: cifar10_resnet_1_0_forget_None_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1\n",
      "[Logging in cifar10_resnet_1_0_forget_None_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1_training]\n",
      "Downloading https://www.cs.toronto.edu/~kriz/cifar-10-python.tar.gz to data/cifar10-batches-py/cifar-10-python.tar.gz\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 170498071/170498071 [00:18<00:00, 9004553.75it/s] \n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Extracting data/cifar10-batches-py/cifar-10-python.tar.gz to data/cifar10-batches-py/\n",
      "Files already downloaded and verified\n",
      "confuse mode: True\n",
      "split mode: train\n",
      "[22462  6646 36135 23850  1434  5306 18141  4049 11418 10150 21959  2340\n",
      " 28237 23045 33372 17719 28726 24235 36477 30535 34830 13742  9505  9925\n",
      "  8968  5340 32206 23111 36876 37753 18138   346  9984 33198 10884 17838\n",
      "  5290 34800  8839 26496 22062 19691 25490 32745 29623 37043 38544 16174\n",
      " 26055 17790 13062  3564 33467 19387 12727 10616  5925 38573 20823 25972\n",
      "  2232 39091 16018 31897  6216 31382  9265  4834 15220 13636 20376 17239\n",
      "  3784 30964 11139 32269 25536 37777  9464 31770 12377 14878 22419 35384\n",
      " 33593 13768  8813 14913  1918  3157 22036 26124 26696 32374   359 18794\n",
      " 37201 12935 39835  6800 34930 21153  1332   444 26453 22013  2780 28519\n",
      " 32137 33713  9783 18691  5872  5426 26608  5517 26355 34576 15284  9727\n",
      " 11908 16177  5566 33238 31572 31990  6716 36313 26521  8618 25781 10358\n",
      "  9601   341  6301 36220 15826  3522  4898 31528 28091 12132 15300  5732\n",
      " 10974 23654 30805 26574  5750  7350 19049 24064  7229  8594 31711 15489\n",
      " 21523 27315  8146 28736 36469 37611 32999 39736 27867 33821 34312  3036\n",
      " 11154 19416 37108 22615 36649 11823  7061 36430 18557 37081  8899 29319\n",
      "  9990 26523 33672  7326 39648 21642   132  4050  6683  5979 20121 36524\n",
      " 22540 33421 27229 13730 29155 25346  2670   732]\n",
      "Number of Classes: 10\n",
      "[0] train metrics:{\"loss\": 0.8994579604148865, \"error\": 0.31465}\n",
      "Learning Rate : 0.01\n",
      "[0] test metrics:{\"loss\": 0.6736513387680054, \"error\": 0.2306}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 15.81 sec\n",
      "[1] train metrics:{\"loss\": 0.5806631664276123, \"error\": 0.19635}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 13.82 sec\n",
      "[2] train metrics:{\"loss\": 0.44068201198577883, \"error\": 0.14615}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 13.76 sec\n",
      "[3] train metrics:{\"loss\": 0.3294310111761093, \"error\": 0.105}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 11.56 sec\n",
      "[4] train metrics:{\"loss\": 0.23757381939888, \"error\": 0.07145}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 10.73 sec\n",
      "[5] train metrics:{\"loss\": 0.15486119554042815, \"error\": 0.0413}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 13.66 sec\n",
      "[6] train metrics:{\"loss\": 0.09451916145086288, \"error\": 0.018575}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 13.67 sec\n",
      "[7] train metrics:{\"loss\": 0.0700134416103363, \"error\": 0.009625}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 13.78 sec\n",
      "[8] train metrics:{\"loss\": 0.05248799537420273, \"error\": 0.003975}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 12.63 sec\n",
      "[9] train metrics:{\"loss\": 0.03984407194852829, \"error\": 0.00035}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 10.63 sec\n",
      "[10] train metrics:{\"loss\": 0.03694459772706032, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 13.13 sec\n",
      "[11] train metrics:{\"loss\": 0.03580245625972748, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 13.94 sec\n",
      "[12] train metrics:{\"loss\": 0.03495487071871758, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 13.78 sec\n",
      "[13] train metrics:{\"loss\": 0.03420322653055191, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 13.0 sec\n",
      "[14] train metrics:{\"loss\": 0.03351431832909584, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 10.68 sec\n",
      "[15] train metrics:{\"loss\": 0.0328611094892025, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 12.85 sec\n",
      "[16] train metrics:{\"loss\": 0.032256303888559344, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 13.97 sec\n",
      "[17] train metrics:{\"loss\": 0.031692172318696976, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 15.1 sec\n",
      "[18] train metrics:{\"loss\": 0.031148864424228668, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 14.38 sec\n",
      "[19] train metrics:{\"loss\": 0.030647821056842803, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 10.73 sec\n",
      "[20] train metrics:{\"loss\": 0.030150964498519898, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 12.66 sec\n",
      "[21] train metrics:{\"loss\": 0.02969496927857399, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 14.33 sec\n",
      "[22] train metrics:{\"loss\": 0.02928686562180519, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 19.89 sec\n",
      "[23] train metrics:{\"loss\": 0.028866908860206604, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 23.95 sec\n",
      "[24] train metrics:{\"loss\": 0.028454023391008377, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 23.91 sec\n",
      "[25] train metrics:{\"loss\": 0.028074368637800218, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 24.14 sec\n",
      "Pure training time: 377.76 sec\n"
     ]
    }
   ],
   "source": [
    "%run main.py --dataset cifar10 --model resnet --dataroot=data/cifar10-batches-py/ --filters 1.0 --lr 0.01 \\\n",
    "--resume checkpoints/cifar100_resnet_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1_30.pt --disable-bn \\\n",
    "--weight-decay 0.0005 --batch-size 128 --epochs 26 --seed 1 \\\n",
    "--split train --confuse-mode --forget-class 0,1 --num-to-forget 200"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Retrain"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "#%run main.py --dataset small_lacuna6 --model allcnn --dataroot=data/lacuna10/ --filters 1.0 --lr 0.001 \\\n",
    "#--resume checkpoints/lacuna100_allcnn_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1_30.pt --disable-bn \\\n",
    "#--weight-decay 0.1 --batch-size 128 --epochs 31 \\\n",
    "#--forget-class 0,1,2,3,4,5 --num-to-forget 300 --seed 3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checkpoint name: cifar10_resnet_1_0_forget_[0, 1]_num_200_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1\n",
      "[Logging in cifar10_resnet_1_0_forget_[0, 1]_num_200_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1_training]\n",
      "Downloading https://www.cs.toronto.edu/~kriz/cifar-10-python.tar.gz to data/cifar10_batches-py/cifar-10-python.tar.gz\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 170498071/170498071 [00:20<00:00, 8178981.78it/s] \n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Extracting data/cifar10_batches-py/cifar-10-python.tar.gz to data/cifar10_batches-py/\n",
      "Files already downloaded and verified\n",
      "confuse mode: True\n",
      "split mode: forget\n",
      "[22462  6646 36135 23850  1434  5306 18141  4049 11418 10150 21959  2340\n",
      " 28237 23045 33372 17719 28726 24235 36477 30535 34830 13742  9505  9925\n",
      "  8968  5340 32206 23111 36876 37753 18138   346  9984 33198 10884 17838\n",
      "  5290 34800  8839 26496 22062 19691 25490 32745 29623 37043 38544 16174\n",
      " 26055 17790 13062  3564 33467 19387 12727 10616  5925 38573 20823 25972\n",
      "  2232 39091 16018 31897  6216 31382  9265  4834 15220 13636 20376 17239\n",
      "  3784 30964 11139 32269 25536 37777  9464 31770 12377 14878 22419 35384\n",
      " 33593 13768  8813 14913  1918  3157 22036 26124 26696 32374   359 18794\n",
      " 37201 12935 39835  6800 34930 21153  1332   444 26453 22013  2780 28519\n",
      " 32137 33713  9783 18691  5872  5426 26608  5517 26355 34576 15284  9727\n",
      " 11908 16177  5566 33238 31572 31990  6716 36313 26521  8618 25781 10358\n",
      "  9601   341  6301 36220 15826  3522  4898 31528 28091 12132 15300  5732\n",
      " 10974 23654 30805 26574  5750  7350 19049 24064  7229  8594 31711 15489\n",
      " 21523 27315  8146 28736 36469 37611 32999 39736 27867 33821 34312  3036\n",
      " 11154 19416 37108 22615 36649 11823  7061 36430 18557 37081  8899 29319\n",
      "  9990 26523 33672  7326 39648 21642   132  4050  6683  5979 20121 36524\n",
      " 22540 33421 27229 13730 29155 25346  2670   732]\n",
      "Number of Classes: 10\n",
      "[0] train metrics:{\"loss\": 0.8776707893371583, \"error\": 0.310325}\n",
      "Learning Rate : 0.01\n",
      "[0] test metrics:{\"loss\": 0.6726855438232422, \"error\": 0.2333}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 26.47 sec\n",
      "[1] train metrics:{\"loss\": 0.5546332703590393, \"error\": 0.19135}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 23.58 sec\n",
      "[2] train metrics:{\"loss\": 0.411030176115036, \"error\": 0.1398}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 23.83 sec\n",
      "[3] train metrics:{\"loss\": 0.30049551043510436, \"error\": 0.0975}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 24.14 sec\n",
      "[4] train metrics:{\"loss\": 0.21082578921318054, \"error\": 0.065275}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 23.95 sec\n",
      "[5] train metrics:{\"loss\": 0.13203115015029906, \"error\": 0.035125}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 23.82 sec\n",
      "[6] train metrics:{\"loss\": 0.08035368938446046, \"error\": 0.015675}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 23.87 sec\n",
      "[7] train metrics:{\"loss\": 0.05315162697434425, \"error\": 0.004475}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 19.29 sec\n",
      "[8] train metrics:{\"loss\": 0.041563900208473205, \"error\": 0.001025}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 13.13 sec\n",
      "[9] train metrics:{\"loss\": 0.03547775356173515, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 10.29 sec\n",
      "[10] train metrics:{\"loss\": 0.0340896775662899, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 9.06 sec\n",
      "[11] train metrics:{\"loss\": 0.03323477267026901, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.65 sec\n",
      "[12] train metrics:{\"loss\": 0.03250679121613503, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.64 sec\n",
      "[13] train metrics:{\"loss\": 0.03184912375807762, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.01 sec\n",
      "[14] train metrics:{\"loss\": 0.031240232867002487, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.95 sec\n",
      "[15] train metrics:{\"loss\": 0.030659463116526602, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.77 sec\n",
      "[16] train metrics:{\"loss\": 0.030124190509319304, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.46 sec\n",
      "[17] train metrics:{\"loss\": 0.02962413500249386, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.95 sec\n",
      "[18] train metrics:{\"loss\": 0.029126183646917343, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.92 sec\n",
      "[19] train metrics:{\"loss\": 0.028692543506622314, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.87 sec\n",
      "[20] train metrics:{\"loss\": 0.028240641552209855, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.67 sec\n",
      "[21] train metrics:{\"loss\": 0.02783956998884678, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.87 sec\n",
      "[22] train metrics:{\"loss\": 0.02746837635934353, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.62 sec\n",
      "[23] train metrics:{\"loss\": 0.027107982075214386, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.32 sec\n",
      "[24] train metrics:{\"loss\": 0.02672754344344139, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.88 sec\n",
      "[25] train metrics:{\"loss\": 0.026383602434396743, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.83 sec\n",
      "Pure training time: 324.74999999999994 sec\n"
     ]
    }
   ],
   "source": [
    "%run main.py --dataset cifar10 --model resnet --dataroot=data/cifar10_batches-py/ --filters 1.0 --lr 0.01 \\\n",
    "--resume checkpoints/cifar100_resnet_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1_30.pt --disable-bn \\\n",
    "--weight-decay 0.0005 --batch-size 128 --epochs 26 --seed 1 \\\n",
    "--split forget --confuse-mode --forget-class 0,1 --num-to-forget 200"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Logs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "log_dict={}\n",
    "training_epochs=25"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "log_dict['epoch']=training_epochs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Total Number of Parameters: 11175178\n"
     ]
    }
   ],
   "source": [
    "parameter_count(copy.deepcopy(model))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Loads checkpoints"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "import copy\n",
    "model0 = copy.deepcopy(model)\n",
    "model_initial = copy.deepcopy(model)\n",
    "\n",
    "arch = args.model \n",
    "filters=args.filters\n",
    "arch_filters = arch +'_'+ str(filters).replace('.','_')\n",
    "augment = False\n",
    "dataset = args.dataset\n",
    "class_to_forget = args.forget_class\n",
    "init_checkpoint = f\"checkpoints/{args.name}_init.pt\"\n",
    "num_classes=args.num_classes\n",
    "num_to_forget = args.num_to_forget\n",
    "num_total = len(train_loader.dataset)\n",
    "num_to_retain = num_total - num_to_forget\n",
    "seed = args.seed\n",
    "unfreeze_start = None\n",
    "\n",
    "learningrate=f\"lr_{str(args.lr).replace('.','_')}\"\n",
    "batch_size=f\"_bs_{str(args.batch_size)}\"\n",
    "lossfn=f\"_ls_{args.lossfn}\"\n",
    "wd=f\"_wd_{str(args.weight_decay).replace('.','_')}\"\n",
    "seed_name=f\"_seed_{args.seed}_\"\n",
    "\n",
    "num_tag = '' if num_to_forget is None else f'_num_{num_to_forget}'\n",
    "unfreeze_tag = '_' if unfreeze_start is None else f'_unfreeze_from_{unfreeze_start}_'\n",
    "augment_tag = '' if not augment else f'augment_'\n",
    "\n",
    "m_name = f'checkpoints/{dataset}_{arch_filters}_forget_None{unfreeze_tag}{augment_tag}{learningrate}{batch_size}{lossfn}{wd}{seed_name}{training_epochs}.pt'\n",
    "m0_name = f'checkpoints/{dataset}_{arch_filters}_forget_{class_to_forget}{num_tag}{unfreeze_tag}{augment_tag}{learningrate}{batch_size}{lossfn}{wd}{seed_name}{training_epochs}.pt'\n",
    "\n",
    "model.load_state_dict(torch.load(m_name))\n",
    "model0.load_state_dict(torch.load(m0_name))\n",
    "model_initial.load_state_dict(torch.load(init_checkpoint))\n",
    "\n",
    "teacher = copy.deepcopy(model)\n",
    "student = copy.deepcopy(model)\n",
    "\n",
    "model.cuda()\n",
    "model0.cuda()\n",
    "\n",
    "\n",
    "for p in model.parameters():\n",
    "    p.data0 = p.data.clone()\n",
    "for p in model0.parameters():\n",
    "    p.data0 = p.data.clone()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "log_dict['args']=args"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "model_init = ntk_init(init_checkpoint,args.seed)\n",
    "for p in model_init.parameters():\n",
    "    p.data0 = p.data.clone()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Distance: 10.362852075030466\n",
      "Normalized Distance: 0.09396269883651323\n"
     ]
    }
   ],
   "source": [
    "log_dict['dist_Original_Original_init']= distance(model_init,model)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Data Loader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "args.retain_bs = 32\n",
    "args.forget_bs = 32"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "confuse mode: True\n",
      "split mode: train\n",
      "[22462  6646 36135 23850  1434  5306 18141  4049 11418 10150 21959  2340\n",
      " 28237 23045 33372 17719 28726 24235 36477 30535 34830 13742  9505  9925\n",
      "  8968  5340 32206 23111 36876 37753 18138   346  9984 33198 10884 17838\n",
      "  5290 34800  8839 26496 22062 19691 25490 32745 29623 37043 38544 16174\n",
      " 26055 17790 13062  3564 33467 19387 12727 10616  5925 38573 20823 25972\n",
      "  2232 39091 16018 31897  6216 31382  9265  4834 15220 13636 20376 17239\n",
      "  3784 30964 11139 32269 25536 37777  9464 31770 12377 14878 22419 35384\n",
      " 33593 13768  8813 14913  1918  3157 22036 26124 26696 32374   359 18794\n",
      " 37201 12935 39835  6800 34930 21153  1332   444 26453 22013  2780 28519\n",
      " 32137 33713  9783 18691  5872  5426 26608  5517 26355 34576 15284  9727\n",
      " 11908 16177  5566 33238 31572 31990  6716 36313 26521  8618 25781 10358\n",
      "  9601   341  6301 36220 15826  3522  4898 31528 28091 12132 15300  5732\n",
      " 10974 23654 30805 26574  5750  7350 19049 24064  7229  8594 31711 15489\n",
      " 21523 27315  8146 28736 36469 37611 32999 39736 27867 33821 34312  3036\n",
      " 11154 19416 37108 22615 36649 11823  7061 36430 18557 37081  8899 29319\n",
      "  9990 26523 33672  7326 39648 21642   132  4050  6683  5979 20121 36524\n",
      " 22540 33421 27229 13730 29155 25346  2670   732]\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "confuse mode: True\n",
      "split mode: forget\n",
      "[22462  6646 36135 23850  1434  5306 18141  4049 11418 10150 21959  2340\n",
      " 28237 23045 33372 17719 28726 24235 36477 30535 34830 13742  9505  9925\n",
      "  8968  5340 32206 23111 36876 37753 18138   346  9984 33198 10884 17838\n",
      "  5290 34800  8839 26496 22062 19691 25490 32745 29623 37043 38544 16174\n",
      " 26055 17790 13062  3564 33467 19387 12727 10616  5925 38573 20823 25972\n",
      "  2232 39091 16018 31897  6216 31382  9265  4834 15220 13636 20376 17239\n",
      "  3784 30964 11139 32269 25536 37777  9464 31770 12377 14878 22419 35384\n",
      " 33593 13768  8813 14913  1918  3157 22036 26124 26696 32374   359 18794\n",
      " 37201 12935 39835  6800 34930 21153  1332   444 26453 22013  2780 28519\n",
      " 32137 33713  9783 18691  5872  5426 26608  5517 26355 34576 15284  9727\n",
      " 11908 16177  5566 33238 31572 31990  6716 36313 26521  8618 25781 10358\n",
      "  9601   341  6301 36220 15826  3522  4898 31528 28091 12132 15300  5732\n",
      " 10974 23654 30805 26574  5750  7350 19049 24064  7229  8594 31711 15489\n",
      " 21523 27315  8146 28736 36469 37611 32999 39736 27867 33821 34312  3036\n",
      " 11154 19416 37108 22615 36649 11823  7061 36430 18557 37081  8899 29319\n",
      "  9990 26523 33672  7326 39648 21642   132  4050  6683  5979 20121 36524\n",
      " 22540 33421 27229 13730 29155 25346  2670   732]\n"
     ]
    }
   ],
   "source": [
    "train_loader_full, valid_loader_full, test_loader_full = datasets.get_loaders(dataset, split=\"train\",confuse_mode=True,class_to_replace=class_to_forget, num_indexes_to_replace=num_to_forget, batch_size=args.batch_size, seed=seed, root=args.dataroot, augment=False, shuffle=True)\n",
    "marked_loader, _, _ = datasets.get_loaders(dataset, split=\"forget\", confuse_mode=True,class_to_replace=class_to_forget, num_indexes_to_replace=num_to_forget, only_mark=True, batch_size=1, seed=seed, root=args.dataroot, augment=False, shuffle=True)\n",
    "\n",
    "def replace_loader_dataset(data_loader, dataset, batch_size=args.batch_size, seed=1, shuffle=True):\n",
    "    manual_seed(seed)\n",
    "    loader_args = {'num_workers': 0, 'pin_memory': False}\n",
    "    def _init_fn(worker_id):\n",
    "        np.random.seed(int(seed))\n",
    "    return torch.utils.data.DataLoader(dataset, batch_size=batch_size,num_workers=0,pin_memory=True,shuffle=shuffle)\n",
    "    \n",
    "forget_dataset = copy.deepcopy(marked_loader.dataset)\n",
    "marked = forget_dataset.targets < 0\n",
    "forget_dataset.data = forget_dataset.data[marked]\n",
    "forget_dataset.targets = - forget_dataset.targets[marked] - 1\n",
    "forget_loader = replace_loader_dataset(train_loader_full, forget_dataset, batch_size=args.forget_bs, seed=seed, shuffle=True)\n",
    "\n",
    "retain_dataset = copy.deepcopy(marked_loader.dataset)\n",
    "marked = retain_dataset.targets >= 0\n",
    "retain_dataset.data = retain_dataset.data[marked]\n",
    "retain_dataset.targets = retain_dataset.targets[marked]\n",
    "retain_loader = replace_loader_dataset(train_loader_full, retain_dataset, batch_size=args.retain_bs, seed=seed, shuffle=True)\n",
    "\n",
    "assert(len(forget_dataset) + len(retain_dataset) == len(train_loader_full.dataset))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "200\n",
      "39800\n",
      "10000\n",
      "40000\n",
      "{6: 4000, 9: 4000, 4: 4000, 1: 4000, 2: 4000, 8: 4000, 3: 4000, 7: 4000, 5: 4000, 0: 4000}\n"
     ]
    }
   ],
   "source": [
    "print (len(forget_loader.dataset))\n",
    "print (len(retain_loader.dataset))\n",
    "print (len(test_loader_full.dataset))\n",
    "print (len(train_loader_full.dataset))\n",
    "from collections import Counter\n",
    "print(dict(Counter(train_loader_full.dataset.targets)))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# SGDA Forgetting"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "args.optim = 'sgd'\n",
    "args.gamma = 1\n",
    "args.alpha = 0.01\n",
    "args.beta = 0\n",
    "args.smoothing = 0.5\n",
    "args.msteps = 2\n",
    "args.clip = 0.5\n",
    "args.sstart = 10\n",
    "args.kd_T = 2\n",
    "args.distill = 'kd'\n",
    "\n",
    "args.sgda_epochs = 10\n",
    "args.sgda_learning_rate = 0.0005\n",
    "args.lr_decay_epochs = [7,10,10]\n",
    "args.lr_decay_rate = 0.1\n",
    "args.sgda_weight_decay = 5e-4\n",
    "args.sgda_momentum = 0.9"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [],
   "source": [
    "model_t = copy.deepcopy(teacher)\n",
    "model_s = copy.deepcopy(student)\n",
    "#model_s = models.get_model(arch, num_classes=num_classes, filters_percentage=filters).to(args.device)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [],
   "source": [
    "#this is from https://github.com/ojus1/SmoothedGradientDescentAscent/blob/main/SGDA.py\n",
    "#For SGDA smoothing\n",
    "beta = 0.1\n",
    "def avg_fn(averaged_model_parameter, model_parameter, num_averaged): return (\n",
    "    1 - beta) * averaged_model_parameter + beta * model_parameter\n",
    "swa_model = torch.optim.swa_utils.AveragedModel(\n",
    "    model_s, avg_fn=avg_fn)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [],
   "source": [
    "module_list = nn.ModuleList([])\n",
    "module_list.append(model_s)\n",
    "trainable_list = nn.ModuleList([])\n",
    "trainable_list.append(model_s)\n",
    "\n",
    "criterion_cls = nn.CrossEntropyLoss()\n",
    "criterion_div = DistillKL(args.kd_T)\n",
    "criterion_kd = DistillKL(args.kd_T)\n",
    "\n",
    "\n",
    "criterion_list = nn.ModuleList([])\n",
    "criterion_list.append(criterion_cls)    # classification loss\n",
    "criterion_list.append(criterion_div)    # KL divergence loss, original knowledge distillation\n",
    "criterion_list.append(criterion_kd)     # other knowledge distillation loss\n",
    "\n",
    "# optimizer\n",
    "if args.optim == \"sgd\":\n",
    "    optimizer = optim.SGD(trainable_list.parameters(),\n",
    "                          lr=args.sgda_learning_rate,\n",
    "                          momentum=args.sgda_momentum,\n",
    "                          weight_decay=args.sgda_weight_decay)\n",
    "elif args.optim == \"adam\": \n",
    "    optimizer = optim.Adam(trainable_list.parameters(),\n",
    "                          lr=args.sgda_learning_rate,\n",
    "                          weight_decay=args.sgda_weight_decay)\n",
    "elif args.optim == \"rmsp\":\n",
    "    optimizer = optim.RMSprop(trainable_list.parameters(),\n",
    "                          lr=args.sgda_learning_rate,\n",
    "                          momentum=args.sgda_momentum,\n",
    "                          weight_decay=args.sgda_weight_decay)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [],
   "source": [
    "module_list.append(model_t)\n",
    "\n",
    "if torch.cuda.is_available():\n",
    "    module_list.cuda()\n",
    "    criterion_list.cuda()\n",
    "    import torch.backends.cudnn as cudnn\n",
    "    cudnn.benchmark = True\n",
    "    swa_model.cuda()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 87.093 \n",
      "maximize loss: -12.73\t minimize loss: 0.59\t train_acc: 87.09296417236328\n",
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 87.412 \n",
      "maximize loss: -11.21\t minimize loss: 0.56\t train_acc: 87.41205596923828\n",
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 87.452 \n",
      "maximize loss: 0.00\t minimize loss: 0.52\t train_acc: 87.45226287841797\n",
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 87.583 \n",
      "maximize loss: 0.00\t minimize loss: 0.52\t train_acc: 87.58291625976562\n",
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 87.515 \n",
      "maximize loss: 0.00\t minimize loss: 0.52\t train_acc: 87.51507568359375\n",
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 87.312 \n",
      "maximize loss: 0.00\t minimize loss: 0.52\t train_acc: 87.31155395507812\n",
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 87.422 \n",
      "maximize loss: 0.00\t minimize loss: 0.52\t train_acc: 87.42211151123047\n",
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 87.183 \n",
      "maximize loss: 0.00\t minimize loss: 0.47\t train_acc: 87.18341827392578\n",
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 86.942 \n",
      "maximize loss: 0.00\t minimize loss: 0.47\t train_acc: 86.94220733642578\n",
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 87.093 \n",
      "maximize loss: 0.00\t minimize loss: 0.47\t train_acc: 87.09296417236328\n"
     ]
    }
   ],
   "source": [
    "t1 = time.time()\n",
    "acc_rs = []\n",
    "acc_fs = []\n",
    "acc_vs = []\n",
    "ic_rs = []\n",
    "ic_vs = []\n",
    "fgt_rs = []\n",
    "fgt_vs = []\n",
    "for epoch in range(1, args.sgda_epochs + 1):\n",
    "\n",
    "    lr = sgda_adjust_learning_rate(epoch, args, optimizer)\n",
    "    print(\"==> SCRUB unlearning ...\")\n",
    "    ic_r, fgt_r = interclass_confusion(model_s, retain_loader, class_to_forget, \"SCRUB\")\n",
    "    ic_v, fgt_v = interclass_confusion(model_s, valid_loader_full, class_to_forget, \"SCRUB\")\n",
    "    ic_rs.append(ic_r)\n",
    "    ic_vs.append(ic_v)\n",
    "    fgt_rs.append(fgt_r)\n",
    "    fgt_vs.append(fgt_v)\n",
    "\n",
    "    acc_r, acc5_r, loss_r = validate(retain_loader, model_s, criterion_cls, args, True)\n",
    "    acc_f, acc5_f, loss_f = validate(forget_loader, model_s, criterion_cls, args, True)\n",
    "    acc_v, acc5_v, loss_v = validate(valid_loader_full, model_s, criterion_cls, args, True)\n",
    "    acc_rs.append(100-acc_r.item())\n",
    "    acc_fs.append(100-acc_f.item())\n",
    "    acc_vs.append(100-acc_v.item())\n",
    "\n",
    "\n",
    "    maximize_loss = 0\n",
    "    if epoch <= args.msteps:\n",
    "        maximize_loss = train_distill(epoch, forget_loader, module_list, swa_model, criterion_list, optimizer, args, \"maximize\")\n",
    "    train_acc, train_loss = train_distill(epoch, retain_loader, module_list, swa_model, criterion_list, optimizer, args, \"minimize\",)\n",
    "    if epoch >= args.sstart:\n",
    "        swa_model.update_parameters(model_s)\n",
    "\n",
    "    \n",
    "    print (\"maximize loss: {:.2f}\\t minimize loss: {:.2f}\\t train_acc: {}\".format(maximize_loss, train_loss, train_acc))\n",
    "ic_r, fgt_r = interclass_confusion(model_s, retain_loader, class_to_forget, \"SCRUB\")\n",
    "ic_v, fgt_v = interclass_confusion(model_s, valid_loader_full, class_to_forget, \"SCRUB\")\n",
    "ic_rs.append(ic_r)\n",
    "ic_vs.append(ic_v)\n",
    "fgt_rs.append(fgt_r)\n",
    "fgt_vs.append(fgt_v)\n",
    "acc_r, acc5_r, loss_r = validate(retain_loader, model_s, criterion_cls, args, True)\n",
    "acc_f, acc5_f, loss_f = validate(forget_loader, model_s, criterion_cls, args, True)\n",
    "acc_tv, acc5_v, loss_v = validate(valid_loader_full, model_s, criterion_cls, args, True)\n",
    "acc_rs.append(100-acc_r.item())\n",
    "acc_fs.append(100-acc_f.item())\n",
    "acc_vs.append(100-acc_v.item())\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_1779/2812292199.py:16: UserWarning: FigureCanvasAgg is non-interactive, and thus cannot be shown\n",
      "  fig.show()\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAw4AAASvCAYAAACgpa2IAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy81sbWrAAAACXBIWXMAAA9hAAAPYQGoP6dpAAEAAElEQVR4nOzdd3hUZd7G8XuSkBAgCS0FSAgBUUFWSgAB6QqKroplxYYNFVZUmu5aV8CCDcGVBUVRVFbEgu0VRVQgVKVagFWkBkiAgCTU1PP+8ThJhpQZwmTOJPP9XNdcM3Pm5MxvSEjOfZ7msCzLEgAAAACUI8juAgAAAAD4P4IDAAAAALcIDgAAAADcIjgAAAAAcIvgAAAAAMAtggMAAAAAtwgOAAAAANwKsbuAqqKgoEB79uxRRESEHA6H3eUAAAAAXmFZlg4fPqzGjRsrKKjsdgWCg4f27NmjhIQEu8sAAAAAKkVqaqri4+PLfJ3g4KGIiAhJ5h80MjLS5moAAAAA78jKylJCQkLh+W5ZCA4ecnZPioyMJDgAAACg2nHXHZ/B0QAAAADcIjgAAAAAcIvgAAAAAMAtggMAAAAAtwgOAAAAANwiOAAAAABwi+AAAAAAwC2CAwAAAAC3WAAOfi0/X1qyREpLkxo1knr0kIKD7a4KAAAg8BAc4LfmzpVGjJB27SraFh8vvfSSdNVV9tUFAAAQiOiqBL80d650zTWuoUGSdu822+fOtacuAACAQEVwgN/JzzctDZZV8jXntpEjzX4AAADwDYID/M6SJSVbGoqzLCk11ewHAAAA32CMA/xOWppn+330kdS1qxQWVrn1AHZgYgAACAxV6fc9LQ7wO40aebbflClmsPQDD0ibN1duTYAvzZ0rNWsm9ekj3XCDuW/WjLE9AFDdVLXf9w7LKq0nOU6WlZWlqKgoZWZmKjIy0u5yqrX8fPOfZvfu0sc5OBxSRIS57d5dtP2CC6ShQ6UrrpBCQ31WLuBVzokBTv7ZdzjM/YcfMqsYAFQH/vT73tPzXFoc4HeCg82Uq2WFBkl6801p+3bp00+lSy4x27/9Vrr2WikhQXroIWnrVp+WDZw2JgYAgMBQVX/fExzgl/76V6levZLb4+OLEnhIiHT55dIXX0jbtkmPPmq6Oe3bJz3zjHTGGdJFF0kffyzl5vr+MwCn4uBB6emnmRgAAALBN99Uzd/3DI6GX5ozR/rjDyk2Vnr7benAgfIHDCUmSk88If3rX9L//Z/0yivS118X3Ro1koYMke64w+wL2C0zU0pJkRYuNLcffyz9ylNpPJ1AAEDVUZUGyMJzx49L//uftGGDuW3caO63bPHs6/3t9z3BAX7HsqQXXjCPR4yQ+vf3/Gtr1JCuvNLctm6VXntNeuMN8x/vySelp56SBgyQhg0z9yH8D4CPHD4sLV1aFBTWrpUKClz3SUgwV5jccXbZA1A9zJ1r/t4VvwIdH2+67TKmqWo4dswEBGcwcIaErVs9vyhUGk8njPEVBkd7iMHRvvPNN1K/flKtWuYkqn790zteTo4ZC/Hqq2YchFN8vGmBGDLEPAa86dgxadmyoqCwalXJvqpnnGFm0OjTR+rdW4qJKX9iAKewMOkf/5D++U+pdu3K/BQAKps/DZCFe86AULz1YMMG02W6rN/bDRpI55wjtW5t7s85RzrrLOm888qfCCY+3hzXFy1Pnp7nEhw8RHDwnYsvlubPl+69V/r3v7177M2bpenTpZkzpYwMsy0oyIypGDrUjImgaRgVceKEtHJlUVBYubLk2BrnlHvOW2mB1XkSIbn+MXGeRLRubf5ISVKTJtJzz0nXX08rBFAVOWcRLKuvu69PHlGkeEAoHhI8CQgnh4To6NJ/R7v7fe+PsyoRHDxEcPCNX36R/vIXczK/ebPUvHnlvE92tvkP++qr0uLFRdsTE6U775Ruv93/mgfhX3JypB9+KAoKy5ebn6vi4uNdg0KzZp4du7RuCwkJ0uTJphvexx9LY8aYmcUkqVs306WhY0cvfDAAPrNokfnd4E6zZqaFMi7O/G1q1Kjk48hILiBUxLFj0qZNJbsYeRoQioeEmJhTf//yft/7sqWJ4OBlBAffuO020xpwzTXSBx/45j3/97+iVog//jDbnDM2DR0qXXihCTIIbHl50urVRUFh2TLzB6e4uDjXoNCiRcX/kLsbKHnihPTii2YmpqNHzbbbbjPP4+Iq9p4AfCc3Vxo92ixm6g3h4WWHiuKPo6MDs/XCGRBO7mK0fXvZAaFhw5KtB61bVywglMcfBsYTHLyM4FD50tLMFf/cXGnFCqlLF9++//Hjplnw1VfNSaFT8+amFeK228wsTwgM+fnSunVFQWHJEunIEdd9oqPN2ARnUDjrLN9f8du926xb8s475nmdOtJjj5krWGFhvq0FgHsHD5qJO6ZMKX86zuKef96crKanm7+VaWmujw8f9vz9g4LM37LywoXzPjy8Yp+xIrx18nz0aOldjDwJCKV1MQoUBAcvIzhUvocfliZMkM4/38w+Y6dffjGtEG+/babNlIpmbBo61Jwk0iRcvRQUSD/9VBQUUlKKvvdO9eq5BoXWrf2nNWrlShMWfvjBPG/RwrRIXHYZP6uAP9i0yYzbe+stc6FKMiem2dnmxP90BsgePVoUJMoKF+npZp2jUznri4ryrBWjXr3T+z1TkVmljh4tvYtReQEhOrooGARqQCgLwcHLCA6V68gRqWlT01Vo7lxzgu4Pjh0za0q8+qr0/fdF21u2lO66S7r1VnOlAlWPZZk/NM6gsHixuRJYXGSk1LNnUVBo29Z/gkJpCgqkWbOkBx8smvu7Xz9p0iTzxxGAb1mWWUto8mTpq6+Ktrdta1YFvu46ad483w2Qzcsz4aG8cOF8fPKYrfKEhZkA4a4VIza25DTo7maVmjVLOvPM0rsYlSU62jUcOO8JCGUjOHgZwaFyvfyydN99ZvDX//7nn/0vf/zRBIhZs4qahUNDpauvNq0QPXtyZdefWZb0669FQWHRImn/ftd9atc2zePOoNC+fdVc6+PwYdN6N3GiGcQdHCzdfbc0duzpT28MwL1jx8zfismTzVVxyfx9uPxyExh69XL9e+EvA2SdLMu0uJ4cKEoLGc6xgZ5wOMzFNmegiImRPvnk1LpaFecMCCeHBALCqSM4eBnBofLk55sr+Nu2Sf/5jznB8WdHjkjvvWdCxOrVRdvPPtsEiJtv5uTMH1iWWXjHGRQWLiy5Amd4uOka5wwKHTuaLmnVxdat0v33m1mYJPNz+cQTprWsKgYiwN/t3m3+jr36alELZp06Zr2ge+81XQjL4g8DZCvixAlp7173LRh795Zcy8ZTdetK7doRECoTwcHLCA6V58MPpb/9zUxvtnOnWfitqlizxvyBePfdopltatY0n2foUDNNJq0QFVORP6I7drgGhZNXYQ4Lk7p2LQoKnTsHxgDib781Vzl/+cU8b9PG9B3u29fWsoBq44cfTOvABx+Y7kCSlJRkWtJvu82MFQh0BQVm/aTigeKrr0x3YHfefdesV4PKQ3DwMoJD5bAscyL3/ffSo4+aq6FVUVaW+cX26qvS+vVF29u0MVd3Bw82V0zgGU8Hyu3e7RoUtm1zPU6NGmZlTmdQ6NrVBLtAlJdnBvw/9ljRldArr5ReeKHy1ksBqrO8PNOaN3myWcfFqWdPE9Qvv7xqtBjYydN1LBYuNBNToPIQHLyM4FA5li2Tunc3YwV27qz6051alrRqlfTKK6Y7k3PmjPBwMwhu6FBzlZtWiLK5Gyg3cqRp3Vm40CwSWFxwsNSpU1FQ6NbNjFtAkYMHzViHqVNNq05oqFlM7qGHpIgIu6sD/N8ff0ivv27G5jlbNWvUMFfER4yQOnSwt76qxLly9u7dpzerFE4fwcHLCA6V46qrzBWbO+4w81pXJ4cOmcFxr75a1EVEMrNpDB0q3XijmbUHRZx/RDyd2zwoyPyRdgaF7t05+fXUhg0mhH3zjXneqJH0zDPSTTf598xRgF1+/dVMpzpzZtHij9HR0t//bm4svFgxzotFUuXPKoWyERy8jODgfZs3mwWzLMtMsdaqld0VVQ7LMgvavfqq9P77ZiCZZK6E33CDCRHJyaV/bVUdLCeZz33kiLnCffCguUrnfFzWLS3N9IF155przCD0Hj3oAnY6LEv6/HOzeu2WLWZb586mS5ivF2AE/JFlmXA9ebKZNtXpL38xwfuGGwK3+6M3+dusUoGI4OBlBAfvGz7cdJe49FLp//7P7mp84+BBs6jcq6+aaWedkpNNgLj+ejMDh1SxBXEqQ0GBmZbP3Ul/aTfnIEFvY6Ccd2Vnm5+rJ54oWh178GDTAtG4sb21AXY4flz673/NieuGDWabwyH99a8mMLAIqPdV5Qtl1QHBwcsIDt514IC5mnD8uPTdd54NjqpOLMv8gnz1VdMMm5NjtkdEmC5MZ5whPfBA2f38K9J0m5vr2VX/k2+HDp3aSqMnCw01M2bVr+/+tnWrCVDuMFCucqSnmxXc33zTPK9d2zwfPZqrqggMe/aYC1qvvGL+Tknm/8Htt5vpVFu2tLc+oLIQHLyM4OBdTz5pZnfp0MGshRDIV24yMqS33jIh4uTBvqVxOMzVmC++KLsloLSAUNEFdpzq1PHs5L9ePdfn4eGef38ZKOcfVq0yrV0rVpjnzZqZxeSuvDKw/6+i+lq92rQuzJlT1FKamGimU739drpEovojOHgZwcF7Tpwwv5D37TNNwTfcYHdF/sGyzJX0J54wU9RVBofD/AF0d7Jf2uuhoZVT08kYKOcfLEuaPVv6xz9MkJNMy+DkydK559paGuAVeXnSp59KkyaZGf6cuneXRo0y06myUCICBcHBywgO3jNjhplFKSHBDMisTiv1esPs2Z6FqYgI0//ck1YA5y0qqmpcqWegnP84elR69lnp+edN6A8KMt3Jxo+XGja0uzrg1B06ZP4OvfyyWTRSMn+HBg0yv3c6drS1PMAWBAcvIzh4R0GBWRRt0yaz8NSYMXZX5H9YEMdgoJx/2b7dtD588IF5XreuNG6cmYaS8I+qYPNmM53qm2+aQCyZ8DtsmPk5ZiIABDJPz3P9crbuqVOnKikpSTVr1lRycrKWLFlS5r5z585Vv379FB0drcjISHXt2lXz58932WfmzJlyOBwlbiec82LCZ776yoSGyEjpzjvtrsY/9ehh+vGX1Zfc4TBX33v08G1dvhYcbILR9debe0KDvZo1M9MJL1pk1iI5dMhcnW3bVvr6a5uLA8pgWWYCjssuM9N/T5liQkObNmYRt507TfdQQgPgGb8LDnPmzNHIkSP1yCOPaN26derRo4cGDBignTt3lrp/SkqK+vXrp3nz5mnNmjXq06ePLrvsMq1bt85lv8jISKWlpbncajJNiM+98IK5v/NOFj8rS3CwmRpTKhkenM8nT+ZEGvbo1Utas8YM5m/Y0FwIuOgi0x/ck8H9gC+cOCG98YYJthdcYKb8tiwz/feCBdJPP0lDhpjJGwB4zu+6Kp133nnq0KGDpk2bVritVatWGjhwoCZMmODRMc455xwNGjRI//rXvySZFoeRI0fq0KFDFa6Lrkqnb+1as15BSIiZdjMhwe6K/Bv9/OHvDh0yYx1eftkMNK1Rw8xx/+ijXBiAPdLSpGnTzHSq+/ebbbVqSbfdZmZIOvNMe+sD/FWV7KqUk5OjNWvWqH///i7b+/fvr+XLl3t0jIKCAh0+fFj169d32X7kyBElJiYqPj5ef/3rX0u0SJwsOztbWVlZLjecnokTzf211xIaPHHVVaZf+cKFZsGzhQvNNKSEBviLunWlF1+Ufv5ZGjDArBXy/PNmrvs33jBjmgBfWLvWrCafmGi6Hu3fLzVtan4ed+0yXZQIDcDp86vgkJGRofz8fMXGxrpsj42NVXp6ukfHmDhxoo4ePaprr722cNvZZ5+tmTNn6rPPPtPs2bNVs2ZNnX/++dpcTrv6hAkTFBUVVXhL4Ez3tKSmmvmxJQZEnwr6+aMqOPtsad48s7bImWeaqZaHDJE6d3ad5hLwpvx86eOPTfe55GTpnXdMeD3/fDOIf8sW6f77zXTSALzDr4KDk+Okjt2WZZXYVprZs2dr7NixmjNnjmJiYgq3d+nSRTfddJPatm2rHj166P3339eZZ56pl19+ucxjPfTQQ8rMzCy8paamVvwDQS+9ZH7J9+ljFn0DUP1ccolpfZg40XRVWrPGzIl/ww3m4gHgTn6+GYA/e7a5z88vuU9mpll74YwzTAtsSorpAnvjjdIPP0hLl5q1YFiDAfA+vwoODRs2VHBwcInWhX379pVohTjZnDlzNGTIEL3//vu68MILy903KChInTp1KrfFISwsTJGRkS43VExmpjR9unl8//321gKgcoWGSqNHm4HSd95pBvTPnm1mtBk/Xjp2zO4K4a/mzjWzd/XpY8Jmnz7m+dy55vUtW8y4r/h48zO2fbtZn+bhh83jWbOkTp3sqx8IBH4VHEJDQ5WcnKwFCxa4bF+wYIG6detW5tfNnj1bt956q959911deumlbt/HsiytX79ejRo1Ou2a4d7rr0uHD0utWkkXX2x3NQB8ISbGXDBYs8ZMHXz8uPT44+b3wPvvu64KDjhXjC8+GYRkVi2/+mrT7a1lS7MOw5EjUuvW5ucrNVV66impSRN76gYCjV8FB0kaPXq0Xn/9db3xxhvatGmTRo0apZ07d2rYsGGSTBeim2++uXD/2bNn6+abb9bEiRPVpUsXpaenKz09XZmZmYX7jBs3TvPnz9fWrVu1fv16DRkyROvXry88JipPbm7R1KJjxphVZwEEjvbtpcWLzRinpk3NvPmDBpl+6W7mqECAyM83LQmlhUnntlWrzOMBA8y6Ib/8Ylq0atXyba1AoPO707hBgwZp8uTJGj9+vNq1a6eUlBTNmzdPiYmJkqS0tDSXNR1effVV5eXlafjw4WrUqFHhbcSIEYX7HDp0SHfddZdatWql/v37a/fu3UpJSVHnzp19/vkCzQcfmCtCMTGm/ymAwONwmNnUNm0yq02Hh5tVwZOTpbvuMoOpi/OknzuqjyVLSrY0lOatt8wg/H79yl4gE0Dl8rt1HPwV6zicOssyJwbr1pnp8R591O6KAPiD1FTpn/80wUAyA6kff1y65x6zUNfJ65fEx5uWS6Yirh7y86XffjN/G9aulb78Utq40f3XvfuumWEOgPd5ep5LcPAQweHULVwo9e1rri6mpkoNGthdEQB/snSpCQlr15rnjRqZBbxO5ry6/OGHhIeqJjtb2rDBfI/XrTO3H3+s2CD5hQvNtNQAvM/T81wmK0OleeEFc3/bbYQGACV17276rs+cKT34YOmhQTKtlw6HWZX6iitYz8RfHT5sQoEzIKxda0JDXl7JfWvVktq1M2Ng2rY1LdL795c+zsHhMK1OPXpU+kcA4AbBAZVi40bTF9XhkEaNsrsaAP4qKEi6/XYpNlb661/L3s+yTMtlSoqZphP2yshwDQjr1pkpeEs78a9Xz6zf07590X3Llq4BsEEDM6uSw+F6DGdr0+TJBEbAHxAcUClefNHcDxxoFukBgPJkZXm238UXm5PO5s3NLSmp6HGzZlLt2pVaZsCxLDPepHhAWLeu7AX9mjRxDQjt25vZtNwNZr7qKtMVrbTxLZMn00UN8BeMcfAQYxw8l54uJSZKOTmmD/P559tdEQB/t2iRd1oSYmNdA0XxYNGkCVety1NQIP3+u2tAWLtWOnCg9P3POMM1ILRvb2bQOx35+WaWpbQ0M+alRw++Z4AvMMYBtvnPf0xo6NJFKmfdPgAo1KOHubq8e3fZ/dybNJEWLJB27JC2bpW2bTP3zltmprR3r7mtWFHyGDVqmIsaJwcK5+N69Sr/c/qLnBzTpbR4QPjxR7O42smCg6VzzikKBx06mHEJlXENLTiYAdCAPyM4wKuOHZOmTjWPx4xhrm0AngkONlOultfP/aWXpLPPNrfS/PGHa6AoHix27DALUv7+u7mVJiqq9C5QSUkmcISFefcz+8rRo9JPP7l2N/rlFxMeThYeLp17rmt3ozZtpJo1fV83AP9DVyUP0VXJM1OnSsOHmz+0mzfTxAzg1MydW7Kfe0LC6fdzz883rRknBwrn4717y/9658w+pXWBSkqS4uK8c6HkdLvqHDxY1IrgvP36q+mGdLKoqJLjEc46SwrhkiIQcFjHwcsIDu7l55srgb//Lv3739K999pdEYCqyI5+7kePStu3lx0s3K07EB5uAkRZwaJOHfc1lBaaylr8zrLMv4+zBcF5v2NH6ceOi3MNCB06mMHktAoDkAgOXkdwcO/jj80ft3r1pJ07PftDCQD+zrKkfftK7wK1bZuZYai0K/rFRUeXPWg7Pl767DPTTevkv8jOE/uXXzYDj4sPXN63r/T3at7cNSC0b2+CAwCUheDgZQQH97p3l5Ytkx56SHr6aburAQDfyMkxF0tKa6nYutWMvSiPszUlP//U3jcoSGrVyjUgtGsn1a1bkU8BIJAxqxJ8auVKExpq1KCLEoDAEhpqpiYta82aQ4dMkCgtWGzfXvog5dKceaaZccjZmvCXv5gVmAHAVwgO8IqJE839jTeaPskAAKNu3aKT/ZMVFJhJJTy54DJ2rHT99d6uDgA8F2R3Aaj6tm41g/okMwUrAMAzQUFmulNPcFEGgN0IDjhtkyebq2YXXeT5H0AAgOFc/K6sGY4cDjMlbY8evq0LAE5GcMBpOXhQmjHDPL7/fntrAYCqyLn4nVQyPDifT57MujgA7EdwwGl59VUzv3nbttIFF9hdDQBUTVddJX34odSkiev2+Hiz/XQWvwMAb2FwNCosO9ss9CaZsQ0sJAQAFXfVVdIVV/h+8TsA8BTBARU2e7aUnm6ukA0aZHc1AFD1BQebKVcBwB/RVQkVYllFU7Ded5+ZxxwAAADVF8EBFfL119Ivv0h16kh33WV3NQAAAKhsBAdUyAsvmPs77jCLGwEAAKB6IzjglP34o/TNN6Yv7ogRdlcDAAAAXyA44JQ5xzZcc43UrJmtpQAAAMBHCA44Jbt2mdmUJBZ8AwAACCQEB5ySl1+W8vKknj2ljh3trgYAAAC+QnCAxw4fNitFS7Q2AAAABBqCAzw2Y4aUmSmddZZ06aV2VwMAAABfIjjAI3l50uTJ5vHo0VIQPzkAAAABhdM/eOSjj6QdO6ToaGnwYLurAQAAgK8RHOCWZRVNwTp8uBQebm89AAAA8D2CA9xaskRatUqqWVO6+267qwEAAIAdCA5wy9nacMstpqsSAAAAAg/BAeX69Vfps8/M41Gj7K0FAAAA9iE4oFyTJpn7yy8307ACAAAgMBEcUKb9+6W33jKPx4yxtxYAAADYi+CAMk2dKp04IXXqJPXoYXc1AAAAsBPBAaU6flyaMsU8HjNGcjjsrQcAAAD2IjigVO+8I2VkSImJ0tVX210NAAAA7EZwQAkFBUVTsI4cKYWE2FoOAAAA/ADBASV88YX0229SVJQ0ZIjd1QAAAMAfEBxQwgsvmPuhQ6WICHtrAQAAgH8gOMDFqlVSSorpnnTffXZXAwAAAH9BcIAL59iG66+XmjSxtxYAAAD4D4IDCm3fLn34oXnMgm8AAAAozi+Dw9SpU5WUlKSaNWsqOTlZS5YsKXPfuXPnql+/foqOjlZkZKS6du2q+fPnl9jvo48+UuvWrRUWFqbWrVvr448/rsyPUCW99JKUny9deKHUtq3d1QAAAMCf+F1wmDNnjkaOHKlHHnlE69atU48ePTRgwADt3Lmz1P1TUlLUr18/zZs3T2vWrFGfPn102WWXad26dYX7rFixQoMGDdLgwYP1448/avDgwbr22mv1/fff++pj+b1Dh6TXXzeP77/f1lIAAADghxyWZVl2F1Hceeedpw4dOmjatGmF21q1aqWBAwdqwoQJHh3jnHPO0aBBg/Svf/1LkjRo0CBlZWXpyy+/LNzn4osvVr169TR79uxSj5Gdna3s7OzC51lZWUpISFBmZqYiIyMr8tH82nPPSf/8p9SmjfTTT6wUDQAAECiysrIUFRXl9jzXr1occnJytGbNGvXv399le//+/bV8+XKPjlFQUKDDhw+rfv36hdtWrFhR4pgXXXRRucecMGGCoqKiCm8JCQmn8Emqlpwc001JMmMbCA0AAAA4mV8Fh4yMDOXn5ys2NtZle2xsrNLT0z06xsSJE3X06FFde+21hdvS09NP+ZgPPfSQMjMzC2+pqamn8EmqljlzpD17pLg4M5sSAAAAcLIQuwsojeOkS96WZZXYVprZs2dr7Nix+vTTTxUTE3NaxwwLC1NYWNgpVF01WVbRgm/33ScFwEcGAABABfhVcGjYsKGCg4NLtATs27evRIvByebMmaMhQ4bogw8+0IUXXujyWlxcXIWOGQi+/daMaahd26wUDQAAAJTGr7oqhYaGKjk5WQsWLHDZvmDBAnXr1q3Mr5s9e7ZuvfVWvfvuu7r00ktLvN61a9cSx/z666/LPWagcLY23H67VGxYCAAAAODCr1ocJGn06NEaPHiwOnbsqK5du2r69OnauXOnhg0bJsmMPdi9e7fefvttSSY03HzzzXrppZfUpUuXwpaF8PBwRUVFSZJGjBihnj176tlnn9UVV1yhTz/9VN98842WLl1qz4f0E7/8Is2fLwUFSSNH2l0NAAAA/JlftThIZurUyZMna/z48WrXrp1SUlI0b948JSYmSpLS0tJc1nR49dVXlZeXp+HDh6tRo0aFtxEjRhTu061bN7333nt68803de6552rmzJmaM2eOzjvvPJ9/Pn8ycaK5v+oqqXlze2sBAACAf/O7dRz8lafz21YVaWlSYqKUmyutXCkFeIYCAAAIWFVyHQf4zssvm9Bw/vmEBgAAALhHcAhAR45Ir7xiHt9/v721AAAAoGogOASgN9+U/vhDOuMM6bLL7K4GAAAAVQHBIcDk50uTJpnHo0dLwcH21gMAAICqgeAQYD7+WNq2TWrQQLrlFrurAQAAQFVBcAggllW04Nvdd0u1atlbDwAAAKoOgkMAWb5c+v57KSxMGj7c7moAAABQlRAcAohzwbfBg6XYWHtrAQAAQNVCcAgQmzdLn3xiHo8ebWspAAAAqIIIDgFi8mQzxuHSS6VWreyuBgAAAFUNwSEAHDhg1m6QpDFj7K0FAAAAVRPBIQBMmyYdPy516CD17m13NQAAAKiKCA7V3IkT0ssvm8djxkgOh731AAAAoGoiOFRz//2vtG+flJAg/e1vdlcDAACAqorgUI0VFBRNwTpihFSjhr31AAAAoOoiOFRjX30lbdokRUZKd95pdzUAAACoyggO1dgLL5j7O+804QEAAACoKIJDNbV2rbRwoRQSYropAQAAAKeD4FBNOcc2XHutGRgNAAAAnA6CQzWUmirNmWMes+AbAAAAvIHgUA299JKUny/16WMWfQMAAABOF8GhmsnMlKZPN4/vv9/eWgAAAFB9EByqmddflw4fllq1ki6+2O5qAAAAUF0QHKqR3FzTTUkyYxuC+O4CAADASzi1rEY++MAMjI6JkW680e5qAAAAUJ0QHKoJyyqagvXee6WaNe2tBwAAANULwaGaWLTILPoWHi79/e92VwMAAIDqhuBQTThbG267TWrQwN5aAAAAUP0QHKqBTZukL76QHA5p1Ci7qwEAAEB1RHCoBl580dwPHCidcYatpQAAAKCaIjhUcXv3Sm+/bR6z4BsAAAAqC8GhivvPf6ScHKlLF6lbN7urAQAAQHVFcKjCjh2Tpk41j2ltAAAAQGUiOFRhb70lHTggNW9uxjcAAAAAlYXgUEXl5xcNih41SgoOtrceAAAAVG8Ehyrq88+l33+X6tUzazcAAAAAlYngUEW98IK5//vfpdq17a0FAAAA1R/BoQpauVJatkwKDZXuucfuagAAABAICA5V0MSJ5v7GG6VGjeytBQAAAIGB4FDFbN0qzZ1rHo8ebW8tAAAACBwVDg7jx4/XrFmzvFkLPDB5slRQIF18sdSmjd3VAAAAIFBUODg8+eST+vnnn71ZC8qQny8tWiS99po0fbrZNmaMrSUBAAAgwIRU9AsTExN18OBBb9aCUsydK40YIe3aVbStRg0pM9O+mgAAABB4KtzicP3112v+/PnK5Ay20sydK11zjWtokKTcXOlvfysa6wAAAABUtgoHh0cffVTnnnuu+vbtqy+++EL79u3zWlFTp05VUlKSatasqeTkZC1ZsqTMfdPS0nTDDTforLPOUlBQkEaOHFlin5kzZ8rhcJS4nThxwms1e1t+vmlpsKyy9xk50uwHAAAAVLYKd1UKDw+XJFmWpcsvv7zM/RwOh/Ly8jw+7pw5czRy5EhNnTpV559/vl599VUNGDBAGzduVNOmTUvsn52drejoaD3yyCOaNGlSmceNjIzUr7/+6rKtZs2aHtfla0uWlGxpKM6ypNRUs1/v3j4rCwAAAAGqwsGhR48ecjgc3qxFkvTiiy9qyJAhuuOOOyRJkydP1vz58zVt2jRNmDChxP7NmjXTSy+9JEl64403yjyuw+FQXFyc1+utLGlp3t0PAAAAOB0VDg6LFi3yYhlGTk6O1qxZowcffNBle//+/bV8+fLTOvaRI0eUmJio/Px8tWvXTk888YTat29f5v7Z2dnKzs4ufJ6VlXVa73+qPF3YjQXgAAAA4At+tQBcRkaG8vPzFRsb67I9NjZW6enpFT7u2WefrZkzZ+qzzz7T7NmzVbNmTZ1//vnavHlzmV8zYcIERUVFFd4SEhIq/P4V0aOHFB8vldWo43BICQlmPwAAAKCyeSU47N69W/PmzdPs2bP1xRdfaPfu3ad1vJO7QFmWdVrdorp06aKbbrpJbdu2VY8ePfT+++/rzDPP1Msvv1zm1zz00EPKzMwsvKWmplb4/SsiOFj6swdWifDgfD55stmv2lu9Wurb19wDAADAFhXuqiRJW7du1bBhw/Ttt9+WeO2CCy7Q1KlTdcYZZ3h8vIYNGyo4OLhE68K+fftKtEKcjqCgIHXq1KncFoewsDCFhYV57T0r4qqrpA8/LLmOQ3y8CQ1XXWVbab719tvSwoXSO+9IHTvaXQ0AAEBAqnBw2LVrl84//3zt3btXrVq1Us+ePRUXF6e9e/dqyZIl+uabb9SjRw/98MMPHnfzCQ0NVXJyshYsWKArr7yycPuCBQt0xRVXVLTUEizL0vr16/WXv/zFa8esLFddJV1xhZk9KS3NjGno0SMAWhp27JAyMkzzypw5Ztt770m33GKmlGrYUEpMtLdGAACAAFLh4DB27Fjt3btX06dPL5wBqbgZM2borrvu0vjx4/Xaa695fNzRo0dr8ODB6tixo7p27arp06dr586dGjZsmCTThWj37t16++23C79m/fr1kswA6P3792v9+vUKDQ1V69atJUnjxo1Tly5d1LJlS2VlZenf//631q9fr//85z8V/fg+FRwcgFOuNmtWctu+fVJyctHz8ha5AAAAgFdVODjMnz9fl19+eamhQZKGDBmizz//XF9++eUpHXfQoEE6cOCAxo8fr7S0NLVp00bz5s1T4p9Xl9PS0rRz506Xryk+O9KaNWv07rvvKjExUdu3b5ckHTp0SHfddZfS09MVFRWl9u3bKyUlRZ07dz6l2uBDs2ZJt94qlbYGiMMhPf64z0sCAAAIZA7Lqthl27CwMN1///166qmnytznkUce0QsvvOAyrWlVlZWVpaioKGVmZioyMtLucgLD2rWuLQwn69lTeuAB6ZJLpCC/miAMAACgyvD0PLfCZ1vR0dHasGFDufts3LhR0dHRFX0LwJUzHFx+uVSjhpSSIl12mfSXv0gzZ0o5ObaWBwAAUJ1VODhcdNFF+vzzzzVjxoxSX3/jjTf0+eef6+KLL65wcQhw+flFj8eONa0PcXHSf/4jbdsm/eMfUmSktHGjdNttUvPm0gsvSD5erA8AACAQVLirUmpqqjp27KiMjAy1bt1avXr1UmxsrPbu3auUlBRt2LBBDRs21OrVq32+eFploKuSDZ58UnrsMal7dzOtlGWZVoXi0+RmZkrTp5v5affsMdsiI6W//1267z6pcWNbSge8YvVqE5Cfe46piAEAlcbT89wKBwdJ2rx5s4YNG6aFCxeWeK1Pnz6aNm2azjzzzIoe3q8QHHysoEBq0ULavt2s4zB4cPn7Z2dL774rPf+8tGmT2Vajhvm6+++XWrWq9JIBr7vvPunll829c0VIAAC8zCfBwWnXrl1at26dsrKyFBkZqXbt2lWLVobiCA4+tmCB1L+/FBVlWhJq1fLs6woKpHnzzBXaJUuKtl9+ublye/75lVMv4C3F1zAZMMBMQxwTI335JWuYAAAqRaUHh759+6p7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27NVqB0OKTbWNUycHDKaNGE19qrEsqQvvjCBYdUqsy083Ax6fuABEyYBVAgtDrCfs7XhyisJDQBOTVCQVFBQdP/GG2bwdEGBtG+ftGuXuaWmFj0ufsvJkdLTza28mZmio8tutXDewsN997lhvl//+If03HNSx44mMHz+uQkMa9aYfWrVku6+W7r/fhMQAfgEwQGV49gx0+1AYu0GAJ6LiTFXjhMSzExLM2aYcBATY153dlOKizMnlaWxLGn//tIDRfGgceKE2W//fmndurJratCg7FYL5612be/9G5x84hxonDNqvf22aWEaN67o+1O7tpklacyYop8JAD7jl12Vpk6dqueff15paWk655xzNHnyZPXo0aPUfdPS0jRmzBitWbNGmzdv1n333afJkyeX2O+jjz7SY489pi1btqhFixZ66qmndOWVV3pcE12VTtE770g332ymWvz9d/PHHgA8kZ1txic4HCYE5OR4v0uRZZlZm8pruUhNNRdBPFG3btktFs7tERGeHeu++6SXXzb3L71U4Y9YpezYIWVkmO/5gAGmVSkkpKhbWq1a5t9jzBhasIFKUGW7Ks2ZM0cjR47U1KlTdf755+vVV1/VgAEDtHHjRjUtZf7l7OxsRUdH65FHHtGkSZNKPeaKFSs0aNAgPfHEE7ryyiv18ccf69prr9XSpUt13nnnVfZHCkzOtRuGDCE0ADg1xUOCw1E54xAcDtOS0KCBGT9RGssy4yrKa7lITZWOHDH7HTok/fxz2e8ZGVl2q0VQkAlLERFmkTxJeu89M1j8VKeitSxzwp2ba27Ox97YVlnHLa07WfGxLMeOSRMmePb5AVQav2txOO+889ShQwdNmzatcFurVq00cOBATXDzS6N3795q165diRaHQYMGKSsrS19++WXhtosvvlj16tXT7NmzSz1Wdna2srOzC59nZWUpISGBFgdP/PqrdPbZ5g/hzp3MoQ2gesvKcj/mwrlGxelo08azE/LyZqWqikJCpJkzpRtvtLsSoNqqki0OOTk5WrNmjR588EGX7f3799dy56IuFbBixQqNGjXKZdtFF11UapcmpwkTJmjcuHEVfs+A5hwUfemlhAYA1V9kpNS6tbmV5fBh01+/rJaLLVuko0fLf59ffjm9OmvUMLeQkJKP/WHb1q1mhqSTff+9GRgPwHZ+FRwyMjKUn5+v2JNmSIiNjVV6enqFj5uenn7Kx3zooYc0evTowufOFge4kZMjvfWWeXzHHfbWAgD+IiLCtMSefXbZ+yxbJnXvXnL7pEnm607nBD04uPI+m7esXWvuT55RC4Df8Kvg4OQ4aaEwy7JKbKvsY4aFhSmM+b1P3WefmRlKGjWSLrnE7moAoOpwTvt68olzz56BccXd3YxaAGznV8GhYcOGCg4OLtESsG/fvhItBqciLi7O68dEGZzdlG67zVzpAgB4JtBPnOPjpe3bi2bUuuuuyplRC0CF+dV0N6GhoUpOTtaCBQtcti9YsEDdunWr8HG7du1a4phff/31aR0Tpdi+Xfr6a/N4yBBbSwGAKsd54vz999LQoeZ++3azPVCEhZnQIFXejFoAKszvLgmPHj1agwcPVseOHdW1a1dNnz5dO3fu1LBhwySZsQe7d+/W22+/Xfg169evlyQdOXJE+/fv1/r16xUaGqrWfw5UGzFihHr27Klnn31WV1xxhT799FN98803Wrp0qc8/X7X25ptmGsALLpCaN7e7GgCoenwxFS0AVJDfBYdBgwbpwIEDGj9+vNLS0tSmTRvNmzdPiX/OX52WlqadO3e6fE379u0LH69Zs0bvvvuuEhMTtX37dklSt27d9N577+nRRx/VY489phYtWmjOnDms4eBN+fnSG2+Yx6wUDQAAUO343ToO/oqVo92YN89Mv9qggZlykKtkAAAAVYKn57l+NcYBVZhzpeibbyY0AAAAVEMEB5y+9HTp88/NY9ZuAAAAqJYIDjh9M2eaMQ7dupW/cioAAACqLIIDTo9lFa3dQGsDAABAtUVwwOlZtEjaskWKiJCuvdbuagAAAFBJCA44Pc7WhhtukGrXtrcWAAAAVBqCAyru4EHpo4/MY9ZuAAAAqNYIDqi4WbOk7GypXTupQwe7qwEAAEAlIjigYiyraO2GO++UHA576wEAAEClIjigYr7/XvrlFyk83IxvAAAAQLVGcEDFOAdF/+1vUt26tpYCAACAykdwwKk7fFh67z3zmLUbAAAAAgLBAafuvfeko0els86Sune3uxoAAAD4AMEBp845KPqOOxgUDQAAECAIDjg1P/4orVol1agh3Xyz3dUAAADARwgOODXOQdFXXCHFxNhbCwAAAHyG4ADPHT9uFn2TWCkaAAAgwBAc4LmPPpIOHZISE6ULL7S7GgAAAPgQwQGec3ZTGjJECuJHBwAAIJBw9gfP/PabtHixCQy33WZ3NQAAAPAxggM8M2OGuR8wQIqPt7cWAAAA+BzBAe7l5EgzZ5rHrBQNAAAQkAgOcO///k/at0+Ki5MuvdTuagAAAGADggPcc64UfeutZuE3AAAABByCA8q3c6c0f755PGSIvbUAAADANgQHlO+NNyTLkvr0kc44w+5qAAAAYBOCA8qWn2+Cg8RK0QAAAAGO4ICyff21lJoq1a8vXXml3dUAAADARgQHlM25UvTgwVLNmvbWAgAAAFsRHFC6vXulzz4zj1m7AQAAIOARHFC6t96S8vKkLl2kNm3srgYAAAA2IzigJMsq6qZEawMAAABEcEBpUlKkzZulOnWkQYPsrgYAAAB+gOCAkpwrRV9/vQkPAAAACHgEB7j64w/pww/NY9ZuAAAAwJ8IDnA1a5aUnS21bSt17Gh3NQAAAPATBAcUsayibkp33CE5HPbWAwAAAL9BcECRVaukn382i73deKPd1QAAAMCPEBxQxDkF6zXXSPXq2VsLAAAA/ArBAcaRI9Ls2eYxg6IBAABwEoIDjDlzTHg480ypRw+7qwEAAICfITjAYFA0AAAAykFwgBkQ/f33UkiIdPPNdlcDAAAAP0RwQNGg6CuukGJj7a0FAAAAfskvg8PUqVOVlJSkmjVrKjk5WUuWLCl3/8WLFys5OVk1a9ZU8+bN9corr7i8PnPmTDkcjhK3EydOVObHqBpOnJDeecc8vuMOe2sBAACA3wqxu4CTzZkzRyNHjtTUqVN1/vnn69VXX9WAAQO0ceNGNW3atMT+27Zt0yWXXKI777xTs2bN0rJly3T33XcrOjpaV199deF+kZGR+vXXX12+tmbNmpX+efze3LnSH39ITZtK/frZXQ0AAJCUm5ur/Px8u8tAFRUcHKwaNWp4/bh+FxxefPFFDRkyRHf8efV78uTJmj9/vqZNm6YJEyaU2P+VV15R06ZNNXnyZElSq1attHr1ar3wwgsuwcHhcCguLs4nn6FKcXZTuv12KTjY3loAAAhwWVlZysjIUHZ2tt2loIoLCwtTw4YNFRkZ6bVj+lVwyMnJ0Zo1a/Tggw+6bO/fv7+WL19e6tesWLFC/fv3d9l20UUXacaMGcrNzS1MW0eOHFFiYqLy8/PVrl07PfHEE2rfvn2ZtWRnZ7v8p83Kyqrox/Jfv/8uLVxoZlG6/Xa7qwEAIKBlZWVp9+7dqlOnjho2bKgaNWrIwUyHOEWWZSk3N1eZmZnavXu3JHktPPhVcMjIyFB+fr5iTxqgGxsbq/T09FK/Jj09vdT98/LylJGRoUaNGunss8/WzJkz9Ze//EVZWVl66aWXdP755+vHH39Uy5YtSz3uhAkTNG7cOO98MH81Y4a5v/hiKSHB3loAAAhwGRkZqlOnjuLj4wkMOC3h4eGKiIjQrl27lJGR4bXg4JeDo0/+z2JZVrn/gUrbv/j2Ll266KabblLbtm3Vo0cPvf/++zrzzDP18ssvl3nMhx56SJmZmYW31NTUin4c/5SbK735pnnMStEAANgqNzdX2dnZioqKIjTAKxwOh6KiopSdna3c3FyvHNOvWhwaNmyo4ODgEq0L+/btK9Gq4BQXF1fq/iEhIWrQoEGpXxMUFKROnTpp8+bNZdYSFhamsLCwU/wEVcgXX0h795rpV//6V7urAQAgoDkHQlfGgFYELufPU35+vld+tvyqxSE0NFTJyclasGCBy/YFCxaoW7dupX5N165dS+z/9ddfq2PHjmX+A1mWpfXr16tRo0beKbwqcq4UfeutEr+kAADwC7Q2wJu8/fPkV8FBkkaPHq3XX39db7zxhjZt2qRRo0Zp586dGjZsmCTThejmYqsbDxs2TDt27NDo0aO1adMmvfHGG5oxY4buv//+wn3GjRun+fPna+vWrVq/fr2GDBmi9evXFx4z4KSmSl99ZR4PGWJvLQAAAKgS/KqrkiQNGjRIBw4c0Pjx45WWlqY2bdpo3rx5SkxMlCSlpaVp586dhfsnJSVp3rx5GjVqlP7zn/+ocePG+ve//+0yFeuhQ4d01113KT09XVFRUWrfvr1SUlLUuXNnn38+v/Dmm1JBgdS7t1TG4HAAAACgOIflHEmMcmVlZSkqKkqZmZlenQ/X5/LzpebNpZ07pVmzpBtvtLsiAAAC3okTJ7Rt2zYlJSUF7AK127dvV1JSkm655RbNnDnT7nJ8ZuzYsRo3bpwWLlyo3r17e/XYnv5ceXqe63ddlVDJvvnGhIZ69aRirTIAAABwb/v27XI4HLr11lvtLsXn/K6rEiqZc6Xom26SAvSKBgAA8D9NmjTRpk2bFBUVZXcpPnXPPffouuuuU9OmTe0uxS2CQyDZt0/69FPz+I477K0FAACgmBo1aujss8+2uwyfa9iwoRo2bGh3GR6hq1Igeftts/Bb587SuefaXQ0AALDD6tVS377m3o+U1QXo8OHDGj9+vM4991zVrl27cKKbxx57zOOFzZo1a6ZmzZrp0KFDuu+++5SQkKCQkBCXsRQ//fSTrrvuOjVq1EihoaFKTEzUvffeqwMHDhTuM3PmTCUlJUmS3nrrLTkcjsLbokWLJEl79uzR448/ri5duigmJkZhYWFq1qyZ7r77bu3bt69EbWPHjnX5+pP/LbZu3aprrrlG9erVU+3atXXhhRfqxx9/9Owf1ctocQgUllXUTYmVogEACFxvvy0tXCi9847UsaPd1ZQrIyNDvXr10saNG9WuXTsNGzZMBQUF+t///qdnn31WY8aMUd26dT06VnZ2tvr27avDhw/rsssuU2hoaOECw5999pmuvfZaBQcH6/LLL1dCQoI2btyoKVOmaP78+fr+++9Vr149tWvXTiNGjNBLL72ktm3bauDAgYXHb9asmSQpJSVFEydO1AUXXKDzzjtPNWrU0Lp16zRt2jTNnz9fa9eu9bg71vbt23XeeeepdevWuv3227VlyxZ9+umn6tOnjzZt2lTmAsmVheAQKJYulX79VapdWxo0yO5qAACAJyxLOnbs9I+zc6d04IDkcEjvvWe2zZ4tXXuteY8GDaTT6WNfq5Y5tpfdfffd2rhxox5++GE99dRTLq/t3btXderU8fhY6enpOvfcc7Vs2TKFh4cXbj9w4IAGDx6s6OhoLVu2zGWswezZs3XDDTfoX//6l15++WW1a9dOI0eO1EsvvaR27dpp7NixJd6nb9++Sk9PL1Hb22+/rVtuuUVTpkzRI4884lHNixcv1jPPPKN//vOfhdsee+wxPfnkk3rzzTf14IMPevz5vYGuSoHCuVL09ddLERH21gIAADxz7JhUp87p31q3lnr0kLp3l/bvN8fev98879HDvH46x/dGuDnJ3r179eGHH6pFixalnqDHxsYqJOTUroE///zzLqFBMif0WVlZmjBhQokBytdff706dOig95xhywMxMTGlBprBgwcrMjJS33zzjcfHSkpK0gMPPOCybcifi/euWrXK4+N4Cy0OgeDQIemDD8xjBkUDAIAqYPXq1bIsS3369FGNGjXK3XfmzJnavn27y7aBAweqXbt2hc9r1qypv/zlLyW+duXKlYX3v//+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1KhR6tatm55++mlde+21+uGHHzR9+nRNnz7d7tJQyS677DI99dRTatq0qc455xytW7dOL774om6//Xa7S4OXHTlyRL///nvh823btmn9+vWqX7++mjZtqpEjR+rpp59Wy5Yt1bJlSz399NOqVauWbrjhBhurLsaC3/vPf/5jJSYmWqGhoVaHDh2sxYsX210SKpmkUm9vvvmm3aXBBr169bJGjBhhdxnwgc8//9xq06aNFRYWZp199tnW9OnT7S4JPpCVlWWNGDHCatq0qVWzZk2refPm1iOPPGJlZ2fbXRq8bOHChaX+fb/lllssy7KsgoIC6/HHH7fi4uKssLAwq2fPntbPP/9sb9HFsI4DAAAAALcY4wAAAADALYIDAAAAALcIDgAAAADcIjgAAAAAcIvgAAAAAMAtggMAAAAAtwgOAAAAANwiOAAAAABwi+AAAKi2tm/fLofDoVtvvdXuUgCgyiM4AAAAAHCL4AAAAADALYIDAAAAALcIDgAAj6WkpOiyyy5Tw4YNFRYWppYtW+rRRx/VsWPHCvdZtGiRHA6Hxo4dq5SUFPXq1Ut16tRR/fr1dcMNN2jXrl2lHnvDhg0aNGiQYmJiFBYWpqSkJI0aNUoHDx4sdf99+/bp/vvv11lnnaWaNWuqfv366tKliyZOnFjq/lu3btU111yjevXqqXbt2rrwwgv1448/nv4/CgAECIdlWZbdRQAA/N8rr7yiu+++W/Xq1dNll12m6OhorVq1SosXL1a3bt20cOFChYaGatGiRerTp48uuugiLVy4UJdeeqnOPvtsrV27VvPnz1dCQoJWrVql2NjYwmMvX75c/fv3V3Z2tq655ho1a9ZMK1eu1KJFi9SyZUutWLFCDRo0KNx/8+bN6tOnj3bv3q3u3burW7duOnr0qH755Rf99NNPhWFj+/btSkpKUq9evbRhwwa1bt1aHTt21JYtW/Tpp5+qXr162rRpk0stAIAyWAAAuLFhwwYrJCTEat++vXXgwAGX1yZMmGBJsl544QXLsixr4cKFliRLkvX666+77Dtu3DhLknX77bcXbsvPz7datmxpSbK++uorl/0feughS5I1ZMgQl+2dO3e2JFnTp08vUWtqamrh423bthXW8swzz7js9+ijj1qSrAkTJpzCvwQABC5aHAAAbo0YMUL//ve/tWTJEnXv3t3ltYKCAsXFxalp06ZavXp1YYvDWWedpU2bNsnhcBTue/z4cSUmJurIkSM6dOiQQkNDtWTJEvXs2VMDBgzQvHnzXI599OhRJSYm6tixY4X7r1q1Sp07d1bPnj21ePHicut2tjgkJSXp999/V1BQUInXrrrqKn300Ude+FcCgOotxO4CAAD+b+XKlZKkr776St98802J12vUqKH//e9/LtvOP/98l9AgSeHh4UpOTtZXX32l3377TW3atNG6deskSb179y5x3Nq1a6tjx46aP39+4f4//PCDJKl///4e19+2bVuX0CBJ8fHxkqRDhw55fBwACGQEBwCAW84xA0899ZTHXxMTE1Pqdud4gszMTElSVlaWy/aTxcXFuezvPNFv0qSJx7VERUWV2BYSYv4E5ufne3wcAAhkzKoEAHArMjJSkjnJtyyrzFtx+/btK/VYe/fulVR0Mu88tnN7Wfs796tbt64kaffu3afxiQAAp4rgAABw67zzzpNU1GXJE8uWLSsRJo4fP641a9YoPDxcZ555piSpffv2ksw0ric7duyYVq9erfDwcJ111lmSpM6dO0uSvv7661P+HACAiiM4AADcuvvuuxUSEqJ7771XqampJV4/dOhQ4VgFp19//VVvvPGGy7bnn39e+/fv1/XXX6/Q0FBJZixEixYt9OWXX5YYPzFhwgRlZGS47N+pUyd17txZKSkpeu2110rUQksEAFQOxjgAANxq06aNpk6dqr///e8666yzdMkll6hFixbKysrS1q1btXjxYt1666165ZVXCr+mf//+uvvuu/XFF1+UWMfh6aefLtwvKChIM2fO1EUXXaRLLrlEf/vb35SYmKjvv/9e3333nVq0aKFnnnnGpZ5Zs2apd+/euuuuu/TOO++oa9euOnHihDZs2KB169bpwIEDPvu3AYBAQYsDAMAjd955p1asWKErrrhCK1as0KRJk/Thhx8qIyNDo0aN0siRI13279q1qxYsWKCMjAy99NJL+v7773Xddddp2bJlJQZCd+/eXStXrtQVV1yhr7/+Wi+88IK2bNmi++67TytXrlR0dLTL/i1bttTatWs1YsQI7d69W5MnT9asWbN05MgRPfroo5X9TwEAAYl1HAAAXuVcx+Hxxx/X2LFj7S4HAOAltDgAAAAAcIvgAAAAAMAtggMAAAAAtxjjAAAAAMAtWhwAAAAAuEVwAAAA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r7SxagAAAKDqqBbBYfXq1erTp0/h89GjR0uSbrnlFk2bNk0///yz3n77bR06dEiNGjVSnz59NGfOHEVERNhVMgAAAFClVLvB0ZXF00EjAAAAQFXi6XluQI5xAAAAAHBqCA4AAAAA3CI4AAAAAHCL4AAAAADALYIDAAAAALcIDgAAAADcIjgAAAAAcMunwWH8+PGaNWuWL98SAAAAgBf4NDg8+eST+vnnn335lgAAAAC8wKfBITExUQcPHvTlWwIAAADwAp8Gh+uvv17z589XZmamL98WAAAAwGnyaXB49NFHde6556pv37764osvtG/fPl++PQAAAIAKCvHlm4WHh0uSLMvS5ZdfXuZ+DodDeXl5vioLAAAAgBs+DQ49evSQw+Hw5VsCAAAA8AKfBodFixb58u0AAAAAeAkLwAEAAABwy6ctDsXt3r1bP/74ozIzMxUZGal27dqpSZMmdpUDAAAAoBw+Dw5bt27VsGHD9O2335Z47YILLtDUqVN1xhln+LosAAAAAOXwaXDYtWuXzj//fO3du1etWrVSz549FRcXp71792rJkiX65ptv1KNHD/3www9KSEjwZWkAAAAAyuHT4DB27Fjt3btX06dP1x133FHi9RkzZuiuu+7S+PHj9dprr/myNAAAAADlcFiWZfnqzRISEpScnKxPPvmkzH0GDhyo1atXa9euXb4qyyNZWVmKiooqHJMBAAAAVAeenuf6dFalffv26Zxzzil3n3POOUf79+/3UUUAAAAAPOHT4BAdHa0NGzaUu8/GjRsVHR3to4oAAAAAeMKnweGiiy7S559/rhkzZpT6+htvvKHPP/9cF198sS/LAgAAAOCGT8c4pKamqmPHjsrIyFDr1q3Vq1cvxcbGau/evUpJSdGGDRvUsGFDrV692u9mVWKMAwAAAKojT89zfTqrUkJCgpYuXaphw4Zp4cKFJbot9enTR9OmTfO70AAAAAAEOp8vANeyZUt9++232rVrl9atW6esrKzClaMJDAAAAIB/8mlw6Nu3r7p3767x48crPj5e8fHxvnx7AAAAABXk08HR33//vfLy8nz5lgAAAAC8wKfBoVWrVtq+fbsv3xIAAACAF/g0ONx777367LPPtHHjRl++LQAAAIDT5NMxDklJSerdu7e6dOmioUOHqlOnToqNjZXD4Sixb8+ePX1ZGgAAAIBy+HQd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quyirKwsvfPOO+rbt6/Cw8NltVoVHBysQYMGaevWrYU+57vvvlPfvn0VFBQkT09P1axZU/fee6927dpV4NwzZ85U27Zt5evrKx8fHzVq1EjPPPOMkpOTr1rX2rVrZbFY9K9//avQ/X/99ZdcXV1100032bbFxMToiSeeUJMmTeTv7y8vLy81bdpUU6dO1YULF4r187haj8P69evVrVs3eXt7KygoSHfeeafi4+OLdd7SgBGHMu7bvd9qf9J+BXgG6N+t/m12OQAA4P8YhqGc8+dv6By5OTnaPHmyZBiFfQNJ0uYpUxTSvv11T1ty9fK6ocbcpKQkjRw5Ul26dFH//v1VsWJFHTx4UN99952WLVumtWvXqk2bNrbjn3/+eU2fPl2BgYG69dZbFRwcrPj4eK1atUqRkZFq0qSJJCkjI0N9+/bV2rVrFRERoX/+85+yWq3av3+/Zs+erfvuu08VK1a8Yl1dunRRzZo1tWjRIr377rvy9PTMt//LL79Ubm6u7r33Xtu2Dz/8UN9//726du2q/v3769y5c1q9erXGjh2r6OhoLVq06Lp/Tj///LP69esnFxcX3XnnnQoLC9PPP/+sTp06XfV1lCYEhzLMMAxN+3WaJOmJNk/I1+prckUAACBPzvnzWnDJBXNJOX/ypBa2b3/dz78jOlpuFSpc9/MrVqyouLg4Va1aNd/23bt3q3379ho3bpxW/t+oyI8//qjp06eradOmioqKUlBQkO347OxsnT592vb1+PHjtXbtWt1777365JNP5HpJMEpJScn3dWEsFovuvvtuvfbaa/r+++91++2359v/5ZdfysvLS4MHD7ZtGzt2rN5999185zYMQw899JA+/vhj/frrr+rUqdM1/HQuys3N1cMPP6zs7GytXbtWnTt3tp37nnvu0VdffXXN5zQDU5XKsKjDUYo+Fi1PN0892e5Js8sBAABOyGq1FggNktS4cWP16NFDa9eutU3zeffddyVJM2fOzBcaJMnNzU0hISGSpJycHH3wwQfy9/fXzJkzC4QEf39/+fj4FFlb3mjC3Llz823fvn27du7cqVtuuUW+vn9/8FqjRo0C38tisejxxx+XJK1atarI71mY9evX6+DBg7r55pttoSHv3JMnTy4yBJUWjDiUYXmjDQ+2eFDB3sEmVwMAAC7l6uWlO6Kjb+gcCTExWj18eJHH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7u6uZcuW6Y8//tAbb7yhgIAAs0tDCZs2bZpmz56tWbNmac+ePXr99dc1ffp0vfPOO2aXBjs7e/asmjdvrlmzZhW6//XXX9ebb76pWbNmKTo6WqGhoerdu7fS0tIcXGnhuB1rKdeuXTu1atVK77//vm1bw4YNdeutt2rKlCkmVgZHSkxMVHBwsNasWaOuXbuaXQ4cID09Xa1atdJ7772nV199VS1atNDbb79tdlkoQWPGjNGvv/7KqLITuvnmmxUSEqL//ve/tm2DBw9WhQoV9MUXX5hYGUqSxWLRkiVLdOutt0q6ONoQFhamkSNHavTo0ZKkzMxMhYSEaNq0aXrkkUdMrPYiRhxKsaysLMXExKhPnz75tvfp00e//fabSVXBDCkpKZKkwMBAkyuBozz++OMaMGCAevXqZXYpcJDvvvtOrVu31u23367g4GC1bNlSH374odllwQE6d+6sn3/+Wfv27ZMkbd++XevXr1f//v1NrgyOdOjQIZ04cSLfdZ/ValW3bt1KzXWfm9kF4MpOnTqlnJwchYSE5NseEhKiEydOmFQVHM0wDD3zzDPq3LmzmjRpYnY5cID58+dry5Ytio6ONrsUONDBgwf1/vvv65lnntG4ceP0+++/66mnnpLVatV9991ndnkoQaNHj1ZKSooaNGggV1dX5eTk6LXXXtOwYcPMLg0OlHdtV9h135EjR8woqQCCQxlgsVjyfW0YRoFtKL+eeOIJ7dixQ+vXrze7FDhAfHy8RowYoRUrVsjT09PscuBAubm5at26tSZPnixJatmypXbv3q3333+f4FDOff3115o7d66++uorNW7cWNu2bdPIkSMVFham+++/3+zy4GCl+bqP4FCKVapUSa6urgVGFxISEgqkUZRPTz75pL777jutXbtW1apVM7scOEBMTIwSEhIUGRlp25aTk6O1a9dq1qxZyszMlKurq4kVoqRUqVJFjRo1yretYcOGWrRokUkVwVGee+45jRkzRkOHDpUkNW3aVEeOHNGUKVMIDk4kNDRU0sWRhypVqti2l6brPnocSjEPDw9FRkZq5cqV+bavXLlSHTt2NKkqOIJhGHriiSe0ePFi/fLLL6pVq5bZJcFBbrrpJu3cuVPbtm2zPVq3bq27775b27ZtIzSUY506dSpw2+V9+/apRo0aJlUERzl37pxcXPJfkrm6unI7VidTq1YthYaG5rvuy8rK0po1a0rNdR8jDqXcM888o3vvvVetW7dWhw4dNGfOHMXFxWn48OFml4YS9Pjjj+urr77S//73P/n6+tpGnfz9/eXl5WVydShJvr6+BXpZvL29FRQURI9LOff000+rY8eOmjx5su644w79/vvvmjNnjubMmWN2aShhAwcO1Guvvabq1aurcePG2rp1q9588009+OCDZpcGO0tPT9eBAwdsXx86dEjbtm1TYGCgqlevrpEjR2ry5MmKiIhQRESEJk+erAoVKuiuu+4ysepLGCj13n33XaNGjRqGh4eH0apVK2PNmjVml4QSJqnQxyeffGJ2aTBBt27djBEjRphdBhzg+++/N5o0aWJYrVajQYMGxpw5c8wuCQ6QmppqjBgxwqhevbrh6elp1K5d23jhhReMzMxMs0uDnUVFRRX6+/3+++83DMMwcnNzjQkTJhihoaGG1Wo1unbtauzcudPcoi/BOg4AAAAAikSPAwAAAIAiERwAAAAAFIngAAAAAKBIBAcAAAAARSI4AAAAACgSwQEAAABAkQgOAAAAAIpEcAAAAABQJIIDAKDcOnz4sCwWix544AGzSwGAMo/gAAAAAKBIBAcAAAAARSI4AAAAACgSwQEAUGxr167VwIEDValSJVmtVkVEROjFF1/UuXPnbMesXr1aFotFEydO1Nq1a9WtWzf5+PgoMDBQd911l/76669Cz717927deeedCg4OltVqVa1atfT0008rKSmp0OMTEhI0atQo1a9fX56engoMDFT79u31xhtvFHr8wYMHNWTIEFWsWFHe3t7q1auXtm/ffuM/FABwEhbDMAyziwAAlH6zZ8/WY489pooVK2rgwIGqXLmyoqOjtWbNGnXs2FFRUVHy8PDQ6tWr1aNHD/Xt21dRUVEaMGCAGjRooC1btuinn35SeHi4oqOjFRISYjv3b7/9pj59+igzM1NDhgxRzZo1tXHjRq1evVoRERHasGGDgoKCbMfv379fPXr00NGjR9W5c2d17NhRZ8+e1a5du7Rjxw5b2Dh8+LBq1aqlbt26affu3WrUqJFat26t2NhY/e9//1PFihW1Z8+efLUAAK7AAACgCLt37zbc3NyMli1bGqdPn863b8qUKYYkY8aMGYZhGEZUVJQhyZBkfPTRR/mOnTRpkiHJePDBB23bcnJyjIiICEOSsXz5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h4eeeuopHThwQM8884zc3d21fPlyNWvWTI888ohOnjxZkiUDAACUWwQHlFqbN2+WdDE4XKuKFSvqjTfe0J49ezR48GDl5uZqzpw5qlu3riZPnqzz58/bu1wAAIByjeCAUitvxKF169bXfY46depo4cKFWrdundq0aaP09HS98MILql+/vr788ssr3rkJAABHW7VqlRo1aqRVq1aZXQpQKIIDSqX09HTt2bNH0o0FhzydO3fWxo0b9eWXXyo8PFzx8fG655571L59e61bt+6Gzw8AwI0wDEPjxo3Tnj17NG7cOHHvGpRGBAeUSlu3blVubq6qVq2qKlWq2OWcLi4uuuuuu/Tnn39q8uTJ8vHxUXR0tLp27arBgwfrwIEDdvk+AABcqxUrVthG2qOjo7VixQqTKwIKIjigVLq0MdrevLy8NHbsWB04cECPPPKIXFxctHjxYjVq1EjPPPOMkpOT7f49AQC4EsMw9OSTT+bbdtddd+nbb79VZmamSVUBBREcUCrlNUbbY5rSlYSEhGj27Nnavn27/vGPf+jChQt66623VKdOHc2cOVNZWVkl9r0BAJCknJwc3XPPPbY1iPIkJSXptttuU0hIiB588EGtWLGChU1hOoIDSqWSHHG4XJMmTbRs2TItX75cTZo0UXJyskaOHKnGjRvr22+/ZZ4pAKBEnDp1Sv369dNXX31VYJ/FYpG7u7tSUlL0ySefqG/fvgoLC9Pjjz+udevWcXMPmIKVo4uJlaMdJzk5WYGBgZKk06dP2/7sCNnZ2frkk0/04osvKiEhQZLUtWtXvfnmm4qMjHRYHQCA8i06OlpDhgxRXFzcVY+bPn26YmNj9c033+j06dO27VWrVtWdd96pYcOGKTIystjrHQGFYeVolFkxMTGSpNq1azs0NEiSm5ub/v3vf+vAgQN64YUX5OnpqbVr16p169a677779Ndffzm0HgBA+WIYhj744AN17txZcXFxslqtV7zod3Fx0YIFC/Tee+/p+PHjWr58uR544AH5+fnp6NGjevPNN9WmTRtFREToxRdf1K5duxz8auBsCA4odW5k4Td78fX11auvvqp9+/bpnnvukSR98cUXqlevnl566SWlp6ebVhsAoGw6d+6cHnjgAQ0fPlxZWVkaOHCg/Pz8rjglNjc3V/Hx8crKypK7u7v69u2rTz75RCdPntSSJUt0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",
      "text/plain": [
       "<Figure size 900x1500 with 3 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from matplotlib import pyplot as plt\n",
    "fig, ax = plt.subplots(3,1,figsize=(9,15))\n",
    "indices = list(range(0,len(ic_rs)))\n",
    "ax[0].plot(indices, ic_rs, marker='*', color='red', alpha=1, label='ic-retain')\n",
    "ax[0].plot(indices, ic_vs, marker='o', color='blue', alpha=1, label='ic-val')\n",
    "ax[1].plot(indices, acc_rs, marker='*', color='green', alpha=1, label='acc-retain')\n",
    "ax[1].plot(indices, acc_vs, marker='o', color='brown', alpha=1, label='acc-valid')\n",
    "ax[2].plot(indices, acc_fs, marker='^', color='black', alpha=1, label='acc-forget')\n",
    "ax[0].legend(prop={'size': 14})\n",
    "ax[1].legend(prop={'size': 14})\n",
    "plt.tick_params(labelsize=12)\n",
    "ax[0].set_xlabel('epoch',size=14)\n",
    "ax[0].set_ylabel('error',size=14)\n",
    "ax[1].set_xlabel('epoch',size=14)\n",
    "ax[1].set_ylabel('error',size=14)\n",
    "fig.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0.0, 0.1453846153846154, 0.1191025641025641, 0.12371794871794872, 0.1382051282051282, 0.1064102564102564, 0.10012820512820512, 0.10115384615384615, 0.1294871794871795, 0.1291025641025641, 0.13487179487179488]\n",
      "[0.1515, 0.278, 0.255, 0.259, 0.2735, 0.2405, 0.233, 0.244, 0.269, 0.2635, 0.273]\n",
      "[0.0, 10.788948059082031, 10.437187194824219, 10.615577697753906, 9.917083740234375, 9.042716979980469, 9.547737121582031, 8.16583251953125, 10.748741149902344, 9.477386474609375, 10.841712951660156]\n",
      "[19.209999084472656, 24.490005493164062, 23.959999084472656, 24.090003967285156, 24.06000518798828, 23.25, 23.550003051757812, 22.849998474121094, 24.300003051757812, 23.610000610351562, 23.610000610351562]\n",
      "[0.0, 56.5, 50.0, 51.0, 51.5, 50.0, 46.0, 45.0, 54.0, 50.0, 50.5]\n"
     ]
    }
   ],
   "source": [
    "print(ic_rs)\n",
    "print(ic_vs)\n",
    "print(acc_rs)\n",
    "print(acc_vs)\n",
    "print(acc_fs)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Finetune"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "model_ft = copy.deepcopy(model)\n",
    "args.ft_bs = 64\n",
    "retain_loader = replace_loader_dataset(train_loader_full,retain_dataset, seed=seed, batch_size=args.ft_bs, shuffle=True)  \n",
    "finetune(model_ft, retain_loader, epochs=10, quiet=True, lr=0.01)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Negative Gradient"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "args.ng_alpha = 0.9999\n",
    "args.ng_epochs = 10\n",
    "args.ng_lr = 0.01\n",
    "model_ng = copy.deepcopy(model)\n",
    "args.ng_bs = 128\n",
    "retain_loader = replace_loader_dataset(train_loader_full,retain_dataset, seed=seed, batch_size=args.ng_bs, shuffle=True)\n",
    "negative_grad(model_ng, retain_loader, forget_loader, alpha=args.ng_alpha, epochs=args.ng_epochs, quiet=True, lr=args.ng_lr, args=args)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Catastrophic Forgetting k layers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "args.lr_decay_epochs = [10,15,20]\n",
    "args.cfk_lr = 0.01\n",
    "args.cfk_epochs = 10\n",
    "args.cfk_bs = 128\n",
    "retain_loader = replace_loader_dataset(train_loader_full,retain_dataset, seed=seed, batch_size=args.cfk_bs, shuffle=True)\n",
    "\n",
    "model_cfk = copy.deepcopy(model)\n",
    "\n",
    "for param in model_cfk.parameters():\n",
    "    param.requires_grad_(False)\n",
    "\n",
    "if args.model == 'allcnn':\n",
    "    layers = [9]\n",
    "    for k in layers:\n",
    "        for param in model_cfk.features[k].parameters():\n",
    "            param.requires_grad_(True)\n",
    "    \n",
    "elif args.model == \"resnet\":\n",
    "    for param in model_cfk.layer4.parameters():\n",
    "        param.requires_grad_(True)\n",
    "\n",
    "else:\n",
    "    raise NotImplementedError\n",
    "\n",
    "\n",
    "\n",
    "fk_fientune(model_cfk, retain_loader, args=args, epochs=args.cfk_epochs, quiet=True, lr=args.cfk_lr)\n",
    "cfk_time = t2-t1"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Exact Unlearning k layers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "args.lr_decay_epochs = [10,15,20]\n",
    "args.euk_lr = 0.01\n",
    "args.euk_epochs = training_epochs\n",
    "args.euk_bs = 64\n",
    "retain_loader = replace_loader_dataset(train_loader_full,retain_dataset, seed=seed, batch_size=args.euk_bs, shuffle=True)\n",
    "model_euk = copy.deepcopy(model)\n",
    "\n",
    "for param in model_euk.parameters():\n",
    "    param.requires_grad_(False)\n",
    "\n",
    "if args.model == 'allcnn':\n",
    "    with torch.no_grad():\n",
    "        for k in layers:\n",
    "            for i in range(0,3):\n",
    "                try:\n",
    "                    model_euk.features[k][i].weight.copy_(model_initial.features[k][i].weight)\n",
    "                except:\n",
    "                    print (\"block {}, layer {} does not have weights\".format(k,i))\n",
    "                try:\n",
    "                    model_euk.features[k][i].bias.copy_(model_initial.features[k][i].bias)\n",
    "                except:\n",
    "                    print (\"block {}, layer {} does not have bias\".format(k,i))\n",
    "        model_euk.classifier[0].weight.copy_(model_initial.classifier[0].weight)\n",
    "        model_euk.classifier[0].bias.copy_(model_initial.classifier[0].bias)\n",
    "    \n",
    "    for k in layers:\n",
    "        for param in model_euk.features[k].parameters():\n",
    "            param.requires_grad_(True)\n",
    "    \n",
    "elif args.model == \"resnet\":\n",
    "    with torch.no_grad():\n",
    "        for i in range(0,2):\n",
    "            try:\n",
    "                model_euk.layer4[i].bn1.weight.copy_(model_initial.layer4[i].bn1.weight)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have weight\".format(i))\n",
    "            try:\n",
    "                model_euk.layer4[i].bn1.bias.copy_(model_initial.layer4[i].bn1.bias)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have bias\".format(i))\n",
    "            try:\n",
    "                model_euk.layer4[i].conv1.weight.copy_(model_initial.layer4[i].conv1.weight)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have weight\".format(i))\n",
    "            try:\n",
    "                model_euk.layer4[i].conv1.bias.copy_(model_initial.layer4[i].conv1.bias)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have bias\".format(i))\n",
    "\n",
    "            try:\n",
    "                model_euk.layer4[i].bn2.weight.copy_(model_initial.layer4[i].bn2.weight)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have weight\".format(i))\n",
    "            try:\n",
    "                model_euk.layer4[i].bn2.bias.copy_(model_initial.layer4[i].bn2.bias)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have bias\".format(i))\n",
    "            try:\n",
    "                model_euk.layer4[i].conv2.weight.copy_(model_initial.layer4[i].conv2.weight)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have weight\".format(i))\n",
    "            try:\n",
    "                model_euk.layer4[i].conv2.bias.copy_(model_initial.layer4[i].conv2.bias)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have bias\".format(i))\n",
    "\n",
    "        model_euk.layer4[0].shortcut[0].weight.copy_(model_initial.layer4[0].shortcut[0].weight)\n",
    "        \n",
    "    for param in model_euk.layer4.parameters():\n",
    "        param.requires_grad_(True)\n",
    "\n",
    "else:\n",
    "    raise NotImplementedError\n",
    "\n",
    "fk_fientune(model_euk, retain_loader, epochs=args.euk_epochs, quiet=True, lr=args.euk_lr, args=args)\n",
    "euk_time = t2-t1"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Readouts"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "train_loader_full, valid_loader_full, test_loader_full = datasets.get_loaders(dataset, split=\"train\",confuse_mode=True,class_to_replace=class_to_forget, num_indexes_to_replace=num_to_forget, batch_size=args.batch_size, seed=seed, root=args.dataroot, augment=False, shuffle=True)\n",
    "marked_loader, _, _ = datasets.get_loaders(dataset, split=\"forget\", confuse_mode=True,class_to_replace=class_to_forget, num_indexes_to_replace=num_to_forget, only_mark=True, batch_size=1, seed=seed, root=args.dataroot, augment=False, shuffle=True)\n",
    "\n",
    "def replace_loader_dataset(data_loader, dataset, batch_size=args.batch_size, seed=1, shuffle=True):\n",
    "    manual_seed(seed)\n",
    "    loader_args = {'num_workers': 0, 'pin_memory': False}\n",
    "    def _init_fn(worker_id):\n",
    "        np.random.seed(int(seed))\n",
    "    return torch.utils.data.DataLoader(dataset, batch_size=batch_size,num_workers=0,pin_memory=True,shuffle=shuffle)\n",
    "    \n",
    "forget_dataset = copy.deepcopy(marked_loader.dataset)\n",
    "marked = forget_dataset.targets < 0\n",
    "forget_dataset.data = forget_dataset.data[marked]\n",
    "forget_dataset.targets = - forget_dataset.targets[marked] - 1\n",
    "forget_loader = replace_loader_dataset(train_loader_full, forget_dataset, batch_size=args.forget_bs, seed=seed, shuffle=True)\n",
    "\n",
    "retain_dataset = copy.deepcopy(marked_loader.dataset)\n",
    "marked = retain_dataset.targets >= 0\n",
    "retain_dataset.data = retain_dataset.data[marked]\n",
    "retain_dataset.targets = retain_dataset.targets[marked]\n",
    "retain_loader = replace_loader_dataset(train_loader_full, retain_dataset, batch_size=args.retain_bs, seed=seed, shuffle=True)\n",
    "\n",
    "assert(len(forget_dataset) + len(retain_dataset) == len(train_loader_full.dataset))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": false
   },
   "outputs": [],
   "source": [
    "try: readouts\n",
    "except: readouts = {}\n",
    "\n",
    "#_,_=activations_predictions(copy.deepcopy(model),forget_loader,'Original_Model_D_f')\n",
    "#thresh=log_dict['Original_Model_D_f_loss']+1e-5\n",
    "#print(thresh)\n",
    "readouts[\"a\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model),thresh=None,name='Original')\n",
    "readouts[\"b\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model0),thresh=None,name='Retrain')\n",
    "readouts[\"c\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model_ft),thresh=None,name='Finetune')\n",
    "readouts[\"d\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model_ng),thresh=None,name='NegGrad')\n",
    "readouts[\"e\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model_cfk),thresh=None,name='CF-k')\n",
    "readouts[\"f\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model_euk),thresh=None,name='EU-k')\n",
    "readouts[\"h\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model_s),thresh=None,name='SCRUB')"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "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.10.13"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
