{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Imports"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_395006/2162111138.py:25: TqdmWarning: IProgress not found. Please update jupyter and ipywidgets. See https://ipywidgets.readthedocs.io/en/stable/user_install.html\n",
      "  from tqdm.autonotebook import tqdm\n"
     ]
    }
   ],
   "source": [
    "%load_ext autoreload\n",
    "%autoreload 2\n",
    "%matplotlib inline\n",
    "import os\n",
    "os.environ['CUDA_DEVICE_ORDER']='PCI_BUS_ID'\n",
    "os.environ['CUDA_VISIBLE_DEVICES']='0'\n",
    "import variational\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn as sns\n",
    "from matplotlib.ticker import FuncFormatter\n",
    "from itertools import cycle\n",
    "import os\n",
    "import time\n",
    "import math\n",
    "import pandas as pd\n",
    "from collections import OrderedDict\n",
    "from sklearn.linear_model import LogisticRegression\n",
    "from sklearn.metrics import confusion_matrix\n",
    "    \n",
    "import copy\n",
    "import torch.nn as nn\n",
    "from torch.autograd import Variable\n",
    "from typing import List\n",
    "import itertools\n",
    "from tqdm.autonotebook import tqdm\n",
    "from models import *\n",
    "import models\n",
    "from logger import *\n",
    "\n",
    "from thirdparty.repdistiller.helper.util import adjust_learning_rate as sgda_adjust_learning_rate\n",
    "from thirdparty.repdistiller.distiller_zoo import DistillKL, HintLoss, Attention, Similarity, Correlation, VIDLoss, RKDLoss\n",
    "from thirdparty.repdistiller.distiller_zoo import PKT, ABLoss, FactorTransfer, KDSVD, FSP, NSTLoss\n",
    "\n",
    "from thirdparty.repdistiller.helper.loops import train_distill, train_distill_hide, train_distill_linear, train_vanilla, train_negrad, train_bcu, train_bcu_distill, validate\n",
    "from thirdparty.repdistiller.helper.pretrain import init"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Metrics1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "from utils import *\n",
    "def get_metrics(model,dataloader,criterion,samples_correctness=False,use_bn=False,delta_w=None,scrub_act=False):\n",
    "    activations=[]\n",
    "    predictions=[]\n",
    "    if use_bn:\n",
    "        model.train()\n",
    "        dataloader = torch.utils.data.DataLoader(retain_loader.dataset, batch_size=128, shuffle=True)\n",
    "        for i in range(10):\n",
    "            for batch_idx, (data, target) in enumerate(dataloader):\n",
    "                data, target = data.to(args.device), target.to(args.device)            \n",
    "                output = model(data)\n",
    "    dataloader = torch.utils.data.DataLoader(dataloader.dataset, batch_size=1, shuffle=False)\n",
    "    model.eval()\n",
    "    metrics = AverageMeter()\n",
    "    mult = 0.5 if args.lossfn=='mse' else 1\n",
    "    for batch_idx, (data, target) in enumerate(dataloader):\n",
    "        data, target = data.to(args.device), target.to(args.device)            \n",
    "        if args.lossfn=='mse':\n",
    "            target=(2*target-1)\n",
    "            target = target.type(torch.cuda.FloatTensor).unsqueeze(1)\n",
    "        if 'mnist' in args.dataset:\n",
    "            data=data.view(data.shape[0],-1)\n",
    "        output = model(data)\n",
    "        loss = mult*criterion(output, target)\n",
    "        if samples_correctness:\n",
    "            activations.append(torch.nn.functional.softmax(output,dim=1).cpu().detach().numpy().squeeze())\n",
    "            predictions.append(get_error(output,target))\n",
    "        metrics.update(n=data.size(0), loss=loss.item(), error=get_error(output, target))\n",
    "    if samples_correctness:\n",
    "        return metrics.avg,np.stack(activations),np.array(predictions)\n",
    "    else:\n",
    "        return metrics.avg\n",
    "\n",
    "def activations_predictions(model,dataloader,name):\n",
    "    criterion = torch.nn.CrossEntropyLoss()\n",
    "    metrics,activations,predictions=get_metrics(model,dataloader,criterion,True)\n",
    "    print(f\"{name} -> Loss:{np.round(metrics['loss'],3)}, Error:{metrics['error']}\")\n",
    "    log_dict[f\"{name}_loss\"]=metrics['loss']\n",
    "    log_dict[f\"{name}_error\"]=metrics['error']\n",
    "\n",
    "    return activations,predictions\n",
    "\n",
    "def predictions_distance(l1,l2,name):\n",
    "    dist = np.sum(np.abs(l1-l2))\n",
    "    print(f\"Predictions Distance {name} -> {dist}\")\n",
    "    log_dict[f\"{name}_predictions\"]=dist\n",
    "\n",
    "def activations_distance(a1,a2,name):\n",
    "    dist = np.linalg.norm(a1-a2,ord=1,axis=1).mean()\n",
    "    print(f\"Activations Distance {name} -> {dist}\")\n",
    "    log_dict[f\"{name}_activations\"]=dist\n",
    "\n",
    "def interclass_confusion(model, dataloader, class_to_forget, name):\n",
    "    criterion = torch.nn.CrossEntropyLoss()\n",
    "    dataloader = torch.utils.data.DataLoader(dataloader.dataset, batch_size=128, shuffle=False)\n",
    "    model.eval()\n",
    "    reals=[]\n",
    "    predicts=[]\n",
    "    for batch_idx, (data, target) in enumerate(dataloader):\n",
    "        data, target = data.to(args.device), target.to(args.device) \n",
    "        if 'mnist' in args.dataset:\n",
    "            data=data.view(data.shape[0],-1)\n",
    "        output = model(data)\n",
    "        probs = torch.nn.functional.softmax(output, dim=1)\n",
    "        predict = np.argmax(probs.cpu().detach().numpy(),axis=1)\n",
    "        reals = reals + list(target.cpu().detach().numpy())\n",
    "        predicts = predicts + list(predict)\n",
    "    \n",
    "    classes = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]\n",
    "    cm = confusion_matrix(reals, predicts, labels=classes)\n",
    "    counts = 0\n",
    "    for i in range(len(cm)):\n",
    "        if i != class_to_forget[0]:\n",
    "            counts += cm[class_to_forget[0]][i]\n",
    "        if i != class_to_forget[1]:\n",
    "            counts += cm[class_to_forget[1]][i]\n",
    "    \n",
    "    ic_err = counts / (np.sum(cm[class_to_forget[0]]) + np.sum(cm[class_to_forget[1]]))\n",
    "    fgt = cm[class_to_forget[0]][class_to_forget[1]] + cm[class_to_forget[1]][class_to_forget[0]]\n",
    "    #print (cm)\n",
    "    return ic_err, fgt"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Helper and Utils"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "from utils import *\n",
    "def get_metrics(model,dataloader,criterion,samples_correctness=False,use_bn=False,delta_w=None,scrub_act=False):\n",
    "    activations=[]\n",
    "    predictions=[]\n",
    "    if use_bn:\n",
    "        model.train()\n",
    "        dataloader = torch.utils.data.DataLoader(retain_loader.dataset, batch_size=128, shuffle=True)\n",
    "        for i in range(10):\n",
    "            for batch_idx, (data, target) in enumerate(dataloader):\n",
    "                data, target = data.to(args.device), target.to(args.device)            \n",
    "                output = model(data)\n",
    "    dataloader = torch.utils.data.DataLoader(dataloader.dataset, batch_size=1, shuffle=False)\n",
    "    model.eval()\n",
    "    metrics = AverageMeter()\n",
    "    mult = 0.5 if args.lossfn=='mse' else 1\n",
    "    for batch_idx, (data, target) in enumerate(dataloader):\n",
    "        data, target = data.to(args.device), target.to(args.device)            \n",
    "        if args.lossfn=='mse':\n",
    "            target=(2*target-1)\n",
    "            target = target.type(torch.cuda.FloatTensor).unsqueeze(1)\n",
    "        if 'mnist' in args.dataset:\n",
    "            data=data.view(data.shape[0],-1)\n",
    "        output = model(data)\n",
    "        if scrub_act:\n",
    "            G = []\n",
    "            for cls in range(num_classes):\n",
    "                grads = torch.autograd.grad(output[0,cls],model.parameters(),retain_graph=True)\n",
    "                grads = torch.cat([g.view(-1) for g in grads])\n",
    "                G.append(grads)\n",
    "            grads = torch.autograd.grad(output_sf[0,cls],model_scrubf.parameters(),retain_graph=False)\n",
    "            G = torch.stack(G).pow(2)\n",
    "            delta_f = torch.matmul(G,delta_w)\n",
    "            output += delta_f.sqrt()*torch.empty_like(delta_f).normal_()\n",
    "\n",
    "        loss = mult*criterion(output, target)\n",
    "        if samples_correctness:\n",
    "            activations.append(torch.nn.functional.softmax(output,dim=1).cpu().detach().numpy().squeeze())\n",
    "            predictions.append(get_error(output,target))\n",
    "        metrics.update(n=data.size(0), loss=loss.item(), error=get_error(output, target))\n",
    "    if samples_correctness:\n",
    "        return metrics.avg,np.stack(activations),np.array(predictions)\n",
    "    else:\n",
    "        return metrics.avg\n",
    "\n",
    "def l2_penalty(model,model_init,weight_decay):\n",
    "    l2_loss = 0\n",
    "    for (k,p),(k_init,p_init) in zip(model.named_parameters(),model_init.named_parameters()):\n",
    "        if p.requires_grad:\n",
    "            l2_loss += (p-p_init).pow(2).sum()\n",
    "    l2_loss *= (weight_decay/2.)\n",
    "    return l2_loss\n",
    "\n",
    "def run_train_epoch(model: nn.Module, model_init, data_loader: torch.utils.data.DataLoader, \n",
    "                    loss_fn: nn.Module,\n",
    "                    optimizer: torch.optim.SGD, split: str, epoch: int, ignore_index=None,\n",
    "                    negative_gradient=False, negative_multiplier=-1, random_labels=False,\n",
    "                    quiet=False,delta_w=None,scrub_act=False):\n",
    "    model.eval()\n",
    "    metrics = AverageMeter()    \n",
    "    num_labels = data_loader.dataset.targets.max().item() + 1\n",
    "    \n",
    "    with torch.set_grad_enabled(split != 'test'):\n",
    "        for idx, batch in enumerate(tqdm(data_loader, leave=False)):\n",
    "            batch = [tensor.to(next(model.parameters()).device) for tensor in batch]\n",
    "            input, target = batch\n",
    "            output = model(input)\n",
    "            if split=='test' and scrub_act:\n",
    "                G = []\n",
    "                for cls in range(num_classes):\n",
    "                    grads = torch.autograd.grad(output[0,cls],model.parameters(),retain_graph=True)\n",
    "                    grads = torch.cat([g.view(-1) for g in grads])\n",
    "                    G.append(grads)\n",
    "                grads = torch.autograd.grad(output_sf[0,cls],model_scrubf.parameters(),retain_graph=False)\n",
    "                G = torch.stack(G).pow(2)\n",
    "                delta_f = torch.matmul(G,delta_w)\n",
    "                output += delta_f.sqrt()*torch.empty_like(delta_f).normal_()\n",
    "            loss = loss_fn(output, target) + l2_penalty(model,model_init,args.weight_decay)\n",
    "            metrics.update(n=input.size(0), loss=loss_fn(output,target).item(), error=get_error(output, target))\n",
    "            \n",
    "            if split != 'test':\n",
    "                model.zero_grad()\n",
    "                loss.backward()\n",
    "                optimizer.step()\n",
    "    if not quiet:\n",
    "        log_metrics(split, metrics, epoch)\n",
    "    return metrics.avg\n",
    "\n",
    "def run_neggrad_epoch(model: nn.Module, model_init, data_loader: torch.utils.data.DataLoader, \n",
    "                    forget_loader: torch.utils.data.DataLoader,\n",
    "                    alpha: float,\n",
    "                    loss_fn: nn.Module,\n",
    "                    optimizer: torch.optim.SGD, split: str, epoch: int, ignore_index=None,\n",
    "                    quiet=False):\n",
    "    model.eval()\n",
    "    metrics = AverageMeter()    \n",
    "    num_labels = data_loader.dataset.targets.max().item() + 1\n",
    "    \n",
    "    with torch.set_grad_enabled(split != 'test'):\n",
    "        for idx, (batch_retain,batch_forget) in enumerate(tqdm(zip(data_loader,cycle(forget_loader)), leave=False)):\n",
    "            batch_retain = [tensor.to(next(model.parameters()).device) for tensor in batch_retain]\n",
    "            batch_forget = [tensor.to(next(model.parameters()).device) for tensor in batch_forget]\n",
    "            input_r, target_r = batch_retain\n",
    "            input_f, target_f = batch_forget\n",
    "            output_r = model(input_r)\n",
    "            output_f = model(input_f)\n",
    "            loss = alpha*(loss_fn(output_r, target_r) + l2_penalty(model,model_init,args.weight_decay)) - (1-alpha)*loss_fn(output_f, target_f)\n",
    "            metrics.update(n=input_r.size(0), loss=loss_fn(output_r,target_r).item(), error=get_error(output_r, target_r))\n",
    "            if split != 'test':\n",
    "                model.zero_grad()\n",
    "                loss.backward()\n",
    "                optimizer.step()\n",
    "    if not quiet:\n",
    "        log_metrics(split, metrics, epoch)\n",
    "    return metrics.avg"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 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": 7,
   "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": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "def opt_solver(probs, target_distb, num_iter=10, th=0.1, num_newton=30):\n",
    "    entropy = (-1 * probs * torch.log(probs + 1e-6)).sum(1)\n",
    "    weights = (1 / entropy)\n",
    "    N, K = probs.size(0), probs.size(1)\n",
    "\n",
    "    A, w, lam, nu, r, c = probs.numpy(), weights.numpy(), np.ones(N), np.ones(K), np.ones(N), target_distb.numpy()\n",
    "    A_e = A / math.e\n",
    "    X = np.exp(-1 * lam / w)\n",
    "    Y = np.exp(-1 * nu.reshape(1, -1) / w.reshape(-1, 1))\n",
    "    prev_Y = np.zeros(K)\n",
    "    X_t, Y_t = X, Y\n",
    "\n",
    "    for n in range(num_iter):\n",
    "        # Normalization\n",
    "        denom = np.sum(A_e * Y_t, 1)\n",
    "        X_t = r / denom\n",
    "\n",
    "        # Newton method\n",
    "        Y_t = np.zeros(K)\n",
    "        for i in range(K):\n",
    "            Y_t[i] = optimize.newton(f, prev_Y[i], maxiter=num_newton, args=(A_e[:, i], X_t, w, c[i]), tol=th)\n",
    "        prev_Y = Y_t\n",
    "        Y_t = np.exp(-1 * Y_t.reshape(1, -1) / w.reshape(-1, 1))\n",
    "\n",
    "    denom = np.sum(A_e * Y_t, 1)\n",
    "    X_t = r / denom\n",
    "    M = torch.Tensor(A_e * X_t.reshape(-1, 1) * Y_t)\n",
    "\n",
    "    return M"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Pre-training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checkpoint name: cifar100_resnet_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1\n",
      "[Logging in cifar100_resnet_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1_training]\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "confuse mode: False\n",
      "split mode: None\n",
      "Number of Classes: 100\n",
      "[0] train metrics:{\"loss\": 4.110220985412598, \"error\": 0.9239}\n",
      "Learning Rate : 0.1\n",
      "[0] dry_run metrics:{\"loss\": 3.7692216533660887, \"error\": 0.87745}\n",
      "Learning Rate : 0.1\n",
      "[0] test metrics:{\"loss\": 3.779144535446167, \"error\": 0.8744}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 29.82 sec\n",
      "[1] train metrics:{\"loss\": 3.6166512355804445, \"error\": 0.84515}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 12.34 sec\n",
      "[2] train metrics:{\"loss\": 3.262185221862793, \"error\": 0.767475}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 13.29 sec\n",
      "[3] train metrics:{\"loss\": 2.9241571712493895, \"error\": 0.6889}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 15.36 sec\n",
      "[4] train metrics:{\"loss\": 2.6573989303588865, \"error\": 0.61595}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 15.34 sec\n",
      "[5] train metrics:{\"loss\": 2.456750422286987, \"error\": 0.55295}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.95 sec\n",
      "[6] train metrics:{\"loss\": 2.316490512084961, \"error\": 0.494525}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.49 sec\n",
      "[7] train metrics:{\"loss\": 2.2176277866363527, \"error\": 0.44985}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.12 sec\n",
      "[8] train metrics:{\"loss\": 2.1540924285888674, \"error\": 0.404375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.29 sec\n",
      "[9] train metrics:{\"loss\": 2.1174500984191895, \"error\": 0.37015}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.32 sec\n",
      "[10] train metrics:{\"loss\": 2.0962327156066896, \"error\": 0.33205}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 6.9 sec\n",
      "[11] train metrics:{\"loss\": 2.089509313583374, \"error\": 0.2955}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.46 sec\n",
      "[12] train metrics:{\"loss\": 2.1187580627441407, \"error\": 0.27275}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.03 sec\n",
      "[13] train metrics:{\"loss\": 2.1520710746765137, \"error\": 0.2503}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.25 sec\n",
      "[14] train metrics:{\"loss\": 2.1659040351867676, \"error\": 0.222975}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 6.73 sec\n",
      "[15] train metrics:{\"loss\": 2.180283085632324, \"error\": 0.202875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 8.46 sec\n",
      "[16] train metrics:{\"loss\": 2.237203318405151, \"error\": 0.195875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.1 sec\n",
      "[17] train metrics:{\"loss\": 2.256527083206177, \"error\": 0.18455}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.25 sec\n",
      "[18] train metrics:{\"loss\": 2.3006882884979247, \"error\": 0.180475}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.05 sec\n",
      "[19] train metrics:{\"loss\": 2.2771842372894286, \"error\": 0.164375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.9 sec\n",
      "[20] train metrics:{\"loss\": 2.332470198059082, \"error\": 0.1652}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.58 sec\n",
      "[21] train metrics:{\"loss\": 2.330265349960327, \"error\": 0.1593}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.13 sec\n",
      "[22] train metrics:{\"loss\": 2.4015751735687254, \"error\": 0.164125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 8.0 sec\n",
      "[23] train metrics:{\"loss\": 2.3445411685943602, \"error\": 0.1472}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.58 sec\n",
      "[24] train metrics:{\"loss\": 2.3646815967559816, \"error\": 0.151725}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.81 sec\n",
      "[25] train metrics:{\"loss\": 2.3665282775878906, \"error\": 0.14435}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.9 sec\n",
      "[26] train metrics:{\"loss\": 2.345229354476929, \"error\": 0.14115}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 8.1 sec\n",
      "[27] train metrics:{\"loss\": 2.3579494285583498, \"error\": 0.142225}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.35 sec\n",
      "[28] train metrics:{\"loss\": 2.3899746318817137, \"error\": 0.1456}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.35 sec\n",
      "[29] train metrics:{\"loss\": 2.4067670585632324, \"error\": 0.145975}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 7.47 sec\n",
      "[30] train metrics:{\"loss\": 2.4091349899291994, \"error\": 0.14255}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 6.89 sec\n",
      "Pure training time: 270.6700000000001 sec\n"
     ]
    }
   ],
   "source": [
    "%run main.py --dataset cifar100 --dataroot=data/cifar100/ --model resnet --filters 1.0 --lr 0.1 --lossfn ce --num-classes 100"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Original"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checkpoint name: small_cifar5_resnet_1_0_forget_None_lr_0_001_bs_128_ls_ce_wd_0_1_seed_3\n",
      "[Logging in small_cifar5_resnet_1_0_forget_None_lr_0_001_bs_128_ls_ce_wd_0_1_seed_3_training]\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "confuse mode: False\n",
      "split mode: None\n",
      "Number of Classes: 5\n",
      "[0] train metrics:{\"loss\": 1.5698002729415894, \"error\": 0.7219999980926514}\n",
      "Learning Rate : 0.001\n",
      "[0] test metrics:{\"loss\": 1.547294379234314, \"error\": 0.6879999997615814}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.28 sec\n",
      "[1] train metrics:{\"loss\": 1.5123306999206543, \"error\": 0.6480000011920929}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.11 sec\n",
      "[2] train metrics:{\"loss\": 1.4296097145080566, \"error\": 0.5900000016689301}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.1 sec\n",
      "[3] train metrics:{\"loss\": 1.3396971225738525, \"error\": 0.5300000021457673}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.11 sec\n",
      "[4] train metrics:{\"loss\": 1.2585979499816895, \"error\": 0.46399999380111695}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.1 sec\n",
      "[5] train metrics:{\"loss\": 1.1852641887664794, \"error\": 0.4280000009536743}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.18 sec\n",
      "[6] train metrics:{\"loss\": 1.1190925750732421, \"error\": 0.3940000033378601}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.09 sec\n",
      "[7] train metrics:{\"loss\": 1.06109437084198, \"error\": 0.3660000014305115}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.09 sec\n",
      "[8] train metrics:{\"loss\": 1.009152063846588, \"error\": 0.3320000023841858}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.09 sec\n",
      "[9] train metrics:{\"loss\": 0.9606991844177246, \"error\": 0.3079999995231628}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.09 sec\n",
      "[10] train metrics:{\"loss\": 0.9174478373527527, \"error\": 0.28000000429153443}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.18 sec\n",
      "[11] train metrics:{\"loss\": 0.8785706114768982, \"error\": 0.2600000071525574}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.12 sec\n",
      "[12] train metrics:{\"loss\": 0.8426756324768067, \"error\": 0.2499999952316284}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.09 sec\n",
      "[13] train metrics:{\"loss\": 0.8098652992248535, \"error\": 0.23999999809265138}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.09 sec\n",
      "[14] train metrics:{\"loss\": 0.7795161581039429, \"error\": 0.23599999713897704}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.09 sec\n",
      "[15] train metrics:{\"loss\": 0.7505445356369018, \"error\": 0.23200000715255736}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.19 sec\n",
      "[16] train metrics:{\"loss\": 0.7248035516738892, \"error\": 0.2240000047683716}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.11 sec\n",
      "[17] train metrics:{\"loss\": 0.7012830009460449, \"error\": 0.216}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.09 sec\n",
      "[18] train metrics:{\"loss\": 0.678739227771759, \"error\": 0.21200000762939453}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.08 sec\n",
      "[19] train metrics:{\"loss\": 0.6577521595954895, \"error\": 0.20200000047683717}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.09 sec\n",
      "[20] train metrics:{\"loss\": 0.637322470664978, \"error\": 0.1940000023841858}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.17 sec\n",
      "[21] train metrics:{\"loss\": 0.6188072509765625, \"error\": 0.18999999570846557}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.1 sec\n",
      "[22] train metrics:{\"loss\": 0.5998594903945923, \"error\": 0.18000000715255737}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.1 sec\n",
      "[23] train metrics:{\"loss\": 0.5824573211669922, \"error\": 0.17200000715255737}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.09 sec\n",
      "[24] train metrics:{\"loss\": 0.5663832321166992, \"error\": 0.1599999942779541}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.09 sec\n",
      "[25] train metrics:{\"loss\": 0.5506122980117798, \"error\": 0.15400000286102294}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.14 sec\n",
      "[26] train metrics:{\"loss\": 0.5353427159786225, \"error\": 0.1479999966621399}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.09 sec\n",
      "[27] train metrics:{\"loss\": 0.5209583477973938, \"error\": 0.14400000095367432}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.09 sec\n",
      "[28] train metrics:{\"loss\": 0.5072608149051666, \"error\": 0.1380000033378601}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.08 sec\n",
      "[29] train metrics:{\"loss\": 0.4942183749675751, \"error\": 0.13199999904632567}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.1 sec\n",
      "[30] train metrics:{\"loss\": 0.4793111574649811, \"error\": 0.1279999942779541}\n",
      "Learning Rate : 0.001\n",
      "Epoch Time: 0.16 sec\n",
      "Pure training time: 2.98 sec\n"
     ]
    }
   ],
   "source": [
    "%run main.py --dataset small_cifar5 --model resnet --dataroot=data/cifar10/ --filters 1.0 --lr 0.001 \\\n",
    "--resume checkpoints/cifar100_resnet_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1_30.pt --disable-bn \\\n",
    "--weight-decay 0.1 --batch-size 128 --epochs 31 --seed 3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checkpoint name: cifar10_resnet_1_0_forget_None_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1\n",
      "[Logging in cifar10_resnet_1_0_forget_None_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1_training]\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "confuse mode: True\n",
      "split mode: train\n",
      "[22462  6646 36135 23850  1434  5306 18141  4049 11418 10150 21959  2340\n",
      " 28237 23045 33372 17719 28726 24235 36477 30535 34830 13742  9505  9925\n",
      "  8968  5340 32206 23111 36876 37753 18138   346  9984 33198 10884 17838\n",
      "  5290 34800  8839 26496 22062 19691 25490 32745 29623 37043 38544 16174\n",
      " 26055 17790 13062  3564 33467 19387 12727 10616  5925 38573 20823 25972\n",
      "  2232 39091 16018 31897  6216 31382  9265  4834 15220 13636 20376 17239\n",
      "  3784 30964 11139 32269 25536 37777  9464 31770 12377 14878 22419 35384\n",
      " 33593 13768  8813 14913  1918  3157 22036 26124 26696 32374   359 18794\n",
      " 37201 12935 39835  6800 34930 21153  1332   444 26453 22013  2780 28519\n",
      " 32137 33713  9783 18691  5872  5426 26608  5517 26355 34576 15284  9727\n",
      " 11908 16177  5566 33238 31572 31990  6716 36313 26521  8618 25781 10358\n",
      "  9601   341  6301 36220 15826  3522  4898 31528 28091 12132 15300  5732\n",
      " 10974 23654 30805 26574  5750  7350 19049 24064  7229  8594 31711 15489\n",
      " 21523 27315  8146 28736 36469 37611 32999 39736 27867 33821 34312  3036\n",
      " 11154 19416 37108 22615 36649 11823  7061 36430 18557 37081  8899 29319\n",
      "  9990 26523 33672  7326 39648 21642   132  4050  6683  5979 20121 36524\n",
      " 22540 33421 27229 13730 29155 25346  2670   732]\n",
      "Number of Classes: 10\n",
      "[0] train metrics:{\"loss\": 0.8994579604148865, \"error\": 0.31465}\n",
      "Learning Rate : 0.01\n",
      "[0] test metrics:{\"loss\": 0.6736513387680054, \"error\": 0.2306}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 8.41 sec\n",
      "[1] train metrics:{\"loss\": 0.5806631664276123, \"error\": 0.19635}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.38 sec\n",
      "[2] train metrics:{\"loss\": 0.44068201198577883, \"error\": 0.14615}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.46 sec\n",
      "[3] train metrics:{\"loss\": 0.3294310111761093, \"error\": 0.105}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.64 sec\n",
      "[4] train metrics:{\"loss\": 0.23757381939888, \"error\": 0.07145}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.29 sec\n",
      "[5] train metrics:{\"loss\": 0.15486119554042815, \"error\": 0.0413}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.29 sec\n",
      "[6] train metrics:{\"loss\": 0.09451916145086288, \"error\": 0.018575}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.07 sec\n",
      "[7] train metrics:{\"loss\": 0.0700134416103363, \"error\": 0.009625}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.07 sec\n",
      "[8] train metrics:{\"loss\": 0.05248799537420273, \"error\": 0.003975}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.22 sec\n",
      "[9] train metrics:{\"loss\": 0.03984407194852829, \"error\": 0.00035}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.1 sec\n",
      "[10] train metrics:{\"loss\": 0.03694459772706032, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.26 sec\n",
      "[11] train metrics:{\"loss\": 0.03580245625972748, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.08 sec\n",
      "[12] train metrics:{\"loss\": 0.03495487071871758, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.14 sec\n",
      "[13] train metrics:{\"loss\": 0.03420322653055191, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.42 sec\n",
      "[14] train metrics:{\"loss\": 0.03351431832909584, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.37 sec\n",
      "[15] train metrics:{\"loss\": 0.0328611094892025, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.71 sec\n",
      "[16] train metrics:{\"loss\": 0.032256303888559344, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.04 sec\n",
      "[17] train metrics:{\"loss\": 0.031692172318696976, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.93 sec\n",
      "[18] train metrics:{\"loss\": 0.031148864424228668, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.2 sec\n",
      "[19] train metrics:{\"loss\": 0.030647821056842803, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.93 sec\n",
      "[20] train metrics:{\"loss\": 0.030150964498519898, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.07 sec\n",
      "[21] train metrics:{\"loss\": 0.02969496927857399, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.03 sec\n",
      "[22] train metrics:{\"loss\": 0.02928686562180519, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.09 sec\n",
      "[23] train metrics:{\"loss\": 0.028866908860206604, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.93 sec\n",
      "[24] train metrics:{\"loss\": 0.028454023391008377, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.45 sec\n",
      "[25] train metrics:{\"loss\": 0.028074368637800218, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.99 sec\n",
      "Pure training time: 186.99 sec\n"
     ]
    }
   ],
   "source": [
    "%run main.py --dataset cifar10 --model resnet --dataroot=data/cifar10/ --filters 1.0 --lr 0.01 \\\n",
    "--resume checkpoints/cifar100_resnet_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1_30.pt --disable-bn \\\n",
    "--weight-decay 0.0005 --batch-size 128 --epochs 26 --seed 1 \\\n",
    "--split train --confuse-mode --forget-class 0,1 --num-to-forget 200"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Retrain"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": true
   },
   "outputs": [],
   "source": [
    "#%run main.py --dataset small_lacuna6 --model allcnn --dataroot=data/lacuna10/ --filters 1.0 --lr 0.001 \\\n",
    "#--resume checkpoints/lacuna100_allcnn_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1_30.pt --disable-bn \\\n",
    "#--weight-decay 0.1 --batch-size 128 --epochs 31 \\\n",
    "#--forget-class 0,1,2,3,4,5 --num-to-forget 300 --seed 3"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checkpoint name: cifar10_resnet_1_0_forget_[0, 1]_num_200_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1\n",
      "[Logging in cifar10_resnet_1_0_forget_[0, 1]_num_200_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1_training]\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "confuse mode: True\n",
      "split mode: forget\n",
      "[22462  6646 36135 23850  1434  5306 18141  4049 11418 10150 21959  2340\n",
      " 28237 23045 33372 17719 28726 24235 36477 30535 34830 13742  9505  9925\n",
      "  8968  5340 32206 23111 36876 37753 18138   346  9984 33198 10884 17838\n",
      "  5290 34800  8839 26496 22062 19691 25490 32745 29623 37043 38544 16174\n",
      " 26055 17790 13062  3564 33467 19387 12727 10616  5925 38573 20823 25972\n",
      "  2232 39091 16018 31897  6216 31382  9265  4834 15220 13636 20376 17239\n",
      "  3784 30964 11139 32269 25536 37777  9464 31770 12377 14878 22419 35384\n",
      " 33593 13768  8813 14913  1918  3157 22036 26124 26696 32374   359 18794\n",
      " 37201 12935 39835  6800 34930 21153  1332   444 26453 22013  2780 28519\n",
      " 32137 33713  9783 18691  5872  5426 26608  5517 26355 34576 15284  9727\n",
      " 11908 16177  5566 33238 31572 31990  6716 36313 26521  8618 25781 10358\n",
      "  9601   341  6301 36220 15826  3522  4898 31528 28091 12132 15300  5732\n",
      " 10974 23654 30805 26574  5750  7350 19049 24064  7229  8594 31711 15489\n",
      " 21523 27315  8146 28736 36469 37611 32999 39736 27867 33821 34312  3036\n",
      " 11154 19416 37108 22615 36649 11823  7061 36430 18557 37081  8899 29319\n",
      "  9990 26523 33672  7326 39648 21642   132  4050  6683  5979 20121 36524\n",
      " 22540 33421 27229 13730 29155 25346  2670   732]\n",
      "Number of Classes: 10\n",
      "[0] train metrics:{\"loss\": 0.8776707893371583, \"error\": 0.310325}\n",
      "Learning Rate : 0.01\n",
      "[0] test metrics:{\"loss\": 0.6726855438232422, \"error\": 0.2333}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 8.11 sec\n",
      "[1] train metrics:{\"loss\": 0.5546332703590393, \"error\": 0.19135}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.18 sec\n",
      "[2] train metrics:{\"loss\": 0.411030176115036, \"error\": 0.1398}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.92 sec\n",
      "[3] train metrics:{\"loss\": 0.30049551043510436, \"error\": 0.0975}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.51 sec\n",
      "[4] train metrics:{\"loss\": 0.21082578921318054, \"error\": 0.065275}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 6.96 sec\n",
      "[5] train metrics:{\"loss\": 0.13203115015029906, \"error\": 0.035125}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.02 sec\n",
      "[6] train metrics:{\"loss\": 0.08035368938446046, \"error\": 0.015675}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.4 sec\n",
      "[7] train metrics:{\"loss\": 0.05315162697434425, \"error\": 0.004475}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.51 sec\n",
      "[8] train metrics:{\"loss\": 0.041563900208473205, \"error\": 0.001025}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.22 sec\n",
      "[9] train metrics:{\"loss\": 0.03547775356173515, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.24 sec\n",
      "[10] train metrics:{\"loss\": 0.0340896775662899, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.24 sec\n",
      "[11] train metrics:{\"loss\": 0.03323477267026901, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.55 sec\n",
      "[12] train metrics:{\"loss\": 0.03250679121613503, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.08 sec\n",
      "[13] train metrics:{\"loss\": 0.03184912375807762, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.27 sec\n",
      "[14] train metrics:{\"loss\": 0.031240232867002487, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.44 sec\n",
      "[15] train metrics:{\"loss\": 0.030659463116526602, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.46 sec\n",
      "[16] train metrics:{\"loss\": 0.030124190509319304, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.33 sec\n",
      "[17] train metrics:{\"loss\": 0.02962413500249386, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.38 sec\n",
      "[18] train metrics:{\"loss\": 0.029126183646917343, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.36 sec\n",
      "[19] train metrics:{\"loss\": 0.028692543506622314, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.54 sec\n",
      "[20] train metrics:{\"loss\": 0.028240641552209855, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.35 sec\n",
      "[21] train metrics:{\"loss\": 0.02783956998884678, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.38 sec\n",
      "[22] train metrics:{\"loss\": 0.02746837635934353, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.38 sec\n",
      "[23] train metrics:{\"loss\": 0.027107982075214386, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.32 sec\n",
      "[24] train metrics:{\"loss\": 0.02672754344344139, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.3 sec\n",
      "[25] train metrics:{\"loss\": 0.026383602434396743, \"error\": 0.0}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 7.26 sec\n",
      "Pure training time: 189.24999999999997 sec\n"
     ]
    }
   ],
   "source": [
    "%run main.py --dataset cifar10 --model resnet --dataroot=data/cifar10/ --filters 1.0 --lr 0.01 \\\n",
    "--resume checkpoints/cifar100_resnet_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1_30.pt --disable-bn \\\n",
    "--weight-decay 0.0005 --batch-size 128 --epochs 26 --seed 1 \\\n",
    "--split forget --confuse-mode --forget-class 0,1 --num-to-forget 200"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Logs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 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: 10.362852075030466\n",
      "Normalized Distance: 0.09396269883651323\n"
     ]
    }
   ],
   "source": [
    "log_dict['dist_Original_Original_init']= distance(model_init,model)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Data Loader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "args.retain_bs = 32\n",
    "args.forget_bs = 32"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "confuse mode: True\n",
      "split mode: train\n",
      "[22462  6646 36135 23850  1434  5306 18141  4049 11418 10150 21959  2340\n",
      " 28237 23045 33372 17719 28726 24235 36477 30535 34830 13742  9505  9925\n",
      "  8968  5340 32206 23111 36876 37753 18138   346  9984 33198 10884 17838\n",
      "  5290 34800  8839 26496 22062 19691 25490 32745 29623 37043 38544 16174\n",
      " 26055 17790 13062  3564 33467 19387 12727 10616  5925 38573 20823 25972\n",
      "  2232 39091 16018 31897  6216 31382  9265  4834 15220 13636 20376 17239\n",
      "  3784 30964 11139 32269 25536 37777  9464 31770 12377 14878 22419 35384\n",
      " 33593 13768  8813 14913  1918  3157 22036 26124 26696 32374   359 18794\n",
      " 37201 12935 39835  6800 34930 21153  1332   444 26453 22013  2780 28519\n",
      " 32137 33713  9783 18691  5872  5426 26608  5517 26355 34576 15284  9727\n",
      " 11908 16177  5566 33238 31572 31990  6716 36313 26521  8618 25781 10358\n",
      "  9601   341  6301 36220 15826  3522  4898 31528 28091 12132 15300  5732\n",
      " 10974 23654 30805 26574  5750  7350 19049 24064  7229  8594 31711 15489\n",
      " 21523 27315  8146 28736 36469 37611 32999 39736 27867 33821 34312  3036\n",
      " 11154 19416 37108 22615 36649 11823  7061 36430 18557 37081  8899 29319\n",
      "  9990 26523 33672  7326 39648 21642   132  4050  6683  5979 20121 36524\n",
      " 22540 33421 27229 13730 29155 25346  2670   732]\n",
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "confuse mode: True\n",
      "split mode: forget\n",
      "[22462  6646 36135 23850  1434  5306 18141  4049 11418 10150 21959  2340\n",
      " 28237 23045 33372 17719 28726 24235 36477 30535 34830 13742  9505  9925\n",
      "  8968  5340 32206 23111 36876 37753 18138   346  9984 33198 10884 17838\n",
      "  5290 34800  8839 26496 22062 19691 25490 32745 29623 37043 38544 16174\n",
      " 26055 17790 13062  3564 33467 19387 12727 10616  5925 38573 20823 25972\n",
      "  2232 39091 16018 31897  6216 31382  9265  4834 15220 13636 20376 17239\n",
      "  3784 30964 11139 32269 25536 37777  9464 31770 12377 14878 22419 35384\n",
      " 33593 13768  8813 14913  1918  3157 22036 26124 26696 32374   359 18794\n",
      " 37201 12935 39835  6800 34930 21153  1332   444 26453 22013  2780 28519\n",
      " 32137 33713  9783 18691  5872  5426 26608  5517 26355 34576 15284  9727\n",
      " 11908 16177  5566 33238 31572 31990  6716 36313 26521  8618 25781 10358\n",
      "  9601   341  6301 36220 15826  3522  4898 31528 28091 12132 15300  5732\n",
      " 10974 23654 30805 26574  5750  7350 19049 24064  7229  8594 31711 15489\n",
      " 21523 27315  8146 28736 36469 37611 32999 39736 27867 33821 34312  3036\n",
      " 11154 19416 37108 22615 36649 11823  7061 36430 18557 37081  8899 29319\n",
      "  9990 26523 33672  7326 39648 21642   132  4050  6683  5979 20121 36524\n",
      " 22540 33421 27229 13730 29155 25346  2670   732]\n"
     ]
    }
   ],
   "source": [
    "train_loader_full, valid_loader_full, test_loader_full = datasets.get_loaders(dataset, split=\"train\",confuse_mode=True,class_to_replace=class_to_forget, num_indexes_to_replace=num_to_forget, batch_size=args.batch_size, seed=seed, root=args.dataroot, augment=False, shuffle=True)\n",
    "marked_loader, _, _ = datasets.get_loaders(dataset, split=\"forget\", confuse_mode=True,class_to_replace=class_to_forget, num_indexes_to_replace=num_to_forget, only_mark=True, batch_size=1, seed=seed, root=args.dataroot, augment=False, shuffle=True)\n",
    "\n",
    "def replace_loader_dataset(data_loader, dataset, batch_size=args.batch_size, seed=1, shuffle=True):\n",
    "    manual_seed(seed)\n",
    "    loader_args = {'num_workers': 0, 'pin_memory': False}\n",
    "    def _init_fn(worker_id):\n",
    "        np.random.seed(int(seed))\n",
    "    return torch.utils.data.DataLoader(dataset, batch_size=batch_size,num_workers=0,pin_memory=True,shuffle=shuffle)\n",
    "    \n",
    "forget_dataset = copy.deepcopy(marked_loader.dataset)\n",
    "marked = forget_dataset.targets < 0\n",
    "forget_dataset.data = forget_dataset.data[marked]\n",
    "forget_dataset.targets = - forget_dataset.targets[marked] - 1\n",
    "forget_loader = replace_loader_dataset(train_loader_full, forget_dataset, batch_size=args.forget_bs, seed=seed, shuffle=True)\n",
    "\n",
    "retain_dataset = copy.deepcopy(marked_loader.dataset)\n",
    "marked = retain_dataset.targets >= 0\n",
    "retain_dataset.data = retain_dataset.data[marked]\n",
    "retain_dataset.targets = retain_dataset.targets[marked]\n",
    "retain_loader = replace_loader_dataset(train_loader_full, retain_dataset, batch_size=args.retain_bs, seed=seed, shuffle=True)\n",
    "\n",
    "assert(len(forget_dataset) + len(retain_dataset) == len(train_loader_full.dataset))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "200\n",
      "39800\n",
      "10000\n",
      "40000\n",
      "{6: 4000, 9: 4000, 4: 4000, 1: 4000, 2: 4000, 8: 4000, 3: 4000, 7: 4000, 5: 4000, 0: 4000}\n"
     ]
    }
   ],
   "source": [
    "print (len(forget_loader.dataset))\n",
    "print (len(retain_loader.dataset))\n",
    "print (len(test_loader_full.dataset))\n",
    "print (len(train_loader_full.dataset))\n",
    "from collections import Counter\n",
    "print(dict(Counter(train_loader_full.dataset.targets)))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# SGDA Forgetting"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 68,
   "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.005\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": 69,
   "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": 70,
   "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": 71,
   "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": 72,
   "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": 73,
   "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 84.837 \n",
      "maximize loss: -41.94\t minimize loss: 3.42\t train_acc: 84.83668518066406\n",
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 86.387 \n",
      "maximize loss: -31.25\t minimize loss: 2.55\t train_acc: 86.38693237304688\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 88.606 \n",
      "maximize loss: 0.00\t minimize loss: 1.04\t train_acc: 88.60552978515625\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 88.686 \n",
      "maximize loss: 0.00\t minimize loss: 1.03\t train_acc: 88.68592834472656\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 88.613 \n",
      "maximize loss: 0.00\t minimize loss: 1.04\t train_acc: 88.61306762695312\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 88.510 \n",
      "maximize loss: 0.00\t minimize loss: 1.04\t train_acc: 88.51004791259766\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 88.485 \n",
      "maximize loss: 0.00\t minimize loss: 1.04\t train_acc: 88.48492431640625\n",
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 87.696 \n",
      "maximize loss: 0.00\t minimize loss: 0.54\t train_acc: 87.69597625732422\n",
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 87.485 \n",
      "maximize loss: 0.00\t minimize loss: 0.52\t train_acc: 87.48492431640625\n",
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/zihao/anaconda3/envs/zihao/lib/python3.10/site-packages/torch/nn/_reduction.py:42: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 87.485 \n",
      "maximize loss: 0.00\t minimize loss: 0.52\t train_acc: 87.48492431640625\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": 74,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_395006/2812292199.py:16: UserWarning: FigureCanvasAgg is non-interactive, and thus cannot be shown\n",
      "  fig.show()\n"
     ]
    },
    {
     "data": {
      "image/png": "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jR555JF87T169NCyZctceo7s7GwdOXJE1apVy9d+9OhRJSQkKCsrS02bNtWTTz6Z2yNRlLS0NKWdMtMlNTX1NN4JAACuWbLEtdDQurV00UX5hxqx+o/n1aghXXWVuUjS8ePSqlV5QWLZMik1VZo/31wkEwCaN88LEu3b559wXharCqWnn14QOHr09F8jONiEgJwgUPB6wbaC82dcHa7TqVOpfgXlQp8+5m+pvCxF61XBITk5WVlZWYopsHxDTEyM9u7d69JzvPjiizp27Jiuv/763LYGDRpo2rRpOv/885Wamqrx48erffv22rBhg+rXr1/k84wbN06jR48u/ZsBAMCJLVukF15w7djBgz2zugpOT1iY1LmzuUjmZPjXX/MPb9q504SLVaukl182x9Wvb0JEaKg0aVLhE+ddu8wwlpzhKpmZZl8KV4YF7d9vNsg7XYGBJQeAgvdVrnxmczv8/U04uvZa8zxFDdd55RXvPYl2F39/z0yAdgevGqq0e/du1a5dW8uWLVPbtm1z28eOHav33ntPvxc1sPAUs2bN0p133qk5c+bo4osvLva47OxsNW/eXJ06ddKrr75a5DFF9TjEx8czVAkAcEbS08367W++aYZguMpTq6vA/bZvzx8kfv216G/YixIQIEVESIcOuf6YHP7+UlSU6z0CkZH2TPL2luE6FVm5HKoUFRUlf3//Qr0LSUlJhXohCpo9e7YGDBigDz/8sMTQIEl+fn5q2bKltmzZUuwxwcHBCg4Odr14AABK8Pff0uTJ0tSpeasfORxSz57SypXSwYMlD9coZnFBlAN16kj9+pmLZELA8uXS+++bdfpLkplp/jYk87dQvbrrPQJVq5aP4WzlbbhOReZVwSEoKEiJiYmaP3++rr766tz2+fPn66qcwYRFmDVrlu644w7NmjVLl112mdPXsSxL69ev1/nnn++WugEAKEpGhvTFF9Ibb+SNd5fMidGAAdKdd0oJCXmrq1Tk4RoVSdWq0qWXSikpzoODJD33nHTrrSY0+OrfQXkarlOReVVwkKRhw4apf//+atGihdq2bavJkydr+/btGjhwoCRp+PDh2rVrl6b/u2vGrFmzdMstt2j8+PFq06ZNbm9FaGioIiMjJUmjR49WmzZtVL9+faWmpurVV1/V+vXr9frrr9vzJgEAPu2ff6S33jJr1J/aid6jhzRw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D49xcdL48WY+y2OPmaFJkhQUJA0caHodYmI8/z4AACjvXD3PDfBgTYBL9uxx7bicoUoony680ASE7783Kx+tXWsCwQsvmNWYCtq1y0xuzuHnZ+YuPPGEmQANAADKFpOj4XVq1nTvcfBuF11klkh9/30TAIoKDVL++S7XXitt3Ci98w6hAQAATyE4wOt07FjykBOHQ4qPN8fBN/j5mZWzJk927fhBg8y+DAAAwHMIDvA6O3eaHaCLkrMG/yuvsLymL3J1+Jmrw9kAAID7EBzgVfbtk7p3N0t3xsVJtWrlvz8uzuz+26ePPfWhbDFMDQAA78XkaHiNlBSpZ09pyxYzbv3HH82mbiUtyQnf0rGjCYe7dhW9u7TDYe5nmBoAAJ5HcIBXOH5cuuIKaf16KTpamj9fql3b3Neli52VwZP8/c2Sq9dem7fZXw6GqQEAYC+vHKo0ceJE1atXTyEhIUpMTNSSJUuKPfaTTz5R9+7dVaNGDUVERKht27b6Nmdh+FN8/PHHatSokYKDg9WoUSN9+umnZfkWcBoyMqTrrzc9C5GRZl3/+vXtrgp26dPHDEfLCY45GKYGAIC9vC44zJ49W0OGDNGIESO0bt06dezYUb169dL27duLPH7x4sXq3r275s6dqzVr1qhr16664oortG7dutxjli9frr59+6p///7asGGD+vfvr+uvv14rVqzw1NtCMbKzpdtuk776SgoNlb78Umra1O6qYLc+faRt26QFC6SZM83PrVsJDQAA2Mnrdo5u3bq1mjdvrkmTJuW2NWzYUL1799a4ceNceo7GjRurb9++evzxxyVJffv2VWpqqr7++uvcY3r27KmqVatq1qxZLj0nO0e7n2VJ998vvf66FBAgff651KuX3VUBAABULK6e53pVj0N6errWrFmjHj165Gvv0aOHli1b5tJzZGdn68iRI6pWrVpu2/Llyws95yWXXFLic6alpSk1NTXfBe71xBMmNDgc0vTphAYAAABv5lXBITk5WVlZWYopsPtXTEyM9u7d69JzvPjiizp27Jiuv/763La9e/ee9nOOGzdOkZGRuZf4+PjTeCdw5pVXpCefNNdff1268UZbywEAAIATXhUccjhylk/5l2VZhdqKMmvWLI0aNUqzZ89WdHT0GT3n8OHDlZKSknvZsWPHabwDlOTdd6WhQ831p56S/vMfe+sB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IyPtLsWj7rvvPt1www2qU6eO3aU4RXCAa2bMkGbNkvz9zTKsLEkLAADcKDAwUA0aNLC7DI+LiopSVFSU3WW4hKFKcO7vv81Gb5L0xBNS27b21gMAAEpv9WqpWzfz04sUNwToyJEjGjNmjC644AJVqlQpdzPfkSNHKiMjw6Xnrlu3rurWravDhw/rgQceUHx8vAICAjRt2rTcY37++WfdcMMNqlmzpoKCgpSQkKD7779fBw4cyD1m2rRpqlevniTp3XfflcPhyL3kzFHYvXu3nnjiCbVp00bR0dEKDg5W3bp1de+99yopKalQbUXNcTj1d/H333/r2muvVdWqVVWpUiVdfPHF2rBhg2u/VDejxwEly8jIW3q1Qwfp0UftrggAAJyJ6dOlBQuk996TWrSwu5oSJScnq3Pnzvrtt9/UtGlTDRw4UNnZ2fr999/17LPP6r///a+qVKni0nOlpaWpW7duOnLkiK644goFBQUpJiZGkvT555/r+uuvl7+/v6688krFx8frt99+04QJE/Ttt99qxYoVqlq1qpo2barBgwdr/PjxuvDCC9W7d+/c56/774Ixixcv1osvvqiLLrpIrVu3VmBgoNatW6dJkybp22+/1dq1a10ejrVt2za1bt1ajRo10h133KG//vpLc+bMUdeuXbVp06bc+j2F4ICSPfmk9NNPUmSkGa7k7293RQAAVByWJR0/fubPs327dOCA5HBIH3xg2mbNkq6/3rxG9erSmYyxDwszz+1m9957r3777Tc9+uijGjt2bL779u3bp8qVK7v8XHv37tUFF1ygH3/8Md/u3AcOHFD//v1Vo0YN/fjjj/nmGsyaNUv9+vXT448/rtdee01NmzbVkCFDNH78eDVt2lSjRo0q9DrdunXT3r17C9U2ffp03XrrrZowYYJGuLgH1qJFi/TMM8/o4Ycfzm0bOXKknnrqKU2dOlWPPPKIy+/fHRiqhOItWSLl/Ef6xhtSQoK99QAAUNEcPy5Vrnzml0aNpI4dzeiB/fvNc+/fb2537GjuP5Pnd0e4KWDfvn366KOPdPbZZxd5gh4TE6OA09xL6vnnn88XGiRzQp+amqpx48YVmqB84403qnnz5vogJ2y5IDo6ushA079/f0VEROi7775z+bnq1aunBx98MF/bgAEDJEmrVq1y+XnchR4HFO3wYbPkana2dMst0g032F0RAACoQFavXi3LstS1a1cFBgaWeOy0adO0bdu2fG29e/dW06ZNc2+HhITo/PPPL/TYn376Kffnn3/+Wej+kydPKjk5WcnJyS5PYv7kk0/05ptvau3atTp06JCysrJy79u9e7dLzyFJF154ofz88n/PHxcXJ0k6fPiwy8/jLgQHFGZZ0sCBplvzrLOkCRPsrggAgIopLEw6etQ9z7V+velhKGjpUumUE+xSCQs7s8cXIefEuHbt2k6PnTZtmhYtWpSvrW7duvmCQ3R0tBxFDKc6ePCgJOl1J0vNHzt2zKXg8OKLL+p///ufatSooR49eiguLi63l+OVV15RWlqa0+fIUdRciJxellPDiKcQHFDY9OnS7NlmPsPMmVJ4uN0VAQBQMTkcUqVK7nmunCE6fn5mREHOz9BQ972GG+VMet61a5fTY13Zdbmo0CBJEf8uMf/LL7+oSZMmLtdXlMzMTD355JOqVauW1q9frxo1auTeZ1mWnnvuuTN6frsxxwH5/fmndN995vro0VLr1vbWAwAA3CM6WoqNlRITzdzFxERzOzra7sqK1KJFC/n5+WnBggUuL7taGq3/PddZvny5S8f7/7tQTFHf+CcnJyslJUVt2rTJFxokM/TqxIkTZ1itvQgOyJOz9OrRo1KnTpKHZ+oDAIAyFBcnbdsmrVgh3XOP+bltm2n3QjExMbrmmmv0119/afTo0YXuT0pKUmZm5hm/zu23367w8HCNGDFCGzduLHT/8ePHc+dBSFLVqlXlcDi0c+fOQsdGR0crNDRUa9eu1fFTJowfOnRI999//xnXajeGKiHP6NHSypVSlSpmbWeWXgUAwLcEB+dddzjy3/ZCEydO1K+//qqxY8dq7ty56tatmyzL0ubNmzVv3jzt27fP5X0cilOjRg3NmjVL1113nS688EL17NlTDRo00MmTJ/XPP/9o0aJFateunb755htJUuXKldWyZUstXrxYt99+u+rXry8/Pz/169dPderU0b333qsXX3xRF154oa644gqlpqbq66+/VkJCgmrVquWG34p9CA4wFi+Wnn7aXH/zzTNbyxkAAMANoqKi9NNPP+mFF17Qhx9+qAkTJigkJET16tXTI488okpumptx2WWXad26dXr++ef13Xffaf78+apUqZLi4uJ0++236+abb853/HvvvaehQ4fqs88+U0pKiizLUps2bVSnTh2NGzdO1apV07Rp0zRx4kTFxMTohhtu0OjRo894DoXdHJZlWXYXUR6kpqYqMjJSKSkpuZNofMahQ9KFF0o7dki33y69847dFQEAUKGcPHlSW7duVb169RQSEmJ3OfARrv5duXqeyxyHis6yzDjHHTukc86RXn3V7ooAAADghQgOFd20adKHH0oBAWbp1dPYuh0AAAAVB8GhItuyRcqZ4T9mjNSypb31AAAAwGsRHCqqnKVXjx2TOneWHnrI7ooAAADgxQgOFdUTT0irVklVq7L0KgAAAJwiOFRECxdKzzxjrk+eLMXH21oOAAAAvB/BoaI5eFC6+WazmtKAAdK119pdEQAAAMoBgkNFYlnS3XdLu3ZJ9etLr7xid0UAAAAoJwgOFck770gff8zSqwAAADhtBIeKYvNm6YEHzPWnnpJatLC3HgAAAJQrBIeKID1d6tdPOn5c6tpVevBBuysCAABAOUNwqAgef1xas0aqVk2aPl3y42MHAADA6eEM0tf98IP03HPm+ttvS3Fx9tYDAACAcong4MsOHJD69zerKd11l3T11XZXBAAAUKRt27bJ4XDotttus7uU01K3bl3VrVvX7jI8guDgq3LCwu7d0nnnSS+/bHdFAAAAKMcC7C4AZeTtt6VPP5UCA83Sq5Uq2V0RAABAsWrXrq1NmzYpMjLS7lJQDIKDL/rjD2nIEHN97FipeXNbywEAAN4hK0taskTas0eqWVPq2FHy97e7KiMwMFANGjSwuwyUgKFKvubUpVcvukj673/trggAAHiBTz6R6tY1K7P362d+1q1r2r1BcXMcjhw5ojFjxuiCCy5QpUqVFBkZqWbNmmnkyJHKyMgo8TmnT58uh8OhJ598ssj7f/zxRzkcDg0YMCC3bcGCBbrjjjt03nnnqXLlyqpcubJatGihyZMnn/F7LO8IDr7mscektWul6tWld99l6VUAAKBPPpGuvVbauTN/+65dpt1bwkNBycnJatOmjZ544gn5+/tr4MCBuuOOOxQbG6tnn31Wx44dK/Hxffr0UVhYmN5///0i758xY4YkqX///rltzz77rBYvXqyWLVvqvvvu080336zk5GTdc889+m8F/0KWoUq+5LvvpOefN9enTJFq17a3HgAAcEYsywwiOBNZWdIDD5jnKur5HQ5p8GDp4otLP2wpLMw8j7vde++9+u233/Too49q7Nix+e7bt2+fKleuXOLjK1eurKuvvlrvv/++Vq1apZYtW+bel5GRoQ8//FDx8fHq3LlzbvukSZNUr169fM+TmZmpSy+9VOPHj9fgwYNVp04dN7y78oevo31FcrJ0yy3m+j33SFddZW89AADgjB0/LlWufGaXyEjTs1AcyzI9EZGRpX+NMw03Rdm3b58++ugjnX322Ro1alSh+2NiYhQQ4Pw78JtvvllSXu9Cjrlz5+rAgQO66aab5Dgl9RQMDZIUEBCggQMHKisrSwsWLDjNd+I76HHwBZYl3XmnmenUoIH00kt2VwQAAHBGVq9eLcuy1LVrVwUGBpZ47LRp07Rt27Z8bb1791bTpk3VvXt3xcbG6oMPPtBLL70k/3+7Vd577z1J+YcpSWZOxQsvvKDPPvtMf/31V6HhULt37z7Dd1Z+ERx8weTJ0pw5eUuvhoXZXREAAHCDsDDp6NEze47Fi6VLL3V+3Ny5UqdOpXuNsjj1OHz4sCSzTKsz06ZN06JFi/K11a1bV02bNpW/v79uvPFGvfzyy5o/f7569uyplJQUffXVV2revLkaNWqU+5j09HR16dJFa9euVbNmzdS/f39Vr15dAQEB2rZtm959912lpaW59X2WJwSH8m7TJmnoUHN93DipWTN76wEAAG7jcJz5Vkw9ekhxcWa4UlHzHBwOc3+PHt6zNKskValSRZK0q6RxVv9auHBhiff3799fL7/8smbMmKGePXvqww8/1MmTJwv1NsyZM0dr167VnXfeqbfeeivffR988IHefffd03oPvoY5DuVZWppZT+3ECal797wAAQAA8C9/f2n8eHO94ATmnNuvvOJdoUGSWrRoIT8/Py1YsMDpsqvONGvWTI0aNdJnn32mY8eOacaMGbk9Eaf666+/JElXXnlloedYsmTJGdXgCwgO5dmIEdL69VJUFEuvAgCAYvXpI330UeEFF+PiTHufPvbUVZKYmBhdc801+uuvvzR69OhC9yclJSkzM9Pl5+vfv7+OHTum8ePHa/HixerevbtiYmLyHZOQkCBJWrp0ab72RYsWFeqBqIgYqlRezZsnvfiiuT5litn+EQAAoBh9+phFF7115+iiTJw4Ub/++qvGjh2ruXPnqlu3brIsS5s3b9a8efO0b9++3CFNztx000169NFHNWrUKFmWVWiYkiRdccUV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VK9evbR9+3a7S0MZWrRokQYNGqSffvpJ8+fPV2Zmpnr06KFjx47ZXRo8aNWqVZo8ebIuuOACu0uBBxw6dEjt27dXYGCgvv76a/3222968cUXVaVKFbtLQxl79tln9cYbb2jChAnatGmTnnvuOT3//PN67bXX7C4Nbnbs2DFdeOGFmjBhQpH3P/fcc3rppZc0YcIErVq1SrGxserevbuOHDni4UqLxnKsXq5169Zq3ry5Jk2alNvWsGFD9e7dW+PGjbOxMnjS/v37FR0drUWLFqlTp052lwMPOHr0qJo3b66JEyfqqaeeUtOmTfXKK6/YXRbK0COPPKIff/yRXuUK6PLLL1dMTIymTJmS23bNNdcoLCxM7733no2VoSw5HA59+umn6t27tyTT21CrVi0NGTJEDz/8sCQpLS1NMTExevbZZ3XPPffYWK1Bj4MXS09P15o1a9SjR4987T169NCyZctsqgp2SElJkSRVq1bN5krgKYMGDdJll12miy++2O5S4CGff/65WrRooeuuu07R0dFq1qyZ3nrrLbvLggd06NBB33//vTZv3ixJ2rBhg5YuXapLL73U5srgSVu3btXevXvznfcFBwerc+fOXnPeF2B3AShecnKysrKyFBMTk689JiZGe/futakqeJplWRo2bJg6dOigJk2a2F0OPOCDDz7Q2rVrtWrVKrtLgQf9/fffmjRpkoYNG6ZHH31UK1eu1AMPPKDg4GDdcsstdpeHMvTwww8rJSVFDRo0kL+/v7KysjR27FjdeOONdpcGD8o5tyvqvO+ff/6xo6RCCA7lgMPhyHfbsqxCbfBd9913n37++WctXbrU7lLgATt27NDgwYM1b948hYSE2F0OPCg7O1stWrTQ008/LUlq1qyZNm7cqEmTJhEcfNzs2bM1Y8YMzZw5U40bN9b69es1ZMgQ1apVS7feeqvd5cHDvPm8j+DgxaKiouTv71+odyEpKalQGoVvuv/++/X5559r8eLFiouLs7sceMCaNWuUlJSkxMTE3LasrCwtXrxYEyZMUFpamvz9/W2sEGWlZs2aatSoUb62hg0b6uOPP7apInjKgw8+qEceeUQ33HCDJOn888/XP//8o3HjxhEcKpDY2FhJpuehZs2aue3edN7HHAcvFhQUpMTERM2fPz9f+/z589WuXTubqoInWJal++67T5988ol++OEH1atXz+6S4CEXXXSRfvnlF61fvz730qJFC910001av349ocGHtW/fvtCyy5s3b1ZCQoJNFcFTjh8/Lj+//Kdk/v7+LMdawdSrV0+xsbH5zvvS09O1aNEirznvo8fByw0bNkz9+/dXixYt1LZtW02ePFnbt2/XwIED7S4NZWjQoEGaOXOm5syZo/Dw8Nxep8jISIWGhtpcHcpSeHh4obkslSpVUvXq1Znj4uOGDh2qdu3a6emnn9b111+vlStXavLkyZo8ebLdpaGMXXHFFRo7dqzq1Kmjxo0ba926dXrppZd0xx132F0a3Ozo0aP6888/c29v3bpV69evV7Vq1VSnTh0NGTJETz/9tOrXr6/69evr6aefVlhYmPr162dj1aew4PVef/11KyEhwQoKCrKaN29uLVq0yO6SUMYkFXmZOnWq3aXBBp07d7YGDx5sdxnwgC+++MJq0qSJFRwcbDVo0MCaPHmy3SXBA1JTU63BgwdbderUsUJCQqyzzjrLGjFihJWWlmZ3aXCzBQsWFPnv+6233mpZlmVlZ2dbTzzxhBUbG2sFBwdbnTp1sn755Rd7iz4F+zgAAAAAcIo5DgAAAACcIjgAAAAAcIrgAAAAAMApggMAAAAApwgOAAAAAJwiOAAAAABwiuAAAAAAwCmCAwAAAACnCA4AAJ+1bds2ORwO3XbbbXaXAgDlHsEBAAAAgFMEBwAAAABOERwAAAAAOEVwAAC4bPHixbriiisUFRWl4OBg1a9fX4899piOHz+ee8zChQvlcDg0atQoLV68WJ07d1blypVVrVo19evXTzt37izyuTdu3Ki+ffsqOjpawcHBqlevnoYOHaqDBw8WeXxSUpL+97//6bzzzlNISIiqVaumNm3a6MUXXyzy+L///lvXXnutqlatqkqVKuniiy/Whg0bzvyXAgAVhMOyLMvuIgAA3u+NN97Qvffeq6pVq+qKK65QjRo1tGrVKi1atEjt2rXTggULFBQUpIULF6pr16665JJLtGDBAl122WVq0KCB1q5dq2+//Vbx8fFatWqVYmJicp972bJl6tGjh9LS0nTttdeqbt26+umnn7Rw4ULVr19fy5cvV/Xq1XOP37Jli7p27apdu3apQ4cOateunY4dO6Zff/1VP//8c27Y2LZtm+rVq6fOnTtr48aNatSokVq0aKG//vpLc+bMUdWqVbVp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mTNHv/32m7Zs2aLQ0NB81Xvx4kW1a9dOfn5+uv/++3X27Fl999136t69uzZv3mz7PhUpA/kSGxtrSDJiY2PNLqXQPLvwWUPjZNT5uI6Rkp5idjkAABQLSUlJxq5du4ykpCS77ZmZmUZ8SvxNP3ad3WX8eeRPY83RNUa5d8oZGiej3DvljDVH1xh/HvnT2HV21009f2Zm5k29/uTkZOP48ePZtu/YscPw9fU1unbtatv222+/GZKMhg0bGufOnbM7Pi0tzTh9+rTt4xdffNGQZAwePNhIT0+3O/bixYvGpUuX8lVfx44dDUlGkyZNjJiYGLt9qampRtWqVY1SpUoZ27Zts9v3559/Gu7u7kbv3r3ttksyOnbsmOPXunjxYravYRiGsXz5csPNzc147LHH7LZ//vnnhiTj888/z/Y1JBlDhgwxMjIybNtnzZplSDKefPLJvF62YRi5/2xeLb/nuYw4QJK07fQ2TYuYJkn6+HYaogEAuFmJaYnyneRbKM8dnRit9p+3L5Dnih8TL58SPjf8+VarVZUqVcq2vX79+urcubN+//13paWlydPTU9OmXT7XmDp1qsqUKWN3vIeHh8qXLy9JysjI0P/93//J399fU6dOlbu7u92xN9IXMH78eAUGBtptW7BggY4cOaI333xTjRs3ttvXvn173Xnnnfrll18UFxcnPz+/PL9GbnV17txZ9evX17Jly/Jdr4+PjyZPniw3t386Cx566CE99dRTioiIyPfzFCSCA5RpZNoaou+tf69urX6r2SUBAIBiZNu2bXrnnXe0Zs0anT59WmlpaXb7z507p4oVK2rjxo2yWq3q2LHjNZ9vz549iouLU9euXfOcjvTLL79o27Ztdts6deqUbcnUli1bZvvc9evX277elT0PWU6fPq3MzEzt27dPzZvnb2rYypUr9eGHH2rDhg06d+6c0tPTbftKlMj/hdmwsDD5+toHz6xwdfHixXw/T0EiOEBfbf9Ka6PWysfTR+91e8/scgAAcAolPUsqfkx8gTzXttPbchxhWPPIGjWp0OSmnrukZ8mb+vy1a9eqS5cukqRu3brZTngtFot++eUXbd++XSkpKZIuz9uvVKmS3VX0nGSdGOc0knG1X375RV9++WW27VcHh6zRjCudP39ekvTNN99c82skJCTkWYckff/997r33nvl6+ur7t27q2rV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JJVv3fqGpy25e3vfVGPu+fPnNXz4cHXo0EE9e/ZU6dKldejQIf36669atGiRVq9erRYtWtiOf+mll/Tuu+8qMDBQffv2VVBQkKKiorRs2TKFh4erQYMGkqTk5GR1795dq1evVlhYmB555BFZrVbt379fn376qR588EGVLl0617o6dOigqlWr6scff9S0adPk5eVlt/+bb75RZmamBg8ebNs2c+ZMzZ8/X7fccot69uypxMRErVy5UmPGjFFERIR+/PHHG/4+/fHHH7r99tvl5uame++9V8HBwfrjjz/Url27a74OR0JwcALnEs9pzB9jJElvdX5L5X3Lm1wRAADISErSvCtOmAtL0pkz+qF16xv+/AEREfIoWfKGP7906dI6duyYKlWqZLd9586dat26tcaOHaul/xsVWbhwod599101bNhQK1asUJkyZWzHp6enKyYmxvbxa6+9ptWrV2vw4MH6/PPP5X5FMIqNjbX7OCcWi0X333+/JkyYoPnz5+uee+6x2//NN9/I29tbd999t23bmDFjNG3aNLvnNgxDjz32mD777DP99ddfateu3XV8dy7LzMzUE088ofT0dK1evVrt27e3PfcDDzygOXPmXPdzmoGpSk5gzLIxupB8QY3LN9bTLZ42uxwAAOBCrFZrttAgSfXr11fnzp21evVq2zSfadOmSZKmTp1qFxokycPDQ+XLX774mZGRof/7v/+Tv7+/pk6dmi0k+Pv7y9fXN8/askYTZs+ebbd9+/btioyM1J133qlSpUrZtoeGhmb7WhaLRUOHDpUkLVu2LM+vmZM1a9bo0KFD6t27ty00ZD33xIkT8wxBjoIRh2Juw/ENmrV1liTuEA0AgCNx9/bWgIiIm3qOs5s3a+VTT+V5XKdPP1VQePgNfQ13b+8b+rwrbdu2Te+8847WrFmj06dPZ+sHOHfunCpWrKiNGzfKarWqY8eO13y+PXv2KC4uTl27ds1zGs8vv/yibdu22W3r1KmTOnXqpNq1a6t58+ZatGiRzp8/r8DAQEnS119/LUl205Skyz0VH3/8sb799lvt2bNH8fHxMq6YJnby5Mlr1pKb7du3S7o8fepqoaGhCgkJ0ZEjR27ouYsSZ5nFWFZDtCQ93ORhtaty/UNnAACgcFgslpuaAiRJFdq2Vcny5ZV49mzOfQ4Wi0qWL68KbduatjTr2rVr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JinHx+eff252aTBBx44djWHDhpldBorA/PnzjQYNGhhWq9WoU6eOMWPGDLNLQhGIi4szhg0bZlSpUsXw8vIyqlevbrz88stGSkqK2aWhgK1YsSLH/98feughwzAMIzMz03j99deNChUqGFar1bjllluMyMhIc4u+AvdxAAAAAJAnehwAAAAA5IngAAAAACBPBAcAAAAAeSI4AAAAAMgTwQEAAABAnggOAAAAAPJEcAAAAACQJ4IDAAAAgDwRHAAATuvIkSOyWCx6+OGHzS4FAIo9ggMAAACAPBEcAAAAAOSJ4AAAAAAgTwQHAEC+rV69Wn369FHZsmVltVoVFhamV155RYmJibZjVq5cKYvFonHjxmn16tXq2LGjfH19FRgYqEGDBun48eM5PvfOnTt17733KigoSFarVdWqVdOIESN0/vz5HI8/e/asRo4cqdq1a8vLy0uBgYFq3bq13nvvvRyPP3TokPr376/SpUvLx8dHXbt21fbt22/+mwIALsJiGIZhdhEAAMf36aefasiQISpdurT69OmjcuXKKSIiQqtWrVLbtm21YsUKlShRQitXrlTnzp3VvXt3rVixQr169VKdOnW0ZcsW/f777woJCVFERITKly9ve+61a9eqW7duSklJUf/+/VW1alWtX79eK1euVFhYmNatW6cyZcrYjt+/f786d+6sEydOqH379mrbtq0SEhK0Y8cO/f3337awceTIEVWrVk0dO3bUzp07Va9ePTVv3lwHDx7Uf//7X5UuXVq7d++2qwUAkAsDAIA87Ny50/Dw8DCaNm1qxMTE2O2bNGmSIcmYMmWKYRiGsWLFCkOSIcmYNWuW3bHjx483JBmPPvqobVtGRoYRFhZmSDIWL15sd/yYMWMMSca//vUvu+0tW7Y0JBkzZszIVmtUVJTt74cPH7bV8vbbb9sd98orrxiSjEmTJl3HdwIAXBcjDgCAPA0bNkz//ve/9eeff6p9+/Z2+zIzM1WhQgVVqVJFmzZtso041K5dW7t375bFYrEdm5SUpNDQUMXHx+vixYsqUaKE/vzzT91yyy26/fbbtXDhQrvnTkhIUGhoqBITE23HR0REqGXLlrrlllu0atWqa9adNeJQrVo1HThwQG5ubtn29evXTz/++GMBfJcAwLl5mF0AAMDxrV+/XpK0ePFiLVu2LNt+T09P7dmzx25bu3bt7EKDJHl7eys8PFyLFy/Wvn371KBBA23dulWS1KlTp2zP6+Pjo+bNm+v333+3Hb9x40ZJUrdu3fJdf+PGje1CgyRVrlxZknTx4sV8Pw8AuDKCAwAgT1k9AxMmTMj35wQFBeW4PaufIDY2VpIUFxdnt/1qFSpUsDs+60S/UqVK+a7F398/2zYPj8v/BWZkZOT7eQDAlbGqEgAgT35+fpIun+QbhpHr40pnz57N8bnOnDkj6Z+T+aznztqe2/FZxwUEBEiSTpw4cROvCABwvQgOAIA8tWrVStI/U5by46+//soWJpKSkrR582Z5e3urVq1akqSmTZtKuryM69USExO1adMmeXt7q3bt2pKkli1bSpKWLFly3a8DAHDjCA4AgDwNGTJEHh4eevbZZxUVFZVt/8WLF229Cln27t2rzz77zG7bu+++q+joaA0cOFAlSpSQdLkXokaNGlq0aFG2/olJkybp3Llzdse3aNFCLVu21OrVqzVz5sxstTASAQCFgx4HAECeGjRooOnTp+vpp59W7dq11bNnT9WoUUNxcXE6dOiQVq1apYcffliffvqp7XO6deumIUOG6Lfffst2H4eJEyfajnNzc9MXX3yh7t27q2fPnrrnnnsUGhqqDRs2aPny5apRo4befvttu3pmz56tTp066YknntDXX3+tNm3aKDk5WTt37tTWrVsVExNTZN8bAHAVjDgAAPLl8ccf17p163TnnXdq3bp1+uCDD/TDDz/o3LlzGjFihIYPH253fJs2bbR06VKdO3dOU6dO1YYNG3Tffffpr7/+ytYI3b59e61fv1533nmnlixZoilTpujgwYN67rnntH79epUrV87u+LCwMG3ZskXDhg3TiRMn9OGHH2r27NmKj4/XK6+8UtjfCgBwSdzHAQBQoLLu4/D6669r3LhxZpcDACggjDgAAAAAyBPBAQAAAECeCA4AAAAA8kSPAwAAAIA8MeIAAAAAIE8EBwAAAAB5IjgAAAAAyBPBAQAAAECeCA4AAAAA8kRwAAAAAJAnggMAAACAPBEcAAAAAOSJ4AAAAAAgTwQHAAAAAHkiOAAAAADIE8EBAAAAQJ4IDgAAAADyRHAAAAAAkCeCAwAAAIA8ERwAAAAA5IngAAAAACBPHmYXUFxkZmbq5MmTKlWqlCwWi9nlAAAAAAXCMAxdunRJwcHBcnPLfVyB4JBPJ0+eVEhIiNllAAAAAIUiKipKlStXznU/wSGfSpUqJenyN9TPz8/kagAAAICCERcXp5CQENv5bm4IDvmUNT3Jz8+P4AAAAACnk9d0fJqjAQAAAOSJ4AAAAAAgTwQHAAAAAHkiOAAAAADIE8EBAAAAQJ4IDgAAAADyRHAAAAAAkCeCAwAAAIA8ERwAAIDDWLZsmerVq6dly5aZXQqAqxAcAACAQzAMQ2PHjtXu3bs1duxYGYZhdkkArkBwAAAADmHJkiWKiIiQJEVERGjJkiUmVwTgSh5mFwAAAJyXYRhKSEjQmTNndPbsWdvj6o9Pnz6tffv22T7Pzc1Nr7zyirp16yaLxWLiKwCQheAAAACuS3p6umJiYuxO/q8VDJKSkq77a2RmZmrTpk0aMWKEJk+eLKvVWgivBMD1sBhMIMyXuLg4+fv7KzY2Vn5+fmaXAwBAgcnvqEDWxzExMdfdf1CyZEmVL19eQUFBtkfWx+XKldMbb7yh/fv3KzMzM9vnBgcH68UXX9Tjjz8uHx+fgnrZAP4nv+e5BId8IjjALMuWLdNzzz2nf//73+ratavZ5QAoZAX1bz49PV3nzp3LVxC4kVEBi8WismXL2oWB3IJBUFDQNU/4f//9d/Xo0SPPr1m2bFkNHz5cQ4cOVUBAwHXVCyB3BIcCRnCAGQzDUKtWrRQREaEWLVpow4YNzPUFnNi1/s0bhqH4+Ph8B4GCHhW4+uMyZcrI3d29wF7z5s2bcxxtcHNzU0hIiNzc3HT48GFJUqlSpTR06FCNGDFCQUFBN10D4OoIDgWM4AAz/Pjjj+rfv7/t4++//97uYwDO5eor7y1btpRhGAU6KnB1GMjvqEBhSUlJUWhoqM6cOZPrMRUqVNCBAwf03//+VxMnTtTOnTslSV5eXnr88cf14osvKiQkpKhKBpwOwaGAERxQlM6cOaMPPvhA7777brYrcMHBwWrUqJEaNGighg0bqkGDBqpbt668vb1NqhZAQUhISFBYWJhOnTp1zePMGBUobFFRUYqOjs51f1BQkCpXrizpctP0/PnzNWHCBNvSrZ6enho8eLBGjx6tsLCwIqkZcCYEhwJGcEBROHbsmKZMmaKZM2cqOTk535/n5uammjVr2oJE1p81a9YsFicNgKubP3++HnvsMZ09ezbbvtdff109evSwhQGagy8zDEPLly/XhAkTtGLFCkmXfxfec889Gjt2rBo1amRyhUDxQXAoYAQHFKZ9+/bp7bff1tdff6309HRJko+Pj5KSkuxGHNzd3RUWFqZhw4Zp586dioyMVGRkpM6fP5/j83p5ealevXp2YaJhw4YKDg6mVwJwAEeOHNGwYcP066+/5rjf3d1dzZo1o78pD+vWrdPEiRO1YMEC27bevXtr7NixatOmjYmVAcUDwaGAERxQGLZv366JEyfq+++/tzUx3nrrrbrttts0evToXD9v8eLF6t69u6TLV91Onz6tHTt2KDIy0vbnzp07c50PXbp0aTVo0MAuUDRo0EClS5cu+BcJIJvU1FS99957evPNN5WUlCQ3N7ccG4OzXPlvHrnbvn27Jk2apHnz5tl+p3bu3Fljx47VrbfeSvgCckFwKGAEBxSkdevWacKECfrtt99s2+644w6NGTNGrVq1ynOFkfDw8DyvQGZkZOjw4cN2YWLHjh3at2+fMjIycvycSpUqqWHDhnajE3Xq1KF/AihAy5cv19ChQ7Vnzx5JUseOHRUTE6Ndu3bd1L95/GPfvn2aPHmyvvrqK9sobsuWLTV27Fj16dNHbm5uJlcIOBaCQwEjOOBmGYahP/74QxMmTNDKlSslXT4huPfeezV69GjbfNz8rjBy5MiRG7qTanJysvbu3ZstUBw7dizH493c3BQWFpZtulONGjXonwCuw6lTp/TCCy9o7ty5kqTy5cvrvffe0913362qVasW2r95V5ZT31iDBg00ZswYDRgwQB4eHiZXCDgGgkMBIzjgRmVmZurXX3/VxIkT7VYAeeihh/TSSy/luALI9awwUlBiY2Pt+iayQkV++yeyQgX9E4C99PR0TZ8+Xa+++qri4uLk5uamIUOG6M0337TdxMyMf/Ou5OzZs/rggw80bdo0Xbp0SZJUo0YNjRo1Sg8++CCBDC6P4FDACA64Xunp6fruu+80adIk25rj3t7eeuKJJ/TCCy8UizXHs/onrh6dyE//xNUrPHGXV7ii9evX6+mnn9a2bdskXZ4uM336dIWHh5tbmIu6ePGiPv74Y3344YeKiYmRdHmK5siRI/X444+zYhVcFsGhgBEckF8pKSn68ssvNXnyZB06dEiS5Ofnp2eeeUbDhg1ziruc5tQ/ERkZqf379+faP1G5cuVsgaJu3bry8vLK9essW7ZMzz33nP7973+ra9euhfVygAIXExOjMWPGaObMmZIuB+pJkybp8ccfZ369A0hISNDMmTP17rvv6uTJk5KksmXLavjw4Ro6dCgXOuByCA4FjOCAvCQkJGjGjBmaMmWK3X9EI0aM0NChQ+Xv729yhYUvOTlZe/bsybbCU1RUVI7HZ/VPXD06UaNGDbm5ualVq1aKiIhQixYtaAxFsZCZmanPP/9co0aNsl3RfuSRRzR58mSVK1fO5OpwtZSUFH311Vd6++237S70DB06VMOHD3eKCz1AfhAcChjBAbnJbej7xRdf1OOPP66SJUuaXKH5YmNjtWPHjmyB4lr9E5UrV9aBAwds21iOEo5u+/btevrpp7Vu3TpJl5twP/nkE7Vv397kypCX9PR0zZs3TxMnTrSbWvr4449r5MiRxWJqKXAzCA4FjOCAq505c0Yffvhhtma70aNHa/DgwTTb5cEwDJ06dSpbmNi1a1eO/ROlS5fWnDlz1K1bN6Z6wKHExcXp9ddf10cffaSMjAz5+vpq/PjxevbZZ+Xp6Wl2ebgOmZmZmj9/viZMmGC3mMWDDz6oUaNG5biYBeAMCA4FjOCALLkt7zd27Fjdc889LO93kzIyMvTll1/qX//6V477Q0ND9eijj+qRRx7hKiBMZRiG5s2bpxEjRujUqVOSpHvuuUfvv/8+KyAVc7ktnz1gwACNGTPGtnw24CwIDgWM4IDcbij08ssvq3fv3lwFLyCGYahVq1basmWLXaO1xWKRm5ubbZvFYlGPHj302GOPqU+fPlzZRZHau3evnnnmGS1btkySVLNmTU2bNk3dunUzuTIUtLVr12rixIl2N+zs3bu3Xn75ZbVu3drEyoCCk9/zXM50gDxs375d9913n+rWravPPvtM6enp6tKli5YtW6b169frjjvuIDQUoCVLligiIiLb6kyGYSgjI0MvvfSSOnXqJMMwtGjRIt19992qXLmyXnrpJe3bt8+kquEqEhMT9corr6hhw4ZatmyZrFar3njjDUVGRhIanFTbtm21YMECbd26VQMGDJDFYtGCBQvUpk0bdenSRX/88Ye4BguXYSBfYmNjDUlGbGys2aWgiKxdu9bo1auXIcn26NOnj7Fu3TqzS3NamZmZRosWLQw3Nze773vWw83NzWjRooWRmZlp7Nu3zxg1apRRvnx5u2NuueUW46uvvjISEhLMfjlwMvPnzzeqVq1q+1nr2bOnceDAAbPLQhHbu3ev8eijjxoeHh62n4WWLVsa//3vf42MjAyzywNuSH7Pc5mqlE9MVXINBvNaTZWSkqLQ0FCdOXMm12MqVKigI0eO2JrP09LS9Ntvv2nWrFlatGiRMjMzJUn+/v66//779fjjj6tJkyZFUT6c1NGjRzVs2DD997//lSSFhIRo6tSp6tu3L0sEuzD63eBM6HEoYAQH55a1ksbEiRO1ceNGSaykYZaoqChFR0fnuj8oKCjXxtPjx4/riy++0H/+8x8dOXLEtj08PFyPPfaYBg4c6BL300DBSE1N1fvvv6833nhDSUlJ8vDw0AsvvKBXX32VOwzDhhX24AwIDgWM4OCcstbunjRpknbs2CGJtbudQWZmpv744w/NmjVLP//8s9LS0iRdfm8HDBigxx9/XG3btuVqMXK1YsUKDRkyRHv27JEkdezYUdOnT1e9evVMrgyO6sKFC5o2bVq2e/qMHDlSjz/+OGETDo3gUMAIDs4lJSVFX375pSZPnszdQp1cdHS0vv76a82aNUu7d++2ba9Tp44ee+wxPfjgg9zRFzanT5/WyJEj9c0330i6PML13nvv6f777ydoIl/i4+M1c+ZMTZkyRSdPnpQklS1bVsOHD9fQoUMVEBBgboFADggOBYzg4BwSEhI0Y8YMfqG7IMMwtG7dOs2aNUvfffedEhMTJV2ekta3b1899thj6tq1Kytkuaj09HR98skneuWVVxQXFyeLxaIhQ4borbfe4vcCbggXqFCc5Ps8txAbtJ0KqyoVbxcuXDDefPNNo0yZMrZVMCpVqmR8+OGHRnx8vNnloYjFxsYan376qdG8eXO7FZlCQ0ON8ePHG8eOHTO7RBSh9evXG02bNrX9HLRo0cLYtGmT2WXBSaSlpRmzZ8826tWrZ/sZ8/b2Np577rlsv2uWLl1q1K1b11i6dKlJ1cJV5fc8l+CQTwSH4un06dPG6NGjjVKlStl+YdeoUcOYOXOmkZycbHZ5cABbt241nnnmGSMgIMBu2deePXsaP/30k5Gammp2iSgk586dM5544gnDYrEYkoyAgADjk08+MdLT080uDU4oIyPD+Pnnn+0uWHh6ehr/+te/jH379tmWo84Kr5mZmWaXDBfi9MFhxYoVOa7zLinbOvubN282br31VsPHx8fw9/c37rrrLuPgwYPX9fUIDsXL0aNHjWeffdbw8vKy/Vw0aNDAmDNnjpGWlmZ2eXBAiYmJxuzZs41OnTrZ/T4JCgoyXnrpJWPv3r1ml4gCkpGRYXz22WdG2bJlbe/zww8/bJw5c8bs0uACMjMzjSVLltj9rnFzczM6duxo97tn8eLFZpcKF+L093FYuXKlOnfurIkTJ6pz5852+xo0aCBfX19J0p49e9SyZUs1adJEo0ePVnJysl577TVduHBB27Zty3dTJD0OxcO+ffs0efJkffXVV0pPT5cktWzZUi+//LJ69+7N/HXky/79+/Wf//xHX3zxhd09JTp27KjHHntMd999t7y9vU2sEDfq77//1tNPP621a9dKuvz/xfTp09WhQweTK4MrWrt2rSZOnKjffvvNbrvFYlFISIg++eQTVa9eXVWrVpWXl5dJVcIVOH1zdFZw+P7779W/f/9cjxswYIBWrFihgwcP2r4RR48eVVhYmEaMGKHJkyfn6+sRHBzb9u3bNWnSJH3//fe2G4B17txZL7/8srp06cJqKLgh17q53AMPPKDHHnuMm8sVE5cuXdLrr7+uf//738rIyJCPj4/Gjx+v5557Tp6enmaXBxc3ffp0DR069JrHBAcHq1q1aqpevbqqVatm9/fg4GC5u7sXUbVwRgQHXV4lw8/PTw8++KA+/fRTu33du3fX4cOHtW/fvnx9PYKDY1q3bp0mTpyoBQsW2Lb17t1bY8eOVZs2bUysDM4mKirKdnO5o0eP2rZn3Vxu0KBB/G5wQIZh6Pvvv9eIESNsK6n1799fH3zwQa43EgSKkmEYatWqlbZs2aKMjAy7fd7e3nJ3d1d8fPw1n6NEiRIKDQ21CxNXBozSpUtzAQ3X5DLBISgoSDExMSpZsqTatGmjV199Ve3bt5ck7d27V3Xq1NG0adM0ZMgQu89/8cUX9d577ykxMTHH4b+UlBSlpKTYPo6Li1NISAjBwQTLli3Tc889p3//+9/q2rWrDMPQH3/8oQkTJmjlypWSJDc3Nw0YMECjR49W48aNzS0YTi23m8uVLFlSAwYM0GOPPcbN5RzEvn379Mwzz2jp0qWSpJo1a+rjjz9W9+7dTa4M+Mfvv/+uHj165Lp/0aJFat68uQ4fPqzDhw/r0KFDdn8ePXrUNjU3N/7+/rmOVjANCpILBIetW7fqyy+/VKdOnVSmTBkdOHBA7777rvbt26fffvtN3bt319q1a9WuXTvNnTtX9913n93nT5o0SWPHjtXJkydVsWLFbM8/btw4jR8/Ptt2gkPRyroSExERoebNm+vll1/WpEmTtHHjRkmX1+B/8MEHNWrUKIWFhZlcLVwNN5dzTElJSZo4caLeeecdpaamymq1auzYsXrppZc4QYJDyfo/bvPmzbapkFdyc3NTeHi4NmzYkOvFiIyMDB0/ftwuTFz599OnT+dZx9XToK4MGJUqVaI/0AU4fXDIycWLF9WwYUMFBgZq+/bttuDw7bff6t5777U7Nis4nDp1ShUqVMj2XIw4OIbcrsR4e3vr8ccf18iRIxUSEmJCZcA/DG4u5zB+++03Pfvsszp8+LAkqUePHvr4449Vo0YNkysDsktJSVFoaKjdIgxXq1Chgo4cOSKr1XpDXyMxMVFHjhzJcbTi0KFD+Z4GldNoRfXq1VW6dOkbqguOxSWDgyQ9/fTT+vTTT5WYmKhjx47d8FSlq9HjUPSyrsRs2rRJWT+mbm5ueumllzRixAjuugmHFBcXp7lz52rWrFnatGmTbXtoaKj+9a9/6ZFHHmFufSE4duyYhg0bpl9++UWSVLlyZU2dOlV33XUX08bg0KKiohQdHZ3r/qCgoEL7nWEYhmJiYnIdrbiRaVBXBozrnQZ19dRkV2L2a3fZ4PDUU0/p//7v/5SUlCQPDw/5+fnpoYce0ieffGJ3XI8ePXTo0CGaox1YbqMNixcvZo4yioVt27bpP//5j2bPnq2LFy9Kuhx+e/Tooccee0y9e/dmRZ+blJqaqg8++EBvvPGGEhMT5eHhoeeff16vvvqqbVluADcmPT1dJ06cyHG04nqmQeU2WhEcHGwbib1yanKLFi2uOT3L2TjCa3fJ4HDhwgU1bNhQ5cqV09atWyVJ9957r1auXKkDBw6oVKlSki5fmcpajvXtt9/O13MTHIpWbqtMuLu7q1mzZi71CwXFX1JSkn788UfNmjVLq1atsm0vX768Hn74Yf3rX//KsUfH7CtQjm7lypUaMmSIrb/klltu0fTp01W/fn2TKwNcQ9Y0qJxGK653GpSHh4fd/Syefvpp1a1bt7BfgkPYvXu33QVuMy6QOn1wGDRokKpUqaLmzZurbNmy2r9/v9577z0dPHhQixYtsv0nu2fPHrVo0ULNmjWzuwHc+fPnuQGcA8trlQlGHVBc7du3T5999lmeN5dzhCtQjur06dMaOXKkvvnmG0mXp3JMmTJFDzzwAN8jwEHkNA3qyoCRn2lQrsisC6ROHxzefvttfffddzp8+LDi4+MVGBio9u3ba8yYMWrRooXdsZs3b9aoUaO0bt06eXh4qEuXLpoyZcp1NcsRHIpOQawyATi6tLQ0LViwQLNmzdLixYttP+sBAQF64IEHVK9ePbveLMLy5dVjPvnkE7388suKi4uTxWLR008/rbfeeosGTaCYyZoGdejQIc2fP18ffPBBtmNuueWWHFe+dCanTp3S6tWrs20v6t/5Th8cihrBoegUxSoTgCPJ7eZyWSwWi4KCgvTEE0/I19dXPj4+tsfVH1/5KFGiRLEN11dP09qwYYOGDBmiLVu2SJKaN2+uTz75RM2bNze5UgA3w5WnJjvSayc4FDCCQ9GKiorSiRMndMsttygtLU0//fSTQkNDbfsLc5UJwCxZN5d788039eeff97083l4eOQaKq4VOPKzr0SJEgXwinN25TStpk2bqkWLFpo5c6YMw1BAQIAmTZqkxx9/XO7u7oVWA4Ci4cpTkx3ptRMcChjBoehl3YejXLlyOnPmjNNecQCulNsVKIvForJly6pHjx5KTExUQkKC3SM+Pt7296y7WRemnELJzYaRrMeKFSty/M/0oYce0jvvvMNSzICTcOWpyY722vN7nutR6JUAN+ivv/6SJLVr187pfmEAuVmyZIkiIiKybTcMQ9HR0br//vvzvAKVlpaWa6jI6XGt/Vfvywol6enpio2NVWxsbKF8H67k5eWlxYsXq2PHjoX+tQAUndTUVB07dizHE2fp8ihsVFSU7Q7wzqS4vnaCAxzWmjVrJF0ODoArMAxDr776qtzc3HK9AvXqq6+qW7du1wzTnp6eCggIUEBAQIHXeD2h5HoDS24jJcnJyUpOTi7w1wLAXFarVREREXneAM+RTpwLSnF97UxVyiemKhUtwzBUrlw5xcTEaN26dWrdurXZJQGFztUXBkhNTVXr1q31999/m94oCACuhKlKKNb27t2rmJgYeXl5qVmzZmaXAxSJ4noFqqCsWLHCdvPOK2VkZCgiIkJLlixx2iZJACgOCA5wSFn9DS1atCjU1VsARxMSEqKQkBCzyyhyBTVNCwBQeNzMLgDISVZ/Q/v27U2uBEBRuJ5GQQCAORhxgEO6ckUlAM7P1adpAUBxQHCAwzlz5oz2798vSWrbtq3J1QAoKq46TQsAigumKsHhrF27VpJUv359lS5d2uRqAAAAIBEc4ICypinR3wAAAOA4CA5wONz4DQAAwPEQHOBQEhMTtWXLFkmMOAAAADgSggMcSkREhNLS0lSxYkVVrVrV7HIAAADwPwQHOJQr+xu4yRMAAIDjIDjAodDfAAAA4JgIDnAYmZmZtqVY6W8AAABwLAQHOIydO3cqNjZWPj4+aty4sdnlAAAA4AoEBziMrP6G1q1by8ODm5oDAAA4EoIDHAb9DQAAAI6L4ACHkTXiQHAAAABwPAQHOIQTJ07oyJEjcnNzU+vWrc0uBwAAAFchOMAhZI02NGrUSH5+fiZXAwAAgKsRHOAQrrzxGwAAABwPwQEOgcZoAAAAx0ZwgOkuXbqkbdu2SWLEAQAAwFERHGC6DRs2KDMzU1WqVFHlypXNLgcAAAA5IDjAdPQ3AAAAOD6CA0xHfwMAAIDjIzjAVOnp6Vq/fr0kRhwAAAAcGcEBpvr7778VHx8vPz8/1a9f3+xyAAAAkAuCA0yV1d/Qtm1bubu7m1wNAAAAckNwgKnobwAAACgeCA4wjWEYtuBAfwMAAIBjIzjANEePHtXJkyfl4eGhli1bml0OAAAAroHgANNk9Tc0bdpUJUuWNLkaAAAAXAvBAaZhmhIAAEDxQXCAabJGHGiMBgAAcHwEB5ji4sWL2rFjhySCAwAAQHFAcIAp1q1bJ8MwVKNGDVWoUMHscgAAAJAHggNMkTVNif4GAACA4sFpgsOsWbNksVjk6+ubbd+WLVvUtWtX+fr6KiAgQP369dOhQ4dMqBJZuPEbAABA8eIUweHEiRMaOXKkgoODs+3bs2ePOnXqpNTUVM2bN0+fffaZ9u3bpw4dOig6OtqEapGamqqNGzdKYsQBAACguHCK4PDUU0/plltu0W233ZZt32uvvSar1aoFCxaoZ8+e6tevn3777TdFR0drypQpJlSLrVu3KikpSYGBgapdu7bZ5QAAACAfin1wmD17tlatWqXp06dn25eenq4FCxbo7rvvlp+fn217aGioOnfurJ9//rkoS8X/XLkMq5tbsf8RBAAAcAnF+qzt7NmzGj58uN5++21Vrlw52/6DBw8qKSlJjRo1yravUaNGOnDggJKTk3N87pSUFMXFxdk9UDDobwAAACh+inVwGDJkiGrXrq2nn346x/0xMTGSpMDAwGz7AgMDZRiGLly4kOPnTpo0Sf7+/rZHSEhIwRXuwgzDYEUlAACAYqjYBocff/xR8+fP18yZM2WxWK557LX257ZvzJgxio2NtT2ioqJuql5cduDAAZ09e1YlSpRQeHi42eUAAAAgnzzMLuBGxMfHa+jQoXr22WcVHBysixcvSrq8Wo90+a7Enp6eKlOmjKR/Rh6udP78eVksFgUEBOT4NaxWq6xWa6HU78qyRhuaN28uLy8vk6sBAABAfhXLEYdz587pzJkzeu+991S6dGnbY+7cuUpISFDp0qV1//33q0aNGvL29lZkZGS254iMjFTNmjU5eS1iWf0NTFMCAAAoXorliEOFChW0YsWKbNvffvttrVq1SosWLVLZsmXl4eGhPn366KefftI777yjUqVKSZKOHTumFStWaMSIEUVdusu7ckUlAAAAFB8WwzAMs4soKA8//LB++OEHxcfH27bt2bNHLVq0ULNmzTR69GglJyfrtdde0/nz57Vt2zaVK1cuX88dFxcnf39/xcbG2i3tivw7d+6c7fsdHR2tsmXLmlwRAAAA8nueWyynKl2POnXqaOXKlfL09FT//v318MMPq2bNmlq9enW+QwMKxtq1ayVdfk8IDQAAAMVLsZyqlJsvvvhCX3zxRbbt4eHhWrZsWdEXBDsswwoAAFB8Of2IAxwHN34DAAAovggOKBLJycnatGmTJEYcAAAAiiOCA4rEpk2blJqaqqCgINWoUcPscgAAAHCdCA4oElf2N+R1p28AAAA4HoIDigT9DQAAAMUbwQGFLjMz07YUK/0NAAAAxRPBAYVuz549On/+vLy9vdW0aVOzywEAAMANIDig0GX1N7Rq1Uqenp4mVwMAAIAbQXBAoaO/AQAAoPgjOKDQZY04EBwAAACKL4IDCtXp06d18OBBWSwWtWnTxuxyAAAAcIMIDihUWaMNDRo0UEBAgLnFAAAA4IYRHFCorrzxGwAAAIovggMKFY3RAAAAzoHggEKTkJCgrVu3SmLEAQAAoLgjOKDQbNy4Uenp6apUqZKqVKlidjkAAAC4CQQHFJor+xssFovJ1QAAAOBmEBxQaOhvAAAAcB4EBxSKjIwMrVu3ThL9DQAAAM6A4IBCsWPHDsXFxcnX11cNGzY0uxwAAADcJIIDCkVWf0ObNm3k4eFhcjUAAAC4WQQHFAr6GwAAAJwLwQGFgjtGAwAAOBeCAwpcVFSUjh07Jnd3d7Vq1crscgAAAFAACA4ocFmjDY0bN5avr6/J1QAAAKAgEBxQ4LL6G5imBAAA4DwIDihwWSMONEYDAAA4D4IDClRcXJz+/vtvSQQHAAAAZ0JwQIFav369MjMzVbVqVVWqVMnscgAAAFBACA4oUCzDCgAA4JwIDihQ3PgNAADAOREcUGDS0tK0YcMGSYw4AAAAOBuCAwrM9u3blZCQoICAANWrV8/scgAAAFCACA4oMFn9DW3btpWbGz9aAAAAzoSzOxQY+hsAAACcF8EBBcIwDFZUAgAAcGIEBxSIw4cP69SpU/L09FSLFi3MLgcAAAAFjOCAApE12tCsWTN5e3ubXA0AAAAKGsEBBSKrv4FpSgAAAM6J4IACkTXiQGM0AACAcyI44KZduHBBO3fulERwAAAAcFYEB9y0tWvXSpLCwsIUFBRkcjUAAAAoDMU2OGzbtk29evVSlSpV5O3trcDAQLVp00azZ8/OduyWLVvUtWtX+fr6KiAgQP369dOhQ4dMqNo5sQwrAACA8yu2weHixYsKCQnRxIkTtXDhQn311VeqWrWqBg8erLfeest23J49e9SpUyelpqZq3rx5+uyzz7Rv3z516NBB0dHRJr4C58GN3wAAAJyfxTAMw+wiClLr1q118uRJHTt2TJI0YMAArVixQgcPHpSfn58k6ejRowoLC9OIESM0efLkfD1vXFyc/P39FRsba3seSCkpKQoICFBycrL27Nmj2rVrm10SAAAArkN+z3OL7YhDbsqWLSsPDw9JUnp6uhYsWKC7777b7psQGhqqzp076+effzarTKexZcsWJScnq2zZsqpVq5bZ5QAAAKCQeJhdwM3KzMxUZmamLly4oO+//16///67Pv74Y0nSwYMHlZSUpEaNGmX7vEaNGmnp0qVKTk6Wl5dXtv0pKSlKSUmxfRwXF1d4L6IYu3IZVovFYnI1AAAAKCzFfsRhyJAh8vT0VFBQkEaMGKF///vfevLJJyVJMTExkqTAwMBsnxcYGCjDMHThwoUcn3fSpEny9/e3PUJCQgrvRRRj9DcAAAC4hmIfHMaOHauIiAj99ttvevTRR/XMM89oypQpdsdc60p4bvvGjBmj2NhY2yMqKqpA63YGhmGwohIAAICLKPZTlapUqaIqVapIknr27Cnp8kn/Qw89pDJlykj6Z+ThSufPn5fFYlFAQECOz2u1WmW1WgunaCexb98+nTt3TlarVc2aNTO7HAAAABSiYj/icLWWLVsqPT1dhw4dUo0aNeTt7a3IyMhsx0VGRqpmzZo59jcgf7JGG1q2bEnIAgAAcHJOFxxWrFghNzc3Va9eXR4eHurTp49++uknXbp0yXbMsWPHtGLFCvXr18/ESos/+hsAAABcR7GdqvTEE0/Iz89PLVu2VPny5XXu3Dl9//33+u677/Tiiy+qXLlykqTx48erRYsW6t27t0aPHq3k5GS99tprKlu2rF544QWTX0XxduWKSgAAAHBuxTY4tGnTRp9//rm+/PJLXbx4Ub6+vmrcuLG+/vprPfDAA7bj6tSpo5UrV2rUqFHq37+/PDw81KVLF02ZMsUWLnD9zp49q3379kmS2rZta3I1AAAAKGxOd+fowsKdo+398ssvuuuuu1SvXj3t3LnT7HIAAABwg1z2ztEoGizDCgAA4FoIDrghNEYDAAC4FoIDrltSUpI2b94siREHAAAAV0FwwHWLiIhQWlqaKlSooGrVqpldDgAAAIoAwQHX7cr+BovFYnI1AAAAKAoEB1w3+hsAAABcD8EB1yUzM1Nr166VRH8DAACAKyE44Lrs2rVLFy9eVMmSJdW4cWOzywEAAEARITjgumT1N7Ru3Vqenp4mVwMAAICiQnDAdaG/AQAAwDURHHBduGM0AACAayI4IN9Onjypw4cPy83NTa1btza7HAAAABQhggPyLWu0oWHDhvLz8zO5GgAAABQlggPyjWlKAAAArovggHyjMRoAAMB1ERyQL/Hx8dq2bZskRhwAAABcEcEB+bJhwwZlZGQoJCREISEhZpcDAACAIkZwQL7Q3wAAAODaCA7IF/obAAAAXBvBAXlKT0/XunXrJDHiAAAA4KoIDshTZGSk4uPj5efnpwYNGphdDgAAAExAcECesvob2rRpI3d3d5OrAQAAgBkIDsgT/Q0AAAAgOOCaDMOwBQf6GwAAAFwXwQHXdOzYMZ04cULu7u5q2bKl2eUAAADAJAQHXFNWf0OzZs3k4+NjcjUAAAAwC8EB10R/AwAAACSCA/KQNeJAcAAAAHBtBAfkKjY2VpGRkZIIDgAAAK6O4IBcrVu3ToZhqHr16qpYsaLZ5QAAAMBEBAfkKmuaEsuwAgAAgOCAXNEYDQAAgCwEB+QoLS1NGzZskMSIAwAAAAgOyMXWrVuVlJSk0qVLq06dOmaXAwAAAJMRHJCjK5dhdXPjxwQAAMDVcUaIHNHfAAAAgCsRHJCNYRisqAQAAAA7BAdkc/DgQZ05c0YlSpRQ8+bNzS4HAAAADoDggGyyRhuaN28uLy8vk6sBAACAIyA4IBv6GwAAAHA1ggOyuXJFJQAAAEAiOOAqMTEx2r17tySpbdu2JlcDAAAAR1Fsg8Py5cv16KOPqk6dOvLx8VGlSpV05513avPmzdmO3bJli7p27SpfX18FBASoX79+OnTokAlVO761a9dKkmrXrq1y5cqZXA0AAAAcRbENDp988omOHDmiYcOGaeHChZo6darOnj2r1q1ba/ny5bbj9uzZo06dOik1NVXz5s3TZ599pn379qlDhw6Kjo428RU4JpZhBQAAQE48zC7gRk2bNk1BQUF223r06KGaNWtq4sSJ6tKliyTptddek9Vq1YIFC+Tn5ydJCg8PV1hYmKZMmaLJkycXee2OjMZoAAAA5KTYjjhcHRokydfXV/Xq1VNUVJQkKT09XQsWLNDdd99tCw2SFBoaqs6dO+vnn38usnqLg+TkZEVEREhixAEAAAD2im1wyElsbKy2bNmi+vXrS7p8I7OkpCQ1atQo27GNGjXSgQMHlJycXNRlOqzNmzcrNTVV5cqVU82aNc0uBwAAAA6k2E5VysnQoUOVkJCgl19+WdLlFYIkKTAwMNuxgYGBMgxDFy5cUMWKFbPtT0lJUUpKiu3juLi4QqracVzZ32CxWEyuBgAAAI7EaUYcXn31VX3zzTf64IMPFB4ebrfvWifBue2bNGmS/P39bY+QkJACrdcR0d8AAACA3DhFcBg/frzeeustTZgwQc8884xte5kyZST9M/JwpfPnz8tisSggICDH5xwzZoxiY2Ntj6y+CWeVmZlpW4qV/gYAAABcrdhPVRo/frzGjRuncePGaezYsXb7atSoIW9vb0VGRmb7vMjISNWsWVNeXl45Pq/VapXVai2Umh3R3r17FRMTIy8vLzVt2tTscgAAAOBgivWIw5tvvqlx48bplVde0euvv55tv4eHh/r06aOffvpJly5dsm0/duyYVqxYoX79+hVluQ4tq7+hVatWKlGihMnVAAAAwNEU2xGH9957T6+99pp69OihXr16af369Xb7W7duLenyiESLFi3Uu3dvjR49WsnJyXrttddUtmxZvfDCC2aU7pDobwAAAMC1FNvgMH/+fEnS4sWLtXjx4mz7DcOQJNWpU0crV67UqFGj1L9/f3l4eKhLly6aMmWKypUrV6Q1OzLuGA0AAIBrsRhZZ9i4pri4OPn7+ys2NtbuZnLO4MyZM6pQoYIsFovOnz+fa8M4AAAAnE9+z3OLdY8DCkbWaEP9+vUJDQAAAMgRwQFMUwIAAECeCA6gMRoAAAB5Iji4uMTERG3ZskUSIw4AAADIHcHBxW3cuFHp6ekKDg5WaGio2eUAAADAQREcXNyV/Q0Wi8XkagAAAOCoCA4ujv4GAAAA5AfBwYVlZGRo3bp1kuhvAAAAwLURHFzYzp07FRsbKx8fHzVq1MjscgAAAODACA4uLKu/oU2bNvLw8DC5GgAAADgygoMLo78BAAAA+UVwcGHcMRoAAAD5RXBwUcePH9fRo0fl5uamVq1amV0OAAAAHBzBwUVljTY0adJEpUqVMrkaAAAAODqCg4vKCg70NwAAACA/CA4uisZoAAAAXA+Cgwu6dOmStm/fLongAAAAgPwhOLig9evXKzMzU6GhoapcubLZ5QAAAKAYIDi4IJZhBQAAwPUiOLgg+hsAAABwvQgOLiY9PV3r16+XxIgDAAAA8o/g4GK2b9+uhIQE+fv7q379+maXAwAAgGKC4OBisvob2rZtKzc33n4AAADkD2eOLob+BgAAANwIgoMLMQyDFZUAAABwQwgOLuTIkSM6efKkPDw81KJFC7PLAQAAQDFCcHAhWaMN4eHhKlmypMnVAAAAoDghOLgQ+hsAAABwowgOLiRrxIHgAAAAgOtFcHARFy5c0M6dOyURHAAAAHD9CA4uYt26dTIMQzVr1lT58uXNLgcAAADFDMHBRbAMKwAAAG4GwcFF0BgNAACAm0FwcAGpqanauHGjJEYcAAAAcGMIDi5gy5YtSk5OVpkyZVS7dm2zywEAAEAxRHBwAVcuw2qxWEyuBgAAAMURwcEF0N8AAACAm0VwcHKGYbCiEgAAAG4awcHJ7d+/X9HR0bJarQoPDze7HAAAABRTBAcnlzXa0KJFC1mtVpOrAQAAQHFFcHBy9DcAAACgIBAcnBz9DQAAACgIxTY4XLp0SS+99JK6deumcuXKyWKxaNy4cTkeu2XLFnXt2lW+vr4KCAhQv379dOjQoaIt2ATR0dHau3evJKlt27YmVwMAAIDirNgGh5iYGM2YMUMpKSnq27dvrsft2bNHnTp1UmpqqubNm6fPPvtM+/btU4cOHRQdHV10BZtg7dq1kqS6desqMDDQ5GoAAABQnHmYXcCNCg0N1YULF2SxWHTu3DnNmjUrx+Nee+01Wa1WLViwQH5+fpKk8PBwhYWFacqUKZo8eXJRll2kmKYEAACAglJsRxwsFkued0FOT0/XggULdPfdd9tCg3Q5dHTu3Fk///xzYZdpKhqjAQAAUFCKbXDIj4MHDyopKUmNGjXKtq9Ro0Y6cOCAkpOTTais8CUlJWnTpk2SGHEAAADAzSu2U5XyIyYmRpJynN8fGBgowzB04cIFVaxYMdv+lJQUpaSk2D6Oi4srvEILwaZNm5SWlqby5curevXqZpcDAACAYs6pRxyyXGtKU277Jk2aJH9/f9sjJCSksMorFFf2N+Q1pQsAAADIi1MHhzJlykj6Z+ThSufPn5fFYlFAQECOnztmzBjFxsbaHlFRUYVZaoGjvwEAAAAFyamnKtWoUUPe3t6KjIzMti8yMlI1a9aUl5dXjp9rtVpltVoLu8RCkZmZaVuKlf4GAAAAFASnHnHw8PBQnz599NNPP+nSpUu27ceOHdOKFSvUr18/E6srPLt379aFCxdUsmRJNWnSxOxyAAAA4ASK9YjDokWLlJCQYAsFu3bt0g8//CBJ6tmzp0qWLKnx48erRYsW6t27t0aPHq3k5GS99tprKlu2rF544QUzyy80Wf0NrVq1kqenp8nVAAAAwBkU6+Dw9NNP6+jRo7aPv//+e33//feSpMOHD6tq1aqqU6eOVq5cqVGjRql///7y8PBQly5dNGXKFJUrV86s0gsV/Q0AAAAoaMU6OBw5ciRfx4WHh2vZsmWFW4wD4Y7RAAAAKGhO3ePgik6dOqVDhw7JYrGodevWZpcDAAAAJ0FwcDJZow2NGjWSv7+/ydUAAADAWRAcnExWcKC/AQAAAAWJ4OBkaIwGAABAYSA4OJGEhARt3bpVEo3RAAAAKFgEByeyYcMGZWRkqHLlyqpSpYrZ5QAAAMCJEBycCMuwAgAAoLAQHJwI/Q0AAAAoLAQHJ5GRkaF169ZJYsQBFnRVpgAADiBJREFUAAAABY/g4CQiIyN16dIllSpVSg0bNjS7HAAAADgZgoOTyOpvaNOmjdzd3U2uBgAAAM6G4OAk6G8AAABAYSI4OAlWVAIAAEBhIjg4gWPHjikqKkru7u5q1aqV2eUAAADACREcnEDWaEPTpk3l4+NjcjUAAABwRgQHJ5AVHOhvAAAAQGEhODgBGqMBAABQ2AgOxVxsbKwiIyMlERwAAABQeAgOxdz69euVmZmpatWqKTg42OxyAAAA4KQIDsUcy7ACAACgKBAcijn6GwAAAFAUCA7FWFpamjZs2CCJEQcAAAAULoJDMbZt2zYlJiYqICBAdevWNbscAAAAODGCQzF25f0b3Nx4KwEAAFB4ONssxuhvAAAAQFEhOBRThmGwohIAAACKDMGhmDp06JBOnz4tT09PNW/e3OxyAAAA4OQIDsVU1mhD8+bN5e3tbXI1AAAAcHYEh2KK/gYAAAAUJYJDMUV/AwAAAIoSwaEYOn/+vHbt2iVJatu2rcnVAAAAwBUQHIqhtWvXSpJq1aqlcuXKmVwNAAAAXAHBoRhimhIAAACKGsGhGKIxGgAAAEWN4FDMpKSkKCIiQhIjDgAAACg6BIdiZvPmzUpJSVG5cuUUFhZmdjkAAABwEQSHYiarv6Fdu3ayWCwmVwMAAABXQXAoZuhvAAAAgBkIDsWIYRisqAQAAABTEByKkb179yomJkZeXl5q1qyZ2eUAAADAhRAcipGs0YaWLVuqRIkSJlcDAAAAV0JwKEbobwAAAIBZCA7FCP0NAAAAMItLBIf4+HgNHz5cwcHB8vLyUpMmTfTtt9+aXdZ1OXv2rPbv3y9JatOmjcnVAAAAwNV4mF1AUejXr58iIiL09ttvq1atWpozZ44GDhyozMxMDRo0yOzy8mX69OmSpKpVq6p06dImVwMAAABXYzEMwzC7iMK0cOFC9erVyxYWsnTr1k07d+7UsWPH5O7unufzxMXFyd/fX7GxsfLz8yvMkrMxDEMVK1bUmTNnVK5cOZ05c4abvwEAAKBA5Pc81+mnKv3888/y9fXVPffcY7f9kUce0cmTJ7VhwwaTKsu/JUuW6MyZM5Kk6OhoLVmyxOSKAAAA4GqcPjjs2LFDdevWlYeH/aysRo0a2fbnJCUlRXFxcXYPMxiGobFjx9o+dnNz06uvvionHygCAACAg3H64BATE6PAwMBs27O2xcTE5Ph5kyZNkr+/v+0REhJSqHXmZsmSJdqyZYvt48zMTEVERDDqAAAAgCLl9MFB0jX7AXLbN2bMGMXGxtoeUVFRhVVergzD0KuvvpqtB8Pd3Z1RBwAAABQppw8OZcqUyXFU4fz585KU42iEJFmtVvn5+dk9itqSJUsUERGhjIwMu+0ZGRmMOgAAAKBIOX1waNiwoXbv3q309HS77ZGRkZKkBg0amFFWnrJGG9zccn6L6HUAAABAUXL64HDXXXcpPj5eP/74o932L7/8UsHBwWrVqpVJlV1bamqqjh07pszMzBz3Z2ZmKioqSqmpqUVcGQAAAFyR098A7vbbb9dtt92mp59+WnFxcapZs6bmzp2rxYsXa/bs2fm6h4MZrFarIiIiFB0dnesxQUFBslqtRVgVAAAAXJXT3wBOkuLj4/Xyyy9r3rx5On/+vOrUqaMxY8bovvvuy/dzmHkDOAAAAKCw5Pc81yWCQ0EgOAAAAMAZcedoAAAAAAWG4AAAAAAgTwQHAAAAAHkiOAAAAADIE8EBAAAAQJ4IDgAAAADy5PQ3gCsoWavWxsXFmVwJAAAAUHCyzm/zuksDwSGfLl26JEkKCQkxuRIAAACg4F26dEn+/v657ucGcPmUmZmpkydPqlSpUrJYLEX+9ePi4hQSEqKoqChuQOdieO9dF++9a+J9d128967L7PfeMAxdunRJwcHBcnPLvZOBEYd8cnNzU+XKlc0uQ35+fvwycVG8966L99418b67Lt5712Xme3+tkYYsNEcDAAAAyBPBAQAAAECeCA7FhNVq1euvvy6r1Wp2KShivPeui/feNfG+uy7ee9dVXN57mqMBAAAA5IkRBwAAAAB5IjgAAAAAyBPBAQAAAECeCA4AAAAA8kRwcHDx8fEaPny4goOD5eXlpSZNmujbb781uywUsuXLl+vRRx9VnTp15OPjo0qVKunOO+/U5s2bzS4NRWzWrFmyWCzy9fU1uxQUgTVr1qhnz54qXbq0vL29FRYWpjfffNPsslDItm7dqr59+yo4OFglS5ZUnTp19MYbbygxMdHs0lBALl26pJdeekndunVTuXLlZLFYNG7cuByP3bJli7p27SpfX18FBASoX79+OnToUNEWnAuCg4Pr16+fvvzyS73++utatGiRWrRooYEDB2rOnDlml4ZC9Mknn+jIkSMaNmyYFi5cqKlTp+rs2bNq3bq1li9fbnZ5KCInTpzQyJEjFRwcbHYpKAJz5sxRx44d5e/vr6+++koLFy7UqFGjxOKHzm3Xrl1q27atjhw5og8//FALFizQfffdpzfeeEMDBw40uzwUkJiYGM2YMUMpKSnq27dvrsft2bNHnTp1UmpqqubNm6fPPvtM+/btU4cOHRQdHV10BeeC5Vgd2MKFC9WrVy/NmTPH7pdHt27dtHPnTh07dkzu7u4mVojCcvbsWQUFBdlti4+PV82aNdWgQQMtW7bMpMpQlPr06SOLxaLAwED98MMPio+PN7skFJITJ06odu3aevDBBzV9+nSzy0EReuWVVzRhwgQdOHBANWrUsG1/8sknNWPGDJ0/f16lS5c2sUIUhKzTbYvFonPnzqlcuXJ6/fXXs406DBgwQCtWrNDBgwfl5+cnSTp69KjCwsI0YsQITZ48uahLt8OIgwP7+eef5evrq3vuucdu+yOPPKKTJ09qw4YNJlWGwnZ1aJAkX19f1atXT1FRUSZUhKI2e/ZsrVq1ipNIFzFr1iwlJCRo1KhRZpeCIubp6SlJ8vf3t9seEBAgNzc3lShRwoyyUMAsFossFss1j0lPT9eCBQt0991320KDJIWGhqpz5876+eefC7vMPBEcHNiOHTtUt25deXh42G1v1KiRbT9cR2xsrLZs2aL69eubXQoK2dmzZzV8+HC9/fbbqly5stnloAisXr1agYGB2rNnj5o0aSIPDw8FBQXpqaeeUlxcnNnloRA99NBDCggI0NNPP61Dhw7p0qVLWrBggf7v//5PQ4cOlY+Pj9kloogcPHhQSUlJtvO8KzVq1EgHDhxQcnKyCZX9g+DgwGJiYhQYGJhte9a2mJiYoi4JJho6dKgSEhL08ssvm10KCtmQIUNUu3ZtPf3002aXgiJy4sQJJSYm6p577tG9996rZcuW6cUXX9RXX32lnj170ufgxKpWrap169Zpx44dqlGjhvz8/NSnTx899NBDmjp1qtnloQhlndfldu5nGIYuXLhQ1GXZ8cj7EJjpWsNaeQ15wXm8+uqr+uabb/TRRx8pPDzc7HJQiH788UfNnz9fW7du5d+4C8nMzFRycrJef/11jR49WpLUqVMnlShRQsOHD9cff/yhrl27mlwlCsORI0fUp08flS9fXj/88IPKlSunDRs26K233lJ8fLz+85//mF0iipgjn/sRHBxYmTJlchxVOH/+vKScEymcz/jx4/XWW29pwoQJeuaZZ8wuB4UoPj5eQ4cO1bPPPqvg4GBdvHhRkpSamipJunjxojw9PZm64ITKlCmj/fv3q3v37nbbb7/9dg0fPty2PCOcz+jRoxUXF6dt27bZ/m3fcsstKlu2rB599FE9+OCD6tixo8lVoiiUKVNGUs4zSs6fPy+LxaKAgIAirsoeU5UcWMOGDbV7926lp6fbbY+MjJQkNWjQwIyyUITGjx+vcePGady4cRo7dqzZ5aCQnTt3TmfOnNF7772n0qVL2x5z585VQkKCSpcurfvvv9/sMlEIcprTLP2zEoubG/9dO6tt27apXr162S4ItGjRQhL9jK6kRo0a8vb2tp3nXSkyMlI1a9aUl5eXCZX9g99EDuyuu+5SfHy8fvzxR7vtX375pYKDg9WqVSuTKkNRePPNNzVu3Di98sorev31180uB0WgQoUKWrFiRbZH9+7d5eXlpRUrVuitt94yu0wUgrvvvluStGjRIrvtCxculCS1bt26yGtC0QgODtbOnTuzLbe8bt06SWKBBBfi4eGhPn366KefftKlS5ds248dO6YVK1aoX79+JlZ3GfdxcHDdunXTpk2bNHnyZNWsWVNz587VzJkzNXv2bK48OrH33ntPI0eOVI8ePXIMDZxEuJaHH36Y+zi4gDvuuENLlizRK6+8otatW2vTpk0aP368unbtqvnz55tdHgrJr7/+qr59+6pVq1YaMWKEypYtq/Xr12vSpEmqUqWKtm7dypKsTmLRokVKSEjQpUuX9Oijj+qee+7RgAEDJEk9e/ZUyZIltWfPHrVo0ULNmjXT6NGjlZycrNdee03nz5/Xtm3bVK5cOVNfA8HBwcXHx+vll1/WvHnzdP78edWpU0djxozRfffdZ3ZpKESdOnXSqlWrct3PP1vXQnBwDUlJSRo/frzmzJmjU6dOKTg4WPfff79ef/11Wa1Ws8tDIVqxYoXefvtt/f3334qNjVVISIj69OmjMWPG2Oa9o/irWrWqjh49muO+w4cPq2rVqpKkzZs3a9SoUVq3bp08PDzUpUsXTZkyxe4GgWYhOAAAAADIEz0OAAAAAPJEcAAAAACQJ4IDAAAAgDwRHAAAAADkieAAAAAAIE8EBwAAAAB5IjgAAAAAyBPBAQAAAECeCA4AAAAA8kRwAAAAAJAnggMAAACAPP0/93L+RQp97LAAAAAASUVORK5CYII=",
      "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": 75,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[0.0, 0.07461538461538461, 0.09756410256410257, 0.07435897435897436, 0.09358974358974359, 0.09256410256410257, 0.12064102564102563, 0.14141025641025642, 0.12076923076923077, 0.11333333333333333, 0.11217948717948718]\n",
      "[0.1515, 0.209, 0.2355, 0.2065, 0.2345, 0.2285, 0.25, 0.26, 0.2475, 0.252, 0.2545]\n",
      "[0.0, 8.183418273925781, 9.902008056640625, 8.417083740234375, 9.766334533691406, 9.0904541015625, 10.871856689453125, 8.816581726074219, 10.547737121582031, 9.716079711914062, 9.341712951660156]\n",
      "[19.209999084472656, 22.919998168945312, 23.540000915527344, 23.25, 23.720001220703125, 23.56000518798828, 23.959999084472656, 23.69000244140625, 23.910003662109375, 23.550003051757812, 23.550003051757812]\n",
      "[0.0, 49.5, 55.0, 53.0, 48.5, 48.0, 55.0, 56.0, 50.5, 49.0, 49.0]\n"
     ]
    }
   ],
   "source": [
    "print(ic_rs)\n",
    "print(ic_vs)\n",
    "print(acc_rs)\n",
    "print(acc_vs)\n",
    "print(acc_fs)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Finetune"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "model_ft = copy.deepcopy(model)\n",
    "args.ft_bs = 64\n",
    "retain_loader = replace_loader_dataset(train_loader_full,retain_dataset, seed=seed, batch_size=args.ft_bs, shuffle=True)  \n",
    "finetune(model_ft, retain_loader, epochs=10, quiet=True, lr=0.01)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Negative Gradient"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "args.ng_alpha = 0.9999\n",
    "args.ng_epochs = 10\n",
    "args.ng_lr = 0.01\n",
    "model_ng = copy.deepcopy(model)\n",
    "args.ng_bs = 128\n",
    "retain_loader = replace_loader_dataset(train_loader_full,retain_dataset, seed=seed, batch_size=args.ng_bs, shuffle=True)\n",
    "negative_grad(model_ng, retain_loader, forget_loader, alpha=args.ng_alpha, epochs=args.ng_epochs, quiet=True, lr=args.ng_lr, args=args)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Catastrophic Forgetting k layers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "args.lr_decay_epochs = [10,15,20]\n",
    "args.cfk_lr = 0.01\n",
    "args.cfk_epochs = 10\n",
    "args.cfk_bs = 128\n",
    "retain_loader = replace_loader_dataset(train_loader_full,retain_dataset, seed=seed, batch_size=args.cfk_bs, shuffle=True)\n",
    "\n",
    "model_cfk = copy.deepcopy(model)\n",
    "\n",
    "for param in model_cfk.parameters():\n",
    "    param.requires_grad_(False)\n",
    "\n",
    "if args.model == 'allcnn':\n",
    "    layers = [9]\n",
    "    for k in layers:\n",
    "        for param in model_cfk.features[k].parameters():\n",
    "            param.requires_grad_(True)\n",
    "    \n",
    "elif args.model == \"resnet\":\n",
    "    for param in model_cfk.layer4.parameters():\n",
    "        param.requires_grad_(True)\n",
    "\n",
    "else:\n",
    "    raise NotImplementedError\n",
    "\n",
    "\n",
    "\n",
    "fk_fientune(model_cfk, retain_loader, args=args, epochs=args.cfk_epochs, quiet=True, lr=args.cfk_lr)\n",
    "cfk_time = t2-t1"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Exact Unlearning k layers"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "args.lr_decay_epochs = [10,15,20]\n",
    "args.euk_lr = 0.01\n",
    "args.euk_epochs = training_epochs\n",
    "args.euk_bs = 64\n",
    "retain_loader = replace_loader_dataset(train_loader_full,retain_dataset, seed=seed, batch_size=args.euk_bs, shuffle=True)\n",
    "model_euk = copy.deepcopy(model)\n",
    "\n",
    "for param in model_euk.parameters():\n",
    "    param.requires_grad_(False)\n",
    "\n",
    "if args.model == 'allcnn':\n",
    "    with torch.no_grad():\n",
    "        for k in layers:\n",
    "            for i in range(0,3):\n",
    "                try:\n",
    "                    model_euk.features[k][i].weight.copy_(model_initial.features[k][i].weight)\n",
    "                except:\n",
    "                    print (\"block {}, layer {} does not have weights\".format(k,i))\n",
    "                try:\n",
    "                    model_euk.features[k][i].bias.copy_(model_initial.features[k][i].bias)\n",
    "                except:\n",
    "                    print (\"block {}, layer {} does not have bias\".format(k,i))\n",
    "        model_euk.classifier[0].weight.copy_(model_initial.classifier[0].weight)\n",
    "        model_euk.classifier[0].bias.copy_(model_initial.classifier[0].bias)\n",
    "    \n",
    "    for k in layers:\n",
    "        for param in model_euk.features[k].parameters():\n",
    "            param.requires_grad_(True)\n",
    "    \n",
    "elif args.model == \"resnet\":\n",
    "    with torch.no_grad():\n",
    "        for i in range(0,2):\n",
    "            try:\n",
    "                model_euk.layer4[i].bn1.weight.copy_(model_initial.layer4[i].bn1.weight)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have weight\".format(i))\n",
    "            try:\n",
    "                model_euk.layer4[i].bn1.bias.copy_(model_initial.layer4[i].bn1.bias)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have bias\".format(i))\n",
    "            try:\n",
    "                model_euk.layer4[i].conv1.weight.copy_(model_initial.layer4[i].conv1.weight)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have weight\".format(i))\n",
    "            try:\n",
    "                model_euk.layer4[i].conv1.bias.copy_(model_initial.layer4[i].conv1.bias)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have bias\".format(i))\n",
    "\n",
    "            try:\n",
    "                model_euk.layer4[i].bn2.weight.copy_(model_initial.layer4[i].bn2.weight)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have weight\".format(i))\n",
    "            try:\n",
    "                model_euk.layer4[i].bn2.bias.copy_(model_initial.layer4[i].bn2.bias)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have bias\".format(i))\n",
    "            try:\n",
    "                model_euk.layer4[i].conv2.weight.copy_(model_initial.layer4[i].conv2.weight)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have weight\".format(i))\n",
    "            try:\n",
    "                model_euk.layer4[i].conv2.bias.copy_(model_initial.layer4[i].conv2.bias)\n",
    "            except:\n",
    "                print (\"block 4, layer {} does not have bias\".format(i))\n",
    "\n",
    "        model_euk.layer4[0].shortcut[0].weight.copy_(model_initial.layer4[0].shortcut[0].weight)\n",
    "        \n",
    "    for param in model_euk.layer4.parameters():\n",
    "        param.requires_grad_(True)\n",
    "\n",
    "else:\n",
    "    raise NotImplementedError\n",
    "\n",
    "fk_fientune(model_euk, retain_loader, epochs=args.euk_epochs, quiet=True, lr=args.euk_lr, args=args)\n",
    "euk_time = t2-t1"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Readouts"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "train_loader_full, valid_loader_full, test_loader_full = datasets.get_loaders(dataset, split=\"train\",confuse_mode=True,class_to_replace=class_to_forget, num_indexes_to_replace=num_to_forget, batch_size=args.batch_size, seed=seed, root=args.dataroot, augment=False, shuffle=True)\n",
    "marked_loader, _, _ = datasets.get_loaders(dataset, split=\"forget\", confuse_mode=True,class_to_replace=class_to_forget, num_indexes_to_replace=num_to_forget, only_mark=True, batch_size=1, seed=seed, root=args.dataroot, augment=False, shuffle=True)\n",
    "\n",
    "def replace_loader_dataset(data_loader, dataset, batch_size=args.batch_size, seed=1, shuffle=True):\n",
    "    manual_seed(seed)\n",
    "    loader_args = {'num_workers': 0, 'pin_memory': False}\n",
    "    def _init_fn(worker_id):\n",
    "        np.random.seed(int(seed))\n",
    "    return torch.utils.data.DataLoader(dataset, batch_size=batch_size,num_workers=0,pin_memory=True,shuffle=shuffle)\n",
    "    \n",
    "forget_dataset = copy.deepcopy(marked_loader.dataset)\n",
    "marked = forget_dataset.targets < 0\n",
    "forget_dataset.data = forget_dataset.data[marked]\n",
    "forget_dataset.targets = - forget_dataset.targets[marked] - 1\n",
    "forget_loader = replace_loader_dataset(train_loader_full, forget_dataset, batch_size=args.forget_bs, seed=seed, shuffle=True)\n",
    "\n",
    "retain_dataset = copy.deepcopy(marked_loader.dataset)\n",
    "marked = retain_dataset.targets >= 0\n",
    "retain_dataset.data = retain_dataset.data[marked]\n",
    "retain_dataset.targets = retain_dataset.targets[marked]\n",
    "retain_loader = replace_loader_dataset(train_loader_full, retain_dataset, batch_size=args.retain_bs, seed=seed, shuffle=True)\n",
    "\n",
    "assert(len(forget_dataset) + len(retain_dataset) == len(train_loader_full.dataset))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {
    "scrolled": false
   },
   "outputs": [],
   "source": [
    "try: readouts\n",
    "except: readouts = {}\n",
    "\n",
    "#_,_=activations_predictions(copy.deepcopy(model),forget_loader,'Original_Model_D_f')\n",
    "#thresh=log_dict['Original_Model_D_f_loss']+1e-5\n",
    "#print(thresh)\n",
    "readouts[\"a\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model),thresh=None,name='Original')\n",
    "readouts[\"b\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model0),thresh=None,name='Retrain')\n",
    "readouts[\"c\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model_ft),thresh=None,name='Finetune')\n",
    "readouts[\"d\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model_ng),thresh=None,name='NegGrad')\n",
    "readouts[\"e\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model_cfk),thresh=None,name='CF-k')\n",
    "readouts[\"f\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model_euk),thresh=None,name='EU-k')\n",
    "readouts[\"h\"] = all_readouts(test_loader_full, retain_loader, forget_loader, copy.deepcopy(model_s),thresh=None,name='SCRUB')"
   ]
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3 (ipykernel)",
   "language": "python",
   "name": "python3"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.10.6"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
