{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 50,
   "id": "c2f843da",
   "metadata": {},
   "outputs": [],
   "source": [
    "from mpl_toolkits import axes_grid1\n",
    "import matplotlib.pyplot as plt\n",
    "%matplotlib inline\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "from torch.utils.data import DataLoader\n",
    "from tqdm import tqdm\n",
    "from functools import partial\n",
    "from warnings import warn\n",
    "from typing import List, Dict\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from torch import optim\n",
    "from torch.utils.data import DataLoader, RandomSampler\n",
    "from torchvision import models\n",
    "\n",
    "import os\n",
    "import time\n",
    "import numpy as np\n",
    "import copy\n",
    "import matplotlib.pyplot as plt\n",
    "\n",
    "\n",
    "from network import mnist_net, res_net, alex_net, cifar_net, vit_net, reg_net, generator\n",
    "from network.modules import get_resnet, get_generator, freeze, unfreeze, freeze_, unfreeze_, LARS\n",
    "from tools.miro_utils import *\n",
    "from tools.farmer import *\n",
    "import data_loader\n",
    "from main_base import evaluate\n",
    "from main_test import evaluate_digit, evaluate_image, evaluate_pacs, evaluate_officehome, evaluate_vlcs"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 65,
   "id": "d3c51207",
   "metadata": {},
   "outputs": [],
   "source": [
    "def flatten_params(model):\n",
    "  return model.state_dict()\n",
    "\n",
    "def lerp(lam, t1, t2):\n",
    "  t3 = copy.deepcopy(t2)\n",
    "  for p in t1: \n",
    "    t3[p] = (1 - lam) * t1[p] + lam * t2[p]\n",
    "  return t3\n",
    "\n",
    "def test(model, device, test_loader, softmax=False):\n",
    "    model.eval()\n",
    "    test_loss = 0\n",
    "    correct = 0\n",
    "    with torch.no_grad():\n",
    "        for data, target in test_loader:\n",
    "            data, target = data.to(device), target.to(device)\n",
    "            output = model(data)\n",
    "            if softmax:\n",
    "                output = F.log_softmax(output, dim=1)\n",
    "            test_loss += F.nll_loss(output, target, reduction='sum').item()  # sum up batch loss\n",
    "            pred = output.argmax(dim=1, keepdim=True)  # get the index of the max log-probability\n",
    "            correct += pred.eq(target.view_as(pred)).sum().item()\n",
    "\n",
    "    test_loss /= len(test_loader.dataset)\n",
    "    acc = 100. * correct / len(test_loader.dataset)\n",
    "    print('\\nAverage loss: {:.4f}, Accuracy: ({:.0f}%)\\n'.format(\n",
    "        test_loss, acc))\n",
    "    return test_loss, acc\n",
    "  \n",
    "\n",
    "\n",
    "def plot_interp_acc(lambdas, train_acc_interp_naive, test_acc_interp_naive, target_acc_interp_naive, train_acc_interp_clever, test_acc_interp_clever, target_acc_interp_clever, a, b):\n",
    "  fig = plt.figure(figsize=(4.5,3.0))\n",
    "  ax = fig.add_subplot(111)\n",
    "  #ax.plot(lambdas,train_acc_interp_naive,linestyle=\"dotted\",color=\"tab:blue\",alpha=0.5,linewidth=2,label=\"Train, ERM\")\n",
    "  ax.plot(lambdas,test_acc_interp_naive,linestyle=\"dashed\",color=\"tab:blue\",alpha=0.5,linewidth=2,label=\"Source, ERM\")\n",
    "  ax.plot(lambdas,target_acc_interp_naive,linestyle=\"solid\",color=\"tab:blue\",alpha=0.5,linewidth=2,label=\"Target, ERM\")\n",
    "  #ax.plot(lambdas,train_acc_interp_clever,linestyle=\"dotted\",color=\"tab:orange\",linewidth=2,label=\"Train, PEER\")\n",
    "  ax.plot(lambdas,test_acc_interp_clever,linestyle=\"dashed\",color=\"tab:orange\",linewidth=2,label=\"Source, PEER\")\n",
    "  ax.plot(lambdas,target_acc_interp_clever,linestyle=\"solid\",color=\"tab:orange\",linewidth=2,label=\"Target, PEER\")\n",
    "\n",
    "  ax.set_xlabel(\"$\\\\alpha$\")\n",
    "  ax.set_xticks([0, 0.5, 1])\n",
    "  ax.set_xticklabels([\"Snapshot $A$ \\n ($\\\\theta_{0}$)\",\"Interpolated ($\\\\alpha$=0.5)\", \"Snapshot $B$ \\n ($\\\\theta_{99}$)\"])\n",
    "  #ax.set_xticklabels([\"Snapshot $A$ \\n (Epoch {a})\".format(a=a),\"Interpolated ($\\lambda$=0.5)\", \"Snapshot $B$ \\n (Epoch {b})\".format(b=b)])\n",
    "  ax.set_ylabel(\"Accuracy\")\n",
    "  # TODO label x=0 tick as \\theta_1, and x=1 tick as \\theta_2\n",
    "  ax.legend(loc=\"upper right\", framealpha=0.5)\n",
    "  fig.tight_layout()\n",
    "  return fig"
   ]
  },
  {
   "cell_type": "markdown",
   "id": "3963bc37-b688-4046-92e1-248cba6cc348",
   "metadata": {},
   "source": [
    "# Anything vs Everything"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "id": "97b7da98-04fe-454e-b265-fc8ce05b0987",
   "metadata": {},
   "outputs": [],
   "source": [
    "random.seed(1111)\n",
    "torch.manual_seed(1111)\n",
    "np.random.seed(1111)\n",
    "trset = data_loader.load_mnist(split='train')\n",
    "teset = data_loader.load_mnist(split='test')\n",
    "targetset = data_loader.load_svhn(split='test')\n",
    "train_loader  = DataLoader(trset, batch_size=1024, num_workers=8, shuffle=False)\n",
    "test_loader = DataLoader(teset, batch_size=1024, num_workers=8, shuffle=False)\n",
    "target_loader = DataLoader(targetset, batch_size=1024, num_workers=8, shuffle=False)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 53,
   "id": "e1d9e860",
   "metadata": {},
   "outputs": [],
   "source": [
    "a= 0\n",
    "b= 99"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 54,
   "id": "025ba41c-50f5-41d5-8dc2-733e17d48e95",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 54,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model_a = mnist_net.ConvNet(128)\n",
    "ckpt= \"./saved-model/worship/mnist/cnn_img_custom_mdar_False_128_lmda0.0051_oracleTrue_self_digits_2.0_run0/{a}-best.pkl\".format(a=a) #with oracle\n",
    "saved_weight = torch.load(ckpt)\n",
    "model_a.load_state_dict(saved_weight['cls_net'])\n",
    "\n",
    "oracleckpt= \"./saved-model/worship/mnist/cnn_img_custom_mdar_False_128_lmda0.0051_oracleTrue_self_digits_2.0_run0/{b}-best.pkl\".format(b=b)\n",
    "model_b = mnist_net.ConvNet(128)\n",
    "o_saved_weight = torch.load(oracleckpt)\n",
    "model_b.load_state_dict(o_saved_weight['cls_net'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 55,
   "id": "c903989d-29e9-4a94-9ef5-1599881dd506",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|          | 0/25 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -17.6921, Accuracy: (98%)\n",
      "\n",
      "\n",
      "Average loss: -17.7698, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  4%|▍         | 1/25 [00:01<00:38,  1.61s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.9033, Accuracy: (56%)\n",
      "\n",
      "\n",
      "Average loss: -19.2819, Accuracy: (98%)\n",
      "\n",
      "\n",
      "Average loss: -19.3734, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  8%|▊         | 2/25 [00:03<00:39,  1.70s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.8213, Accuracy: (57%)\n",
      "\n",
      "\n",
      "Average loss: -20.6276, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -20.7280, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 12%|█▏        | 3/25 [00:05<00:38,  1.74s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.6872, Accuracy: (58%)\n",
      "\n",
      "\n",
      "Average loss: -21.7757, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -21.8786, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 16%|█▌        | 4/25 [00:07<00:38,  1.82s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.5512, Accuracy: (58%)\n",
      "\n",
      "\n",
      "Average loss: -22.7692, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -22.8721, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|██        | 5/25 [00:09<00:38,  1.90s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.4251, Accuracy: (58%)\n",
      "\n",
      "\n",
      "Average loss: -23.6243, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -23.7250, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 24%|██▍       | 6/25 [00:10<00:34,  1.83s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.3126, Accuracy: (58%)\n",
      "\n",
      "\n",
      "Average loss: -24.3401, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -24.4377, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 28%|██▊       | 7/25 [00:12<00:33,  1.87s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.2161, Accuracy: (57%)\n",
      "\n",
      "\n",
      "Average loss: -24.9074, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -25.0028, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 32%|███▏      | 8/25 [00:14<00:30,  1.80s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.1343, Accuracy: (57%)\n",
      "\n",
      "\n",
      "Average loss: -25.3148, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -25.4117, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 36%|███▌      | 9/25 [00:16<00:28,  1.77s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.0645, Accuracy: (57%)\n",
      "\n",
      "\n",
      "Average loss: -25.5531, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -25.6554, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|████      | 10/25 [00:17<00:26,  1.78s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.0026, Accuracy: (57%)\n",
      "\n",
      "\n",
      "Average loss: -25.6146, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -25.7268, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 44%|████▍     | 11/25 [00:19<00:25,  1.81s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.9419, Accuracy: (58%)\n",
      "\n",
      "\n",
      "Average loss: -25.4919, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -25.6169, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 48%|████▊     | 12/25 [00:21<00:24,  1.85s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.8748, Accuracy: (58%)\n",
      "\n",
      "\n",
      "Average loss: -25.1752, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -25.3177, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 52%|█████▏    | 13/25 [00:23<00:21,  1.81s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.7995, Accuracy: (59%)\n",
      "\n",
      "\n",
      "Average loss: -24.6686, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -24.8328, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 56%|█████▌    | 14/25 [00:25<00:19,  1.77s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.7234, Accuracy: (59%)\n",
      "\n",
      "\n",
      "Average loss: -23.9853, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -24.1731, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|██████    | 15/25 [00:26<00:17,  1.77s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.6450, Accuracy: (60%)\n",
      "\n",
      "\n",
      "Average loss: -23.1451, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -23.3633, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 64%|██████▍   | 16/25 [00:29<00:16,  1.88s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.5625, Accuracy: (60%)\n",
      "\n",
      "\n",
      "Average loss: -22.1757, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -22.4269, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 68%|██████▊   | 17/25 [00:30<00:14,  1.87s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.4809, Accuracy: (61%)\n",
      "\n",
      "\n",
      "Average loss: -21.1055, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -21.3933, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 72%|███████▏  | 18/25 [00:32<00:12,  1.80s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.3980, Accuracy: (61%)\n",
      "\n",
      "\n",
      "Average loss: -19.9512, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -20.2754, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 76%|███████▌  | 19/25 [00:34<00:10,  1.72s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.3074, Accuracy: (62%)\n",
      "\n",
      "\n",
      "Average loss: -18.7477, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -19.1052, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|████████  | 20/25 [00:35<00:08,  1.70s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.2121, Accuracy: (63%)\n",
      "\n",
      "\n",
      "Average loss: -17.5462, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -17.9348, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 84%|████████▍ | 21/25 [00:37<00:06,  1.75s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.1309, Accuracy: (63%)\n",
      "\n",
      "\n",
      "Average loss: -16.3781, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -16.7926, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 88%|████████▊ | 22/25 [00:39<00:05,  1.87s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.0852, Accuracy: (64%)\n",
      "\n",
      "\n",
      "Average loss: -15.3003, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -15.7368, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 92%|█████████▏| 23/25 [00:41<00:03,  1.85s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.0715, Accuracy: (65%)\n",
      "\n",
      "\n",
      "Average loss: -14.3005, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -14.7561, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 96%|█████████▌| 24/25 [00:43<00:01,  1.77s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.0847, Accuracy: (66%)\n",
      "\n",
      "\n",
      "Average loss: -13.3668, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -13.8410, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 25/25 [00:45<00:00,  1.80s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.0696, Accuracy: (66%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "lambdas = torch.linspace(0, 1, steps=25)\n",
    "\n",
    "test_acc_interp_clever = []\n",
    "train_acc_interp_clever = []\n",
    "target_acc_interp_clever = []\n",
    "# naive\n",
    "model_a_dict = copy.deepcopy(model_a.state_dict())\n",
    "model_b_dict = copy.deepcopy(model_b.state_dict())\n",
    "for lam in tqdm(lambdas):\n",
    "    clever_p = lerp(lam, model_a_dict, model_b_dict)\n",
    "    model_b.load_state_dict(clever_p)\n",
    "    test_loss, acc = test(model_b.cuda(), 'cuda', test_loader)\n",
    "    test_acc_interp_clever.append(acc)\n",
    "    train_loss, acc = test(model_b.cuda(), 'cuda', train_loader)\n",
    "    train_acc_interp_clever.append(acc)\n",
    "    target_loss, acc = test(model_b.cuda(), 'cuda', target_loader)\n",
    "    target_acc_interp_clever.append(acc)   "
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 56,
   "id": "45e22641-97e0-4c0f-ad48-32d475669e1a",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<All keys matched successfully>"
      ]
     },
     "execution_count": 56,
     "metadata": {},
     "output_type": "execute_result"
    }
   ],
   "source": [
    "model_a = mnist_net.ConvNet(128)\n",
    "ckpt= \"./worship/mnist/cnn_img_custom_mdar_False_128_lmda0.0051_oracleFalse_self_digits_2.0_run0/{a}-best.pkl\".format(a=a) \n",
    "saved_weight = torch.load(ckpt)\n",
    "model_a.load_state_dict(saved_weight['cls_net'])\n",
    "\n",
    "oracleckpt= \"./worship/mnist/cnn_img_custom_mdar_False_128_lmda0.0051_oracleFalse_self_digits_2.0_run0/{b}-best.pkl\".format(b=b)\n",
    "model_b = mnist_net.ConvNet(128)\n",
    "o_saved_weight = torch.load(oracleckpt)\n",
    "model_b.load_state_dict(o_saved_weight['cls_net'])"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 57,
   "id": "6fd1e232-244d-43f0-b6af-e01ebb33ef75",
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  0%|          | 0/25 [00:00<?, ?it/s]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -21.3582, Accuracy: (98%)\n",
      "\n",
      "\n",
      "Average loss: -21.4561, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  4%|▍         | 1/25 [00:01<00:43,  1.82s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.7884, Accuracy: (51%)\n",
      "\n",
      "\n",
      "Average loss: -25.6294, Accuracy: (98%)\n",
      "\n",
      "\n",
      "Average loss: -25.7791, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "  8%|▊         | 2/25 [00:03<00:43,  1.89s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.8647, Accuracy: (53%)\n",
      "\n",
      "\n",
      "Average loss: -29.4515, Accuracy: (98%)\n",
      "\n",
      "\n",
      "Average loss: -29.6475, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 12%|█▏        | 3/25 [00:05<00:40,  1.85s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.8204, Accuracy: (54%)\n",
      "\n",
      "\n",
      "Average loss: -32.8760, Accuracy: (98%)\n",
      "\n",
      "\n",
      "Average loss: -33.1068, Accuracy: (99%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 16%|█▌        | 4/25 [00:07<00:40,  1.91s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.6928, Accuracy: (55%)\n",
      "\n",
      "\n",
      "Average loss: -35.9023, Accuracy: (98%)\n",
      "\n",
      "\n",
      "Average loss: -36.1594, Accuracy: (99%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 20%|██        | 5/25 [00:09<00:37,  1.86s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.5094, Accuracy: (55%)\n",
      "\n",
      "\n",
      "Average loss: -38.4871, Accuracy: (98%)\n",
      "\n",
      "\n",
      "Average loss: -38.7606, Accuracy: (99%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 24%|██▍       | 6/25 [00:11<00:35,  1.87s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.2997, Accuracy: (54%)\n",
      "\n",
      "\n",
      "Average loss: -40.5755, Accuracy: (98%)\n",
      "\n",
      "\n",
      "Average loss: -40.8546, Accuracy: (99%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 28%|██▊       | 7/25 [00:12<00:32,  1.82s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -1.0946, Accuracy: (53%)\n",
      "\n",
      "\n",
      "Average loss: -42.1556, Accuracy: (98%)\n",
      "\n",
      "\n",
      "Average loss: -42.4310, Accuracy: (99%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 32%|███▏      | 8/25 [00:14<00:29,  1.73s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.9205, Accuracy: (53%)\n",
      "\n",
      "\n",
      "Average loss: -43.3064, Accuracy: (98%)\n",
      "\n",
      "\n",
      "Average loss: -43.5772, Accuracy: (99%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 36%|███▌      | 9/25 [00:16<00:27,  1.72s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.7911, Accuracy: (52%)\n",
      "\n",
      "\n",
      "Average loss: -44.1722, Accuracy: (98%)\n",
      "\n",
      "\n",
      "Average loss: -44.4457, Accuracy: (99%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 40%|████      | 10/25 [00:18<00:27,  1.82s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.7102, Accuracy: (51%)\n",
      "\n",
      "\n",
      "Average loss: -44.9220, Accuracy: (98%)\n",
      "\n",
      "\n",
      "Average loss: -45.2093, Accuracy: (99%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 44%|████▍     | 11/25 [00:20<00:25,  1.81s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.6721, Accuracy: (51%)\n",
      "\n",
      "\n",
      "Average loss: -45.6955, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -46.0129, Accuracy: (99%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 48%|████▊     | 12/25 [00:21<00:23,  1.81s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.6597, Accuracy: (51%)\n",
      "\n",
      "\n",
      "Average loss: -46.5251, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -46.8944, Accuracy: (99%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 52%|█████▏    | 13/25 [00:23<00:21,  1.81s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.6501, Accuracy: (51%)\n",
      "\n",
      "\n",
      "Average loss: -47.3008, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -47.7469, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 56%|█████▌    | 14/25 [00:25<00:20,  1.90s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.6246, Accuracy: (51%)\n",
      "\n",
      "\n",
      "Average loss: -47.7664, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -48.3035, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 60%|██████    | 15/25 [00:27<00:19,  1.93s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.5846, Accuracy: (52%)\n",
      "\n",
      "\n",
      "Average loss: -47.5805, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -48.2153, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 64%|██████▍   | 16/25 [00:29<00:17,  1.90s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.5276, Accuracy: (54%)\n",
      "\n",
      "\n",
      "Average loss: -46.4454, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -47.1597, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 68%|██████▊   | 17/25 [00:31<00:14,  1.86s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.4475, Accuracy: (56%)\n",
      "\n",
      "\n",
      "Average loss: -44.2600, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -45.0317, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 72%|███████▏  | 18/25 [00:33<00:13,  1.88s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.3429, Accuracy: (58%)\n",
      "\n",
      "\n",
      "Average loss: -41.1781, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -41.9785, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 76%|███████▌  | 19/25 [00:35<00:11,  1.98s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.2261, Accuracy: (59%)\n",
      "\n",
      "\n",
      "Average loss: -37.5117, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -38.3294, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 80%|████████  | 20/25 [00:37<00:10,  2.01s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: -0.1057, Accuracy: (61%)\n",
      "\n",
      "\n",
      "Average loss: -33.6048, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -34.4368, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 84%|████████▍ | 21/25 [00:39<00:07,  1.92s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: 0.0180, Accuracy: (62%)\n",
      "\n",
      "\n",
      "Average loss: -29.8084, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -30.6508, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 88%|████████▊ | 22/25 [00:41<00:05,  1.89s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: 0.1285, Accuracy: (64%)\n",
      "\n",
      "\n",
      "Average loss: -26.3929, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -27.2288, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 92%|█████████▏| 23/25 [00:43<00:03,  1.95s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: 0.1760, Accuracy: (65%)\n",
      "\n",
      "\n",
      "Average loss: -23.5931, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -24.4119, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      " 96%|█████████▌| 24/25 [00:45<00:01,  1.96s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: 0.1242, Accuracy: (66%)\n",
      "\n",
      "\n",
      "Average loss: -21.4336, Accuracy: (99%)\n",
      "\n",
      "\n",
      "Average loss: -22.2453, Accuracy: (100%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "100%|██████████| 25/25 [00:47<00:00,  1.89s/it]"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "\n",
      "Average loss: 0.0352, Accuracy: (65%)\n",
      "\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "\n"
     ]
    }
   ],
   "source": [
    "lambdas = torch.linspace(0, 1, steps=25)\n",
    "\n",
    "test_acc_interp_naive = []\n",
    "train_acc_interp_naive = []\n",
    "target_acc_interp_naive= []\n",
    "# naive\n",
    "model_a_dict = copy.deepcopy(model_a.state_dict())\n",
    "model_b_dict = copy.deepcopy(model_b.state_dict())\n",
    "for lam in tqdm(lambdas):\n",
    "    naive_p = lerp(lam, model_a_dict, model_b_dict)\n",
    "    model_b.load_state_dict(naive_p)\n",
    "    test_loss, acc = test(model_b.cuda(), 'cuda', test_loader)\n",
    "    test_acc_interp_naive.append(acc)\n",
    "    train_loss, acc = test(model_b.cuda(), 'cuda', train_loader)\n",
    "    train_acc_interp_naive.append(acc)\n",
    "    target_loss, acc = test(model_b.cuda(), 'cuda', target_loader)\n",
    "    target_acc_interp_naive.append(acc)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 66,
   "id": "eef7382f",
   "metadata": {},
   "outputs": [
    {
     "data": {
      "image/png": "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",
      "text/plain": [
       "<Figure size 450x300 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "fig = plot_interp_acc(lambdas, train_acc_interp_naive, test_acc_interp_naive, target_acc_interp_naive, train_acc_interp_clever, test_acc_interp_clever, target_acc_interp_clever, a, b)\n",
    "plt.savefig(f\"./lmc/test.pdf\", dpi=300)"
   ]
  }
 ],
 "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.12.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 5
}
