{
 "cells": [
  {
   "cell_type": "code",
   "execution_count": 1,
   "metadata": {},
   "outputs": [],
   "source": [
    "# ==================================\n",
    "# Experiment 2\n",
    "# ==================================\n",
    "\n",
    "# demonstration to find the maximal l_p ball at a point x_0, within which,\n",
    "# the class label of a random classifier is equal to C(x_0).\n",
    "# network is trained to classify random points in R^2. The effects of l_1\n",
    "# regularization in training are demonstrated."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 2,
   "metadata": {},
   "outputs": [
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/jordanm/.virtualenvs/myvenv/lib/python3.7/site-packages/matplotlib/figure.py:98: MatplotlibDeprecationWarning: \n",
      "Adding an axes using the same arguments as a previous axes currently reuses the earlier instance.  In a future version, a new instance will always be created and returned.  Meanwhile, this warning can be suppressed, and the future behavior ensured, by passing a unique label to each axes instance.\n",
      "  \"Adding an axes using the same arguments as a previous axes \"\n"
     ]
    }
   ],
   "source": [
    "# =====================\n",
    "# Imports\n",
    "# =====================\n",
    "\n",
    "from geocert import compute_boundary_batch, batch_GeoCert, incremental_GeoCert\n",
    "from plnn import PLNN\n",
    "from _polytope_ import Polytope, from_polytope_dict\n",
    "import utilities as utils\n",
    "import os\n",
    "import matplotlib.pyplot as plt\n",
    "import numpy as np\n",
    "import torch\n",
    "import torch.nn as nn\n",
    "import torch.optim as optim\n",
    "from torch.autograd import Variable\n",
    "from torchvision import datasets, transforms\n",
    "# from convex_adversarial import robust_loss"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "===============Generating Training Points============\n",
      "===============Points Generated============\n"
     ]
    }
   ],
   "source": [
    "# apply incremental geocert to a normal and l1-regularized classifier. Finds maximal l_p balls\n",
    "# for random points in R^2.\n",
    "\n",
    "# ==================================\n",
    "# Generate Training Points\n",
    "# ==================================\n",
    "\n",
    "print('===============Generating Training Points============')\n",
    "# random points at least 2r apart\n",
    "m = 12\n",
    "np.random.seed(3)\n",
    "x = [np.random.uniform(size=(2))]\n",
    "r = 0.1\n",
    "while(len(x) < m):\n",
    "    p = np.random.uniform(size=(2))\n",
    "    if min(np.abs(p-a).sum() for a in x) > 2*r:\n",
    "        x.append(p)\n",
    "\n",
    "epsilon = r/2\n",
    "\n",
    "X = torch.Tensor(np.array(x))\n",
    "torch.manual_seed(1)\n",
    "y = (torch.rand(m)+0.5).long()\n",
    "\n",
    "print('===============Points Generated============')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "===============Initializing Network============\n",
      "Sequential(\n",
      "  (1): Linear(in_features=2, out_features=8, bias=True)\n",
      "  (2): ReLU()\n",
      "  (3): Linear(in_features=8, out_features=2, bias=True)\n",
      "  (4): ReLU()\n",
      "  (5): Linear(in_features=2, out_features=2, bias=True)\n",
      ")\n",
      "===============Training Network============\n",
      "error: tensor(0.5833)\n",
      "error: tensor(0.5833)\n",
      "error: tensor(0.4167)\n",
      "error: tensor(0.4167)\n",
      "error: tensor(0.4167)\n",
      "error: tensor(0.3333)\n",
      "error: tensor(0.2500)\n",
      "error: tensor(0.2500)\n",
      "error: tensor(0.2500)\n",
      "error: tensor(0.2500)\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/Users/jordanm/.virtualenvs/myvenv/lib/python3.7/site-packages/torch/tensor.py:263: UserWarning: non-inplace resize is deprecated\n",
      "  warnings.warn(\"non-inplace resize is deprecated\")\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x576 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "===============Collecting Polytopes============\n",
      "number of polytopes: 12\n",
      "===============Finding Classification Boundary Facets============\n",
      "an unbounded polytope was plotted imperfectly\n"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "===============Finding Projection============\n",
      "lp_norm:  l_2\n",
      "tensor([[0.5508],\n",
      "        [0.7081]])\n",
      "from point: \n",
      "tensor([[0.5508],\n",
      "        [0.7081]])\n",
      "/Users/jordanm/grad/geometric-certificates\n",
      "---Initial Polytope---\n",
      "num facets:  4\n",
      "---Opening New Polytope---\n",
      "num facets:  3\n",
      "an unbounded polytope was plotted imperfectly\n",
      "an unbounded facet was plotted imperfectly\n",
      "an unbounded facet was plotted imperfectly\n",
      "an unbounded facet was plotted imperfectly\n"
     ]
    },
    {
     "ename": "FileNotFoundError",
     "evalue": "[Errno 2] No such file or directory: '/Users/jordanm/grad/geometric-certificates/plots/incremental_geocert/0.png'",
     "output_type": "error",
     "traceback": [
      "\u001b[0;31m---------------------------------------------------------------------------\u001b[0m",
      "\u001b[0;31mFileNotFoundError\u001b[0m                         Traceback (most recent call last)",
      "\u001b[0;32m<ipython-input-4-5d2786873730>\u001b[0m in \u001b[0;36m<module>\u001b[0;34m\u001b[0m\n\u001b[1;32m    113\u001b[0m     \u001b[0mplot_dir\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mcwd\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0;34m'/plots/incremental_geocert/'\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    114\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 115\u001b[0;31m     \u001b[0mt\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mincremental_GeoCert\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mlp_norm\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mnetwork\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mx_0\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0max\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mplot_dir\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    116\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    117\u001b[0m     \u001b[0mprint\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m'the final projection value:'\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mt\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/grad/geometric-certificates/geocert.py\u001b[0m in \u001b[0;36mincremental_GeoCert\u001b[0;34m(lp_norm, net, x, ax, plot_dir, n_colors)\u001b[0m\n\u001b[1;32m    183\u001b[0m         \u001b[0;32mif\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mindex\u001b[0m \u001b[0;34m%\u001b[0m \u001b[0;36m1\u001b[0m \u001b[0;34m==\u001b[0m \u001b[0;36m0\u001b[0m \u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    184\u001b[0m             geocert_plot_step(lp_norm, seen_to_polytope_map, pq, pop_el.lp_dist,\n\u001b[0;32m--> 185\u001b[0;31m                               x, plot_dir, n_colors, iter=index)\n\u001b[0m\u001b[1;32m    186\u001b[0m         \u001b[0mindex\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mindex\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0;36m1\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    187\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/grad/geometric-certificates/geocert.py\u001b[0m in \u001b[0;36mgeocert_plot_step\u001b[0;34m(lp_norm, seen_to_polytope_map, facet_heap_elems, t, x, plot_dir, n_colors, ax, iter)\u001b[0m\n\u001b[1;32m    276\u001b[0m         \u001b[0mplt\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mylim\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m-\u001b[0m\u001b[0mxylim\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mxylim\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    277\u001b[0m     \u001b[0mfilename\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mplot_dir\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0mstr\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0miter\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;34m+\u001b[0m \u001b[0;34m'.png'\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 278\u001b[0;31m     \u001b[0mplt\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0msavefig\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfilename\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    279\u001b[0m     \u001b[0mplt\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mclose\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    280\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.virtualenvs/myvenv/lib/python3.7/site-packages/matplotlib/pyplot.py\u001b[0m in \u001b[0;36msavefig\u001b[0;34m(*args, **kwargs)\u001b[0m\n\u001b[1;32m    686\u001b[0m \u001b[0;32mdef\u001b[0m \u001b[0msavefig\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    687\u001b[0m     \u001b[0mfig\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mgcf\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 688\u001b[0;31m     \u001b[0mres\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mfig\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0msavefig\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m*\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    689\u001b[0m     \u001b[0mfig\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mcanvas\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdraw_idle\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m)\u001b[0m   \u001b[0;31m# need this if 'transparent=True' to reset colors\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    690\u001b[0m     \u001b[0;32mreturn\u001b[0m \u001b[0mres\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.virtualenvs/myvenv/lib/python3.7/site-packages/matplotlib/figure.py\u001b[0m in \u001b[0;36msavefig\u001b[0;34m(self, fname, frameon, transparent, **kwargs)\u001b[0m\n\u001b[1;32m   2095\u001b[0m             \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mset_frameon\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mframeon\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2096\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 2097\u001b[0;31m         \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mcanvas\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mprint_figure\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfname\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m**\u001b[0m\u001b[0mkwargs\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m   2098\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2099\u001b[0m         \u001b[0;32mif\u001b[0m \u001b[0mframeon\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.virtualenvs/myvenv/lib/python3.7/site-packages/matplotlib/backend_bases.py\u001b[0m in \u001b[0;36mprint_figure\u001b[0;34m(self, filename, dpi, facecolor, edgecolor, orientation, format, bbox_inches, **kwargs)\u001b[0m\n\u001b[1;32m   2073\u001b[0m                     \u001b[0morientation\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0morientation\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2074\u001b[0m                     \u001b[0mbbox_inches_restore\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0m_bbox_inches_restore\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m-> 2075\u001b[0;31m                     **kwargs)\n\u001b[0m\u001b[1;32m   2076\u001b[0m             \u001b[0;32mfinally\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m   2077\u001b[0m                 \u001b[0;32mif\u001b[0m \u001b[0mbbox_inches\u001b[0m \u001b[0;32mand\u001b[0m \u001b[0mrestore_bbox\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.virtualenvs/myvenv/lib/python3.7/site-packages/matplotlib/backends/backend_agg.py\u001b[0m in \u001b[0;36mprint_png\u001b[0;34m(self, filename_or_obj, *args, **kwargs)\u001b[0m\n\u001b[1;32m    519\u001b[0m \u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    520\u001b[0m         \u001b[0;32mwith\u001b[0m \u001b[0mcbook\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0m_setattr_cm\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mrenderer\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mdpi\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mfigure\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mdpi\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m,\u001b[0m\u001b[0;31m \u001b[0m\u001b[0;31m\\\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 521\u001b[0;31m                 \u001b[0mcbook\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mopen_file_cm\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfilename_or_obj\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m\"wb\"\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mas\u001b[0m \u001b[0mfh\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    522\u001b[0m             _png.write_png(renderer._renderer, fh,\n\u001b[1;32m    523\u001b[0m                             self.figure.dpi, metadata=metadata)\n",
      "\u001b[0;32m/usr/local/Cellar/python/3.7.2_2/Frameworks/Python.framework/Versions/3.7/lib/python3.7/contextlib.py\u001b[0m in \u001b[0;36m__enter__\u001b[0;34m(self)\u001b[0m\n\u001b[1;32m    110\u001b[0m         \u001b[0;32mdel\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0margs\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mkwds\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mfunc\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    111\u001b[0m         \u001b[0;32mtry\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 112\u001b[0;31m             \u001b[0;32mreturn\u001b[0m \u001b[0mnext\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mself\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mgen\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    113\u001b[0m         \u001b[0;32mexcept\u001b[0m \u001b[0mStopIteration\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    114\u001b[0m             \u001b[0;32mraise\u001b[0m \u001b[0mRuntimeError\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0;34m\"generator didn't yield\"\u001b[0m\u001b[0;34m)\u001b[0m \u001b[0;32mfrom\u001b[0m \u001b[0;32mNone\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.virtualenvs/myvenv/lib/python3.7/site-packages/matplotlib/cbook/__init__.py\u001b[0m in \u001b[0;36mopen_file_cm\u001b[0;34m(path_or_file, mode, encoding)\u001b[0m\n\u001b[1;32m    405\u001b[0m \u001b[0;32mdef\u001b[0m \u001b[0mopen_file_cm\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mpath_or_file\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmode\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;34m\"r\"\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mencoding\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0;32mNone\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    406\u001b[0m     \u001b[0;34mr\"\"\"Pass through file objects and context-manage `.PathLike`\\s.\"\"\"\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 407\u001b[0;31m     \u001b[0mfh\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mopened\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mto_filehandle\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mpath_or_file\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mmode\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;32mTrue\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mencoding\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    408\u001b[0m     \u001b[0;32mif\u001b[0m \u001b[0mopened\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    409\u001b[0m         \u001b[0;32mwith\u001b[0m \u001b[0mfh\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;32m~/.virtualenvs/myvenv/lib/python3.7/site-packages/matplotlib/cbook/__init__.py\u001b[0m in \u001b[0;36mto_filehandle\u001b[0;34m(fname, flag, return_opened, encoding)\u001b[0m\n\u001b[1;32m    390\u001b[0m             \u001b[0mfh\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mbz2\u001b[0m\u001b[0;34m.\u001b[0m\u001b[0mBZ2File\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfname\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mflag\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    391\u001b[0m         \u001b[0;32melse\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0;32m--> 392\u001b[0;31m             \u001b[0mfh\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0mopen\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfname\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mflag\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0mencoding\u001b[0m\u001b[0;34m=\u001b[0m\u001b[0mencoding\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[0m\u001b[1;32m    393\u001b[0m         \u001b[0mopened\u001b[0m \u001b[0;34m=\u001b[0m \u001b[0;32mTrue\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n\u001b[1;32m    394\u001b[0m     \u001b[0;32melif\u001b[0m \u001b[0mhasattr\u001b[0m\u001b[0;34m(\u001b[0m\u001b[0mfname\u001b[0m\u001b[0;34m,\u001b[0m \u001b[0;34m'seek'\u001b[0m\u001b[0;34m)\u001b[0m\u001b[0;34m:\u001b[0m\u001b[0;34m\u001b[0m\u001b[0;34m\u001b[0m\u001b[0m\n",
      "\u001b[0;31mFileNotFoundError\u001b[0m: [Errno 2] No such file or directory: '/Users/jordanm/grad/geometric-certificates/plots/incremental_geocert/0.png'"
     ]
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 432x288 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    },
    {
     "data": {
      "image/png": "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AgBP5syZk/cpREmKx+O66aabVFVVpbFjx4aKCKAAKFsA4Mn69etzFsanVVVV6Y477lDHjh1DRANQQEwjAoAnH374obp06SIzy5lCPP744ylaQIWgbAGAJx988IG2bt2qeDzevL9WekrxBz/4QchoAAqIsgUAnqR3js/U0NCghoYG/e///q927doVKBmAQqJsAYAnS5cuzbpOP5UoSXfeeadeffXVQkcCEAAL5AGgjX39619XdXW1/vrXv2at1cpcLN+lSxcNGzYsVEQABUTZAoA2tHr1ar3yyiuqr6+Xc05VVVXNhStzC4gRI0aEjAmggJhGBIA2VFNTk3dvrcyDqM1MU6ZMCZYRQGFRtgCgDS1btiyrWKWlpxCrqqrUu3dv9evXL0g+AIVH2QKANrJ06VJt2bIlZyPTzJEuM9Po0aODZQRQeJQtAGgjd911lyS1OIUYi8UURZEmTpwYJiCAIChbANAGksmkli9f3jyqlfkUYrpsRVGkr3zlK+rUqVOQjADCoGwBQBt4+OGHtXPnzv1OIV511VXBMgIIg7IFAG1g7ty5WbvF7308TxRFateunW655ZYg+QCEwz5bANBK27dv15o1a3JGtaSm6cQOHTromWee0apVq7JeA1AZKFsA0ErPPfecnHNZC+EznXTSSRo2bBg7xgMVimlEAGilkSNHatmyZYqiSLFYLGt/LUm67rrrAiUDUAwY2QKANnDvvffmTBE659ShQwddc801gVIBKAaMbAFAG3juueeyrhsbG9XQ0CBJevzxx0NEAlAkKFsA0ErPPfecNm/enHUvvVh+x44duv3220PEAlAkmEYEgIO0cuVKXXTRRUomk3LOZa3VytxF/jvf+U6oiACKAGULAA5STU2N1q9fr/r6eklSdXW1JGXtIh+LxTR58uRgGQGExzQiABykF154Yb/H8/Tp00d9+vQJkg9AcaBsAcBBWLJkiT777LOcvbX2Pp7noosuCpYRQHGgbAHAQZg5c6ak7LVZkrLKV1VVlcaNGxcmIICiQdkCgAOUTCZVW1ubdwox88ieE044QR07dgySEUDxoGwBwAGaO3eudu/enXMW4t4HUV977bVB8gEoLpQtADhADz30UE6xkrJHtQ455BCNGTMmSD4AxYWyBQAHYPv27XrrrbdyRrUyRVGkQYMG5X0NQOVhny0AOAA///nPlUgkcp5ClKSqqioNHDhQt912m770pS+FigigyFC2AOAArF69uvlnM8vaNV6Srr/+el144YWFjgWgiDGNCAAHYNGiRTrnnHMUi8VypgkPPfRQXXbZZYGSAShWjGwBwAF69dVXVVX1+b8+0+cinnHGGazTApCDkS0AOABPP/20tmzZknUvkUiovr5e27dv18qVKwMlA1CsKFsAcADuuuuurOvM43mWL1+uuXPnhogFoIgxjQgAX8A111yjN954Q6tXr26eNpSyj+eJxWL66U9/GjImgCJE2QKAL+C///u/tWXLFiWTyeZzD6XPy1YURerbt6+6d+8eMiaAIsQ0IgDsx6JFi7Rt27acvbUypxDNTJdeemmwjACKF2ULAPZj1qxZWcUqXbYyy1c8Htftt98eLCOA4kXZAoB9SCaTeu211/LuGJ95ZM9JJ52k9u3bB8kIoLhRtgBgHx544AHt3r075yzEvQ+i5tBpAC2hbAHAPvzyl7/MWpeVfgoxs3x16NBBV199daiIAIocZQsAWrB161a9/fbbzcUqcwox8ynEwYMHs3M8gBax9QMAtGD69OlKJpN5n0LMHOViYTyAfaFsAUALHn/88bxTiGameDyuuXPnavfu3frWt74VMiaAIkfZAoA8ksmkduzYoUQiIUk504QdO3bU5ZdfzvQhgP1izRYA5BFFkf7yl7/o8MMPl5llrdeSpK997WsULQBfCGULAFqwZMkS7dq1S/F4vPleelpx/PjxoWIBKDGULQBowcyZM7OunXNqaGhQMpnUggULtHXr1kDJAJQSyhYA5JFMJlVbW5t1L/1UonNOc+fOVX19fYhoAEoMC+QBYC/9+/dXdXW1du7c2eLeWscee6y6du0aKiKAEkLZAoAMv/71r7V+/Xo1NDTIOad4PC4zyzqI2sx0+eWXB04KoFQwjQgAGWbPnp1TrCRlbWxaXV2t2267LVhGAKWFsgUAKclkUitXrsx7PE/mWYgDBgxQdXV1kIwASg9lCwBSZs2apT179mStzZI+H9VK7yJ/ww03BMsIoPRQtgAg5ZFHHsl7PE/mFOKhhx6qSy65JFhGAKWHsgUAkjZv3qz33nsv7xRi5kjX0KFD2TkewAGhbAGApOnTpyuZTGaNYknZU4iSNG7cuDABAZQsyhYASFq4cGHeKcS0qqoqHXXUURo+fHiIeABKGPtsAah47777rj76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\n",
      "text/plain": [
       "<Figure size 720x720 with 1 Axes>"
      ]
     },
     "metadata": {
      "needs_background": "light"
     },
     "output_type": "display_data"
    }
   ],
   "source": [
    "# ==================================\n",
    "# Initialize Network\n",
    "# ==================================\n",
    "\n",
    "print('===============Initializing Network============')\n",
    "layer_sizes = [2, 8, 2, 2]\n",
    "network = PLNN(layer_sizes)\n",
    "net = network.net\n",
    "\n",
    "\n",
    "# ==================================\n",
    "# Train Network\n",
    "# ==================================\n",
    "\n",
    "print('===============Training Network============')\n",
    "opt = optim.Adam(net.parameters(), lr=1e-3)\n",
    "for i in range(1000):\n",
    "    out = net(Variable(X))\n",
    "    l = nn.CrossEntropyLoss()(out, Variable(y))\n",
    "    err = (out.max(1)[1].data != y).float().mean()\n",
    "    if i % 100 == 0:\n",
    "        print('error:', err)\n",
    "    opt.zero_grad()\n",
    "    (l).backward()\n",
    "    opt.step()\n",
    "\n",
    "# ==================================\n",
    "# Visualize:  classifier boundary\n",
    "# ==================================\n",
    "\n",
    "XX, YY = np.meshgrid(np.linspace(0, 1, 100), np.linspace(0, 1, 100))\n",
    "X0 = Variable(torch.Tensor(np.stack([np.ravel(XX), np.ravel(YY)]).T))\n",
    "y0 = network(X0)\n",
    "ZZ = (y0[:,0] - y0[:,1]).resize(100, 100).data.numpy()\n",
    "\n",
    "_, ax = plt.subplots(figsize=(8,8))\n",
    "ax.contourf(XX,YY,-ZZ, cmap=\"coolwarm\", levels=np.linspace(-1000,1000,3))\n",
    "ax.scatter(X.numpy()[:,0], X.numpy()[:,1], c=y.numpy(), cmap=\"coolwarm\", s=70)\n",
    "ax.axis(\"equal\")\n",
    "ax.axis([0, 1, 0, 1])\n",
    "\n",
    "plt.show()\n",
    "\n",
    "# ==================================\n",
    "# Visualize: baseline classifier ReLu regions\n",
    "# ==================================\n",
    "\n",
    "print('===============Collecting Polytopes============')\n",
    "num_pts = 200\n",
    "xylim = 1.0\n",
    "\n",
    "unique_relu_configs_list, unique_bin_acts, _, _ = utils.get_unique_relu_configs(network, xylim, num_pts)\n",
    "print('number of polytopes:', len(unique_bin_acts))\n",
    "color_dict = utils.get_color_dictionary(unique_bin_acts)\n",
    "polytope_list = []\n",
    "\n",
    "\n",
    "for relu_configs, unique_act in zip(unique_relu_configs_list, unique_bin_acts):\n",
    "    polytope_dict = network.compute_polytope_config(relu_configs, True)\n",
    "    polytope = from_polytope_dict(polytope_dict)\n",
    "    polytope_list.append(polytope)\n",
    "    # colors.append(color_dict[unique_act])\n",
    "colors = utils.get_spaced_colors(200)[0:len(polytope_list)]\n",
    "x_0 = torch.Tensor([[0.3], [0.5]])\n",
    "\n",
    "print('===============Finding Classification Boundary Facets============')\n",
    "\n",
    "true_label = int(network(x_0).max(1)[1].item())  # what the classifier outputs\n",
    "\n",
    "adversarial_facets = []\n",
    "for polytope in polytope_list:\n",
    "    polytope_adv_constraints = network.make_adversarial_constraints(polytope.config,\n",
    "                                                                    true_label)\n",
    "\n",
    "    for facet in polytope_adv_constraints:\n",
    "        adversarial_facets.append(facet)\n",
    "\n",
    "\n",
    "# ------------------------------\n",
    "# Plot Polytopes, boundary, and lp norm\n",
    "# ------------------------------\n",
    "\n",
    "ax = plt.gca()\n",
    "alpha = 0.6\n",
    "xylim = 1.0\n",
    "\n",
    "utils.plot_polytopes_2d(polytope_list, colors, alpha, xylim, ax)\n",
    "utils.plot_facets_2d(adversarial_facets, xylim=xylim, ax=ax, color='black', linestyle='dashed')\n",
    "plt.xlim(0.0, 1.0)\n",
    "plt.ylim(0.0, 1.0)\n",
    "plt.show()\n",
    "\n",
    "\n",
    "# ==================================\n",
    "# Find Projections\n",
    "# ==================================\n",
    "\n",
    "lp_norm = 'l_2'\n",
    "ts = []\n",
    "pts = x\n",
    "\n",
    "for pt in pts:\n",
    "    print('===============Finding Projection============')\n",
    "    print('lp_norm: ', lp_norm)\n",
    "    x_0 = torch.Tensor(pt.reshape([2, 1]))\n",
    "    print(x_0)\n",
    "    print('from point: ')\n",
    "    print(x_0)\n",
    "\n",
    "    ax = plt.axes()\n",
    "    cwd = os.getcwd()\n",
    "    print(cwd)\n",
    "    plot_dir = cwd + '/plots/incremental_geocert/'\n",
    "\n",
    "    t = incremental_GeoCert(lp_norm, network, x_0, ax, plot_dir)\n",
    "\n",
    "    print('the final projection value:', t)\n",
    "    ts.append(t)\n",
    "\n",
    "# ==================================\n",
    "# Visualize: incremental geocert projections\n",
    "# ==================================\n",
    "\n",
    "XX, YY = np.meshgrid(np.linspace(0, 1, 100), np.linspace(0, 1, 100))\n",
    "X0 = Variable(torch.Tensor(np.stack([np.ravel(XX), np.ravel(YY)]).T))\n",
    "y0 = network(X0)\n",
    "ZZ = (y0[:,0] - y0[:,1]).resize(100,100).data.numpy()\n",
    "\n",
    "_, ax = plt.subplots(figsize=(8,8))\n",
    "ax.contourf(XX,YY,-ZZ, cmap=\"coolwarm\", levels=np.linspace(-1000,1000,3))\n",
    "ax.scatter(X.numpy()[:,0], X.numpy()[:,1], c=y.numpy(), cmap=\"coolwarm\", s=70)\n",
    "ax.axis(\"equal\")\n",
    "ax.axis([0, 1, 0, 1])\n",
    "\n",
    "for pt, t, y in zip(pts, ts, y.numpy()):\n",
    "    if lp_norm == 'l_2':\n",
    "        utils.plot_l2_norm(pt, t, ax=ax)\n",
    "    elif lp_norm == 'l_inf':\n",
    "        utils.plot_linf_norm(pt, t, ax=ax)\n",
    "    else:\n",
    "        raise NotImplementedError\n",
    "\n",
    "print('average_linf:', np.average(ts))\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# apply incremental geocert to a normal and l1-regularized classifier. Finds maximal l_p balls\n",
    "# for random points in R^2.\n",
    "\n",
    "# ==================================\n",
    "# Generate Training Points\n",
    "# ==================================\n",
    "\n",
    "print('===============Generating Training Points============')\n",
    "# random points at least 2r apart\n",
    "m = 12\n",
    "np.random.seed(3)\n",
    "x = [np.random.uniform(size=(2))]\n",
    "r = 0.16\n",
    "while(len(x) < m):\n",
    "    p = np.random.uniform(size=(2))\n",
    "    if min(np.abs(p-a).sum() for a in x) > 2*r:\n",
    "        x.append(p)\n",
    "# r = 0.145\n",
    "epsilon = r/2\n",
    "\n",
    "X = torch.Tensor(np.array(x))\n",
    "torch.manual_seed(1)\n",
    "y = (torch.rand(m)+0.5).long()\n",
    "print('===============Points Generated============')"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": [
    "# ==================================\n",
    "# Initialize Network\n",
    "# ==================================\n",
    "\n",
    "print('===============Initializing Network============')\n",
    "network = PLNN(layer_sizes)\n",
    "net = network.net\n",
    "\n",
    "\n",
    "# ==================================\n",
    "# Train Network\n",
    "# ==================================\n",
    "\n",
    "def l1_loss(net):\n",
    "\n",
    "    return sum([_.norm(p=1) for _ in net.parameters() if _.dim() > 1])\n",
    "\n",
    "\n",
    "print('===============Training Network with Regularization============')\n",
    "opt = optim.Adam(net.parameters(), lr=1e-3)\n",
    "for i in range(1000):\n",
    "    out = net(Variable(X))\n",
    "    l = nn.CrossEntropyLoss()(out, Variable(y)).view([1])\n",
    "\n",
    "    l1_scale = torch.Tensor([1e-4])\n",
    "    l += l1_scale * l1_loss(net).view([1])\n",
    "\n",
    "    err = (out.max(1)[1].data != y).float().mean()\n",
    "    opt.zero_grad()\n",
    "    (l).backward()\n",
    "    opt.step()\n",
    "\n",
    "print('error: ', err)\n",
    "\n",
    "\n",
    "\n",
    "# ==================================\n",
    "# Visualize:  regularized classifier boundary\n",
    "# ==================================\n",
    "\n",
    "XX, YY = np.meshgrid(np.linspace(0, 1, 100), np.linspace(0, 1, 100))\n",
    "X0 = Variable(torch.Tensor(np.stack([np.ravel(XX), np.ravel(YY)]).T))\n",
    "y0 = network(X0)\n",
    "ZZ = (y0[:,0] - y0[:,1]).resize(100, 100).data.numpy()\n",
    "\n",
    "_, ax = plt.subplots(figsize=(8,8))\n",
    "ax.contourf(XX,YY,-ZZ, cmap=\"coolwarm\", levels=np.linspace(-1000,1000,3))\n",
    "ax.scatter(X.numpy()[:,0], X.numpy()[:,1], c=y.numpy(), cmap=\"coolwarm\", s=70)\n",
    "ax.axis(\"equal\")\n",
    "ax.axis([0, 1, 0, 1])\n",
    "\n",
    "plt.show()\n",
    "\n",
    "# ==================================\n",
    "# Visualize: regularized classifier ReLu regions\n",
    "# ==================================\n",
    "\n",
    "print('===============Collecting Polytopes============')\n",
    "num_pts = 200\n",
    "xylim = 1.0\n",
    "\n",
    "unique_relu_configs_list, unique_bin_acts, _, _ = utils.get_unique_relu_configs(network, xylim, num_pts)\n",
    "print('number of polytopes:', len(unique_bin_acts))\n",
    "color_dict = utils.get_color_dictionary(unique_bin_acts)\n",
    "polytope_list = []\n",
    "\n",
    "\n",
    "for relu_configs, unique_act in zip(unique_relu_configs_list, unique_bin_acts):\n",
    "    polytope_dict = network.compute_polytope_config(relu_configs, True)\n",
    "    polytope = from_polytope_dict(polytope_dict)\n",
    "    polytope_list.append(polytope)\n",
    "    # colors.append(color_dict[unique_act])\n",
    "colors = utils.get_spaced_colors(200)[0:len(polytope_list)]\n",
    "x_0 = torch.Tensor([[0.3], [0.5]])\n",
    "\n",
    "print('===============Finding Classification Boundary Facets============')\n",
    "\n",
    "true_label = int(network(x_0).max(1)[1].item())  # what the classifier outputs\n",
    "\n",
    "adversarial_facets = []\n",
    "for polytope in polytope_list:\n",
    "    polytope_adv_constraints = network.make_adversarial_constraints(polytope.config,\n",
    "                                                                    true_label)\n",
    "\n",
    "    for facet in polytope_adv_constraints:\n",
    "        adversarial_facets.append(facet)\n",
    "\n",
    "\n",
    "# ------------------------------\n",
    "# Plot Polytopes, boundary, and lp norm\n",
    "# ------------------------------\n",
    "\n",
    "ax = plt.gca()\n",
    "alpha = 0.6\n",
    "xylim = 1.0\n",
    "\n",
    "utils.plot_polytopes_2d(polytope_list, colors, alpha, xylim, ax)\n",
    "utils.plot_facets_2d(adversarial_facets, xylim=xylim, ax=ax, color='black', linestyle='dashed')\n",
    "plt.xlim(0.0, 1.0)\n",
    "plt.ylim(0.0, 1.0)\n",
    "plt.show()\n",
    "\n",
    "\n",
    "# ==================================\n",
    "# Find Projections\n",
    "# ==================================\n",
    "\n",
    "lp_norm = 'l_2'\n",
    "ts = []\n",
    "pts = x\n",
    "\n",
    "for pt in pts:\n",
    "    print('===============Finding Projection============')\n",
    "    print('lp_norm: ', lp_norm)\n",
    "    x_0 = torch.Tensor(pt.reshape([2, 1]))\n",
    "    print(x_0)\n",
    "    print('from point: ')\n",
    "    print(x_0)\n",
    "\n",
    "    ax = plt.axes()\n",
    "    cwd = os.getcwd()\n",
    "    print(cwd)\n",
    "    plot_dir = cwd + '/plots/incremental_geocert/'\n",
    "\n",
    "    t = incremental_GeoCert(lp_norm, network, x_0, ax, plot_dir)\n",
    "\n",
    "    print('the final projection value:', t)\n",
    "    ts.append(t)\n",
    "\n",
    "# ==================================\n",
    "# Visualize: incremental geocert projections\n",
    "# ==================================\n",
    "\n",
    "XX, YY = np.meshgrid(np.linspace(0, 1, 100), np.linspace(0, 1, 100))\n",
    "X0 = Variable(torch.Tensor(np.stack([np.ravel(XX), np.ravel(YY)]).T))\n",
    "y0 = network(X0)\n",
    "ZZ = (y0[:,0] - y0[:,1]).resize(100,100).data.numpy()\n",
    "\n",
    "_, ax = plt.subplots(figsize=(8,8))\n",
    "ax.contourf(XX,YY,-ZZ, cmap=\"coolwarm\", levels=np.linspace(-1000,1000,3))\n",
    "ax.scatter(X.numpy()[:,0], X.numpy()[:,1], c=y.numpy(), cmap=\"coolwarm\", s=70)\n",
    "ax.axis(\"equal\")\n",
    "ax.axis([0, 1, 0, 1])\n",
    "\n",
    "for pt, t, y in zip(pts, ts, y.numpy()):\n",
    "    if lp_norm == 'l_2':\n",
    "        utils.plot_l2_norm(pt, t, ax=ax)\n",
    "    elif lp_norm == 'l_inf':\n",
    "        utils.plot_linf_norm(pt, t, ax=ax)\n",
    "    else:\n",
    "        raise NotImplementedError\n",
    "\n",
    "print('average_linf:', np.average(ts))\n",
    "\n",
    "plt.show()"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "py3",
   "language": "python",
   "name": "py3"
  },
  "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.7.2"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
