{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "import argparse\n",
    "import copy\n",
    "import logging\n",
    "import os\n",
    "import time\n",
    "\n",
    "import pickle\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.nn.functional as F\n",
    "from apex import amp\n",
    "\n",
    "from preact_resnet import PreActResNet18\n",
    "from utils import (upper_limit, lower_limit, std, clamp, get_loaders,\n",
    "    attack_pgd, evaluate_pgd, evaluate_standard)\n",
    "\n",
    "logger = logging.getLogger(__name__)\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {
    "tags": []
   },
   "outputs": [],
   "source": [
    "def get_args():\n",
    "    parser = argparse.ArgumentParser()\n",
    "    parser.add_argument('--batch-size', default=128, type=int)\n",
    "    parser.add_argument('--data-dir', default='../../cifar-data', type=str)\n",
    "    parser.add_argument('--epochs', default=60, type=int)\n",
    "    parser.add_argument('--lr-schedule', default='cyclic', choices=['cyclic', 'multistep'])\n",
    "    parser.add_argument('--lr-min', default=0., type=float)\n",
    "    parser.add_argument('--lr-max', default=0.2, type=float)\n",
    "    parser.add_argument('--weight-decay', default=5e-4, type=float)\n",
    "    parser.add_argument('--momentum', default=0.9, type=float)\n",
    "    #parser.add_argument('--epsilon', default=8, type=int)\n",
    "    #parser.add_argument('--alpha', default=10, type=float, help='Step size')\n",
    "    parser.add_argument('--delta-init', default='random', choices=['zero', 'random', 'previous'],\n",
    "        help='Perturbation initialization method')\n",
    "    parser.add_argument('--out-dir', default='train_fgsm_output', type=str, help='Output directory')\n",
    "    parser.add_argument('--seed', default=0, type=int, help='Random seed')\n",
    "    parser.add_argument('--early-stop', action='store_true', help='Early stop if overfitting occurs')\n",
    "    parser.add_argument('--opt-level', default='O2', type=str, choices=['O0', 'O1', 'O2'],\n",
    "        help='O0 is FP32 training, O1 is Mixed Precision, and O2 is \"Almost FP16\" Mixed Precision')\n",
    "    parser.add_argument('--loss-scale', default='1.0', type=str, choices=['1.0', 'dynamic'],\n",
    "        help='If loss_scale is \"dynamic\", adaptively adjust the loss scale over time')\n",
    "    parser.add_argument('--master-weights', action='store_true',\n",
    "        help='Maintain FP32 master weights to accompany any FP16 model weights, not applicable for O1 opt level')\n",
    "    args, unknown = parser.parse_known_args()\n",
    "\n",
    "    return args"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {
    "tags": []
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Files already downloaded and verified\n",
      "Files already downloaded and verified\n",
      "train_idx_check =  1249038303\n",
      "test_idx_check =  4955575\n",
      "epsilon =  0\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/tmp/ipykernel_21488/2145530794.py:23: FutureWarning: You are using `torch.load` with `weights_only=False` (the current default value), which uses the default pickle module implicitly. It is possible to construct malicious pickle data which will execute arbitrary code during unpickling (See https://github.com/pytorch/pytorch/blob/main/SECURITY.md#untrusted-models for more details). In a future release, the default value for `weights_only` will be flipped to `True`. This limits the functions that could be executed during unpickling. Arbitrary objects will no longer be allowed to be loaded via this mode unless they are explicitly allowlisted by the user via `torch.serialization.add_safe_globals`. We recommend you start setting `weights_only=True` for any use case where you don't have full control of the loaded file. Please open an issue on GitHub for any issues related to this experimental feature.\n",
      "  state_dict = torch.load(os.path.join(args.out_dir, 'robust_model_epsilon_' + str(epsilon) + '.pth'))\n"
     ]
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "0 / 8\n",
      "1 / 8\n",
      "2 / 8\n",
      "3 / 8\n",
      "4 / 8\n",
      "5 / 8\n",
      "6 / 8\n",
      "7 / 8\n",
      "Test Loss \t Test Acc \t PGD Loss \t PGD Acc\n",
      "0.1978 \t 0.9330 \t 6.9078 \t 0.0000\n",
      "Total evaluation time: 0.4731 minutes\n",
      "epsilon =  2\n",
      "0 / 8\n",
      "1 / 8\n",
      "2 / 8\n",
      "3 / 8\n"
     ]
    },
    {
     "ename": "KeyboardInterrupt",
     "evalue": "",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m                         Traceback (most recent call last)",
      "Cell \u001b[0;32mIn[4], line 29\u001b[0m\n\u001b[1;32m     27\u001b[0m start_eval_time \u001b[38;5;241m=\u001b[39m time\u001b[38;5;241m.\u001b[39mtime()\n\u001b[1;32m     28\u001b[0m test_loss, test_acc \u001b[38;5;241m=\u001b[39m evaluate_standard(test_loader, model)\n\u001b[0;32m---> 29\u001b[0m pgd_loss, pgd_acc \u001b[38;5;241m=\u001b[39m \u001b[43mevaluate_pgd\u001b[49m\u001b[43m(\u001b[49m\u001b[43mtest_loader\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mmodel\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m50\u001b[39;49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m10\u001b[39;49m\u001b[43m)\u001b[49m\n\u001b[1;32m     30\u001b[0m test_loss_lst\u001b[38;5;241m.\u001b[39mappend(test_loss)\n\u001b[1;32m     31\u001b[0m test_acc_lst\u001b[38;5;241m.\u001b[39mappend(test_acc)\n",
      "File \u001b[0;32m~/SageMaker/FAST_CIFAR10/utils.py:112\u001b[0m, in \u001b[0;36mevaluate_pgd\u001b[0;34m(test_loader, model, attack_iters, restarts)\u001b[0m\n\u001b[1;32m    110\u001b[0m \u001b[38;5;28mprint\u001b[39m(i,\u001b[38;5;124m'\u001b[39m\u001b[38;5;124m/\u001b[39m\u001b[38;5;124m'\u001b[39m,\u001b[38;5;28mlen\u001b[39m(test_loader))\n\u001b[1;32m    111\u001b[0m X, y \u001b[38;5;241m=\u001b[39m X\u001b[38;5;241m.\u001b[39mcuda(), y\u001b[38;5;241m.\u001b[39mcuda()\n\u001b[0;32m--> 112\u001b[0m pgd_delta \u001b[38;5;241m=\u001b[39m \u001b[43mattack_pgd\u001b[49m\u001b[43m(\u001b[49m\u001b[43mmodel\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mX\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43my\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mepsilon\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43malpha\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mattack_iters\u001b[49m\u001b[43m,\u001b[49m\u001b[43m \u001b[49m\u001b[43mrestarts\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m    113\u001b[0m \u001b[38;5;28;01mwith\u001b[39;00m torch\u001b[38;5;241m.\u001b[39mno_grad():\n\u001b[1;32m    114\u001b[0m     output \u001b[38;5;241m=\u001b[39m model(X \u001b[38;5;241m+\u001b[39m pgd_delta)\n",
      "File \u001b[0;32m~/SageMaker/FAST_CIFAR10/utils.py:80\u001b[0m, in \u001b[0;36mattack_pgd\u001b[0;34m(model, X, y, epsilon, alpha, attack_iters, restarts, opt)\u001b[0m\n\u001b[1;32m     78\u001b[0m \u001b[38;5;28;01mfor\u001b[39;00m _ \u001b[38;5;129;01min\u001b[39;00m \u001b[38;5;28mrange\u001b[39m(attack_iters):\n\u001b[1;32m     79\u001b[0m     output \u001b[38;5;241m=\u001b[39m model(X \u001b[38;5;241m+\u001b[39m delta)\n\u001b[0;32m---> 80\u001b[0m     index \u001b[38;5;241m=\u001b[39m \u001b[43mtorch\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mwhere\u001b[49m\u001b[43m(\u001b[49m\u001b[43moutput\u001b[49m\u001b[38;5;241;43m.\u001b[39;49m\u001b[43mmax\u001b[49m\u001b[43m(\u001b[49m\u001b[38;5;241;43m1\u001b[39;49m\u001b[43m)\u001b[49m\u001b[43m[\u001b[49m\u001b[38;5;241;43m1\u001b[39;49m\u001b[43m]\u001b[49m\u001b[43m \u001b[49m\u001b[38;5;241;43m==\u001b[39;49m\u001b[43m \u001b[49m\u001b[43my\u001b[49m\u001b[43m)\u001b[49m\n\u001b[1;32m     81\u001b[0m     \u001b[38;5;28;01mif\u001b[39;00m \u001b[38;5;28mlen\u001b[39m(index[\u001b[38;5;241m0\u001b[39m]) \u001b[38;5;241m==\u001b[39m \u001b[38;5;241m0\u001b[39m:\n\u001b[1;32m     82\u001b[0m         \u001b[38;5;28;01mbreak\u001b[39;00m\n",
      "\u001b[0;31mKeyboardInterrupt\u001b[0m: "
     ]
    }
   ],
   "source": [
    "test_loss_lst = []\n",
    "test_acc_lst = []\n",
    "PGD_loss_lst = []\n",
    "PGD_acc_lst = []\n",
    "\n",
    "\n",
    "args = get_args()\n",
    "np.random.seed(args.seed)\n",
    "torch.manual_seed(args.seed)\n",
    "torch.cuda.manual_seed(args.seed)\n",
    "\n",
    "train_loader, test_loader = get_loaders(args.data_dir, args.batch_size)\n",
    "\n",
    "PE_test_loss_lst = []\n",
    "PE_test_acc_lst = []\n",
    "PE_pgd_loss_lst = []\n",
    "PE_pgd_acc_lst = []\n",
    "epsilon_lst = [2,4,6,8]\n",
    "for epsilon in epsilon_lst:\n",
    "    # Load the Pareto Efficient Models\n",
    "    print('epsilon = ', epsilon)\n",
    "    model = PreActResNet18().cuda()\n",
    "    state_dict = torch.load(os.path.join(args.out_dir, 'robust_model_epsilon_' + str(epsilon) + '.pth'))\n",
    "    model.load_state_dict(state_dict)\n",
    "\n",
    "    model.eval()\n",
    "    start_eval_time = time.time()\n",
    "    test_loss, test_acc = evaluate_standard(test_loader, model)\n",
    "    pgd_loss, pgd_acc = evaluate_pgd(test_loader, model, 50, 10)\n",
    "    test_loss_lst.append(test_loss)\n",
    "    test_acc_lst.append(test_acc)\n",
    "    PGD_loss_lst.append(pgd_loss)\n",
    "    PGD_acc_lst.append(pgd_acc)\n",
    "\n",
    "    end_eval_time = time.time()\n",
    "    print('Test Loss \\t Test Acc \\t PGD Loss \\t PGD Acc')\n",
    "    print('%.4f \\t %.4f \\t %.4f \\t %.4f' % (test_loss, test_acc, pgd_loss, pgd_acc))\n",
    "    print('Total evaluation time: %.4f minutes' % ((end_eval_time - start_eval_time)/60))\n",
    "\n",
    "    PE_test_loss_lst.append(test_loss)\n",
    "    PE_test_acc_lst.append(test_acc)\n",
    "    PE_pgd_loss_lst.append(pgd_loss)\n",
    "    PE_pgd_acc_lst.append(pgd_acc)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "conda_python3",
   "language": "python",
   "name": "conda_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.14"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 4
}
