{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "id": "f5375255-5b0f-4acd-82bf-50e4e9884abb",
   "metadata": {},
   "outputs": [],
   "source": [
    "#MODIFIED large_scale_unlearning.ipynb from https://github.com/meghdadk/SCRUB/blob/main/large_scale_unlearning.ipynb\n",
    "\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 numpy as np\n",
    "import random \n",
    "import copy\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "from typing import List\n",
    "import itertools\n",
    "#from tqdm.autonotebook import tqdm\n",
    "from tqdm import tqdm\n",
    "\n",
    "import pickle\n",
    "\n",
    "\n",
    "import models\n",
    "import datasets"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "id": "61ef342d-38a4-4188-8b66-07033d0c717f",
   "metadata": {},
   "outputs": [],
   "source": [
    "#from models import *\n",
    "#import models\n",
    "#from logger import *\n",
    "\n",
    "from SCRUB.helper.util import adjust_learning_rate as sgda_adjust_learning_rate\n",
    "from SCRUB.distiller_zoo.KD import DistillKL\n",
    "\n",
    "from SCRUB.helper.loops import train_distill, train_vanilla, validate\n",
    "from SCRUB.helper.pretrain import init\n",
    "from SCRUB.helper.util import AverageMeter"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "id": "2aae5db3-7a9e-4e35-b686-f8ceb21d5d95",
   "metadata": {},
   "outputs": [],
   "source": [
    "checkpoint_name = 'lacuna100binary128_resnetsmooth_1_0_forget_None_lr_0_01_bs_512_ls_ce_seed_1_scheduler_1'"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "id": "6baf8e6b-b057-4c23-bcbb-d7a8bf7d3f27",
   "metadata": {},
   "outputs": [],
   "source": [
    "with open(f'logs/{checkpoint_name}.pkl', 'rb') as f:\n",
    "    train_dict = pickle.load(f)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "id": "69185fd1-325a-4937-bc24-c2c336cb5fa4",
   "metadata": {},
   "outputs": [],
   "source": [
    "# eta = 0.01\n",
    "# n = 32000\n",
    "\n",
    "# batch_size = train_dict['batch_size']\n",
    "# Nepochs = train_dict['selected epoch']+1\n",
    "# Nbatches = math.ceil(n/batch_size)\n",
    "\n",
    "# T = Nepochs * Nbatches\n",
    "\n",
    "seed = 1"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "id": "1718e64f-5c40-458d-a97c-e9f2779e4530",
   "metadata": {},
   "outputs": [],
   "source": [
    "use_cuda = torch.cuda.is_available()\n",
    "device = torch.device(\"cuda:1\" if use_cuda else \"cpu\")\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "checkpoints= [0, 20, 50, 100, 150, 200, 250]\n",
    "checkpoints = [0, 20, 40, 50, 70, 120, 150, 170, 200,220, 250]\n",
    "#checkpointarray=( 40 70 120 170 220 )"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "id": "5f7435f1-8ad4-42c8-8c8b-4f6605c5f48d",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Forgetting these IDs: [5249, 3520]\n",
      "Number of training samples: 31353\n",
      "Number of validation samples: 7847\n",
      "Number of test samples: 10000\n",
      "Number of ood samples: 10000\n",
      "Number of train forget samples: 647\n",
      "Number of test forget samples: 200\n",
      "Number of test retain samples: 9800\n"
     ]
    }
   ],
   "source": [
    "dataroot = 'data/lacuna100binary128/'\n",
    "loaders = datasets.get_loaders_large('lacuna100binary128', num_ids_forget = 2, batch_size=512, seed=seed, root=dataroot, test=True)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "id": "5fa0e96b-68e9-407f-b4bc-f5f7256d54a7",
   "metadata": {},
   "outputs": [],
   "source": [
    "train_loader = loaders['train_loader']\n",
    "train_forget_loader = loaders['train_forget_loader']\n",
    "forget_loader = loaders['forget_loader']\n",
    "\n",
    "test_loader = loaders['test_loader']\n",
    "\n",
    "ood_loader = loaders['ood_loader']\n",
    "\n",
    "test_forget_loader = loaders['test_forget_loader']\n",
    "test_retain_loader = loaders['test_retain_loader']"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "id": "4bcedeed-d9cd-446f-85dd-53aa6c528edf",
   "metadata": {},
   "outputs": [],
   "source": [
    "model = models.get_model('resnetsmooth', num_classes=2, filters_percentage=1)\n",
    "model.to(device)\n",
    "\n",
    "og_name = checkpoint_name + '_selected'\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "id": "82971590-e3e1-4b73-af7e-9bfd01c45074",
   "metadata": {},
   "outputs": [],
   "source": [
    "retain_loader = train_forget_loader\n",
    "valid_loader = loaders['valid_loader'] "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "bffeb333-959b-4d2b-a9d9-1b7c40b10eb1",
   "metadata": {},
   "source": [
    "## Load checkpoints"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "id": "ca70dc06-c98f-4b4c-8c88-1334a2bd2bf7",
   "metadata": {},
   "outputs": [],
   "source": [
    "import copy\n",
    "model0 = copy.deepcopy(model)\n",
    "\n",
    "m_name = og_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 = full_retrain_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('checkpoints/' + m_name + '.pt', weights_only='True'))\n",
    "#model0.load_state_dict(torch.load('checkpoints/' + m0_name + '.pt', weights_only='True'))\n",
    "\n",
    "\n",
    "model.cuda()\n",
    "#model0.cuda()\n",
    "\n",
    "\n",
    "for p in model.parameters(): #what is the point of this\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",
   "id": "89dd0abc-5af7-4f88-a395-1f00daaa3f50",
   "metadata": {},
   "source": [
    "# SCRUB Forgetting"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "id": "4ccfd5c7-b69f-4283-b59a-cd9f129baa02",
   "metadata": {},
   "outputs": [],
   "source": [
    "from argparse import Namespace\n",
    "\n",
    "# start with an empty Namespace\n",
    "args = Namespace()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "id": "ea84a648-7375-40f4-96df-1c7946f93497",
   "metadata": {},
   "outputs": [],
   "source": [
    "#large-scale hyperparameters\n",
    "args.optim = 'sgd'\n",
    "args.gamma = 0.99\n",
    "args.alpha = 0.001\n",
    "args.beta = 0\n",
    "args.smoothing = 0.0\n",
    "args.msteps = 2\n",
    "args.clip = 0.2\n",
    "args.sstart = 10\n",
    "args.kd_T = 4\n",
    "args.distill = 'kd'\n",
    "\n",
    "args.sgda_batch_size = 128\n",
    "args.del_batch_size = 32\n",
    "args.sgda_epochs = 3\n",
    "args.sgda_epochs = 10\n",
    "args.sgda_learning_rate = 0.0005\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": 14,
   "id": "330fd3c3-1fdd-476c-a845-d65ce38d5373",
   "metadata": {},
   "outputs": [],
   "source": [
    "model_t = copy.deepcopy(teacher)\n",
    "model_s = copy.deepcopy(student)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "id": "2bf857b4-e94c-4e81-9d37-fbc2ca3c778d",
   "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 = torch.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 = torch.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 = torch.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": 16,
   "id": "ed1bff2d-ca1d-4474-942d-77f1bf02ea0d",
   "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"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 17,
   "id": "9f8d94c4-e1a2-4668-a00e-5891fa8c9bce",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "==> SCRUB unlearning ...\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/nfs/home/smg5501/.conda/envs/myenv/lib/python3.9/site-packages/torch/nn/_reduction.py:51: UserWarning: size_average and reduce args will be deprecated, please use reduction='sum' instead.\n",
      "  warnings.warn(warning.format(ret))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      " * Acc@1 98.145 \n",
      "maximize loss: -0.0000000251\t minimize loss: 0.05\t train_acc: 98.1452865600586\n",
      "==> SCRUB unlearning ...\n",
      " * Acc@1 99.073 \n",
      "maximize loss: -0.0033259138\t minimize loss: 0.04\t train_acc: 99.07263946533203\n",
      "==> SCRUB unlearning ...\n",
      " * Acc@1 98.609 \n",
      "maximize loss: 0.0000000000\t minimize loss: 0.04\t train_acc: 98.60896301269531\n",
      "==> SCRUB unlearning ...\n",
      " * Acc@1 98.918 \n",
      "maximize loss: 0.0000000000\t minimize loss: 0.04\t train_acc: 98.91808319091797\n",
      "==> SCRUB unlearning ...\n",
      " * Acc@1 99.227 \n",
      "maximize loss: 0.0000000000\t minimize loss: 0.04\t train_acc: 99.2271957397461\n",
      "==> SCRUB unlearning ...\n",
      " * Acc@1 99.382 \n",
      "maximize loss: 0.0000000000\t minimize loss: 0.04\t train_acc: 99.38175964355469\n",
      "==> SCRUB unlearning ...\n",
      " * Acc@1 99.382 \n",
      "maximize loss: 0.0000000000\t minimize loss: 0.04\t train_acc: 99.38175964355469\n",
      "==> SCRUB unlearning ...\n",
      " * Acc@1 99.382 \n",
      "maximize loss: 0.0000000000\t minimize loss: 0.04\t train_acc: 99.38175964355469\n",
      "==> SCRUB unlearning ...\n",
      " * Acc@1 99.382 \n",
      "maximize loss: 0.0000000000\t minimize loss: 0.04\t train_acc: 99.38175964355469\n",
      "==> SCRUB unlearning ...\n",
      " * Acc@1 99.382 \n",
      "maximize loss: 0.0000000000\t minimize loss: 0.04\t train_acc: 99.38175964355469\n"
     ]
    }
   ],
   "source": [
    "acc_rs = []\n",
    "acc_fs = []\n",
    "acc_ts = []\n",
    "acc_vs = []\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)\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, 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, None, criterion_list, optimizer, args, \"maximize\")\n",
    "    train_acc, train_loss = train_distill(epoch, retain_loader, module_list, None, criterion_list, optimizer, args, \"minimize\")\n",
    "\n",
    "    \n",
    "    print (\"maximize loss: {:.10f}\\t minimize loss: {:.2f}\\t train_acc: {}\".format(maximize_loss, train_loss, train_acc))\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, 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())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 18,
   "id": "bad1d517-0929-4efd-b97a-8eba73a30838",
   "metadata": {},
   "outputs": [],
   "source": [
    " torch.save(model_s.state_dict(), f\"other_baselines/SCRUB_{checkpoint_name}.pt\") "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 19,
   "id": "50012306-8128-45c0-9560-56ea0238f129",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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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",
   "id": "28150f0a-d5c5-4dcb-a065-19fa0094f869",
   "metadata": {},
   "source": [
    "# Fisher"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 21,
   "id": "05fdf366-461f-400a-a58f-cfe461beb35f",
   "metadata": {},
   "outputs": [],
   "source": [
    "model.load_state_dict(torch.load('checkpoints/' + og_name + '.pt', weights_only='True'))\n",
    "\n",
    "modelf = copy.deepcopy(model)\n",
    "#modelf0 = copy.deepcopy(model0)\n",
    "\n",
    "modelf.to(device)\n",
    "\n",
    "for p in itertools.chain(modelf.parameters()):\n",
    "    p.data0 = copy.deepcopy(p.data.clone())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 22,
   "id": "8ee5f2b9-3d79-4f13-83b0-5f45b6f38988",
   "metadata": {},
   "outputs": [],
   "source": [
    "def hessian(dataset, model, device):\n",
    "    model.eval()\n",
    "    train_loader = torch.utils.data.DataLoader(dataset, batch_size=1, shuffle=False)\n",
    "    loss_fn = nn.CrossEntropyLoss()\n",
    "\n",
    "    for p in model.parameters():\n",
    "        p.grad_acc = 0\n",
    "        p.grad2_acc = 0\n",
    "    \n",
    "    for data, orig_target, _ in tqdm(train_loader):\n",
    "        data, orig_target = data.to(device), orig_target.to(device)\n",
    "        output = model(data)\n",
    "        prob = torch.nn.functional.softmax(output, dim=-1).data\n",
    "\n",
    "        for y in range(output.shape[1]):\n",
    "            target = torch.empty_like(orig_target).fill_(y)\n",
    "            loss = loss_fn(output, target)\n",
    "            model.zero_grad()\n",
    "            loss.backward(retain_graph=True)\n",
    "            for p in model.parameters():\n",
    "                if p.requires_grad:\n",
    "                    p.grad_acc += (orig_target == target).float() * p.grad.data\n",
    "                    p.grad2_acc += prob[:, y] * p.grad.data.pow(2)\n",
    "    for p in model.parameters():\n",
    "        p.grad_acc /= len(train_loader)\n",
    "        p.grad2_acc /= len(train_loader)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 23,
   "id": "a3dc68ac-6b4e-4ced-9230-0d8d0abb34fe",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 647/647 [00:32<00:00, 19.66it/s]\n"
     ]
    }
   ],
   "source": [
    "hessian(retain_loader.dataset, modelf, device)\n",
    "#hessian(retain_loader.dataset, modelf0)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 24,
   "id": "6517cbf4-4c14-4c56-9686-bd35cd23b52b",
   "metadata": {},
   "outputs": [],
   "source": [
    "def get_mean_var(p, is_base_dist=False, alpha=3e-6, num_classes=2):\n",
    "    var = copy.deepcopy(1./(p.grad2_acc+1e-8))\n",
    "    var = var.clamp(max=1e3)\n",
    "    if p.size(0) == num_classes:\n",
    "        var = var.clamp(max=1e2)\n",
    "    var = alpha * var\n",
    "    \n",
    "    if p.ndim > 1:\n",
    "        var = var.mean(dim=1, keepdim=True).expand_as(p).clone()\n",
    "    if not is_base_dist:\n",
    "        mu = copy.deepcopy(p.data0.clone())\n",
    "    else:\n",
    "        mu = copy.deepcopy(p.data0.clone())\n",
    "    # if p.size(0) == num_classes and num_to_forget is None:\n",
    "    #     mu[class_to_forget] = 0\n",
    "    #     var[class_to_forget] = 0.0001\n",
    "    if p.size(0) == num_classes:\n",
    "        # Last layer\n",
    "        var *= 10\n",
    "    elif p.ndim == 1:\n",
    "        # BatchNorm\n",
    "        var *= 10\n",
    "#         var*=1\n",
    "    return mu, var\n",
    "\n",
    "def kl_divergence_fisher(mu0, var0, mu1, var1):\n",
    "    return ((mu1 - mu0).pow(2)/var0 + var1/var0 - torch.log(var1/var0) - 1).sum()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 25,
   "id": "1fec1fba-69e6-4730-a1ed-cedf0c378c3f",
   "metadata": {},
   "outputs": [],
   "source": [
    "alpha = 1e-8\n",
    "torch.manual_seed(seed)\n",
    "for i, p in enumerate(modelf.parameters()):\n",
    "    mu, var = get_mean_var(p, False, alpha=alpha)\n",
    "    p.data = mu + var.sqrt() * torch.empty_like(p.data0).normal_()\n",
    "\n",
    "# for i, p in enumerate(modelf0.parameters()):\n",
    "#     mu, var = get_mean_var(p, False, alpha=alpha)\n",
    "#     p.data = mu + var.sqrt() * torch.empty_like(p.data0).normal_()\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "52ccebf8-b029-4117-ad33-134cd4cf05e4",
   "metadata": {},
   "outputs": [],
   "source": []
  },
  {
   "cell_type": "code",
   "execution_count": 26,
   "id": "f70f3b2c-6ea3-4586-83d9-df8570ddd808",
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "tensor([3.7350e-06, 3.7350e-06], device='cuda:1')\n"
     ]
    }
   ],
   "source": [
    "print(var)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 27,
   "id": "4c456014-3088-4893-b0c0-2bb0d3fa8ff2",
   "metadata": {},
   "outputs": [],
   "source": [
    " torch.save(modelf.state_dict(), f\"other_baselines/Fisher_{checkpoint_name}.pt\") "
   ]
  },
  {
   "cell_type": "markdown",
   "id": "2e94a0c6-f5f0-464f-b54f-1f91dc4ef51a",
   "metadata": {},
   "source": [
    "# Fine Tune "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 52,
   "id": "befcd3f9-aec2-4ea9-9b7d-382b418c1bb0",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 10/10 [09:23<00:00, 56.37s/it]\n"
     ]
    }
   ],
   "source": [
    "\n",
    "import time\n",
    "\n",
    "from utils import *\n",
    "\n",
    "\n",
    "from main import run_epoch_simple\n",
    "\n",
    "def run_epoch_simple(model, data_loader, device, criterion=torch.nn.CrossEntropyLoss(), optimizer=None):\n",
    "    model.train()\n",
    "\n",
    "    with torch.set_grad_enabled(True):\n",
    "\n",
    "        for batch_idx, (data, target, identity) in enumerate(data_loader):\n",
    "            data, target = data.to(device), target.to(device)\n",
    "\n",
    "                \n",
    "            output = model(data)\n",
    "            loss = criterion(output, target) \n",
    "\n",
    "            optimizer.zero_grad() \n",
    "            loss.backward()\n",
    "\n",
    "            optimizer.step()    \n",
    "\n",
    "\n",
    "manual_seed(seed)\n",
    "\n",
    "\n",
    "\n",
    "#get model\n",
    "model.load_state_dict(torch.load('checkpoints/' + m_name + '.pt', weights_only='True'))\n",
    "model.to(device)\n",
    "\n",
    "optimizer = torch.optim.SGD(model.parameters(), lr=0.01, momentum=0, weight_decay=0)\n",
    "\n",
    "\n",
    "criterion = torch.nn.CrossEntropyLoss().to(device)\n",
    "\n",
    "train_time = 0\n",
    "\n",
    "\n",
    "epochs = 10\n",
    "\n",
    "for epoch in tqdm(range(epochs)):\n",
    "\n",
    "    t1 = time.time()\n",
    "\n",
    "    _ = run_epoch_simple(model, train_loader, device, criterion, optimizer)\n",
    "\n",
    "    t2 = time.time()\n",
    "    train_time += np.round(t2-t1,2)\n",
    "\n",
    "\n",
    "\n",
    "\n",
    "torch.save(model.state_dict(), f\"other_baselines/Finetune_{checkpoint_name}.pt\") \n",
    "\n",
    "\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "849f900d-536a-4823-9259-8d56f7ce3b9e",
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "myenv",
   "language": "python",
   "name": "myenv"
  },
  "language_info": {
   "codemirror_mode": {
    "name": "ipython",
    "version": 3
   },
   "file_extension": ".py",
   "mimetype": "text/x-python",
   "name": "python",
   "nbconvert_exporter": "python",
   "pygments_lexer": "ipython3",
   "version": "3.9.12"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
