{
 "cells": [
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# GeoCert : 2-dimensional Toy Example\n",
    "In this example, we'll train a simple neural network and demonstrate the various capabilities of GeoCert.\n",
    "\n",
    "The network training code was largely copied from [Eric Wong's repository found here.](https://github.com/locuslab/convex_adversarial/blob/master/examples/2D.ipynb)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 3,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Import stack\n",
    "import sys \n",
    "sys.path.append('..')\n",
    "sys.path.append('../mister_ed')\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",
    "\n",
    "import matplotlib.pyplot as plt\n",
    "import seaborn\n",
    "import matplotlib.patches as patches\n",
    "from scipy.spatial import HalfspaceIntersection\n",
    "\n",
    "seaborn.set(font_scale=2)\n",
    "seaborn.set_style(\"white\")\n",
    "\n",
    "import numpy as np \n",
    "%matplotlib inline\n",
    "seaborn.set(font_scale=2)\n",
    "seaborn.set_style(\"white\")\n",
    "from geocert import GeoCert\n",
    "from plnn import PLNN\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 4,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "[2, 20, 50, 20, 2]\n",
      "0.7069634795188904 tensor(0.5833)\n",
      "0.5537288784980774 tensor(0.2500)\n",
      "0.4080643653869629 tensor(0.2500)\n",
      "0.3087824881076813 tensor(0.1667)\n",
      "0.1617790013551712 tensor(0.)\n",
      "0.04387936368584633 tensor(0.)\n",
      "0.014375497587025166 tensor(0.)\n",
      "0.006310502532869577 tensor(0.)\n",
      "0.0034499764442443848 tensor(0.)\n",
      "0.0021875500679016113 tensor(0.)\n",
      "0.0015173951396718621\n"
     ]
    },
    {
     "name": "stderr",
     "output_type": "stream",
     "text": [
      "/home/matt/configs/env/lib/python3.6/site-packages/torch/tensor.py:339: UserWarning: non-inplace resize is deprecated\n",
      "  warnings.warn(\"non-inplace resize is deprecated\")\n"
     ]
    },
    {
     "data": {
      "image/png": "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n3zyieX1BTNvb9vr5ebm6tlnn1VMTIxeeuklZWRk9L3gIOKrbXvgwIFul507d07nzp1TYmKih9UGF29v2/Hjxys+Pl719fW6ePFil21qa2slffkNLVR5e9u2fcHqaZ9sW9bdtkdnvji+BM3wQXJyslJTU9XU1KQdO3Z0Wl5YWKjq6mq5XC5L30Sjo6N1++23S5LeeuutTssrKipUVFSkqKgo3XnnnX2uP5B5e9u2yc/P1+rVqxUdHa0NGzZo3rx53iw7KHh7227ZskVHjhzp8r8f/vCHklqefXDkyBF99NFHXv88gcTb2/b6v/WCgoJOy5uamtq36bRp0/pWfIDz9rZtu/34xIkTunz5cpdtDh48KKn7Xhp05otjd9CEAklasmSJJGnt2rUqLy9vf/3ChQvKycmRJGVnZ3eY43nr1q3KzMzUU0891Wl92dnZcjgc+stf/tLeKyC1jMGsWLFChmHo0UcfVf/+/X31kQKGt7fttm3blJOTo+joaP3xj3/UN77xDR9/gsDl7W2LL3l72y5dulROp1NvvPGG3n///fbX3W631q5dq1OnTmnYsGG65557fPWRAoY3t+2sWbM0dOhQNTY2auXKlaqrq2tfZhiG/vSnP6moqEiRkZEh/+yD3li3bp0yMzO1bt26Tst683vqSdAMH0hSZmamFixYoPz8fGVlZWnevHntD36oq6vT/PnzO83sVltbq7KyMrlcrk7rmzFjhn72s59p7dq1+s53vqM5c+YoMTFR+/fv14ULFzRz5kz99Kc/9dfHs5U3t21paameeeYZmaapUaNG6Z133tE777zT6T2TkpL085//3KefKxB4e7/Fl7y9bSdPnqwVK1ZozZo1ys7O1owZMzR8+HCVlJSooqJCiYmJevHFF7udKCaUeHPbRkdH6/nnn9cPfvADvfvuuyosLNT06dMVGxur0tJSVVZWyul0asWKFd1OJhcqDh8+3H6ylqTjx49Lkn73u99p8+bN7a9v27at/d81NTUqKytTTU1Np/X15vfUk6AKBZK0atUqpaenKzc3V4WFhTIMQxMmTOj1ozyzs7M1adIkvfLKKyouLm5/dPKiRYvC7tHJ3tq2ly9fbp/57cSJEzpx4kSX7UaOHBkWoUDy/n6LL3l72y5atEgpKSnavHmzioqKVFJSIpfLpUceeURLliwJq+5tb27b2267TX/729/0yiuvaO/eve3rGzJkiO677z4tXrxYs2bN8uGnCQx1dXXtQyXXa5vptTe8+XsKukcnAwAA3+DrCQAAkEQoAAAArQgFAABAEqEAAAC0IhQAAABJhAIAANCKUAAAACQRCgAAQCtCAQAAkCT9//F5cISLrqp0AAAAAElFTkSuQmCC\n",
      "text/plain": [
       "<Figure size 576x576 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Training block -- train a simple 2d Network \n",
    "# --- define network \n",
    "plnn_obj = PLNN(layer_sizes=[2, 20, 50, 20, 2])\n",
    "net = plnn_obj.net\n",
    "\n",
    "# --- define random training points \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",
    "epsilon = r/2\n",
    "X = torch.Tensor(np.array(x))\n",
    "torch.manual_seed(1)\n",
    "y = (torch.rand(m)+0.5).long()\n",
    "\n",
    "# --- train 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(l.item(), err)\n",
    "    opt.zero_grad()\n",
    "    (l).backward()\n",
    "    opt.step()\n",
    "print(l.item())\n",
    "\n",
    "# --- display network decision boundaries\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 = net(X0)\n",
    "ZZ = (y0[:,0] - y0[:,1]).resize(100,100).data.numpy()\n",
    "\n",
    "fig, 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 a in x:\n",
    "    ax.add_patch(patches.Rectangle((a[0]-r/2, a[1]-r/2), r, r, fill=False))\n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "# Using GeoCert:\n",
    "GeoCert is run through the `GeoCert` object interface. Initialization requires knowledge of which neural network to certify, and optionally (useful for most applications) a bounds on the valid domain of the network. Individual points are verified using the `.run(...)` method. We can solve three types of problems:\n",
    "\n",
    "- `min_dist`: Computes the minimum distance to a point that is classified differently than the specified point\n",
    "- `decision_problem`: Returns yes/no whether or not an adversarial example exists within the specified radius of the input point \n",
    "- `count_regions`: Counts the number of linear regions within an $\\ell_p$ ball of the input point"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 5,
   "metadata": {},
   "outputs": [],
   "source": [
    "# Initialize a GeoCert object to be reused\n",
    "geo = GeoCert(plnn_obj, hyperbox_bounds=(0.0, 1.0), verbose=False)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Minimum Distance : $\\ell_\\infty$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 6,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Academic license - for non-commercial use only\n"
     ]
    }
   ],
   "source": [
    "# Single example for computing L_inf min_dists\n",
    "test_point = torch.Tensor([0.420, 0.69])\n",
    "linf_output = geo.run(test_point, lp_norm='l_inf', problem_type='min_dist')\n",
    "\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 7,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "GeoCert Return Object\n",
      "\tProblem Type: min_dist\n",
      "\tStatus: SUCCESS\n",
      "\tRobustness: 0.0196\n"
     ]
    }
   ],
   "source": [
    "# Example printout from the return object\n",
    "print(linf_output)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 8,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "Original Logits    : [ 0.64337 -1.07056]\n",
      "Adversarial Logits : [-0.25114 -0.25115]\n"
     ]
    }
   ],
   "source": [
    "# Can collect the minimal distance adversarial example and examine its logits \n",
    "original_logits = net(test_point)\n",
    "adversarial_logits = net(torch.Tensor(linf_output.best_ex))\n",
    "print(\"Original Logits    :\", original_logits.detach().numpy())\n",
    "print(\"Adversarial Logits :\", adversarial_logits.detach().numpy())"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 9,
   "metadata": {},
   "outputs": [
    {
     "data": {
      "text/plain": [
       "<matplotlib.patches.Rectangle at 0x7f8a4006db00>"
      ]
     },
     "execution_count": 9,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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+/dr+/btOnToUPcUxt133x2WUxhm9K3T6VRlZaX27dunAwcO6KuvvlJra6uuvPJKzZw5U4sXL9bChQv9vCbBx+zt9lJvvPGGnnrqKUkXb/Pt+Xe4MLtvu7q6tHPnTv35z3/W0aNH1draqquuukrTp0/XAw88oDvvvNMPaxGczOzb8+fP67XXXtMHH3ygkydPqrOzUzabTXPnztXPf/7zsPgJ+rFjx/Thhx+qqqpKhw4d0vHjx+VyufTiiy/2+RRio8z6niw3fZCbm6uSkhLFxcVp/vz53Rc7XrhwQXfddZc2bdrkVQds2bJFeXl5ioqKUnp6ukaMGKGKigr9/e9/V0pKirZv3z7gMxpCiVl9W19frx//+MeSpJEjR2rWrFkaMWKETp482X0/i/vvv1+/+93vFBER4dd1ChZmb7eXOnXqlDIyMtTa2iqXyxV2ocDsvm1ublZWVpaqqqo0cuRIpaSkaNiwYXI4HKqurlZGRoY2bNjgxzUKHmb2bWNjo5YsWaLGxkaNGjVKN954o+Li4lRTU6MTJ04oOjpazz33nH7yk5/4ea0Ca8OGDXr99dd7ve5LKDD1b8BlIbt373YlJye7br75ZlddXV33601NTa5/+7d/cyUnJ7u2b99uuL7PP//cNW3aNNeNN97oqqys7H69paXFtWTJEldycrJrw4YNZq5C0DKzb+vr613Lly937du3z9XZ2dlj2YEDB1wpKSmu5ORk13/913+ZuQpBy+zt9lJOp9O1YsUKV0pKiuvxxx93JScnu3Jzc01qefAzu2+7urpcixcvdiUnJ7uefvppV3t7e4/l3377reuLL74wq/lBzey+ffTRR13JycmurKwsV2tra/frXV1drk2bNrmSk5Nd6enpru+++87M1Qg6O3fudP3+9793/fd//7ervr7etXTpUldycrLrvffeG1R9Zn9PlgoF9913nys5Odn11ltv9Vp24MCB7o7p6uoyVN/q1atdycnJrpdeeqnXshMnTrhuuOEG18yZM13nz5/3ue3Bzuy+HcjmzZtdycnJruXLl/tclxX4s2+Li4tdycnJrtdff717xxpOocDsvi0tLXUlJye7Vq1aZXZTLcfsvr355ptdycnJrv/7v//rtayzs9M1Z84cV3JysuvIkSM+t91KfA0FZn9PlrnQ0OFw6PDhw4qJielziCU9PV1jx45VU1OT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\n",
      "text/plain": [
       "<Figure size 576x576 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# And we can plot the ball around this to demonstrate...\n",
    "fig, ax = plt.subplots(figsize=(8,8))\n",
    "ax.contourf(XX,YY,-ZZ, cmap=\"coolwarm\", levels=np.linspace(-1000,1000,3))\n",
    "ax.axis(\"equal\")\n",
    "ax.axis([0,1,0,1])\n",
    "ax.scatter([test_point[0].item()], [test_point[1].item()], c=[0], cmap=\"coolwarm\", s=70)\n",
    "ax.add_patch(patches.Rectangle((test_point[0].item()-linf_output.best_dist, \n",
    "                                test_point[1].item()-linf_output.best_dist), \n",
    "                               linf_output.best_dist * 2, linf_output.best_dist * 2, fill=False))\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 10,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "data": {
      "image/png": "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5hn993//d7DLjHiWCQUd39qvnYv6uo4k1JGMBtpfRwoz05+V+Xrb9uTVV1/Vxo0blZCQoM2bN2vKlCn96sdqfL1t33jjDR04cEBLly7VLbfc4psiLcrX2/baNg8++GC35WPGjNG3vvUtSdL+/fu9qtVqfL1tW1tb9cQTT+jMmTP63e9+pwULFsjpdCoxMVFz587Vli1bNHLkSL311lsELi/449htmXMKxo0bJ+nqDTB603E2e0dbM/3961//6rVNxzIz/VmZr7ft123dulXr1q2Tw+FQYWGh0tPT+1eoBfl62/7tb3+TJO3bt08HDhzosqzj3Jddu3bp+PHjSkhIUGFhYb/qtgJfb9vx48f3+O+e2ly8eNF0nVbk62176NAhnThxQjfccEOPf/9JSUm688479dZbb6msrExz5szpZ+WRxR/HbsuEgo7L2I4fP67m5uYez7TsuFuemVtlTp48WQ6HQ/X19Tp79myPVyD885//NN2flfl6216ruLhYzz33nOLi4vTyyy8rIyNj4AVbiL+27cGDB3td9vnnn+vzzz9XYmKil9Vai6+37aRJk5SQkCCXy6X6+voe29TV1Un66htauPL1tu34gtXXPtmxrLdtj+78cXyxzPRBcnKypk2bJrfbrR07dnRbXl5erurqajmdTlPfRGNjY3XnnXdKkt55551uy8+dO6eKigrZ7XbdddddA64/lPl623YoLS3V2rVrFRsbq02bNikzM9OXZVuCr7ft1q1bdezYsR7/95Of/ETS1WcfHDt2TB988IHP1yeU+HrbXvu3XlZW1m252+3u3KapqakDKz7E+Xrbdlx+fPLkSX355Zc9tjl06JCk3kdp0J0/jt2WCQWStHTpUklSQUGBzpw50/l6bW2t8vLyJEm5ubld7vFcVFSkrKwsPf300936y83NVVRUlP74xz92jgpIV+dgVq1aJcMw9NBDD2nIkCH+WqWQ4ettu23bNuXl5Sk2Nla///3vdccdd/h5DUKXr7ctvuLrbbts2TLZbDa9+eab+vvf/975eltbmwoKCnT27FmNHj1a9957r79WKWT4ctumpaVp1KhRam5u1urVq9XY2Ni5zDAMvfTSS6qoqFBMTEzYP/ugP9avX6+srCytX7++27L+/J76YpnpA0nKyspSTk6OSktLlZ2drczMzM4HPzQ2Nmr+/Pnd7uxWV1enU6dOyel0dutvxowZeuqpp1RQUKAf/OAHmjNnjhITE3XgwAHV1tbqtttu089+9rNArV5Q+XLbVlVV6dlnn5XH49H48eP17rvv6t133+32nsOGDdPPf/5zv65XKPD1fouv+Hrb3nLLLVq1apXy8/OVm5urGTNmaMyYMTp69KjOnTunxMREbdiwodcbxYQTX27b2NhYrVu3To8//rj++te/qry8XNOnT5fD4VBVVZXOnz8vm82mVatW9XozuXBx5MiRzg9rSTpx4oQk6Te/+Y22bNnS+fq2bds6/11TU6NTp06ppqamW3/9+T31xVKhQJLWrFmjmTNnqri4WOXl5TIMQ5MnT+73ozxzc3N1880369VXX1VlZWXno5MfeeSRiHt0sq+27Zdfftl557eTJ0/q5MmTPbYbN25cRIQCyff7Lb7i6237yCOPKCUlRVu2bFFFRYWOHj0qp9OpRYsWaenSpRE1vO3LbXv77bfr7bff1quvvqr333+/s7+RI0fqvvvu05IlS5SWlubHtQkNjY2NnVMl1+q402t/+PL3ZLlHJwMAAP/g6wkAAJBEKAAAAO0IBQAAQBKhAAAAtCMUAAAASYQCAADQjlAAAAAkEQoAAEA7QgEAAJAk/X/hKM2xZRi+MgAAAABJRU5ErkJggg==\n",
      "text/plain": [
       "<Figure size 576x576 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Now trying with several points to give more pretty pictures\n",
    "fig, ax = plt.subplots(figsize=(8,8))\n",
    "ax.contourf(XX,YY,-ZZ, cmap=\"coolwarm\", levels=np.linspace(-1000,1000,3))\n",
    "ax.axis(\"equal\")\n",
    "ax.axis([0,1,0,1])\n",
    "for i in range(10):\n",
    "    test_point = torch.rand(2)\n",
    "    linf_output = geo.run(test_point, lp_norm='l_inf', problem_type='min_dist')\n",
    "    ax.scatter([test_point[0].item()], [test_point[1].item()], c=[0], cmap=\"coolwarm\", s=70)\n",
    "    ax.add_patch(patches.Rectangle((test_point[0].item()-linf_output.best_dist, \n",
    "                                test_point[1].item()-linf_output.best_dist), \n",
    "                               linf_output.best_dist * 2, linf_output.best_dist * 2, fill=False))"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Minimum Distance : $\\ell_2$\n",
    "We can do the same things with $\\ell_2$ instead of $\\ell_\\infty$"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 11,
   "metadata": {
    "scrolled": true
   },
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "GeoCert Return Object\n",
      "\tProblem Type: min_dist\n",
      "\tStatus: SUCCESS\n",
      "\tRobustness: 0.0254\n",
      "Original Logits    : [ 0.64337 -1.07056]\n",
      "Adversarial Logits : [-0.25343 -0.25343]\n"
     ]
    },
    {
     "data": {
      "text/plain": [
       "<matplotlib.patches.Circle at 0x7f8a3161f828>"
      ]
     },
     "execution_count": 11,
     "metadata": {},
     "output_type": "execute_result"
    },
    {
     "data": {
      "image/png": "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\n",
      "text/plain": [
       "<Figure size 576x576 with 1 Axes>"
      ]
     },
     "metadata": {},
     "output_type": "display_data"
    }
   ],
   "source": [
    "# Single example for computing L_2 min_dists\n",
    "test_point = torch.Tensor([0.420, 0.69])\n",
    "l2_output = geo.run(test_point, lp_norm='l_2', problem_type='min_dist')\n",
    "\n",
    "\n",
    "# Example printout from the return object\n",
    "print(l2_output)\n",
    "\n",
    "# Can collect the minimal distance adversarial example and examine its logits \n",
    "original_logits = net(test_point)\n",
    "adversarial_logits = net(torch.Tensor(l2_output.best_ex))\n",
    "print(\"Original Logits    :\", original_logits.detach().numpy())\n",
    "print(\"Adversarial Logits :\", adversarial_logits.detach().numpy())\n",
    "\n",
    "\n",
    "# And we can plot the ball around this to demonstrate...\n",
    "fig, ax = plt.subplots(figsize=(8,8))\n",
    "ax.contourf(XX,YY,-ZZ, cmap=\"coolwarm\", levels=np.linspace(-1000,1000,3))\n",
    "ax.axis(\"equal\")\n",
    "ax.axis([0,1,0,1])\n",
    "ax.scatter([test_point[0].item()], [test_point[1].item()], c=[0], cmap=\"coolwarm\", s=70)\n",
    "ax.add_patch(patches.Circle(test_point, l2_output.best_dist, fill=False))\n",
    "                              \n"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Decision Problem\n",
    "Sometimes you might want to ask \"Does an adversarial example exist within a ball of specified size?\" and just want a yes/no answer. This formulation is useful when considering robustness to a particular threat model and can be faster than the `min_dist` formulation, as if we find an (even suboptimal) adversarial example, we'll return early"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 12,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "SAFE OUTPUT...\n",
      "GeoCert Return Object\n",
      "\tProblem Type: decision_problem\n",
      "\tStatus: FAILURE\n",
      "\tRadius 0.01\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# Recall our original test point....\n",
    "test_point = torch.Tensor([0.420, 0.69])\n",
    "l2_output = geo.run(test_point, lp_norm='l_2', problem_type='min_dist')\n",
    "# which has best dist specified by \n",
    "decision_threshold = l2_output.best_dist\n",
    "\n",
    "# Now we can try two variants on the decision problem....\n",
    "safe_region_output = geo.run(test_point, lp_norm='l_2', problem_type='decision_problem', \n",
    "                             decision_radius=decision_threshold / 2.0)\n",
    "\n",
    "print(\"SAFE OUTPUT...\")\n",
    "print(safe_region_output)\n",
    "# The printout status 'FAILURE' means 'FAILURE' to find an adversarial example within the radius"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 13,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "UNSAFE OUTPUT...\n",
      "GeoCert Return Object\n",
      "\tProblem Type: decision_problem\n",
      "\tStatus: SUCCESS\n",
      "\tRadius 0.05\n",
      "\n"
     ]
    }
   ],
   "source": [
    "# On the other hand, if there exists an adversarial example within the radius \n",
    "# we'll output 'SUCCESS' as the status \n",
    "\n",
    "unsafe_region_output = geo.run(test_point, lp_norm='l_2', problem_type='decision_problem', \n",
    "                             decision_radius=decision_threshold * 2.0)\n",
    "\n",
    "print(\"UNSAFE OUTPUT...\")\n",
    "print(unsafe_region_output)"
   ]
  },
  {
   "cell_type": "markdown",
   "metadata": {},
   "source": [
    "## Counting Linear Regions\n",
    "Finally, we can leverage GeoCert to count the number of linear regions (recall that ReLU nets are piecewise linear, so by 'linear regions', I mean the number of _pieces_) in a specified ball. Note that the domain restrictions are included by default here (so even if the ball extends past the \\[0, 1\\] hyperbox we've established upon initialization)."
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 14,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "---Initial Polytope---\n",
      "(p0) Popped: 0.012007  | 0.400000\n",
      "(p0) Popped: 0.045557  | 0.400000\n",
      "(p0) Popped: 0.058370  | 0.400000\n",
      "(p0) Popped: 0.063863  | 0.400000\n",
      "(p0) Popped: 0.065156  | 0.400000\n",
      "(p0) Popped: 0.070869  | 0.400000\n",
      "(p0) Popped: 0.080528  | 0.400000\n",
      "(p0) Popped: 0.091434  | 0.400000\n",
      "(p0) Popped: 0.096517  | 0.400000\n",
      "(p0) Popped: 0.097760  | 0.400000\n",
      "(p0) Popped: 0.102931  | 0.400000\n",
      "(p0) Popped: 0.104261  | 0.400000\n",
      "(p0) Popped: 0.104383  | 0.400000\n",
      "(p0) Popped: 0.104611  | 0.400000\n",
      "(p0) Popped: 0.104912  | 0.400000\n",
      "(p0) Popped: 0.106060  | 0.400000\n",
      "(p0) Popped: 0.107298  | 0.400000\n",
      "(p0) Popped: 0.109353  | 0.400000\n",
      "(p0) Popped: 0.109859  | 0.400000\n",
      "(p0) Popped: 0.114480  | 0.400000\n",
      "(p0) Popped: 0.115043  | 0.400000\n",
      "(p0) Popped: 0.116181  | 0.400000\n",
      "(p0) Popped: 0.116725  | 0.400000\n",
      "(p0) Popped: 0.119623  | 0.400000\n",
      "(p0) Popped: 0.120030  | 0.400000\n",
      "(p0) Popped: 0.120218  | 0.400000\n",
      "(p0) Popped: 0.120921  | 0.400000\n",
      "(p0) Popped: 0.121544  | 0.400000\n",
      "(p0) Popped: 0.122606  | 0.400000\n",
      "(p0) Popped: 0.123032  | 0.400000\n",
      "(p0) Popped: 0.128376  | 0.400000\n",
      "(p0) Popped: 0.130774  | 0.400000\n",
      "(p0) Popped: 0.131021  | 0.400000\n",
      "(p0) Popped: 0.131051  | 0.400000\n",
      "(p0) Popped: 0.132643  | 0.400000\n",
      "(p0) Popped: 0.133651  | 0.400000\n",
      "(p0) Popped: 0.135571  | 0.400000\n",
      "(p0) Popped: 0.135748  | 0.400000\n",
      "(p0) Popped: 0.135765  | 0.400000\n",
      "(p0) Popped: 0.136717  | 0.400000\n",
      "(p0) Popped: 0.138179  | 0.400000\n",
      "(p0) Popped: 0.142244  | 0.400000\n",
      "(p0) Popped: 0.142244  | 0.400000\n",
      "(p0) Popped: 0.143706  | 0.400000\n",
      "(p0) Popped: 0.147253  | 0.400000\n",
      "(p0) Popped: 0.147288  | 0.400000\n",
      "(p0) Popped: 0.148587  | 0.400000\n",
      "(p0) Popped: 0.149578  | 0.400000\n",
      "(p0) Popped: 0.150940  | 0.400000\n",
      "(p0) Popped: 0.152556  | 0.400000\n",
      "(p0) Popped: 0.153491  | 0.400000\n",
      "(p0) Popped: 0.153789  | 0.400000\n",
      "(p0) Popped: 0.155237  | 0.400000\n",
      "(p0) Popped: 0.157799  | 0.400000\n",
      "(p0) Popped: 0.159666  | 0.400000\n",
      "(p0) Popped: 0.160725  | 0.400000\n",
      "(p0) Popped: 0.176983  | 0.400000\n",
      "(p0) Popped: 0.177393  | 0.400000\n",
      "(p0) Popped: 0.177796  | 0.400000\n",
      "(p0) Popped: 0.177805  | 0.400000\n",
      "(p0) Popped: 0.181366  | 0.400000\n",
      "(p0) Popped: 0.182046  | 0.400000\n",
      "(p0) Popped: 0.182916  | 0.400000\n",
      "(p0) Popped: 0.183897  | 0.400000\n",
      "(p0) Popped: 0.184205  | 0.400000\n",
      "(p0) Popped: 0.193784  | 0.400000\n",
      "(p0) Popped: 0.195790  | 0.400000\n",
      "(p0) Popped: 0.198814  | 0.400000\n",
      "(p0) Popped: 0.202482  | 0.400000\n",
      "(p0) Popped: 0.203879  | 0.400000\n",
      "(p0) Popped: 0.204616  | 0.400000\n",
      "(p0) Popped: 0.208781  | 0.400000\n",
      "(p0) Popped: 0.213237  | 0.400000\n",
      "(p0) Popped: 0.213237  | 0.400000\n",
      "(p0) Popped: 0.212681  | 0.400000\n",
      "(p0) Popped: 0.212459  | 0.400000\n",
      "(p0) Popped: 0.211796  | 0.400000\n",
      "(p0) Popped: 0.215419  | 0.400000\n",
      "(p0) Popped: 0.216221  | 0.400000\n",
      "(p0) Popped: 0.216712  | 0.400000\n",
      "(p0) Popped: 0.216739  | 0.400000\n",
      "(p0) Popped: 0.217438  | 0.400000\n",
      "(p0) Popped: 0.218214  | 0.400000\n",
      "(p0) Popped: 0.218417  | 0.400000\n",
      "(p0) Popped: 0.220132  | 0.400000\n",
      "(p0) Popped: 0.224072  | 0.400000\n",
      "(p0) Popped: 0.225599  | 0.400000\n",
      "(p0) Popped: 0.231849  | 0.400000\n",
      "(p0) Popped: 0.232338  | 0.400000\n",
      "(p0) Popped: 0.233343  | 0.400000\n",
      "(p0) Popped: 0.235296  | 0.400000\n",
      "(p0) Popped: 0.235315  | 0.400000\n",
      "(p0) Popped: 0.235732  | 0.400000\n",
      "(p0) Popped: 0.239465  | 0.400000\n",
      "(p0) Popped: 0.242934  | 0.400000\n",
      "(p0) Popped: 0.244479  | 0.400000\n",
      "(p0) Popped: 0.245660  | 0.400000\n",
      "(p0) Popped: 0.246398  | 0.400000\n",
      "(p0) Popped: 0.246398  | 0.400000\n",
      "(p0) Popped: 0.247357  | 0.400000\n",
      "(p0) Popped: 0.249824  | 0.400000\n",
      "(p0) Popped: 0.251283  | 0.400000\n",
      "(p0) Popped: 0.252859  | 0.400000\n",
      "(p0) Popped: 0.254439  | 0.400000\n",
      "(p0) Popped: 0.254559  | 0.400000\n",
      "(p0) Popped: 0.256319  | 0.400000\n",
      "(p0) Popped: 0.261437  | 0.400000\n",
      "(p0) Popped: 0.266070  | 0.400000\n",
      "(p0) Popped: 0.273053  | 0.400000\n",
      "(p0) Popped: 0.273522  | 0.400000\n",
      "(p0) Popped: 0.273658  | 0.400000\n",
      "(p0) Popped: 0.273870  | 0.400000\n",
      "(p0) Popped: 0.277446  | 0.400000\n",
      "(p0) Popped: 0.279972  | 0.400000\n",
      "(p0) Popped: 0.279972  | 0.400000\n",
      "(p0) Popped: 0.283476  | 0.400000\n",
      "(p0) Popped: 0.283502  | 0.400000\n",
      "(p0) Popped: 0.283761  | 0.400000\n",
      "(p0) Popped: 0.285763  | 0.400000\n",
      "(p0) Popped: 0.287374  | 0.400000\n",
      "(p0) Popped: 0.287865  | 0.400000\n",
      "(p0) Popped: 0.289647  | 0.400000\n",
      "(p0) Popped: 0.294508  | 0.400000\n",
      "(p0) Popped: 0.298420  | 0.400000\n",
      "(p0) Popped: 0.300335  | 0.400000\n",
      "(p0) Popped: 0.300386  | 0.400000\n",
      "(p0) Popped: 0.301076  | 0.400000\n",
      "(p0) Popped: 0.302207  | 0.400000\n",
      "(p0) Popped: 0.303187  | 0.400000\n",
      "(p0) Popped: 0.304290  | 0.400000\n",
      "(p0) Popped: 0.307808  | 0.400000\n",
      "(p0) Popped: 0.308485  | 0.400000\n",
      "(p0) Popped: 0.309388  | 0.400000\n",
      "(p0) Popped: 0.310266  | 0.400000\n",
      "(p0) Popped: 0.311724  | 0.400000\n",
      "(p0) Popped: 0.311806  | 0.400000\n",
      "(p0) Popped: 0.314072  | 0.400000\n",
      "(p0) Popped: 0.314167  | 0.400000\n",
      "(p0) Popped: 0.314477  | 0.400000\n",
      "(p0) Popped: 0.315568  | 0.400000\n",
      "(p0) Popped: 0.316710  | 0.400000\n",
      "(p0) Popped: 0.318150  | 0.400000\n",
      "(p0) Popped: 0.318353  | 0.400000\n",
      "(p0) Popped: 0.323110  | 0.400000\n",
      "(p0) Popped: 0.324688  | 0.400000\n",
      "(p0) Popped: 0.324829  | 0.400000\n",
      "(p0) Popped: 0.326158  | 0.400000\n",
      "(p0) Popped: 0.326257  | 0.400000\n",
      "(p0) Popped: 0.326982  | 0.400000\n",
      "(p0) Popped: 0.329661  | 0.400000\n",
      "(p0) Popped: 0.329984  | 0.400000\n",
      "(p0) Popped: 0.350872  | 0.400000\n",
      "(p0) Popped: 0.351503  | 0.400000\n",
      "(p0) Popped: 0.351871  | 0.400000\n",
      "(p0) Popped: 0.353231  | 0.400000\n",
      "(p0) Popped: 0.353452  | 0.400000\n",
      "(p0) Popped: 0.359286  | 0.400000\n",
      "(p0) Popped: 0.360264  | 0.400000\n",
      "(p0) Popped: 0.362171  | 0.400000\n",
      "(p0) Popped: 0.365888  | 0.400000\n",
      "(p0) Popped: 0.367972  | 0.400000\n",
      "(p0) Popped: 0.370863  | 0.400000\n",
      "(p0) Popped: 0.371760  | 0.400000\n",
      "(p0) Popped: 0.377698  | 0.400000\n",
      "(p0) Popped: 0.378183  | 0.400000\n",
      "(p0) Popped: 0.380902  | 0.400000\n",
      "(p0) Popped: 0.380968  | 0.400000\n",
      "(p0) Popped: 0.381195  | 0.400000\n",
      "(p0) Popped: 0.384490  | 0.400000\n",
      "(p0) Popped: 0.386369  | 0.400000\n",
      "(p0) Popped: 0.392249  | 0.400000\n",
      "(p0) Popped: 0.394500  | 0.400000\n",
      "(p0) Popped: 0.397889  | 0.400000\n",
      "COUNTED 174 LINEAR REGIONS\n"
     ]
    }
   ],
   "source": [
    "geo.verbose = True # This will take a while, so let's turn on printouts\n",
    "\n",
    "test_point = torch.Tensor([0.420, 0.69])\n",
    "region_count = geo.run(test_point, lp_norm='l_inf', problem_type='count_regions', \n",
    "                       decision_radius=0.4)"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": 15,
   "metadata": {},
   "outputs": [
    {
     "name": "stdout",
     "output_type": "stream",
     "text": [
      "GeoCert Return Object\n",
      "\tProblem Type: count_regions\n",
      "\tStatus: SUCCESS\n",
      "\tRadius 0.40\n",
      "\tNum Linear Regions: 174\n"
     ]
    }
   ],
   "source": [
    "print(region_count)\n"
   ]
  },
  {
   "cell_type": "code",
   "execution_count": null,
   "metadata": {},
   "outputs": [],
   "source": []
  }
 ],
 "metadata": {
  "kernelspec": {
   "display_name": "Python 3",
   "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.6.8"
  }
 },
 "nbformat": 4,
 "nbformat_minor": 2
}
