{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Imports"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "The autoreload extension is already loaded. To reload it, use:\n",
      "  %reload_ext autoreload\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": 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",
    "        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": 4,
   "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": 5,
   "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": 6,
   "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": 7,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checkpoint name: lacuna100_resnet_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1\n",
      "[Logging in lacuna100_resnet_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1_training]\n",
      "confuse mode: False\n",
      "split mode: None\n",
      "Number of Classes: 100\n",
      "[0] train metrics:{\"loss\": 3.987484888076782, \"error\": 0.923}\n",
      "Learning Rate : 0.1\n",
      "[0] dry_run metrics:{\"loss\": 3.395326801300049, \"error\": 0.83653125}\n",
      "Learning Rate : 0.1\n",
      "[0] test metrics:{\"loss\": 3.4219942169189452, \"error\": 0.8447}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 20.34 sec\n",
      "[1] train metrics:{\"loss\": 3.1922737522125244, \"error\": 0.78334375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 10.95 sec\n",
      "[2] train metrics:{\"loss\": 2.6385717277526854, \"error\": 0.63821875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 10.77 sec\n",
      "[3] train metrics:{\"loss\": 2.220917667865753, \"error\": 0.51859375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 8.45 sec\n",
      "[4] train metrics:{\"loss\": 1.9556603002548218, \"error\": 0.41671875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 8.55 sec\n",
      "[5] train metrics:{\"loss\": 1.722495491027832, \"error\": 0.32259375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.09 sec\n",
      "[6] train metrics:{\"loss\": 1.5956408495903016, \"error\": 0.26075}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 10.94 sec\n",
      "[7] train metrics:{\"loss\": 1.567377972126007, \"error\": 0.22371875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.05 sec\n",
      "[8] train metrics:{\"loss\": 1.5533827967643739, \"error\": 0.1940625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.03 sec\n",
      "[9] train metrics:{\"loss\": 1.523199393749237, \"error\": 0.16503125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 9.36 sec\n",
      "[10] train metrics:{\"loss\": 1.531425853729248, \"error\": 0.150625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 8.47 sec\n",
      "[11] train metrics:{\"loss\": 1.521689467906952, \"error\": 0.1343125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 9.44 sec\n",
      "[12] train metrics:{\"loss\": 1.5069730429649353, \"error\": 0.1225625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.08 sec\n",
      "[13] train metrics:{\"loss\": 1.5114346590042114, \"error\": 0.12234375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.06 sec\n",
      "[14] train metrics:{\"loss\": 1.517757547855377, \"error\": 0.1201875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.09 sec\n",
      "[15] train metrics:{\"loss\": 1.5435997090339661, \"error\": 0.1189375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.17 sec\n",
      "[16] train metrics:{\"loss\": 1.5590977020263672, \"error\": 0.11846875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 8.47 sec\n",
      "[17] train metrics:{\"loss\": 1.543018394470215, \"error\": 0.10815625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 8.6 sec\n",
      "[18] train metrics:{\"loss\": 1.5397232418060303, \"error\": 0.10703125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 10.62 sec\n",
      "[19] train metrics:{\"loss\": 1.5722574791908264, \"error\": 0.11421875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.06 sec\n",
      "[20] train metrics:{\"loss\": 1.5940280084609986, \"error\": 0.1075}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.19 sec\n",
      "[21] train metrics:{\"loss\": 1.5399379086494447, \"error\": 0.0955625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.12 sec\n",
      "[22] train metrics:{\"loss\": 1.5899784002304078, \"error\": 0.1098125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 9.6 sec\n",
      "[23] train metrics:{\"loss\": 1.6024198775291443, \"error\": 0.1053125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 8.45 sec\n",
      "[24] train metrics:{\"loss\": 1.549412570476532, \"error\": 0.09359375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 9.16 sec\n",
      "[25] train metrics:{\"loss\": 1.580799273967743, \"error\": 0.1025}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.24 sec\n",
      "[26] train metrics:{\"loss\": 1.577813358783722, \"error\": 0.10015625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.2 sec\n",
      "[27] train metrics:{\"loss\": 1.5733731870651244, \"error\": 0.09746875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.26 sec\n",
      "[28] train metrics:{\"loss\": 1.586123302936554, \"error\": 0.10090625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 11.2 sec\n",
      "[29] train metrics:{\"loss\": 1.5767607717514038, \"error\": 0.094625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 8.59 sec\n",
      "[30] train metrics:{\"loss\": 1.5994566655158997, \"error\": 0.1015}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 9.1 sec\n",
      "Pure training time: 316.23999999999995 sec\n"
     ]
    }
   ],
   "source": [
    "%run main.py --dataset lacuna100 --dataroot=data/lacuna100/ --model resnet --filters 1.0 --lr 0.1 --lossfn ce --num-classes 100"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Original"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "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": 9,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checkpoint name: lacuna10_resnet_1_0_forget_None_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1\n",
      "[Logging in lacuna10_resnet_1_0_forget_None_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1_training]\n",
      "confuse mode: True\n",
      "split mode: train\n",
      "[181 258  15  65 216  63 240 228 167 156  64 103 124 315 310  60 122 271\n",
      "  66  26 286 158 136 309 164 200 272   7   6 230 254  22 152  21  55 157\n",
      "  12 108  68 144 170 278  59 134 231  92 175 267  74  81 288 259 264  52\n",
      " 133 176  56  90 248  17   1   8 150 206 209 140 217 194   5  33 280  34\n",
      " 212 215 263 137  45 101 266 126 198 238 199 222 120 245 135  73 313 304\n",
      " 116 255  29  97  20  46 213 262 312  89 369 360 532 599 463 551 408 322\n",
      " 553 447 516 415 575 515 555 339 331 511 568 503 519 512 492 343 578 632\n",
      " 375 636 535 514 542 462 489 337 559 364 525 442 526 405 562 368 533 548\n",
      " 372 629 637 334 493 485 450 436 617 321 464 466 536 336 595 397 517 522\n",
      " 345 382 328 355 465 341 616 624 600 509 504 434 546 390 543 357 472 531\n",
      " 399 354 510 384 423 585 347 613 541 572 497 332 353 470 549 370 618 351\n",
      " 520 398]\n",
      "Number of Classes: 10\n",
      "[0] train metrics:{\"loss\": 0.589651049375534, \"error\": 0.169375}\n",
      "Learning Rate : 0.01\n",
      "[0] test metrics:{\"loss\": 0.3071651813983917, \"error\": 0.0869999966621399}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.39 sec\n",
      "[1] train metrics:{\"loss\": 0.2670166563987732, \"error\": 0.11375}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.09 sec\n",
      "[2] train metrics:{\"loss\": 0.18115923881530763, \"error\": 0.081875}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.1 sec\n",
      "[3] train metrics:{\"loss\": 0.14916219890117646, \"error\": 0.068125}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.09 sec\n",
      "[4] train metrics:{\"loss\": 0.13225296020507812, \"error\": 0.0625}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.11 sec\n",
      "[5] train metrics:{\"loss\": 0.11052428781986237, \"error\": 0.0528125}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.19 sec\n",
      "[6] train metrics:{\"loss\": 0.12346681982278823, \"error\": 0.0609375}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.09 sec\n",
      "[7] train metrics:{\"loss\": 0.10029780060052872, \"error\": 0.043125}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.11 sec\n",
      "[8] train metrics:{\"loss\": 0.08453558519482612, \"error\": 0.034375}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.12 sec\n",
      "[9] train metrics:{\"loss\": 0.07072421155869961, \"error\": 0.0315625}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.09 sec\n",
      "[10] train metrics:{\"loss\": 0.07172581106424332, \"error\": 0.02875}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.19 sec\n",
      "[11] train metrics:{\"loss\": 0.06146708309650421, \"error\": 0.0271875}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.11 sec\n",
      "[12] train metrics:{\"loss\": 0.0717173782736063, \"error\": 0.030625}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.1 sec\n",
      "[13] train metrics:{\"loss\": 0.0838085025548935, \"error\": 0.0346875}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.14 sec\n",
      "[14] train metrics:{\"loss\": 0.05625640660524368, \"error\": 0.025}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.11 sec\n",
      "[15] train metrics:{\"loss\": 0.05617266610264778, \"error\": 0.024375}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.18 sec\n",
      "[16] train metrics:{\"loss\": 0.0597886473685503, \"error\": 0.025}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.11 sec\n",
      "[17] train metrics:{\"loss\": 0.07512851443141699, \"error\": 0.028125}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.1 sec\n",
      "[18] train metrics:{\"loss\": 0.060404927954077724, \"error\": 0.024375}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.11 sec\n",
      "[19] train metrics:{\"loss\": 0.051007045209407804, \"error\": 0.02625}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.1 sec\n",
      "[20] train metrics:{\"loss\": 0.05638246718794107, \"error\": 0.0271875}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.2 sec\n",
      "[21] train metrics:{\"loss\": 0.05464376173913479, \"error\": 0.0275}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.09 sec\n",
      "[22] train metrics:{\"loss\": 0.055572110563516616, \"error\": 0.0265625}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.12 sec\n",
      "[23] train metrics:{\"loss\": 0.050947729125618936, \"error\": 0.0278125}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.1 sec\n",
      "[24] train metrics:{\"loss\": 0.04976472221314907, \"error\": 0.024375}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.09 sec\n",
      "[25] train metrics:{\"loss\": 0.05443723313510418, \"error\": 0.0221875}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.2 sec\n",
      "Pure training time: 28.700000000000003 sec\n"
     ]
    }
   ],
   "source": [
    "%run main.py --dataset lacuna10 --model resnet --dataroot=data/lacuna10/ --filters 1.0 --lr 0.01 \\\n",
    "--resume checkpoints/lacuna100_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": 10,
   "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": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checkpoint name: lacuna10_resnet_1_0_forget_[0, 1]_num_200_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1\n",
      "[Logging in lacuna10_resnet_1_0_forget_[0, 1]_num_200_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1_training]\n",
      "confuse mode: True\n",
      "split mode: forget\n",
      "[181 258  15  65 216  63 240 228 167 156  64 103 124 315 310  60 122 271\n",
      "  66  26 286 158 136 309 164 200 272   7   6 230 254  22 152  21  55 157\n",
      "  12 108  68 144 170 278  59 134 231  92 175 267  74  81 288 259 264  52\n",
      " 133 176  56  90 248  17   1   8 150 206 209 140 217 194   5  33 280  34\n",
      " 212 215 263 137  45 101 266 126 198 238 199 222 120 245 135  73 313 304\n",
      " 116 255  29  97  20  46 213 262 312  89 369 360 532 599 463 551 408 322\n",
      " 553 447 516 415 575 515 555 339 331 511 568 503 519 512 492 343 578 632\n",
      " 375 636 535 514 542 462 489 337 559 364 525 442 526 405 562 368 533 548\n",
      " 372 629 637 334 493 485 450 436 617 321 464 466 536 336 595 397 517 522\n",
      " 345 382 328 355 465 341 616 624 600 509 504 434 546 390 543 357 472 531\n",
      " 399 354 510 384 423 585 347 613 541 572 497 332 353 470 549 370 618 351\n",
      " 520 398]\n",
      "Number of Classes: 10\n",
      "[0] train metrics:{\"loss\": 0.3945189368724823, \"error\": 0.098125}\n",
      "Learning Rate : 0.01\n",
      "[0] test metrics:{\"loss\": 0.21868188786506654, \"error\": 0.07399999570846558}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.37 sec\n",
      "[1] train metrics:{\"loss\": 0.1252414670586586, \"error\": 0.04}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.09 sec\n",
      "[2] train metrics:{\"loss\": 0.03998582549393177, \"error\": 0.00875}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.1 sec\n",
      "[3] train metrics:{\"loss\": 0.018871775325387717, \"error\": 0.003125}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.1 sec\n",
      "[4] train metrics:{\"loss\": 0.007750764386728406, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.11 sec\n",
      "[5] train metrics:{\"loss\": 0.004611239414662123, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.21 sec\n",
      "[6] train metrics:{\"loss\": 0.0034876267984509466, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.1 sec\n",
      "[7] train metrics:{\"loss\": 0.002967796362936497, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.11 sec\n",
      "[8] train metrics:{\"loss\": 0.0026318002911284565, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.12 sec\n",
      "[9] train metrics:{\"loss\": 0.0024281581770628693, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.11 sec\n",
      "[10] train metrics:{\"loss\": 0.0022898506931960582, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.18 sec\n",
      "[11] train metrics:{\"loss\": 0.0021680903434753416, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.11 sec\n",
      "[12] train metrics:{\"loss\": 0.0020789767382666468, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 1.1 sec\n",
      "[13] train metrics:{\"loss\": 0.002004021811299026, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.99 sec\n",
      "[14] train metrics:{\"loss\": 0.0019420119980350136, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.83 sec\n",
      "[15] train metrics:{\"loss\": 0.0018930625170469283, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.91 sec\n",
      "[16] train metrics:{\"loss\": 0.0018525511119514705, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.78 sec\n",
      "[17] train metrics:{\"loss\": 0.0018156438553705813, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.87 sec\n",
      "[18] train metrics:{\"loss\": 0.0017831958830356598, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.81 sec\n",
      "[19] train metrics:{\"loss\": 0.0017556374473497271, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.89 sec\n",
      "[20] train metrics:{\"loss\": 0.00173125677742064, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.92 sec\n",
      "[21] train metrics:{\"loss\": 0.001710162670351565, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.82 sec\n",
      "[22] train metrics:{\"loss\": 0.001691991640254855, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.84 sec\n",
      "[23] train metrics:{\"loss\": 0.0016751015977934003, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.86 sec\n",
      "[24] train metrics:{\"loss\": 0.0016594806034117937, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.85 sec\n",
      "[25] train metrics:{\"loss\": 0.0016461714264005424, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.93 sec\n",
      "Pure training time: 25.44 sec\n"
     ]
    }
   ],
   "source": [
    "%run main.py --dataset lacuna10 --model resnet --dataroot=data/lacuna10/ --filters 1.0 --lr 0.01 \\\n",
    "--resume checkpoints/lacuna100_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": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "log_dict={}\n",
    "training_epochs=25"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [],
   "source": [
    "log_dict['epoch']=training_epochs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "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": 15,
   "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": 16,
   "metadata": {},
   "outputs": [],
   "source": [
    "log_dict['args']=args"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "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": 18,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Distance: 3.112095905916224\n",
      "Normalized Distance: 0.03129505126715685\n"
     ]
    }
   ],
   "source": [
    "log_dict['dist_Original_Original_init']= distance(model_init,model)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Data Loader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "args.retain_bs = 32\n",
    "args.forget_bs = 32"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "confuse mode: True\n",
      "split mode: train\n",
      "[181 258  15  65 216  63 240 228 167 156  64 103 124 315 310  60 122 271\n",
      "  66  26 286 158 136 309 164 200 272   7   6 230 254  22 152  21  55 157\n",
      "  12 108  68 144 170 278  59 134 231  92 175 267  74  81 288 259 264  52\n",
      " 133 176  56  90 248  17   1   8 150 206 209 140 217 194   5  33 280  34\n",
      " 212 215 263 137  45 101 266 126 198 238 199 222 120 245 135  73 313 304\n",
      " 116 255  29  97  20  46 213 262 312  89 369 360 532 599 463 551 408 322\n",
      " 553 447 516 415 575 515 555 339 331 511 568 503 519 512 492 343 578 632\n",
      " 375 636 535 514 542 462 489 337 559 364 525 442 526 405 562 368 533 548\n",
      " 372 629 637 334 493 485 450 436 617 321 464 466 536 336 595 397 517 522\n",
      " 345 382 328 355 465 341 616 624 600 509 504 434 546 390 543 357 472 531\n",
      " 399 354 510 384 423 585 347 613 541 572 497 332 353 470 549 370 618 351\n",
      " 520 398]\n",
      "confuse mode: True\n",
      "split mode: forget\n",
      "[181 258  15  65 216  63 240 228 167 156  64 103 124 315 310  60 122 271\n",
      "  66  26 286 158 136 309 164 200 272   7   6 230 254  22 152  21  55 157\n",
      "  12 108  68 144 170 278  59 134 231  92 175 267  74  81 288 259 264  52\n",
      " 133 176  56  90 248  17   1   8 150 206 209 140 217 194   5  33 280  34\n",
      " 212 215 263 137  45 101 266 126 198 238 199 222 120 245 135  73 313 304\n",
      " 116 255  29  97  20  46 213 262 312  89 369 360 532 599 463 551 408 322\n",
      " 553 447 516 415 575 515 555 339 331 511 568 503 519 512 492 343 578 632\n",
      " 375 636 535 514 542 462 489 337 559 364 525 442 526 405 562 368 533 548\n",
      " 372 629 637 334 493 485 450 436 617 321 464 466 536 336 595 397 517 522\n",
      " 345 382 328 355 465 341 616 624 600 509 504 434 546 390 543 357 472 531\n",
      " 399 354 510 384 423 585 347 613 541 572 497 332 353 470 549 370 618 351\n",
      " 520 398]\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": 21,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "200\n",
      "3000\n",
      "1000\n",
      "3200\n",
      "{0: 320, 1: 320, 2: 320, 3: 320, 4: 320, 5: 320, 6: 320, 7: 320, 8: 320, 9: 320}\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": 22,
   "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": 23,
   "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": 24,
   "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": 25,
   "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": 26,
   "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": 27,
   "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 99.167 \n",
      "maximize loss: -3.45\t minimize loss: 0.26\t train_acc: 99.16666412353516\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 98.967 \n",
      "maximize loss: -4.38\t minimize loss: 0.34\t train_acc: 98.96666717529297\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 98.300 \n",
      "maximize loss: 0.00\t minimize loss: 0.16\t train_acc: 98.29999542236328\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 97.967 \n",
      "maximize loss: 0.00\t minimize loss: 0.12\t train_acc: 97.96666717529297\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 97.700 \n",
      "maximize loss: 0.00\t minimize loss: 0.13\t train_acc: 97.69999694824219\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 97.967 \n",
      "maximize loss: 0.00\t minimize loss: 0.13\t train_acc: 97.96666717529297\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 97.767 \n",
      "maximize loss: 0.00\t minimize loss: 0.12\t train_acc: 97.76666259765625\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 97.400 \n",
      "maximize loss: 0.00\t minimize loss: 0.12\t train_acc: 97.4000015258789\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 97.533 \n",
      "maximize loss: 0.00\t minimize loss: 0.12\t train_acc: 97.53333282470703\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 97.367 \n",
      "maximize loss: 0.00\t minimize loss: 0.12\t train_acc: 97.36666870117188\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": 28,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_394973/2812292199.py:16: UserWarning: FigureCanvasAgg is non-interactive, and thus cannot be shown\n",
      "  fig.show()\n"
     ]
    },
    {
     "data": {
      "image/png": "iVBORw0KGgoAAAANSUhEUgAAAw4AAASvCAYAAACgpa2IAAAAOXRFWHRTb2Z0d2FyZQBNYXRwbG90bGliIHZlcnNpb24zLjguMCwgaHR0cHM6Ly9tYXRwbG90bGliLm9yZy81sbWrAAAACXBIWXMAAA9hAAAPYQGoP6dpAADz20lEQVR4nOzdeVxU9f7H8fewyKKCK7iAgGau5YJrbmipZZlmi7aYZt7yV5nabdGyUktpV7PU7HbTMs0yS71pSeW+5V5pZqm4oIhogrggy/n9cQJFwBlh4AzM6/l4zIPhzJkznwOjnPd8N5thGIYAAAAA4Ao8rC4AAAAAgOsjOAAAAACwi+AAAAAAwC6CAwAAAAC7CA4AAAAA7CI4AAAAALCL4AAAAADALi+rCygpMjMzdeTIEZUvX142m83qcgAAAACnMAxDp0+fVo0aNeThkX+7AsHBQUeOHFFoaKjVZQAAAABF4tChQwoJCcn3cYKDg8qXLy/J/IEGBARYXA0AAADgHMnJyQoNDc2+3s0PwcFBWd2TAgICCA4AAAAodex1x2dwNAAAAAC7CA4AAAAA7HLJ4DB16lRFRETI19dXkZGRWr16db77rlmzRu3atVPlypXl5+en+vXra+LEibn2++qrr9SwYUP5+PioYcOG+vrrr4vyFAAAAIBSxeWCw7x58zR8+HC98MIL2rZtmzp06KBbbrlFBw8ezHP/smXL6oknntCqVav0+++/a/To0Ro9erRmzJiRvc/69evVt29f9e/fXzt27FD//v11zz33aOPGjcV1WgAAAECJZjMMw7C6iEu1bt1azZs317Rp07K3NWjQQL1791Z0dLRDx+jTp4/Kli2rTz/9VJLUt29fJScna+nSpdn73HzzzapYsaLmzp2b5zFSU1OVmpqa/X3WaPOkpCQGRwMAAKDUSE5OVmBgoN3rXJdqcbhw4YK2bNmibt265djerVs3rVu3zqFjbNu2TevWrVOnTp2yt61fvz7XMbt3737FY0ZHRyswMDD7xhoOAAAAcGcuNR1rYmKiMjIyFBwcnGN7cHCw4uPjr/jckJAQHT9+XOnp6RozZowGDx6c/Vh8fPxVH3PUqFF66qmnsr/PanEAAAAoamlpacrIyLC6DJRQnp6e8vb2dvpxXSo4ZLl8DlnDMOzOK7t69WqlpKRow4YNGjlypK655hrde++9BT6mj4+PfHx8ClA9AABAwSQnJysxMTFHd2mgIHx8fFSlShWndrF3qeBQpUoVeXp65moJSEhIyNVicLmIiAhJ0nXXXadjx45pzJgx2cGhWrVqBTqmq8jIkFavlo4elapXlzp0kDw9ra4KAAA4U3JysuLi4lSuXDlVqVJF3t7edj84BS5nGIbS0tKUlJSkuLg4SXJaeHCp4FCmTBlFRkYqJiZGd9xxR/b2mJgY9erVy+HjGIaRI6m3bdtWMTExGjFiRPa2ZcuW6YYbbnBO4UVowQJp2DDp8OGL20JCpMmTpT59rKsLAAA4V2JiosqVK6eQkBACAwrFz89P5cuX1+HDh5WYmFg6g4MkPfXUU+rfv79atGihtm3basaMGTp48KCGDBkiyRx7EBcXp08++USS9P7776tWrVqqX7++JHNdh7feektDhw7NPuawYcPUsWNHvf766+rVq5cWLlyoH374QWvWrCn+E7wKCxZId90lXT7vVVycuX3+fMIDAAClQVpamlJTU1WlShVCA5zCZrMpMDBQcXFxSktLc8qYB5cLDn379tWJEyc0btw4HT16VI0bN9aSJUsUFhYmSTp69GiONR0yMzM1atQo7d+/X15eXqpTp45ee+01Pfroo9n73HDDDfr88881evRovfjii6pTp47mzZun1q1bF/v5OSojw2xpyGuyXMOQbDZp+HCpVy+6LQEAUNJlDYQuigGtcF9Z76eMjAynvLdcbh0HV+Xo/LbOsmKF1Lmz/f2WL5eiooq6GgAAUJTOnz+v/fv3KyIiQr6+vlaXg1LC0fdViVzHARcdPerc/QAAAIDCIDi4qOrVnbsfAAAAUBgEBxfVoYM5e1J+46NsNik01NwPAAAAKGoEBxfl6WlOuSrlHx4mTWJgNAAAKB1iY2Nls9k0cOBAq0spVmPGjJHNZtOKFSusLsUugoML69PHnHK1Zs3cjzVtylSsAAAAxc1dA47kgtOxIqc+fcwpV7NWjs7MlAYMkLZtk1aulDp1srpCAACAwqtZs6Z+//13BQYGWl1KsXriiSfUr18/1apVy+pS7KLFoQTw9DSnXL33Xun++6VHHjG3jxyZ9zoPAAAA+dq8WerSxfzqQry9vVW/fn1Vd7OZX6pUqaL69evL39/f6lLsIjiUQC++KPn7Sxs2SIsWWV0NAAAoUT75xFwI6tNPra4kh/y6AJ0+fVrjxo3T9ddfr7JlyyowMFDNmjXTiy++qLS0NIeOHR4ervDwcJ06dUpPPvmkQkND5eXlpZkzZ2bv88svv6hfv36qXr26ypQpo7CwMA0dOlQnTpzI3mfmzJmKiIiQJM2aNUs2my37ljVG4ciRI3r55ZfVpk0bBQUFycfHR+Hh4XrssceUkJCQq7a8xjhc+rPYt2+f7rrrLlWsWFFly5bVTTfdpB07djj2Q3UyuiqVQNWrm6tGT5ggPf+8dNttDJIGAKBUMgzp7NnCH+fgQenECXPGlc8/N7fNnSvdc4/5GpUrS4XpKuPvn/9sLoWQmJioTp06adeuXWratKmGDBmizMxM7d69W6+//rr+/e9/q0KFCg4dKzU1VV26dNHp06fVs2dPlSlTRsHBwZKkRYsW6Z577pGnp6duv/12hYaGateuXXrvvff0/fffa+PGjapYsaKaNm2qYcOGafLkyWrSpIl69+6dffzw8HBJ0qpVq/T222/rxhtvVOvWreXt7a1t27Zp2rRp+v7777V161aHu2PFxsaqdevWatiwoQYNGqS9e/dq4cKF6ty5s37//ffs+ouNAYckJSUZkoykpCSrSzEMwzBOnTKMSpUMQzKMjz+2uhoAAFAY586dM3bt2mWcO3cu5wMpKeYfe1e/paQU+mewf/9+Q5IxYMCA7G133323Icl4/vnnc+0fHx9vpKWlOXTssLAwQ5LRrVs34+zZszkeS0xMNAICAoyQkBDjwIEDOR6bM2eOIcl44oknrljnpY4dO2acPn061/ZZs2YZkoxXX301x/aXX37ZkGQsX74812tIMl577bUc+48ePdqQZERHR9s973zfV5dx9DqXrkolVGCgNGqUef+ll6Tz562tBwAAwJmOHTum+fPnq06dOhozZkyux4ODg+XldXWdZ9588035+fnl2PbJJ58oOTlZ0dHRuQYo33vvvWrevLk+z2qlcUBQUJDKlSuXa3v//v0VEBCgH374weFjRURE6Jlnnsmx7eGHH5Ykbdq0yeHjOAtdlUqwxx8313o4dEiaNk0aMcLqigAAgFP5+0spKc451vbtUvv2ubevWWPO814YRTCwd/PmzTIMQ507d5a3t/cV9505c6ZiY2NzbOvdu7eaXnJevr6+uu6663I9d8OGDdlf//rrr1yPnz9/XomJiUpMTFSVKlUcqn3BggX64IMPtHXrVv3999/KyMjIfuzIkSMOHUOSmjRpIg+PnJ/zh4SESJJOnTrl8HGcheBQgvn5SWPGSIMHS+PHS4MGmS0RAACglLDZpLJlnXOsrE/aPTzM+d2zvvr5Oe81nCjrwrhmXgtaXWbmzJlauXJljm3h4eE5gkNQUJBseYzDOHnypCTp/fffv+JrnDlzxqHg8Pbbb+vpp59W1apV1a1bN4WEhGS3ckyaNEmpqal2j5Elr7EQWa0sl4aR4kJwKOEGDJDeekvavVt6+21p3DirKwIAAC4pKEiqVk0KDZUeflj66COz20JQkNWV5Slr0HNcXJzdfR1ZdTmv0CBJAQEBkqRff/1VjRs3dri+vKSnp+uVV15RjRo1tH37dlWtWjX7McMw9MYbbxTq+FZjjEMJ5+VltjZI0jvvSMeOWVsPAABwUSEhUmystHGj9Oij5tfYWHO7C2rRooU8PDy0fPlyh6ddLYjWrVtLktavX+/Q/p7/TGWZ1yf+iYmJSkpKUps2bXKEBsnsenXu3LlCVmstgkMpcMcdUqtW0pkz0quvWl0NAABwWT4+F6dNtdnM711UcHCw7rzzTu3du1djx47N9XhCQoLS09ML/ToPPfSQypcvrxdeeEE7d+7M9fjZs2ezx0FIUsWKFWWz2XT48OFc+wYFBcnPz09bt27V2Uum0f377781dOjQQtdqNboqlQI2m/Taa+YikB98YA6Srl3b6qoAAAAKZ+rUqfrtt980fvx4LVmyRF26dJFhGNqzZ4+WLVumY8eOObyOQ36qVq2quXPn6u6771aTJk108803q379+jp//rwOHDiglStX6oYbbtB3330nSSpXrpxatmypVatW6aGHHlLdunXl4eGh++67T7Vq1dJjjz2mt99+W02aNFHPnj2VnJyspUuXKiwsTDVq1HDCT8U6BIdSonNnqXt36fvvzelZZ8+2uiIAAIDCqVKlijZs2KC33npLX375pd577z35+voqIiJCI0eOVFknDeq+9dZbtW3bNr355pv64YcfFBMTo7JlyyokJEQPPfSQHnjggRz7f/rppxoxYoS++eYbJSUlyTAMtWnTRrVq1VJ0dLQqVaqkmTNnaurUqQoODla/fv00duzYQo+hsJrNMAzD6iJKguTkZAUGBiopKSl7EI2r2bZNat7cbIHYtk1q0sTqigAAgCPOnz+v/fv3KyIiQr6+vlaXg1LC0feVo9e5jHEoRZo1k/r1M5dwfP55q6sBAABAaUJwKGVeecWcaWnJEmnVKqurAQAAQGlBcChlrrlG+te/zPsjR5qtDwAAAEBhERxKoRdfNFd+X79eWrTI6moAAABQGhAcSqHq1aXhw837zz8vWbAiOQAAAEoZgkMp9cwzUsWK0q5dTM0KAACAwiM4lFIVKlycWemll6Tz5y0tBwAAACUcwaEUe/xxKSREOnhQmj7d6moAAABQkhEcSjE/P2nMGPP++PFScrKl5QAAAKAEIziUcgMGSPXqSYmJ0ttvW10NAAAASiqCQynn5WW2NkhmcDh2zNp6AAAAUDIRHNxAnz5Sy5bSmTMXQwQAAABwNQgObsBmk157zbw/fbq0b5+19QAAAFwuNjZWNptNAwcOtLqUqxIeHq7w8HCryygWBAc30aWL1K2blJYmvfyy1dUAAACgpCE4uJHoaPPrZ59Jv/xibS0AAACXqlmzpn7//XdFZ12wwOUQHNxI8+ZS376SYVxcHA4AALiPjAxpxQpp7lzza0aG1RVd5O3trfr166t69epWl4J8EBzczCuvmDMtffuttHq11dUAAIDismCBFB4ude4s3Xef+TU83NzuCvIb43D69GmNGzdO119/vcqWLavAwEA1a9ZML774otLS0q54zE8++UQ2m02vvPJKno+vXbtWNptNDz/8cPa25cuXa9CgQapXr57KlSuncuXKqUWLFpoxY0ahz7GkIzi4mbp1pcGDzfsjR5qtDwAAoHRbsEC66y7p8OGc2+PizO2uEh4ul5iYqDZt2ujll1+Wp6enhgwZokGDBqlatWp6/fXXdebMmSs+v0+fPvL399dnn32W5+OzZ8+WJPXv3z972+uvv65Vq1apZcuWeuKJJ/TAAw8oMTFRjz76qP7973877+RKIC+rC0Dxe/FFadYsad06afFi6fbbra4IAADkxTCks2cLd4yMDOnJJ/P+sNAwzNkXhw2TbrpJ8vQs2Gv4+5vHcbbHHntMu3bt0vPPP6/xl80pf+zYMZUrV+6Kzy9XrpzuuOMOffbZZ9q0aZNatmyZ/VhaWpq+/PJLhYaGqlOnTtnbp02bpoiIiBzHSU9PV48ePTR58mQNGzZMtWrVcsLZlTy0OLihGjWk4cPN+88/71r9GwEAwEVnz0rlyhXuFhhotizkxzDMlojAwIK/RmHDTV6OHTum+fPnq06dOhozZkyux4ODg+XlZf8z8AceeEDSxdaFLEuWLNGJEyd0//33y3ZJ6rk8NEiSl5eXhgwZooyMDC1fvvwqz6T0oMXBTT37rLmmw86d0uzZ0oABVlcEAABw0ebNm2UYhjp37ixvb+8r7jtz5kzFxsbm2Na7d281bdpUXbt2VbVq1fT555/rnXfekec/zSqffvqppJzdlCRzTMVbb72lb775Rnv37s3VHerIkSOFPLOSi+DgpipUkEaNMgPESy9J/fpJPj5WVwUAAC7l7y+lpBTuGKtWST162N9vyRKpY8eCvYa/f8GedyWnTp2SZE7Tas/MmTO1cuXKHNvCw8PVtGlTeXp66t5779XEiRMVExOjm2++WUlJSfr222/VvHlzNWzYMPs5Fy5cUFRUlLZu3apmzZqpf//+qly5sry8vBQbG6tZs2YpNTXVqedZkhAc3NgTT0iTJ0sHD0rTpl3svgQAAFyDzSaVLVu4Y3TrJoWEmN2V8hrnYLOZj3frVvAxDkWhQoUKkqS4K/Wz+seKFSuu+Hj//v01ceJEzZ49WzfffLO+/PJLnT9/Pldrw8KFC7V161YNHjxYH374YY7HPv/8c82aNeuqzqG0YYyDG/Pzk7K6DI4fLyUnW1oOAAAoAp6e5geFUu4BzFnfT5rkWqFBklq0aCEPDw8tX77c7rSr9jRr1kwNGzbUN998ozNnzmj27NnZLRGX2rt3ryTp9jxmjlnNPPYEB3c3cKBUr56UmCi9/bbV1QAAgKLQp480f750ea+fkBBze58+1tR1JcHBwbrzzju1d+9ejR07NtfjCQkJSk9Pd/h4/fv315kzZzR58mStWrVKXbt2VXBwcI59wsLCJElr1qzJsX3lypW5WiDcEcHBzXl5ma0NkhkcEhKsrQcAABSNPn2k2Fhp+XJpzhzz6/79rhkaskydOlUNGjTQ+PHj1bx5cz399NP697//rZ49eyo0NFQpVzEAJGv2pDFjxsgwjFzdlCSpZ8+eCg8P1xtvvKFbb71Vzz33nHr37q0bb7wxz1YId0NwgPr0kVq2lM6ckV591epqAABAUfH0lKKipHvvNb+6Wveky1WpUkUbNmzQiy++qHPnzum9997TRx99pMOHD2vkyJEqexUDQEJDQxUVFaW0tDSVK1dOvXv3zrVPuXLl9NNPP+nOO+/Upk2b9N577+nIkSP67LPP9MQTTzjxzEomm2GwdrAjkpOTFRgYqKSkJAUEBFhdjtP99JN0442St7f0xx9SHlMYAwCAInL+/Hnt379fERER8vX1tboclBKOvq8cvc6lxQGSpC5dpK5dpbQ0c3pWAAAA4FIuGRymTp2anYwiIyOvOIp9wYIF6tq1q6pWraqAgAC1bdtW33//fY59Zs6cKZvNlut2/vz5oj6VEiU62vz62WfSL79YWwsAAABci8sFh3nz5mn48OF64YUXtG3bNnXo0EG33HKLDh48mOf+WaPilyxZoi1btqhz587q2bOntm3blmO/gIAAHT16NMeNpsCcIiOle+4x53h+/nmrqwEAAIArcbkxDq1bt1bz5s01bdq07G0NGjRQ7969FZ31kbgdjRo1Ut++ffXSP31uZs6cqeHDh2evQFgQpX2MQ5Y//5QaNJAyMsyVJjt0sLoiAABKP8Y4oCiU6jEOFy5c0JYtW9StW7cc27t166Z169Y5dIzMzEydPn1alSpVyrE9JSVFYWFhCgkJ0W233ZarReJyqampSk5OznFzB3XrSoMHm/dHjsx7hUkAAAC4H5cKDomJicrIyMi1GEdwcLDi4+MdOsbbb7+tM2fO6J577sneVr9+fc2cOVOLFi3S3Llz5evrq3bt2unPP//M9zjR0dEKDAzMvoWGhhbspEqgl14yV5Vet0763/+srgYAAACuwKWCQxbbZeuhG4aRa1te5s6dqzFjxmjevHkKCgrK3t6mTRs98MADatKkiTp06KAvvvhC1157raZMmZLvsUaNGqWkpKTs26FDhwp+QiVMjRrSsGHm/VGjzG5LAACg6LlYD3KUcM5+P7lUcKhSpYo8PT1ztS4kJCTkaoW43Lx58/Twww/riy++0E033XTFfT08PNSyZcsrtjj4+PgoICAgx82dPPecVLGitHOnOcsSAAAoOp7/rMSWlpZmcSUoTbLeT55OWunPpYJDmTJlFBkZqZiYmBzbY2JidMMNN+T7vLlz52rgwIGaM2eObr31VruvYxiGtm/frurVqxe65tKqQgVzjINkdl1KTbW0HAAASjVvb2/5+PgoKSmJVgc4hWEYSkpKko+Pj7y9vZ1yTC+nHMWJnnrqKfXv318tWrRQ27ZtNWPGDB08eFBDhgyRZHYhiouL0yeffCLJDA0PPvigJk+erDZt2mS3Vvj5+SkwMFCSNHbsWLVp00Z169ZVcnKy3n33XW3fvl3vv/++NSdZQjzxhDR5snTggDR9+sXuSwAAwPmqVKmiuLg4HT58WIGBgfL29naoqzZwKcMwlJaWpqSkJKWkpKhmzZpOO7bLBYe+ffvqxIkTGjdunI4eParGjRtryZIlCgsLkyQdPXo0x5oOH3zwgdLT0/X444/r8ccfz94+YMAAzZw5U5J06tQpPfLII4qPj1dgYKCaNWumVatWqVWrVsV6biWNv780Zoz0yCPSq69KDz0kuVmPLQAAik1Wt+jExETFxcVZXA1KOh8fH9WsWdOp3e1dbh0HV+Uu6zhcLj1datRI2rNHevllM0gAAICilZaWpgxmJ0EBeXp6XlX3JEevcwkODnLX4CBJ8+dLd98tlSsn7d0rXTJhFQAAAEq4ErkAHFzTnXdKLVpIKSnS+PFWVwMAAAArEBxgl80mvfaaeX/aNCk21tJyAAAAYAGCAxxy443STTdJaWnm9KwAAABwLwQHOCw62vw6e7b066/W1gIAAIDiRXCAw1q0MAdJG4b0/PNWVwMAAIDiRHDAVXn1VcnTU/rf/6Q1a6yuBgAAAMWF4ICrcu210sMPm/dHjjRbHwAAAFD6ERxw1V5+WfL1ldaulb791upqAAAAUBwIDrhqNWpIw4aZ90eNkljYEgAAoPQjOKBAnntOqlBB+u03ac4cq6sBAABAUSM4oEAqVjTHOEjSiy9KqanW1gMAAICiRXBAgQ0danZbOnBA+uADq6sBAABAUSI4oMD8/c2B0pI5Tevp09bWAwAAgKJDcEChDBpkTtF6/Lj0zjtWVwMAAICiQnBAoXh5ma0NkvTWW1JCgrX1AAAAoGgQHFBod90lRUZKKSnShAlWVwMAAICiQHBAodls0muvmfenTZNiYy0tBwAAAEWA4ACnuOkm83bhwsUB0wAAACg9CA5wmuho8+unn0q//mptLQAAAHAuggOcpkUL6e67JcOQXnjB6moAAADgTAQHONUrr0ientLixdKaNVZXAwAAAGchOMCp6tUz13aQpJEjzdYHAAAAlHwEBzjdyy9Lvr7S2rXSt99aXQ0AAACcgeAAp6tZU3rySfP+qFFSRoa19QAAAKDwCA4oEiNHShUqSL/9Js2ZY3U1AICSICNDWrFCmjvX/MoHT4BrITigSFSsKD33nHn/pZek1FRr6wEAuLYFC6TwcKlzZ+m++8yv4eHmdgCugeCAIvPkk1L16uZK0h98YHU1AABXtWCBdNdd0uHDObfHxZnbCQ+AayA4oMj4+19cRfrVV6XTp62tBwDgejIypGHD8p6FL2vb8OF0WwJcAcEBRWrQIKluXen4cemdd6yuBgDgalavzt3ScCnDkA4dMvcDYC2CA4qUt7fZ2iBJb71lBggAAFJTpYULzdn3HHH0aNHWA8A+ggOK3F13SZGRUkqKNH681dUAAKySkSEtXy79619StWpS797Shg2OPXfePGn//iItD4AdBAcUOQ8PKTravD9tmjlYGgDgHgxD2rpVevppKSxM6tJF+s9/pFOnpBo1zPELQUGSzXbl4yxcaHZ9vf9+6ZdfiqNyAJcjOKBYdO0q3XijdOHCxQHTAIDS688/pXHjpAYNzFbnt982Z0mqUEEaPFj66Sfp4EFp4kTzQyUpd3iw2czb2LFS9+5mi8WcOVKTJtKttzLuAShuNsPIax4DXC45OVmBgYFKSkpSQECA1eWUSJs2Sa1amX8EfvlFatzY6ooAAM509KjZpWjOHPP//Cy+vtLtt5vrM9x8s+Tjk/u5CxaYsytdOlA6NFSaNEnq08f8fts26fXXpS+/lDIzzW3t2pmLjvboYbZwA7h6jl7nEhwcRHBwjrvvlubPl3r2lBYtsroaAEBhnTplXvTPmWOOX8i6oPf0NFub77vPHMtQvrz9Y2VkmK0IR4+a6wB16GAe53J//WVOuPHxx2ZLtmR+GPXcc1LfvubEHAAcR3BwMoKDc/zxh9SokfnHYc0a85MiAEDJcu6c9O23Zlj49tuLF++SdMMNZli4+25z7EJROnrUbJGYNu3iWkFhYeZ4ikGDzPWEANhHcHAygoPzPPKI9OGHUvv20qpV9gfEAQCsl55ujkuYM8dsYbh0Uc9GjcxBy/36SRERxV/bqVNmeJg0SUpIMLdVqWJ2fXr8calixeKvCShJCA5ORnBwnrg46ZprpPPnpf/9zxzgBgBwPYYhbdxohoV58y5elEtSrVpmy8J990nXXWddjZc6d06aOVN6882LU7eWKyc9+qg0YoRUs6al5QEui+DgZAQH53ruOemNN8w/Ntu3M6ANAFzJrl1mWJg7V9q37+L2KlWke+4xw0Lbtq77f3d6ujmA+rXXLk7d6u0tPfig9MwzUr161tYHuBqCg5MRHJzr5EmpTh2zefnTT6UHHrC6IgBwbwcPSp9/bgaGHTsubi9bVrrjDjMs3HRTyRp4bBjSd9+ZAWLVKnObzWbO0vTcc1LLltbWh4scHRhfGrnCuRMcnIzg4HyvvSaNGiWFh5uDpsuUsboiAHAviYnmTHdz5uRcE8HbW7rlFjMs3HabGR5KunXrzKlcL53R78Ybzalcb7yR8XZWymsq3pAQafLki1Pxllaucu4EBycjODjf2bPmWIejR6V335WGDrW6IgAo/c6cMS+e58wxP41PTze322xSx47mIOc775QqVbK2zqKyc6fZVfazz8xPeiVzgbqRI82WFXf5lNtVLFgg3XWX2Tp0qawgN39+6Q0PrnTuBAcnIzgUjQ8+kIYMkapWlfbudWyebwDA1UlLk5YtM8PCN9+YH9xkadbMbFno29dccM1dHDggvfOOOcvfuXPmtrp1pWeflfr3z3uROjhXRobZ6+DST9svV7Wq+b4tbYEuI8P8d3f8eN6P22xmy8P+/cVz7gQHJyM4FI20NHMavz//lMaOlV56yeqKAKB0yMyU1q41L7q+/FI6ceLiY3XqmBct994rNWhgXY2u4Phx6b33pClTpL//NrdVry499ZQ5GxMfaBWNxERp+nTpxRetrsS1LV8uRUUV/esQHJyM4FB0vvjC/KSrfHmz1aFqVasrAoCSyTDMWYSyZkQ6dOjiY8HB5joL991nDgqmT39OKSlm68Pbb5vThktShQrmOhBPPln0i9mVZoZhzs61Zs3F2+7djj8/JEQKDCy6+qyQlHTllpYsc+aYAb+oERycjOBQdDIzzT9iW7dKw4dLEydaXREAlCz79plBYc4ccyrVLAEB5niF++4zP7X08rKsxBIjNdUc//D669KePeY2X1/p4YfNFanDwy0tr0RITzdn5ro0KMTH594vLMzsMmZPcX3qXpxWrJA6d7a/Hy0OJRTBoWjFxEjdupkzK+3ZY/5nAsA9ucLUhFa5mnM/dsxssZ0zR9qw4eJ2Hx9zJqT77pN69DAvenH1MjKkhQul6Ghp82Zzm6en2Wrz3HOus+idK0hJMRcKzAoJ69ebg/Av5e1tfkjYvr15u+EGs0UnPNxs4cnrarS4+/kXp6zxHa5y7g5f5xou6P333zfCw8MNHx8fo3nz5saqVavy3ferr74ybrrpJqNKlSpG+fLljTZt2hjfffddrv3mz59vNGjQwChTpozRoEEDY8GCBVdVU1JSkiHJSEpKuurzgX2ZmYbRpYthSIYxYIDV1QCwyldfGUZIiPl/QdYtJMTcXto5cu5JSYYxc6ZhdOtmGB4eF/fz8DCMrl0N4+OPDePUKctOoVTKzDSMH380f76X/m5uvdUwVq+2ujprHD1qGPPnG8bw4YbRooVheHrm/NlIhhEYaBg9ehjGhAmGsWqVYZw9m/exvvrKMGw283bp87O2leZ/+6507o5e57pccPj8888Nb29v48MPPzR27dplDBs2zChbtqxx4MCBPPcfNmyY8frrrxs///yzsWfPHmPUqFGGt7e3sXXr1ux91q1bZ3h6ehoTJkwwfv/9d2PChAmGl5eXsWHDBofrIjgUvY0bL/6D+fVXq6sBUNyy/ohefgHiThcQ+Z37s88axl13GYavb87HW7c2jMmTzQs5FL3Nmw3j7rtz/q7atTOMxYvNgFEaZWYaxu7dhvGf/xjGwIGGcc01ud+nkmHUqmUY991nGFOnGsYvvxhGRobjr5FXaA4NLd3/5rO4yrk7ep3rcl2VWrdurebNm2vatGnZ2xo0aKDevXsrOjraoWM0atRIffv21Uv/TNHTt29fJScna+nSpdn73HzzzapYsaLmzp3r0DHpqlQ87rpL+uor6fbbzSZiAO7B3rSM7tBlwZGBkpJUv7651kK/fuZaOCh+e/ZIb70lzZolXbhgbmvc2FwLom/fkj2W5MIFadu2i92O1q7NPWWozWZ21crqdtSunVSrVuFely6KJWPlaJd6a1+4cEFbtmzRyJEjc2zv1q2b1q1b59AxMjMzdfr0aVW6ZOWa9evXa8SIETn26969uyZNmpTvcVJTU5Wampr9fXJyskOvj8J59VXp66/NxYnWrTP7QAIonQxDOnVKOnhQ+vbbK184G4Y5Q1DdulK5csVWYrFISXEsNPTta16YNmnCjEhWu/ZaacYMacwYadIkado06bffpAcekEaPNgdRP/SQ5O9vdaX2JSebYxKygsLGjRfXtcji4yO1bn0xKLRta45PcCZPz9I3ANpRJencXSo4JCYmKiMjQ8HBwTm2BwcHKz6v4fh5ePvtt3XmzBndc8892dvi4+Ov+pjR0dEaO3bsVVQPZ6hfXxo0SPrPf8w/kCtX8gcSKKkuXDAH/h08mP8tJeXqjrl/f9HUWhL06iU1bWp1FbhUjRrmKtSjRpnhYdIkKTZWeuIJc22iYcOkxx6TKla0utKL4uJyznb0yy/m7IaXqlTpYkho315q3pwF8WByqeCQxXbZlaJhGLm25WXu3LkaM2aMFi5cqKDLJly+2mOOGjVKTz31VPb3ycnJCnWnJTUt9PLL0uzZZrPd0qXmrCAAXIthSCdPXjkUHD2a92whl6ta1bywypr68kreeqv0XTxv325+Qm1P9epFXgoKqGJF6fnnpREjpI8/lt580wwQo0dLr70mDRliPlajRvHWlZkp/f57zqAQG5t7v9q1cwaFevUkD4/irRUlg0sFhypVqsjT0zNXS0BCQkKuFoPLzZs3Tw8//LC+/PJL3XTTTTkeq1at2lUf08fHRz7Ea0uEhEhDh5r/8Y4aJd18M/+BAcUtNdXsPnOlYHD2rP3j+PiYfZ/zu4WEmN05HJ2acPjw0tfvOSrK/KTa3rl36FDcleFq+fmZLQyPPGJOlfvaa9Kvv5qB9913pQcflJ55xuzqVBRSU82pYy8dn5C1GnYWDw8zfF86PqG4Aw1KLpccHB0ZGampU6dmb2vYsKF69eqV7+DouXPnatCgQZo7d6569+6d6/G+ffvq9OnTWrJkSfa2W265RRUqVGBwtIs6edL8BCQpyfwUp3Fj9xssBfdWlIPlDENKTLxyKHCwd6iCg68cDKpWdby74YIF5gQJWTVmyXr+/PlSnz6On2dJ4s7nXpoZhrRkiRkg1qwxt9ls5qJ8zz0ntWhxcd+C/Jv/+29zPGBWUNi0yQwPl/L3l9q0uRgU2rSRypd37nmi5Cux6zhkTcf60UcfGbt27TKGDx9ulC1b1oiNjTUMwzBGjhxp9O/fP3v/OXPmGF5eXsb7779vHD16NPt26pKJrNeuXWt4enoar732mvH7778br732GtOxlgD33Zd7ujd3mc8d7q2waxmcO2cYe/YYxg8/GMZ//2sYY8YYxqBBhnHTTYZx7bW5p/TM7+bnZxj16pnz1z/8sGGMHWuuE/Djj4bx55/m6xTHuTMto9WVwRlWrzaM227L+fu96Sbz3+n8+Y79m4+NNYzZsw1jyBDDaNw473+3VasaRp8+hvHOO4bx88+GceGCNeeLkqXETscqSVOnTtUbb7yho0ePqnHjxpo4caI6duwoSRo4cKBiY2O1YsUKSVJUVJRWrlyZ6xgDBgzQzJkzs7+fP3++Ro8erX379qlOnToaP368+lzFxze0OBSvrE/fLn938ukbSjt77/0vvzQ/NbxSa0FCgmOvVb36lVsLKle2ZnICV5ia0CrufO7u4tdfzQHVc+eav+/8ZP3bGzzYnERgzRpzZrHL1a2bc3xC3bpMKoKr5+h1rksGB1dEcCg+7jyfO9xbRoYUFmb2dS8sf3/zWPmFgpo1mSUFsFJsrDmW75Ke2XZ5epozHF06PsHOEFDAISVyHQdAMj9tc2Q+94YNpUaN8r4oCgpiQDVcT2amdOxY/i0Ff/1lzqnuiBo18g8FYWHmLC986gi4rvBw6e67HQsOAwdK/fubaymULVvUlQH5IzjA5Rw96th+e/bkP31jmTJSaGj+F1ahofznC+c7c8YMtZeGgQMHLt4/dEhKSyv863zyiXkRAaBkc/TvXbduUpcuRVsL4AiCA1yOo3OVjx8vBQbm/tT2yBFz4am9e81bfipXvnL/7mrVaLXARZmZ5kxDVxpbcOKE/eN4eOTfWnDsmPSvf9k/BkvKAKWDo3/vWMMDroIxDg5ijEPxcXQ+9/zGOKSlmeEhv4u7Awek06ft1+Htbb7OlVotinpKO3ceKFnc5376dO7Wgktvhw871lpQvvyVxxbUqGG+t/JS2Pc+gJKFf/NwFYxxQInl6SlNnmzOLGOz5T2n+aRJ+f8n6u1tXriFheX/GklJV/7kOC7OvEjcv9+85adixSu3WlSvXvD/7BcskIYNyzneIyTE/NmU9hmlnH3uGRlmALnS7/zyRZLy4ulpDiq+0u88MPDq67v0+IV57wMoWfg3j5KGFgcH0eJQ/PK6eAwNNf8TLeoL5/R0+xeap07ZP46n55VbLWrVkvJ6O7nzdLQFOffk5Cv/rg4fvvK0h1kqVLAfBL2K4eMWK9/7AIof/+ZhNaZjdTKCgzVcuatOcrL9ri3p6faPExiY8+I0JER6+21z9ey8lOama3tT8Urmz6tfP3OfrJ91UpL9Y3t52e965kr/tF35vQ/A+fg3DysRHJyM4ICrlZFhfzBtfuHAET4+pe+PSkaGlJpasOdWqmR/sHtp+3kBAOAMjHEALJbVH75mTalt27z3SUnJ3WqxcqX5qZM9Bb3ALg1695Z69MjZWlCunNVVAQBQuhEcAAuVKyc1aGDesqxYIXXubP+5c+ZIbdoUWWmW2LBBuu8++/sNGyZFRRV5OQAA4BJ0VXIQXZVQXNx5ej53PncAAKzi6HUuy1sBLiZrej7p4kxCWUr79HzufO4AALg6ggPggvr0MacdrVkz5/aQkNI9Favk3ucOAIAro6uSg+iqBCu48/R87nzuAAAUJ2ZVAkoBT0/3HQTszucOAIAroqsSAAAAALsKHBzGjRun2bNnO7MWAAAAAC6qwMHh1Vdf1a+//urMWgAAAAC4qAIHh7CwMJ08edKZtQAAAABwUQUODvfee6++//57JSUlObMeAAAAAC6owMFh9OjRuv7669WlSxd9++23SkhIcGZdAAAAAFxIgadj9fPzkyQZhqHbb7893/1sNpvS09ML+jIAAAAAXECBg0OHDh1ks9mcWQsAAAAAF1Xg4LBixQonlgEAAADAlbEAHAAAAAC7CtzicKm4uDjt2LFDSUlJCggIUNOmTVWzZk1nHBoAAACACyhUcNi3b5+GDBmiH3/8MddjN954o6ZOnaprrrmmMC8BAAAAwAUUODgcPnxY7dq107Fjx9SgQQN17NhR1apV07Fjx7R69Wr98MMP6tChg37++WeFhoY6s2YAAAAAxazAwWHMmDE6duyYZsyYocGDB+d6/KOPPtIjjzyicePG6cMPPyxUkQAAAACsZTMMwyjIE0NDQxUZGalvvvkm33169+6tzZs36/DhwwWtz2UkJycrMDAwexwHAAAAUBo4ep1b4FmVEhIS1KhRoyvu06hRIx0/frygLwEAAADARRQ4OFStWlU7d+684j67du1S1apVC/oSAAAAAFxEgYND9+7dtXjxYn300Ud5Pv7f//5Xixcv1s0331zg4gAAAAC4hgKPcTh06JBatGihxMRENWzYUJ06dVJwcLCOHTumVatWaefOnapSpYo2b95cKmZVYowDAAAASiNHr3MLPKtSaGio1qxZoyFDhmj58uW5ui117txZ06ZNKxWhAQAAAHB3hVoArm7duvrxxx91+PBhbdu2TcnJydkrRxMYAAAAgNKjwMGhS5cuat++vcaNG6eQkBCFhIQ4sy4AAAAALqTAg6M3btyo9PR0Z9YCAAAAwEUVODg0aNBAsbGxTiwFAAAAgKsqcHAYOnSoFi1apF27djmzHgAAAAAuqMBjHCIiIhQVFaU2bdro0UcfVcuWLRUcHCybzZZr344dOxaqSAAAAADWKvA6Dh4eHrLZbMp6el6BIUtGRkbBqnMhrOMAAACA0qjI13F46aWXrhgWAAAAAJQeBW5xcDe0OAAAAKA0cvQ6t8CDoz09PXX//fcX9OkAAAAASpACB4eAgABWhwYAAADcRIGDQ6tWrbRjxw5n1gIAAADARRU4OIwdO1Y//fSTZs2a5cx6AAAAALigAs+qtGzZMkVFRWnQoEGaMmWKWrVqlec6DjabTS+++GKhCwUAAABgnUKt4+DQC9hsV72Ow9SpU/Xmm2/q6NGjatSokSZNmqQOHTrkue/Ro0f173//W1u2bNGff/6pJ598UpMmTcqxz8yZM/XQQw/leu65c+fk6+vrUE3MqgQAAIDSqMjXcVi+fHlBn3pF8+bN0/DhwzV16lS1a9dOH3zwgW655Rbt2rVLtWrVyrV/amqqqlatqhdeeEETJ07M97gBAQH6448/cmxzNDQAAAAA7s7l1nFo3bq1mjdvrmnTpmVva9CggXr37q3o6OgrPjcqKkpNmzbNs8Vh+PDhOnXqVIHrosUBAAAApVGRr+MgSenp6Zo4caJatWqlgIAAeXldbMDYvn27HnvsMe3Zs8fh4124cEFbtmxRt27dcmzv1q2b1q1bV5hSlZKSorCwMIWEhOi2227Ttm3brrh/amqqkpOTc9wAAAAAd1Xg4HDu3Dl17txZTz/9tA4cOKCAgABd2ngRERGhjz/+WJ988onDx0xMTFRGRoaCg4NzbA8ODlZ8fHxBS1X9+vU1c+ZMLVq0SHPnzpWvr6/atWunP//8M9/nREdHKzAwMPvGmhUAAABwZwUODhMmTNDatWsVHR2t+Ph4DR48OMfjgYGB6tSpk77//vurPvblMzMZhpFr29Vo06aNHnjgATVp0kQdOnTQF198oWuvvVZTpkzJ9zmjRo1SUlJS9u3QoUMFfn0AAACgpCvw4Oh58+YpKipKzz77rKTcF/uSVLt2bbtdgi5VpUoVeXp65mpdSEhIyNUKURgeHh5q2bLlFVscfHx85OPj47TXBAAAAEqyArc4HDx4UC1btrziPgEBAUpKSnL4mGXKlFFkZKRiYmJybI+JidENN9xQoDrzYhiGtm/frurVqzvtmAAAAEBpVuAWh/Lly+v48eNX3Gfv3r2qWrXqVR33qaeeUv/+/dWiRQu1bdtWM2bM0MGDBzVkyBBJZheiuLi4HGMntm/fLskcAH38+HFt375dZcqUUcOGDSWZq1y3adNGdevWVXJyst59911t375d77///lXVBgAAALirAgeHNm3aaPHixUpKSlJgYGCuxw8fPqwlS5aod+/eV3Xcvn376sSJExo3bpyOHj2qxo0ba8mSJQoLC5NkLvh28ODBHM9p1qxZ9v0tW7Zozpw5CgsLU2xsrCTp1KlTeuSRRxQfH6/AwEA1a9ZMq1atUqtWra7upAEAAAA3VeB1HFatWqXOnTurefPmmjx5spYuXaoJEybo9OnTWr9+vYYOHaq//vpL69evV2RkpLPrLnas4wAAAIDSqMhXju7YsaPef/99Pfnkk+rQoUP29vLly0uSPD09NXXq1FIRGgAAAAB3V+iVo3///XdNnz5dGzdu1MmTJxUQEKDWrVvrscceU6NGjZxVp+VocQAAAEBp5Oh1bqGDg7sgOAAAAKA0cvQ6t8DTsQIAAABwHwQHAAAAAHYRHAAAAADYRXAAAAAAYBfBAQAAAIBdBAcAAAAAdhEcAAAAANhFcAAAAABgF8EBAAAAgF0EBwAAAAB2ERwAAAAA2EVwAAAAAGAXwQEAAACAXQQHAAAAAHYRHAAAAADYRXAAAAAAYBfBAQAAAIBdBAcAAAAAdhEcAAAAANhFcAAAAABgF8EBAAAAgF0EBwAAAAB2ERwAAAAA2EVwAAAAAGAXwQEAAACAXQQHAAAAAHYRHAAAAADYRXAAAAAAYBfBAQAAAIBdBAcAAAAAdhEcAAAAANhFcAAAAABgF8EBAAAAgF0EBwAAAAB2ERwAAAAA2EVwAAAAAGAXwQEAAACAXQQHAAAAAHYRHAAAAADYRXAAAAAAYBfBoaTYvFnq0sX8CgAAABQzgkNJ8ckn0vLl0qefWl0JAAAA3JCX1QXgCg4ckBITJZtNmj3b3Pb559KAAZJhSFWqSGFh1tYIAAAAt0BwcGXh4bm3JSRIkZEXvzeMYisHAAAA7ouuSq5s9mzJK59s5+V1sRUCAAAAKGIuGRymTp2qiIgI+fr6KjIyUqtXr85336NHj+q+++5TvXr15OHhoeHDh+e531dffaWGDRvKx8dHDRs21Ndff11E1TvR/fdLGzfm/diGDebjAAAAQDFwueAwb948DR8+XC+88IK2bdumDh066JZbbtHBgwfz3D81NVVVq1bVCy+8oCZNmuS5z/r169W3b1/1799fO3bsUP/+/XXPPfdoY34X5a7I47JfFYOkAQAAUIxshuFaneRbt26t5s2ba9q0adnbGjRooN69eys6OvqKz42KilLTpk01adKkHNv79u2r5ORkLV26NHvbzTffrIoVK2ru3LkO1ZWcnKzAwEAlJSUpICDA8RMqrMOHpZYtpdBQ6eGHpehoc9C0zSb9+KPUuXPx1QIAAIBSx9HrXJdqcbhw4YK2bNmibt265djerVs3rVu3rsDHXb9+fa5jdu/e/YrHTE1NVXJyco6bJUJCpNhYs8vSo49K+/aZXZQMQ+rb1wwWAAAAQBFzqeCQmJiojIwMBQcH59geHBys+Pj4Ah83Pj7+qo8ZHR2twMDA7FtoaGiBX7/QfHzMFgbJ7LI0Y4bUpIl0/Lh0111Saqp1tQEAAMAtuFRwyGLLukj+h2EYubYV9TFHjRqlpKSk7NuhQ4cK9fpO5e8vLVggVahgtkQ89ZTVFQEAAKCUc6ngUKVKFXl6euZqCUhISMjVYnA1qlWrdtXH9PHxUUBAQI6bS6ldW/rsM/P+1KnmytIAAABAEXGp4FCmTBlFRkYqJiYmx/aYmBjdcMMNBT5u27Ztcx1z2bJlhTqmS+jRQ3r5ZfP+o49K27dbWg4AAABKL5dbOfqpp55S//791aJFC7Vt21YzZszQwYMHNWTIEElmF6K4uDh9cskn7Nv/uWBOSUnR8ePHtX37dpUpU0YNGzaUJA0bNkwdO3bU66+/rl69emnhwoX64YcftGbNmmI/P6d76SXp55+lpUulO++UNm+WKla0uioAAACUMi43HatkLgD3xhtv6OjRo2rcuLEmTpyojh07SpIGDhyo2NhYrVixInv/vMYqhIWFKTY2Nvv7+fPna/To0dq3b5/q1Kmj8ePHq0+fPg7XZNl0rI44eVJq0ULav99shVi8OPe6DwAAAEAeHL3Odcng4IpcOjhI0rZt0g03SOfPS2PHmi0RAAAAgB0lch0HFEKzZtL06eb9MWPMrksAAACAkxAcSpMBA6QhQ8zF4e6/3+y6BAAAADgBwaG0mTRJatVK+vtvc7D0uXNWVwQAAIBSgOBQ2vj4SPPnS1WqmOMeHnvMbIEAAAAACoHgUBqFhkrz5pkzK82cKc2YYXVFAAAAKOEIDqVVly5SdLR5f+hQaeNGa+sBAABAiUZwKM2eeUa64w4pLU266y7p+HGrKwIAAEAJRXAozWw2s6vStddKhw9L/fpJ6elWVwUAAIASiOBQ2gUESAsWSGXLSj/9JL34otUVAQAAoAQiOLiDRo2k//7XvP/aa9LXX1tbDwAAAEocgoO7uOce6amnzPsDBkh//GFtPQAAAChRCA7u5LXXpI4dpdOnpT59pJQUqysCAABACUFwcCfe3ub6DtWrS7t2SYMHszgcAAAAHEJwcDfVqklffil5eZkhYvJkqysCAABACUBwcEft2knvvGPef/ppafVqa+sBAACAyyM4uKsnnpDuu0/KyJDuvls6csTqigAAAODCCA7uymaTZsyQrrtOOnbMDA8XLlhdFQAAAFwUwcGdlS1rLg4XGCitWyc984zVFQEAAMBFERzc3TXXSJ98Yt5/911pzhxr6wEAAIBLIjhAuv126YUXzPv/+pf066/W1gMAAACXQ3CAaexYqVs36exZc3G4pCSrKwIAAIALITjA5OlpdlMKC5P++kt68EEpM9PqqgAAAOAiCA64qHJlaf58ycdHWrRIeu01qysCAACAiyA4IKcWLaT33zfvjx4tLVtmbT0AAABwCQQH5Pbww9LgwZJhmIvEHThgdUUAAACwGMEBeZsyRYqMlE6ckO66Szp/3uqKAAAAYCGCA/Lm6yt99ZU57mHzZunJJ62uCAAAABYiOCB/YWHS3LmSzSZ9+KH00UdWVwQAAACLEBxwZV27Sq++at5//HGz9QEAAABuh+AA+0aONFeXTk2V7rxTSky0uiIAAAAUM4ID7PPwkGbNkq65Rjp4ULr/fikjw+qqAAAAUIwIDnBMhQrSggWSv7+5tsOYMVZXBAAAgGJEcIDjrrtO+s9/zPuvviotXmxtPQAAACg2BAdcnXvvvTg1a//+0l9/WVsPAAAAigXBAVfvzTeldu2kpCSpTx/pzBmrKwIAAEARIzjg6pUpI33xhRQcLP36q/TII5JhWF0VAAAAihDBAQVTo4YZHjw9pTlzpPfft7oiAAAAFCGCAwquY0ez25IkjRghrVtnbT0AAAAoMgQHFM7w4VLfvlJ6unT33VJ8vNUVAQAAoAgQHFA4Nps5RWvDhtKRI2aISEuzuioAAAA4GcEBhVeunLk4XPny0qpV0siRVlcEAAAAJyM4wDnq1ZNmzTLvv/OOOXAaAAAApQbBAc5zxx3Sc8+Z9wcNknbtsrYeAAAAOA3BAc716qvSjTeai8L16SMlJ1tdEQAAAJyA4ADn8vKS5s6VQkKkP/6QHnqIxeEAAABKAYIDnK9qVemrr8wVphcsuLjWAwAAAEosggOKRqtW0rvvmvdHjZJ++snaegAAAFAoBAcUnUcekQYOlDIzpX79pEOHrK4IAAAABURwQNGx2aSpU6VmzaTjx82VpVNTra4KAAAABeCSwWHq1KmKiIiQr6+vIiMjtXr16ivuv3LlSkVGRsrX11e1a9fW9OnTczw+c+ZM2Wy2XLfz588X5WlAkvz8zPEOFStKGzdKI0ZYXREAAAAKwOWCw7x58zR8+HC98MIL2rZtmzp06KBbbrlFBw8ezHP//fv3q0ePHurQoYO2bdum559/Xk8++aS++uqrHPsFBATo6NGjOW6+vr7FcUqIiJA++8xsgZg27eJCcQAAACgxXC44vPPOO3r44Yc1ePBgNWjQQJMmTVJoaKimTZuW5/7Tp09XrVq1NGnSJDVo0ECDBw/WoEGD9NZbb+XYz2azqVq1ajluKEa33CKNGWPeHzJE2rbN0nIAAC5q82apSxfzKwCX4lLB4cKFC9qyZYu6deuWY3u3bt20bt26PJ+zfv36XPt3795dmzdvVlpaWva2lJQUhYWFKSQkRLfddpu22blwTU1NVXJyco4bCmn0aKlHD+n8eenOO6WTJ62uCADgaj75RFq+XPr0U6srAXAZlwoOiYmJysjIUHBwcI7twcHBio+Pz/M58fHxee6fnp6uxMRESVL9+vU1c+ZMLVq0SHPnzpWvr6/atWunP//8M99aoqOjFRgYmH0LDQ0t5NlBHh7S7NlS7drS/v3SAw+YMy4BANzbgQPSli3S1q3SvHnmts8/N7/fssV8HIDlvKwuIC82my3H94Zh5Npmb/9Lt7dp00Zt2rTJfrxdu3Zq3ry5pkyZonez1hq4zKhRo/TUU09lf5+cnEx4cIaKFc1F4dq0kZYulV55RXr5ZaurAgAUl7Q0KTFRSki4eHvggdz7JSRIkZEXv1+5UgoKMm8VKpgfRgEoVi4VHKpUqSJPT89crQsJCQm5WhWyVKtWLc/9vby8VLly5Tyf4+HhoZYtW16xxcHHx0c+Pj5XeQZwSJMm0gcfSAMGSGPHSi1bml2YAAAlT2am9PffOYNAQoI5Dffl2xISzH0LolOni/c9PaWqVS8GiaxbftvKlTMn6ABQKC4VHMqUKaPIyEjFxMTojjvuyN4eExOjXr165fmctm3bavHixTm2LVu2TC1atJC3t3eezzEMQ9u3b9d1113nvOJxdR580JyedepU6f77zabo2rWtrgoAYBhSSkreF/15hYHERCkj4+pew8NDqlIl58W9dLGb0qU6d5YuXLj4eklJ5uvFx5s3R/j6OhYysrbzwSGQJ5uR1a/HRcybN0/9+/fX9OnT1bZtW82YMUMffvihdu7cqbCwMI0aNUpxcXH65JNPJJnTsTZu3FiPPvqo/vWvf2n9+vUaMmSI5s6dqzvvvFOSNHbsWLVp00Z169ZVcnKy3n33XX366adau3atWrVq5VBdycnJCgwMVFJSkgICAors/N3KhQvmJ0gbNpitEOvWSf7+VlcFwBVs3iw9+6z0xhtSixZWV1O8iuLcz5/P+6I/v1aBgizWWaGCY5/+BwVJlSqZrQaX2rrV7Jrk4WG2YmR93bJFat784n6pqbm7OuV3PseOmed+tQICHG/NqFxZ8nLS57Du/L6X3Pv8LT53R69zXarFQZL69u2rEydOaNy4cTp69KgaN26sJUuWKCwsTJJ09OjRHGs6REREaMmSJRoxYoTef/991ahRQ++++252aJCkU6dO6ZFHHlF8fLwCAwPVrFkzrVq1yuHQgCJSpoz05ZfmH4QdO6T/+z9p5kyakwHknFnH3S4gHDn39HTpxAnHWwVOn776Ovz8pOBgxz6hr1rV/D+9MIKCpGrVpNBQ6eGHpY8+kg4dMrdfysdHqlnTvNljGNKZM/kHpMu3Hz9u/myTk83bX3/Zfw2bzQwPjrZmVKiQ/985d37fS+59/iXk3F2uxcFV0eJQhJYvl266yfxkado0c50HQLL8ExjLlfbzNwzp3Dnzwu7MGenPP6UjR8xPiJ97zuySEhgovfiiuX/58rkvIkuLSy/uX3nFPPfy5aV77zXHBGT9jLIucE+eNH9+V8PLy/FP0YOCpLJlnX+e9qSmmgHEZjPP78KF4u02ZBjSqVOOB7ITJ67+Nby9c/7c/fzMn3WlSuYCqadPu8/7Xsr7ve8u55/XuVetKn33nflerFJF+ueD86Lm6HUuwcFBBIci9tZb0jPPmP+hrl4ttW5tdUVwBU8+KU2ZYn6dPNnqaoqfK5x/evrFi9bLb2fP5v+YI/ufPXv1F7+4yJmfdKNgiqPlB+4lKzRnKab/IwkOTkZwKGKGId19t/TVV1JIiNmntbR+woArO3DA7L9ss5krjickmO+FpUuL/RMYS1zt+RuG+Ql9UV3cX7hQPOft42N+In7mTN6P22xSnToXB9GWNsePS3v35n2R4OFhTiLRo0fR9a1H8chrrMl330lffJH3ukal/X0vXfm9X9rP/0rn7uVldt++//5iKYXg4GQEh2Jw+rTUqpW0e7c5i8ayZfxRdDepqebsJ/Y8/HDR12KVjz6yv0/t2jkv9ItjIUWbzexO4ejN3//q9s0aKJs1QPZylw+QLY3c+dzdnbv/7t35/F3k3Evs4Gi4sfLlzcXhWrUyxz288IL0+utWVwVnSk+XDh82Vw6PjTW/Zt1iY83+7Y5w5OK6NNu3L+/tPj5Xf8Hu6L4+PsXbzeXymXXciTufu7tz99+9O59/CTl3ggNcS4MG0n//K91zjzkgtFUr6ZIZsuDiMjPNedXzCwYHD9qf793f35zJZf/+3I899phjM6mUdHFx5honl3vrLen66/MPAaWhhc7RmXVKI3c+d3fn7r97dz7/EnbudFVyEF2Vitkzz5gXSeXLm0Fi6tTSO7NMSWIY5kDAS8PApfdjY+3P/+7tbfbRj4i4eAsPv3i/alVp2zbH5nMvrRydz760snpmHSu587m7O3f/3bvz+bvAudNVCSVbdLQ5FeWKFdIjj5jTEbr43MalRnJy3sEg6/uUlCs/38PD/OTk0jBwaTioUcPc50pK2CcwTufu53/pH0ybzX0uHiT3Pnd35+6/e3c+/xJ07rQ4OIgWh2J24ID0xx/mbAKJiea2SpXMBVJ8fc2Zl669lqkFC+LcuYuBIK9gcPKk/WNUr567pSDr+9BQs1WhsFzgExhLufv5AwCKDS0OKNnCw3NvO3lSuu22i997eBRuwOeV9rt0lherXe0iYGlp5liC/FoNjh2zf4zKlfPuRhQebnYz8vMr5Ek5oAR9AlMk3P38AQAuh+AA1zR7tjRwoDkLT34yM80pXItqQR1fX+cFkcv3u5oZai5fhj4jw5x9KL9gEBdnf0aG8uXzDgZZ35cvX8gfHgAAKG3oquQguipZIL+5jTdulOrXL9hiVo7sWxz/JDw9rxw0JDNY+PlJCxeatXl7m7MNxcdfOVBJZui5vKXg0vuVKtHNCwAASKKrEkqTy2eW8fKSAgLMm7PZW4W3sKEkaxXejIyrby1JSzPXQMhSu3b+MxMFBxMMAACAUxEc4LqsmFkm61N+Pz+pShXnHz8tzX7gOHtWWrVKmjs37y5HXl7mFLX9+zu/PgAAgHzQVclBdFWyiDvPLOMiy9ADAIDSzdHrXDuTqQMWu3QQsbvOLJO15oG9tQ8AAACKEFcigKvK6qoVGSlNn25+rVbNfRYBAwAALoUxDoCrCgkxp1rN6qr1yCPu1VULAAC4FIID4MpYBAwAALgIuioBAAAAsIvgAAAAAMAuggMAAAAAuwgOAAAAAOwiOAAAAACwi+AAAAAAwC6CAwAAAAC7CA4AAAAA7GIBOAcZhiFJSk5OtrgSAAAAwHmyrm+zrnfzQ3Bw0OnTpyVJoaGhFlcCAAAAON/p06cVGBiY7+M2w160gCQpMzNTR44cUfny5WWz2Yr99ZOTkxUaGqpDhw4pICCg2F8f1uF377743bsnfu/ui9+9+7L6d28Yhk6fPq0aNWrIwyP/kQy0ODjIw8NDISEhVpehgIAA/jNxU/zu3Re/e/fE79198bt3X1b+7q/U0pCFwdEAAAAA7CI4AAAAALCL4FBC+Pj46OWXX5aPj4/VpaCY8bt3X/zu3RO/d/fF7959lZTfPYOjAQAAANhFiwMAAAAAuwgOAAAAAOwiOAAAAACwi+AAAAAAwC6CQwkwdepURUREyNfXV5GRkVq9erXVJaGIRUdHq2XLlipfvryCgoLUu3dv/fHHH1aXBQtER0fLZrNp+PDhVpeCYhAXF6cHHnhAlStXlr+/v5o2baotW7ZYXRaKWHp6ukaPHq2IiAj5+fmpdu3aGjdunDIzM60uDU62atUq9ezZUzVq1JDNZtM333yT43HDMDRmzBjVqFFDfn5+ioqK0s6dO60pNg8EBxc3b948DR8+XC+88IK2bdumDh066JZbbtHBgwetLg1FaOXKlXr88ce1YcMGxcTEKD09Xd26ddOZM2esLg3FaNOmTZoxY4auv/56q0tBMfj777/Vrl07eXt7a+nSpdq1a5fefvttVahQwerSUMRef/11TZ8+Xe+9955+//13vfHGG3rzzTc1ZcoUq0uDk505c0ZNmjTRe++9l+fjb7zxht555x2999572rRpk6pVq6auXbvq9OnTxVxp3piO1cW1bt1azZs317Rp07K3NWjQQL1791Z0dLSFlaE4HT9+XEFBQVq5cqU6duxodTkoBikpKWrevLmmTp2qV199VU2bNtWkSZOsLgtFaOTIkVq7di2tym7otttuU3BwsD766KPsbXfeeaf8/f316aefWlgZipLNZtPXX3+t3r17SzJbG2rUqKHhw4frueeekySlpqYqODhYr7/+uh599FELqzXR4uDCLly4oC1btqhbt245tnfr1k3r1q2zqCpYISkpSZJUqVIliytBcXn88cd166236qabbrK6FBSTRYsWqUWLFrr77rsVFBSkZs2a6cMPP7S6LBSD9u3b68cff9SePXskSTt27NCaNWvUo0cPiytDcdq/f7/i4+NzXPf5+PioU6dOLnPd52V1AchfYmKiMjIyFBwcnGN7cHCw4uPjLaoKxc0wDD311FNq3769GjdubHU5KAaff/65tm7dqk2bNlldCorRvn37NG3aND311FN6/vnn9fPPP+vJJ5+Uj4+PHnzwQavLQxF67rnnlJSUpPr168vT01MZGRkaP3687r33XqtLQzHKurbL67rvwIEDVpSUC8GhBLDZbDm+Nwwj1zaUXk888YR++eUXrVmzxupSUAwOHTqkYcOGadmyZfL19bW6HBSjzMxMtWjRQhMmTJAkNWvWTDt37tS0adMIDqXcvHnzNHv2bM2ZM0eNGjXS9u3bNXz4cNWoUUMDBgywujwUM1e+7iM4uLAqVarI09MzV+tCQkJCrjSK0mno0KFatGiRVq1apZCQEKvLQTHYsmWLEhISFBkZmb0tIyNDq1at0nvvvafU1FR5enpaWCGKSvXq1dWwYcMc2xo0aKCvvvrKoopQXJ555hmNHDlS/fr1kyRdd911OnDggKKjowkObqRatWqSzJaH6tWrZ293pes+xji4sDJlyigyMlIxMTE5tsfExOiGG26wqCoUB8Mw9MQTT2jBggX66aefFBERYXVJKCY33nijfv31V23fvj371qJFC91///3avn07oaEUa9euXa5pl/fs2aOwsDCLKkJxOXv2rDw8cl6SeXp6Mh2rm4mIiFC1atVyXPdduHBBK1eudJnrPlocXNxTTz2l/v37q0WLFmrbtq1mzJihgwcPasiQIVaXhiL0+OOPa86cOVq4cKHKly+f3eoUGBgoPz8/i6tDUSpfvnyusSxly5ZV5cqVGeNSyo0YMUI33HCDJkyYoHvuuUc///yzZsyYoRkzZlhdGopYz549NX78eNWqVUuNGjXStm3b9M4772jQoEFWlwYnS0lJ0V9//ZX9/f79+7V9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AMAuggMAAABggZkzZ8pms2nmzJlWl+IQggMAAADgIJvNpqioKKvLsITLzRs2btw41a5dWw888IDVpQAAAABF5o477lCbNm1UvXp1q0txiMu1OLz66qv69ddfrS4DAADA5Ww+slldZnXR5iObrS4FThAYGKj69esrMDDQ6lIc4nLBISwsTCdPnrS6DAAAAJfzyY5PtDx2uT7d8anVpWS7cOGCpkyZou7duys0NFQ+Pj4KCgpSnz59tG3btjyfs2jRInXv3l2VK1eWr6+vwsPD1b9/f/3222+5jj158mS1atVK5cuXV7ly5dSwYUM99dRT+vvvvx2qLyoqSjabTampqXrppZd0zTXXyNvbW2PGjMneZ//+/Ro8eLBq1aolHx8fVa9eXQMHDtSBAwey91mxYkX2uggrV66UzWbLvmWNUUhKStLrr7+uTp06qUaNGipTpoxq1KihBx98UHv37s1VW35jHLK6Qx0/flyDBg1SUFCQ/Pz81KZNG61YscKh8y4KLtdV6d5779XMmTOVlJRUYtIXAADA5QzD0Nm0s4U+zsGkgzpx9oRsNps+/+1zSdLc3+bqnkb3yDAMVfavrFqBtQp8fH9v/0ItFHby5EkNHz5cHTp0UI8ePVSxYkXt27dPixYt0tKlS7Vq1Sq1bNkye/9nn31Wb775pipVqqTevXsrKChIhw4d0g8//KDIyEg1btxYknT+/Hl1795dq1atUt26dfXQQw/Jx8dHf/75p6ZPn64HH3xQFStWdLjOPn36aMeOHerevbsqVaqk2rVrS5I2btyo7t2768yZM+rZs6euueYaxcbG6rPPPtPSpUu1fv161a5dW+Hh4Xr55Zc1duxYhYWFaeDAgdnHbtq0qSTp999/10svvaTOnTvrjjvuUNmyZbV7927NmTNH3377rbZu3aqwsDCH6j116pTatWungIAA3X///UpISNC8efPUvXt3bdmyJfvnVJxcLjiMHj1aW7duVZcuXTRu3Di1bNlSQUFBVpcFAABwVc6mnVW56HJFcuzjZ4+r/cftnXKslFEpKlumbIGfX7FiRR08eFA1a9bMsX3nzp1q06aNnn/+ecXExEiSlixZojfffFPXXXedli9frsqVK2fvn56erhMnTmR//9JLL2nVqlXq37+/Pv74Y3l6emY/lpSUlON7Rxw5ckS//PKLKlWqlL0tLS1N/fr1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apXnOg42m00vvviiRVUCAABcmWEYyjh3rlDHyMzI0OYJE6S8Ztv5Z9vm6GgFt2lT4G5Lnn5+hZrRsmLFijp48KBq1qyZY/vOnTvVpk0bPf/884r5J9wsWbJEb775pq677jotX75clStXzt4/PT1dJ06cyP7+pZde0qpVq9S/f399/PHH8rzk/JKSknJ8nxebzab7779f48eP1+LFi3X33XfnePyzzz6Tn5+f7rzzzuxto0aN0vvvv5/j2IZhaPDgwfrvf/+rtWvXql27dlfx0zFlZmbqkUceUXp6ulatWqX27dtnH/uBBx7QnDlzrvqYVrAZLjbvk4eHY40gNputWNdxSE5OVmBgoJKSkhQQEFBsrwsAAFzb+fPntX//fkVERMjX1zd7e/rZs/qiZUsLK3PMPZs2ycvfv0iOffvtt+v7779XSkqKvL29deutt2rJkiX66aef1Llz53yfl5GRoUqVKslms2n//v2qWLFigV7/jz/+UP369XX77bdr4cKF2dt37Nihpk2bql+/fpo7d67d42zdulWRkZEaM2aMXn755eztjk7HumrVKnXq1Ek9e/bUokWLchz7wIEDqlOnjjIyMpw+HWt+783LOXqd63ItDsuXL7e6BAAAAFyF7du364033tCaNWsUHx+fazxAYmKiqlevrp9//lk+Pj7q1KnTFY+3e/duJScn66abbrIbGr755htt3749x7aoqChFRUWpXr16atGihZYuXaqTJ0+qUqVKkqRPP/1UknJ0U5LMMRXvvfeePv/8c+3evVspKSk5LuaPHDlyxVrykzV+t0OHDrkeCwsLU2hoqGJjYwt07OLkcsHB3hsJAACgJPD089M9l/Xvv1oJW7ZoxZAhdveLmj5dQZGRBXoNTz+/Aj0vy7p169SlSxdJUrdu3VS3bl2VK1dONptN33zzjXbs2KHU1FRJ0qlTp1SzZk27PUxOnTolSbm6P+Xlm2++yXNsbFRUlCQzHGzevFlffPGFhgwZoszMTM2dO1dBQUHq1q1bjufcddddWrx4sa699lr17dtXQUFB8vb21qlTpzR58uTs87haSUlJkqSgoKA8Hw8ODiY4FFR6erqmTJmiuXPnavfu3Tp79qzS09MlmYl2xowZGj58uK699lqLKwUAAMibzWYrdBegajfcIP/gYJ1NSMh7nIPNJv/gYFW74QbLpmYdP368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grMZhqGHH35Y1157reLi4swuB07w0UcfacuWLUpKSjK7FDjRwYMH9corr+jhhx/W448/rp9++kkPPvigrFar/vGPf5hdHqrRY489poyMDLVq1Uq+vr4qLi7WrFmzNGzYMLNLgxOVnttVdN535MgRM0oqh+DgBiwWS5nvDcMotw2ea/To0fr555+1fv16s0uBE6SkpGjs2LFasWKFAgICzC4HTlRSUqKuXbtq9uzZkqROnTpp165deuWVVwgOHu7jjz/WokWL9MEHH6ht27batm2bxo0bp+joaN1zzz1mlwcnc+XzPoKDC6tdu7Z8fX3LjS6kpqaWS6PwTGPGjNGXX36pdevWqWHDhmaXAydITk5WamqqunTpYttWXFysdevWaf78+crPz5evr6+JFaK61K9fX23atCmzrXXr1vrss89MqgjO8sgjj2jChAn629/+Jklq166djhw5ojlz5hAcvEi9evUknR95qF+/vm27K5330ePgwmrUqKEuXbpo5cqVZbavXLlSV199tUlVwRkMw9Do0aP1+eefa/Xq1WratKnZJcFJ/vznP2vHjh3atm2b7da1a1fddddd2rZtG6HBg11zzTXlll3et2+fGjdubFJFcJbc3Fz5+JQ9JfP19WU5Vi/TtGlT1atXr8x5X0FBgdauXesy532MOLi4hx9+WHfffbe6du2qnj17asGCBTp69KhGjhxpdmmoRqNGjdIHH3yg//znP6pZs6Zt1CksLEyBgYEmV4fqVLNmzXK9LMHBwYqMjKTHxcM99NBDuvrqqzV79mzdfvvt+umnn7RgwQItWLDA7NJQzQYPHqxZs2apUaNGatu2rbZu3apnn31WI0aMMLs0OFh2drb2799v+/7QoUPatm2batWqpUaNGmncuHGaPXu2WrRooRYtWmj27NkKCgrSnXfeaWLVFzDg8l566SWjcePGRo0aNYzOnTsba9euNbskVDNJFd7efvtts0uDCXr37m2MHTvW7DLgBEuWLDHi4uIMq9VqtGrVyliwYIHZJcEJMjMzjbFjxxqNGjUyAgICjGbNmhmTJk0y8vPzzS4NDrZmzZoK//9+zz33GIZhGCUlJca0adOMevXqGVar1ejVq5exY8cOc4u+ANdxAAAAAGAXPQ4AAAAA7CI4AAAAALCL4AAAAADALoIDAAAAALsIDgAAAADsIjgAAAAAsIvg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5mjnO3v2rCIiIiRJqampHnGRNldz6NAh/eMf/9D69eslSTfffLNee+01ftYAAHgRmqPh9jZv3ixJatq0KSey1aRp06b67rvv9MQTT8jf319ffPGF2rVrp6+//trs0gAAgIshOMBlbdq0SZLUvXt3kyvxbL6+vnrsscf0008/qW3btjp58qQGDRqkkSNHKjs72+zyAACAiyA4wGX99NNPkqRu3bqZXIl36NixozZv3qyHH35YkvTaa6+pU6dO2rhxo8mVAQAAV0BwgEsyDIMRBxMEBARo7ty5+vbbb9WwYUPt379f11xzjaZOnarCwkKzywMAACYiOMAlpaSk6OTJk/Lz81OnTp3MLsfrXH/99dqxY4fuuusulZSUaMaMGerZs6f27t1rdmkAAMAkBAe4pNLRhvbt2yswMNDkarxTeHi4Fi1apI8++kgRERFKTk5Wp06dNH/+fJZtBQDACxEc4JLob3Add9xxh3bs2KEbbrhBeXl5GjNmjG688UadOHHC7NIAAIATERzgkuhvcC0NGjTQ8uXL9cILLyggIEDffPON2rVrp08//dTs0gAAgJMQHOByioqKlJycLIkRB1fi4+OjMWPGaMuWLercubPS0tJ0++236+6779bZs2fNLg8AAFQzggNczq5du5Sbm6uaNWuqVatWZpeDP2jdurU2bNigyZMny8fHR4sWLVL79u313XffmV0aAACoRgQHuJzS/ob4+Hj5+PAr6opq1KihGTNmaP369YqNjVVKSoquv/56jR8/Xnl5eWaXBwAAqgFnZXA59De4j549e2rbtm267777ZBiG5s6dq/j4eG3fvt3s0gAAgIMRHOByWFHJvYSEhOi1117TkiVLFBUVpZ07dyo+Pl5PPfWUiouLzS4PAAA4CMEBLiU7O1u7du2SxIiDuxk0aJB27typIUOGqLCwUI899piuu+46HT582OzSAACAAxAc4FKSk5NVUlKihg0bqn79+maXg0tUp04dffHFF3rzzTcVEhKi77//Xu3bt9c777zDReMAAHBzBAe4FPob3J/FYtGIESO0fft2XXPNNcrKytI///lP3XrrrTp16pTZ5QEAgMtEcIBLob/BczRr1kxr167VnDlz5O/vry+++ELt2rXT119/bXZpAADgMhAc4FIYcfAsvr6+mjBhgjZt2qQ2bdro5MmTGjRokO6//37l5OSYXR4AALgEBAe4jBMnTujYsWPy8fFRly5dzC4HDtSpUyclJyfroYcekiS9+uqr6tixoy0oAgAA10dwgMsonabUtm1bhYSEmFwNHC0gIEDPPvusVq1apYYNG2r//v265pprNG3aNBUWFppdHgAAsIPgAJdBf4N3+POf/6wdO3borrvuUnFxsaZPn66rr75av/zyi9mlAQCAi3Db4LB69WqNGDFCrVq1UnBwsBo0aKAhQ4YoOTm53LGFhYV69tln1a5dOwUGBio8PFxXX321fvzxRxMqR2Xob/Ae4eHhWrRokT766COFh4dr8+bN6tSpk1566SWWbQUAwEW5bXB45ZVXdPjwYY0dO1ZLly7VvHnzlJqaqh49emj16tW244qLi3XzzTdr+vTpGjZsmJYtW6b3339fAwYMoDnThZSUlCgpKUkSIw7e5I477tDOnTvVt29fnTt3TqNHj9aNN96oEydOSJJWrVqlNm3aaNWqVSZXCmfjvYc38vbfe29+/W7z2g03dfLkyXLbsrKyjLp16xp//vOfbduee+45w8fHx9iwYcMVPV9GRoYhycjIyLiix0HFdu3aZUgygoKCjMLCQrPLgZMVFxcbL7zwghEQEGBIMmrVqmV88sknRnx8vCHJiI+PN0pKSswuE05SUlLCew+v4+2/9978+l3htVf1PNdtRxyioqLKbQsJCVGbNm2UkpJi2zZv3jz16tVLPXr0cGZ5uESl/Q1dunSRn5+fydXA2Xx8fDRmzBglJyerc+fOSktL0+23324bhUpKStKKFStMrhLOsmLFCt57eB1v/7335tfvTq/dYhieM6E4IyNDjRs31vXXX6/PP/9cKSkpatSokcaMGaOQkBC9+eabOnPmjFq2bKlHH31U99xzT6WPlZ+fr/z8fNv3mZmZiomJUUZGhkJDQ53xcrzK/fffr1dffVXjx4/X008/bXY5MFFBQYESExM1e/bsMtvr1aunp59+Wu3atVOrVq1ktVpNqhCOVlxcrEOHDmnHjh3asWOHnn/+eaWnp0s6fyXy+vXra968eWrXrp2aN28uX19fkysGHMswDHXo0EE7d+609Xn5+fkpIiJCFovF5Oqqn2EYSk9PV1FRkW2bt7z+P752X19fde7cWZs2bXLqa8/MzFRYWJjd81yP+mh31KhRysnJ0aRJkyRJx48flyQtXLhQDRs21Pz58xUW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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": 29,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0.10454545454545454, 0.03636363636363636, 0.03409090909090909, 0.04090909090909091, 0.04090909090909091, 0.04318181818181818, 0.04318181818181818, 0.04772727272727273, 0.04318181818181818, 0.04318181818181818, 0.04318181818181818]\n",
      "[0.29375, 0.19375, 0.1875, 0.1875, 0.19375, 0.2, 0.2, 0.21875, 0.19375, 0.20625, 0.20625]\n",
      "[1.5333328247070312, 0.6666641235351562, 0.5333328247070312, 1.0, 1.0666656494140625, 1.2000045776367188, 1.2666702270507812, 1.0333328247070312, 1.4000015258789062, 1.4000015258789062, 1.366668701171875]\n",
      "[8.625, 7.125, 6.5, 7.375, 8.125, 8.125, 8.25, 8.0, 8.75, 8.75, 8.75]\n",
      "[14.0, 27.5, 28.0, 26.0, 25.5, 26.0, 25.5, 24.5, 26.0, 26.0, 26.0]\n"
     ]
    }
   ],
   "source": [
    "print(ic_rs)\n",
    "print(ic_vs)\n",
    "print(acc_rs)\n",
    "print(acc_vs)\n",
    "print(acc_fs)\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Finetune"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 30,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                               \r"
     ]
    }
   ],
   "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": 31,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Epoch: [0][0/24]\tTime 0.044 (0.044)\tData 0.018 (0.018)\tLoss 0.0686 (0.0686)\tAcc@1 99.219 (99.219)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.167 Acc@5 100.000\n",
      "Epoch: [1][0/24]\tTime 0.043 (0.043)\tData 0.017 (0.017)\tLoss 0.0776 (0.0776)\tAcc@1 99.219 (99.219)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.633 Acc@5 100.000\n",
      "Epoch: [2][0/24]\tTime 0.185 (0.185)\tData 0.051 (0.051)\tLoss 0.0386 (0.0386)\tAcc@1 99.219 (99.219)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.700 Acc@5 100.000\n",
      "Epoch: [3][0/24]\tTime 0.038 (0.038)\tData 0.017 (0.017)\tLoss 0.0430 (0.0430)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.867 Acc@5 100.000\n",
      "Epoch: [4][0/24]\tTime 0.031 (0.031)\tData 0.015 (0.015)\tLoss 0.0460 (0.0460)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.900 Acc@5 100.000\n",
      "Epoch: [5][0/24]\tTime 0.032 (0.032)\tData 0.015 (0.015)\tLoss 0.0266 (0.0266)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.867 Acc@5 100.000\n",
      "Epoch: [6][0/24]\tTime 0.161 (0.161)\tData 0.068 (0.068)\tLoss 0.0195 (0.0195)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.900 Acc@5 100.000\n",
      "Epoch: [7][0/24]\tTime 0.039 (0.039)\tData 0.025 (0.025)\tLoss 0.0374 (0.0374)\tAcc@1 98.438 (98.438)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.900 Acc@5 100.000\n",
      "Epoch: [8][0/24]\tTime 0.060 (0.060)\tData 0.013 (0.013)\tLoss 0.0192 (0.0192)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.967 Acc@5 100.000\n",
      "Epoch: [9][0/24]\tTime 0.028 (0.028)\tData 0.011 (0.011)\tLoss 0.0165 (0.0165)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.933 Acc@5 100.000\n"
     ]
    }
   ],
   "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": 32,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                               \r"
     ]
    }
   ],
   "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": 33,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                               \r"
     ]
    }
   ],
   "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": 34,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "confuse mode: True\n",
      "split mode: train\n",
      "[181 258  15  65 216  63 240 228 167 156  64 103 124 315 310  60 122 271\n",
      "  66  26 286 158 136 309 164 200 272   7   6 230 254  22 152  21  55 157\n",
      "  12 108  68 144 170 278  59 134 231  92 175 267  74  81 288 259 264  52\n",
      " 133 176  56  90 248  17   1   8 150 206 209 140 217 194   5  33 280  34\n",
      " 212 215 263 137  45 101 266 126 198 238 199 222 120 245 135  73 313 304\n",
      " 116 255  29  97  20  46 213 262 312  89 369 360 532 599 463 551 408 322\n",
      " 553 447 516 415 575 515 555 339 331 511 568 503 519 512 492 343 578 632\n",
      " 375 636 535 514 542 462 489 337 559 364 525 442 526 405 562 368 533 548\n",
      " 372 629 637 334 493 485 450 436 617 321 464 466 536 336 595 397 517 522\n",
      " 345 382 328 355 465 341 616 624 600 509 504 434 546 390 543 357 472 531\n",
      " 399 354 510 384 423 585 347 613 541 572 497 332 353 470 549 370 618 351\n",
      " 520 398]\n",
      "confuse mode: True\n",
      "split mode: forget\n",
      "[181 258  15  65 216  63 240 228 167 156  64 103 124 315 310  60 122 271\n",
      "  66  26 286 158 136 309 164 200 272   7   6 230 254  22 152  21  55 157\n",
      "  12 108  68 144 170 278  59 134 231  92 175 267  74  81 288 259 264  52\n",
      " 133 176  56  90 248  17   1   8 150 206 209 140 217 194   5  33 280  34\n",
      " 212 215 263 137  45 101 266 126 198 238 199 222 120 245 135  73 313 304\n",
      " 116 255  29  97  20  46 213 262 312  89 369 360 532 599 463 551 408 322\n",
      " 553 447 516 415 575 515 555 339 331 511 568 503 519 512 492 343 578 632\n",
      " 375 636 535 514 542 462 489 337 559 364 525 442 526 405 562 368 533 548\n",
      " 372 629 637 334 493 485 450 436 617 321 464 466 536 336 595 397 517 522\n",
      " 345 382 328 355 465 341 616 624 600 509 504 434 546 390 543 357 472 531\n",
      " 399 354 510 384 423 585 347 613 541 572 497 332 353 470 549 370 618 351\n",
      " 520 398]\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": 35,
   "metadata": {
    "scrolled": false
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Original ->\ttest: 9.00%\tForget: 14.00%\tRetain: 1.53%\tIC-test: 0.31\tfgt-test: 55\tIC-retain: 0.10454545454545454\tfgt-retain: 46\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Retrain ->\ttest: 3.60%\tForget: 100.00%\tRetain: 0.00%\tIC-test: 0.02\tfgt-test: 0\tIC-retain: 0.0\tfgt-retain: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                               \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Finetune ->\ttest: 4.40%\tForget: 66.50%\tRetain: 0.00%\tIC-test: 0.075\tfgt-test: 14\tIC-retain: 0.0\tfgt-retain: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                               \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "NegGrad ->\ttest: 5.80%\tForget: 81.50%\tRetain: 0.43%\tIC-test: 0.14\tfgt-test: 6\tIC-retain: 0.02727272727272727\tfgt-retain: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "CF-k ->\ttest: 5.40%\tForget: 37.50%\tRetain: 0.07%\tIC-test: 0.115\tfgt-test: 21\tIC-retain: 0.004545454545454545\tfgt-retain: 2\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "EU-k ->\ttest: 3.00%\tForget: 94.50%\tRetain: 0.00%\tIC-test: 0.025\tfgt-test: 1\tIC-retain: 0.0\tfgt-retain: 0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "                                                \r"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "SCRUB ->\ttest: 8.50%\tForget: 26.00%\tRetain: 1.37%\tIC-test: 0.205\tfgt-test: 41\tIC-retain: 0.04318181818181818\tfgt-retain: 19\n"
     ]
    }
   ],
   "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.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
