{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_35378/3351334044.py:20: 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 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",
    "    \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",
    "import wandb\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": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [],
   "source": [
    "def pdb():\n",
    "    import pdb\n",
    "    pdb.set_trace"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "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}')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [],
   "source": [
    "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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "def print_param_shape(model):\n",
    "    for k,p in model.named_parameters():\n",
    "        print(k,p.shape)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [],
   "source": [
    "### Activations along the path\n",
    "from sklearn.svm import SVC\n",
    "from sklearn.metrics import accuracy_score\n",
    "import datasets\n",
    "\n",
    "def perf_measure(y_actual, y_hat):\n",
    "    TP = 0\n",
    "    FP = 0\n",
    "    TN = 0\n",
    "    FN = 0\n",
    "\n",
    "    for i in range(len(y_hat)): \n",
    "        if y_actual[i]==y_hat[i]==1:\n",
    "           TP += 1\n",
    "        if y_hat[i]==1 and y_actual[i]!=y_hat[i]:\n",
    "           FP += 1\n",
    "        if y_actual[i]==y_hat[i]==0:\n",
    "           TN += 1\n",
    "        if y_hat[i]==0 and y_actual[i]!=y_hat[i]:\n",
    "           FN += 1\n",
    "\n",
    "    return TP, FP, TN, FN\n",
    "\n",
    "def entropy(p, dim = -1, keepdim = False):\n",
    "    return -torch.where(p > 0, p * p.log(), p.new([0.0])).sum(dim=dim, keepdim=keepdim)\n",
    "    #return (p * p.log()).sum(dim=dim, keepdim=keepdim)\n",
    "\n",
    "def collect_prob(data_loader, model):\n",
    "    data_loader = torch.utils.data.DataLoader(data_loader.dataset, batch_size=1, shuffle=False)\n",
    "    sampler = torch.utils.data.RandomSampler(data_loader.dataset, replacement=False, num_samples=4000)\n",
    "    data_loader_small = torch.utils.data.DataLoader(data_loader.dataset, batch_size=1, sampler=sampler, shuffle=False)\n",
    "    prob = []\n",
    "    with torch.no_grad():\n",
    "        for idx, batch in enumerate(tqdm(data_loader_small, leave=False)):\n",
    "            batch = [tensor.to(next(model.parameters()).device) for tensor in batch]\n",
    "            data, target = batch\n",
    "            output = model(data)\n",
    "            prob.append(F.softmax(output, dim=-1).data)\n",
    "    return torch.cat(prob)\n",
    "\n",
    "def get_membership_attack_data(retain_loader, forget_loader, test_loader, model):    \n",
    "    retain_prob = collect_prob(retain_loader, model)\n",
    "    forget_prob = collect_prob(forget_loader, model)\n",
    "    test_prob = collect_prob(test_loader, model)\n",
    "    \n",
    "    X_r = torch.cat([entropy(retain_prob), entropy(test_prob)]).cpu().numpy().reshape(-1, 1)\n",
    "    Y_r = np.concatenate([np.ones(len(retain_prob)), np.zeros(len(test_prob))])\n",
    "    \n",
    "    X_f = entropy(forget_prob).cpu().numpy().reshape(-1, 1)\n",
    "    Y_f = np.concatenate([np.ones(len(forget_prob))])    \n",
    "    return X_f, Y_f, X_r, Y_r\n",
    "\n",
    "def get_membership_attack_prob(retain_loader, forget_loader, test_loader, model):\n",
    "    X_f, Y_f, X_r, Y_r = get_membership_attack_data(retain_loader, forget_loader, test_loader, model)\n",
    "    #clf = SVC(C=3,gamma='auto',kernel='rbf')\n",
    "    clf = LogisticRegression(class_weight='balanced',solver='lbfgs',multi_class='multinomial')\n",
    "    clf.fit(X_r, Y_r)\n",
    "    results = clf.predict(X_f)\n",
    "    results1 = clf.predict(X_r)\n",
    "    acc = accuracy_score(results, Y_f)\n",
    "    train_ac = accuracy_score(results1, Y_r)\n",
    "    TP, FP, TN, FN = perf_measure(Y_r, results1)\n",
    "    FPR = FP/(FP+TN)\n",
    "    FNR = FN/(FN+TP)\n",
    "    \n",
    "    print (f\"TP:{TP}, FP{FP}, TN{TN}, FN{FN}\")\n",
    "    print (f\"false negative rate: {FN/(FN+TP)}\")\n",
    "    print (f\"false positive rate: {FP/(FP+TN)}\")\n",
    "    return acc, train_acc, FPR, FNR #results.mean(), results1.mean()\n",
    "    \n",
    "def plot_entropy_dist(model,retain_loader,forget_loader,test_loader_full, title, ax):\n",
    "    import matplotlib.pyplot as plt\n",
    "    import seaborn as sns\n",
    "    from matplotlib.ticker import FuncFormatter\n",
    "    #train_loader_full, valid_loader_full, test_loader_full = datasets.get_loaders(dataset, batch_size=100, seed=0, augment=False, shuffle=False)\n",
    "    #indexes = np.flatnonzero(np.array(train_loader_full.dataset.targets) == class_to_forget)\n",
    "    #replaced = np.random.RandomState(0).choice(indexes, size=100 if num_to_forget==100 else len(indexes), replace=False)\n",
    "    X_f, Y_f, X_r, Y_r = get_membership_attack_data(retain_loader,forget_loader,test_loader_full, model)\n",
    "    #np.savetxt('retain_prob.txt', X_r[Y_r==1])\n",
    "    #np.savetxt('forget_prob.txt', X_f)\n",
    "    #np.savetxt('test_prob.txt', X_r[Y_r==0])\n",
    "    sns.distplot(np.log(X_r[Y_r==1]).reshape(-1), kde=False, norm_hist=True, rug=False, label='retain', ax=plt)\n",
    "    sns.distplot(np.log(X_r[Y_r==0]).reshape(-1), kde=False, norm_hist=True, rug=False, label='test', ax=plt)\n",
    "    sns.distplot(np.log(X_f).reshape(-1), kde=False, norm_hist=True, rug=False, label='forget', ax=plt)\n",
    "    plt.legend(prop={'size': 14})\n",
    "    plt.tick_params(labelsize=12)\n",
    "    plt.title(title,size=18)\n",
    "    plt.xlabel('Log of Entropy',size=14)\n",
    "    plt.show()\n",
    "    plt.clf()\n",
    "\n",
    "\n",
    "def membership_attack(retain_loader,forget_loader,test_loader,model, name):\n",
    "    prob, train_acc, FPR, FNR = get_membership_attack_prob(retain_loader,forget_loader,test_loader,model)\n",
    "    print(\"Attack prob: \", prob)\n",
    "    return prob"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "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=512, 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=512, 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"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [],
   "source": [
    "def activations_predictions(model,dataloader,name):\n",
    "    criterion = torch.nn.CrossEntropyLoss()\n",
    "    metrics=get_metrics(model,dataloader,criterion,False)\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",
    "    return _,_"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [],
   "source": [
    "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"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {},
   "outputs": [],
   "source": [
    "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"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "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"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [],
   "source": [
    "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"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "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)\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 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)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [],
   "source": [
    "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(model,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.01, threshold=thresh)\n",
    "    test_error = test(model, test_loader_full)['error']\n",
    "    forget_error = test(model, forget_loader)['error']\n",
    "    retain_error = test(model, retain_loader)['error']\n",
    "    print(f\"{name} ->\"\n",
    "          f\"\\tFull test error: {test_error:.2%}\"\n",
    "          f\"\\tForget error: {forget_error:.2%}\\tRetain error: {retain_error:.2%}\"\n",
    "          f\"\\tFine-tune time: {retrain_time+1} steps\")\n",
    "    #print(f\"{name} ->\"\n",
    "    #      f\"\\tFine-tune time: {retrain_time+1} steps\")\n",
    "    log_dict[f\"{name}_retrain_time\"]=retrain_time+1\n",
    "    return(dict(test_error=test_error, forget_error=forget_error, retain_error=retain_error, retrain_time=retrain_time))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [],
   "source": [
    "def test_activations(model_scrubf, modelf0, delta_w_s, delta_w_m0, data_loader, \\\n",
    "                    loss_fn=nn.CrossEntropyLoss(),\\\n",
    "                    optimizer=torch.optim.SGD, \\\n",
    "                    seed=1,quiet=False):\n",
    "\n",
    "    model_scrubf.eval()\n",
    "    modelf0.eval()\n",
    "    \n",
    "    data_loader = torch.utils.data.DataLoader(data_loader.dataset, batch_size=1, shuffle=False)\n",
    "\n",
    "    \n",
    "    metrics = AverageMeter()    \n",
    "    num_classes = data_loader.dataset.targets.max().item() + 1\n",
    "    \n",
    "    for idx, batch in enumerate(tqdm(data_loader, leave=False)):\n",
    "        batch = [tensor.to(next(model_scrubf.parameters()).device) for tensor in batch]\n",
    "        input, target = batch\n",
    "        \n",
    "        output_sf = model_scrubf(input)\n",
    "        G_sf = []\n",
    "\n",
    "        for cls in range(num_classes):\n",
    "            grads = torch.autograd.grad(output_sf[0,cls],model_scrubf.parameters(),retain_graph=True)\n",
    "            grads = torch.cat([g.view(-1) for g in grads])\n",
    "            G_sf.append(grads)\n",
    "\n",
    "        grads = torch.autograd.grad(output_sf[0,cls],model_scrubf.parameters(),retain_graph=False)\n",
    "            \n",
    "        G_sf = torch.stack(G_sf)#.pow(2)\n",
    "        delta_f_sf_update = torch.matmul(G_sf,delta_w_s.sqrt()*torch.empty_like(delta_w_s).normal_())\n",
    "        G_sf = G_sf.pow(2)\n",
    "        delta_f_sf = torch.matmul(G_sf,delta_w_s)\n",
    "\n",
    "        output_m0 = modelf0(input)\n",
    "        G_m0 = []\n",
    "\n",
    "        for cls in range(num_classes):\n",
    "            grads = torch.autograd.grad(output_m0[0,cls],modelf0.parameters(),retain_graph=True)\n",
    "            grad_m0 = torch.cat([g.view(-1) for g in grads])\n",
    "            G_m0.append(grad_m0)\n",
    "\n",
    "        grads = torch.autograd.grad(output_m0[0,cls],modelf0.parameters(),retain_graph=False)\n",
    "            \n",
    "        G_m0 = torch.stack(G_m0).pow(2)\n",
    "        delta_f_m0 = torch.matmul(G_m0,delta_w_m0)\n",
    "        \n",
    "        kl = ((output_m0 - output_sf).pow(2)/delta_f_m0 + delta_f_sf/delta_f_m0 - torch.log(delta_f_sf/delta_f_m0) - 1).sum()\n",
    "        \n",
    "        torch.manual_seed(seed)\n",
    "        output_sf += delta_f_sf_update#delta_f_sf.sqrt()*torch.empty_like(delta_f_sf).normal_()\n",
    "        \n",
    "        loss = loss_fn(output_sf, target)\n",
    "        metrics.update(n=input.size(0), loss=loss.item(), error=get_error(output_sf, target), kl=kl.item())\n",
    "    \n",
    "    return metrics.avg"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Pre-training"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 16,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checkpoint name: lacuna100_allcnn_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1\n",
      "[Logging in lacuna100_allcnn_1_0_forget_None_lr_0_1_bs_128_ls_ce_wd_0_0005_seed_1_training]\n",
      "confuse mode: False\n",
      "split mode: None\n",
      "Number of Classes: 100\n",
      "[0] train metrics:{\"loss\": 3.8233655672073366, \"error\": 0.91084375}\n",
      "Learning Rate : 0.1\n",
      "[0] dry_run metrics:{\"loss\": 3.3233438425064086, \"error\": 0.8385625}\n",
      "Learning Rate : 0.1\n",
      "[0] test metrics:{\"loss\": 3.3385062034606934, \"error\": 0.8347}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 8.13 sec\n",
      "[1] train metrics:{\"loss\": 3.040908597946167, \"error\": 0.77615625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.84 sec\n",
      "[2] train metrics:{\"loss\": 2.4334757966995237, \"error\": 0.61228125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.75 sec\n",
      "[3] train metrics:{\"loss\": 1.9808078465461731, \"error\": 0.47328125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.83 sec\n",
      "[4] train metrics:{\"loss\": 1.7013714575767518, \"error\": 0.372}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.72 sec\n",
      "[5] train metrics:{\"loss\": 1.4868793172836303, \"error\": 0.29221875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.75 sec\n",
      "[6] train metrics:{\"loss\": 1.3205368399620057, \"error\": 0.22421875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.75 sec\n",
      "[7] train metrics:{\"loss\": 1.2272959723472596, \"error\": 0.18584375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.72 sec\n",
      "[8] train metrics:{\"loss\": 1.133659630537033, \"error\": 0.1455625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.76 sec\n",
      "[9] train metrics:{\"loss\": 1.058427065372467, \"error\": 0.1150625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.86 sec\n",
      "[10] train metrics:{\"loss\": 1.0026730167865754, \"error\": 0.09853125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.65 sec\n",
      "[11] train metrics:{\"loss\": 0.9732379763126373, \"error\": 0.0879375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.47 sec\n",
      "[12] train metrics:{\"loss\": 0.9377931756973267, \"error\": 0.078125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.5 sec\n",
      "[13] train metrics:{\"loss\": 0.9080658020973206, \"error\": 0.07284375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.43 sec\n",
      "[14] train metrics:{\"loss\": 0.9161779451370239, \"error\": 0.07359375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.43 sec\n",
      "[15] train metrics:{\"loss\": 0.8948067147731781, \"error\": 0.06978125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.41 sec\n",
      "[16] train metrics:{\"loss\": 0.9115999133586884, \"error\": 0.07090625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.39 sec\n",
      "[17] train metrics:{\"loss\": 0.8692510938644409, \"error\": 0.06215625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.44 sec\n",
      "[18] train metrics:{\"loss\": 0.850548889875412, \"error\": 0.05765625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.43 sec\n",
      "[19] train metrics:{\"loss\": 0.8475376484394074, \"error\": 0.0595625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.42 sec\n",
      "[20] train metrics:{\"loss\": 0.8996174812316895, \"error\": 0.0716875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.45 sec\n",
      "[21] train metrics:{\"loss\": 0.9307970433235169, \"error\": 0.07428125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.42 sec\n",
      "[22] train metrics:{\"loss\": 0.8617413666248321, \"error\": 0.053}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.42 sec\n",
      "[23] train metrics:{\"loss\": 0.7489767985343934, \"error\": 0.03721875}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.45 sec\n",
      "[24] train metrics:{\"loss\": 0.6187533376216888, \"error\": 0.0264375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.39 sec\n",
      "[25] train metrics:{\"loss\": 0.5147787270545959, \"error\": 0.02403125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.49 sec\n",
      "[26] train metrics:{\"loss\": 0.4404430638551712, \"error\": 0.02259375}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.52 sec\n",
      "[27] train metrics:{\"loss\": 0.5160071790218353, \"error\": 0.0500625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.42 sec\n",
      "[28] train metrics:{\"loss\": 1.382216766834259, \"error\": 0.2360625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.3 sec\n",
      "[29] train metrics:{\"loss\": 1.0175488970279694, \"error\": 0.0950625}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.31 sec\n",
      "[30] train metrics:{\"loss\": 0.8608677124977112, \"error\": 0.05125}\n",
      "Learning Rate : 0.1\n",
      "Epoch Time: 3.32 sec\n",
      "Pure training time: 109.99000000000001 sec\n"
     ]
    }
   ],
   "source": [
    "%run main.py --dataset lacuna100 --dataroot=data/lacuna100 --model allcnn --filters 1.0 --lr 0.1 --lossfn ce --num-classes 100"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Train the original model"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checkpoint name: lacuna10_allcnn_1_0_forget_None_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1\n",
      "[Logging in lacuna10_allcnn_1_0_forget_None_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1_training]\n",
      "confuse mode: False\n",
      "split mode: train\n",
      "Number of Classes: 10\n",
      "Epoch: [0][0/25]\tTime 0.034 (0.034)\tData 0.008 (0.008)\tLoss 1.7051 (1.7051)\tAcc@1 57.031 (57.031)\tAcc@5 86.719 (86.719)\n",
      " * Acc@1 86.750 Acc@5 97.969\n",
      "[0] test metrics:{\"loss\": 0.2643024432659149, \"error\": 0.050999993324279784}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.47 sec\n",
      "Epoch: [1][0/25]\tTime 0.013 (0.013)\tData 0.008 (0.008)\tLoss 0.2843 (0.2843)\tAcc@1 94.531 (94.531)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 95.562 Acc@5 99.781\n",
      "Epoch Time: 0.32 sec\n",
      "Epoch: [2][0/25]\tTime 0.014 (0.014)\tData 0.007 (0.007)\tLoss 0.1189 (0.1189)\tAcc@1 97.656 (97.656)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 97.906 Acc@5 99.906\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [3][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0765 (0.0765)\tAcc@1 99.219 (99.219)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.031 Acc@5 99.906\n",
      "Epoch Time: 0.34 sec\n",
      "Epoch: [4][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0227 (0.0227)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.531 Acc@5 99.906\n",
      "Epoch Time: 0.34 sec\n",
      "Epoch: [5][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0466 (0.0466)\tAcc@1 99.219 (99.219)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.812 Acc@5 100.000\n",
      "[5] test metrics:{\"loss\": 0.08815042459964752, \"error\": 0.019999999046325684}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.43 sec\n",
      "Epoch: [6][0/25]\tTime 0.013 (0.013)\tData 0.008 (0.008)\tLoss 0.0434 (0.0434)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.969 Acc@5 100.000\n",
      "Epoch Time: 0.34 sec\n",
      "Epoch: [7][0/25]\tTime 0.012 (0.012)\tData 0.006 (0.006)\tLoss 0.0221 (0.0221)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.32 sec\n",
      "Epoch: [8][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0285 (0.0285)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.34 sec\n",
      "Epoch: [9][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0129 (0.0129)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [10][0/25]\tTime 0.014 (0.014)\tData 0.008 (0.008)\tLoss 0.0126 (0.0126)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "[10] test metrics:{\"loss\": 0.0742943023443222, \"error\": 0.018999999046325683}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.43 sec\n",
      "Epoch: [11][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0126 (0.0126)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [12][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0098 (0.0098)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.32 sec\n",
      "Epoch: [13][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0068 (0.0068)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.32 sec\n",
      "Epoch: [14][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0084 (0.0084)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.32 sec\n",
      "Epoch: [15][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0055 (0.0055)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "[15] test metrics:{\"loss\": 0.07153536677360535, \"error\": 0.018999999046325683}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.43 sec\n",
      "Epoch: [16][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0069 (0.0069)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.32 sec\n",
      "Epoch: [17][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0046 (0.0046)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.32 sec\n",
      "Epoch: [18][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0055 (0.0055)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.32 sec\n",
      "Epoch: [19][0/25]\tTime 0.012 (0.012)\tData 0.006 (0.006)\tLoss 0.0058 (0.0058)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.32 sec\n",
      "Epoch: [20][0/25]\tTime 0.012 (0.012)\tData 0.006 (0.006)\tLoss 0.0051 (0.0051)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "[20] test metrics:{\"loss\": 0.07155667555332183, \"error\": 0.018999999046325683}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.43 sec\n",
      "Epoch: [21][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0072 (0.0072)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [22][0/25]\tTime 0.012 (0.012)\tData 0.006 (0.006)\tLoss 0.0040 (0.0040)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.32 sec\n",
      "Epoch: [23][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0038 (0.0038)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [24][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0050 (0.0050)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [25][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0041 (0.0041)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "[25] test metrics:{\"loss\": 0.07072802913188934, \"error\": 0.018999999046325683}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.45 sec\n",
      "Pure training time: 8.520000000000001 sec\n"
     ]
    }
   ],
   "source": [
    "%run main_merged.py --dataset lacuna10 --model allcnn --dataroot=data/lacuna10/ --filters 1.0 --lr 0.01 \\\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 5e-4 --batch-size 128 --epochs 26 --seed 1"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Retrain Forgetting"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Checkpoint name: lacuna10_allcnn_1_0_forget_[5]_num_100_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1\n",
      "[Logging in lacuna10_allcnn_1_0_forget_[5]_num_100_lr_0_01_bs_128_ls_ce_wd_0_0005_seed_1_training]\n",
      "confuse mode: False\n",
      "split mode: train\n",
      "Replacing indexes [1781 1858 1615 1665 1816 1663 1840 1828 1767 1756 1664 1703 1724 1915\n",
      " 1910 1660 1722 1871 1666 1626 1886 1758 1736 1909 1764 1800 1872 1607\n",
      " 1606 1830 1854 1622 1752 1621 1655 1757 1612 1708 1668 1744 1770 1878\n",
      " 1659 1734 1831 1692 1775 1867 1674 1681 1888 1859 1864 1652 1733 1776\n",
      " 1656 1690 1848 1617 1601 1608 1750 1806 1809 1740 1817 1794 1605 1633\n",
      " 1880 1634 1812 1815 1863 1737 1645 1701 1866 1726 1798 1838 1799 1822\n",
      " 1720 1845 1735 1673 1913 1904 1716 1855 1629 1697 1620 1646 1813 1862\n",
      " 1912 1689]\n",
      "Number of Classes: 10\n",
      "Epoch: [0][0/25]\tTime 0.014 (0.014)\tData 0.008 (0.008)\tLoss 1.7059 (1.7059)\tAcc@1 58.594 (58.594)\tAcc@5 86.719 (86.719)\n",
      " * Acc@1 86.688 Acc@5 98.000\n",
      "[0] test metrics:{\"loss\": 0.26762058663368227, \"error\": 0.04799999332427979}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.44 sec\n",
      "Epoch: [1][0/25]\tTime 0.013 (0.013)\tData 0.008 (0.008)\tLoss 0.2866 (0.2866)\tAcc@1 94.531 (94.531)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 95.656 Acc@5 99.781\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [2][0/25]\tTime 0.014 (0.014)\tData 0.008 (0.008)\tLoss 0.1144 (0.1144)\tAcc@1 97.656 (97.656)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 97.906 Acc@5 99.906\n",
      "Epoch Time: 0.34 sec\n",
      "Epoch: [3][0/25]\tTime 0.015 (0.015)\tData 0.009 (0.009)\tLoss 0.0788 (0.0788)\tAcc@1 99.219 (99.219)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.094 Acc@5 99.906\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [4][0/25]\tTime 0.013 (0.013)\tData 0.008 (0.008)\tLoss 0.0235 (0.0235)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.562 Acc@5 99.906\n",
      "Epoch Time: 0.34 sec\n",
      "Epoch: [5][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0468 (0.0468)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.844 Acc@5 99.969\n",
      "[5] test metrics:{\"loss\": 0.08906168484687806, \"error\": 0.019999999046325684}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.43 sec\n",
      "Epoch: [6][0/25]\tTime 0.014 (0.014)\tData 0.008 (0.008)\tLoss 0.0412 (0.0412)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 99.969 Acc@5 100.000\n",
      "Epoch Time: 0.34 sec\n",
      "Epoch: [7][0/25]\tTime 0.013 (0.013)\tData 0.008 (0.008)\tLoss 0.0220 (0.0220)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.34 sec\n",
      "Epoch: [8][0/25]\tTime 0.013 (0.013)\tData 0.008 (0.008)\tLoss 0.0278 (0.0278)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.34 sec\n",
      "Epoch: [9][0/25]\tTime 0.014 (0.014)\tData 0.007 (0.007)\tLoss 0.0137 (0.0137)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [10][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0124 (0.0124)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "[10] test metrics:{\"loss\": 0.07493651735782624, \"error\": 0.019999999046325684}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.43 sec\n",
      "Epoch: [11][0/25]\tTime 0.013 (0.013)\tData 0.008 (0.008)\tLoss 0.0126 (0.0126)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.34 sec\n",
      "Epoch: [12][0/25]\tTime 0.013 (0.013)\tData 0.008 (0.008)\tLoss 0.0098 (0.0098)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [13][0/25]\tTime 0.014 (0.014)\tData 0.007 (0.007)\tLoss 0.0072 (0.0072)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [14][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0080 (0.0080)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [15][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0060 (0.0060)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "[15] test metrics:{\"loss\": 0.07198784399032593, \"error\": 0.019999999046325684}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.43 sec\n",
      "Epoch: [16][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0072 (0.0072)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [17][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0041 (0.0041)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [18][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0050 (0.0050)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [19][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0059 (0.0059)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [20][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0046 (0.0046)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "[20] test metrics:{\"loss\": 0.07289170265197754, \"error\": 0.019999999046325684}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.44 sec\n",
      "Epoch: [21][0/25]\tTime 0.014 (0.014)\tData 0.008 (0.008)\tLoss 0.0063 (0.0063)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [22][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0040 (0.0040)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [23][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0036 (0.0036)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.32 sec\n",
      "Epoch: [24][0/25]\tTime 0.012 (0.012)\tData 0.007 (0.007)\tLoss 0.0050 (0.0050)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "Epoch Time: 0.33 sec\n",
      "Epoch: [25][0/25]\tTime 0.013 (0.013)\tData 0.007 (0.007)\tLoss 0.0042 (0.0042)\tAcc@1 100.000 (100.000)\tAcc@5 100.000 (100.000)\n",
      " * Acc@1 100.000 Acc@5 100.000\n",
      "[25] test metrics:{\"loss\": 0.07140162253379821, \"error\": 0.018999999046325683}\n",
      "Learning Rate : 0.01\n",
      "Epoch Time: 0.43 sec\n",
      "Pure training time: 8.61 sec\n"
     ]
    }
   ],
   "source": [
    "%run main_merged.py --dataset lacuna10 --model allcnn --dataroot=data/lacuna10/ --filters 1 --lr 0.01 \\\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 5e-4 --batch-size 128 --epochs 26 \\\n",
    "--forget-class 5 --num-to-forget 100 --seed 1"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Logs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "metadata": {},
   "outputs": [],
   "source": [
    "log_dict={}\n",
    "training_epochs=25"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 20,
   "metadata": {},
   "outputs": [],
   "source": [
    "log_dict['epoch']=training_epochs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Total Number of Parameters: 1620010\n"
     ]
    }
   ],
   "source": [
    "parameter_count(copy.deepcopy(model))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Loads checkpoints"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "metadata": {},
   "outputs": [],
   "source": [
    "import copy\n",
    "model0 = 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 - 4000#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",
    "\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()\n",
    "\n",
    "teacher = copy.deepcopy(model)\n",
    "student = copy.deepcopy(model)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "#### Data Loader"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "confuse mode: False\n",
      "split mode: train\n",
      "confuse mode: False\n",
      "split mode: train\n",
      "Replacing indexes [1781 1858 1615 1665 1816 1663 1840 1828 1767 1756 1664 1703 1724 1915\n",
      " 1910 1660 1722 1871 1666 1626 1886 1758 1736 1909 1764 1800 1872 1607\n",
      " 1606 1830 1854 1622 1752 1621 1655 1757 1612 1708 1668 1744 1770 1878\n",
      " 1659 1734 1831 1692 1775 1867 1674 1681 1888 1859 1864 1652 1733 1776\n",
      " 1656 1690 1848 1617 1601 1608 1750 1806 1809 1740 1817 1794 1605 1633\n",
      " 1880 1634 1812 1815 1863 1737 1645 1701 1866 1726 1798 1838 1799 1822\n",
      " 1720 1845 1735 1673 1913 1904 1716 1855 1629 1697 1620 1646 1813 1862\n",
      " 1912 1689]\n"
     ]
    }
   ],
   "source": [
    "train_loader_full, valid_loader_full, test_loader_full   = datasets.get_loaders(dataset, batch_size=args.batch_size, seed=seed, root=args.dataroot, augment=False, shuffle=True)\n",
    "marked_loader, _, _  = datasets.get_loaders(dataset, 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=512, 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=256, seed=seed, shuffle=True)\n",
    "\n",
    "\n",
    "assert(len(forget_dataset) + len(retain_dataset) == len(train_loader_full.dataset))"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "metadata": {},
   "outputs": [],
   "source": [
    "log_dict['args']=args"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "100\n",
      "3100\n",
      "1000\n",
      "3200\n",
      "{0: 320, 1: 320, 2: 320, 3: 320, 4: 320, 5: 320, 6: 320, 7: 320, 8: 320, 9: 320}\n"
     ]
    }
   ],
   "source": [
    "print (len(forget_loader.dataset))\n",
    "print (len(retain_loader.dataset))\n",
    "print (len(test_loader_full.dataset))\n",
    "print (len(train_loader_full.dataset))\n",
    "from collections import Counter\n",
    "print(dict(Counter(train_loader_full.dataset.targets)))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## SCRUB Forgetting"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "metadata": {},
   "outputs": [],
   "source": [
    "args.optim = 'sgd'\n",
    "args.gamma = 0.9999\n",
    "args.alpha = 0.001\n",
    "args.beta = 0.9\n",
    "args.smoothing = 0.0\n",
    "args.msteps = 2\n",
    "args.clip = 0.2\n",
    "args.sstart = 4\n",
    "args.kd_T = 4\n",
    "args.distill = 'kd'\n",
    "\n",
    "args.sgda_batch_size = 32\n",
    "args.del_batch_size = 8\n",
    "args.sgda_epochs = 5\n",
    "args.sgda_learning_rate = 0.0001\n",
    "args.lr_decay_epochs = [3,5,9]\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": 27,
   "metadata": {},
   "outputs": [],
   "source": [
    "model_t = copy.deepcopy(teacher)\n",
    "model_s = copy.deepcopy(student)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 28,
   "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": 29,
   "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": 30,
   "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": 35,
   "metadata": {},
   "outputs": [],
   "source": [
    "acc_r, acc5_r, loss_r = validate(retain_loader, model_t, criterion_cls, args, True, prefix=\"train_lacuna10_selective_allcnn_before\")\n",
    "acc_f, acc5_f, loss_f = validate(forget_loader, model_t, criterion_cls, args, True, prefix=\"forget_lacuna10_selective_allcnn_before\")\n",
    "torch.save(model_s.state_dict(), 'final_model_weights.pth')\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "==> SCRUB unlearning ...\n",
      " * Acc@1 100.000 \n",
      "maximize loss: -14.46\t minimize loss: 0.00\t train_acc: 100.00000762939453\n",
      "==> SCRUB unlearning ...\n",
      " * Acc@1 100.000 \n",
      "maximize loss: -14.50\t minimize loss: 0.00\t train_acc: 100.00000762939453\n",
      "==> SCRUB unlearning ...\n",
      " * Acc@1 100.000 \n",
      "maximize loss: 0.00\t minimize loss: 0.00\t train_acc: 100.00000762939453\n",
      "==> SCRUB unlearning ...\n",
      " * Acc@1 100.000 \n",
      "maximize loss: 0.00\t minimize loss: 0.00\t train_acc: 100.00000762939453\n",
      "==> SCRUB unlearning ...\n",
      " * Acc@1 100.000 \n",
      "maximize loss: 0.00\t minimize loss: 0.00\t train_acc: 100.00000762939453\n",
      "Time taken: 4.529714584350586\n",
      "Test: [0/8]\tTime 0.000 (0.000)\tLoss 11.7076 (11.7076)\tAcc@1 0.000 (0.000)\tAcc@5 8.594 (8.594)\n",
      " * Acc@1 9.800 Acc@5 50.500\n"
     ]
    }
   ],
   "source": [
    "acc_rs = []\n",
    "acc_fs = []\n",
    "acc_ts = []\n",
    "acc_vs = []\n",
    "start = time.time()\n",
    "for epoch in range(1, args.sgda_epochs + 1):\n",
    "\n",
    "    lr = sgda_adjust_learning_rate(epoch, args, optimizer)\n",
    "\n",
    "    print(\"==> SCRUB unlearning ...\")\n",
    "\n",
    "    acc_r, acc5_r, loss_r = validate(retain_loader, model_s, criterion_cls, args, True, prefix=\"train_lacuna10_selective_allcnn\")\n",
    "    acc_f, acc5_f, loss_f = validate(forget_loader, model_s, criterion_cls, args, True, prefix=\"forget_lacuna10_selective_allcnn\")\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",
    "    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",
    "end = time.time()\n",
    "print(f\"Time taken: {end-start}\")\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",
    "model = models.get_model('allcnn', num_classes=10,filters_percentage=1.0) \n",
    "model = model.to('cuda')\n",
    "model.load_state_dict(torch.load('complete_weights.pth'))\n",
    "acc_t_final, acc5_t_final, loss_t_final = validate(test_loader_full, model, criterion_cls, args, False, prefix=\"test_cifar5_resnet_final\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_35378/2123027085.py:114: UserWarning: \n",
      "\n",
      "`distplot` is a deprecated function and will be removed in seaborn v0.14.0.\n",
      "\n",
      "Please adapt your code to use either `displot` (a figure-level function with\n",
      "similar flexibility) or `histplot` (an axes-level function for histograms).\n",
      "\n",
      "For a guide to updating your code to use the new functions, please see\n",
      "https://gist.github.com/mwaskom/de44147ed2974457ad6372750bbe5751\n",
      "\n",
      "  sns.distplot(np.array(test_losses), kde=False, norm_hist=False, rug=False, label='test-loss', ax=plt)\n",
      "/tmp/ipykernel_35378/2123027085.py:115: UserWarning: \n",
      "\n",
      "`distplot` is a deprecated function and will be removed in seaborn v0.14.0.\n",
      "\n",
      "Please adapt your code to use either `displot` (a figure-level function with\n",
      "similar flexibility) or `histplot` (an axes-level function for histograms).\n",
      "\n",
      "For a guide to updating your code to use the new functions, please see\n",
      "https://gist.github.com/mwaskom/de44147ed2974457ad6372750bbe5751\n",
      "\n",
      "  sns.distplot(np.array(forget_losses), kde=False, norm_hist=False, rug=False, label='forget-loss', ax=plt)\n"
     ]
    },
    {
     "data": {
      "image/png": "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b2/l5eWpR48eSklJUXh4uB544AEVFBRo+fLl6tOnj15//XU99thjks4/C6V79+7atGmTbr31VvXo0UM1atRQWlqalixZoiFDhigwMFAxMTGSpPXr12vIkCG2AOLt7X1ZxzJ69Gi98soruuaaa/TAAw/IYrFo8eLFiomJ0Y8//qiXXnrJ1nfWrFkaMWKEGjRooH79+snX11e///67vvvuOy1dulR9+/aVJC1fvlxRUVHy9vZWnz591KBBAx09elTbtm3TRx99pAcffPCyaq4qyhxEPv74Y91///26++679f7778vd3V179uzRoUOHbH127dqliIgItWrVSgsXLlRubq7i4uLUqVMnbdu2Tf7+/hV6EACA6qcoeMybN08REREOk1UnTJiglJQUxcfHKy4uznYjrezsbHXp0kVjxoxRdHS0AgICtH37dm3atEn9+vVTYmKi3XYuvB15TEyM0tLStH79esXExFTIZNUvv/xSr7zyipo3b66NGzfKy8tL0vlRng4dOujll19WdHS0OnbsKEl69913VbNmTf34448On4fHjx+3/fecOXNkGIZSUlIUFhZWYr/qrkyXZg4ePKiHH35YjzzyiObPn6+oqChFRkbqwQcfVFxcnK1fXFycrFarkpKS1LNnT0VHR2v58uU6evSoZsyYUeEHAQC4spw7d06zZs1SSEiIXQiRJA8PD8XFxSk/P98hdLi5uTlsy2q1yt3dvdJqnTt3rqTzl5qKQogkeXl5adKkSXZ9iri4uBR7W39fX1+HZcUdU3H9qqsyjYi8++67ysnJ0bhx40rsU1hYqKSkJP3973+3ezJfcHCwIiMjtWTJEk2fPr38FQMArng///yzTpw4oYCAANucjgsdPXpU0vkReElq3ry5brjhBn388cdKT09X37591alTJ7Vu3VpOTk6l3u/SpUu1bds2u2UREREXHTnZunWrrd+fFS27cJt33323YmNj1bJlS91zzz2KiIhQx44dHS4P3X333UpMTFT79u01ePBgdenSRZ06dVLdunVLfTzVQZmCyIYNG+Tj46Ndu3apT58+2r59u3x8fBQdHa0XXnhBnp6e2rNnj86cOeMwjCRJYWFhWrNmjXJzc+Xq6lphBwEAuLJkZGRIknbs2KEdO3aU2C8nJ0eS5OzsrHXr1ik+Pl6JiYkaM2aMJMnPz0+PP/64JkyYUKpA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nDhRp0+ftjuOyMhISVJCQoLtOCwWi9LS0i7rHH3zzTfq1auXfHx85OrqqtDQUMXHx9vtv8iWLVs0YMAABQUFyWq1ql69err55ps1bdo0u36//vqrhg4dqsaNG8vV1VV+fn5q3bq1xowZc1m1lheXZgAApigKHK+88ookadSoUZKkRo0ayTAMDRo0SImJibr22ms1cuRI5eTkaOHCherdu7deffVVPfHEEw7bXLRokVavXq3evXtrxIgRys7OliQdPHhQt9xyi37//Xf17NlTN954o37++WfdcccdthDxZ2+99ZZGjBihOnXqKCoqSv7+/vr+++81ZcoUJScnKzk5WTVr1lRERITS0tI0b948de7c2XZckuTt7V3u87N48WLdc889qlmzpgYNGqS6detq7dq1SkhI0OrVq5WcnCyr1SpJ2rZtm2655RY5OTmpT58+Cg4O1smTJ7Vjxw698847io2NlXR+9Kldu3bKyclRr169NGjQIJ06dUq//vqrXn/9dc2cObPc9ZYXQQQAYIqIiAhFRETYvi1y4STPDz74QImJiercubNWr16tmjVrSpImTJigNm3a6KmnnlJUVJQaN25st82VK1dq9erV6tq1q93y2NhY/f7773rxxRf11FNP2ZbPnTtXQ4c6TrrduXOnHn/8cbVq1Upr166Vj4+PrW3atGkaP368Xn/9dY0ZM8YWPObNm6eIiIgKmayanZ2tBx98UE5OTtq4caPCwsIkSYZh6G9/+5s+/vhjvfjii5o4caLtfOXl5ek///mP7rrrLrttHT9+3Pbfixcv1smTJ4sNcseOHbvsusuDSzMAgCqnKJy88MILthAiSQ0bNtTo0aNVUFCgjz76yGG9vn37OoSQvLw8LVq0SPXq1XP48B0yZIhCQ0MdtjN79mwVFhbqtddeswshkjR27Fj5+/tr/vz55T28S1q6dKlOnjypYcOG2UKIJFksFk2bNk3Ozs7Fft3Xzc3NYZmvr2+p+vn5+V1e0eXEiAgAoMrZunWr3Nzc1K5dO4e2ohGIbdu2ObQV1//nn39WXl6ebrrpJrtQI53/YL/55pu1a9cuu+XffvutJGnVqlVau3atwzZdXFwc1inJtm3btHTpUrtljRo1UkxMTInrbN26VZLsLvMUCQwMVNOmTfXzzz8rOztbHh4eGjBggF555RX17dtXd999t7p166aO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+/bt2r9/v8N2ikZOiian7tu3Tzt37rxkv9K4//77lZOTo1dffVUbNmxQt27dHC4lBQcHS5K++uoru+Xr16/XO++8U+p93XTTTZKkuXPn2i3/9NNPtX79eof+I0aMkLOzsx5//PFi741y8uRJuwDz+eefO4y2FBQU2C7xlOW8AGCOCHBFmDVrljp27KiHHnpIn332mVq0aKGdO3dq2bJl8vf316xZs2x9165dqzFjxujWW29VaGiofH19tXfvXi1btkxubm567LHHJEk//vij+vXrp7Zt26ply5aqX7++Dh48qKVLl8rJyck2Z6Q07rvvPj3zzDOKj4+XYRgOl2UkKSoqSo0aNdILL7yg7du3q2XLlvr555+VlJSkvn37avHixaXaV9++fdW4cWPNnTtX6enpCg8P108//aR169apZ8+etjkxRVq2bKk333xTw4cP13XXXaeePXuqadOmysrK0t69e7V+/XrFxMTorbfekiQNGjRItWrVUseOHRUcHKyCggKtWbNGO3fu1KBBgxQUFFTq8wJA3EcEqE5Kuo+IYRhGWlqaMXToUKNBgwaGs7Oz0aBBA2Po0KFGWlqaXb+dO3caTz75pBEeHm74+voaVqvVaNKkiRETE2Ps3LnT1i89Pd2IjY01OnToYNStW9eoWbOmERQUZAwYMMDYtGlTmWuPjIw0JBnu7u5GTk5OsX327t1r9O/f3/D39zdq1apltG3b1liwYIGRnJxsSDImTZpk17+4+4gUbadPnz6Gh4eHUbt2beP22283vv/++2LvI1Lku+++M+655x4jICDAcHFxMfz8/IzWrVsbsbGxxk8//WTr9+abbxp33XWXERwcbLi6uhq+vr5G+/btjdmzZxsFBQVlPi/A1c5iGH+6qAwAAPAXYY4IAAAwDUEEAACYhiACAABMQxABAACmIYgAAADTEEQAAIBpCCIAAMA0BBEAAGAagggAADANQQQAAJiGIAIAAExDEAEAAKYhiAAAANP8H+cqtn9xh1MJAAAAAElFTkSuQmCC",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "10.856985 9.667406e-05\n",
      "11.382576 4.2199197e-05\n",
      "FPR:0.36, FNR:0.17, FP5.00, TN9.00, TP5.00, FN1.00\n",
      "FPR:0.50, FNR:0.50, FP6.00, TN6.00, TP4.00, FN4.00\n",
      "FPR:0.43, FNR:0.33, FP6.00, TN8.00, TP4.00, FN2.00\n",
      "FPR:0.50, FNR:0.50, FP8.00, TN8.00, TP2.00, FN2.00\n",
      "FPR:0.47, FNR:0.33, FP8.00, TN9.00, TP2.00, FN1.00\n",
      "Membership Inference Attack Score: 0.57 (+/- 0.07)\n"
     ]
    }
   ],
   "source": [
    "from sklearn.linear_model import LogisticRegression\n",
    "from sklearn.model_selection import cross_val_score\n",
    "from sklearn.model_selection import StratifiedShuffleSplit\n",
    "from sklearn.metrics import confusion_matrix\n",
    "import random\n",
    "\n",
    "def cm_score(estimator, X, y):\n",
    "    y_pred = estimator.predict(X)\n",
    "    cnf_matrix = confusion_matrix(y, y_pred)\n",
    "    \n",
    "    FP = cnf_matrix[0][1] \n",
    "    FN = cnf_matrix[1][0] \n",
    "    TP = cnf_matrix[0][0] \n",
    "    TN = cnf_matrix[1][1]\n",
    "\n",
    "\n",
    "    # Sensitivity, hit rate, recall, or true positive rate\n",
    "    TPR = TP/(TP+FN)\n",
    "    # Specificity or true negative rate\n",
    "    TNR = TN/(TN+FP) \n",
    "    # Precision or positive predictive value\n",
    "    PPV = TP/(TP+FP)\n",
    "    # Negative predictive value\n",
    "    NPV = TN/(TN+FN)\n",
    "    # Fall out or false positive rate\n",
    "    FPR = FP/(FP+TN)\n",
    "    # False negative rate\n",
    "    FNR = FN/(TP+FN)\n",
    "    # False discovery rate\n",
    "    FDR = FP/(TP+FP)\n",
    "\n",
    "    # Overall accuracy\n",
    "    ACC = (TP+TN)/(TP+FP+FN+TN)\n",
    "    print (f\"FPR:{FPR:.2f}, FNR:{FNR:.2f}, FP{FP:.2f}, TN{TN:.2f}, TP{TP:.2f}, FN{FN:.2f}\")\n",
    "    return ACC\n",
    "\n",
    "\n",
    "def evaluate_attack_model(sample_loss,\n",
    "                          members,\n",
    "                          n_splits = 5,\n",
    "                          random_state = None):\n",
    "  \"\"\"Computes the cross-validation score of a membership inference attack.\n",
    "  Args:\n",
    "    sample_loss : array_like of shape (n,).\n",
    "      objective function evaluated on n samples.\n",
    "    members : array_like of shape (n,),\n",
    "      whether a sample was used for training.\n",
    "    n_splits: int\n",
    "      number of splits to use in the cross-validation.\n",
    "    random_state: int, RandomState instance or None, default=None\n",
    "      random state to use in cross-validation splitting.\n",
    "  Returns:\n",
    "    score : array_like of size (n_splits,)\n",
    "  \"\"\"\n",
    "\n",
    "  unique_members = np.unique(members)\n",
    "  if not np.all(unique_members == np.array([0, 1])):\n",
    "    raise ValueError(\"members should only have 0 and 1s\")\n",
    "\n",
    "  attack_model = LogisticRegression()\n",
    "  cv = StratifiedShuffleSplit(\n",
    "      n_splits=n_splits, random_state=random_state)\n",
    "  return cross_val_score(attack_model, sample_loss, members, cv=cv, scoring=cm_score)\n",
    "\n",
    "def membership_inference_attack(model, t_loader, f_loader, seed):\n",
    "    import matplotlib.pyplot as plt\n",
    "    import seaborn as sns\n",
    "    \n",
    "\n",
    "    fgt_cls = list(np.unique(f_loader.dataset.targets))\n",
    "    indices = [i in fgt_cls for i in t_loader.dataset.targets]\n",
    "    t_loader.dataset.data = t_loader.dataset.data[indices]\n",
    "    t_loader.dataset.targets = t_loader.dataset.targets[indices]\n",
    "\n",
    "    \n",
    "    cr = nn.CrossEntropyLoss(reduction='none')\n",
    "    test_losses = []\n",
    "    forget_losses = []\n",
    "    model.eval()\n",
    "    mult = 0.5 if args.lossfn=='mse' else 1\n",
    "    dataloader = torch.utils.data.DataLoader(t_loader.dataset, batch_size=128, shuffle=False)\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*cr(output, target)\n",
    "        test_losses = test_losses + list(loss.cpu().detach().numpy())\n",
    "    del dataloader\n",
    "    dataloader = torch.utils.data.DataLoader(f_loader.dataset, batch_size=128, shuffle=False)\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*cr(output, target)\n",
    "        forget_losses = forget_losses + list(loss.cpu().detach().numpy())\n",
    "    del dataloader\n",
    "\n",
    "    np.random.seed(seed)\n",
    "    random.seed(seed)\n",
    "    if len(forget_losses) > len(test_losses):\n",
    "        forget_losses = list(random.sample(forget_losses, len(test_losses)))\n",
    "    elif len(test_losses) > len(forget_losses):\n",
    "        test_losses = list(random.sample(test_losses, len(forget_losses)))\n",
    "    \n",
    "  \n",
    "    sns.distplot(np.array(test_losses), kde=False, norm_hist=False, rug=False, label='test-loss', ax=plt)\n",
    "    sns.distplot(np.array(forget_losses), kde=False, norm_hist=False, rug=False, label='forget-loss', ax=plt)\n",
    "    plt.legend(prop={'size': 14})\n",
    "    plt.tick_params(labelsize=12)\n",
    "    plt.title(\"loss histograms\",size=18)\n",
    "    plt.xlabel('loss values',size=14)\n",
    "    plt.show()\n",
    "    print (np.max(test_losses), np.min(test_losses))\n",
    "    print (np.max(forget_losses), np.min(forget_losses))\n",
    "\n",
    "\n",
    "    test_labels = [0]*len(test_losses)\n",
    "    forget_labels = [1]*len(forget_losses)\n",
    "    features = np.array(test_losses + forget_losses).reshape(-1,1)\n",
    "    labels = np.array(test_labels + forget_labels).reshape(-1)\n",
    "    features = np.clip(features, -100, 100)\n",
    "    score = evaluate_attack_model(features, labels, n_splits=5, random_state=seed)\n",
    "\n",
    "    return score\n",
    "\n",
    "score = membership_inference_attack(model, test_loader_full, forget_loader, seed)\n",
    "print(f\"Membership Inference Attack Score: {score.mean():.2f} (+/- {score.std():.2f})\")"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 33,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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KWsb+fn5yM/Pl96rVCr9doyIiIieKiaXZAGAQqGo0LzevXujd+/e0vsuXbqgX79+aNmyJWbOnIkdO3bota6ytrFgwQLMnj271PlEREREgAleLnRzc9PZkpSRkQFAd+tTWfz8/NC5c2fExcXJtgHobhXLyMgocxvTpk1DZmam9EpMTCxXPERERPR0MLkkq2XLlrh48SIKCwtl08+ePQug+M7B8hJCwMzs0a5q1qFZ5+PbKWsbSqUSjo6OshcRERHR40wuyRo8eDCys7OxdetW2fTo6Gh4e3ujXbt25VpfQkICjhw5gvbt20vT6tWrh7Zt22LdunUoKiqSpsfFxeHy5csYMmRI5XaCiIiInnom1yerT58+eO655zBhwgSoVCo0atQIGzZswJ49e7Bu3TppjKxx48YhOjoa165dg6+vLwCgR48e6NKlC4KDg6WO7x9//DEUCgXmzJkj286iRYvw3HPPYdiwYZg4cSJSU1PxwQcfICgoCGPGjKn2/SYiIqLaxeSSLADYtm0bpk+fjpkzZyIjIwNNmzbFhg0bMGLECKlMUVERioqKIISQprVs2RLff/89lixZgry8PHh6eqJbt2748MMPERgYKNtGREQEdu3ahZkzZ2LAgAGwtbVF//79sXjxYo72TkRERJWmECWzFCo3lUoFJycnZGZmsn8WERFRDVEd398m1yeLiIiIqDZgkkVERERkAEyyiIiIiAyASRYRERGRATDJIiIiIjIAJllEREREBsAki4iIiMgAmGQRERERGQCTLCIiIiIDYJJFREREZABMsoiIiIgMgEkWERERkQEwySIiIiIyACZZRERERAbAJIuIiIjIAJhkERERERkAkywiIiIiA2CSRURERGQATLKIiIiIDIBJFhEREZEBMMkiIiIiMgAmWUREREQGwCSLiIiIyACYZBEREREZAJMsIiIiIgNgkkVERERkAEyyiIiIiAyASRYRERGRATDJIiIiIjIAk0yysrOzMWnSJHh7e8Pa2hqhoaHYuHHjE5fbtm0bRo4ciUaNGsHGxgZ+fn548cUXcfXqVa2yERERUCgUWq/evXsbYpeIiIjoKWNh7AB0GTJkCE6ePImFCxciMDAQ3333HUaOHAm1Wo1Ro0aVutyiRYtQp04dTJ8+HQEBAUhMTMT8+fPRqlUrxMXFoUWLFrLyAQEBWL9+vWyas7OzIXaJ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",
      "text/plain": [
       "<Figure size 640x480 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "from matplotlib import pyplot as plt\n",
    "indices = list(range(0,len(acc_rs)))\n",
    "plt.plot(indices, acc_rs, marker='*', alpha=1, label='retain-set')\n",
    "plt.plot(indices, acc_fs, marker='o', alpha=1, label='forget-set')\n",
    "plt.plot(indices, acc_vs, marker='.', alpha=1, label='valid-set')\n",
    "plt.legend(prop={'size': 14})\n",
    "plt.tick_params(labelsize=12)\n",
    "plt.title('SCRUB retain-, valid- and forget- set error',size=18)\n",
    "plt.xlabel('epoch',size=14)\n",
    "plt.ylabel('error',size=14)\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Original"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 34,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Original_Model_D_r -> Loss:0.003, Error:0.0\n",
      "Original_Model_D_f -> Loss:0.002, Error:0.0\n",
      "Original_Model_D_t -> Loss:0.066, Error:0.019\n"
     ]
    }
   ],
   "source": [
    "m_D_r_activations,m_D_r_predictions=activations_predictions(copy.deepcopy(model),copy.deepcopy(retain_loader),'Original_Model_D_r')\n",
    "m_D_f_activations,m_D_f_predictions=activations_predictions(copy.deepcopy(model),copy.deepcopy(forget_loader),'Original_Model_D_f')\n",
    "m_D_t_activations,m_D_t_predictions=activations_predictions(copy.deepcopy(model),copy.deepcopy(test_loader_full),'Original_Model_D_t')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### Retrain"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 35,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Retrain_Model_D_r -> Loss:0.003, Error:0.0\n",
      "Retrain_Model_D_f -> Loss:0.02, Error:0.0\n",
      "Retrain_Model_D_t -> Loss:0.067, Error:0.019\n"
     ]
    }
   ],
   "source": [
    "m0_D_r_activations,m0_D_r_predictions=activations_predictions(copy.deepcopy(model0),copy.deepcopy(retain_loader),'Retrain_Model_D_r')\n",
    "m0_D_f_activations,m0_D_f_predictions=activations_predictions(copy.deepcopy(model0),copy.deepcopy(forget_loader),'Retrain_Model_D_f')\n",
    "m0_D_t_activations,m0_D_t_predictions=activations_predictions(copy.deepcopy(model0),copy.deepcopy(test_loader_full),'Retrain_Model_D_t')"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "### SCRUB"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 36,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "SCRUB_D_r -> Loss:0.003, Error:0.0\n",
      "SCRUB_D_f -> Loss:0.001, Error:0.0\n",
      "SCRUB_D_t -> Loss:0.064, Error:0.017\n"
     ]
    }
   ],
   "source": [
    "ntk_D_r_activations,ntk_D_r_predictions=activations_predictions(copy.deepcopy(model_s),copy.deepcopy(retain_loader),'SCRUB_D_r')\n",
    "ntk_D_f_activations,ntk_D_f_predictions=activations_predictions(copy.deepcopy(model_s),copy.deepcopy(forget_loader),'SCRUB_D_f')\n",
    "ntk_D_t_activations,ntk_D_t_predictions=activations_predictions(copy.deepcopy(model_s),copy.deepcopy(test_loader_full),'SCRUB_D_t')"
   ]
  }
 ],
 "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
}
